mirror of
https://github.com/overleaf/overleaf.git
synced 2024-11-21 20:47:08 -05:00
91 lines
3.8 KiB
CoffeeScript
91 lines
3.8 KiB
CoffeeScript
async = require("async")
|
|
Settings = require('settings-sharelatex')
|
|
logger = require("logger-sharelatex")
|
|
redis = require("redis-sharelatex")
|
|
rclient = redis.createClient(Settings.redis.realtime)
|
|
Keys = Settings.redis.realtime.key_schema
|
|
|
|
ONE_HOUR_IN_S = 60 * 60
|
|
ONE_DAY_IN_S = ONE_HOUR_IN_S * 24
|
|
FOUR_DAYS_IN_S = ONE_DAY_IN_S * 4
|
|
|
|
USER_TIMEOUT_IN_S = ONE_HOUR_IN_S / 4
|
|
REFRESH_TIMEOUT_IN_S = 10 # only show clients which have responded to a refresh request in the last 10 seconds
|
|
|
|
module.exports =
|
|
# Use the same method for when a user connects, and when a user sends a cursor
|
|
# update. This way we don't care if the connected_user key has expired when
|
|
# we receive a cursor update.
|
|
updateUserPosition: (project_id, client_id, user, cursorData, callback = (err)->)->
|
|
logger.log project_id:project_id, client_id:client_id, "marking user as joined or connected"
|
|
|
|
multi = rclient.multi()
|
|
|
|
multi.sadd Keys.clientsInProject({project_id}), client_id
|
|
multi.expire Keys.clientsInProject({project_id}), FOUR_DAYS_IN_S
|
|
|
|
multi.hset Keys.connectedUser({project_id, client_id}), "last_updated_at", Date.now()
|
|
multi.hset Keys.connectedUser({project_id, client_id}), "user_id", user._id
|
|
multi.hset Keys.connectedUser({project_id, client_id}), "first_name", user.first_name or ""
|
|
multi.hset Keys.connectedUser({project_id, client_id}), "last_name", user.last_name or ""
|
|
multi.hset Keys.connectedUser({project_id, client_id}), "email", user.email or ""
|
|
|
|
if cursorData?
|
|
multi.hset Keys.connectedUser({project_id, client_id}), "cursorData", JSON.stringify(cursorData)
|
|
multi.expire Keys.connectedUser({project_id, client_id}), USER_TIMEOUT_IN_S
|
|
|
|
multi.exec (err)->
|
|
if err?
|
|
logger.err err:err, project_id:project_id, client_id:client_id, "problem marking user as connected"
|
|
callback(err)
|
|
|
|
refreshClient: (project_id, client_id, callback = (err) ->) ->
|
|
logger.log project_id:project_id, client_id:client_id, "refreshing connected client"
|
|
multi = rclient.multi()
|
|
multi.hset Keys.connectedUser({project_id, client_id}), "last_updated_at", Date.now()
|
|
multi.expire Keys.connectedUser({project_id, client_id}), USER_TIMEOUT_IN_S
|
|
multi.exec (err)->
|
|
if err?
|
|
logger.err err:err, project_id:project_id, client_id:client_id, "problem refreshing connected client"
|
|
callback(err)
|
|
|
|
markUserAsDisconnected: (project_id, client_id, callback)->
|
|
logger.log project_id:project_id, client_id:client_id, "marking user as disconnected"
|
|
multi = rclient.multi()
|
|
multi.srem Keys.clientsInProject({project_id}), client_id
|
|
multi.expire Keys.clientsInProject({project_id}), FOUR_DAYS_IN_S
|
|
multi.del Keys.connectedUser({project_id, client_id})
|
|
multi.exec callback
|
|
|
|
|
|
_getConnectedUser: (project_id, client_id, callback)->
|
|
rclient.hgetall Keys.connectedUser({project_id, client_id}), (err, result)->
|
|
if !result? or Object.keys(result).length == 0 or !result.user_id
|
|
result =
|
|
connected : false
|
|
client_id:client_id
|
|
else
|
|
result.connected = true
|
|
result.client_id = client_id
|
|
result.client_age = (Date.now() - parseInt(result.last_updated_at,10)) / 1000
|
|
if result.cursorData?
|
|
try
|
|
result.cursorData = JSON.parse(result.cursorData)
|
|
catch e
|
|
logger.error {err: e, project_id, client_id, cursorData: result.cursorData}, "error parsing cursorData JSON"
|
|
return callback e
|
|
callback err, result
|
|
|
|
getConnectedUsers: (project_id, callback)->
|
|
self = @
|
|
rclient.smembers Keys.clientsInProject({project_id}), (err, results)->
|
|
return callback(err) if err?
|
|
jobs = results.map (client_id)->
|
|
(cb)->
|
|
self._getConnectedUser(project_id, client_id, cb)
|
|
async.series jobs, (err, users = [])->
|
|
return callback(err) if err?
|
|
users = users.filter (user) ->
|
|
user?.connected && user?.client_age < REFRESH_TIMEOUT_IN_S
|
|
callback null, users
|
|
|