2020-02-19 06:15:25 -05:00
|
|
|
/* eslint-disable
|
|
|
|
no-return-assign,
|
|
|
|
no-unused-vars,
|
|
|
|
*/
|
|
|
|
// TODO: This file was created by bulk-decaffeinate.
|
|
|
|
// Fix any style issues and re-enable lint.
|
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 should = require('chai').should();
|
|
|
|
const SandboxedModule = require('sandboxed-module');
|
|
|
|
const assert = require('assert');
|
|
|
|
const path = require('path');
|
|
|
|
const sinon = require('sinon');
|
|
|
|
const modulePath = path.join(__dirname, "../../../app/js/StaticServerForbidSymlinks");
|
|
|
|
const { expect } = require("chai");
|
|
|
|
|
|
|
|
describe("StaticServerForbidSymlinks", function() {
|
|
|
|
|
|
|
|
beforeEach(function() {
|
|
|
|
|
|
|
|
this.settings = {
|
|
|
|
path: {
|
2015-02-27 10:38:57 -05:00
|
|
|
compilesDir: "/compiles/here"
|
2020-02-19 06:15:08 -05:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
this.fs = {};
|
|
|
|
this.ForbidSymlinks = SandboxedModule.require(modulePath, { requires: {
|
|
|
|
"settings-sharelatex":this.settings,
|
|
|
|
"logger-sharelatex": {
|
|
|
|
log() {},
|
|
|
|
warn() {},
|
|
|
|
error() {}
|
|
|
|
},
|
|
|
|
"fs":this.fs
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
this.dummyStatic = (rootDir, options) =>
|
|
|
|
(req, res, next) =>
|
|
|
|
// console.log "dummyStatic serving file", rootDir, "called with", req.url
|
|
|
|
// serve it
|
2015-02-27 10:38:57 -05:00
|
|
|
next()
|
2020-02-19 06:15:08 -05:00
|
|
|
|
|
|
|
;
|
2015-02-27 10:38:57 -05:00
|
|
|
|
2020-02-19 06:15:08 -05:00
|
|
|
this.StaticServerForbidSymlinks = this.ForbidSymlinks(this.dummyStatic, this.settings.path.compilesDir);
|
|
|
|
this.req = {
|
|
|
|
params: {
|
2015-02-27 10:38:57 -05:00
|
|
|
project_id:"12345"
|
2020-02-19 06:15:08 -05:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
this.res = {};
|
|
|
|
return this.req.url = "/12345/output.pdf";
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
describe("sending a normal file through", function() {
|
|
|
|
beforeEach(function() {
|
|
|
|
return this.fs.realpath = sinon.stub().callsArgWith(1, null, `${this.settings.path.compilesDir}/${this.req.params.project_id}/output.pdf`);
|
|
|
|
});
|
|
|
|
|
|
|
|
return it("should call next", function(done){
|
|
|
|
this.res.sendStatus = function(resCode){
|
|
|
|
resCode.should.equal(200);
|
|
|
|
return done();
|
|
|
|
};
|
|
|
|
return this.StaticServerForbidSymlinks(this.req, this.res, done);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
describe("with a missing file", function() {
|
|
|
|
beforeEach(function() {
|
|
|
|
return this.fs.realpath = sinon.stub().callsArgWith(1, {code: 'ENOENT'}, `${this.settings.path.compilesDir}/${this.req.params.project_id}/unknown.pdf`);
|
|
|
|
});
|
|
|
|
|
|
|
|
return it("should send a 404", function(done){
|
|
|
|
this.res.sendStatus = function(resCode){
|
|
|
|
resCode.should.equal(404);
|
|
|
|
return done();
|
|
|
|
};
|
|
|
|
return this.StaticServerForbidSymlinks(this.req, this.res);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
describe("with a symlink file", function() {
|
|
|
|
beforeEach(function() {
|
|
|
|
return this.fs.realpath = sinon.stub().callsArgWith(1, null, `/etc/${this.req.params.project_id}/output.pdf`);
|
|
|
|
});
|
|
|
|
|
|
|
|
return it("should send a 404", function(done){
|
|
|
|
this.res.sendStatus = function(resCode){
|
|
|
|
resCode.should.equal(404);
|
|
|
|
return done();
|
|
|
|
};
|
|
|
|
return this.StaticServerForbidSymlinks(this.req, this.res);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
describe("with a relative file", function() {
|
|
|
|
beforeEach(function() {
|
|
|
|
return this.req.url = "/12345/../67890/output.pdf";
|
|
|
|
});
|
|
|
|
|
|
|
|
return it("should send a 404", function(done){
|
|
|
|
this.res.sendStatus = function(resCode){
|
|
|
|
resCode.should.equal(404);
|
|
|
|
return done();
|
|
|
|
};
|
|
|
|
return this.StaticServerForbidSymlinks(this.req, this.res);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
describe("with a unnormalized file containing .", function() {
|
|
|
|
beforeEach(function() {
|
|
|
|
return this.req.url = "/12345/foo/./output.pdf";
|
|
|
|
});
|
|
|
|
|
|
|
|
return it("should send a 404", function(done){
|
|
|
|
this.res.sendStatus = function(resCode){
|
|
|
|
resCode.should.equal(404);
|
|
|
|
return done();
|
|
|
|
};
|
|
|
|
return this.StaticServerForbidSymlinks(this.req, this.res);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
describe("with a file containing an empty path", function() {
|
|
|
|
beforeEach(function() {
|
|
|
|
return this.req.url = "/12345/foo//output.pdf";
|
|
|
|
});
|
|
|
|
|
|
|
|
return it("should send a 404", function(done){
|
|
|
|
this.res.sendStatus = function(resCode){
|
|
|
|
resCode.should.equal(404);
|
|
|
|
return done();
|
|
|
|
};
|
|
|
|
return this.StaticServerForbidSymlinks(this.req, this.res);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("with a non-project file", function() {
|
|
|
|
beforeEach(function() {
|
|
|
|
return this.req.url = "/.foo/output.pdf";
|
|
|
|
});
|
|
|
|
|
|
|
|
return it("should send a 404", function(done){
|
|
|
|
this.res.sendStatus = function(resCode){
|
|
|
|
resCode.should.equal(404);
|
|
|
|
return done();
|
|
|
|
};
|
|
|
|
return this.StaticServerForbidSymlinks(this.req, this.res);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("with a file outside the compiledir", function() {
|
|
|
|
beforeEach(function() {
|
|
|
|
return this.req.url = "/../bar/output.pdf";
|
|
|
|
});
|
|
|
|
|
|
|
|
return it("should send a 404", function(done){
|
|
|
|
this.res.sendStatus = function(resCode){
|
|
|
|
resCode.should.equal(404);
|
|
|
|
return done();
|
|
|
|
};
|
|
|
|
return this.StaticServerForbidSymlinks(this.req, this.res);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
describe("with a file with no leading /", function() {
|
|
|
|
beforeEach(function() {
|
|
|
|
return this.req.url = "./../bar/output.pdf";
|
|
|
|
});
|
|
|
|
|
|
|
|
return it("should send a 404", function(done){
|
|
|
|
this.res.sendStatus = function(resCode){
|
|
|
|
resCode.should.equal(404);
|
|
|
|
return done();
|
|
|
|
};
|
|
|
|
return this.StaticServerForbidSymlinks(this.req, this.res);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("with a github style path", function() {
|
|
|
|
beforeEach(function() {
|
|
|
|
this.req.url = "/henryoswald-latex_example/output/output.log";
|
|
|
|
return this.fs.realpath = sinon.stub().callsArgWith(1, null, `${this.settings.path.compilesDir}/henryoswald-latex_example/output/output.log`);
|
|
|
|
});
|
|
|
|
|
|
|
|
return it("should call next", function(done){
|
|
|
|
this.res.sendStatus = function(resCode){
|
|
|
|
resCode.should.equal(200);
|
|
|
|
return done();
|
|
|
|
};
|
|
|
|
return this.StaticServerForbidSymlinks(this.req, this.res, done);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
return describe("with an error from fs.realpath", function() {
|
|
|
|
|
|
|
|
beforeEach(function() {
|
|
|
|
return this.fs.realpath = sinon.stub().callsArgWith(1, "error");
|
|
|
|
});
|
|
|
|
|
|
|
|
return it("should send a 500", function(done){
|
|
|
|
this.res.sendStatus = function(resCode){
|
|
|
|
resCode.should.equal(500);
|
|
|
|
return done();
|
|
|
|
};
|
|
|
|
return this.StaticServerForbidSymlinks(this.req, this.res);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
2015-02-27 10:38:57 -05:00
|
|
|
|