2014-02-12 10:40:42 +00:00
|
|
|
RedisManager = require "./RedisManager"
|
|
|
|
PersistenceManager = require "./PersistenceManager"
|
|
|
|
DocOpsManager = require "./DocOpsManager"
|
|
|
|
DiffCodec = require "./DiffCodec"
|
|
|
|
logger = require "logger-sharelatex"
|
|
|
|
Metrics = require "./Metrics"
|
2016-01-20 15:05:31 +00:00
|
|
|
TrackChangesManager = require "./TrackChangesManager"
|
2014-02-12 10:40:42 +00:00
|
|
|
|
|
|
|
module.exports = DocumentManager =
|
|
|
|
getDoc: (project_id, doc_id, _callback = (error, lines, version) ->) ->
|
|
|
|
timer = new Metrics.Timer("docManager.getDoc")
|
|
|
|
callback = (args...) ->
|
|
|
|
timer.done()
|
|
|
|
_callback(args...)
|
|
|
|
|
2016-01-20 14:31:25 +00:00
|
|
|
RedisManager.getDoc doc_id, (error, lines, version, alreadyLoaded) ->
|
2014-02-12 10:40:42 +00:00
|
|
|
return callback(error) if error?
|
|
|
|
if !lines? or !version?
|
|
|
|
logger.log project_id: project_id, doc_id: doc_id, "doc not in redis so getting from persistence API"
|
2014-02-10 15:17:08 +00:00
|
|
|
PersistenceManager.getDoc project_id, doc_id, (error, lines, version) ->
|
2014-02-12 10:40:42 +00:00
|
|
|
return callback(error) if error?
|
2014-02-10 15:17:08 +00:00
|
|
|
logger.log project_id: project_id, doc_id: doc_id, lines: lines, version: version, "got doc from persistence API"
|
|
|
|
RedisManager.putDocInMemory project_id, doc_id, lines, version, (error) ->
|
2014-02-12 10:40:42 +00:00
|
|
|
return callback(error) if error?
|
2016-01-20 14:31:25 +00:00
|
|
|
callback null, lines, version, false
|
2014-02-12 10:40:42 +00:00
|
|
|
else
|
2016-01-20 14:31:25 +00:00
|
|
|
callback null, lines, version, true
|
2014-02-12 10:40:42 +00:00
|
|
|
|
|
|
|
getDocAndRecentOps: (project_id, doc_id, fromVersion, _callback = (error, lines, version, recentOps) ->) ->
|
|
|
|
timer = new Metrics.Timer("docManager.getDocAndRecentOps")
|
|
|
|
callback = (args...) ->
|
|
|
|
timer.done()
|
|
|
|
_callback(args...)
|
|
|
|
|
|
|
|
DocumentManager.getDoc project_id, doc_id, (error, lines, version) ->
|
|
|
|
return callback(error) if error?
|
|
|
|
if fromVersion == -1
|
|
|
|
callback null, lines, version, []
|
|
|
|
else
|
|
|
|
DocOpsManager.getPreviousDocOps project_id, doc_id, fromVersion, version, (error, ops) ->
|
|
|
|
return callback(error) if error?
|
|
|
|
callback null, lines, version, ops
|
|
|
|
|
2014-03-11 12:47:26 +00:00
|
|
|
setDoc: (project_id, doc_id, newLines, source, user_id, _callback = (error) ->) ->
|
2014-02-12 10:40:42 +00:00
|
|
|
timer = new Metrics.Timer("docManager.setDoc")
|
|
|
|
callback = (args...) ->
|
|
|
|
timer.done()
|
|
|
|
_callback(args...)
|
|
|
|
|
|
|
|
if !newLines?
|
|
|
|
return callback(new Error("No lines were provided to setDoc"))
|
|
|
|
|
|
|
|
UpdateManager = require "./UpdateManager"
|
2016-01-20 14:31:25 +00:00
|
|
|
DocumentManager.getDoc project_id, doc_id, (error, oldLines, version, alreadyLoaded) ->
|
2014-02-12 10:40:42 +00:00
|
|
|
return callback(error) if error?
|
|
|
|
|
|
|
|
if oldLines? and oldLines.length > 0 and oldLines[0].text?
|
|
|
|
logger.log doc_id: doc_id, project_id: project_id, oldLines: oldLines, newLines: newLines, "document is JSON so not updating"
|
|
|
|
return callback(null)
|
|
|
|
|
|
|
|
logger.log doc_id: doc_id, project_id: project_id, oldLines: oldLines, newLines: newLines, "setting a document via http"
|
|
|
|
DiffCodec.diffAsShareJsOp oldLines, newLines, (error, op) ->
|
|
|
|
return callback(error) if error?
|
|
|
|
update =
|
|
|
|
doc: doc_id
|
|
|
|
op: op
|
|
|
|
v: version
|
|
|
|
meta:
|
|
|
|
type: "external"
|
2014-03-11 12:47:26 +00:00
|
|
|
source: source
|
|
|
|
user_id: user_id
|
2014-02-12 10:40:42 +00:00
|
|
|
UpdateManager.applyUpdates project_id, doc_id, [update], (error) ->
|
|
|
|
return callback(error) if error?
|
2016-01-20 14:31:25 +00:00
|
|
|
# If the document was loaded already, then someone has it open
|
|
|
|
# in a project, and the usual flushing mechanism will happen.
|
|
|
|
# Otherwise we should remove it immediately since nothing else
|
|
|
|
# is using it.
|
|
|
|
if alreadyLoaded
|
|
|
|
DocumentManager.flushDocIfLoaded project_id, doc_id, (error) ->
|
|
|
|
return callback(error) if error?
|
|
|
|
callback null
|
|
|
|
else
|
|
|
|
DocumentManager.flushAndDeleteDoc project_id, doc_id, (error) ->
|
|
|
|
return callback(error) if error?
|
|
|
|
callback null
|
2014-02-12 10:40:42 +00:00
|
|
|
|
|
|
|
|
|
|
|
flushDocIfLoaded: (project_id, doc_id, _callback = (error) ->) ->
|
|
|
|
timer = new Metrics.Timer("docManager.flushDocIfLoaded")
|
|
|
|
callback = (args...) ->
|
|
|
|
timer.done()
|
|
|
|
_callback(args...)
|
|
|
|
|
|
|
|
RedisManager.getDoc doc_id, (error, lines, version) ->
|
|
|
|
return callback(error) if error?
|
|
|
|
if !lines? or !version?
|
|
|
|
logger.log project_id: project_id, doc_id: doc_id, "doc is not loaded so not flushing"
|
|
|
|
callback null
|
|
|
|
else
|
2014-02-10 15:17:08 +00:00
|
|
|
logger.log project_id: project_id, doc_id: doc_id, version: version, "flushing doc"
|
|
|
|
PersistenceManager.setDoc project_id, doc_id, lines, version, (error) ->
|
2014-02-12 10:40:42 +00:00
|
|
|
return callback(error) if error?
|
2015-12-01 15:58:40 +00:00
|
|
|
callback null
|
2014-02-12 10:40:42 +00:00
|
|
|
|
|
|
|
flushAndDeleteDoc: (project_id, doc_id, _callback = (error) ->) ->
|
|
|
|
timer = new Metrics.Timer("docManager.flushAndDeleteDoc")
|
|
|
|
callback = (args...) ->
|
|
|
|
timer.done()
|
|
|
|
_callback(args...)
|
|
|
|
|
|
|
|
DocumentManager.flushDocIfLoaded project_id, doc_id, (error) ->
|
|
|
|
return callback(error) if error?
|
2016-01-20 15:05:31 +00:00
|
|
|
|
|
|
|
# Flush in the background since it requires and http request
|
|
|
|
# to track changes
|
|
|
|
TrackChangesManager.flushDocChanges project_id, doc_id, (err) ->
|
|
|
|
if err?
|
|
|
|
logger.err {err, project_id, doc_id}, "error flushing to track changes"
|
|
|
|
|
2014-02-12 10:40:42 +00:00
|
|
|
RedisManager.removeDocFromMemory project_id, doc_id, (error) ->
|
|
|
|
return callback(error) if error?
|
|
|
|
callback null
|
|
|
|
|
|
|
|
getDocWithLock: (project_id, doc_id, callback = (error, lines, version) ->) ->
|
|
|
|
UpdateManager = require "./UpdateManager"
|
|
|
|
UpdateManager.lockUpdatesAndDo DocumentManager.getDoc, project_id, doc_id, callback
|
|
|
|
|
|
|
|
getDocAndRecentOpsWithLock: (project_id, doc_id, fromVersion, callback = (error, lines, version) ->) ->
|
|
|
|
UpdateManager = require "./UpdateManager"
|
|
|
|
UpdateManager.lockUpdatesAndDo DocumentManager.getDocAndRecentOps, project_id, doc_id, fromVersion, callback
|
|
|
|
|
2014-03-11 12:47:26 +00:00
|
|
|
setDocWithLock: (project_id, doc_id, lines, source, user_id, callback = (error) ->) ->
|
2014-02-12 10:40:42 +00:00
|
|
|
UpdateManager = require "./UpdateManager"
|
2014-03-11 12:47:26 +00:00
|
|
|
UpdateManager.lockUpdatesAndDo DocumentManager.setDoc, project_id, doc_id, lines, source, user_id, callback
|
2014-02-12 10:40:42 +00:00
|
|
|
|
|
|
|
flushDocIfLoadedWithLock: (project_id, doc_id, callback = (error) ->) ->
|
|
|
|
UpdateManager = require "./UpdateManager"
|
|
|
|
UpdateManager.lockUpdatesAndDo DocumentManager.flushDocIfLoaded, project_id, doc_id, callback
|
|
|
|
|
|
|
|
flushAndDeleteDocWithLock: (project_id, doc_id, callback = (error) ->) ->
|
|
|
|
UpdateManager = require "./UpdateManager"
|
|
|
|
UpdateManager.lockUpdatesAndDo DocumentManager.flushAndDeleteDoc, project_id, doc_id, callback
|