Handling Errors & Retries

When functions fail, Inngest can retry them automatically. Whether Inngest retries your function is determined by how it fails. To learn how to handle function failures, read the reference guide here.

Types of failures

There are two ways that functions are determined to have failed:

export default inngest.createFunction(
  { id: "import-item-data" },
  { event: "import.requested" },
  async ({ event }) => {
    throw new Error("Failed to fetch item from ecommerce API");
  }
);
  • A function throws a non-retriable error. ❌ This will not be retried.
import { NonRetriableError } from "inngest";

export default inngest.createFunction(
  { id: "mark-store-imported" },
  { event: "import.completed" },
  async ({ event }) => {
    try {
      const result = await database.updateStore(
        { id: event.data.storeId },
        { imported: true }
      );
      return result.ok === true;
    } catch (err) {
      // Passing the original error via `cause` enables you to view the error in function logs
      throw new NonRetriableError("Store not found", { cause: err });
    }
  }
);

Attempt counter

Every time your function is executed, an attempt count is passed in as input, which is the current zero-based attempt number for this run.

The first attempt will be 0, the second 1, and so on. It is incremented every time the function throws an error and is retried.

inngest.createFunction(
  { id: "product-check" },
  { event: "product.check.requested" },
  async ({ attempt }) => {
    // `attempt` is the zero-based attempt number
  }
);

If retries are caused by a step failing which then succeeds before retries are exhausted, attempt will reset back to 0. See the Errors within steps section below.

Errors within steps

Steps are individually retried. That means Inngest will handle the type of error, just as above, but on the per-step basis.

import { NonRetriableError } from "inngest";

export default inngest.createFunction(
  { id: "import-store-items" },
  { event: "ecommerce/import.required" },
  async ({ event, tools }) => {
    const items = await step.run("get-items-from-api", async () => {
      // Third party APIs can often fail (e.g. because of a network or rate limit issue),
      // if this call fails, Inngest will attempt to retry this step
      return await ecommerceAPI.getItems(event.data.itemIds);
    });

    const store = await step.run("get-store-in-database", async () => {
      try {
        return await database.getStore(event.data.storeId);
      } catch (err) {
        // Store was not found - the step and function should not be retried
        throw new NonRetriableError("Could not find store in database", {
          cause: err,
        });
      }
    });

    // If either of the above steps
    await step.run("import-items-for-store", () => {
      /* ... */
    });
  }
);

To illustrate how this logic might work, let's use the above code and explain how the two types of errors will be handled across the life of an entire function run:

Scenario: The API has a major outage and all 3 attempts of the first step fail:

┌ Function: "Import store items" triggered by new "ecommerce/import.required" event
├─┬ Step: "Get items from API" started
│ ├ Attempt 1: ❌ Error thrown
│ ├ Attempt 2: ❌ Error thrown
│ └ Attempt 3: ❌ Error thrown
├── Step: "Get store in database" skipped
├── Step: "Import items for store" skipped
└ ❌ Function run failed

Scenario: The API responds perfectly, but somehow the user has since deleted the "store" from the database:

┌ Function: "Import store items" triggered by new "ecommerce/import.required" event
├─┬ Step: "Get items from API" started
│ └ Attempt 1: ✅ Successful
├─┬ Step: "Get store in database" started
│ └ Attempt 1: ❌ NonRetriableError thrown
├── Step: "Import items for store" skipped
└ ❌ Function run failed

Scenario: The API has a minor blip, but it's retried and everything else runs smoothly:

┌ Function: "Import store items" triggered by new "ecommerce/import.required" event
├─┬ Step: "Get items from API" started
│ ├ Attempt 1: ❌ Error thrown
│ ├ Attempt 2: ✅ Successful
├─┬ Step: "Get store in database" started
│ └ Attempt 1: ✅ Successful
├─┬ Step: "Import items for store" skipped
│ └ Attempt 1: ✅ Successful
└ ✅ Function run successful

Retry policies

By default, each function is retried 3 times using backoff with jitter.

  • Successful - No error thrown. This will not be retried.
  • Non-retriable error - A NonRetriableError was thrown (think: 404). This will not be retried.
  • Error - Any error was thrown indicating a potentially temporary failure (think: 500). This will be retried according to the retry policy (3 times, by default).

You can customize the number of retries directly from your function configuration:

import { Inngest } from "inngest";

const inngest = new Inngest({ id: "my-app" });

export default inngest.createFunction(
  {
    id: "handle-form",
    retries: 10, // Choose the number of retries you'd liie.
  },
  { event: "api/form.submitted" },
  async ({ event, step }) => {
    // ...
  }
);

Retries follow this backoff schedule, with 0-30 seconds of random jitter added.