2020-02-17 17:35:01 +00:00
|
|
|
/* eslint-disable
|
|
|
|
camelcase,
|
|
|
|
no-return-assign,
|
|
|
|
no-unused-vars,
|
|
|
|
*/
|
|
|
|
// TODO: This file was created by bulk-decaffeinate.
|
|
|
|
// Fix any style issues and re-enable lint.
|
2020-02-17 17:34:50 +00:00
|
|
|
/*
|
|
|
|
* decaffeinate suggestions:
|
|
|
|
* DS101: Remove unnecessary use of Array.from
|
|
|
|
* DS102: Remove unnecessary code created because of implicit returns
|
|
|
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/master/docs/suggestions.md
|
|
|
|
*/
|
2020-02-17 17:35:16 +00:00
|
|
|
const sinon = require('sinon')
|
|
|
|
const chai = require('chai')
|
|
|
|
const should = chai.should()
|
|
|
|
const { expect } = chai
|
|
|
|
const modulePath = '../../../../app/js/RedisManager.js'
|
|
|
|
const SandboxedModule = require('sandboxed-module')
|
|
|
|
|
2020-06-04 08:24:21 +00:00
|
|
|
describe('RedisManager', function () {
|
|
|
|
beforeEach(function () {
|
2020-02-17 17:35:16 +00:00
|
|
|
this.RedisManager = SandboxedModule.require(modulePath, {
|
|
|
|
requires: {
|
2020-11-10 11:32:05 +00:00
|
|
|
'@overleaf/redis-wrapper': {
|
2020-02-17 17:35:16 +00:00
|
|
|
createClient: () => {
|
|
|
|
return (this.rclient = {
|
|
|
|
auth: sinon.stub(),
|
|
|
|
multi: () => this.rclient
|
|
|
|
})
|
|
|
|
}
|
|
|
|
},
|
|
|
|
'settings-sharelatex': {
|
|
|
|
redis: {
|
|
|
|
history: {
|
|
|
|
key_schema: {
|
|
|
|
uncompressedHistoryOps({ doc_id }) {
|
|
|
|
return `UncompressedHistoryOps:${doc_id}`
|
|
|
|
},
|
|
|
|
docsWithHistoryOps({ project_id }) {
|
|
|
|
return `DocsWithHistoryOps:${project_id}`
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
this.doc_id = 'doc-id-123'
|
|
|
|
this.project_id = 'project-id-123'
|
|
|
|
this.batchSize = 100
|
|
|
|
return (this.callback = sinon.stub())
|
|
|
|
})
|
|
|
|
|
2020-06-04 08:24:21 +00:00
|
|
|
describe('getOldestDocUpdates', function () {
|
|
|
|
beforeEach(function () {
|
2020-02-17 17:35:16 +00:00
|
|
|
this.rawUpdates = [
|
|
|
|
{ v: 42, op: 'mock-op-42' },
|
|
|
|
{ v: 45, op: 'mock-op-45' }
|
|
|
|
]
|
2020-06-04 08:24:21 +00:00
|
|
|
this.jsonUpdates = Array.from(this.rawUpdates).map((update) =>
|
2020-02-17 17:35:16 +00:00
|
|
|
JSON.stringify(update)
|
|
|
|
)
|
|
|
|
this.rclient.lrange = sinon.stub().callsArgWith(3, null, this.jsonUpdates)
|
|
|
|
return this.RedisManager.getOldestDocUpdates(
|
|
|
|
this.doc_id,
|
|
|
|
this.batchSize,
|
|
|
|
this.callback
|
|
|
|
)
|
|
|
|
})
|
|
|
|
|
2020-06-04 08:24:21 +00:00
|
|
|
it('should read the updates from redis', function () {
|
2020-02-17 17:35:16 +00:00
|
|
|
return this.rclient.lrange
|
|
|
|
.calledWith(
|
|
|
|
`UncompressedHistoryOps:${this.doc_id}`,
|
|
|
|
0,
|
|
|
|
this.batchSize - 1
|
|
|
|
)
|
|
|
|
.should.equal(true)
|
|
|
|
})
|
|
|
|
|
2020-06-04 08:24:21 +00:00
|
|
|
it('should call the callback with the unparsed ops', function () {
|
2020-02-17 17:35:16 +00:00
|
|
|
return this.callback.calledWith(null, this.jsonUpdates).should.equal(true)
|
|
|
|
})
|
|
|
|
|
2020-06-04 08:24:21 +00:00
|
|
|
describe('expandDocUpdates', function () {
|
|
|
|
beforeEach(function () {
|
2020-02-17 17:35:16 +00:00
|
|
|
return this.RedisManager.expandDocUpdates(
|
|
|
|
this.jsonUpdates,
|
|
|
|
this.callback
|
|
|
|
)
|
|
|
|
})
|
|
|
|
|
2020-06-04 08:24:21 +00:00
|
|
|
return it('should call the callback with the parsed ops', function () {
|
2020-02-17 17:35:16 +00:00
|
|
|
return this.callback
|
|
|
|
.calledWith(null, this.rawUpdates)
|
|
|
|
.should.equal(true)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2020-06-04 08:24:21 +00:00
|
|
|
return describe('deleteAppliedDocUpdates', function () {
|
|
|
|
beforeEach(function () {
|
2020-02-17 17:35:16 +00:00
|
|
|
this.rclient.lrem = sinon.stub()
|
|
|
|
this.rclient.srem = sinon.stub()
|
|
|
|
this.rclient.exec = sinon.stub().callsArgWith(0)
|
|
|
|
return this.RedisManager.deleteAppliedDocUpdates(
|
|
|
|
this.project_id,
|
|
|
|
this.doc_id,
|
|
|
|
this.jsonUpdates,
|
|
|
|
this.callback
|
|
|
|
)
|
|
|
|
})
|
|
|
|
|
2020-06-04 08:24:21 +00:00
|
|
|
it('should delete the first update from redis', function () {
|
2020-02-17 17:35:16 +00:00
|
|
|
return this.rclient.lrem
|
|
|
|
.calledWith(
|
|
|
|
`UncompressedHistoryOps:${this.doc_id}`,
|
|
|
|
1,
|
|
|
|
this.jsonUpdates[0]
|
|
|
|
)
|
|
|
|
.should.equal(true)
|
|
|
|
})
|
|
|
|
|
2020-06-04 08:24:21 +00:00
|
|
|
it('should delete the second update from redis', function () {
|
2020-02-17 17:35:16 +00:00
|
|
|
return this.rclient.lrem
|
|
|
|
.calledWith(
|
|
|
|
`UncompressedHistoryOps:${this.doc_id}`,
|
|
|
|
1,
|
|
|
|
this.jsonUpdates[1]
|
|
|
|
)
|
|
|
|
.should.equal(true)
|
|
|
|
})
|
|
|
|
|
2020-06-04 08:24:21 +00:00
|
|
|
it('should delete the doc from the set of docs with history ops', function () {
|
2020-02-17 17:35:16 +00:00
|
|
|
return this.rclient.srem
|
|
|
|
.calledWith(`DocsWithHistoryOps:${this.project_id}`, this.doc_id)
|
|
|
|
.should.equal(true)
|
|
|
|
})
|
|
|
|
|
2020-06-04 08:24:21 +00:00
|
|
|
return it('should call the callback ', function () {
|
2020-02-17 17:35:16 +00:00
|
|
|
return this.callback.called.should.equal(true)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2020-06-04 08:24:21 +00:00
|
|
|
return describe('getDocIdsWithHistoryOps', function () {
|
|
|
|
beforeEach(function () {
|
2020-02-17 17:35:16 +00:00
|
|
|
this.doc_ids = ['mock-id-1', 'mock-id-2']
|
|
|
|
this.rclient.smembers = sinon.stub().callsArgWith(1, null, this.doc_ids)
|
|
|
|
return this.RedisManager.getDocIdsWithHistoryOps(
|
|
|
|
this.project_id,
|
|
|
|
this.callback
|
|
|
|
)
|
|
|
|
})
|
|
|
|
|
2020-06-04 08:24:21 +00:00
|
|
|
it('should read the doc_ids from redis', function () {
|
2020-02-17 17:35:16 +00:00
|
|
|
return this.rclient.smembers
|
|
|
|
.calledWith(`DocsWithHistoryOps:${this.project_id}`)
|
|
|
|
.should.equal(true)
|
|
|
|
})
|
|
|
|
|
2020-06-04 08:24:21 +00:00
|
|
|
return it('should call the callback with the doc_ids', function () {
|
2020-02-17 17:35:16 +00:00
|
|
|
return this.callback.calledWith(null, this.doc_ids).should.equal(true)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|