Trigger.dev Realtime is a set of APIs that allow you to subscribe to runs and get real-time updates on the run status. This is useful for monitoring runs, updating UIs, and building realtime dashboards.

How it works

The Realtime API is built on top of Electric SQL, an open-source PostgreSQL syncing engine. The Trigger.dev API wraps Electric SQL and provides a simple API to subscribe to runs and get real-time updates.

Walkthrough

Usage

After you trigger a task, you can subscribe to the run using the runs.subscribeToRun function. This function returns an async iterator that you can use to get updates on the run status.

import { runs, tasks } from "@trigger.dev/sdk/v3";

// Somewhere in your backend code
async function myBackend() {
  const handle = await tasks.trigger("my-task", { some: "data" });

  for await (const run of runs.subscribeToRun(handle.id)) {
    // This will log the run every time it changes
    console.log(run);
  }
}

Every time the run changes, the async iterator will yield the updated run. You can use this to update your UI, log the run status, or take any other action.

Alternatively, you can subscribe to changes to any run that includes a specific tag (or tags) using the runs.subscribeToRunsWithTag function.

import { runs } from "@trigger.dev/sdk/v3";

// Somewhere in your backend code
for await (const run of runs.subscribeToRunsWithTag("user:1234")) {
  // This will log the run every time it changes, for all runs with the tag "user:1234"
  console.log(run);
}

If you’ve used batchTrigger to trigger multiple runs, you can also subscribe to changes to all the runs triggered in the batch using the runs.subscribeToBatch function.

import { runs } from "@trigger.dev/sdk/v3";

// Somewhere in your backend code
for await (const run of runs.subscribeToBatch("batch-id")) {
  // This will log the run every time it changes, for all runs in the batch with the ID "batch-id"
  console.log(run);
}

React hooks

We also provide a set of React hooks that make it easy to use the Realtime API in your React components. See the React hooks doc for more information.

Run changes

You will receive updates whenever a run changes for the following reasons:

Run object

The run object returned by the async iterator is NOT the same as the run object returned by the runs.retrieve function. This is because Electric SQL streams changes from a single PostgreSQL table, and the run object returned by runs.retrieve is a combination of multiple tables.

The run object returned by the async iterator has the following fields:

id
string
required

The run ID.

taskIdentifier
string
required

The task identifier.

payload
object
required

The input payload for the run.

output
object

The output result of the run.

createdAt
Date
required

Timestamp when the run was created.

updatedAt
Date
required

Timestamp when the run was last updated.

number
number
required

Sequential number assigned to the run.

status
RunStatus
required

Current status of the run.

durationMs
number
required

Duration of the run in milliseconds.

costInCents
number
required

Total cost of the run in cents.

baseCostInCents
number
required

Base cost of the run in cents before any additional charges.

tags
string[]
required

Array of tags associated with the run.

idempotencyKey
string

Key used to ensure idempotent execution.

expiredAt
Date

Timestamp when the run expired.

ttl
string

Time-to-live duration for the run.

finishedAt
Date

Timestamp when the run finished.

startedAt
Date

Timestamp when the run started.

delayedUntil
Date

Timestamp until which the run is delayed.

queuedAt
Date

Timestamp when the run was queued.

metadata
Record<string, DeserializedJson>

Additional metadata associated with the run.

error
SerializedError

Error information if the run failed.

isTest
boolean
required

Indicates whether this is a test run.

Type-safety

You can infer the types of the run’s payload and output by passing the type of the task to the subscribeToRun function. This will give you type-safe access to the run’s payload and output.

import { runs, tasks } from "@trigger.dev/sdk/v3";
import type { myTask } from "./trigger/my-task";

// Somewhere in your backend code
async function myBackend() {
  const handle = await tasks.trigger("my-task", { some: "data" });

  for await (const run of runs.subscribeToRun<typeof myTask>(handle.id)) {
    // This will log the run every time it changes
    console.log(run.payload.some);

    if (run.output) {
      // This will log the output if it exists
      console.log(run.output.some);
    }
  }
}

When using subscribeToRunsWithTag, you can pass a union of task types for all the possible tasks that can have the tag.

import { runs } from "@trigger.dev/sdk/v3";
import type { myTask, myOtherTask } from "./trigger/my-task";

// Somewhere in your backend code
for await (const run of runs.subscribeToRunsWithTag<typeof myTask | typeof myOtherTask>("my-tag")) {
  // You can narrow down the type based on the taskIdentifier
  switch (run.taskIdentifier) {
    case "my-task": {
      console.log("Run output:", run.output.foo); // This will be type-safe
      break;
    }
    case "my-other-task": {
      console.log("Run output:", run.output.bar); // This will be type-safe
      break;
    }
  }
}

Run metadata

The run metadata API gives you the ability to add or update custom metadata on a run, which will cause the run to be updated. This allows you to extend the realtime API with custom data attached to a run that can be used for various purposes. Some common use cases include:

  • Adding a link to a related resource
  • Adding a reference to a user or organization
  • Adding a custom status with progress information

See our run metadata docs for more on how to use this feature.

Using w/Realtime & React hooks

We suggest combining run metadata with the realtime API and our React hooks to bridge the gap between your trigger.dev tasks and your UI. This allows you to update your UI in real-time based on changes to the run metadata. As a simple example, you could add a custom status to a run with a progress value, and update your UI based on that progress.

We have a full demo app repo available here

Realtime streams

See our dedicated Realtime streams documentation for more information on how to use the Realtime streams API.

Limits

The Realtime API in the Trigger.dev Cloud limits the number of concurrent subscriptions, depending on your plan. If you exceed the limit, you will receive an error when trying to subscribe to a run. For more information, see our pricing page.

Known issues

There is currently a known issue where the realtime API does not work if subscribing to a run that has a large payload or large output and are stored in object store instead of the database. We are working on a fix for this issue: https://github.com/triggerdotdev/trigger.dev/issues/1451. As a workaround you’ll need to keep payloads and outputs below 128KB when using the realtime API.