Files
overleaf-cep/services/web/test/unit/src/Project/ProjectController.test.mjs
Andrew Rumble 819b2a3a46 Merge pull request #28367 from overleaf/ar-migrate-routermjs-to-zod
[web] migrate router.mjs to zod

GitOrigin-RevId: d3fc21a11351f3e2deb5011cd1beeb86286a300b
2025-09-16 08:05:26 +00:00

1758 lines
54 KiB
JavaScript

import { beforeEach, describe, it, vi, expect } from 'vitest'
import path from 'path'
import sinon from 'sinon'
import mongodb from 'mongodb-legacy'
const { ObjectId } = mongodb
const MODULE_PATH = path.join(
import.meta.dirname,
'../../../../app/src/Features/Project/ProjectController'
)
describe('ProjectController', function () {
beforeEach(async function (ctx) {
ctx.project_id = new ObjectId('abcdefabcdefabcdefabcdef')
ctx.user = {
_id: new ObjectId('123456123456123456123456'),
email: 'test@overleaf.com',
first_name: 'bjkdsjfk',
features: {},
emails: [{ email: 'test@overleaf.com' }],
}
ctx.settings = {
apis: {
chat: {
url: 'chat.com',
},
},
siteUrl: 'https://overleaf.com',
algolia: {},
plans: [],
features: {},
}
ctx.brandVariationDetails = {
id: '12',
active: true,
brand_name: 'The journal',
home_url: 'http://www.thejournal.com/',
publish_menu_link_html: 'Submit your paper to the <em>The Journal</em>',
}
ctx.token = 'some-token'
ctx.ProjectDeleter = {
promises: {
deleteProject: sinon.stub().resolves(),
restoreProject: sinon.stub().resolves(),
},
findArchivedProjects: sinon.stub(),
}
ctx.ProjectDuplicator = {
promises: {
duplicate: sinon.stub().resolves({ _id: ctx.project_id }),
},
}
ctx.ProjectCreationHandler = {
promises: {
createExampleProject: sinon.stub().resolves({ _id: ctx.project_id }),
createBasicProject: sinon.stub().resolves({ _id: ctx.project_id }),
},
}
ctx.SubscriptionLocator = {
promises: {
getUsersSubscription: sinon.stub().resolves(),
},
}
ctx.SubscriptionController = {
promises: {
getRecommendedCurrency: sinon.stub().resolves({ currency: 'USD' }),
},
}
ctx.LimitationsManager = {
hasPaidSubscription: sinon.stub(),
promises: {
userIsMemberOfGroupSubscription: sinon.stub().resolves(false),
},
}
ctx.TagsHandler = {
promises: {
getTagsForProject: sinon.stub().resolves([
{
name: 'test',
project_ids: [ctx.project_id],
},
]),
},
addProjectToTags: sinon.stub(),
}
ctx.UserModel = {
findById: sinon.stub().returns({ exec: sinon.stub().resolves() }),
updateOne: sinon.stub().returns({ exec: sinon.stub().resolves() }),
}
ctx.AuthorizationManager = {
promises: {
getPrivilegeLevelForProject: sinon.stub(),
},
isRestrictedUser: sinon.stub().returns(false),
}
ctx.EditorController = {
promises: {
renameProject: sinon.stub().resolves(),
},
}
ctx.InactiveProjectManager = {
promises: { reactivateProjectIfRequired: sinon.stub() },
}
ctx.ProjectUpdateHandler = {
promises: {
markAsOpened: sinon.stub().resolves(),
},
}
ctx.ProjectGetter = {
promises: {
findAllUsersProjects: sinon.stub().resolves(),
getProject: sinon.stub().resolves(),
},
}
ctx.ProjectHelper = {
isArchived: sinon.stub(),
isTrashed: sinon.stub(),
isArchivedOrTrashed: sinon.stub(),
getAllowedImagesForUser: sinon.stub().returns([]),
}
ctx.SessionManager = {
getLoggedInUserId: sinon.stub().returns(ctx.user._id),
getSessionUser: sinon.stub().returns(ctx.user),
isUserLoggedIn: sinon.stub().returns(true),
}
ctx.UserController = {
logout: sinon.stub(),
}
ctx.TokenAccessHandler = {
getRequestToken: sinon.stub().returns(ctx.token),
}
ctx.CollaboratorsGetter = {
promises: {
userIsTokenMember: sinon.stub().resolves(false),
isUserInvitedMemberOfProject: sinon.stub().resolves(true),
userIsReadWriteTokenMember: sinon.stub().resolves(false),
isUserInvitedReadWriteMemberOfProject: sinon.stub().resolves(true),
},
}
ctx.CollaboratorsHandler = {
promises: {
setCollaboratorPrivilegeLevel: sinon.stub().resolves(),
},
}
ctx.ProjectEntityHandler = {}
ctx.UserGetter = {
getUserFullEmails: sinon.stub().yields(null, []),
getUser: sinon.stub().resolves({ lastLoginIp: '192.170.18.2' }),
promises: {
getUserFeatures: sinon.stub().resolves(null, { collaborators: 1 }),
},
}
ctx.Features = {
hasFeature: sinon.stub(),
}
ctx.FeaturesUpdater = {
featuresEpochIsCurrent: sinon.stub().returns(true),
promises: {
refreshFeatures: sinon.stub().resolves(ctx.user),
},
}
ctx.BrandVariationsHandler = {
promises: {
getBrandVariationById: sinon.stub().resolves(ctx.brandVariationDetails),
},
}
ctx.TpdsProjectFlusher = {
promises: {
flushProjectToTpdsIfNeeded: sinon.stub().resolves(),
},
}
ctx.Metrics = {
Timer: class {
done() {}
},
inc: sinon.stub(),
}
ctx.SplitTestHandler = {
promises: {
getAssignment: sinon.stub().resolves({ variant: 'default' }),
getAssignmentForUser: sinon.stub().resolves({ variant: 'default' }),
hasUserBeenAssignedToVariant: sinon.stub().resolves(false),
},
getAssignment: sinon.stub().yields(null, { variant: 'default' }),
}
ctx.SplitTestSessionHandler = {
promises: {
sessionMaintenance: sinon.stub().resolves(),
},
}
ctx.InstitutionsFeatures = {
promises: {
hasLicence: sinon.stub().resolves(false),
},
}
ctx.InstitutionsGetter = {
promises: {
getCurrentAffiliations: sinon.stub().resolves([]),
},
}
ctx.SurveyHandler = {
getSurvey: sinon.stub().yields(null, {}),
}
ctx.ProjectAuditLogHandler = {
promises: {
addEntry: sinon.stub().resolves(),
},
}
ctx.TutorialHandler = {
getInactiveTutorials: sinon.stub().returns([]),
}
ctx.OnboardingDataCollectionManager = {
getOnboardingDataValue: sinon.stub().resolves(null),
}
ctx.Modules = {
promises: { hooks: { fire: sinon.stub().resolves() } },
}
vi.doMock('mongodb-legacy', () => ({
default: { ObjectId },
}))
vi.doMock('@overleaf/settings', () => ({
default: ctx.settings,
}))
vi.doMock('@overleaf/metrics', () => ({
default: ctx.Metrics,
}))
vi.doMock(
'../../../../app/src/Features/Collaborators/CollaboratorsHandler',
() => ({
default: ctx.CollaboratorsHandler,
})
)
vi.doMock(
'../../../../app/src/Features/SplitTests/SplitTestHandler',
() => ({
default: ctx.SplitTestHandler,
})
)
vi.doMock(
'../../../../app/src/Features/SplitTests/SplitTestSessionHandler',
() => ({
default: ctx.SplitTestSessionHandler,
})
)
vi.doMock('../../../../app/src/Features/Project/ProjectDeleter', () => ({
default: ctx.ProjectDeleter,
}))
vi.doMock('../../../../app/src/Features/Project/ProjectDuplicator', () => ({
default: ctx.ProjectDuplicator,
}))
vi.doMock(
'../../../../app/src/Features/Project/ProjectCreationHandler',
() => ({
default: ctx.ProjectCreationHandler,
})
)
vi.doMock('../../../../app/src/Features/Editor/EditorController', () => ({
default: ctx.EditorController,
}))
vi.doMock('../../../../app/src/Features/User/UserController', () => ({
default: ctx.UserController,
}))
vi.doMock('../../../../app/src/Features/Project/ProjectHelper', () => ({
default: ctx.ProjectHelper,
}))
vi.doMock(
'../../../../app/src/Features/Subscription/SubscriptionLocator',
() => ({
default: ctx.SubscriptionLocator,
})
)
vi.doMock(
'../../../../app/src/Features/Subscription/SubscriptionController',
() => ({
default: ctx.SubscriptionController,
})
)
vi.doMock(
'../../../../app/src/Features/Subscription/LimitationsManager',
() => ({
default: ctx.LimitationsManager,
})
)
vi.doMock('../../../../app/src/Features/Tags/TagsHandler', () => ({
default: ctx.TagsHandler,
}))
vi.doMock('../../../../app/src/models/User', () => ({
User: ctx.UserModel,
}))
vi.doMock('../../../../app/src/models/Subscription', () => ({}))
vi.doMock(
'../../../../app/src/Features/Authorization/AuthorizationManager',
() => ({
default: ctx.AuthorizationManager,
})
)
vi.doMock(
'../../../../app/src/Features/InactiveData/InactiveProjectManager',
() => ({
default: ctx.InactiveProjectManager,
})
)
vi.doMock(
'../../../../app/src/Features/Project/ProjectUpdateHandler',
() => ({
default: ctx.ProjectUpdateHandler,
})
)
vi.doMock('../../../../app/src/Features/Project/ProjectGetter', () => ({
default: ctx.ProjectGetter,
}))
vi.doMock(
'../../../../app/src/Features/Project/ProjectDetailsHandler',
() => ({
default: ctx.ProjectDetailsHandler,
})
)
vi.doMock(
'../../../../app/src/Features/Authentication/SessionManager',
() => ({
default: ctx.SessionManager,
})
)
vi.doMock(
'../../../../app/src/Features/TokenAccess/TokenAccessHandler',
() => ({
default: ctx.TokenAccessHandler,
})
)
vi.doMock(
'../../../../app/src/Features/Collaborators/CollaboratorsGetter',
() => ({
default: ctx.CollaboratorsGetter,
})
)
vi.doMock(
'../../../../app/src/Features/Project/ProjectEntityHandler',
() => ({
default: ctx.ProjectEntityHandler,
})
)
vi.doMock('../../../../app/src/infrastructure/Features', () => ({
default: ctx.Features,
}))
vi.doMock(
'../../../../app/src/Features/Subscription/FeaturesUpdater',
() => ({
default: ctx.FeaturesUpdater,
})
)
vi.doMock('../../../../app/src/Features/User/UserGetter', () => ({
default: ctx.UserGetter,
}))
vi.doMock(
'../../../../app/src/Features/BrandVariations/BrandVariationsHandler',
() => ({
default: ctx.BrandVariationsHandler,
})
)
vi.doMock(
'../../../../app/src/Features/ThirdPartyDataStore/TpdsProjectFlusher',
() => ({
default: ctx.TpdsProjectFlusher,
})
)
vi.doMock('../../../../app/src/models/Project', () => ({}))
vi.doMock(
'../../../../app/src/Features/Analytics/AnalyticsManager',
() => ({
default: {
recordEventForUserInBackground: () => {},
},
})
)
vi.doMock(
'../../../../app/src/Features/Subscription/SubscriptionViewModelBuilder',
() => ({
default: ctx.SubscriptionViewModelBuilder,
})
)
vi.doMock('../../../../app/src/Features/Spelling/SpellingHandler', () => ({
default: {
promises: {
getUserDictionary: sinon.stub().resolves([]),
},
},
}))
vi.doMock(
'../../../../app/src/Features/Institutions/InstitutionsFeatures',
() => ({
default: ctx.InstitutionsFeatures,
})
)
vi.doMock(
'../../../../app/src/Features/Institutions/InstitutionsGetter',
() => ({
default: ctx.InstitutionsGetter,
})
)
vi.doMock('../../../../app/src/Features/Survey/SurveyHandler', () => ({
default: ctx.SurveyHandler,
}))
vi.doMock(
'../../../../app/src/Features/Project/ProjectAuditLogHandler',
() => ({
default: ctx.ProjectAuditLogHandler,
})
)
vi.doMock('../../../../app/src/Features/Tutorial/TutorialHandler', () => ({
default: ctx.TutorialHandler,
}))
vi.doMock(
'../../../../app/src/Features/OnboardingDataCollection/OnboardingDataCollectionManager',
() => ({
default: ctx.OnboardingDataCollectionManager,
})
)
vi.doMock('../../../../app/src/Features/User/UserUpdater', () => ({
default: {
promises: {
updateUser: sinon.stub().resolves(),
},
},
}))
vi.doMock('../../../../app/src/infrastructure/Modules', () => ({
default: ctx.Modules,
}))
ctx.ProjectController = (await import(MODULE_PATH)).default
ctx.projectName = '£12321jkj9ujkljds'
ctx.req = {
query: {},
params: {
Project_id: ctx.project_id,
},
headers: {},
connection: {
remoteAddress: '192.170.18.1',
},
session: {
user: ctx.user,
},
body: {
projectName: ctx.projectName,
},
i18n: {
translate() {},
},
ip: '192.170.18.1',
capabilitySet: new Set(['chat']),
}
ctx.res = {
locals: {
jsPath: 'js path here',
},
setTimeout: sinon.stub(),
}
})
describe('updateProjectSettings', function () {
beforeEach(function (ctx) {
ctx.req.params.Project_id = ctx.project_id.toString()
})
it('should update the name', async function (ctx) {
await new Promise(resolve => {
ctx.EditorController.promises.renameProject = sinon.stub().resolves()
ctx.req.body = { name: (ctx.projectName = 'New name') }
ctx.res.sendStatus = code => {
ctx.EditorController.promises.renameProject
.calledWith(ctx.project_id, ctx.projectName)
.should.equal(true)
code.should.equal(204)
resolve()
}
ctx.ProjectController.updateProjectSettings(ctx.req, ctx.res)
})
})
it('should update the compiler', async function (ctx) {
await new Promise(resolve => {
ctx.EditorController.promises.setCompiler = sinon.stub().resolves()
ctx.req.body = { compiler: (ctx.compiler = 'pdflatex') }
ctx.res.sendStatus = code => {
ctx.EditorController.promises.setCompiler
.calledWith(ctx.project_id, ctx.compiler)
.should.equal(true)
code.should.equal(204)
resolve()
}
ctx.ProjectController.updateProjectSettings(ctx.req, ctx.res)
})
})
it('should update the imageName', async function (ctx) {
await new Promise(resolve => {
ctx.EditorController.promises.setImageName = sinon.stub().resolves()
ctx.req.body = { imageName: (ctx.imageName = 'texlive-1234.5') }
ctx.res.sendStatus = code => {
ctx.EditorController.promises.setImageName
.calledWith(ctx.project_id, ctx.imageName)
.should.equal(true)
code.should.equal(204)
resolve()
}
ctx.ProjectController.updateProjectSettings(ctx.req, ctx.res)
})
})
it('should update the spell check language', async function (ctx) {
await new Promise(resolve => {
ctx.EditorController.promises.setSpellCheckLanguage = sinon
.stub()
.resolves()
ctx.req.body = { spellCheckLanguage: (ctx.languageCode = 'fr') }
ctx.res.sendStatus = code => {
ctx.EditorController.promises.setSpellCheckLanguage
.calledWith(ctx.project_id, ctx.languageCode)
.should.equal(true)
code.should.equal(204)
resolve()
}
ctx.ProjectController.updateProjectSettings(ctx.req, ctx.res)
})
})
it('should update the root doc', async function (ctx) {
await new Promise(resolve => {
ctx.EditorController.promises.setRootDoc = sinon.stub().resolves()
ctx.req.body = {
rootDocId: (ctx.rootDocId = 'abc123def456abc123def456'),
}
ctx.res.sendStatus = code => {
ctx.EditorController.promises.setRootDoc
.calledWith(ctx.project_id, ctx.rootDocId)
.should.equal(true)
code.should.equal(204)
resolve()
}
ctx.ProjectController.updateProjectSettings(ctx.req, ctx.res)
})
})
})
describe('updateProjectAdminSettings', function () {
it('should update the public access level', async function (ctx) {
ctx.Features.hasFeature.withArgs('link-sharing').returns(true)
ctx.EditorController.promises.setPublicAccessLevel = sinon
.stub()
.resolves()
ctx.req.params.Project_id = ctx.project_id.toString()
ctx.req.body = {
publicAccessLevel: 'tokenBased',
}
await new Promise(resolve => {
ctx.res.sendStatus = code => {
ctx.EditorController.promises.setPublicAccessLevel
.calledWith(ctx.project_id, 'tokenBased')
.should.equal(true)
code.should.equal(204)
resolve()
}
ctx.ProjectController.updateProjectAdminSettings(ctx.req, ctx.res)
})
})
it('should record the change in the project audit log', async function (ctx) {
ctx.Features.hasFeature.withArgs('link-sharing').returns(true)
ctx.EditorController.promises.setPublicAccessLevel = sinon
.stub()
.resolves()
ctx.req.body = {
publicAccessLevel: 'tokenBased',
}
ctx.req.params.Project_id = ctx.project_id.toString()
await new Promise(resolve => {
ctx.res.sendStatus = code => {
ctx.ProjectAuditLogHandler.promises.addEntry
.calledWith(
ctx.project_id,
'toggle-access-level',
ctx.user._id,
ctx.req.ip,
{
publicAccessLevel: 'tokenBased',
status: 'OK',
}
)
.should.equal(true)
resolve()
}
ctx.ProjectController.updateProjectAdminSettings(ctx.req, ctx.res)
})
})
it('should refuse to update the public access level when link sharing is disabled', async function (ctx) {
ctx.Features.hasFeature.withArgs('link-sharing').returns(false)
ctx.EditorController.promises.setPublicAccessLevel = sinon
.stub()
.resolves()
ctx.req.params.Project_id = ctx.project_id.toString()
ctx.req.body = {
publicAccessLevel: 'tokenBased',
}
await new Promise(resolve => {
ctx.res.sendStatus = code => {
ctx.EditorController.promises.setPublicAccessLevel.called.should.equal(
false
)
code.should.equal(403) // Forbidden
resolve()
}
ctx.ProjectController.updateProjectAdminSettings(ctx.req, ctx.res)
})
})
})
describe('deleteProject', function () {
it('should call the project deleter', async function (ctx) {
await new Promise(resolve => {
ctx.res.sendStatus = code => {
ctx.ProjectDeleter.promises.deleteProject
.calledWith(ctx.project_id, {
deleterUser: ctx.user,
ipAddress: ctx.req.ip,
})
.should.equal(true)
code.should.equal(200)
resolve()
}
ctx.ProjectController.deleteProject(ctx.req, ctx.res)
})
})
})
describe('restoreProject', function () {
it('should tell the project deleter', async function (ctx) {
await new Promise(resolve => {
ctx.res.sendStatus = code => {
ctx.ProjectDeleter.promises.restoreProject
.calledWith(ctx.project_id)
.should.equal(true)
code.should.equal(200)
resolve()
}
ctx.ProjectController.restoreProject(ctx.req, ctx.res)
})
})
})
describe('cloneProject', function () {
it('should call the project duplicator', async function (ctx) {
await new Promise(resolve => {
ctx.res.json = json => {
ctx.ProjectDuplicator.promises.duplicate
.calledWith(ctx.user, ctx.project_id, ctx.projectName)
.should.equal(true)
json.project_id.should.equal(ctx.project_id)
resolve()
}
ctx.ProjectController.cloneProject(ctx.req, ctx.res)
})
})
})
describe('newProject', function () {
it('should call the projectCreationHandler with createExampleProject', async function (ctx) {
await new Promise(resolve => {
ctx.req.body.template = 'example'
ctx.res.json = json => {
ctx.ProjectCreationHandler.promises.createExampleProject
.calledWith(ctx.user._id, ctx.projectName)
.should.equal(true)
ctx.ProjectCreationHandler.promises.createBasicProject.called.should.equal(
false
)
resolve()
}
ctx.ProjectController.newProject(ctx.req, ctx.res)
})
})
it('should call the projectCreationHandler with createBasicProject', async function (ctx) {
await new Promise(resolve => {
ctx.req.body.template = 'basic'
ctx.res.json = json => {
ctx.ProjectCreationHandler.promises.createExampleProject.called.should.equal(
false
)
ctx.ProjectCreationHandler.promises.createBasicProject
.calledWith(ctx.user._id, ctx.projectName)
.should.equal(true)
resolve()
}
ctx.ProjectController.newProject(ctx.req, ctx.res)
})
})
})
describe('renameProject', function () {
beforeEach(function (ctx) {
ctx.newProjectName = 'my supper great new project'
ctx.req.body.newProjectName = ctx.newProjectName
ctx.req.params.Project_id = ctx.project_id.toString()
})
it('should call the editor controller', async function (ctx) {
await new Promise(resolve => {
ctx.EditorController.promises.renameProject.resolves()
ctx.res.sendStatus = code => {
code.should.equal(200)
expect(ctx.EditorController.promises.renameProject).to.have.been
.called
expect(
ctx.EditorController.promises.renameProject.args[0][0].toString()
).to.equal(ctx.project_id.toString())
expect(
ctx.EditorController.promises.renameProject.args[0][1]
).to.equal(ctx.newProjectName)
resolve()
}
ctx.ProjectController.renameProject(ctx.req, ctx.res)
})
})
it('should send an error to next() if there is a problem', async function (ctx) {
await new Promise(resolve => {
let error
ctx.EditorController.promises.renameProject.rejects(
(error = new Error('problem'))
)
const next = e => {
e.should.equal(error)
resolve()
}
ctx.ProjectController.renameProject(ctx.req, ctx.res, next)
})
})
})
describe('loadEditor', function () {
beforeEach(function (ctx) {
ctx.settings.editorIsOpen = true
ctx.project = {
name: 'my proj',
_id: '213123kjlkj',
owner_ref: '59fc84d5fbea77482d436e1b',
}
ctx.brandedProject = {
name: 'my branded proj',
_id: '3252332',
owner_ref: '59fc84d5fbea77482d436e1b',
brandVariationId: '12',
}
ctx.user = {
_id: ctx.user._id,
ace: {
fontSize: 'massive',
theme: 'sexy',
},
email: 'bob@bob.com',
refProviders: {
mendeley: { encrypted: 'aaaa' },
zotero: { encrypted: 'bbbb' },
},
}
ctx.ProjectGetter.promises.getProject.resolves(ctx.project)
ctx.UserModel.findById.returns({
exec: sinon.stub().resolves(ctx.user),
})
ctx.SubscriptionLocator.promises.getUsersSubscription.resolves({})
ctx.AuthorizationManager.promises.getPrivilegeLevelForProject.resolves(
'owner'
)
ctx.ProjectDeleter.unmarkAsDeletedByExternalSource = sinon.stub()
ctx.InactiveProjectManager.promises.reactivateProjectIfRequired.resolves()
ctx.ProjectUpdateHandler.promises.markAsOpened.resolves()
})
it('should render the project/ide-react page', async function (ctx) {
await new Promise(resolve => {
ctx.res.render = (pageName, opts) => {
pageName.should.equal('project/ide-react')
resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
it('should redirect to domain capture page', async function (ctx) {
ctx.Features.hasFeature.withArgs('saas').returns(true)
ctx.SplitTestHandler.promises.getAssignment
.withArgs(ctx.req, ctx.res, 'domain-capture-redirect')
.resolves({ variant: 'enabled' })
ctx.Modules.promises.hooks.fire
.withArgs('findDomainCaptureGroupUserCouldBePartOf', ctx.user._id)
.resolves([{ _id: new ObjectId(), managedUsersEnabled: true }])
await new Promise(resolve => {
ctx.res.redirect = url => {
url.should.equal('/domain-capture')
resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
it('should add user', async function (ctx) {
await new Promise(resolve => {
ctx.res.render = (pageName, opts) => {
opts.user.email.should.equal(ctx.user.email)
resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
it('should sanitize refProviders', async function (ctx) {
await new Promise(resolve => {
ctx.res.render = (_pageName, opts) => {
expect(opts.user.refProviders).to.deep.equal({
mendeley: true,
zotero: true,
})
resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
it('should add on userSettings', async function (ctx) {
await new Promise(resolve => {
ctx.res.render = (pageName, opts) => {
opts.userSettings.fontSize.should.equal(ctx.user.ace.fontSize)
opts.userSettings.editorTheme.should.equal(ctx.user.ace.theme)
resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
it('should add isRestrictedTokenMember', async function (ctx) {
await new Promise(resolve => {
ctx.AuthorizationManager.isRestrictedUser.returns(false)
ctx.res.render = (pageName, opts) => {
opts.isRestrictedTokenMember.should.exist
opts.isRestrictedTokenMember.should.equal(false)
return resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
it('should set isRestrictedTokenMember when appropriate', async function (ctx) {
await new Promise(resolve => {
ctx.AuthorizationManager.isRestrictedUser.returns(true)
ctx.res.render = (pageName, opts) => {
opts.isRestrictedTokenMember.should.exist
opts.isRestrictedTokenMember.should.equal(true)
return resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
it('should invoke the session maintenance for logged in user', async function (ctx) {
await new Promise(resolve => {
ctx.res.render = () => {
ctx.SplitTestSessionHandler.promises.sessionMaintenance.should.have.been.calledWith(
ctx.req,
ctx.user
)
resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
it('should invoke the session maintenance for anonymous user', async function (ctx) {
await new Promise(resolve => {
ctx.SessionManager.getLoggedInUserId.returns(null)
ctx.res.render = () => {
ctx.SplitTestSessionHandler.promises.sessionMaintenance.should.have.been.calledWith(
ctx.req
)
resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
it('should render the closed page if the editor is closed', async function (ctx) {
await new Promise(resolve => {
ctx.settings.editorIsOpen = false
ctx.res.render = (pageName, opts) => {
pageName.should.equal('general/closed')
resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
it('should not render the page if the project can not be accessed', async function (ctx) {
await new Promise(resolve => {
ctx.AuthorizationManager.promises.getPrivilegeLevelForProject = sinon
.stub()
.resolves(null)
ctx.res.sendStatus = (resCode, opts) => {
resCode.should.equal(401)
ctx.AuthorizationManager.promises.getPrivilegeLevelForProject.should.have.been.calledWith(
ctx.user._id,
ctx.project_id,
'some-token'
)
resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
it('should reactivateProjectIfRequired', async function (ctx) {
await new Promise(resolve => {
ctx.res.render = (pageName, opts) => {
ctx.InactiveProjectManager.promises.reactivateProjectIfRequired
.calledWith(ctx.project_id)
.should.equal(true)
resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
it('should mark user as active', async function (ctx) {
await new Promise(resolve => {
ctx.res.render = (pageName, opts) => {
expect(ctx.UserModel.updateOne).to.have.been.calledOnce
expect(ctx.UserModel.updateOne.args[0][0]).to.deep.equal({
_id: new ObjectId(ctx.user._id),
})
expect(ctx.UserModel.updateOne.args[0][1].$set.lastActive).to.exist
resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
it('should mark project as opened', async function (ctx) {
await new Promise(resolve => {
ctx.res.render = (pageName, opts) => {
ctx.ProjectUpdateHandler.promises.markAsOpened
.calledWith(ctx.project_id)
.should.equal(true)
resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
it('should call the brand variations handler for branded projects', async function (ctx) {
await new Promise(resolve => {
ctx.ProjectGetter.promises.getProject.resolves(ctx.brandedProject)
ctx.res.render = (pageName, opts) => {
ctx.BrandVariationsHandler.promises.getBrandVariationById
.calledWith()
.should.equal(true)
resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
it('should not call the brand variations handler for unbranded projects', async function (ctx) {
await new Promise(resolve => {
ctx.res.render = (pageName, opts) => {
ctx.BrandVariationsHandler.promises.getBrandVariationById.called.should.equal(
false
)
resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
it('should expose the brand variation details as locals for branded projects', async function (ctx) {
await new Promise(resolve => {
ctx.ProjectGetter.promises.getProject.resolves(ctx.brandedProject)
ctx.res.render = (pageName, opts) => {
opts.brandVariation.should.deep.equal(ctx.brandVariationDetails)
resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
it('flushes the project to TPDS if a flush is pending', async function (ctx) {
await new Promise(resolve => {
ctx.res.render = () => {
ctx.TpdsProjectFlusher.promises.flushProjectToTpdsIfNeeded.should.have.been.calledWith(
ctx.project_id
)
resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
it('should refresh the user features if the epoch is outdated', async function (ctx) {
await new Promise(resolve => {
ctx.FeaturesUpdater.featuresEpochIsCurrent = sinon.stub().returns(false)
ctx.res.render = () => {
ctx.FeaturesUpdater.promises.refreshFeatures.should.have.been.calledWith(
ctx.user._id,
'load-editor'
)
resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
describe('wsUrl', function () {
function checkLoadEditorWsMetric(metric) {
it(`should inc metric ${metric}`, async function (ctx) {
await new Promise(resolve => {
ctx.res.render = () => {
ctx.Metrics.inc.calledWith(metric).should.equal(true)
resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
}
function checkWsFallback(isBeta, isV2) {
describe('with ws=fallback', function () {
beforeEach(function (ctx) {
ctx.req.query = {}
ctx.req.query.ws = 'fallback'
})
it('should unset the wsUrl', async function (ctx) {
await new Promise(resolve => {
ctx.res.render = (pageName, opts) => {
;(opts.wsUrl || '/socket.io').should.equal('/socket.io')
resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
checkLoadEditorWsMetric(
`load-editor-ws${isBeta ? '-beta' : ''}${
isV2 ? '-v2' : ''
}-fallback`
)
})
}
beforeEach(function (ctx) {
ctx.settings.wsUrl = '/other.socket.io'
})
it('should set the custom wsUrl', async function (ctx) {
await new Promise(resolve => {
ctx.res.render = (pageName, opts) => {
opts.wsUrl.should.equal('/other.socket.io')
resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
checkLoadEditorWsMetric('load-editor-ws')
checkWsFallback(false)
describe('beta program', function () {
beforeEach(function (ctx) {
ctx.settings.wsUrlBeta = '/beta.socket.io'
})
describe('for a normal user', function () {
it('should set the normal custom wsUrl', async function (ctx) {
await new Promise(resolve => {
ctx.res.render = (pageName, opts) => {
opts.wsUrl.should.equal('/other.socket.io')
resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
checkLoadEditorWsMetric('load-editor-ws')
checkWsFallback(false)
})
describe('for a beta user', function () {
beforeEach(function (ctx) {
ctx.user.betaProgram = true
})
it('should set the beta wsUrl', async function (ctx) {
await new Promise(resolve => {
ctx.res.render = (pageName, opts) => {
opts.wsUrl.should.equal('/beta.socket.io')
resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
checkLoadEditorWsMetric('load-editor-ws-beta')
checkWsFallback(true)
})
})
describe('v2-rollout', function () {
beforeEach(function (ctx) {
ctx.settings.wsUrlBeta = '/beta.socket.io'
ctx.settings.wsUrlV2 = '/socket.io.v2'
})
function checkNonMatch() {
it('should set the normal custom wsUrl', async function (ctx) {
await new Promise(resolve => {
ctx.res.render = (pageName, opts) => {
opts.wsUrl.should.equal('/other.socket.io')
resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
checkLoadEditorWsMetric('load-editor-ws')
checkWsFallback(false)
}
function checkMatch() {
it('should set the v2 wsUrl', async function (ctx) {
await new Promise(resolve => {
ctx.res.render = (pageName, opts) => {
opts.wsUrl.should.equal('/socket.io.v2')
resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
checkLoadEditorWsMetric('load-editor-ws-v2')
checkWsFallback(false, true)
}
function checkForBetaUser() {
describe('for a beta user', function () {
beforeEach(function (ctx) {
ctx.user.betaProgram = true
})
it('should set the beta wsUrl', async function (ctx) {
await new Promise(resolve => {
ctx.res.render = (pageName, opts) => {
opts.wsUrl.should.equal('/beta.socket.io')
resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
checkLoadEditorWsMetric('load-editor-ws-beta')
checkWsFallback(true)
})
}
describe('when the roll out percentage is 0', function () {
beforeEach(function (ctx) {
ctx.settings.wsUrlV2Percentage = 0
})
describe('when the projectId does not match (0)', function () {
beforeEach(function (ctx) {
ctx.req.params.Project_id = ObjectId.createFromTime(0)
})
checkNonMatch()
})
describe('when the projectId does not match (42)', function () {
beforeEach(function (ctx) {
ctx.req.params.Project_id = ObjectId.createFromTime(42)
})
checkNonMatch()
})
checkForBetaUser()
})
describe('when the roll out percentage is 1', function () {
beforeEach(function (ctx) {
ctx.settings.wsUrlV2Percentage = 1
})
describe('when the projectId matches (0)', function () {
beforeEach(function (ctx) {
ctx.req.params.Project_id = ObjectId.createFromTime(0)
})
checkMatch()
checkForBetaUser()
})
describe('when the projectId does not match (1)', function () {
beforeEach(function (ctx) {
ctx.req.params.Project_id = ObjectId.createFromTime(1)
})
checkNonMatch()
checkForBetaUser()
})
describe('when the projectId does not match (42)', function () {
beforeEach(function (ctx) {
ctx.req.params.Project_id = ObjectId.createFromTime(42)
})
checkNonMatch()
})
})
describe('when the roll out percentage is 10', function () {
beforeEach(function (ctx) {
ctx.settings.wsUrlV2Percentage = 10
})
describe('when the projectId matches (0)', function () {
beforeEach(function (ctx) {
ctx.req.params.Project_id = ObjectId.createFromTime(0)
})
checkMatch()
})
describe('when the projectId matches (9)', function () {
beforeEach(function (ctx) {
ctx.req.params.Project_id = ObjectId.createFromTime(9)
})
checkMatch()
checkForBetaUser()
})
describe('when the projectId does not match (10)', function () {
beforeEach(function (ctx) {
ctx.req.params.Project_id = ObjectId.createFromTime(10)
})
checkNonMatch()
})
describe('when the projectId does not match (42)', function () {
beforeEach(function (ctx) {
ctx.req.params.Project_id = ObjectId.createFromTime(42)
})
checkNonMatch()
checkForBetaUser()
})
})
describe('when the roll out percentage is 100', function () {
beforeEach(function (ctx) {
ctx.settings.wsUrlV2Percentage = 100
})
describe('when the projectId matches (0)', function () {
beforeEach(function (ctx) {
ctx.req.params.Project_id = ObjectId.createFromTime(0)
})
checkMatch()
checkForBetaUser()
})
describe('when the projectId matches (10)', function () {
beforeEach(function (ctx) {
ctx.req.params.Project_id = ObjectId.createFromTime(10)
})
checkMatch()
})
describe('when the projectId matches (42)', function () {
beforeEach(function (ctx) {
ctx.req.params.Project_id = ObjectId.createFromTime(42)
})
checkMatch()
})
describe('when the projectId matches (99)', function () {
beforeEach(function (ctx) {
ctx.req.params.Project_id = ObjectId.createFromTime(99)
})
checkMatch()
})
})
})
})
describe('upgrade prompt (on header and share project modal)', function () {
beforeEach(function (ctx) {
// default to saas enabled
ctx.Features.hasFeature.withArgs('saas').returns(true)
// default to without a subscription
ctx.SubscriptionLocator.promises.getUsersSubscription = sinon
.stub()
.resolves(null)
})
it('should not show without the saas feature', async function (ctx) {
ctx.Features.hasFeature.withArgs('saas').returns(false)
await new Promise(resolve => {
ctx.res.render = (pageName, opts) => {
expect(opts.showUpgradePrompt).to.equal(false)
resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
it('should show for a user without a subscription or only non-paid affiliations', async function (ctx) {
await new Promise(resolve => {
ctx.res.render = (pageName, opts) => {
expect(opts.showUpgradePrompt).to.equal(true)
resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
it('should not show for a user with a personal subscription', async function (ctx) {
ctx.SubscriptionLocator.promises.getUsersSubscription = sinon
.stub()
.resolves({})
await new Promise(resolve => {
ctx.res.render = (pageName, opts) => {
expect(opts.showUpgradePrompt).to.equal(false)
resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
it('should not show for a user who is a member of a group subscription', async function (ctx) {
ctx.InstitutionsFeatures.promises.hasLicence = sinon
.stub()
.resolves(true)
await new Promise(resolve => {
ctx.res.render = (pageName, opts) => {
expect(opts.showUpgradePrompt).to.equal(false)
resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
it('should not show for a user with an affiliated paid university', async function (ctx) {
await new Promise(resolve => {
ctx.LimitationsManager.promises.userIsMemberOfGroupSubscription =
sinon.stub().resolves({ isMember: true })
ctx.res.render = (pageName, opts) => {
expect(opts.showUpgradePrompt).to.equal(false)
resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
describe('when user is a read write token member (and not already a named editor)', function () {
beforeEach(function (ctx) {
ctx.CollaboratorsGetter.promises.userIsTokenMember.resolves(true)
ctx.CollaboratorsGetter.promises.userIsReadWriteTokenMember.resolves(
true
)
ctx.CollaboratorsGetter.promises.isUserInvitedReadWriteMemberOfProject.resolves(
false
)
})
it('should redirect to the sharing-updates page', async function (ctx) {
await new Promise(resolve => {
ctx.res.redirect = url => {
expect(url).to.equal(`/project/${ctx.project_id}/sharing-updates`)
resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
})
describe('when user is a read write token member but also a named editor', function () {
beforeEach(function (ctx) {
ctx.CollaboratorsGetter.promises.userIsTokenMember.resolves(true)
ctx.CollaboratorsGetter.promises.userIsReadWriteTokenMember.resolves(
true
)
ctx.CollaboratorsGetter.promises.isUserInvitedReadWriteMemberOfProject.resolves(
true
)
})
it('should not redirect to the sharing-updates page, and should load the editor', async function (ctx) {
await new Promise(resolve => {
ctx.res.render = (pageName, opts) => {
resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
})
it('should call the collaborator limit enforcement check', async function (ctx) {
await new Promise(resolve => {
ctx.res.render = (pageName, opts) => {
ctx.Modules.promises.hooks.fire.should.have.been.calledWith(
'enforceCollaboratorLimit',
ctx.project_id
)
resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
})
describe('capabilitySet', function () {
it('should be passed as an array when loading the editor', async function (ctx) {
ctx.Features.hasFeature = sinon.stub().withArgs('chat').returns(false)
await new Promise(resolve => {
ctx.res.render = (pageName, opts) => {
expect(opts.capabilities).to.deep.equal(['chat'])
resolve()
}
ctx.ProjectController.loadEditor(ctx.req, ctx.res)
})
})
})
})
describe('userProjectsJson', function () {
beforeEach(function (ctx) {
const projects = [
{
archived: true,
trashed: true,
id: 'a',
name: 'A',
accessLevel: 'a',
somethingElse: 1,
},
{
archived: false,
id: 'b',
name: 'B',
accessLevel: 'b',
somethingElse: 1,
},
{
archived: false,
trashed: true,
id: 'c',
name: 'C',
accessLevel: 'c',
somethingElse: 1,
},
{
archived: false,
trashed: false,
id: 'd',
name: 'D',
accessLevel: 'd',
somethingElse: 1,
},
]
ctx.ProjectHelper.isArchivedOrTrashed
.withArgs(projects[0], ctx.user._id)
.returns(true)
ctx.ProjectHelper.isArchivedOrTrashed
.withArgs(projects[1], ctx.user._id)
.returns(false)
ctx.ProjectHelper.isArchivedOrTrashed
.withArgs(projects[2], ctx.user._id)
.returns(true)
ctx.ProjectHelper.isArchivedOrTrashed
.withArgs(projects[3], ctx.user._id)
.returns(false)
ctx.ProjectGetter.promises.findAllUsersProjects = sinon
.stub()
.resolves([])
ctx.ProjectController._buildProjectList = sinon.stub().returns(projects)
ctx.SessionManager.getLoggedInUserId = sinon.stub().returns(ctx.user._id)
})
it('should produce a list of projects', async function (ctx) {
await new Promise((resolve, reject) => {
ctx.res.json = data => {
expect(data).to.deep.equal({
projects: [
{ _id: 'b', name: 'B', accessLevel: 'b' },
{ _id: 'd', name: 'D', accessLevel: 'd' },
],
})
resolve()
}
ctx.ProjectController.userProjectsJson(
ctx.req,
ctx.res,
ctx.rejectOnError(reject)
)
})
})
})
describe('projectEntitiesJson', function () {
beforeEach(function (ctx) {
ctx.SessionManager.getLoggedInUserId = sinon.stub().returns('abc')
ctx.req.params = { Project_id: 'abcd' }
ctx.project = { _id: 'abcd' }
ctx.docs = [
{ path: '/things/b.txt', doc: true },
{ path: '/main.tex', doc: true },
]
ctx.files = [{ path: '/things/a.txt' }]
ctx.ProjectGetter.promises.getProject = sinon.stub().resolves(ctx.project)
ctx.ProjectEntityHandler.getAllEntitiesFromProject = sinon
.stub()
.returns({ docs: ctx.docs, files: ctx.files })
})
it('should produce a list of entities', async function (ctx) {
await new Promise(resolve => {
ctx.res.json = data => {
expect(data).to.deep.equal({
project_id: 'abcd',
entities: [
{ path: '/main.tex', type: 'doc' },
{ path: '/things/a.txt', type: 'file' },
{ path: '/things/b.txt', type: 'doc' },
],
})
expect(ctx.ProjectGetter.promises.getProject.callCount).to.equal(1)
expect(
ctx.ProjectEntityHandler.getAllEntitiesFromProject.callCount
).to.equal(1)
resolve()
}
ctx.ProjectController.projectEntitiesJson(ctx.req, ctx.res, ctx.next)
})
})
it('should call next with an error if the project file tree is invalid', async function (ctx) {
ctx.ProjectEntityHandler.getAllEntitiesFromProject = sinon.stub().throws()
await new Promise(resolve => {
ctx.next = err => {
expect(err).to.be.an.instanceof(Error)
resolve()
}
ctx.ProjectController.projectEntitiesJson(ctx.req, ctx.res, ctx.next)
})
})
})
describe('_buildProjectViewModel', function () {
beforeEach(function (ctx) {
ctx.ProjectHelper.isArchived.returns(false)
ctx.ProjectHelper.isTrashed.returns(false)
ctx.project = {
_id: 'abcd',
name: 'netsenits',
lastUpdated: 1,
lastUpdatedBy: 2,
publicAccesLevel: 'private',
archived: false,
owner_ref: 'defg',
tokens: {
readAndWrite: '1abcd',
readAndWritePrefix: '1',
readOnly: 'neiotsranteoia',
},
}
})
describe('project not being archived or trashed', function () {
it('should produce a model of the project', function (ctx) {
const result = ctx.ProjectController._buildProjectViewModel(
ctx.project,
'readAndWrite',
'owner',
ctx.user._id
)
expect(result).to.exist
expect(result).to.be.an('object')
expect(result).to.deep.equal({
id: 'abcd',
name: 'netsenits',
lastUpdated: 1,
lastUpdatedBy: 2,
publicAccessLevel: 'private',
accessLevel: 'readAndWrite',
source: 'owner',
archived: false,
trashed: false,
owner_ref: 'defg',
isV1Project: false,
})
})
})
describe('project being simultaneously archived and trashed', function () {
beforeEach(function (ctx) {
ctx.ProjectHelper.isArchived.returns(true)
ctx.ProjectHelper.isTrashed.returns(true)
})
it('should produce a model of the project', function (ctx) {
const result = ctx.ProjectController._buildProjectViewModel(
ctx.project,
'readAndWrite',
'owner',
ctx.user._id
)
expect(result).to.exist
expect(result).to.be.an('object')
expect(result).to.deep.equal({
id: 'abcd',
name: 'netsenits',
lastUpdated: 1,
lastUpdatedBy: 2,
publicAccessLevel: 'private',
accessLevel: 'readAndWrite',
source: 'owner',
archived: true,
trashed: false,
owner_ref: 'defg',
isV1Project: false,
})
})
})
describe('when token-read-only access', function () {
it('should redact the owner and last-updated data', function (ctx) {
const result = ctx.ProjectController._buildProjectViewModel(
ctx.project,
'readOnly',
'token',
ctx.user._id
)
expect(result).to.exist
expect(result).to.be.an('object')
expect(result).to.deep.equal({
id: 'abcd',
name: 'netsenits',
lastUpdated: 1,
lastUpdatedBy: null,
publicAccessLevel: 'private',
accessLevel: 'readOnly',
source: 'token',
archived: false,
trashed: false,
owner_ref: null,
isV1Project: false,
})
})
})
})
describe('_isInPercentageRollout', function () {
const ids = [
'5a05cd7621f9fe22be131740',
'5a05cd7821f9fe22be131741',
'5a05cd7921f9fe22be131742',
'5a05cd7a21f9fe22be131743',
'5a05cd7b21f9fe22be131744',
'5a05cd7c21f9fe22be131745',
'5a05cd7d21f9fe22be131746',
'5a05cd7e21f9fe22be131747',
'5a05cd7f21f9fe22be131748',
'5a05cd8021f9fe22be131749',
'5a05cd8021f9fe22be13174a',
'5a05cd8121f9fe22be13174b',
'5a05cd8221f9fe22be13174c',
'5a05cd8221f9fe22be13174d',
'5a05cd8321f9fe22be13174e',
'5a05cd8321f9fe22be13174f',
'5a05cd8421f9fe22be131750',
'5a05cd8421f9fe22be131751',
'5a05cd8421f9fe22be131752',
'5a05cd8521f9fe22be131753',
]
it('should produce the expected results', function (ctx) {
expect(
ids.map(i =>
ctx.ProjectController._isInPercentageRollout('abcd', i, 50)
)
).to.deep.equal([
false,
false,
false,
false,
false,
false,
true,
false,
true,
true,
true,
true,
true,
true,
false,
false,
false,
true,
false,
true,
])
expect(
ids.map(i =>
ctx.ProjectController._isInPercentageRollout('efgh', i, 50)
)
).to.deep.equal([
false,
false,
false,
false,
true,
false,
false,
true,
false,
false,
true,
true,
true,
false,
true,
false,
true,
true,
false,
false,
])
})
})
})