name: trigger-authoring-tasks description: > Covers writing backend Trigger.dev tasks with @trigger.dev/sdk: defining task() and schemaTask(), the run function and its ctx, retries, waits, queues and concurrency, idempotency keys, run metadata, logging, triggering other tasks (and the Result shape), scheduled/cron tasks, and the essentials of trigger.config.ts. Load this whenever you are authoring or editing code inside a /trigger directory, defining a task, or writing backend code that triggers tasks. Realtime/React hooks and AI chat are covered by separate skills. type: core library: trigger.dev
Authoring Trigger.dev Tasks
The full, version-pinned reference for authoring tasks ships inside your installed @trigger.dev/sdk. Read it before writing code — it always matches the SDK version in this project, so it never drifts:
- Skill:
node_modules/@trigger.dev/sdk/skills/trigger-authoring-tasks/SKILL.md— the complete guide (setup,schemaTask, retries, triggering + the Result shape, idempotency, waits, metadata, scheduled tasks, queues/concurrency,trigger.config.ts). - Docs: the full, version-pinned docs ship bundled at
node_modules/@trigger.dev/sdk/docs/; the skill above lists the exact pages it draws from in itssources:frontmatter. Grep for an API, e.g.grep -rl "schemaTask" node_modules/@trigger.dev/sdk/docs/.
If those paths don't exist, @trigger.dev/sdk isn't installed yet — install it first. In a non-hoisted layout, resolve the package with node -p "require.resolve('@trigger.dev/sdk/package.json')" and read skills/ + docs/ beside it.
Always import from @trigger.dev/sdk — never @trigger.dev/sdk/v3 (deprecated alias) or @trigger.dev/core.
Common mistakes
CRITICAL: Treating the wait result as the output.
triggerAndWaitandwait.forTokenreturn a Result object, not the raw output.- Wrong:
const out = await childTask.triggerAndWait(p); use(out.foo); - Correct:
const r = await childTask.triggerAndWait(p); if (r.ok) use(r.output.foo);(or.unwrap()).
- Wrong:
Wrapping
triggerAndWait/batchTriggerAndWait/waitinPromise.all.- Wrong:
await Promise.all([childTask.triggerAndWait(a), childTask.triggerAndWait(b)]); - Correct:
await childTask.batchTriggerAndWait([{ payload: a }, { payload: b }]);(or a sequential for-loop).
- Wrong:
Importing the task instance into backend code.
- Wrong:
import { emailSequence } from "~/trigger/emails";in a route handler. - Correct:
import type { emailSequence }plustasks.trigger<typeof emailSequence>("email-sequence", payload).
- Wrong:
Calling
metadata.set/getoutsiderun().- Wrong: setting metadata at module scope or in unrelated backend code (a no-op;
getreturnsundefined). - Correct: call inside
run()or a task lifecycle hook.
- Wrong: setting metadata at module scope or in unrelated backend code (a no-op;
Assuming child tasks inherit the parent's queue or metadata.
- Wrong: expecting a subtask to share the parent's
concurrencyLimitor see its metadata. - Correct: subtasks run on their own queue; pass metadata explicitly via
{ metadata: metadata.current() }, or push up withmetadata.parent.*.
- Wrong: expecting a subtask to share the parent's
Bundling native/WASM packages.
- Wrong: leaving
sharp,re2,sqlite3, or WASM packages in the default bundle. - Correct: add them to
build.externalintrigger.config.ts.
- Wrong: leaving
Relying on a raw string idempotency key being global.
- Wrong:
trigger(p, { idempotencyKey: "welcome-email" })expecting once-ever (true only in v4.3.0 and earlier). - Correct:
await idempotencyKeys.create("welcome-email", { scope: "global" }).
- Wrong:
References
Sibling skills: trigger-realtime-and-frontend (subscribe to runs, trigger from the frontend), trigger-authoring-chat-agent and trigger-chat-agent-advanced (AI chat agents).