# Make HTTP Requests with Node.js
HTTP requests are fundamental to working with APIs or other web services. You can make HTTP requests to retrieve data from APIs, fetch HTML from websites, or do pretty much anything your web browser can do.
Below, we'll review how to make HTTP requests using Node.js code on Pipedream.
We'll use the axios and got HTTP clients in the examples below, but you can use any npm package you'd like on Pipedream, so feel free to experiment with other clients, too.
If you're new to HTTP, see our glossary of HTTP terms for a helpful introduction.
- Basic axios usage notes
- Send a GET request to fetch data
- Send a POST request to submit data
- Pass query string parameters to a GET request
- Send a request with HTTP headers
- Send a request with a secret or API key
- Send multiple HTTP requests using a loop
- Send a multipart/form-data request
- Download a file to the /tmp directory
- Upload a file from the /tmp directory
- Use an HTTP proxy to proxy requests through another host
- IP addresses for HTTP requests made from Pipedream workflows
- Forward an incoming HTTP request to another URL
- Stream a downloaded file directly to another URL
# Basic axios usage notes
 To use axios on Pipedream, you'll just need to require the axios npm package:
const axios = require("axios");
You make HTTP requests by passing a JavaScript object to axios that defines the parameters of the request. For example, you'll typically want to define the HTTP method and the URL you're sending data to:
{
  method: "GET",
  url: `https://swapi.co/api/films/`
}
axios returns a Promise, which is just a fancy way of saying that it makes the HTTP request in the background (asynchronously) while the rest of your code runs. On Pipedream, all asynchronous code must be run synchronously, which means you'll need to wait for the HTTP request to finish before moving on to the next step. You do this by adding an await in front of the call to axios.
Putting all of this together, here's how to make a basic HTTP request on Pipedream:
const resp = await axios({
  method: "GET",
  url: `https://swapi.co/api/films/`,
});
The response object resp contains a lot of information about the response: its data, headers, and more. Typically, you just care about the data, which you can access in the data property of the response:
const resp = await axios({
  method: "GET",
  url: `https://swapi.co/api/films/`,
});
// HTTP response data is in the data property
const data = resp.data;
Alternatively, you can access the data using object destructuring, which is equivalent to the above and preferred in modern JavaScript:
const { data } = resp;
# Send a GET request to fetch data
 Make a request to retrieve Star Wars films from the Star Wars API:
const axios = require("axios");
// Make an HTTP GET request using axios
const resp = await axios({
  method: "GET",
  url: `https://swapi.co/api/films/`,
});
// Retrieve just the data from the response
const { data } = resp;
Copy this workflow to run this example on Pipedream.
# Send a POST request to submit data
 POST sample JSON to JSONPlaceholder, a free mock API service:
const axios = require("axios");
// Make an HTTP POST request using axios
const resp = await axios({
  method: "POST",
  url: `https://jsonplaceholder.typicode.com/posts`
  data: {
    name: "Luke",
  }
});
// Retrieve just the data from the response
const { data } = resp;
When you make a POST request, you pass POST as the method, and include the data you'd like to send in the data object.
Copy this workflow to run this example on Pipedream.
# Pass query string parameters to a GET request
 Retrieve fake comment data on a specific post using JSONPlaceholder, a free mock API service. Here, you fetch data from the /comments resource, retrieving data for a specific post by query string parameter: /comments?postId=1.
const axios = require("axios");
// Make an HTTP GET request using axios
const resp = await axios({
  method: "GET",
  url: `https://jsonplaceholder.typicode.com/comments`,
  params: {
    postId: 1,
  },
});
// Retrieve just the data from the response
const { data } = resp;
You should pass query string parameters using the params object, like above. When you do, axios automatically URL-encodes the parameters for you, which you'd otherwise have to do manually.
Copy this workflow to run this code on Pipedream.
# Send a request with HTTP headers
You pass HTTP headers in the headers object of the axios request:
const axios = require("axios");
// Make an HTTP GET request using axios
const resp = await axios({
  method: "POST",
  url: `https://jsonplaceholder.typicode.com/posts`,
  headers: {
    "Content-Type": "application/json",
  },
  data: {
    name: "Luke",
  },
});
# Send a request with a secret or API key
Most APIs require you authenticate HTTP requests with an API key or other token. Please review the docs for your service to understand how they accept this data.
Here's an example showing an API key passed in an HTTP header:
const axios = require("axios");
// Make an HTTP GET request using axios
const resp = await axios({
  method: "POST",
  url: `https://jsonplaceholder.typicode.com/posts`,
  headers: {
    "Content-Type": "application/json",
    "X-API-Key": "123", // API KEY
  },
  data: {
    name: "Luke",
  },
});
Copy this workflow to run this code on Pipedream.
# Send multiple HTTP requests using a loop
There are many ways to make multiple HTTP requests. This code shows you a simple example that sends the numbers 1, 2, and 3 in the body of an HTTP POST request:
const axios = require("axios");
// We'll store each response and return them in this array
const responses = [];
for (const num of [1, 2, 3]) {
  const resp = await axios({
    method: "POST",
    url: params.url,
    data: {
      num, // Will send the current value of num in the loop
    },
  });
  responses.push(resp.data);
}
return responses;
This sends each HTTP request in sequence, one after another, and returns an array of response data returned from the URL to which you send the POST request.
Copy this workflow and fill in your destination URL to see how this works. This workflow iterates over the value of a Pipedream step export - data returned from a previous step. Since you often want to iterate over data returned from a Pipedream action or other code step, this is a common use case.
# Send a multipart/form-data request
 const axios = require("axios");
