2014-11-13 16:03:37 +00:00
|
|
|
SandboxedModule = require('sandboxed-module')
|
|
|
|
sinon = require('sinon')
|
|
|
|
require('chai').should()
|
|
|
|
modulePath = require('path').join __dirname, '../../../app/js/WebsocketLoadBalancer'
|
|
|
|
|
|
|
|
describe "WebsocketLoadBalancer", ->
|
|
|
|
beforeEach ->
|
2017-05-02 14:51:17 +00:00
|
|
|
@rclient = {}
|
2019-07-19 07:49:57 +00:00
|
|
|
@RoomEvents = {on: sinon.stub()}
|
2014-11-13 16:03:37 +00:00
|
|
|
@WebsocketLoadBalancer = SandboxedModule.require modulePath, requires:
|
2019-07-09 11:20:59 +00:00
|
|
|
"./RedisClientManager":
|
|
|
|
createClientList: () => []
|
2015-12-01 11:05:49 +00:00
|
|
|
"logger-sharelatex": @logger = { log: sinon.stub(), error: sinon.stub() }
|
|
|
|
"./SafeJsonParse": @SafeJsonParse =
|
|
|
|
parse: (data, cb) => cb null, JSON.parse(data)
|
2019-03-19 10:55:12 +00:00
|
|
|
"./EventLogger": {checkEventOrder: sinon.stub()}
|
2019-04-15 13:05:26 +00:00
|
|
|
"./HealthCheckManager": {check: sinon.stub()}
|
2019-07-19 07:49:57 +00:00
|
|
|
"./RoomManager" : @RoomManager = {eventSource: sinon.stub().returns @RoomEvents}
|
|
|
|
"./ChannelManager": @ChannelManager = {publish: sinon.stub()}
|
2014-11-13 16:03:37 +00:00
|
|
|
@io = {}
|
2019-06-29 11:28:54 +00:00
|
|
|
@WebsocketLoadBalancer.rclientPubList = [{publish: sinon.stub()}]
|
|
|
|
@WebsocketLoadBalancer.rclientSubList = [{
|
2014-11-13 16:03:37 +00:00
|
|
|
subscribe: sinon.stub()
|
|
|
|
on: sinon.stub()
|
2019-06-29 11:28:54 +00:00
|
|
|
}]
|
|
|
|
|
2014-11-13 16:03:37 +00:00
|
|
|
@room_id = "room-id"
|
|
|
|
@message = "message-to-editor"
|
|
|
|
@payload = ["argument one", 42]
|
|
|
|
|
|
|
|
describe "emitToRoom", ->
|
|
|
|
beforeEach ->
|
|
|
|
@WebsocketLoadBalancer.emitToRoom(@room_id, @message, @payload...)
|
|
|
|
|
|
|
|
it "should publish the message to redis", ->
|
2019-07-19 07:49:57 +00:00
|
|
|
@ChannelManager.publish
|
|
|
|
.calledWith(@WebsocketLoadBalancer.rclientPubList[0], "editor-events", @room_id, JSON.stringify(
|
2014-11-13 16:03:37 +00:00
|
|
|
room_id: @room_id,
|
|
|
|
message: @message
|
|
|
|
payload: @payload
|
|
|
|
))
|
|
|
|
.should.equal true
|
|
|
|
|
|
|
|
describe "emitToAll", ->
|
|
|
|
beforeEach ->
|
|
|
|
@WebsocketLoadBalancer.emitToRoom = sinon.stub()
|
|
|
|
@WebsocketLoadBalancer.emitToAll @message, @payload...
|
|
|
|
|
|
|
|
it "should emit to the room 'all'", ->
|
|
|
|
@WebsocketLoadBalancer.emitToRoom
|
|
|
|
.calledWith("all", @message, @payload...)
|
|
|
|
.should.equal true
|
|
|
|
|
|
|
|
describe "listenForEditorEvents", ->
|
|
|
|
beforeEach ->
|
|
|
|
@WebsocketLoadBalancer._processEditorEvent = sinon.stub()
|
|
|
|
@WebsocketLoadBalancer.listenForEditorEvents()
|
|
|
|
|
|
|
|
it "should subscribe to the editor-events channel", ->
|
2019-06-29 11:28:54 +00:00
|
|
|
@WebsocketLoadBalancer.rclientSubList[0].subscribe
|
2014-11-13 16:03:37 +00:00
|
|
|
.calledWith("editor-events")
|
|
|
|
.should.equal true
|
|
|
|
|
|
|
|
it "should process the events with _processEditorEvent", ->
|
2019-06-29 11:28:54 +00:00
|
|
|
@WebsocketLoadBalancer.rclientSubList[0].on
|
2014-11-13 16:03:37 +00:00
|
|
|
.calledWith("message", sinon.match.func)
|
|
|
|
.should.equal true
|
|
|
|
|
|
|
|
describe "_processEditorEvent", ->
|
2015-12-01 11:05:49 +00:00
|
|
|
describe "with bad JSON", ->
|
|
|
|
beforeEach ->
|
|
|
|
@SafeJsonParse.parse = sinon.stub().callsArgWith 1, new Error("oops")
|
|
|
|
@WebsocketLoadBalancer._processEditorEvent(@io, "editor-events", "blah")
|
|
|
|
|
|
|
|
it "should log an error", ->
|
|
|
|
@logger.error.called.should.equal true
|
|
|
|
|
2014-11-13 16:03:37 +00:00
|
|
|
describe "with a designated room", ->
|
|
|
|
beforeEach ->
|
|
|
|
@io.sockets =
|
2019-04-09 13:48:00 +00:00
|
|
|
clients: sinon.stub().returns([
|
|
|
|
{id: 'client-id-1', emit: @emit1 = sinon.stub()}
|
|
|
|
{id: 'client-id-2', emit: @emit2 = sinon.stub()}
|
|
|
|
{id: 'client-id-1', emit: @emit3 = sinon.stub()} # duplicate client
|
|
|
|
])
|
2014-11-13 16:03:37 +00:00
|
|
|
data = JSON.stringify
|
|
|
|
room_id: @room_id
|
|
|
|
message: @message
|
|
|
|
payload: @payload
|
|
|
|
@WebsocketLoadBalancer._processEditorEvent(@io, "editor-events", data)
|
|
|
|
|
2019-04-09 13:48:00 +00:00
|
|
|
it "should send the message to all (unique) clients in the room", ->
|
|
|
|
@io.sockets.clients
|
2014-11-13 16:03:37 +00:00
|
|
|
.calledWith(@room_id)
|
|
|
|
.should.equal true
|
2019-04-09 13:48:00 +00:00
|
|
|
@emit1.calledWith(@message, @payload...).should.equal true
|
|
|
|
@emit2.calledWith(@message, @payload...).should.equal true
|
|
|
|
@emit3.called.should.equal false # duplicate client should be ignored
|
2014-11-13 16:03:37 +00:00
|
|
|
|
|
|
|
describe "when emitting to all", ->
|
|
|
|
beforeEach ->
|
|
|
|
@io.sockets =
|
|
|
|
emit: @emit = sinon.stub()
|
|
|
|
data = JSON.stringify
|
|
|
|
room_id: "all"
|
|
|
|
message: @message
|
|
|
|
payload: @payload
|
|
|
|
@WebsocketLoadBalancer._processEditorEvent(@io, "editor-events", data)
|
|
|
|
|
|
|
|
it "should send the message to all clients", ->
|
|
|
|
@emit.calledWith(@message, @payload...).should.equal true
|
|
|
|
|