mirror of
https://github.com/yu-i-i/overleaf-cep.git
synced 2025-07-27 17:00:06 +02:00

* [history-v1-ot] initial implementation of using doc-level history-v1-ot * [web] fix advancing of the otMigrationStage Use 'nextStage' for the user provided, desired stage when advancing. Co-authored-by: Brian Gough <brian.gough@overleaf.com> * [document-updater] document size check in editor-core * [history-ot] rename history-v1-ot to history-ot and add types * [history-ot] apply review feedback - remove extra !! - merge variable assignment when processing diff-match-match output - add helper function for getting docstore lines view of StringFileData Co-authored-by: Alf Eaton <alf.eaton@overleaf.com> * Revert "[document-updater] add safe rollback point for history-ot (#25283)" This reverts commit d7230dd14a379a27d2c6ab03a006463a18979d06 Signed-off-by: Jakob Ackermann <jakob.ackermann@overleaf.com> --------- Signed-off-by: Jakob Ackermann <jakob.ackermann@overleaf.com> Co-authored-by: Brian Gough <brian.gough@overleaf.com> Co-authored-by: Alf Eaton <alf.eaton@overleaf.com> GitOrigin-RevId: 89c497782adb0427635d50d02263d6f535b12481
175 lines
4.6 KiB
JavaScript
175 lines
4.6 KiB
JavaScript
/* eslint-disable
|
|
no-return-assign,
|
|
*/
|
|
// TODO: This file was created by bulk-decaffeinate.
|
|
// Fix any style issues and re-enable lint.
|
|
/*
|
|
* decaffeinate suggestions:
|
|
* 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
|
|
*/
|
|
let Client
|
|
const { XMLHttpRequest } = require('../../libs/XMLHttpRequest')
|
|
const io = require('socket.io-client')
|
|
const async = require('async')
|
|
|
|
const request = require('request')
|
|
const Settings = require('@overleaf/settings')
|
|
const redis = require('@overleaf/redis-wrapper')
|
|
const rclient = redis.createClient(Settings.redis.websessions)
|
|
|
|
const uid = require('uid-safe').sync
|
|
const signature = require('cookie-signature')
|
|
|
|
io.util.request = function () {
|
|
const xhr = new XMLHttpRequest()
|
|
const _open = xhr.open
|
|
xhr.open = function () {
|
|
_open.apply(xhr, arguments)
|
|
if (Client.cookie != null) {
|
|
return xhr.setRequestHeader('Cookie', Client.cookie)
|
|
}
|
|
}
|
|
return xhr
|
|
}
|
|
|
|
module.exports = Client = {
|
|
cookie: null,
|
|
|
|
setSession(session, callback) {
|
|
if (callback == null) {
|
|
callback = function () {}
|
|
}
|
|
const sessionId = uid(24)
|
|
session.cookie = {}
|
|
return rclient.set('sess:' + sessionId, JSON.stringify(session), error => {
|
|
if (error != null) {
|
|
return callback(error)
|
|
}
|
|
Client.cookieSignedWith = {}
|
|
// prepare cookie strings for all supported session secrets
|
|
for (const secretName of [
|
|
'sessionSecret',
|
|
'sessionSecretFallback',
|
|
'sessionSecretUpcoming',
|
|
]) {
|
|
const secret = Settings.security[secretName]
|
|
const cookieKey = 's:' + signature.sign(sessionId, secret)
|
|
Client.cookieSignedWith[secretName] =
|
|
`${Settings.cookieName}=${cookieKey}`
|
|
}
|
|
// default to the current session secret
|
|
Client.cookie = Client.cookieSignedWith.sessionSecret
|
|
return callback()
|
|
})
|
|
},
|
|
|
|
setAnonSession(projectId, anonymousAccessToken, callback) {
|
|
Client.setSession(
|
|
{
|
|
anonTokenAccess: {
|
|
[projectId]: anonymousAccessToken,
|
|
},
|
|
},
|
|
callback
|
|
)
|
|
},
|
|
|
|
unsetSession(callback) {
|
|
if (callback == null) {
|
|
callback = function () {}
|
|
}
|
|
Client.cookie = null
|
|
return callback()
|
|
},
|
|
|
|
connect(projectId, callback) {
|
|
const client = io.connect('http://127.0.0.1:3026', {
|
|
'force new connection': true,
|
|
query: new URLSearchParams({ projectId }).toString(),
|
|
})
|
|
let disconnected = false
|
|
client.on('disconnect', () => {
|
|
disconnected = true
|
|
})
|
|
client.on('connectionRejected', err => {
|
|
// Wait for disconnect ahead of continuing with the test sequence.
|
|
setTimeout(() => {
|
|
if (!disconnected) {
|
|
throw new Error('should disconnect after connectionRejected')
|
|
}
|
|
callback(err)
|
|
}, 10)
|
|
})
|
|
client.on('joinProjectResponse', resp => {
|
|
const { publicId, project, permissionsLevel, protocolVersion } = resp
|
|
client.publicId = publicId
|
|
callback(null, project, permissionsLevel, protocolVersion)
|
|
})
|
|
return client
|
|
},
|
|
|
|
getConnectedClients(callback) {
|
|
if (callback == null) {
|
|
callback = function () {}
|
|
}
|
|
return request.get(
|
|
{
|
|
url: 'http://127.0.0.1:3026/clients',
|
|
json: true,
|
|
},
|
|
(error, response, data) => callback(error, data)
|
|
)
|
|
},
|
|
|
|
countConnectedClients(projectId, callback) {
|
|
request.get(
|
|
{
|
|
url: `http://127.0.0.1:3026/project/${projectId}/count-connected-clients`,
|
|
json: true,
|
|
},
|
|
(error, response, data) => callback(error, data)
|
|
)
|
|
},
|
|
|
|
getConnectedClient(clientId, callback) {
|
|
if (callback == null) {
|
|
callback = function () {}
|
|
}
|
|
return request.get(
|
|
{
|
|
url: `http://127.0.0.1:3026/clients/${clientId}`,
|
|
json: true,
|
|
},
|
|
(error, response, data) => {
|
|
if (response?.statusCode === 404) {
|
|
callback(new Error('not found'))
|
|
} else {
|
|
callback(error, data)
|
|
}
|
|
}
|
|
)
|
|
},
|
|
|
|
disconnectClient(clientId, callback) {
|
|
request.post(
|
|
{
|
|
url: `http://127.0.0.1:3026/client/${clientId}/disconnect`,
|
|
},
|
|
(error, response, data) => callback(error, data)
|
|
)
|
|
return null
|
|
},
|
|
|
|
disconnectAllClients(callback) {
|
|
return Client.getConnectedClients((error, clients) => {
|
|
if (error) return callback(error)
|
|
async.each(
|
|
clients,
|
|
(clientView, cb) => Client.disconnectClient(clientView.client_id, cb),
|
|
callback
|
|
)
|
|
})
|
|
},
|
|
}
|