overleaf/services/real-time/app/coffee/WebsocketLoadBalancer.coffee

57 lines
2.2 KiB
CoffeeScript
Raw Normal View History

Settings = require 'settings-sharelatex'
2014-11-24 07:09:12 -05:00
logger = require 'logger-sharelatex'
redis = require("redis-sharelatex")
2015-12-01 06:05:49 -05:00
SafeJsonParse = require "./SafeJsonParse"
rclientPub = redis.createClient(Settings.redis.realtime)
rclientSub = redis.createClient(Settings.redis.realtime)
2019-03-19 06:55:12 -04:00
EventLogger = require "./EventLogger"
module.exports = WebsocketLoadBalancer =
rclientPub: rclientPub
rclientSub: rclientSub
emitToRoom: (room_id, message, payload...) ->
2014-11-24 07:09:12 -05:00
if !room_id?
logger.warn {message, payload}, "no room_id provided, ignoring emitToRoom"
2014-11-24 07:09:12 -05:00
return
data = JSON.stringify
room_id: room_id
message: message
payload: payload
logger.log {room_id, message, payload, length: data.length}, "emitting to room"
@rclientPub.publish "editor-events", data
emitToAll: (message, payload...) ->
@emitToRoom "all", message, payload...
listenForEditorEvents: (io) ->
@rclientSub.subscribe "editor-events"
@rclientSub.on "message", (channel, message) ->
2019-04-11 10:00:25 -04:00
EventLogger.debugEvent(channel, message) if Settings.debugEvents > 0
WebsocketLoadBalancer._processEditorEvent io, channel, message
_processEditorEvent: (io, channel, message) ->
2015-12-01 06:05:49 -05:00
SafeJsonParse.parse message, (error, message) ->
if error?
logger.error {err: error, channel}, "error parsing JSON"
return
if message.room_id == "all"
io.sockets.emit(message.message, message.payload...)
else if message.room_id?
2019-03-19 06:55:12 -04:00
if message._id?
2019-04-11 07:53:43 -04:00
status = EventLogger.checkEventOrder("editor-events", message._id, message)
if status is "duplicate"
return # skip duplicate events
# send messages only to unique clients (due to duplicate entries in io.sockets.clients)
clientList = io.sockets.clients(message.room_id)
2019-04-11 07:53:43 -04:00
# avoid unnecessary work if no clients are connected
return if clientList.length is 0
logger.log {channel:channel, message: message.message, room_id: message.room_id, message_id: message._id, socketIoClients: (client.id for client in clientList)}, "distributing event to clients"
seen = {}
for client in clientList when not seen[client.id]
seen[client.id] = true
client.emit(message.message, message.payload...)
else if message.health_check?
logger.debug {message}, "got health check message in editor events channel"