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
						
					
					
				| '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')
 | |
|   }
 | |
| })
 |