Path = require "path" async = require "async" Settings = require "settings-sharelatex" request = require('request') Project = require("../../models/Project").Project ProjectEntityHandler = require("../Project/ProjectEntityHandler") logger = require "logger-sharelatex" url = require("url") ClsiRequestManager = require("./ClsiRequestManager") module.exports = ClsiManager = _makeRequest: (project_id, opts, callback)-> ClsiRequestManager.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, callback sendRequest: (project_id, options = {}, callback = (error, success) ->) -> ClsiManager._buildRequest project_id, options, (error, req) -> return callback(error) if error? logger.log project_id: project_id, "sending compile to CLSI" ClsiManager._postToClsi project_id, req, options.compileGroup, (error, response) -> return callback(error) if error? logger.log project_id: project_id, response: response, "received compile response from CLSI" callback( null response?.compile?.status ClsiManager._parseOutputFiles(project_id, response?.compile?.outputFiles) ) deleteAuxFiles: (project_id, options, callback = (error) ->) -> compilerUrl = @_getCompilerUrl(options?.compileGroup) opts = url:"#{compilerUrl}/project/#{project_id}" method:"DELETE" ClsiManager._makeRequest project_id, opts, callback _getCompilerUrl: (compileGroup) -> if compileGroup == "priority" return Settings.apis.clsi_priority.url else return Settings.apis.clsi.url _postToClsi: (project_id, req, compileGroup, callback = (error, response) ->) -> compilerUrl = @_getCompilerUrl(compileGroup) opts = url: "#{compilerUrl}/project/#{project_id}/compile" json: req 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 _parseOutputFiles: (project_id, rawOutputFiles = []) -> outputFiles = [] for file in rawOutputFiles outputFiles.push path: url.parse(file.url).path.replace("/project/#{project_id}/output/", "") type: file.type build: file.build return outputFiles VALID_COMPILERS: ["pdflatex", "latex", "xelatex", "lualatex"] _buildRequest: (project_id, options={}, callback = (error, request) ->) -> Project.findById project_id, {compiler: 1, rootDoc_id: 1, imageName: 1}, (error, project) -> return callback(error) if error? return callback(new Errors.NotFoundError("project does not exist: #{project_id}")) if !project? if project.compiler not in ClsiManager.VALID_COMPILERS project.compiler = "pdflatex" ProjectEntityHandler.getAllDocs project_id, (error, docs = {}) -> return callback(error) if error? ProjectEntityHandler.getAllFiles project_id, (error, files = {}) -> return callback(error) if error? resources = [] rootResourcePath = null rootResourcePathOverride = null 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 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" 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() callback null, { compile: options: compiler: project.compiler timeout: options.timeout imageName: project.imageName draft: !!options.draft rootResourcePath: rootResourcePath resources: resources } wordCount: (project_id, file, options, callback = (error, response) ->) -> ClsiManager._buildRequest project_id, options, (error, req) -> compilerUrl = ClsiManager._getCompilerUrl(options?.compileGroup) filename = file || req?.compile?.rootResourcePath wordcount_url = "#{compilerUrl}/project/#{project_id}/wordcount?file=#{encodeURIComponent(filename)}" if req.compile.options.imageName? wordcount_url += "&image=#{encodeURIComponent(req.compile.options.imageName)}" opts = url: wordcount_url method: "GET" ClsiManager._makeRequest project_id, opts, (error, response, body) -> 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