2021-10-05 21:28:05 +00:00
|
|
|
# webrtcsink
|
|
|
|
|
|
|
|
All-batteries included GStreamer WebRTC producer, that tries its best to do The Right Thing™.
|
|
|
|
|
|
|
|
## Use case
|
|
|
|
|
|
|
|
The [webrtcbin] element in GStreamer is extremely flexible and powerful, but using
|
|
|
|
it can be a difficult exercise. When all you want to do is serve a fixed set of streams
|
|
|
|
to any number of consumers, `webrtcsink` (which wraps `webrtcbin` internally) can be a
|
|
|
|
useful alternative.
|
|
|
|
|
|
|
|
[webrtcbin]: https://gstreamer.freedesktop.org/documentation/webrtc/index.html
|
|
|
|
|
|
|
|
## Features
|
|
|
|
|
|
|
|
`webrtcsink` implements the following features:
|
|
|
|
|
2022-03-03 03:30:44 +00:00
|
|
|
* Built-in signaller: when using the default signalling server, this element will
|
|
|
|
perform signalling without requiring application interaction.
|
|
|
|
This makes it usable directly from `gst-launch`.
|
2021-10-05 21:28:05 +00:00
|
|
|
|
|
|
|
* Application-provided signalling: `webrtcsink` can be instantiated by an application
|
|
|
|
with a custom signaller. That signaller must be a GObject, and must implement the
|
|
|
|
`Signallable` interface as defined [here](plugins/src/webrtcsink/mod.rs). The
|
|
|
|
[default signaller](plugins/src/signaller/mod.rs) can be used as an example.
|
|
|
|
|
2021-12-20 23:32:51 +00:00
|
|
|
An [example project] is also available to use as a boilerplate for implementing
|
|
|
|
and using a custom signaller.
|
|
|
|
|
2021-10-05 21:28:05 +00:00
|
|
|
* Sandboxed consumers: when a consumer is added, its encoder / payloader / webrtcbin
|
|
|
|
elements run in a separately managed pipeline. This provides a certain level of
|
|
|
|
sandboxing, as opposed to having those elements running inside the element itself.
|
|
|
|
|
|
|
|
It is important to note that at this moment, encoding is not shared between consumers.
|
|
|
|
While this is not on the roadmap at the moment, nothing in the design prevents
|
|
|
|
implementing this optimization.
|
|
|
|
|
2022-03-25 01:32:55 +00:00
|
|
|
* Congestion control: the element leverages transport-wide congestion control
|
2021-11-04 17:26:50 +00:00
|
|
|
feedback messages in order to adapt the bitrate of individual consumers' video
|
|
|
|
encoders to the available bandwidth.
|
|
|
|
|
2022-03-25 01:32:55 +00:00
|
|
|
* Configuration: the level of user control over the element is slowly expanding,
|
|
|
|
consult `gst-inspect-1.0` for more information on the available properties and
|
|
|
|
signals.
|
2021-10-05 21:28:05 +00:00
|
|
|
|
2022-03-25 01:32:55 +00:00
|
|
|
* Packet loss mitigation: webrtcsink now supports sending protection packets for
|
|
|
|
Forward Error Correction, modulating the amount as a function of the available
|
|
|
|
bandwidth, and can honor retransmission requests. Both features can be disabled
|
|
|
|
via properties.
|
2021-10-05 21:28:05 +00:00
|
|
|
|
|
|
|
It is important to note that full control over the individual elements used by
|
|
|
|
`webrtcsink` is *not* on the roadmap, as it will act as a black box in that respect,
|
|
|
|
for example `webrtcsink` wants to reserve control over the bitrate for congestion
|
|
|
|
control.
|
|
|
|
|
2022-03-25 01:32:55 +00:00
|
|
|
A signal is now available however for the application to provide the initial
|
|
|
|
configuration for the encoders `webrtcsink` instantiates.
|
|
|
|
|
2021-10-05 21:28:05 +00:00
|
|
|
If more granular control is required, applications should use `webrtcbin` directly,
|
|
|
|
`webrtcsink` will focus on trying to just do the right thing, although it might
|
2022-03-25 01:32:55 +00:00
|
|
|
expose more interfaces to guide and tune the heuristics it employs.
|
2021-10-05 21:28:05 +00:00
|
|
|
|
2021-12-20 23:32:51 +00:00
|
|
|
[example project]: https://github.com/centricular/webrtcsink-custom-signaller
|
|
|
|
|
2021-10-05 21:28:05 +00:00
|
|
|
## Building
|
|
|
|
|
|
|
|
### Prerequisites
|
|
|
|
|
2022-01-11 16:14:52 +00:00
|
|
|
The element has only been tested for now against GStreamer main.
|
2021-10-05 21:28:05 +00:00
|
|
|
|
|
|
|
For testing, it is recommended to simply build GStreamer locally and run
|
|
|
|
in the uninstalled devenv.
|
|
|
|
|
|
|
|
> Make sure to install the development packages for some codec libraries
|
|
|
|
> beforehand, such as libx264, libvpx and libopusenc, exact names depend
|
|
|
|
> on your distribution.
|
|
|
|
|
|
|
|
```
|
2022-01-11 16:14:52 +00:00
|
|
|
git clone --depth 1 --single-branch --branch main https://gitlab.freedesktop.org/gstreamer/gstreamer
|
|
|
|
cd gstreamer
|
2021-10-05 21:28:05 +00:00
|
|
|
meson build
|
|
|
|
ninja -C build
|
|
|
|
ninja -C build devenv
|
|
|
|
```
|
|
|
|
|
|
|
|
### Compiling
|
|
|
|
|
|
|
|
``` shell
|
|
|
|
cargo build
|
|
|
|
```
|
|
|
|
|
|
|
|
## Usage
|
|
|
|
|
|
|
|
Open three terminals. In the first, run:
|
|
|
|
|
|
|
|
``` shell
|
2022-03-03 03:30:44 +00:00
|
|
|
WEBRTCSINK_SIGNALLING_SERVER_LOG=debug cargo run --bin server
|
2021-10-05 21:28:05 +00:00
|
|
|
```
|
|
|
|
|
|
|
|
In the second, run:
|
|
|
|
|
|
|
|
``` shell
|
2022-04-28 09:35:08 +00:00
|
|
|
python3 -m http.server -d www/
|
2021-10-05 21:28:05 +00:00
|
|
|
```
|
|
|
|
|
|
|
|
In the third, run:
|
|
|
|
|
2021-11-04 17:26:50 +00:00
|
|
|
``` shell
|
2021-10-05 21:28:05 +00:00
|
|
|
export GST_PLUGIN_PATH=$PWD/target/debug:$GST_PLUGIN_PATH
|
|
|
|
gst-launch-1.0 webrtcsink name=ws videotestsrc ! ws. audiotestsrc ! ws.
|
|
|
|
```
|
|
|
|
|
|
|
|
When the pipeline above is running succesfully, open a browser and
|
2022-03-03 03:30:44 +00:00
|
|
|
point it to the http server:
|
2021-10-05 21:28:05 +00:00
|
|
|
|
2021-11-04 17:26:50 +00:00
|
|
|
``` shell
|
2021-10-05 21:28:05 +00:00
|
|
|
xdg-open http://127.0.0.1:8000
|
|
|
|
```
|
|
|
|
|
|
|
|
You should see an identifier listed in the left-hand panel, click on
|
|
|
|
it. You should see a test video stream, and hear a test tone.
|
|
|
|
|
2021-11-04 17:26:50 +00:00
|
|
|
## Configuration
|
|
|
|
|
|
|
|
The element itself can be configured through its properties, see
|
|
|
|
`gst-inspect-1.0 webrtcsink` for more information about that, in addition the
|
|
|
|
default signaller also exposes properties for configuring it, in
|
|
|
|
particular setting the signalling server address, those properties
|
|
|
|
can be accessed through the `gst::ChildProxy` interface, for example
|
|
|
|
with gst-launch:
|
|
|
|
|
|
|
|
``` shell
|
|
|
|
gst-launch-1.0 webrtcsink signaller::address="ws://127.0.0.1:8443" ..
|
|
|
|
```
|
|
|
|
|
|
|
|
The signaller object can not be inspected, refer to [the source code]
|
|
|
|
for the list of properties.
|
|
|
|
|
|
|
|
[the source code]: plugins/src/signaller/imp.rs
|
|
|
|
|
2021-12-24 12:26:26 +00:00
|
|
|
|
|
|
|
### Enable 'navigation' a.k.a user interactivity with the content
|
|
|
|
|
|
|
|
`webrtcsink` implements the [`GstNavigation`] interface which allows interacting
|
|
|
|
with the content, for example move with your mouse, entering keys with the
|
|
|
|
keyboard, etc... On top of that a `WebRTCDataChannel` based protocol has been
|
|
|
|
implemented and can be activated with the `enable-data-channel-navigation=true`
|
|
|
|
property. The [demo](www/) implements the protocol and you can easily test this
|
|
|
|
feature, using the [`wpesrc`] for example.
|
|
|
|
|
|
|
|
As an example, the following pipeline allows you to navigate the GStreamer
|
|
|
|
documentation inside the video running within your web browser (in
|
|
|
|
http://127.0.0.1:8000 if you followed previous steps of that readme):
|
|
|
|
|
|
|
|
```
|
|
|
|
gst-launch-1.0 wpesrc location=https://gstreamer.freedesktop.org/documentation/ ! webrtcsink enable-data-channel-navigation=true
|
|
|
|
```
|
|
|
|
|
|
|
|
[`GstNavigation`]: https://gstreamer.freedesktop.org/documentation/video/gstnavigation.html
|
|
|
|
[`wpesrc`]: https://gstreamer.freedesktop.org/documentation/wpe/wpesrc.html
|
|
|
|
|
2021-11-04 17:26:50 +00:00
|
|
|
## Testing congestion control
|
|
|
|
|
|
|
|
For the purpose of testing congestion in a reproducible manner, a
|
|
|
|
[simple tool] has been used, I only used it on Linux but it is documented
|
|
|
|
as usable on MacOS too. I had to run the client browser on a separate
|
|
|
|
machine on my local network for congestion to actually be applied, I didn't
|
|
|
|
look into why that was necessary.
|
|
|
|
|
|
|
|
My testing procedure was:
|
|
|
|
|
|
|
|
* identify the server machine network interface (eg with `ifconfig` on Linux)
|
|
|
|
|
|
|
|
* identify the client machine IP address (eg with `ifconfig` on Linux)
|
|
|
|
|
|
|
|
* start the various services as explained in the Usage section (use
|
|
|
|
`GST_DEBUG=webrtcsink:7` to get detailed logs about congestion control)
|
|
|
|
|
|
|
|
* start playback in the client browser
|
|
|
|
|
|
|
|
* Run a `comcast` command on the server machine, for instance:
|
|
|
|
|
|
|
|
``` shell
|
|
|
|
/home/meh/go/bin/comcast --device=$SERVER_INTERFACE --target-bw 3000 --target-addr=$CLIENT_IP --target-port=1:65535 --target-proto=udp
|
|
|
|
```
|
|
|
|
|
|
|
|
* Observe the bitrate sharply decreasing, playback should slow down briefly
|
|
|
|
then catch back up
|
|
|
|
|
|
|
|
* Remove the bandwidth limitation, and observe the bitrate eventually increasing
|
|
|
|
back to a maximum:
|
|
|
|
|
|
|
|
``` shell
|
|
|
|
/home/meh/go/bin/comcast --device=$SERVER_INTERFACE --stop
|
|
|
|
```
|
|
|
|
|
|
|
|
For comparison, the congestion control property can be set to disabled on
|
|
|
|
webrtcsink, then the above procedure applied again, the expected result is
|
|
|
|
for playback to simply crawl down to a halt until the bandwidth limitation
|
|
|
|
is lifted:
|
|
|
|
|
|
|
|
``` shell
|
|
|
|
gst-launch-1.0 webrtcsink congestion-control=disabled
|
|
|
|
```
|
|
|
|
|
|
|
|
[simple tool]: https://github.com/tylertreat/comcast
|
|
|
|
|
2021-12-09 23:06:46 +00:00
|
|
|
## Monitoring tool
|
|
|
|
|
|
|
|
An example server / client application for monitoring per-consumer stats
|
|
|
|
can be found [here].
|
|
|
|
|
|
|
|
[here]: plugins/examples/README.md
|
|
|
|
|
2021-10-05 21:28:05 +00:00
|
|
|
## License
|
|
|
|
|
|
|
|
All code in this repository is licensed under the [MIT license].
|
|
|
|
|
|
|
|
[MIT license]: https://opensource.org/licenses/MIT
|