overleaf/services/track-changes/test/unit/js/UpdatesManager/UpdatesManagerTests.js
2021-07-13 12:04:43 +01:00

1333 lines
40 KiB
JavaScript

/* eslint-disable
camelcase,
handle-callback-err,
no-return-assign,
no-unused-vars,
*/
// TODO: This file was created by bulk-decaffeinate.
// Fix any style issues and re-enable lint.
/*
* decaffeinate suggestions:
* DS101: Remove unnecessary use of Array.from
* DS102: Remove unnecessary code created because of implicit returns
* DS207: Consider shorter variations of null checks
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/master/docs/suggestions.md
*/
const sinon = require('sinon')
const { expect } = require('chai')
const { ObjectId } = require('mongodb')
const modulePath = '../../../../app/js/UpdatesManager.js'
const SandboxedModule = require('sandboxed-module')
describe('UpdatesManager', function () {
beforeEach(function () {
this.UpdatesManager = SandboxedModule.require(modulePath, {
singleOnly: true,
requires: {
'./UpdateCompressor': (this.UpdateCompressor = {}),
'./MongoManager': (this.MongoManager = {}),
'./PackManager': (this.PackManager = {}),
'./RedisManager': (this.RedisManager = {}),
'./LockManager': (this.LockManager = {}),
'./WebApiManager': (this.WebApiManager = {}),
'./UpdateTrimmer': (this.UpdateTrimmer = {}),
'./DocArchiveManager': (this.DocArchiveManager = {}),
'@overleaf/settings': {
redis: {
lock: {
key_schema: {
historyLock({ doc_id }) {
return `HistoryLock:${doc_id}`
},
},
},
},
},
},
})
this.doc_id = 'doc-id-123'
this.project_id = 'project-id-123'
this.callback = sinon.stub()
return (this.temporary = 'temp-mock')
})
describe('compressAndSaveRawUpdates', function () {
describe('when there are no raw ops', function () {
beforeEach(function () {
this.MongoManager.peekLastCompressedUpdate = sinon.stub()
return this.UpdatesManager.compressAndSaveRawUpdates(
this.project_id,
this.doc_id,
[],
this.temporary,
this.callback
)
})
it('should not need to access the database', function () {
return this.MongoManager.peekLastCompressedUpdate.called.should.equal(
false
)
})
return it('should call the callback', function () {
return this.callback.called.should.equal(true)
})
})
describe('when there is no compressed history to begin with', function () {
beforeEach(function () {
this.rawUpdates = [
{ v: 12, op: 'mock-op-12' },
{ v: 13, op: 'mock-op-13' },
]
this.compressedUpdates = [{ v: 13, op: 'compressed-op-12' }]
this.MongoManager.peekLastCompressedUpdate = sinon
.stub()
.callsArgWith(1, null, null)
this.PackManager.insertCompressedUpdates = sinon.stub().callsArg(5)
this.UpdateCompressor.compressRawUpdates = sinon
.stub()
.returns(this.compressedUpdates)
return this.UpdatesManager.compressAndSaveRawUpdates(
this.project_id,
this.doc_id,
this.rawUpdates,
this.temporary,
this.callback
)
})
it('should look at the last compressed op', function () {
return this.MongoManager.peekLastCompressedUpdate
.calledWith(this.doc_id)
.should.equal(true)
})
it('should save the compressed ops as a pack', function () {
return this.PackManager.insertCompressedUpdates
.calledWith(
this.project_id,
this.doc_id,
null,
this.compressedUpdates,
this.temporary
)
.should.equal(true)
})
return it('should call the callback', function () {
return this.callback.called.should.equal(true)
})
})
describe('when the raw ops need appending to existing history', function () {
beforeEach(function () {
this.lastCompressedUpdate = { v: 11, op: 'compressed-op-11' }
this.compressedUpdates = [
{ v: 12, op: 'compressed-op-11+12' },
{ v: 13, op: 'compressed-op-12' },
]
this.MongoManager.peekLastCompressedUpdate = sinon
.stub()
.callsArgWith(
1,
null,
this.lastCompressedUpdate,
this.lastCompressedUpdate.v
)
this.PackManager.insertCompressedUpdates = sinon.stub().callsArg(5)
return (this.UpdateCompressor.compressRawUpdates = sinon
.stub()
.returns(this.compressedUpdates))
})
describe('when the raw ops start where the existing history ends', function () {
beforeEach(function () {
this.rawUpdates = [
{ v: 12, op: 'mock-op-12' },
{ v: 13, op: 'mock-op-13' },
]
return this.UpdatesManager.compressAndSaveRawUpdates(
this.project_id,
this.doc_id,
this.rawUpdates,
this.temporary,
this.callback
)
})
it('should look at the last compressed op', function () {
return this.MongoManager.peekLastCompressedUpdate
.calledWith(this.doc_id)
.should.equal(true)
})
it('should compress the raw ops', function () {
return this.UpdateCompressor.compressRawUpdates
.calledWith(null, this.rawUpdates)
.should.equal(true)
})
it('should save the new compressed ops into a pack', function () {
return this.PackManager.insertCompressedUpdates
.calledWith(
this.project_id,
this.doc_id,
this.lastCompressedUpdate,
this.compressedUpdates,
this.temporary
)
.should.equal(true)
})
return it('should call the callback', function () {
return this.callback.called.should.equal(true)
})
})
describe('when the raw ops start where the existing history ends and the history is in a pack', function () {
beforeEach(function () {
this.lastCompressedUpdate = {
pack: [{ v: 11, op: 'compressed-op-11' }],
v: 11,
}
this.rawUpdates = [
{ v: 12, op: 'mock-op-12' },
{ v: 13, op: 'mock-op-13' },
]
this.MongoManager.peekLastCompressedUpdate = sinon
.stub()
.callsArgWith(
1,
null,
this.lastCompressedUpdate,
this.lastCompressedUpdate.v
)
return this.UpdatesManager.compressAndSaveRawUpdates(
this.project_id,
this.doc_id,
this.rawUpdates,
this.temporary,
this.callback
)
})
it('should look at the last compressed op', function () {
return this.MongoManager.peekLastCompressedUpdate
.calledWith(this.doc_id)
.should.equal(true)
})
it('should compress the raw ops', function () {
return this.UpdateCompressor.compressRawUpdates
.calledWith(null, this.rawUpdates)
.should.equal(true)
})
it('should save the new compressed ops into a pack', function () {
return this.PackManager.insertCompressedUpdates
.calledWith(
this.project_id,
this.doc_id,
this.lastCompressedUpdate,
this.compressedUpdates,
this.temporary
)
.should.equal(true)
})
return it('should call the callback', function () {
return this.callback.called.should.equal(true)
})
})
describe('when some raw ops are passed that have already been compressed', function () {
beforeEach(function () {
this.rawUpdates = [
{ v: 10, op: 'mock-op-10' },
{ v: 11, op: 'mock-op-11' },
{ v: 12, op: 'mock-op-12' },
{ v: 13, op: 'mock-op-13' },
]
return this.UpdatesManager.compressAndSaveRawUpdates(
this.project_id,
this.doc_id,
this.rawUpdates,
this.temporary,
this.callback
)
})
return it('should only compress the more recent raw ops', function () {
return this.UpdateCompressor.compressRawUpdates
.calledWith(null, this.rawUpdates.slice(-2))
.should.equal(true)
})
})
describe('when the raw ops do not follow from the last compressed op version', function () {
beforeEach(function () {
this.rawUpdates = [{ v: 13, op: 'mock-op-13' }]
return this.UpdatesManager.compressAndSaveRawUpdates(
this.project_id,
this.doc_id,
this.rawUpdates,
this.temporary,
this.callback
)
})
it('should call the callback with an error', function () {
return this.callback
.calledWith(
sinon.match.has(
'message',
'Tried to apply raw op at version 13 to last compressed update with version 11 from unknown time'
)
)
.should.equal(true)
})
return it('should not insert any update into mongo', function () {
return this.PackManager.insertCompressedUpdates.called.should.equal(
false
)
})
})
return describe('when the raw ops are out of order', function () {
beforeEach(function () {
this.rawUpdates = [
{ v: 13, op: 'mock-op-13' },
{ v: 12, op: 'mock-op-12' },
]
return this.UpdatesManager.compressAndSaveRawUpdates(
this.project_id,
this.doc_id,
this.rawUpdates,
this.temporary,
this.callback
)
})
it('should call the callback with an error', function () {
return this.callback
.calledWith(sinon.match.has('message'))
.should.equal(true)
})
return it('should not insert any update into mongo', function () {
return this.PackManager.insertCompressedUpdates.called.should.equal(
false
)
})
})
})
return describe('when the raw ops need appending to existing history which is in S3', function () {
beforeEach(function () {
this.lastCompressedUpdate = null
this.lastVersion = 11
this.compressedUpdates = [{ v: 13, op: 'compressed-op-12' }]
this.MongoManager.peekLastCompressedUpdate = sinon
.stub()
.callsArgWith(1, null, null, this.lastVersion)
this.PackManager.insertCompressedUpdates = sinon.stub().callsArg(5)
return (this.UpdateCompressor.compressRawUpdates = sinon
.stub()
.returns(this.compressedUpdates))
})
return describe('when the raw ops start where the existing history ends', function () {
beforeEach(function () {
this.rawUpdates = [
{ v: 12, op: 'mock-op-12' },
{ v: 13, op: 'mock-op-13' },
]
return this.UpdatesManager.compressAndSaveRawUpdates(
this.project_id,
this.doc_id,
this.rawUpdates,
this.temporary,
this.callback
)
})
it('should try to look at the last compressed op', function () {
return this.MongoManager.peekLastCompressedUpdate
.calledWith(this.doc_id)
.should.equal(true)
})
it('should compress the last compressed op and the raw ops', function () {
return this.UpdateCompressor.compressRawUpdates
.calledWith(this.lastCompressedUpdate, this.rawUpdates)
.should.equal(true)
})
it('should save the compressed ops', function () {
return this.PackManager.insertCompressedUpdates
.calledWith(
this.project_id,
this.doc_id,
null,
this.compressedUpdates,
this.temporary
)
.should.equal(true)
})
return it('should call the callback', function () {
return this.callback.called.should.equal(true)
})
})
})
})
describe('processUncompressedUpdates', function () {
beforeEach(function () {
this.UpdatesManager.compressAndSaveRawUpdates = sinon
.stub()
.callsArgWith(4)
this.RedisManager.deleteAppliedDocUpdates = sinon.stub().callsArg(3)
this.MongoManager.backportProjectId = sinon.stub().callsArg(2)
return (this.UpdateTrimmer.shouldTrimUpdates = sinon
.stub()
.callsArgWith(1, null, (this.temporary = 'temp mock')))
})
describe('when there is fewer than one batch to send', function () {
beforeEach(function () {
this.updates = ['mock-update']
this.RedisManager.getOldestDocUpdates = sinon
.stub()
.callsArgWith(2, null, this.updates)
this.RedisManager.expandDocUpdates = sinon
.stub()
.callsArgWith(1, null, this.updates)
return this.UpdatesManager.processUncompressedUpdates(
this.project_id,
this.doc_id,
this.temporary,
this.callback
)
})
it('should get the oldest updates', function () {
return this.RedisManager.getOldestDocUpdates
.calledWith(this.doc_id, this.UpdatesManager.REDIS_READ_BATCH_SIZE)
.should.equal(true)
})
it('should compress and save the updates', function () {
return this.UpdatesManager.compressAndSaveRawUpdates
.calledWith(
this.project_id,
this.doc_id,
this.updates,
this.temporary
)
.should.equal(true)
})
it('should delete the batch of uncompressed updates that was just processed', function () {
return this.RedisManager.deleteAppliedDocUpdates
.calledWith(this.project_id, this.doc_id, this.updates)
.should.equal(true)
})
return it('should call the callback', function () {
return this.callback.called.should.equal(true)
})
})
return describe('when there are multiple batches to send', function () {
beforeEach(function (done) {
this.UpdatesManager.REDIS_READ_BATCH_SIZE = 2
this.updates = [
'mock-update-0',
'mock-update-1',
'mock-update-2',
'mock-update-3',
'mock-update-4',
]
this.redisArray = this.updates.slice()
this.RedisManager.getOldestDocUpdates = (
doc_id,
batchSize,
callback
) => {
if (callback == null) {
callback = function (error, updates) {}
}
const updates = this.redisArray.slice(0, batchSize)
this.redisArray = this.redisArray.slice(batchSize)
return callback(null, updates)
}
sinon.spy(this.RedisManager, 'getOldestDocUpdates')
this.RedisManager.expandDocUpdates = (jsonUpdates, callback) => {
return callback(null, jsonUpdates)
}
sinon.spy(this.RedisManager, 'expandDocUpdates')
return this.UpdatesManager.processUncompressedUpdates(
this.project_id,
this.doc_id,
this.temporary,
(...args) => {
this.callback(...Array.from(args || []))
return done()
}
)
})
it('should get the oldest updates in three batches ', function () {
return this.RedisManager.getOldestDocUpdates.callCount.should.equal(3)
})
it('should compress and save the updates in batches', function () {
this.UpdatesManager.compressAndSaveRawUpdates
.calledWith(
this.project_id,
this.doc_id,
this.updates.slice(0, 2),
this.temporary
)
.should.equal(true)
this.UpdatesManager.compressAndSaveRawUpdates
.calledWith(
this.project_id,
this.doc_id,
this.updates.slice(2, 4),
this.temporary
)
.should.equal(true)
return this.UpdatesManager.compressAndSaveRawUpdates
.calledWith(
this.project_id,
this.doc_id,
this.updates.slice(4, 5),
this.temporary
)
.should.equal(true)
})
it('should delete the batches of uncompressed updates', function () {
return this.RedisManager.deleteAppliedDocUpdates.callCount.should.equal(
3
)
})
return it('should call the callback', function () {
return this.callback.called.should.equal(true)
})
})
})
describe('processCompressedUpdatesWithLock', function () {
beforeEach(function () {
this.UpdateTrimmer.shouldTrimUpdates = sinon
.stub()
.callsArgWith(1, null, (this.temporary = 'temp mock'))
this.MongoManager.backportProjectId = sinon.stub().callsArg(2)
this.UpdatesManager._processUncompressedUpdates = sinon.stub().callsArg(3)
this.LockManager.runWithLock = sinon.stub().callsArg(2)
return this.UpdatesManager.processUncompressedUpdatesWithLock(
this.project_id,
this.doc_id,
this.callback
)
})
it('should check if the updates are temporary', function () {
return this.UpdateTrimmer.shouldTrimUpdates
.calledWith(this.project_id)
.should.equal(true)
})
it('should backport the project id', function () {
return this.MongoManager.backportProjectId
.calledWith(this.project_id, this.doc_id)
.should.equal(true)
})
it('should run processUncompressedUpdates with the lock', function () {
return this.LockManager.runWithLock
.calledWith(`HistoryLock:${this.doc_id}`)
.should.equal(true)
})
return it('should call the callback', function () {
return this.callback.called.should.equal(true)
})
})
describe('getDocUpdates', function () {
beforeEach(function () {
this.updates = ['mock-updates']
this.options = { to: 'mock-to', limit: 'mock-limit' }
this.PackManager.getOpsByVersionRange = sinon
.stub()
.callsArgWith(4, null, this.updates)
this.UpdatesManager.processUncompressedUpdatesWithLock = sinon
.stub()
.callsArg(2)
return this.UpdatesManager.getDocUpdates(
this.project_id,
this.doc_id,
this.options,
this.callback
)
})
it('should process outstanding updates', function () {
return this.UpdatesManager.processUncompressedUpdatesWithLock
.calledWith(this.project_id, this.doc_id)
.should.equal(true)
})
it('should get the updates from the database', function () {
return this.PackManager.getOpsByVersionRange
.calledWith(
this.project_id,
this.doc_id,
this.options.from,
this.options.to
)
.should.equal(true)
})
return it('should return the updates', function () {
return this.callback.calledWith(null, this.updates).should.equal(true)
})
})
describe('getDocUpdatesWithUserInfo', function () {
beforeEach(function () {
this.updates = ['mock-updates']
this.options = { to: 'mock-to', limit: 'mock-limit' }
this.updatesWithUserInfo = ['updates-with-user-info']
this.UpdatesManager.getDocUpdates = sinon
.stub()
.callsArgWith(3, null, this.updates)
this.UpdatesManager.fillUserInfo = sinon
.stub()
.callsArgWith(1, null, this.updatesWithUserInfo)
return this.UpdatesManager.getDocUpdatesWithUserInfo(
this.project_id,
this.doc_id,
this.options,
this.callback
)
})
it('should get the updates', function () {
return this.UpdatesManager.getDocUpdates
.calledWith(this.project_id, this.doc_id, this.options)
.should.equal(true)
})
it('should file the updates with the user info', function () {
return this.UpdatesManager.fillUserInfo
.calledWith(this.updates)
.should.equal(true)
})
return it('should return the updates with the filled details', function () {
return this.callback
.calledWith(null, this.updatesWithUserInfo)
.should.equal(true)
})
})
describe('processUncompressedUpdatesForProject', function () {
beforeEach(function (done) {
this.doc_ids = ['mock-id-1', 'mock-id-2']
this.UpdateTrimmer.shouldTrimUpdates = sinon
.stub()
.callsArgWith(1, null, (this.temporary = 'temp mock'))
this.MongoManager.backportProjectId = sinon.stub().callsArg(2)
this.UpdatesManager._processUncompressedUpdatesForDocWithLock = sinon
.stub()
.callsArg(3)
this.RedisManager.getDocIdsWithHistoryOps = sinon
.stub()
.callsArgWith(1, null, this.doc_ids)
return this.UpdatesManager.processUncompressedUpdatesForProject(
this.project_id,
() => {
this.callback()
return done()
}
)
})
it('should get all the docs with history ops', function () {
return this.RedisManager.getDocIdsWithHistoryOps
.calledWith(this.project_id)
.should.equal(true)
})
it('should process the doc ops for the each doc_id', function () {
return Array.from(this.doc_ids).map(doc_id =>
this.UpdatesManager._processUncompressedUpdatesForDocWithLock
.calledWith(this.project_id, doc_id, this.temporary)
.should.equal(true)
)
})
return it('should call the callback', function () {
return this.callback.called.should.equal(true)
})
})
describe('getSummarizedProjectUpdates', function () {
beforeEach(function () {
this.updates = [
{
doc_id: 123,
v: 456,
op: 'mock-updates',
meta: { user_id: 123, start_ts: 1233, end_ts: 1234 },
},
]
this.options = { before: 'mock-before', limit: 'mock-limit' }
this.summarizedUpdates = [
{
meta: { user_ids: [123], start_ts: 1233, end_ts: 1234 },
docs: { 123: { fromV: 456, toV: 456 } },
},
]
this.updatesWithUserInfo = ['updates-with-user-info']
this.done_state = false
this.iterator = {
next: cb => {
this.done_state = true
return cb(null, this.updates)
},
done: () => {
return this.done_state
},
}
this.PackManager.makeProjectIterator = sinon
.stub()
.callsArgWith(2, null, this.iterator)
this.UpdatesManager.processUncompressedUpdatesForProject = sinon
.stub()
.callsArg(1)
this.UpdatesManager.fillSummarizedUserInfo = sinon
.stub()
.callsArgWith(1, null, this.updatesWithUserInfo)
return this.UpdatesManager.getSummarizedProjectUpdates(
this.project_id,
this.options,
this.callback
)
})
it('should process any outstanding updates', function () {
return this.UpdatesManager.processUncompressedUpdatesForProject
.calledWith(this.project_id)
.should.equal(true)
})
it('should get the updates', function () {
return this.PackManager.makeProjectIterator
.calledWith(this.project_id, this.options.before)
.should.equal(true)
})
it('should fill the updates with the user info', function () {
return this.UpdatesManager.fillSummarizedUserInfo
.calledWith(this.summarizedUpdates)
.should.equal(true)
})
return it('should return the updates with the filled details', function () {
return this.callback
.calledWith(null, this.updatesWithUserInfo)
.should.equal(true)
})
})
// describe "_extendBatchOfSummarizedUpdates", ->
// beforeEach ->
// @before = Date.now()
// @min_count = 2
// @existingSummarizedUpdates = ["summarized-updates-3"]
// @summarizedUpdates = ["summarized-updates-3", "summarized-update-2", "summarized-update-1"]
// describe "when there are updates to get", ->
// beforeEach ->
// @updates = [
// {op: "mock-op-1", meta: end_ts: @before - 10},
// {op: "mock-op-1", meta: end_ts: @nextBeforeTimestamp = @before - 20}
// ]
// @existingSummarizedUpdates = ["summarized-updates-3"]
// @summarizedUpdates = ["summarized-updates-3", "summarized-update-2", "summarized-update-1"]
// @UpdatesManager._summarizeUpdates = sinon.stub().returns(@summarizedUpdates)
// @UpdatesManager.getProjectUpdatesWithUserInfo = sinon.stub().callsArgWith(2, null, @updates)
// @UpdatesManager._extendBatchOfSummarizedUpdates @project_id, @existingSummarizedUpdates, @before, @min_count, @callback
// it "should get the updates", ->
// @UpdatesManager.getProjectUpdatesWithUserInfo
// .calledWith(@project_id, { before: @before, limit: 3 * @min_count })
// .should.equal true
// it "should summarize the updates", ->
// @UpdatesManager._summarizeUpdates
// .calledWith(@updates, @existingSummarizedUpdates)
// .should.equal true
// it "should call the callback with the summarized updates and the next before timestamp", ->
// @callback.calledWith(null, @summarizedUpdates, @nextBeforeTimestamp).should.equal true
// describe "when there are no more updates", ->
// beforeEach ->
// @updates = []
// @UpdatesManager._summarizeUpdates = sinon.stub().returns(@summarizedUpdates)
// @UpdatesManager.getProjectUpdatesWithUserInfo = sinon.stub().callsArgWith(2, null, @updates)
// @UpdatesManager._extendBatchOfSummarizedUpdates @project_id, @existingSummarizedUpdates, @before, @min_count, @callback
// it "should call the callback with the summarized updates and null for nextBeforeTimestamp", ->
// @callback.calledWith(null, @summarizedUpdates, null).should.equal true
// describe "getSummarizedProjectUpdates", ->
// describe "when one batch of updates is enough to meet the limit", ->
// beforeEach ->
// @before = Date.now()
// @min_count = 2
// @updates = ["summarized-updates-3", "summarized-updates-2"]
// @nextBeforeTimestamp = @before - 100
// @UpdatesManager._extendBatchOfSummarizedUpdates = sinon.stub().callsArgWith(4, null, @updates, @nextBeforeTimestamp)
// @UpdatesManager.getSummarizedProjectUpdates @project_id, { before: @before, min_count: @min_count }, @callback
// it "should get the batch of summarized updates", ->
// @UpdatesManager._extendBatchOfSummarizedUpdates
// .calledWith(@project_id, [], @before, @min_count)
// .should.equal true
// it "should call the callback with the updates", ->
// @callback.calledWith(null, @updates, @nextBeforeTimestamp).should.equal true
// describe "when multiple batches are needed to meet the limit", ->
// beforeEach ->
// @before = Date.now()
// @min_count = 4
// @firstBatch = [{ toV: 6, fromV: 6 }, { toV: 5, fromV: 5 }]
// @nextBeforeTimestamp = @before - 100
// @secondBatch = [{ toV: 4, fromV: 4 }, { toV: 3, fromV: 3 }]
// @nextNextBeforeTimestamp = @before - 200
// @UpdatesManager._extendBatchOfSummarizedUpdates = (project_id, existingUpdates, before, desiredLength, callback) =>
// if existingUpdates.length == 0
// callback null, @firstBatch, @nextBeforeTimestamp
// else
// callback null, @firstBatch.concat(@secondBatch), @nextNextBeforeTimestamp
// sinon.spy @UpdatesManager, "_extendBatchOfSummarizedUpdates"
// @UpdatesManager.getSummarizedProjectUpdates @project_id, { before: @before, min_count: @min_count }, @callback
// it "should get the first batch of summarized updates", ->
// @UpdatesManager._extendBatchOfSummarizedUpdates
// .calledWith(@project_id, [], @before, @min_count)
// .should.equal true
// it "should get the second batch of summarized updates", ->
// @UpdatesManager._extendBatchOfSummarizedUpdates
// .calledWith(@project_id, @firstBatch, @nextBeforeTimestamp, @min_count)
// .should.equal true
// it "should call the callback with all the updates", ->
// @callback.calledWith(null, @firstBatch.concat(@secondBatch), @nextNextBeforeTimestamp).should.equal true
// describe "when the end of the database is hit", ->
// beforeEach ->
// @before = Date.now()
// @min_count = 4
// @updates = [{ toV: 6, fromV: 6 }, { toV: 5, fromV: 5 }]
// @UpdatesManager._extendBatchOfSummarizedUpdates = sinon.stub().callsArgWith(4, null, @updates, null)
// @UpdatesManager.getSummarizedProjectUpdates @project_id, { before: @before, min_count: @min_count }, @callback
// it "should get the batch of summarized updates", ->
// @UpdatesManager._extendBatchOfSummarizedUpdates
// .calledWith(@project_id, [], @before, @min_count)
// .should.equal true
// it "should call the callback with the updates", ->
// @callback.calledWith(null, @updates, null).should.equal true
describe('fillUserInfo', function () {
describe('with valid users', function () {
beforeEach(function (done) {
this.user_id_1 = ObjectId().toString()
this.user_id_2 = ObjectId().toString()
this.updates = [
{
meta: {
user_id: this.user_id_1,
},
op: 'mock-op-1',
},
{
meta: {
user_id: this.user_id_1,
},
op: 'mock-op-2',
},
{
meta: {
user_id: this.user_id_2,
},
op: 'mock-op-3',
},
]
this.user_info = {}
this.user_info[this.user_id_1] = { email: 'user1@sharelatex.com' }
this.user_info[this.user_id_2] = { email: 'user2@sharelatex.com' }
this.WebApiManager.getUserInfo = (user_id, callback) => {
if (callback == null) {
callback = function (error, userInfo) {}
}
return callback(null, this.user_info[user_id])
}
sinon.spy(this.WebApiManager, 'getUserInfo')
return this.UpdatesManager.fillUserInfo(
this.updates,
(error, results) => {
this.results = results
return done()
}
)
})
it('should only call getUserInfo once for each user_id', function () {
this.WebApiManager.getUserInfo.calledTwice.should.equal(true)
this.WebApiManager.getUserInfo
.calledWith(this.user_id_1)
.should.equal(true)
return this.WebApiManager.getUserInfo
.calledWith(this.user_id_2)
.should.equal(true)
})
return it('should return the updates with the user info filled', function () {
return expect(this.results).to.deep.equal([
{
meta: {
user: {
email: 'user1@sharelatex.com',
},
},
op: 'mock-op-1',
},
{
meta: {
user: {
email: 'user1@sharelatex.com',
},
},
op: 'mock-op-2',
},
{
meta: {
user: {
email: 'user2@sharelatex.com',
},
},
op: 'mock-op-3',
},
])
})
})
return describe('with invalid user ids', function () {
beforeEach(function (done) {
this.updates = [
{
meta: {
user_id: null,
},
op: 'mock-op-1',
},
{
meta: {
user_id: 'anonymous-user',
},
op: 'mock-op-2',
},
]
this.WebApiManager.getUserInfo = (user_id, callback) => {
if (callback == null) {
callback = function (error, userInfo) {}
}
return callback(null, this.user_info[user_id])
}
sinon.spy(this.WebApiManager, 'getUserInfo')
return this.UpdatesManager.fillUserInfo(
this.updates,
(error, results) => {
this.results = results
return done()
}
)
})
it('should not call getUserInfo', function () {
return this.WebApiManager.getUserInfo.called.should.equal(false)
})
return it('should return the updates without the user info filled', function () {
return expect(this.results).to.deep.equal([
{
meta: {},
op: 'mock-op-1',
},
{
meta: {},
op: 'mock-op-2',
},
])
})
})
})
return describe('_summarizeUpdates', function () {
beforeEach(function () {
this.now = Date.now()
this.user_1 = { id: 'mock-user-1' }
return (this.user_2 = { id: 'mock-user-2' })
})
it('should concat updates that are close in time', function () {
const result = this.UpdatesManager._summarizeUpdates([
{
doc_id: 'doc-id-1',
meta: {
user_id: this.user_1.id,
start_ts: this.now + 20,
end_ts: this.now + 30,
},
v: 5,
},
{
doc_id: 'doc-id-1',
meta: {
user_id: this.user_2.id,
start_ts: this.now,
end_ts: this.now + 10,
},
v: 4,
},
])
return expect(result).to.deep.equal([
{
docs: {
'doc-id-1': {
fromV: 4,
toV: 5,
},
},
meta: {
user_ids: [this.user_1.id, this.user_2.id],
start_ts: this.now,
end_ts: this.now + 30,
},
},
])
})
it('should leave updates that are far apart in time', function () {
const oneDay = 1000 * 60 * 60 * 24
const result = this.UpdatesManager._summarizeUpdates([
{
doc_id: 'doc-id-1',
meta: {
user_id: this.user_2.id,
start_ts: this.now + oneDay,
end_ts: this.now + oneDay + 10,
},
v: 5,
},
{
doc_id: 'doc-id-1',
meta: {
user_id: this.user_1.id,
start_ts: this.now,
end_ts: this.now + 10,
},
v: 4,
},
])
return expect(result).to.deep.equal([
{
docs: {
'doc-id-1': {
fromV: 5,
toV: 5,
},
},
meta: {
user_ids: [this.user_2.id],
start_ts: this.now + oneDay,
end_ts: this.now + oneDay + 10,
},
},
{
docs: {
'doc-id-1': {
fromV: 4,
toV: 4,
},
},
meta: {
user_ids: [this.user_1.id],
start_ts: this.now,
end_ts: this.now + 10,
},
},
])
})
it('should concat onto existing summarized updates', function () {
const result = this.UpdatesManager._summarizeUpdates(
[
{
doc_id: 'doc-id-2',
meta: {
user_id: this.user_1.id,
start_ts: this.now + 20,
end_ts: this.now + 30,
},
v: 5,
},
{
doc_id: 'doc-id-2',
meta: {
user_id: this.user_2.id,
start_ts: this.now,
end_ts: this.now + 10,
},
v: 4,
},
],
[
{
docs: {
'doc-id-1': {
fromV: 6,
toV: 8,
},
},
meta: {
user_ids: [this.user_1.id],
start_ts: this.now + 40,
end_ts: this.now + 50,
},
},
]
)
return expect(result).to.deep.equal([
{
docs: {
'doc-id-1': {
toV: 8,
fromV: 6,
},
'doc-id-2': {
toV: 5,
fromV: 4,
},
},
meta: {
user_ids: [this.user_1.id, this.user_2.id],
start_ts: this.now,
end_ts: this.now + 50,
},
},
])
})
it('should include null user values', function () {
const result = this.UpdatesManager._summarizeUpdates([
{
doc_id: 'doc-id-1',
meta: {
user_id: this.user_1.id,
start_ts: this.now + 20,
end_ts: this.now + 30,
},
v: 5,
},
{
doc_id: 'doc-id-1',
meta: {
user_id: null,
start_ts: this.now,
end_ts: this.now + 10,
},
v: 4,
},
])
return expect(result).to.deep.equal([
{
docs: {
'doc-id-1': {
fromV: 4,
toV: 5,
},
},
meta: {
user_ids: [this.user_1.id, null],
start_ts: this.now,
end_ts: this.now + 30,
},
},
])
})
it('should include null user values, when the null is earlier in the updates list', function () {
const result = this.UpdatesManager._summarizeUpdates([
{
doc_id: 'doc-id-1',
meta: {
user_id: null,
start_ts: this.now,
end_ts: this.now + 10,
},
v: 4,
},
{
doc_id: 'doc-id-1',
meta: {
user_id: this.user_1.id,
start_ts: this.now + 20,
end_ts: this.now + 30,
},
v: 5,
},
])
return expect(result).to.deep.equal([
{
docs: {
'doc-id-1': {
fromV: 4,
toV: 5,
},
},
meta: {
user_ids: [null, this.user_1.id],
start_ts: this.now,
end_ts: this.now + 30,
},
},
])
})
it('should roll several null user values into one', function () {
const result = this.UpdatesManager._summarizeUpdates([
{
doc_id: 'doc-id-1',
meta: {
user_id: this.user_1.id,
start_ts: this.now + 20,
end_ts: this.now + 30,
},
v: 5,
},
{
doc_id: 'doc-id-1',
meta: {
user_id: null,
start_ts: this.now,
end_ts: this.now + 10,
},
v: 4,
},
{
doc_id: 'doc-id-1',
meta: {
user_id: null,
start_ts: this.now + 2,
end_ts: this.now + 4,
},
v: 4,
},
])
return expect(result).to.deep.equal([
{
docs: {
'doc-id-1': {
fromV: 4,
toV: 5,
},
},
meta: {
user_ids: [this.user_1.id, null],
start_ts: this.now,
end_ts: this.now + 30,
},
},
])
})
return it('should split updates before a big delete', function () {
const result = this.UpdatesManager._summarizeUpdates([
{
doc_id: 'doc-id-1',
op: [{ d: 'this is a long long long long long delete', p: 34 }],
meta: {
user_id: this.user_1.id,
start_ts: this.now + 20,
end_ts: this.now + 30,
},
v: 5,
},
{
doc_id: 'doc-id-1',
meta: {
user_id: this.user_2.id,
start_ts: this.now,
end_ts: this.now + 10,
},
v: 4,
},
])
return expect(result).to.deep.equal([
{
docs: {
'doc-id-1': {
fromV: 5,
toV: 5,
},
},
meta: {
user_ids: [this.user_1.id],
start_ts: this.now + 20,
end_ts: this.now + 30,
},
},
{
docs: {
'doc-id-1': {
fromV: 4,
toV: 4,
},
},
meta: {
user_ids: [this.user_2.id],
start_ts: this.now,
end_ts: this.now + 10,
},
},
])
})
})
})