How to use async/await with Promises in Node
Posted
Updated
Europe’s developer-focused job platform
Let companies apply to you
Developer-focused, salary and tech stack upfront.
Just one profile, no job applications!
This article is based on Node v16.14.0.
Async/Await can be used to write asynchronous code in Node.js that reads like synchronous code and is available in Node since v.7.6 and up (officially rolled out with Node v8 and ECMAScript 2017). Any code that uses Promises can be converted to use async/await. For an overview of promises in Node.js have a look at the article: Promises in Node.js
💰 The Pragmatic Programmer: journey to mastery. 💰 One of the best books in software development, sold over 200,000 times.
The basic idea behind async/await
is to write async code that behaves similarly to synchronous code, it should execute in the order it was written.
Let's look at an example to compare code execution with Promises
and with async/await
:
Create or add a project folder.
mkdir node-async-await
Initialize project with npm init -y
to be able to install node packages.
cd node-async-await
npm init -y
Install node-fetch
to make fetch requests.
npm install node-fetch
Create an index.js
file.
touch index.js
Copy example code:
const fetch = require('node-fetch');
async function run() {
const status = await fetch(
'https://jsonplaceholder.typicode.com/todos/1',
)
.then(response => response.json())
.then(res => res);
console.log(status, 'Complete!');
}
run();
We need a second file for the comparison code for Promises.
touch index-promise.js
Copy the example code:
const fetch = require('node-fetch');
const data = fetch('https://jsonplaceholder.typicode.com/todos/1')
.then(response => response.json())
.then(res => res);
console.log(data, 'Complete!');
Now execute both files.
With async/await
code execution inside the run function is paused without blocking the Node.js process until the Promise chain resolves.
Without async/await
the console.log
statement will run before the Promise has resolved. The output in the command line will look like this Promise { <pending> } Complete!
.
If we want the same behaviour as the async/await
with just Promises
the console.log
has to be nested to ensure data was present when using it.
With async/await
we don't have to nest the code, so it is more readable.
Async functions will implicitly return a Promise, which resolves with the return value from the function.
Using await
will pause execution in the async function, until the awaited Promise resolves.
Using await doesn't block the main thread like a synchronous operation would, but pauses execution within the async function where it is used.
Once the Promise resolves, execution will begin again where it left off.
The await
keyword can only be used inside an async
function. It can't be used in the global context and must enclose it in an async function.
Error handling in async/await
comes with a big improvement.
When an error or an exception is thrown, it will reference the origin of the error (the function it originated from).
Hence, the context of the call is not lost, and the stack trace can be used to debug easier.
Error are handled by wrapping the await
calls in a try/catch
block, instead of using .catch
.
Let's add error handling to the example from above:
const fetch = require('node-fetch');
async function run() {
try {
const status = await fetch(
'https://jsonplaceholder.typicode.com/todos/1',
)
.then(response => response.json())
.then(res => res);
console.log(status, 'Complete!');
} catch (error) {
console.log(error);
}
}
run();
An async
function return a Promise
implicitly, so handling errors with a catch handler would also be possible.
Though handling errors with a try/catch
block is the common way of doing it, so the error handling is not delegated to the outside of async function.
const fetch = require('node-fetch');
async function run() {
const status = await fetch(
'https://jsonplaceholder.typicode.com/todos/1',
)
.then(response => response.json())
.then(res => res);
console.log(status, 'Complete!');
}
run.catch(error => console.log(error));
async/await
is the modern standard for asynchronous code in Node.jsThanks for reading and if you have any questions, use the comment function or send me a message @mariokandut.
If you want to know more about Node, have a look at these Node Tutorials.
Never miss an article.