expect = require("chai").expect Async = require("async") User = require "./helpers/User" request = require "./helpers/request" settings = require "settings-sharelatex" CollaboratorsEmailHandler = require "../../../app/js/Features/Collaborators/CollaboratorsEmailHandler" createInvite = (sendingUser, projectId, email, callback=(err, invite)->) -> sendingUser.getCsrfToken (err) -> return callback(err) if err sendingUser.request.post { uri: "/project/#{projectId}/invite", json: email: email privileges: 'readAndWrite' }, (err, response, body) -> return callback(err) if err callback(null, body.invite) createProject = (owner, projectName, callback=(err, projectId, project)->) -> owner.createProject projectName, (err, projectId) -> throw err if err fakeProject = { _id: projectId, name: projectName, owner_ref: owner } callback(err, projectId, fakeProject) createProjectAndInvite = (owner, projectName, email, callback=(err, project, invite)->) -> createProject owner, projectName, (err, projectId, project) -> return callback(err) if err createInvite owner, projectId, email, (err, invite) -> return callback(err) if err link = CollaboratorsEmailHandler._buildInviteUrl(project, invite) callback(null, project, invite, link) revokeInvite = (sendingUser, projectId, inviteId, callback=(err)->) -> sendingUser.getCsrfToken (err) -> return callback(err) if err sendingUser.request.delete { uri: "/project/#{projectId}/invite/#{inviteId}", }, (err, response, body) -> return callback(err) if err callback(null) # Actions tryFollowInviteLink = (user, link, callback=(err, response, body)->) -> user.request.get { uri: link baseUrl: null }, callback tryAcceptInvite = (user, invite, callback=(err, response, body)->) -> user.request.post { uri: "/project/#{invite.projectId}/invite/token/#{invite.token}/accept" json: token: invite.token }, callback tryRegisterUser = (user, email, callback=(err, response, body)->) -> user.getCsrfToken (error) => return callback(error) if error? user.request.post { url: "/register" json: email: email password: "some_weird_password" }, callback tryFollowLoginLink = (user, loginLink, callback=(err, response, body)->) -> user.getCsrfToken (error) => return callback(error) if error? user.request.get loginLink, callback tryLoginUser = (user, callback=(err, response, body)->) -> user.getCsrfToken (error) => return callback(error) if error? user.request.post { url: "/login" json: email: user.email password: user.password }, callback tryGetInviteList = (user, projectId, callback=(err, response, body)->) -> user.getCsrfToken (error) => return callback(error) if error? user.request.get { url: "/project/#{projectId}/invites" json: true }, callback tryJoinProject = (user, projectId, callback=(err, response, body)->) -> user.getCsrfToken (error) => return callback(error) if error? user.request.post { url: "/project/#{projectId}/join" qs: {user_id: user._id} auth: user: settings.apis.web.user pass: settings.apis.web.pass sendImmediately: true json: true jar: false }, callback # Expectations expectProjectAccess = (user, projectId, callback=(err,result)->) -> # should have access to project user.openProject projectId, (err) => expect(err).to.be.oneOf [null, undefined] callback() expectNoProjectAccess = (user, projectId, callback=(err,result)->) -> # should not have access to project page user.openProject projectId, (err) => expect(err).to.be.instanceof Error callback() expectInvitePage = (user, link, callback=(err,result)->) -> # view invite tryFollowInviteLink user, link, (err, response, body) -> expect(err).to.be.oneOf [null, undefined] expect(response.statusCode).to.equal 200 expect(body).to.match new RegExp("Project Invite - .*") callback() expectInvalidInvitePage = (user, link, callback=(err,result)->) -> # view invalid invite tryFollowInviteLink user, link, (err, response, body) -> expect(err).to.be.oneOf [null, undefined] expect(response.statusCode).to.equal 200 expect(body).to.match new RegExp("Invalid Invite - .*") callback() expectInviteRedirectToRegister = (user, link, callback=(err,result)->) -> # view invite, redirect to `/register` tryFollowInviteLink user, link, (err, response, body) -> expect(err).to.be.oneOf [null, undefined] expect(response.statusCode).to.equal 302 expect(response.headers.location).to.match new RegExp("^/register.*$") # follow redirect to register page and extract the redirectUrl from form user.request.get response.headers.location, (err, response, body) -> callback(null) expectLoginPage = (user, callback=(err, result)->) -> tryFollowLoginLink user, "/login", (err, response, body) -> expect(err).to.be.oneOf [null, undefined] expect(response.statusCode).to.equal 200 expect(body).to.match new RegExp("Login - .*") callback(null) expectLoginRedirectToInvite = (user, link, callback=(err, result)->) -> tryLoginUser user, (err, response, body) -> expect(err).to.be.oneOf [null, undefined] expect(response.statusCode).to.equal 200 callback(null, null) expectRegistrationRedirectToInvite = (user, email, link, callback=(err, result)->) -> tryRegisterUser user, email, (err, response, body) -> expect(err).to.be.oneOf [null, undefined] expect(response.statusCode).to.equal 200 callback(null, null) expectInviteRedirectToProject = (user, link, invite, callback=(err,result)->) -> # view invite, redirect straight to project tryFollowInviteLink user, link, (err, response, body) -> expect(err).to.be.oneOf [null, undefined] expect(response.statusCode).to.equal 302 expect(response.headers.location).to.equal "/project/#{invite.projectId}" callback() expectAcceptInviteAndRedirect = (user, invite, callback=(err,result)->) -> # should accept the invite and redirect to project tryAcceptInvite user, invite, (err, response, body) => expect(err).to.be.oneOf [null, undefined] expect(response.statusCode).to.equal 302 expect(response.headers.location).to.equal "/project/#{invite.projectId}" callback() expectInviteListCount = (user, projectId, count, callback=(err)->) -> tryGetInviteList user, projectId, (err, response, body) -> expect(err).to.be.oneOf [null, undefined] expect(response.statusCode).to.equal 200 expect(body).to.have.all.keys ['invites'] expect(body.invites.length).to.equal count callback() expectInvitesInJoinProjectCount = (user, projectId, count, callback=(err,result)->) -> tryJoinProject user, projectId, (err, response, body) -> expect(err).to.be.oneOf [null, undefined] expect(response.statusCode).to.equal 200 expect(body.project).to.contain.keys ['invites'] expect(body.project.invites.length).to.equal count callback() describe "ProjectInviteTests", -> before (done) -> @sendingUser = new User() @user = new User() @site_admin = new User({email: "admin@example.com"}) @email = 'smoketestuser@example.com' @projectName = 'sharing test' Async.series [ (cb) => @user.login cb (cb) => @user.logout cb (cb) => @sendingUser.login cb ], done describe 'creating invites', -> beforeEach (done) -> @projectName = "wat" @projectId = null @fakeProject = null done() afterEach -> describe 'creating two invites', -> beforeEach (done) -> Async.series [ (cb) => createProject @sendingUser, @projectName, (err, projectId, project) => @projectId = projectId @fakeProject = project cb() ], done afterEach (done) -> Async.series [ (cb) => @sendingUser.deleteProject(@projectId, cb) (cb) => @sendingUser.deleteProject(@projectId, cb) ], done it 'should allow the project owner to create and remove invites', (done) -> @invite = null Async.series [ (cb) => expectProjectAccess @sendingUser, @projectId, cb (cb) => expectInviteListCount @sendingUser, @projectId, 0, cb # create invite, check invite list count (cb) => createInvite @sendingUser, @projectId, @email, (err, invite) => return cb(err) if err @invite = invite cb() (cb) => expectInviteListCount @sendingUser, @projectId, 1, cb (cb) => revokeInvite @sendingUser, @projectId, @invite._id, cb (cb) => expectInviteListCount @sendingUser, @projectId, 0, cb # and a second time (cb) => createInvite @sendingUser, @projectId, @email, (err, invite) => return cb(err) if err @invite = invite cb() (cb) => expectInviteListCount @sendingUser, @projectId, 1, cb # check the joinProject view (cb) => expectInvitesInJoinProjectCount @sendingUser, @projectId, 1, cb # revoke invite (cb) => revokeInvite @sendingUser, @projectId, @invite._id, cb (cb) => expectInviteListCount @sendingUser, @projectId, 0, cb (cb) => expectInvitesInJoinProjectCount @sendingUser, @projectId, 0, cb ], done it 'should allow the project owner to many invites at once', (done) -> @inviteOne = null @inviteTwo = null Async.series [ (cb) => expectProjectAccess @sendingUser, @projectId, cb (cb) => expectInviteListCount @sendingUser, @projectId, 0, cb # create first invite (cb) => createInvite @sendingUser, @projectId, @email, (err, invite) => return cb(err) if err @inviteOne = invite cb() (cb) => expectInviteListCount @sendingUser, @projectId, 1, cb # and a second (cb) => createInvite @sendingUser, @projectId, @email, (err, invite) => return cb(err) if err @inviteTwo = invite cb() # should have two (cb) => expectInviteListCount @sendingUser, @projectId, 2, cb (cb) => expectInvitesInJoinProjectCount @sendingUser, @projectId, 2, cb # revoke first (cb) => revokeInvite @sendingUser, @projectId, @inviteOne._id, cb (cb) => expectInviteListCount @sendingUser, @projectId, 1, cb # revoke second (cb) => revokeInvite @sendingUser, @projectId, @inviteTwo._id, cb (cb) => expectInviteListCount @sendingUser, @projectId, 0, cb ], done describe 'clicking the invite link', -> beforeEach (done) -> @projectId = null @fakeProject = null done() describe "user is logged in already", -> beforeEach (done) -> Async.series [ (cb) => createProjectAndInvite @sendingUser, @projectName, @email, (err, project, invite, link) => @projectId = project._id @fakeProject = project @invite = invite @link = link cb() (cb) => @user.login (err) => if err throw err cb() ], done afterEach (done) -> Async.series [ (cb) => @sendingUser.deleteProject(@projectId, cb) (cb) => @sendingUser.deleteProject(@projectId, cb) (cb) => revokeInvite(@sendingUser, @projectId, @invite._id, cb) ], done describe 'user is already a member of the project', -> beforeEach (done) -> Async.series [ (cb) => expectInvitePage @user, @link, cb (cb) => expectAcceptInviteAndRedirect @user, @invite, cb (cb) => expectProjectAccess @user, @invite.projectId, cb ], done describe 'when user clicks on the invite a second time', -> it 'should just redirect to the project page', (done) -> Async.series [ (cb) => expectProjectAccess @user, @invite.projectId, cb (cb) => expectInviteRedirectToProject @user, @link, @invite, cb (cb) => expectProjectAccess @user, @invite.projectId, cb ], done describe 'when the user recieves another invite to the same project', -> it 'should redirect to the project page', (done) -> Async.series [ (cb) => createInvite @sendingUser, @projectId, @email, (err, invite) => if err throw err @secondInvite = invite @secondLink = CollaboratorsEmailHandler._buildInviteUrl(@fakeProject, invite) cb() (cb) => expectInviteRedirectToProject @user, @secondLink, @secondInvite, cb (cb) => expectProjectAccess @user, @invite.projectId, cb (cb) => revokeInvite @sendingUser, @projectId, @secondInvite._id, cb ], done describe 'user is not a member of the project', -> it 'should not grant access if the user does not accept the invite', (done) -> Async.series( [ (cb) => expectInvitePage @user, @link, cb (cb) => expectNoProjectAccess @user, @invite.projectId, cb ], done ) it 'should render the invalid-invite page if the token is invalid', (done) -> Async.series( [ (cb) => link = @link.replace(@invite.token, 'not_a_real_token') expectInvalidInvitePage @user, link, cb (cb) => expectNoProjectAccess @user, @invite.projectId, cb (cb) => expectNoProjectAccess @user, @invite.projectId, cb ], done ) it 'should allow the user to accept the invite and access the project', (done) -> Async.series( [ (cb) => expectInvitePage @user, @link, cb (cb) => expectAcceptInviteAndRedirect @user, @invite, cb (cb) => expectProjectAccess @user, @invite.projectId, cb ], done ) describe 'user is not logged in initially', -> before (done) -> @user.logout done beforeEach (done) -> Async.series [ (cb) => createProjectAndInvite @sendingUser, @projectName, @email, (err, project, invite, link) => @projectId = project._id @fakeProject = project @invite = invite @link = link cb() ], done afterEach (done) -> Async.series [ (cb) => @sendingUser.deleteProject(@projectId, cb) (cb) => @sendingUser.deleteProject(@projectId, cb) (cb) => revokeInvite(@sendingUser, @projectId, @invite._id, cb) ], done describe 'registration prompt workflow with valid token', -> it 'should redirect to the register page', (done) -> Async.series [ (cb) => expectInviteRedirectToRegister(@user, @link, cb) ], done it 'should allow user to accept the invite if the user registers a new account', (done) -> Async.series [ (cb) => expectInviteRedirectToRegister @user, @link, cb (cb) => expectRegistrationRedirectToInvite @user, "some_email@example.com", @link, cb (cb) => expectInvitePage @user, @link, cb (cb) => expectAcceptInviteAndRedirect @user, @invite, cb (cb) => expectProjectAccess @user, @invite.projectId, cb ], done describe 'registration prompt workflow with non-valid token', -> before (done)-> @user.logout done it 'should redirect to the register page', (done) -> Async.series [ (cb) => expectInviteRedirectToRegister(@user, @link, cb) (cb) => expectNoProjectAccess @user, @invite.projectId, cb ], done it 'should display invalid-invite if the user registers a new account', (done) -> badLink = @link.replace(@invite.token, 'not_a_real_token') Async.series [ (cb) => expectInviteRedirectToRegister @user, badLink, cb (cb) => expectRegistrationRedirectToInvite @user, "some_email@example.com", badLink, cb (cb) => expectInvalidInvitePage @user, badLink, cb (cb) => expectNoProjectAccess @user, @invite.projectId, cb ], done describe 'login workflow with valid token', -> before (done)-> @user.logout done it 'should redirect to the register page', (done) -> Async.series [ (cb) => expectInviteRedirectToRegister(@user, @link, cb) (cb) => expectNoProjectAccess @user, @invite.projectId, cb ], done it 'should allow the user to login to view the invite', (done) -> Async.series [ (cb) => expectInviteRedirectToRegister @user, @link, cb (cb) => expectLoginPage @user, cb (cb) => expectLoginRedirectToInvite @user, @link, cb (cb) => expectInvitePage @user, @link, cb (cb) => expectNoProjectAccess @user, @invite.projectId, cb ], done it 'should allow user to accept the invite if the user registers a new account', (done) -> Async.series [ (cb) => expectInvitePage @user, @link, cb (cb) => expectAcceptInviteAndRedirect @user, @invite, cb (cb) => expectProjectAccess @user, @invite.projectId, cb ], done describe 'login workflow with non-valid token', -> before (done)-> @user.logout done it 'should redirect to the register page', (done) -> Async.series [ (cb) => expectInviteRedirectToRegister(@user, @link, cb) (cb) => expectNoProjectAccess @user, @invite.projectId, cb ], done it 'should show the invalid-invite page once the user has logged in', (done) -> badLink = @link.replace(@invite.token, 'not_a_real_token') Async.series [ (cb) => expectInviteRedirectToRegister @user, badLink, cb (cb) => expectLoginPage @user, cb (cb) => expectLoginRedirectToInvite @user, badLink, cb (cb) => expectInvalidInvitePage @user, badLink, cb (cb) => expectNoProjectAccess @user, @invite.projectId, cb ], done