2020-02-16 09:03:03 -05:00
|
|
|
/* eslint-disable
|
|
|
|
camelcase,
|
|
|
|
handle-callback-err,
|
|
|
|
no-unused-vars,
|
|
|
|
*/
|
|
|
|
// TODO: This file was created by bulk-decaffeinate.
|
|
|
|
// Fix any style issues and re-enable lint.
|
2020-02-16 09:03:02 -05:00
|
|
|
/*
|
|
|
|
* decaffeinate suggestions:
|
|
|
|
* DS102: Remove unnecessary code created because of implicit returns
|
|
|
|
* DS207: Consider shorter variations of null checks
|
|
|
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/master/docs/suggestions.md
|
|
|
|
*/
|
2020-08-28 08:13:19 -04:00
|
|
|
const { db, ObjectId } = require('../../../app/js/mongodb')
|
2021-03-12 17:16:01 -05:00
|
|
|
const { expect } = require('chai')
|
2020-02-16 09:03:09 -05:00
|
|
|
const DocstoreApp = require('./helpers/DocstoreApp')
|
2020-07-23 14:43:33 -04:00
|
|
|
const Errors = require('../../../app/js/Errors')
|
2021-07-12 12:47:20 -04:00
|
|
|
const Settings = require('@overleaf/settings')
|
2014-04-29 11:36:10 -04:00
|
|
|
|
2020-02-16 09:03:09 -05:00
|
|
|
const DocstoreClient = require('./helpers/DocstoreClient')
|
2014-04-29 11:36:10 -04:00
|
|
|
|
2021-02-15 08:13:48 -05:00
|
|
|
function deleteTestSuite(deleteDoc) {
|
2020-05-28 09:20:54 -04:00
|
|
|
beforeEach(function (done) {
|
2020-02-16 09:03:09 -05:00
|
|
|
this.project_id = ObjectId()
|
|
|
|
this.doc_id = ObjectId()
|
|
|
|
this.lines = ['original', 'lines']
|
|
|
|
this.version = 42
|
|
|
|
this.ranges = []
|
|
|
|
return DocstoreApp.ensureRunning(() => {
|
|
|
|
return DocstoreClient.createDoc(
|
|
|
|
this.project_id,
|
|
|
|
this.doc_id,
|
|
|
|
this.lines,
|
|
|
|
this.version,
|
|
|
|
this.ranges,
|
2021-07-13 07:04:48 -04:00
|
|
|
error => {
|
2020-02-16 09:03:09 -05:00
|
|
|
if (error != null) {
|
|
|
|
throw error
|
|
|
|
}
|
|
|
|
return done()
|
|
|
|
}
|
|
|
|
)
|
|
|
|
})
|
|
|
|
})
|
2014-04-29 11:36:10 -04:00
|
|
|
|
2021-01-12 12:29:06 -05:00
|
|
|
it('should show as not deleted on /deleted', function (done) {
|
|
|
|
DocstoreClient.isDocDeleted(
|
|
|
|
this.project_id,
|
|
|
|
this.doc_id,
|
|
|
|
(error, res, body) => {
|
|
|
|
if (error) return done(error)
|
|
|
|
expect(res.statusCode).to.equal(200)
|
|
|
|
expect(body).to.have.property('deleted').to.equal(false)
|
|
|
|
done()
|
|
|
|
}
|
|
|
|
)
|
|
|
|
})
|
|
|
|
|
2020-05-28 09:20:54 -04:00
|
|
|
describe('when the doc exists', function () {
|
|
|
|
beforeEach(function (done) {
|
2021-02-15 08:13:48 -05:00
|
|
|
deleteDoc(this.project_id, this.doc_id, (error, res, doc) => {
|
|
|
|
this.res = res
|
|
|
|
return done()
|
|
|
|
})
|
2020-02-16 09:03:09 -05:00
|
|
|
})
|
2014-04-29 11:36:10 -04:00
|
|
|
|
2020-05-28 09:20:54 -04:00
|
|
|
afterEach(function (done) {
|
2020-02-16 09:03:09 -05:00
|
|
|
return db.docs.remove({ _id: this.doc_id }, done)
|
|
|
|
})
|
2014-04-29 11:36:10 -04:00
|
|
|
|
2021-01-12 12:29:06 -05:00
|
|
|
it('should mark the doc as deleted on /deleted', function (done) {
|
|
|
|
DocstoreClient.isDocDeleted(
|
|
|
|
this.project_id,
|
|
|
|
this.doc_id,
|
|
|
|
(error, res, body) => {
|
|
|
|
if (error) return done(error)
|
|
|
|
expect(res.statusCode).to.equal(200)
|
|
|
|
expect(body).to.have.property('deleted').to.equal(true)
|
|
|
|
done()
|
|
|
|
}
|
|
|
|
)
|
|
|
|
})
|
|
|
|
|
2021-01-04 08:36:19 -05:00
|
|
|
it('should insert a deleted doc into the docs collection', function (done) {
|
2020-08-24 05:01:08 -04:00
|
|
|
return db.docs.find({ _id: this.doc_id }).toArray((error, docs) => {
|
2020-02-16 09:03:09 -05:00
|
|
|
docs[0]._id.should.deep.equal(this.doc_id)
|
|
|
|
docs[0].lines.should.deep.equal(this.lines)
|
|
|
|
docs[0].deleted.should.equal(true)
|
|
|
|
return done()
|
|
|
|
})
|
|
|
|
})
|
2021-01-04 08:36:19 -05:00
|
|
|
|
|
|
|
it('should not export the doc to s3', function (done) {
|
|
|
|
setTimeout(() => {
|
2021-07-13 07:04:48 -04:00
|
|
|
DocstoreClient.getS3Doc(this.project_id, this.doc_id, error => {
|
2021-01-04 08:36:19 -05:00
|
|
|
expect(error).to.be.instanceOf(Errors.NotFoundError)
|
|
|
|
done()
|
|
|
|
})
|
|
|
|
}, 1000)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
describe('when archiveOnSoftDelete is enabled', function () {
|
|
|
|
let archiveOnSoftDelete
|
2021-02-15 08:13:48 -05:00
|
|
|
beforeEach('overwrite settings', function () {
|
2021-01-04 08:36:19 -05:00
|
|
|
archiveOnSoftDelete = Settings.docstore.archiveOnSoftDelete
|
|
|
|
Settings.docstore.archiveOnSoftDelete = true
|
|
|
|
})
|
2021-02-15 08:13:48 -05:00
|
|
|
afterEach('restore settings', function () {
|
2021-01-04 08:36:19 -05:00
|
|
|
Settings.docstore.archiveOnSoftDelete = archiveOnSoftDelete
|
|
|
|
})
|
|
|
|
|
2021-02-15 08:13:48 -05:00
|
|
|
beforeEach('delete Doc', function (done) {
|
|
|
|
deleteDoc(this.project_id, this.doc_id, (error, res) => {
|
2021-01-04 08:36:19 -05:00
|
|
|
this.res = res
|
|
|
|
done()
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
beforeEach(function waitForBackgroundFlush(done) {
|
|
|
|
setTimeout(done, 500)
|
|
|
|
})
|
|
|
|
|
|
|
|
afterEach(function cleanupDoc(done) {
|
|
|
|
db.docs.remove({ _id: this.doc_id }, done)
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should set the deleted flag in the doc', function (done) {
|
|
|
|
db.docs.findOne({ _id: this.doc_id }, (error, doc) => {
|
|
|
|
if (error) {
|
|
|
|
return done(error)
|
|
|
|
}
|
|
|
|
expect(doc.deleted).to.equal(true)
|
|
|
|
done()
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should set inS3 and unset lines and ranges in the doc', function (done) {
|
|
|
|
db.docs.findOne({ _id: this.doc_id }, (error, doc) => {
|
|
|
|
if (error) {
|
|
|
|
return done(error)
|
|
|
|
}
|
|
|
|
expect(doc.lines).to.not.exist
|
|
|
|
expect(doc.ranges).to.not.exist
|
|
|
|
expect(doc.inS3).to.equal(true)
|
|
|
|
done()
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should set the doc in s3 correctly', function (done) {
|
|
|
|
DocstoreClient.getS3Doc(this.project_id, this.doc_id, (error, s3_doc) => {
|
|
|
|
if (error) {
|
|
|
|
return done(error)
|
|
|
|
}
|
|
|
|
expect(s3_doc.lines).to.deep.equal(this.lines)
|
|
|
|
expect(s3_doc.ranges).to.deep.equal(this.ranges)
|
|
|
|
done()
|
|
|
|
})
|
|
|
|
})
|
2020-02-16 09:03:09 -05:00
|
|
|
})
|
2014-04-29 11:36:10 -04:00
|
|
|
|
2021-01-12 12:29:06 -05:00
|
|
|
describe('when the doc exists in another project', function () {
|
|
|
|
const otherProjectId = ObjectId()
|
|
|
|
|
|
|
|
it('should show as not existing on /deleted', function (done) {
|
|
|
|
DocstoreClient.isDocDeleted(otherProjectId, this.doc_id, (error, res) => {
|
|
|
|
if (error) return done(error)
|
|
|
|
expect(res.statusCode).to.equal(404)
|
|
|
|
done()
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should return a 404 when trying to delete', function (done) {
|
2021-02-15 08:13:48 -05:00
|
|
|
deleteDoc(otherProjectId, this.doc_id, (error, res) => {
|
2021-01-12 12:29:06 -05:00
|
|
|
if (error) return done(error)
|
|
|
|
expect(res.statusCode).to.equal(404)
|
|
|
|
done()
|
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2020-05-28 09:20:54 -04:00
|
|
|
return describe('when the doc does not exist', function () {
|
2021-01-12 12:29:06 -05:00
|
|
|
it('should show as not existing on /deleted', function (done) {
|
|
|
|
const missing_doc_id = ObjectId()
|
|
|
|
DocstoreClient.isDocDeleted(
|
|
|
|
this.project_id,
|
|
|
|
missing_doc_id,
|
|
|
|
(error, res) => {
|
|
|
|
if (error) return done(error)
|
|
|
|
expect(res.statusCode).to.equal(404)
|
|
|
|
done()
|
|
|
|
}
|
|
|
|
)
|
|
|
|
})
|
|
|
|
|
2020-05-28 09:20:54 -04:00
|
|
|
return it('should return a 404', function (done) {
|
2020-02-16 09:03:09 -05:00
|
|
|
const missing_doc_id = ObjectId()
|
2021-02-15 08:13:48 -05:00
|
|
|
deleteDoc(this.project_id, missing_doc_id, (error, res, doc) => {
|
|
|
|
res.statusCode.should.equal(404)
|
|
|
|
return done()
|
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
describe('Delete via PATCH', function () {
|
|
|
|
deleteTestSuite(DocstoreClient.deleteDoc)
|
|
|
|
|
|
|
|
describe('when providing a custom doc name in the delete request', function () {
|
|
|
|
beforeEach(function (done) {
|
|
|
|
DocstoreClient.deleteDocWithName(
|
2020-02-16 09:03:09 -05:00
|
|
|
this.project_id,
|
2021-02-15 08:13:48 -05:00
|
|
|
this.doc_id,
|
|
|
|
'wombat.tex',
|
|
|
|
done
|
|
|
|
)
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should insert the doc name into the docs collection', function (done) {
|
|
|
|
db.docs.find({ _id: this.doc_id }).toArray((error, docs) => {
|
|
|
|
if (error) return done(error)
|
|
|
|
expect(docs[0].name).to.equal('wombat.tex')
|
|
|
|
done()
|
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
describe('when providing a custom deletedAt date in the delete request', function () {
|
|
|
|
beforeEach('record date and delay', function (done) {
|
|
|
|
this.deletedAt = new Date()
|
|
|
|
setTimeout(done, 5)
|
|
|
|
})
|
|
|
|
|
|
|
|
beforeEach('perform deletion with past date', function (done) {
|
|
|
|
DocstoreClient.deleteDocWithDate(
|
|
|
|
this.project_id,
|
|
|
|
this.doc_id,
|
|
|
|
this.deletedAt,
|
|
|
|
done
|
|
|
|
)
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should insert the date into the docs collection', function (done) {
|
|
|
|
db.docs.find({ _id: this.doc_id }).toArray((error, docs) => {
|
|
|
|
if (error) return done(error)
|
|
|
|
expect(docs[0].deletedAt.toISOString()).to.equal(
|
|
|
|
this.deletedAt.toISOString()
|
|
|
|
)
|
|
|
|
done()
|
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
describe('when providing no doc name in the delete request', function () {
|
|
|
|
beforeEach(function (done) {
|
|
|
|
DocstoreClient.deleteDocWithName(
|
|
|
|
this.project_id,
|
|
|
|
this.doc_id,
|
|
|
|
'',
|
|
|
|
(error, res) => {
|
|
|
|
this.res = res
|
|
|
|
done(error)
|
|
|
|
}
|
|
|
|
)
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should reject the request', function () {
|
|
|
|
expect(this.res.statusCode).to.equal(400)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
describe('when providing no date in the delete request', function () {
|
|
|
|
beforeEach(function (done) {
|
|
|
|
DocstoreClient.deleteDocWithDate(
|
|
|
|
this.project_id,
|
|
|
|
this.doc_id,
|
|
|
|
'',
|
|
|
|
(error, res) => {
|
|
|
|
this.res = res
|
|
|
|
done(error)
|
2020-02-16 09:03:09 -05:00
|
|
|
}
|
|
|
|
)
|
|
|
|
})
|
2021-02-15 08:13:48 -05:00
|
|
|
|
|
|
|
it('should reject the request', function () {
|
|
|
|
expect(this.res.statusCode).to.equal(400)
|
|
|
|
})
|
2020-02-16 09:03:09 -05:00
|
|
|
})
|
2021-02-18 05:10:14 -05:00
|
|
|
|
|
|
|
describe('before deleting anything', function () {
|
|
|
|
it('should show nothing in deleted docs response', function (done) {
|
|
|
|
DocstoreClient.getAllDeletedDocs(
|
|
|
|
this.project_id,
|
|
|
|
(error, deletedDocs) => {
|
|
|
|
if (error) return done(error)
|
|
|
|
expect(deletedDocs).to.deep.equal([])
|
|
|
|
done()
|
|
|
|
}
|
|
|
|
)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
describe('when the doc gets a name on delete', function () {
|
|
|
|
beforeEach(function (done) {
|
|
|
|
DocstoreClient.deleteDoc(this.project_id, this.doc_id, done)
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should show the doc in deleted docs response', function (done) {
|
|
|
|
DocstoreClient.getAllDeletedDocs(
|
|
|
|
this.project_id,
|
|
|
|
(error, deletedDocs) => {
|
|
|
|
if (error) return done(error)
|
|
|
|
expect(deletedDocs).to.deep.equal([
|
2021-07-13 07:04:48 -04:00
|
|
|
{ _id: this.doc_id.toString(), name: 'main.tex' },
|
2021-02-18 05:10:14 -05:00
|
|
|
])
|
|
|
|
done()
|
|
|
|
}
|
|
|
|
)
|
|
|
|
})
|
|
|
|
|
|
|
|
describe('after deleting multiple docs', function () {
|
|
|
|
beforeEach('create doc2', function (done) {
|
|
|
|
this.doc_id2 = ObjectId()
|
|
|
|
DocstoreClient.createDoc(
|
|
|
|
this.project_id,
|
|
|
|
this.doc_id2,
|
|
|
|
this.lines,
|
|
|
|
this.version,
|
|
|
|
this.ranges,
|
|
|
|
done
|
|
|
|
)
|
|
|
|
})
|
|
|
|
beforeEach('delete doc2', function (done) {
|
|
|
|
DocstoreClient.deleteDocWithName(
|
|
|
|
this.project_id,
|
|
|
|
this.doc_id2,
|
|
|
|
'two.tex',
|
|
|
|
done
|
|
|
|
)
|
|
|
|
})
|
|
|
|
beforeEach('create doc3', function (done) {
|
|
|
|
this.doc_id3 = ObjectId()
|
|
|
|
DocstoreClient.createDoc(
|
|
|
|
this.project_id,
|
|
|
|
this.doc_id3,
|
|
|
|
this.lines,
|
|
|
|
this.version,
|
|
|
|
this.ranges,
|
|
|
|
done
|
|
|
|
)
|
|
|
|
})
|
|
|
|
beforeEach('delete doc3', function (done) {
|
|
|
|
DocstoreClient.deleteDocWithName(
|
|
|
|
this.project_id,
|
|
|
|
this.doc_id3,
|
|
|
|
'three.tex',
|
|
|
|
done
|
|
|
|
)
|
|
|
|
})
|
|
|
|
it('should show all the docs as deleted', function (done) {
|
|
|
|
DocstoreClient.getAllDeletedDocs(
|
|
|
|
this.project_id,
|
|
|
|
(error, deletedDocs) => {
|
|
|
|
if (error) return done(error)
|
|
|
|
|
|
|
|
expect(deletedDocs).to.deep.equal([
|
|
|
|
{ _id: this.doc_id3.toString(), name: 'three.tex' },
|
|
|
|
{ _id: this.doc_id2.toString(), name: 'two.tex' },
|
2021-07-13 07:04:48 -04:00
|
|
|
{ _id: this.doc_id.toString(), name: 'main.tex' },
|
2021-02-18 05:10:14 -05:00
|
|
|
])
|
|
|
|
done()
|
|
|
|
}
|
|
|
|
)
|
|
|
|
})
|
|
|
|
|
|
|
|
describe('with one more than max_deleted_docs permits', function () {
|
|
|
|
let maxDeletedDocsBefore
|
|
|
|
beforeEach(function () {
|
|
|
|
maxDeletedDocsBefore = Settings.max_deleted_docs
|
|
|
|
Settings.max_deleted_docs = 2
|
|
|
|
})
|
|
|
|
afterEach(function () {
|
|
|
|
Settings.max_deleted_docs = maxDeletedDocsBefore
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should omit the first deleted doc', function (done) {
|
|
|
|
DocstoreClient.getAllDeletedDocs(
|
|
|
|
this.project_id,
|
|
|
|
(error, deletedDocs) => {
|
|
|
|
if (error) return done(error)
|
|
|
|
|
|
|
|
expect(deletedDocs).to.deep.equal([
|
|
|
|
{ _id: this.doc_id3.toString(), name: 'three.tex' },
|
2021-07-13 07:04:48 -04:00
|
|
|
{ _id: this.doc_id2.toString(), name: 'two.tex' },
|
2021-02-18 05:10:14 -05:00
|
|
|
// dropped main.tex
|
|
|
|
])
|
|
|
|
done()
|
|
|
|
}
|
|
|
|
)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|
2020-02-16 09:03:09 -05:00
|
|
|
})
|
2014-04-29 11:36:10 -04:00
|
|
|
|
2020-05-28 09:20:54 -04:00
|
|
|
describe("Destroying a project's documents", function () {
|
|
|
|
describe('when the doc exists', function () {
|
|
|
|
beforeEach(function (done) {
|
2020-02-16 09:03:09 -05:00
|
|
|
return db.docOps.insert(
|
|
|
|
{ doc_id: ObjectId(this.doc_id), version: 1 },
|
2020-05-28 09:20:54 -04:00
|
|
|
function (err) {
|
2020-02-16 09:03:09 -05:00
|
|
|
if (err != null) {
|
|
|
|
return done(err)
|
|
|
|
}
|
|
|
|
return DocstoreClient.destroyAllDoc(this.project_id, done)
|
|
|
|
}
|
|
|
|
)
|
|
|
|
})
|
2019-07-02 07:45:54 -04:00
|
|
|
|
2020-05-28 09:20:54 -04:00
|
|
|
it('should remove the doc from the docs collection', function (done) {
|
2020-08-24 05:01:08 -04:00
|
|
|
return db.docs.find({ _id: this.doc_id }).toArray((err, docs) => {
|
2020-02-16 09:03:09 -05:00
|
|
|
expect(err).not.to.exist
|
|
|
|
expect(docs).to.deep.equal([])
|
|
|
|
return done()
|
|
|
|
})
|
|
|
|
})
|
2019-07-02 07:45:54 -04:00
|
|
|
|
2020-05-28 09:20:54 -04:00
|
|
|
return it('should remove the docOps from the docOps collection', function (done) {
|
2020-08-24 05:01:08 -04:00
|
|
|
return db.docOps.find({ doc_id: this.doc_id }).toArray((err, docOps) => {
|
2020-02-16 09:03:09 -05:00
|
|
|
expect(err).not.to.exist
|
|
|
|
expect(docOps).to.deep.equal([])
|
|
|
|
return done()
|
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|
2019-07-02 07:45:54 -04:00
|
|
|
|
2020-05-28 09:20:54 -04:00
|
|
|
return describe('when the doc is archived', function () {
|
|
|
|
beforeEach(function (done) {
|
2021-07-13 07:04:48 -04:00
|
|
|
return DocstoreClient.archiveAllDoc(this.project_id, err => {
|
2020-02-16 09:03:09 -05:00
|
|
|
if (err != null) {
|
|
|
|
return done(err)
|
|
|
|
}
|
|
|
|
return DocstoreClient.destroyAllDoc(this.project_id, done)
|
|
|
|
})
|
|
|
|
})
|
2019-07-02 07:45:54 -04:00
|
|
|
|
2020-05-28 09:20:54 -04:00
|
|
|
it('should remove the doc from the docs collection', function (done) {
|
2020-08-24 05:01:08 -04:00
|
|
|
return db.docs.find({ _id: this.doc_id }).toArray((err, docs) => {
|
2020-02-16 09:03:09 -05:00
|
|
|
expect(err).not.to.exist
|
|
|
|
expect(docs).to.deep.equal([])
|
|
|
|
return done()
|
|
|
|
})
|
|
|
|
})
|
2019-07-02 07:45:54 -04:00
|
|
|
|
2020-05-28 09:20:54 -04:00
|
|
|
it('should remove the docOps from the docOps collection', function (done) {
|
2020-08-24 05:01:08 -04:00
|
|
|
return db.docOps.find({ doc_id: this.doc_id }).toArray((err, docOps) => {
|
2020-02-16 09:03:09 -05:00
|
|
|
expect(err).not.to.exist
|
|
|
|
expect(docOps).to.deep.equal([])
|
|
|
|
return done()
|
|
|
|
})
|
|
|
|
})
|
2019-07-02 07:45:54 -04:00
|
|
|
|
2020-05-28 09:20:54 -04:00
|
|
|
return it('should remove the doc contents from s3', function (done) {
|
2021-07-13 07:04:48 -04:00
|
|
|
return DocstoreClient.getS3Doc(this.project_id, this.doc_id, error => {
|
2020-07-23 14:43:33 -04:00
|
|
|
expect(error).to.be.instanceOf(Errors.NotFoundError)
|
|
|
|
done()
|
|
|
|
})
|
2020-02-16 09:03:09 -05:00
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|