mirror of
https://github.com/yu-i-i/overleaf-cep.git
synced 2026-05-23 17:19:37 +02:00
[web] migrate file-view to download from history-v1 (via web) 1/2 GitOrigin-RevId: b787e90c57af5e2704b06ba63502aa6fc09ea1df
175 lines
4.9 KiB
JavaScript
175 lines
4.9 KiB
JavaScript
import logger from '@overleaf/logger'
|
|
import metrics from '@overleaf/metrics'
|
|
import fs from 'node:fs'
|
|
import Path from 'node:path'
|
|
import FileSystemImportManager from './FileSystemImportManager.js'
|
|
import ProjectUploadManager from './ProjectUploadManager.js'
|
|
import SessionManager from '../Authentication/SessionManager.js'
|
|
import EditorController from '../Editor/EditorController.js'
|
|
import ProjectLocator from '../Project/ProjectLocator.js'
|
|
import Settings from '@overleaf/settings'
|
|
import { InvalidZipFileError } from './ArchiveErrors.js'
|
|
import multer from 'multer'
|
|
import lodash from 'lodash'
|
|
import { expressify } from '@overleaf/promise-utils'
|
|
import { DuplicateNameError } from '../Errors/Errors.js'
|
|
|
|
const defaultsDeep = lodash.defaultsDeep
|
|
|
|
const upload = multer(
|
|
defaultsDeep(
|
|
{
|
|
dest: Settings.path.uploadFolder,
|
|
limits: {
|
|
fileSize: Settings.maxUploadSize,
|
|
},
|
|
},
|
|
Settings.multerOptions
|
|
)
|
|
)
|
|
|
|
function uploadProject(req, res, next) {
|
|
const timer = new metrics.Timer('project-upload')
|
|
const userId = SessionManager.getLoggedInUserId(req.session)
|
|
const { path } = req.file
|
|
const name = Path.basename(req.body.name, '.zip')
|
|
return ProjectUploadManager.createProjectFromZipArchive(
|
|
userId,
|
|
name,
|
|
path,
|
|
function (error, project) {
|
|
fs.unlink(path, function () {})
|
|
timer.done()
|
|
if (error != null) {
|
|
logger.error(
|
|
{ err: error, filePath: path, fileName: name },
|
|
'error uploading project'
|
|
)
|
|
if (error instanceof InvalidZipFileError) {
|
|
return res.status(422).json({
|
|
success: false,
|
|
error: req.i18n.translate(error.message),
|
|
})
|
|
} else {
|
|
return res.status(500).json({
|
|
success: false,
|
|
error: req.i18n.translate('upload_failed'),
|
|
})
|
|
}
|
|
} else {
|
|
return res.json({ success: true, project_id: project._id })
|
|
}
|
|
}
|
|
)
|
|
}
|
|
|
|
async function uploadFile(req, res, next) {
|
|
const timer = new metrics.Timer('file-upload')
|
|
const name = req.body.name
|
|
const path = req.file?.path
|
|
const projectId = req.params.Project_id
|
|
let { folder_id: folderId } = req.query
|
|
if (name == null || name.length === 0 || name.length > 150) {
|
|
return res.status(422).json({
|
|
success: false,
|
|
error: 'invalid_filename',
|
|
})
|
|
}
|
|
|
|
// preserve the directory structure from an uploaded folder
|
|
const { relativePath } = req.body
|
|
// NOTE: Uppy sends a "null" string for `relativePath` when the file is not nested in a folder
|
|
if (relativePath && relativePath !== 'null') {
|
|
const { path } = await ProjectLocator.promises.findElement({
|
|
project_id: projectId,
|
|
element_id: folderId,
|
|
type: 'folder',
|
|
})
|
|
const { lastFolder } = await EditorController.promises.mkdirp(
|
|
projectId,
|
|
Path.dirname(Path.join('/', path.fileSystem, relativePath))
|
|
)
|
|
folderId = lastFolder._id
|
|
}
|
|
|
|
const userId = SessionManager.getLoggedInUserId(req.session)
|
|
|
|
return FileSystemImportManager.addEntity(
|
|
userId,
|
|
projectId,
|
|
folderId,
|
|
name,
|
|
path,
|
|
true,
|
|
function (error, entity) {
|
|
fs.unlink(path, function () {})
|
|
timer.done()
|
|
if (error != null) {
|
|
if (error.name === 'InvalidNameError') {
|
|
return res.status(422).json({
|
|
success: false,
|
|
error: 'invalid_filename',
|
|
})
|
|
} else if (error instanceof DuplicateNameError) {
|
|
return res.status(422).json({
|
|
success: false,
|
|
error: 'duplicate_file_name',
|
|
})
|
|
} else if (error.message === 'project_has_too_many_files') {
|
|
return res.status(422).json({
|
|
success: false,
|
|
error: 'project_has_too_many_files',
|
|
})
|
|
} else if (error.message === 'folder_not_found') {
|
|
return res.status(422).json({
|
|
success: false,
|
|
error: 'folder_not_found',
|
|
})
|
|
} else {
|
|
logger.error(
|
|
{
|
|
err: error,
|
|
projectId,
|
|
filePath: path,
|
|
fileName: name,
|
|
folderId,
|
|
},
|
|
'error uploading file'
|
|
)
|
|
return res.status(422).json({ success: false })
|
|
}
|
|
} else {
|
|
return res.json({
|
|
success: true,
|
|
entity_id: entity?._id,
|
|
entity_type: entity?.type,
|
|
hash: entity?.hash,
|
|
})
|
|
}
|
|
}
|
|
)
|
|
}
|
|
|
|
function multerMiddleware(req, res, next) {
|
|
if (upload == null) {
|
|
return res
|
|
.status(500)
|
|
.json({ success: false, error: req.i18n.translate('upload_failed') })
|
|
}
|
|
return upload.single('qqfile')(req, res, function (err) {
|
|
if (err instanceof multer.MulterError && err.code === 'LIMIT_FILE_SIZE') {
|
|
return res
|
|
.status(422)
|
|
.json({ success: false, error: req.i18n.translate('file_too_large') })
|
|
}
|
|
|
|
return next(err)
|
|
})
|
|
}
|
|
|
|
export default {
|
|
uploadProject,
|
|
uploadFile: expressify(uploadFile),
|
|
multerMiddleware,
|
|
}
|