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.
		
		
		
		
		
			
		
			
				
					185 lines
				
				5.5 KiB
			
		
		
			
		
	
	
					185 lines
				
				5.5 KiB
			| 
											3 years ago
										 | 'use strict' | ||
|  | 
 | ||
|  | const tap = require('tap') | ||
|  | const test = tap.test | ||
|  | const Fastify = require('fastify') | ||
|  | const plugin = require('../') | ||
|  | const qs = require('qs') | ||
|  | const formAutoContent = require('form-auto-content') | ||
|  | 
 | ||
|  | 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({ port: 0 }, (err) => { | ||
|  |     if (err) tap.error(err) | ||
|  |     fastify.server.unref() | ||
|  | 
 | ||
|  |     fastify.inject({ path: '/test1', method: 'POST', ...formAutoContent({ foo: 'foo' }) }, (err, response) => { | ||
|  |       t.error(err) | ||
|  |       t.equal(response.statusCode, 200) | ||
|  |       t.same(JSON.parse(response.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({ port: 0 }, (err) => { | ||
|  |     if (err) tap.error(err) | ||
|  |     fastify.server.unref() | ||
|  | 
 | ||
|  |     const payload = require('crypto').randomBytes(128).toString('hex') | ||
|  |     fastify.inject({ path: '/limited', method: 'POST', ...formAutoContent({ foo: payload }) }, (err, response) => { | ||
|  |       t.error(err) | ||
|  |       t.equal(response.statusCode, 413) | ||
|  |       t.equal(JSON.parse(response.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({ port: 0 }, (err) => { | ||
|  |     if (err) tap.error(err) | ||
|  |     fastify.server.unref() | ||
|  | 
 | ||
|  |     let sent = false | ||
|  |     const payload = require('stream').Readable({ | ||
|  |       read: function () { | ||
|  |         this.push(sent ? null : Buffer.alloc(70000, 'a')) | ||
|  |         sent = true | ||
|  |       } | ||
|  |     }) | ||
|  |     fastify.inject({ path: '/limited', method: 'POST', headers: { 'content-type': 'application/x-www-form-urlencoded' }, body: payload }, (err, response) => { | ||
|  |       t.error(err) | ||
|  |       t.equal(response.statusCode, 413) | ||
|  |       t.equal(JSON.parse(response.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({ port: 0 }, (err) => { | ||
|  |     if (err) tap.error(err) | ||
|  |     fastify.server.unref() | ||
|  | 
 | ||
|  |     const payload = require('crypto').randomBytes(128).toString('hex') | ||
|  |     fastify.inject({ path: '/limited', method: 'POST', ...formAutoContent({ foo: payload }) }, (err, response) => { | ||
|  |       t.error(err) | ||
|  |       t.equal(response.statusCode, 413) | ||
|  |       t.equal(JSON.parse(response.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({ port: 0 }, (err) => { | ||
|  |     if (err) tap.error(err) | ||
|  |     fastify.server.unref() | ||
|  | 
 | ||
|  |     const payload = require('crypto').randomBytes(128).toString('hex') | ||
|  |     fastify.inject({ path: '/limited', method: 'POST', ...formAutoContent({ foo: payload }) }, (err, response) => { | ||
|  |       t.error(err) | ||
|  |       t.equal(response.statusCode, 413) | ||
|  |       t.equal(JSON.parse(response.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({ port: 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({ port: 0 }, (err) => { | ||
|  |     if (err) tap.error(err) | ||
|  |     fastify.server.unref() | ||
|  | 
 | ||
|  |     fastify.inject({ path: '/test1', method: 'POST', ...formAutoContent({ 'foo[one]': 'foo', 'foo[two]': 'bar' }) }, (err, response) => { | ||
|  |       t.error(err) | ||
|  |       t.equal(response.statusCode, 200) | ||
|  |       t.same(JSON.parse(response.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({ port: 0 }, (err) => { | ||
|  |     if (err) tap.error(err) | ||
|  |     fastify.server.unref() | ||
|  | 
 | ||
|  |     fastify.inject({ path: '/test1', method: 'POST', ...formAutoContent({ 'foo[one]': 'foo', 'foo[two]': 'bar' }) }, (err, response) => { | ||
|  |       t.error(err) | ||
|  |       t.equal(response.statusCode, 200) | ||
|  |       t.same(JSON.parse(response.body), { foo: { one: 'foo', two: 'bar' }, message: 'done' }) | ||
|  |     }) | ||
|  |   }) | ||
|  | }) |