mirror of
https://gitlab.freedesktop.org/gstreamer/gst-plugins-rs.git
synced 2025-01-02 23:38:45 +00:00
net/webrtc: implement AWS KVS signaller
And expose a wrapper webrtcsink variant, aws-kvs-webrtcsink. This adds support in webrtcsink for processing a consumer offer, instead of producing one. Part-of: <https://gitlab.freedesktop.org/gstreamer/gst-plugins-rs/-/merge_requests/1114>
This commit is contained in:
parent
82f1789589
commit
584392049c
10 changed files with 1743 additions and 219 deletions
|
@ -5904,6 +5904,37 @@
|
|||
"rswebrtc": {
|
||||
"description": "GStreamer plugin for high level WebRTC elements and a simple signaling server",
|
||||
"elements": {
|
||||
"awskvswebrtcsink": {
|
||||
"author": "Mathieu Duponchelle <mathieu@centricular.com>",
|
||||
"description": "WebRTC sink",
|
||||
"hierarchy": [
|
||||
"GstAwsKvsWebRTCSink",
|
||||
"GstWebRTCSink",
|
||||
"GstBin",
|
||||
"GstElement",
|
||||
"GstObject",
|
||||
"GInitiallyUnowned",
|
||||
"GObject"
|
||||
],
|
||||
"interfaces": [
|
||||
"GstChildProxy",
|
||||
"GstNavigation"
|
||||
],
|
||||
"klass": "Sink/Network/WebRTC",
|
||||
"pad-templates": {
|
||||
"audio_%%u": {
|
||||
"caps": "audio/x-raw:\n",
|
||||
"direction": "sink",
|
||||
"presence": "request"
|
||||
},
|
||||
"video_%%u": {
|
||||
"caps": "video/x-raw:\n\nvideo/x-raw(memory:CUDAMemory):\n\nvideo/x-raw(memory:GLMemory):\n\nvideo/x-raw(memory:NVMM):\n",
|
||||
"direction": "sink",
|
||||
"presence": "request"
|
||||
}
|
||||
},
|
||||
"rank": "none"
|
||||
},
|
||||
"webrtcsink": {
|
||||
"author": "Mathieu Duponchelle <mathieu@centricular.com>",
|
||||
"description": "WebRTC sink",
|
||||
|
|
|
@ -33,6 +33,18 @@ gst_plugin_webrtc_protocol = { path="protocol", package = "gst-plugin-webrtc-sig
|
|||
human_bytes = "0.4"
|
||||
url = "2"
|
||||
|
||||
aws-config = "0.52.0"
|
||||
aws-types = "0.52.0"
|
||||
aws-sig-auth = "0.52.0"
|
||||
aws-smithy-http = { version = "0.52.0", features = [ "rt-tokio" ] }
|
||||
aws-smithy-types = "0.52.0"
|
||||
aws-sdk-kinesisvideo = "0.22.0"
|
||||
aws-sdk-kinesisvideosignaling = "0.22.0"
|
||||
http = "0.2.7"
|
||||
chrono = "0.4"
|
||||
data-encoding = "2.3.3"
|
||||
url-escape = "0.1.1"
|
||||
|
||||
[dev-dependencies]
|
||||
tracing = { version = "0.1", features = ["log"] }
|
||||
tracing-subscriber = { version = "0.3", features = ["registry", "env-filter"] }
|
||||
|
|
|
@ -197,5 +197,19 @@ Parts of the JavaScript code in the www/ example are licensed under the [Apache
|
|||
the rest is licensed under the [Mozilla Public License Version 2.0] unless advertised in the
|
||||
header.
|
||||
|
||||
## Using the AWS KVS signaller
|
||||
|
||||
* Setup AWS Kinesis Video Streams
|
||||
|
||||
* Create a channel from the AWS console (<https://us-east-1.console.aws.amazon.com/kinesisvideo/home?region=us-east-1#/signalingChannels/create>)
|
||||
|
||||
* Start a producer:
|
||||
|
||||
```
|
||||
AWS_ACCESS_KEY_ID="XXX" AWS_SECRET_ACCESS_KEY="XXX" gst-launch-1.0 videotestsrc pattern=ball ! video/x-raw, width=1280, height=720 ! videoconvert ! textoverlay text="Hello from GStreamer!" ! videoconvert ! awskvswebrtcsink name=ws signaller::channel-name="XXX"
|
||||
```
|
||||
|
||||
* Connect a viewer @ <https://awslabs.github.io/amazon-kinesis-video-streams-webrtc-sdk-js/examples/index.html>
|
||||
|
||||
[Mozilla Public License Version 2.0]: http://opensource.org/licenses/MPL-2.0
|
||||
[Apache License, Version 2.0]: https://www.apache.org/licenses/LICENSE-2.1
|
||||
|
|
734
net/webrtc/src/aws_kvs_signaller/imp.rs
Normal file
734
net/webrtc/src/aws_kvs_signaller/imp.rs
Normal file
|
@ -0,0 +1,734 @@
|
|||
// SPDX-License-Identifier: MPL-2.0
|
||||
|
||||
use super::protocol as p;
|
||||
use crate::webrtcsink::WebRTCSink;
|
||||
use crate::RUNTIME;
|
||||
use anyhow::{anyhow, Error};
|
||||
use async_tungstenite::tungstenite::Message as WsMessage;
|
||||
use futures::channel::mpsc;
|
||||
use futures::prelude::*;
|
||||
use gst::glib;
|
||||
use gst::glib::prelude::*;
|
||||
use gst::subclass::prelude::*;
|
||||
use once_cell::sync::Lazy;
|
||||
use std::path::PathBuf;
|
||||
use std::sync::Mutex;
|
||||
use tokio::task;
|
||||
|
||||
use aws_config::default_provider::credentials::DefaultCredentialsChain;
|
||||
use aws_sdk_kinesisvideo::{
|
||||
model::{ChannelProtocol, ChannelRole, SingleMasterChannelEndpointConfiguration},
|
||||
Client, Endpoint,
|
||||
};
|
||||
use aws_sdk_kinesisvideosignaling::Client as SignalingClient;
|
||||
use aws_sig_auth::signer::{self, HttpSignatureType, OperationSigningConfig, RequestConfig};
|
||||
use aws_smithy_http::body::SdkBody;
|
||||
use aws_types::credentials::ProvideCredentials;
|
||||
use aws_types::region::{Region, SigningRegion};
|
||||
use aws_types::{Credentials, SigningService};
|
||||
use chrono::prelude::*;
|
||||
use data_encoding::BASE64;
|
||||
use http::Uri;
|
||||
use std::time::{Duration, SystemTime};
|
||||
|
||||
const DEFAULT_AWS_REGION: &str = "us-east-1";
|
||||
const DEFAULT_PING_TIMEOUT: i32 = 30;
|
||||
|
||||
static CAT: Lazy<gst::DebugCategory> = Lazy::new(|| {
|
||||
gst::DebugCategory::new(
|
||||
"webrtc-aws-kvs-signaller",
|
||||
gst::DebugColorFlags::empty(),
|
||||
Some("WebRTC AWS KVS signaller"),
|
||||
)
|
||||
});
|
||||
|
||||
#[derive(Default)]
|
||||
struct State {
|
||||
/// Sender for the websocket messages
|
||||
websocket_sender: Option<mpsc::Sender<p::OutgoingMessage>>,
|
||||
send_task_handle: Option<task::JoinHandle<Result<(), Error>>>,
|
||||
receive_task_handle: Option<task::JoinHandle<()>>,
|
||||
}
|
||||
|
||||
#[derive(Clone)]
|
||||
struct Settings {
|
||||
address: Option<String>,
|
||||
cafile: Option<PathBuf>,
|
||||
access_key: Option<String>,
|
||||
secret_access_key: Option<String>,
|
||||
session_token: Option<String>,
|
||||
channel_name: Option<String>,
|
||||
ping_timeout: i32,
|
||||
}
|
||||
|
||||
impl Default for Settings {
|
||||
fn default() -> Self {
|
||||
Self {
|
||||
address: Some("ws://127.0.0.1:8443".to_string()),
|
||||
cafile: None,
|
||||
access_key: None,
|
||||
secret_access_key: None,
|
||||
session_token: None,
|
||||
channel_name: None,
|
||||
ping_timeout: DEFAULT_PING_TIMEOUT,
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
#[derive(Default)]
|
||||
pub struct Signaller {
|
||||
state: Mutex<State>,
|
||||
settings: Mutex<Settings>,
|
||||
}
|
||||
|
||||
impl Signaller {
|
||||
fn handle_message(element: &WebRTCSink, msg: String) {
|
||||
if let Ok(msg) = serde_json::from_str::<p::IncomingMessage>(&msg) {
|
||||
match BASE64.decode(&msg.message_payload.into_bytes()) {
|
||||
Ok(payload) => {
|
||||
let payload = String::from_utf8_lossy(&payload);
|
||||
match msg.message_type.as_str() {
|
||||
"SDP_OFFER" => {
|
||||
if let Ok(sdp_msg) = serde_json::from_str::<p::SdpOffer>(&payload) {
|
||||
gst::log!(
|
||||
CAT,
|
||||
"Consumer {} got SDP offer: {}",
|
||||
msg.sender_client_id,
|
||||
sdp_msg.sdp
|
||||
);
|
||||
if let Err(err) = element.start_session(
|
||||
&msg.sender_client_id,
|
||||
&msg.sender_client_id,
|
||||
Some(&gst_webrtc::WebRTCSessionDescription::new(
|
||||
gst_webrtc::WebRTCSDPType::Offer,
|
||||
gst_sdp::SDPMessage::parse_buffer(sdp_msg.sdp.as_bytes())
|
||||
.unwrap(),
|
||||
)),
|
||||
) {
|
||||
gst::warning!(CAT, obj: element, "{err}");
|
||||
}
|
||||
} else {
|
||||
gst::warning!(
|
||||
CAT,
|
||||
obj: element,
|
||||
"Failed to parse SDP_OFFER: {payload}"
|
||||
);
|
||||
}
|
||||
}
|
||||
"ICE_CANDIDATE" => {
|
||||
if let Ok(ice_msg) = serde_json::from_str::<p::IceCandidate>(&payload) {
|
||||
gst::log!(
|
||||
CAT,
|
||||
"Consumer {} got candidate {} for m_line {} and mid {}",
|
||||
msg.sender_client_id,
|
||||
ice_msg.candidate,
|
||||
ice_msg.sdp_m_line_index,
|
||||
ice_msg.sdp_mid
|
||||
);
|
||||
if let Err(err) = element.handle_ice(
|
||||
&msg.sender_client_id,
|
||||
Some(ice_msg.sdp_m_line_index),
|
||||
Some(ice_msg.sdp_mid),
|
||||
&ice_msg.candidate,
|
||||
) {
|
||||
gst::warning!(CAT, obj: element, "{err}");
|
||||
}
|
||||
} else {
|
||||
gst::warning!(
|
||||
CAT,
|
||||
obj: element,
|
||||
"Failed to parse ICE_CANDIDATE: {payload}"
|
||||
);
|
||||
}
|
||||
}
|
||||
_ => {
|
||||
gst::log!(
|
||||
CAT,
|
||||
obj: element,
|
||||
"Ignoring unsupported message type {}",
|
||||
msg.message_type
|
||||
);
|
||||
}
|
||||
}
|
||||
}
|
||||
Err(e) => {
|
||||
gst::error!(
|
||||
CAT,
|
||||
obj: element,
|
||||
"Failed to decode message payload from server: {e}"
|
||||
);
|
||||
element.handle_signalling_error(
|
||||
anyhow!("Failed to decode message payload from server: {e}").into(),
|
||||
);
|
||||
}
|
||||
}
|
||||
} else {
|
||||
gst::log!(CAT, obj: element, "Unknown message from server: [{msg}]");
|
||||
}
|
||||
}
|
||||
|
||||
async fn connect(&self, element: &WebRTCSink) -> Result<(), Error> {
|
||||
let settings = self.settings.lock().unwrap().clone();
|
||||
|
||||
let connector = if let Some(path) = settings.cafile {
|
||||
let cert = tokio::fs::read_to_string(&path).await?;
|
||||
let cert = tokio_native_tls::native_tls::Certificate::from_pem(cert.as_bytes())?;
|
||||
let mut connector_builder = tokio_native_tls::native_tls::TlsConnector::builder();
|
||||
let connector = connector_builder.add_root_certificate(cert).build()?;
|
||||
Some(tokio_native_tls::TlsConnector::from(connector))
|
||||
} else {
|
||||
None
|
||||
};
|
||||
|
||||
let region = Region::new(DEFAULT_AWS_REGION);
|
||||
let access_key = settings.access_key.as_ref();
|
||||
let secret_access_key = settings.secret_access_key.as_ref();
|
||||
let session_token = settings.session_token.clone();
|
||||
|
||||
let credentials = match (access_key, secret_access_key) {
|
||||
(Some(key), Some(secret_key)) => {
|
||||
gst::debug!(
|
||||
CAT,
|
||||
imp: self,
|
||||
"Using provided access and secret access key"
|
||||
);
|
||||
Ok(Credentials::new(
|
||||
key.clone(),
|
||||
secret_key.clone(),
|
||||
session_token,
|
||||
None,
|
||||
"kvs",
|
||||
))
|
||||
}
|
||||
_ => {
|
||||
gst::debug!(CAT, imp: self, "Using default AWS credentials");
|
||||
let cred = DefaultCredentialsChain::builder()
|
||||
.region(region.clone())
|
||||
.build()
|
||||
.await;
|
||||
cred.provide_credentials().await
|
||||
}
|
||||
};
|
||||
|
||||
let credentials = match credentials {
|
||||
Err(e) => {
|
||||
anyhow::bail!("Failed to retrieve credentials with error {e}");
|
||||
}
|
||||
Ok(credentials) => credentials,
|
||||
};
|
||||
|
||||
let Some(channel_name) = settings.channel_name else { anyhow::bail!("Channel name cannot be None!"); };
|
||||
|
||||
let client = Client::new(
|
||||
&aws_config::from_env()
|
||||
.credentials_provider(credentials.clone())
|
||||
.load()
|
||||
.await,
|
||||
);
|
||||
|
||||
let resp = client
|
||||
.describe_signaling_channel()
|
||||
.set_channel_name(Some(channel_name.clone()))
|
||||
.send()
|
||||
.await?;
|
||||
|
||||
let Some(cinfo) = resp.channel_info() else { anyhow::bail!("No description found for {channel_name}"); };
|
||||
|
||||
gst::debug!(CAT, "Channel description: {cinfo:?}");
|
||||
|
||||
let Some(channel_arn) = cinfo.channel_arn() else { anyhow::bail!("No channel ARN found for {channel_name}"); };
|
||||
|
||||
let config = SingleMasterChannelEndpointConfiguration::builder()
|
||||
.set_protocols(Some(vec![ChannelProtocol::Wss, ChannelProtocol::Https]))
|
||||
.set_role(Some(ChannelRole::Master))
|
||||
.build();
|
||||
|
||||
let resp = client
|
||||
.get_signaling_channel_endpoint()
|
||||
.set_channel_arn(Some(channel_arn.to_string()))
|
||||
.set_single_master_channel_endpoint_configuration(Some(config))
|
||||
.send()
|
||||
.await?;
|
||||
|
||||
gst::debug!(CAT, "Endpoints: {:?}", resp.resource_endpoint_list());
|
||||
|
||||
let endpoint_wss_uri =
|
||||
match resp
|
||||
.resource_endpoint_list()
|
||||
.unwrap()
|
||||
.iter()
|
||||
.find_map(|endpoint| {
|
||||
if endpoint.protocol == Some(ChannelProtocol::Wss) {
|
||||
Some(endpoint.resource_endpoint().unwrap().to_owned())
|
||||
} else {
|
||||
None
|
||||
}
|
||||
}) {
|
||||
Some(endpoint_uri_str) => Uri::from_maybe_shared(endpoint_uri_str).unwrap(),
|
||||
None => {
|
||||
anyhow::bail!("No WSS endpoint found for {channel_name}");
|
||||
}
|
||||
};
|
||||
|
||||
let endpoint_https_uri =
|
||||
match resp
|
||||
.resource_endpoint_list()
|
||||
.unwrap()
|
||||
.iter()
|
||||
.find_map(|endpoint| {
|
||||
if endpoint.protocol == Some(ChannelProtocol::Https) {
|
||||
Some(endpoint.resource_endpoint().unwrap().to_owned())
|
||||
} else {
|
||||
None
|
||||
}
|
||||
}) {
|
||||
Some(endpoint_uri_str) => Uri::from_maybe_shared(endpoint_uri_str).unwrap(),
|
||||
None => {
|
||||
anyhow::bail!("No HTTPS endpoint found for {channel_name}");
|
||||
}
|
||||
};
|
||||
|
||||
gst::debug!(
|
||||
CAT,
|
||||
"Endpoints: {:?} {:?}",
|
||||
endpoint_wss_uri,
|
||||
endpoint_https_uri
|
||||
);
|
||||
|
||||
let signaling_config = aws_sdk_kinesisvideosignaling::config::Builder::from(
|
||||
&aws_config::from_env()
|
||||
.credentials_provider(credentials.clone())
|
||||
.load()
|
||||
.await,
|
||||
)
|
||||
.endpoint_resolver(Endpoint::immutable_uri(endpoint_https_uri.clone())?)
|
||||
.build();
|
||||
|
||||
let signaling_client = SignalingClient::from_conf(signaling_config);
|
||||
|
||||
let resp = signaling_client
|
||||
.get_ice_server_config()
|
||||
.set_channel_arn(Some(channel_arn.to_string()))
|
||||
.send()
|
||||
.await?;
|
||||
|
||||
let ice_servers: Vec<String> = resp
|
||||
.ice_server_list()
|
||||
.unwrap()
|
||||
.iter()
|
||||
.filter_map(|server| {
|
||||
Option::zip(server.username(), server.password())
|
||||
.map(|(username, password)| (username, password, server))
|
||||
})
|
||||
.flat_map(|(username, password, server)| {
|
||||
server
|
||||
.uris()
|
||||
.unwrap()
|
||||
.iter()
|
||||
.filter_map(move |uri| {
|
||||
uri.split_once(':').map(|(protocol, host)| {
|
||||
let (timestamp, username) = username.split_once(':').unwrap();
|
||||
|
||||
format!("{protocol}://{timestamp}%3A{encoded_user_name}:{encoded_password}@{host}",
|
||||
encoded_user_name=url_escape::encode_userinfo(username),
|
||||
encoded_password=url_escape::encode_userinfo(password),
|
||||
)
|
||||
})
|
||||
})
|
||||
})
|
||||
.collect();
|
||||
|
||||
gst::info!(CAT, "Ice servers: {:?}", ice_servers);
|
||||
|
||||
element.connect_closure(
|
||||
"consumer-added",
|
||||
false,
|
||||
glib::closure!(|_webrtcsink: &gst::Element,
|
||||
_consumer_identifier: &str,
|
||||
webrtcbin: &gst::Element| {
|
||||
webrtcbin.set_property(
|
||||
"stun-server",
|
||||
format!("stun://stun.kinesisvideo.{DEFAULT_AWS_REGION}.amazonaws.com:443"),
|
||||
);
|
||||
for ice_server in &ice_servers {
|
||||
let res = webrtcbin.emit_by_name::<bool>("add-turn-server", &[&ice_server]);
|
||||
gst::debug!(CAT, "Added ICE server {ice_server}, res: {res}");
|
||||
}
|
||||
}),
|
||||
);
|
||||
|
||||
let current_time = Utc::now();
|
||||
|
||||
let signer = signer::SigV4Signer::new();
|
||||
let mut operation_config = OperationSigningConfig::default_config();
|
||||
operation_config.signature_type = HttpSignatureType::HttpRequestQueryParams;
|
||||
operation_config.expires_in = Some(Duration::from_secs(5 * 60)); // See commit a3db85d.
|
||||
|
||||
let request_config = RequestConfig {
|
||||
request_ts: SystemTime::from(current_time),
|
||||
region: &SigningRegion::from(region.clone()),
|
||||
service: &SigningService::from_static("kinesisvideo"),
|
||||
payload_override: None,
|
||||
};
|
||||
|
||||
let transcribe_uri = Uri::builder()
|
||||
.scheme("wss")
|
||||
.authority(endpoint_wss_uri.authority().unwrap().to_owned())
|
||||
.path_and_query(format!(
|
||||
"/?X-Amz-ChannelARN={}",
|
||||
aws_smithy_http::query::fmt_string(channel_arn)
|
||||
))
|
||||
.build()
|
||||
.map_err(|err| {
|
||||
gst::error!(CAT, imp: self, "Failed to build HTTP request URI: {err}");
|
||||
anyhow!("Failed to build HTTP request URI: {err}")
|
||||
})?;
|
||||
let mut request = http::Request::builder()
|
||||
.uri(transcribe_uri)
|
||||
.body(SdkBody::empty())
|
||||
.expect("Failed to build valid request");
|
||||
|
||||
let _signature = signer
|
||||
.sign(
|
||||
&operation_config,
|
||||
&request_config,
|
||||
&credentials,
|
||||
&mut request,
|
||||
)
|
||||
.map_err(|err| {
|
||||
gst::error!(CAT, imp: self, "Failed to sign HTTP request: {err}");
|
||||
anyhow!("Failed to sign HTTP request: {err}")
|
||||
})?;
|
||||
|
||||
let url = request.uri().to_string();
|
||||
|
||||
gst::debug!(CAT, "Signed URL: {url}");
|
||||
|
||||
let (ws, _) =
|
||||
async_tungstenite::tokio::connect_async_with_tls_connector(url, connector).await?;
|
||||
|
||||
gst::info!(CAT, obj: element, "connected");
|
||||
|
||||
// Channel for asynchronously sending out websocket message
|
||||
let (mut ws_sink, mut ws_stream) = ws.split();
|
||||
|
||||
// 1000 is completely arbitrary, we simply don't want infinite piling
|
||||
// up of messages as with unbounded
|
||||
let (mut _websocket_sender, mut websocket_receiver) =
|
||||
mpsc::channel::<p::OutgoingMessage>(1000);
|
||||
let element_clone = element.downgrade();
|
||||
let ping_timeout = settings.ping_timeout;
|
||||
let send_task_handle = task::spawn(async move {
|
||||
loop {
|
||||
match tokio::time::timeout(
|
||||
std::time::Duration::from_secs(ping_timeout as u64),
|
||||
websocket_receiver.next(),
|
||||
)
|
||||
.await
|
||||
{
|
||||
Ok(Some(msg)) => {
|
||||
if let Some(element) = element_clone.upgrade() {
|
||||
gst::trace!(
|
||||
CAT,
|
||||
obj: element,
|
||||
"Sending websocket message {}",
|
||||
serde_json::to_string(&msg).unwrap()
|
||||
);
|
||||
}
|
||||
ws_sink
|
||||
.send(WsMessage::Text(serde_json::to_string(&msg).unwrap()))
|
||||
.await?;
|
||||
}
|
||||
Ok(None) => {
|
||||
break;
|
||||
}
|
||||
Err(_) => {
|
||||
ws_sink.send(WsMessage::Ping(vec![])).await?;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if let Some(element) = element_clone.upgrade() {
|
||||
gst::info!(CAT, obj: element, "Done sending");
|
||||
}
|
||||
|
||||
ws_sink.send(WsMessage::Close(None)).await?;
|
||||
ws_sink.close().await?;
|
||||
|
||||
Ok::<(), Error>(())
|
||||
});
|
||||
|
||||
let element_clone = element.downgrade();
|
||||
let receive_task_handle = task::spawn(async move {
|
||||
while let Some(msg) = tokio_stream::StreamExt::next(&mut ws_stream).await {
|
||||
if let Some(element) = element_clone.upgrade() {
|
||||
match msg {
|
||||
Ok(WsMessage::Text(msg)) => {
|
||||
gst::trace!(CAT, "received message [{msg}]");
|
||||
Signaller::handle_message(&element, msg);
|
||||
}
|
||||
Ok(WsMessage::Close(reason)) => {
|
||||
gst::info!(
|
||||
CAT,
|
||||
obj: element,
|
||||
"websocket connection closed: {:?}",
|
||||
reason
|
||||
);
|
||||
element.shutdown();
|
||||
break;
|
||||
}
|
||||
Ok(_) => (),
|
||||
Err(err) => {
|
||||
element
|
||||
.handle_signalling_error(anyhow!("Error receiving: {err}").into());
|
||||
break;
|
||||
}
|
||||
}
|
||||
} else {
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if let Some(element) = element_clone.upgrade() {
|
||||
gst::info!(CAT, obj: element, "Stopped websocket receiving");
|
||||
}
|
||||
});
|
||||
|
||||
let mut state = self.state.lock().unwrap();
|
||||
state.websocket_sender = Some(_websocket_sender);
|
||||
state.send_task_handle = Some(send_task_handle);
|
||||
state.receive_task_handle = Some(receive_task_handle);
|
||||
|
||||
Ok(())
|
||||
}
|
||||
|
||||
pub fn start(&self, element: &WebRTCSink) {
|
||||
let this = self.obj().clone();
|
||||
let element_clone = element.clone();
|
||||
task::spawn(async move {
|
||||
let this = this.imp();
|
||||
if let Err(err) = this.connect(&element_clone).await {
|
||||
element_clone.handle_signalling_error(err.into());
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
pub fn handle_sdp(
|
||||
&self,
|
||||
element: &WebRTCSink,
|
||||
session_id: &str,
|
||||
sdp: &gst_webrtc::WebRTCSessionDescription,
|
||||
) {
|
||||
let state = self.state.lock().unwrap();
|
||||
|
||||
let msg = p::OutgoingMessage {
|
||||
action: "SDP_ANSWER".to_string(),
|
||||
message_payload: BASE64.encode(
|
||||
&serde_json::to_string(&p::SdpAnswer {
|
||||
type_: "answer".to_string(),
|
||||
sdp: sdp.sdp().as_text().unwrap(),
|
||||
})
|
||||
.unwrap()
|
||||
.into_bytes(),
|
||||
),
|
||||
recipient_client_id: session_id.to_string(),
|
||||
};
|
||||
|
||||
if let Some(mut sender) = state.websocket_sender.clone() {
|
||||
let element = element.downgrade();
|
||||
RUNTIME.spawn(async move {
|
||||
if let Err(err) = sender.send(msg).await {
|
||||
if let Some(element) = element.upgrade() {
|
||||
element.handle_signalling_error(anyhow!("Error: {err}").into());
|
||||
}
|
||||
}
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
pub fn handle_ice(
|
||||
&self,
|
||||
element: &WebRTCSink,
|
||||
session_id: &str,
|
||||
candidate: &str,
|
||||
sdp_m_line_index: Option<u32>,
|
||||
_sdp_mid: Option<String>,
|
||||
) {
|
||||
let state = self.state.lock().unwrap();
|
||||
|
||||
let msg = p::OutgoingMessage {
|
||||
action: "ICE_CANDIDATE".to_string(),
|
||||
message_payload: BASE64.encode(
|
||||
&serde_json::to_string(&p::OutgoingIceCandidate {
|
||||
candidate: candidate.to_string(),
|
||||
sdp_mid: sdp_m_line_index.unwrap().to_string(),
|
||||
sdp_m_line_index: sdp_m_line_index.unwrap(),
|
||||
})
|
||||
.unwrap()
|
||||
.into_bytes(),
|
||||
),
|
||||
recipient_client_id: session_id.to_string(),
|
||||
};
|
||||
|
||||
if let Some(mut sender) = state.websocket_sender.clone() {
|
||||
let element = element.downgrade();
|
||||
RUNTIME.spawn(async move {
|
||||
if let Err(err) = sender.send(msg).await {
|
||||
if let Some(element) = element.upgrade() {
|
||||
element.handle_signalling_error(anyhow!("Error: {err}").into());
|
||||
}
|
||||
}
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
pub fn stop(&self, element: &WebRTCSink) {
|
||||
gst::info!(CAT, obj: element, "Stopping now");
|
||||
|
||||
let mut state = self.state.lock().unwrap();
|
||||
let send_task_handle = state.send_task_handle.take();
|
||||
let receive_task_handle = state.receive_task_handle.take();
|
||||
if let Some(mut sender) = state.websocket_sender.take() {
|
||||
let element = element.downgrade();
|
||||
RUNTIME.block_on(async move {
|
||||
sender.close_channel();
|
||||
|
||||
if let Some(handle) = send_task_handle {
|
||||
if let Err(err) = handle.await {
|
||||
if let Some(element) = element.upgrade() {
|
||||
gst::warning!(
|
||||
CAT,
|
||||
obj: element,
|
||||
"Error while joining send task: {err}"
|
||||
);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if let Some(handle) = receive_task_handle {
|
||||
if let Err(err) = handle.await {
|
||||
if let Some(element) = element.upgrade() {
|
||||
gst::warning!(
|
||||
CAT,
|
||||
obj: element,
|
||||
"Error while joining receive task: {err}"
|
||||
);
|
||||
}
|
||||
}
|
||||
}
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
pub fn end_session(&self, element: &WebRTCSink, session_id: &str) {
|
||||
gst::info!(CAT, obj: element, "Signalling session {session_id} ended");
|
||||
|
||||
// We can seemingly not do anything beyond that
|
||||
}
|
||||
}
|
||||
|
||||
#[glib::object_subclass]
|
||||
impl ObjectSubclass for Signaller {
|
||||
const NAME: &'static str = "GstAwsKvsWebRTCSinkSignaller";
|
||||
type Type = super::AwsKvsSignaller;
|
||||
type ParentType = glib::Object;
|
||||
}
|
||||
|
||||
impl ObjectImpl for Signaller {
|
||||
fn properties() -> &'static [glib::ParamSpec] {
|
||||
static PROPERTIES: Lazy<Vec<glib::ParamSpec>> = Lazy::new(|| {
|
||||
vec![
|
||||
glib::ParamSpecString::builder("address")
|
||||
.nick("Address")
|
||||
.blurb("Address of the signalling server")
|
||||
.default_value("ws://127.0.0.1:8443")
|
||||
.build(),
|
||||
glib::ParamSpecString::builder("cafile")
|
||||
.nick("CA file")
|
||||
.blurb("Path to a Certificate file to add to the set of roots the TLS connector will trust")
|
||||
.build(),
|
||||
glib::ParamSpecString::builder("channel-name")
|
||||
.nick("Channel name")
|
||||
.blurb("Name of the channel to connect as master to")
|
||||
.build(),
|
||||
glib::ParamSpecInt::builder("ping-timeout")
|
||||
.nick("Ping Timeout")
|
||||
.blurb("How often (in seconds) to send pings to keep the websocket alive")
|
||||
.default_value(DEFAULT_PING_TIMEOUT)
|
||||
.minimum(1)
|
||||
.build(),
|
||||
]
|
||||
});
|
||||
|
||||
PROPERTIES.as_ref()
|
||||
}
|
||||
|
||||
fn set_property(&self, _id: usize, value: &glib::Value, pspec: &glib::ParamSpec) {
|
||||
match pspec.name() {
|
||||
"address" => {
|
||||
let address: Option<_> = value.get().expect("type checked upstream");
|
||||
|
||||
if let Some(address) = address {
|
||||
gst::info!(CAT, "Signaller address set to {address}");
|
||||
|
||||
let mut settings = self.settings.lock().unwrap();
|
||||
settings.address = Some(address);
|
||||
} else {
|
||||
gst::error!(CAT, "address can't be None");
|
||||
}
|
||||
}
|
||||
"cafile" => {
|
||||
let value: String = value.get().unwrap();
|
||||
let mut settings = self.settings.lock().unwrap();
|
||||
settings.cafile = Some(value.into());
|
||||
}
|
||||
"access-key" => {
|
||||
let mut settings = self.settings.lock().unwrap();
|
||||
settings.access_key = value.get().expect("type checked upstream");
|
||||
}
|
||||
"secret-access-key" => {
|
||||
let mut settings = self.settings.lock().unwrap();
|
||||
settings.secret_access_key = value.get().expect("type checked upstream");
|
||||
}
|
||||
"session-token" => {
|
||||
let mut settings = self.settings.lock().unwrap();
|
||||
settings.session_token = value.get().expect("type checked upstream");
|
||||
}
|
||||
"channel-name" => {
|
||||
let mut settings = self.settings.lock().unwrap();
|
||||
settings.channel_name = value.get().expect("type checked upstream");
|
||||
}
|
||||
"ping-timeout" => {
|
||||
let mut settings = self.settings.lock().unwrap();
|
||||
settings.ping_timeout = value.get().expect("type checked upstream");
|
||||
}
|
||||
_ => unimplemented!(),
|
||||
}
|
||||
}
|
||||
|
||||
fn property(&self, _id: usize, pspec: &glib::ParamSpec) -> glib::Value {
|
||||
match pspec.name() {
|
||||
"address" => self.settings.lock().unwrap().address.to_value(),
|
||||
"cafile" => {
|
||||
let settings = self.settings.lock().unwrap();
|
||||
let cafile = settings.cafile.as_ref();
|
||||
cafile.and_then(|file| file.to_str()).to_value()
|
||||
}
|
||||
"access-key" => {
|
||||
let settings = self.settings.lock().unwrap();
|
||||
settings.access_key.to_value()
|
||||
}
|
||||
"secret-access-key" => {
|
||||
let settings = self.settings.lock().unwrap();
|
||||
settings.secret_access_key.to_value()
|
||||
}
|
||||
"session-token" => {
|
||||
let settings = self.settings.lock().unwrap();
|
||||
settings.session_token.to_value()
|
||||
}
|
||||
"channel-name" => self.settings.lock().unwrap().channel_name.to_value(),
|
||||
"ping-timeout" => self.settings.lock().unwrap().ping_timeout.to_value(),
|
||||
_ => unimplemented!(),
|
||||
}
|
||||
}
|
||||
}
|
65
net/webrtc/src/aws_kvs_signaller/mod.rs
Normal file
65
net/webrtc/src/aws_kvs_signaller/mod.rs
Normal file
|
@ -0,0 +1,65 @@
|
|||
// SPDX-License-Identifier: MPL-2.0
|
||||
|
||||
use crate::webrtcsink::{Signallable, WebRTCSink};
|
||||
use gst::glib;
|
||||
use gst::subclass::prelude::*;
|
||||
use std::error::Error;
|
||||
|
||||
mod imp;
|
||||
mod protocol;
|
||||
|
||||
glib::wrapper! {
|
||||
pub struct AwsKvsSignaller(ObjectSubclass<imp::Signaller>);
|
||||
}
|
||||
|
||||
unsafe impl Send for AwsKvsSignaller {}
|
||||
unsafe impl Sync for AwsKvsSignaller {}
|
||||
|
||||
impl Signallable for AwsKvsSignaller {
|
||||
fn start(&mut self, element: &WebRTCSink) -> Result<(), Box<dyn Error>> {
|
||||
let signaller = self.imp();
|
||||
signaller.start(element);
|
||||
|
||||
Ok(())
|
||||
}
|
||||
|
||||
fn handle_sdp(
|
||||
&mut self,
|
||||
element: &WebRTCSink,
|
||||
peer_id: &str,
|
||||
sdp: &gst_webrtc::WebRTCSessionDescription,
|
||||
) -> Result<(), Box<dyn Error>> {
|
||||
let signaller = self.imp();
|
||||
signaller.handle_sdp(element, peer_id, sdp);
|
||||
Ok(())
|
||||
}
|
||||
|
||||
fn handle_ice(
|
||||
&mut self,
|
||||
element: &WebRTCSink,
|
||||
session_id: &str,
|
||||
candidate: &str,
|
||||
sdp_mline_index: Option<u32>,
|
||||
sdp_mid: Option<String>,
|
||||
) -> Result<(), Box<dyn Error>> {
|
||||
let signaller = self.imp();
|
||||
signaller.handle_ice(element, session_id, candidate, sdp_mline_index, sdp_mid);
|
||||
Ok(())
|
||||
}
|
||||
|
||||
fn stop(&mut self, element: &WebRTCSink) {
|
||||
let signaller = self.imp();
|
||||
signaller.stop(element);
|
||||
}
|
||||
|
||||
fn session_ended(&mut self, element: &WebRTCSink, session_id: &str) {
|
||||
let signaller = self.imp();
|
||||
signaller.end_session(element, session_id);
|
||||
}
|
||||
}
|
||||
|
||||
impl Default for AwsKvsSignaller {
|
||||
fn default() -> Self {
|
||||
glib::Object::new()
|
||||
}
|
||||
}
|
53
net/webrtc/src/aws_kvs_signaller/protocol.rs
Normal file
53
net/webrtc/src/aws_kvs_signaller/protocol.rs
Normal file
|
@ -0,0 +1,53 @@
|
|||
// SPDX-License-Identifier: MPL-2.0
|
||||
|
||||
/// The default protocol used by the signalling server
|
||||
use serde::{Deserialize, Serialize};
|
||||
|
||||
#[derive(Serialize, Deserialize, Debug)]
|
||||
#[serde(rename_all = "camelCase")]
|
||||
pub struct SdpOffer {
|
||||
#[serde(rename = "type")]
|
||||
pub type_: String,
|
||||
pub sdp: String,
|
||||
}
|
||||
|
||||
#[derive(Serialize, Deserialize, Debug)]
|
||||
#[serde(rename_all = "camelCase")]
|
||||
pub struct IceCandidate {
|
||||
pub candidate: String,
|
||||
pub sdp_mid: String,
|
||||
pub sdp_m_line_index: u32,
|
||||
pub username_fragment: Option<String>,
|
||||
}
|
||||
|
||||
#[derive(Serialize, Deserialize, Debug)]
|
||||
#[serde(rename_all = "camelCase")]
|
||||
pub struct IncomingMessage {
|
||||
pub message_type: String,
|
||||
pub message_payload: String,
|
||||
pub sender_client_id: String,
|
||||
}
|
||||
|
||||
#[derive(Serialize, Deserialize, Debug)]
|
||||
#[serde(rename_all = "camelCase")]
|
||||
pub struct SdpAnswer {
|
||||
#[serde(rename = "type")]
|
||||
pub type_: String,
|
||||
pub sdp: String,
|
||||
}
|
||||
|
||||
#[derive(Serialize, Deserialize, Debug)]
|
||||
#[serde(rename_all = "camelCase")]
|
||||
pub struct OutgoingIceCandidate {
|
||||
pub candidate: String,
|
||||
pub sdp_mid: String,
|
||||
pub sdp_m_line_index: u32,
|
||||
}
|
||||
|
||||
#[derive(Serialize, Deserialize, Debug)]
|
||||
#[serde(rename_all = "camelCase")]
|
||||
pub struct OutgoingMessage {
|
||||
pub action: String,
|
||||
pub message_payload: String,
|
||||
pub recipient_client_id: String,
|
||||
}
|
|
@ -10,6 +10,7 @@ use gst::glib;
|
|||
use once_cell::sync::Lazy;
|
||||
use tokio::runtime;
|
||||
|
||||
mod aws_kvs_signaller;
|
||||
mod signaller;
|
||||
pub mod utils;
|
||||
pub mod webrtcsink;
|
||||
|
|
|
@ -139,7 +139,7 @@ impl Signaller {
|
|||
peer_id,
|
||||
} => {
|
||||
if let Err(err) =
|
||||
element.start_session(&session_id, &peer_id)
|
||||
element.start_session(&session_id, &peer_id, None)
|
||||
{
|
||||
gst::warning!(CAT, obj: element, "{}", err);
|
||||
}
|
||||
|
|
File diff suppressed because it is too large
Load diff
|
@ -18,8 +18,9 @@ glib::wrapper! {
|
|||
pub struct WebRTCSink(ObjectSubclass<imp::WebRTCSink>) @extends gst::Bin, gst::Element, gst::Object, @implements gst::ChildProxy, gst_video::Navigation;
|
||||
}
|
||||
|
||||
unsafe impl Send for WebRTCSink {}
|
||||
unsafe impl Sync for WebRTCSink {}
|
||||
glib::wrapper! {
|
||||
pub struct AwsKvsWebRTCSink(ObjectSubclass<imp::AwsKvsWebRTCSink>) @extends WebRTCSink, gst::Bin, gst::Element, gst::Object, @implements gst::ChildProxy, gst_video::Navigation;
|
||||
}
|
||||
|
||||
#[derive(thiserror::Error, Debug)]
|
||||
pub enum WebRTCSinkError {
|
||||
|
@ -79,12 +80,6 @@ pub trait SignallableObject: AsRef<glib::Object> + Signallable {}
|
|||
|
||||
impl<T: AsRef<glib::Object> + Signallable> SignallableObject for T {}
|
||||
|
||||
impl Default for WebRTCSink {
|
||||
fn default() -> Self {
|
||||
glib::Object::new()
|
||||
}
|
||||
}
|
||||
|
||||
impl WebRTCSink {
|
||||
pub fn with_signaller(signaller: Box<dyn SignallableObject>) -> Self {
|
||||
let ret = glib::Object::new::<WebRTCSink>();
|
||||
|
@ -123,9 +118,19 @@ impl WebRTCSink {
|
|||
ws.handle_signalling_error(self, anyhow::anyhow!(error));
|
||||
}
|
||||
|
||||
pub fn start_session(&self, session_id: &str, peer_id: &str) -> Result<(), WebRTCSinkError> {
|
||||
pub fn shutdown(&self) {
|
||||
let ws = self.imp();
|
||||
ws.start_session(self, session_id, peer_id)
|
||||
ws.shutdown(self);
|
||||
}
|
||||
|
||||
pub fn start_session(
|
||||
&self,
|
||||
session_id: &str,
|
||||
peer_id: &str,
|
||||
offer: Option<&gst_webrtc::WebRTCSessionDescription>,
|
||||
) -> Result<(), WebRTCSinkError> {
|
||||
let ws = self.imp();
|
||||
ws.start_session(self, session_id, peer_id, offer)
|
||||
}
|
||||
|
||||
pub fn end_session(&self, session_id: &str) -> Result<(), WebRTCSinkError> {
|
||||
|
@ -163,5 +168,13 @@ pub fn register(plugin: &gst::Plugin) -> Result<(), glib::BoolError> {
|
|||
"webrtcsink",
|
||||
gst::Rank::None,
|
||||
WebRTCSink::static_type(),
|
||||
)
|
||||
)?;
|
||||
gst::Element::register(
|
||||
Some(plugin),
|
||||
"awskvswebrtcsink",
|
||||
gst::Rank::None,
|
||||
AwsKvsWebRTCSink::static_type(),
|
||||
)?;
|
||||
|
||||
Ok(())
|
||||
}
|
||||
|
|
Loading…
Reference in a new issue