Skip to content Skip to sidebar Skip to footer

Multiple File Upload Only Uploads The First One After Using Express File Upload

So I'm working on a function which takes in an array of file data that is acquired from user input from the front end. The files are inside req.files.fileInput which is an array of

Solution 1:

This is kind of a quick and dirty solution that should nonetheless work:

  • wrap the whole loop in the Promise executor function.
  • introduce a local count state to keep track of the number of loop iteration that have completed successfully.
  • introduce a isInError state to keep track of whether a loop iteration has failed.
  • In all cases, resolve - or reject - the promise only on the first iteration that results in an error, or when all files have completed upload successfully.

const uploadFunc = (fileDataArr) => {
    return new Promise((resolve, reject) => {
        var count = fileDataArr.length
        var isInError = false

        for (let eachFile of fileDataArr) {
            const fileName = eachFile.name;
            const uploadPath = `./images/${fileName}`;

            eachFile.mv(uploadPath, function (err) {
                if (err && !isInError) {
                    isInError = true
                    return resolve({
                        success: false,
                        message: 'Something went wrong. Please try again!',
                        data: null
                    });
                }

                if (--count == 0 && !isInError) {
                    return resolve({
                        success: true,
                        message: 'Files Uploaded Successfully',
                        data: null
                    });
                }
            });
        }
    });
};

Note I have stuck with your logic of resolving the promise when an error occurs - but rejecting may actually make more sense and is actually more semantically correct.

As I pointed out in the comments, look up promisifying asynchronous functions and the async/await syntax for a cleaner solution.


Solution 2:

@IAmDranged solution is great. However, you can also do this

  • Move file.mv() process into its own promise function called fileMoveAsync(file)
  • Map through all the files and pass each to the fileMoveAsync(file)
  • Use Promise.all() to await all the fileMoveAsync(file)
  • If all goes well with the Promise.all() then resolve otherwise reject promise accordingly

See code below.

The fileMoveAsync function

const fileMoveAsync = async (file) =>
  new Promise((resolve, reject) => {
    const uploadPath = `./resources/images/${file.name}`;
    return file.mv(uploadPath, (err) => {
      if (err) {
        console.error(err);

        return reject({
          success: false,
          message: 'Something went wrong. Please upload again!',
          data: null,
        });
      }

      return resolve({
        success: true,
        message: 'File Uploaded Successfully!',
        data: file.name,
      });
    });
  });
const uploadFunc = async (file) => {
  if (Array.isArray(file)) {
    try {
      const data = await Promise.all(file.map((x) => fileMoveAsync(x)));
      return Promise.resolve({
        success: true,
        message: 'Files Uploaded Successfully!',
        data,
      });
    } catch (e) {
      return Promise.reject({
        success: false,
        message: 'Something went wrong. Please upload again!',
        data: null,
      });
    }
  } else if (typeof file === 'object') {
    return fileMoveAsync(file);
  }
};

Use it like this

const { success, message, data } = await uploadFunc(req.files.fileInput);

Post a Comment for "Multiple File Upload Only Uploads The First One After Using Express File Upload"