Skip to content

Commit

Permalink
Add Introduction to Elixir WebRTC tutorial (#136)
Browse files Browse the repository at this point in the history
  • Loading branch information
LVala authored Jul 22, 2024
1 parent ef355c0 commit 9e18881
Show file tree
Hide file tree
Showing 9 changed files with 714 additions and 1 deletion.
1 change: 1 addition & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@ end
## Getting started

To get started with Elixir WebRTC, check out:
* the [Introduction to Elixir Webrtc](https://hexdocs.pm/ex_webrtc/intro.html) tutorial
* the [examples directory](https://github.com/elixir-webrtc/ex_webrtc/tree/master/examples) that contains a bunch of very simple usage examples of the library
* the [`apps` repo](https://github.com/elixir-webrtc/apps) with example applications built on top of `ex_webrtc`
* the [documentation](https://hexdocs.pm/ex_webrtc/readme.html), especially the [`PeerConnection` module page](https://hexdocs.pm/ex_webrtc/ExWebRTC.PeerConnection.html)
Expand Down
File renamed without changes.
27 changes: 27 additions & 0 deletions guides/introduction/consuming.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
# Consuming media data

Other than just forwarding, we probably would like to be able to use the media right in the Elixir app to
e..g feed it to a machine learning model or create a recording of a meeting.

In this tutorial, we are going to build on top of the simple app from the previous tutorial by, instead of just sending the packets back, depayloading and decoding
the media, using a machine learning model to somehow augment the video, encode and payload it back into RTP packets and only then send it to the web browser.

## Deplayloading RTP

We refer to the process of taking the media payload out of RTP packets as _depayloading_.

> #### Codecs {: .info}
> A media codec is a program used to encode/decode digital video and audio streams. Codecs also compress the media data,
> otherwise, it would be too big to send over the network (bitrate of raw 24-bit color depth, FullHD, 60 fps video is about 3 Gbit/s!).
>
> In WebRTC, most likely you will encounter VP8, H264 or AV1 video codecs and Opus audio codec. Codecs that will be used during the session are negotiated in
> the SDP offer/answer exchange. You can tell what codec is carried in an RTP packet by inspecting its payload type (`packet.payload_type`,
> a non-negative integer field) and match it with one of the codecs listed in this track's transceiver's `codecs` field (you have to find
> the `transceiver` by iterating over `PeerConnection.get_transceivers` as shown previously in this tutorial series).
_TBD_

## Decoding the media to raw format

_TBD_

195 changes: 195 additions & 0 deletions guides/introduction/forwarding.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,195 @@
# Forwarding media data

Elixir WebRTC, in contrast to the JavaScript API, provides you with the actual media data transmitted via WebRTC.
That means you can be much more flexible with what you do with the data, but you also need to know a bit more
about how WebRTC actually works under the hood.

All of the media data received by the `PeerConnection` is sent to the user in the form of messages like this:

```elixir
receive do
{:ex_webrtc, ^pc, {:rtp, track_id, _rid, packet}} ->
# do something with the packet
# also, for now, you can assume that _rid is always nil and ignore it
end
```

The `track_id` corresponds to one of the tracks that we received in `{:ex_webrtc, _from, {:track, %MediaStreamTrack{id: track_id}}}` messages.
The `packet` is an RTP packet. It contains the media data alongside some other useful information.

> #### The RTP protocol {: .info}
> RTP is a network protocol created for carrying real-time data (like media) and is used by WebRTC.
> It provides some useful features like:
>
> * sequence numbers: UDP (which is usually used by WebRTC) does not provide ordering, thus we need this to catch missing or out-of-order packets
> * timestamp: these can be used to correctly play the media back to the user (e.g. using the right framerate for the video)
> * payload type: thanks to this combined with information in the SDP offer/answer, we can tell which codec is carried by this packet
>
> and many more. Check out the [RFC 3550](https://datatracker.ietf.org/doc/html/rfc3550) to learn more about RTP.
Next, we will learn what you can do with the RTP packets.
For now, we won't actually look into the packets themselves, our goal for this part of the tutorial will be to forward the received data back to the same web browser.

```mermaid
flowchart LR
subgraph Elixir
PC[PeerConnection] --> Forwarder --> PC
end
WB((Web Browser)) <-.-> PC
```

The only thing we have to implement is the `Forwarder` GenServer. Let's combine the ideas from the previous section to write it.

```elixir
defmodule Forwarder do
use GenServer

alias ExWebRTC.{PeerConnection, ICEAgent, MediaStreamTrack, SessionDescription}

@ice_servers [%{urls: "stun:stun.l.google.com:19302"}]

@impl true
def init(_) do
{:ok, pc} = PeerConnection.start_link(ice_servers: @ice_servers)

# we expect to receive two tracks from the web browser - one for audio, one for video
# so we also need to add two tracks here, we will use these to forward media
# from each of the web browser tracks
stream_id = MediaStreamTrack.generate_stream_id()
audio_track = MediaStreamTrack.new(:audio, [stream_id])
video_track = MediaStreamTrack.new(:video, [stream_id])

{:ok, _sender} = PeerConnection.add_track(pc, audio_track)
{:ok, _sender} = PeerConnection.add_track(pc, video_track)

# in_tracks (tracks we will receive media from) = %{id => kind}
# out_tracks (tracks we will send media to) = %{kind => id}
out_tracks = %{audio: audio_track.id, video: video_track.id}
{:ok, %{pc: pc, out_tracks: out_tracks, in_tracks: %{}}}
end

# ...
end
```

We started by creating the PeerConnection and adding two tracks (one for audio and one for video).
Remember that these tracks will be used to *send* data to the web browser peer. Remote tracks (the ones we will set up on the JavaScript side, like in the previous tutorial)
will arrive as messages after the negotiation is completed.

> #### Where are the tracks? {: .tip}
> In the context of Elixir WebRTC, a track is simply a _track id_, _ids_ of streams this track belongs to, and a _kind_ (audio/video).
> We can either add tracks to the PeerConnection (these tracks will be used to *send* data when calling `PeerConnection.send_rtp/4` and
> for each one of the tracks, the remote peer should fire the `track` event)
> or handle remote tracks (which you are notified about with messages from the PeerConnection process: `{:ex_webrtc, _from, {:track, track}}`).
> These are used when handling messages with RTP packets: `{:ex_webrtc, _from, {:rtp, _rid, track_id, packet}}`.
> You cannot use the same track to send AND receive, keep that in mind.
>
> Alternatively, all of the tracks can be obtained by iterating over the transceivers:
>
> ```elixir
> tracks =
> peer_connection
> |> PeerConnection.get_transceivers()
> |> Enum.map(&(&1.receiver.track))
> ```
>
> If you want to know more about transceivers, read the [Mastering Transceivers](https://hexdocs.pm/ex_webrtc/mastering_transceivers.html) guide.
Next, we need to take care of the offer/answer and ICE candidate exchange. As in the previous tutorial, we assume that there's some kind
of WebSocket relay service available that will forward our offer/answer/candidate messages to the web browser and back to us.
```elixir
@impl true
def handle_info({:web_socket, {:offer, offer}}, state) do
:ok = PeerConnection.set_remote_description(state.pc, offer)
{:ok, answer} = PeerConnection.create_answer(state.pc)
:ok = PeerConnection.set_local_description(state.pc, answer)
web_socket_send(answer)
{:noreply, state}
end
@impl true
def handle_info({:web_socket, {:ice_candidate, cand}}, state) do
:ok = PeerConnection.add_ice_candidate(state.pc, cand)
{:noreply, state}
end
@impl true
def handle_info({:ex_webrtc, _from, {:ice_candidate, cand}}, state) do
web_socket_send(cand)
{:noreply, state}
end
```
Now we can expect to receive messages with notifications about new remote tracks.
Let's handle these and match them with the tracks that we are going to send to.
We need to be careful not to send packets from the audio track on a video track by mistake!
```elixir
@impl true
def handle_info({:ex_webrtc, _from, {:track, track}}, state) do
state = put_in(state.in_tracks[track.id], track.kind)
{:noreply, state}
end
```
We are ready to handle the incoming RTP packets!

```elixir
@impl true
def handle_info({:ex_webrtc, _from, {:rtp, track_id, nil, packet}}, state) do
kind = Map.fetch!(state.in_tracks, track_id)
id = Map.fetch!(state.out_tracks, kind)
:ok = PeerConnection.send_rtp(state.pc, id, packet)

{:noreply, state}
end
```

> #### RTP packet rewriting {: .info}
> In the example above we just receive the RTP packet and immediately send it back. In reality, a lot of stuff in the packet header must be rewritten.
> That includes SSRC (a number that identifies to which stream the packet belongs), payload type (indicates the codec, even though the codec does not
> change between two tracks, the payload types are dynamically assigned and may differ between RTP sessions), and some RTP header extensions. All of that is
> done by Elixir WebRTC behind the scenes, but be aware - it is not as simple as forwarding the same piece of data!
Lastly, let's take care of the client-side code. It's nearly identical to what we have written in the previous tutorial.

```js
const localStream = await navigator.mediaDevices.getUserMedia({audio: true, video: true});
const pc = new RTCPeerConnection({iceServers: [{urls: "stun:stun.l.google.com:19302"}]});
localStream.getTracks().forEach(track => pc.addTrack(track, localStream));

// these will be the tracks that we added using `PeerConnection.add_track`
pc.ontrack = event => videoPlayer.srcObject = event.stream[0];

// sending/receiving the offer/answer/candidates to the other peer is your responsibility
pc.onicecandidate = event => send_to_other_peer(event.candidate);
on_cand_received(cand => pc.addIceCandidate(cand));

// remember that we set up the Elixir app to just handle the incoming offer
// so we need to generate and send it (and thus, start the negotiation) here
const offer = await pc.createOffer();
await pc.setLocalDescription(offer)
send_offer_to_other_peer(offer);

const answer = await receive_answer_from_other_peer();
await pc.setRemoteDescription(answer);
```

And that's it! The other peer should be able to see and hear the echoed video and audio.

> #### PeerConnection state {: .info}
> Before we can send anything on a PeerConnection, its state must change to `connected` which is signaled
> by the `{:ex_webrtc, _from, {:connection_state_change, :connected}}` message. In this particular example, we want
> to send packets on the very same PeerConnection that we received the packets from, thus it must be connected
> from the first RTP packet received.
What you've seen here is a simplified version of the [echo](https://github.com/elixir-webrtc/ex_webrtc/tree/master/examples/echo) example available
in the Elixir WebRTC Github repo. Check it out and play with it!

Now, you might be thinking that forwarding the media back to the same web browser does not seem very useful, and you're probably right!
But thankfully, you can use the gained knowledge to build more complex apps.

In the next part of the tutorial, we will learn how to actually do something with media data in the Elixir app.
33 changes: 33 additions & 0 deletions guides/introduction/intro.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
# Introduction to WebRTC

In this series of tutorials, we are going to learn what is WebRTC, and go through some simple use cases of Elixir WebRTC.
Its purpose is to teach you where you'd want to use WebRTC, show you what the WebRTC API looks like, and how it should
be used, focusing on some common caveats.

> #### Before You Start {: .info}
> This guide assumes little prior knowledge of the WebRTC API, but it would be highly beneficial
> to go through the [MDN WebRTC tutorial](https://developer.mozilla.org/en-US/docs/Web/API/WebRTC_API)
> as the Elixir API tries to closely mimic the browser JavaScript API.
## What is WebRTC

WebRTC is an open, real-time communication standard that allows you to send video, audio, and generic data between peers over the network.
It places a lot of emphasis on low latency (targeting values in low hundreds of milliseconds end-to-end) and was designed to be used peer-to-peer.

WebRTC is implemented by all of the major web browsers and is available as a JavaScript API, there's also native WebRTC clients for Android and iOS
and implementation in other programming languages ([Pion](https://github.com/pion/webrtc), [webrtc.rs](https://github.com/webrtc-rs/webrtc),
and now [Elixir WebRTC](https://github.com/elixir-webrtc/ex_webrtc)).

## Where would you use WebRTC

WebRTC is the obvious choice in applications where low latency is important. It's also probably the easiest way to obtain the voice and video from a user of
your web application. Here are some example use cases:

* videoconferencing apps (one-on-one meetings of fully fledged meeting rooms, like Microsoft Teams or Google Meet)
* ingress for broadcasting services (as a presenter, you can use WebRTC to get media to a server, which will then broadcast it to viewers using WebRTC or different protocols)
* obtaining voice and video from web app users to use it for machine learning model inference on the back end.

In general, all of the use cases come down to getting media from one peer to another. In the case of Elixir WebRTC, one of the peers is usually a server,
like your Phoenix app (although it doesn't have to - there's no concept of server/client in WebRTC, so you might as well connect two browsers or two Elixir peers).

This is what the next section of this tutorial series will focus on - we will try to get media from a web browser to a simple Elixir app.
Loading

0 comments on commit 9e18881

Please sign in to comment.