2020-02-19 06:14:28 -05:00
|
|
|
/* eslint-disable
|
|
|
|
no-return-assign,
|
|
|
|
*/
|
|
|
|
// TODO: This file was created by bulk-decaffeinate.
|
|
|
|
// Fix any style issues and re-enable lint.
|
2020-02-19 06:14:14 -05:00
|
|
|
/*
|
|
|
|
* decaffeinate suggestions:
|
|
|
|
* DS101: Remove unnecessary use of Array.from
|
|
|
|
* 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
|
|
|
|
*/
|
2020-02-19 06:14:37 -05:00
|
|
|
const UrlFetcher = require('./UrlFetcher')
|
2021-07-12 12:47:21 -04:00
|
|
|
const Settings = require('@overleaf/settings')
|
2024-11-08 05:21:56 -05:00
|
|
|
const fs = require('node:fs')
|
|
|
|
const Path = require('node:path')
|
|
|
|
const { callbackify } = require('node:util')
|
2024-01-17 04:22:46 -05:00
|
|
|
const Metrics = require('./Metrics')
|
2014-02-12 12:27:43 -05:00
|
|
|
|
2021-10-06 04:11:59 -04:00
|
|
|
const PENDING_DOWNLOADS = new Map()
|
2014-02-12 12:27:43 -05:00
|
|
|
|
2021-10-06 04:11:59 -04:00
|
|
|
function getProjectDir(projectId) {
|
|
|
|
return Path.join(Settings.path.clsiCacheDir, projectId)
|
|
|
|
}
|
2014-02-12 12:27:43 -05:00
|
|
|
|
2021-10-06 04:11:59 -04:00
|
|
|
function getCachePath(projectId, url, lastModified) {
|
|
|
|
// The url is a filestore URL.
|
|
|
|
// It is sufficient to look at the path and mtime for uniqueness.
|
|
|
|
const mtime = (lastModified && lastModified.getTime()) || 0
|
|
|
|
const key = new URL(url).pathname.replace(/\//g, '-') + '-' + mtime
|
|
|
|
return Path.join(getProjectDir(projectId), key)
|
|
|
|
}
|
2014-02-12 12:27:43 -05:00
|
|
|
|
2024-01-17 04:22:46 -05:00
|
|
|
async function clearProject(projectId, options) {
|
|
|
|
const timer = new Metrics.Timer('url_cache', {
|
|
|
|
status: options?.reason || 'unknown',
|
|
|
|
path: 'delete',
|
|
|
|
})
|
2024-01-08 05:43:41 -05:00
|
|
|
await fs.promises.rm(getProjectDir(projectId), {
|
|
|
|
force: true,
|
|
|
|
recursive: true,
|
|
|
|
})
|
2024-01-17 04:22:46 -05:00
|
|
|
timer.done()
|
2021-10-06 04:11:59 -04:00
|
|
|
}
|
2014-02-12 12:27:43 -05:00
|
|
|
|
2021-10-06 04:11:59 -04:00
|
|
|
async function createProjectDir(projectId) {
|
|
|
|
await fs.promises.mkdir(getProjectDir(projectId), { recursive: true })
|
|
|
|
}
|
2014-02-12 12:27:43 -05:00
|
|
|
|
2021-10-06 04:11:59 -04:00
|
|
|
async function downloadUrlToFile(projectId, url, destPath, lastModified) {
|
|
|
|
const cachePath = getCachePath(projectId, url, lastModified)
|
|
|
|
try {
|
2024-01-17 04:22:46 -05:00
|
|
|
const timer = new Metrics.Timer('url_cache', {
|
|
|
|
status: 'cache-hit',
|
|
|
|
path: 'copy',
|
|
|
|
})
|
2021-10-06 04:11:59 -04:00
|
|
|
await fs.promises.copyFile(cachePath, destPath)
|
2024-01-17 04:22:46 -05:00
|
|
|
// the metric is only updated if the file is present in the cache
|
|
|
|
timer.done()
|
2021-10-06 04:11:59 -04:00
|
|
|
return
|
|
|
|
} catch (e) {
|
|
|
|
if (e.code !== 'ENOENT') {
|
|
|
|
throw e
|
2020-02-19 06:14:37 -05:00
|
|
|
}
|
2021-10-06 04:11:59 -04:00
|
|
|
}
|
2024-01-17 04:22:46 -05:00
|
|
|
// time the download
|
|
|
|
{
|
|
|
|
const timer = new Metrics.Timer('url_cache', {
|
|
|
|
status: 'cache-miss',
|
|
|
|
path: 'download',
|
|
|
|
})
|
|
|
|
try {
|
|
|
|
await download(url, cachePath)
|
|
|
|
} finally {
|
|
|
|
timer.done()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// time the file copy
|
|
|
|
{
|
|
|
|
const timer = new Metrics.Timer('url_cache', {
|
|
|
|
status: 'cache-miss',
|
|
|
|
path: 'copy',
|
|
|
|
})
|
|
|
|
await fs.promises.copyFile(cachePath, destPath)
|
|
|
|
timer.done()
|
|
|
|
}
|
2021-10-06 04:11:59 -04:00
|
|
|
}
|
2014-02-12 12:27:43 -05:00
|
|
|
|
2021-10-06 04:11:59 -04:00
|
|
|
async function download(url, cachePath) {
|
|
|
|
let pending = PENDING_DOWNLOADS.get(cachePath)
|
|
|
|
if (pending) {
|
|
|
|
return pending
|
|
|
|
}
|
2018-07-30 11:22:04 -04:00
|
|
|
|
2021-10-06 04:11:59 -04:00
|
|
|
pending = UrlFetcher.promises.pipeUrlToFileWithRetry(url, cachePath)
|
|
|
|
PENDING_DOWNLOADS.set(cachePath, pending)
|
|
|
|
try {
|
|
|
|
await pending
|
|
|
|
} finally {
|
|
|
|
PENDING_DOWNLOADS.delete(cachePath)
|
|
|
|
}
|
|
|
|
}
|
2014-02-12 12:27:43 -05:00
|
|
|
|
2021-10-06 04:11:59 -04:00
|
|
|
module.exports = {
|
|
|
|
clearProject: callbackify(clearProject),
|
|
|
|
createProjectDir: callbackify(createProjectDir),
|
|
|
|
downloadUrlToFile: callbackify(downloadUrlToFile),
|
|
|
|
promises: {
|
|
|
|
clearProject,
|
|
|
|
createProjectDir,
|
|
|
|
downloadUrlToFile,
|
2021-07-13 07:04:48 -04:00
|
|
|
},
|
2020-02-19 06:14:37 -05:00
|
|
|
}
|