Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

fixes(#1283): Add documentation for Multer functions #1284

Open
wants to merge 1 commit into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
251 changes: 251 additions & 0 deletions index.js
Original file line number Diff line number Diff line change
Expand Up @@ -4,10 +4,43 @@ var diskStorage = require('./storage/disk')
var memoryStorage = require('./storage/memory')
var MulterError = require('./lib/multer-error')

/**
* Default file filter function that allows all files to be processed.
*
* @param {Object} req - The HTTP request object.
* @param {Object} file - The file object containing file details such as `fieldname`, `originalname`, `encoding`, and `mimetype`.
* @param {Function} cb - A callback function that takes two arguments:
* - `err` (Error, if any, or `null` to allow the file)
Copy link
Member

@IamLizu IamLizu Dec 26, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@arham-sayyed hey 👋

Can you please help me understand this line (13) according the code written in the function?

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hey 👋

Sure! Let me explain line 13:

 * - `err` (Error, if any, or `null` to allow the file)

This describes the first parameter of the callback function (cb) used in the allowAll function. The err parameter is intended to indicate if there's an error that should prevent the file from being processed.

  • If you pass null as the err value (as done in cb(null, true)), it means there's no error, and the file is allowed to proceed.
  • If you pass an Error object instead of null, it will stop the processing of that specific file and trigger an error in the file upload flow.

Essentially, this mechanism allows you to handle file validation or other conditions dynamically, depending on your use case. In this case, since allowAll is a default filter that accepts all files, we simply pass null for err to indicate no error. 😊

Let me know if you have any further questions!

* - `acceptFile` (Boolean, `true` to accept the file, `false` to reject it)
*/
function allowAll (req, file, cb) {
cb(null, true)
}

/**
* Multer constructor function that configures the file handling middleware.
*
* @class
* @param {Object} options - Configuration options for Multer.
* @param {Object} [options.storage] - Storage engine to use for saving files.
* @param {string} [options.dest] - Directory path for storing files, if no custom storage engine is provided.
* @param {Object} [options.limits] - Limits for file size, field size, etc.
* @param {boolean} [options.preservePath=false] - Whether to preserve the file's original path.
* @param {Function} [options.fileFilter=allowAll] - Function to filter which files to accept or reject. Receives `req`, `file`, and a callback function as arguments.
*
* @throws {TypeError} Throws an error if `options` is neither undefined nor an object.
*
* @property {Object} storage - Storage engine used for saving files.
* @property {Object} limits - File size and count limits.
* @property {boolean} preservePath - Flag for preserving original file paths.
* @property {Function} fileFilter - Filter function to determine if a file should be saved.
*
* @example
* const upload = new Multer({ dest: 'uploads/' });
*
* @description
* Multer is a file handling middleware for Node.js. It provides configuration for file storage location, size limits, and file filtering, enabling file uploads for multipart form-data.
*/
function Multer (options) {
if (options.storage) {
this.storage = options.storage
Expand All @@ -22,6 +55,29 @@ function Multer (options) {
this.fileFilter = options.fileFilter || allowAll
}

/**
* Creates a middleware function for handling file uploads with custom settings.
*
* @private
* @param {Array<{name: string, maxCount: number}>} fields - Array of field configurations. Each field object specifies a `name` and a `maxCount` for the field, defining which fields to handle and how many files are allowed per field.
* @param {string} fileStrategy - Strategy for handling files, which can be 'VALUE', 'ARRAY', 'OBJECT', or 'NONE'.
*
* @returns {Function} Middleware function that can handle multipart file uploads.
*
* @description
* `_makeMiddleware` generates middleware for processing file uploads, controlling how fields and files are managed based on specified fields and file strategy.
*
* - **File Strategy**: Determines how uploaded files are organized:
* - `VALUE`: Allows a single file per field.
* - `ARRAY`: Allows multiple files to be collected as an array.
* - `OBJECT`: Stores files in an object keyed by field name.
* - `NONE`: No files are allowed.
*
* @example
* const upload = new Multer({ dest: 'uploads/' });
* const singleUploadMiddleware = upload.single('avatar');
* const arrayUploadMiddleware = upload.array('photos', 5);
*/
Multer.prototype._makeMiddleware = function (fields, fileStrategy) {
function setup () {
var fileFilter = this.fileFilter
Expand All @@ -35,6 +91,33 @@ Multer.prototype._makeMiddleware = function (fields, fileStrategy) {
}
})

/**
* Wraps the user-defined file filter to enforce field-specific file count limits.
*
* @param {Object} req - The Express request object.
* @param {Object} file - The file object representing the current file being processed.
* @param {Function} cb - Callback function to indicate whether the file should be accepted or rejected. Calls with `cb(null, true)` to accept or `cb(new Error)` to reject.
*
* @throws {MulterError} Throws a `LIMIT_UNEXPECTED_FILE` error if the file exceeds the allowed count for its field.
*
* @description
* `wrappedFileFilter` is an internal function used to wrap the user-defined `fileFilter`. It ensures that each field does not exceed its specified file count (`maxCount`), calling the provided `fileFilter` function only when this condition is met.
*
* - **Exceeding Limit**: If the number of files for a field exceeds its `maxCount`, the function calls `cb` with a `MulterError`, preventing additional files for that field from being processed.
* - **Calling User File Filter**: After verifying the limit, it passes the file to the user-defined `fileFilter` for custom processing or further validation.
*
* @example
* const upload = multer({ limits: { fileSize: 1000000 } });
* const fileFilter = (req, file, cb) => {
* if (file.mimetype === 'image/jpeg') cb(null, true);
* else cb(new Error('Only JPEGs are allowed'));
* };
*
* const middleware = upload.fields([
* { name: 'avatar', maxCount: 1 },
* { name: 'gallery', maxCount: 5 }
* ]);
*/
function wrappedFileFilter (req, file, cb) {
if ((filesLeft[file.fieldname] || 0) <= 0) {
return cb(new MulterError('LIMIT_UNEXPECTED_FILE', file.fieldname))
Expand All @@ -56,22 +139,148 @@ Multer.prototype._makeMiddleware = function (fields, fileStrategy) {
return makeMiddleware(setup.bind(this))
}

/**
* Creates middleware for handling a single file upload for a specified field.
*
* @param {string} name - The name of the form field to accept a single file upload.
* @returns {Function} Middleware function to handle single file uploads.
*
* @description
* `Multer.prototype.single` is a method that generates middleware specifically for handling single file uploads for a specified field. This middleware will only accept one file for the given `name` field, rejecting additional files if provided in the same field.
*
* - **File Handling Strategy**: Sets the `fileStrategy` to `'VALUE'`, meaning only a single file is stored as `req.file` rather than in an array or object.
* - **Field-Specific Limit**: Allows only one file in the specified field name and rejects additional files in that field.
*
* @example
* // Usage example:
* const multer = require('multer');
* const upload = multer({ dest: 'uploads/' });
*
* app.post('/profile', upload.single('avatar'), (req, res) => {
* // Access the uploaded file with req.file
* res.send('Single file upload successful!');
* });
*
* @throws {MulterError} Throws `LIMIT_UNEXPECTED_FILE` if additional files are uploaded in the specified field.
*/
Multer.prototype.single = function (name) {
return this._makeMiddleware([{ name: name, maxCount: 1 }], 'VALUE')
}

/**
* Creates middleware for handling multiple file uploads for a specified field.
*
* @param {string} name - The name of the form field to accept multiple file uploads.
* @param {number} [maxCount] - Optional maximum number of files allowed for the specified field. If not specified, there is no limit.
* @returns {Function} Middleware function to handle multiple file uploads.
*
* @description
* `Multer.prototype.array` is a method that generates middleware for handling multiple file uploads on a specified field name. This middleware allows multiple files to be uploaded under the same field, storing them in `req.files` as an array.
*
* - **File Handling Strategy**: Sets the `fileStrategy` to `'ARRAY'`, meaning files are stored as an array under `req.files`, making each file accessible as individual elements in the array.
* - **Field-Specific Limit**: Accepts multiple files but can limit the number of files in the specified field if `maxCount` is provided.
*
* @example
* // Usage example:
* const multer = require('multer');
* const upload = multer({ dest: 'uploads/' });
*
* app.post('/photos', upload.array('photos', 10), (req, res) => {
* // Access uploaded files with req.files
* res.send(`Uploaded ${req.files.length} photos successfully!`);
* });
*
* @throws {MulterError} Throws `LIMIT_UNEXPECTED_FILE` if the number of uploaded files exceeds `maxCount`.
*/
Multer.prototype.array = function (name, maxCount) {
return this._makeMiddleware([{ name: name, maxCount: maxCount }], 'ARRAY')
}

/**
* Creates middleware for handling multiple file uploads for multiple fields.
*
* @param {Array<Object>} fields - An array of objects specifying fields to accept files for. Each object in the array should have:
* - `name` (string): The name of the field to accept file uploads.
* - `maxCount` (number, optional): Maximum number of files allowed for this field. Defaults to `Infinity` if not provided.
* @returns {Function} Middleware function to handle file uploads across multiple fields.
*
* @description
* `Multer.prototype.fields` generates middleware that allows handling file uploads across multiple form fields, with each field accepting a defined number of files. Uploaded files are organized in `req.files` as an object where each key is a field name, and each value is an array of files for that field.
*
* - **File Handling Strategy**: Sets the `fileStrategy` to `'OBJECT'`, organizing files in `req.files` by field name.
* - **Field-Specific Limits**: Each field can specify a `maxCount` to restrict the number of files for that particular field.
*
* @example
* // Usage example:
* const multer = require('multer');
* const upload = multer({ dest: 'uploads/' });
*
* app.post('/profile', upload.fields([
* { name: 'avatar', maxCount: 1 },
* { name: 'gallery', maxCount: 5 }
* ]), (req, res) => {
* // Access uploaded files with req.files
* res.send(`Uploaded ${req.files['gallery'].length} gallery images successfully!`);
* });
*
* @throws {MulterError} Throws `LIMIT_UNEXPECTED_FILE` if the number of uploaded files exceeds the specified `maxCount` for any field.
*/
Multer.prototype.fields = function (fields) {
return this._makeMiddleware(fields, 'OBJECT')
}

/**
* Creates middleware for handling form data without any file uploads.
*
* @returns {Function} Middleware function that processes only text fields and rejects any file uploads.
*
* @description
* `Multer.prototype.none` generates middleware that processes only form fields without allowing any file uploads.
* If any file is uploaded, it will result in an error, as this middleware is strictly for text-based form submissions.
*
* - **File Handling Strategy**: Sets the `fileStrategy` to `'NONE'`, ensuring `req.files` will be empty.
* - **Text-Only Forms**: Suitable for forms that only contain text fields.
*
* @example
* // Usage example:
* const multer = require('multer');
* const upload = multer();
*
* app.post('/submit', upload.none(), (req, res) => {
* // Access form fields through req.body
* res.send(`Received submission with name: ${req.body.name}`);
* });
*
* @throws {MulterError} Throws `LIMIT_UNEXPECTED_FILE` if any file upload is attempted.
*/
Multer.prototype.none = function () {
return this._makeMiddleware([], 'NONE')
}

/**
* Creates middleware for handling any number of file uploads on any field name.
*
* @returns {Function} Middleware function that accepts files uploaded on any field name, storing them in `req.files` as an array.
*
* @description
* `Multer.prototype.any` generates middleware that allows unlimited file uploads on any field, storing all files in `req.files` as an array.
* This middleware is flexible for forms where the field names for file uploads are dynamic or undefined in advance.
*
* - **File Handling Strategy**: Sets the `fileStrategy` to `'ARRAY'`, ensuring all uploaded files are appended to `req.files`.
* - **Flexible Field Handling**: Accepts files on any field name, unlike `single`, `array`, or `fields`, which target specific fields.
*
* @example
* // Usage example:
* const multer = require('multer');
* const upload = multer();
*
* app.post('/upload', upload.any(), (req, res) => {
* // Access all uploaded files through req.files array
* res.send(`Uploaded ${req.files.length} files`);
* });
*
* @throws {MulterError} Throws an error for any violations of size, count, or field limits as configured.
*/
Multer.prototype.any = function () {
function setup () {
return {
Expand All @@ -86,6 +295,48 @@ Multer.prototype.any = function () {
return makeMiddleware(setup.bind(this))
}

/**
* Creates a new Multer instance for handling file uploads.
*
* @param {Object} [options] - Optional configuration options for the Multer instance.
* @param {Object} [options.storage] - A custom storage engine, such as `diskStorage` or `memoryStorage`.
* @param {String} [options.dest] - The destination directory for file uploads (if using disk storage).
* @param {Object} [options.limits] - Limits for file upload size, field counts, etc.
* @param {Boolean} [options.preservePath] - Whether to preserve the original file path.
* @param {Function} [options.fileFilter] - A function to filter file uploads based on their attributes.
*
* @returns {Multer} A new Multer instance configured with the provided options.
*
* @throws {TypeError} Throws an error if the options argument is not an object.
*
* @example
* // Example of using multer to handle file uploads:
* const multer = require('multer');
*
* // Using disk storage and custom file filter
* const upload = multer({
* storage: multer.diskStorage({
* destination: function (req, file, cb) {
* cb(null, 'uploads/')
* },
* filename: function (req, file, cb) {
* cb(null, Date.now() + path.extname(file.originalname))
* }
* }),
* limits: { fileSize: 10 * 1024 * 1024 }, // 10MB limit
* fileFilter: function (req, file, cb) {
* if (file.mimetype === 'image/jpeg' || file.mimetype === 'image/png') {
* cb(null, true); // Accept file
* } else {
* cb(new Error('Only JPEG and PNG files are allowed'), false); // Reject file
* }
* }
* });
*
* app.post('/upload', upload.single('image'), (req, res) => {
* res.send('File uploaded!');
* });
*/
function multer (options) {
if (options === undefined) {
return new Multer({})
Expand Down
22 changes: 22 additions & 0 deletions lib/counter.js
Original file line number Diff line number Diff line change
@@ -1,24 +1,46 @@
var EventEmitter = require('events').EventEmitter

/**
* A counter class that can be incremented or decremented.
* Emits a 'zero' event when the counter reaches zero.
* @extends EventEmitter
*/
function Counter () {
EventEmitter.call(this)
/** @type {number} The current value of the counter. */
this.value = 0
}

Counter.prototype = Object.create(EventEmitter.prototype)

/**
* Increments the counter by 1.
*/
Counter.prototype.increment = function increment () {
this.value++
}

/**
* Decrements the counter by 1.
* If the counter reaches zero, emits a 'zero' event.
*/
Counter.prototype.decrement = function decrement () {
if (--this.value === 0) this.emit('zero')
}

/**
* Checks if the counter value is zero.
* @returns {boolean} True if the counter is zero, false otherwise.
*/
Counter.prototype.isZero = function isZero () {
return (this.value === 0)
}

/**
* Calls a function when the counter reaches zero.
* If the counter is already zero, calls the function immediately.
* @param {Function} fn - The function to call when the counter reaches zero.
*/
Counter.prototype.onceZero = function onceZero (fn) {
if (this.isZero()) return fn()

Expand Down
Loading