2014-02-12 10:23:40 +00:00
|
|
|
Settings = require('settings-sharelatex')
|
2017-05-04 14:22:54 +00:00
|
|
|
RedisWrapper = require("../../infrastructure/RedisWrapper")
|
|
|
|
rclient = RedisWrapper.client("clsi_recently_compiled")
|
2018-02-15 12:18:43 +00:00
|
|
|
ProjectGetter = require('../Project/ProjectGetter')
|
2014-02-12 10:23:40 +00:00
|
|
|
ProjectRootDocManager = require "../Project/ProjectRootDocManager"
|
2014-11-28 14:26:21 +00:00
|
|
|
UserGetter = require "../User/UserGetter"
|
2014-02-12 10:23:40 +00:00
|
|
|
ClsiManager = require "./ClsiManager"
|
2017-04-03 15:18:30 +00:00
|
|
|
Metrics = require('metrics-sharelatex')
|
2014-02-12 10:23:40 +00:00
|
|
|
logger = require("logger-sharelatex")
|
2014-02-28 17:59:54 +00:00
|
|
|
rateLimiter = require("../../infrastructure/RateLimiter")
|
2014-02-12 10:23:40 +00:00
|
|
|
|
|
|
|
module.exports = CompileManager =
|
2016-05-18 09:09:22 +00:00
|
|
|
|
|
|
|
|
2014-11-28 14:26:21 +00:00
|
|
|
compile: (project_id, user_id, options = {}, _callback = (error) ->) ->
|
2014-02-12 10:23:40 +00:00
|
|
|
timer = new Metrics.Timer("editor.compile")
|
|
|
|
callback = (args...) ->
|
|
|
|
timer.done()
|
|
|
|
_callback(args...)
|
|
|
|
|
2018-04-17 13:43:54 +00:00
|
|
|
logger.log project_id: project_id, user_id: user_id, "compiling project"
|
|
|
|
CompileManager._checkIfRecentlyCompiled project_id, user_id, (error, recentlyCompiled) ->
|
|
|
|
return callback(error) if error?
|
|
|
|
if recentlyCompiled
|
|
|
|
logger.warn {project_id, user_id}, "project was recently compiled so not continuing"
|
|
|
|
return callback null, "too-recently-compiled", []
|
|
|
|
|
|
|
|
CompileManager._checkIfAutoCompileLimitHasBeenHit options.isAutoCompile, "everyone", (err, canCompile)->
|
|
|
|
if !canCompile
|
|
|
|
return callback null, "autocompile-backoff", []
|
2014-06-01 17:26:33 +00:00
|
|
|
|
2014-02-12 10:23:40 +00:00
|
|
|
CompileManager._ensureRootDocumentIsSet project_id, (error) ->
|
|
|
|
return callback(error) if error?
|
2017-07-28 14:01:05 +00:00
|
|
|
CompileManager.getProjectCompileLimits project_id, (error, limits) ->
|
2014-02-12 10:23:40 +00:00
|
|
|
return callback(error) if error?
|
2017-07-28 14:01:05 +00:00
|
|
|
for key, value of limits
|
|
|
|
options[key] = value
|
2017-10-09 15:31:01 +00:00
|
|
|
# Put a lower limit on autocompiles for free users, based on compileGroup
|
|
|
|
CompileManager._checkCompileGroupAutoCompileLimit options.isAutoCompile, limits.compileGroup, (err, canCompile)->
|
2017-10-03 15:16:21 +00:00
|
|
|
if !canCompile
|
|
|
|
return callback null, "autocompile-backoff", []
|
|
|
|
# only pass user_id down to clsi if this is a per-user compile
|
|
|
|
compileAsUser = if Settings.disablePerUserCompiles then undefined else user_id
|
|
|
|
ClsiManager.sendRequest project_id, compileAsUser, options, (error, status, outputFiles, clsiServerId, validationProblems) ->
|
|
|
|
return callback(error) if error?
|
|
|
|
logger.log files: outputFiles, "output files"
|
|
|
|
callback(null, status, outputFiles, clsiServerId, limits, validationProblems)
|
2017-07-28 14:01:05 +00:00
|
|
|
|
2016-07-14 13:48:46 +00:00
|
|
|
|
|
|
|
stopCompile: (project_id, user_id, callback = (error) ->) ->
|
|
|
|
CompileManager.getProjectCompileLimits project_id, (error, limits) ->
|
|
|
|
return callback(error) if error?
|
|
|
|
ClsiManager.stopCompile project_id, user_id, limits, callback
|
|
|
|
|
2016-05-31 15:20:24 +00:00
|
|
|
deleteAuxFiles: (project_id, user_id, callback = (error) ->) ->
|
2014-12-01 12:19:01 +00:00
|
|
|
CompileManager.getProjectCompileLimits project_id, (error, limits) ->
|
|
|
|
return callback(error) if error?
|
2016-05-31 15:20:24 +00:00
|
|
|
ClsiManager.deleteAuxFiles project_id, user_id, limits, callback
|
2014-02-12 10:23:40 +00:00
|
|
|
|
2014-11-28 14:26:21 +00:00
|
|
|
getProjectCompileLimits: (project_id, callback = (error, limits) ->) ->
|
2018-02-15 12:18:43 +00:00
|
|
|
ProjectGetter.getProject project_id, owner_ref: 1, (error, project) ->
|
2014-11-28 14:26:21 +00:00
|
|
|
return callback(error) if error?
|
|
|
|
UserGetter.getUser project.owner_ref, {"features":1}, (err, owner)->
|
|
|
|
return callback(error) if error?
|
|
|
|
callback null, {
|
|
|
|
timeout: owner.features?.compileTimeout || Settings.defaultFeatures.compileTimeout
|
|
|
|
compileGroup: owner.features?.compileGroup || Settings.defaultFeatures.compileGroup
|
|
|
|
}
|
2014-05-19 15:10:41 +00:00
|
|
|
|
2014-02-12 10:23:40 +00:00
|
|
|
COMPILE_DELAY: 1 # seconds
|
|
|
|
_checkIfRecentlyCompiled: (project_id, user_id, callback = (error, recentlyCompiled) ->) ->
|
|
|
|
key = "compile:#{project_id}:#{user_id}"
|
|
|
|
rclient.set key, true, "EX", @COMPILE_DELAY, "NX", (error, ok) ->
|
|
|
|
return callback(error) if error?
|
|
|
|
if ok == "OK"
|
|
|
|
return callback null, false
|
|
|
|
else
|
|
|
|
return callback null, true
|
|
|
|
|
2017-10-09 15:31:01 +00:00
|
|
|
_checkCompileGroupAutoCompileLimit: (isAutoCompile, compileGroup, callback = (err, canCompile)->)->
|
2017-10-12 15:18:14 +00:00
|
|
|
if !isAutoCompile
|
|
|
|
return callback(null, true)
|
|
|
|
if compileGroup is "standard"
|
|
|
|
# apply extra limits to the standard compile group
|
2017-10-09 15:31:01 +00:00
|
|
|
CompileManager._checkIfAutoCompileLimitHasBeenHit isAutoCompile, compileGroup, callback
|
|
|
|
else
|
|
|
|
Metrics.inc "auto-compile-#{compileGroup}"
|
|
|
|
return callback(null, true) # always allow priority group users to compile
|
2017-10-03 15:16:21 +00:00
|
|
|
|
2017-10-09 15:31:01 +00:00
|
|
|
_checkIfAutoCompileLimitHasBeenHit: (isAutoCompile, compileGroup, callback = (err, canCompile)->)->
|
2017-10-03 15:16:21 +00:00
|
|
|
if !isAutoCompile
|
|
|
|
return callback(null, true)
|
2017-10-09 15:31:01 +00:00
|
|
|
Metrics.inc "auto-compile-#{compileGroup}"
|
2017-10-03 15:16:21 +00:00
|
|
|
opts =
|
|
|
|
endpointName:"auto_compile"
|
|
|
|
timeInterval:20
|
2017-10-09 15:31:01 +00:00
|
|
|
subjectName:compileGroup
|
|
|
|
throttle: Settings?.rateLimit?.autoCompile?[compileGroup] || 25
|
2017-10-03 15:16:21 +00:00
|
|
|
rateLimiter.addCount opts, (err, canCompile)->
|
|
|
|
if err?
|
|
|
|
canCompile = false
|
|
|
|
if !canCompile
|
2017-10-09 15:31:01 +00:00
|
|
|
Metrics.inc "auto-compile-#{compileGroup}-limited"
|
2014-02-12 10:23:40 +00:00
|
|
|
callback err, canCompile
|
|
|
|
|
|
|
|
_ensureRootDocumentIsSet: (project_id, callback = (error) ->) ->
|
2018-02-15 12:18:43 +00:00
|
|
|
ProjectGetter.getProject project_id, rootDoc_id: 1, (error, project) ->
|
2014-02-12 10:23:40 +00:00
|
|
|
return callback(error) if error?
|
|
|
|
if !project?
|
|
|
|
return callback new Error("project not found")
|
|
|
|
|
|
|
|
if project.rootDoc_id?
|
|
|
|
callback()
|
|
|
|
else
|
|
|
|
ProjectRootDocManager.setRootDocAutomatically project_id, callback
|
|
|
|
|
2016-05-31 15:20:24 +00:00
|
|
|
wordCount: (project_id, user_id, file, callback = (error) ->) ->
|
2015-09-10 15:41:48 +00:00
|
|
|
CompileManager.getProjectCompileLimits project_id, (error, limits) ->
|
|
|
|
return callback(error) if error?
|
2016-05-31 15:20:24 +00:00
|
|
|
ClsiManager.wordCount project_id, user_id, file, limits, callback
|