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.
		
		
		
		
		
			
		
			
				
					193 lines
				
				4.1 KiB
			
		
		
			
		
	
	
					193 lines
				
				4.1 KiB
			| 
											3 years ago
										 | 'use strict' | ||
|  | 
 | ||
|  | const fs = require('fs') | ||
|  | const crypto = require('crypto') | ||
|  | const test = require('tap').test | ||
|  | const FormData = require('form-data') | ||
|  | const Fastify = require('fastify') | ||
|  | const multipart = require('..') | ||
|  | const http = require('http') | ||
|  | const EventEmitter = require('events') | ||
|  | const { once } = EventEmitter | ||
|  | 
 | ||
|  | test('should throw fileSize limitation error when consuming the stream', async function (t) { | ||
|  |   t.plan(4) | ||
|  | 
 | ||
|  |   const fastify = Fastify() | ||
|  |   t.teardown(fastify.close.bind(fastify)) | ||
|  | 
 | ||
|  |   fastify.register(multipart, { | ||
|  |     throwFileSizeLimit: true, | ||
|  |     limits: { | ||
|  |       fileSize: 524288 | ||
|  |     } | ||
|  |   }) | ||
|  | 
 | ||
|  |   fastify.post('/', async function (req, reply) { | ||
|  |     t.ok(req.isMultipart()) | ||
|  | 
 | ||
|  |     const part = await req.file() | ||
|  |     t.pass('the file is not consumed yet') | ||
|  | 
 | ||
|  |     try { | ||
|  |       await part.toBuffer() | ||
|  |       t.fail('it should throw') | ||
|  |     } catch (error) { | ||
|  |       t.ok(error) | ||
|  |       reply.send(error) | ||
|  |     } | ||
|  |   }) | ||
|  | 
 | ||
|  |   await fastify.listen({ port: 0 }) | ||
|  | 
 | ||
|  |   // request
 | ||
|  |   const form = new FormData() | ||
|  |   const opts = { | ||
|  |     hostname: '127.0.0.1', | ||
|  |     port: fastify.server.address().port, | ||
|  |     path: '/', | ||
|  |     headers: form.getHeaders(), | ||
|  |     method: 'POST' | ||
|  |   } | ||
|  | 
 | ||
|  |   const randomFileBuffer = Buffer.alloc(600_000) | ||
|  |   crypto.randomFillSync(randomFileBuffer) | ||
|  | 
 | ||
|  |   const req = http.request(opts) | ||
|  |   form.append('upload', randomFileBuffer) | ||
|  | 
 | ||
|  |   form.pipe(req) | ||
|  | 
 | ||
|  |   try { | ||
|  |     const [res] = await once(req, 'response') | ||
|  |     t.equal(res.statusCode, 413) | ||
|  |     res.resume() | ||
|  |     await once(res, 'end') | ||
|  |   } catch (error) { | ||
|  |     t.error(error, 'request') | ||
|  |   } | ||
|  | }) | ||
|  | 
 | ||
