46

I have an API endpoint that lets the client post their csv to our server then post it to someone else server. I have done our server part which save uploaded file to our server, but I can't get the other part done. I keep getting error { message: 'File not found', code: 400 } which may mean the file never reach the server. I'm using axios as an agent, does anyone know how to get this done? Thanks.

// file = uploaded file
const form_data = new FormData();
form_data.append("file", fs.createReadStream(file.path));
const request_config = {
    method: "post",
    url: url,
    headers: {
        "Authorization": "Bearer " + access_token,
        "Content-Type": "multipart/form-data"
    },
    data: form_data
};
return axios(request_config);

Update

As axios doc states as below and the API I'm trying to call requires a file

// data is the data to be sent as the request body // Only applicable for request methods 'PUT', 'POST', and 'PATCH' // When no transformRequest is set, must be of one of the following types: // - string, plain object, ArrayBuffer, ArrayBufferView, URLSearchParams // - Browser only: FormData, File, Blob // - Node only: Stream, Buffer

Is there any way to make axios send a file as a whole? Thanks.

Tri Nguyen
  • 1,688
  • 3
  • 18
  • 46

7 Answers7

68

The 2 oldest answers did not work for me. This, however, did the trick:

const FormData = require('form-data'); // npm install --save form-data

const form = new FormData();
form.append('file', fs.createReadStream(file.path));

const request_config = {
  headers: {
    'Authorization': `Bearer ${access_token}`,
    ...form.getHeaders()
  }
};

return axios.post(url, form, request_config);

form.getHeaders() returns an Object with the content-type as well as the boundary.
For example:

{ "content-type": "multipart/form-data; boundary=-------------------0123456789" }
blex
  • 24,941
  • 5
  • 39
  • 72
15

I'm thinking the createReadStream is your issue because its async. try this. Since createReadStream extends the event emitter, we can "listen" for when it finishes/ends.

var newFile = fs.createReadStream(file.path);

// personally I'd function out the inner body here and just call 
// to the function and pass in the newFile
newFile.on('end', function() {
  const form_data = new FormData();
  form_data.append("file", newFile, "filename.ext");
  const request_config = {
    method: "post",
    url: url,
    headers: {
        "Authorization": "Bearer " + access_token,
        "Content-Type": "multipart/form-data"
    },
    data: form_data
  };
  return axios(request_config);
});
Marcello B.
  • 4,177
  • 11
  • 45
  • 65
james emanon
  • 11,185
  • 11
  • 56
  • 97
  • 2
    Hi James. Your solution does work. However, does `createReadStream` read the entire file into memory, if yes, I might encounter memory problem for massive files. – Tri Nguyen Oct 29 '18 at 09:27
  • 3
    So, with the Request module (import it), you could do that easily like so: newFile.pipe(request(request_config)) . I think axios might be able to do it too. But the the idea is, stream in the file, and pipe it to the request/axios to transfer. – james emanon Oct 29 '18 at 19:58
  • @jamesemanon have you any code snippet to pipe the file stream for axios upload? if so, please post here – Kiran Maniya Nov 23 '19 at 05:58
  • How to do it in case one wants to upload multiple files ? @jamesemanon – Salim Shamim May 21 '20 at 14:33
  • what's the solution if we have used multer – HexaCrop Apr 12 '21 at 09:38
  • 2
    Hi! Thanks for the answer! In typescript I get an error: `Argument of type 'ReadStream' is not assignable to parameter of type 'string | Blob'. Type 'ReadStream' is missing the following properties from type 'Blob': size, type, arrayBuffer, slice, and 2 more.` Any ideas? – Gabriel Augusto Oct 24 '21 at 12:13
  • 1
    For typescript add `import FormData from "form-data";` – CleverPatrick Jul 17 '22 at 12:17
10

This is what you really need:

const form_data = new FormData();
form_data.append("file", fs.createReadStream(file.path));

const request_config = {
  headers: {
    "Authorization": "Bearer " + access_token,
    "Content-Type": "multipart/form-data"
  },
  data: form_data
};

return axios
  .post(url, form_data, request_config);
Alex Povar
  • 4,890
  • 3
  • 29
  • 44
  • It seems there is a little mistake in your sample: you miss the filename... The right code is ` form_data.append("file", fs.createReadStream(file.path), "filename.ext"); – Didier68 May 10 '23 at 08:40
8

In my case, fs.createReadStream(file.path) did not work.
I had to use buffer instead.

const form = new FormData();
form.append('file', fs.readFileSync(filePath), fileName);

const config = {
  headers: {
    Authorization: `Bearer ${auth.access_token}`,
    ...form.getHeaders(),
  },
};

axios.post(api, form.getBuffer(), config);

Dharman
  • 30,962
  • 25
  • 85
  • 135
Marcelo Souza
  • 146
  • 1
  • 4
1

I have made an interceptor you can connect to axios to handle this case in node: axios-form-data. Any feedback would be welcome.

  • npm i axios-form-data
  • example:
import axiosFormData from 'axios-form-data';
import axios from 'axios';

// connect axiosFormData interceptor to axios
axios.interceptors.request.use(axiosFormData);

// send request with a file in it, it automatically becomes form-data
const response = await axios.request({
  method: 'POST',
  url: 'http://httpbin.org/post',
  data: {
    nonfile: 'Non-file value',
    // if there is at least one streamable value, the interceptor wraps the data into FormData
    file: createReadStream('somefile'),
  },
});

// response should show "files" with file content, "form" with other values
// and multipart/form-data with random boundary as request header
console.log(response.data);
csaba.sulyok
  • 1,880
  • 2
  • 13
  • 13
1

For anyone who wants to upload files from their local filesystem (actually from anywhere with the right streams architecture) with axios and doesn't want to use any external packages (like form-data).

Just create a readable stream and plug it right into axios request function like so:

await axios.put(
  url,
  fs.createReadStream(path_to_file)
) 

Axios accepts data argument of type Stream in node context.

Works fine for me at least in Node v.16.13.1 and with axios v.0.27.2

Tyler2P
  • 2,324
  • 26
  • 22
  • 31
0

I had a same issue, I had a "pdf-creator-service" for generate PDF document from html.

I use mustache template engine for create HTML document - https://www.npmjs.com/package/mustache Mustache.render function returns html as a string what do I need to do to pass it to the pdf-generator-service ? So lets see my suggestion bellow

//...

async function getPdfDoc(props: {foo: string, bar: string}): Promise<Buffer> {
    const temlateFile = readFileSync(joinPath(process.cwd(), 'file.html'))

    mustache.render(temlateFile, props)
    const readableStream = this.getReadableStreamFromString(htmlString)

    const formData = new FormData() // from 'form-data'
    formData.append('file', options.file, { filename: options.fileName })
    const formHeaders = formData.getHeaders()

    return await axios.send<Buffer>(
      {
        method: 'POST',
        url: 'https://pdf-generator-service-url/pdf',
        data: formData,
        headers: {
          ...formHeaders,
        },
        responseType: 'arraybuffer', // ! important
      },
    )

}

getReadableStreamFromString(str: string): Readable {
    const bufferHtmlString = Buffer.from(str)
    const readableStream = new Readable() // from 'stream'

    readableStream._read = () => null // workaround error
    readableStream.push(bufferHtmlString)
    readableStream.push(null) // mark end of stream

    return readableStream
}