2014-02-12 05:23:40 -05:00
|
|
|
sinon = require('sinon')
|
|
|
|
chai = require('chai')
|
|
|
|
should = chai.should()
|
2014-10-16 11:34:21 -04:00
|
|
|
assert = require("chai").assert
|
2014-02-12 05:23:40 -05:00
|
|
|
expect = chai.expect
|
|
|
|
modulePath = "../../../../app/js/Features/Compile/CompileController.js"
|
|
|
|
SandboxedModule = require('sandboxed-module')
|
|
|
|
MockRequest = require "../helpers/MockRequest"
|
|
|
|
MockResponse = require "../helpers/MockResponse"
|
|
|
|
|
|
|
|
describe "CompileController", ->
|
|
|
|
beforeEach ->
|
|
|
|
@CompileManager =
|
|
|
|
compile: sinon.stub()
|
2014-10-16 11:02:31 -04:00
|
|
|
@ClsiManager = {}
|
2014-10-16 11:34:21 -04:00
|
|
|
@UserGetter =
|
|
|
|
getUser:sinon.stub()
|
2014-02-12 05:23:40 -05:00
|
|
|
@CompileController = SandboxedModule.require modulePath, requires:
|
|
|
|
"settings-sharelatex": @settings =
|
|
|
|
apis:
|
|
|
|
clsi:
|
|
|
|
url: "clsi.example.com"
|
2014-03-27 13:00:41 -04:00
|
|
|
"request": @request = sinon.stub()
|
2014-02-12 05:23:40 -05:00
|
|
|
"../../models/Project": Project: @Project = {}
|
|
|
|
"logger-sharelatex": @logger = { log: sinon.stub(), error: sinon.stub() }
|
|
|
|
"../../infrastructure/Metrics": @Metrics = { inc: sinon.stub() }
|
|
|
|
"./CompileManager":@CompileManager
|
2014-10-16 11:34:21 -04:00
|
|
|
"../User/UserGetter":@UserGetter
|
2014-05-19 11:10:41 -04:00
|
|
|
"./ClsiManager": @ClsiManager
|
2014-05-19 10:28:35 -04:00
|
|
|
"../Authentication/AuthenticationController": @AuthenticationController = {}
|
2014-02-12 05:23:40 -05:00
|
|
|
@project_id = "project-id"
|
2014-10-16 11:34:21 -04:00
|
|
|
@user =
|
|
|
|
features:
|
|
|
|
compileGroup: "premium"
|
|
|
|
compileTimeout: 100
|
2014-02-12 05:23:40 -05:00
|
|
|
@next = sinon.stub()
|
|
|
|
@req = new MockRequest()
|
|
|
|
@res = new MockResponse()
|
|
|
|
|
2014-05-19 10:28:35 -04:00
|
|
|
describe "compile", ->
|
2014-10-17 08:23:29 -04:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2014-05-19 10:28:35 -04:00
|
|
|
describe "when not an auto compile", ->
|
|
|
|
beforeEach ->
|
|
|
|
@req.params =
|
|
|
|
Project_id: @project_id
|
|
|
|
@AuthenticationController.getLoggedInUserId = sinon.stub().callsArgWith(1, null, @user_id = "mock-user-id")
|
|
|
|
@CompileManager.compile = sinon.stub().callsArgWith(3, null, @status = "success", @outputFiles = ["mock-output-files"])
|
2014-10-16 11:34:21 -04:00
|
|
|
@UserGetter.getUser.callsArgWith(2, null, @user)
|
2014-05-19 10:28:35 -04:00
|
|
|
@CompileController.compile @req, @res, @next
|
|
|
|
|
|
|
|
it "should look up the user id", ->
|
|
|
|
@AuthenticationController.getLoggedInUserId
|
|
|
|
.calledWith(@req)
|
|
|
|
.should.equal true
|
|
|
|
|
|
|
|
it "should do the compile without the auto compile flag", ->
|
|
|
|
@CompileManager.compile
|
2014-10-16 11:34:21 -04:00
|
|
|
.calledWith(@project_id, @user_id, { isAutoCompile: false, settingsOverride:{timeout:@user.features.compileTimeout, compiler:@user.features.compileGroup} })
|
2014-05-19 10:28:35 -04:00
|
|
|
.should.equal true
|
|
|
|
|
|
|
|
it "should set the content-type of the response to application/json", ->
|
|
|
|
@res.contentType
|
|
|
|
.calledWith("application/json")
|
|
|
|
.should.equal true
|
|
|
|
|
|
|
|
it "should send a successful response reporting the status and files", ->
|
|
|
|
@res.statusCode.should.equal 200
|
|
|
|
@res.body.should.equal JSON.stringify({
|
|
|
|
status: @status
|
|
|
|
outputFiles: @outputFiles
|
|
|
|
})
|
|
|
|
|
2014-10-16 11:34:21 -04:00
|
|
|
it "should get the compile timeout from the users features",->
|
|
|
|
@UserGetter.getUser.args[0][0].should.equal @user_id
|
|
|
|
assert.deepEqual @UserGetter.getUser.args[0][1], {"features.compileGroup":1, "features.compileTimeout":1}
|
|
|
|
|
2014-10-17 08:23:29 -04:00
|
|
|
it "should set the timeout", ->
|
|
|
|
assert @res.timout > 1000 * 60 * 3
|
2014-10-16 11:34:21 -04:00
|
|
|
|
2014-05-19 10:28:35 -04:00
|
|
|
describe "when an auto compile", ->
|
|
|
|
beforeEach ->
|
|
|
|
@req.params =
|
|
|
|
Project_id: @project_id
|
|
|
|
@req.query =
|
|
|
|
auto_compile: "true"
|
|
|
|
@AuthenticationController.getLoggedInUserId = sinon.stub().callsArgWith(1, null, @user_id = "mock-user-id")
|
|
|
|
@CompileManager.compile = sinon.stub().callsArgWith(3, null, @status = "success", @outputFiles = ["mock-output-files"])
|
2014-10-16 11:34:21 -04:00
|
|
|
@UserGetter.getUser.callsArgWith(2, null, @user)
|
2014-05-19 10:28:35 -04:00
|
|
|
@CompileController.compile @req, @res, @next
|
|
|
|
|
|
|
|
it "should do the compile with the auto compile flag", ->
|
2014-10-16 11:34:21 -04:00
|
|
|
@CompileManager.compile.calledWith(@project_id, @user_id, { isAutoCompile: true, settingsOverride:{timeout:@user.features.compileTimeout, compiler:@user.features.compileGroup} }).should.equal true
|
2014-05-19 10:28:35 -04:00
|
|
|
|
2014-02-12 05:23:40 -05:00
|
|
|
describe "downloadPdf", ->
|
|
|
|
beforeEach ->
|
|
|
|
@req.params =
|
|
|
|
Project_id: @project_id
|
|
|
|
@project =
|
|
|
|
getSafeProjectName: () => @safe_name = "safe-name"
|
|
|
|
|
|
|
|
@Project.findById = sinon.stub().callsArgWith(2, null, @project)
|
|
|
|
|
|
|
|
describe "when downloading for embedding", ->
|
|
|
|
beforeEach ->
|
|
|
|
@project.useClsi2 = true
|
|
|
|
@CompileController.proxyToClsi = sinon.stub()
|
|
|
|
@CompileController.downloadPdf(@req, @res, @next)
|
|
|
|
|
|
|
|
it "should look up the project", ->
|
|
|
|
@Project.findById
|
|
|
|
.calledWith(@project_id, {name: 1})
|
|
|
|
.should.equal true
|
|
|
|
|
|
|
|
it "should set the content-type of the response to application/pdf", ->
|
|
|
|
@res.contentType
|
|
|
|
.calledWith("application/pdf")
|
|
|
|
.should.equal true
|
|
|
|
|
|
|
|
it "should set the content-disposition header with the project name", ->
|
|
|
|
@res.header
|
|
|
|
.calledWith("Content-Disposition", "filename=#{@safe_name}.pdf")
|
|
|
|
.should.equal true
|
|
|
|
|
|
|
|
it "should increment the pdf-downloads metric", ->
|
|
|
|
@Metrics.inc
|
|
|
|
.calledWith("pdf-downloads")
|
|
|
|
.should.equal true
|
|
|
|
|
|
|
|
it "should proxy the PDF from the CLSI", ->
|
|
|
|
@CompileController.proxyToClsi
|
|
|
|
.calledWith("/project/#{@project_id}/output/output.pdf", @req, @res, @next)
|
|
|
|
.should.equal true
|
|
|
|
|
|
|
|
describe "proxyToClsi", ->
|
|
|
|
beforeEach ->
|
2014-03-27 13:00:41 -04:00
|
|
|
@request.returns(@proxy = {
|
2014-02-12 05:23:40 -05:00
|
|
|
pipe: sinon.stub()
|
|
|
|
on: sinon.stub()
|
|
|
|
})
|
2014-03-27 13:00:41 -04:00
|
|
|
@upstream =
|
|
|
|
statusCode: 204
|
|
|
|
headers: { "mock": "header" }
|
|
|
|
@req.method = "mock-method"
|
2014-10-21 04:16:23 -04:00
|
|
|
@CompileController.proxyToClsi(@url = "/test", @req, @res, @next)
|
|
|
|
|
|
|
|
it "should open a request to the CLSI", ->
|
|
|
|
@request
|
|
|
|
.calledWith(
|
|
|
|
method: @req.method
|
|
|
|
url: "#{@settings.apis.clsi.url}#{@url}",
|
|
|
|
timeout: 60 * 1000
|
|
|
|
)
|
|
|
|
.should.equal true
|
2014-02-12 05:23:40 -05:00
|
|
|
|
2014-10-21 04:16:23 -04:00
|
|
|
it "should pass the request on to the client", ->
|
|
|
|
@proxy.pipe
|
|
|
|
.calledWith(@res)
|
|
|
|
.should.equal true
|
2014-02-12 05:23:40 -05:00
|
|
|
|
2014-10-21 04:16:23 -04:00
|
|
|
it "should bind an error handle to the request proxy", ->
|
|
|
|
@proxy.on.calledWith("error").should.equal true
|
2014-02-12 05:23:40 -05:00
|
|
|
|
2014-03-27 13:00:41 -04:00
|
|
|
describe "deleteAuxFiles", ->
|
|
|
|
beforeEach ->
|
2014-05-19 11:10:41 -04:00
|
|
|
@ClsiManager.deleteAuxFiles = sinon.stub().callsArg(1)
|
2014-03-27 13:00:41 -04:00
|
|
|
@req.params =
|
|
|
|
Project_id: @project_id
|
2014-05-19 11:10:41 -04:00
|
|
|
@res.send = sinon.stub()
|
2014-03-27 13:00:41 -04:00
|
|
|
@CompileController.deleteAuxFiles @req, @res, @next
|
|
|
|
|
|
|
|
it "should proxy to the CLSI", ->
|
2014-05-19 11:10:41 -04:00
|
|
|
@ClsiManager.deleteAuxFiles
|
|
|
|
.calledWith(@project_id)
|
|
|
|
.should.equal true
|
|
|
|
|
|
|
|
it "should return a 200", ->
|
|
|
|
@res.send
|
|
|
|
.calledWith(200)
|
2014-03-27 13:00:41 -04:00
|
|
|
.should.equal true
|
2014-02-12 05:23:40 -05:00
|
|
|
|
|
|
|
describe "compileAndDownloadPdf", ->
|
|
|
|
beforeEach ->
|
|
|
|
@req =
|
|
|
|
params:
|
|
|
|
project_id:@project_id
|
|
|
|
@CompileManager.compile.callsArgWith(3)
|
|
|
|
@CompileController.proxyToClsi = sinon.stub()
|
|
|
|
@res =
|
|
|
|
send:=>
|
|
|
|
|
|
|
|
it "should call compile in the compile manager", (done)->
|
|
|
|
@CompileController.compileAndDownloadPdf @req, @res
|
|
|
|
@CompileManager.compile.calledWith(@project_id).should.equal true
|
|
|
|
done()
|
|
|
|
|
|
|
|
it "should proxy the res to the clsi with correct url", (done)->
|
|
|
|
@CompileController.compileAndDownloadPdf @req, @res
|
|
|
|
@CompileController.proxyToClsi.calledWith("/project/#{@project_id}/output/output.pdf", @req, @res).should.equal true
|
|
|
|
done()
|