2020-05-06 10:09:15 +00:00
|
|
|
/* eslint-disable
|
|
|
|
no-unused-vars,
|
|
|
|
*/
|
|
|
|
// TODO: This file was created by bulk-decaffeinate.
|
|
|
|
// Fix any style issues and re-enable lint.
|
2020-05-06 10:08:21 +00:00
|
|
|
/*
|
|
|
|
* decaffeinate suggestions:
|
|
|
|
* DS101: Remove unnecessary use of Array.from
|
|
|
|
* DS102: Remove unnecessary code created because of implicit returns
|
|
|
|
* DS207: Consider shorter variations of null checks
|
|
|
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/master/docs/suggestions.md
|
|
|
|
*/
|
2021-07-12 16:47:15 +00:00
|
|
|
const Settings = require('@overleaf/settings')
|
2024-01-30 15:35:54 +00:00
|
|
|
const { promisifyAll } = require('@overleaf/promise-utils')
|
2020-11-10 11:32:04 +00:00
|
|
|
const rclient = require('@overleaf/redis-wrapper').createClient(
|
2020-05-06 10:09:33 +00:00
|
|
|
Settings.redis.documentupdater
|
|
|
|
)
|
2020-11-10 11:32:04 +00:00
|
|
|
const pubsubClient = require('@overleaf/redis-wrapper').createClient(
|
2020-05-06 10:09:33 +00:00
|
|
|
Settings.redis.pubsub
|
|
|
|
)
|
|
|
|
const Keys = Settings.redis.documentupdater.key_schema
|
2021-10-06 09:10:28 +00:00
|
|
|
const logger = require('@overleaf/logger')
|
2024-11-08 10:21:56 +00:00
|
|
|
const os = require('node:os')
|
|
|
|
const crypto = require('node:crypto')
|
2020-05-06 10:09:33 +00:00
|
|
|
const metrics = require('./Metrics')
|
2019-03-21 12:10:15 +00:00
|
|
|
|
2020-05-06 10:09:33 +00:00
|
|
|
const HOST = os.hostname()
|
|
|
|
const RND = crypto.randomBytes(4).toString('hex') // generate a random key for this process
|
|
|
|
let COUNT = 0
|
2016-06-17 11:17:22 +00:00
|
|
|
|
2020-05-06 10:09:33 +00:00
|
|
|
const MAX_OPS_PER_ITERATION = 8 // process a limited number of ops for safety
|
2017-05-12 12:11:04 +00:00
|
|
|
|
2024-01-30 15:35:54 +00:00
|
|
|
const RealTimeRedisManager = {
|
2023-03-21 12:06:13 +00:00
|
|
|
getPendingUpdatesForDoc(docId, callback) {
|
2024-05-29 15:50:34 +00:00
|
|
|
// Make sure that this MULTI operation only operates on doc
|
|
|
|
// specific keys, i.e. keys that have the doc id in curly braces.
|
|
|
|
// The curly braces identify a hash key for Redis and ensures that
|
|
|
|
// the MULTI's operations are all done on the same node in a
|
|
|
|
// cluster environment.
|
2020-05-06 10:09:33 +00:00
|
|
|
const multi = rclient.multi()
|
2024-07-18 13:01:09 +00:00
|
|
|
multi.llen(Keys.pendingUpdates({ doc_id: docId }))
|
2023-03-21 12:06:13 +00:00
|
|
|
multi.lrange(
|
|
|
|
Keys.pendingUpdates({ doc_id: docId }),
|
|
|
|
0,
|
|
|
|
MAX_OPS_PER_ITERATION - 1
|
|
|
|
)
|
|
|
|
multi.ltrim(
|
|
|
|
Keys.pendingUpdates({ doc_id: docId }),
|
|
|
|
MAX_OPS_PER_ITERATION,
|
|
|
|
-1
|
|
|
|
)
|
2020-05-06 10:09:33 +00:00
|
|
|
return multi.exec(function (error, replys) {
|
|
|
|
if (error != null) {
|
|
|
|
return callback(error)
|
|
|
|
}
|
2024-07-18 13:01:09 +00:00
|
|
|
const [llen, jsonUpdates, _trimResult] = replys
|
|
|
|
metrics.histogram(
|
|
|
|
'redis.pendingUpdates.llen',
|
|
|
|
llen,
|
|
|
|
[0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 15, 20, 25, 50, 75, 100]
|
|
|
|
)
|
|
|
|
for (const jsonUpdate of jsonUpdates) {
|
2020-05-06 10:09:33 +00:00
|
|
|
// record metric for each update removed from queue
|
|
|
|
metrics.summary('redis.pendingUpdates', jsonUpdate.length, {
|
2021-07-13 11:04:42 +00:00
|
|
|
status: 'pop',
|
2020-05-06 10:09:33 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
const updates = []
|
2024-07-18 13:01:09 +00:00
|
|
|
for (const jsonUpdate of jsonUpdates) {
|
2021-10-26 08:08:56 +00:00
|
|
|
let update
|
2020-05-06 10:09:33 +00:00
|
|
|
try {
|
|
|
|
update = JSON.parse(jsonUpdate)
|
|
|
|
} catch (e) {
|
|
|
|
return callback(e)
|
|
|
|
}
|
|
|
|
updates.push(update)
|
|
|
|
}
|
|
|
|
return callback(error, updates)
|
|
|
|
})
|
|
|
|
},
|
2016-06-17 11:17:22 +00:00
|
|
|
|
2023-03-21 12:06:13 +00:00
|
|
|
getUpdatesLength(docId, callback) {
|
|
|
|
return rclient.llen(Keys.pendingUpdates({ doc_id: docId }), callback)
|
2020-05-06 10:09:33 +00:00
|
|
|
},
|
2016-06-17 11:17:22 +00:00
|
|
|
|
2024-07-18 13:01:09 +00:00
|
|
|
sendCanaryAppliedOp({ projectId, docId, op }) {
|
|
|
|
const ack = JSON.stringify({ v: op.v, doc: docId }).length
|
|
|
|
// Updates with op.dup===true will not get sent to other clients, they only get acked.
|
|
|
|
const broadcast = op.dup ? 0 : JSON.stringify(op).length
|
|
|
|
|
|
|
|
const payload = JSON.stringify({
|
|
|
|
message: 'canary-applied-op',
|
|
|
|
payload: {
|
|
|
|
ack,
|
|
|
|
broadcast,
|
|
|
|
docId,
|
|
|
|
projectId,
|
|
|
|
source: op.meta.source,
|
|
|
|
},
|
|
|
|
})
|
|
|
|
|
|
|
|
// Publish on the editor-events channel of the project as real-time already listens to that before completing the connection startup.
|
|
|
|
|
|
|
|
// publish on separate channels for individual projects and docs when
|
|
|
|
// configured (needs realtime to be configured for this too).
|
|
|
|
if (Settings.publishOnIndividualChannels) {
|
|
|
|
return pubsubClient.publish(`editor-events:${projectId}`, payload)
|
|
|
|
} else {
|
|
|
|
return pubsubClient.publish('editor-events', payload)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2020-05-06 10:09:33 +00:00
|
|
|
sendData(data) {
|
|
|
|
// create a unique message id using a counter
|
2023-03-21 12:06:13 +00:00
|
|
|
const messageId = `doc:${HOST}:${RND}-${COUNT++}`
|
2020-05-06 10:09:33 +00:00
|
|
|
if (data != null) {
|
2023-03-21 12:06:13 +00:00
|
|
|
data._id = messageId
|
2020-05-06 10:09:33 +00:00
|
|
|
}
|
2020-03-30 09:31:43 +00:00
|
|
|
|
2020-05-06 10:09:33 +00:00
|
|
|
const blob = JSON.stringify(data)
|
|
|
|
metrics.summary('redis.publish.applied-ops', blob.length)
|
2020-03-30 09:31:43 +00:00
|
|
|
|
2020-05-06 10:09:33 +00:00
|
|
|
// publish on separate channels for individual projects and docs when
|
|
|
|
// configured (needs realtime to be configured for this too).
|
|
|
|
if (Settings.publishOnIndividualChannels) {
|
|
|
|
return pubsubClient.publish(`applied-ops:${data.doc_id}`, blob)
|
|
|
|
} else {
|
|
|
|
return pubsubClient.publish('applied-ops', blob)
|
|
|
|
}
|
2021-07-13 11:04:42 +00:00
|
|
|
},
|
2020-05-06 10:09:33 +00:00
|
|
|
}
|
2024-01-30 15:35:54 +00:00
|
|
|
|
|
|
|
module.exports = RealTimeRedisManager
|
|
|
|
module.exports.promises = promisifyAll(RealTimeRedisManager, {
|
|
|
|
without: ['sendData'],
|
|
|
|
})
|