overleaf/services/web/app/coffee/Features/Compile/ClsiManager.coffee

174 lines
7 KiB
CoffeeScript
Raw Normal View History

2014-02-12 10:23:40 +00:00
Path = require "path"
async = require "async"
Settings = require "settings-sharelatex"
request = require('request')
Project = require("../../models/Project").Project
2014-05-06 11:54:26 +00:00
ProjectEntityHandler = require("../Project/ProjectEntityHandler")
2014-02-12 10:23:40 +00:00
logger = require "logger-sharelatex"
Url = require("url")
ClsiCookieManager = require("./ClsiCookieManager")
_ = require("underscore")
async = require("async")
2016-06-02 12:09:11 +00:00
ClsiFormatChecker = require("./ClsiFormatChecker")
2014-02-12 10:23:40 +00:00
module.exports = ClsiManager =
2016-04-19 15:48:51 +00:00
sendRequest: (project_id, user_id, options = {}, callback = (error, status, outputFiles, clsiServerId, validationProblems) ->) ->
ClsiManager._buildRequest project_id, options, (error, req) ->
2014-02-12 10:23:40 +00:00
return callback(error) if error?
2014-05-06 11:54:26 +00:00
logger.log project_id: project_id, "sending compile to CLSI"
2016-06-02 12:09:11 +00:00
ClsiFormatChecker.checkRecoursesForProblems req.compile?.resources, (err, validationProblems)->
if err?
logger.err err, project_id, "could not check resources for potential problems before sending to clsi"
return callback(err)
2016-06-02 12:09:11 +00:00
if validationProblems?
logger.log project_id:project_id, validationProblems:validationProblems, "problems with users latex before compile was attempted"
return callback(null, "validation-problems", null, null, validationProblems)
ClsiManager._postToClsi project_id, user_id, req, options.compileGroup, (error, response) ->
if error?
logger.err err:error, project_id:project_id, "error sending request to clsi"
return callback(error)
logger.log project_id: project_id, outputFilesLength: response?.outputFiles?.length, status: response?.status, "received compile response from CLSI"
ClsiCookieManager._getServerId project_id, (err, clsiServerId)->
if err?
logger.err err:err, project_id:project_id, "error getting server id"
return callback(err)
outputFiles = ClsiManager._parseOutputFiles(project_id, response?.compile?.outputFiles)
callback(null, response?.compile?.status, outputFiles, clsiServerId)
2014-02-12 10:23:40 +00:00
2016-07-14 13:48:46 +00:00
stopCompile: (project_id, user_id, options, callback = (error) ->) ->
compilerUrl = @_getCompilerUrl(options?.compileGroup, project_id, user_id, "compile/stop")
opts =
url:compilerUrl
method:"POST"
ClsiManager._makeRequest project_id, opts, callback
deleteAuxFiles: (project_id, user_id, options, callback = (error) ->) ->
compilerUrl = @_getCompilerUrl(options?.compileGroup, project_id, user_id)
2016-04-19 15:48:51 +00:00
opts =
url:compilerUrl
2016-04-19 15:48:51 +00:00
method:"DELETE"
ClsiManager._makeRequest project_id, opts, callback
_makeRequest: (project_id, opts, callback)->
ClsiCookieManager.getCookieJar project_id, (err, jar)->
if err?
logger.err err:err, "error getting cookie jar for clsi request"
return callback(err)
opts.jar = jar
request opts, (err, response, body)->
if err?
logger.err err:err, project_id:project_id, url:opts?.url, "error making request to clsi"
return callback(err)
ClsiCookieManager.setServerId project_id, response, (err)->
if err?
logger.warn err:err, project_id:project_id, "error setting server id"
return callback err, response, body
_getCompilerUrl: (compileGroup, project_id, user_id, action) ->
host = Settings.apis.clsi.url
path = "/project/#{project_id}"
path += "/user/#{user_id}" if user_id?
path += "/#{action}" if action?
return "#{host}#{path}"
_postToClsi: (project_id, user_id, req, compileGroup, callback = (error, response) ->) ->
compileUrl = @_getCompilerUrl(compileGroup, project_id, user_id, "compile")
opts =
url: compileUrl
2014-02-12 10:23:40 +00:00
json: req
2016-04-19 15:48:51 +00:00
method: "POST"
ClsiManager._makeRequest project_id, opts, (error, response, body) ->
return callback(error) if error?
if 200 <= response.statusCode < 300
callback null, body
else if response.statusCode == 413
callback null, compile:status:"project-too-large"
else
error = new Error("CLSI returned non-success code: #{response.statusCode}")
logger.error err: error, project_id: project_id, "CLSI returned failure code"
callback error, body
2014-02-12 10:23:40 +00:00
_parseOutputFiles: (project_id, rawOutputFiles = []) ->
2014-02-12 10:23:40 +00:00
outputFiles = []
for file in rawOutputFiles
outputFiles.push
2016-05-26 15:26:58 +00:00
path: file.path # the clsi is now sending this to web
2016-06-02 14:41:33 +00:00
url: Url.parse(file.url).path # the location of the file on the clsi, excluding the host part
2014-02-12 10:23:40 +00:00
type: file.type
build: file.build
2014-02-12 10:23:40 +00:00
return outputFiles
VALID_COMPILERS: ["pdflatex", "latex", "xelatex", "lualatex"]
2016-02-02 14:50:48 +00:00
_buildRequest: (project_id, options={}, callback = (error, request) ->) ->
2016-01-14 16:35:42 +00:00
Project.findById project_id, {compiler: 1, rootDoc_id: 1, imageName: 1}, (error, project) ->
2014-05-06 11:54:26 +00:00
return callback(error) if error?
return callback(new Errors.NotFoundError("project does not exist: #{project_id}")) if !project?
2014-02-12 10:23:40 +00:00
2014-05-06 11:54:26 +00:00
if project.compiler not in ClsiManager.VALID_COMPILERS
project.compiler = "pdflatex"
2014-02-12 10:23:40 +00:00
2014-05-06 11:54:26 +00:00
ProjectEntityHandler.getAllDocs project_id, (error, docs = {}) ->
return callback(error) if error?
ProjectEntityHandler.getAllFiles project_id, (error, files = {}) ->
return callback(error) if error?
2014-02-12 10:23:40 +00:00
2014-05-06 11:54:26 +00:00
resources = []
rootResourcePath = null
rootResourcePathOverride = null
2014-02-12 10:23:40 +00:00
2014-05-06 11:54:26 +00:00
for path, doc of docs
path = path.replace(/^\//, "") # Remove leading /
resources.push
path: path
content: doc.lines.join("\n")
if project.rootDoc_id? and doc._id.toString() == project.rootDoc_id.toString()
rootResourcePath = path
2016-02-02 14:50:48 +00:00
if options.rootDoc_id? and doc._id.toString() == options.rootDoc_id.toString()
rootResourcePathOverride = path
rootResourcePath = rootResourcePathOverride if rootResourcePathOverride?
if !rootResourcePath?
logger.warn {project_id}, "no root document found, setting to main.tex"
rootResourcePath = "main.tex"
2014-02-12 10:23:40 +00:00
2014-05-06 11:54:26 +00:00
for path, file of files
path = path.replace(/^\//, "") # Remove leading /
resources.push
path: path
url: "#{Settings.apis.filestore.url}/project/#{project._id}/file/#{file._id}"
modified: file.created?.getTime()
2014-02-12 10:23:40 +00:00
callback null, {
compile:
options:
compiler: project.compiler
timeout: options.timeout
imageName: project.imageName
draft: !!options.draft
2016-07-26 15:25:19 +00:00
check: options.check
rootResourcePath: rootResourcePath
resources: resources
}
2015-09-10 15:41:48 +00:00
wordCount: (project_id, user_id, file, options, callback = (error, response) ->) ->
2015-09-11 12:53:06 +00:00
ClsiManager._buildRequest project_id, options, (error, req) ->
2015-10-19 21:14:52 +00:00
filename = file || req?.compile?.rootResourcePath
wordcount_url = ClsiManager._getCompilerUrl(options?.compileGroup, project_id, user_id, "wordcount")
2016-04-19 15:48:51 +00:00
opts =
url: wordcount_url
qs:
file: filename
image: req.compile.options.imageName
2016-04-19 15:48:51 +00:00
method: "GET"
ClsiManager._makeRequest project_id, opts, (error, response, body) ->
2015-09-11 12:53:06 +00:00
return callback(error) if error?
if 200 <= response.statusCode < 300
callback null, body
else
error = new Error("CLSI returned non-success code: #{response.statusCode}")
logger.error err: error, project_id: project_id, "CLSI returned failure code"
callback error, body