mirror of
https://github.com/yu-i-i/overleaf-cep.git
synced 2026-05-31 12:51:35 +02:00
* Add docModified hook in ds-mobile-app module * use Object.entries when iterating over promises * avoid project lookup * update tests GitOrigin-RevId: 88676746f56558a97ce31010b57f5eeb254fefef
256 lines
6.7 KiB
JavaScript
256 lines
6.7 KiB
JavaScript
import { vi } from 'vitest'
|
|
import sinon from 'sinon'
|
|
import MockRequest from '../helpers/MockRequest.js'
|
|
import MockResponse from '../helpers/MockResponse.js'
|
|
import Errors from '../../../../app/src/Features/Errors/Errors.js'
|
|
|
|
const MODULE_PATH =
|
|
'../../../../app/src/Features/Documents/DocumentController.mjs'
|
|
|
|
describe('DocumentController', function () {
|
|
beforeEach(async function (ctx) {
|
|
ctx.res = new MockResponse()
|
|
ctx.req = new MockRequest()
|
|
ctx.next = sinon.stub()
|
|
ctx.doc = { _id: 'doc-id-123' }
|
|
ctx.doc_lines = ['one', 'two', 'three']
|
|
ctx.version = 42
|
|
ctx.ranges = {
|
|
comments: [
|
|
{
|
|
id: 'comment1',
|
|
op: {
|
|
c: 'foo',
|
|
p: 123,
|
|
t: 'comment1',
|
|
},
|
|
},
|
|
{
|
|
id: 'comment2',
|
|
op: {
|
|
c: 'bar',
|
|
p: 456,
|
|
t: 'comment2',
|
|
},
|
|
},
|
|
],
|
|
}
|
|
ctx.pathname = '/a/b/c/file.tex'
|
|
ctx.lastUpdatedAt = new Date().getTime()
|
|
ctx.lastUpdatedBy = 'fake-last-updater-id'
|
|
ctx.rev = 5
|
|
ctx.project = {
|
|
_id: 'project-id-123',
|
|
overleaf: {
|
|
history: {
|
|
id: 1234,
|
|
display: true,
|
|
},
|
|
},
|
|
}
|
|
ctx.resolvedThreadIds = [
|
|
'comment2',
|
|
'comment4', // Comment in project but not in doc
|
|
]
|
|
|
|
ctx.ProjectGetter = {
|
|
promises: {
|
|
getProject: sinon.stub().resolves(ctx.project),
|
|
},
|
|
}
|
|
ctx.ProjectLocator = {
|
|
promises: {
|
|
findElement: sinon
|
|
.stub()
|
|
.resolves({ element: ctx.doc, path: { fileSystem: ctx.pathname } }),
|
|
},
|
|
}
|
|
ctx.ProjectEntityHandler = {
|
|
promises: {
|
|
getDoc: sinon.stub().resolves({
|
|
lines: ctx.doc_lines,
|
|
rev: ctx.rev,
|
|
version: ctx.version,
|
|
ranges: ctx.ranges,
|
|
}),
|
|
},
|
|
}
|
|
ctx.ProjectEntityUpdateHandler = {
|
|
promises: {
|
|
updateDocLines: sinon.stub().resolves(),
|
|
},
|
|
}
|
|
|
|
ctx.ChatApiHandler = {
|
|
promises: {
|
|
getResolvedThreadIds: sinon.stub().resolves(ctx.resolvedThreadIds),
|
|
},
|
|
}
|
|
|
|
ctx.Modules = {
|
|
promises: {
|
|
hooks: {
|
|
fire: sinon.stub().resolves(),
|
|
},
|
|
},
|
|
}
|
|
|
|
vi.doMock('../../../../app/src/Features/Project/ProjectGetter', () => ({
|
|
default: ctx.ProjectGetter,
|
|
}))
|
|
|
|
vi.doMock('../../../../app/src/Features/Project/ProjectLocator', () => ({
|
|
default: ctx.ProjectLocator,
|
|
}))
|
|
|
|
vi.doMock(
|
|
'../../../../app/src/Features/Project/ProjectEntityHandler',
|
|
() => ({
|
|
default: ctx.ProjectEntityHandler,
|
|
})
|
|
)
|
|
|
|
vi.doMock(
|
|
'../../../../app/src/Features/Project/ProjectEntityUpdateHandler',
|
|
() => ({
|
|
default: ctx.ProjectEntityUpdateHandler,
|
|
})
|
|
)
|
|
|
|
vi.doMock('../../../../app/src/Features/Chat/ChatApiHandler', () => ({
|
|
default: ctx.ChatApiHandler,
|
|
}))
|
|
|
|
vi.doMock('../../../../app/src/infrastructure/Modules.js', () => ({
|
|
default: ctx.Modules,
|
|
}))
|
|
|
|
ctx.DocumentController = (await import(MODULE_PATH)).default
|
|
})
|
|
|
|
describe('getDocument', function () {
|
|
beforeEach(function (ctx) {
|
|
ctx.req.params = {
|
|
Project_id: ctx.project._id,
|
|
doc_id: ctx.doc._id,
|
|
}
|
|
})
|
|
|
|
describe('when project exists with project history enabled', function () {
|
|
beforeEach(async function (ctx) {
|
|
await new Promise(resolve => {
|
|
ctx.res.callback = err => {
|
|
resolve(err)
|
|
}
|
|
ctx.DocumentController.getDocument(ctx.req, ctx.res, ctx.next)
|
|
})
|
|
})
|
|
|
|
it('should return the history id and display setting to the client as JSON', function (ctx) {
|
|
ctx.res.type.should.equal('application/json')
|
|
JSON.parse(ctx.res.body).should.deep.equal({
|
|
lines: ctx.doc_lines,
|
|
version: ctx.version,
|
|
ranges: ctx.ranges,
|
|
pathname: ctx.pathname,
|
|
projectHistoryId: ctx.project.overleaf.history.id,
|
|
projectHistoryType: 'project-history',
|
|
resolvedCommentIds: ['comment2'],
|
|
historyRangesSupport: false,
|
|
otMigrationStage: 0,
|
|
})
|
|
})
|
|
})
|
|
|
|
describe('when the project does not exist', function () {
|
|
beforeEach(async function (ctx) {
|
|
await new Promise(resolve => {
|
|
ctx.ProjectGetter.promises.getProject.resolves(null)
|
|
ctx.res.callback = err => {
|
|
resolve(err)
|
|
}
|
|
ctx.DocumentController.getDocument(ctx.req, ctx.res, ctx.next)
|
|
})
|
|
})
|
|
|
|
it('returns a 404', function (ctx) {
|
|
ctx.res.statusCode.should.equal(404)
|
|
})
|
|
})
|
|
})
|
|
|
|
describe('setDocument', function () {
|
|
beforeEach(function (ctx) {
|
|
ctx.req.params = {
|
|
Project_id: ctx.project._id,
|
|
doc_id: ctx.doc._id,
|
|
}
|
|
})
|
|
|
|
describe('when the document exists', function () {
|
|
beforeEach(async function (ctx) {
|
|
await new Promise(resolve => {
|
|
ctx.req.body = {
|
|
lines: ctx.doc_lines,
|
|
version: ctx.version,
|
|
ranges: ctx.ranges,
|
|
lastUpdatedAt: ctx.lastUpdatedAt,
|
|
lastUpdatedBy: ctx.lastUpdatedBy,
|
|
}
|
|
ctx.res.callback = err => {
|
|
resolve(err)
|
|
}
|
|
ctx.DocumentController.setDocument(ctx.req, ctx.res, ctx.next)
|
|
})
|
|
})
|
|
|
|
it('should update the document in Mongo', function (ctx) {
|
|
sinon.assert.calledWith(
|
|
ctx.ProjectEntityUpdateHandler.promises.updateDocLines,
|
|
ctx.project._id,
|
|
ctx.doc._id,
|
|
ctx.doc_lines,
|
|
ctx.version,
|
|
ctx.ranges,
|
|
ctx.lastUpdatedAt,
|
|
ctx.lastUpdatedBy
|
|
)
|
|
})
|
|
|
|
it('should return a successful response', function (ctx) {
|
|
ctx.res.success.should.equal(true)
|
|
})
|
|
|
|
it('should call the docModified hook', function (ctx) {
|
|
sinon.assert.calledWith(
|
|
ctx.Modules.promises.hooks.fire,
|
|
'docModified',
|
|
ctx.project._id,
|
|
ctx.doc._id
|
|
)
|
|
})
|
|
})
|
|
|
|
describe("when the document doesn't exist", function () {
|
|
beforeEach(async function (ctx) {
|
|
await new Promise(resolve => {
|
|
ctx.ProjectEntityUpdateHandler.promises.updateDocLines.rejects(
|
|
new Errors.NotFoundError('document does not exist')
|
|
)
|
|
ctx.req.body = { lines: ctx.doc_lines }
|
|
ctx.next.callsFake(() => {
|
|
resolve()
|
|
})
|
|
ctx.DocumentController.setDocument(ctx.req, ctx.res, ctx.next)
|
|
})
|
|
})
|
|
|
|
it('should call next with the NotFoundError', function (ctx) {
|
|
ctx.next
|
|
.calledWith(sinon.match.instanceOf(Errors.NotFoundError))
|
|
.should.equal(true)
|
|
})
|
|
})
|
|
})
|
|
})
|