Skip to content

Start typing to search the documentation.

Supabase

AI-generated, awaiting review View as Markdown @flue/postgres

Quickstart

Add durable, shared Postgres persistence to an existing Flue project with the Supabase blueprint. Run the following command in your terminal or coding agent of choice:

flue add database supabase

Overview

The Supabase blueprint installs @flue/postgres and pg, adds the matching @types/pg development dependency, and creates a transaction-safe db.ts in the project’s source-root. It uses the project’s existing secret convention and updates an existing environment example or environment documentation when one is present.

The primary generated adapter uses one checked-out pg client for every query in a transaction:

import { postgres } from '@flue/postgres';
import { Pool } from 'pg';

const pool = new Pool({ connectionString: process.env.SUPABASE_DATABASE_URL });

export default postgres({
  query: async (text, params) => (await pool.query(text, params)).rows,
  transaction: async (fn) => {
    const client = await pool.connect();
    // ...
  },
  close: () => pool.end(),
});

Flue discovers the adapter during a Node build, runs its migrations at server startup, and persists agent sessions, accepted submissions, workflow runs, and event state in Supabase so that state survives process restarts and can be shared across replicas. Application business data remains application-owned.

Configure

VariablePurpose
SUPABASE_DATABASE_URLRequired — Connection string from Supabase Dashboard > Connect.

The blueprint installs the existing @flue/postgres adapter with pg and writes a source-root db.ts. There is no Supabase-specific Flue package. Flue discovers the file at build time and wires it into the generated Node server.

This integration is Node.js only. The Cloudflare target uses Durable Object SQLite automatically and rejects db.ts at build time. See Database for persistence by target.

Copy a connection string from Supabase Dashboard > Connect and provide it at runtime as SUPABASE_DATABASE_URL:

DeploymentRecommended connection
Persistent, IPv6-capable Node serverDirect connection
Persistent, IPv4-only Node serverShared pooler in session mode

The provider-specific environment variable makes the secret’s source clear. If your project already uses another database variable convention, use it consistently in db.ts instead. Supply the value through your platform’s secret store and never commit it. For local development, flue dev --env <file> and flue run --env <file> load any .env-format file.

Transaction-mode pooling is not the default. It can preserve an explicit transaction performed on one checked-out client and does not inherently break BEGIN/COMMIT, but it does not support prepared statements or session state. If your deployment requires transaction mode, keep pg queries unnamed as in the example: do not pass a name in query configuration or otherwise enable named prepared statements, and do not depend on session state.

Use the transaction-safe runner

@flue/postgres accepts a runner so the application owns driver configuration, pooling, TLS, and credentials. The canonical pg runner checks out one client for the entire transaction callback:

import { postgres } from '@flue/postgres';
import { Pool } from 'pg';

const pool = new Pool({ connectionString: process.env.SUPABASE_DATABASE_URL });

export default postgres({
  query: async (text, params) => (await pool.query(text, params)).rows,
  transaction: async (fn) => {
    const client = await pool.connect();
    try {
      await client.query('BEGIN');
      const result = await fn({
        query: async (text, params) => (await client.query(text, params)).rows,
      });
      await client.query('COMMIT');
      return result;
    } catch (error) {
      await client.query('ROLLBACK');
      throw error;
    } finally {
      client.release();
    }
  },
  close: () => pool.end(),
});

Every query in the callback uses the checked-out client. Sending those queries through the pool could move work onto another connection and outside the transaction. @flue/postgres uses transaction-scoped pg_advisory_xact_lock, not session advisory locks, to serialize session updates; each lock is released with its transaction.

Migrations

The adapter’s migrate() hook runs automatically when the generated Node server starts. It creates Flue’s flue_* tables idempotently and stamps a schema version, so a fresh Supabase database is provisioned on first boot and an existing one is reused on restart. There is no separate migration command, and a database written by a newer Flue version refuses to start rather than risking incompatible writes.

What gets stored

A Flue database stores runtime state, not the application’s whole data model.

Stored by FlueNot stored by Flue
Agent session messages and compaction stateSandbox files and installed dependencies
Accepted direct prompts and dispatch(...) submissionsExternal API side effects
Durable turn journals, workflow-run records, persisted events, and run indexesApplication-owned business data
Recovery state for accepted workProvider credentials or secrets

See Durable Agents for recovery behavior and the Data Persistence API for the adapter contract.

Verify

Build the configured Node target and confirm db.ts is discovered. Against a non-production Supabase project, start the server and confirm the flue_* tables are created. Create agent or workflow state, restart the server, and confirm that state is reloaded. If you use the shared pooler, verify that its mode matches the deployment and that transaction mode, when explicitly chosen, uses no named prepared statements or session state.

When to choose Supabase

Choose Supabase when your persistent Node deployment needs durable shared Flue state and Supabase already provides its managed Postgres. Use the direct connection where IPv6 is available, or session-mode shared pooling for an IPv4-only server. For another managed or self-hosted Postgres deployment, use the general Postgres guide.