Pyramid of Doom

2024, Nov 24    

Understanding the Pyramid of Doom and How to Avoid It

The “Pyramid of Doom” is a term used to describe deeply nested and hard-to-read code, particularly in JavaScript. While this pattern is less common today due to modern improvements in handling asynchronous operations, it’s still useful to understand the issue and how to refactor it effectively. If you’ve worked with callback-heavy JavaScript, you may have encountered this problem, which is similar to the excessive nesting often seen in poorly structured HTML (sometimes called “div soup”).

This post revisits the concept of the Pyramid of Doom, originally covered in my book Survive JavaScript - a Web Developer’s Guide, which I wrote a decade ago. Although the book is outdated, some core ideas remain relevant today.

What is the Pyramid of Doom?

The Pyramid of Doom arises from excessive callback nesting. This was especially common in early Node.js development when asynchronous operations relied heavily on callbacks, with each function handling an error-first parameter convention.

Example of the Pyramid of Doom

import db from "./db";

db.set("key1", "value1", (err) => {
  if (err) throw err;

  db.set("key2", "value2", (err) => {
    if (err) throw err;

    db.set("key3", "value3", (err) => {
      if (err) throw err;

      db.get("key1", (err, value) => {
        if (err) throw err;

        console.log(value + "bar");
      });
    });
  });
});

The deeply nested structure makes the code difficult to read and maintain. Thankfully, modern JavaScript provides better ways to handle asynchronous execution.

Solution 1: Using Promises

The introduction of Promises allowed developers to replace deeply nested callbacks with a more manageable structure. Promises provide a .then() method to chain operations and a .catch() method for handling errors.

Refactored Example with Promises

import db from "./db";

db.set("key1", "value1")
  .then(() => db.set("key2", "value2"))
  .then(() => db.set("key3", "value3"))
  .then(() => db.get("key1"))
  .then((value) => console.log(value + "bar"))
  .catch((err) => console.error(err));

This refactored version significantly reduces nesting, improving readability and maintainability.

Solution 2: Using async/await

Async/await further simplifies asynchronous code, making it look more like synchronous code while still being non-blocking. This syntax is built on top of Promises but eliminates chaining.

Refactored Example with async/await

import db from "./db";

async function main() {
  try {
    await db.set("key1", "value1");
    await db.set("key2", "value2");
    await db.set("key3", "value3");

    const value = await db.get("key1");
    console.log(value + "bar");
  } catch (err) {
    console.error(err);
  }
}

main();

This structure makes the control flow much easier to follow while handling errors using try/catch.

Solution 3: Parallel Execution with Promise.all

One limitation of the previous solutions is that they execute each operation sequentially, potentially slowing down execution. If the operations do not depend on each other, they can be executed in parallel using Promise.all.

Example with Promise.all

import db from "./db";

async function main() {
  try {
    await Promise.all([
      db.set("key1", "value1"),
      db.set("key2", "value2"),
      db.set("key3", "value3"),
    ]);

    const value = await db.get("key1");
    console.log(value + "bar");
  } catch (err) {
    console.error(err);
  }
}

main();

This approach speeds up execution but introduces a tradeoff—if any promise fails, Promise.all will reject immediately, potentially leaving the system in an inconsistent state. In scenarios where partial failures must be handled gracefully, Promise.allSettled may be a better option.

Handling Errors in Asynchronous Code

One of the most common mistakes when using async/await is neglecting proper error handling. Since await is just syntactic sugar over Promises, you can handle errors using:

  • .catch() for Promise chains.
  • try/catch for async functions.
  • Promise.allSettled when handling multiple async operations with independent error handling needs.

Conclusion

The Pyramid of Doom, once a prevalent issue in JavaScript development, has largely been mitigated by modern features like Promises and async/await. Understanding these techniques will not only improve your code readability but also help you write more efficient and maintainable applications.

While Promises and async/await cover most use cases, other models like reactive programming (e.g., RxJS) offer alternative solutions for handling complex asynchronous workflows. The key takeaway is to avoid unnecessary nesting, structure your code logically, and use the right tools for the job.