• Stars
    star
    166
  • Rank 220,979 (Top 5 %)
  • Language
    JavaScript
  • License
    MIT License
  • Created almost 8 years ago
  • Updated over 1 year ago

Reviews

There are no reviews yet. Be the first to send feedback to the community and the maintainers!

Repository Details

multipart plugin for egg

egg-multipart

NPM version Node.js CI Test coverage Known Vulnerabilities npm download

Use co-busboy to upload file by streaming and process it without save to disk(using the stream mode).

Just use ctx.multipart() to got file stream, then pass to image processing liberary such as gm or upload to cloud storage such as oss.

Whitelist of file extensions

For security, if uploading file extension is not in white list, will response as 400 Bad request.

Default Whitelist:

const whitelist = [
  // images
  '.jpg', '.jpeg', // image/jpeg
  '.png', // image/png, image/x-png
  '.gif', // image/gif
  '.bmp', // image/bmp
  '.wbmp', // image/vnd.wap.wbmp
  '.webp',
  '.tif',
  '.psd',
  // text
  '.svg',
  '.js', '.jsx',
  '.json',
  '.css', '.less',
  '.html', '.htm',
  '.xml',
  // tar
  '.zip',
  '.gz', '.tgz', '.gzip',
  // video
  '.mp3',
  '.mp4',
  '.avi',
];

fileSize

The default fileSize that multipart can accept is 10mb. if you upload a large file, you should specify this config.

// config/config.default.js
exports.multipart = {
  fileSize: '50mb',
};

Custom Config

Developer can custom additional file extensions:

// config/config.default.js
exports.multipart = {
  // will append to whilelist
  fileExtensions: [
    '.foo',
    '.apk',
  ],
};

Can also override built-in whitelist, such as only allow png:

// config/config.default.js
exports.multipart = {
  whitelist: [
    '.png',
  ],
};

Or by function:

exports.multipart = {
  whitelist: (filename) => [ '.png' ].includes(path.extname(filename) || '')
};

Note: if define whitelist, then fileExtensions will be ignored.

Examples

More examples please follow:

file mode: the easy way

If you don't know the Node.js Stream work, maybe you should use the file mode to get started.

The usage very similar to bodyParser.

  • ctx.request.body: Get all the multipart fields and values, except file.
  • ctx.request.files: Contains all file from the multipart request, it's an Array object.

WARNING: you should remove the temporary upload files after you use it, the async ctx.cleanupRequestFiles() method will be very helpful.

Enable file mode on config

You need to set config.multipart.mode = 'file' to enable file mode:

// config/config.default.js
exports.multipart = {
  mode: 'file',
};

After file mode enable, egg will remove the old temporary files(don't include today's files) on 04:30 AM every day by default.

config.multipart = {
  mode: 'file',
  tmpdir: path.join(os.tmpdir(), 'egg-multipart-tmp', appInfo.name),
  cleanSchedule: {
    // run tmpdir clean job on every day 04:30 am
    // cron style see https://github.com/eggjs/egg-schedule#cron-style-scheduling
    cron: '0 30 4 * * *',
    disable: false,
  },
};

Default will use the last field which has same name, if need the all fields value, please set allowArrayField in config.

// config/config.default.js
exports.multipart = {
  mode: 'file',
  allowArrayField: true,
};

Upload One File

<form method="POST" action="/upload?_csrf={{ ctx.csrf | safe }}" enctype="multipart/form-data">
  title: <input name="title" />
  file: <input name="file" type="file" />
  <button type="submit">Upload</button>
</form>

Controller which hanlder POST /upload:

// app/controller/upload.js
const Controller = require('egg').Controller;
const fs = require('mz/fs');

module.exports = class extends Controller {
  async upload() {
    const { ctx } = this;
    const file = ctx.request.files[0];
    const name = 'egg-multipart-test/' + path.basename(file.filename);
    let result;
    try {
      // process file or upload to cloud storage
      result = await ctx.oss.put(name, file.filepath);
    } finally {
      // remove tmp files and don't block the request's response
      // cleanupRequestFiles won't throw error even remove file io error happen
      ctx.cleanupRequestFiles();
      // remove tmp files before send response
      // await ctx.cleanupRequestFiles();
    }

    ctx.body = {
      url: result.url,
      // get all field values
      requestBody: ctx.request.body,
    };
  }
};

Upload Multiple Files

<form method="POST" action="/upload?_csrf={{ ctx.csrf | safe }}" enctype="multipart/form-data">
  title: <input name="title" />
  file1: <input name="file1" type="file" />
  file2: <input name="file2" type="file" />
  <button type="submit">Upload</button>
</form>

Controller which hanlder POST /upload:

// app/controller/upload.js
const Controller = require('egg').Controller;
const fs = require('mz/fs');

