mirror of
https://github.com/harivansh-afk/sandbox-agent.git
synced 2026-04-20 08:04:48 +00:00
Rename Factory to Foundry
This commit is contained in:
parent
0a8fda040b
commit
324de36577
256 changed files with 605 additions and 603 deletions
141
foundry/packages/backend/src/db/actor-sqlite.ts
Normal file
141
foundry/packages/backend/src/db/actor-sqlite.ts
Normal file
|
|
@ -0,0 +1,141 @@
|
|||
import { mkdirSync } from "node:fs";
|
||||
import { join } from "node:path";
|
||||
import { db as kvDrizzleDb } from "rivetkit/db/drizzle";
|
||||
|
||||
// Keep this file decoupled from RivetKit's internal type export paths.
|
||||
// RivetKit consumes database providers structurally.
|
||||
export interface RawAccess {
|
||||
execute: (query: string, ...args: unknown[]) => Promise<unknown[]>;
|
||||
close: () => Promise<void>;
|
||||
}
|
||||
|
||||
export interface DatabaseProviderContext {
|
||||
actorId: string;
|
||||
}
|
||||
|
||||
export type DatabaseProvider<DB> = {
|
||||
createClient: (ctx: DatabaseProviderContext) => Promise<DB>;
|
||||
onMigrate: (client: DB) => void | Promise<void>;
|
||||
onDestroy?: (client: DB) => void | Promise<void>;
|
||||
};
|
||||
|
||||
export interface ActorSqliteDbOptions<TSchema extends Record<string, unknown>> {
|
||||
actorName: string;
|
||||
schema?: TSchema;
|
||||
migrations?: {
|
||||
journal?: {
|
||||
entries?: ReadonlyArray<{
|
||||
idx: number;
|
||||
when: number;
|
||||
tag: string;
|
||||
}>;
|
||||
};
|
||||
migrations?: Readonly<Record<string, string>>;
|
||||
};
|
||||
migrationsFolderUrl: URL;
|
||||
/**
|
||||
* Override base directory for per-actor SQLite files.
|
||||
*
|
||||
* Default: `<cwd>/.sandbox-agent-foundry/backend/sqlite`
|
||||
*/
|
||||
baseDir?: string;
|
||||
}
|
||||
|
||||
export function actorSqliteDb<TSchema extends Record<string, unknown>>(
|
||||
options: ActorSqliteDbOptions<TSchema>
|
||||
): DatabaseProvider<any & RawAccess> {
|
||||
const isBunRuntime =
|
||||
typeof (globalThis as any).Bun !== "undefined" && typeof (process as any)?.versions?.bun === "string";
|
||||
|
||||
// Backend tests run in a Node-ish Vitest environment where `bun:sqlite` and
|
||||
// Bun's sqlite-backed Drizzle driver are not supported.
|
||||
//
|
||||
// Additionally, RivetKit's KV-backed SQLite implementation currently has stability
|
||||
// issues under Bun in this repo's setup (wa-sqlite runtime errors). Prefer Bun's
|
||||
// native SQLite driver in production backend execution.
|
||||
if (!isBunRuntime || process.env.VITEST || process.env.NODE_ENV === "test") {
|
||||
return kvDrizzleDb({
|
||||
schema: options.schema,
|
||||
migrations: options.migrations,
|
||||
}) as unknown as DatabaseProvider<any & RawAccess>;
|
||||
}
|
||||
|
||||
const baseDir = options.baseDir ?? join(process.cwd(), ".sandbox-agent-foundry", "backend", "sqlite");
|
||||
return {
|
||||
createClient: async (ctx) => {
|
||||
// Keep Bun-only module out of Vitest/Vite's static import graph.
|
||||
const { Database } = await import(/* @vite-ignore */ "bun:sqlite");
|
||||
const { drizzle } = await import("drizzle-orm/bun-sqlite");
|
||||
|
||||
const dir = join(baseDir, options.actorName);
|
||||
mkdirSync(dir, { recursive: true });
|
||||
|
||||
const dbPath = join(dir, `${ctx.actorId}.sqlite`);
|
||||
const sqlite = new Database(dbPath);
|
||||
sqlite.exec("PRAGMA journal_mode = WAL;");
|
||||
sqlite.exec("PRAGMA foreign_keys = ON;");
|
||||
|
||||
const client = drizzle({
|
||||
client: sqlite,
|
||||
schema: options.schema,
|
||||
});
|
||||
|
||||
return Object.assign(client, {
|
||||
execute: async (query: string, ...args: unknown[]) => {
|
||||
const stmt = sqlite.query(query);
|
||||
try {
|
||||
return stmt.all(args as never) as unknown[];
|
||||
} catch {
|
||||
stmt.run(args as never);
|
||||
return [];
|
||||
}
|
||||
},
|
||||
close: async () => {
|
||||
sqlite.close();
|
||||
},
|
||||
} satisfies RawAccess);
|
||||
},
|
||||
|
||||
onMigrate: async (client) => {
|
||||
await client.execute(
|
||||
"CREATE TABLE IF NOT EXISTS __drizzle_migrations (id INTEGER PRIMARY KEY AUTOINCREMENT, hash TEXT NOT NULL UNIQUE, created_at INTEGER NOT NULL)",
|
||||
);
|
||||
|
||||
const appliedRows = await client.execute("SELECT hash FROM __drizzle_migrations");
|
||||
const applied = new Set(
|
||||
appliedRows
|
||||
.map((row) => (row && typeof row === "object" && "hash" in row ? String((row as { hash: unknown }).hash) : null))
|
||||
.filter((value): value is string => value !== null),
|
||||
);
|
||||
|
||||
for (const entry of options.migrations?.journal?.entries ?? []) {
|
||||
if (applied.has(entry.tag)) {
|
||||
continue;
|
||||
}
|
||||
|
||||
const sql = options.migrations?.migrations?.[`m${String(entry.idx).padStart(4, "0")}`];
|
||||
if (!sql) {
|
||||
continue;
|
||||
}
|
||||
|
||||
const statements = sql
|
||||
.split("--> statement-breakpoint")
|
||||
.map((statement) => statement.trim())
|
||||
.filter((statement) => statement.length > 0);
|
||||
|
||||
for (const statement of statements) {
|
||||
await client.execute(statement);
|
||||
}
|
||||
await client.execute(
|
||||
"INSERT INTO __drizzle_migrations (hash, created_at) VALUES (?, ?)",
|
||||
entry.tag,
|
||||
entry.when ?? Date.now(),
|
||||
);
|
||||
}
|
||||
},
|
||||
|
||||
onDestroy: async (client) => {
|
||||
await client.close();
|
||||
},
|
||||
};
|
||||
}
|
||||
Loading…
Add table
Add a link
Reference in a new issue