2014-01-27 17:51:09 +00:00
|
|
|
sinon = require "sinon"
|
|
|
|
chai = require("chai")
|
|
|
|
chai.should()
|
2014-02-26 12:11:45 +00:00
|
|
|
expect = chai.expect
|
2014-01-27 17:51:09 +00:00
|
|
|
mongojs = require "../../../app/js/mongojs"
|
|
|
|
ObjectId = mongojs.ObjectId
|
|
|
|
Settings = require "settings-sharelatex"
|
|
|
|
request = require "request"
|
2014-02-26 12:11:45 +00:00
|
|
|
rclient = require("redis").createClient() # Only works locally for now
|
2014-01-27 17:51:09 +00:00
|
|
|
|
2014-03-04 15:27:03 +00:00
|
|
|
TrackChangesClient = require "./helpers/TrackChangesClient"
|
2014-05-16 14:59:12 +00:00
|
|
|
MockWebApi = require "./helpers/MockWebApi"
|
2014-02-26 12:38:47 +00:00
|
|
|
|
2014-01-27 17:51:09 +00:00
|
|
|
describe "Appending doc ops to the history", ->
|
|
|
|
describe "when the history does not exist yet", ->
|
|
|
|
before (done) ->
|
2014-03-19 16:14:17 +00:00
|
|
|
@project_id = ObjectId().toString()
|
2014-01-27 17:51:09 +00:00
|
|
|
@doc_id = ObjectId().toString()
|
|
|
|
@user_id = ObjectId().toString()
|
2014-05-16 14:59:12 +00:00
|
|
|
MockWebApi.projects[@project_id] = features: versioning: false
|
2014-03-21 13:17:58 +00:00
|
|
|
TrackChangesClient.pushRawUpdates @project_id, @doc_id, [{
|
2014-01-27 17:51:09 +00:00
|
|
|
op: [{ i: "f", p: 3 }]
|
|
|
|
meta: { ts: Date.now(), user_id: @user_id }
|
2014-02-26 12:11:45 +00:00
|
|
|
v: 3
|
2014-01-27 17:51:09 +00:00
|
|
|
}, {
|
|
|
|
op: [{ i: "o", p: 4 }]
|
|
|
|
meta: { ts: Date.now(), user_id: @user_id }
|
2014-02-26 12:11:45 +00:00
|
|
|
v: 4
|
2014-01-27 17:51:09 +00:00
|
|
|
}, {
|
|
|
|
op: [{ i: "o", p: 5 }]
|
|
|
|
meta: { ts: Date.now(), user_id: @user_id }
|
2014-02-26 12:11:45 +00:00
|
|
|
v: 5
|
2014-02-26 12:38:47 +00:00
|
|
|
}], (error) =>
|
|
|
|
throw error if error?
|
2014-03-19 16:14:17 +00:00
|
|
|
TrackChangesClient.flushAndGetCompressedUpdates @project_id, @doc_id, (error, @updates) =>
|
2014-02-26 12:38:47 +00:00
|
|
|
throw error if error?
|
|
|
|
done()
|
2014-01-27 17:51:09 +00:00
|
|
|
|
2014-02-26 12:11:45 +00:00
|
|
|
it "should insert the compressed op into mongo", ->
|
2014-03-11 15:24:38 +00:00
|
|
|
expect(@updates[0].op).to.deep.equal [{
|
2014-02-26 12:11:45 +00:00
|
|
|
p: 3, i: "foo"
|
2014-03-11 15:24:38 +00:00
|
|
|
}]
|
2014-01-27 17:51:09 +00:00
|
|
|
|
2014-02-26 12:11:45 +00:00
|
|
|
it "should insert the correct version number into mongo", ->
|
2014-02-26 12:38:47 +00:00
|
|
|
expect(@updates[0].v).to.equal 5
|
2014-02-26 12:11:45 +00:00
|
|
|
|
2014-03-19 16:40:55 +00:00
|
|
|
it "should store the doc id", ->
|
|
|
|
expect(@updates[0].doc_id.toString()).to.equal @doc_id
|
|
|
|
|
|
|
|
it "should store the project id", ->
|
|
|
|
expect(@updates[0].project_id.toString()).to.equal @project_id
|
|
|
|
|
2014-03-21 13:17:58 +00:00
|
|
|
it "should clear the doc from the DocsWithHistoryOps set", (done) ->
|
|
|
|
rclient.sismember "DocsWithHistoryOps:#{@project_id}", @doc_id, (error, member) ->
|
|
|
|
member.should.equal 0
|
|
|
|
done()
|
|
|
|
|
2014-01-27 18:20:38 +00:00
|
|
|
describe "when the history has already been started", ->
|
|
|
|
beforeEach (done) ->
|
2014-03-19 16:14:17 +00:00
|
|
|
@project_id = ObjectId().toString()
|
2014-01-27 18:20:38 +00:00
|
|
|
@doc_id = ObjectId().toString()
|
|
|
|
@user_id = ObjectId().toString()
|
2014-05-16 14:59:12 +00:00
|
|
|
MockWebApi.projects[@project_id] = features: versioning: false
|
2014-03-21 13:17:58 +00:00
|
|
|
TrackChangesClient.pushRawUpdates @project_id, @doc_id, [{
|
2014-01-27 18:20:38 +00:00
|
|
|
op: [{ i: "f", p: 3 }]
|
|
|
|
meta: { ts: Date.now(), user_id: @user_id }
|
2014-02-26 12:38:47 +00:00
|
|
|
v: 3
|
2014-01-27 18:20:38 +00:00
|
|
|
}, {
|
|
|
|
op: [{ i: "o", p: 4 }]
|
|
|
|
meta: { ts: Date.now(), user_id: @user_id }
|
2014-02-26 12:38:47 +00:00
|
|
|
v: 4
|
2014-01-27 18:20:38 +00:00
|
|
|
}, {
|
|
|
|
op: [{ i: "o", p: 5 }]
|
|
|
|
meta: { ts: Date.now(), user_id: @user_id }
|
2014-02-26 12:38:47 +00:00
|
|
|
v: 5
|
|
|
|
}], (error) =>
|
|
|
|
throw error if error?
|
2014-03-19 16:14:17 +00:00
|
|
|
TrackChangesClient.flushAndGetCompressedUpdates @project_id, @doc_id, (error, updates) =>
|
2014-02-26 12:38:47 +00:00
|
|
|
throw error if error?
|
|
|
|
done()
|
2014-01-27 18:20:38 +00:00
|
|
|
|
|
|
|
describe "when the updates are recent and from the same user", ->
|
|
|
|
beforeEach (done) ->
|
2014-03-21 13:17:58 +00:00
|
|
|
TrackChangesClient.pushRawUpdates @project_id, @doc_id, [{
|
2014-01-27 18:20:38 +00:00
|
|
|
op: [{ i: "b", p: 6 }]
|
|
|
|
meta: { ts: Date.now(), user_id: @user_id }
|
2014-02-26 12:38:47 +00:00
|
|
|
v: 6
|
2014-01-27 18:20:38 +00:00
|
|
|
}, {
|
|
|
|
op: [{ i: "a", p: 7 }]
|
|
|
|
meta: { ts: Date.now(), user_id: @user_id }
|
2014-02-26 12:38:47 +00:00
|
|
|
v: 7
|
2014-01-27 18:20:38 +00:00
|
|
|
}, {
|
|
|
|
op: [{ i: "r", p: 8 }]
|
|
|
|
meta: { ts: Date.now(), user_id: @user_id }
|
2014-02-26 12:38:47 +00:00
|
|
|
v: 8
|
|
|
|
}], (error) =>
|
|
|
|
throw error if error?
|
2014-03-19 16:14:17 +00:00
|
|
|
TrackChangesClient.flushAndGetCompressedUpdates @project_id, @doc_id, (error, @updates) =>
|
2014-02-26 12:38:47 +00:00
|
|
|
throw error if error?
|
|
|
|
done()
|
2014-01-27 18:20:38 +00:00
|
|
|
|
2014-02-26 12:38:47 +00:00
|
|
|
it "should combine all the updates into one", ->
|
2014-03-11 15:24:38 +00:00
|
|
|
expect(@updates[0].op).to.deep.equal [{
|
2014-02-26 12:38:47 +00:00
|
|
|
p: 3, i: "foobar"
|
2014-03-11 15:24:38 +00:00
|
|
|
}]
|
2014-01-27 18:20:38 +00:00
|
|
|
|
2014-02-26 12:38:47 +00:00
|
|
|
it "should insert the correct version number into mongo", ->
|
|
|
|
expect(@updates[0].v).to.equal 8
|
2014-01-27 18:20:38 +00:00
|
|
|
|
|
|
|
|
|
|
|
describe "when the updates are far apart", ->
|
|
|
|
beforeEach (done) ->
|
|
|
|
oneDay = 24 * 60 * 60 * 1000
|
2014-03-21 13:17:58 +00:00
|
|
|
TrackChangesClient.pushRawUpdates @project_id, @doc_id, [{
|
2014-01-27 18:20:38 +00:00
|
|
|
op: [{ i: "b", p: 6 }]
|
|
|
|
meta: { ts: Date.now() + oneDay, user_id: @user_id }
|
2014-02-26 12:38:47 +00:00
|
|
|
v: 6
|
2014-01-27 18:20:38 +00:00
|
|
|
}, {
|
|
|
|
op: [{ i: "a", p: 7 }]
|
|
|
|
meta: { ts: Date.now() + oneDay, user_id: @user_id }
|
2014-02-26 12:38:47 +00:00
|
|
|
v: 7
|
2014-01-27 18:20:38 +00:00
|
|
|
}, {
|
|
|
|
op: [{ i: "r", p: 8 }]
|
|
|
|
meta: { ts: Date.now() + oneDay, user_id: @user_id }
|
2014-02-26 12:38:47 +00:00
|
|
|
v: 8
|
|
|
|
}], (error) =>
|
|
|
|
throw error if error?
|
2014-03-19 16:14:17 +00:00
|
|
|
TrackChangesClient.flushAndGetCompressedUpdates @project_id, @doc_id, (error, @updates) =>
|
2014-02-26 12:38:47 +00:00
|
|
|
throw error if error?
|
|
|
|
done()
|
2014-01-27 18:20:38 +00:00
|
|
|
|
2014-02-26 12:38:47 +00:00
|
|
|
it "should keep the updates separate", ->
|
2014-03-11 15:24:38 +00:00
|
|
|
expect(@updates[0].op).to.deep.equal [{
|
2014-02-26 12:38:47 +00:00
|
|
|
p: 3, i: "foo"
|
2014-03-11 15:24:38 +00:00
|
|
|
}]
|
|
|
|
expect(@updates[1].op).to.deep.equal [{
|
2014-02-26 12:38:47 +00:00
|
|
|
p: 6, i: "bar"
|
2014-03-11 15:24:38 +00:00
|
|
|
}]
|
2014-02-26 12:44:13 +00:00
|
|
|
|
|
|
|
describe "when the updates need processing in batches", ->
|
|
|
|
before (done) ->
|
2014-03-19 16:14:17 +00:00
|
|
|
@project_id = ObjectId().toString()
|
2014-02-26 12:44:13 +00:00
|
|
|
@doc_id = ObjectId().toString()
|
|
|
|
@user_id = ObjectId().toString()
|
2014-05-16 14:59:12 +00:00
|
|
|
MockWebApi.projects[@project_id] = features: versioning: false
|
2014-02-26 12:44:13 +00:00
|
|
|
updates = []
|
2014-03-11 15:24:38 +00:00
|
|
|
@expectedOp = [{ p:0, i: "" }]
|
2014-02-26 12:44:13 +00:00
|
|
|
for i in [0..250]
|
|
|
|
updates.push {
|
|
|
|
op: [{i: "a", p: 0}]
|
|
|
|
meta: { ts: Date.now(), user_id: @user_id }
|
|
|
|
v: i
|
|
|
|
}
|
2014-03-11 15:24:38 +00:00
|
|
|
@expectedOp[0].i = "a" + @expectedOp[0].i
|
2014-02-26 12:44:13 +00:00
|
|
|
|
2014-03-21 13:17:58 +00:00
|
|
|
TrackChangesClient.pushRawUpdates @project_id, @doc_id, updates, (error) =>
|
2014-02-26 12:44:13 +00:00
|
|
|
throw error if error?
|
2014-03-19 16:14:17 +00:00
|
|
|
TrackChangesClient.flushAndGetCompressedUpdates @project_id, @doc_id, (error, @updates) =>
|
2014-02-26 12:44:13 +00:00
|
|
|
throw error if error?
|
|
|
|
done()
|
|
|
|
|
|
|
|
it "should concat the compressed op into mongo", ->
|
|
|
|
expect(@updates[0].op).to.deep.equal @expectedOp
|
|
|
|
|
|
|
|
it "should insert the correct version number into mongo", ->
|
|
|
|
expect(@updates[0].v).to.equal 250
|
|
|
|
|
2014-03-11 15:24:38 +00:00
|
|
|
|
|
|
|
describe "when there are multiple ops in each update", ->
|
|
|
|
before (done) ->
|
2014-03-19 16:14:17 +00:00
|
|
|
@project_id = ObjectId().toString()
|
2014-03-11 15:24:38 +00:00
|
|
|
@doc_id = ObjectId().toString()
|
|
|
|
@user_id = ObjectId().toString()
|
2014-05-16 14:59:12 +00:00
|
|
|
MockWebApi.projects[@project_id] = features: versioning: false
|
2014-03-11 15:24:38 +00:00
|
|
|
oneDay = 24 * 60 * 60 * 1000
|
2014-03-21 13:17:58 +00:00
|
|
|
TrackChangesClient.pushRawUpdates @project_id, @doc_id, [{
|
2014-03-11 15:24:38 +00:00
|
|
|
op: [{ i: "f", p: 3 }, { i: "o", p: 4 }, { i: "o", p: 5 }]
|
|
|
|
meta: { ts: Date.now(), user_id: @user_id }
|
|
|
|
v: 3
|
|
|
|
}, {
|
|
|
|
op: [{ i: "b", p: 6 }, { i: "a", p: 7 }, { i: "r", p: 8 }]
|
|
|
|
meta: { ts: Date.now() + oneDay, user_id: @user_id }
|
|
|
|
v: 4
|
|
|
|
}], (error) =>
|
|
|
|
throw error if error?
|
2014-03-19 16:14:17 +00:00
|
|
|
TrackChangesClient.flushAndGetCompressedUpdates @project_id, @doc_id, (error, @updates) =>
|
2014-03-11 15:24:38 +00:00
|
|
|
throw error if error?
|
|
|
|
done()
|
|
|
|
|
|
|
|
it "should insert the compressed ops into mongo", ->
|
|
|
|
expect(@updates[0].op).to.deep.equal [{
|
|
|
|
p: 3, i: "foo"
|
|
|
|
}]
|
|
|
|
expect(@updates[1].op).to.deep.equal [{
|
|
|
|
p: 6, i: "bar"
|
|
|
|
}]
|
|
|
|
|
|
|
|
it "should insert the correct version numbers into mongo", ->
|
|
|
|
expect(@updates[0].v).to.equal 3
|
|
|
|
expect(@updates[1].v).to.equal 4
|
|
|
|
|
2014-03-25 11:40:48 +00:00
|
|
|
describe "when there is a no-op update", ->
|
|
|
|
before (done) ->
|
|
|
|
@project_id = ObjectId().toString()
|
|
|
|
@doc_id = ObjectId().toString()
|
|
|
|
@user_id = ObjectId().toString()
|
2014-05-16 14:59:12 +00:00
|
|
|
MockWebApi.projects[@project_id] = features: versioning: false
|
2014-03-25 11:40:48 +00:00
|
|
|
oneDay = 24 * 60 * 60 * 1000
|
|
|
|
TrackChangesClient.pushRawUpdates @project_id, @doc_id, [{
|
|
|
|
op: []
|
|
|
|
meta: { ts: Date.now(), user_id: @user_id }
|
|
|
|
v: 3
|
|
|
|
}, {
|
|
|
|
op: [{ i: "foo", p: 3 }]
|
|
|
|
meta: { ts: Date.now() + oneDay, user_id: @user_id }
|
|
|
|
v: 4
|
|
|
|
}], (error) =>
|
|
|
|
throw error if error?
|
|
|
|
TrackChangesClient.flushAndGetCompressedUpdates @project_id, @doc_id, (error, @updates) =>
|
|
|
|
throw error if error?
|
|
|
|
done()
|
|
|
|
|
|
|
|
it "should insert the compressed no-op into mongo", ->
|
|
|
|
expect(@updates[0].op).to.deep.equal []
|
|
|
|
|
|
|
|
|
|
|
|
it "should insert the compressed next update into mongo", ->
|
|
|
|
expect(@updates[1].op).to.deep.equal [{
|
|
|
|
p: 3, i: "foo"
|
|
|
|
}]
|
|
|
|
|
|
|
|
it "should insert the correct version numbers into mongo", ->
|
|
|
|
expect(@updates[0].v).to.equal 3
|
|
|
|
expect(@updates[1].v).to.equal 4
|
2014-05-16 14:59:12 +00:00
|
|
|
|
|
|
|
describe "when the project has versioning enabled", ->
|
|
|
|
before (done) ->
|
|
|
|
@project_id = ObjectId().toString()
|
|
|
|
@doc_id = ObjectId().toString()
|
|
|
|
@user_id = ObjectId().toString()
|
|
|
|
MockWebApi.projects[@project_id] = features: versioning: true
|
|
|
|
|
|
|
|
TrackChangesClient.pushRawUpdates @project_id, @doc_id, [{
|
|
|
|
op: [{ i: "f", p: 3 }]
|
|
|
|
meta: { ts: Date.now(), user_id: @user_id }
|
|
|
|
v: 3
|
|
|
|
}], (error) =>
|
|
|
|
throw error if error?
|
|
|
|
TrackChangesClient.flushAndGetCompressedUpdates @project_id, @doc_id, (error, @updates) =>
|
|
|
|
throw error if error?
|
|
|
|
done()
|
|
|
|
|
2014-05-16 15:41:14 +00:00
|
|
|
it "should not add a expiresAt entry in the update in mongo", ->
|
|
|
|
expect(@updates[0].expiresAt).to.be.undefined
|
2014-05-16 14:59:12 +00:00
|
|
|
|
|
|
|
describe "when the project does not have versioning enabled", ->
|
|
|
|
before (done) ->
|
|
|
|
@project_id = ObjectId().toString()
|
|
|
|
@doc_id = ObjectId().toString()
|
|
|
|
@user_id = ObjectId().toString()
|
|
|
|
MockWebApi.projects[@project_id] = features: versioning: false
|
|
|
|
|
|
|
|
TrackChangesClient.pushRawUpdates @project_id, @doc_id, [{
|
|
|
|
op: [{ i: "f", p: 3 }]
|
|
|
|
meta: { ts: Date.now(), user_id: @user_id }
|
|
|
|
v: 3
|
|
|
|
}], (error) =>
|
|
|
|
throw error if error?
|
|
|
|
TrackChangesClient.flushAndGetCompressedUpdates @project_id, @doc_id, (error, @updates) =>
|
|
|
|
throw error if error?
|
|
|
|
done()
|
|
|
|
|
2014-05-16 15:41:14 +00:00
|
|
|
it "should add a expiresAt entry in the update in mongo", ->
|
|
|
|
expect(@updates[0].expiresAt).to.exist
|