Créé dans le cadre du projet de fin d'année de la promo 2018 de CIR2 de l'ISEN Brest/Rennes, le Burger Quizz est une adaptation numérique du jeu télévisé éponyme, plus précisément d'une épreuve spécifique de ce jeu : le "Sel ou Poivre".

client.js 5.1KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247
  1. /**
  2. * Module dependencies.
  3. */
  4. var parser = require('socket.io-parser');
  5. var debug = require('debug')('socket.io:client');
  6. /**
  7. * Module exports.
  8. */
  9. module.exports = Client;
  10. /**
  11. * Client constructor.
  12. *
  13. * @param {Server} server instance
  14. * @param {Socket} connection
  15. * @api private
  16. */
  17. function Client(server, conn){
  18. this.server = server;
  19. this.conn = conn;
  20. this.encoder = new parser.Encoder();
  21. this.decoder = new parser.Decoder();
  22. this.id = conn.id;
  23. this.request = conn.request;
  24. this.setup();
  25. this.sockets = [];
  26. this.nsps = {};
  27. this.connectBuffer = [];
  28. }
  29. /**
  30. * Sets up event listeners.
  31. *
  32. * @api private
  33. */
  34. Client.prototype.setup = function(){
  35. this.onclose = this.onclose.bind(this);
  36. this.ondata = this.ondata.bind(this);
  37. this.onerror = this.onerror.bind(this);
  38. this.ondecoded = this.ondecoded.bind(this);
  39. this.decoder.on('decoded', this.ondecoded);
  40. this.conn.on('data', this.ondata);
  41. this.conn.on('error', this.onerror);
  42. this.conn.on('close', this.onclose);
  43. };
  44. /**
  45. * Connects a client to a namespace.
  46. *
  47. * @param {String} namespace name
  48. * @api private
  49. */
  50. Client.prototype.connect = function(name){
  51. debug('connecting to namespace %s', name);
  52. if (!this.server.nsps[name]) {
  53. this.packet({ type: parser.ERROR, nsp: name, data : 'Invalid namespace'});
  54. return;
  55. }
  56. var nsp = this.server.of(name);
  57. if ('/' != name && !this.nsps['/']) {
  58. this.connectBuffer.push(name);
  59. return;
  60. }
  61. var self = this;
  62. var socket = nsp.add(this, function(){
  63. self.sockets.push(socket);
  64. self.nsps[nsp.name] = socket;
  65. if ('/' == nsp.name && self.connectBuffer.length > 0) {
  66. self.connectBuffer.forEach(self.connect, self);
  67. self.connectBuffer = [];
  68. }
  69. });
  70. };
  71. /**
  72. * Disconnects from all namespaces and closes transport.
  73. *
  74. * @api private
  75. */
  76. Client.prototype.disconnect = function(){
  77. var socket;
  78. // we don't use a for loop because the length of
  79. // `sockets` changes upon each iteration
  80. while (socket = this.sockets.shift()) {
  81. socket.disconnect();
  82. }
  83. this.close();
  84. };
  85. /**
  86. * Removes a socket. Called by each `Socket`.
  87. *
  88. * @api private
  89. */
  90. Client.prototype.remove = function(socket){
  91. var i = this.sockets.indexOf(socket);
  92. if (~i) {
  93. var nsp = this.sockets[i].nsp.name;
  94. this.sockets.splice(i, 1);
  95. delete this.nsps[nsp];
  96. } else {
  97. debug('ignoring remove for %s', socket.id);
  98. }
  99. };
  100. /**
  101. * Closes the underlying connection.
  102. *
  103. * @api private
  104. */
  105. Client.prototype.close = function(){
  106. if ('open' == this.conn.readyState) {
  107. debug('forcing transport close');
  108. this.conn.close();
  109. this.onclose('forced server close');
  110. }
  111. };
  112. /**
  113. * Writes a packet to the transport.
  114. *
  115. * @param {Object} packet object
  116. * @param {Boolean} whether packet is already encoded
  117. * @param {Boolean} whether packet is volatile
  118. * @api private
  119. */
  120. Client.prototype.packet = function(packet, preEncoded, volatile){
  121. var self = this;
  122. // this writes to the actual connection
  123. function writeToEngine(encodedPackets) {
  124. if (volatile && !self.conn.transport.writable) return;
  125. for (var i = 0; i < encodedPackets.length; i++) {
  126. self.conn.write(encodedPackets[i]);
  127. }
  128. }
  129. if ('open' == this.conn.readyState) {
  130. debug('writing packet %j', packet);
  131. if(!preEncoded) { // not broadcasting, need to encode
  132. this.encoder.encode(packet, function (encodedPackets) { // encode, then write results to engine
  133. writeToEngine(encodedPackets);
  134. });
  135. } else { // a broadcast pre-encodes a packet
  136. writeToEngine(packet);
  137. }
  138. } else {
  139. debug('ignoring packet write %j', packet);
  140. }
  141. };
  142. /**
  143. * Called with incoming transport data.
  144. *
  145. * @api private
  146. */
  147. Client.prototype.ondata = function(data){
  148. // try/catch is needed for protocol violations (GH-1880)
  149. try {
  150. this.decoder.add(data);
  151. } catch(e) {
  152. this.onerror(e);
  153. }
  154. };
  155. /**
  156. * Called when parser fully decodes a packet.
  157. *
  158. * @api private
  159. */
  160. Client.prototype.ondecoded = function(packet) {
  161. if (parser.CONNECT == packet.type) {
  162. this.connect(packet.nsp);
  163. } else {
  164. var socket = this.nsps[packet.nsp];
  165. if (socket) {
  166. socket.onpacket(packet);
  167. } else {
  168. debug('no socket for namespace %s', packet.nsp);
  169. }
  170. }
  171. };
  172. /**
  173. * Handles an error.
  174. *
  175. * @param {Objcet} error object
  176. * @api private
  177. */
  178. Client.prototype.onerror = function(err){
  179. this.sockets.forEach(function(socket){
  180. socket.onerror(err);
  181. });
  182. this.onclose('client error');
  183. };
  184. /**
  185. * Called upon transport close.
  186. *
  187. * @param {String} reason
  188. * @api private
  189. */
  190. Client.prototype.onclose = function(reason){
  191. debug('client close with reason %s', reason);
  192. // ignore a potential subsequent `close` event
  193. this.destroy();
  194. // `nsps` and `sockets` are cleaned up seamlessly
  195. var socket;
  196. while (socket = this.sockets.shift()) {
  197. socket.onclose(reason);
  198. }
  199. this.decoder.destroy(); // clean up decoder
  200. };
  201. /**
  202. * Cleans up event listeners.
  203. *
  204. * @api private
  205. */
  206. Client.prototype.destroy = function(){
  207. this.conn.removeListener('data', this.ondata);
  208. this.conn.removeListener('error', this.onerror);
  209. this.conn.removeListener('close', this.onclose);
  210. this.decoder.removeListener('decoded', this.ondecoded);
  211. };