2019-07-18 10:25:10 +00:00
|
|
|
logger = require 'logger-sharelatex'
|
2019-07-24 15:25:45 +00:00
|
|
|
metrics = require "metrics-sharelatex"
|
2019-07-18 10:25:10 +00:00
|
|
|
{EventEmitter} = require 'events'
|
|
|
|
|
|
|
|
IdMap = new Map() # keep track of whether ids are from projects or docs
|
2019-07-24 14:41:25 +00:00
|
|
|
RoomEvents = new EventEmitter() # emits {project,doc}-active and {project,doc}-empty events
|
2019-07-18 10:25:10 +00:00
|
|
|
|
|
|
|
# Manage socket.io rooms for individual projects and docs
|
|
|
|
#
|
|
|
|
# The first time someone joins a project or doc we emit a 'project-active' or
|
|
|
|
# 'doc-active' event.
|
|
|
|
#
|
|
|
|
# When the last person leaves a project or doc, we emit 'project-empty' or
|
|
|
|
# 'doc-empty' event.
|
|
|
|
#
|
|
|
|
# The pubsub side is handled by ChannelManager
|
|
|
|
|
|
|
|
module.exports = RoomManager =
|
|
|
|
|
2019-07-23 16:02:09 +00:00
|
|
|
joinProject: (client, project_id, callback = () ->) ->
|
|
|
|
@joinEntity client, "project", project_id, callback
|
2019-07-18 10:25:10 +00:00
|
|
|
|
2019-07-23 16:02:09 +00:00
|
|
|
joinDoc: (client, doc_id, callback = () ->) ->
|
|
|
|
@joinEntity client, "doc", doc_id, callback
|
2019-07-18 10:25:10 +00:00
|
|
|
|
|
|
|
leaveDoc: (client, doc_id) ->
|
2019-07-22 10:23:33 +00:00
|
|
|
@leaveEntity client, "doc", doc_id
|
2019-07-18 10:25:10 +00:00
|
|
|
|
|
|
|
leaveProjectAndDocs: (client) ->
|
2019-07-19 07:56:38 +00:00
|
|
|
# what rooms is this client in? we need to leave them all. socket.io
|
|
|
|
# will cause us to leave the rooms, so we only need to manage our
|
|
|
|
# channel subscriptions... but it will be safer if we leave them
|
|
|
|
# explicitly, and then socket.io will just regard this as a client that
|
|
|
|
# has not joined any rooms and do a final disconnection.
|
2019-07-29 14:19:08 +00:00
|
|
|
roomsToLeave = @_roomsClientIsIn(client)
|
|
|
|
logger.log {client: client.id, roomsToLeave: roomsToLeave}, "client leaving project"
|
|
|
|
for id in roomsToLeave
|
2019-07-18 10:25:10 +00:00
|
|
|
entity = IdMap.get(id)
|
2019-07-22 10:23:33 +00:00
|
|
|
@leaveEntity client, entity, id
|
2019-07-18 10:25:10 +00:00
|
|
|
|
2019-07-24 13:30:48 +00:00
|
|
|
emitOnCompletion: (promiseList, eventName) ->
|
2020-06-17 08:29:12 +00:00
|
|
|
Promise.all(promiseList)
|
|
|
|
.then(() -> RoomEvents.emit(eventName))
|
|
|
|
.catch((err) -> RoomEvents.emit(eventName, err))
|
2019-07-24 13:30:48 +00:00
|
|
|
|
2019-07-18 10:25:10 +00:00
|
|
|
eventSource: () ->
|
|
|
|
return RoomEvents
|
|
|
|
|
2019-07-23 16:02:09 +00:00
|
|
|
joinEntity: (client, entity, id, callback) ->
|
2019-07-18 10:25:10 +00:00
|
|
|
beforeCount = @_clientsInRoom(client, id)
|
2019-07-26 07:07:49 +00:00
|
|
|
# client joins room immediately but joinDoc request does not complete
|
|
|
|
# until room is subscribed
|
|
|
|
client.join id
|
2019-07-18 10:25:10 +00:00
|
|
|
# is this a new room? if so, subscribe
|
2019-07-23 16:02:09 +00:00
|
|
|
if beforeCount == 0
|
2019-07-18 10:25:10 +00:00
|
|
|
logger.log {entity, id}, "room is now active"
|
2019-07-23 16:02:09 +00:00
|
|
|
RoomEvents.once "#{entity}-subscribed-#{id}", (err) ->
|
2019-07-24 14:41:25 +00:00
|
|
|
# only allow the client to join when all the relevant channels have subscribed
|
2019-07-26 07:07:49 +00:00
|
|
|
logger.log {client: client.id, entity, id, beforeCount}, "client joined new room and subscribed to channel"
|
2019-07-23 16:02:09 +00:00
|
|
|
callback(err)
|
2019-07-18 10:25:10 +00:00
|
|
|
RoomEvents.emit "#{entity}-active", id
|
|
|
|
IdMap.set(id, entity)
|
2019-07-24 15:25:45 +00:00
|
|
|
# keep track of the number of listeners
|
|
|
|
metrics.gauge "room-listeners", RoomEvents.eventNames().length
|
2019-07-23 16:02:09 +00:00
|
|
|
else
|
|
|
|
logger.log {client: client.id, entity, id, beforeCount}, "client joined existing room"
|
|
|
|
client.join id
|
|
|
|
callback()
|
2019-07-18 10:25:10 +00:00
|
|
|
|
2019-07-22 10:23:33 +00:00
|
|
|
leaveEntity: (client, entity, id) ->
|
2019-07-29 14:19:08 +00:00
|
|
|
# Ignore any requests to leave when the client is not actually in the
|
|
|
|
# room. This can happen if the client sends spurious leaveDoc requests
|
|
|
|
# for old docs after a reconnection.
|
2020-05-19 16:41:20 +00:00
|
|
|
# This can now happen all the time, as we skip the join for clients that
|
|
|
|
# disconnect before joinProject/joinDoc completed.
|
2019-07-29 14:19:08 +00:00
|
|
|
if !@_clientAlreadyInRoom(client, id)
|
2020-05-19 16:41:20 +00:00
|
|
|
logger.log {client: client.id, entity, id}, "ignoring request from client to leave room it is not in"
|
2019-07-29 14:19:08 +00:00
|
|
|
return
|
2019-07-18 10:25:10 +00:00
|
|
|
client.leave id
|
|
|
|
afterCount = @_clientsInRoom(client, id)
|
2019-07-23 16:02:09 +00:00
|
|
|
logger.log {client: client.id, entity, id, afterCount}, "client left room"
|
2019-07-18 10:25:10 +00:00
|
|
|
# is the room now empty? if so, unsubscribe
|
2019-07-22 10:23:33 +00:00
|
|
|
if !entity?
|
|
|
|
logger.error {entity: id}, "unknown entity when leaving with id"
|
|
|
|
return
|
2019-07-23 16:02:09 +00:00
|
|
|
if afterCount == 0
|
2019-07-18 10:25:10 +00:00
|
|
|
logger.log {entity, id}, "room is now empty"
|
|
|
|
RoomEvents.emit "#{entity}-empty", id
|
2019-07-22 10:23:33 +00:00
|
|
|
IdMap.delete(id)
|
2019-07-24 15:25:45 +00:00
|
|
|
metrics.gauge "room-listeners", RoomEvents.eventNames().length
|
2019-07-22 10:23:33 +00:00
|
|
|
|
|
|
|
# internal functions below, these access socket.io rooms data directly and
|
|
|
|
# will need updating for socket.io v2
|
|
|
|
|
|
|
|
_clientsInRoom: (client, room) ->
|
|
|
|
nsp = client.namespace.name
|
|
|
|
name = (nsp + '/') + room;
|
|
|
|
return (client.manager?.rooms?[name] || []).length
|
|
|
|
|
|
|
|
_roomsClientIsIn: (client) ->
|
|
|
|
roomList = for fullRoomPath of client.manager.roomClients?[client.id] when fullRoomPath isnt ''
|
|
|
|
# strip socket.io prefix from room to get original id
|
|
|
|
[prefix, room] = fullRoomPath.split('/', 2)
|
|
|
|
room
|
|
|
|
return roomList
|
2019-07-29 14:19:08 +00:00
|
|
|
|
|
|
|
_clientAlreadyInRoom: (client, room) ->
|
|
|
|
nsp = client.namespace.name
|
|
|
|
name = (nsp + '/') + room;
|
|
|
|
return client.manager.roomClients?[client.id]?[name]
|