// Copyright (c) Microsoft Corporation. // Licensed under the MIT License. using System; using UnityEngine; namespace Microsoft.MixedReality.WebRTC.Unity { /// /// Media line abstraction for a peer connection. /// /// This container binds together a source component () and/or a receiver /// component () on one side, with a transceiver on the other side. The media line /// is a declarative representation of this association, which is then turned into a binding by the implementation /// during an SDP negotiation. This forms the core of the algorithm allowing automatic transceiver pairing /// between the two peers based on the declaration of intent of the user. /// /// Assigning Unity components to the and properties serves /// as an indication of the user intent to send and/or receive media through the transceiver, and is /// used during the SDP exchange to derive the to negotiate. /// After the SDP negotiation is completed, the property refers to the transceiver /// associated with this media line, and which the sender and receiver will use. /// /// Users typically interact with this class through the peer connection transceiver collection in the Unity /// inspector window, though direct manipulation via code is also possible. /// [Serializable] public class MediaLine { /// /// Kind of media of the media line and its attached transceiver. /// /// This is assiged when the media line is created with /// and is immutable for the lifetime of the peer connection. /// public MediaKind MediaKind => _mediaKind; /// /// Media source producing the media to send through the transceiver attached to this media line. /// /// /// This must be an instance of a class derived from or /// depending on whether is /// or , respectively. /// /// Internally the peer connection will automatically create and manage a media track to bridge the /// media source with the transceiver. /// /// If this is non-null then the peer connection will negotiate sending some media, otherwise /// it will signal the remote peer that it does not wish to send (receive-only or inactive). /// /// If is valid, that is a first session negotiation has already been completed, /// then changing this value raises a event on the /// peer connection of . /// /// Must be changed on the main Unity app thread. /// public MediaTrackSource Source { get { return _source; } set { if (_source == value) { return; } if (value != null && value.MediaKind != MediaKind) { throw new ArgumentException("Wrong media kind", nameof(Receiver)); } var oldTrack = LocalTrack; if (_source != null && _peer.IsAwake) { _source.OnRemovedFromMediaLine(this); } _source = value; if (_source != null && _peer.IsAwake) { _source.OnAddedToMediaLine(this); CreateLocalTrackIfNeeded(); } // Dispose the old track *after* replacing it with the new one // so that there is no gap in sending. oldTrack?.Dispose(); // Whatever the change, keep the direction consistent. UpdateTransceiverDesiredDirection(); } } /// /// Name of the local media track this component will create when calling . /// If left empty, the implementation will generate a unique name for the track (generally a GUID). /// /// /// This value must comply with the 'msid' attribute rules as defined in /// https://tools.ietf.org/html/draft-ietf-mmusic-msid-05#section-2, which in /// particular constraints the set of allowed characters to those allowed for a /// 'token' element as specified in https://tools.ietf.org/html/rfc4566#page-43: /// - Symbols [!#$%'*+-.^_`{|}~] and ampersand & /// - Alphanumerical characters [A-Za-z0-9] /// /// Users can manually test if a string is a valid SDP token with the utility method /// . The property setter will /// use this and throw an if the token is not a valid /// SDP token. /// /// The sender track name is taken into account each time the track is created. If this /// property is assigned after the track was created (already negotiated), the value will /// be used only for the next negotiation, and the current sender track will keep its /// current track name (either a previous value or a generated one). /// /// public string SenderTrackName { get { return _senderTrackName; } set { SdpTokenAttribute.Validate(_senderTrackName); _senderTrackName = value; } } /// /// Local track created from a local source. /// /// /// This is non-null when a live source is attached to the , and the owning /// is connected. /// public LocalMediaTrack LocalTrack => Transceiver?.LocalTrack; /// /// Media receiver consuming the media received through the transceiver attached to this media line. /// /// /// This must be an instance of a class derived from or /// depending on whether is /// or , respectively. /// /// If this is non-null then the peer connection will negotiate receiving some media, otherwise /// it will signal the remote peer that it does not wish to receive (send-only or inactive). /// /// If is valid, that is a first session negotiation has already been conducted, /// then changing this value raises a event on the /// peer connection of . /// /// Must be changed on the main Unity app thread. /// public MediaReceiver Receiver { get { return _receiver; } set { if (_receiver == value) { return; } if (value != null && value.MediaKind != MediaKind) { throw new ArgumentException("Wrong media kind", nameof(Receiver)); } if (_receiver != null && _peer.IsAwake) { if (_remoteTrack != null) { _receiver.OnUnpaired(_remoteTrack); } _receiver.OnRemovedFromMediaLine(this); } _receiver = value; if (_receiver != null && _peer.IsAwake) { _receiver.OnAddedToMediaLine(this); if (_remoteTrack != null) { _receiver.OnPaired(_remoteTrack); } } // Whatever the change, keep the direction consistent. UpdateTransceiverDesiredDirection(); } } /// /// Transceiver attached with this media line. /// /// On the offering peer this changes during , while this is updated by /// when receiving an offer on the answering peer. /// /// Because transceivers cannot be destroyed, once this property is assigned a non-null value it keeps that /// value until the peer connection owning the media line is closed. /// public Transceiver Transceiver { get; private set; } /// /// owning this . /// public PeerConnection Peer { get => _peer; internal set { Debug.Assert(Peer == null || Peer == value); _peer = value; } } #region Private fields private PeerConnection _peer; /// /// Backing field to serialize the property. /// /// [SerializeField] private MediaKind _mediaKind; /// /// Backing field to serialize the property. /// /// [SerializeField] private MediaTrackSource _source; /// /// Backing field to serialize the property. /// /// [SerializeField] private MediaReceiver _receiver; /// /// Backing field to serialize the sender track's name. /// [SerializeField] [Tooltip("SDP track name")] [SdpToken(allowEmpty: true)] private string _senderTrackName; // Cache for the remote track opened by the latest negotiation. // Comparing it to Transceiver.RemoteTrack will tell if streaming has just started/stopped. private MediaTrack _remoteTrack; #endregion /// /// Constructor called internally by . /// /// Immutable value assigned to the property on construction. internal MediaLine(PeerConnection peer, MediaKind kind) { Peer = peer; _mediaKind = kind; } private void UpdateTransceiverDesiredDirection() { if (Transceiver != null) { // Avoid races on the desired direction by limiting changes to the main thread. // Note that EnsureIsMainAppThread cannot be used if _peer is not awake, so only // check when there is a transceiver (meaning _peer is enabled). Peer.EnsureIsMainAppThread(); bool wantsSend = _source != null && _source.IsLive; bool wantsRecv = (_receiver != null); Transceiver.DesiredDirection = Transceiver.DirectionFromSendRecv(wantsSend, wantsRecv); } } // Initializes and attaches a local track if all the preconditions are satisfied. private void CreateLocalTrackIfNeeded() { if (_source != null && _source.IsLive && Transceiver != null) { if (MediaKind == MediaKind.Audio) { var audioSource = (AudioTrackSource)_source; var initConfig = new LocalAudioTrackInitConfig { trackName = _senderTrackName }; var audioTrack = LocalAudioTrack.CreateFromSource(audioSource.Source, initConfig); Transceiver.LocalAudioTrack = audioTrack; } else { Debug.Assert(MediaKind == MediaKind.Video); var videoSource = (VideoTrackSource)_source; var initConfig = new LocalVideoTrackInitConfig { trackName = _senderTrackName }; var videoTrack = LocalVideoTrack.CreateFromSource(videoSource.Source, initConfig); Transceiver.LocalVideoTrack = videoTrack; } } } // Detaches and disposes the local track if there is one. private void DestroyLocalTrackIfAny() { var localTrack = Transceiver?.LocalTrack; if (localTrack != null) { if (MediaKind == MediaKind.Audio) { Transceiver.LocalAudioTrack = null; } else { Debug.Assert(MediaKind == MediaKind.Video); Transceiver.LocalVideoTrack = null; } localTrack.Dispose(); } } internal void UpdateAfterSdpReceived() { Debug.Assert(Transceiver != null); // Callbacks must be called on the main Unity app thread. Peer.EnsureIsMainAppThread(); var newRemoteTrack = Transceiver.RemoteTrack; if (_receiver != null) { bool wasReceiving = _remoteTrack != null; bool isReceiving = newRemoteTrack != null; if (isReceiving && !wasReceiving) { // Transceiver started receiving, and user actually wants to receive _receiver.OnPaired(newRemoteTrack); } else if (!isReceiving && wasReceiving) { // Transceiver stopped receiving (user intent does not matter here) _receiver.OnUnpaired(_remoteTrack); } } _remoteTrack = newRemoteTrack; } /// /// Pair the given transceiver with the current media line. /// /// The transceiver to pair with. /// /// The transceiver associated in the offer with the same media line index as the current media line /// has a different media kind than the media line. This is generally a result of the two peers having /// mismatching media line configurations. /// internal void PairTransceiver(Transceiver tr) { Peer.EnsureIsMainAppThread(); Debug.Assert(tr != null); Debug.Assert(Transceiver == null); // Check consistency before assigning if (tr.MediaKind != MediaKind) { throw new InvalidTransceiverMediaKindException(); } Transceiver = tr; // Initialize the transceiver direction in sync with Sender and Receiver. UpdateTransceiverDesiredDirection(); // Start the local track if there is a live source. CreateLocalTrackIfNeeded(); } internal void UnpairTransceiver() { Peer.EnsureIsMainAppThread(); // Notify the receiver. if (_remoteTrack != null && _receiver != null) { _receiver.OnUnpaired(_remoteTrack); } _remoteTrack = null; DestroyLocalTrackIfAny(); Transceiver = null; } /// /// Internal callback when the underlying source providing media frames to the sender track /// is created, and therefore the local media track needs to be created too. /// /// /// internal void AttachSource() { Debug.Assert(Source.IsLive); CreateLocalTrackIfNeeded(); UpdateTransceiverDesiredDirection(); } /// /// Internal callback when the underlying source providing media frames to the sender track /// is destroyed, and therefore the local media track needs to be destroyed too. /// /// /// internal void DetachSource() { Debug.Assert(Source.IsLive); DestroyLocalTrackIfAny(); UpdateTransceiverDesiredDirection(); } internal void OnReceiverDestroyed() { // Different from `Receiver = null`. Don't need to call Receiver.OnRemovedFromMediaLine // or Receiver.OnUnpaired since the Receiver itself has called this. _receiver = null; UpdateTransceiverDesiredDirection(); } // Called by PeerConnection.Awake. internal void Awake() { if (_source) { // Fill the list of media lines for the source. _source.OnAddedToMediaLine(this); } if (_receiver) { _receiver.OnAddedToMediaLine(this); } } // Called by PeerConnection.OnDestroy. internal void OnDestroy() { if (_source) { // Fill the list of media lines for the source. _source.OnRemovedFromMediaLine(this); } if (_receiver) { _receiver.OnRemovedFromMediaLine(this); } } } }