index.js 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263
  1. /**
  2. * Module dependencies.
  3. */
  4. var Emitter = require('events').EventEmitter;
  5. /**
  6. * Module exports.
  7. */
  8. module.exports = Adapter;
  9. /**
  10. * Memory adapter constructor.
  11. *
  12. * @param {Namespace} nsp
  13. * @api public
  14. */
  15. function Adapter(nsp){
  16. this.nsp = nsp;
  17. this.rooms = {};
  18. this.sids = {};
  19. this.encoder = nsp.server.encoder;
  20. }
  21. /**
  22. * Inherits from `EventEmitter`.
  23. */
  24. Adapter.prototype.__proto__ = Emitter.prototype;
  25. /**
  26. * Adds a socket to a room.
  27. *
  28. * @param {String} socket id
  29. * @param {String} room name
  30. * @param {Function} callback
  31. * @api public
  32. */
  33. Adapter.prototype.add = function(id, room, fn){
  34. return this.addAll(id, [ room ], fn);
  35. };
  36. /**
  37. * Adds a socket to a list of room.
  38. *
  39. * @param {String} socket id
  40. * @param {String} rooms
  41. * @param {Function} callback
  42. * @api public
  43. */
  44. Adapter.prototype.addAll = function(id, rooms, fn){
  45. for (var i = 0; i < rooms.length; i++) {
  46. var room = rooms[i];
  47. this.sids[id] = this.sids[id] || {};
  48. this.sids[id][room] = true;
  49. this.rooms[room] = this.rooms[room] || Room();
  50. this.rooms[room].add(id);
  51. }
  52. if (fn) process.nextTick(fn.bind(null, null));
  53. };
  54. /**
  55. * Removes a socket from a room.
  56. *
  57. * @param {String} socket id
  58. * @param {String} room name
  59. * @param {Function} callback
  60. * @api public
  61. */
  62. Adapter.prototype.del = function(id, room, fn){
  63. if (this.sids[id]) delete this.sids[id][room];
  64. if (this.rooms.hasOwnProperty(room)) {
  65. this.rooms[room].del(id);
  66. if (this.rooms[room].length === 0) delete this.rooms[room];
  67. }
  68. if (fn) process.nextTick(fn.bind(null, null));
  69. };
  70. /**
  71. * Removes a socket from all rooms it's joined.
  72. *
  73. * @param {String} socket id
  74. * @param {Function} callback
  75. * @api public
  76. */
  77. Adapter.prototype.delAll = function(id, fn){
  78. var rooms = this.sids[id];
  79. if (rooms) {
  80. for (var room in rooms) {
  81. if (this.rooms.hasOwnProperty(room)) {
  82. this.rooms[room].del(id);
  83. if (this.rooms[room].length === 0) delete this.rooms[room];
  84. }
  85. }
  86. }
  87. delete this.sids[id];
  88. if (fn) process.nextTick(fn.bind(null, null));
  89. };
  90. /**
  91. * Broadcasts a packet.
  92. *
  93. * Options:
  94. * - `flags` {Object} flags for this packet
  95. * - `except` {Array} sids that should be excluded
  96. * - `rooms` {Array} list of rooms to broadcast to
  97. *
  98. * @param {Object} packet object
  99. * @api public
  100. */
  101. Adapter.prototype.broadcast = function(packet, opts){
  102. var rooms = opts.rooms || [];
  103. var except = opts.except || [];
  104. var flags = opts.flags || {};
  105. var packetOpts = {
  106. preEncoded: true,
  107. volatile: flags.volatile,
  108. compress: flags.compress
  109. };
  110. var ids = {};
  111. var self = this;
  112. var socket;
  113. packet.nsp = this.nsp.name;
  114. this.encoder.encode(packet, function(encodedPackets) {
  115. if (rooms.length) {
  116. for (var i = 0; i < rooms.length; i++) {
  117. var room = self.rooms[rooms[i]];
  118. if (!room) continue;
  119. var sockets = room.sockets;
  120. for (var id in sockets) {
  121. if (sockets.hasOwnProperty(id)) {
  122. if (ids[id] || ~except.indexOf(id)) continue;
  123. socket = self.nsp.connected[id];
  124. if (socket) {
  125. socket.packet(encodedPackets, packetOpts);
  126. ids[id] = true;
  127. }
  128. }
  129. }
  130. }
  131. } else {
  132. for (var id in self.sids) {
  133. if (self.sids.hasOwnProperty(id)) {
  134. if (~except.indexOf(id)) continue;
  135. socket = self.nsp.connected[id];
  136. if (socket) socket.packet(encodedPackets, packetOpts);
  137. }
  138. }
  139. }
  140. });
  141. };
  142. /**
  143. * Gets a list of clients by sid.
  144. *
  145. * @param {Array} explicit set of rooms to check.
  146. * @param {Function} callback
  147. * @api public
  148. */
  149. Adapter.prototype.clients = function(rooms, fn){
  150. if ('function' == typeof rooms){
  151. fn = rooms;
  152. rooms = null;
  153. }
  154. rooms = rooms || [];
  155. var ids = {};
  156. var sids = [];
  157. var socket;
  158. if (rooms.length) {
  159. for (var i = 0; i < rooms.length; i++) {
  160. var room = this.rooms[rooms[i]];
  161. if (!room) continue;
  162. var sockets = room.sockets;
  163. for (var id in sockets) {
  164. if (sockets.hasOwnProperty(id)) {
  165. if (ids[id]) continue;
  166. socket = this.nsp.connected[id];
  167. if (socket) {
  168. sids.push(id);
  169. ids[id] = true;
  170. }
  171. }
  172. }
  173. }
  174. } else {
  175. for (var id in this.sids) {
  176. if (this.sids.hasOwnProperty(id)) {
  177. socket = this.nsp.connected[id];
  178. if (socket) sids.push(id);
  179. }
  180. }
  181. }
  182. if (fn) process.nextTick(fn.bind(null, null, sids));
  183. };
  184. /**
  185. * Gets the list of rooms a given client has joined.
  186. *
  187. * @param {String} socket id
  188. * @param {Function} callback
  189. * @api public
  190. */
  191. Adapter.prototype.clientRooms = function(id, fn){
  192. var rooms = this.sids[id];
  193. if (fn) process.nextTick(fn.bind(null, null, rooms ? Object.keys(rooms) : null));
  194. };
  195. /**
  196. * Room constructor.
  197. *
  198. * @api private
  199. */
  200. function Room(){
  201. if (!(this instanceof Room)) return new Room();
  202. this.sockets = {};
  203. this.length = 0;
  204. }
  205. /**
  206. * Adds a socket to a room.
  207. *
  208. * @param {String} socket id
  209. * @api private
  210. */
  211. Room.prototype.add = function(id){
  212. if (!this.sockets.hasOwnProperty(id)) {
  213. this.sockets[id] = true;
  214. this.length++;
  215. }
  216. };
  217. /**
  218. * Removes a socket from a room.
  219. *
  220. * @param {String} socket id
  221. * @api private
  222. */
  223. Room.prototype.del = function(id){
  224. if (this.sockets.hasOwnProperty(id)) {
  225. delete this.sockets[id];
  226. this.length--;
  227. }
  228. };