2019-05-29 05:21:06 -04:00
|
|
|
const modulePath = '../../../../app/src/Features/Project/ProjectDeleter'
|
|
|
|
const SandboxedModule = require('sandboxed-module')
|
|
|
|
const sinon = require('sinon')
|
2019-07-18 10:18:56 -04:00
|
|
|
const chai = require('chai')
|
|
|
|
const { expect } = chai
|
|
|
|
const tk = require('timekeeper')
|
|
|
|
const moment = require('moment')
|
|
|
|
const { Project } = require('../helpers/models/Project')
|
|
|
|
const { DeletedProject } = require('../helpers/models/DeletedProject')
|
|
|
|
const { ObjectId } = require('mongoose').Types
|
2019-05-29 05:21:06 -04:00
|
|
|
|
2019-07-18 10:18:56 -04:00
|
|
|
describe('ProjectDeleter', () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
tk.freeze(Date.now())
|
|
|
|
this.project_id = ObjectId('588fffffffffffffffffffff')
|
|
|
|
this.ip = '192.170.18.1'
|
2019-05-29 05:21:06 -04:00
|
|
|
this.project = {
|
|
|
|
_id: this.project_id,
|
2019-07-18 10:18:56 -04:00
|
|
|
lastUpdated: new Date(),
|
2019-05-29 05:21:06 -04:00
|
|
|
rootFolder: [],
|
|
|
|
collaberator_refs: ['collab1', 'collab2'],
|
|
|
|
readOnly_refs: ['readOnly1', 'readOnly2'],
|
2019-07-18 10:18:56 -04:00
|
|
|
tokenAccessReadAndWrite_refs: ['tokenCollab1', 'tokenCollab2'],
|
|
|
|
tokenAccessReadOnly_refs: ['tokenReadOnly1', 'tokenReadOnly2'],
|
|
|
|
owner_ref: ObjectId('588aaaaaaaaaaaaaaaaaaaaa'),
|
|
|
|
tokens: {
|
|
|
|
readOnly: 'wombat',
|
|
|
|
readAndWrite: 'potato'
|
|
|
|
},
|
|
|
|
name: 'a very scientific analysis of spooky ghosts'
|
2019-05-29 05:21:06 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
this.user = {
|
|
|
|
_id: '588f3ddae8ebc1bac07c9fa4',
|
|
|
|
first_name: 'bjkdsjfk',
|
|
|
|
features: {}
|
|
|
|
}
|
|
|
|
|
2019-07-18 10:18:56 -04:00
|
|
|
this.doc = {
|
|
|
|
_id: '5bd975f54f62e803cb8a8fec',
|
|
|
|
lines: ['a bunch of lines', 'for a sunny day', 'in London town'],
|
|
|
|
ranges: {},
|
|
|
|
project_id: '5cf9270b4eff6e186cf8b05e'
|
2019-05-29 05:21:06 -04:00
|
|
|
}
|
2019-07-18 10:18:56 -04:00
|
|
|
|
|
|
|
this.deletedProjects = [
|
|
|
|
{
|
|
|
|
_id: '5cf7f145c1401f0ca0eb1aaa',
|
|
|
|
deleterData: {
|
|
|
|
_id: '5cf7f145c1401f0ca0eb1aac',
|
|
|
|
deletedAt: moment()
|
|
|
|
.subtract(95, 'days')
|
|
|
|
.toDate(),
|
|
|
|
deleterId: '588f3ddae8ebc1bac07c9fa4',
|
|
|
|
deleterIpAddress: '172.19.0.1'
|
|
|
|
},
|
|
|
|
project: {
|
|
|
|
_id: '5cf9270b4eff6e186cf8b05e'
|
|
|
|
}
|
|
|
|
},
|
|
|
|
{
|
|
|
|
_id: '5cf8eb11c1401f0ca0eb1ad7',
|
|
|
|
deleterData: {
|
|
|
|
_id: '5b74360c0fbe57011ae9938f',
|
|
|
|
deletedAt: moment()
|
|
|
|
.subtract(95, 'days')
|
|
|
|
.toDate(),
|
|
|
|
deleterId: '588f3ddae8ebc1bac07c9fa4',
|
|
|
|
deleterIpAddress: '172.20.0.1',
|
|
|
|
deletedProjectId: '5cf8f95a0c87371362c23919'
|
|
|
|
},
|
|
|
|
project: {
|
|
|
|
_id: '5cf8f95a0c87371362c23919'
|
2019-05-29 05:21:06 -04:00
|
|
|
}
|
|
|
|
}
|
2019-07-18 10:18:56 -04:00
|
|
|
]
|
|
|
|
|
2019-05-29 05:21:06 -04:00
|
|
|
this.documentUpdaterHandler = {
|
|
|
|
flushProjectToMongoAndDelete: sinon.stub().callsArgWith(1)
|
|
|
|
}
|
|
|
|
this.editorController = {
|
|
|
|
notifyUsersProjectHasBeenDeletedOrRenamed: sinon.stub().callsArgWith(1)
|
|
|
|
}
|
|
|
|
this.TagsHandler = {
|
|
|
|
removeProjectFromAllTags: sinon.stub().callsArgWith(2)
|
|
|
|
}
|
|
|
|
this.CollaboratorsHandler = {
|
|
|
|
removeUserFromAllProjets: sinon.stub().yields(),
|
|
|
|
getMemberIds: sinon
|
|
|
|
.stub()
|
|
|
|
.withArgs(this.project_id)
|
|
|
|
.yields(null, ['member-id-1', 'member-id-2'])
|
|
|
|
}
|
2019-07-18 10:18:56 -04:00
|
|
|
|
|
|
|
this.logger = {
|
|
|
|
err: sinon.stub(),
|
|
|
|
log: sinon.stub(),
|
|
|
|
warn: sinon.stub()
|
|
|
|
}
|
|
|
|
|
|
|
|
this.ProjectDetailsHandler = {
|
|
|
|
generateUniqueName: sinon.stub().yields(null, this.project.name)
|
|
|
|
}
|
|
|
|
|
|
|
|
this.db = {
|
|
|
|
projects: {
|
|
|
|
insert: sinon.stub().yields()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
this.DocstoreManager = {
|
|
|
|
destroyProject: sinon.stub().yields()
|
|
|
|
}
|
|
|
|
|
|
|
|
this.ProjectMock = sinon.mock(Project)
|
|
|
|
this.DeletedProjectMock = sinon.mock(DeletedProject)
|
|
|
|
|
|
|
|
this.ProjectDeleter = SandboxedModule.require(modulePath, {
|
2019-05-29 05:21:06 -04:00
|
|
|
requires: {
|
|
|
|
'../Editor/EditorController': this.editorController,
|
2019-07-18 10:18:56 -04:00
|
|
|
'../../models/Project': { Project: Project },
|
|
|
|
'../../models/DeletedProject': { DeletedProject: DeletedProject },
|
2019-05-29 05:21:06 -04:00
|
|
|
'../DocumentUpdater/DocumentUpdaterHandler': this
|
|
|
|
.documentUpdaterHandler,
|
|
|
|
'../Tags/TagsHandler': this.TagsHandler,
|
|
|
|
'../FileStore/FileStoreHandler': (this.FileStoreHandler = {}),
|
|
|
|
'../Collaborators/CollaboratorsHandler': this.CollaboratorsHandler,
|
2019-07-18 10:18:56 -04:00
|
|
|
'../Docstore/DocstoreManager': this.DocstoreManager,
|
|
|
|
'./ProjectDetailsHandler': this.ProjectDetailsHandler,
|
|
|
|
'../../infrastructure/mongojs': { db: this.db },
|
|
|
|
'logger-sharelatex': this.logger
|
|
|
|
},
|
|
|
|
globals: {
|
|
|
|
console: console
|
2019-05-29 05:21:06 -04:00
|
|
|
}
|
2019-07-18 10:18:56 -04:00
|
|
|
})
|
2019-05-29 05:21:06 -04:00
|
|
|
})
|
|
|
|
|
2019-07-18 10:18:56 -04:00
|
|
|
afterEach(() => {
|
|
|
|
tk.reset()
|
|
|
|
this.DeletedProjectMock.restore()
|
|
|
|
this.ProjectMock.restore()
|
|
|
|
})
|
|
|
|
|
|
|
|
describe('mark as deleted by external source', () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
this.ProjectMock.expects('update')
|
|
|
|
.withArgs(
|
|
|
|
{ _id: this.project_id },
|
|
|
|
{ deletedByExternalDataSource: true }
|
|
|
|
)
|
|
|
|
.yields()
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should update the project with the flag set to true', done => {
|
|
|
|
this.ProjectDeleter.markAsDeletedByExternalSource(this.project_id, () => {
|
|
|
|
this.ProjectMock.verify()
|
|
|
|
done()
|
2019-05-29 05:21:06 -04:00
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2019-07-18 10:18:56 -04:00
|
|
|
it('should tell the editor controler so users are notified', done => {
|
|
|
|
this.ProjectDeleter.markAsDeletedByExternalSource(this.project_id, () => {
|
2019-05-29 05:21:06 -04:00
|
|
|
this.editorController.notifyUsersProjectHasBeenDeletedOrRenamed
|
2019-07-18 10:18:56 -04:00
|
|
|
.calledWith(this.project_id)
|
2019-05-29 05:21:06 -04:00
|
|
|
.should.equal(true)
|
2019-07-18 10:18:56 -04:00
|
|
|
done()
|
2019-05-29 05:21:06 -04:00
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2019-07-18 10:18:56 -04:00
|
|
|
describe('unmarkAsDeletedByExternalSource', done => {
|
|
|
|
beforeEach(() => {
|
|
|
|
this.ProjectMock.expects('update')
|
|
|
|
.withArgs(
|
2019-05-29 05:21:06 -04:00
|
|
|
{ _id: this.project_id },
|
|
|
|
{ deletedByExternalDataSource: false }
|
|
|
|
)
|
2019-07-18 10:18:56 -04:00
|
|
|
.yields()
|
|
|
|
this.ProjectDeleter.unmarkAsDeletedByExternalSource(this.project_id, done)
|
2019-05-29 05:21:06 -04:00
|
|
|
})
|
|
|
|
|
2019-07-18 10:18:56 -04:00
|
|
|
it('should remove the flag from the project', () => {
|
|
|
|
this.ProjectMock.verify()
|
2019-05-29 05:21:06 -04:00
|
|
|
})
|
2019-07-18 10:18:56 -04:00
|
|
|
})
|
2019-05-29 05:21:06 -04:00
|
|
|
|
2019-07-18 10:18:56 -04:00
|
|
|
describe('deleteUsersProjects', () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
this.ProjectMock.expects('find')
|
|
|
|
.withArgs({ owner_ref: this.user._id })
|
|
|
|
.yields(null, [{ _id: 'wombat' }, { _id: 'potato' }])
|
|
|
|
|
|
|
|
this.ProjectDeleter.deleteProject = sinon.stub().yields()
|
2019-05-29 05:21:06 -04:00
|
|
|
})
|
|
|
|
|
2019-07-18 10:18:56 -04:00
|
|
|
it('should find all the projects owned by the user_id', done => {
|
|
|
|
this.ProjectDeleter.deleteUsersProjects(this.user._id, () => {
|
|
|
|
this.ProjectMock.verify()
|
|
|
|
done()
|
2019-05-29 05:21:06 -04:00
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2019-07-18 10:18:56 -04:00
|
|
|
it('should call deleteProject once for each project', done => {
|
|
|
|
this.ProjectDeleter.deleteUsersProjects(this.user._id, () => {
|
|
|
|
sinon.assert.calledTwice(this.ProjectDeleter.deleteProject)
|
|
|
|
sinon.assert.calledWith(this.ProjectDeleter.deleteProject, 'wombat')
|
|
|
|
sinon.assert.calledWith(this.ProjectDeleter.deleteProject, 'potato')
|
|
|
|
done()
|
2019-05-29 05:21:06 -04:00
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2019-07-18 10:18:56 -04:00
|
|
|
it('should remove all the projects the user is a collaborator of', done => {
|
|
|
|
this.ProjectDeleter.deleteUsersProjects(this.user._id, () => {
|
|
|
|
sinon.assert.calledWith(
|
|
|
|
this.CollaboratorsHandler.removeUserFromAllProjets,
|
|
|
|
this.user._id
|
|
|
|
)
|
|
|
|
sinon.assert.calledOnce(
|
|
|
|
this.CollaboratorsHandler.removeUserFromAllProjets
|
|
|
|
)
|
|
|
|
done()
|
2019-05-29 05:21:06 -04:00
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2019-07-18 10:18:56 -04:00
|
|
|
describe('deleteProject', () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
this.deleterData = {
|
|
|
|
deletedAt: new Date(),
|
|
|
|
deletedProjectId: this.project._id,
|
|
|
|
deletedProjectOwnerId: this.project.owner_ref,
|
|
|
|
deletedProjectCollaboratorIds: this.project.collaberator_refs,
|
|
|
|
deletedProjectReadOnlyIds: this.project.readOnly_refs,
|
|
|
|
deletedProjectReadWriteTokenAccessIds: this.project
|
|
|
|
.tokenAccessReadAndWrite_refs,
|
|
|
|
deletedProjectReadOnlyTokenAccessIds: this.project
|
|
|
|
.tokenAccessReadOnly_refs,
|
|
|
|
deletedProjectReadWriteToken: this.project.tokens.readAndWrite,
|
|
|
|
deletedProjectReadOnlyToken: this.project.tokens.readOnly,
|
|
|
|
deletedProjectLastUpdatedAt: this.project.lastUpdated
|
|
|
|
}
|
|
|
|
|
|
|
|
this.ProjectMock.expects('findOne')
|
|
|
|
.withArgs({ _id: this.project_id })
|
|
|
|
.chain('exec')
|
|
|
|
.resolves(this.project)
|
2019-05-29 05:21:06 -04:00
|
|
|
})
|
|
|
|
|
2019-07-18 10:18:56 -04:00
|
|
|
it('should save a DeletedProject with additional deleterData', done => {
|
|
|
|
this.deleterData.deleterIpAddress = this.ip
|
|
|
|
this.deleterData.deleterId = this.user._id
|
|
|
|
|
|
|
|
this.ProjectMock.expects('remove')
|
|
|
|
.chain('exec')
|
|
|
|
.resolves()
|
|
|
|
this.DeletedProjectMock.expects('create')
|
|
|
|
.withArgs({
|
|
|
|
project: this.project,
|
|
|
|
deleterData: this.deleterData
|
|
|
|
})
|
|
|
|
.resolves()
|
|
|
|
|
|
|
|
this.ProjectDeleter.deleteProject(
|
2019-05-29 05:21:06 -04:00
|
|
|
this.project_id,
|
|
|
|
{ deleterUser: this.user, ipAddress: this.ip },
|
2019-07-18 10:18:56 -04:00
|
|
|
err => {
|
|
|
|
expect(err).not.to.exist
|
|
|
|
this.DeletedProjectMock.verify()
|
|
|
|
done()
|
2019-05-29 05:21:06 -04:00
|
|
|
}
|
|
|
|
)
|
|
|
|
})
|
|
|
|
|
2019-07-18 10:18:56 -04:00
|
|
|
it('should flushProjectToMongoAndDelete in doc updater', done => {
|
|
|
|
this.ProjectMock.expects('remove')
|
|
|
|
.chain('exec')
|
|
|
|
.resolves()
|
|
|
|
this.DeletedProjectMock.expects('create').resolves()
|
|
|
|
|
|
|
|
this.ProjectDeleter.deleteProject(
|
2019-05-29 05:21:06 -04:00
|
|
|
this.project_id,
|
|
|
|
{ deleterUser: this.user, ipAddress: this.ip },
|
|
|
|
() => {
|
|
|
|
this.documentUpdaterHandler.flushProjectToMongoAndDelete
|
|
|
|
.calledWith(this.project_id)
|
|
|
|
.should.equal(true)
|
2019-07-18 10:18:56 -04:00
|
|
|
done()
|
2019-05-29 05:21:06 -04:00
|
|
|
}
|
|
|
|
)
|
|
|
|
})
|
|
|
|
|
2019-07-18 10:18:56 -04:00
|
|
|
it('should removeProjectFromAllTags', done => {
|
|
|
|
this.ProjectMock.expects('remove')
|
|
|
|
.chain('exec')
|
|
|
|
.resolves()
|
|
|
|
this.DeletedProjectMock.expects('create').resolves()
|
|
|
|
|
|
|
|
this.ProjectDeleter.deleteProject(this.project_id, () => {
|
|
|
|
sinon.assert.calledWith(
|
|
|
|
this.TagsHandler.removeProjectFromAllTags,
|
|
|
|
'member-id-1',
|
|
|
|
this.project_id
|
|
|
|
)
|
|
|
|
sinon.assert.calledWith(
|
|
|
|
this.TagsHandler.removeProjectFromAllTags,
|
|
|
|
'member-id-2',
|
|
|
|
this.project_id
|
|
|
|
)
|
|
|
|
done()
|
2019-05-29 05:21:06 -04:00
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2019-07-18 10:18:56 -04:00
|
|
|
it('should remove the project from Mongo', done => {
|
|
|
|
this.ProjectMock.expects('remove')
|
|
|
|
.withArgs({ _id: this.project_id })
|
|
|
|
.chain('exec')
|
|
|
|
.resolves()
|
|
|
|
this.DeletedProjectMock.expects('create').resolves()
|
|
|
|
|
|
|
|
this.ProjectDeleter.deleteProject(this.project_id, () => {
|
|
|
|
this.ProjectMock.verify()
|
|
|
|
done()
|
2019-05-29 05:21:06 -04:00
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2019-07-18 10:18:56 -04:00
|
|
|
describe('expireDeletedProjectsAfterDuration', () => {
|
|
|
|
beforeEach(done => {
|
|
|
|
this.ProjectDeleter.expireDeletedProject = sinon
|
|
|
|
.stub()
|
|
|
|
.callsArgWith(1, null)
|
|
|
|
|
|
|
|
this.DeletedProjectMock.expects('find')
|
|
|
|
.withArgs({
|
|
|
|
'deleterData.deletedAt': {
|
|
|
|
$lt: new Date(moment().subtract(90, 'days'))
|
|
|
|
},
|
|
|
|
project: {
|
|
|
|
$ne: null
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.yields(null, this.deletedProjects)
|
|
|
|
|
|
|
|
this.ProjectDeleter.expireDeletedProjectsAfterDuration(done)
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should call find with a date 90 days earlier than today', () => {
|
|
|
|
this.DeletedProjectMock.verify()
|
2019-05-29 05:21:06 -04:00
|
|
|
})
|
|
|
|
|
2019-07-18 10:18:56 -04:00
|
|
|
it('should call expireDeletedProject', done => {
|
|
|
|
expect(this.ProjectDeleter.expireDeletedProject).to.have.been.calledWith(
|
|
|
|
this.deletedProjects[0].deletedProjectId
|
|
|
|
)
|
|
|
|
done()
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
describe('expireDeletedProject', () => {
|
|
|
|
beforeEach(done => {
|
|
|
|
this.DeletedProjectMock.expects('update')
|
|
|
|
.withArgs(
|
|
|
|
{
|
|
|
|
_id: this.deletedProjects[0]._id
|
|
|
|
},
|
|
|
|
{
|
|
|
|
$set: {
|
|
|
|
'deleterData.deleterIpAddress': null,
|
|
|
|
project: null
|
2019-05-29 05:21:06 -04:00
|
|
|
}
|
2019-07-18 10:18:56 -04:00
|
|
|
}
|
|
|
|
)
|
|
|
|
.chain('exec')
|
|
|
|
.resolves()
|
|
|
|
|
|
|
|
this.DeletedProjectMock.expects('findOne')
|
|
|
|
.withArgs({
|
|
|
|
'deleterData.deletedProjectId': this.deletedProjects[0].project._id
|
|
|
|
})
|
|
|
|
.chain('exec')
|
|
|
|
.resolves(this.deletedProjects[0])
|
|
|
|
|
|
|
|
this.ProjectDeleter.expireDeletedProject(
|
|
|
|
this.deletedProjects[0].project._id,
|
|
|
|
done
|
|
|
|
)
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should find the specified deletedProject and remove its project and ip address', () => {
|
|
|
|
this.DeletedProjectMock.verify()
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should destroy the docs in docstore', () => {
|
|
|
|
expect(this.DocstoreManager.destroyProject).to.have.been.calledWith(
|
|
|
|
this.deletedProjects[0].project._id
|
|
|
|
)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
describe('archiveProject', () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
this.ProjectMock.expects('update')
|
|
|
|
.withArgs(
|
|
|
|
{
|
|
|
|
_id: this.project_id
|
|
|
|
},
|
|
|
|
{
|
|
|
|
$set: { archived: true }
|
|
|
|
}
|
|
|
|
)
|
|
|
|
.yields()
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should update the project', done => {
|
|
|
|
this.ProjectDeleter.archiveProject(this.project_id, () => {
|
|
|
|
this.ProjectMock.verify()
|
|
|
|
done()
|
2019-05-29 05:21:06 -04:00
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2019-07-18 10:18:56 -04:00
|
|
|
describe('restoreProject', () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
this.ProjectMock.expects('update')
|
|
|
|
.withArgs(
|
|
|
|
{
|
|
|
|
_id: this.project_id
|
|
|
|
},
|
|
|
|
{
|
|
|
|
$unset: { archived: true }
|
|
|
|
}
|
|
|
|
)
|
|
|
|
.yields()
|
2019-05-29 05:21:06 -04:00
|
|
|
})
|
|
|
|
|
2019-07-18 10:18:56 -04:00
|
|
|
it('should unset the archive attribute', done => {
|
|
|
|
this.ProjectDeleter.restoreProject(this.project_id, () => {
|
|
|
|
this.ProjectMock.verify()
|
|
|
|
done()
|
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
describe('undeleteProject', () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
this.deletedProject = {
|
|
|
|
_id: 'deleted',
|
|
|
|
project: this.project,
|
|
|
|
deleterData: {
|
|
|
|
deletedProjectId: this.project._id,
|
|
|
|
deletedProjectOwnerId: this.project.owner_ref
|
|
|
|
}
|
|
|
|
}
|
|
|
|
this.purgedProject = {
|
|
|
|
_id: 'purged',
|
|
|
|
deleterData: {
|
|
|
|
deletedProjectId: 'purgedProject',
|
|
|
|
deletedProjectOwnerId: 'potato'
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
this.DeletedProjectMock.expects('findOne')
|
|
|
|
.withArgs({ 'deleterData.deletedProjectId': this.project._id })
|
|
|
|
.chain('exec')
|
|
|
|
.resolves(this.deletedProject)
|
|
|
|
this.DeletedProjectMock.expects('findOne')
|
|
|
|
.withArgs({ 'deleterData.deletedProjectId': 'purgedProject' })
|
|
|
|
.chain('exec')
|
|
|
|
.resolves(this.purgedProject)
|
|
|
|
this.DeletedProjectMock.expects('findOne')
|
|
|
|
.withArgs({ 'deleterData.deletedProjectId': 'wombat' })
|
|
|
|
.chain('exec')
|
|
|
|
.resolves(null)
|
|
|
|
this.DeletedProjectMock.expects('deleteOne')
|
|
|
|
.chain('exec')
|
|
|
|
.resolves()
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should return not found if the project does not exist', done => {
|
|
|
|
this.ProjectDeleter.undeleteProject('wombat', err => {
|
|
|
|
expect(err).to.exist
|
|
|
|
expect(err.name).to.equal('NotFoundError')
|
|
|
|
expect(err.message).to.equal('project_not_found')
|
|
|
|
done()
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should return not found if the project has been expired', done => {
|
|
|
|
this.ProjectDeleter.undeleteProject('purgedProject', err => {
|
|
|
|
expect(err.name).to.equal('NotFoundError')
|
|
|
|
expect(err.message).to.equal('project_too_old_to_restore')
|
|
|
|
done()
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should insert the project into the collection', done => {
|
|
|
|
this.ProjectDeleter.undeleteProject(this.project._id, err => {
|
|
|
|
expect(err).not.to.exist
|
|
|
|
sinon.assert.calledWith(
|
|
|
|
this.db.projects.insert,
|
|
|
|
sinon.match({
|
|
|
|
_id: this.project._id,
|
|
|
|
name: this.project.name
|
|
|
|
})
|
|
|
|
)
|
|
|
|
done()
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should remove the DeletedProject', done => {
|
|
|
|
// need to change the mock just to include the methods we want
|
|
|
|
this.DeletedProjectMock.restore()
|
|
|
|
this.DeletedProjectMock = sinon.mock(DeletedProject)
|
|
|
|
this.DeletedProjectMock.expects('findOne')
|
|
|
|
.withArgs({ 'deleterData.deletedProjectId': this.project._id })
|
|
|
|
.chain('exec')
|
|
|
|
.resolves(this.deletedProject)
|
|
|
|
this.DeletedProjectMock.expects('deleteOne')
|
|
|
|
.withArgs({ _id: 'deleted' })
|
|
|
|
.chain('exec')
|
|
|
|
.resolves()
|
|
|
|
|
|
|
|
this.ProjectDeleter.undeleteProject(this.project._id, err => {
|
|
|
|
expect(err).not.to.exist
|
|
|
|
this.DeletedProjectMock.verify()
|
|
|
|
done()
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should clear the archive bit', done => {
|
|
|
|
this.project.archived = true
|
|
|
|
this.ProjectDeleter.undeleteProject(this.project._id, err => {
|
|
|
|
expect(err).not.to.exist
|
|
|
|
sinon.assert.calledWith(
|
|
|
|
this.db.projects.insert,
|
|
|
|
sinon.match({ archived: undefined })
|
|
|
|
)
|
|
|
|
done()
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should generate a unique name for the project', done => {
|
|
|
|
this.ProjectDeleter.undeleteProject(this.project._id, err => {
|
|
|
|
expect(err).not.to.exist
|
|
|
|
sinon.assert.calledWith(
|
|
|
|
this.ProjectDetailsHandler.generateUniqueName,
|
|
|
|
this.project.owner_ref
|
|
|
|
)
|
|
|
|
done()
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should add a suffix to the project name', done => {
|
|
|
|
this.ProjectDeleter.undeleteProject(this.project._id, err => {
|
|
|
|
expect(err).not.to.exist
|
|
|
|
sinon.assert.calledWith(
|
|
|
|
this.ProjectDetailsHandler.generateUniqueName,
|
|
|
|
this.project.owner_ref,
|
|
|
|
this.project.name + ' (Restored)'
|
|
|
|
)
|
|
|
|
done()
|
2019-05-29 05:21:06 -04:00
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|