From 363500a0d4a3f71be86f2dc7efedcd8957d1a86b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=C5=81ukasz=20Wala?= <84684231+LVala@users.noreply.github.com> Date: Tue, 28 Nov 2023 12:10:43 +0100 Subject: [PATCH] Allow for sending data (#21) --- examples/example.exs | 65 +++++---- examples/example.js | 23 +-- lib/ex_webrtc/dtls_transport.ex | 75 +++++++--- lib/ex_webrtc/media_stream_track.ex | 9 +- lib/ex_webrtc/peer_connection.ex | 185 +++++++++++++++++------- lib/ex_webrtc/rtp_receiver.ex | 9 +- lib/ex_webrtc/rtp_sender.ex | 13 ++ lib/ex_webrtc/rtp_transceiver.ex | 25 ++-- mix.lock | 2 +- test/ex_webrtc/dtls_transport_test.exs | 2 +- test/ex_webrtc/peer_connection_test.exs | 4 +- 11 files changed, 277 insertions(+), 135 deletions(-) create mode 100644 lib/ex_webrtc/rtp_sender.ex diff --git a/examples/example.exs b/examples/example.exs index 40d58cb2..39265717 100644 --- a/examples/example.exs +++ b/examples/example.exs @@ -8,19 +8,19 @@ defmodule Peer do require Logger - alias ExWebRTC.{IceCandidate, PeerConnection, SessionDescription} + alias ExWebRTC.{IceCandidate, PeerConnection, SessionDescription, RTPTransceiver} @ice_servers [ # %{urls: "stun:stun.stunprotocol.org:3478"}, %{urls: "stun:stun.l.google.com:19302"} ] - def start_link(mode \\ :passive) do - GenServer.start_link(__MODULE__, mode) + def start_link() do + GenServer.start_link(__MODULE__, nil) end @impl true - def init(mode) do + def init(_) do {:ok, conn} = :gun.open({127, 0, 0, 1}, 4000) {:ok, _protocol} = :gun.await_up(conn) :gun.ws_upgrade(conn, "/websocket") @@ -32,16 +32,7 @@ defmodule Peer do {:ok, pc} = PeerConnection.start_link(ice_servers: @ice_servers) - if mode == :active do - {:ok, _transceiver} = PeerConnection.add_transceiver(pc, :audio) - {:ok, offer} = PeerConnection.create_offer(pc) - :ok = PeerConnection.set_local_description(pc, offer) - msg = %{"type" => "offer", "sdp" => offer.sdp} - :gun.ws_send(conn, stream, {:text, Jason.encode!(msg)}) - Logger.info("Send SDP offer: #{offer.sdp}") - end - - {:ok, %{conn: conn, stream: stream, peer_connection: pc, mode: mode}} + {:ok, %{conn: conn, stream: stream, peer_connection: pc, track_id: nil}} other -> Logger.error("Couldn't connect to the signalling server: #{inspect(other)}") @@ -64,9 +55,10 @@ defmodule Peer do @impl true def handle_info({:gun_ws, _, _, {:text, msg}}, state) do - msg - |> Jason.decode!() - |> handle_ws_message(state) + state = + msg + |> Jason.decode!() + |> handle_ws_message(state) {:noreply, state} end @@ -79,9 +71,7 @@ defmodule Peer do @impl true def handle_info({:ex_webrtc, _pid, msg}, state) do - Logger.info("Received ExWebRTC message: #{inspect(msg)}") handle_webrtc_message(msg, state) - {:noreply, state} end @@ -91,20 +81,30 @@ defmodule Peer do {:noreply, state} end - defp handle_ws_message(%{"type" => "offer", "sdp" => sdp}, %{mode: :passive} = state) do + defp handle_ws_message(%{"type" => "offer", "sdp" => sdp}, %{peer_connection: pc} = state) do Logger.info("Received SDP offer: #{inspect(sdp)}") offer = %SessionDescription{type: :offer, sdp: sdp} - :ok = PeerConnection.set_remote_description(state.peer_connection, offer) - {:ok, answer} = PeerConnection.create_answer(state.peer_connection) - :ok = PeerConnection.set_local_description(state.peer_connection, answer) + :ok = PeerConnection.set_remote_description(pc, offer) + {:ok, answer} = PeerConnection.create_answer(pc) + :ok = PeerConnection.set_local_description(pc, answer) msg = %{"type" => "answer", "sdp" => answer.sdp} :gun.ws_send(state.conn, state.stream, {:text, Jason.encode!(msg)}) + + track = ExWebRTC.MediaStreamTrack.new(:video) + {:ok, _} = PeerConnection.add_transceiver(pc, track) + {:ok, offer} = PeerConnection.create_offer(pc) + :ok = PeerConnection.set_local_description(pc, offer) + msg = %{"type" => "offer", "sdp" => offer.sdp} + :gun.ws_send(state.conn, state.stream, {:text, Jason.encode!(msg)}) + + %{state | track_id: track.id} end - defp handle_ws_message(%{"type" => "answer", "sdp" => sdp}, %{mode: :active} = state) do + defp handle_ws_message(%{"type" => "answer", "sdp" => sdp}, state) do Logger.info("Received SDP answer: #{inspect(sdp)}") answer = %SessionDescription{type: :answer, sdp: sdp} :ok = PeerConnection.set_remote_description(state.peer_connection, answer) + state end defp handle_ws_message(%{"type" => "ice", "data" => data}, state) do @@ -118,10 +118,13 @@ defmodule Peer do } :ok = PeerConnection.add_ice_candidate(state.peer_connection, candidate) + + state end - defp handle_ws_message(msg, _state) do + defp handle_ws_message(msg, state) do Logger.info("Received unexpected message: #{inspect(msg)}") + state end defp handle_webrtc_message({:ice_candidate, candidate}, state) do @@ -136,13 +139,21 @@ defmodule Peer do :gun.ws_send(state.conn, state.stream, {:text, Jason.encode!(msg)}) end + defp handle_webrtc_message({:rtp, _mid, _packet}, %{track_id: nil}) do + Logger.warning("Received RTP, but out transceiver has not beed created") + end + + defp handle_webrtc_message({:rtp, _mid, packet}, state) do + Logger.info("Received RTP: #{inspect(packet)}") + PeerConnection.send_rtp(state.peer_connection, state.track_id, packet) + end + defp handle_webrtc_message(msg, _state) do Logger.warning("Received unknown ex_webrtc message: #{inspect(msg)}") end end -mode = :active -{:ok, pid} = Peer.start_link(mode) +{:ok, pid} = Peer.start_link() ref = Process.monitor(pid) receive do diff --git a/examples/example.js b/examples/example.js index ad63e977..2cb579f0 100644 --- a/examples/example.js +++ b/examples/example.js @@ -13,7 +13,14 @@ const start_connection = async (ws) => { pc.oniceconnectionstatechange = _ => console.log("ICE connection state changed:", pc.iceConnectionState); pc.onicegatheringstatechange = _ => console.log("ICE gathering state changed:", pc.iceGatheringState); pc.onsignalingstatechange = _ => console.log("Signaling state changed:", pc.signalingState); - pc.ontrack = event => console.log("New track:", event); + pc.ontrack = event => { + const videoPlayer = document.createElement("video"); + videoPlayer.srcObject = event.streams[0]; + videoPlayer.onloadedmetadata = () => { + videoPlayer.play(); + }; + document.body.appendChild(videoPlayer); + }; pc.onicecandidate = event => { console.log("New local ICE candidate:", event.candidate); @@ -22,7 +29,7 @@ const start_connection = async (ws) => { } }; - const localStream = await navigator.mediaDevices.getUserMedia({audio: true}); + const localStream = await navigator.mediaDevices.getUserMedia({video: true}); for (const track of localStream.getTracks()) { pc.addTrack(track, localStream); } @@ -48,16 +55,12 @@ const start_connection = async (ws) => { } }; - if (mode === "active") { - const desc = await pc.createOffer(); - console.log("Generated SDP offer:", desc); - await pc.setLocalDescription(desc); - ws.send(JSON.stringify(desc)) - } + const desc = await pc.createOffer(); + console.log("Generated SDP offer:", desc); + await pc.setLocalDescription(desc); + ws.send(JSON.stringify(desc)) }; -const mode = "passive" - const ws = new WebSocket("ws://127.0.0.1:4000/websocket"); ws.onclose = event => console.log("WebSocket was closed", event); ws.onopen = _ => start_connection(ws); diff --git a/lib/ex_webrtc/dtls_transport.ex b/lib/ex_webrtc/dtls_transport.ex index 4b80f1d2..2fa1148e 100644 --- a/lib/ex_webrtc/dtls_transport.ex +++ b/lib/ex_webrtc/dtls_transport.ex @@ -56,9 +56,15 @@ defmodule ExWebRTC.DTLSTransport do end @doc false - @spec send_data(dtls_transport(), binary()) :: :ok - def send_data(dtls_transport, data) do - GenServer.cast(dtls_transport, {:send_data, data}) + @spec send_rtp(dtls_transport(), binary()) :: :ok + def send_rtp(dtls_transport, data) do + GenServer.cast(dtls_transport, {:send_rtp, data}) + end + + @doc false + @spec send_rtcp(dtls_transport(), binary()) :: :ok + def send_rtcp(dtls_transport, data) do + GenServer.cast(dtls_transport, {:send_rtcp, data}) end @impl true @@ -67,7 +73,6 @@ defmodule ExWebRTC.DTLSTransport do fingerprint = ExDTLS.get_cert_fingerprint(cert) {:ok, ice_agent} = ice_module.start_link(:controlled, ice_config) - srtp = ExLibSRTP.new() state = %{ owner: owner, @@ -77,9 +82,10 @@ defmodule ExWebRTC.DTLSTransport do cert: cert, pkey: pkey, fingerprint: fingerprint, + in_srtp: ExLibSRTP.new(), + out_srtp: ExLibSRTP.new(), # sha256 hex dump peer_fingerprint: nil, - srtp: srtp, dtls_state: :new, dtls: nil, mode: nil @@ -125,18 +131,25 @@ defmodule ExWebRTC.DTLSTransport do end @impl true - def handle_cast({:send_data, _data}, %{dtls_state: :connected, ice_state: ice_state} = state) + def handle_cast({:send_rtp, data}, %{dtls_state: :connected, ice_state: ice_state} = state) when ice_state in [:connected, :completed] do - # TODO + case ExLibSRTP.protect(state.out_srtp, data) do + {:ok, protected} -> ICEAgent.send_data(state.ice_agent, protected) + {:error, reason} -> Logger.error("Unable to protect RTP: #{inspect(reason)}") + end + {:noreply, state} end @impl true - def handle_cast({:send_data, _data}, state) do - Logger.error( - "Attempted to send data when DTLS handshake was not finished or ICE Transport is unavailable" - ) + def handle_cast({:send_rtp, _data}, state) do + Logger.warning("Attemped to send RTP before DTLS handshake has been finished. Ignoring.") + {:noreply, state} + end + @impl true + def handle_cast({:send_rtcp, _data}, state) do + # TODO: implement {:noreply, state} end @@ -183,6 +196,7 @@ defmodule ExWebRTC.DTLSTransport do end defp handle_ice({:data, <> = data}, state) when f in 20..64 do + # TODO: handle {:connection_closed, _} case ExDTLS.handle_data(state.dtls, data) do {:handshake_packets, packets, timeout} when state.ice_state in [:connected, :completed] -> :ok = ICEAgent.send_data(state.ice_agent, packets) @@ -199,7 +213,7 @@ defmodule ExWebRTC.DTLSTransport do state = %{state | buffered_packets: packets} update_dtls_state(state, :connecting) - {:handshake_finished, _, remote_keying_material, profile, packets} -> + {:handshake_finished, lkm, rkm, profile, packets} -> Logger.debug("DTLS handshake finished") ICEAgent.send_data(state.ice_agent, packets) @@ -210,16 +224,16 @@ defmodule ExWebRTC.DTLSTransport do |> Utils.hex_dump() if peer_fingerprint == state.peer_fingerprint do - state = setup_srtp(state, remote_keying_material, profile) + :ok = setup_srtp(state, lkm, rkm, profile) update_dtls_state(state, :connected) else Logger.debug("Non-matching peer cert fingerprint.") update_dtls_state(state, :failed) end - {:handshake_finished, _, remote_keying_material, profile} -> + {:handshake_finished, lkm, rkm, profile} -> Logger.debug("DTLS handshake finished") - state = setup_srtp(state, remote_keying_material, profile) + :ok = setup_srtp(state, lkm, rkm, profile) update_dtls_state(state, :connected) :handshake_want_read -> @@ -229,13 +243,18 @@ defmodule ExWebRTC.DTLSTransport do defp handle_ice({:data, <> = data}, %{dtls_state: :connected} = state) when f in 128..191 do - case ExLibSRTP.unprotect(state.srtp, data) do + {type, unprotect} = + case data do + <<_, s, _::binary>> when s in 192..223 -> {:rtcp, &ExLibSRTP.unprotect_rtcp/2} + _ -> {:rtp, &ExLibSRTP.unprotect/2} + end + + case unprotect.(state.in_srtp, data) do {:ok, payload} -> - # TODO: temporarily, everything goes to peer connection process - notify(state.owner, {:rtp_data, payload}) + notify(state.owner, {type, payload}) {:error, reason} -> - Logger.warning("Failed to decrypt SRTP, reason: #{inspect(reason)}") + Logger.error("Failed to decrypt SRTP/SRTCP, reason: #{inspect(reason)}") end state @@ -280,19 +299,29 @@ defmodule ExWebRTC.DTLSTransport do defp handle_ice(_msg, state), do: state - defp setup_srtp(state, remote_keying_material, profile) do + defp setup_srtp(state, local_keying_material, remote_keying_material, profile) do {:ok, crypto_profile} = ExLibSRTP.Policy.crypto_profile_from_dtls_srtp_protection_profile(profile) - policy = %ExLibSRTP.Policy{ + inbound_policy = %ExLibSRTP.Policy{ ssrc: :any_inbound, key: remote_keying_material, rtp: crypto_profile, rtcp: crypto_profile } - :ok = ExLibSRTP.add_stream(state.srtp, policy) - state + :ok = ExLibSRTP.add_stream(state.in_srtp, inbound_policy) + + outbound_policy = %ExLibSRTP.Policy{ + ssrc: :any_outbound, + key: local_keying_material, + rtp: crypto_profile, + rtcp: crypto_profile + } + + :ok = ExLibSRTP.add_stream(state.out_srtp, outbound_policy) + + :ok end defp update_dtls_state(%{dtls_state: dtls_state} = state, dtls_state), do: state diff --git a/lib/ex_webrtc/media_stream_track.ex b/lib/ex_webrtc/media_stream_track.ex index d68cf249..49351980 100644 --- a/lib/ex_webrtc/media_stream_track.ex +++ b/lib/ex_webrtc/media_stream_track.ex @@ -5,15 +5,14 @@ defmodule ExWebRTC.MediaStreamTrack do @type t() :: %__MODULE__{ kind: :audio | :video, - id: integer(), - mid: String.t() + id: integer() } @enforce_keys [:id, :kind] - defstruct @enforce_keys ++ [:mid] + defstruct @enforce_keys - def from_transceiver(tr) do - %__MODULE__{kind: tr.kind, id: generate_id(), mid: tr.mid} + def new(kind) do + %__MODULE__{kind: kind, id: generate_id()} end defp generate_id() do diff --git a/lib/ex_webrtc/peer_connection.ex b/lib/ex_webrtc/peer_connection.ex index 2eeee569..d1ad1a5d 100644 --- a/lib/ex_webrtc/peer_connection.ex +++ b/lib/ex_webrtc/peer_connection.ex @@ -15,6 +15,8 @@ defmodule ExWebRTC.PeerConnection do IceCandidate, MediaStreamTrack, RTPTransceiver, + RTPReceiver, + RTPSender, SDPUtils, SessionDescription, Utils @@ -93,7 +95,11 @@ defmodule ExWebRTC.PeerConnection do GenServer.call(peer_connection, :get_transceivers) end - @spec add_transceiver(peer_connection(), RTPTransceiver.kind(), transceiver_options()) :: + @spec add_transceiver( + peer_connection(), + RTPTransceiver.kind() | MediaStreamTrack.t(), + transceiver_options() + ) :: {:ok, RTPTransceiver.t()} | {:error, :TODO} def add_transceiver(peer_connection, kind, options \\ []) do GenServer.call(peer_connection, {:add_transceiver, kind, options}) @@ -104,6 +110,11 @@ defmodule ExWebRTC.PeerConnection do GenServer.stop(peer_connection) end + @spec send_rtp(peer_connection(), String.t(), ExRTP.Packet.t()) :: :ok + def send_rtp(peer_connection, track_id, packet) do + GenServer.cast(peer_connection, {:send_rtp, track_id, packet}) + end + #### CALLBACKS #### @impl true @@ -318,26 +329,56 @@ defmodule ExWebRTC.PeerConnection do @impl true def handle_call({:add_transceiver, kind, options}, _from, state) when kind in [:audio, :video] do - direction = Keyword.get(options, :direction, :sendrcv) + transceiver = new_transceiver(kind, nil, options, state.config) + transceivers = state.transceivers ++ [transceiver] - {rtp_hdr_exts, codecs} = - case kind do - :audio -> {state.config.audio_rtp_hdr_exts, state.config.audio_codecs} - :video -> {state.config.video_rtp_hdr_exts, state.config.video_codecs} - end + {:reply, {:ok, transceiver}, %{state | transceivers: transceivers}} + end - transceiver = %RTPTransceiver{ - mid: nil, - direction: direction, - kind: kind, - codecs: codecs, - rtp_hdr_exts: rtp_hdr_exts - } + @impl true + def handle_call({:add_transceiver, %MediaStreamTrack{} = track, options}, _from, state) do + transceiver = new_transceiver(track.kind, track, options, state.config) + transceivers = state.transceivers ++ [transceiver] - transceivers = List.insert_at(state.transceivers, -1, transceiver) {:reply, {:ok, transceiver}, %{state | transceivers: transceivers}} end + @impl true + def handle_cast({:send_rtp, track_id, packet}, state) do + sdes_id = + Enum.find_value(state.demuxer.extensions, fn + {ext_id, {ExRTP.Packet.Extension.SourceDescription, :mid}} -> ext_id + _ -> nil + end) + + # TODO: iterating over transceivers is not optimal + # but this is, most likely, going to be refactored anyways + transceiver = + Enum.find(state.transceivers, fn + %{sender: %{track: %{id: id}}} -> id == track_id + _ -> false + end) + + case transceiver do + %RTPTransceiver{mid: mid} -> + mid_ext = + %ExRTP.Packet.Extension.SourceDescription{text: mid} + |> ExRTP.Packet.Extension.SourceDescription.to_raw(sdes_id) + + packet + |> ExRTP.Packet.set_extension(:two_byte, [mid_ext]) + |> ExRTP.Packet.encode() + |> then(&DTLSTransport.send_rtp(state.dtls_transport, &1)) + + nil -> + Logger.warning( + "Attempted to send packet to track with unrecognized id: #{inspect(track_id)}" + ) + end + + {:noreply, state} + end + @impl true def handle_info({:ex_ice, _from, {:connection_state_change, new_ice_state}}, state) do state = %{state | ice_state: new_ice_state} @@ -375,11 +416,16 @@ defmodule ExWebRTC.PeerConnection do end @impl true - def handle_info({:dtls_transport, _pid, {:rtp_data, data}}, state) do - case Demuxer.demux(state.demuxer, data) do - {:ok, demuxer, mid, packet} -> - notify(state.owner, {:data, {mid, packet}}) - {:noreply, %{state | demuxer: demuxer}} + def handle_info({:dtls_transport, _pid, {:rtp, data}}, state) do + with {:ok, demuxer, mid, packet} <- Demuxer.demux(state.demuxer, data), + %RTPTransceiver{} = t <- Enum.find(state.transceivers, &(&1.mid == mid)) do + track_id = t.receiver.track.id + notify(state.owner, {:rtp, track_id, packet}) + {:noreply, %{state | demuxer: demuxer}} + else + nil -> + Logger.warning("Received RTP with unrecognized MID: #{inspect(data)}") + {:noreply, state} {:error, reason} -> Logger.error("Unable to demux RTP, reason: #{inspect(reason)}") @@ -393,6 +439,28 @@ defmodule ExWebRTC.PeerConnection do {:noreply, state} end + defp new_transceiver(kind, sender_track, options, config) do + direction = Keyword.get(options, :direction, :sendrcv) + + {rtp_hdr_exts, codecs} = + case kind do + :audio -> {config.audio_rtp_hdr_exts, config.audio_codecs} + :video -> {config.video_rtp_hdr_exts, config.video_codecs} + end + + track = MediaStreamTrack.new(kind) + + %RTPTransceiver{ + mid: nil, + direction: direction, + kind: kind, + codecs: codecs, + rtp_hdr_exts: rtp_hdr_exts, + receiver: %RTPReceiver{track: track}, + sender: %RTPSender{track: sender_track} + } + end + defp apply_local_description(type, sdp, state) do new_transceivers = update_local_transceivers(type, state.transceivers, sdp) state = set_description(:local, type, sdp, state) @@ -403,15 +471,12 @@ defmodule ExWebRTC.PeerConnection do pt_to_mid: SDPUtils.get_payload_types(sdp) } - dtls = - if type == :answer do - {:setup, setup} = ExSDP.Media.get_attribute(hd(sdp.media), :setup) - :ok = DTLSTransport.start_dtls(state.dtls_transport, setup, state.peer_fingerprint) - else - state.dtls_transport - end + if type == :answer do + {:setup, setup} = ExSDP.Media.get_attribute(hd(sdp.media), :setup) + DTLSTransport.start_dtls(state.dtls_transport, setup, state.peer_fingerprint) + end - {:ok, %{state | transceivers: new_transceivers, dtls_transport: dtls, demuxer: demuxer}} + {:ok, %{state | transceivers: new_transceivers, demuxer: demuxer}} end defp update_local_transceivers(:offer, transceivers, _sdp) do @@ -434,6 +499,8 @@ defmodule ExWebRTC.PeerConnection do :ok = ICEAgent.set_remote_credentials(state.ice_agent, ice_ufrag, ice_pwd) :ok = ICEAgent.gather_candidates(state.ice_agent) + # TODO: this needs a look + new_remote_tracks = new_transceivers # only take new transceivers that can receive tracks @@ -441,7 +508,7 @@ defmodule ExWebRTC.PeerConnection do RTPTransceiver.find_by_mid(state.transceivers, tr.mid) == nil and tr.direction in [:recvonly, :sendrecv] end) - |> Enum.map(fn tr -> MediaStreamTrack.from_transceiver(tr) end) + |> Enum.map(fn tr -> tr.receiver.track end) for track <- new_remote_tracks do notify(state.owner, {:track, track}) @@ -449,26 +516,22 @@ defmodule ExWebRTC.PeerConnection do state = set_description(:remote, type, sdp, state) - dtls = - if type == :answer do - {:setup, setup} = ExSDP.Media.get_attribute(hd(sdp.media), :setup) + if type == :answer do + {:setup, setup} = ExSDP.Media.get_attribute(hd(sdp.media), :setup) - setup = - case setup do - :active -> :passive - :passive -> :active - end + setup = + case setup do + :active -> :passive + :passive -> :active + end - :ok = DTLSTransport.start_dtls(state.dtls_transport, setup, peer_fingerprint) - else - state.dtls_transport - end + DTLSTransport.start_dtls(state.dtls_transport, setup, peer_fingerprint) + end {:ok, %{ state | transceivers: new_transceivers, - dtls_transport: dtls, peer_fingerprint: peer_fingerprint }} else @@ -506,24 +569,34 @@ defmodule ExWebRTC.PeerConnection do defp find_next_mid(state) do # next mid must be unique, it's acomplished by looking for values # greater than any mid in remote description or our own transceivers - crd_mids = - if is_nil(state.current_remote_desc) do - [] + crd_mids = get_desc_mids(state.current_remote_desc) + tsc_mids = get_transceiver_mids(state.transceivers) + + Enum.max(crd_mids ++ tsc_mids, &>=/2, fn -> -1 end) + 1 + end + + defp get_desc_mids(nil), do: [] + + defp get_desc_mids({_, remote_desc}) do + Enum.flat_map(remote_desc.media, fn mline -> + with {:mid, mid} <- ExSDP.Media.get_attribute(mline, :mid), + {mid, ""} <- Integer.parse(mid) do + [mid] else - for mline <- state.current_remote_desc.media, - {:mid, mid} <- ExSDP.Media.get_attribute(mline, :mid), - {mid, ""} <- Integer.parse(mid) do - mid - end + _ -> [] end + end) + end - tsc_mids = - for %RTPTransceiver{mid: mid} when mid != nil <- state.transceivers, - {mid, ""} <- Integer.parse(mid) do - mid + defp get_transceiver_mids(transceivers) do + Enum.flat_map(transceivers, fn transceiver -> + with mid when mid != nil <- transceiver.mid, + {mid, ""} <- Integer.parse(mid) do + [mid] + else + _ -> [] end - - Enum.max(crd_mids ++ tsc_mids, &>=/2, fn -> -1 end) + 1 + end) end defp check_desc_altered(:offer, sdp, %{last_offer: offer}) when sdp == offer, do: :ok diff --git a/lib/ex_webrtc/rtp_receiver.ex b/lib/ex_webrtc/rtp_receiver.ex index fd1a8338..5356cf88 100644 --- a/lib/ex_webrtc/rtp_receiver.ex +++ b/lib/ex_webrtc/rtp_receiver.ex @@ -2,5 +2,12 @@ defmodule ExWebRTC.RTPReceiver do @moduledoc """ RTPReceiver """ - defstruct [:ssrc] + + alias ExWebRTC.MediaStreamTrack + + @type t() :: %__MODULE__{ + track: MediaStreamTrack.t() | nil + } + + defstruct [:track] end diff --git a/lib/ex_webrtc/rtp_sender.ex b/lib/ex_webrtc/rtp_sender.ex new file mode 100644 index 00000000..a497b2f9 --- /dev/null +++ b/lib/ex_webrtc/rtp_sender.ex @@ -0,0 +1,13 @@ +defmodule ExWebRTC.RTPSender do + @moduledoc """ + RTPSender + """ + + alias ExWebRTC.MediaStreamTrack + + @type t() :: %__MODULE__{ + track: MediaStreamTrack.t() | nil + } + + defstruct [:track] +end diff --git a/lib/ex_webrtc/rtp_transceiver.ex b/lib/ex_webrtc/rtp_transceiver.ex index 82d24c4a..e8397c37 100644 --- a/lib/ex_webrtc/rtp_transceiver.ex +++ b/lib/ex_webrtc/rtp_transceiver.ex @@ -3,7 +3,13 @@ defmodule ExWebRTC.RTPTransceiver do RTPTransceiver """ - alias ExWebRTC.{PeerConnection.Configuration, RTPCodecParameters, RTPReceiver} + alias ExWebRTC.{ + PeerConnection.Configuration, + RTPCodecParameters, + RTPReceiver, + RTPSender, + MediaStreamTrack + } @type direction() :: :sendonly | :recvonly | :sendrecv | :inactive | :stopped @type kind() :: :audio | :video @@ -14,11 +20,13 @@ defmodule ExWebRTC.RTPTransceiver do kind: kind(), rtp_hdr_exts: [ExSDP.Attribute.Extmap.t()], codecs: [RTPCodecParameters.t()], - rtp_receiver: nil + receiver: RTPReceiver.t(), + sender: RTPSender.t() } @enforce_keys [:mid, :direction, :kind] - defstruct @enforce_keys ++ [codecs: [], rtp_hdr_exts: [], rtp_receiver: %RTPReceiver{}] + defstruct @enforce_keys ++ + [codecs: [], rtp_hdr_exts: [], receiver: %RTPReceiver{}, sender: %RTPSender{}] @doc false def find_by_mid(transceivers, mid) do @@ -61,7 +69,8 @@ defmodule ExWebRTC.RTPTransceiver do nil -> codecs = get_codecs(mline, config) rtp_hdr_exts = get_rtp_hdr_extensions(mline, config) - ssrc = ExSDP.Media.get_attributes(mline, ExSDP.Attribute.SSRC) + + track = MediaStreamTrack.new(mline.type) tr = %__MODULE__{ mid: mid, @@ -69,7 +78,7 @@ defmodule ExWebRTC.RTPTransceiver do kind: mline.type, codecs: codecs, rtp_hdr_exts: rtp_hdr_exts, - rtp_receiver: %RTPReceiver{ssrc: ssrc} + receiver: %RTPReceiver{track: track} } transceivers ++ [tr] @@ -128,14 +137,12 @@ defmodule ExWebRTC.RTPTransceiver do defp update(transceiver, mline, config) do codecs = get_codecs(mline, config) rtp_hdr_exts = get_rtp_hdr_extensions(mline, config) - ssrc = ExSDP.Media.get_attributes(mline, ExSDP.Attribute.SSRC) - rtp_receiver = %RTPReceiver{ssrc: ssrc} + # TODO: potentially update tracks %__MODULE__{ transceiver | codecs: codecs, - rtp_hdr_exts: rtp_hdr_exts, - rtp_receiver: rtp_receiver + rtp_hdr_exts: rtp_hdr_exts } end diff --git a/mix.lock b/mix.lock index aef0377a..05fedca8 100644 --- a/mix.lock +++ b/mix.lock @@ -10,7 +10,7 @@ "earmark_parser": {:hex, :earmark_parser, "1.4.38", "b42252eddf63bda05554ba8be93a1262dc0920c721f1aaf989f5de0f73a2e367", [:mix], [], "hexpm", "2cd0907795aaef0c7e8442e376633c5b3bd6edc8dbbdc539b22f095501c1cdb6"}, "erlex": {:hex, :erlex, "0.2.6", "c7987d15e899c7a2f34f5420d2a2ea0d659682c06ac607572df55a43753aa12e", [:mix], [], "hexpm", "2ed2e25711feb44d52b17d2780eabf998452f6efda104877a3881c2f8c0c0c75"}, "ex_doc": {:hex, :ex_doc, "0.30.9", "d691453495c47434c0f2052b08dd91cc32bc4e1a218f86884563448ee2502dd2", [:mix], [{:earmark_parser, "~> 1.4.31", [hex: :earmark_parser, repo: "hexpm", optional: false]}, {:makeup_elixir, "~> 0.14", [hex: :makeup_elixir, repo: "hexpm", optional: false]}, {:makeup_erlang, "~> 0.1", [hex: :makeup_erlang, repo: "hexpm", optional: false]}], "hexpm", "d7aaaf21e95dc5cddabf89063327e96867d00013963eadf2c6ad135506a8bc10"}, - "ex_dtls": {:git, "https://github.com/elixir-webrtc/ex_dtls.git", "876e2119f97eee053671b6f6fd069575f0da8d77", []}, + "ex_dtls": {:git, "https://github.com/elixir-webrtc/ex_dtls.git", "f4324f9f0200612116f662fec7ac63f0d7d3dd91", []}, "ex_ice": {:hex, :ex_ice, "0.2.0", "6002de5b1c0561f8ff50238a6d04cb99eabf9b7db91031b6f0580d67888f0365", [:mix], [{:ex_stun, "~> 0.1.0", [hex: :ex_stun, repo: "hexpm", optional: false]}], "hexpm", "1def9449009241918b31ff269f344e5c4bd15257ba5cf95d1edc1250c28af39b"}, "ex_libsrtp": {:hex, :ex_libsrtp, "0.6.0", "d96cd7fc1780157614f0bf47d31587e5eab953b43067f4885849f8177ec452a9", [:mix], [{:bunch, "~> 1.3", [hex: :bunch, repo: "hexpm", optional: false]}, {:unifex, "~> 1.0", [hex: :unifex, repo: "hexpm", optional: false]}], "hexpm", "e9ce8a507a658f7e2df72fae82a4b3ba0a056c175f0bc490e79ab03058e094d5"}, "ex_rctp": {:git, "https://github.com/elixir-webrtc/ex_rtcp.git", "c0cf2b7f995e34d13cee4cbb228376a55700fb6a", []}, diff --git a/test/ex_webrtc/dtls_transport_test.exs b/test/ex_webrtc/dtls_transport_test.exs index a7fa2101..75f25251 100644 --- a/test/ex_webrtc/dtls_transport_test.exs +++ b/test/ex_webrtc/dtls_transport_test.exs @@ -61,7 +61,7 @@ defmodule ExWebRTC.DTLSTransportTest do end test "cannot send data when handshake not finished", %{dtls: dtls} do - DTLSTransport.send_data(dtls, <<1, 2, 3>>) + DTLSTransport.send_rtp(dtls, <<1, 2, 3>>) refute_receive {:fake_ice, _data} end diff --git a/test/ex_webrtc/peer_connection_test.exs b/test/ex_webrtc/peer_connection_test.exs index 71629581..8bf10387 100644 --- a/test/ex_webrtc/peer_connection_test.exs +++ b/test/ex_webrtc/peer_connection_test.exs @@ -31,7 +31,7 @@ defmodule ExWebRTC.PeerConnectionTest do offer = %SessionDescription{type: :offer, sdp: File.read!("test/fixtures/audio_sdp.txt")} :ok = PeerConnection.set_remote_description(pc, offer) - assert_receive {:ex_webrtc, ^pc, {:track, %MediaStreamTrack{mid: "0", kind: :audio}}} + assert_receive {:ex_webrtc, ^pc, {:track, %MediaStreamTrack{kind: :audio}}} offer = %SessionDescription{ type: :offer, @@ -40,7 +40,7 @@ defmodule ExWebRTC.PeerConnectionTest do :ok = PeerConnection.set_remote_description(pc, offer) - assert_receive {:ex_webrtc, ^pc, {:track, %MediaStreamTrack{mid: "1", kind: :video}}} + assert_receive {:ex_webrtc, ^pc, {:track, %MediaStreamTrack{kind: :video}}} refute_receive {:ex_webrtc, ^pc, {:track, %MediaStreamTrack{}}} end