2020-06-23 17:29:44 +00:00
|
|
|
const request = require('request')
|
2024-01-08 13:55:30 +00:00
|
|
|
const _ = require('lodash')
|
2020-08-20 13:05:50 +00:00
|
|
|
const OError = require('@overleaf/o-error')
|
2021-12-14 13:00:35 +00:00
|
|
|
const logger = require('@overleaf/logger')
|
2021-07-12 16:47:18 +00:00
|
|
|
const settings = require('@overleaf/settings')
|
2020-11-25 11:57:22 +00:00
|
|
|
const metrics = require('@overleaf/metrics')
|
2020-08-20 11:52:59 +00:00
|
|
|
const {
|
2020-08-20 11:59:52 +00:00
|
|
|
ClientRequestedMissingOpsError,
|
2020-08-20 11:52:59 +00:00
|
|
|
DocumentUpdaterRequestFailedError,
|
|
|
|
NullBytesInOpError,
|
2021-07-13 11:04:45 +00:00
|
|
|
UpdateTooLargeError,
|
2020-08-20 11:52:59 +00:00
|
|
|
} = require('./Errors')
|
2014-11-12 15:54:55 +00:00
|
|
|
|
2020-11-10 11:32:06 +00:00
|
|
|
const rclient = require('@overleaf/redis-wrapper').createClient(
|
2020-06-23 17:29:44 +00:00
|
|
|
settings.redis.documentupdater
|
|
|
|
)
|
|
|
|
const Keys = settings.redis.documentupdater.key_schema
|
2014-11-13 17:07:05 +00:00
|
|
|
|
2020-07-13 09:42:50 +00:00
|
|
|
const DocumentUpdaterManager = {
|
2023-03-20 14:10:40 +00:00
|
|
|
getDocument(projectId, docId, fromVersion, callback) {
|
2020-06-23 17:29:44 +00:00
|
|
|
const timer = new metrics.Timer('get-document')
|
2023-03-20 14:10:40 +00:00
|
|
|
const url = `${settings.apis.documentupdater.url}/project/${projectId}/doc/${docId}?fromVersion=${fromVersion}`
|
2021-09-14 08:36:24 +00:00
|
|
|
logger.debug(
|
2023-03-20 14:10:40 +00:00
|
|
|
{ projectId, docId, fromVersion },
|
2020-06-23 17:29:44 +00:00
|
|
|
'getting doc from document updater'
|
|
|
|
)
|
2020-07-07 10:06:02 +00:00
|
|
|
request.get(url, function (err, res, body) {
|
2020-06-23 17:29:44 +00:00
|
|
|
timer.done()
|
2020-07-07 10:06:02 +00:00
|
|
|
if (err) {
|
2020-08-20 13:05:50 +00:00
|
|
|
OError.tag(err, 'error getting doc from doc updater')
|
2020-06-23 17:29:44 +00:00
|
|
|
return callback(err)
|
|
|
|
}
|
|
|
|
if (res.statusCode >= 200 && res.statusCode < 300) {
|
2021-09-14 08:36:24 +00:00
|
|
|
logger.debug(
|
2023-03-20 14:10:40 +00:00
|
|
|
{ projectId, docId },
|
2020-06-23 17:29:44 +00:00
|
|
|
'got doc from document document updater'
|
|
|
|
)
|
|
|
|
try {
|
|
|
|
body = JSON.parse(body)
|
|
|
|
} catch (error) {
|
2020-08-20 13:05:50 +00:00
|
|
|
OError.tag(error, 'error parsing doc updater response')
|
2020-06-23 17:29:44 +00:00
|
|
|
return callback(error)
|
|
|
|
}
|
2020-07-07 10:06:02 +00:00
|
|
|
body = body || {}
|
|
|
|
callback(null, body.lines, body.version, body.ranges, body.ops)
|
2020-06-23 17:29:44 +00:00
|
|
|
} else if ([404, 422].includes(res.statusCode)) {
|
2020-08-20 11:59:52 +00:00
|
|
|
callback(new ClientRequestedMissingOpsError(res.statusCode))
|
2020-06-23 17:29:44 +00:00
|
|
|
} else {
|
2020-08-20 11:52:59 +00:00
|
|
|
callback(
|
|
|
|
new DocumentUpdaterRequestFailedError('getDocument', res.statusCode)
|
2020-06-23 17:29:44 +00:00
|
|
|
)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
},
|
2014-11-14 15:53:59 +00:00
|
|
|
|
2023-03-20 14:10:40 +00:00
|
|
|
checkDocument(projectId, docId, callback) {
|
2020-07-13 09:42:50 +00:00
|
|
|
// in this call fromVersion = -1 means get document without docOps
|
2023-03-20 14:10:40 +00:00
|
|
|
DocumentUpdaterManager.getDocument(projectId, docId, -1, callback)
|
2020-07-13 09:42:50 +00:00
|
|
|
},
|
|
|
|
|
2023-03-20 14:10:40 +00:00
|
|
|
flushProjectToMongoAndDelete(projectId, callback) {
|
2020-06-23 17:29:44 +00:00
|
|
|
// this method is called when the last connected user leaves the project
|
2023-03-20 14:10:40 +00:00
|
|
|
logger.debug({ projectId }, 'deleting project from document updater')
|
2020-06-23 17:29:44 +00:00
|
|
|
const timer = new metrics.Timer('delete.mongo.project')
|
|
|
|
// flush the project in the background when all users have left
|
|
|
|
const url =
|
2023-03-20 14:10:40 +00:00
|
|
|
`${settings.apis.documentupdater.url}/project/${projectId}?background=true` +
|
2020-06-23 17:29:44 +00:00
|
|
|
(settings.shutDownInProgress ? '&shutdown=true' : '')
|
2020-07-07 10:06:02 +00:00
|
|
|
request.del(url, function (err, res) {
|
2020-06-23 17:29:44 +00:00
|
|
|
timer.done()
|
2020-07-07 10:06:02 +00:00
|
|
|
if (err) {
|
2020-08-20 13:05:50 +00:00
|
|
|
OError.tag(err, 'error deleting project from document updater')
|
2020-07-07 10:06:02 +00:00
|
|
|
callback(err)
|
2020-06-23 17:29:44 +00:00
|
|
|
} else if (res.statusCode >= 200 && res.statusCode < 300) {
|
2023-03-20 14:10:40 +00:00
|
|
|
logger.debug({ projectId }, 'deleted project from document updater')
|
2020-07-07 10:06:02 +00:00
|
|
|
callback(null)
|
2020-06-23 17:29:44 +00:00
|
|
|
} else {
|
2020-08-20 11:52:59 +00:00
|
|
|
callback(
|
|
|
|
new DocumentUpdaterRequestFailedError(
|
|
|
|
'flushProjectToMongoAndDelete',
|
|
|
|
res.statusCode
|
|
|
|
)
|
2020-06-23 17:29:44 +00:00
|
|
|
)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
},
|
2019-05-31 21:27:05 +00:00
|
|
|
|
2021-02-08 12:19:25 +00:00
|
|
|
_getPendingUpdateListKey() {
|
2021-02-09 10:48:40 +00:00
|
|
|
const shard = _.random(0, settings.pendingUpdateListShardCount - 1)
|
2021-02-08 12:19:25 +00:00
|
|
|
if (shard === 0) {
|
|
|
|
return 'pending-updates-list'
|
|
|
|
} else {
|
|
|
|
return `pending-updates-list-${shard}`
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2023-03-20 14:10:40 +00:00
|
|
|
queueChange(projectId, docId, change, callback) {
|
2020-06-23 17:29:44 +00:00
|
|
|
const allowedKeys = [
|
|
|
|
'doc',
|
|
|
|
'op',
|
|
|
|
'v',
|
|
|
|
'dupIfSource',
|
|
|
|
'meta',
|
|
|
|
'lastV',
|
2021-07-13 11:04:45 +00:00
|
|
|
'hash',
|
2020-06-23 17:29:44 +00:00
|
|
|
]
|
|
|
|
change = _.pick(change, allowedKeys)
|
|
|
|
const jsonChange = JSON.stringify(change)
|
|
|
|
if (jsonChange.indexOf('\u0000') !== -1) {
|
|
|
|
// memory corruption check
|
2020-08-20 10:30:22 +00:00
|
|
|
return callback(new NullBytesInOpError(jsonChange))
|
2020-06-23 17:29:44 +00:00
|
|
|
}
|
2020-03-24 10:22:28 +00:00
|
|
|
|
2020-06-23 17:29:44 +00:00
|
|
|
const updateSize = jsonChange.length
|
|
|
|
if (updateSize > settings.maxUpdateSize) {
|
2020-08-20 10:16:26 +00:00
|
|
|
return callback(new UpdateTooLargeError(updateSize))
|
2020-06-23 17:29:44 +00:00
|
|
|
}
|
2020-03-24 10:22:28 +00:00
|
|
|
|
2020-06-23 17:29:44 +00:00
|
|
|
// record metric for each update added to queue
|
|
|
|
metrics.summary('redis.pendingUpdates', updateSize, { status: 'push' })
|
2020-04-06 15:24:10 +00:00
|
|
|
|
2023-03-20 14:10:40 +00:00
|
|
|
const docKey = `${projectId}:${docId}`
|
2020-06-23 17:29:44 +00:00
|
|
|
// Push onto pendingUpdates for doc_id first, because once the doc updater
|
|
|
|
// gets an entry on pending-updates-list, it starts processing.
|
2021-07-13 11:04:45 +00:00
|
|
|
rclient.rpush(
|
2023-03-20 14:10:40 +00:00
|
|
|
Keys.pendingUpdates({ doc_id: docId }),
|
2021-07-13 11:04:45 +00:00
|
|
|
jsonChange,
|
|
|
|
function (error) {
|
2021-02-09 12:42:58 +00:00
|
|
|
if (error) {
|
2021-07-13 11:04:45 +00:00
|
|
|
error = new OError('error pushing update into redis').withCause(error)
|
|
|
|
return callback(error)
|
2020-08-20 13:05:50 +00:00
|
|
|
}
|
2021-07-13 11:04:45 +00:00
|
|
|
const queueKey = DocumentUpdaterManager._getPendingUpdateListKey()
|
2023-03-20 14:10:40 +00:00
|
|
|
rclient.rpush(queueKey, docKey, function (error) {
|
2021-07-13 11:04:45 +00:00
|
|
|
if (error) {
|
|
|
|
error = new OError('error pushing doc_id into redis')
|
|
|
|
.withInfo({ queueKey })
|
|
|
|
.withCause(error)
|
|
|
|
}
|
|
|
|
callback(error)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
)
|
|
|
|
},
|
2020-06-23 17:29:44 +00:00
|
|
|
}
|
2020-07-13 09:42:50 +00:00
|
|
|
|
|
|
|
module.exports = DocumentUpdaterManager
|