You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							223 lines
						
					
					
						
							6.3 KiB
						
					
					
				
			
		
		
	
	
							223 lines
						
					
					
						
							6.3 KiB
						
					
					
				| 'use strict'
 | |
| 
 | |
| const tap = require('tap')
 | |
| const test = tap.test
 | |
| const Fastify = require('fastify')
 | |
| const request = require('request')
 | |
| const plugin = require('../')
 | |
| const qs = require('qs')
 | |
| 
 | |
| test('succes route succeeds', (t) => {
 | |
|   t.plan(3)
 | |
|   const fastify = Fastify()
 | |
| 
 | |
|   fastify.register(plugin)
 | |
|   fastify.post('/test1', (req, res) => {
 | |
|     res.send(Object.assign({}, req.body, { message: 'done' }))
 | |
|   })
 | |
| 
 | |
|   fastify.listen(0, (err) => {
 | |
|     if (err) tap.error(err)
 | |
|     fastify.server.unref()
 | |
| 
 | |
|     const reqOpts = {
 | |
|       method: 'POST',
 | |
|       baseUrl: 'http://localhost:' + fastify.server.address().port
 | |
|     }
 | |
|     const req = request.defaults(reqOpts)
 | |
|     req({ uri: '/test1', form: { foo: 'foo' } }, (err, response, body) => {
 | |
|       t.error(err)
 | |
|       t.equal(response.statusCode, 200)
 | |
|       t.same(JSON.parse(body), { foo: 'foo', message: 'done' })
 | |
|     })
 | |
|   })
 | |
| })
 | |
| 
 | |
| test('cannot exceed body limit', (t) => {
 | |
|   t.plan(3)
 | |
|   const fastify = Fastify()
 | |
| 
 | |
|   fastify.register(plugin, { bodyLimit: 10 })
 | |
|   fastify.post('/limited', (req, res) => {
 | |
|     res.send(Object.assign({}, req.body, { message: 'done' }))
 | |
|   })
 | |
| 
 | |
|   fastify.listen(0, (err) => {
 | |
|     if (err) tap.error(err)
 | |
|     fastify.server.unref()
 | |
| 
 | |
|     const reqOpts = {
 | |
|       method: 'POST',
 | |
|       baseUrl: 'http://localhost:' + fastify.server.address().port
 | |
|     }
 | |
|     const req = request.defaults(reqOpts)
 | |
|     const payload = require('crypto').randomBytes(128).toString('hex')
 | |
|     req({ uri: '/limited', form: { foo: payload } }, (err, response, body) => {
 | |
|       t.error(err)
 | |
|       t.equal(response.statusCode, 413)
 | |
|       t.equal(JSON.parse(body).message, 'Request body is too large')
 | |
|     })
 | |
|   })
 | |
| })
 | |
| 
 | |
| test('cannot exceed body limit when Content-Length is not available', (t) => {
 | |
|   t.plan(3)
 | |
|   const fastify = Fastify()
 | |
| 
 | |
|   fastify.addHook('onSend', (request, reply, payload, next) => {
 | |
|     reply.send = function mockSend (arg) {
 | |
|       t.fail('reply.send() was called multiple times')
 | |
|     }
 | |
|     setTimeout(next, 1)
 | |
|   })
 | |
| 
 | |
|   fastify.register(plugin, { bodyLimit: 10 })
 | |
|   fastify.post('/limited', (req, res) => {
 | |
|     res.send(Object.assign({}, req.body, { message: 'done' }))
 | |
|   })
 | |
| 
 | |
|   fastify.listen(0, (err) => {
 | |
|     if (err) tap.error(err)
 | |
|     fastify.server.unref()
 | |
| 
 | |
|     const reqOpts = {
 | |
|       method: 'POST',
 | |
|       baseUrl: 'http://localhost:' + fastify.server.address().port,
 | |
|       headers: {
 | |
|         'content-type': 'application/x-www-form-urlencoded'
 | |
|       }
 | |
|     }
 | |
|     const req = request.defaults(reqOpts)
 | |
|     let sent = false
 | |
|     const payload = require('stream').Readable({
 | |
|       read: function () {
 | |
|         this.push(sent ? null : Buffer.alloc(70000, 'a'))
 | |
|         sent = true
 | |
|       }
 | |
|     })
 | |
|     req({ uri: '/limited', body: payload }, (err, response, body) => {
 | |
|       t.error(err)
 | |
|       t.equal(response.statusCode, 413)
 | |
|       t.equal(JSON.parse(body).message, 'Request body is too large')
 | |
|     })
 | |
|   })
 | |
| })
 | |
| 
 | |
