chai = require('chai') should = chai.should() sinon = require("sinon") expect = chai.expect modulePath = "../../../app/js/WebsocketController.js" SandboxedModule = require('sandboxed-module') tk = require "timekeeper" describe 'WebsocketController', -> beforeEach -> tk.freeze(new Date()) @project_id = "project-id-123" @user = { _id: @user_id = "user-id-123" first_name: "James" last_name: "Allen" email: "james@example.com" signUpDate: new Date("2014-01-01") loginCount: 42 } @callback = sinon.stub() @client = disconnected: false id: @client_id = "mock-client-id-123" publicId: "other-id-#{Math.random()}" ol_context: {} join: sinon.stub() leave: sinon.stub() @WebsocketController = SandboxedModule.require modulePath, requires: "./WebApiManager": @WebApiManager = {} "./AuthorizationManager": @AuthorizationManager = {} "./DocumentUpdaterManager": @DocumentUpdaterManager = {} "./ConnectedUsersManager": @ConnectedUsersManager = {} "./WebsocketLoadBalancer": @WebsocketLoadBalancer = {} "logger-sharelatex": @logger = { log: sinon.stub(), error: sinon.stub(), warn: sinon.stub() } "metrics-sharelatex": @metrics = inc: sinon.stub() set: sinon.stub() "./RoomManager": @RoomManager = {} afterEach -> tk.reset() describe "joinProject", -> describe "when authorised", -> beforeEach -> @client.id = "mock-client-id" @project = { name: "Test Project" owner: { _id: @owner_id = "mock-owner-id-123" } } @privilegeLevel = "owner" @ConnectedUsersManager.updateUserPosition = sinon.stub().callsArg(4) @isRestrictedUser = true @WebApiManager.joinProject = sinon.stub().callsArgWith(2, null, @project, @privilegeLevel, @isRestrictedUser) @RoomManager.joinProject = sinon.stub().callsArg(2) @WebsocketController.joinProject @client, @user, @project_id, @callback it "should load the project from web", -> @WebApiManager.joinProject .calledWith(@project_id, @user) .should.equal true it "should join the project room", -> @RoomManager.joinProject.calledWith(@client, @project_id).should.equal true it "should set the privilege level on the client", -> @client.ol_context["privilege_level"].should.equal @privilegeLevel it "should set the user's id on the client", -> @client.ol_context["user_id"].should.equal @user._id it "should set the user's email on the client", -> @client.ol_context["email"].should.equal @user.email it "should set the user's first_name on the client", -> @client.ol_context["first_name"].should.equal @user.first_name it "should set the user's last_name on the client", -> @client.ol_context["last_name"].should.equal @user.last_name it "should set the user's sign up date on the client", -> @client.ol_context["signup_date"].should.equal @user.signUpDate it "should set the user's login_count on the client", -> @client.ol_context["login_count"].should.equal @user.loginCount it "should set the connected time on the client", -> @client.ol_context["connected_time"].should.equal new Date() it "should set the project_id on the client", -> @client.ol_context["project_id"].should.equal @project_id it "should set the project owner id on the client", -> @client.ol_context["owner_id"].should.equal @owner_id it "should set the is_restricted_user flag on the client", -> @client.ol_context["is_restricted_user"].should.equal @isRestrictedUser it "should call the callback with the project, privilegeLevel and protocolVersion", -> @callback .calledWith(null, @project, @privilegeLevel, @WebsocketController.PROTOCOL_VERSION) .should.equal true it "should mark the user as connected in ConnectedUsersManager", -> @ConnectedUsersManager.updateUserPosition .calledWith(@project_id, @client.publicId, @user, null) .should.equal true it "should increment the join-project metric", -> @metrics.inc.calledWith("editor.join-project").should.equal true describe "when not authorized", -> beforeEach -> @WebApiManager.joinProject = sinon.stub().callsArgWith(2, null, null, null) @WebsocketController.joinProject @client, @user, @project_id, @callback it "should return an error", -> @callback .calledWith(sinon.match({message: "not authorized"})) .should.equal true it "should not log an error", -> @logger.error.called.should.equal false describe "when the subscribe failed", -> beforeEach -> @client.id = "mock-client-id" @project = { name: "Test Project" owner: { _id: @owner_id = "mock-owner-id-123" } } @privilegeLevel = "owner" @ConnectedUsersManager.updateUserPosition = sinon.stub().callsArg(4) @isRestrictedUser = true @WebApiManager.joinProject = sinon.stub().callsArgWith(2, null, @project, @privilegeLevel, @isRestrictedUser) @RoomManager.joinProject = sinon.stub().callsArgWith(2, new Error("subscribe failed")) @WebsocketController.joinProject @client, @user, @project_id, @callback it "should return an error", -> @callback .calledWith(sinon.match({message: "subscribe failed"})) .should.equal true @callback.args[0][0].message.should.equal "subscribe failed" describe "when the client has disconnected", -> beforeEach -> @client.disconnected = true @WebApiManager.joinProject = sinon.stub().callsArg(2) @WebsocketController.joinProject @client, @user, @project_id, @callback it "should not call WebApiManager.joinProject", -> expect(@WebApiManager.joinProject.called).to.equal(false) it "should call the callback with no details", -> expect(@callback.args[0]).to.deep.equal [] it "should increment the editor.join-project.disconnected metric with a status", -> expect(@metrics.inc.calledWith('editor.join-project.disconnected', 1, {status: 'immediately'})).to.equal(true) describe "when the client disconnects while WebApiManager.joinProject is running", -> beforeEach -> @WebApiManager.joinProject = (project, user, cb) => @client.disconnected = true cb(null, @project, @privilegeLevel, @isRestrictedUser) @WebsocketController.joinProject @client, @user, @project_id, @callback it "should call the callback with no details", -> expect(@callback.args[0]).to.deep.equal [] it "should increment the editor.join-project.disconnected metric with a status", -> expect(@metrics.inc.calledWith('editor.join-project.disconnected', 1, {status: 'after-web-api-call'})).to.equal(true) describe "leaveProject", -> beforeEach -> @DocumentUpdaterManager.flushProjectToMongoAndDelete = sinon.stub().callsArg(1) @ConnectedUsersManager.markUserAsDisconnected = sinon.stub().callsArg(2) @WebsocketLoadBalancer.emitToRoom = sinon.stub() @RoomManager.leaveProjectAndDocs = sinon.stub() @clientsInRoom = [] @io = sockets: clients: (room_id) => if room_id != @project_id throw "expected room_id to be project_id" return @clientsInRoom @client.ol_context.project_id = @project_id @client.ol_context.user_id = @user_id @WebsocketController.FLUSH_IF_EMPTY_DELAY = 0 tk.reset() # Allow setTimeout to work. describe "when the client did not joined a project yet", -> beforeEach (done) -> @client.ol_context = {} @WebsocketController.leaveProject @io, @client, done it "should bail out when calling leaveProject", () -> @WebsocketLoadBalancer.emitToRoom.called.should.equal false @RoomManager.leaveProjectAndDocs.called.should.equal false @ConnectedUsersManager.markUserAsDisconnected.called.should.equal false it "should not inc any metric", () -> @metrics.inc.called.should.equal false describe "when the project is empty", -> beforeEach (done) -> @clientsInRoom = [] @WebsocketController.leaveProject @io, @client, done it "should end clientTracking.clientDisconnected to the project room", -> @WebsocketLoadBalancer.emitToRoom .calledWith(@project_id, "clientTracking.clientDisconnected", @client.publicId) .should.equal true it "should mark the user as disconnected", -> @ConnectedUsersManager.markUserAsDisconnected .calledWith(@project_id, @client.publicId) .should.equal true it "should flush the project in the document updater", -> @DocumentUpdaterManager.flushProjectToMongoAndDelete .calledWith(@project_id) .should.equal true it "should increment the leave-project metric", -> @metrics.inc.calledWith("editor.leave-project").should.equal true it "should track the disconnection in RoomManager", -> @RoomManager.leaveProjectAndDocs .calledWith(@client) .should.equal true describe "when the project is not empty", -> beforeEach -> @clientsInRoom = ["mock-remaining-client"] @WebsocketController.leaveProject @io, @client it "should not flush the project in the document updater", -> @DocumentUpdaterManager.flushProjectToMongoAndDelete .called.should.equal false describe "when client has not authenticated", -> beforeEach (done) -> @client.ol_context.user_id = null @client.ol_context.project_id = null @WebsocketController.leaveProject @io, @client, done it "should not end clientTracking.clientDisconnected to the project room", -> @WebsocketLoadBalancer.emitToRoom .calledWith(@project_id, "clientTracking.clientDisconnected", @client.publicId) .should.equal false it "should not mark the user as disconnected", -> @ConnectedUsersManager.markUserAsDisconnected .calledWith(@project_id, @client.publicId) .should.equal false it "should not flush the project in the document updater", -> @DocumentUpdaterManager.flushProjectToMongoAndDelete .calledWith(@project_id) .should.equal false it "should not increment the leave-project metric", -> @metrics.inc.calledWith("editor.leave-project").should.equal false describe "when client has not joined a project", -> beforeEach (done) -> @client.ol_context.user_id = @user_id @client.ol_context.project_id = null @WebsocketController.leaveProject @io, @client, done it "should not end clientTracking.clientDisconnected to the project room", -> @WebsocketLoadBalancer.emitToRoom .calledWith(@project_id, "clientTracking.clientDisconnected", @client.publicId) .should.equal false it "should not mark the user as disconnected", -> @ConnectedUsersManager.markUserAsDisconnected .calledWith(@project_id, @client.publicId) .should.equal false it "should not flush the project in the document updater", -> @DocumentUpdaterManager.flushProjectToMongoAndDelete .calledWith(@project_id) .should.equal false it "should not increment the leave-project metric", -> @metrics.inc.calledWith("editor.leave-project").should.equal false describe "joinDoc", -> beforeEach -> @doc_id = "doc-id-123" @doc_lines = ["doc", "lines"] @version = 42 @ops = ["mock", "ops"] @ranges = { "mock": "ranges" } @options = {} @client.ol_context.project_id = @project_id @client.ol_context.is_restricted_user = false @AuthorizationManager.addAccessToDoc = sinon.stub() @AuthorizationManager.assertClientCanViewProject = sinon.stub().callsArgWith(1, null) @DocumentUpdaterManager.getDocument = sinon.stub().callsArgWith(3, null, @doc_lines, @version, @ranges, @ops) @RoomManager.joinDoc = sinon.stub().callsArg(2) describe "works", -> beforeEach -> @WebsocketController.joinDoc @client, @doc_id, -1, @options, @callback it "should check that the client is authorized to view the project", -> @AuthorizationManager.assertClientCanViewProject .calledWith(@client) .should.equal true it "should get the document from the DocumentUpdaterManager with fromVersion", -> @DocumentUpdaterManager.getDocument .calledWith(@project_id, @doc_id, -1) .should.equal true it "should add permissions for the client to access the doc", -> @AuthorizationManager.addAccessToDoc .calledWith(@client, @doc_id) .should.equal true it "should join the client to room for the doc_id", -> @RoomManager.joinDoc .calledWith(@client, @doc_id) .should.equal true it "should call the callback with the lines, version, ranges and ops", -> @callback .calledWith(null, @doc_lines, @version, @ops, @ranges) .should.equal true it "should increment the join-doc metric", -> @metrics.inc.calledWith("editor.join-doc").should.equal true describe "with a fromVersion", -> beforeEach -> @fromVersion = 40 @WebsocketController.joinDoc @client, @doc_id, @fromVersion, @options, @callback it "should get the document from the DocumentUpdaterManager with fromVersion", -> @DocumentUpdaterManager.getDocument .calledWith(@project_id, @doc_id, @fromVersion) .should.equal true describe "with doclines that need escaping", -> beforeEach -> @doc_lines.push ["räksmörgås"] @WebsocketController.joinDoc @client, @doc_id, -1, @options, @callback it "should call the callback with the escaped lines", -> escaped_lines = @callback.args[0][1] escaped_word = escaped_lines.pop() escaped_word.should.equal 'räksmörgÃ¥s' # Check that unescaping works decodeURIComponent(escape(escaped_word)).should.equal "räksmörgås" describe "with comments that need encoding", -> beforeEach -> @ranges.comments = [{ op: { c: "räksmörgås" } }] @WebsocketController.joinDoc @client, @doc_id, -1, { encodeRanges: true }, @callback it "should call the callback with the encoded comment", -> encoded_comments = @callback.args[0][4] encoded_comment = encoded_comments.comments.pop() encoded_comment_text = encoded_comment.op.c encoded_comment_text.should.equal 'räksmörgÃ¥s' describe "with changes that need encoding", -> it "should call the callback with the encoded insert change", -> @ranges.changes = [{ op: { i: "räksmörgås" } }] @WebsocketController.joinDoc @client, @doc_id, -1, { encodeRanges: true }, @callback encoded_changes = @callback.args[0][4] encoded_change = encoded_changes.changes.pop() encoded_change_text = encoded_change.op.i encoded_change_text.should.equal 'räksmörgÃ¥s' it "should call the callback with the encoded delete change", -> @ranges.changes = [{ op: { d: "räksmörgås" } }] @WebsocketController.joinDoc @client, @doc_id, -1, { encodeRanges: true }, @callback encoded_changes = @callback.args[0][4] encoded_change = encoded_changes.changes.pop() encoded_change_text = encoded_change.op.d encoded_change_text.should.equal 'räksmörgÃ¥s' describe "when not authorized", -> beforeEach -> @AuthorizationManager.assertClientCanViewProject = sinon.stub().callsArgWith(1, @err = new Error("not authorized")) @WebsocketController.joinDoc @client, @doc_id, -1, @options, @callback it "should call the callback with an error", -> @callback.calledWith(sinon.match({message: "not authorized"})).should.equal true it "should not call the DocumentUpdaterManager", -> @DocumentUpdaterManager.getDocument.called.should.equal false describe "with a restricted client", -> beforeEach -> @ranges.comments = [{op: {a: 1}}, {op: {a: 2}}] @client.ol_context.is_restricted_user = true @WebsocketController.joinDoc @client, @doc_id, -1, @options, @callback it "should overwrite ranges.comments with an empty list", -> ranges = @callback.args[0][4] expect(ranges.comments).to.deep.equal [] describe "when the client has disconnected", -> beforeEach -> @client.disconnected = true @WebsocketController.joinDoc @client, @doc_id, -1, @options, @callback it "should call the callback with no details", -> expect(@callback.args[0]).to.deep.equal([]) it "should increment the editor.join-doc.disconnected metric with a status", -> expect(@metrics.inc.calledWith('editor.join-doc.disconnected', 1, {status: 'immediately'})).to.equal(true) it "should not get the document", -> expect(@DocumentUpdaterManager.getDocument.called).to.equal(false) describe "when the client disconnects while RoomManager.joinDoc is running", -> beforeEach -> @RoomManager.joinDoc = (client, doc_id, cb) => @client.disconnected = true cb() @WebsocketController.joinDoc @client, @doc_id, -1, @options, @callback it "should call the callback with no details", -> expect(@callback.args[0]).to.deep.equal([]) it "should increment the editor.join-doc.disconnected metric with a status", -> expect(@metrics.inc.calledWith('editor.join-doc.disconnected', 1, {status: 'after-joining-room'})).to.equal(true) it "should not get the document", -> expect(@DocumentUpdaterManager.getDocument.called).to.equal(false) describe "when the client disconnects while DocumentUpdaterManager.getDocument is running", -> beforeEach -> @DocumentUpdaterManager.getDocument = (project_id, doc_id, fromVersion, callback) => @client.disconnected = true callback(null, @doc_lines, @version, @ranges, @ops) @WebsocketController.joinDoc @client, @doc_id, -1, @options, @callback it "should call the callback with no details", -> expect(@callback.args[0]).to.deep.equal [] it "should increment the editor.join-doc.disconnected metric with a status", -> expect(@metrics.inc.calledWith('editor.join-doc.disconnected', 1, {status: 'after-doc-updater-call'})).to.equal(true) describe "leaveDoc", -> beforeEach -> @doc_id = "doc-id-123" @client.ol_context.project_id = @project_id @RoomManager.leaveDoc = sinon.stub() @WebsocketController.leaveDoc @client, @doc_id, @callback it "should remove the client from the doc_id room", -> @RoomManager.leaveDoc .calledWith(@client, @doc_id).should.equal true it "should call the callback", -> @callback.called.should.equal true it "should increment the leave-doc metric", -> @metrics.inc.calledWith("editor.leave-doc").should.equal true describe "getConnectedUsers", -> beforeEach -> @client.ol_context.project_id = @project_id @users = ["mock", "users"] @WebsocketLoadBalancer.emitToRoom = sinon.stub() @ConnectedUsersManager.getConnectedUsers = sinon.stub().callsArgWith(1, null, @users) describe "when authorized", -> beforeEach (done) -> @AuthorizationManager.assertClientCanViewProject = sinon.stub().callsArgWith(1, null) @WebsocketController.getConnectedUsers @client, (args...) => @callback(args...) done() it "should check that the client is authorized to view the project", -> @AuthorizationManager.assertClientCanViewProject .calledWith(@client) .should.equal true it "should broadcast a request to update the client list", -> @WebsocketLoadBalancer.emitToRoom .calledWith(@project_id, "clientTracking.refresh") .should.equal true it "should get the connected users for the project", -> @ConnectedUsersManager.getConnectedUsers .calledWith(@project_id) .should.equal true it "should return the users", -> @callback.calledWith(null, @users).should.equal true it "should increment the get-connected-users metric", -> @metrics.inc.calledWith("editor.get-connected-users").should.equal true describe "when not authorized", -> beforeEach -> @AuthorizationManager.assertClientCanViewProject = sinon.stub().callsArgWith(1, @err = new Error("not authorized")) @WebsocketController.getConnectedUsers @client, @callback it "should not get the connected users for the project", -> @ConnectedUsersManager.getConnectedUsers .called .should.equal false it "should return an error", -> @callback.calledWith(@err).should.equal true describe "when restricted user", -> beforeEach -> @client.ol_context.is_restricted_user = true @AuthorizationManager.assertClientCanViewProject = sinon.stub().callsArgWith(1, null) @WebsocketController.getConnectedUsers @client, @callback it "should return an empty array of users", -> @callback.calledWith(null, []).should.equal true it "should not get the connected users for the project", -> @ConnectedUsersManager.getConnectedUsers .called .should.equal false describe "when the client has disconnected", -> beforeEach -> @client.disconnected = true @AuthorizationManager.assertClientCanViewProject = sinon.stub() @WebsocketController.getConnectedUsers @client, @callback it "should call the callback with no details", -> expect(@callback.args[0]).to.deep.equal([]) it "should not check permissions", -> expect(@AuthorizationManager.assertClientCanViewProject.called).to.equal(false) describe "updateClientPosition", -> beforeEach -> @WebsocketLoadBalancer.emitToRoom = sinon.stub() @ConnectedUsersManager.updateUserPosition = sinon.stub().callsArgWith(4) @AuthorizationManager.assertClientCanViewProjectAndDoc = sinon.stub().callsArgWith(2, null) @update = { doc_id: @doc_id = "doc-id-123" row: @row = 42 column: @column = 37 } describe "with a logged in user", -> beforeEach -> @client.ol_context = { project_id: @project_id first_name: @first_name = "Douglas" last_name: @last_name = "Adams" email: @email = "joe@example.com" user_id: @user_id = "user-id-123" } @WebsocketController.updateClientPosition @client, @update @populatedCursorData = doc_id: @doc_id, id: @client.publicId name: "#{@first_name} #{@last_name}" row: @row column: @column email: @email user_id: @user_id it "should send the update to the project room with the user's name", -> @WebsocketLoadBalancer.emitToRoom.calledWith(@project_id, "clientTracking.clientUpdated", @populatedCursorData).should.equal true it "should send the cursor data to the connected user manager", (done)-> @ConnectedUsersManager.updateUserPosition.calledWith(@project_id, @client.publicId, { _id: @user_id, email: @email, first_name: @first_name, last_name: @last_name }, { row: @row column: @column doc_id: @doc_id }).should.equal true done() it "should increment the update-client-position metric at 0.1 frequency", -> @metrics.inc.calledWith("editor.update-client-position", 0.1).should.equal true describe "with a logged in user who has no last_name set", -> beforeEach -> @client.ol_context = { project_id: @project_id first_name: @first_name = "Douglas" last_name: undefined email: @email = "joe@example.com" user_id: @user_id = "user-id-123" } @WebsocketController.updateClientPosition @client, @update @populatedCursorData = doc_id: @doc_id, id: @client.publicId name: "#{@first_name}" row: @row column: @column email: @email user_id: @user_id it "should send the update to the project room with the user's name", -> @WebsocketLoadBalancer.emitToRoom.calledWith(@project_id, "clientTracking.clientUpdated", @populatedCursorData).should.equal true it "should send the cursor data to the connected user manager", (done)-> @ConnectedUsersManager.updateUserPosition.calledWith(@project_id, @client.publicId, { _id: @user_id, email: @email, first_name: @first_name, last_name: undefined }, { row: @row column: @column doc_id: @doc_id }).should.equal true done() it "should increment the update-client-position metric at 0.1 frequency", -> @metrics.inc.calledWith("editor.update-client-position", 0.1).should.equal true describe "with a logged in user who has no first_name set", -> beforeEach -> @client.ol_context = { project_id: @project_id first_name: undefined last_name: @last_name = "Adams" email: @email = "joe@example.com" user_id: @user_id = "user-id-123" } @WebsocketController.updateClientPosition @client, @update @populatedCursorData = doc_id: @doc_id, id: @client.publicId name: "#{@last_name}" row: @row column: @column email: @email user_id: @user_id it "should send the update to the project room with the user's name", -> @WebsocketLoadBalancer.emitToRoom.calledWith(@project_id, "clientTracking.clientUpdated", @populatedCursorData).should.equal true it "should send the cursor data to the connected user manager", (done)-> @ConnectedUsersManager.updateUserPosition.calledWith(@project_id, @client.publicId, { _id: @user_id, email: @email, first_name: undefined, last_name: @last_name }, { row: @row column: @column doc_id: @doc_id }).should.equal true done() it "should increment the update-client-position metric at 0.1 frequency", -> @metrics.inc.calledWith("editor.update-client-position", 0.1).should.equal true describe "with a logged in user who has no names set", -> beforeEach -> @client.ol_context = { project_id: @project_id first_name: undefined last_name: undefined email: @email = "joe@example.com" user_id: @user_id = "user-id-123" } @WebsocketController.updateClientPosition @client, @update it "should send the update to the project name with no name", -> @WebsocketLoadBalancer.emitToRoom .calledWith(@project_id, "clientTracking.clientUpdated", { doc_id: @doc_id, id: @client.publicId, user_id: @user_id, name: "", row: @row, column: @column, email: @email }) .should.equal true describe "with an anonymous user", -> beforeEach -> @client.ol_context = { project_id: @project_id } @WebsocketController.updateClientPosition @client, @update it "should send the update to the project room with no name", -> @WebsocketLoadBalancer.emitToRoom .calledWith(@project_id, "clientTracking.clientUpdated", { doc_id: @doc_id, id: @client.publicId name: "" row: @row column: @column }) .should.equal true it "should not send cursor data to the connected user manager", (done)-> @ConnectedUsersManager.updateUserPosition.called.should.equal false done() describe "when the client has disconnected", -> beforeEach -> @client.disconnected = true @AuthorizationManager.assertClientCanViewProjectAndDoc = sinon.stub() @WebsocketController.updateClientPosition @client, @update, @callback it "should call the callback with no details", -> expect(@callback.args[0]).to.deep.equal([]) it "should not check permissions", -> expect(@AuthorizationManager.assertClientCanViewProjectAndDoc.called).to.equal(false) describe "applyOtUpdate", -> beforeEach -> @update = {op: {p: 12, t: "foo"}} @client.ol_context.user_id = @user_id @client.ol_context.project_id = @project_id @WebsocketController._assertClientCanApplyUpdate = sinon.stub().yields() @DocumentUpdaterManager.queueChange = sinon.stub().callsArg(3) describe "succesfully", -> beforeEach -> @WebsocketController.applyOtUpdate @client, @doc_id, @update, @callback it "should set the source of the update to the client id", -> @update.meta.source.should.equal @client.publicId it "should set the user_id of the update to the user id", -> @update.meta.user_id.should.equal @user_id it "should queue the update", -> @DocumentUpdaterManager.queueChange .calledWith(@project_id, @doc_id, @update) .should.equal true it "should call the callback", -> @callback.called.should.equal true it "should increment the doc updates", -> @metrics.inc.calledWith("editor.doc-update").should.equal true describe "unsuccessfully", -> beforeEach -> @client.disconnect = sinon.stub() @DocumentUpdaterManager.queueChange = sinon.stub().callsArgWith(3, @error = new Error("Something went wrong")) @WebsocketController.applyOtUpdate @client, @doc_id, @update, @callback it "should disconnect the client", -> @client.disconnect.called.should.equal true it "should log an error", -> @logger.error.called.should.equal true it "should call the callback with the error", -> @callback.calledWith(@error).should.equal true describe "when not authorized", -> beforeEach -> @client.disconnect = sinon.stub() @WebsocketController._assertClientCanApplyUpdate = sinon.stub().yields(@error = new Error("not authorized")) @WebsocketController.applyOtUpdate @client, @doc_id, @update, @callback # This happens in a setTimeout to allow the client a chance to receive the error first. # I'm not sure how to unit test, but it is acceptance tested. # it "should disconnect the client", -> # @client.disconnect.called.should.equal true it "should log a warning", -> @logger.warn.called.should.equal true it "should call the callback with the error", -> @callback.calledWith(@error).should.equal true describe "update_too_large", -> beforeEach (done) -> @client.disconnect = sinon.stub() @client.emit = sinon.stub() @client.ol_context.user_id = @user_id @client.ol_context.project_id = @project_id error = new Error("update is too large") error.updateSize = 7372835 @DocumentUpdaterManager.queueChange = sinon.stub().callsArgWith(3, error) @WebsocketController.applyOtUpdate @client, @doc_id, @update, @callback setTimeout -> done() , 1 it "should call the callback with no error", -> @callback.called.should.equal true @callback.args[0].should.deep.equal [] it "should log a warning with the size and context", -> @logger.warn.called.should.equal true @logger.warn.args[0].should.deep.equal [{ @user_id, @project_id, @doc_id, updateSize: 7372835 }, 'update is too large'] describe "after 100ms", -> beforeEach (done) -> setTimeout done, 100 it "should send an otUpdateError the client", -> @client.emit.calledWith('otUpdateError').should.equal true it "should disconnect the client", -> @client.disconnect.called.should.equal true describe "when the client disconnects during the next 100ms", -> beforeEach (done) -> @client.disconnected = true setTimeout done, 100 it "should not send an otUpdateError the client", -> @client.emit.calledWith('otUpdateError').should.equal false it "should not disconnect the client", -> @client.disconnect.called.should.equal false it "should increment the editor.doc-update.disconnected metric with a status", -> expect(@metrics.inc.calledWith('editor.doc-update.disconnected', 1, {status:'at-otUpdateError'})).to.equal(true) describe "_assertClientCanApplyUpdate", -> beforeEach -> @edit_update = { op: [{i: "foo", p: 42}, {c: "bar", p: 132}] } # comments may still be in an edit op @comment_update = { op: [{c: "bar", p: 132}] } @AuthorizationManager.assertClientCanEditProjectAndDoc = sinon.stub() @AuthorizationManager.assertClientCanViewProjectAndDoc = sinon.stub() describe "with a read-write client", -> it "should return successfully", (done) -> @AuthorizationManager.assertClientCanEditProjectAndDoc.yields(null) @WebsocketController._assertClientCanApplyUpdate @client, @doc_id, @edit_update, (error) -> expect(error).to.be.null done() describe "with a read-only client and an edit op", -> it "should return an error", (done) -> @AuthorizationManager.assertClientCanEditProjectAndDoc.yields(new Error("not authorized")) @AuthorizationManager.assertClientCanViewProjectAndDoc.yields(null) @WebsocketController._assertClientCanApplyUpdate @client, @doc_id, @edit_update, (error) -> expect(error.message).to.equal "not authorized" done() describe "with a read-only client and a comment op", -> it "should return successfully", (done) -> @AuthorizationManager.assertClientCanEditProjectAndDoc.yields(new Error("not authorized")) @AuthorizationManager.assertClientCanViewProjectAndDoc.yields(null) @WebsocketController._assertClientCanApplyUpdate @client, @doc_id, @comment_update, (error) -> expect(error).to.be.null done() describe "with a totally unauthorized client", -> it "should return an error", (done) -> @AuthorizationManager.assertClientCanEditProjectAndDoc.yields(new Error("not authorized")) @AuthorizationManager.assertClientCanViewProjectAndDoc.yields(new Error("not authorized")) @WebsocketController._assertClientCanApplyUpdate @client, @doc_id, @comment_update, (error) -> expect(error.message).to.equal "not authorized" done()