2020-02-19 06:15:08 -05:00
|
|
|
/*
|
|
|
|
* decaffeinate suggestions:
|
|
|
|
* DS102: Remove unnecessary code created because of implicit returns
|
|
|
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/master/docs/suggestions.md
|
|
|
|
*/
|
|
|
|
const SandboxedModule = require('sandboxed-module');
|
|
|
|
const sinon = require('sinon');
|
|
|
|
require('chai').should();
|
|
|
|
const modulePath = require('path').join(__dirname, '../../../app/js/CompileController');
|
|
|
|
const tk = require("timekeeper");
|
|
|
|
|
|
|
|
describe("CompileController", function() {
|
|
|
|
beforeEach(function() {
|
|
|
|
this.CompileController = SandboxedModule.require(modulePath, { requires: {
|
|
|
|
"./CompileManager": (this.CompileManager = {}),
|
|
|
|
"./RequestParser": (this.RequestParser = {}),
|
|
|
|
"settings-sharelatex": (this.Settings = {
|
|
|
|
apis: {
|
|
|
|
clsi: {
|
2014-02-12 12:27:43 -05:00
|
|
|
url: "http://clsi.example.com"
|
2020-02-19 06:15:08 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}),
|
|
|
|
"./ProjectPersistenceManager": (this.ProjectPersistenceManager = {}),
|
|
|
|
"logger-sharelatex": (this.logger = { log: sinon.stub(), error: sinon.stub(), err:sinon.stub(), warn: sinon.stub()})
|
|
|
|
}
|
|
|
|
});
|
|
|
|
this.Settings.externalUrl = "http://www.example.com";
|
|
|
|
this.req = {};
|
|
|
|
this.res = {};
|
|
|
|
return this.next = sinon.stub();
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("compile", function() {
|
|
|
|
beforeEach(function() {
|
|
|
|
this.req.body = {
|
2014-02-12 12:27:43 -05:00
|
|
|
compile: "mock-body"
|
2020-02-19 06:15:08 -05:00
|
|
|
};
|
|
|
|
this.req.params =
|
|
|
|
{project_id: (this.project_id = "project-id-123")};
|
|
|
|
this.request = {
|
2014-02-12 12:27:43 -05:00
|
|
|
compile: "mock-parsed-request"
|
2020-02-19 06:15:08 -05:00
|
|
|
};
|
|
|
|
this.request_with_project_id = {
|
|
|
|
compile: this.request.compile,
|
|
|
|
project_id: this.project_id
|
|
|
|
};
|
|
|
|
this.output_files = [{
|
|
|
|
path: "output.pdf",
|
|
|
|
type: "pdf",
|
2015-02-26 10:30:57 -05:00
|
|
|
build: 1234
|
2014-02-12 12:27:43 -05:00
|
|
|
}, {
|
2020-02-19 06:15:08 -05:00
|
|
|
path: "output.log",
|
|
|
|
type: "log",
|
2015-02-26 10:30:57 -05:00
|
|
|
build: 1234
|
2020-02-19 06:15:08 -05:00
|
|
|
}];
|
|
|
|
this.RequestParser.parse = sinon.stub().callsArgWith(1, null, this.request);
|
|
|
|
this.ProjectPersistenceManager.markProjectAsJustAccessed = sinon.stub().callsArg(1);
|
|
|
|
this.res.status = sinon.stub().returnsThis();
|
|
|
|
return this.res.send = sinon.stub();
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("successfully", function() {
|
|
|
|
beforeEach(function() {
|
|
|
|
this.CompileManager.doCompileWithLock = sinon.stub().callsArgWith(1, null, this.output_files);
|
|
|
|
return this.CompileController.compile(this.req, this.res);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should parse the request", function() {
|
|
|
|
return this.RequestParser.parse
|
|
|
|
.calledWith(this.req.body)
|
|
|
|
.should.equal(true);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should run the compile for the specified project", function() {
|
|
|
|
return this.CompileManager.doCompileWithLock
|
|
|
|
.calledWith(this.request_with_project_id)
|
|
|
|
.should.equal(true);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should mark the project as accessed", function() {
|
|
|
|
return this.ProjectPersistenceManager.markProjectAsJustAccessed
|
|
|
|
.calledWith(this.project_id)
|
|
|
|
.should.equal(true);
|
|
|
|
});
|
|
|
|
|
|
|
|
return it("should return the JSON response", function() {
|
|
|
|
this.res.status.calledWith(200).should.equal(true);
|
|
|
|
return this.res.send
|
|
|
|
.calledWith({
|
|
|
|
compile: {
|
|
|
|
status: "success",
|
|
|
|
error: null,
|
|
|
|
outputFiles: this.output_files.map(file => {
|
|
|
|
return {
|
|
|
|
url: `${this.Settings.apis.clsi.url}/project/${this.project_id}/build/${file.build}/output/${file.path}`,
|
|
|
|
path: file.path,
|
|
|
|
type: file.type,
|
|
|
|
build: file.build
|
|
|
|
};
|
|
|
|
})
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.should.equal(true);
|
|
|
|
});
|
|
|
|
});
|
2014-02-12 12:27:43 -05:00
|
|
|
|
2020-02-19 06:15:08 -05:00
|
|
|
describe("with an error", function() {
|
|
|
|
beforeEach(function() {
|
|
|
|
this.CompileManager.doCompileWithLock = sinon.stub().callsArgWith(1, new Error(this.message = "error message"), null);
|
|
|
|
return this.CompileController.compile(this.req, this.res);
|
|
|
|
});
|
2014-02-12 12:27:43 -05:00
|
|
|
|
2020-02-19 06:15:08 -05:00
|
|
|
return it("should return the JSON response with the error", function() {
|
|
|
|
this.res.status.calledWith(500).should.equal(true);
|
|
|
|
return this.res.send
|
|
|
|
.calledWith({
|
|
|
|
compile: {
|
|
|
|
status: "error",
|
|
|
|
error: this.message,
|
2014-02-12 12:27:43 -05:00
|
|
|
outputFiles: []
|
2020-02-19 06:15:08 -05:00
|
|
|
}
|
|
|
|
})
|
|
|
|
.should.equal(true);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("when the request times out", function() {
|
|
|
|
beforeEach(function() {
|
|
|
|
this.error = new Error(this.message = "container timed out");
|
|
|
|
this.error.timedout = true;
|
|
|
|
this.CompileManager.doCompileWithLock = sinon.stub().callsArgWith(1, this.error, null);
|
|
|
|
return this.CompileController.compile(this.req, this.res);
|
|
|
|
});
|
2014-05-19 07:18:57 -04:00
|
|
|
|
2020-02-19 06:15:08 -05:00
|
|
|
return it("should return the JSON response with the timeout status", function() {
|
|
|
|
this.res.status.calledWith(200).should.equal(true);
|
|
|
|
return this.res.send
|
|
|
|
.calledWith({
|
|
|
|
compile: {
|
|
|
|
status: "timedout",
|
|
|
|
error: this.message,
|
2014-05-19 07:18:57 -04:00
|
|
|
outputFiles: []
|
2020-02-19 06:15:08 -05:00
|
|
|
}
|
|
|
|
})
|
|
|
|
.should.equal(true);
|
|
|
|
});
|
|
|
|
});
|
2014-05-19 07:18:57 -04:00
|
|
|
|
2020-02-19 06:15:08 -05:00
|
|
|
return describe("when the request returns no output files", function() {
|
|
|
|
beforeEach(function() {
|
|
|
|
this.CompileManager.doCompileWithLock = sinon.stub().callsArgWith(1, null, []);
|
|
|
|
return this.CompileController.compile(this.req, this.res);
|
|
|
|
});
|
2014-05-19 07:18:57 -04:00
|
|
|
|
2020-02-19 06:15:08 -05:00
|
|
|
return it("should return the JSON response with the failure status", function() {
|
|
|
|
this.res.status.calledWith(200).should.equal(true);
|
|
|
|
return this.res.send
|
|
|
|
.calledWith({
|
|
|
|
compile: {
|
|
|
|
error: null,
|
|
|
|
status: "failure",
|
2014-05-19 07:18:57 -04:00
|
|
|
outputFiles: []
|
2020-02-19 06:15:08 -05:00
|
|
|
}
|
|
|
|
})
|
|
|
|
.should.equal(true);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("syncFromCode", function() {
|
|
|
|
beforeEach(function() {
|
|
|
|
this.file = "main.tex";
|
|
|
|
this.line = 42;
|
|
|
|
this.column = 5;
|
|
|
|
this.project_id = "mock-project-id";
|
|
|
|
this.req.params =
|
|
|
|
{project_id: this.project_id};
|
|
|
|
this.req.query = {
|
|
|
|
file: this.file,
|
|
|
|
line: this.line.toString(),
|
|
|
|
column: this.column.toString()
|
|
|
|
};
|
|
|
|
this.res.json = sinon.stub();
|
|
|
|
|
|
|
|
this.CompileManager.syncFromCode = sinon.stub().callsArgWith(5, null, (this.pdfPositions = ["mock-positions"]));
|
|
|
|
return this.CompileController.syncFromCode(this.req, this.res, this.next);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should find the corresponding location in the PDF", function() {
|
|
|
|
return this.CompileManager.syncFromCode
|
|
|
|
.calledWith(this.project_id, undefined, this.file, this.line, this.column)
|
|
|
|
.should.equal(true);
|
|
|
|
});
|
|
|
|
|
|
|
|
return it("should return the positions", function() {
|
|
|
|
return this.res.json
|
|
|
|
.calledWith({
|
|
|
|
pdf: this.pdfPositions
|
|
|
|
})
|
|
|
|
.should.equal(true);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("syncFromPdf", function() {
|
|
|
|
beforeEach(function() {
|
|
|
|
this.page = 5;
|
|
|
|
this.h = 100.23;
|
|
|
|
this.v = 45.67;
|
|
|
|
this.project_id = "mock-project-id";
|
|
|
|
this.req.params =
|
|
|
|
{project_id: this.project_id};
|
|
|
|
this.req.query = {
|
|
|
|
page: this.page.toString(),
|
|
|
|
h: this.h.toString(),
|
|
|
|
v: this.v.toString()
|
|
|
|
};
|
|
|
|
this.res.json = sinon.stub();
|
|
|
|
|
|
|
|
this.CompileManager.syncFromPdf = sinon.stub().callsArgWith(5, null, (this.codePositions = ["mock-positions"]));
|
|
|
|
return this.CompileController.syncFromPdf(this.req, this.res, this.next);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should find the corresponding location in the code", function() {
|
|
|
|
return this.CompileManager.syncFromPdf
|
|
|
|
.calledWith(this.project_id, undefined, this.page, this.h, this.v)
|
|
|
|
.should.equal(true);
|
|
|
|
});
|
|
|
|
|
|
|
|
return it("should return the positions", function() {
|
|
|
|
return this.res.json
|
|
|
|
.calledWith({
|
|
|
|
code: this.codePositions
|
|
|
|
})
|
|
|
|
.should.equal(true);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
return describe("wordcount", function() {
|
|
|
|
beforeEach(function() {
|
|
|
|
this.file = "main.tex";
|
|
|
|
this.project_id = "mock-project-id";
|
|
|
|
this.req.params =
|
|
|
|
{project_id: this.project_id};
|
|
|
|
this.req.query = {
|
|
|
|
file: this.file,
|
|
|
|
image: (this.image = "example.com/image")
|
|
|
|
};
|
|
|
|
this.res.json = sinon.stub();
|
|
|
|
|
|
|
|
this.CompileManager.wordcount = sinon.stub().callsArgWith(4, null, (this.texcount = ["mock-texcount"]));
|
|
|
|
return this.CompileController.wordcount(this.req, this.res, this.next);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should return the word count of a file", function() {
|
|
|
|
return this.CompileManager.wordcount
|
|
|
|
.calledWith(this.project_id, undefined, this.file, this.image)
|
|
|
|
.should.equal(true);
|
|
|
|
});
|
|
|
|
|
|
|
|
return it("should return the texcount info", function() {
|
|
|
|
return this.res.json
|
|
|
|
.calledWith({
|
|
|
|
texcount: this.texcount
|
|
|
|
})
|
|
|
|
.should.equal(true);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|