| id | store-adapters |
|---|---|
| title | Store Adapters |
Store adapters implement WorkflowExecutionStore from
@tanstack/workflow-runtime.
The runtime store contract includes:
| Capability | Methods |
|---|---|
| Run lifecycle | createRun, loadRun, loadExecution, markRunPaused, markRunFinished, markRunErrored |
| Event log | appendEvents, readEvents, optional subscribeEvents |
| Leases | claimRun, heartbeatRunLease, releaseRunLease, claimStaleRuns |
| Timers | scheduleTimer, claimDueTimers |
| Signals | deliverSignal |
| Approvals | deliverApproval |
| Schedules | upsertSchedule, claimDueScheduleBuckets, markScheduleBucketStarted |
| Query | listRuns, getRunTimeline |
Adapter authors should run the shared contract tests from
@tanstack/workflow-runtime.
Install:
pnpm add @tanstack/workflow-store-drizzle-postgres drizzle-orm pgCreate the store:
import { drizzle } from 'drizzle-orm/node-postgres'
import { Pool } from 'pg'
import { createDrizzlePostgresWorkflowStore } from '@tanstack/workflow-store-drizzle-postgres'
const db = drizzle(new Pool({ connectionString: process.env.DATABASE_URL }))
const store = createDrizzlePostgresWorkflowStore({ db })Apply the package-owned migration during setup/deploy:
psql "$DATABASE_URL" -f node_modules/@tanstack/workflow-store-drizzle-postgres/migrations/0000_workflow_store.sqlUse it in the runtime:
const runtime = defineWorkflowRuntime({
store,
workflows,
})const store = createDrizzlePostgresWorkflowStore({
db,
schema: 'public',
tables: {
runs: 'workflow_runs',
},
})Options:
| Option | Purpose |
|---|---|
db |
Drizzle-compatible database object with execute and optional transaction. |
schema |
Optional Postgres schema name. |
tables |
Optional table name overrides. |
The Drizzle/Postgres package owns the durable Workflow schema. Apps should apply
the SQL artifact from the installed package instead of mirroring workflow_*
tables in application Drizzle schema files.
psql "$DATABASE_URL" -f node_modules/@tanstack/workflow-store-drizzle-postgres/migrations/0000_workflow_store.sqlProgrammatic setup can use the exported migration helpers:
import {
getDrizzlePostgresWorkflowStoreMigrationSql,
getDrizzlePostgresWorkflowStoreMigrations,
} from '@tanstack/workflow-store-drizzle-postgres'Schema changes are versioned with @tanstack/workflow-store-drizzle-postgres.
Apply new package migrations when upgrading the adapter. Runtime code assumes the
database schema is compatible with the installed store adapter version.
The initial migration creates workflow_schema_migrations and records applied
Workflow store migrations. Future migrations will be appended as new numbered SQL
files in the package. Apply them in order during deploy/setup before running the
new adapter version.
Maintainers changing the Drizzle/Postgres store schema should follow the
package-local SCHEMA_MIGRATIONS.md checklist.
Install:
pnpm add @tanstack/workflow-store-cloudflare-d1Create the store from a D1 binding:
import { createCloudflareD1WorkflowStore } from '@tanstack/workflow-store-cloudflare-d1'
const store = createCloudflareD1WorkflowStore({
db: env.WORKFLOW_DB,
})Apply the package-owned D1 migration during setup/deploy:
node_modules/@tanstack/workflow-store-cloudflare-d1/migrations/0000_workflow_store.sqlProgrammatic setup can use the exported migration helpers:
import {
getCloudflareD1WorkflowStoreMigrationSql,
getCloudflareD1WorkflowStoreMigrations,
} from '@tanstack/workflow-store-cloudflare-d1'D1 stores JSON payloads as text and uses SQLite integer timestamps. Its claim
operations use atomic conditional updates and leases rather than Postgres
FOR UPDATE SKIP LOCKED.
Creates the required tables and indexes if they do not exist.
await store.ensureSchema()Use this from tests, local demos, or an explicit admin bootstrap script, not from every request or sweep. Runtime and host adapters assume the schema exists. Production deploys should prefer the package-owned SQL migration artifact.
defaultDrizzlePostgresWorkflowStoreTables contains:
| Table key | Default table |
|---|---|
schemaMigrations |
workflow_schema_migrations |
runs |
workflow_runs |
runStates |
workflow_run_states |
eventLocks |
workflow_event_locks |
events |
workflow_events |
timers |
workflow_timers |
signalDeliveries |
workflow_signal_deliveries |
schedules |
workflow_schedules |
scheduleBuckets |
workflow_schedule_buckets |
The package exports Drizzle table definitions for apps that explicitly want typed read access for dashboards, diagnostics, or admin tooling:
import {
workflowEvents,
workflowRuns,
workflowSchedules,
workflowSchemaMigrations,
} from '@tanstack/workflow-store-drizzle-postgres'These exports are optional. Normal runtime use does not require adding Workflow tables to your app schema.
For tests and demos:
import { inMemoryWorkflowExecutionStore } from '@tanstack/workflow-runtime'
const store = inMemoryWorkflowExecutionStore()The in-memory store implements the same runtime contract, but it is not durable.
A production store adapter should satisfy:
- compare-and-swap append semantics for events
- idempotent run creation
- idempotent signal and approval delivery
- atomic claim and lease semantics
- due timer and due schedule indexes
- stale lease recovery
- timeline and list APIs
- migration strategy
- tests against the shared contract suite