diff --git a/test/exports.js b/test/exports.js
index 57b3265bd6..e5fb6f2a9c 100644
--- a/test/exports.js
+++ b/test/exports.js
@@ -14,6 +14,11 @@ describe('exports', function(){
assert.equal(express.json.length, 1)
})
+ it('should expose static middleware', function () {
+ assert.equal(typeof express.static, 'function')
+ assert.equal(express.static.length, 2)
+ })
+
it('should expose urlencoded middleware', function () {
assert.equal(typeof express.urlencoded, 'function')
assert.equal(express.urlencoded.length, 1)
diff --git a/test/express.static.js b/test/express.static.js
new file mode 100644
index 0000000000..7c9852243e
--- /dev/null
+++ b/test/express.static.js
@@ -0,0 +1,813 @@
+
+var assert = require('assert')
+var Buffer = require('safe-buffer').Buffer
+var express = require('..')
+var path = require('path')
+var request = require('supertest')
+var utils = require('./support/utils')
+
+var fixtures = path.join(__dirname, '/fixtures')
+var relative = path.relative(process.cwd(), fixtures)
+
+var skipRelative = ~relative.indexOf('..') || path.resolve(relative) === relative
+
+describe('express.static()', function () {
+ describe('basic operations', function () {
+ before(function () {
+ this.app = createApp()
+ })
+
+ it('should require root path', function () {
+ assert.throws(express.static.bind(), /root path required/)
+ })
+
+ it('should require root path to be string', function () {
+ assert.throws(express.static.bind(null, 42), /root path.*string/)
+ })
+
+ it('should serve static files', function (done) {
+ request(this.app)
+ .get('/todo.txt')
+ .expect(200, '- groceries', done)
+ })
+
+ it('should support nesting', function (done) {
+ request(this.app)
+ .get('/users/tobi.txt')
+ .expect(200, 'ferret', done)
+ })
+
+ it('should set Content-Type', function (done) {
+ request(this.app)
+ .get('/todo.txt')
+ .expect('Content-Type', 'text/plain; charset=UTF-8')
+ .expect(200, done)
+ })
+
+ it('should set Last-Modified', function (done) {
+ request(this.app)
+ .get('/todo.txt')
+ .expect('Last-Modified', /\d{2} \w{3} \d{4}/)
+ .expect(200, done)
+ })
+
+ it('should default max-age=0', function (done) {
+ request(this.app)
+ .get('/todo.txt')
+ .expect('Cache-Control', 'public, max-age=0')
+ .expect(200, done)
+ })
+
+ it('should support urlencoded pathnames', function (done) {
+ request(this.app)
+ .get('/%25%20of%20dogs.txt')
+ .expect(200, '20%', done)
+ })
+
+ it('should not choke on auth-looking URL', function (done) {
+ request(this.app)
+ .get('//todo@txt')
+ .expect(404, 'Not Found', done)
+ })
+
+ it('should support index.html', function (done) {
+ request(this.app)
+ .get('/users/')
+ .expect(200)
+ .expect('Content-Type', /html/)
+ .expect('
tobi, loki, jane
', done)
+ })
+
+ it('should support ../', function (done) {
+ request(this.app)
+ .get('/users/../todo.txt')
+ .expect(200, '- groceries', done)
+ })
+
+ it('should support HEAD', function (done) {
+ request(this.app)
+ .head('/todo.txt')
+ .expect(200)
+ .expect(utils.shouldNotHaveBody())
+ .end(done)
+ })
+
+ it('should skip POST requests', function (done) {
+ request(this.app)
+ .post('/todo.txt')
+ .expect(404, 'Not Found', done)
+ })
+
+ it('should support conditional requests', function (done) {
+ var app = this.app
+
+ request(app)
+ .get('/todo.txt')
+ .end(function (err, res) {
+ if (err) throw err
+ request(app)
+ .get('/todo.txt')
+ .set('If-None-Match', res.headers.etag)
+ .expect(304, done)
+ })
+ })
+
+ it('should support precondition checks', function (done) {
+ request(this.app)
+ .get('/todo.txt')
+ .set('If-Match', '"foo"')
+ .expect(412, done)
+ })
+
+ it('should serve zero-length files', function (done) {
+ request(this.app)
+ .get('/empty.txt')
+ .expect(200, '', done)
+ })
+
+ it('should ignore hidden files', function (done) {
+ request(this.app)
+ .get('/.name')
+ .expect(404, 'Not Found', done)
+ })
+ });
+
+ (skipRelative ? describe.skip : describe)('current dir', function () {
+ before(function () {
+ this.app = createApp('.')
+ })
+
+ it('should be served with "."', function (done) {
+ var dest = relative.split(path.sep).join('/')
+ request(this.app)
+ .get('/' + dest + '/todo.txt')
+ .expect(200, '- groceries', done)
+ })
+ })
+
+ describe('acceptRanges', function () {
+ describe('when false', function () {
+ it('should not include Accept-Ranges', function (done) {
+ request(createApp(fixtures, { 'acceptRanges': false }))
+ .get('/nums.txt')
+ .expect(utils.shouldNotHaveHeader('Accept-Ranges'))
+ .expect(200, '123456789', done)
+ })
+
+ it('should ignore Rage request header', function (done) {
+ request(createApp(fixtures, { 'acceptRanges': false }))
+ .get('/nums.txt')
+ .set('Range', 'bytes=0-3')
+ .expect(utils.shouldNotHaveHeader('Accept-Ranges'))
+ .expect(utils.shouldNotHaveHeader('Content-Range'))
+ .expect(200, '123456789', done)
+ })
+ })
+
+ describe('when true', function () {
+ it('should include Accept-Ranges', function (done) {
+ request(createApp(fixtures, { 'acceptRanges': true }))
+ .get('/nums.txt')
+ .expect('Accept-Ranges', 'bytes')
+ .expect(200, '123456789', done)
+ })
+
+ it('should obey Rage request header', function (done) {
+ request(createApp(fixtures, { 'acceptRanges': true }))
+ .get('/nums.txt')
+ .set('Range', 'bytes=0-3')
+ .expect('Accept-Ranges', 'bytes')
+ .expect('Content-Range', 'bytes 0-3/9')
+ .expect(206, '1234', done)
+ })
+ })
+ })
+
+ describe('cacheControl', function () {
+ describe('when false', function () {
+ it('should not include Cache-Control', function (done) {
+ request(createApp(fixtures, { 'cacheControl': false }))
+ .get('/nums.txt')
+ .expect(utils.shouldNotHaveHeader('Cache-Control'))
+ .expect(200, '123456789', done)
+ })
+
+ it('should ignore maxAge', function (done) {
+ request(createApp(fixtures, { 'cacheControl': false, 'maxAge': 12000 }))
+ .get('/nums.txt')
+ .expect(utils.shouldNotHaveHeader('Cache-Control'))
+ .expect(200, '123456789', done)
+ })
+ })
+
+ describe('when true', function () {
+ it('should include Cache-Control', function (done) {
+ request(createApp(fixtures, { 'cacheControl': true }))
+ .get('/nums.txt')
+ .expect('Cache-Control', 'public, max-age=0')
+ .expect(200, '123456789', done)
+ })
+ })
+ })
+
+ describe('extensions', function () {
+ it('should be not be enabled by default', function (done) {
+ request(createApp(fixtures))
+ .get('/todo')
+ .expect(404, done)
+ })
+
+ it('should be configurable', function (done) {
+ request(createApp(fixtures, { 'extensions': 'txt' }))
+ .get('/todo')
+ .expect(200, '- groceries', done)
+ })
+
+ it('should support disabling extensions', function (done) {
+ request(createApp(fixtures, { 'extensions': false }))
+ .get('/todo')
+ .expect(404, done)
+ })
+
+ it('should support fallbacks', function (done) {
+ request(createApp(fixtures, { 'extensions': ['htm', 'html', 'txt'] }))
+ .get('/todo')
+ .expect(200, 'groceries', done)
+ })
+
+ it('should 404 if nothing found', function (done) {
+ request(createApp(fixtures, { 'extensions': ['htm', 'html', 'txt'] }))
+ .get('/bob')
+ .expect(404, done)
+ })
+ })
+
+ describe('fallthrough', function () {
+ it('should default to true', function (done) {
+ request(createApp())
+ .get('/does-not-exist')
+ .expect(404, 'Not Found', done)
+ })
+
+ describe('when true', function () {
+ before(function () {
+ this.app = createApp(fixtures, { 'fallthrough': true })
+ })
+
+ it('should fall-through when OPTIONS request', function (done) {
+ request(this.app)
+ .options('/todo.txt')
+ .expect(404, 'Not Found', done)
+ })
+
+ it('should fall-through when URL malformed', function (done) {
+ request(this.app)
+ .get('/%')
+ .expect(404, 'Not Found', done)
+ })
+
+ it('should fall-through when traversing past root', function (done) {
+ request(this.app)
+ .get('/users/../../todo.txt')
+ .expect(404, 'Not Found', done)
+ })
+
+ it('should fall-through when URL too long', function (done) {
+ var app = express()
+ var root = fixtures + Array(10000).join('/foobar')
+
+ app.use(express.static(root, { 'fallthrough': true }))
+ app.use(function (req, res, next) {
+ res.sendStatus(404)
+ })
+
+ request(app)
+ .get('/')
+ .expect(404, 'Not Found', done)
+ })
+
+ describe('with redirect: true', function () {
+ before(function () {
+ this.app = createApp(fixtures, { 'fallthrough': true, 'redirect': true })
+ })
+
+ it('should fall-through when directory', function (done) {
+ request(this.app)
+ .get('/pets/')
+ .expect(404, 'Not Found', done)
+ })
+
+ it('should redirect when directory without slash', function (done) {
+ request(this.app)
+ .get('/pets')
+ .expect(301, /Redirecting/, done)
+ })
+ })
+
+ describe('with redirect: false', function () {
+ before(function () {
+ this.app = createApp(fixtures, { 'fallthrough': true, 'redirect': false })
+ })
+
+ it('should fall-through when directory', function (done) {
+ request(this.app)
+ .get('/pets/')
+ .expect(404, 'Not Found', done)
+ })
+
+ it('should fall-through when directory without slash', function (done) {
+ request(this.app)
+ .get('/pets')
+ .expect(404, 'Not Found', done)
+ })
+ })
+ })
+
+ describe('when false', function () {
+ before(function () {
+ this.app = createApp(fixtures, { 'fallthrough': false })
+ })
+
+ it('should 405 when OPTIONS request', function (done) {
+ request(this.app)
+ .options('/todo.txt')
+ .expect('Allow', 'GET, HEAD')
+ .expect(405, done)
+ })
+
+ it('should 400 when URL malformed', function (done) {
+ request(this.app)
+ .get('/%')
+ .expect(400, /BadRequestError/, done)
+ })
+
+ it('should 403 when traversing past root', function (done) {
+ request(this.app)
+ .get('/users/../../todo.txt')
+ .expect(403, /ForbiddenError/, done)
+ })
+
+ it('should 404 when URL too long', function (done) {
+ var app = express()
+ var root = fixtures + Array(10000).join('/foobar')
+
+ app.use(express.static(root, { 'fallthrough': false }))
+ app.use(function (req, res, next) {
+ res.sendStatus(404)
+ })
+
+ request(app)
+ .get('/')
+ .expect(404, /ENAMETOOLONG/, done)
+ })
+
+ describe('with redirect: true', function () {
+ before(function () {
+ this.app = createApp(fixtures, { 'fallthrough': false, 'redirect': true })
+ })
+
+ it('should 404 when directory', function (done) {
+ request(this.app)
+ .get('/pets/')
+ .expect(404, /NotFoundError|ENOENT/, done)
+ })
+
+ it('should redirect when directory without slash', function (done) {
+ request(this.app)
+ .get('/pets')
+ .expect(301, /Redirecting/, done)
+ })
+ })
+
+ describe('with redirect: false', function () {
+ before(function () {
+ this.app = createApp(fixtures, { 'fallthrough': false, 'redirect': false })
+ })
+
+ it('should 404 when directory', function (done) {
+ request(this.app)
+ .get('/pets/')
+ .expect(404, /NotFoundError|ENOENT/, done)
+ })
+
+ it('should 404 when directory without slash', function (done) {
+ request(this.app)
+ .get('/pets')
+ .expect(404, /NotFoundError|ENOENT/, done)
+ })
+ })
+ })
+ })
+
+ describe('hidden files', function () {
+ before(function () {
+ this.app = createApp(fixtures, { 'dotfiles': 'allow' })
+ })
+
+ it('should be served when dotfiles: "allow" is given', function (done) {
+ request(this.app)
+ .get('/.name')
+ .expect(200)
+ .expect(utils.shouldHaveBody(Buffer.from('tobi')))
+ .end(done)
+ })
+ })
+
+ describe('immutable', function () {
+ it('should default to false', function (done) {
+ request(createApp(fixtures))
+ .get('/nums.txt')
+ .expect('Cache-Control', 'public, max-age=0', done)
+ })
+
+ it('should set immutable directive in Cache-Control', function (done) {
+ request(createApp(fixtures, { 'immutable': true, 'maxAge': '1h' }))
+ .get('/nums.txt')
+ .expect('Cache-Control', 'public, max-age=3600, immutable', done)
+ })
+ })
+
+ describe('lastModified', function () {
+ describe('when false', function () {
+ it('should not include Last-Modifed', function (done) {
+ request(createApp(fixtures, { 'lastModified': false }))
+ .get('/nums.txt')
+ .expect(utils.shouldNotHaveHeader('Last-Modified'))
+ .expect(200, '123456789', done)
+ })
+ })
+
+ describe('when true', function () {
+ it('should include Last-Modifed', function (done) {
+ request(createApp(fixtures, { 'lastModified': true }))
+ .get('/nums.txt')
+ .expect('Last-Modified', /^\w{3}, \d+ \w+ \d+ \d+:\d+:\d+ \w+$/)
+ .expect(200, '123456789', done)
+ })
+ })
+ })
+
+ describe('maxAge', function () {
+ it('should accept string', function (done) {
+ request(createApp(fixtures, { 'maxAge': '30d' }))
+ .get('/todo.txt')
+ .expect('cache-control', 'public, max-age=' + (60 * 60 * 24 * 30))
+ .expect(200, done)
+ })
+
+ it('should be reasonable when infinite', function (done) {
+ request(createApp(fixtures, { 'maxAge': Infinity }))
+ .get('/todo.txt')
+ .expect('cache-control', 'public, max-age=' + (60 * 60 * 24 * 365))
+ .expect(200, done)
+ })
+ })
+
+ describe('redirect', function () {
+ before(function () {
+ this.app = express()
+ this.app.use(function (req, res, next) {
+ req.originalUrl = req.url =
+ req.originalUrl.replace(/\/snow(\/|$)/, '/snow \u2603$1')
+ next()
+ })
+ this.app.use(express.static(fixtures))
+ })
+
+ it('should redirect directories', function (done) {
+ request(this.app)
+ .get('/users')
+ .expect('Location', '/users/')
+ .expect(301, done)
+ })
+
+ it('should include HTML link', function (done) {
+ request(this.app)
+ .get('/users')
+ .expect('Location', '/users/')
+ .expect(301, //, done)
+ })
+
+ it('should redirect directories with query string', function (done) {
+ request(this.app)
+ .get('/users?name=john')
+ .expect('Location', '/users/?name=john')
+ .expect(301, done)
+ })
+
+ it('should not redirect to protocol-relative locations', function (done) {
+ request(this.app)
+ .get('//users')
+ .expect('Location', '/users/')
+ .expect(301, done)
+ })
+
+ it('should ensure redirect URL is properly encoded', function (done) {
+ request(this.app)
+ .get('/snow')
+ .expect('Location', '/snow%20%E2%98%83/')
+ .expect('Content-Type', /html/)
+ .expect(301, />Redirecting to \/snow%20%E2%98%83\/<\/a>, done)
+ })
+
+ it('should respond with default Content-Security-Policy', function (done) {
+ request(this.app)
+ .get('/users')
+ .expect('Content-Security-Policy', "default-src 'self'")
+ .expect(301, done)
+ })
+
+ it('should not redirect incorrectly', function (done) {
+ request(this.app)
+ .get('/')
+ .expect(404, done)
+ })
+
+ describe('when false', function () {
+ before(function () {
+ this.app = createApp(fixtures, { 'redirect': false })
+ })
+
+ it('should disable redirect', function (done) {
+ request(this.app)
+ .get('/users')
+ .expect(404, done)
+ })
+ })
+ })
+
+ describe('setHeaders', function () {
+ before(function () {
+ this.app = express()
+ this.app.use(express.static(fixtures, { 'setHeaders': function (res) {
+ res.setHeader('x-custom', 'set')
+ } }))
+ })
+
+ it('should reject non-functions', function () {
+ assert.throws(express.static.bind(null, fixtures, { 'setHeaders': 3 }), /setHeaders.*function/)
+ })
+
+ it('should get called when sending file', function (done) {
+ request(this.app)
+ .get('/nums.txt')
+ .expect('x-custom', 'set')
+ .expect(200, done)
+ })
+
+ it('should not get called on 404', function (done) {
+ request(this.app)
+ .get('/bogus')
+ .expect(utils.shouldNotHaveHeader('x-custom'))
+ .expect(404, done)
+ })
+
+ it('should not get called on redirect', function (done) {
+ request(this.app)
+ .get('/users')
+ .expect(utils.shouldNotHaveHeader('x-custom'))
+ .expect(301, done)
+ })
+ })
+
+ describe('when traversing past root', function () {
+ before(function () {
+ this.app = createApp(fixtures, { 'fallthrough': false })
+ })
+
+ it('should catch urlencoded ../', function (done) {
+ request(this.app)
+ .get('/users/%2e%2e/%2e%2e/todo.txt')
+ .expect(403, done)
+ })
+
+ it('should not allow root path disclosure', function (done) {
+ request(this.app)
+ .get('/users/../../fixtures/todo.txt')
+ .expect(403, done)
+ })
+ })
+
+ describe('when request has "Range" header', function () {
+ before(function () {
+ this.app = createApp()
+ })
+
+ it('should support byte ranges', function (done) {
+ request(this.app)
+ .get('/nums.txt')
+ .set('Range', 'bytes=0-4')
+ .expect('12345', done)
+ })
+
+ it('should be inclusive', function (done) {
+ request(this.app)
+ .get('/nums.txt')
+ .set('Range', 'bytes=0-0')
+ .expect('1', done)
+ })
+
+ it('should set Content-Range', function (done) {
+ request(this.app)
+ .get('/nums.txt')
+ .set('Range', 'bytes=2-5')
+ .expect('Content-Range', 'bytes 2-5/9', done)
+ })
+
+ it('should support -n', function (done) {
+ request(this.app)
+ .get('/nums.txt')
+ .set('Range', 'bytes=-3')
+ .expect('789', done)
+ })
+
+ it('should support n-', function (done) {
+ request(this.app)
+ .get('/nums.txt')
+ .set('Range', 'bytes=3-')
+ .expect('456789', done)
+ })
+
+ it('should respond with 206 "Partial Content"', function (done) {
+ request(this.app)
+ .get('/nums.txt')
+ .set('Range', 'bytes=0-4')
+ .expect(206, done)
+ })
+
+ it('should set Content-Length to the # of octets transferred', function (done) {
+ request(this.app)
+ .get('/nums.txt')
+ .set('Range', 'bytes=2-3')
+ .expect('Content-Length', '2')
+ .expect(206, '34', done)
+ })
+
+ describe('when last-byte-pos of the range is greater than current length', function () {
+ it('is taken to be equal to one less than the current length', function (done) {
+ request(this.app)
+ .get('/nums.txt')
+ .set('Range', 'bytes=2-50')
+ .expect('Content-Range', 'bytes 2-8/9', done)
+ })
+
+ it('should adapt the Content-Length accordingly', function (done) {
+ request(this.app)
+ .get('/nums.txt')
+ .set('Range', 'bytes=2-50')
+ .expect('Content-Length', '7')
+ .expect(206, done)
+ })
+ })
+
+ describe('when the first- byte-pos of the range is greater than the current length', function () {
+ it('should respond with 416', function (done) {
+ request(this.app)
+ .get('/nums.txt')
+ .set('Range', 'bytes=9-50')
+ .expect(416, done)
+ })
+
+ it('should include a Content-Range header of complete length', function (done) {
+ request(this.app)
+ .get('/nums.txt')
+ .set('Range', 'bytes=9-50')
+ .expect('Content-Range', 'bytes */9')
+ .expect(416, done)
+ })
+ })
+
+ describe('when syntactically invalid', function () {
+ it('should respond with 200 and the entire contents', function (done) {
+ request(this.app)
+ .get('/nums.txt')
+ .set('Range', 'asdf')
+ .expect('123456789', done)
+ })
+ })
+ })
+
+ describe('when index at mount point', function () {
+ before(function () {
+ this.app = express()
+ this.app.use('/users', express.static(fixtures + '/users'))
+ })
+
+ it('should redirect correctly', function (done) {
+ request(this.app)
+ .get('/users')
+ .expect('Location', '/users/')
+ .expect(301, done)
+ })
+ })
+
+ describe('when mounted', function () {
+ before(function () {
+ this.app = express()
+ this.app.use('/static', express.static(fixtures))
+ })
+
+ it('should redirect relative to the originalUrl', function (done) {
+ request(this.app)
+ .get('/static/users')
+ .expect('Location', '/static/users/')
+ .expect(301, done)
+ })
+
+ it('should not choke on auth-looking URL', function (done) {
+ request(this.app)
+ .get('//todo@txt')
+ .expect(404, done)
+ })
+ })
+
+ //
+ // NOTE: This is not a real part of the API, but
+ // over time this has become something users
+ // are doing, so this will prevent unseen
+ // regressions around this use-case.
+ //
+ describe('when mounted "root" as a file', function () {
+ before(function () {
+ this.app = express()
+ this.app.use('/todo.txt', express.static(fixtures + '/todo.txt'))
+ })
+
+ it('should load the file when on trailing slash', function (done) {
+ request(this.app)
+ .get('/todo.txt')
+ .expect(200, '- groceries', done)
+ })
+
+ it('should 404 when trailing slash', function (done) {
+ request(this.app)
+ .get('/todo.txt/')
+ .expect(404, done)
+ })
+ })
+
+ describe('when responding non-2xx or 304', function () {
+ it('should not alter the status', function (done) {
+ var app = express()
+
+ app.use(function (req, res, next) {
+ res.status(501)
+ next()
+ })
+ app.use(express.static(fixtures))
+
+ request(app)
+ .get('/todo.txt')
+ .expect(501, '- groceries', done)
+ })
+ })
+
+ describe('when index file serving disabled', function () {
+ before(function () {
+ this.app = express()
+ this.app.use('/static', express.static(fixtures, { 'index': false }))
+ this.app.use(function (req, res, next) {
+ res.sendStatus(404)
+ })
+ })
+
+ it('should next() on directory', function (done) {
+ request(this.app)
+ .get('/static/users/')
+ .expect(404, 'Not Found', done)
+ })
+
+ it('should redirect to trailing slash', function (done) {
+ request(this.app)
+ .get('/static/users')
+ .expect('Location', '/static/users/')
+ .expect(301, done)
+ })
+
+ it('should next() on mount point', function (done) {
+ request(this.app)
+ .get('/static/')
+ .expect(404, 'Not Found', done)
+ })
+
+ it('should redirect to trailing slash mount point', function (done) {
+ request(this.app)
+ .get('/static')
+ .expect('Location', '/static/')
+ .expect(301, done)
+ })
+ })
+})
+
+function createApp (dir, options, fn) {
+ var app = express()
+ var root = dir || fixtures
+
+ app.use(express.static(root, options))
+
+ app.use(function (req, res, next) {
+ res.sendStatus(404)
+ })
+
+ return app
+}
diff --git a/test/fixtures/empty.txt b/test/fixtures/empty.txt
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/test/fixtures/nums.txt b/test/fixtures/nums.txt
new file mode 100644
index 0000000000..e2e107ac61
--- /dev/null
+++ b/test/fixtures/nums.txt
@@ -0,0 +1 @@
+123456789
\ No newline at end of file
diff --git a/test/fixtures/pets/names.txt b/test/fixtures/pets/names.txt
new file mode 100644
index 0000000000..91407a3e04
--- /dev/null
+++ b/test/fixtures/pets/names.txt
@@ -0,0 +1 @@
+tobi,loki
\ No newline at end of file
diff --git "a/test/fixtures/snow \342\230\203/.gitkeep" "b/test/fixtures/snow \342\230\203/.gitkeep"
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/test/fixtures/todo.html b/test/fixtures/todo.html
new file mode 100644
index 0000000000..e7af6d7998
--- /dev/null
+++ b/test/fixtures/todo.html
@@ -0,0 +1 @@
+groceries
\ No newline at end of file
diff --git a/test/fixtures/todo.txt b/test/fixtures/todo.txt
new file mode 100644
index 0000000000..8c3539d946
--- /dev/null
+++ b/test/fixtures/todo.txt
@@ -0,0 +1 @@
+- groceries
\ No newline at end of file
diff --git a/test/fixtures/users/index.html b/test/fixtures/users/index.html
new file mode 100644
index 0000000000..00a2db41f7
--- /dev/null
+++ b/test/fixtures/users/index.html
@@ -0,0 +1 @@
+tobi, loki, jane
\ No newline at end of file
diff --git a/test/fixtures/users/tobi.txt b/test/fixtures/users/tobi.txt
new file mode 100644
index 0000000000..9d9529d47d
--- /dev/null
+++ b/test/fixtures/users/tobi.txt
@@ -0,0 +1 @@
+ferret
\ No newline at end of file
diff --git a/test/support/utils.js b/test/support/utils.js
index ec6b801bc0..579f042a0c 100644
--- a/test/support/utils.js
+++ b/test/support/utils.js
@@ -3,14 +3,48 @@
* Module dependencies.
* @private
*/
+
var assert = require('assert');
+var Buffer = require('safe-buffer').Buffer
/**
* Module exports.
* @public
*/
+
+exports.shouldHaveBody = shouldHaveBody
+exports.shouldNotHaveBody = shouldNotHaveBody
exports.shouldNotHaveHeader = shouldNotHaveHeader;
+/**
+ * Assert that a supertest response has a specific body.
+ *
+ * @param {Buffer} buf
+ * @returns {function}
+ */
+
+function shouldHaveBody (buf) {
+ return function (res) {
+ var body = !Buffer.isBuffer(res.body)
+ ? Buffer.from(res.text)
+ : res.body
+ assert.ok(body, 'response has body')
+ assert.strictEqual(body.toString('hex'), buf.toString('hex'))
+ }
+}
+
+/**
+ * Assert that a supertest response does not have a body.
+ *
+ * @returns {function}
+ */
+
+function shouldNotHaveBody () {
+ return function (res) {
+ assert.ok(res.text === '' || res.text === undefined)
+ }
+}
+
/**
* Assert that a supertest response does not have a header.
*