module.exports = class extends Controller {
  async upload() {
    const { ctx } = this;
    console.log(ctx.request.body);
    console.log('got %d files', ctx.request.files.length);
    for (const file of ctx.request.files) {
      console.log('field: ' + file.fieldname);
      console.log('filename: ' + file.filename);
      console.log('encoding: ' + file.encoding);
      console.log('mime: ' + file.mime);
      console.log('tmp filepath: ' + file.filepath);
      let result;
      try {
        // process file or upload to cloud storage
        result = await ctx.oss.put('egg-multipart-test/' + file.filename, file.filepath);
      } finally {
        // remove tmp files and don't block the request's response
        // cleanupRequestFiles won't throw error even remove file io error happen
        ctx.cleanupRequestFiles([ file ]);
      }
      console.log(result);
    }
  }
};

stream mode: the hard way

If you're well-known about know the Node.js Stream work, you should use the stream mode.

Use with for await...of

<form method="POST" action="/upload?_csrf={{ ctx.csrf | safe }}" enctype="multipart/form-data">
  title: <input name="title" />
  file1: <input name="file1" type="file" />
  file2: <input name="file2" type="file" />
  <button type="submit">Upload</button>
</form>

Controller which hanlder POST /upload:

// app/controller/upload.js
const { Controller } = require('egg');
const fs = require('fs');
const stream = require('stream');
const util = require('util');
const { randomUUID } = require('crypto');
const pipeline = util.promisify(stream.pipeline);

module.exports = class UploadController extends Controller {
  async upload() {
    const parts = this.ctx.multipart();
    const fields = {};
    const files = {};

    for await (const part of parts) {
      if (Array.isArray(part)) {
        // fields
        console.log('field: ' + part[0]);
        console.log('value: ' + part[1]);
      } else {
        // otherwise, it's a stream
        const { filename, fieldname, encoding, mime } = part;

        console.log('field: ' + fieldname);
        console.log('filename: ' + filename);
        console.log('encoding: ' + encoding);
        console.log('mime: ' + mime);

        // how to handler?
        // 1. save to tmpdir with pipeline
        // 2. or send to oss
        // 3. or just consume it with another for await

        // WARNING: You should almost never use the origin filename as it could contain malicious input.
        const targetPath = path.join(os.tmpdir(), randomUUID() + path.extname(filename));
        await pipeline(part, createWriteStream(targetPath)); // use `pipeline` not `pipe`
      }
    }

    this.ctx.body = 'ok';
  }
};

Upload One File (DEPRECATED)

You can got upload stream by ctx.getFileStream*().

<form method="POST" action="/upload?_csrf={{ ctx.csrf | safe }}" enctype="multipart/form-data">
  title: <input name="title" />
  file: <input name="file" type="file" />
  <button type="submit">Upload</button>
</form>

Controller which handler POST /upload:

// app/controller/upload.js
const path = require('path');
const sendToWormhole = require('stream-wormhole');
const Controller = require('egg').Controller;

module.exports = class extends Controller {
  async upload() {
    const { ctx } = this;
    // file not exists will response 400 error
    const stream = await ctx.getFileStream();
    const name = 'egg-multipart-test/' + path.basename(stream.filename);
    // process file or upload to cloud storage
    const result = await ctx.oss.put(name, stream);

    ctx.body = {
      url: result.url,
      // process form fields by `stream.fields`
      fields: stream.fields,
    };
  }

  async uploadNotRequiredFile() {
    const { ctx } = this;
    // file not required
    const stream = await ctx.getFileStream({ requireFile: false });
    let result;
    if (stream.filename) {
      const name = 'egg-multipart-test/' + path.basename(stream.filename);
      // process file or upload to cloud storage
      const result = await ctx.oss.put(name, stream);
    } else {
      // must consume the empty stream
      await sendToWormhole(stream);
    }

    ctx.body = {
      url: result && result.url,
      // process form fields by `stream.fields`
      fields: stream.fields,
    };
  }
};

Upload Multiple Files (DEPRECATED)

<form method="POST" action="/upload?_csrf={{ ctx.csrf | safe }}" enctype="multipart/form-data">
  title: <input name="title" />
  file1: <input name="file1" type="file" />
  file2: <input name="file2" type="file" />
  <button type="submit">Upload</button>
</form>

Controller which hanlder POST /upload:

// app/controller/upload.js
const Controller = require('egg').Controller;

