From 15d2c39c4d46b7233c24169be0056c4698c47a82 Mon Sep 17 00:00:00 2001 From: Marius Melzer Date: Wed, 10 Jun 2020 23:43:38 +0200 Subject: [PATCH] Enable renegotation No "perfect renegotiation" because that's too new and unsupported by browsers. Instead, a workaround by deferring adding the own offer: https://stackoverflow.com/questions/65059808/downward-compatibility-of-webrtc-renegotiation --- coffee/browser.coffee | 6 ++ coffee/gum.coffee | 32 +++++++++- coffee/local_peer.coffee | 7 +++ coffee/remote_peer.coffee | 83 +++++++++++++++++--------- coffee/room.coffee | 1 + palava.bundle.js | 2 +- palava.js | 120 ++++++++++++++++++++++++++++++++++---- palava.min.js | 2 +- 8 files changed, 213 insertions(+), 40 deletions(-) diff --git a/coffee/browser.coffee b/coffee/browser.coffee index 1596f42..73bf67f 100644 --- a/coffee/browser.coffee +++ b/coffee/browser.coffee @@ -15,6 +15,12 @@ palava.browser.isMozilla = -> palava.browser.isChrome = -> adapter.browserDetails.browser == 'chrome' +# Checks whether the browser supports the modern asymmetric +# offer/answer mechanism via +palava.browser.modernWebrtc(status) = -> + (status.user_agent == "chrome" && status.user_agent_version >= 83) || + (status.user_agent == "firefox" && status.user_agent_version >= 77) + # Checks which browser is used # # @return [String] A well defined id of the browser (firefox, chrome, safari, or unknown) diff --git a/coffee/gum.coffee b/coffee/gum.coffee index 7e8f64f..d2f230d 100644 --- a/coffee/gum.coffee +++ b/coffee/gum.coffee @@ -5,7 +5,9 @@ palava = @palava class palava.Gum extends @EventEmitter constructor: (config) -> @config = config || { video: true, audio: true } - @stream = null + @stream = null # this stream switches from localStream to displayStream on request + @localStream = null + @displayStream = null changeConfig: (config) => @config = config @@ -17,6 +19,7 @@ class palava.Gum extends @EventEmitter @config ).then( (stream) => + @localStream = stream.clone() @stream = stream @emit 'stream_ready', stream ).catch( @@ -24,9 +27,36 @@ class palava.Gum extends @EventEmitter @emit 'stream_error', error ) + requestDisplaySharing: => + navigator.mediaDevices.getDisplayMedia( + {video:true} + ).then( + (stream) => + # add audio track to the display stream (if any) + if @localStream.getAudioTracks().length > 0 + stream.addTrack(@localStream.getAudioTracks()[0], @localStream) + @displayStream = stream.clone() + @stream = stream + @emit 'display_stream_ready', stream + ).catch( + (error) => + @emit 'display_stream_error', error + ) + + stopDisplaySharing: => + @stream = @localStream + @emit 'display_stream_stop', @displayStream + @displayStream = null + getStream: => @stream + getLocalStream: => + @localStream + + getDisplayStream: => + @displayStream + releaseStream: => if @stream @stream.getAudioTracks().forEach( (track) => track.stop() ) diff --git a/coffee/local_peer.coffee b/coffee/local_peer.coffee index b4a524f..b9fee7e 100644 --- a/coffee/local_peer.coffee +++ b/coffee/local_peer.coffee @@ -33,6 +33,10 @@ class palava.LocalPeer extends palava.Peer @emit 'stream_ready', e @userMedia.on 'stream_error', (e) => @emit 'stream_error', e + @userMedia.on 'display_stream_ready', (e) => + @emit 'display_stream_ready', e + @userMedia.on 'display_stream_error', (e) => + @emit 'display_stream_error', e if @getStream() @ready = true @emit 'stream_ready' @@ -54,6 +58,9 @@ class palava.LocalPeer extends palava.Peer getStream: => @userMedia.getStream() + requestDisplaySharing: => + @userMedia.requestDisplaySharing() + # Updates the status of the local peer. The status is extended or updated with the given items. # # @param status [Object] Object containing the new items diff --git a/coffee/remote_peer.coffee b/coffee/remote_peer.coffee index 5de671f..09daf8c 100644 --- a/coffee/remote_peer.coffee +++ b/coffee/remote_peer.coffee @@ -28,10 +28,14 @@ class palava.RemotePeer extends palava.Peer @dataChannels = {} + @offers = offers + @makingOffer = false + @ignoreOffer = false + @setRemoteAnswerPending = false + @setupRoom() - @setupPeerConnection(offers) @setupDistributor() - + @setupPeerConnection(offers) if offers @sendOffer() @@ -62,8 +66,7 @@ class palava.RemotePeer extends palava.Peer credential: @turnCredentials.password {iceServers: options} - # Sets up the peer connection and its events - # + # Sets up the peer connection and its events # # @nodoc # setupPeerConnection: (offers) => @@ -87,6 +90,9 @@ class palava.RemotePeer extends palava.Peer @ready = false @emit 'stream_removed' + @peerConnection.onnegotiationneeded = () => + @sendOffer() + @peerConnection.oniceconnectionstatechange = (event) => connectionState = event.target.iceConnectionState @@ -107,12 +113,33 @@ class palava.RemotePeer extends palava.Peer @error = "connection_closed" @emit 'connection_closed' - # TODO onsignalingstatechange - if @room.localPeer.getStream() - @peerConnection.addStream @room.localPeer.getStream() - else - # not suppored yet + for track in @room.localPeer.getStream().getTracks() + @peerConnection.addTrack(track, @room.localPeer.getStream()) + + @room.localPeer.on 'display_stream_ready', (stream) => + videoSender = @peerConnection.getSenders().find( + (sender) => sender.track.kind == "video" + ) + if videoSender + # if there is a local video track then replace it because that's faster + # and does not need renegotiation + videoSender.replaceTrack(track) + else + @peerConnection.addTrack(stream.getVideoTracks()[0], + @room.localPeer.getStream()) + + @room.localPeer.on 'display_stream_stop', (stream) => + localVideoTracks = @room.localPeer.getLocalStream().getVideoTracks() + if localVideoTracks.length > 0 + # if there was a local video track then reuse the display video track + # because it's faster and does not need renegotiation + videoSender = @peerConnection.getSenders().find( + (sender) => sender.track.kind == "video" + ) + videoSender.replaceTrack(localVideoTracks[0]) + else + @peerConnection.removeTrack(stream.getVideoTracks()[0]) # data channel setup @@ -140,8 +167,6 @@ class palava.RemotePeer extends palava.Peer # @nodoc # setupDistributor: => - # TODO _ in events also in rtc-server - # TODO consistent protocol naming @distributor = new palava.Distributor(@room.channel, @id) @distributor.on 'peer_left', (msg) => @@ -157,16 +182,20 @@ class palava.RemotePeer extends palava.Peer return if msg.candidate == "" candidate = new RTCIceCandidate({candidate: msg.candidate, sdpMLineIndex: msg.sdpmlineindex, sdpMid: msg.sdpmid}) unless @room.options.filterIceCandidateTypes.includes(candidate.type) - @peerConnection.addIceCandidate(candidate) + await @peerConnection.addIceCandidate(candidate) @distributor.on 'offer', (msg) => - @peerConnection.setRemoteDescription(new RTCSessionDescription(msg.sdp)) - @emit 'offer' # ignored so far + # we sent an offer already and are the preferred offerer, so we don't back down + return if @offers && @pendingOffer + @pendingOffer = null + + await @peerConnection.setRemoteDescription(new RTCSessionDescription(msg.sdp)) @sendAnswer() @distributor.on 'answer', (msg) => - @peerConnection.setRemoteDescription(new RTCSessionDescription(msg.sdp)) - @emit 'answer' # ignored so far + await @peerConnection.setLocalDescription(@pendingOffer) if @pendingOffer + @pendingOffer = null + await @peerConnection.setRemoteDescription(new RTCSessionDescription(msg.sdp)) @distributor.on 'peer_updated_status', (msg) => @status = msg.status @@ -199,30 +228,32 @@ class palava.RemotePeer extends palava.Peer @on 'oaerror', (e) => @room.emit('peer_oaerror', @, e) @on 'channel_ready', (n, c) => @room.emit('peer_channel_ready', @, n, c) - # Sends the offer for a peer connection - # - # @nodoc + sendMessage: (data) => + @distributor.send + event: 'message' + data: data + + # Sends the offer to create a peer connection # sendOffer: => - @peerConnection.createOffer @sdpSender('offer'), @oaError, palava.browser.getConstraints() + @peerConnection.createOffer @sdpSender('offer'), @oaError, palava.browser.getConstraints() if @peerConnection.signalingState == "stable" && !@pendingOffer # Sends the answer to create a peer connection # sendAnswer: => @peerConnection.createAnswer @sdpSender('answer'), @oaError, palava.browser.getConstraints() - sendMessage: (data) => - @distributor.send - event: 'message' - data: data - # Helper for sending sdp + # Send offer/answer # # @nodoc # sdpSender: (event) => (sdp) => - @peerConnection.setLocalDescription(sdp) + if event == 'offer' + @pendingOffer = sdp + else + await @peerConnection.setLocalDescription(sdp) @distributor.send event: event sdp: sdp diff --git a/coffee/room.coffee b/coffee/room.coffee index a4fd1c3..10d49bb 100644 --- a/coffee/room.coffee +++ b/coffee/room.coffee @@ -85,6 +85,7 @@ class palava.Room extends @EventEmitter @options.ownStatus[key] = status[key] for key in status @options.ownStatus.user_agent ||= palava.browser.getUserAgent() + @options.ownStatus.user_agent_version ||= palava.browser.getUserAgentVersion() @distributor.send event: 'join_room' diff --git a/palava.bundle.js b/palava.bundle.js index 89e6869..3621890 100644 --- a/palava.bundle.js +++ b/palava.bundle.js @@ -3547,4 +3547,4 @@ GNU Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with this program. If not, see . */ -(function(){}).call(this),function(){this.palava={browser:{}}}.call(this),function(){"object"==typeof module&&"object"==typeof module.exports&&(module.exports=this.palava),"object"!=typeof EventEmitter&&"function"==typeof require?this.EventEmitter=require("wolfy87-eventemitter"):this.EventEmitter=EventEmitter,"object"!=typeof adapter&&"function"==typeof require?this.adapter=require("webrtc-adapter/out/adapter_no_edge"):this.adapter=adapter}.call(this),function(){var t,r;r=this.palava,t=this.adapter,r.browser.isMozilla=function(){return"firefox"===t.browserDetails.browser},r.browser.isChrome=function(){return"chrome"===t.browserDetails.browser},r.browser.getUserAgent=function(){return t.browserDetails.browser},r.browser.getUserAgentVersion=function(){return t.browserDetails.version},r.browser.checkForWebrtcError=function(){try{new window.RTCPeerConnection({iceServers:[]})}catch(t){return t}return!(window.RTCPeerConnection&&window.RTCIceCandidate&&window.RTCSessionDescription&&navigator.mediaDevices&&navigator.mediaDevices.getUserMedia)},r.browser.getConstraints=function(){return{optional:[],mandatory:{OfferToReceiveAudio:!0,OfferToReceiveVideo:!0}}},r.browser.getPeerConnectionOptions=function(){return r.browser.isChrome()?{optional:[{DtlsSrtpKeyAgreement:!0}]}:{}},r.browser.registerFullscreen=function(t,e){return console.log("DEPRECATED: palava.browser.registerFullscreen will be removed from the palava library in early 2021"),t.requestFullscreen?t.addEventListener(e,function(){return this.requestFullscreen()}):t.mozRequestFullScreen?t.addEventListener(e,function(){return this.mozRequestFullScreen()}):t.webkitRequestFullscreen?t.addEventListener(e,function(){return this.webkitRequestFullscreen()}):void 0},r.browser.attachMediaStream=function(t,e){return e?t.srcObject=e:(t.pause(),t.srcObject=null)},r.browser.attachPeer=function(t,e){var n;return n=function(){return r.browser.attachMediaStream(t,e.getStream()),e.isLocal()&&t.setAttribute("muted",!0),t.play()},e.getStream()?n():e.on("stream_ready",function(){return n()})}}.call(this),function(){var n=function(t,e){return function(){return t.apply(e,arguments)}},r=function(t,e){function n(){this.constructor=t}for(var r in e)i.call(e,r)&&(t[r]=e[r]);return n.prototype=e.prototype,t.prototype=new n,t.__super__=e.prototype,t},i={}.hasOwnProperty;this.palava.Gum=function(t){function e(t){this.releaseStream=n(this.releaseStream,this),this.getStream=n(this.getStream,this),this.requestStream=n(this.requestStream,this),this.changeConfig=n(this.changeConfig,this),this.config=t||{video:!0,audio:!0},this.stream=null}return r(e,t),e.prototype.changeConfig=function(t){return this.config=t,this.releaseStream(),this.requestStream()},e.prototype.requestStream=function(){return navigator.mediaDevices.getUserMedia(this.config).then((n=this,function(t){return n.stream=t,n.emit("stream_ready",t)}))["catch"]((e=this,function(t){return e.emit("stream_error",t)}));var e,n},e.prototype.getStream=function(){return this.stream},e.prototype.releaseStream=function(){return!!this.stream&&(this.stream.getAudioTracks().forEach(function(t){return t.stop()}),this.stream.getVideoTracks().forEach(function(t){return t.stop()}),this.stream=null,this.emit("stream_released",this),!0)},e}(this.EventEmitter)}.call(this),function(){var e,n=function(t,e){return function(){return t.apply(e,arguments)}};(e=this.palava).Identity=function(){function t(t){this.getStatus=n(this.getStatus,this),this.getName=n(this.getName,this),this.userMediaConfig=t.userMediaConfig,this.status=t.status||{},this.status.name=t.name}return t.prototype.newUserMedia=function(){return new e.Gum(this.userMediaConfig)},t.prototype.getName=function(){return this.name},t.prototype.getStatus=function(){return this.status},t}()}.call(this),function(){var r,i=function(t,e){return function(){return t.apply(e,arguments)}},n=function(t,e){function n(){this.constructor=t}for(var r in e)o.call(e,r)&&(t[r]=e[r]);return n.prototype=e.prototype,t.prototype=new n,t.__super__=e.prototype,t},o={}.hasOwnProperty;(r=this.palava).Peer=function(t){function e(t,e){var n;this.isRemote=i(this.isRemote,this),this.isLocal=i(this.isLocal,this),this.isReady=i(this.isReady,this),this.isMuted=i(this.isMuted,this),this.getError=i(this.getError,this),this.hasError=i(this.hasError,this),this.hasVideo=i(this.hasVideo,this),this.transmitsVideo=i(this.transmitsVideo,this),this.hasAudio=i(this.hasAudio,this),this.transmitsAudio=i(this.transmitsAudio,this),this.id=t,this.status=e||{},(n=this.status).user_agent||(n.user_agent=r.browser.getUserAgent()),this.joinTime=(new Date).getTime(),this.ready=!1,this.error=null}return n(e,t),e.prototype.transmitsAudio=function(){var t,e,n;return!!(null!=(t=this.getStream())&&null!=(e=t.getAudioTracks())&&null!=(n=e[0])?n.enabled:void 0)},e.prototype.hasAudio=function(){var t,e;return!!(null!=(t=this.getStream())&&null!=(e=t.getAudioTracks())?e[0]:void 0)},e.prototype.transmitsVideo=function(){var t,e,n;return!!(null!=(t=this.getStream())&&null!=(e=t.getVideoTracks())&&null!=(n=e[0])?n.enabled:void 0)},e.prototype.hasVideo=function(){var t,e;return!!(null!=(t=this.getStream())&&null!=(e=t.getVideoTracks())?e[0]:void 0)},e.prototype.hasError=function(){return!!this.error},e.prototype.getError=function(){return this.error},e.prototype.isMuted=function(){return!!this.muted},e.prototype.isReady=function(){return!!this.ready},e.prototype.isLocal=function(){return!!this.local},e.prototype.isRemote=function(){return!this.local},e}(this.EventEmitter)}.call(this),function(){var i,o=function(t,e){return function(){return t.apply(e,arguments)}},e=function(t,e){function n(){this.constructor=t}for(var r in e)s.call(e,r)&&(t[r]=e[r]);return n.prototype=e.prototype,t.prototype=new n,t.__super__=e.prototype,t},s={}.hasOwnProperty;(i=this.palava).LocalPeer=function(t){function r(t,e,n){this.leave=o(this.leave,this),this.enableVideo=o(this.enableVideo,this),this.disableVideo=o(this.disableVideo,this),this.enableAudio=o(this.enableAudio,this),this.disableAudio=o(this.disableAudio,this),this.updateStatus=o(this.updateStatus,this),this.getStream=o(this.getStream,this),this.setupRoom=o(this.setupRoom,this),this.setupUserMedia=o(this.setupUserMedia,this),this.muted=!0,this.local=!0,r.__super__.constructor.call(this,t,e),this.room=n,this.userMedia=n.userMedia,this.setupRoom(),this.setupUserMedia()}return e(r,t),r.prototype.setupUserMedia=function(){var t,e,n;if(this.userMedia.on("stream_released",(t=this,function(){return t.ready=!1,t.emit("stream_removed")})),this.userMedia.on("stream_ready",(e=this,function(t){return e.ready=!0,e.emit("stream_ready",t)})),this.userMedia.on("stream_error",(n=this,function(t){return n.emit("stream_error",t)})),this.getStream())return this.ready=!0,this.emit("stream_ready")},r.prototype.setupRoom=function(){var t,e,n;return this.room.peers[this.id]=this.room.localPeer=this,this.on("update",(t=this,function(){return t.room.emit("peer_update",t)})),this.on("stream_ready",(e=this,function(){return e.room.emit("peer_stream_ready",e)})),this.on("stream_removed",(n=this,function(){return n.room.emit("peer_stream_removed",n)}))},r.prototype.getStream=function(){return this.userMedia.getStream()},r.prototype.updateStatus=function(t){var e,n;if(!(t&&t instanceof Object&&0!==Object.keys(t).length))return t;for(n in t)this.status[n]=t[n];return(e=this.status).user_agent||(e.user_agent=i.browser.getUserAgent()),this.room.channel.send({event:"update_status",status:this.status}),this.status},r.prototype.disableAudio=function(){var t,e,n,r,i;if(this.ready){for(r=[],t=0,e=(n=this.getStream().getAudioTracks()).length;tthis.MAX_BUFFER)return void setTimeout(this.actualSend.bind(this),1);e=(r=this.sendBuffer[0])[0],t=r[1];try{this.channel.send(e)}catch(i){return n=i,void setTimeout(this.actualSend.bind(this),1)}try{"function"==typeof t&&t()}catch(i){n=i,console.log("Exception in write callback:",n)}this.sendBuffer.shift()}else console.log("Not sending when not open!")},e}(this.EventEmitter)}.call(this),function(){var c,s=function(t,e){return function(){return t.apply(e,arguments)}},e=function(t,e){function n(){this.constructor=t}for(var r in e)i.call(e,r)&&(t[r]=e[r]);return n.prototype=e.prototype,t.prototype=new n,t.__super__=e.prototype,t},i={}.hasOwnProperty;(c=this.palava).RemotePeer=function(t){function o(t,e,n,r,i){this.closePeerConnection=s(this.closePeerConnection,this),this.oaError=s(this.oaError,this),this.sdpSender=s(this.sdpSender,this),this.sendMessage=s(this.sendMessage,this),this.sendAnswer=s(this.sendAnswer,this),this.sendOffer=s(this.sendOffer,this),this.setupRoom=s(this.setupRoom,this),this.setupDistributor=s(this.setupDistributor,this),this.setupPeerConnection=s(this.setupPeerConnection,this),this.generateIceOptions=s(this.generateIceOptions,this),this.toggleMute=s(this.toggleMute,this),this.getStream=s(this.getStream,this),this.muted=!1,this.local=!1,o.__super__.constructor.call(this,t,e),this.room=n,this.remoteStream=null,this.turnCredentials=i,this.dataChannels={},this.setupRoom(),this.setupPeerConnection(r),this.setupDistributor(),r&&this.sendOffer()}return e(o,t),o.prototype.getStream=function(){return this.remoteStream},o.prototype.toggleMute=function(){return this.muted=!this.muted},o.prototype.generateIceOptions=function(){var t;return t=[],this.room.options.stun&&t.push({urls:[this.room.options.stun]}),this.room.options.turnUrls&&this.turnCredentials&&t.push({urls:this.room.options.turnUrls,username:this.turnCredentials.user,credential:this.turnCredentials.password}),{iceServers:t}},o.prototype.setupPeerConnection=function(t){var e,n,r,i,o,s,a,u,h;if(this.peerConnection=new RTCPeerConnection(this.generateIceOptions(),c.browser.getPeerConnectionOptions()),this.peerConnection.onicecandidate=(o=this,function(t){if(t.candidate)return o.distributor.send({event:"ice_candidate",sdpmlineindex:t.candidate.sdpMLineIndex,sdpmid:t.candidate.sdpMid,candidate:t.candidate.candidate})}),this.peerConnection.ontrack=(s=this,function(t){return s.remoteStream=t.streams[0],s.ready=!0,s.emit("stream_ready")}),this.peerConnection.onremovestream=(a=this,function(){return a.remoteStream=null,a.ready=!1,a.emit("stream_removed")}),this.peerConnection.oniceconnectionstatechange=(u=this,function(t){switch(t.target.iceConnectionState){case"connecting":return u.error=null,u.emit("connection_pending");case"connected":return u.error=null,u.emit("connection_established");case"failed":return u.error="connection_failed",u.emit("connection_failed");case"disconnected":return u.error="connection_disconnected",u.emit("connection_disconnected");case"closed":return u.error="connection_closed",u.emit("connection_closed")}}),this.room.localPeer.getStream()&&this.peerConnection.addStream(this.room.localPeer.getStream()),null!=this.room.options.dataChannels)if(h=this,i=function(t){var e,n;return e=t.label,n=new c.DataChannel(t),h.dataChannels[e]=n,h.emit("channel_ready",e,n)},t)for(e in r=this.room.options.dataChannels)n=r[e],this.peerConnection.createDataChannel(e,n).onopen=function(){return i(this)};else this.peerConnection.ondatachannel=function(t){return i(t.channel)};return this.peerConnection},o.prototype.setupDistributor=function(){var t,n,e,r,i,o;return this.distributor=new c.Distributor(this.room.channel,this.id),this.distributor.on("peer_left",(t=this,function(){return t.ready&&(t.remoteStream=null,t.emit("stream_removed"),t.ready=!1),t.peerConnection.close(),t.emit("left")})),this.distributor.on("ice_candidate",(n=this,function(t){var e;if(""!==t.candidate)return e=new RTCIceCandidate({candidate:t.candidate,sdpMLineIndex:t.sdpmlineindex,sdpMid:t.sdpmid}),n.room.options.filterIceCandidateTypes.includes(e.type)?void 0:n.peerConnection.addIceCandidate(e)})),this.distributor.on("offer",(e=this,function(t){return e.peerConnection.setRemoteDescription(new RTCSessionDescription(t.sdp)),e.emit("offer"),e.sendAnswer()})),this.distributor.on("answer",(r=this,function(t){return r.peerConnection.setRemoteDescription(new RTCSessionDescription(t.sdp)),r.emit("answer")})),this.distributor.on("peer_updated_status",(i=this,function(t){return i.status=t.status,i.emit("update")})),this.distributor.on("message",(o=this,function(t){return o.emit("message",t.data)})),this.distributor},o.prototype.setupRoom=function(){var t,e,n,r,i,o,s,a,u,h,c,p,d;return(this.room.peers[this.id]=this).on("left",(t=this,function(){return delete t.room.peers[t.id],t.room.emit("peer_left",t)})),this.on("offer",(e=this,function(){return e.room.emit("peer_offer",e)})),this.on("answer",(n=this,function(){return n.room.emit("peer_answer",n)})),this.on("update",(r=this,function(){return r.room.emit("peer_update",r)})),this.on("stream_ready",(i=this,function(){return i.room.emit("peer_stream_ready",i)})),this.on("stream_removed",(o=this,function(){return o.room.emit("peer_stream_removed",o)})),this.on("connection_pending",(s=this,function(){return s.room.emit("peer_connection_pending",s)})),this.on("connection_established",(a=this,function(){return a.room.emit("peer_connection_established",a)})),this.on("connection_failed",(u=this,function(){return u.room.emit("peer_connection_failed",u)})),this.on("connection_disconnected",(h=this,function(){return h.room.emit("peer_connection_disconnected",h)})),this.on("connection_closed",(c=this,function(){return c.room.emit("peer_connection_closed",c)})),this.on("oaerror",(p=this,function(t){return p.room.emit("peer_oaerror",p,t)})),this.on("channel_ready",(d=this,function(t,e){return d.room.emit("peer_channel_ready",d,t,e)}))},o.prototype.sendOffer=function(){return this.peerConnection.createOffer(this.sdpSender("offer"),this.oaError,c.browser.getConstraints())},o.prototype.sendAnswer=function(){return this.peerConnection.createAnswer(this.sdpSender("answer"),this.oaError,c.browser.getConstraints())},o.prototype.sendMessage=function(t){return this.distributor.send({event:"message",data:t})},o.prototype.sdpSender=function(e){return n=this,function(t){return n.peerConnection.setLocalDescription(t),n.distributor.send({event:e,sdp:t})};var n},o.prototype.oaError=function(t){return this.emit("oaerror",t)},o.prototype.closePeerConnection=function(){var t;return null!=(t=this.peerConnection)&&t.close(),this.peerConnection=null},o}(c.Peer)}.call(this),function(){var u,i=function(t,e){return function(){return t.apply(e,arguments)}},n=function(t,e){function n(){this.constructor=t}for(var r in e)o.call(e,r)&&(t[r]=e[r]);return n.prototype=e.prototype,t.prototype=new n,t.__super__=e.prototype,t},o={}.hasOwnProperty;(u=this.palava).Room=function(t){function e(t,e,n,r){null==r&&(r={}),this.getAllPeers=i(this.getAllPeers,this),this.getRemotePeers=i(this.getRemotePeers,this),this.getLocalPeer=i(this.getLocalPeer,this),this.getPeerById=i(this.getPeerById,this),this.destroy=i(this.destroy,this),this.leave=i(this.leave,this),this.join=i(this.join,this),this.setupDistributor=i(this.setupDistributor,this),this.setupOptions=i(this.setupOptions,this),this.setupUserMedia=i(this.setupUserMedia,this),this.id=t,this.userMedia=n,this.channel=e,this.peers={},this.options=r,this.setupUserMedia(),this.setupDistributor(),this.setupOptions()}return n(e,t),e.prototype.setupUserMedia=function(){var e,n,t;return this.userMedia.on("stream_ready",(e=this,function(t){return e.emit("local_stream_ready",t)})),this.userMedia.on("stream_error",(n=this,function(t){return n.emit("local_stream_error",t)})),this.userMedia.on("stream_released",(t=this,function(){return t.emit("local_stream_removed")}))},e.prototype.setupOptions=function(){var t,e,n;return(t=this.options).joinTimeout||(t.joinTimeout=1e3),(e=this.options).ownStatus||(e.ownStatus={}),(n=this.options).filterIceCandidateTypes||(n.filterIceCandidateTypes=[])},e.prototype.setupDistributor=function(){var a,r,e,n;return this.distributor=new u.Distributor(this.channel),this.distributor.on("joined_room",(a=this,function(t){var e,n,r,i,o,s;for(clearTimeout(a.joinCheckTimeout),s=t.turn_user?{user:t.turn_user,password:t.turn_password}:null,new u.LocalPeer(t.own_id,a.options.ownStatus,a),e=0,n=(o=t.peers).length;ethis.MAX_BUFFER)return void setTimeout(this.actualSend.bind(this),1);e=(n=this.sendBuffer[0])[0],t=n[1];try{this.channel.send(e)}catch(i){return r=i,void setTimeout(this.actualSend.bind(this),1)}try{"function"==typeof t&&t()}catch(i){r=i,console.log("Exception in write callback:",r)}this.sendBuffer.shift()}else console.log("Not sending when not open!")},e}(this.EventEmitter)}.call(this),function(){var _,s=function(t,e){return function(){return t.apply(e,arguments)}},e=function(t,e){function r(){this.constructor=t}for(var n in e)i.call(e,n)&&(t[n]=e[n]);return r.prototype=e.prototype,t.prototype=new r,t.__super__=e.prototype,t},i={}.hasOwnProperty;(_=this.palava).RemotePeer=function(t){function o(t,e,r,n,i){this.closePeerConnection=s(this.closePeerConnection,this),this.oaError=s(this.oaError,this),this.sdpSender=s(this.sdpSender,this),this.sendMessage=s(this.sendMessage,this),this.sendAnswer=s(this.sendAnswer,this),this.sendOffer=s(this.sendOffer,this),this.setupRoom=s(this.setupRoom,this),this.setupDistributor=s(this.setupDistributor,this),this.setupPeerConnection=s(this.setupPeerConnection,this),this.generateIceOptions=s(this.generateIceOptions,this),this.toggleMute=s(this.toggleMute,this),this.getStream=s(this.getStream,this),this.muted=!1,this.local=!1,o.__super__.constructor.call(this,t,e),this.room=r,this.remoteStream=null,this.turnCredentials=i,this.dataChannels={},this.setupRoom(),this.setupPeerConnection(n),this.setupDistributor(),this.offers=n}return e(o,t),o.prototype.getStream=function(){return this.remoteStream},o.prototype.toggleMute=function(){return this.muted=!this.muted},o.prototype.generateIceOptions=function(){var t;return t=[],this.room.options.stun&&t.push({urls:[this.room.options.stun]}),this.room.options.turnUrls&&this.turnCredentials&&t.push({urls:this.room.options.turnUrls,username:this.turnCredentials.user,credential:this.turnCredentials.password}),{iceServers:t}},o.prototype.setupPeerConnection=function(t){var e,r,n,i,o,s,a,u,h,c,p,l,d,m,f,g;if(this.peerConnection=new RTCPeerConnection(this.generateIceOptions(),_.browser.getPeerConnectionOptions()),this.peerConnection.onicecandidate=(h=this,function(t){if(t.candidate)return h.distributor.send({event:"ice_candidate",sdpmlineindex:t.candidate.sdpMLineIndex,sdpmid:t.candidate.sdpMid,candidate:t.candidate.candidate})}),this.peerConnection.ontrack=(c=this,function(t){return c.remoteStream=t.streams[0],c.ready=!0,c.emit("stream_ready")}),this.peerConnection.onremovestream=(p=this,function(){return p.remoteStream=null,p.ready=!1,p.emit("stream_removed")}),this.peerConnection.onnegotiationneeded=(l=this,function(){return l.sendOffer()}),this.peerConnection.oniceconnectionstatechange=(d=this,function(t){switch(t.target.iceConnectionState){case"connecting":return d.error=null,d.emit("connection_pending");case"connected":return d.error=null,d.emit("connection_established");case"failed":return d.error="connection_failed",d.emit("connection_failed");case"disconnected":return d.error="connection_disconnected",d.emit("connection_disconnected");case"closed":return d.error="connection_closed",d.emit("connection_closed")}}),this.room.localPeer.getStream())for(e=0,n=(o=this.room.localPeer.getStream().getTracks()).length;e. function Gum(config) { this.releaseStream = bind(this.releaseStream, this); + this.getDisplayStream = bind(this.getDisplayStream, this); + this.getLocalStream = bind(this.getLocalStream, this); this.getStream = bind(this.getStream, this); + this.stopDisplaySharing = bind(this.stopDisplaySharing, this); + this.requestDisplaySharing = bind(this.requestDisplaySharing, this); this.requestStream = bind(this.requestStream, this); this.changeConfig = bind(this.changeConfig, this); this.config = config || { @@ -179,6 +183,8 @@ along with this program. If not, see . audio: true }; this.stream = null; + this.localStream = null; + this.displayStream = null; } Gum.prototype.changeConfig = function(config) { @@ -190,6 +196,7 @@ along with this program. If not, see . Gum.prototype.requestStream = function() { return navigator.mediaDevices.getUserMedia(this.config).then((function(_this) { return function(stream) { + _this.localStream = stream.clone(); _this.stream = stream; return _this.emit('stream_ready', stream); }; @@ -200,10 +207,43 @@ along with this program. If not, see . })(this)); }; + Gum.prototype.requestDisplaySharing = function() { + return navigator.mediaDevices.getDisplayMedia({ + video: true + }).then((function(_this) { + return function(stream) { + if (_this.localStream.getAudioTracks().length > 0) { + stream.addTrack(_this.localStream.getAudioTracks()[0], _this.localStream); + } + _this.displayStream = stream.clone(); + _this.stream = stream; + return _this.emit('display_stream_ready', stream); + }; + })(this))["catch"]((function(_this) { + return function(error) { + return _this.emit('display_stream_error', error); + }; + })(this)); + }; + + Gum.prototype.stopDisplaySharing = function() { + this.stream = this.localStream; + this.emit('display_stream_stop', this.displayStream); + return this.displayStream = null; + }; + Gum.prototype.getStream = function() { return this.stream; }; + Gum.prototype.getLocalStream = function() { + return this.localStream; + }; + + Gum.prototype.getDisplayStream = function() { + return this.displayStream; + }; + Gum.prototype.releaseStream = function() { if (this.stream) { this.stream.getAudioTracks().forEach((function(_this) { @@ -379,6 +419,7 @@ along with this program. If not, see . this.enableAudio = bind(this.enableAudio, this); this.disableAudio = bind(this.disableAudio, this); this.updateStatus = bind(this.updateStatus, this); + this.requestDisplaySharing = bind(this.requestDisplaySharing, this); this.getStream = bind(this.getStream, this); this.setupRoom = bind(this.setupRoom, this); this.setupUserMedia = bind(this.setupUserMedia, this); @@ -409,6 +450,16 @@ along with this program. If not, see . return _this.emit('stream_error', e); }; })(this)); + this.userMedia.on('display_stream_ready', (function(_this) { + return function(e) { + return _this.emit('display_stream_ready', e); + }; + })(this)); + this.userMedia.on('display_stream_error', (function(_this) { + return function(e) { + return _this.emit('display_stream_error', e); + }; + })(this)); if (this.getStream()) { this.ready = true; return this.emit('stream_ready'); @@ -438,6 +489,10 @@ along with this program. If not, see . return this.userMedia.getStream(); }; + LocalPeer.prototype.requestDisplaySharing = function() { + return this.userMedia.requestDisplaySharing(); + }; + LocalPeer.prototype.updateStatus = function(status) { var base, key; if (!status || !(status instanceof Object) || Object.keys(status).length === 0) { @@ -681,9 +736,7 @@ along with this program. If not, see . this.setupRoom(); this.setupPeerConnection(offers); this.setupDistributor(); - if (offers) { - this.sendOffer(); - } + this.offers = offers; } RemotePeer.prototype.getStream = function() { @@ -715,7 +768,7 @@ along with this program. If not, see . }; RemotePeer.prototype.setupPeerConnection = function(offers) { - var channel, label, options, ref, registerChannel; + var channel, i, label, len, options, ref, ref1, registerChannel, track; this.peerConnection = new RTCPeerConnection(this.generateIceOptions(), palava.browser.getPeerConnectionOptions()); this.peerConnection.onicecandidate = (function(_this) { return function(event) { @@ -743,6 +796,11 @@ along with this program. If not, see . return _this.emit('stream_removed'); }; })(this); + this.peerConnection.onnegotiationneeded = (function(_this) { + return function() { + return _this.sendOffer(); + }; + })(this); this.peerConnection.oniceconnectionstatechange = (function(_this) { return function(event) { var connectionState; @@ -767,10 +825,39 @@ along with this program. If not, see . }; })(this); if (this.room.localPeer.getStream()) { - this.peerConnection.addStream(this.room.localPeer.getStream()); - } else { - + ref = this.room.localPeer.getStream().getTracks(); + for (i = 0, len = ref.length; i < len; i++) { + track = ref[i]; + this.peerConnection.addTrack(track, this.room.localPeer.getStream()); + } } + this.room.localPeer.on('display_stream_ready', (function(_this) { + return function(stream) { + var videoSender; + videoSender = _this.peerConnection.getSenders().find(function(sender) { + return sender.track.kind === "video"; + }); + if (videoSender) { + return videoSender.replaceTrack(track); + } else { + return _this.peerConnection.addTrack(stream.getVideoTracks()[0], _this.room.localPeer.getStream()); + } + }; + })(this)); + this.room.localPeer.on('display_stream_stop', (function(_this) { + return function(stream) { + var localVideoTracks, videoSender; + localVideoTracks = _this.room.localPeer.getLocalStream().getVideoTracks(); + if (localVideoTracks.length > 0) { + videoSender = _this.peerConnection.getSenders().find(function(sender) { + return sender.track.kind === "video"; + }); + return videoSender.replaceTrack(localVideoTracks[0]); + } else { + return _this.peerConnection.removeTrack(stream.getVideoTracks()[0]); + } + }; + })(this)); if (this.room.options.dataChannels != null) { registerChannel = (function(_this) { return function(channel) { @@ -782,9 +869,9 @@ along with this program. If not, see . }; })(this); if (offers) { - ref = this.room.options.dataChannels; - for (label in ref) { - options = ref[label]; + ref1 = this.room.options.dataChannels; + for (label in ref1) { + options = ref1[label]; channel = this.peerConnection.createDataChannel(label, options); channel.onopen = function() { return registerChannel(this); @@ -832,6 +919,14 @@ along with this program. If not, see . })(this)); this.distributor.on('offer', (function(_this) { return function(msg) { + if (_this.peerConnection.signalingState !== "stable") { + if (_this.offers) { + return; + } + _this.peerConnection.setLocalDescription({ + type: "rollback" + }); + } _this.peerConnection.setRemoteDescription(new RTCSessionDescription(msg.sdp)); _this.emit('offer'); return _this.sendAnswer(); @@ -945,6 +1040,9 @@ along with this program. If not, see . RemotePeer.prototype.sdpSender = function(event) { return (function(_this) { return function(sdp) { + if (event === 'offer' && _this.peerConnection.signalingState !== 'stable') { + return; + } _this.peerConnection.setLocalDescription(sdp); return _this.distributor.send({ event: event, @@ -1598,7 +1696,7 @@ along with this program. If not, see . palava.LIB_VERSION = '2.2.0'; - palava.LIB_COMMIT = 'v2.1.0-12-g412482867a-dirty'; + palava.LIB_COMMIT = 'v2.2.0-1-gb4ef48ea59-dirty'; palava.protocol_identifier = function() { return palava.PROTOCOL_NAME = "palava.1.0"; diff --git a/palava.min.js b/palava.min.js index cd3dfa8..3a0efc4 100644 --- a/palava.min.js +++ b/palava.min.js @@ -21,4 +21,4 @@ GNU Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with this program. If not, see . */ -(function(){}).call(this),function(){this.palava={browser:{}}}.call(this),function(){"object"==typeof module&&"object"==typeof module.exports&&(module.exports=this.palava),"object"!=typeof EventEmitter&&"function"==typeof require?this.EventEmitter=require("wolfy87-eventemitter"):this.EventEmitter=EventEmitter,"object"!=typeof adapter&&"function"==typeof require?this.adapter=require("webrtc-adapter/out/adapter_no_edge"):this.adapter=adapter}.call(this),function(){var t,r;r=this.palava,t=this.adapter,r.browser.isMozilla=function(){return"firefox"===t.browserDetails.browser},r.browser.isChrome=function(){return"chrome"===t.browserDetails.browser},r.browser.getUserAgent=function(){return t.browserDetails.browser},r.browser.getUserAgentVersion=function(){return t.browserDetails.version},r.browser.checkForWebrtcError=function(){try{new window.RTCPeerConnection({iceServers:[]})}catch(t){return t}return!(window.RTCPeerConnection&&window.RTCIceCandidate&&window.RTCSessionDescription&&navigator.mediaDevices&&navigator.mediaDevices.getUserMedia)},r.browser.getConstraints=function(){return{optional:[],mandatory:{OfferToReceiveAudio:!0,OfferToReceiveVideo:!0}}},r.browser.getPeerConnectionOptions=function(){return r.browser.isChrome()?{optional:[{DtlsSrtpKeyAgreement:!0}]}:{}},r.browser.registerFullscreen=function(t,e){return console.log("DEPRECATED: palava.browser.registerFullscreen will be removed from the palava library in early 2021"),t.requestFullscreen?t.addEventListener(e,function(){return this.requestFullscreen()}):t.mozRequestFullScreen?t.addEventListener(e,function(){return this.mozRequestFullScreen()}):t.webkitRequestFullscreen?t.addEventListener(e,function(){return this.webkitRequestFullscreen()}):void 0},r.browser.attachMediaStream=function(t,e){return e?t.srcObject=e:(t.pause(),t.srcObject=null)},r.browser.attachPeer=function(t,e){var n;return n=function(){return r.browser.attachMediaStream(t,e.getStream()),e.isLocal()&&t.setAttribute("muted",!0),t.play()},e.getStream()?n():e.on("stream_ready",function(){return n()})}}.call(this),function(){var n=function(t,e){return function(){return t.apply(e,arguments)}},r=function(t,e){function n(){this.constructor=t}for(var r in e)i.call(e,r)&&(t[r]=e[r]);return n.prototype=e.prototype,t.prototype=new n,t.__super__=e.prototype,t},i={}.hasOwnProperty;this.palava.Gum=function(t){function e(t){this.releaseStream=n(this.releaseStream,this),this.getStream=n(this.getStream,this),this.requestStream=n(this.requestStream,this),this.changeConfig=n(this.changeConfig,this),this.config=t||{video:!0,audio:!0},this.stream=null}return r(e,t),e.prototype.changeConfig=function(t){return this.config=t,this.releaseStream(),this.requestStream()},e.prototype.requestStream=function(){return navigator.mediaDevices.getUserMedia(this.config).then((n=this,function(t){return n.stream=t,n.emit("stream_ready",t)}))["catch"]((e=this,function(t){return e.emit("stream_error",t)}));var e,n},e.prototype.getStream=function(){return this.stream},e.prototype.releaseStream=function(){return!!this.stream&&(this.stream.getAudioTracks().forEach(function(t){return t.stop()}),this.stream.getVideoTracks().forEach(function(t){return t.stop()}),this.stream=null,this.emit("stream_released",this),!0)},e}(this.EventEmitter)}.call(this),function(){var e,n=function(t,e){return function(){return t.apply(e,arguments)}};(e=this.palava).Identity=function(){function t(t){this.getStatus=n(this.getStatus,this),this.getName=n(this.getName,this),this.userMediaConfig=t.userMediaConfig,this.status=t.status||{},this.status.name=t.name}return t.prototype.newUserMedia=function(){return new e.Gum(this.userMediaConfig)},t.prototype.getName=function(){return this.name},t.prototype.getStatus=function(){return this.status},t}()}.call(this),function(){var r,i=function(t,e){return function(){return t.apply(e,arguments)}},n=function(t,e){function n(){this.constructor=t}for(var r in e)o.call(e,r)&&(t[r]=e[r]);return n.prototype=e.prototype,t.prototype=new n,t.__super__=e.prototype,t},o={}.hasOwnProperty;(r=this.palava).Peer=function(t){function e(t,e){var n;this.isRemote=i(this.isRemote,this),this.isLocal=i(this.isLocal,this),this.isReady=i(this.isReady,this),this.isMuted=i(this.isMuted,this),this.getError=i(this.getError,this),this.hasError=i(this.hasError,this),this.hasVideo=i(this.hasVideo,this),this.transmitsVideo=i(this.transmitsVideo,this),this.hasAudio=i(this.hasAudio,this),this.transmitsAudio=i(this.transmitsAudio,this),this.id=t,this.status=e||{},(n=this.status).user_agent||(n.user_agent=r.browser.getUserAgent()),this.joinTime=(new Date).getTime(),this.ready=!1,this.error=null}return n(e,t),e.prototype.transmitsAudio=function(){var t,e,n;return!!(null!=(t=this.getStream())&&null!=(e=t.getAudioTracks())&&null!=(n=e[0])?n.enabled:void 0)},e.prototype.hasAudio=function(){var t,e;return!!(null!=(t=this.getStream())&&null!=(e=t.getAudioTracks())?e[0]:void 0)},e.prototype.transmitsVideo=function(){var t,e,n;return!!(null!=(t=this.getStream())&&null!=(e=t.getVideoTracks())&&null!=(n=e[0])?n.enabled:void 0)},e.prototype.hasVideo=function(){var t,e;return!!(null!=(t=this.getStream())&&null!=(e=t.getVideoTracks())?e[0]:void 0)},e.prototype.hasError=function(){return!!this.error},e.prototype.getError=function(){return this.error},e.prototype.isMuted=function(){return!!this.muted},e.prototype.isReady=function(){return!!this.ready},e.prototype.isLocal=function(){return!!this.local},e.prototype.isRemote=function(){return!this.local},e}(this.EventEmitter)}.call(this),function(){var i,o=function(t,e){return function(){return t.apply(e,arguments)}},e=function(t,e){function n(){this.constructor=t}for(var r in e)s.call(e,r)&&(t[r]=e[r]);return n.prototype=e.prototype,t.prototype=new n,t.__super__=e.prototype,t},s={}.hasOwnProperty;(i=this.palava).LocalPeer=function(t){function r(t,e,n){this.leave=o(this.leave,this),this.enableVideo=o(this.enableVideo,this),this.disableVideo=o(this.disableVideo,this),this.enableAudio=o(this.enableAudio,this),this.disableAudio=o(this.disableAudio,this),this.updateStatus=o(this.updateStatus,this),this.getStream=o(this.getStream,this),this.setupRoom=o(this.setupRoom,this),this.setupUserMedia=o(this.setupUserMedia,this),this.muted=!0,this.local=!0,r.__super__.constructor.call(this,t,e),this.room=n,this.userMedia=n.userMedia,this.setupRoom(),this.setupUserMedia()}return e(r,t),r.prototype.setupUserMedia=function(){var t,e,n;if(this.userMedia.on("stream_released",(t=this,function(){return t.ready=!1,t.emit("stream_removed")})),this.userMedia.on("stream_ready",(e=this,function(t){return e.ready=!0,e.emit("stream_ready",t)})),this.userMedia.on("stream_error",(n=this,function(t){return n.emit("stream_error",t)})),this.getStream())return this.ready=!0,this.emit("stream_ready")},r.prototype.setupRoom=function(){var t,e,n;return this.room.peers[this.id]=this.room.localPeer=this,this.on("update",(t=this,function(){return t.room.emit("peer_update",t)})),this.on("stream_ready",(e=this,function(){return e.room.emit("peer_stream_ready",e)})),this.on("stream_removed",(n=this,function(){return n.room.emit("peer_stream_removed",n)}))},r.prototype.getStream=function(){return this.userMedia.getStream()},r.prototype.updateStatus=function(t){var e,n;if(!(t&&t instanceof Object&&0!==Object.keys(t).length))return t;for(n in t)this.status[n]=t[n];return(e=this.status).user_agent||(e.user_agent=i.browser.getUserAgent()),this.room.channel.send({event:"update_status",status:this.status}),this.status},r.prototype.disableAudio=function(){var t,e,n,r,i;if(this.ready){for(r=[],t=0,e=(n=this.getStream().getAudioTracks()).length;tthis.MAX_BUFFER)return void setTimeout(this.actualSend.bind(this),1);e=(r=this.sendBuffer[0])[0],t=r[1];try{this.channel.send(e)}catch(i){return n=i,void setTimeout(this.actualSend.bind(this),1)}try{"function"==typeof t&&t()}catch(i){n=i,console.log("Exception in write callback:",n)}this.sendBuffer.shift()}else console.log("Not sending when not open!")},e}(this.EventEmitter)}.call(this),function(){var c,s=function(t,e){return function(){return t.apply(e,arguments)}},e=function(t,e){function n(){this.constructor=t}for(var r in e)i.call(e,r)&&(t[r]=e[r]);return n.prototype=e.prototype,t.prototype=new n,t.__super__=e.prototype,t},i={}.hasOwnProperty;(c=this.palava).RemotePeer=function(t){function o(t,e,n,r,i){this.closePeerConnection=s(this.closePeerConnection,this),this.oaError=s(this.oaError,this),this.sdpSender=s(this.sdpSender,this),this.sendMessage=s(this.sendMessage,this),this.sendAnswer=s(this.sendAnswer,this),this.sendOffer=s(this.sendOffer,this),this.setupRoom=s(this.setupRoom,this),this.setupDistributor=s(this.setupDistributor,this),this.setupPeerConnection=s(this.setupPeerConnection,this),this.generateIceOptions=s(this.generateIceOptions,this),this.toggleMute=s(this.toggleMute,this),this.getStream=s(this.getStream,this),this.muted=!1,this.local=!1,o.__super__.constructor.call(this,t,e),this.room=n,this.remoteStream=null,this.turnCredentials=i,this.dataChannels={},this.setupRoom(),this.setupPeerConnection(r),this.setupDistributor(),r&&this.sendOffer()}return e(o,t),o.prototype.getStream=function(){return this.remoteStream},o.prototype.toggleMute=function(){return this.muted=!this.muted},o.prototype.generateIceOptions=function(){var t;return t=[],this.room.options.stun&&t.push({urls:[this.room.options.stun]}),this.room.options.turnUrls&&this.turnCredentials&&t.push({urls:this.room.options.turnUrls,username:this.turnCredentials.user,credential:this.turnCredentials.password}),{iceServers:t}},o.prototype.setupPeerConnection=function(t){var e,n,r,i,o,s,a,u,h;if(this.peerConnection=new RTCPeerConnection(this.generateIceOptions(),c.browser.getPeerConnectionOptions()),this.peerConnection.onicecandidate=(o=this,function(t){if(t.candidate)return o.distributor.send({event:"ice_candidate",sdpmlineindex:t.candidate.sdpMLineIndex,sdpmid:t.candidate.sdpMid,candidate:t.candidate.candidate})}),this.peerConnection.ontrack=(s=this,function(t){return s.remoteStream=t.streams[0],s.ready=!0,s.emit("stream_ready")}),this.peerConnection.onremovestream=(a=this,function(){return a.remoteStream=null,a.ready=!1,a.emit("stream_removed")}),this.peerConnection.oniceconnectionstatechange=(u=this,function(t){switch(t.target.iceConnectionState){case"connecting":return u.error=null,u.emit("connection_pending");case"connected":return u.error=null,u.emit("connection_established");case"failed":return u.error="connection_failed",u.emit("connection_failed");case"disconnected":return u.error="connection_disconnected",u.emit("connection_disconnected");case"closed":return u.error="connection_closed",u.emit("connection_closed")}}),this.room.localPeer.getStream()&&this.peerConnection.addStream(this.room.localPeer.getStream()),null!=this.room.options.dataChannels)if(h=this,i=function(t){var e,n;return e=t.label,n=new c.DataChannel(t),h.dataChannels[e]=n,h.emit("channel_ready",e,n)},t)for(e in r=this.room.options.dataChannels)n=r[e],this.peerConnection.createDataChannel(e,n).onopen=function(){return i(this)};else this.peerConnection.ondatachannel=function(t){return i(t.channel)};return this.peerConnection},o.prototype.setupDistributor=function(){var t,n,e,r,i,o;return this.distributor=new c.Distributor(this.room.channel,this.id),this.distributor.on("peer_left",(t=this,function(){return t.ready&&(t.remoteStream=null,t.emit("stream_removed"),t.ready=!1),t.peerConnection.close(),t.emit("left")})),this.distributor.on("ice_candidate",(n=this,function(t){var e;if(""!==t.candidate)return e=new RTCIceCandidate({candidate:t.candidate,sdpMLineIndex:t.sdpmlineindex,sdpMid:t.sdpmid}),n.room.options.filterIceCandidateTypes.includes(e.type)?void 0:n.peerConnection.addIceCandidate(e)})),this.distributor.on("offer",(e=this,function(t){return e.peerConnection.setRemoteDescription(new RTCSessionDescription(t.sdp)),e.emit("offer"),e.sendAnswer()})),this.distributor.on("answer",(r=this,function(t){return r.peerConnection.setRemoteDescription(new RTCSessionDescription(t.sdp)),r.emit("answer")})),this.distributor.on("peer_updated_status",(i=this,function(t){return i.status=t.status,i.emit("update")})),this.distributor.on("message",(o=this,function(t){return o.emit("message",t.data)})),this.distributor},o.prototype.setupRoom=function(){var t,e,n,r,i,o,s,a,u,h,c,p,d;return(this.room.peers[this.id]=this).on("left",(t=this,function(){return delete t.room.peers[t.id],t.room.emit("peer_left",t)})),this.on("offer",(e=this,function(){return e.room.emit("peer_offer",e)})),this.on("answer",(n=this,function(){return n.room.emit("peer_answer",n)})),this.on("update",(r=this,function(){return r.room.emit("peer_update",r)})),this.on("stream_ready",(i=this,function(){return i.room.emit("peer_stream_ready",i)})),this.on("stream_removed",(o=this,function(){return o.room.emit("peer_stream_removed",o)})),this.on("connection_pending",(s=this,function(){return s.room.emit("peer_connection_pending",s)})),this.on("connection_established",(a=this,function(){return a.room.emit("peer_connection_established",a)})),this.on("connection_failed",(u=this,function(){return u.room.emit("peer_connection_failed",u)})),this.on("connection_disconnected",(h=this,function(){return h.room.emit("peer_connection_disconnected",h)})),this.on("connection_closed",(c=this,function(){return c.room.emit("peer_connection_closed",c)})),this.on("oaerror",(p=this,function(t){return p.room.emit("peer_oaerror",p,t)})),this.on("channel_ready",(d=this,function(t,e){return d.room.emit("peer_channel_ready",d,t,e)}))},o.prototype.sendOffer=function(){return this.peerConnection.createOffer(this.sdpSender("offer"),this.oaError,c.browser.getConstraints())},o.prototype.sendAnswer=function(){return this.peerConnection.createAnswer(this.sdpSender("answer"),this.oaError,c.browser.getConstraints())},o.prototype.sendMessage=function(t){return this.distributor.send({event:"message",data:t})},o.prototype.sdpSender=function(e){return n=this,function(t){return n.peerConnection.setLocalDescription(t),n.distributor.send({event:e,sdp:t})};var n},o.prototype.oaError=function(t){return this.emit("oaerror",t)},o.prototype.closePeerConnection=function(){var t;return null!=(t=this.peerConnection)&&t.close(),this.peerConnection=null},o}(c.Peer)}.call(this),function(){var u,i=function(t,e){return function(){return t.apply(e,arguments)}},n=function(t,e){function n(){this.constructor=t}for(var r in e)o.call(e,r)&&(t[r]=e[r]);return n.prototype=e.prototype,t.prototype=new n,t.__super__=e.prototype,t},o={}.hasOwnProperty;(u=this.palava).Room=function(t){function e(t,e,n,r){null==r&&(r={}),this.getAllPeers=i(this.getAllPeers,this),this.getRemotePeers=i(this.getRemotePeers,this),this.getLocalPeer=i(this.getLocalPeer,this),this.getPeerById=i(this.getPeerById,this),this.destroy=i(this.destroy,this),this.leave=i(this.leave,this),this.join=i(this.join,this),this.setupDistributor=i(this.setupDistributor,this),this.setupOptions=i(this.setupOptions,this),this.setupUserMedia=i(this.setupUserMedia,this),this.id=t,this.userMedia=n,this.channel=e,this.peers={},this.options=r,this.setupUserMedia(),this.setupDistributor(),this.setupOptions()}return n(e,t),e.prototype.setupUserMedia=function(){var e,n,t;return this.userMedia.on("stream_ready",(e=this,function(t){return e.emit("local_stream_ready",t)})),this.userMedia.on("stream_error",(n=this,function(t){return n.emit("local_stream_error",t)})),this.userMedia.on("stream_released",(t=this,function(){return t.emit("local_stream_removed")}))},e.prototype.setupOptions=function(){var t,e,n;return(t=this.options).joinTimeout||(t.joinTimeout=1e3),(e=this.options).ownStatus||(e.ownStatus={}),(n=this.options).filterIceCandidateTypes||(n.filterIceCandidateTypes=[])},e.prototype.setupDistributor=function(){var a,r,e,n;return this.distributor=new u.Distributor(this.channel),this.distributor.on("joined_room",(a=this,function(t){var e,n,r,i,o,s;for(clearTimeout(a.joinCheckTimeout),s=t.turn_user?{user:t.turn_user,password:t.turn_password}:null,new u.LocalPeer(t.own_id,a.options.ownStatus,a),e=0,n=(o=t.peers).length;ethis.MAX_BUFFER)return void setTimeout(this.actualSend.bind(this),1);e=(n=this.sendBuffer[0])[0],t=n[1];try{this.channel.send(e)}catch(i){return r=i,void setTimeout(this.actualSend.bind(this),1)}try{"function"==typeof t&&t()}catch(i){r=i,console.log("Exception in write callback:",r)}this.sendBuffer.shift()}else console.log("Not sending when not open!")},e}(this.EventEmitter)}.call(this),function(){var _,s=function(t,e){return function(){return t.apply(e,arguments)}},e=function(t,e){function r(){this.constructor=t}for(var n in e)i.call(e,n)&&(t[n]=e[n]);return r.prototype=e.prototype,t.prototype=new r,t.__super__=e.prototype,t},i={}.hasOwnProperty;(_=this.palava).RemotePeer=function(t){function o(t,e,r,n,i){this.closePeerConnection=s(this.closePeerConnection,this),this.oaError=s(this.oaError,this),this.sdpSender=s(this.sdpSender,this),this.sendMessage=s(this.sendMessage,this),this.sendAnswer=s(this.sendAnswer,this),this.sendOffer=s(this.sendOffer,this),this.setupRoom=s(this.setupRoom,this),this.setupDistributor=s(this.setupDistributor,this),this.setupPeerConnection=s(this.setupPeerConnection,this),this.generateIceOptions=s(this.generateIceOptions,this),this.toggleMute=s(this.toggleMute,this),this.getStream=s(this.getStream,this),this.muted=!1,this.local=!1,o.__super__.constructor.call(this,t,e),this.room=r,this.remoteStream=null,this.turnCredentials=i,this.dataChannels={},this.setupRoom(),this.setupPeerConnection(n),this.setupDistributor(),this.offers=n}return e(o,t),o.prototype.getStream=function(){return this.remoteStream},o.prototype.toggleMute=function(){return this.muted=!this.muted},o.prototype.generateIceOptions=function(){var t;return t=[],this.room.options.stun&&t.push({urls:[this.room.options.stun]}),this.room.options.turnUrls&&this.turnCredentials&&t.push({urls:this.room.options.turnUrls,username:this.turnCredentials.user,credential:this.turnCredentials.password}),{iceServers:t}},o.prototype.setupPeerConnection=function(t){var e,r,n,i,o,s,a,u,h,c,p,l,d,m,f,g;if(this.peerConnection=new RTCPeerConnection(this.generateIceOptions(),_.browser.getPeerConnectionOptions()),this.peerConnection.onicecandidate=(h=this,function(t){if(t.candidate)return h.distributor.send({event:"ice_candidate",sdpmlineindex:t.candidate.sdpMLineIndex,sdpmid:t.candidate.sdpMid,candidate:t.candidate.candidate})}),this.peerConnection.ontrack=(c=this,function(t){return c.remoteStream=t.streams[0],c.ready=!0,c.emit("stream_ready")}),this.peerConnection.onremovestream=(p=this,function(){return p.remoteStream=null,p.ready=!1,p.emit("stream_removed")}),this.peerConnection.onnegotiationneeded=(l=this,function(){return l.sendOffer()}),this.peerConnection.oniceconnectionstatechange=(d=this,function(t){switch(t.target.iceConnectionState){case"connecting":return d.error=null,d.emit("connection_pending");case"connected":return d.error=null,d.emit("connection_established");case"failed":return d.error="connection_failed",d.emit("connection_failed");case"disconnected":return d.error="connection_disconnected",d.emit("connection_disconnected");case"closed":return d.error="connection_closed",d.emit("connection_closed")}}),this.room.localPeer.getStream())for(e=0,n=(o=this.room.localPeer.getStream().getTracks()).length;e