Vous ne pouvez pas sélectionner plus de 25 sujets Les noms de sujets doivent commencer par une lettre ou un nombre, peuvent contenir des tirets ('-') et peuvent comporter jusqu'à 35 caractères.
 
 
 
 
 
 

358 lignes
8.6 KiB

  1. var app = require('express')();
  2. var server = require('http').Server(app);
  3. var io = require('socket.io')(server);
  4. var cookie = require('cookie')
  5. var fs = require('fs');
  6. var path = require('path');
  7. var request = require('superagent');
  8. var { get_conf, get_redis_subscriber } = require('./node_utils');
  9. var conf = get_conf();
  10. var flags = {};
  11. var files_struct = {
  12. name: null,
  13. type: null,
  14. size: 0,
  15. data: [],
  16. slice: 0,
  17. site_name: null,
  18. is_private: 0
  19. };
  20. var subscriber = get_redis_subscriber();
  21. // serve socketio
  22. server.listen(conf.socketio_port, function () {
  23. console.log('listening on *:', conf.socketio_port); //eslint-disable-line
  24. });
  25. // on socket connection
  26. io.on('connection', function (socket) {
  27. if (get_hostname(socket.request.headers.host) != get_hostname(socket.request.headers.origin)) {
  28. return;
  29. }
  30. if (!socket.request.headers.cookie) {
  31. return;
  32. }
  33. var sid = cookie.parse(socket.request.headers.cookie).sid
  34. if (!sid) {
  35. return;
  36. }
  37. socket.user = cookie.parse(socket.request.headers.cookie).user_id;
  38. socket.files = {};
  39. // frappe.chat
  40. socket.on("frappe.chat.room:subscribe", function (rooms) {
  41. if (!Array.isArray(rooms)) {
  42. rooms = [rooms];
  43. }
  44. for (var room of rooms) {
  45. console.log('frappe.chat: Subscribing ' + socket.user + ' to room ' + room);
  46. room = get_chat_room(socket, room);
  47. console.log('frappe.chat: Subscribing ' + socket.user + ' to event ' + room);
  48. socket.join(room);
  49. }
  50. });
  51. socket.on("frappe.chat.message:typing", function (data) {
  52. const user = data.user;
  53. const room = get_chat_room(socket, data.room);
  54. console.log('frappe.chat: Dispatching ' + user + ' typing to room ' + room);
  55. io.to(room).emit('frappe.chat.room:typing', {
  56. room: data.room,
  57. user: user
  58. });
  59. });
  60. // end frappe.chat
  61. let join_chat_room = () => {
  62. request.get(get_url(socket, '/api/method/frappe.realtime.get_user_info'))
  63. .type('form')
  64. .query({
  65. sid: sid
  66. })
  67. .then(res => {
  68. const room = get_user_room(socket, res.body.message.user);
  69. socket.join(room);
  70. socket.join(get_site_room(socket));
  71. })
  72. .catch(e => {
  73. if (e.code === 'ECONNREFUSED') {
  74. // retry after 1s
  75. return setTimeout(join_chat_room, 1000);
  76. }
  77. log(e.code);
  78. });
  79. };
  80. join_chat_room();
  81. socket.on('disconnect', function () {
  82. delete socket.files;
  83. })
  84. socket.on('task_subscribe', function (task_id) {
  85. var room = get_task_room(socket, task_id);
  86. socket.join(room);
  87. });
  88. socket.on('task_unsubscribe', function (task_id) {
  89. var room = get_task_room(socket, task_id);
  90. socket.leave(room);
  91. });
  92. socket.on('progress_subscribe', function (task_id) {
  93. var room = get_task_room(socket, task_id);
  94. socket.join(room);
  95. send_existing_lines(task_id, socket);
  96. });
  97. socket.on('doc_subscribe', function (doctype, docname) {
  98. can_subscribe_doc({
  99. socket: socket,
  100. sid: sid,
  101. doctype: doctype,
  102. docname: docname,
  103. callback: function (err, res) {
  104. var room = get_doc_room(socket, doctype, docname);
  105. socket.join(room);
  106. }
  107. });
  108. });
  109. socket.on('doc_unsubscribe', function (doctype, docname) {
  110. var room = get_doc_room(socket, doctype, docname);
  111. socket.leave(room);
  112. });
  113. socket.on('task_unsubscribe', function (task_id) {
  114. var room = 'task:' + task_id;
  115. socket.leave(room);
  116. });
  117. socket.on('doc_open', function (doctype, docname) {
  118. // show who is currently viewing the form
  119. can_subscribe_doc({
  120. socket: socket,
  121. sid: sid,
  122. doctype: doctype,
  123. docname: docname,
  124. callback: function (err, res) {
  125. var room = get_open_doc_room(socket, doctype, docname);
  126. socket.join(room);
  127. send_viewers({
  128. socket: socket,
  129. doctype: doctype,
  130. docname: docname,
  131. });
  132. }
  133. });
  134. });
  135. socket.on('doc_close', function (doctype, docname) {
  136. // remove this user from the list of 'who is currently viewing the form'
  137. var room = get_open_doc_room(socket, doctype, docname);
  138. socket.leave(room);
  139. send_viewers({
  140. socket: socket,
  141. doctype: doctype,
  142. docname: docname,
  143. });
  144. });
  145. socket.on('upload-accept-slice', (data) => {
  146. try {
  147. if (!socket.files[data.name]) {
  148. socket.files[data.name] = Object.assign({}, files_struct, data);
  149. socket.files[data.name].data = [];
  150. }
  151. //convert the ArrayBuffer to Buffer
  152. data.data = new Buffer(new Uint8Array(data.data));
  153. //save the data
  154. socket.files[data.name].data.push(data.data);
  155. socket.files[data.name].slice++;
  156. if (socket.files[data.name].slice * 24576 >= socket.files[data.name].size) {
  157. // do something with the data
  158. var fileBuffer = Buffer.concat(socket.files[data.name].data);
  159. const file_url = path.join((socket.files[data.name].is_private ? 'private' : 'public'),
  160. 'files', data.name);
  161. const file_path = path.join('sites', get_site_name(socket), file_url);
  162. fs.writeFile(file_path, fileBuffer, (err) => {
  163. delete socket.files[data.name];
  164. if (err) return socket.emit('upload error');
  165. socket.emit('upload-end', {
  166. file_url: '/' + file_url
  167. });
  168. });
  169. } else {
  170. socket.emit('upload-request-slice', {
  171. currentSlice: socket.files[data.name].slice
  172. });
  173. }
  174. } catch (e) {
  175. console.log(e);
  176. socket.emit('upload-error', {
  177. error: e.message
  178. });
  179. }
  180. });
  181. });
  182. subscriber.on("message", function (channel, message, room) {
  183. message = JSON.parse(message);
  184. if (message.room) {
  185. io.to(message.room).emit(message.event, message.message);
  186. } else {
  187. io.emit(message.event, message.message);
  188. }
  189. });
  190. subscriber.subscribe("events");
  191. function send_existing_lines(task_id, socket) {
  192. var room = get_task_room(socket, task_id);
  193. subscriber.hgetall('task_log:' + task_id, function (err, lines) {
  194. io.to(room).emit('task_progress', {
  195. "task_id": task_id,
  196. "message": {
  197. "lines": lines
  198. }
  199. });
  200. });
  201. }
  202. function get_doc_room(socket, doctype, docname) {
  203. return get_site_name(socket) + ':doc:' + doctype + '/' + docname;
  204. }
  205. function get_open_doc_room(socket, doctype, docname) {
  206. return get_site_name(socket) + ':open_doc:' + doctype + '/' + docname;
  207. }
  208. function get_user_room(socket, user) {
  209. return get_site_name(socket) + ':user:' + user;
  210. }
  211. function get_site_room(socket) {
  212. return get_site_name(socket) + ':all';
  213. }
  214. function get_task_room(socket, task_id) {
  215. return get_site_name(socket) + ':task_progress:' + task_id;
  216. }
  217. // frappe.chat
  218. // If you're thinking on multi-site or anything, please
  219. // update frappe.async as well.
  220. function get_chat_room(socket, room) {
  221. var room = get_site_name(socket) + ":room:" + room;
  222. return room
  223. }
  224. function get_site_name(socket) {
  225. if (socket.request.headers['x-frappe-site-name']) {
  226. return get_hostname(socket.request.headers['x-frappe-site-name']);
  227. } else if (['localhost', '127.0.0.1'].indexOf(socket.request.headers.host) !== -1 &&
  228. conf.default_site) {
  229. // from currentsite.txt since host is localhost
  230. return conf.default_site;
  231. } else if (socket.request.headers.origin) {
  232. return get_hostname(socket.request.headers.origin);
  233. } else {
  234. return get_hostname(socket.request.headers.host);
  235. }
  236. }
  237. function get_hostname(url) {
  238. if (!url) return undefined;
  239. if (url.indexOf("://") > -1) {
  240. url = url.split('/')[2];
  241. }
  242. return (url.match(/:/g)) ? url.slice(0, url.indexOf(":")) : url
  243. }
  244. function get_url(socket, path) {
  245. if (!path) {
  246. path = '';
  247. }
  248. return socket.request.headers.origin + path;
  249. }
  250. function can_subscribe_doc(args) {
  251. if (!args) return;
  252. if (!args.doctype || !args.docname) return;
  253. request.get(get_url(args.socket, '/api/method/frappe.realtime.can_subscribe_doc'))
  254. .type('form')
  255. .query({
  256. sid: args.sid,
  257. doctype: args.doctype,
  258. docname: args.docname
  259. })
  260. .end(function (err, res) {
  261. if (!res) {
  262. console.log("No response for doc_subscribe");
  263. } else if (res.status == 403) {
  264. return;
  265. } else if (err) {
  266. console.log(err);
  267. } else if (res.status == 200) {
  268. args.callback(err, res);
  269. } else {
  270. console.log("Something went wrong", err, res);
  271. }
  272. });
  273. }
  274. function send_viewers(args) {
  275. // send to doc room, 'users currently viewing this document'
  276. if (!(args && args.doctype && args.docname)) {
  277. return;
  278. }
  279. // open doc room
  280. var room = get_open_doc_room(args.socket, args.doctype, args.docname);
  281. var socketio_room = io.sockets.adapter.rooms[room] || {};
  282. // for compatibility with both v1.3.7 and 1.4.4
  283. var clients_dict = ("sockets" in socketio_room) ? socketio_room.sockets : socketio_room;
  284. // socket ids connected to this room
  285. var clients = Object.keys(clients_dict || {});
  286. var viewers = [];
  287. for (var i in io.sockets.sockets) {
  288. var s = io.sockets.sockets[i];
  289. if (clients.indexOf(s.id) !== -1) {
  290. // this socket is connected to the room
  291. viewers.push(s.user);
  292. }
  293. }
  294. // notify
  295. io.to(room).emit("doc_viewers", {
  296. doctype: args.doctype,
  297. docname: args.docname,
  298. viewers: viewers
  299. });
  300. }