overleaf/services/clsi/app/js/UrlFetcher.js

56 lines
1.4 KiB
JavaScript
Raw Normal View History

const fs = require('fs')
const logger = require('@overleaf/logger')
const Settings = require('@overleaf/settings')
const { URL } = require('url')
const { promisify } = require('util')
const fetch = require('node-fetch')
const pipeline = promisify(require('stream').pipeline)
async function pipeUrlToFileWithRetry(url, filePath) {
let remainingAttempts = 3
let lastErr
while (remainingAttempts-- > 0) {
try {
await pipeUrlToFile(url, filePath)
return
} catch (err) {
logger.warn(
{ err, url, filePath, remainingAttempts },
'error downloading url'
)
lastErr = err
}
}
throw lastErr
}
2015-05-15 09:07:15 -04:00
async function pipeUrlToFile(url, filePath) {
const u = new URL(url)
if (
Settings.filestoreDomainOveride &&
u.host !== Settings.apis.clsiPerf.host
) {
url = `${Settings.filestoreDomainOveride}${u.pathname}${u.search}`
}
const res = await fetch(url, { signal: AbortSignal.timeout(60 * 1000) })
if (res.status !== 200) {
throw new Error('non success response: ' + res.statusText)
}
const atomicWrite = filePath + '~'
try {
await pipeline(res.body, fs.createWriteStream(atomicWrite))
await fs.promises.rename(atomicWrite, filePath)
} catch (err) {
try {
await fs.promises.unlink(atomicWrite)
} catch (e) {}
throw err
}
}
module.exports.promises = {
pipeUrlToFileWithRetry,
}