Skip to main content
Sign In
Persistence

SQLite

Use embedded SQLite in Rivet Actors with raw SQL queries.

What is SQLite?

  • Database per actor: each actor instance has its own SQLite database, scoped to that actor.
  • High performance: Rivet Actors keep compute and storage together, so queries avoid network round trips to an external database.
  • Larger-than-memory storage: SQLite stores data on disk, so you can work with datasets that do not fit in actor memory.
  • Embedded relational database: use tables, indexes, and SQL queries directly inside actor logic.

SQLite features

  • Indexes: speed up lookups on frequently queried fields.
  • Search and filtering: use WHERE, LIKE, and ORDER BY instead of manual in-memory loops.
  • Relationships: use multiple tables and JOIN queries for connected data.
  • Constraints: use primary keys, unique constraints, and foreign keys for data integrity.
  • Transactions: apply multiple writes atomically when changes must stay consistent.

Raw SQL vs ORM (Drizzle)

Rivet supports both raw SQL and Drizzle for actor-local SQLite.

Use raw SQL when you want direct query control and minimal abstraction.

await c.db.execute("INSERT INTO todos (title) VALUES (?)", title);
const rows = await c.db.execute("SELECT id, title FROM todos ORDER BY id DESC");

Use Drizzle when you want typed schema and typed query APIs.

await c.vars.drizzle.insert(todos).values({ title });
const rows = await c.vars.drizzle.select().from(todos).orderBy(desc(todos.id));

You can mix both in the same actor.

For Drizzle setup, see SQLite + Drizzle.

Basic setup

Define db: db({ onMigrate }) on your actor, create your schema in onMigrate, and execute SQL with c.db.execute(...).

Queries

c.db.execute(...) returns an array of row objects for SELECT queries.

const rows = await c.db.execute(
  "SELECT id, title FROM todos WHERE title LIKE ?",
  `%${query}%`,
);

Parameterized queries

Use ? placeholders for dynamic values and pass parameters in order after the SQL string.

await c.db.execute("INSERT INTO todos (title) VALUES (?)", title);

Transactions

Use transactions when multiple writes must succeed or fail together.

  • Start with BEGIN.
  • End with COMMIT if all queries succeed.
  • On error, run ROLLBACK and rethrow.
  • A transaction is global for the entire shared c.db connection. Be careful with other interleaved queries.
await c.db.execute("BEGIN");

try {
  await c.db.execute("INSERT INTO todos (title) VALUES (?)", title);
  await c.db.execute(
    "INSERT INTO comments (todo_id, body) VALUES (last_insert_rowid(), ?)",
    body,
  );
  await c.db.execute("COMMIT");
} catch (error) {
  await c.db.execute("ROLLBACK");
  throw error;
}

Queues

It’s recommended to use queues for mutations and actions for read-only queries. This is the same code structure as the basic setup, but mutation writes are routed through queues.

Debugging

  • GET /inspector/summary includes isDatabaseEnabled so you can confirm SQLite is configured.
  • GET /inspector/traces helps inspect slow query paths and SQL-heavy actions.
  • Keep a small read-only action for quick query verification while debugging.
  • In non-dev mode, inspector endpoints require authorization.

Recommendations

  • Keep schema creation and migration steps in onMigrate.
  • Use ? placeholders for dynamic values.
  • Prefer queue-driven writes for ordered or background work.
  • Use transactions for related multi-step mutations when atomicity matters.

Read more