overleaf/services/clsi/app/js/OutputFileArchiveManager.js

97 lines
2.6 KiB
JavaScript
Raw Normal View History

const archiver = require('archiver')
const OutputCacheManager = require('./OutputCacheManager')
const OutputFileFinder = require('./OutputFileFinder')
const Settings = require('@overleaf/settings')
const { open } = require('node:fs/promises')
const path = require('path')
const { NotFoundError } = require('./Errors')
const logger = require('@overleaf/logger')
// NOTE: Updating this list requires a corresponding change in
// * services/web/frontend/js/features/pdf-preview/util/file-list.js
const ignoreFiles = ['output.fls', 'output.fdb_latexmk']
function getContentDir(projectId, userId) {
let subDir
if (userId != null) {
subDir = `${projectId}-${userId}`
} else {
subDir = projectId
}
return `${Settings.path.outputDir}/${subDir}/`
}
module.exports = {
async archiveFilesForBuild(projectId, userId, build) {
logger.debug({ projectId, userId, build }, 'Will create zip file')
const outputFiles = await this._getAllOutputFiles(projectId, userId, build)
const archive = archiver('zip')
archive.on('error', err => {
logger.warn({ err }, 'error emitted when creating output files archive')
})
const missingFiles = []
for (const file of outputFiles) {
let fileHandle
try {
fileHandle = await open(file, 'r')
} catch (error) {
logger.warn(
{ file, error },
'error opening file to add to output files archive'
)
missingFiles.push(file)
continue
}
const fileStream = fileHandle.createReadStream()
archive.append(fileStream, {
name: path.basename(file),
})
}
if (missingFiles.length > 0) {
archive.append(missingFiles.join('\n'), {
name: 'missing_files.txt',
})
}
await archive.finalize()
return archive
},
async _getAllOutputFiles(projectId, userId, build) {
const contentDir = getContentDir(projectId, userId)
try {
const { outputFiles } = await OutputFileFinder.promises.findOutputFiles(
[],
`${contentDir}${OutputCacheManager.path(build, '.')}`
)
return outputFiles
.filter(
// Ignore the pdf and also ignore the files ignored by the frontend.
({ path }) => path !== 'output.pdf' && !ignoreFiles.includes(path)
)
.map(
({ path }) => `${contentDir}${OutputCacheManager.path(build, path)}`
)
} catch (error) {
if (
error.code === 'ENOENT' ||
error.code === 'ENOTDIR' ||
error.code === 'EACCES'
) {
throw new NotFoundError('Output files not found')
}
throw error
}
},
}