123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311 |
- 'use strict';
- const fs = require('fs');
- const path = require('path');
- const { Readable } = require('stream');
- // Parameters for safe file name parsing.
- const SAFE_FILE_NAME_REGEX = /[^\w-]/g;
- const MAX_EXTENSION_LENGTH = 3;
- // Parameters to generate unique temporary file names:
- const TEMP_COUNTER_MAX = 65536;
- const TEMP_PREFIX = 'tmp';
- let tempCounter = 0;
- /**
- * Logs message to console if debug option set to true.
- * @param {Object} options - options object.
- * @param {string} msg - message to log.
- * @returns {boolean} - false if debug is off.
- */
- const debugLog = (options, msg) => {
- const opts = options || {};
- if (!opts.debug) return false;
- console.log(`Express-file-upload: ${msg}`); // eslint-disable-line
- return true;
- };
- /**
- * Generates unique temporary file name. e.g. tmp-5000-156788789789.
- * @param {string} prefix - a prefix for generated unique file name.
- * @returns {string}
- */
- const getTempFilename = (prefix = TEMP_PREFIX) => {
- tempCounter = tempCounter >= TEMP_COUNTER_MAX ? 1 : tempCounter + 1;
- return `${prefix}-${tempCounter}-${Date.now()}`;
- };
- /**
- * isFunc: Checks if argument is a function.
- * @returns {boolean} - Returns true if argument is a function.
- */
- const isFunc = func => func && func.constructor && func.call && func.apply ? true: false;
- /**
- * Set errorFunc to the same value as successFunc for callback mode.
- * @returns {Function}
- */
- const errorFunc = (resolve, reject) => isFunc(reject) ? reject : resolve;
- /**
- * Return a callback function for promise resole/reject args.
- * Ensures that callback is called only once.
- * @returns {Function}
- */
- const promiseCallback = (resolve, reject) => {
- let hasFired = false;
- return (err) => {
- if (hasFired) {
- return;
- }
- hasFired = true;
- return err ? errorFunc(resolve, reject)(err) : resolve();
- };
- };
- /**
- * Builds instance options from arguments objects(can't be arrow function).
- * @returns {Object} - result options.
- */
- const buildOptions = function() {
- const result = {};
- [...arguments].forEach(options => {
- if (!options || typeof options !== 'object') return;
- Object.keys(options).forEach(i => result[i] = options[i]);
- });
- return result;
- };
- // The default prototypes for both objects and arrays.
- // Used by isSafeFromPollution
- const OBJECT_PROTOTYPE_KEYS = Object.getOwnPropertyNames(Object.prototype);
- const ARRAY_PROTOTYPE_KEYS = Object.getOwnPropertyNames(Array.prototype);
- /**
- * Determines whether a key insertion into an object could result in a prototype pollution
- * @param {Object} base - The object whose insertion we are checking
- * @param {string} key - The key that will be inserted
- */
- const isSafeFromPollution = (base, key) => {
- // We perform an instanceof check instead of Array.isArray as the former is more
- // permissive for cases in which the object as an Array prototype but was not constructed
- // via an Array constructor or literal.
- const TOUCHES_ARRAY_PROTOTYPE = (base instanceof Array) && ARRAY_PROTOTYPE_KEYS.includes(key);
- const TOUCHES_OBJECT_PROTOTYPE = OBJECT_PROTOTYPE_KEYS.includes(key);
- return !TOUCHES_ARRAY_PROTOTYPE && !TOUCHES_OBJECT_PROTOTYPE;
- };
- /**
- * Builds request fields (using to build req.body and req.files)
- * @param {Object} instance - request object.
- * @param {string} field - field name.
- * @param {any} value - field value.
- * @returns {Object}
- */
- const buildFields = (instance, field, value) => {
- // Do nothing if value is not set.
- if (value === null || value === undefined) return instance;
- instance = instance || Object.create(null);
- if (!isSafeFromPollution(instance, field)) {
- return instance;
- }
- // Non-array fields
- if (!instance[field]) {
- instance[field] = value;
- return instance;
- }
- // Array fields
- if (instance[field] instanceof Array) {
- instance[field].push(value);
- } else {
- instance[field] = [instance[field], value];
- }
- return instance;
- };
- /**
- * Creates a folder for file specified in the path variable
- * @param {Object} fileUploadOptions
- * @param {string} filePath
- * @returns {boolean}
- */
- const checkAndMakeDir = (fileUploadOptions, filePath) => {
- // Check upload options were set.
- if (!fileUploadOptions) return false;
- if (!fileUploadOptions.createParentPath) return false;
- // Check whether folder for the file exists.
- if (!filePath) return false;
- const parentPath = path.dirname(filePath);
- // Create folder if it doesn't exist.
- if (!fs.existsSync(parentPath)) fs.mkdirSync(parentPath, { recursive: true });
- // Checks folder again and return a results.
- return fs.existsSync(parentPath);
- };
- /**
- * Deletes a file.
- * @param {string} file - Path to the file to delete.
- * @param {Function} callback
- */
- const deleteFile = (file, callback) => fs.unlink(file, callback);
- /**
- * Copy file via streams
- * @param {string} src - Path to the source file
- * @param {string} dst - Path to the destination file.
- */
- const copyFile = (src, dst, callback) => {
- // cbCalled flag and runCb helps to run cb only once.
- let cbCalled = false;
- let runCb = (err) => {
- if (cbCalled) return;
- cbCalled = true;
- callback(err);
- };
- // Create read stream
- let readable = fs.createReadStream(src);
- readable.on('error', runCb);
- // Create write stream
- let writable = fs.createWriteStream(dst);
- writable.on('error', (err)=>{
- readable.destroy();
- runCb(err);
- });
- writable.on('close', () => runCb());
- // Copy file via piping streams.
- readable.pipe(writable);
- };
- /**
- * moveFile: moves the file from src to dst.
- * Firstly trying to rename the file if no luck copying it to dst and then deleteing src.
- * @param {string} src - Path to the source file
- * @param {string} dst - Path to the destination file.
- * @param {Function} callback - A callback function.
- */
- const moveFile = (src, dst, callback) => fs.rename(src, dst, err => (err
- ? copyFile(src, dst, err => err ? callback(err) : deleteFile(src, callback))
- : callback()
- ));
- /**
- * Save buffer data to a file.
- * @param {Buffer} buffer - buffer to save to a file.
- * @param {string} filePath - path to a file.
- */
- const saveBufferToFile = (buffer, filePath, callback) => {
- if (!Buffer.isBuffer(buffer)) {
- return callback(new Error('buffer variable should be type of Buffer!'));
- }
- // Setup readable stream from buffer.
- let streamData = buffer;
- let readStream = Readable();
- readStream._read = () => {
- readStream.push(streamData);
- streamData = null;
- };
- // Setup file system writable stream.
- let fstream = fs.createWriteStream(filePath);
- // console.log("Calling saveBuffer");
- fstream.on('error', err => {
- // console.log("err cb")
- callback(err);
- });
- fstream.on('close', () => {
- // console.log("close cb");
- callback();
- });
- // Copy file via piping streams.
- readStream.pipe(fstream);
- };
- /**
- * Decodes uriEncoded file names.
- * @param fileName {String} - file name to decode.
- * @returns {String}
- */
- const uriDecodeFileName = (opts, fileName) => {
- return opts.uriDecodeFileNames ? decodeURIComponent(fileName) : fileName;
- };
- /**
- * Parses filename and extension and returns object {name, extension}.
- * @param {boolean|integer} preserveExtension - true/false or number of characters for extension.
- * @param {string} fileName - file name to parse.
- * @returns {Object} - { name, extension }.
- */
- const parseFileNameExtension = (preserveExtension, fileName) => {
- const preserveExtensionLength = parseInt(preserveExtension);
- const result = {name: fileName, extension: ''};
- if (!preserveExtension && preserveExtensionLength !== 0) return result;
- // Define maximum extension length
- const maxExtLength = isNaN(preserveExtensionLength)
- ? MAX_EXTENSION_LENGTH
- : Math.abs(preserveExtensionLength);
- const nameParts = fileName.split('.');
- if (nameParts.length < 2) return result;
- let extension = nameParts.pop();
- if (
- extension.length > maxExtLength &&
- maxExtLength > 0
- ) {
- nameParts[nameParts.length - 1] +=
- '.' +
- extension.substr(0, extension.length - maxExtLength);
- extension = extension.substr(-maxExtLength);
- }
- result.extension = maxExtLength ? extension : '';
- result.name = nameParts.join('.');
- return result;
- };
- /**
- * Parse file name and extension.
- * @param {Object} opts - middleware options.
- * @param {string} fileName - Uploaded file name.
- * @returns {string}
- */
- const parseFileName = (opts, fileName) => {
- // Check fileName argument
- if (!fileName || typeof fileName !== 'string') return getTempFilename();
- // Cut off file name if it's lenght more then 255.
- let parsedName = fileName.length <= 255 ? fileName : fileName.substr(0, 255);
- // Decode file name if uriDecodeFileNames option set true.
- parsedName = uriDecodeFileName(opts, parsedName);
- // Stop parsing file name if safeFileNames options hasn't been set.
- if (!opts.safeFileNames) return parsedName;
- // Set regular expression for the file name.
- const nameRegex = typeof opts.safeFileNames === 'object' && opts.safeFileNames instanceof RegExp
- ? opts.safeFileNames
- : SAFE_FILE_NAME_REGEX;
- // Parse file name extension.
- let {name, extension} = parseFileNameExtension(opts.preserveExtension, parsedName);
- if (extension.length) extension = '.' + extension.replace(nameRegex, '');
- return name.replace(nameRegex, '').concat(extension);
- };
- module.exports = {
- isFunc,
- debugLog,
- copyFile, // For testing purpose.
- moveFile,
- errorFunc,
- deleteFile, // For testing purpose.
- buildFields,
- buildOptions,
- parseFileName,
- getTempFilename,
- promiseCallback,
- checkAndMakeDir,
- saveBufferToFile,
- uriDecodeFileName,
- isSafeFromPollution
- };
|