2014-08-15 11:13:35 +00:00
|
|
|
async = require "async"
|
2015-03-04 16:43:59 +00:00
|
|
|
ASpellWorkerPool = require "./ASpellWorkerPool"
|
|
|
|
LRU = require "lru-cache"
|
2015-03-09 15:57:39 +00:00
|
|
|
logger = require 'logger-sharelatex'
|
2016-03-04 11:58:37 +00:00
|
|
|
fs = require 'fs'
|
2016-12-13 09:14:09 +00:00
|
|
|
settings = require("settings-sharelatex")
|
|
|
|
Path = require("path")
|
2014-08-15 11:13:35 +00:00
|
|
|
|
2016-03-04 11:58:37 +00:00
|
|
|
OneMinute = 60 * 1000
|
2019-01-13 21:43:12 +00:00
|
|
|
opts = {max:10000, maxAge: OneMinute * 60 * 10}
|
|
|
|
cache = LRU(opts)
|
2016-03-04 11:58:37 +00:00
|
|
|
|
2016-12-13 09:14:09 +00:00
|
|
|
cacheFsPath = Path.resolve(settings.cacheDir, "spell.cache")
|
|
|
|
cacheFsPathTmp = cacheFsPath + ".tmp"
|
|
|
|
|
2016-03-04 11:58:37 +00:00
|
|
|
# load any existing cache
|
|
|
|
try
|
2016-12-13 09:14:09 +00:00
|
|
|
oldCache = fs.readFileSync cacheFsPath
|
2016-03-04 11:58:37 +00:00
|
|
|
cache.load JSON.parse(oldCache)
|
|
|
|
catch err
|
2016-12-13 09:14:09 +00:00
|
|
|
logger.log err:err, cacheFsPath:cacheFsPath, "could not load the cache file"
|
2016-03-04 11:58:37 +00:00
|
|
|
|
|
|
|
# write the cache every 30 minutes
|
|
|
|
setInterval () ->
|
|
|
|
dump = JSON.stringify cache.dump()
|
2016-12-13 09:14:09 +00:00
|
|
|
fs.writeFile cacheFsPathTmp, dump, (err) ->
|
2016-03-04 11:58:37 +00:00
|
|
|
if err?
|
|
|
|
logger.log {err}, "error writing cache file"
|
2016-12-13 09:14:09 +00:00
|
|
|
fs.unlink cacheFsPathTmp
|
2016-03-04 11:58:37 +00:00
|
|
|
else
|
2016-12-13 09:14:09 +00:00
|
|
|
fs.rename cacheFsPathTmp, cacheFsPath
|
|
|
|
logger.log {len: dump.length, cacheFsPath:cacheFsPath}, "wrote cache file"
|
2016-03-04 11:58:37 +00:00
|
|
|
, 30 * OneMinute
|
2014-08-15 11:13:35 +00:00
|
|
|
|
|
|
|
class ASpellRunner
|
|
|
|
checkWords: (language, words, callback = (error, result) ->) ->
|
|
|
|
@runAspellOnWords language, words, (error, output) =>
|
|
|
|
return callback(error) if error?
|
2015-03-04 16:43:59 +00:00
|
|
|
#output = @removeAspellHeader(output)
|
2015-03-11 14:50:24 +00:00
|
|
|
suggestions = @getSuggestions(language, output)
|
2014-08-15 11:13:35 +00:00
|
|
|
results = []
|
2015-03-09 15:57:39 +00:00
|
|
|
hits = 0
|
2015-03-11 14:50:24 +00:00
|
|
|
addToCache = {}
|
2014-08-15 11:13:35 +00:00
|
|
|
for word, i in words
|
2015-03-04 17:00:19 +00:00
|
|
|
key = language + ':' + word
|
|
|
|
cached = cache.get(key)
|
|
|
|
if cached?
|
2015-03-09 15:57:39 +00:00
|
|
|
hits++
|
|
|
|
if cached == true
|
|
|
|
# valid word, no need to do anything
|
|
|
|
continue
|
|
|
|
else
|
|
|
|
results.push index: i, suggestions: cached
|
|
|
|
else
|
2015-03-11 14:50:24 +00:00
|
|
|
if suggestions[key]?
|
|
|
|
addToCache[key] = suggestions[key]
|
|
|
|
results.push index: i, suggestions: suggestions[key]
|
2015-03-09 15:57:39 +00:00
|
|
|
else
|
|
|
|
# a valid word, but uncached
|
2015-03-11 14:50:24 +00:00
|
|
|
addToCache[key] = true
|
|
|
|
|
|
|
|
# update the cache after processing all words, to avoid cache
|
|
|
|
# changing while we use it
|
2015-03-11 15:57:33 +00:00
|
|
|
for k, v of addToCache
|
2015-03-11 14:50:24 +00:00
|
|
|
cache.set(k, v)
|
|
|
|
|
2019-01-09 15:37:03 +00:00
|
|
|
logger.info hits: hits, total: words.length, hitrate: (hits/words.length).toFixed(2), "cache hit rate"
|
2014-08-15 11:13:35 +00:00
|
|
|
callback null, results
|
|
|
|
|
2015-03-11 14:50:24 +00:00
|
|
|
getSuggestions: (language, output) ->
|
2014-08-15 11:13:35 +00:00
|
|
|
lines = output.split("\n")
|
|
|
|
suggestions = {}
|
|
|
|
for line in lines
|
|
|
|
if line[0] == "&" # Suggestions found
|
|
|
|
parts = line.split(" ")
|
|
|
|
if parts.length > 1
|
|
|
|
word = parts[1]
|
|
|
|
suggestionsString = line.slice(line.indexOf(":") + 2)
|
2015-03-11 14:50:24 +00:00
|
|
|
suggestions[language + ":" + word] = suggestionsString.split(", ")
|
2014-08-15 11:13:35 +00:00
|
|
|
else if line[0] == "#" # No suggestions
|
|
|
|
parts = line.split(" ")
|
|
|
|
if parts.length > 1
|
|
|
|
word = parts[1]
|
2015-03-11 14:50:24 +00:00
|
|
|
suggestions[language + ":" + word] = []
|
2014-08-15 11:13:35 +00:00
|
|
|
return suggestions
|
|
|
|
|
2015-03-04 16:43:59 +00:00
|
|
|
#removeAspellHeader: (output) -> output.slice(1)
|
2014-08-15 11:13:35 +00:00
|
|
|
|
2015-03-04 16:43:59 +00:00
|
|
|
runAspellOnWords: (language, words, callback = (error, output) ->) ->
|
|
|
|
# send words to aspell, get back string output for those words
|
|
|
|
# find a free pipe for the language (or start one)
|
|
|
|
# send the words down the pipe
|
|
|
|
# send an END marker that will generate a "*" line in the output
|
|
|
|
# when the output pipe receives the "*" return the data sofar and reset the pipe to be available
|
|
|
|
#
|
|
|
|
# @open(language)
|
|
|
|
# @captureOutput(callback)
|
|
|
|
# @setTerseMode()
|
|
|
|
# start = new Date()
|
2015-03-02 16:58:10 +00:00
|
|
|
|
2015-03-04 16:43:59 +00:00
|
|
|
newWord = {}
|
|
|
|
for word in words
|
2015-03-04 17:00:19 +00:00
|
|
|
newWord[word] = true if !newWord[word] && !cache.has(language + ':' + word)
|
2015-03-04 16:43:59 +00:00
|
|
|
words = Object.keys(newWord)
|
2015-03-02 16:58:10 +00:00
|
|
|
|
2015-03-04 16:43:59 +00:00
|
|
|
if words.length
|
|
|
|
WorkerPool.check(language, words, ASpell.ASPELL_TIMEOUT, callback)
|
|
|
|
else
|
2015-03-04 17:00:19 +00:00
|
|
|
callback null, ""
|
2014-08-15 11:13:35 +00:00
|
|
|
|
|
|
|
module.exports = ASpell =
|
|
|
|
# The description of how to call aspell from another program can be found here:
|
|
|
|
# http://aspell.net/man-html/Through-A-Pipe.html
|
|
|
|
checkWords: (language, words, callback = (error, result) ->) ->
|
|
|
|
runner = new ASpellRunner()
|
|
|
|
runner.checkWords language, words, callback
|
|
|
|
ASPELL_TIMEOUT : 4000
|
2015-03-04 16:43:59 +00:00
|
|
|
|
|
|
|
WorkerPool = new ASpellWorkerPool()
|