mirror of
https://github.com/yu-i-i/overleaf-cep.git
synced 2026-05-23 17:19:37 +02:00
[clsi] Use clsi-nginx for downloading pandoc exports GitOrigin-RevId: b6013fae6f53c7af714634d700ceed491d724653
136 lines
4.1 KiB
JavaScript
136 lines
4.1 KiB
JavaScript
import crypto from 'node:crypto'
|
|
import logger from '@overleaf/logger'
|
|
import { expressify } from '@overleaf/promise-utils'
|
|
import fs from 'node:fs/promises'
|
|
import fsSync from 'node:fs'
|
|
import ConversionManager from './ConversionManager.js'
|
|
import ConversionOutputCleaner from './ConversionOutputCleaner.js'
|
|
import OutputCacheManager from './OutputCacheManager.js'
|
|
import ResourceWriter from './ResourceWriter.js'
|
|
import RequestParser from './RequestParser.js'
|
|
import { pipeline } from 'node:stream/promises'
|
|
import Settings from '@overleaf/settings'
|
|
import Path from 'node:path'
|
|
|
|
const CONVERSION_CONFIGS = {
|
|
docx: { extension: 'docx' },
|
|
markdown: { extension: 'zip' },
|
|
}
|
|
|
|
async function convertDocumentToLaTeX(req, res) {
|
|
const { path } = req.file
|
|
const conversionType = req.query.type
|
|
if (!Settings.enablePandocConversions) {
|
|
await fs.unlink(path).catch(() => {})
|
|
return res.sendStatus(404)
|
|
}
|
|
if (!conversionType || !['docx', 'markdown'].includes(conversionType)) {
|
|
await fs.unlink(path).catch(() => {})
|
|
return res.sendStatus(400)
|
|
}
|
|
logger.debug({ path, conversionType }, 'received file for conversion')
|
|
const conversionId = crypto.randomUUID()
|
|
let zipPath
|
|
try {
|
|
zipPath = await ConversionManager.promises.convertToLaTeXWithLock(
|
|
conversionId,
|
|
path,
|
|
conversionType
|
|
)
|
|
} finally {
|
|
await fs.unlink(path).catch(() => {})
|
|
}
|
|
|
|
try {
|
|
const zipStat = await fs.stat(zipPath)
|
|
|
|
res.setHeader('Content-Length', zipStat.size)
|
|
res.attachment('conversion.zip')
|
|
res.setHeader('X-Content-Type-Options', 'nosniff')
|
|
|
|
const readStream = fsSync.createReadStream(zipPath)
|
|
await pipeline(readStream, res)
|
|
} finally {
|
|
await fs
|
|
.rm(Path.dirname(zipPath), { recursive: true, force: true })
|
|
.catch(() => {})
|
|
}
|
|
}
|
|
|
|
async function convertProjectToDocument(req, res) {
|
|
if (!Settings.enablePandocConversions) {
|
|
return res.sendStatus(404)
|
|
}
|
|
|
|
const type = req.query.type
|
|
if (!Object.hasOwn(CONVERSION_CONFIGS, type)) {
|
|
return res.sendStatus(400)
|
|
}
|
|
const config = CONVERSION_CONFIGS[type]
|
|
|
|
const request = await RequestParser.promises.parse(req.body)
|
|
request.project_id = req.params.project_id
|
|
request.user_id = req.params.user_id
|
|
request.metricsOpts = {}
|
|
|
|
const responseFormat = req.query.responseFormat === 'json' ? 'json' : 'stream'
|
|
|
|
const conversionId = crypto.randomUUID()
|
|
const conversionDir = Path.join(Settings.path.compilesDir, conversionId)
|
|
|
|
logger.debug(
|
|
{
|
|
projectId: request.project_id,
|
|
userId: request.user_id,
|
|
rootResourcePath: request.rootResourcePath,
|
|
type,
|
|
},
|
|
'syncing resources for project-to-document conversion'
|
|
)
|
|
|
|
try {
|
|
await ResourceWriter.promises.syncResourcesToDisk(request, conversionDir)
|
|
|
|
const documentPath =
|
|
await ConversionManager.promises.convertLaTeXToDocumentInDirWithLock(
|
|
conversionId,
|
|
conversionDir,
|
|
request.rootResourcePath,
|
|
type
|
|
)
|
|
|
|
const outputName = `output.${config.extension}`
|
|
if (responseFormat === 'json') {
|
|
// TODO: drop the streaming branch once web is migrated to the two-step flow
|
|
const buildId = await OutputCacheManager.promises.generateBuildId()
|
|
const buildDir = Path.join(
|
|
Settings.path.outputDir,
|
|
conversionId,
|
|
OutputCacheManager.CACHE_SUBDIR,
|
|
buildId
|
|
)
|
|
try {
|
|
await fs.mkdir(buildDir, { recursive: true })
|
|
await fs.copyFile(documentPath, Path.join(buildDir, outputName))
|
|
res.json({ conversionId, buildId, file: outputName })
|
|
} finally {
|
|
ConversionOutputCleaner.scheduleCleanup(conversionId)
|
|
}
|
|
} else {
|
|
const documentStat = await fs.stat(documentPath)
|
|
res.setHeader('Content-Length', documentStat.size)
|
|
res.attachment(outputName)
|
|
res.setHeader('X-Content-Type-Options', 'nosniff')
|
|
const readStream = fsSync.createReadStream(documentPath)
|
|
await pipeline(readStream, res)
|
|
}
|
|
} finally {
|
|
await fs.rm(conversionDir, { recursive: true, force: true }).catch(() => {})
|
|
}
|
|
}
|
|
|
|
export default {
|
|
convertDocumentToLaTeX: expressify(convertDocumentToLaTeX),
|
|
convertProjectToDocument: expressify(convertProjectToDocument),
|
|
}
|