const FormData = require("form-data");
const formData = new FormData();
formData.append("name", "Luke Skywalker");
const headers = formData.getHeaders();
const config = {
  method: "POST",
  url: params.url,
  headers,
  data: formData,
};
return await axios(config);
Copy this workflow to run this example.
# Download a file to the /tmp directory
 This example shows you how to download a file to a file in the /tmp directory. This can be especially helpful for downloading large files: it streams the file to disk, minimizing the memory the workflow uses when downloading the file.
const fs = require("fs");
const { default: got } = await import("got");
const stream = require("stream");
const { promisify } = require("util");
// DOWNLOAD
const pipeline = promisify(stream.pipeline);
await pipeline(
  got.stream(params.downloadURL),
  fs.createWriteStream(params.filePath)
);
Copy this workflow to run this example.
# Upload a file from the /tmp directory
 This example shows you how to make a multipart/form-data request with a file as a form part. You can store and read any files from the /tmp directory.
This can be especially helpful for uploading large files: it streams the file from disk, minimizing the memory the workflow uses when uploading the file.
const axios = require("axios");
const fs = require("fs");
const FormData = require("form-data");
const formData = new FormData();
formData.append("file", fs.createReadStream(params.pathToYourFile));
const headers = formData.getHeaders();
const config = {
  method: "POST",
  url: params.url,
  headers,
  data: formData,
};
return await axios(config);
Copy this workflow to run this example.
# Use an HTTP proxy to proxy requests through another host
When you make HTTP requests to certain services, they might require you whitelist a set of IP addresses those requests come from. Often, this is to improve the security of the target service.
By default, HTTP requests made from Pipedream can come from a range of IP addresses. If you need to make requests from a single IP address, you can route traffic through an HTTP proxy:
const axios = require("axios");
let httpsProxyAgent = require("https-proxy-agent");
const agent = new httpsProxyAgent(`http://${user}:${pass}@${host}:${port}`);
const config = {
  method: "GET",
  url,
  httpsAgent: agent,
};
const resp = await axios.request(config);
If you don't have access to an HTTP proxy, and you are a paying Pipedream customer, reach out to our team. We operate a proxy that you can use for HTTP requests made through Pipedream.
Copy this workflow to run this code on Pipedream.
# IP addresses for HTTP requests made from Pipedream workflows
By default, HTTP requests made from Pipedream can come from a large range of IP addresses. If you need to restrict the IP addresses HTTP requests come from, you have two options:
- Use an HTTP proxy to proxy requests
- If you don't need to access the HTTP response data, you can use $send.http()to send requests from a limited set of IP addresses.
# Forward an incoming HTTP request to another URL
Often, you'll want to forward an incoming HTTP request from Pipedream to another service, with the same HTTP method, headers, body, etc. This workflow does just that.
Once you Copy the workflow, enter the URL where you'd like to forward an HTTP request in the forward_http_request step. Every HTTP request you send to the workflow's HTTP endpoint will get forwarded to that URL.
const config = {
  method: event.method || "POST",
  url: params.url,
};
const { query } = event;
if (Object.keys(query).length) {
  config.params = query;
}
// Headers, removing the original Host
const { headers } = event;
delete headers.host;
if (Object.keys(headers).length) {
  config.headers = headers;
}
if (event.body) config.data = event.body;
return await require("@pipedreamhq/platform").axios(this, config);
You can modify this workflow in any way you'd like. For example, if you wanted to forward only certain types of requests, you could add another Node.js code step before the forward_http_request step, ending the workflow early if the request doesn't contain a specific key in the HTTP payload:
if (!event.body.myImportantData) {
  $end("myImportantData not present in HTTP payload. Exiting");
}
# Stream a downloaded file directly to another URL
Sometimes you need to upload a downloaded file directly to another service, without processing the downloaded file. You could download the file and then upload it to the other URL, but these intermediate steps are unnecessary: you can just stream the download to the other service directly, without saving the file to disk.
This method is especially effective for large files that exceed the limits of the /tmp directory.
Copy this workflow or paste this code into a new Node.js code step:
const stream = require("stream");
const { promisify } = require("util");
const fs = require("fs");
const { default: got } = await import("got");
const pipeline = promisify(stream.pipeline);
await pipeline(
  got.stream(params.downloadURL),
  got.stream.post(params.uploadURL)
);
You'll be asked to provide the Download URL — the URL of the content you want to download — and the Upload URL — the place you want to upload the content to. got streams the content directly, downloading the file using a GET request and uploading it as a POST request.
If you need to modify this behavior, see the got Stream API.