overleaf/services/web/test/UnitTests/coffee/Project/ProjectControllerTests.coffee

467 lines
16 KiB
CoffeeScript
Raw Normal View History

should = require('chai').should()
SandboxedModule = require('sandboxed-module')
assert = require('assert')
path = require('path')
sinon = require('sinon')
modulePath = path.join __dirname, "../../../../app/js/Features/Project/ProjectController"
expect = require("chai").expect
describe "ProjectController", ->
beforeEach ->
@project_id = "123213jlkj9kdlsaj"
2016-09-07 11:40:49 -04:00
@user =
_id:"588f3ddae8ebc1bac07c9fa4"
2016-09-07 11:40:49 -04:00
first_name: "bjkdsjfk"
@settings =
apis:
chat:
url:"chat.com"
siteUrl: "mysite.com"
@token = 'some-token'
2016-09-07 11:40:49 -04:00
@ProjectDeleter =
2014-06-04 07:45:04 -04:00
archiveProject: sinon.stub().callsArg(1)
deleteProject: sinon.stub().callsArg(1)
restoreProject: sinon.stub().callsArg(1)
findArchivedProjects: sinon.stub()
@ProjectDuplicator =
duplicate: sinon.stub().callsArgWith(3, null, {_id:@project_id})
@ProjectCreationHandler =
createExampleProject: sinon.stub().callsArgWith(2, null, {_id:@project_id})
createBasicProject: sinon.stub().callsArgWith(2, null, {_id:@project_id})
@SubscriptionLocator =
getUsersSubscription: sinon.stub()
2016-09-07 11:40:49 -04:00
@LimitationsManager =
userHasSubscriptionOrIsGroupMember: sinon.stub()
@TagsHandler =
getAllTags: sinon.stub()
2016-01-22 00:41:22 -05:00
@NotificationsHandler =
getUserNotifications: sinon.stub()
@UserModel =
findById: sinon.stub()
@AuthorizationManager =
getPrivilegeLevelForProject:sinon.stub()
2016-09-07 11:40:49 -04:00
@EditorController =
2014-04-28 12:47:47 -04:00
renameProject:sinon.stub()
@InactiveProjectManager =
reactivateProjectIfRequired:sinon.stub()
@ProjectUpdateHandler =
markAsOpened: sinon.stub()
@ReferencesSearchHandler =
indexProjectReferences: sinon.stub()
@ProjectGetter =
findAllUsersProjects: sinon.stub()
getProject: sinon.stub()
2016-09-07 11:40:49 -04:00
@AuthenticationController =
getLoggedInUser: sinon.stub().callsArgWith(1, null, @user)
getLoggedInUserId: sinon.stub().returns(@user._id)
getSessionUser: sinon.stub().returns(@user)
isUserLoggedIn: sinon.stub().returns(true)
@AnalyticsManager =
getLastOccurance: sinon.stub()
@TokenAccessHandler =
getRequestToken: sinon.stub().returns(@token)
protectTokens: sinon.stub()
@ProjectController = SandboxedModule.require modulePath, requires:
"settings-sharelatex":@settings
2016-09-07 11:40:49 -04:00
"logger-sharelatex":
2014-04-28 12:47:47 -04:00
log:->
err:->
"metrics-sharelatex":
Timer:->
done:->
inc:->
"./ProjectDeleter": @ProjectDeleter
"./ProjectDuplicator": @ProjectDuplicator
"./ProjectCreationHandler": @ProjectCreationHandler
2014-04-28 12:47:47 -04:00
"../Editor/EditorController": @EditorController
"../Subscription/SubscriptionLocator": @SubscriptionLocator
"../Subscription/LimitationsManager": @LimitationsManager
"../Tags/TagsHandler":@TagsHandler
2016-01-22 00:41:22 -05:00
"../Notifications/NotificationsHandler":@NotificationsHandler
"../../models/User":User:@UserModel
"../Authorization/AuthorizationManager":@AuthorizationManager
"../InactiveData/InactiveProjectManager":@InactiveProjectManager
"./ProjectUpdateHandler":@ProjectUpdateHandler
"../ReferencesSearch/ReferencesSearchHandler": @ReferencesSearchHandler
"./ProjectGetter": @ProjectGetter
2016-09-07 11:40:49 -04:00
'../Authentication/AuthenticationController': @AuthenticationController
"../Analytics/AnalyticsManager": @AnalyticsManager
"../TokenAccess/TokenAccessHandler": @TokenAccessHandler
@projectName = "£12321jkj9ujkljds"
2016-09-07 11:40:49 -04:00
@req =
params:
Project_id: @project_id
session:
user: @user
body:
2016-09-07 11:40:49 -04:00
projectName: @projectName
2016-01-22 15:08:39 -05:00
i18n:
translate:->
2016-09-07 11:40:49 -04:00
@res =
locals:
jsPath:"js path here"
2014-06-25 08:51:02 -04:00
describe "updateProjectSettings", ->
it "should update the name", (done) ->
@EditorController.renameProject = sinon.stub().callsArg(2)
@req.body =
name: @name = "New name"
2015-07-08 11:56:38 -04:00
@res.sendStatus = (code) =>
2014-06-25 08:51:02 -04:00
@EditorController.renameProject
.calledWith(@project_id, @name)
.should.equal true
code.should.equal 204
done()
@ProjectController.updateProjectSettings @req, @res
it "should update the compiler", (done) ->
@EditorController.setCompiler = sinon.stub().callsArg(2)
@req.body =
compiler: @compiler = "pdflatex"
2015-07-08 11:56:38 -04:00
@res.sendStatus = (code) =>
2014-06-25 08:51:02 -04:00
@EditorController.setCompiler
.calledWith(@project_id, @compiler)
.should.equal true
code.should.equal 204
done()
@ProjectController.updateProjectSettings @req, @res
it "should update the spell check language", (done) ->
@EditorController.setSpellCheckLanguage = sinon.stub().callsArg(2)
@req.body =
spellCheckLanguage: @languageCode = "fr"
2015-07-08 11:56:38 -04:00
@res.sendStatus = (code) =>
2014-06-25 08:51:02 -04:00
@EditorController.setSpellCheckLanguage
.calledWith(@project_id, @languageCode)
.should.equal true
code.should.equal 204
done()
@ProjectController.updateProjectSettings @req, @res
it "should update the root doc", (done) ->
@EditorController.setRootDoc = sinon.stub().callsArg(2)
@req.body =
rootDocId: @rootDocId = "root-doc-id"
2015-07-08 11:56:38 -04:00
@res.sendStatus = (code) =>
2014-06-25 08:51:02 -04:00
@EditorController.setRootDoc
.calledWith(@project_id, @rootDocId)
.should.equal true
code.should.equal 204
done()
@ProjectController.updateProjectSettings @req, @res
2016-09-07 11:40:49 -04:00
describe "updateProjectAdminSettings", ->
it "should update the public access level", (done) ->
@EditorController.setPublicAccessLevel = sinon.stub().callsArg(2)
@req.body =
publicAccessLevel: @publicAccessLevel = "readonly"
@res.sendStatus = (code) =>
@EditorController.setPublicAccessLevel
.calledWith(@project_id, @publicAccessLevel)
.should.equal true
code.should.equal 204
done()
@ProjectController.updateProjectAdminSettings @req, @res
2014-06-25 08:51:02 -04:00
describe "deleteProject", ->
2014-06-04 07:45:04 -04:00
it "should tell the project deleter to archive when forever=false", (done)->
2015-07-08 11:56:38 -04:00
@res.sendStatus = (code)=>
@ProjectDeleter.archiveProject.calledWith(@project_id).should.equal true
code.should.equal 200
done()
@ProjectController.deleteProject @req, @res
2014-06-04 07:45:04 -04:00
it "should tell the project deleter to delete when forever=true", (done)->
@req.query = forever: "true"
2015-07-08 11:56:38 -04:00
@res.sendStatus = (code)=>
2014-06-04 07:45:04 -04:00
@ProjectDeleter.deleteProject.calledWith(@project_id).should.equal true
code.should.equal 200
done()
@ProjectController.deleteProject @req, @res
describe "restoreProject", ->
it "should tell the project deleter", (done)->
2015-07-08 11:56:38 -04:00
@res.sendStatus = (code)=>
@ProjectDeleter.restoreProject.calledWith(@project_id).should.equal true
code.should.equal 200
done()
@ProjectController.restoreProject @req, @res
describe "cloneProject", ->
2016-09-07 11:40:49 -04:00
it "should call the project duplicator", (done)->
@res.send = (json)=>
@ProjectDuplicator.duplicate.calledWith(@user, @project_id, @projectName).should.equal true
json.project_id.should.equal @project_id
done()
@ProjectController.cloneProject @req, @res
describe "newProject", ->
it "should call the projectCreationHandler with createExampleProject", (done)->
@req.body.template = "example"
@res.send = (json)=>
@ProjectCreationHandler.createExampleProject.calledWith(@user._id, @projectName).should.equal true
@ProjectCreationHandler.createBasicProject.called.should.equal false
done()
@ProjectController.newProject @req, @res
it "should call the projectCreationHandler with createBasicProject", (done)->
@req.body.template = "basic"
@res.send = (json)=>
@ProjectCreationHandler.createExampleProject.called.should.equal false
@ProjectCreationHandler.createBasicProject.calledWith(@user._id, @projectName).should.equal true
done()
@ProjectController.newProject @req, @res
describe "projectListPage", ->
beforeEach ->
@tags = [{name:1, project_ids:["1","2","3"]}, {name:2, project_ids:["a","1"]}, {name:3, project_ids:["a", "b", "c", "d"]}]
2016-01-22 00:41:22 -05:00
@notifications = [{_id:'1',user_id:'2',templateKey:'3',messageOpts:'4',key:'5'}]
@projects = [
{_id:1, lastUpdated:1, owner_ref: "user-1"},
{_id:2, lastUpdated:2, owner_ref: "user-2"}
]
@collabertions = [
{_id:5, lastUpdated:5, owner_ref: "user-1"}
]
@readOnly = [
{_id:3, lastUpdated:3, owner_ref: "user-1"}
]
@tokenReadAndWrite = [
{_id:6, lastUpdated:5, owner_ref: "user-4"}
]
@tokenReadOnly = [
{_id:7, lastUpdated:4, owner_ref: "user-5"}
]
@allProjects = {
owned: @projects,
readAndWrite: @collabertions,
readOnly: @readOnly,
tokenReadAndWrite: @tokenReadAndWrite,
tokenReadOnly: @tokenReadOnly
}
2014-06-16 08:34:38 -04:00
@users =
2016-09-07 11:40:49 -04:00
'user-1':
2014-06-16 08:34:38 -04:00
first_name: 'James'
'user-2':
first_name: 'Henry'
@users[@user._id] = @user # Owner
2014-06-16 08:34:38 -04:00
@UserModel.findById = (id, fields, callback) =>
callback null, @users[id]
@LimitationsManager.userHasSubscriptionOrIsGroupMember.callsArgWith(1, null, false)
@TagsHandler.getAllTags.callsArgWith(1, null, @tags, {})
2016-01-22 00:41:22 -05:00
@NotificationsHandler.getUserNotifications = sinon.stub().callsArgWith(1, null, @notifications, {})
@ProjectGetter.findAllUsersProjects.callsArgWith(2, null, @allProjects)
it "should render the project/list page", (done)->
@res.render = (pageName, opts)=>
pageName.should.equal "project/list"
done()
@ProjectController.projectListPage @req, @res
it "should send the tags", (done)->
@res.render = (pageName, opts)=>
2014-06-16 08:34:38 -04:00
opts.tags.length.should.equal @tags.length
done()
@ProjectController.projectListPage @req, @res
it "should send the projects", (done)->
@res.render = (pageName, opts)=>
opts.projects.length.should.equal (@projects.length + @collabertions.length + @readOnly.length + @tokenReadAndWrite.length + @tokenReadOnly.length)
2014-06-16 08:34:38 -04:00
done()
@ProjectController.projectListPage @req, @res
it "should send the user", (done)->
@res.render = (pageName, opts)=>
opts.user.should.deep.equal @user
done()
@ProjectController.projectListPage @req, @res
2014-06-16 08:34:38 -04:00
it "should inject the users", (done) ->
@res.render = (pageName, opts)=>
opts.projects[0].owner.should.equal (@users[@projects[0].owner_ref])
opts.projects[1].owner.should.equal (@users[@projects[1].owner_ref])
done()
@ProjectController.projectListPage @req, @res
describe "projectListPage with duplicate projects", ->
beforeEach ->
@tags = [{name:1, project_ids:["1","2","3"]}, {name:2, project_ids:["a","1"]}, {name:3, project_ids:["a", "b", "c", "d"]}]
@notifications = [{_id:'1',user_id:'2',templateKey:'3',messageOpts:'4',key:'5'}]
@projects = [
{_id:1, lastUpdated:1, owner_ref: "user-1"},
{_id:2, lastUpdated:2, owner_ref: "user-2"}
]
@collabertions = [
{_id:5, lastUpdated:5, owner_ref: "user-1"}
]
@readOnly = [
{_id:3, lastUpdated:3, owner_ref: "user-1"}
]
@tokenReadAndWrite = [
{_id:6, lastUpdated:5, owner_ref: "user-4"}
]
@tokenReadOnly = [
{_id:6, lastUpdated:5, owner_ref: "user-4"} # Also in tokenReadAndWrite
{_id:7, lastUpdated:4, owner_ref: "user-5"}
]
@allProjects = {
owned: @projects,
readAndWrite: @collabertions,
readOnly: @readOnly,
tokenReadAndWrite: @tokenReadAndWrite,
tokenReadOnly: @tokenReadOnly
}
@users =
'user-1':
first_name: 'James'
'user-2':
first_name: 'Henry'
@users[@user._id] = @user # Owner
@UserModel.findById = (id, fields, callback) =>
callback null, @users[id]
@LimitationsManager.userHasSubscriptionOrIsGroupMember.callsArgWith(1, null, false)
@TagsHandler.getAllTags.callsArgWith(1, null, @tags, {})
@NotificationsHandler.getUserNotifications = sinon.stub().callsArgWith(1, null, @notifications, {})
@ProjectGetter.findAllUsersProjects.callsArgWith(2, null, @allProjects)
it "should render the project/list page", (done)->
@res.render = (pageName, opts)=>
pageName.should.equal "project/list"
done()
@ProjectController.projectListPage @req, @res
it "should omit one of the projects", (done)->
@res.render = (pageName, opts)=>
opts.projects.length.should.equal (
@projects.length + @collabertions.length + @readOnly.length + @tokenReadAndWrite.length + @tokenReadOnly.length - 1
)
done()
@ProjectController.projectListPage @req, @res
2014-04-28 12:47:47 -04:00
describe "renameProject", ->
beforeEach ->
@newProjectName = "my supper great new project"
@req.body.newProjectName = @newProjectName
it "should call the editor controller", (done)->
@EditorController.renameProject.callsArgWith(2)
2015-07-08 11:56:38 -04:00
@res.sendStatus = (code)=>
2014-04-28 12:47:47 -04:00
code.should.equal 200
@EditorController.renameProject.calledWith(@project_id, @newProjectName).should.equal true
done()
@ProjectController.renameProject @req, @res
it "should send an error to next() if there is a problem", (done)->
@EditorController.renameProject.callsArgWith(2, error = new Error("problem"))
next = (e)=>
e.should.equal error
2014-04-28 12:47:47 -04:00
done()
@ProjectController.renameProject @req, @res, next
describe "loadEditor", ->
beforeEach ->
@settings.editorIsOpen = true
2016-09-07 11:40:49 -04:00
@project =
name:"my proj"
_id:"213123kjlkj"
2016-09-07 11:40:49 -04:00
@user =
_id: "588f3ddae8ebc1bac07c9fa4"
ace:
fontSize:"massive"
theme:"sexy"
email: "bob@bob.com"
@ProjectGetter.getProject.callsArgWith 2, null, @project
@UserModel.findById.callsArgWith(1, null, @user)
@SubscriptionLocator.getUsersSubscription.callsArgWith(1, null, {})
2017-09-27 09:01:52 -04:00
@AuthorizationManager.getPrivilegeLevelForProject.callsArgWith 3, null, "owner"
@ProjectDeleter.unmarkAsDeletedByExternalSource = sinon.stub()
@InactiveProjectManager.reactivateProjectIfRequired.callsArgWith(1)
@AnalyticsManager.getLastOccurance.yields(null, {"mock": "event"})
@ProjectUpdateHandler.markAsOpened.callsArgWith(1)
it "should render the project/editor page", (done)->
@res.render = (pageName, opts)=>
pageName.should.equal "project/editor"
done()
@ProjectController.loadEditor @req, @res
2014-07-09 06:05:00 -04:00
it "should add user", (done)->
@res.render = (pageName, opts)=>
2014-07-09 06:05:00 -04:00
opts.user.email.should.equal @user.email
done()
@ProjectController.loadEditor @req, @res
2014-07-09 06:05:00 -04:00
it "should add on userSettings", (done)->
@res.render = (pageName, opts)=>
2014-07-09 06:05:00 -04:00
opts.userSettings.fontSize.should.equal @user.ace.fontSize
opts.userSettings.theme.should.equal @user.ace.theme
done()
@ProjectController.loadEditor @req, @res
it "should render the closed page if the editor is closed", (done)->
@settings.editorIsOpen = false
@res.render = (pageName, opts)=>
pageName.should.equal "general/closed"
done()
@ProjectController.loadEditor @req, @res
it "should not render the page if the project can not be accessed", (done)->
2017-09-27 09:01:52 -04:00
@AuthorizationManager.getPrivilegeLevelForProject = sinon.stub().callsArgWith 3, null, null
2015-07-08 11:56:38 -04:00
@res.sendStatus = (resCode, opts)=>
resCode.should.equal 401
done()
@ProjectController.loadEditor @req, @res
it "should reactivateProjectIfRequired", (done)->
@res.render = (pageName, opts)=>
@InactiveProjectManager.reactivateProjectIfRequired.calledWith(@project_id).should.equal true
done()
@ProjectController.loadEditor @req, @res
it "should mark project as opened", (done)->
@res.render = (pageName, opts)=>
@ProjectUpdateHandler.markAsOpened.calledWith(@project_id).should.equal true
done()
@ProjectController.loadEditor @req, @res
it "should set showTrackChangesOnboarding = false if there is an event", (done) ->
@AnalyticsManager.getLastOccurance.yields(null, {"mock": "event"})
@res.render = (pageName, opts)=>
opts.showTrackChangesOnboarding.should.equal false
done()
@ProjectController.loadEditor @req, @res
it "should set showTrackChangesOnboarding = true if there is no event", (done) ->
@AnalyticsManager.getLastOccurance.yields(null, null)
@res.render = (pageName, opts)=>
opts.showTrackChangesOnboarding.should.equal true
done()
@ProjectController.loadEditor @req, @res
it "should set showTrackChangesOnboarding = false if there is an error", (done) ->
@AnalyticsManager.getLastOccurance.yields(new Error("oops"), null)
@res.render = (pageName, opts)=>
opts.showTrackChangesOnboarding.should.equal false
done()
@ProjectController.loadEditor @req, @res
it "should set showTrackChangesOnboarding = false if the user signed up after release", (done) ->
@AuthenticationController.getLoggedInUserId.returns("58c11a608ba0d6e49e8ce5d5")
@AnalyticsManager.getLastOccurance.yields(null, null)
@res.render = (pageName, opts)=>
opts.showTrackChangesOnboarding.should.equal false
done()
@ProjectController.loadEditor @req, @res