2020-05-06 06:11:22 -04:00
|
|
|
/* eslint-disable
|
|
|
|
no-return-assign,
|
|
|
|
no-unused-vars,
|
|
|
|
*/
|
|
|
|
// TODO: This file was created by bulk-decaffeinate.
|
|
|
|
// Fix any style issues and re-enable lint.
|
2020-05-06 06:10:51 -04:00
|
|
|
/*
|
|
|
|
* decaffeinate suggestions:
|
|
|
|
* 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
|
|
|
|
*/
|
2020-05-06 06:11:36 -04:00
|
|
|
const sinon = require('sinon')
|
2021-04-01 15:51:00 -04:00
|
|
|
const { expect } = require('chai')
|
2020-05-06 06:11:36 -04:00
|
|
|
const modulePath = '../../../../app/js/ShareJsDB.js'
|
|
|
|
const SandboxedModule = require('sandboxed-module')
|
|
|
|
const Errors = require('../../../../app/js/Errors')
|
|
|
|
|
|
|
|
describe('ShareJsDB', function () {
|
|
|
|
beforeEach(function () {
|
|
|
|
this.doc_id = 'document-id'
|
|
|
|
this.project_id = 'project-id'
|
|
|
|
this.doc_key = `${this.project_id}:${this.doc_id}`
|
|
|
|
this.callback = sinon.stub()
|
|
|
|
this.ShareJsDB = SandboxedModule.require(modulePath, {
|
|
|
|
requires: {
|
2024-07-23 05:36:52 -04:00
|
|
|
'./RedisManager': (this.RedisManager = {
|
|
|
|
getPreviousDocOps: sinon.stub(),
|
|
|
|
}),
|
2021-07-13 07:04:42 -04:00
|
|
|
'./Errors': Errors,
|
2024-07-18 09:01:09 -04:00
|
|
|
'@overleaf/metrics': {
|
|
|
|
inc: sinon.stub(),
|
|
|
|
histogram: sinon.stub(),
|
|
|
|
Timer: class Timer {
|
|
|
|
done() {}
|
|
|
|
},
|
|
|
|
},
|
2021-07-13 07:04:42 -04:00
|
|
|
},
|
2020-05-06 06:11:36 -04:00
|
|
|
})
|
|
|
|
|
|
|
|
this.version = 42
|
|
|
|
this.lines = ['one', 'two', 'three']
|
|
|
|
return (this.db = new this.ShareJsDB(
|
|
|
|
this.project_id,
|
|
|
|
this.doc_id,
|
|
|
|
this.lines,
|
|
|
|
this.version
|
|
|
|
))
|
|
|
|
})
|
|
|
|
|
|
|
|
describe('getSnapshot', function () {
|
|
|
|
describe('successfully', function () {
|
|
|
|
beforeEach(function () {
|
|
|
|
return this.db.getSnapshot(this.doc_key, this.callback)
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should return the doc lines', function () {
|
|
|
|
return this.callback.args[0][1].snapshot.should.equal(
|
|
|
|
this.lines.join('\n')
|
|
|
|
)
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should return the doc version', function () {
|
|
|
|
return this.callback.args[0][1].v.should.equal(this.version)
|
|
|
|
})
|
|
|
|
|
|
|
|
return it('should return the type as text', function () {
|
|
|
|
return this.callback.args[0][1].type.should.equal('text')
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
return describe('when the key does not match', function () {
|
|
|
|
beforeEach(function () {
|
|
|
|
return this.db.getSnapshot('bad:key', this.callback)
|
|
|
|
})
|
|
|
|
|
|
|
|
return it('should return the callback with a NotFoundError', function () {
|
|
|
|
return this.callback
|
2020-05-15 14:29:49 -04:00
|
|
|
.calledWith(sinon.match.instanceOf(Errors.NotFoundError))
|
2020-05-06 06:11:36 -04:00
|
|
|
.should.equal(true)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
describe('getOps', function () {
|
|
|
|
describe('with start == end', function () {
|
|
|
|
beforeEach(function () {
|
|
|
|
this.start = this.end = 42
|
|
|
|
return this.db.getOps(this.doc_key, this.start, this.end, this.callback)
|
|
|
|
})
|
|
|
|
|
2024-07-23 05:36:52 -04:00
|
|
|
it('should not talk to redis', function () {
|
|
|
|
this.RedisManager.getPreviousDocOps.should.not.have.been.called
|
|
|
|
})
|
|
|
|
|
2020-05-06 06:11:36 -04:00
|
|
|
return it('should return an empty array', function () {
|
|
|
|
return this.callback.calledWith(null, []).should.equal(true)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2024-07-23 05:36:52 -04:00
|
|
|
describe('with start == redis-version and end unset', function () {
|
|
|
|
beforeEach(function () {
|
|
|
|
const start = this.version
|
|
|
|
const end = null
|
|
|
|
this.db.getOps(this.doc_key, start, end, this.callback)
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should not talk to redis', function () {
|
|
|
|
this.RedisManager.getPreviousDocOps.should.not.have.been.called
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should return an empty array', function () {
|
|
|
|
this.callback.should.have.been.calledWith(null, [])
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2020-05-06 06:11:36 -04:00
|
|
|
describe('with a non empty range', function () {
|
|
|
|
beforeEach(function () {
|
|
|
|
this.start = 35
|
|
|
|
this.end = 42
|
2024-07-18 09:01:09 -04:00
|
|
|
this.ops = new Array(this.end - this.start)
|
2020-05-06 06:11:36 -04:00
|
|
|
this.RedisManager.getPreviousDocOps = sinon
|
|
|
|
.stub()
|
|
|
|
.callsArgWith(3, null, this.ops)
|
|
|
|
return this.db.getOps(this.doc_key, this.start, this.end, this.callback)
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should get the range from redis', function () {
|
|
|
|
return this.RedisManager.getPreviousDocOps
|
|
|
|
.calledWith(this.doc_id, this.start, this.end - 1)
|
|
|
|
.should.equal(true)
|
|
|
|
})
|
|
|
|
|
|
|
|
return it('should return the ops', function () {
|
|
|
|
return this.callback.calledWith(null, this.ops).should.equal(true)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
return describe('with no specified end', function () {
|
|
|
|
beforeEach(function () {
|
|
|
|
this.start = 35
|
|
|
|
this.end = null
|
2024-07-18 09:01:09 -04:00
|
|
|
this.ops = []
|
2020-05-06 06:11:36 -04:00
|
|
|
this.RedisManager.getPreviousDocOps = sinon
|
|
|
|
.stub()
|
|
|
|
.callsArgWith(3, null, this.ops)
|
|
|
|
return this.db.getOps(this.doc_key, this.start, this.end, this.callback)
|
|
|
|
})
|
|
|
|
|
|
|
|
return it('should get until the end of the list', function () {
|
|
|
|
return this.RedisManager.getPreviousDocOps
|
|
|
|
.calledWith(this.doc_id, this.start, -1)
|
|
|
|
.should.equal(true)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
return describe('writeOps', function () {
|
|
|
|
return describe('writing an op', function () {
|
|
|
|
beforeEach(function () {
|
|
|
|
this.opData = {
|
|
|
|
op: { p: 20, t: 'foo' },
|
|
|
|
meta: { source: 'bar' },
|
2021-07-13 07:04:42 -04:00
|
|
|
v: this.version,
|
2020-05-06 06:11:36 -04:00
|
|
|
}
|
|
|
|
return this.db.writeOp(this.doc_key, this.opData, this.callback)
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should write into appliedOps', function () {
|
|
|
|
return expect(this.db.appliedOps[this.doc_key]).to.deep.equal([
|
2021-07-13 07:04:42 -04:00
|
|
|
this.opData,
|
2020-05-06 06:11:36 -04:00
|
|
|
])
|
|
|
|
})
|
|
|
|
|
|
|
|
return it('should call the callback without an error', function () {
|
|
|
|
this.callback.called.should.equal(true)
|
|
|
|
return (this.callback.args[0][0] != null).should.equal(false)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|