module.exports = class extends Controller {
  async upload() {
    const { ctx } = this;
    const parts = ctx.multipart();
    let part;
    while ((part = await parts()) != null) {
      if (part.length) {
        // arrays are busboy fields
        console.log('field: ' + part[0]);
        console.log('value: ' + part[1]);
        console.log('valueTruncated: ' + part[2]);
        console.log('fieldnameTruncated: ' + part[3]);
      } else {
        if (!part.filename) {
          // user click `upload` before choose a file,
          // `part` will be file stream, but `part.filename` is empty
          // must handler this, such as log error.
          continue;
        }
        // otherwise, it's a stream
        console.log('field: ' + part.fieldname);
        console.log('filename: ' + part.filename);
        console.log('encoding: ' + part.encoding);
        console.log('mime: ' + part.mime);
        const result = await ctx.oss.put('egg-multipart-test/' + part.filename, part);
        console.log(result);
      }
    }
    console.log('and we are done parsing the form!');
  }
};

Support file and stream mode in the same time

If the default mode is stream, use the fileModeMatch options to match the request urls switch to file mode.

config.multipart = {
  mode: 'stream',
  // let POST /upload_file request use the file mode, other requests use the stream mode.
  fileModeMatch: /^\/upload_file$/,
  // or glob
  // fileModeMatch: '/upload_file',
};

NOTICE: fileModeMatch options only work on stream mode.

License

MIT

More Repositories

1

egg

🥚 Born to build better enterprise frameworks and apps with Node.js & Koa
JavaScript
18,650
star
2

examples

Store all egg examples in one place
JavaScript
1,748
star
3

awesome-egg

Awesome Egg.js Web Framework and Plugin.
869
star
4

egg-sequelize

Sequelize for Egg.js
JavaScript
609
star
5

egg-mongoose

JavaScript
437
star
6

egg-graphql

JavaScript
371
star
7

egg-mysql

MySQL plugin for egg
TypeScript
328
star
8

egg-ant-design-pro

showcase for Egg loves Ant Design
JavaScript
318
star
9

egg-redis

redis plugin for egg
JavaScript
267
star
10

egg-validate

validate plugin for egg
JavaScript
259
star
11

egg-security

Security plugin for egg, force performance too.
JavaScript
237
star
12

egg-socket.io

socket.io plugin for eggjs.
JavaScript
237
star
13

egg-cluster

cluster manager for egg
JavaScript
218
star
14

egg-core

A core Pluggable framework based on koa.
JavaScript
217
star
15

tegg

Strong Type framework with eggjs.
TypeScript
185
star
16

egg-bin

egg developer tool
TypeScript
183
star
17

egg-ts-helper

🍳 Generate TypeScript definition files(d.ts) for Egg
TypeScript
157
star
18

egg-init

Init egg app helper tools
JavaScript
151
star
19

egg-view-vue

vue view plugin for egg
JavaScript
150
star
20

egg-cors

CORS plugin for egg
JavaScript
150
star
21

egg-logger

Egg logger
JavaScript
150
star
22

vscode-eggjs

vscode extension for https://eggjs.org/
TypeScript
145
star
23

egg-mock

Mock library for egg testing.
JavaScript
144
star
24

egg-router-plus

The missing router feature for eggjs
JavaScript
140
star
25

egg-scripts

deploy tool for egg projects
JavaScript
112
star
26

egg-rest

Restful API plugin for egg
JavaScript
107
star
27

egg-passport

passport plugin for egg
JavaScript
106
star
28

egg-view-react

egg view plugin for react
JavaScript
89
star
29

egg-view-nunjucks

nunjucks view plugin for egg
JavaScript
89
star
30

egg-static

static server plugin for egg
JavaScript
89
star
31

egg-schedule

Schedule plugin for egg
JavaScript
88
star
32

egg-oss

aliyun oss plugin for egg
JavaScript
87
star
33

egg-grpc

grpc plugin for egg
JavaScript
81
star
34

egg-sofa-rpc

SOFARPC plugin for egg
JavaScript
80
star
35

egg-userrole

user role plugin for egg
JavaScript
77
star
36

eslint-config-egg

Node Style Guide for Egg.
JavaScript
67
star
37

egg-view-ejs

egg view plugin for ejs.
JavaScript
64
star
38

egg-onerror

error handler for egg
Mustache
56
star
39

egg-alinode

alinode plugin for egg
JavaScript
54
star
40

egg-logrotator

Log rotate plugin for egg
JavaScript
53
star
41

egg-view-assets

Manage frontend assets in development and production.
JavaScript
52
star
42

egg-router

router for eggjs, fork from koa-router with some additional features
JavaScript
50
star
43

egg-orm

Object relational mapping for Egg framework
JavaScript
49
star
44

aliyun-egg

node web framework for aliyun, base on eggjs
JavaScript
49
star
45

egg-cancan

cancancan like authorization plugin for Egg.js
JavaScript
48
star
46

egg-session

session plugin for egg
JavaScript
47
star
47

egg-healthy