| test('cannot exceed body limit set on Fastify instance', (t) => {
 | |
|   t.plan(3)
 | |
|   const fastify = Fastify({ bodyLimit: 10 })
 | |
| 
 | |
|   fastify.register(plugin)
 | |
|   fastify.post('/limited', (req, res) => {
 | |
|     res.send(Object.assign({}, req.body, { message: 'done' }))
 | |
|   })
 | |
| 
 | |
|   fastify.listen(0, (err) => {
 | |
|     if (err) tap.error(err)
 | |
|     fastify.server.unref()
 | |
| 
 | |
|     const reqOpts = {
 | |
|       method: 'POST',
 | |
|       baseUrl: 'http://localhost:' + fastify.server.address().port
 | |
|     }
 | |
|     const req = request.defaults(reqOpts)
 | |
|     const payload = require('crypto').randomBytes(128).toString('hex')
 | |
|     req({ uri: '/limited', form: { foo: payload } }, (err, response, body) => {
 | |
|       t.error(err)
 | |
|       t.equal(response.statusCode, 413)
 | |
|       t.equal(JSON.parse(body).message, 'Request body is too large')
 | |
|     })
 | |
|   })
 | |
| })
 | |
| 
 | |
| test('plugin bodyLimit should overwrite Fastify instance bodyLimit', (t) => {
 | |
|   t.plan(3)
 | |
|   const fastify = Fastify({ bodyLimit: 100000 })
 | |
| 
 | |
|   fastify.register(plugin, { bodyLimit: 10 })
 | |
|   fastify.post('/limited', (req, res) => {
 | |
|     res.send(Object.assign({}, req.body, { message: 'done' }))
 | |
|   })
 | |
| 
 | |
|   fastify.listen(0, (err) => {
 | |
|     if (err) tap.error(err)
 | |
|     fastify.server.unref()
 | |
| 
 | |
|     const reqOpts = {
 | |
|       method: 'POST',
 | |
|       baseUrl: 'http://localhost:' + fastify.server.address().port
 | |
|     }
 | |
|     const req = request.defaults(reqOpts)
 | |
|     const payload = require('crypto').randomBytes(128).toString('hex')
 | |
|     req({ uri: '/limited', form: { foo: payload } }, (err, response, body) => {
 | |
|       t.error(err)
 | |
|       t.equal(response.statusCode, 413)
 | |
|       t.equal(JSON.parse(body).message, 'Request body is too large')
 | |
|     })
 | |
|   })
 | |
| })
 | |
| 
 | |
| test('plugin should throw if opts.parser is not a function', (t) => {
 | |
|   t.plan(2)
 | |
|   const fastify = Fastify()
 | |
|   fastify.register(plugin, { parser: 'invalid' })
 | |
|   fastify.listen(0, (err) => {
 | |
|     t.ok(err)
 | |
|     t.match(err.message, /parser must be a function/)
 | |
|     fastify.server.unref()
 | |
|   })
 | |
| })
 | |
| 
 | |
| test('plugin should not parse nested objects by default', (t) => {
 | |
|   t.plan(3)
 | |
|   const fastify = Fastify()
 | |
| 
 | |
|   fastify.register(plugin)
 | |
|   fastify.post('/test1', (req, res) => {
 | |
|     res.send(Object.assign({}, req.body, { message: 'done' }))
 | |
|   })
 | |
| 
 | |
|   fastify.listen(0, (err) => {
 | |
|     if (err) tap.error(err)
 | |
|     fastify.server.unref()
 | |
| 
 | |
|     const reqOpts = {
 | |
|       method: 'POST',
 | |
|       baseUrl: 'http://localhost:' + fastify.server.address().port
 | |
|     }
 | |
|     const req = request.defaults(reqOpts)
 | |
|     req({ uri: '/test1', form: { 'foo[one]': 'foo', 'foo[two]': 'bar' } }, (err, response, body) => {
 | |
|       t.error(err)
 | |
|       t.equal(response.statusCode, 200)
 | |
|       t.same(JSON.parse(body), { 'foo[one]': 'foo', 'foo[two]': 'bar', message: 'done' })
 | |
|     })
 | |
|   })
 | |
| })
 | |
| 
 | |
| test('plugin should allow providing custom parser as option', (t) => {
 | |
|   t.plan(3)
 | |
|   const fastify = Fastify()
 | |
| 
 | |
|   // this makes sure existing users for <= 4 have upgrade path
 | |
|   fastify.register(plugin, { parser: str => qs.parse(str) })
 | |
|   fastify.post('/test1', (req, res) => {
 | |
|     res.send(Object.assign({}, req.body, { message: 'done' }))
 | |
|   })
 | |
| 
 | |
|   fastify.listen(0, (err) => {
 | |
|     if (err) tap.error(err)
 | |
|     fastify.server.unref()
 | |
| 
 | |
|     const reqOpts = {
 | |
|       method: 'POST',
 | |
|       baseUrl: 'http://localhost:' + fastify.server.address().port
 | |
|     }
 | |
|     const req = request.defaults(reqOpts)
 | |
|     req({ uri: '/test1', form: { 'foo[one]': 'foo', 'foo[two]': 'bar' } }, (err, response, body) => {
 | |
|       t.error(err)
 | |
|       t.equal(response.statusCode, 200)
 | |
|       t.same(JSON.parse(body), { foo: { one: 'foo', two: 'bar' }, message: 'done' })
 | |
|     })
 | |
|   })
 | |
| })
 |