2020-02-19 06:14:28 -05:00
|
|
|
/* eslint-disable
|
|
|
|
camelcase,
|
|
|
|
handle-callback-err,
|
|
|
|
no-return-assign,
|
|
|
|
no-unused-vars,
|
|
|
|
no-useless-escape,
|
|
|
|
*/
|
|
|
|
// 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
|
|
|
let ResourceWriter
|
|
|
|
const UrlCache = require('./UrlCache')
|
|
|
|
const Path = require('path')
|
|
|
|
const fs = require('fs')
|
|
|
|
const async = require('async')
|
|
|
|
const OutputFileFinder = require('./OutputFileFinder')
|
|
|
|
const ResourceStateManager = require('./ResourceStateManager')
|
|
|
|
const Metrics = require('./Metrics')
|
|
|
|
const logger = require('logger-sharelatex')
|
|
|
|
const settings = require('settings-sharelatex')
|
2016-05-23 09:31:27 -04:00
|
|
|
|
2020-02-19 06:14:37 -05:00
|
|
|
const parallelFileDownloads = settings.parallelFileDownloads || 1
|
2014-02-12 12:27:43 -05:00
|
|
|
|
2020-02-19 06:14:37 -05:00
|
|
|
module.exports = ResourceWriter = {
|
|
|
|
syncResourcesToDisk(request, basePath, callback) {
|
|
|
|
if (callback == null) {
|
2020-08-10 12:01:11 -04:00
|
|
|
callback = function (error, resourceList) {}
|
2020-02-19 06:14:37 -05:00
|
|
|
}
|
|
|
|
if (request.syncType === 'incremental') {
|
|
|
|
logger.log(
|
|
|
|
{ project_id: request.project_id, user_id: request.user_id },
|
|
|
|
'incremental sync'
|
|
|
|
)
|
|
|
|
return ResourceStateManager.checkProjectStateMatches(
|
|
|
|
request.syncState,
|
|
|
|
basePath,
|
2020-08-10 12:01:11 -04:00
|
|
|
function (error, resourceList) {
|
2020-02-19 06:14:37 -05:00
|
|
|
if (error != null) {
|
|
|
|
return callback(error)
|
|
|
|
}
|
|
|
|
return ResourceWriter._removeExtraneousFiles(
|
|
|
|
resourceList,
|
|
|
|
basePath,
|
2020-08-10 12:01:11 -04:00
|
|
|
function (error, outputFiles, allFiles) {
|
2020-02-19 06:14:37 -05:00
|
|
|
if (error != null) {
|
|
|
|
return callback(error)
|
|
|
|
}
|
|
|
|
return ResourceStateManager.checkResourceFiles(
|
|
|
|
resourceList,
|
|
|
|
allFiles,
|
|
|
|
basePath,
|
2020-08-10 12:01:11 -04:00
|
|
|
function (error) {
|
2020-02-19 06:14:37 -05:00
|
|
|
if (error != null) {
|
|
|
|
return callback(error)
|
|
|
|
}
|
|
|
|
return ResourceWriter.saveIncrementalResourcesToDisk(
|
|
|
|
request.project_id,
|
|
|
|
request.resources,
|
|
|
|
basePath,
|
2020-08-10 12:01:11 -04:00
|
|
|
function (error) {
|
2020-02-19 06:14:37 -05:00
|
|
|
if (error != null) {
|
|
|
|
return callback(error)
|
|
|
|
}
|
|
|
|
return callback(null, resourceList)
|
|
|
|
}
|
|
|
|
)
|
|
|
|
}
|
|
|
|
)
|
|
|
|
}
|
|
|
|
)
|
|
|
|
}
|
|
|
|
)
|
|
|
|
} else {
|
|
|
|
logger.log(
|
|
|
|
{ project_id: request.project_id, user_id: request.user_id },
|
|
|
|
'full sync'
|
|
|
|
)
|
|
|
|
return this.saveAllResourcesToDisk(
|
|
|
|
request.project_id,
|
|
|
|
request.resources,
|
|
|
|
basePath,
|
2020-08-10 12:01:11 -04:00
|
|
|
function (error) {
|
2020-02-19 06:14:37 -05:00
|
|
|
if (error != null) {
|
|
|
|
return callback(error)
|
|
|
|
}
|
|
|
|
return ResourceStateManager.saveProjectState(
|
|
|
|
request.syncState,
|
|
|
|
request.resources,
|
|
|
|
basePath,
|
2020-08-10 12:01:11 -04:00
|
|
|
function (error) {
|
2020-02-19 06:14:37 -05:00
|
|
|
if (error != null) {
|
|
|
|
return callback(error)
|
|
|
|
}
|
|
|
|
return callback(null, request.resources)
|
|
|
|
}
|
|
|
|
)
|
|
|
|
}
|
|
|
|
)
|
|
|
|
}
|
|
|
|
},
|
2017-08-01 09:35:55 -04:00
|
|
|
|
2020-02-19 06:14:37 -05:00
|
|
|
saveIncrementalResourcesToDisk(project_id, resources, basePath, callback) {
|
|
|
|
if (callback == null) {
|
2020-08-10 12:01:11 -04:00
|
|
|
callback = function (error) {}
|
2020-02-19 06:14:37 -05:00
|
|
|
}
|
2020-08-10 12:01:11 -04:00
|
|
|
return this._createDirectory(basePath, (error) => {
|
2020-02-19 06:14:37 -05:00
|
|
|
if (error != null) {
|
|
|
|
return callback(error)
|
|
|
|
}
|
2020-08-10 12:01:11 -04:00
|
|
|
const jobs = Array.from(resources).map((resource) =>
|
|
|
|
((resource) => {
|
|
|
|
return (callback) =>
|
2020-02-19 06:14:37 -05:00
|
|
|
this._writeResourceToDisk(project_id, resource, basePath, callback)
|
|
|
|
})(resource)
|
|
|
|
)
|
|
|
|
return async.parallelLimit(jobs, parallelFileDownloads, callback)
|
|
|
|
})
|
|
|
|
},
|
2017-08-01 09:35:55 -04:00
|
|
|
|
2020-02-19 06:14:37 -05:00
|
|
|
saveAllResourcesToDisk(project_id, resources, basePath, callback) {
|
|
|
|
if (callback == null) {
|
2020-08-10 12:01:11 -04:00
|
|
|
callback = function (error) {}
|
2020-02-19 06:14:37 -05:00
|
|
|
}
|
2020-08-10 12:01:11 -04:00
|
|
|
return this._createDirectory(basePath, (error) => {
|
2020-02-19 06:14:37 -05:00
|
|
|
if (error != null) {
|
|
|
|
return callback(error)
|
|
|
|
}
|
2020-08-10 12:01:11 -04:00
|
|
|
return this._removeExtraneousFiles(resources, basePath, (error) => {
|
2020-02-19 06:14:37 -05:00
|
|
|
if (error != null) {
|
|
|
|
return callback(error)
|
|
|
|
}
|
2020-08-10 12:01:11 -04:00
|
|
|
const jobs = Array.from(resources).map((resource) =>
|
|
|
|
((resource) => {
|
|
|
|
return (callback) =>
|
2020-02-19 06:14:37 -05:00
|
|
|
this._writeResourceToDisk(
|
|
|
|
project_id,
|
|
|
|
resource,
|
|
|
|
basePath,
|
|
|
|
callback
|
|
|
|
)
|
|
|
|
})(resource)
|
|
|
|
)
|
|
|
|
return async.parallelLimit(jobs, parallelFileDownloads, callback)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
},
|
2017-08-01 09:35:55 -04:00
|
|
|
|
2020-02-19 06:14:37 -05:00
|
|
|
_createDirectory(basePath, callback) {
|
|
|
|
if (callback == null) {
|
2020-08-10 12:01:11 -04:00
|
|
|
callback = function (error) {}
|
2020-02-19 06:14:37 -05:00
|
|
|
}
|
2020-08-10 12:01:11 -04:00
|
|
|
return fs.mkdir(basePath, function (err) {
|
2020-02-19 06:14:37 -05:00
|
|
|
if (err != null) {
|
|
|
|
if (err.code === 'EEXIST') {
|
|
|
|
return callback()
|
|
|
|
} else {
|
|
|
|
logger.log({ err, dir: basePath }, 'error creating directory')
|
|
|
|
return callback(err)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return callback()
|
|
|
|
}
|
|
|
|
})
|
|
|
|
},
|
2016-03-31 06:25:25 -04:00
|
|
|
|
2020-02-19 06:14:37 -05:00
|
|
|
_removeExtraneousFiles(resources, basePath, _callback) {
|
|
|
|
if (_callback == null) {
|
2020-08-10 12:01:11 -04:00
|
|
|
_callback = function (error, outputFiles, allFiles) {}
|
2020-02-19 06:14:37 -05:00
|
|
|
}
|
|
|
|
const timer = new Metrics.Timer('unlink-output-files')
|
2020-08-10 12:01:11 -04:00
|
|
|
const callback = function (error, ...result) {
|
2020-02-19 06:14:37 -05:00
|
|
|
timer.done()
|
|
|
|
return _callback(error, ...Array.from(result))
|
|
|
|
}
|
2014-02-12 12:27:43 -05:00
|
|
|
|
2020-08-10 12:01:11 -04:00
|
|
|
return OutputFileFinder.findOutputFiles(resources, basePath, function (
|
2020-02-19 06:14:37 -05:00
|
|
|
error,
|
|
|
|
outputFiles,
|
|
|
|
allFiles
|
|
|
|
) {
|
|
|
|
if (error != null) {
|
|
|
|
return callback(error)
|
|
|
|
}
|
2014-02-12 12:27:43 -05:00
|
|
|
|
2020-02-19 06:14:37 -05:00
|
|
|
const jobs = []
|
|
|
|
for (const file of Array.from(outputFiles || [])) {
|
2020-08-10 12:01:11 -04:00
|
|
|
;(function (file) {
|
2020-02-19 06:14:37 -05:00
|
|
|
const { path } = file
|
|
|
|
let should_delete = true
|
|
|
|
if (
|
|
|
|
path.match(/^output\./) ||
|
|
|
|
path.match(/\.aux$/) ||
|
|
|
|
path.match(/^cache\//)
|
|
|
|
) {
|
|
|
|
// knitr cache
|
|
|
|
should_delete = false
|
|
|
|
}
|
|
|
|
if (path.match(/^output-.*/)) {
|
|
|
|
// Tikz cached figures (default case)
|
|
|
|
should_delete = false
|
|
|
|
}
|
|
|
|
if (path.match(/\.(pdf|dpth|md5)$/)) {
|
|
|
|
// Tikz cached figures (by extension)
|
|
|
|
should_delete = false
|
|
|
|
}
|
|
|
|
if (
|
|
|
|
path.match(/\.(pygtex|pygstyle)$/) ||
|
|
|
|
path.match(/(^|\/)_minted-[^\/]+\//)
|
|
|
|
) {
|
|
|
|
// minted files/directory
|
|
|
|
should_delete = false
|
|
|
|
}
|
|
|
|
if (
|
|
|
|
path.match(/\.md\.tex$/) ||
|
|
|
|
path.match(/(^|\/)_markdown_[^\/]+\//)
|
|
|
|
) {
|
|
|
|
// markdown files/directory
|
|
|
|
should_delete = false
|
|
|
|
}
|
|
|
|
if (path.match(/-eps-converted-to\.pdf$/)) {
|
|
|
|
// Epstopdf generated files
|
|
|
|
should_delete = false
|
|
|
|
}
|
|
|
|
if (
|
|
|
|
path === 'output.pdf' ||
|
|
|
|
path === 'output.dvi' ||
|
|
|
|
path === 'output.log' ||
|
2020-05-20 09:12:08 -04:00
|
|
|
path === 'output.xdv' ||
|
|
|
|
path === 'output.stdout' ||
|
|
|
|
path === 'output.stderr'
|
2020-02-19 06:14:37 -05:00
|
|
|
) {
|
|
|
|
should_delete = true
|
|
|
|
}
|
|
|
|
if (path === 'output.tex') {
|
|
|
|
// created by TikzManager if present in output files
|
|
|
|
should_delete = true
|
|
|
|
}
|
|
|
|
if (should_delete) {
|
2020-08-10 12:01:11 -04:00
|
|
|
return jobs.push((callback) =>
|
2020-02-19 06:14:37 -05:00
|
|
|
ResourceWriter._deleteFileIfNotDirectory(
|
|
|
|
Path.join(basePath, path),
|
|
|
|
callback
|
|
|
|
)
|
|
|
|
)
|
|
|
|
}
|
|
|
|
})(file)
|
|
|
|
}
|
2014-02-12 12:27:43 -05:00
|
|
|
|
2020-08-10 12:01:11 -04:00
|
|
|
return async.series(jobs, function (error) {
|
2020-02-19 06:14:37 -05:00
|
|
|
if (error != null) {
|
|
|
|
return callback(error)
|
|
|
|
}
|
|
|
|
return callback(null, outputFiles, allFiles)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
},
|
2014-02-12 12:27:43 -05:00
|
|
|
|
2020-02-19 06:14:37 -05:00
|
|
|
_deleteFileIfNotDirectory(path, callback) {
|
|
|
|
if (callback == null) {
|
2020-08-10 12:01:11 -04:00
|
|
|
callback = function (error) {}
|
2020-02-19 06:14:37 -05:00
|
|
|
}
|
2020-08-10 12:01:11 -04:00
|
|
|
return fs.stat(path, function (error, stat) {
|
2020-02-19 06:14:37 -05:00
|
|
|
if (error != null && error.code === 'ENOENT') {
|
|
|
|
return callback()
|
|
|
|
} else if (error != null) {
|
|
|
|
logger.err(
|
|
|
|
{ err: error, path },
|
|
|
|
'error stating file in deleteFileIfNotDirectory'
|
|
|
|
)
|
|
|
|
return callback(error)
|
|
|
|
} else if (stat.isFile()) {
|
2020-08-10 12:01:11 -04:00
|
|
|
return fs.unlink(path, function (error) {
|
2020-02-19 06:14:37 -05:00
|
|
|
if (error != null) {
|
|
|
|
logger.err(
|
|
|
|
{ err: error, path },
|
|
|
|
'error removing file in deleteFileIfNotDirectory'
|
|
|
|
)
|
|
|
|
return callback(error)
|
|
|
|
} else {
|
|
|
|
return callback()
|
|
|
|
}
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
return callback()
|
|
|
|
}
|
|
|
|
})
|
|
|
|
},
|
2014-02-12 12:27:43 -05:00
|
|
|
|
2020-02-19 06:14:37 -05:00
|
|
|
_writeResourceToDisk(project_id, resource, basePath, callback) {
|
|
|
|
if (callback == null) {
|
2020-08-10 12:01:11 -04:00
|
|
|
callback = function (error) {}
|
2020-02-19 06:14:37 -05:00
|
|
|
}
|
2020-08-10 12:01:11 -04:00
|
|
|
return ResourceWriter.checkPath(basePath, resource.path, function (
|
2020-02-19 06:14:37 -05:00
|
|
|
error,
|
|
|
|
path
|
|
|
|
) {
|
|
|
|
if (error != null) {
|
|
|
|
return callback(error)
|
|
|
|
}
|
2020-08-10 12:01:11 -04:00
|
|
|
return fs.mkdir(Path.dirname(path), { recursive: true }, function (
|
|
|
|
error
|
|
|
|
) {
|
2020-02-19 06:14:37 -05:00
|
|
|
if (error != null) {
|
|
|
|
return callback(error)
|
|
|
|
}
|
|
|
|
// TODO: Don't overwrite file if it hasn't been modified
|
|
|
|
if (resource.url != null) {
|
|
|
|
return UrlCache.downloadUrlToFile(
|
|
|
|
project_id,
|
|
|
|
resource.url,
|
|
|
|
path,
|
|
|
|
resource.modified,
|
2020-08-10 12:01:11 -04:00
|
|
|
function (err) {
|
2020-02-19 06:14:37 -05:00
|
|
|
if (err != null) {
|
|
|
|
logger.err(
|
|
|
|
{
|
|
|
|
err,
|
|
|
|
project_id,
|
|
|
|
path,
|
|
|
|
resource_url: resource.url,
|
|
|
|
modified: resource.modified
|
|
|
|
},
|
|
|
|
'error downloading file for resources'
|
|
|
|
)
|
2020-03-26 06:18:50 -04:00
|
|
|
Metrics.inc('download-failed')
|
2020-02-19 06:14:37 -05:00
|
|
|
}
|
|
|
|
return callback()
|
|
|
|
}
|
|
|
|
) // try and continue compiling even if http resource can not be downloaded at this time
|
|
|
|
} else {
|
|
|
|
const process = require('process')
|
|
|
|
fs.writeFile(path, resource.content, callback)
|
|
|
|
try {
|
|
|
|
let result
|
|
|
|
return (result = fs.lstatSync(path))
|
|
|
|
} catch (e) {}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|
|
|
|
},
|
2014-02-12 12:27:43 -05:00
|
|
|
|
2020-02-19 06:14:37 -05:00
|
|
|
checkPath(basePath, resourcePath, callback) {
|
|
|
|
const path = Path.normalize(Path.join(basePath, resourcePath))
|
|
|
|
if (path.slice(0, basePath.length + 1) !== basePath + '/') {
|
|
|
|
return callback(new Error('resource path is outside root directory'))
|
|
|
|
} else {
|
|
|
|
return callback(null, path)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|