2014-02-12 10:40:42 +00:00
|
|
|
sinon = require('sinon')
|
|
|
|
chai = require('chai')
|
|
|
|
should = chai.should()
|
|
|
|
modulePath = "../../../../app/js/LockManager.js"
|
|
|
|
SandboxedModule = require('sandboxed-module')
|
|
|
|
|
|
|
|
describe 'LockManager - trying the lock', ->
|
|
|
|
beforeEach ->
|
|
|
|
@LockManager = SandboxedModule.require modulePath, requires:
|
|
|
|
"logger-sharelatex": log:->
|
2016-07-06 10:50:02 +00:00
|
|
|
"redis-sharelatex":
|
2014-02-12 10:40:42 +00:00
|
|
|
createClient : () =>
|
|
|
|
auth:->
|
|
|
|
set: @set = sinon.stub()
|
2016-01-20 17:36:06 +00:00
|
|
|
"./Metrics": {inc: () ->}
|
2017-05-08 15:02:40 +00:00
|
|
|
"settings-sharelatex": {
|
|
|
|
redis:
|
|
|
|
lock:
|
|
|
|
key_schema:
|
|
|
|
blockingKey: ({doc_id}) -> "Blocking:#{doc_id}"
|
|
|
|
}
|
2017-07-12 09:45:44 +00:00
|
|
|
"./Profiler": @Profiler = class Profiler
|
2017-06-08 15:43:27 +00:00
|
|
|
log: sinon.stub().returns { end: sinon.stub() }
|
|
|
|
end: sinon.stub()
|
2016-01-20 17:36:06 +00:00
|
|
|
|
2014-02-12 10:40:42 +00:00
|
|
|
@callback = sinon.stub()
|
|
|
|
@doc_id = "doc-id-123"
|
|
|
|
|
|
|
|
describe "when the lock is not set", ->
|
|
|
|
beforeEach ->
|
2016-04-13 10:59:56 +00:00
|
|
|
@lockValue = "mock-lock-value"
|
|
|
|
@LockManager.randomLock = sinon.stub().returns @lockValue
|
2014-02-12 10:40:42 +00:00
|
|
|
@set.callsArgWith(5, null, "OK")
|
|
|
|
@LockManager.tryLock @doc_id, @callback
|
|
|
|
|
|
|
|
it "should set the lock key with an expiry if it is not set", ->
|
2016-04-13 10:59:56 +00:00
|
|
|
@set.calledWith("Blocking:#{@doc_id}", @lockValue, "EX", 30, "NX")
|
2014-02-12 10:40:42 +00:00
|
|
|
.should.equal true
|
|
|
|
|
2016-04-13 10:59:56 +00:00
|
|
|
it "should return the callback with true and the lock value", ->
|
|
|
|
@callback.calledWith(null, true, @lockValue).should.equal true
|
2014-02-12 10:40:42 +00:00
|
|
|
|
|
|
|
describe "when the lock is already set", ->
|
|
|
|
beforeEach ->
|
|
|
|
@set.callsArgWith(5, null, null)
|
|
|
|
@LockManager.tryLock @doc_id, @callback
|
|
|
|
|
|
|
|
it "should return the callback with false", ->
|
|
|
|
@callback.calledWith(null, false).should.equal true
|
|
|
|
|
2017-07-12 09:45:44 +00:00
|
|
|
describe "when it takes a long time for redis to set the lock", ->
|
|
|
|
beforeEach ->
|
|
|
|
@Profiler.prototype.end = () -> 7000 # take a long time
|
|
|
|
@Profiler.prototype.log = sinon.stub().returns { end: @Profiler.prototype.end }
|
|
|
|
@lockValue = "mock-lock-value"
|
|
|
|
@LockManager.randomLock = sinon.stub().returns @lockValue
|
|
|
|
@LockManager.releaseLock = sinon.stub().callsArgWith(2,null)
|
|
|
|
@set.callsArgWith(5, null, "OK")
|
|
|
|
|
|
|
|
describe "in all cases", ->
|
|
|
|
beforeEach ->
|
|
|
|
@LockManager.tryLock @doc_id, @callback
|
|
|
|
|
|
|
|
it "should set the lock key with an expiry if it is not set", ->
|
|
|
|
@set.calledWith("Blocking:#{@doc_id}", @lockValue, "EX", 30, "NX")
|
|
|
|
.should.equal true
|
|
|
|
|
|
|
|
it "should try to release the lock", ->
|
|
|
|
@LockManager.releaseLock.calledWith(@doc_id, @lockValue).should.equal true
|
|
|
|
|
|
|
|
describe "if the lock is released successfully", ->
|
|
|
|
beforeEach ->
|
|
|
|
@LockManager.releaseLock = sinon.stub().callsArgWith(2,null)
|
|
|
|
@LockManager.tryLock @doc_id, @callback
|
|
|
|
|
|
|
|
it "should return the callback with false", ->
|
|
|
|
@callback.calledWith(null, false).should.equal true
|
|
|
|
|
|
|
|
describe "if the lock has already timed out", ->
|
|
|
|
beforeEach ->
|
|
|
|
@LockManager.releaseLock = sinon.stub().callsArgWith(2, new Error("tried to release timed out lock"))
|
|
|
|
@LockManager.tryLock @doc_id, @callback
|
|
|
|
|
|
|
|
it "should return the callback with an error", ->
|
|
|
|
e = new Error("tried to release timed out lock")
|
|
|
|
@callback.calledWith(e).should.equal true
|