Promises in JavaScript
CodePromises in JavaScript
February 2025

“Promises are only as strong as the person who gives them.”
Stephen Richards

What are promises?

In JavaScript, a Promise is an object that allow us to write some code which will be executed in the background by the event loop withount blocking the main tread. It allows us to handle asynchronous tasks in a more readable and manageable way compared to traditional callback functions. A promise has three different states: pending, fullfilled and rejected:

  • The promise is pending when it is running in the background
  • The fullfilled state rappresent a successful promise.
  • A rejected promise is a completed promise which was unsuccessful

How to create a promise

To create a promise we need to pass a callback which accepts two functions as arguments, the first one is the the one we call to resolve the promise and the second one is to reject it. As an example let's create a promise call bake which after 1 second will return 'Cooked!':

const bake = new Promise((resolve, reject) => {
    setTimeout(() => {
        resolve('Cooked!')
    }, 1000)
})

How to consume a promise

Now that we have defined our bake promise, how do we use it? Let's say we want to log the resolved value to be alered when the backing is over:

bake.then(console.log)

The then method allows us to specify a callback we want to execute once the promise is fullfilled. There are other two methods available which are the catch and the finally:

  • The catch method accept a callback which get executed if the promise is rejected
  • The finally method accep a callback which get executed after a promise has finished, regardless of its result.

We can concatenate the methods:

bake.then((result) => console.log(result))
    .catch((error) => console.log(error))
    .finally(() => console.log('Profised finished'))

Promises methods

Now what about we add one over so that we have two of them so we need to promises to notify us when each oven has finished:

const bakeFirstOven = new Promise((resolve, reject) => {
    setTimeout(() => {
        resolve('Cooked!')
    }, 1000)
})

const bakeSecondOven = new Promise((resolve, reject) => {
    setTimeout(() => {
        resolve('Cooked!')
    }, 2000)
})

Our first bake will take 1 sec while the one in the second oven will take us 2 seconds, we want to be notified only when both have finished, how can we do that?

We can use one method exposed by the Promise class which takes an array of promises and treat them as only one, firing the next when all promises have been fullfilled:

Promise.all([bakeFirstOven, bakeSecondOven])
    .then(([resultFirstOven, resultSecondOven]) => console.log(resultFirstOven, resultSecondOven))

There is an any method available which will call the next method with the first rejected promise value.

Async/Await sugar-syntax

To have a more flatted code we can remove the concatenation by using the async/await pattern combined with the try-catch-finally blocks:

async function backCakes () {
    try {
    await const results = Promise.all([bakeFirstOven, bakeSecondOven])
    console.log(...results)
    } catch (error) {
        console.log(error)
    }
}

This allows us to cover all the functionality provided by the methods concatatanation approach while keeping the code more readable.