2015-03-04 16:43:59 +00:00
|
|
|
ASpellWorker = require "./ASpellWorker"
|
|
|
|
_ = require "underscore"
|
2015-03-10 13:33:05 +00:00
|
|
|
logger = require 'logger-sharelatex'
|
2015-03-12 09:54:57 +00:00
|
|
|
metrics = require('metrics-sharelatex')
|
2015-03-04 16:43:59 +00:00
|
|
|
|
|
|
|
class ASpellWorkerPool
|
2015-03-10 13:33:05 +00:00
|
|
|
MAX_REQUESTS: 100*1024
|
|
|
|
MAX_WORKERS: 32
|
2015-03-12 11:13:58 +00:00
|
|
|
MAX_IDLE_TIME: 1000
|
|
|
|
MAX_REQUEST_TIME: 60*1000
|
2015-03-10 13:33:05 +00:00
|
|
|
|
2015-03-04 16:43:59 +00:00
|
|
|
constructor: (@options) ->
|
|
|
|
@PROCESS_POOL = []
|
|
|
|
|
|
|
|
create: (language) ->
|
2015-03-10 13:33:05 +00:00
|
|
|
if @PROCESS_POOL.length >= @MAX_WORKERS
|
|
|
|
logger.log maxworkers: @MAX_WORKERS, "maximum number of workers already running"
|
|
|
|
return null
|
2015-03-04 16:43:59 +00:00
|
|
|
worker = new ASpellWorker(language, @options)
|
|
|
|
worker.pipe.on 'exit', () =>
|
2015-03-12 11:13:58 +00:00
|
|
|
if worker.killTimer?
|
|
|
|
clearTimeout worker.killTimer
|
|
|
|
worker.killTimer = null
|
|
|
|
if worker.idleTimer?
|
|
|
|
clearTimeout worker.idleTimer
|
|
|
|
worker.idleTimer = null
|
2019-01-09 15:37:03 +00:00
|
|
|
logger.info process: worker.pipe.pid, lang: language, "removing aspell worker from pool"
|
2015-03-11 14:53:06 +00:00
|
|
|
@cleanup()
|
2015-03-04 16:43:59 +00:00
|
|
|
@PROCESS_POOL.push(worker)
|
2015-03-12 09:54:57 +00:00
|
|
|
metrics.gauge 'aspellWorkerPool-size', @PROCESS_POOL.length
|
2015-03-04 16:43:59 +00:00
|
|
|
return worker
|
|
|
|
|
|
|
|
cleanup: () ->
|
|
|
|
active = @PROCESS_POOL.filter (worker) ->
|
|
|
|
worker.state != 'killed'
|
|
|
|
@PROCESS_POOL = active
|
2015-03-12 09:54:57 +00:00
|
|
|
metrics.gauge 'aspellWorkerPool-size', @PROCESS_POOL.length
|
2015-03-04 16:43:59 +00:00
|
|
|
|
|
|
|
check: (language, words, timeout, callback) ->
|
|
|
|
# look for an existing process in the pool
|
|
|
|
availableWorker = _.find @PROCESS_POOL, (cached) ->
|
|
|
|
cached.language == language && cached.isReady()
|
|
|
|
if not availableWorker?
|
|
|
|
worker = @create(language)
|
|
|
|
else
|
|
|
|
worker = availableWorker
|
|
|
|
|
2015-03-10 13:33:05 +00:00
|
|
|
if not worker?
|
|
|
|
# return error if too many workers
|
|
|
|
callback(new Error("no worker available"))
|
|
|
|
return
|
|
|
|
|
|
|
|
if worker.idleTimer?
|
|
|
|
clearTimeout worker.idleTimer
|
|
|
|
worker.idleTimer = null
|
|
|
|
|
2015-03-12 11:13:58 +00:00
|
|
|
worker.killTimer = setTimeout () ->
|
2015-03-12 11:24:04 +00:00
|
|
|
worker.kill("spell check timed out")
|
2015-03-12 11:13:58 +00:00
|
|
|
, timeout || @MAX_REQUEST_TIME
|
2015-03-10 13:33:05 +00:00
|
|
|
|
|
|
|
worker.check words, (err, output) =>
|
2015-03-12 11:13:58 +00:00
|
|
|
if worker.killTimer?
|
|
|
|
clearTimeout worker.killTimer
|
|
|
|
worker.killTimer = null
|
2015-03-04 16:43:59 +00:00
|
|
|
callback(err, output)
|
2015-03-12 10:27:11 +00:00
|
|
|
return if err? # process has shut down
|
2015-03-10 13:33:05 +00:00
|
|
|
if worker.count > @MAX_REQUESTS
|
|
|
|
worker.shutdown("reached limit of " + @MAX_REQUESTS + " requests")
|
|
|
|
else
|
2015-03-12 10:27:11 +00:00
|
|
|
# queue a shutdown if worker is idle
|
2015-03-10 13:33:05 +00:00
|
|
|
worker.idleTimer = setTimeout () ->
|
|
|
|
worker.shutdown("idle worker")
|
|
|
|
worker.idleTimer = null
|
2015-03-12 11:13:58 +00:00
|
|
|
, @MAX_IDLE_TIME
|
2015-03-04 16:43:59 +00:00
|
|
|
|
|
|
|
module.exports = ASpellWorkerPool
|