mirror of
https://github.com/yu-i-i/overleaf-cep.git
synced 2026-05-29 12:01:32 +02:00
Extend file upload to create history blobs GitOrigin-RevId: b3efac7f58d4c027ca9131867e761837ce63facb
298 lines
8.1 KiB
JavaScript
298 lines
8.1 KiB
JavaScript
const { callbackify } = require('util')
|
|
const { fetchJson, fetchNothing } = require('@overleaf/fetch-utils')
|
|
const fs = require('fs')
|
|
const settings = require('@overleaf/settings')
|
|
const OError = require('@overleaf/o-error')
|
|
const UserGetter = require('../User/UserGetter')
|
|
const ProjectGetter = require('../Project/ProjectGetter')
|
|
const HistoryBackupDeletionHandler = require('./HistoryBackupDeletionHandler')
|
|
|
|
async function initializeProject(projectId) {
|
|
const body = await fetchJson(`${settings.apis.project_history.url}/project`, {
|
|
method: 'POST',
|
|
json: { historyId: projectId.toString() },
|
|
})
|
|
const historyId = body && body.project && body.project.id
|
|
if (!historyId) {
|
|
throw new OError('project-history did not provide an id', { body })
|
|
}
|
|
return historyId
|
|
}
|
|
|
|
async function flushProject(projectId) {
|
|
try {
|
|
await fetchNothing(
|
|
`${settings.apis.project_history.url}/project/${projectId}/flush`,
|
|
{ method: 'POST' }
|
|
)
|
|
} catch (err) {
|
|
throw OError.tag(err, 'failed to flush project to project history', {
|
|
projectId,
|
|
})
|
|
}
|
|
}
|
|
|
|
async function deleteProjectHistory(projectId) {
|
|
try {
|
|
await fetchNothing(
|
|
`${settings.apis.project_history.url}/project/${projectId}`,
|
|
{ method: 'DELETE' }
|
|
)
|
|
} catch (err) {
|
|
throw OError.tag(err, 'failed to delete project history', {
|
|
projectId,
|
|
})
|
|
}
|
|
}
|
|
|
|
async function resyncProject(projectId, options = {}) {
|
|
const body = {}
|
|
if (options.force) {
|
|
body.force = options.force
|
|
}
|
|
if (options.origin) {
|
|
body.origin = options.origin
|
|
}
|
|
if (options.historyRangesMigration) {
|
|
body.historyRangesMigration = options.historyRangesMigration
|
|
}
|
|
try {
|
|
await fetchNothing(
|
|
`${settings.apis.project_history.url}/project/${projectId}/resync`,
|
|
{
|
|
method: 'POST',
|
|
json: body,
|
|
signal: AbortSignal.timeout(6 * 60 * 1000),
|
|
}
|
|
)
|
|
} catch (err) {
|
|
throw OError.tag(err, 'failed to resync project history', {
|
|
projectId,
|
|
})
|
|
}
|
|
}
|
|
|
|
async function deleteProject(projectId, historyId) {
|
|
const tasks = []
|
|
tasks.push(_deleteProjectInProjectHistory(projectId))
|
|
if (historyId != null) {
|
|
tasks.push(_deleteProjectInFullProjectHistory(historyId))
|
|
}
|
|
await Promise.all(tasks)
|
|
await HistoryBackupDeletionHandler.deleteProject(projectId)
|
|
}
|
|
|
|
async function _deleteProjectInProjectHistory(projectId) {
|
|
try {
|
|
await fetchNothing(
|
|
`${settings.apis.project_history.url}/project/${projectId}`,
|
|
{ method: 'DELETE' }
|
|
)
|
|
} catch (err) {
|
|
throw OError.tag(
|
|
err,
|
|
'failed to clear project history in project-history',
|
|
{ projectId }
|
|
)
|
|
}
|
|
}
|
|
|
|
async function _deleteProjectInFullProjectHistory(historyId) {
|
|
try {
|
|
await fetchNothing(
|
|
`${settings.apis.v1_history.url}/projects/${historyId}`,
|
|
{
|
|
method: 'DELETE',
|
|
basicAuth: {
|
|
user: settings.apis.v1_history.user,
|
|
password: settings.apis.v1_history.pass,
|
|
},
|
|
}
|
|
)
|
|
} catch (err) {
|
|
throw OError.tag(err, 'failed to clear project history', { historyId })
|
|
}
|
|
}
|
|
|
|
async function uploadBlobFromDisk(historyId, hash, byteLength, fsPath) {
|
|
const outStream = fs.createReadStream(fsPath)
|
|
|
|
const url = `${settings.apis.v1_history.url}/projects/${historyId}/blobs/${hash}`
|
|
await fetchNothing(url, {
|
|
method: 'PUT',
|
|
body: outStream,
|
|
headers: { 'Content-Length': byteLength }, // add the content length to work around problems with chunked encoding in node 18
|
|
signal: AbortSignal.timeout(60 * 1000),
|
|
basicAuth: {
|
|
user: settings.apis.v1_history.user,
|
|
password: settings.apis.v1_history.pass,
|
|
},
|
|
})
|
|
}
|
|
|
|
/**
|
|
* Warning: Don't use this method for large projects. It will eagerly load all
|
|
* the history data and apply all operations.
|
|
* @param {string} projectId
|
|
* @returns Promise<object>
|
|
*/
|
|
async function getCurrentContent(projectId) {
|
|
const project = await ProjectGetter.promises.getProject(projectId, {
|
|
overleaf: true,
|
|
})
|
|
const historyId = project?.overleaf?.history?.id
|
|
if (!historyId) {
|
|
throw new OError('project does not have a history id', { projectId })
|
|
}
|
|
try {
|
|
return await fetchJson(
|
|
`${settings.apis.v1_history.url}/projects/${historyId}/latest/content`,
|
|
{
|
|
method: 'GET',
|
|
basicAuth: {
|
|
user: settings.apis.v1_history.user,
|
|
password: settings.apis.v1_history.pass,
|
|
},
|
|
}
|
|
)
|
|
} catch (err) {
|
|
throw OError.tag(err, 'failed to load project history', { historyId })
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Warning: Don't use this method for large projects. It will eagerly load all
|
|
* the history data and apply all operations.
|
|
* @param {string} projectId
|
|
* @param {number} version
|
|
*
|
|
* @returns Promise<object>
|
|
*/
|
|
async function getContentAtVersion(projectId, version) {
|
|
const project = await ProjectGetter.promises.getProject(projectId, {
|
|
overleaf: true,
|
|
})
|
|
const historyId = project?.overleaf?.history?.id
|
|
if (!historyId) {
|
|
throw new OError('project does not have a history id', { projectId })
|
|
}
|
|
try {
|
|
return await fetchJson(
|
|
`${settings.apis.v1_history.url}/projects/${historyId}/versions/${version}/content`,
|
|
{
|
|
method: 'GET',
|
|
basicAuth: {
|
|
user: settings.apis.v1_history.user,
|
|
password: settings.apis.v1_history.pass,
|
|
},
|
|
}
|
|
)
|
|
} catch (err) {
|
|
throw OError.tag(
|
|
err,
|
|
'failed to load project history snapshot at version',
|
|
{ historyId, version }
|
|
)
|
|
}
|
|
}
|
|
|
|
async function injectUserDetails(data) {
|
|
// data can be either:
|
|
// {
|
|
// diff: [{
|
|
// i: "foo",
|
|
// meta: {
|
|
// users: ["user_id", v1_user_id, ...]
|
|
// ...
|
|
// }
|
|
// }, ...]
|
|
// }
|
|
// or
|
|
// {
|
|
// updates: [{
|
|
// pathnames: ["main.tex"]
|
|
// meta: {
|
|
// users: ["user_id", v1_user_id, ...]
|
|
// ...
|
|
// },
|
|
// ...
|
|
// }, ...]
|
|
// }
|
|
// Either way, the top level key points to an array of objects with a meta.users property
|
|
// that we need to replace user_ids with populated user objects.
|
|
// Note that some entries in the users arrays may be v1 ids returned by the v1 history
|
|
// service. v1 ids will be `numbers`
|
|
let userIds = new Set()
|
|
let v1UserIds = new Set()
|
|
const entries = Array.isArray(data.diff)
|
|
? data.diff
|
|
: Array.isArray(data.updates)
|
|
? data.updates
|
|
: []
|
|
for (const entry of entries) {
|
|
for (const user of (entry.meta && entry.meta.users) || []) {
|
|
if (typeof user === 'string') {
|
|
userIds.add(user)
|
|
} else if (typeof user === 'number') {
|
|
v1UserIds.add(user)
|
|
}
|
|
}
|
|
}
|
|
|
|
userIds = Array.from(userIds)
|
|
v1UserIds = Array.from(v1UserIds)
|
|
const projection = { first_name: 1, last_name: 1, email: 1 }
|
|
const usersArray = await UserGetter.promises.getUsers(userIds, projection)
|
|
const users = {}
|
|
for (const user of usersArray) {
|
|
users[user._id.toString()] = _userView(user)
|
|
}
|
|
projection.overleaf = 1
|
|
const v1IdentifiedUsersArray = await UserGetter.promises.getUsersByV1Ids(
|
|
v1UserIds,
|
|
projection
|
|
)
|
|
for (const user of v1IdentifiedUsersArray) {
|
|
users[user.overleaf.id] = _userView(user)
|
|
}
|
|
for (const entry of entries) {
|
|
if (entry.meta != null) {
|
|
entry.meta.users = ((entry.meta && entry.meta.users) || []).map(user => {
|
|
if (typeof user === 'string' || typeof user === 'number') {
|
|
return users[user]
|
|
} else {
|
|
return user
|
|
}
|
|
})
|
|
}
|
|
}
|
|
return data
|
|
}
|
|
|
|
function _userView(user) {
|
|
const { _id, first_name: firstName, last_name: lastName, email } = user
|
|
return { first_name: firstName, last_name: lastName, email, id: _id }
|
|
}
|
|
|
|
module.exports = {
|
|
initializeProject: callbackify(initializeProject),
|
|
flushProject: callbackify(flushProject),
|
|
resyncProject: callbackify(resyncProject),
|
|
deleteProject: callbackify(deleteProject),
|
|
deleteProjectHistory: callbackify(deleteProjectHistory),
|
|
injectUserDetails: callbackify(injectUserDetails),
|
|
getCurrentContent: callbackify(getCurrentContent),
|
|
uploadBlobFromDisk: callbackify(uploadBlobFromDisk),
|
|
promises: {
|
|
initializeProject,
|
|
flushProject,
|
|
resyncProject,
|
|
deleteProject,
|
|
injectUserDetails,
|
|
deleteProjectHistory,
|
|
getCurrentContent,
|
|
getContentAtVersion,
|
|
uploadBlobFromDisk,
|
|
},
|
|
}
|