PackagesSeed

@lunora/seed

Deterministic, schema-driven seeding: realistic fake data generated from defineSchema.

@lunora/seed fills a Lunora database with realistic fake data derived from your defineSchema. It reads every table, maps each column to a generator (field-name aware — a string column named email becomes an email, firstName a first name), inserts foreign-key parents before their children, and lets you override any value.

Generation is deterministic. The same seed value and schema always produce the same rows and ids, so fixtures are reproducible across runs and machines. It is built on a vendored, input-hashed generator (a rebuilt copycat) over @faker-js/faker.

Install

pnpm add @lunora/seed

@lunora/server and @lunora/values are peer dependencies (you already have them in a Lunora app).

Three ways to seed

You want to…Use
Generate rows in memory, write them yourselfseedPlan (from @lunora/seed)
Populate a lunoraTest harnessseed (from @lunora/seed/testing)
Author one table at a time, with typescreateSeedClient (from @lunora/seed, or _generated/seed)
Seed a running dev workerlunora seed (CLI)

seedPlan — the pure core

seedPlan(schema, options) returns one TablePlan per seeded table ({ table, rows }), ordered so a table's FK parents come first. It performs no I/O — every adapter (test harness, CLI, client) builds on it.

import { seedPlan } from "@lunora/seed";

import schema from "./lunora/schema";

const plan = seedPlan(schema, {
    counts: { posts: 30, users: 10 },
    overrides: {
        users: { email: (ctx) => `user${ctx.index}@example.com` },
    },
    seed: 1,
});

// plan === [{ rows: [...], table: "users" }, { rows: [...], table: "posts" }]
// Every post.authorId points at a generated user; each row carries an explicit `_id`.

SeedOptions

Prop

Type

An override function receives an OverrideContext: { field, index, row, store, table }. row is the row built so far; store is a read-only view of every table's rows generated this run (use it to copy a value from the parent row a foreign key points at). Returning undefined defers to the generator, so a partial row that omits a field still gets a generated value.

seed — populate a test harness

@lunora/seed/testing runs seedPlan and inserts every row into a lunoraTest harness in FK order, preserving the planned _ids. It returns the inserted ids keyed by table, for assertions.

import { seed } from "@lunora/seed/testing";
import { lunoraTest } from "@lunora/testing";

import schema from "./lunora/schema";

const harness = lunoraTest(schema);
const ids = await seed(harness, schema, { counts: { posts: 20, users: 5 } });

expect(ids.users).toHaveLength(5);

createSeedClient — typed, one table at a time

For a Snaplet-style DX, createSeedClient exposes each table as a method. Call it with a count, an inclusive range, explicit partial rows, or a count callback. Foreign keys connect to rows seeded earlier in the run, FK parents are seeded automatically, and state accumulates on $store / $ids (clear it with $reset()).

import { createSeedClient } from "@lunora/seed";

import schema from "./lunora/schema";

const seed = createSeedClient(schema, { seed: 1 });

const { users } = await seed.users(5);
const { posts } = await seed.posts((x) => x([10, 20])); // a deterministic count in [10, 20]

// Explicit partial rows — omitted columns are still generated:
await seed.users([{ name: "Alice" }, { email: "bob@example.com", name: "Bob" }]);

// Per-field overrides for one call:
await seed.posts(3, { overrides: { title: (ctx) => `Post ${ctx.index}` } });

seed.$ids.users; // every user id generated this run
seed.$reset(); // clear store/ids to drive a fresh run

Generated, project-bound client

When @lunora/seed is a declared dependency, codegen emits _generated/seed.ts — a createSeedClient with your schema and InsertModel already bound, so every table method is fully typed without passing a type argument.

import { createSeedClient } from "@/lunora/_generated/seed";

const seed = createSeedClient({ seed: 1 });
const { users } = await seed.users(5); // columns typed from InsertModel

Without codegen, pass the type yourself: createSeedClient<InsertModel>(schema).

SeedClientOptions

Prop

Type

Because every row carries an explicit _id, $ids is known whether or not you persist. A persist hook lets you write batches into the test harness or an admin import as they are generated.

lunora seed — seed a running worker

Generates from lunora/schema.ts and bulk-inserts through the worker's admin endpoint.

lunora seed                              # every table, default count (10)
lunora seed --table posts --count 100    # one table; FK parents seeded automatically
lunora seed --seed 42                    # reproducible run
lunora seed --dry-run                    # print the NDJSON, insert nothing
lunora seed --reset                      # wipe local .wrangler/state first (local dev only)

Other flags: --batch-size (rows per HTTP request, default 500), --url (worker URL, default http://localhost:8787), --prod (requires an explicit --url), --token / LUNORA_ADMIN_TOKEN (admin bearer; prefer the env var), --yes (skip the non-local confirmation prompt).

--reset clears local .wrangler/state only — it cannot be combined with --prod or a remote --url.

Limitations

  • .unique() columns are not enforced. Each value is hashed independently, so a unique column over a small value space (a bounded integer, a boolean, a short enum) can collide across rows. Strings such as emails and uuids are effectively unique in practice. Colliding rows are rejected by the import path rather than overwritten.
  • Seeding is deterministic by design. Re-running with the same seed regenerates identical _ids, which the import path skips as conflicts. Use a different seed for fresh rows, or --reset to wipe local state first.