Liveness and Readiness health check for egg application
JavaScript
45
star
48

egg-passport-local

wrap passport-local strategy for egg-passport
JavaScript
45
star
49

egg-aop

AOP plugin for eggjs, add DI, AOP support.
TypeScript
44
star
50

egg-session-redis

redis store for egg session
JavaScript
43
star
51

egg-showcase-aliyun-blog

A blog showcase for aliyun-egg
JavaScript
42
star
52

egg-view

JavaScript
41
star
53

jar2proxy

transfer java facade jar to proxyjs
Java
41
star
54

egg-dubbo-rpc

dubbo rpc plugin for egg
JavaScript
40
star
55

egg-boilerplate-ts

Boilerplate for egg typescript project
TypeScript
40
star
56

egg-i18n

i18n plugin for egg
JavaScript
39
star
57

doctools

doctools for eggjs
JavaScript
38
star
58

egg-prometheus

Prometheus plugin for Egg.js
JavaScript
38
star
59

egg-http-proxy

http proxy for egg
JavaScript
38
star
60

egg-cloud

egg cloud provider tools for developers to quickly build some of the common patterns in distributed systems
JavaScript
38
star
61

egg-rpc-generator

RPC tools for egg framework
JavaScript
37
star
62

egg-errors

Errors for Egg.js.
TypeScript
31
star
63

egg-dingtalk

egg plugin for dingtalk
JavaScript
30
star
64

egg-cookies

cookies module for egg, base on pillarjs/cookies
JavaScript
30
star
65

egg-rpc

rpc plugin for eggjs
JavaScript
29
star
66

egg-tracer

tracer plugin for egg
JavaScript
29
star
67

create-egg

so you could use `npm init egg showcase` to init egg project
JavaScript
28
star
68

docker

Egg official docker image
JavaScript
27
star
69

egg-websocket

egg plugin for websocket
JavaScript
26
star
70

egg-path-matching

url path match/ignore support string, regexp and function
JavaScript
26
star
71

egg-lru

egg lru-cache plugin
JavaScript
25
star
72

egg-boilerplate-microservice

Boilerplate for egg microservice project
JavaScript
25
star
73

egg-watcher

Watcher plugin for egg
JavaScript
25
star
74

egg-parameters

Merge all parameters (ctx.params, ctx.request.query, ctx.request.body) into ctx.params like Rails application.
JavaScript
24
star
75

egg-development-proxyagent

[DEPRECATED] A proxy adapter for debugging httpclient on egg.
JavaScript
21
star
76

egg-userservice

userservice plugin for egg
JavaScript
21
star
77

egg-zookeeper

egg plugin for zookeeper
JavaScript
20
star
78

benchmark

benchmark for egg
JavaScript
20
star
79

egg-opentracing

Implementation of opentracing in Egg.js
JavaScript
19
star
80

egg-datahub

Macaca DataHub plugin for Egg.js
JavaScript
19
star
81

egg-ons

aliyun ons plugin for egg
JavaScript
18
star
82

egg-validate-schema

validate by json-schema plugin for egg
JavaScript
17
star
83

egg-development-proxyworker

**[DEPRECATED]** A proxy worker for debugging worker on egg
JavaScript
17
star
84

egg-boilerplate-simple

Boilerplate for egg simple project
JavaScript
16
star
85

eggjs.github.io

eggjs docs site
JavaScript
16
star
86

egg-logger-sls

Logger transport for aliyun sls.
JavaScript
15
star
87

egg-logview

Provide a log files viewer for development purpose.
JavaScript
15
star
88

egg-ci

Auto gen ci config file
JavaScript
15
star
89

egg-jsonp

jsonp support for egg, with security check inside
JavaScript
14
star
90

egg-boilerplate-plugin

Boilerplate for egg plugin
JavaScript
14
star
91

koa-express-adapter

The adapter for the migration from express to koa
JavaScript
14
star
92

egg-boilerplate-alipay-tiny

支付宝小程序服务端脚手架
JavaScript
14
star
93

egg-view-handlebars

egg view plugin for handlebars
JavaScript
12
star
94

egg-passport-github

github passport plugin for egg
JavaScript
11
star
95

egg-leancloud

leancloud plugin for Egg.js
JavaScript
11
star
96

egg-instrument

Compute the duration of an operation in local environment.
JavaScript
11
star
97

egg-utils

Utils for all egg projects.
TypeScript
11
star
98

egg-tslint-to-eslint

Migrate tslint-config-egg to eslint-config-egg in ts project.
JavaScript
9
star
99

egg-lookout

SOFALookout plugin for egg
JavaScript
9
star
100

egg-host

A host plugin for egg. It supports modify the host config, just like edit the /etc/hosts.
JavaScript
9
star