|  | test('should throw fileSize limitation error when consuming the stream MBs', async function (t) { | ||
|  |   t.plan(4) | ||
|  | 
 | ||
|  |   const fastify = Fastify() | ||
|  |   t.teardown(fastify.close.bind(fastify)) | ||
|  | 
 | ||
|  |   fastify.register(multipart, { | ||
|  |     throwFileSizeLimit: true, | ||
|  |     limits: { | ||
|  |       fileSize: 5_000_000 // 5MB
 | ||
|  |     } | ||
|  |   }) | ||
|  | 
 | ||
|  |   fastify.post('/', async function (req, reply) { | ||
|  |     t.ok(req.isMultipart()) | ||
|  | 
 | ||
|  |     const part = await req.file() | ||
|  |     t.pass('the file is not consumed yet') | ||
|  | 
 | ||
|  |     try { | ||
|  |       await part.toBuffer() | ||
|  |       t.fail('it should throw') | ||
|  |     } catch (error) { | ||
|  |       t.ok(error) | ||
|  |       reply.send(error) | ||
|  |     } | ||
|  |   }) | ||
|  | 
 | ||
|  |   await fastify.listen({ port: 0 }) | ||
|  | 
 | ||
|  |   // request
 | ||
|  |   const form = new FormData() | ||
|  |   const opts = { | ||
|  |     hostname: '127.0.0.1', | ||
|  |     port: fastify.server.address().port, | ||
|  |     path: '/', | ||
|  |     headers: form.getHeaders(), | ||
|  |     method: 'POST' | ||
|  |   } | ||
|  | 
 | ||
|  |   const randomFileBuffer = Buffer.alloc(15_000_000) | ||
|  |   crypto.randomFillSync(randomFileBuffer) | ||
|  | 
 | ||
|  |   const tmpFile = 'test/random-file' | ||
|  |   fs.writeFileSync(tmpFile, randomFileBuffer) | ||
|  | 
 | ||
|  |   const req = http.request(opts) | ||
|  |   form.append('upload', fs.createReadStream(tmpFile)) | ||
|  | 
 | ||
|  |   form.pipe(req) | ||
|  | 
 | ||
|  |   try { | ||
|  |     const [res] = await once(req, 'response') | ||
|  |     t.equal(res.statusCode, 413) | ||
|  |     res.resume() | ||
|  |     await once(res, 'end') | ||
|  | 
 | ||
|  |     fs.unlinkSync(tmpFile) | ||
|  |   } catch (error) { | ||
|  |     t.error(error, 'request') | ||
|  |   } | ||
|  | }) | ||
|  | 
 | ||
|  | test('should NOT throw fileSize limitation error when consuming the stream', async function (t) { | ||
|  |   t.plan(5) | ||
|  | 
 | ||
|  |   const fastify = Fastify() | ||
|  |   t.teardown(fastify.close.bind(fastify)) | ||
|  | 
 | ||
|  |   fastify.register(multipart, { | ||
|  |     throwFileSizeLimit: false, | ||
|  |     limits: { | ||
|  |       fileSize: 524288 | ||
|  |     } | ||
|  |   }) | ||
|  |   const fileInputLength = 600000 | ||
|  | 
 | ||
|  |   fastify.post('/', async function (req, reply) { | ||
|  |     t.ok(req.isMultipart()) | ||
|  | 
 | ||
|  |     const part = await req.file() | ||
|  |     t.pass('the file is not consumed yet') | ||
|  | 
 | ||
|  |     try { | ||
|  |       const buffer = await part.toBuffer() | ||
|  |       t.ok(part.file.truncated) | ||
|  |       t.notSame(buffer.length, fileInputLength) | ||
|  |       reply.send(new fastify.multipartErrors.FilesLimitError()) | ||
|  |     } catch (error) { | ||
|  |       t.fail('it should not throw') | ||
|  |     } | ||
|  |   }) | ||
|  | 
 | ||
|  |   await fastify.listen({ port: 0 }) | ||
|  | 
 | ||
|  |   // request
 | ||
|  |   const form = new FormData() | ||
|  |   const opts = { | ||
|  |     protocol: 'http:', | ||
|  |     hostname: 'localhost', | ||
|  |     port: fastify.server.address().port, | ||
|  |     path: '/', | ||
|  |     headers: form.getHeaders(), | ||
|  |     method: 'POST' | ||
|  |   } | ||
|  | 
 | ||
|  |   const randomFileBuffer = Buffer.alloc(fileInputLength) | ||
|  |   crypto.randomFillSync(randomFileBuffer) | ||
|  | 
 | ||
|  |   const req = http.request(opts) | ||
|  |   form.append('upload', randomFileBuffer) | ||
|  | 
 | ||
|  |   form.pipe(req) | ||
|  | 
 | ||
|  |   try { | ||
|  |     const [res] = await once(req, 'response') | ||
|  |     t.equal(res.statusCode, 413) | ||
|  |     res.resume() | ||
|  |   } catch (error) { | ||
|  |     t.error(error, 'request') | ||
|  |   } | ||
|  | }) |