2020-02-19 11:15:25 +00: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 11:15:08 +00:00
|
|
|
/*
|
|
|
|
* decaffeinate suggestions:
|
|
|
|
* DS102: Remove unnecessary code created because of implicit returns
|
|
|
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/master/docs/suggestions.md
|
|
|
|
*/
|
2020-02-19 11:15:37 +00:00
|
|
|
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: {
|
|
|
|
compilesDir: '/compiles/here'
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
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
|
|
|
|
next()
|
|
|
|
|
|
|
|
this.StaticServerForbidSymlinks = this.ForbidSymlinks(
|
|
|
|
this.dummyStatic,
|
|
|
|
this.settings.path.compilesDir
|
|
|
|
)
|
|
|
|
this.req = {
|
|
|
|
params: {
|
|
|
|
project_id: '12345'
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
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)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|