You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

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