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