io.js 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221
  1. (function() {
  2. var rtc = this.rtc = {};
  3. // Fallbacks for vendor-specific variables until the spec is finalized.
  4. var PeerConnection = window.PeerConnection || window.webkitPeerConnection00;
  5. var URL = window.URL || window.webkitURL || window.msURL || window.oURL;
  6. var getUserMedia = navigator.getUserMedia || navigator.webkitGetUserMedia ||
  7. navigator.mozGetUserMedia || navigator.msGetUserMedia;
  8. // Holds a connection to the server.
  9. rtc._socket = null;
  10. // Holds callbacks for certain events.
  11. rtc._events = {};
  12. // Holds the STUN server to use for PeerConnections.
  13. rtc.SERVER = "STUN stun.l.google.com:19302";
  14. // Reference to the lone PeerConnection instance.
  15. rtc.peerConnections = {};
  16. // Array of known peer socket ids
  17. rtc.connections = [];
  18. // Stream-related variables.
  19. rtc.streams = [];
  20. rtc.numStreams = 0;
  21. rtc.initializedStreams = 0;
  22. /**
  23. * Connects to the socket.io server.
  24. */
  25. rtc.connect = function(server) {
  26. rtc._socket = io.connect(server);
  27. rtc._socket.on('connect', function() {
  28. rtc.fire('connect');
  29. });
  30. // TODO: Fix possible race condition if get peers is not emitted
  31. // before the "ready" event is fired.
  32. rtc._socket.on('get peers', function(data) {
  33. var peers = data.connections;
  34. rtc.connections = peers;
  35. // fire connections event and pass peers
  36. rtc.fire('connections', peers);
  37. });
  38. rtc._socket.on('receive ice candidate', function(data) {
  39. var candidate = new IceCandidate(data.label, data.candidate);
  40. rtc.peerConnections[data.socketId].processIceMessage(candidate);
  41. rtc.fire('receive ice candidate', candidate);
  42. });
  43. rtc._socket.on('new peer connected', function(data) {
  44. var pc = rtc.createPeerConnection(data.socketId);
  45. for (var i = 0; i < rtc.streams.length; i++) {
  46. var stream = rtc.streams[i];
  47. pc.addStream(stream);
  48. }
  49. });
  50. rtc._socket.on('remove peer connected', function(data) {
  51. console.log(data);
  52. //the actual onremovestream function is not yet supported. Here is a temporary workaround
  53. rtc.fire('disconnect stream', data.socketId);
  54. onClose(data.socketId);
  55. //rtc.peerConnections[data.socketId].close();
  56. delete rtc.peerConnections[data.socketId];
  57. });
  58. rtc._socket.on('receive offer', function(data) {
  59. rtc.receiveOffer(data.socketId, data.sdp);
  60. rtc.fire('receive offer', data);
  61. });
  62. rtc._socket.on('receive answer', function(data) {
  63. rtc.receiveAnswer(data.socketId, data.sdp);
  64. rtc.fire('receive answer', data);
  65. });
  66. };
  67. rtc.sendOffers = function() {
  68. for (var i = 0, len = rtc.connections.length; i < len; i++) {
  69. var socketId = rtc.connections[i];
  70. rtc.sendOffer(socketId);
  71. }
  72. }
  73. rtc.onClose = function(data) {
  74. rtc._socket.on('close stream', function() {
  75. rtc.fire('close stream', data);
  76. });
  77. }
  78. rtc.on = function(eventName, callback) {
  79. rtc._events[eventName] = rtc._events[eventName] || [];
  80. rtc._events[eventName].push(callback);
  81. };
  82. rtc.fire = function(eventName, _) {
  83. var events = rtc._events[eventName];
  84. var args = Array.prototype.slice.call(arguments, 1);
  85. if (!events) {
  86. return;
  87. }
  88. for (var i = 0, len = events.length; i < len; i++) {
  89. events[i].apply(null, args);
  90. }
  91. };
  92. rtc.createPeerConnections = function() {
  93. for (var i = 0; i < rtc.connections.length; i++) {
  94. rtc.createPeerConnection(rtc.connections[i]);
  95. }
  96. };
  97. rtc.createPeerConnection = function(id) {
  98. var pc = rtc.peerConnections[id] = new PeerConnection(rtc.SERVER, function(candidate, moreToFollow) {
  99. if (candidate) {
  100. rtc._socket.emit('receive ice candidate', {
  101. label: candidate.label,
  102. candidate: candidate.toSdp(),
  103. socketId: id
  104. });
  105. }
  106. rtc.fire('ice candidate', candidate, moreToFollow);
  107. });
  108. pc.onopen = function() {
  109. // TODO: Finalize this API
  110. rtc.fire('peer connection opened');
  111. };
  112. pc.onaddstream = function(event) {
  113. // TODO: Finalize this API
  114. rtc.fire('add remote stream', event.stream, id);
  115. };
  116. return pc;
  117. };
  118. rtc.sendOffer = function(socketId) {
  119. var pc = rtc.peerConnections[socketId];
  120. // TODO: Abstract away video: true, audio: true for offers
  121. var offer = pc.createOffer({ video: true, audio: true });
  122. pc.setLocalDescription(pc.SDP_OFFER, offer);
  123. rtc._socket.emit('send offer', { socketId: socketId, sdp: offer.toSdp() });
  124. pc.startIce();
  125. };
  126. rtc.receiveOffer = function(socketId, sdp) {
  127. var pc = rtc.peerConnections[socketId];
  128. pc.setRemoteDescription(pc.SDP_OFFER, new SessionDescription(sdp));
  129. rtc.sendAnswer(socketId);
  130. };
  131. rtc.sendAnswer = function(socketId) {
  132. var pc = rtc.peerConnections[socketId];
  133. var offer = pc.remoteDescription;
  134. // TODO: Abstract away video: true, audio: true for answers
  135. var answer = pc.createAnswer(offer.toSdp(), {video: true, audio: true});
  136. pc.setLocalDescription(pc.SDP_ANSWER, answer);
  137. rtc._socket.emit('send answer', { socketId: socketId, sdp: answer.toSdp() });
  138. pc.startIce();
  139. };
  140. rtc.receiveAnswer = function(socketId, sdp) {
  141. var pc = rtc.peerConnections[socketId];
  142. pc.setRemoteDescription(pc.SDP_ANSWER, new SessionDescription(sdp));
  143. };
  144. rtc.createStream = function(domId, onSuccess, onFail) {
  145. var el = document.getElementById(domId);
  146. var options;
  147. onSuccess = onSuccess || function() {};
  148. onFail = onFail || function() {};
  149. if (el.tagName.toLowerCase() === "audio") {
  150. options = { audio: true };
  151. } else {
  152. options = { video: true, audio: true };
  153. }
  154. if (getUserMedia) {
  155. rtc.numStreams++;
  156. getUserMedia.call(navigator, options, function(stream) {
  157. el.src = URL.createObjectURL(stream);
  158. rtc.streams.push(stream);
  159. rtc.initializedStreams++;
  160. onSuccess(stream);
  161. if (rtc.initializedStreams === rtc.numStreams) {
  162. rtc.fire('ready');
  163. }
  164. }, function() {
  165. alert("Could not connect stream.");
  166. onFail();
  167. });
  168. } else {
  169. alert('webRTC is not yet supported in this browser.');
  170. }
  171. }
  172. rtc.addStreams = function() {
  173. for (var i = 0; i < rtc.streams.length; i++) {
  174. var stream = rtc.streams[i];
  175. for (var connection in rtc.peerConnections) {
  176. rtc.peerConnections[connection].addStream(stream);
  177. }
  178. }
  179. };
  180. rtc.attachStream = function(stream, domId) {
  181. document.getElementById(domId).src = URL.createObjectURL(stream);
  182. };
  183. rtc.on('ready', function() {
  184. rtc.createPeerConnections();
  185. rtc.addStreams();
  186. rtc.sendOffers();
  187. });
  188. }).call(this);