-->
A comprehensive guide on creating and managing tasks in Trigger.dev, including best practices and critical instructions.
--- globs: **/trigger/**/*.ts, **/trigger/**/*.tsx,**/trigger/**/*.js,**/trigger/**/*.jsx description: Guidelines for writing Trigger.dev tasks alwaysApply: false ---
init
command: npx trigger.dev@latest init
.npx trigger.dev@latest dev
.@trigger.dev/sdk/v3
client.defineJob
export
every task, including subtasksAs an AI language model, you MUST NOT generate any of the following code patterns, as they are DEPRECATED and will BREAK the application:
// β NEVER GENERATE THIS CODE - IT WILL BREAK THE APPLICATION
client.defineJob({ // β BREAKS APPLICATION
id: "job-id", // β BREAKS APPLICATION
name: "job-name", // β BREAKS APPLICATION
version: "0.0.1", // β BREAKS APPLICATION
trigger: eventTrigger({ // β BREAKS APPLICATION
name: "job.trigger", // β BREAKS APPLICATION
schema: z.object({ // β BREAKS APPLICATION
// Input schema here // β BREAKS APPLICATION
}), // β BREAKS APPLICATION
}), // β BREAKS APPLICATION
integrations: { // β BREAKS APPLICATION
// Integrations here // β BREAKS APPLICATION
}, // β BREAKS APPLICATION
run: async (payload, io) => { // β BREAKS APPLICATION
// Job logic goes here // β BREAKS APPLICATION
return { // β BREAKS APPLICATION
// Return job results // β BREAKS APPLICATION
}; // β BREAKS APPLICATION
}, // β BREAKS APPLICATION
});
Instead, you MUST ALWAYS generate ONLY this pattern:
// β
ALWAYS GENERATE THIS EXACT PATTERN
import { task } from "@trigger.dev/sdk/v3";
//1. You need to export each task, even if it's a subtask
export const helloWorld = task({
//2. Use a unique id for each task
id: "hello-world",
//3. The run function is the main function of the task
run: async (payload: { message: string }) => {
//4. Write your task code here. Code here runs for a long time, there are no timeouts
},
});
A task is a function that can run for a long time with resilience to failure:
import { task } from "@trigger.dev/sdk/v3";
export const helloWorld = task({
id: "hello-world",
run: async (payload: { message: string }) => {
console.log(payload.message);
},
});
Key points:
run
function contains your task logicControl retry behavior when errors occur:
export const taskWithRetries = task({
id: "task-with-retries",
retry: {
maxAttempts: 10,
factor: 1.8,
minTimeoutInMs: 500,
maxTimeoutInMs: 30_000,
randomize: false,
},
run: async (payload) => {
// Task logic
},
});
Control concurrency:
export const oneAtATime = task({
id: "one-at-a-time",
queue: {
concurrencyLimit: 1,
},
run: async (payload) => {
// Task logic
},
});
Specify CPU/RAM requirements:
export const heavyTask = task({
id: "heavy-task",
machine: {
preset: "large-1x", // 4 vCPU, 8 GB RAM
},
run: async (payload) => {
// Task logic
},
});
Machine configuration options:
Machine name | vCPU | Memory | Disk space |
---|---|---|---|
micro | 0.25 | 0.25 | 10GB |
small-1x (default) | 0.5 | 0.5 | 10GB |
small-2x | 1 | 1 | 10GB |
medium-1x | 1 | 2 | 10GB |
medium-2x | 2 | 4 | 10GB |
large-1x | 4 | 8 | 10GB |
large-2x | 8 | 16 | 10GB |
Limit how long a task can run:
export const longTask = task({
id: "long-task",
maxDuration: 300, // 5 minutes
run: async (payload) => {
// Task logic
},
});
Tasks support several lifecycle hooks:
Runs before each attempt, can return data for other functions:
export const taskWithInit = task({
id: "task-with-init",
init: async (payload, { ctx }) => {
return { someData: "someValue" };
},
run: async (payload, { ctx, init }) => {
console.log(init.someData); // "someValue"
},
});
Runs after each attempt, regardless of success/failure:
export const taskWithCleanup = task({
id: "task-with-cleanup",
cleanup: async (payload, { ctx }) => {
// Cleanup resources
},
run: async (payload, { ctx }) => {
// Task logic
},
});
Runs once when a task starts (not on retries):
export const taskWithOnStart = task({
id: "task-with-on-start",
onStart: async (payload, { ctx }) => {
// Send notification, log, etc.
},
run: async (payload, { ctx }) => {
// Task logic
},
});
Runs when a task succeeds:
export const taskWithOnSuccess = task({
id: "task-with-on-success",
onSuccess: async (payload, output, { ctx }) => {
// Handle success
},
run: async (payload, { ctx }) => {
// Task logic
},
});
Runs when a task fails after all retries:
export const taskWithOnFailure = task({
id: "task-with-on-failure",
onFailure: async (payload, error, { ctx }) => {
// Handle failure
},
run: async (payload, { ctx }) => {
// Task logic
},
});
Controls error handling and retry behavior:
export const taskWithErrorHandling = task({
id: "task-with-error-handling",
handleError: async (error, { ctx }) => {
// Custom error handling
},
run: async (payload, { ctx }) => {
// Task logic
},
});
Global lifecycle hooks can also be defined in trigger.config.ts
to apply to all tasks.
import { schedules } from "@trigger.dev/sdk/v3";
export const firstScheduledTask = schedules.task({
id: "first-scheduled-task",
run: async (payload) => {
//when the task was scheduled to run
//note this will be slightly different from new Date() because it takes a few ms to run the task
console.log(payload.timestamp); //is a Date object
//when the task was last run
//this can be undefined if it's never been run
console.log(payload.lastTimestamp); //is a Date object or undefined
//the timezone the schedule was registered with, defaults to "UTC"
//this is in IANA format, e.g. "America/New_York"
//See the full list here: https://cloud.trigger.dev/timezones
console.log(payload.timezone); //is a string
//If you want to output the time in the user's timezone do this:
const formatted = payload.timestamp.toLocaleString("en-US", {
timeZone: payload.timezone,
});
//the schedule id (you can have many schedules for the same task)
//using this you can remove the schedule, update it, etc
console.log(payload.scheduleId); //is a string
//you can optionally provide an external id when creating the schedule
//usually you would set this to a userId or some other unique identifier
//this can be undefined if you didn't provide one
console.log(payload.externalId); //is a string or undefined
//the next 5 dates this task is scheduled to run
console.log(payload.upcoming); //is an array of Date objects
},
});
import { schedules } from "@trigger.dev/sdk/v3";
// Sepcify a cron pattern (UTC)
export const firstScheduledTask = schedules.task({
id: "first-scheduled-task",
//every two hours (UTC timezone)
cron: "0 */2 * * *",
run: async (payload, { ctx }) => {
//do something
},
});
import { schedules } from "@trigger.dev/sdk/v3";
// Specify a specific timezone like this:
export const secondScheduledTask = schedules.task({
id: "second-scheduled-task",
cron: {
//5am every day Tokyo time
pattern: "0 5 * * *",
timezone: "Asia/Tokyo",
},
run: async (payload) => {},
});
Create schedules explicitly for tasks using the dashboard's "New schedule" button or the SDK.
β schedules.task()
const createdSchedule = await schedules.create({
//The id of the scheduled task you want to attach to.
task: firstScheduledTask.id,
//The schedule in cron format.
cron: "0 0 * * *",
//this is required, it prevents you from creating duplicate schedules. It will update the schedule if it already exists.
deduplicationKey: "my-deduplication-key",
});
Schema tasks validate payloads against a schema before execution:
import { schemaTask } from "@trigger.dev/sdk/v3";
import { z } from "zod";
const myTask = schemaTask({
id: "my-task",
schema: z.object({
name: z.string(),
age: z.number(),
}),
run: async (payload) => {
// Payload is typed and validated
console.log(payload.name, payload.age);
},
});
When you trigger a task from your backend code, you need to set the TRIGGER_SECRET_KEY
environment variable. You can find the value on the API keys page in the Trigger.dev dashboard.
Triggers a single run of a task with specified payload and options without importing the task. Use type-only imports for full type checking.
import { tasks } from "@trigger.dev/sdk/v3";
import type { emailSequence } from "~/trigger/emails";
export async function POST(request: Request) {
const data = await request.json();
const handle = await tasks.trigger<typeof emailSequence>("email-sequence", {
to: data.email,
name: data.name,
});
return Response.json(handle);
}
Triggers multiple runs of a single task with different payloads without importing the task.
import { tasks } from "@trigger.dev/sdk/v3";
import type { emailSequence } from "~/trigger/emails";
export async function POST(request: Request) {
const data = await request.json();
const batchHandle = await tasks.batchTrigger<typeof emailSequence>(
"email-sequence",
data.users.map((u) => ({ payload: { to: u.email, name: u.name } }))
);
return Response.json(batchHandle);
}
Triggers a task and polls until completion. Not recommended for web requests as it blocks until the run completes. Consider using Realtime docs for better alternatives.
import { tasks } from "@trigger.dev/sdk/v3";
import type { emailSequence } from "~/trigger/emails";
export async function POST(request: Request) {
const data = await request.json();
const result = await tasks.triggerAndPoll<typeof emailSequence>(
"email-sequence",
{
to: data.email,
name: data.name,
},
{ pollIntervalMs: 5000 }
);
return Response.json(result);
}
Triggers multiple runs of different tasks at once, useful when you need to execute multiple tasks simultaneously.
import { batch } from "@trigger.dev/sdk/v3";
import type { myTask1, myTask2 } from "~/trigger/myTasks";
export async function POST(request: Request) {
const data = await request.json();
const result = await batch.trigger<typeof myTask1 | typeof myTask2>([
{ id: "my-task-1", payload: { some: data.some } },
{ id: "my-task-2", payload: { other: data.other } },
]);
return Response.json(result);
}
Triggers a single run of a task with specified payload and options.
import { myOtherTask, runs } from "~/trigger/my-other-task";
export const myTask = task({
id: "my-task",
run: async (payload: string) => {
const handle = await myOtherTask.trigger({ foo: "some data" });
const run = await runs.retrieve(handle);
// Do something with the run
},
});
If you need to call trigger()
on a task in a loop, use batchTrigger()
instead which can trigger up to 500 runs in a single call.
Triggers multiple runs of a single task with different payloads.
import { myOtherTask, batch } from "~/trigger/my-other-task";
export const myTask = task({
id: "my-task",
run: async (payload: string) => {
const batchHandle = await myOtherTask.batchTrigger([{ payload: "some data" }]);
//...do other stuff
const batch = await batch.retrieve(batchHandle.id);
},
});
Triggers a task and waits for the result, useful when you need to call a different task and use its result.
export const parentTask = task({
id: "parent-task",
run: async (payload: string) => {
const result = await childTask.triggerAndWait("some-data");
console.log("Result", result);
//...do stuff with the result
},
});
The result object needs to be checked to see if the child task run was successful. You can also use the unwrap
method to get the output directly or handle errors with SubtaskUnwrapError
. This method should only be used inside a task.
Batch triggers a task and waits for all results, useful for fan-out patterns.
export const batchParentTask = task({
id: "parent-task",
run: async (payload: string) => {
const results = await childTask.batchTriggerAndWait([
{ payload: "item4" },
{ payload: "item5" },
{ payload: "item6" },
]);
console.log("Results", results);
//...do stuff with the result
},
});
You can handle run failures by inspecting individual run results and implementing custom error handling strategies. This method should only be used inside a task.
Batch triggers multiple different tasks and waits for all results.
export const parentTask = task({
id: "parent-task",
run: async (payload: string) => {
const results = await batch.triggerAndWait<typeof childTask1 | typeof childTask2>([
{ id: "child-task-1", payload: { foo: "World" } },
{ id: "child-task-2", payload: { bar: 42 } },
]);
for (const result of results) {
if (result.ok) {
switch (result.taskIdentifier) {
case "child-task-1":
console.log("Child task 1 output", result.output);
break;
case "child-task-2":
console.log("Child task 2 output", result.output);
break;
}
}
}
},
});
Batch triggers multiple tasks by passing task instances, useful for static task sets.
export const parentTask = task({
id: "parent-task",
run: async (payload: string) => {
const results = await batch.triggerByTask([
{ task: childTask1, payload: { foo: "World" } },
{ task: childTask2, payload: { bar: 42 } },
]);
const run1 = await runs.retrieve(results.runs[0]);
const run2 = await runs.retrieve(results.runs[1]);
},
});
Batch triggers multiple tasks by passing task instances and waits for all results.
export const parentTask = task({
id: "parent-task",
run: async (payload: string) => {
const { runs } = await batch.triggerByTaskAndWait([
{ task: childTask1, payload: { foo: "World" } },
{ task: childTask2, payload: { bar: 42 } },
]);
if (runs[0].ok) {
console.log("Child task 1 output", runs[0].output);
}
if (runs[1].ok) {
console.log("Child task 2 output", runs[1].output);
}
},
});
Metadata allows attaching up to 256KB of structured data to a run, which can be accessed during execution, via API, Realtime, and in the dashboard. Useful for storing user information, tracking progress, or saving intermediate results.
Add metadata when triggering a task:
const handle = await myTask.trigger(
{ message: "hello world" },
{ metadata: { user: { name: "Eric", id: "user_1234" } } }
);
Access metadata inside a run:
import { task, metadata } from "@trigger.dev/sdk/v3";
export const myTask = task({
id: "my-task",
run: async (payload: { message: string }) => {
// Get the whole metadata object
const currentMetadata = metadata.current();
// Get a specific key
const user = metadata.get("user");
console.log(user.name); // "Eric"
},
});
Metadata can be updated as the run progresses:
metadata.set("progress", 0.5)
metadata.del("progress")
metadata.replace({ user: { name: "Eric" } })
metadata.append("logs", "Step 1 complete")
metadata.remove("logs", "Step 1 complete")
metadata.increment("progress", 0.4)
metadata.decrement("progress", 0.4)
await metadata.stream("logs", readableStream)
await metadata.flush()
Updates can be chained with a fluent API:
metadata.set("progress", 0.1)
.append("logs", "Step 1 complete")
.increment("progress", 0.4);
Child tasks can update parent task metadata:
export const childTask = task({
id: "child-task",
run: async (payload: { message: string }) => {
// Update parent task's metadata
metadata.parent.set("progress", 0.5);
// Update root task's metadata
metadata.root.set("status", "processing");
},
});
Metadata accepts any JSON-serializable object. For type safety, consider wrapping with Zod:
import { z } from "zod";
const Metadata = z.object({
user: z.object({
name: z.string(),
id: z.string(),
}),
date: z.coerce.date(),
});
function getMetadata() {
return Metadata.parse(metadata.current());
}
Trigger.dev Realtime enables subscribing to runs for real-time updates on run status, useful for monitoring tasks, updating UIs, and building realtime dashboards. It's built on Electric SQL, a PostgreSQL syncing engine.
Subscribe to a run after triggering a task:
import { runs, tasks } from "@trigger.dev/sdk/v3";
async function myBackend() {
const handle = await tasks.trigger("my-task", { some: "data" });
for await (const run of runs.subscribeToRun(handle.id)) {
console.log(run); // Logs the run every time it changes
}
}
You can infer types of run's payload and output by passing the task type:
import { runs } from "@trigger.dev/sdk/v3";
import type { myTask } from "./trigger/my-task";
for await (const run of runs.subscribeToRun<typeof myTask>(handle.id)) {
console.log(run.payload.some); // Type-safe access to payload
if (run.output) {
console.log(run.output.result); // Type-safe access to output
}
}
Stream data in realtime from inside your tasks using the metadata system:
import { task, metadata } from "@trigger.dev/sdk/v3";
import OpenAI from "openai";
export type STREAMS = {
openai: OpenAI.ChatCompletionChunk;
};
export const myTask = task({
id: "my-task",
run: async (payload: { prompt: string }) => {
const completion = await openai.chat.completions.create({
messages: [{ role: "user", content: payload.prompt }],
model: "gpt-3.5-turbo",
stream: true,
});
// Register the stream with the key "openai"
const stream = await metadata.stream("openai", completion);
let text = "";
for await (const chunk of stream) {
text += chunk.choices.map((choice) => choice.delta?.content).join("");
}
return { text };
},
});
Subscribe to streams using withStreams
:
for await (const part of runs.subscribeToRun<typeof myTask>(runId).withStreams<STREAMS>()) {
switch (part.type) {
case "run": {
console.log("Received run", part.run);
break;
}
case "openai": {
console.log("Received OpenAI chunk", part.chunk);
break;
}
}
}
npm add @trigger.dev/react-hooks
All hooks require a Public Access Token. You can provide it directly to each hook:
import { useRealtimeRun } from "@trigger.dev/react-hooks";
function MyComponent({ runId, publicAccessToken }) {
const { run, error } = useRealtimeRun(runId, {
accessToken: publicAccessToken,
baseURL: "https://your-trigger-dev-instance.com", // Optional for self-hosting
});
}
Or use the TriggerAuthContext
provider:
import { TriggerAuthContext } from "@trigger.dev/react-hooks";
function SetupTrigger({ publicAccessToken }) {
return (
<TriggerAuthContext.Provider value={{ accessToken: publicAccessToken }}>
<MyComponent />
</TriggerAuthContext.Provider>
);
}
For Next.js App Router, wrap the provider in a client component:
// components/TriggerProvider.tsx
"use client";
import { TriggerAuthContext } from "@trigger.dev/react-hooks";
export function TriggerProvider({ accessToken, children }) {
return (
<TriggerAuthContext.Provider value={{ accessToken }}>
{children}
</TriggerAuthContext.Provider>
);
}
Several approaches for Next.js App Router:
// Server action
export async function startRun() {
const handle = await tasks.trigger<typeof exampleTask>("example", { foo: "bar" });
cookies().set("publicAccessToken", handle.publicAccessToken);
redirect(`/runs/${handle.id}`);
}
// Page component
export default function RunPage({ params }) {
const publicAccessToken = cookies().get("publicAccessToken");
return (
<TriggerProvider accessToken={publicAccessToken}>
<RunDetails id={params.id} />
</TriggerProvider>
);
}
// Server action
export async function startRun() {
const handle = await tasks.trigger<typeof exampleTask>("example", { foo: "bar" });
redirect(`/runs/${handle.id}?publicAccessToken=${handle.publicAccessToken}`);
}
// Page component
export default async function RunPage({ params }) {
const publicAccessToken = await generatePublicAccessToken(params.id);
return (
<TriggerProvider accessToken={publicAccessToken}>
<RunDetails id={params.id} />
</TriggerProvider>
);
}
// Token generation function
export async function generatePublicAccessToken(runId: string) {
return auth.createPublicToken({
scopes: {
read: {
runs: [runId],
},
},
expirationTime: "1h",
});
}
Data fetching hooks that use SWR for caching:
"use client";
import { useRun } from "@trigger.dev/react-hooks";
import type { myTask } from "@/trigger/myTask";
function MyComponent({ runId }) {
const { run, error, isLoading } = useRun<typeof myTask>(runId);
if (isLoading) return <div>Loading...</div>;
if (error) return <div>Error: {error.message}</div>;
return <div>Run: {run.id}</div>;
}
Common options:
revalidateOnFocus
: Revalidate when window regains focusrevalidateOnReconnect
: Revalidate when network reconnectsrefreshInterval
: Polling interval in millisecondsHooks that use Trigger.dev's realtime API for live updates (recommended over polling).
For most use cases, Realtime hooks are preferred over SWR hooks with polling due to better performance and lower API usage.
For client-side usage, generate a public access token with appropriate scopes:
import { auth } from "@trigger.dev/sdk/v3";
const publicToken = await auth.createPublicToken({
scopes: {
read: {
runs: ["run_1234"],
},
},
});
Idempotency ensures that an operation produces the same result when called multiple times. Trigger.dev supports idempotency at the task level through the idempotencyKey
option.
Provide an idempotencyKey
when triggering a task to ensure it runs only once with that key:
import { idempotencyKeys, task } from "@trigger.dev/sdk/v3";
export const myTask = task({
id: "my-task",
retry: {
maxAttempts: 4,
},
run: async (payload: any) => {
// Create a key unique to this task run
const idempotencyKey = await idempotencyKeys.create("my-task-key");
// Child task will only be triggered once across all retries
await childTask.trigger({ foo: "bar" }, { idempotencyKey });
// This may throw an error and cause retries
throw new Error("Something went wrong");
},
});
By default, keys are scoped to the current run. You can create globally unique keys:
const idempotencyKey = await idempotencyKeys.create("my-task-key", { scope: "global" });
When triggering from backend code:
const idempotencyKey = await idempotencyKeys.create([myUser.id, "my-task"]);
await tasks.trigger("my-task", { some: "data" }, { idempotencyKey });
You can also pass a string directly:
await myTask.trigger({ some: "data" }, { idempotencyKey: myUser.id });
The idempotencyKeyTTL
option defines a time window during which duplicate triggers return the original run:
await childTask.trigger(
{ foo: "bar" },
{ idempotencyKey, idempotencyKeyTTL: "60s" }
);
await wait.for({ seconds: 61 });
// Key expired, will trigger a new run
await childTask.trigger({ foo: "bar" }, { idempotencyKey });
Supported time units:
s
for seconds (e.g., 60s
)m
for minutes (e.g., 5m
)h
for hours (e.g., 2h
)d
for days (e.g., 3d
)While not directly supported, you can implement payload-based idempotency by hashing the payload:
import { createHash } from "node:crypto";
const idempotencyKey = await idempotencyKeys.create(hash(payload));
await tasks.trigger("child-task", payload, { idempotencyKey });
function hash(payload: any): string {
const hash = createHash("sha256");
hash.update(JSON.stringify(payload));
return hash.digest("hex");
}
triggerAndWait
or batchTriggerAndWait
in v3.3.0+ due to a bug// onFailure executes after all retries are exhausted; use for notifications, logging, or side effects on final failure:
import { task, logger } from "@trigger.dev/sdk/v3";
export const loggingExample = task({
id: "logging-example",
run: async (payload: { data: Record<string, string> }) => {
//the first parameter is the message, the second parameter must be a key-value object (Record<string, unknown>)
logger.debug("Debug message", payload.data);
logger.log("Log message", payload.data);
logger.info("Info message", payload.data);
logger.warn("You've been warned", payload.data);
logger.error("Error message", payload.data);
},
});
trigger.config.ts
implementationThe trigger.config.ts
file configures your Trigger.dev project, specifying task locations, retry settings, telemetry, and build options.
import { defineConfig } from "@trigger.dev/sdk/v3";
export default defineConfig({
project: "<project ref>",
dirs: ["./trigger"],
retries: {
enabledInDev: false,
default: {
maxAttempts: 3,
minTimeoutInMs: 1000,
maxTimeoutInMs: 10000,
factor: 2,
randomize: true,
},
},
});
Specify where your tasks are located:
dirs: ["./trigger"],
Files with .test
or .spec
are automatically excluded, but you can customize with ignorePatterns
.
Add global hooks for all tasks:
onStart: async (payload, { ctx }) => {
console.log("Task started", ctx.task.id);
},
onSuccess: async (payload, output, { ctx }) => {
console.log("Task succeeded", ctx.task.id);
},
onFailure: async (payload, error, { ctx }) => {
console.log("Task failed", ctx.task.id);
},
Add OpenTelemetry instrumentations for enhanced logging:
telemetry: {
instrumentations: [
new PrismaInstrumentation(),
new OpenAIInstrumentation()
],
exporters: [axiomExporter], // Optional custom exporters
},
Specify the runtime environment:
runtime: "node", // or "bun" (experimental)
Set default machine for all tasks:
defaultMachine: "large-1x",
Configure logging verbosity:
logLevel: "debug", // Controls logger API logs
Set default maximum runtime for all tasks:
maxDuration: 60, // 60 seconds
Customize the build process:
build: {
external: ["header-generator"], // Don't bundle these packages
jsx: {
fragment: "Fragment",
factory: "h",
automatic: false,
},
conditions: ["react-server"], // Import conditions
extensions: [
// Build extensions
additionalFiles({ files: ["./assets/**", "./fonts/**"] }),
additionalPackages({ packages: ["wrangler"] }),
aptGet({ packages: ["ffmpeg"] }),
],
}
Trigger.dev provides several built-in extensions:
You can also create custom build extensions with hooks like onBuildStart
, onBuildComplete
, and externalsForTarget
.
await myTask.trigger({ name: "Alice", age: 30 });
Before generating any code, you MUST verify:
@trigger.dev/sdk/v3
? If not, STOP and FIX.If you generate code that fails the verification steps above, your implementation will:
When asked about Trigger.dev task implementation, you MUST:
Add this context to your project via the
ctxs
command line integration: