@sqlite-sync/core 0.2.0 → 0.3.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/{chunk-O4WYEB4H.js → chunk-RKBTBPNC.js} +3 -3
- package/dist/chunk-RKBTBPNC.js.map +1 -0
- package/dist/{crdt-sync-remote-source-Na9Uicu6.d.ts → crdt-sync-remote-source-Da77s4k0.d.ts} +1 -1
- package/dist/index.d.ts +4 -4
- package/dist/index.js +5 -5
- package/dist/index.js.map +1 -1
- package/dist/{reset-state-WqgHBSA4.d.ts → reset-state-0LGwO78x.d.ts} +1 -1
- package/dist/server.d.ts +4 -2
- package/dist/server.js +1 -1
- package/dist/server.js.map +1 -1
- package/dist/worker.d.ts +3 -3
- package/dist/worker.js +1 -1
- package/package.json +1 -1
- package/dist/chunk-O4WYEB4H.js.map +0 -1
|
@@ -991,7 +991,7 @@ function createCrdtApplyFunction({
|
|
|
991
991
|
if (!meta) {
|
|
992
992
|
throw new Error(`Item ${event.item_id} in dataset ${event.dataset} not found`);
|
|
993
993
|
}
|
|
994
|
-
const eventPayload = JSON.parse(event.payload);
|
|
994
|
+
const eventPayload = event.type === "item-deleted" ? { tombstone: 1 } : JSON.parse(event.payload);
|
|
995
995
|
const updatePayload = {};
|
|
996
996
|
let hasUpdates = false;
|
|
997
997
|
for (const [key, value] of Object.entries(eventPayload)) {
|
|
@@ -1028,7 +1028,7 @@ function createCrdtApplyFunction({
|
|
|
1028
1028
|
itemId: event.item_id,
|
|
1029
1029
|
dataset: event.dataset
|
|
1030
1030
|
});
|
|
1031
|
-
if (event.type !== "item-created" && event.type !== "item-updated") {
|
|
1031
|
+
if (event.type !== "item-created" && event.type !== "item-updated" && event.type !== "item-deleted") {
|
|
1032
1032
|
throw new Error(`Unknown event type: ${event.type}`);
|
|
1033
1033
|
}
|
|
1034
1034
|
if (meta) {
|
|
@@ -1796,4 +1796,4 @@ export {
|
|
|
1796
1796
|
createResetStateStore,
|
|
1797
1797
|
createReloadRequestHandler
|
|
1798
1798
|
};
|
|
1799
|
-
//# sourceMappingURL=chunk-
|
|
1799
|
+
//# sourceMappingURL=chunk-RKBTBPNC.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/dummy-kysely.ts","../src/hash.ts","../src/hlc.ts","../src/introspection.ts","../src/logger.ts","../src/sqlite-db-wrapper.ts","../src/sqlite-crdt/crdt-table-schema.ts","../src/migrations/migrator.ts","../src/sqlite-crdt/stored-value.ts","../src/sqlite-kv-store.ts","../src/migrations/system-schema.ts","../src/sqlite-crdt/apply-crdt-event.ts","../src/sqlite-crdt/crdt-storage.ts","../src/sqlite-crdt/event-consistency.ts","../src/sqlite-crdt/crdt-sync-producer.ts","../src/sqlite-crdt/crdt-sync-remote-source.ts","../src/worker-db/reset-state.ts","../src/worker-db/worker-common.ts"],"sourcesContent":["import { DummyDriver, Kysely, SqliteAdapter, SqliteIntrospector, SqliteQueryCompiler } from \"kysely\";\n\nexport const dummyKysely: Kysely<any> = new Kysely({\n dialect: {\n createAdapter: () => new SqliteAdapter(),\n createDriver: () => new DummyDriver(),\n createQueryCompiler: () => new SqliteQueryCompiler(),\n createIntrospector: (db) => new SqliteIntrospector(db),\n },\n});\n","import initXxhash, { type XXHashAPI } from \"xxhash-wasm\";\n\nlet loadPromise: Promise<void> | null = null;\nlet api: XXHashAPI | null = null;\n\nfunction ensureLoaded(): Promise<void> {\n if (!loadPromise) {\n loadPromise = initXxhash().then((hasher) => {\n api = hasher;\n });\n }\n return loadPromise;\n}\n\nfunction h64(input: string, seed = 0n): bigint {\n if (!api) {\n throw new Error(\"xxhash is not initialized; call xxhash.ensureLoaded() first\");\n }\n return api.h64(input, seed);\n}\n\nexport const xxhash = {\n ensureLoaded,\n h64,\n};\n","export interface HLC {\n timestamp: number;\n counter: number;\n nodeId: string;\n}\n\nconst MAX_COUNTER = 36 ** 5 - 1; // 60,466,175 — max value that fits in 5-char base36\nconst DEFAULT_MAX_DRIFT_MS = 6 * 60 * 60 * 1000; // 6 hours\n\nexport class HLCCounter {\n private timestamp: number;\n private counter: number;\n private nodeId: string;\n\n private readonly getTimestamp: () => number;\n private readonly maxDrift: number;\n\n constructor(nodeId: string, getTimestamp: () => number, maxDrift: number = DEFAULT_MAX_DRIFT_MS) {\n this.timestamp = getTimestamp();\n this.counter = 0;\n this.nodeId = nodeId;\n this.getTimestamp = getTimestamp;\n this.maxDrift = maxDrift;\n }\n\n getCurrentHLC(): HLC {\n return {\n timestamp: this.timestamp,\n counter: this.counter,\n nodeId: this.nodeId,\n };\n }\n\n getNextHLC(): HLC {\n const now = this.getTimestamp();\n\n if (now > this.timestamp) {\n this.timestamp = now;\n this.counter = 0;\n return this.getCurrentHLC();\n }\n\n this.counter++;\n if (this.counter > MAX_COUNTER) {\n throw new Error(`HLC counter overflow: exceeded max value ${MAX_COUNTER}`);\n }\n return this.getCurrentHLC();\n }\n\n mergeHLC(hlc: HLC) {\n const now = this.getTimestamp();\n if (hlc.timestamp - now > this.maxDrift) {\n console.warn(\n `HLC: ignoring far-future timestamp (remote=${hlc.timestamp}, local=${now}, drift=${hlc.timestamp - now}ms)`,\n );\n return;\n }\n\n if (this.timestamp === hlc.timestamp) {\n this.counter = Math.max(this.counter, hlc.counter) + 1;\n } else if (this.timestamp > hlc.timestamp) {\n this.counter++;\n } else {\n this.timestamp = hlc.timestamp;\n this.counter = hlc.counter + 1;\n }\n if (this.counter > MAX_COUNTER) {\n throw new Error(`HLC counter overflow: exceeded max value ${MAX_COUNTER}`);\n }\n }\n}\n\nexport function serializeHLC(hlc: HLC) {\n return `${hlc.timestamp.toString().padStart(15, \"0\")}:${hlc.counter.toString(36).padStart(5, \"0\")}:${hlc.nodeId}`;\n}\n\nexport function deserializeHLC(serialized: string): HLC {\n const parts = serialized.split(\":\");\n if (parts.length < 3) {\n throw new Error(`Invalid HLC format: expected at least 3 colon-separated segments, got ${parts.length}`);\n }\n\n const timestamp = parseInt(parts[0], 10);\n const counter = parseInt(parts[1], 36);\n\n if (Number.isNaN(timestamp) || Number.isNaN(counter)) {\n throw new Error(`Invalid HLC values: timestamp=${parts[0]}, counter=${parts[1]}`);\n }\n\n return {\n timestamp,\n counter,\n nodeId: parts.slice(2).join(\":\"),\n };\n}\n\nexport function compareHLC(one: HLC, two: HLC) {\n if (one.timestamp === two.timestamp) {\n if (one.counter === two.counter) {\n if (one.nodeId === two.nodeId) {\n return 0;\n }\n return one.nodeId < two.nodeId ? -1 : 1;\n }\n return one.counter - two.counter;\n }\n return one.timestamp - two.timestamp;\n}\n","import type { Kysely, QueryCreator } from \"kysely\";\nimport { sql } from \"kysely\";\nimport type { SQLiteDbWrapper } from \"./sqlite-db-wrapper\";\n\ninterface SqliteSystemDatabase {\n // https://www.sqlite.org/schematab.html#alternative_names\n sqlite_master: SQliteMasterTable;\n}\n\n// https://www.sqlite.org/schematab.html#interpretation_of_the_schema_table\ninterface SQliteMasterTable {\n name: string;\n rootpage: number | null;\n sql: string;\n tbl_name: string;\n type: \"index\" | \"table\" | \"trigger\" | \"view\";\n}\n\n// https://www.sqlite.org/pragma.html#pragma_table_info\ninterface PragmaTableInfo {\n cid: number;\n dflt_value: unknown;\n name: string;\n notnull: 0 | 1;\n pk: number;\n type: string;\n}\n\nfunction tablesQuery(qb: QueryCreator<SqliteSystemDatabase> | Kysely<SqliteSystemDatabase>) {\n return qb\n .selectFrom(\"sqlite_master\")\n .where(\"type\", \"in\", [\"table\", \"view\"])\n .where(\"name\", \"not like\", \"sqlite_%\")\n .select([\"name\", \"sql\", \"type\"])\n .orderBy(\"name\");\n}\n\nexport type TableMetadata = {\n name: string;\n isView: boolean;\n columns: ColumnMetadata[];\n};\n\nexport type DatabaseIntrospection = Record<string, TableMetadata>;\n\ntype ColumnMetadata = {\n name: string;\n dataType: string;\n isNullable: boolean;\n isAutoIncrementing: boolean;\n hasDefaultValue: boolean;\n comment: undefined;\n};\n\nexport function introspectDb<BaseDatabase>(_db: SQLiteDbWrapper<BaseDatabase>): DatabaseIntrospection {\n const db = _db as unknown as SQLiteDbWrapper<SqliteSystemDatabase>;\n const tables = db.executeKysely((db) => tablesQuery(db as unknown as Kysely<SqliteSystemDatabase>), {\n loggerLevel: \"system\",\n }).rows;\n\n const tablesMetadata = db.executeKysely(\n (db) =>\n db\n .with(\"table_list\", (qb) => tablesQuery(qb as unknown as Kysely<SqliteSystemDatabase>))\n .selectFrom([\"table_list as tl\", sql<PragmaTableInfo>`pragma_table_info(tl.name)`.as(\"p\")])\n .select([\"tl.name as table\", \"p.cid\", \"p.name\", \"p.type\", \"p.notnull\", \"p.dflt_value\", \"p.pk\"])\n .orderBy(\"tl.name\")\n .orderBy(\"p.cid\"),\n { loggerLevel: \"system\" },\n ).rows;\n\n const columnsByTable: Record<string, typeof tablesMetadata> = {};\n for (const row of tablesMetadata) {\n columnsByTable[row.table] ??= [];\n columnsByTable[row.table].push(row);\n }\n\n return Object.fromEntries(\n tables.map(({ name, sql, type }) => {\n // Try to find the name of the column that has `autoincrement`\n let autoIncrementCol = sql\n ?.split(/[(),]/)\n ?.find((it) => it.toLowerCase().includes(\"autoincrement\"))\n ?.trimStart()\n ?.split(/\\s+/)?.[0]\n ?.replace(/[\"`]/g, \"\");\n\n const columns = columnsByTable[name] ?? [];\n\n // Otherwise, check for an INTEGER PRIMARY KEY\n // https://www.sqlite.org/autoinc.html\n if (!autoIncrementCol) {\n const pkCols = columns.filter((r) => r.pk > 0);\n if (pkCols.length === 1 && pkCols[0].type.toLowerCase() === \"integer\") {\n autoIncrementCol = pkCols[0].name;\n }\n }\n\n return [\n name,\n {\n name: name,\n isView: type === \"view\",\n columns: columns.map((col) => ({\n name: col.name,\n dataType: col.type,\n isNullable: !col.notnull,\n isAutoIncrementing: col.name === autoIncrementCol,\n hasDefaultValue: col.dflt_value != null,\n comment: undefined,\n })),\n },\n ];\n }),\n );\n}\n","export type LogLevel = \"info\" | \"warning\" | \"error\" | \"trace\" | \"system\";\n\nexport type Logger = (type: string, message: string, level?: LogLevel) => void;\n\nexport const startPerformanceLogger = (logger: Logger) => {\n let startTime = performance.now();\n\n return {\n restart: () => {\n startTime = performance.now();\n },\n logEnd: (type: string, message: string, level: LogLevel = \"info\") => {\n const elapsed = performance.now() - startTime;\n\n logger(type, `${elapsed.toFixed(2)}ms - ${message}`, level);\n },\n };\n};\n","import type {\n BindableValue,\n FunctionOptions,\n Database as SQLiteDatabase,\n Sqlite3Static,\n SqlValue,\n} from \"@sqlite.org/sqlite-wasm\";\nimport type { Compilable, CompiledQuery, Kysely } from \"kysely\";\nimport { dummyKysely } from \"./dummy-kysely\";\nimport { type DatabaseIntrospection, introspectDb } from \"./introspection\";\nimport { type Logger, startPerformanceLogger } from \"./logger\";\n\nexport type ExecuteParams = {\n sql: string;\n parameters: readonly unknown[];\n};\n\nexport type ExecuteResult<T> = {\n rows: T[];\n};\n\nexport type PreparedStatement<TParams extends unknown[], TResult> = {\n execute: (parameters: TParams) => TResult[];\n finalize: () => void;\n isFinalized: boolean;\n};\n\ntype ScalarFunctionOptions<TArgs extends readonly SqlValue[], TResult extends SqlValue | undefined> = {\n name: string;\n callback: (...args: TArgs) => TResult;\n} & Pick<FunctionOptions, \"deterministic\" | \"directOnly\" | \"innocuous\">;\n\ntype SqliteWrapperOptions = {\n logger?: Logger;\n loggerPrefix?: string;\n sqlite3: Sqlite3Static;\n db: () => SQLiteDatabase;\n};\n\nexport type SQLiteTransactionWrapper<TDatabase = unknown> = Pick<\n SQLiteDbWrapper<TDatabase>,\n \"sql\" | \"execute\" | \"executePrepared\" | \"executePreparedRaw\" | \"executeKysely\"\n>;\n\nexport type InternalSQLiteTransactionWrapper<TDatabase = unknown> = Pick<\n SQLiteDbWrapper<TDatabase>,\n \"executePrepared\" | \"executePreparedRaw\"\n>;\n\nexport type InternalSQLiteWrapper<TDatabase = unknown> = InternalSQLiteTransactionWrapper<TDatabase> & {\n executeTransaction: (callback: (db: InternalSQLiteTransactionWrapper<TDatabase>) => void) => void;\n};\n\ntype QueryMetaOpts = {\n loggerLevel?: \"info\" | \"system\";\n};\n\nexport class SQLiteDbWrapper<TDatabase = unknown> {\n private db: SQLiteDatabase | null = null;\n private sqlite3: Sqlite3Static;\n private logger?: Logger;\n private loggerPrefix?: string;\n\n private loadedDbSchema: DatabaseIntrospection | null = null;\n\n private readonly dataPointers = [] as number[];\n\n private preparedStatements: PreparedStatement<SqlValue[], unknown>[] = [];\n private preparedStatementsMap = new Map<string, TypedStatement<Record<string, unknown>, unknown>>();\n private preparedRawStatementsMap = new Map<string, PreparedStatement<SqlValue[], unknown>>();\n\n constructor(opts: SqliteWrapperOptions) {\n this.db = opts.db();\n this.sqlite3 = opts.sqlite3;\n this.logger = opts.logger;\n this.loggerPrefix = opts.loggerPrefix;\n }\n\n get ensureDb() {\n if (!this.db) {\n throw new Error(\"Database is already closed\");\n }\n return this.db;\n }\n\n get dbSchema() {\n if (!this.loadedDbSchema) {\n this.loadedDbSchema = introspectDb(this);\n }\n return this.loadedDbSchema;\n }\n\n execute<T = unknown>(opts: ExecuteParams | string | CompiledQuery<T>, meta?: QueryMetaOpts): ExecuteResult<T> {\n const sql = typeof opts === \"string\" ? opts : opts.sql;\n const bind = typeof opts === \"string\" ? undefined : opts.parameters;\n\n const perf = this.logger ? startPerformanceLogger(this.logger) : undefined;\n const rows = this.ensureDb.exec({\n sql,\n bind: bind as BindableValue[],\n returnValue: \"resultRows\",\n rowMode: \"object\",\n });\n perf?.logEnd(`${this.loggerPrefix ?? \"\"}:query`, sql, meta?.loggerLevel);\n\n return { rows: rows as T[] };\n }\n\n executeTransaction<T>(callback: (db: SQLiteTransactionWrapper<TDatabase>) => T): T {\n const transaction = this.beginTransaction();\n try {\n const result = callback(this);\n transaction.commit();\n return result;\n } catch (error) {\n transaction.rollback();\n throw error;\n }\n }\n\n isInTransaction() {\n // TODO: Awaiting upstream fix: https://github.com/sqlite/sqlite-wasm/pull/143\n return (this.sqlite3.capi as any).sqlite3_get_autocommit(this.ensureDb) === 0;\n }\n\n beginTransaction() {\n this.executePreparedRaw({\n key: \"$begin-transaction\",\n sql: \"begin\",\n meta: {\n loggerLevel: \"system\",\n },\n });\n\n return {\n commit: () => {\n this.executePreparedRaw({\n key: \"$commit-transaction\",\n sql: \"commit\",\n meta: {\n loggerLevel: \"system\",\n },\n });\n },\n rollback: () => {\n this.executePreparedRaw({\n key: \"$rollback-transaction\",\n sql: \"rollback\",\n meta: {\n loggerLevel: \"system\",\n },\n });\n },\n };\n }\n\n prepare<TParams extends unknown[], TResult>(sql: string, opts?: QueryMetaOpts) {\n const perf = this.logger ? startPerformanceLogger(this.logger) : undefined;\n const stmt = this.ensureDb.prepare(sql);\n perf?.logEnd(`${this.loggerPrefix ?? \"\"}:prepare`, sql, opts?.loggerLevel);\n\n let isFinalized = false;\n\n const execute = (params: TParams) => {\n if (isFinalized) {\n throw new Error(\"Statement is finalized\");\n }\n\n const perf = this.logger ? startPerformanceLogger(this.logger) : undefined;\n try {\n if (params.length > 0) {\n stmt.bind(params as SqlValue[]);\n }\n\n const results = [] as TResult[];\n while (stmt.step()) {\n results.push(stmt.get({}) as TResult);\n }\n\n return results;\n } finally {\n stmt.reset(true);\n perf?.logEnd(`${this.loggerPrefix ?? \"\"}:prepare-execute`, sql, opts?.loggerLevel);\n }\n };\n\n const finalize = () => {\n isFinalized = true;\n stmt.finalize();\n };\n\n const preparedStatement: PreparedStatement<TParams, TResult> = {\n execute,\n finalize,\n get isFinalized() {\n return isFinalized;\n },\n };\n\n this.preparedStatements.push(preparedStatement as PreparedStatement<SqlValue[], unknown>);\n\n return preparedStatement;\n }\n\n prepareKysely<TParams extends Record<string, unknown>>(opts?: QueryMetaOpts) {\n return <TQuery extends Compilable<TResult>, TResult = QueryBuilderOutput<TQuery>>(\n factory: KyselyStatementFactory<TParams, TDatabase, TQuery, TResult>,\n ): TypedStatement<TParams, TResult> => {\n const query = factory(dummyKysely, (key) => key as any).compile();\n const statement = this.prepare<SqlValue[], TResult>(query.sql, opts);\n\n return {\n execute: (parameters) => {\n const params = query.parameters.map((param) => parameters[param as keyof TParams]);\n const result = statement.execute(params as SqlValue[]);\n return result;\n },\n };\n };\n }\n\n executeKysely<TQuery extends Compilable<TResult>, TResult = QueryBuilderOutput<TQuery>>(\n factory: KyselyQueryFactory<TDatabase, TQuery, TResult>,\n meta?: QueryMetaOpts,\n ) {\n const query = factory(dummyKysely).compile();\n return this.execute(query, meta);\n }\n\n executePrepared<\n TParams extends Record<string, unknown>,\n TQuery extends Compilable<TResult>,\n TResult = QueryBuilderOutput<TQuery>,\n >(\n key: string,\n params: TParams,\n factory: KyselyStatementFactory<TParams, TDatabase, TQuery, TResult>,\n meta?: QueryMetaOpts,\n ) {\n let statement = this.preparedStatementsMap.get(key) as TypedStatement<TParams, TResult> | undefined;\n if (!statement) {\n statement = this.prepareKysely<TParams>(meta)(factory);\n this.preparedStatementsMap.set(key, statement as TypedStatement<Record<string, unknown>, unknown>);\n }\n\n return statement.execute(params);\n }\n\n executePreparedRaw<TParams extends unknown[], TResult>({\n key,\n sql,\n params,\n meta,\n }: {\n key: string;\n sql: string;\n params?: TParams;\n meta?: QueryMetaOpts;\n }) {\n let statement = this.preparedRawStatementsMap.get(key) as PreparedStatement<TParams, TResult> | undefined;\n if (!statement) {\n statement = this.prepare(sql, meta);\n this.preparedRawStatementsMap.set(key, statement as PreparedStatement<any[], unknown>);\n }\n\n return statement.execute((params ?? []) as TParams);\n }\n\n sql<T = unknown>(templateOrString: TemplateStringsArray | string, ...parameters: unknown[]) {\n if (typeof templateOrString === \"string\") {\n return this.execute<T>({\n sql: templateOrString,\n parameters,\n });\n }\n return this.execute<T>({\n sql: templateOrString.join(\"?\"),\n parameters,\n });\n }\n\n createScalarFunction<TArgs extends SqlValue[], TResult extends SqlValue | undefined>({\n name,\n callback,\n deterministic,\n directOnly,\n innocuous,\n }: ScalarFunctionOptions<TArgs, TResult>) {\n return this.ensureDb.createFunction({\n name,\n xFunc: (_, ...args) => {\n const result = callback(...(args as TArgs)) as SqlValue;\n return result;\n },\n arity: callback.length,\n deterministic,\n directOnly,\n innocuous,\n });\n }\n\n useSnapshot(snapshot: Uint8Array<ArrayBufferLike>) {\n const perf = this.logger ? startPerformanceLogger(this.logger) : undefined;\n const dataPointer = this.sqlite3.wasm.allocFromTypedArray(snapshot);\n this.dataPointers.push(dataPointer);\n\n const resultCode = this.sqlite3.capi.sqlite3_deserialize(\n this.ensureDb,\n \"main\",\n dataPointer,\n snapshot.byteLength,\n snapshot.byteLength,\n this.sqlite3.capi.SQLITE_DESERIALIZE_FREEONCLOSE | this.sqlite3.capi.SQLITE_DESERIALIZE_RESIZEABLE,\n );\n\n this.ensureDb.checkRc(resultCode);\n\n this.invalidateDbSchema();\n\n perf?.logEnd(\"useSnapshot\", \"success\", \"system\");\n }\n\n createSnapshot() {\n return this.sqlite3.capi.sqlite3_js_db_export(this.ensureDb);\n }\n\n invalidateDbSchema() {\n this.loadedDbSchema = null;\n }\n\n cleanup() {\n this.preparedStatements.forEach((stmt) => {\n stmt.finalize();\n });\n this.preparedStatements.splice(0);\n this.preparedStatementsMap.clear();\n this.preparedRawStatementsMap.clear();\n }\n\n close() {\n this.cleanup();\n\n this.db?.close();\n this.db = null;\n }\n}\n\nexport type QueryBuilderOutput<QB> = QB extends Compilable<infer O> ? O : never;\ntype ParamsGetter<TParams> = <TKey extends keyof TParams>(key: TKey) => TParams[TKey];\n\ntype TypedStatement<TParams extends Record<string, unknown>, TResult> = {\n execute: (parameters: TParams) => TResult[];\n};\nexport type KyselyStatementFactory<\n TParams extends Record<string, unknown>,\n TDatabase,\n TQuery extends Compilable<TResult>,\n TResult = QueryBuilderOutput<TQuery>,\n> = (kysely: Kysely<TDatabase>, params: ParamsGetter<TParams>) => TQuery;\nexport type KyselyQueryFactory<TDatabase, TQuery extends Compilable<TResult>, TResult = QueryBuilderOutput<TQuery>> = (\n kysely: Kysely<TDatabase>,\n) => TQuery;\n","import type { SchemaModule } from \"kysely\";\n\nexport type CrdtEventType = \"item-created\" | \"item-updated\" | \"item-deleted\";\n\nexport type CrdtEventStatus = \"pending\" | \"applied\" | \"failed\" | \"deduped\";\n\n/** Persisted on applied events that were accepted but did not mutate materialized state. */\nexport const CRDT_EVENT_NO_OP_PAYLOAD = \"no-op\";\n\nexport function isNoOpCrdtEventPayload(payload: string) {\n return payload === CRDT_EVENT_NO_OP_PAYLOAD;\n}\n\nexport type CrdtEventOrigin = \"remote\" | \"own\" | \"local\";\n\nexport type PersistedCrdtEvent = {\n schema_version: number;\n sync_id: number;\n status: CrdtEventStatus;\n type: CrdtEventType;\n timestamp: string;\n origin: CrdtEventOrigin;\n source_node_id: string;\n dataset: string;\n item_id: string;\n payload: string;\n};\n\nexport type CrdtUpdateLogItem = {\n dataset: string;\n item_id: string;\n payload: string;\n};\n\nexport type CrdtUpdateLogPayload = Record<string, string>;\n\nexport const crdtSchema = {\n persistedEventsTable: createPersistedEventsTable,\n crdtUpdateLogTable: createCrdtUpdateLogTableQuery,\n};\n\nfunction createPersistedEventsTable(schema: SchemaModule, tableName: string) {\n return schema\n .createTable(tableName)\n .ifNotExists()\n .addColumn(\"sync_id\", \"integer\", (col) => col.notNull().primaryKey())\n .addColumn(\"schema_version\", \"integer\", (col) => col.notNull())\n .addColumn(\"status\", \"text\", (col) => col.notNull())\n .addColumn(\"type\", \"text\", (col) => col.notNull())\n .addColumn(\"timestamp\", \"text\", (col) => col.notNull())\n .addColumn(\"origin\", \"text\", (col) => col.notNull())\n .addColumn(\"dataset\", \"text\", (col) => col.notNull())\n .addColumn(\"item_id\", \"text\", (col) => col.notNull())\n .addColumn(\"payload\", \"text\", (col) => col.notNull());\n}\n\nfunction createCrdtUpdateLogTableQuery(schema: SchemaModule, tableName: string) {\n return schema\n .createTable(tableName)\n .ifNotExists()\n .addColumn(\"dataset\", \"text\", (col) => col.notNull())\n .addColumn(\"item_id\", \"text\", (col) => col.notNull())\n .addColumn(\"payload\", \"text\", (col) => col.notNull())\n .addPrimaryKeyConstraint(`pk_${tableName}`, [\"item_id\", \"dataset\"]);\n}\n","import type {\n ColumnDataType,\n ColumnDefinitionBuilderCallback,\n Compilable,\n CreateIndexBuilder,\n CreateTableBuilder,\n Expression,\n Kysely,\n} from \"kysely\";\nimport { dummyKysely } from \"../dummy-kysely\";\nimport { type CrdtEventType, isNoOpCrdtEventPayload } from \"../sqlite-crdt/crdt-table-schema\";\nimport type { StoredValue } from \"../sqlite-crdt/stored-value\";\n\ntype CrdtEvent = {\n type: CrdtEventType;\n dataset: string;\n item_id: string;\n payload: Record<string, unknown>;\n};\n\nexport type MigratableEvent = {\n schema_version: number;\n type: CrdtEventType;\n dataset: string;\n item_id: string;\n payload: string;\n};\n\ntype CrdtEventTransformer = (event: CrdtEvent) => CrdtEvent | null;\n\ntype MigrationStepSql =\n | {\n sql: string;\n parameters?: readonly unknown[];\n }\n | Compilable\n | ((db: Kysely<unknown>) => Compilable);\n\ntype TableRename = { oldTable: string; newTable: string };\n\ntype MigrationStep = {\n sql: MigrationStepSql | MigrationStepSql[];\n eventTransformer?: MigrationEventTransformers;\n tableRenames?: TableRename[];\n tableDrops?: string[];\n};\n\ntype RawMigrationStep = {\n sql: MigrationSql[];\n eventTransformer?: CompiledMigrationEventTransformer;\n tableRenames?: TableRename[];\n tableDrops?: string[];\n};\n\ntype MigrationSql = { sql: string; parameters: readonly unknown[] };\n\ntype DataTypeExpression = ColumnDataType | Expression<any>;\n\nconst protectedColumns = [\"id\", \"tombstone\"];\n\nfunction assertColumnNotProtected(column: string, operation: string) {\n if (protectedColumns.includes(column)) {\n throw new Error(`Cannot ${operation} protected column \"${column}\"`);\n }\n}\n\nconst migrationSteps = {\n createTable: (table: string, build: (table: CreateTableBuilder<string, never>) => Compilable): MigrationStep => ({\n sql: (db) => build(db.schema.createTable(table)),\n }),\n\n dropTable: (table: string): MigrationStep => ({\n sql: (db) => db.schema.dropTable(table),\n eventTransformer: {\n [table]: () => null,\n },\n tableDrops: [table],\n }),\n\n createIndex: (indexName: string, build: (index: CreateIndexBuilder) => Compilable): MigrationStep => ({\n sql: (db) => build(db.schema.createIndex(indexName)),\n }),\n\n dropIndex: (indexName: string): MigrationStep => ({\n sql: (db) => db.schema.dropIndex(indexName),\n }),\n\n renameTable: ({ oldTable, newTable }: { oldTable: string; newTable: string }): MigrationStep => ({\n sql: (db) => db.schema.alterTable(oldTable).renameTo(newTable),\n eventTransformer: {\n [oldTable]: (event) => {\n event.dataset = newTable;\n return event;\n },\n },\n tableRenames: [{ oldTable, newTable }],\n }),\n\n renameColumn: ({\n table,\n oldColumn,\n newColumn,\n }: {\n table: string;\n oldColumn: string;\n newColumn: string;\n }): MigrationStep => {\n assertColumnNotProtected(oldColumn, \"rename\");\n assertColumnNotProtected(newColumn, \"rename to\");\n return {\n sql: (db) => db.schema.alterTable(table).renameColumn(oldColumn, newColumn),\n eventTransformer: {\n [table]: (event) => {\n if ((event.type !== \"item-updated\" && event.type !== \"item-created\") || !(oldColumn in event.payload)) {\n return event;\n }\n\n const oldVal = event.payload[oldColumn];\n delete event.payload[oldColumn];\n event.payload[newColumn] = oldVal;\n\n return event;\n },\n },\n };\n },\n\n addColumn: ({\n table,\n column,\n type,\n defaultValue,\n build = (e) => e,\n }: {\n table: string;\n column: string;\n type: DataTypeExpression;\n defaultValue: number | boolean | string | null;\n build?: ColumnDefinitionBuilderCallback;\n }): MigrationStep => ({\n sql: (db) => db.schema.alterTable(table).addColumn(column, type, (x) => build(x).defaultTo(defaultValue)),\n eventTransformer: {\n [table]: (event) => {\n if (event.type !== \"item-created\") {\n return event;\n }\n\n event.payload[column] = defaultValue;\n\n return event;\n },\n },\n }),\n\n dropColumn: ({ table, column }: { table: string; column: string }): MigrationStep => {\n assertColumnNotProtected(column, \"drop\");\n return {\n sql: (db) => db.schema.alterTable(table).dropColumn(column),\n eventTransformer: {\n [table]: (event) => {\n if (!(column in event.payload)) {\n return event;\n }\n\n delete event.payload[column];\n\n if (event.type === \"item-updated\" && Object.keys(event.payload).length === 0) {\n return null;\n }\n\n return event;\n },\n },\n };\n },\n};\n\ntype MigrationEventTransformers = Record<string, CrdtEventTransformer>;\ntype CompiledMigrationEventTransformer = (event: CrdtEvent) => CrdtEvent | null;\n\nfunction buildMigrationSql(steps: MigrationStep[]): MigrationSql[] {\n return steps\n .flatMap((step) => (Array.isArray(step.sql) ? step.sql : [step.sql]))\n .map((sql): MigrationSql => {\n if (typeof sql === \"string\") {\n return { sql, parameters: [] };\n }\n\n if (typeof sql === \"function\") {\n const query = sql(dummyKysely).compile();\n return { sql: query.sql, parameters: query.parameters };\n }\n\n if (\"compile\" in sql) {\n const query = sql.compile();\n return { sql: query.sql, parameters: query.parameters };\n }\n\n return {\n sql: sql.sql,\n parameters: sql.parameters ?? [],\n };\n });\n}\n\nfunction buildMigrationEventTransformer(steps: MigrationStep[]): CompiledMigrationEventTransformer | undefined {\n const transformers: Array<[string, CrdtEventTransformer]> = [];\n\n for (const step of steps) {\n if (step.eventTransformer) {\n transformers.push(...Object.entries(step.eventTransformer));\n }\n }\n\n if (transformers.length === 0) {\n return undefined;\n }\n\n return (event: CrdtEvent) => {\n let transformedEvent: CrdtEvent | null = event;\n\n for (const [table, transformer] of transformers) {\n if (transformedEvent === null) {\n return null;\n }\n if (transformedEvent.dataset !== table) {\n continue;\n }\n transformedEvent = transformer(transformedEvent);\n if (transformedEvent === null) {\n return null;\n }\n }\n\n return transformedEvent;\n };\n}\n\nexport function createMigrations(buildMigrations: (builder: typeof migrationSteps) => Record<number, MigrationStep[]>) {\n const migrations: Record<number, RawMigrationStep> = Object.fromEntries(\n Object.entries(buildMigrations(migrationSteps)).map(([version, steps]) => {\n const versionNumber = Number(version);\n\n if (Number.isNaN(versionNumber)) {\n throw new Error(`Invalid migration version: ${version}`);\n }\n\n if (versionNumber < 0) {\n throw new Error(`Migration version cannot be negative: ${version}`);\n }\n\n const tableRenames = steps.flatMap((s) => s.tableRenames ?? []);\n const tableDrops = steps.flatMap((s) => s.tableDrops ?? []);\n\n return [\n version,\n {\n sql: buildMigrationSql(steps),\n eventTransformer: buildMigrationEventTransformer(steps),\n ...(tableRenames.length > 0 && { tableRenames }),\n ...(tableDrops.length > 0 && { tableDrops }),\n },\n ];\n }),\n );\n\n return migrations;\n}\n\nexport type Migrations = ReturnType<typeof createMigrations>;\n\nexport type MigrationsDb = {\n startTransaction: (callback: (tx: MigrationsTransaction) => void) => void;\n};\n\ntype MigrationsTransaction = {\n execute: (sql: string, parameters: readonly unknown[]) => void;\n};\n\nexport function createMigrator({\n migrations,\n schemaVersion,\n updateLogTableName,\n}: {\n migrations: Migrations;\n schemaVersion: StoredValue<number>;\n updateLogTableName?: string;\n}) {\n const latestSchemaVersion = Math.max(...Object.keys(migrations).map(Number));\n\n // Pre-sort migrations once for efficient range lookups\n const sortedMigrations = Object.entries(migrations)\n .map(([v, m]) => [Number(v), m] as const)\n .sort((a, b) => a[0] - b[0]);\n\n const applyMigration = (db: MigrationsDb, version: number, migration: RawMigrationStep) => {\n if (version <= schemaVersion.current) {\n throw new Error(`Cannot apply migration ${version} to schema version ${schemaVersion.current}`);\n }\n\n db.startTransaction((tx) => {\n for (const statement of migration.sql) {\n tx.execute(statement.sql, statement.parameters);\n }\n if (updateLogTableName) {\n if (migration.tableRenames) {\n for (const { oldTable, newTable } of migration.tableRenames) {\n tx.execute(`UPDATE ${updateLogTableName} SET \"dataset\" = ? WHERE \"dataset\" = ?`, [newTable, oldTable]);\n }\n }\n if (migration.tableDrops) {\n for (const table of migration.tableDrops) {\n tx.execute(`DELETE FROM ${updateLogTableName} WHERE \"dataset\" = ?`, [table]);\n }\n }\n }\n schemaVersion.current = version;\n });\n };\n\n const migrateEvent = <Event extends MigratableEvent>(event: Event, targetVersion?: number): Event | null => {\n targetVersion ??= latestSchemaVersion;\n if (targetVersion > schemaVersion.current) {\n throw new Error(\n `Target schema version ${targetVersion} is greater than current schema version ${schemaVersion.current}`,\n );\n }\n\n if (isNoOpCrdtEventPayload(event.payload)) {\n if (event.schema_version < targetVersion) {\n event.schema_version = targetVersion;\n }\n return event;\n }\n\n if (event.schema_version >= targetVersion) {\n return event;\n }\n\n const fromVersion = event.schema_version;\n\n let crdtEvent: CrdtEvent | null = {\n dataset: event.dataset,\n item_id: event.item_id,\n type: event.type,\n payload: JSON.parse(event.payload),\n };\n\n for (let i = 0; i < sortedMigrations.length; i++) {\n const [version, migration] = sortedMigrations[i];\n if (version <= fromVersion) continue;\n if (version > targetVersion) break;\n\n const transformer = migration.eventTransformer;\n if (transformer) {\n crdtEvent = transformer(crdtEvent);\n if (crdtEvent === null) return null;\n }\n }\n\n if (crdtEvent === null) {\n return null;\n }\n\n event.schema_version = targetVersion;\n event.dataset = crdtEvent.dataset;\n event.item_id = crdtEvent.item_id;\n event.type = crdtEvent.type;\n event.payload = JSON.stringify(crdtEvent.payload);\n\n return event;\n };\n\n const migrateEvents = <Event extends MigratableEvent>(events: Event[], targetVersion?: number): Event[] => {\n return events\n .map((event) => migrateEvent(event, targetVersion ?? latestSchemaVersion))\n .filter((event): event is NonNullable<typeof event> => event !== null);\n };\n\n return {\n latestSchemaVersion,\n get currentSchemaVersion() {\n return schemaVersion.current;\n },\n migrateDbToLatest: (db: MigrationsDb) => {\n const currentSchemaVersion = schemaVersion.current;\n\n if (currentSchemaVersion >= latestSchemaVersion) {\n return;\n }\n\n for (let i = 0; i < sortedMigrations.length; i++) {\n const [version, migration] = sortedMigrations[i];\n if (version <= currentSchemaVersion) continue;\n applyMigration(db, version, migration);\n }\n },\n migrateEvent,\n migrateEvents,\n };\n}\n\nexport type SyncDbMigrator = ReturnType<typeof createMigrator>;\n","export type StoredValue<T> = {\n get current(): T;\n set current(newValue: T);\n};\n\nexport function createStoredValue<T>({\n initialValue,\n saveToStorage,\n}: {\n initialValue: T;\n saveToStorage?: (value: T) => void;\n}): StoredValue<T> {\n let currentValue = initialValue;\n\n return {\n get current() {\n return currentValue;\n },\n set current(newValue: T) {\n saveToStorage?.(newValue);\n currentValue = newValue;\n },\n };\n}\n","import { type SchemaModule, sql } from \"kysely\";\nimport { createStoredValue } from \"./sqlite-crdt/stored-value\";\nimport type { InternalSQLiteWrapper } from \"./sqlite-db-wrapper\";\n\nexport type KvStoreItem = {\n key: string;\n value: string;\n};\n\nexport function createKvStoreTableQuery(schema: SchemaModule, tableName: string) {\n return schema\n .createTable(tableName)\n .ifNotExists()\n .addColumn(\"key\", \"text\", (col) => col.notNull().primaryKey())\n .addColumn(\"value\", \"text\", (col) => col.notNull())\n .modifyEnd(sql`without rowid`);\n}\n\nexport function createSQLiteKvStore({ db, metaTableName }: { db: InternalSQLiteWrapper<any>; metaTableName: string }) {\n const metaDb = db as InternalSQLiteWrapper<{\n meta: KvStoreItem;\n }>;\n\n const get = (key: string): string | null => {\n const [result] = metaDb.executePrepared(\n \"get-meta-value\",\n { key },\n (db, params) =>\n db\n .selectFrom(metaTableName as \"meta\")\n .where(\"key\", \"=\", params(\"key\"))\n .select(\"value\")\n .limit((eb) => eb.lit(1)),\n { loggerLevel: \"system\" },\n );\n\n return result?.value ?? null;\n };\n\n const set = (key: string, value: string) => {\n metaDb.executePrepared(\n \"set-meta-value\",\n { key, value },\n (db, params) =>\n db\n .insertInto(metaTableName as \"meta\")\n .values({ key: params(\"key\"), value: params(\"value\") })\n .onConflict((oc) => oc.doUpdateSet({ value: params(\"value\") })),\n { loggerLevel: \"system\" },\n );\n };\n\n const remove = (key: string) => {\n metaDb.executePrepared(\n \"remove-meta-value\",\n { key },\n (db, params) => db.deleteFrom(metaTableName as \"meta\").where(\"key\", \"=\", params(\"key\")),\n { loggerLevel: \"system\" },\n );\n };\n\n const getNumberOrDefault = <T>(key: string, defaultValue: T): T | number => {\n const value = get(key);\n if (!value) return defaultValue;\n const parsedValue = Number.parseInt(value, 10);\n return Number.isNaN(parsedValue) ? defaultValue : parsedValue;\n };\n\n return {\n get,\n set,\n remove,\n createStringStoredValue: (key: string, defaultValue: string) =>\n createStoredValue<string>({\n initialValue: get(key) ?? defaultValue,\n saveToStorage: (val) => set(key, val),\n }),\n createNumberStoredValue: (key: string, defaultValue: number) =>\n createStoredValue<number>({\n initialValue: getNumberOrDefault(key, defaultValue),\n saveToStorage: (val) => set(key, val.toString()),\n }),\n };\n}\n\nexport type KvStore = ReturnType<typeof createSQLiteKvStore>;\n","import type { CrdtUpdateLogItem, PersistedCrdtEvent } from \"../sqlite-crdt/crdt-table-schema\";\nimport type { StoredValue } from \"../sqlite-crdt/stored-value\";\nimport type { SQLiteDbWrapper } from \"../sqlite-db-wrapper\";\nimport { createKvStoreTableQuery, createSQLiteKvStore, type KvStoreItem } from \"../sqlite-kv-store\";\nimport { type ParsedTableName, parseTableName } from \"../utils\";\n\nexport type SystemDbConfig = {\n eventsTable: ParsedTableName;\n updateLogTable: ParsedTableName;\n};\n\nexport function createSystemDbConfig({\n eventsTableName,\n updateLogTableName,\n}: {\n eventsTableName: string;\n updateLogTableName: string;\n}): SystemDbConfig {\n return {\n eventsTable: parseTableName(eventsTableName),\n updateLogTable: parseTableName(updateLogTableName),\n };\n}\n\nexport type WorkerDbSchema = {\n crdt_update_log: CrdtUpdateLogItem;\n \"worker.kv\": KvStoreItem;\n \"worker.crdt_events\": PersistedCrdtEvent;\n};\n\nexport const workerDbConfig = createSystemDbConfig({\n eventsTableName: \"worker.crdt_events\",\n updateLogTableName: \"crdt_update_log\",\n});\n\nexport type MemoryDbSchema = {\n crdt_update_log: CrdtUpdateLogItem;\n persisted_crdt_events: PersistedCrdtEvent;\n};\n\nexport const memoryDbConfig = createSystemDbConfig({\n eventsTableName: \"persisted_crdt_events\",\n updateLogTableName: \"crdt_update_log\",\n});\n\nexport type SystemMigrationContext = SystemDbConfig & {\n execute: (sql: string) => void;\n};\n\nexport type SystemMigration = {\n version: number;\n up: (ctx: SystemMigrationContext) => void;\n};\n\nexport const baseSystemMigrations: SystemMigration[] = [\n {\n version: 0,\n up: (ctx: SystemMigrationContext) => {\n ctx.execute(`CREATE TABLE IF NOT EXISTS ${ctx.eventsTable.fullIdentifier} (\n \"sync_id\" integer NOT NULL PRIMARY KEY,\n \"schema_version\" integer NOT NULL,\n \"status\" text NOT NULL,\n \"type\" text NOT NULL,\n \"timestamp\" text NOT NULL,\n \"origin\" text NOT NULL,\n \"dataset\" text NOT NULL,\n \"item_id\" text NOT NULL,\n \"payload\" text NOT NULL\n )`);\n ctx.execute(`CREATE TABLE IF NOT EXISTS ${ctx.updateLogTable.fullIdentifier} (\n \"dataset\" text NOT NULL,\n \"item_id\" text NOT NULL,\n \"payload\" text NOT NULL,\n PRIMARY KEY (\"item_id\", \"dataset\")\n )`);\n },\n },\n {\n version: 1,\n up: (ctx: SystemMigrationContext) => {\n ctx.execute(`ALTER TABLE ${ctx.eventsTable.fullIdentifier} ADD COLUMN \"source_node_id\" TEXT NOT NULL DEFAULT ''`);\n },\n },\n {\n version: 2,\n up: (ctx: SystemMigrationContext) => {\n const indexName = `${ctx.eventsTable.table}_status_sync_id_idx`;\n ctx.execute(\n `CREATE INDEX IF NOT EXISTS ${ctx.eventsTable.schema}.${indexName} ON ${ctx.eventsTable.table} (\"status\", \"sync_id\")`,\n );\n },\n },\n {\n version: 3,\n up: (ctx: SystemMigrationContext) => {\n const indexName = `${ctx.eventsTable.table}_timestamp_status_sync_id_idx`;\n ctx.execute(\n `CREATE INDEX IF NOT EXISTS ${ctx.eventsTable.schema}.${indexName} ON ${ctx.eventsTable.table} (\"timestamp\", \"status\", \"sync_id\")`,\n );\n },\n },\n];\n\nexport function runSystemMigrations(opts: {\n version: StoredValue<number>;\n migrations: SystemMigration[];\n dbConfig: SystemDbConfig;\n execute: (sql: string) => void;\n transaction: (callback: () => void) => void;\n}): void {\n const ctx: SystemMigrationContext = {\n ...opts.dbConfig,\n execute: opts.execute,\n };\n for (const migration of opts.migrations) {\n if (migration.version > opts.version.current) {\n opts.transaction(() => {\n migration.up(ctx);\n opts.version.current = migration.version;\n });\n }\n }\n}\n\nexport function applyWorkerDbSchema(db: SQLiteDbWrapper<any>) {\n // KV table stays separate — needed before system migrations for version tracking\n db.executeKysely((kysely) => createKvStoreTableQuery(kysely.schema, \"worker.kv\"), { loggerLevel: \"system\" });\n\n // System schema migrations (each in its own transaction)\n const kvStore = createSQLiteKvStore({ db, metaTableName: \"worker.kv\" });\n runSystemMigrations({\n migrations: baseSystemMigrations,\n version: kvStore.createNumberStoredValue(\"internal-schema-version\", -1),\n dbConfig: workerDbConfig,\n execute: (sql) => db.execute(sql, { loggerLevel: \"system\" }),\n transaction: (callback) => db.executeTransaction(callback),\n });\n\n return { kvStore };\n}\n\nexport function applyMemoryDbSchema(db: SQLiteDbWrapper<any>) {\n db.execute(\n `CREATE TABLE IF NOT EXISTS ${memoryDbConfig.eventsTable.fullIdentifier} (\n \"sync_id\" integer NOT NULL PRIMARY KEY,\n \"schema_version\" integer NOT NULL,\n \"status\" text NOT NULL,\n \"type\" text NOT NULL,\n \"timestamp\" text NOT NULL,\n \"origin\" text NOT NULL,\n \"source_node_id\" text NOT NULL DEFAULT '',\n \"dataset\" text NOT NULL,\n \"item_id\" text NOT NULL,\n \"payload\" text NOT NULL\n )`,\n { loggerLevel: \"system\" },\n );\n db.execute(\n `CREATE INDEX IF NOT EXISTS ${memoryDbConfig.eventsTable.table}_status_sync_id_idx ON ${memoryDbConfig.eventsTable.table} (\"status\", \"sync_id\")`,\n {\n loggerLevel: \"system\",\n },\n );\n db.execute(\n `CREATE INDEX IF NOT EXISTS ${memoryDbConfig.eventsTable.table}_timestamp_status_sync_id_idx ON ${memoryDbConfig.eventsTable.table} (\"timestamp\", \"status\", \"sync_id\")`,\n {\n loggerLevel: \"system\",\n },\n );\n db.execute(\n `CREATE TABLE IF NOT EXISTS ${memoryDbConfig.updateLogTable.fullIdentifier} (\n \"dataset\" text NOT NULL,\n \"item_id\" text NOT NULL,\n \"payload\" text NOT NULL,\n PRIMARY KEY (\"item_id\", \"dataset\")\n)`,\n { loggerLevel: \"system\" },\n );\n}\n","import type { SqlValue } from \"@sqlite.org/sqlite-wasm\";\nimport type { Kysely } from \"kysely\";\nimport type { SystemDbConfig } from \"../migrations/system-schema\";\nimport type { InternalSQLiteWrapper } from \"../sqlite-db-wrapper\";\nimport { quoteId } from \"../utils\";\nimport {\n type CrdtEventType,\n type CrdtUpdateLogItem,\n type CrdtUpdateLogPayload,\n isNoOpCrdtEventPayload,\n} from \"./crdt-table-schema\";\n\nexport type PendingCrdtEvent = {\n type: CrdtEventType;\n dataset: string;\n item_id: string;\n timestamp: string;\n payload: string;\n};\n\nexport const createSQLiteCrdtApplyFunction = ({\n db,\n dbConfig,\n}: {\n db: InternalSQLiteWrapper<any>;\n dbConfig: SystemDbConfig;\n}) => {\n const applyCrdtEvent = createCrdtApplyFunction({\n getCrdtUpdateLog(opts) {\n const [metaRow] = db.executePrepared(\n \"get-item-crdt-meta\",\n {\n item_id: opts.itemId,\n dataset: opts.dataset,\n },\n (db, params) => {\n return (db as unknown as Kysely<{ table: CrdtUpdateLogItem }>)\n .selectFrom(dbConfig.updateLogTable.fullIdentifier as \"table\")\n .select(\"payload\")\n .where(\"item_id\", \"=\", params(\"item_id\"))\n .where(\"dataset\", \"=\", params(\"dataset\"));\n },\n { loggerLevel: \"system\" },\n );\n const meta = metaRow ? (JSON.parse(metaRow.payload) as CrdtUpdateLogPayload) : null;\n return meta;\n },\n insertCrdtUpdateLog(opts) {\n db.executePrepared(\n \"insert-crdt-update-log\",\n {\n item_id: opts.itemId,\n dataset: opts.dataset,\n payload: opts.payload,\n },\n (db, params) =>\n (db as unknown as Kysely<{ table: CrdtUpdateLogItem }>)\n .insertInto(dbConfig.updateLogTable.fullIdentifier as \"table\")\n .values({\n item_id: params(\"item_id\"),\n dataset: params(\"dataset\"),\n payload: params(\"payload\"),\n }),\n { loggerLevel: \"system\" },\n );\n },\n updateCrdtUpdateLog(opts) {\n db.executePrepared(\n \"update-crdt-update-log\",\n {\n item_id: opts.itemId,\n dataset: opts.dataset,\n payload: opts.payload,\n },\n (db, params) =>\n (db as unknown as Kysely<{ table: CrdtUpdateLogItem }>)\n .updateTable(dbConfig.updateLogTable.fullIdentifier as \"table\")\n .set({\n payload: params(\"payload\"),\n })\n .where(\"item_id\", \"=\", params(\"item_id\"))\n .where(\"dataset\", \"=\", params(\"dataset\")),\n { loggerLevel: \"system\" },\n );\n },\n insertItem(opts) {\n const insertPayload = {} as Record<string, unknown>;\n for (const key of Object.keys(opts.payload)) {\n insertPayload[key] = key;\n }\n db.executePrepared(\n `crdt-insert-item-${opts.dataset}`,\n opts.payload,\n (db) => db.insertInto(opts.dataset).values(insertPayload),\n { loggerLevel: \"system\" },\n );\n },\n updateItem(opts) {\n const keys = Array.from(Object.keys(opts.payload));\n keys.sort();\n db.executePreparedRaw({\n key: `update-item-${opts.dataset}-${keys.join(\"-\")}`,\n sql: `update ${quoteId(opts.dataset)} set ${keys.map((key) => `${quoteId(key)} = ?`).join(\",\")} where id = ?`,\n params: [...keys.map((key) => opts.payload[key]), opts.itemId] as SqlValue[],\n meta: { loggerLevel: \"system\" },\n });\n },\n });\n\n return applyCrdtEvent;\n};\n\ntype CreateCrdtApplyOpts = {\n getCrdtUpdateLog: (opts: { itemId: string; dataset: string }) => CrdtUpdateLogPayload | null;\n insertItem: (opts: { dataset: string; payload: Record<string, unknown> }) => void;\n insertCrdtUpdateLog: (opts: { dataset: string; itemId: string; payload: string }) => void;\n updateItem: (opts: { dataset: string; itemId: string; payload: Record<string, unknown> }) => void;\n updateCrdtUpdateLog: (opts: { dataset: string; itemId: string; payload: string }) => void;\n};\n\nexport function createCrdtApplyFunction({\n getCrdtUpdateLog,\n insertItem,\n insertCrdtUpdateLog,\n updateItem,\n updateCrdtUpdateLog,\n}: CreateCrdtApplyOpts) {\n type ItemCreatedOpts = {\n event: PendingCrdtEvent;\n };\n const applyItemCreated = ({ event }: ItemCreatedOpts) => {\n const eventPayload = JSON.parse(event.payload);\n\n eventPayload.tombstone = false;\n insertItem({ dataset: event.dataset, payload: eventPayload });\n\n const newUpdateLog = {} as Record<string, string>;\n for (const key of Object.keys(eventPayload)) {\n newUpdateLog[key] = event.timestamp;\n }\n\n insertCrdtUpdateLog({\n dataset: event.dataset,\n itemId: event.item_id,\n payload: JSON.stringify(newUpdateLog),\n });\n };\n\n type ItemUpdatedOpts = {\n event: PendingCrdtEvent;\n meta: CrdtUpdateLogPayload;\n };\n const applyItemUpdated = ({ event, meta }: ItemUpdatedOpts) => {\n if (!meta) {\n throw new Error(`Item ${event.item_id} in dataset ${event.dataset} not found`);\n }\n // A delete carries no field data on the wire — the tombstone is a local\n // materialization of the soft-delete, stamped here with the event's HLC so\n // it competes with concurrent field edits under the same last-write-wins rule.\n const eventPayload = event.type === \"item-deleted\" ? { tombstone: 1 } : JSON.parse(event.payload);\n\n const updatePayload = {} as Record<string, unknown>;\n let hasUpdates = false;\n\n for (const [key, value] of Object.entries(eventPayload)) {\n if (key === \"id\") {\n continue;\n }\n\n const lastUpdateTimestamp = meta[key];\n const currentUpdateTimestamp = event.timestamp;\n\n if (!lastUpdateTimestamp || !currentUpdateTimestamp || currentUpdateTimestamp > lastUpdateTimestamp) {\n updatePayload[key] = value;\n meta[key] = currentUpdateTimestamp;\n hasUpdates = true;\n }\n }\n\n if (!hasUpdates) {\n return;\n }\n\n updateItem({\n dataset: event.dataset,\n itemId: event.item_id,\n payload: updatePayload,\n });\n updateCrdtUpdateLog({\n dataset: event.dataset,\n itemId: event.item_id,\n payload: JSON.stringify(meta),\n });\n };\n\n return (event: PendingCrdtEvent) => {\n if (isNoOpCrdtEventPayload(event.payload)) {\n return;\n }\n\n const meta = getCrdtUpdateLog({\n itemId: event.item_id,\n dataset: event.dataset,\n });\n\n // TODO Check primary key / unique constraints\n\n if (event.type !== \"item-created\" && event.type !== \"item-updated\" && event.type !== \"item-deleted\") {\n throw new Error(`Unknown event type: ${event.type}`);\n }\n\n if (meta) {\n applyItemUpdated({ event, meta });\n return;\n }\n\n if (event.type === \"item-created\") {\n applyItemCreated({ event });\n return;\n }\n\n throw new Error(`Item ${event.item_id} in dataset ${event.dataset} not found`);\n };\n}\n","import { sql } from \"kysely\";\nimport { deserializeHLC, type HLCCounter, serializeHLC } from \"../hlc\";\nimport type { SyncDbMigrator } from \"../migrations/migrator\";\nimport type { SystemDbConfig } from \"../migrations/system-schema\";\nimport type { InternalSQLiteTransactionWrapper, InternalSQLiteWrapper } from \"../sqlite-db-wrapper\";\nimport { createTypedEventTarget, ensureSingletonExecution } from \"../utils\";\nimport { createSQLiteCrdtApplyFunction } from \"./apply-crdt-event\";\nimport {\n CRDT_EVENT_NO_OP_PAYLOAD,\n type CrdtEventOrigin,\n type CrdtEventStatus,\n type CrdtEventType,\n type CrdtUpdateLogItem,\n isNoOpCrdtEventPayload,\n type PersistedCrdtEvent,\n} from \"./crdt-table-schema\";\nimport { createEventHlcAccumulator } from \"./event-consistency\";\nimport type { StoredValue } from \"./stored-value\";\n\ntype LocalCrdtEvent = {\n type: CrdtEventType;\n dataset: string;\n item_id: string;\n payload: string;\n timestamp: string;\n schema_version: number;\n};\n\nexport type OwnCrdtEvent = {\n type: CrdtEventType;\n dataset: string;\n item_id: string;\n payload: string;\n timestamp?: undefined;\n schema_version?: undefined;\n};\n\ntype RemoteCrdtEvent = {\n type: CrdtEventType;\n dataset: string;\n item_id: string;\n payload: string;\n timestamp: string;\n schema_version: number;\n};\n\ntype EnqueuedCrdtEvent = LocalCrdtEvent | OwnCrdtEvent | RemoteCrdtEvent;\n\nexport type GetEventsOptions = {\n afterSyncId?: number;\n status?: CrdtEventStatus;\n excludeOrigin?: string;\n excludeNodeId?: string;\n limit?: number;\n};\n\nexport type GetEventsBatchQuery = {\n afterSyncId: number;\n status: CrdtEventStatus;\n limit: number;\n} & (\n | { excludeOrigin: CrdtEventOrigin; excludeNodeId?: undefined }\n | { excludeOrigin?: undefined; excludeNodeId: string }\n);\n\nexport type GetEventsBatch = {\n events: PersistedCrdtEvent[];\n hasMore: boolean;\n nextSyncId: number;\n};\n\nexport type EnqueueEventsResult = {\n beforeSyncId: number;\n afterSyncId: number;\n /** Resolves when the enqueued events have been processed (applied/deduped/failed). */\n processed: Promise<void>;\n};\n\nexport type EventUpdate = {\n status: CrdtEventStatus;\n schema_version: number;\n type: CrdtEventType;\n dataset: string;\n item_id: string;\n payload: string;\n};\n\ntype StorageHLC = Pick<HLCCounter, \"getNextHLC\" | \"mergeHLC\">;\n\ntype DbSyncerStorage = {\n nodeId: string;\n initialLocalSyncId: number;\n migrator: SyncDbMigrator;\n db: InternalSQLiteWrapper<any>;\n dbConfig: SystemDbConfig;\n hlc: StorageHLC;\n eventHlcAccumulator?: StoredValue<string>;\n onEventApplied?: (event: PersistedCrdtEvent) => void;\n};\n\nexport type CrdtStorage = ReturnType<typeof createCrdtStorage>;\n\ntype EventsAppliedPayload = {\n syncId: number;\n eventHlcSum: string | null;\n};\n\ntype InternalDbSchema = {\n _crdt_events: PersistedCrdtEvent;\n _crdt_update_log: CrdtUpdateLogItem;\n};\n\nexport function createCrdtStorage(storage: DbSyncerStorage) {\n let localSyncId = storage.initialLocalSyncId;\n\n const db = storage.db as InternalSQLiteWrapper<InternalDbSchema>;\n\n const crdtEventsTable = storage.dbConfig.eventsTable.fullIdentifier as \"_crdt_events\";\n\n const eventTarget = createTypedEventTarget<{\n \"events-applied\": EventsAppliedPayload;\n \"remote-event-apply-failed\": { syncId: number };\n }>();\n\n const persistEvent = (tx: InternalSQLiteTransactionWrapper<InternalDbSchema>, event: PersistedCrdtEvent) => {\n tx.executePrepared(\n \"persist-crdt-event\",\n event,\n (db, params) =>\n db.insertInto(crdtEventsTable).values({\n type: params(\"type\"),\n dataset: params(\"dataset\"),\n item_id: params(\"item_id\"),\n payload: params(\"payload\"),\n schema_version: params(\"schema_version\"),\n sync_id: params(\"sync_id\"),\n status: params(\"status\"),\n timestamp: params(\"timestamp\"),\n origin: params(\"origin\"),\n source_node_id: params(\"source_node_id\"),\n }),\n { loggerLevel: \"system\" },\n );\n };\n\n const enqueueEvents = (\n origin: CrdtEventOrigin,\n sourceNodeId: string,\n events: EnqueuedCrdtEvent[],\n ): EnqueueEventsResult => {\n const beforeSyncId = localSyncId;\n if (events.length === 0) {\n return { beforeSyncId, afterSyncId: beforeSyncId, processed: Promise.resolve() };\n }\n\n db.executeTransaction((tx) => {\n for (const event of events) {\n persistEvent(tx, {\n schema_version: event.schema_version ?? storage.migrator.currentSchemaVersion,\n timestamp: event.timestamp ?? serializeHLC(storage.hlc.getNextHLC()),\n type: event.type,\n dataset: event.dataset,\n item_id: event.item_id,\n origin: origin,\n source_node_id: sourceNodeId,\n payload: event.payload,\n sync_id: ++localSyncId,\n status: \"pending\",\n });\n }\n });\n\n return { beforeSyncId, afterSyncId: localSyncId, processed: processEnqueuedEvents() };\n };\n\n const enqueueLocalEvents = (events: LocalCrdtEvent[], sourceNodeId: string): EnqueueEventsResult => {\n return enqueueEvents(\"local\", sourceNodeId, events);\n };\n\n const enqueueOwnEvents = (events: OwnCrdtEvent[]): EnqueueEventsResult => {\n return enqueueEvents(\"own\", storage.nodeId, events);\n };\n\n const enqueueRemoteEvents = (events: RemoteCrdtEvent[]): EnqueueEventsResult => {\n return enqueueEvents(\"remote\", \"\", events);\n };\n\n const notifyEventApplied = (event: PersistedCrdtEvent) => {\n if (event.status === \"applied\") {\n storage.onEventApplied?.(event);\n }\n };\n\n const applyOwnEvent = (event: OwnCrdtEvent, { wrapInTransaction }: { wrapInTransaction?: boolean } = {}) => {\n const persistedEvent: PersistedCrdtEvent = {\n schema_version: storage.migrator.currentSchemaVersion,\n timestamp: serializeHLC(storage.hlc.getNextHLC()),\n type: event.type,\n dataset: event.dataset,\n item_id: event.item_id,\n origin: \"own\",\n source_node_id: storage.nodeId,\n payload: event.payload,\n sync_id: ++localSyncId,\n status: \"pending\",\n };\n\n if (wrapInTransaction) {\n db.executeTransaction((tx) => {\n persistEvent(tx, persistedEvent);\n processPersistedEvent(tx, persistedEvent);\n persistEventHlcAccumulator();\n });\n } else {\n persistEvent(db, persistedEvent);\n processPersistedEvent(db, persistedEvent);\n persistEventHlcAccumulator();\n }\n };\n\n const dispatchEventsApplied = (syncId = localSyncId) => {\n eventTarget.dispatchEvent(\"events-applied\", {\n syncId,\n eventHlcSum: eventHlcAccumulator?.current ?? null,\n });\n };\n\n const hasPendingEvents = (): boolean => {\n const events = db.executePrepared(\n \"has-pending-events\",\n { status: \"pending\" as const },\n (db, params) =>\n db.selectFrom(crdtEventsTable).select(\"sync_id\").where(\"status\", \"=\", params(\"status\")).limit(sql.lit(1)),\n { loggerLevel: \"system\" },\n );\n return events.length > 0;\n };\n\n const getEventsBatch = (options: GetEventsBatchQuery): GetEventsBatch => {\n const limit = options.limit ?? 50;\n\n const queryParams = {\n limit: limit + 1,\n status: options.status ?? null,\n afterSyncId: options.afterSyncId ?? null,\n excludeOrigin: options.excludeOrigin ?? null,\n excludeNodeId: options.excludeNodeId ?? null,\n };\n\n const filterKeys = [\n queryParams.excludeNodeId ? \"nodeid\" : \"no-nodeid\",\n queryParams.excludeOrigin ? \"origin\" : \"no-origin\",\n ];\n\n const events = db.executePrepared(`get-events-batch-${filterKeys.join(\"-\")}`, queryParams, (db, params) =>\n db\n .selectFrom(crdtEventsTable)\n .where(\"sync_id\", \">\", params(\"afterSyncId\"))\n .where(\"status\", \"=\", params(\"status\"))\n .$if(!!queryParams.excludeNodeId, (qb) => qb.where(\"source_node_id\", \"!=\", params(\"excludeNodeId\")))\n .$if(!!queryParams.excludeOrigin, (qb) => qb.where(\"origin\", \"!=\", params(\"excludeOrigin\")))\n .selectAll()\n .limit(params(\"limit\"))\n .orderBy(\"sync_id\", \"asc\"),\n );\n\n const hasMore = events.length > limit;\n if (hasMore) {\n events.pop();\n }\n return {\n events,\n hasMore,\n nextSyncId: events[events.length - 1]?.sync_id ?? options.afterSyncId ?? 0,\n };\n };\n\n // Storage is quiescent when there is nothing left to converge: no events waiting\n // to be applied, and no local applied events past `pushedSyncId` still waiting to\n // be pushed to the remote. When quiescent and caught up, the local and remote node\n // share the exact same set of applied events, so their HLC checksums must match.\n const checkIsQuiescent = (pushedSyncId: number): boolean => {\n if (hasPendingEvents()) {\n return false;\n }\n\n const unpushed = getEventsBatch({\n status: \"applied\",\n afterSyncId: pushedSyncId,\n excludeOrigin: \"remote\",\n limit: 1,\n });\n\n return unpushed.events.length === 0;\n };\n\n const applyCrdtEvent = createSQLiteCrdtApplyFunction({\n db,\n dbConfig: storage.dbConfig,\n });\n const eventHlcAccumulator = storage.eventHlcAccumulator\n ? createEventHlcAccumulator(storage.eventHlcAccumulator.current)\n : null;\n\n const persistEventHlcAccumulator = () => {\n if (eventHlcAccumulator && storage.eventHlcAccumulator) {\n storage.eventHlcAccumulator.current = eventHlcAccumulator.current;\n }\n };\n\n // Rebuild the accumulator from the full applied-event history when it has\n // never been computed for this storage. An empty stored value is the \"never\n // computed\" sentinel: once any event is applied the accumulator is persisted\n // as padded hex, never \"\". This recovers storages created before the\n // accumulator existed and lets us force a recompute by bumping the stored\n // value's key version. The accumulator is a commutative sum, so scan order\n // does not matter.\n const recomputeEventHlcAccumulatorIfNeeded = () => {\n if (!eventHlcAccumulator || !storage.eventHlcAccumulator) {\n return;\n }\n if (storage.eventHlcAccumulator.current !== \"\") {\n return;\n }\n\n const batchSize = 1000;\n let afterSyncId = 0;\n for (;;) {\n const rows = db.executePrepared(\n \"get-applied-event-timestamps\",\n { status: \"applied\" as const, afterSyncId, limit: batchSize },\n (db, params) =>\n db\n .selectFrom(crdtEventsTable)\n .select([\"sync_id\", \"timestamp\"])\n .where(\"status\", \"=\", params(\"status\"))\n .where(\"sync_id\", \">\", params(\"afterSyncId\"))\n .orderBy(\"sync_id\", \"asc\")\n .limit(params(\"limit\")),\n { loggerLevel: \"system\" },\n );\n\n if (rows.length === 0) {\n break;\n }\n\n for (const row of rows) {\n eventHlcAccumulator.add(row.timestamp);\n afterSyncId = row.sync_id;\n }\n\n if (rows.length < batchSize) {\n break;\n }\n }\n\n persistEventHlcAccumulator();\n };\n\n const hasAcceptedEventWithTimestamp = (\n tx: InternalSQLiteTransactionWrapper<InternalDbSchema>,\n event: PersistedCrdtEvent,\n ) => {\n const [existingEvent] = tx.executePrepared(\n \"get-accepted-crdt-event-by-timestamp\",\n {\n timestamp: event.timestamp,\n sync_id: event.sync_id,\n },\n (db, params) =>\n db\n .selectFrom(crdtEventsTable)\n .select(\"sync_id\")\n .where(\"timestamp\", \"=\", params(\"timestamp\"))\n .where(\"sync_id\", \"<\", params(\"sync_id\"))\n .where(\"status\", \"=\", sql.lit(\"applied\"))\n .limit(sql.lit(1)),\n { loggerLevel: \"system\" },\n );\n\n return existingEvent !== undefined;\n };\n\n const processPersistedEvent = (tx: InternalSQLiteTransactionWrapper<InternalDbSchema>, event: PersistedCrdtEvent) => {\n if (event.status !== \"pending\") {\n throw new Error(`Event ${event.sync_id} is not pending`);\n }\n\n try {\n if (hasAcceptedEventWithTimestamp(tx, event)) {\n event.status = \"deduped\";\n return event;\n }\n\n // Always advance HLC, even for no-op events, to maintain monotonic ordering\n if (event.origin === \"local\" || event.origin === \"remote\") {\n storage.hlc.mergeHLC(deserializeHLC(event.timestamp));\n }\n\n if (isNoOpCrdtEventPayload(event.payload)) {\n applyCrdtEvent(event);\n event.status = \"applied\";\n eventHlcAccumulator?.add(event.timestamp);\n return event;\n }\n\n const migratedEvent = storage.migrator.migrateEvent(event, storage.migrator.latestSchemaVersion);\n\n if (migratedEvent === null) {\n // Event was dropped during migration (e.g., table was deleted)\n event.schema_version = storage.migrator.latestSchemaVersion;\n event.payload = CRDT_EVENT_NO_OP_PAYLOAD;\n\n applyCrdtEvent(event);\n event.status = \"applied\";\n eventHlcAccumulator?.add(event.timestamp);\n return event;\n }\n\n event.schema_version = migratedEvent.schema_version;\n event.type = migratedEvent.type;\n event.dataset = migratedEvent.dataset;\n event.item_id = migratedEvent.item_id;\n event.payload = migratedEvent.payload;\n\n applyCrdtEvent(event);\n event.status = \"applied\";\n eventHlcAccumulator?.add(event.timestamp);\n } catch (error) {\n console.error(\"Error applying enqueued CRDT event\", error);\n event.status = \"failed\";\n } finally {\n tx.executePrepared(\n \"update-crdt-event\",\n event,\n (db, params) =>\n db\n .updateTable(crdtEventsTable)\n .set({\n status: params(\"status\"),\n schema_version: params(\"schema_version\"),\n type: params(\"type\"),\n dataset: params(\"dataset\"),\n item_id: params(\"item_id\"),\n payload: params(\"payload\"),\n })\n .where(\"sync_id\", \"=\", params(\"sync_id\")),\n { loggerLevel: \"system\" },\n );\n }\n };\n\n const processEnqueuedEvents = ensureSingletonExecution(async () => {\n let hasMore = true;\n while (hasMore) {\n await Promise.resolve();\n\n const batchSize = 100;\n\n const events = db.executePrepared(\n \"get-enqueued-pending-events\",\n {\n status: \"pending\" as const,\n limit: batchSize + 1,\n },\n (db, params) =>\n db\n .selectFrom(crdtEventsTable)\n .selectAll()\n .where(\"status\", \"=\", params(\"status\"))\n .limit(params(\"limit\"))\n .orderBy(\"sync_id\", \"asc\"),\n );\n hasMore = events.length > batchSize;\n if (hasMore) {\n events.pop();\n }\n\n if (events.length === 0) {\n break;\n }\n\n let appliedSyncId: number | null = null;\n const failedRemoteSyncIds: number[] = [];\n\n db.executeTransaction((tx) => {\n for (const event of events) {\n processPersistedEvent(tx, event);\n notifyEventApplied(event);\n if (event.status === \"applied\") {\n appliedSyncId = event.sync_id;\n } else if (event.status === \"failed\" && event.origin === \"remote\") {\n failedRemoteSyncIds.push(event.sync_id);\n }\n }\n persistEventHlcAccumulator();\n });\n\n if (appliedSyncId !== null) {\n dispatchEventsApplied(appliedSyncId);\n }\n\n // A remote event was accepted by the server but could not be applied\n // locally, which means our local state has diverged from the server.\n for (const syncId of failedRemoteSyncIds) {\n eventTarget.dispatchEvent(\"remote-event-apply-failed\", { syncId });\n }\n }\n });\n\n recomputeEventHlcAccumulatorIfNeeded();\n\n void processEnqueuedEvents();\n\n return {\n getEventsBatch,\n enqueueLocalEvents,\n enqueueOwnEvents,\n enqueueRemoteEvents,\n applyOwnEvent,\n dispatchEventsApplied,\n checkIsQuiescent,\n getEventHlcAccumulator: () => eventHlcAccumulator?.current ?? null,\n\n addEventListener: eventTarget.addEventListener,\n removeEventListener: eventTarget.removeEventListener,\n };\n}\n","import { xxhash } from \"../hash\";\n\nconst MASK_128 = (1n << 128n) - 1n;\nconst HEX_128_PATTERN = /^[0-9a-f]{32}$/;\n\nexport function createEventHlcAccumulator(initialValue: string) {\n let current = parseHex128(initialValue);\n\n return {\n add(timestamp: string) {\n current = (current + hash128BigInt(timestamp)) & MASK_128;\n },\n get current() {\n return toHex128(current);\n },\n };\n}\n\nfunction parseHex128(value: string) {\n if (value === \"\") {\n return 0n;\n }\n const normalized = value.toLowerCase();\n if (!HEX_128_PATTERN.test(normalized)) {\n throw new Error(`Invalid event HLC accumulator value: ${value}`);\n }\n return BigInt(`0x${normalized}`);\n}\n\nfunction toHex128(value: bigint) {\n return value.toString(16).padStart(32, \"0\");\n}\n\nfunction hash128BigInt(value: string) {\n return xxhash.h64(value, 0n) | (xxhash.h64(value, 1n) << 64n);\n}\n","import type { CrdtStorage } from \"./crdt-storage\";\n\ntype CrdtSyncProducer = {\n storage: CrdtStorage;\n broadcastEvents: (request: { newSyncId: number; eventHlcSum: string | null }) => void;\n};\n\nexport const createCrdtSyncProducer = ({ storage, broadcastEvents }: CrdtSyncProducer) => {\n storage.addEventListener(\"events-applied\", (event) => {\n broadcastEvents({\n newSyncId: event.payload.syncId,\n eventHlcSum: event.payload.eventHlcSum,\n });\n });\n};\n","import retryAsPromised from \"retry-as-promised\";\nimport type { SyncDbMigrator } from \"../migrations/migrator\";\nimport { createTypedEventTarget, ensureSingletonExecution, tryCatchAsync } from \"../utils\";\nimport type { EventsPullResponse } from \"../worker-db/worker-common\";\nimport type { PendingCrdtEvent } from \"./apply-crdt-event\";\nimport type { CrdtStorage } from \"./crdt-storage\";\nimport type { StoredValue } from \"./stored-value\";\n\ntype CrdtSyncRemoteSourceConfig = {\n bufferSize: number;\n storage: CrdtStorage;\n migrator: SyncDbMigrator;\n pullSyncId: StoredValue<number>;\n pushSyncId: StoredValue<number>;\n nodeId: string;\n remoteFactory?: CreateRemoteSourceFactory;\n};\n\nexport type EventsPullRequest = {\n afterSyncId: number;\n excludeNodeId?: string;\n};\n\nexport type EventsPushRequest = {\n nodeId: string;\n events: (PendingCrdtEvent & { schema_version: number })[];\n};\nexport type EventsPushResponse = {\n ok: boolean;\n /** Remote sync_id right before the pushed events were enqueued. */\n beforeSyncId?: number;\n /** Remote sync_id right after the pushed events were enqueued. */\n afterSyncId?: number;\n};\n\nexport type CrdtSyncRemoteSource = ReturnType<typeof createCrdtSyncRemoteSource>;\n\nexport type EventsAvailable = {\n newSyncId: number;\n remoteEventHlcSum: string | null;\n};\n\nexport type CreateRemoteSourceFactory = (opts: {\n onEventsAvailable: (event: EventsAvailable) => void;\n}) => RemoteSource | Promise<RemoteSource>;\n\ntype RemoteSource = {\n pullEvents: (request: EventsPullRequest) => Promise<EventsPullResponse>;\n pushEvents: (request: EventsPushRequest) => Promise<EventsPushResponse>;\n disconnect?: () => void | Promise<void>;\n};\n\nexport class SchemaVersionMismatchError extends Error {\n constructor(\n public remoteSchemaVersion: number,\n public localSchemaVersion: number,\n ) {\n super(`Schema version mismatch: remote ${remoteSchemaVersion} != local ${localSchemaVersion}`);\n this.name = \"SchemaVersionMismatchError\";\n }\n}\n\ntype RemoteSourceState =\n | {\n type: \"pending\";\n }\n | {\n type: \"offline\";\n reason: OfflineReason;\n }\n | {\n type: \"online\";\n source: RemoteSource;\n };\n\nexport type OfflineReason =\n | \"NOT_INITIALIZED\"\n | \"INITIALIZATION_FAILED\"\n | \"REMOTE_PUSH_ERROR\"\n | \"REMOTE_PULL_ERROR\"\n | \"DISCONNECTED\";\n\nexport type DeSyncDetectedReason = \"CHECKSUM_MISMATCH\" | \"ERROR_APPLYING_REMOTE_EVENT\";\n\nexport const createCrdtSyncRemoteSource = ({\n bufferSize,\n storage,\n migrator,\n pullSyncId,\n pushSyncId,\n nodeId,\n remoteFactory,\n}: CrdtSyncRemoteSourceConfig) => {\n const eventTarget = createTypedEventTarget<{\n \"state-changed\": RemoteSourceState[\"type\"];\n \"de-sync-detected\": {\n reason: DeSyncDetectedReason;\n };\n \"remote-schema-version-mismatch\": {\n remoteSchemaVersion: number;\n localSchemaVersion: number;\n };\n }>();\n\n let remoteState: RemoteSourceState = { type: \"offline\", reason: \"NOT_INITIALIZED\" };\n\n const setRemoteState = (state: RemoteSourceState) => {\n remoteState = state;\n eventTarget.dispatchEvent(\"state-changed\", state.type);\n };\n\n const initRemote = ensureSingletonExecution(\n async () => {\n if (remoteState.type !== \"offline\") {\n throw new Error(\"Remote source is not offline\");\n }\n\n if (!remoteFactory) {\n console.warn(\"Remote source factory not provided. Going offline.\");\n setRemoteState({ type: \"offline\", reason: \"NOT_INITIALIZED\" });\n return;\n }\n\n setRemoteState({ type: \"pending\" });\n\n const factoryResult = await tryCatchAsync(async () => {\n return await remoteFactory?.({\n onEventsAvailable: ({ newSyncId, remoteEventHlcSum }) => {\n pullEvents({ remoteSyncId: newSyncId, remoteEventHlcSum, includeSelf: false });\n },\n });\n });\n\n if (!factoryResult.success) {\n setRemoteState({ type: \"offline\", reason: \"INITIALIZATION_FAILED\" });\n console.warn(\"Failed to create remote source\", factoryResult.error);\n return;\n }\n\n setRemoteState({\n type: \"online\",\n source: factoryResult.data,\n });\n },\n { queueReExecution: false },\n );\n\n const syncWithRemote = ensureSingletonExecution(\n async () => {\n if (remoteState.type !== \"online\") {\n return;\n }\n\n await pullEvents();\n await startPushingEvents();\n },\n { queueReExecution: false },\n );\n\n const goOffline = ensureSingletonExecution(\n async (reason: OfflineReason) => {\n if (remoteState.type !== \"online\") {\n return;\n }\n const source = remoteState.source;\n\n setRemoteState({ type: \"pending\" });\n\n const disconnectResult = await tryCatchAsync(async () => {\n return await source.disconnect?.();\n });\n\n if (!disconnectResult.success) {\n console.warn(\"Error while disconnecting from remote source\", disconnectResult.error);\n }\n\n setRemoteState({ type: \"offline\", reason });\n },\n { queueReExecution: false },\n );\n\n const goOnline = async () => {\n if (remoteState.type !== \"online\") {\n await initRemote();\n }\n\n if (remoteState.type === \"online\") {\n await syncWithRemote();\n }\n };\n\n let requestedPullSyncId: number | null = null;\n let pullPromise: Promise<void> | null = null;\n const pullEvents = (request?: {\n remoteSyncId?: number;\n remoteEventHlcSum?: string | null;\n includeSelf?: boolean;\n }) => {\n if (remoteState.type !== \"online\") {\n return Promise.resolve();\n }\n\n const remoteSyncId = request?.remoteSyncId;\n\n if (remoteSyncId !== undefined && remoteSyncId <= pullSyncId.current) {\n // We are already caught up to this broadcast, so there is nothing to pull.\n // This is the quiescent moment to verify we have not diverged from the\n // remote (the check is a no-op unless we are exactly aligned: remoteSyncId\n // === pullSyncId.current).\n checkRemoteConsistency(remoteSyncId, request?.remoteEventHlcSum ?? null);\n return Promise.resolve();\n }\n\n if (pullPromise) {\n if (remoteSyncId !== undefined && (!requestedPullSyncId || requestedPullSyncId < remoteSyncId)) {\n requestedPullSyncId = remoteSyncId;\n }\n return pullPromise;\n }\n\n pullPromise = pullAllEvents({\n afterSyncId: pullSyncId.current,\n excludeNodeId: request?.includeSelf ? undefined : nodeId,\n })\n .catch((error) => {\n console.error(\"Error pulling events. Going offline.\", error);\n goOffline(\"REMOTE_PULL_ERROR\");\n })\n .finally(() => {\n pullPromise = null;\n\n const nextTarget = requestedPullSyncId;\n requestedPullSyncId = null;\n\n if (nextTarget && nextTarget > pullSyncId.current) {\n pullEvents({ remoteSyncId: nextTarget });\n }\n });\n return pullPromise;\n };\n\n const pullAllEvents = async (opts: EventsPullRequest) => {\n let hasMore = true;\n let afterSyncId = opts.afterSyncId;\n while (hasMore) {\n if (remoteState.type !== \"online\") {\n return;\n }\n const source = remoteState.source;\n\n const response = await retryAsPromised(\n () =>\n source.pullEvents({\n ...opts,\n afterSyncId,\n }),\n {\n max: 3,\n backoffBase: 100,\n backoffExponent: 1.5,\n backoffJitter: 150,\n timeout: 10000,\n },\n );\n hasMore = response.hasMore;\n afterSyncId = response.nextSyncId;\n\n if (response.events) {\n storage.enqueueRemoteEvents(\n response.events.map((x) => {\n if (x.schema_version > migrator.currentSchemaVersion) {\n eventTarget.dispatchEvent(\"remote-schema-version-mismatch\", {\n remoteSchemaVersion: x.schema_version,\n localSchemaVersion: migrator.currentSchemaVersion,\n });\n throw new SchemaVersionMismatchError(x.schema_version, migrator.currentSchemaVersion);\n }\n return x;\n }),\n );\n }\n if (response.nextSyncId <= pullSyncId.current) {\n break;\n }\n if (response.nextSyncId > pullSyncId.current) {\n pullSyncId.current = response.nextSyncId;\n }\n }\n };\n\n // De-sync detection: when we are exactly caught up to the remote's broadcast\n // sync id and fully quiescent, our applied-event set must equal the remote's,\n // so our HLC checksums must match. A mismatch means the nodes have diverged.\n const checkRemoteConsistency = (remoteSyncId: number, remoteEventHlcSum: string | null) => {\n // A remote with no accumulator gives us nothing to compare against.\n if (remoteEventHlcSum === null) {\n return;\n }\n\n // Only meaningful when we are exactly caught up: if we are behind we still\n // need to pull; if we are ahead our state covers events the remote checksum\n // does not.\n if (remoteSyncId !== pullSyncId.current) {\n return;\n }\n\n // Quiescence: the accumulator only matches the remote's when nothing is left\n // to apply locally and no local applied events are still waiting to be pushed\n // (those are in our accumulator but the remote has not seen them yet).\n if (!storage.checkIsQuiescent(pushSyncId.current)) {\n return;\n }\n\n const localEventHlcSum = storage.getEventHlcAccumulator();\n if (localEventHlcSum === null) {\n return;\n }\n\n if (localEventHlcSum !== remoteEventHlcSum) {\n eventTarget.dispatchEvent(\"de-sync-detected\", { reason: \"CHECKSUM_MISMATCH\" });\n console.warn(\n `[sqlite-sync] De-sync detected at syncId ${remoteSyncId}: local HLC checksum ${localEventHlcSum} != remote ${remoteEventHlcSum}. Local and remote have diverged despite being caught up.`,\n );\n }\n };\n\n const startPushingEvents = ensureSingletonExecution(async () => {\n while (true) {\n const eventsBatch = storage.getEventsBatch({\n status: \"applied\",\n afterSyncId: pushSyncId.current,\n excludeOrigin: \"remote\",\n limit: bufferSize,\n });\n if (eventsBatch.events.length === 0) {\n break;\n }\n\n if (remoteState.type !== \"online\") {\n break;\n }\n const source = remoteState.source;\n\n let response: EventsPushResponse;\n try {\n response = await retryAsPromised(\n () =>\n source.pushEvents({\n nodeId,\n events: eventsBatch.events.map((event) => ({\n schema_version: event.schema_version,\n timestamp: event.timestamp,\n type: event.type,\n dataset: event.dataset,\n item_id: event.item_id,\n payload: event.payload,\n })),\n }),\n {\n max: 3,\n backoffBase: 100,\n backoffExponent: 1.5,\n backoffJitter: 150,\n timeout: 10000,\n },\n );\n } catch (error) {\n console.error(\"Error pushing events. Going offline.\", error);\n goOffline(\"REMOTE_PUSH_ERROR\");\n return;\n }\n\n pushSyncId.current = eventsBatch.nextSyncId;\n\n // Fast-forward the pull cursor: the remote assigns sync ids for the pushed\n // events synchronously, so (beforeSyncId, afterSyncId] contains only this\n // node's own events. If we are caught up to at least beforeSyncId, the skipped\n // range (pullSyncId, afterSyncId] contains only our own events, so there is\n // nothing to pull up to afterSyncId.\n if (\n response.ok &&\n response.beforeSyncId !== undefined &&\n response.afterSyncId !== undefined &&\n response.beforeSyncId <= pullSyncId.current &&\n response.afterSyncId > pullSyncId.current\n ) {\n pullSyncId.current = response.afterSyncId;\n }\n if (!eventsBatch.hasMore) {\n break;\n }\n }\n });\n\n const eventsAppliedSubscription = storage.addEventListener(\"events-applied\", () => {\n startPushingEvents();\n });\n\n const remoteEventApplyFailedSubscription = storage.addEventListener(\"remote-event-apply-failed\", () => {\n eventTarget.dispatchEvent(\"de-sync-detected\", { reason: \"ERROR_APPLYING_REMOTE_EVENT\" });\n });\n\n const getState = (): \"pending\" | \"offline\" | \"online\" => remoteState.type;\n\n const dispose = async () => {\n await goOffline(\"DISCONNECTED\");\n eventsAppliedSubscription.unsubscribe();\n remoteEventApplyFailedSubscription.unsubscribe();\n };\n\n return {\n goOnline,\n goOffline,\n syncWithRemote,\n getState,\n dispose,\n addEventListener: eventTarget.addEventListener,\n removeEventListener: eventTarget.removeEventListener,\n };\n};\n","import { createStore, del, get, set } from \"idb-keyval\";\nimport { generateId } from \"../utils\";\nimport type { WorkerNotificationMessage } from \"./worker-common\";\n\nexport type ResetRequest = {\n epoch: string;\n requestedAt: number;\n};\n\n/** Durable async key-value storage for reset state. Injectable for tests. */\nexport type ResetStore = {\n get: <T>(key: string) => Promise<T | undefined>;\n set: (key: string, value: unknown) => Promise<void>;\n delete: (key: string) => Promise<void>;\n};\n\n/** Default IndexedDB-backed reset store (idb-keyval over a dedicated database). */\nexport function createIdbResetStore(): ResetStore {\n const store = createStore(\"sqlite-sync\", \"kv\");\n return {\n get: (key) => get(key, store),\n set: (key, value) => set(key, value, store),\n delete: (key) => del(key, store),\n };\n}\n\n/**\n * A clean reset is a recovery action for a de-sync detected now. If the reload\n * never happens (broadcast lost, tab crashed mid-reload, browser killed the page),\n * the request must not fire on an arbitrary later cold start and silently wipe\n * local-only writes accumulated since.\n */\nexport const RESET_REQUEST_TTL_MS = 10 * 60 * 1000; // 10 minutes\n\nconst resetRequestKey = (dbId: string) => `sqlite-sync-reset-request-${dbId}`;\nconst resetAppliedKey = (dbId: string) => `sqlite-sync-reset-applied-${dbId}`;\n\ntype ResetStateStoreOptions = {\n store: ResetStore;\n dbId: string;\n now?: () => number;\n};\n\nexport type ResetStateStore = ReturnType<typeof createResetStateStore>;\n\n/**\n * Worker-owned durable reset state. The reset decision must be owned by the\n * elected worker, not by tabs during `createSyncedDb` — otherwise a later\n * worker election could repeat an already-applied wipe.\n */\nexport function createResetStateStore({ store, dbId, now = () => Date.now() }: ResetStateStoreOptions) {\n const requestKey = resetRequestKey(dbId);\n const appliedKey = resetAppliedKey(dbId);\n\n return {\n async writeResetRequest(epoch: string): Promise<ResetRequest> {\n const request: ResetRequest = { epoch, requestedAt: now() };\n await store.set(requestKey, request);\n return request;\n },\n /**\n * Read the pending reset request after winning the worker election.\n * Returns the request only when it has not been applied yet and is within\n * the TTL. Stale requests are deleted so they cannot fire on a later cold start.\n */\n async resolvePendingReset(): Promise<ResetRequest | undefined> {\n const request = await store.get<ResetRequest>(requestKey);\n if (!request) {\n return undefined;\n }\n\n if (now() - request.requestedAt > RESET_REQUEST_TTL_MS) {\n await store.delete(requestKey);\n return undefined;\n }\n\n const appliedEpoch = await store.get<string>(appliedKey);\n if (request.epoch === appliedEpoch) {\n return undefined;\n }\n\n return request;\n },\n /**\n * Record the epoch as applied. Must be called only after the worker has\n * successfully initialized with `clearOnInit: true`, so a failed init can\n * be retried by a later elected worker.\n */\n async markResetApplied(epoch: string): Promise<void> {\n await store.set(appliedKey, epoch);\n },\n };\n}\n\ntype ReloadRequestHandlerOptions = {\n resetState: ResetStateStore;\n broadcast: (message: WorkerNotificationMessage) => void;\n generateEpoch?: () => string;\n};\n\n/**\n * Worker-side `requestReload` RPC implementation. For `clean: true` the reset\n * request is durably stored before broadcasting and before the RPC resolves,\n * so the epoch survives no matter which path triggers the reload.\n */\nexport function createReloadRequestHandler({\n resetState,\n broadcast,\n generateEpoch = generateId,\n}: ReloadRequestHandlerOptions) {\n return async (options: { clean: boolean }): Promise<void> => {\n const reloadEpoch = generateEpoch();\n\n if (options.clean) {\n await resetState.writeResetRequest(reloadEpoch);\n }\n\n broadcast({\n notificationType: \"reload-requested\",\n reloadEpoch,\n clean: options.clean,\n });\n };\n}\n","import type {\n DeSyncDetectedReason,\n EventsPullRequest,\n EventsPushRequest,\n EventsPushResponse,\n} from \"../sqlite-crdt/crdt-sync-remote-source\";\nimport type { CrdtEventType } from \"../sqlite-crdt/crdt-table-schema\";\nimport type { ExecuteParams, ExecuteResult } from \"../sqlite-db-wrapper\";\nimport { TypedBroadcastChannel } from \"../utils\";\n\nexport const syncDbWorkerLockName = \"sync-db-worker-lock\";\nexport const syncDbClientLockName = \"sync-db-client-lock\";\n\nexport type WorkerNotificationMessage =\n | {\n notificationType: \"new-event-chunk-applied\";\n newSyncId: number;\n /** Worker's HLC checksum after applying up to `newSyncId`. */\n eventHlcSum: string | null;\n }\n | {\n notificationType: \"state-changed\";\n state: WorkerState;\n }\n | {\n notificationType: \"reload-requested\";\n reloadEpoch: string;\n clean: boolean;\n }\n | {\n notificationType: \"de-sync-detected\";\n reason: DeSyncDetectedReason;\n }\n | {\n notificationType: \"remote-schema-version-mismatch\";\n remoteSchemaVersion: number;\n localSchemaVersion: number;\n };\n\nexport type WorkerState = {\n remoteState: \"online\" | \"offline\" | \"pending\";\n};\n\nexport type GetSnapshotResponse = {\n file: Uint8Array<ArrayBufferLike>;\n syncId: number;\n schemaVersion: number;\n};\n\nexport type EventsPullResponse = {\n events: {\n schema_version: number;\n type: CrdtEventType;\n timestamp: string;\n dataset: string;\n item_id: string;\n payload: string;\n }[];\n hasMore: boolean;\n nextSyncId: number;\n};\n\nexport interface WorkerRpc {\n getSnapshot: () => GetSnapshotResponse;\n pushTabEvents: (request: EventsPushRequest) => EventsPushResponse;\n execute: (query: ExecuteParams) => ExecuteResult<unknown>;\n pullEvents: (params: EventsPullRequest) => EventsPullResponse;\n postState: () => void;\n goOnline: () => Promise<void>;\n goOffline: () => void;\n requestReload: (options: { clean: boolean }) => Promise<void>;\n}\n\nexport type WorkerRequestMethod = keyof WorkerRpc;\n\nexport type WorkerRequestMessage<TMethod extends WorkerRequestMethod = WorkerRequestMethod> = {\n type: \"request\";\n requestId: string;\n method: TMethod;\n args: Parameters<WorkerRpc[TMethod]>;\n};\n\nexport type WorkerResponseMessage<TMethod extends WorkerRequestMethod = WorkerRequestMethod> = {\n type: \"response\";\n requestId: string;\n data: ReturnType<WorkerRpc[TMethod]>;\n};\n\nexport type WorkerErrorResponseMessage = {\n type: \"error-response\";\n requestId: string;\n error: string;\n};\n\nexport type AsyncRpc<T> = {\n [K in keyof T]: T[K] extends (...args: infer U) => infer V ? (...args: U) => Promise<Awaited<V>> : never;\n};\n\nexport const broadcastChannelNames = {\n requests: \"sync-db-worker-requests\",\n responses: \"sync-db-worker-responses\",\n} as const;\n\nexport type WorkerBroadcastChannels = {\n requests: TypedBroadcastChannel<WorkerRequestMessage>;\n responses: TypedBroadcastChannel<WorkerResponseMessage | WorkerErrorResponseMessage | WorkerNotificationMessage>;\n};\n\nexport const createBroadcastChannels = (prefix: string): WorkerBroadcastChannels => {\n return {\n requests: new TypedBroadcastChannel(`${prefix}-${broadcastChannelNames.requests}`),\n responses: new TypedBroadcastChannel(`${prefix}-${broadcastChannelNames.responses}`),\n };\n};\n\nexport type WorkerConfig<Props = any> = {\n dbId: string;\n clientId: string;\n props: Props;\n};\n\nexport type WorkerInitMessage = {\n type: \"init\";\n config: WorkerConfig;\n};\n\nexport function isWorkerInitMessage(message: unknown): message is WorkerInitMessage {\n return typeof message === \"object\" && message !== null && \"type\" in message && message.type === \"init\";\n}\n\nexport function isWorkerRequestMessage(message: unknown): message is WorkerRequestMessage {\n return typeof message === \"object\" && message !== null && \"type\" in message && message.type === \"request\";\n}\n\nexport function isWorkerResponseMessage(message: unknown): message is WorkerResponseMessage {\n return (\n typeof message === \"object\" && message !== null && \"type\" in message && \"requestId\" in message && \"data\" in message\n );\n}\n\nexport function isWorkerErrorResponseMessage(message: unknown): message is WorkerErrorResponseMessage {\n return typeof message === \"object\" && message !== null && \"type\" in message && message.type === \"error-response\";\n}\n\nexport function isWorkerNotificationMessage(message: unknown): message is WorkerNotificationMessage {\n return typeof message === \"object\" && message !== null && \"notificationType\" in message && !!message.notificationType;\n}\n"],"mappings":"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","names":["db","sql","sql","perf","sql","sql","sql","get","db","set","sql","db","sql","db","sql"]}
|
package/dist/{crdt-sync-remote-source-Na9Uicu6.d.ts → crdt-sync-remote-source-Da77s4k0.d.ts}
RENAMED
|
@@ -169,7 +169,7 @@ type ParsedTableName = {
|
|
|
169
169
|
fullIdentifier: string;
|
|
170
170
|
};
|
|
171
171
|
|
|
172
|
-
type CrdtEventType = "item-created" | "item-updated";
|
|
172
|
+
type CrdtEventType = "item-created" | "item-updated" | "item-deleted";
|
|
173
173
|
type CrdtEventStatus = "pending" | "applied" | "failed" | "deduped";
|
|
174
174
|
/** Persisted on applied events that were accepted but did not mutate materialized state. */
|
|
175
175
|
declare const CRDT_EVENT_NO_OP_PAYLOAD = "no-op";
|
package/dist/index.d.ts
CHANGED
|
@@ -1,9 +1,9 @@
|
|
|
1
1
|
import * as kysely from 'kysely';
|
|
2
2
|
import { Kysely, SchemaModule } from 'kysely';
|
|
3
|
-
import { S as SQLiteDbWrapper, L as Logger, T as TypedEvent, I as InternalSQLiteWrapper, a as StoredValue, C as CrdtStorage, E as ExecuteParams, b as ExecuteResult, Q as QueryBuilderOutput, K as KyselyQueryFactory, c as SQLiteTransactionWrapper, W as WorkerState, D as DeSyncDetectedReason } from './crdt-sync-remote-source-
|
|
4
|
-
export { R as CRDT_EVENT_NO_OP_PAYLOAD, U as CrdtEventOrigin, V as CrdtEventStatus, X as CrdtEventType, F as CrdtSyncRemoteSource, Y as CrdtUpdateLogItem, Z as CrdtUpdateLogPayload, g as DatabaseIntrospection, a5 as DeferredPromise, a6 as DistributiveOmit, J as EventsPullRequest, N as EventsPushRequest, O as EventsPushResponse, H as HLC, f as HLCCounter, a1 as KyselyStatementFactory, j as LogLevel, M as MigratableEvent, n as Migrations, o as MigrationsDb, P as PendingCrdtEvent, $ as PersistedCrdtEvent, a2 as PreparedStatement, p as SyncDbMigrator, w as SystemDbConfig, x as SystemMigration, y as SystemMigrationContext, h as TableMetadata, aa as TypedBroadcastChannel, ab as TypedEventTarget, ae as WorkerConfig, q as applyMemoryDbSchema, r as applyWorkerDbSchema, t as baseSystemMigrations, d as compareHLC, z as createCrdtApplyFunction, B as createCrdtStorage, G as createCrdtSyncRemoteSource, a3 as createDeferredPromise, l as createMigrations, m as createMigrator, A as createSQLiteCrdtApplyFunction, a0 as createStoredValue, u as createSystemDbConfig, a4 as createTypedEventTarget, e as deserializeHLC, a7 as generateId, i as introspectDb, _ as isNoOpCrdtEventPayload, a8 as jsonSafeParse, a9 as quoteId, v as runSystemMigrations, s as serializeHLC, k as startPerformanceLogger, ac as tryCatch, ad as tryCatchAsync } from './crdt-sync-remote-source-
|
|
5
|
-
import { S as SyncDbSchema, C as CrdtTableConfig } from './reset-state-
|
|
6
|
-
export { a as CreateCrdtSchemaOptions, R as RESET_REQUEST_TTL_MS, b as ResetRequest, d as ResetStore, c as createSyncDbSchema } from './reset-state-
|
|
3
|
+
import { S as SQLiteDbWrapper, L as Logger, T as TypedEvent, I as InternalSQLiteWrapper, a as StoredValue, C as CrdtStorage, E as ExecuteParams, b as ExecuteResult, Q as QueryBuilderOutput, K as KyselyQueryFactory, c as SQLiteTransactionWrapper, W as WorkerState, D as DeSyncDetectedReason } from './crdt-sync-remote-source-Da77s4k0.js';
|
|
4
|
+
export { R as CRDT_EVENT_NO_OP_PAYLOAD, U as CrdtEventOrigin, V as CrdtEventStatus, X as CrdtEventType, F as CrdtSyncRemoteSource, Y as CrdtUpdateLogItem, Z as CrdtUpdateLogPayload, g as DatabaseIntrospection, a5 as DeferredPromise, a6 as DistributiveOmit, J as EventsPullRequest, N as EventsPushRequest, O as EventsPushResponse, H as HLC, f as HLCCounter, a1 as KyselyStatementFactory, j as LogLevel, M as MigratableEvent, n as Migrations, o as MigrationsDb, P as PendingCrdtEvent, $ as PersistedCrdtEvent, a2 as PreparedStatement, p as SyncDbMigrator, w as SystemDbConfig, x as SystemMigration, y as SystemMigrationContext, h as TableMetadata, aa as TypedBroadcastChannel, ab as TypedEventTarget, ae as WorkerConfig, q as applyMemoryDbSchema, r as applyWorkerDbSchema, t as baseSystemMigrations, d as compareHLC, z as createCrdtApplyFunction, B as createCrdtStorage, G as createCrdtSyncRemoteSource, a3 as createDeferredPromise, l as createMigrations, m as createMigrator, A as createSQLiteCrdtApplyFunction, a0 as createStoredValue, u as createSystemDbConfig, a4 as createTypedEventTarget, e as deserializeHLC, a7 as generateId, i as introspectDb, _ as isNoOpCrdtEventPayload, a8 as jsonSafeParse, a9 as quoteId, v as runSystemMigrations, s as serializeHLC, k as startPerformanceLogger, ac as tryCatch, ad as tryCatchAsync } from './crdt-sync-remote-source-Da77s4k0.js';
|
|
5
|
+
import { S as SyncDbSchema, C as CrdtTableConfig } from './reset-state-0LGwO78x.js';
|
|
6
|
+
export { a as CreateCrdtSchemaOptions, R as RESET_REQUEST_TTL_MS, b as ResetRequest, d as ResetStore, c as createSyncDbSchema } from './reset-state-0LGwO78x.js';
|
|
7
7
|
import '@sqlite.org/sqlite-wasm';
|
|
8
8
|
|
|
9
9
|
declare const dummyKysely: Kysely<any>;
|
package/dist/index.js
CHANGED
|
@@ -32,7 +32,7 @@ import {
|
|
|
32
32
|
startPerformanceLogger,
|
|
33
33
|
syncDbClientLockName,
|
|
34
34
|
xxhash
|
|
35
|
-
} from "./chunk-
|
|
35
|
+
} from "./chunk-RKBTBPNC.js";
|
|
36
36
|
import {
|
|
37
37
|
TypedBroadcastChannel,
|
|
38
38
|
TypedEvent,
|
|
@@ -399,10 +399,10 @@ function createCrdtStorageMutator({ storage }) {
|
|
|
399
399
|
};
|
|
400
400
|
case "item-deleted":
|
|
401
401
|
return {
|
|
402
|
-
type: "item-
|
|
402
|
+
type: "item-deleted",
|
|
403
403
|
dataset: event.dataset,
|
|
404
404
|
item_id: event.item_id,
|
|
405
|
-
payload:
|
|
405
|
+
payload: "{}"
|
|
406
406
|
};
|
|
407
407
|
}
|
|
408
408
|
};
|
|
@@ -588,10 +588,10 @@ function registerCrdtFunctions({
|
|
|
588
588
|
callback: (dataset, itemId) => {
|
|
589
589
|
storage.applyOwnEvent(
|
|
590
590
|
{
|
|
591
|
-
type: "item-
|
|
591
|
+
type: "item-deleted",
|
|
592
592
|
dataset,
|
|
593
593
|
item_id: itemId,
|
|
594
|
-
payload:
|
|
594
|
+
payload: "{}"
|
|
595
595
|
},
|
|
596
596
|
{
|
|
597
597
|
wrapInTransaction: false
|
package/dist/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/memory-db/sqlite-reactive-db.ts","../src/bound-map.ts","../src/sqlite-crdt/crdt-schema.ts","../src/sqlite-crdt/crdt-storage-mutator.ts","../src/sqlite-crdt/make-crdt-table.ts","../src/db-id.ts","../src/devtools-registry.ts","../src/memory-db/memory-db.ts","../src/worker-db/db-worker-client.ts","../src/sync-db.ts"],"sourcesContent":["import sqlite3InitModule, { type Sqlite3Static } from \"@sqlite.org/sqlite-wasm\";\nimport { BoundMap } from \"../bound-map\";\nimport { type Logger, startPerformanceLogger } from \"../logger\";\nimport { type PreparedStatement, SQLiteDbWrapper } from \"../sqlite-db-wrapper\";\nimport { createTypedEventTarget, type TypedEvent } from \"../utils\";\n\nlet sqliteModule: Sqlite3Static | null = null;\n\ntype TableName<Database> = keyof Database extends string ? keyof Database : never;\n\ntype SQLiteReactiveDbOptions = {\n snapshot: Uint8Array<ArrayBufferLike>;\n logger: Logger;\n};\n\ntype EventsMap = {\n \"transaction-committed\": undefined;\n \"transaction-rolled-back\": undefined;\n \"any-table-changed\": undefined;\n} & Record<`table:${string}`, void>;\n\nexport function createSQLiteReactiveDb<Database>(opts: SQLiteReactiveDbOptions) {\n return SQLiteReactiveDb.create<Database>(opts);\n}\n\nexport class SQLiteReactiveDb<Database> {\n readonly db: SQLiteDbWrapper<Database>;\n private sqlite3: Sqlite3Static;\n\n private readonly logger: Logger;\n\n private tablesUsedStatement: PreparedStatement<[string], { name: string; isWrite: boolean }> | null = null;\n\n private eventTarget = createTypedEventTarget<EventsMap>();\n\n private constructor(sqlite3: Sqlite3Static, logger: Logger) {\n this.sqlite3 = sqlite3;\n this.logger = logger;\n\n this.db = new SQLiteDbWrapper({\n db: () => new sqlite3.oo1.DB({ filename: \":memory:\" }),\n logger: this.logger,\n loggerPrefix: \"memory\",\n sqlite3,\n });\n }\n\n static async create<Database>(opts: SQLiteReactiveDbOptions) {\n const logger = opts.logger;\n const perf = startPerformanceLogger(logger);\n if (!sqliteModule) {\n sqliteModule = await sqlite3InitModule();\n }\n\n const db = new SQLiteReactiveDb<Database>(sqliteModule, logger);\n\n if (opts.snapshot) {\n db.useSnapshot(opts.snapshot);\n }\n db.registerDbHooks();\n\n perf.logEnd(\"createSQLiteMemoryDb\", \"success\", \"system\");\n\n return db;\n }\n\n private liveQueryStatements = new BoundMap<string, PreparedStatement<any[], unknown>>({\n maxSize: 100,\n onRemove(_, value) {\n value.finalize();\n },\n });\n\n createLiveQuery<TResult>(query: { sql: string; parameters: readonly unknown[] }) {\n const fetchRows = (parameters: readonly unknown[]) => {\n let statement = this.liveQueryStatements.get(query.sql);\n if (!statement) {\n statement = this.db.prepare<any[], any>(query.sql);\n this.liveQueryStatements.set(query.sql, statement);\n }\n return statement.execute(parameters as any) as TResult[];\n };\n\n let rows: TResult[] | null = null;\n\n const getRows = () => {\n if (!rows) {\n rows = fetchRows(query.parameters);\n }\n return rows;\n };\n\n let subscriber: (() => void) | null = null;\n\n let lastParameters: readonly unknown[] = query.parameters;\n const refresh = (parameters?: readonly unknown[]) => {\n if (parameters) {\n lastParameters = parameters;\n }\n rows = fetchRows(lastParameters);\n subscriber?.();\n };\n\n const subscribe = (onchange: () => void) => {\n if (subscriber) {\n throw new Error(\"Subscriber already exists\");\n }\n\n subscriber = onchange;\n const subscription = this.subscribeToQueryChanges({\n sql: query.sql,\n onDataChange: refresh,\n });\n\n return () => {\n subscription.unsubscribe();\n subscriber = null;\n };\n };\n\n return { getRows, refresh, subscribe };\n }\n\n subscribeToQueryChanges(params: { sql: string; onDataChange: () => void }) {\n const { sql, onDataChange } = params;\n\n const tables = this.getTablesUsed(sql);\n const readTables = new Set<string>();\n for (const table of tables) {\n if (!readTables.has(table.name)) {\n readTables.add(table.name);\n } else if (table.isWrite) {\n throw new Error(\"This query writes and reads from the same table. This may cause infinite loops.\");\n }\n }\n\n const notifyDataChange = createDebouncedCallback(() => {\n onDataChange();\n }, 30);\n\n for (const table of readTables) {\n this.eventTarget.addEventListener(`table:${table}`, notifyDataChange);\n }\n this.eventTarget.addEventListener(\"any-table-changed\", notifyDataChange);\n\n return {\n unsubscribe: () => {\n for (const table of readTables) {\n this.eventTarget.removeEventListener(`table:${table}`, notifyDataChange);\n }\n this.eventTarget.removeEventListener(\"any-table-changed\", notifyDataChange);\n },\n };\n }\n\n subscribeToTableChanges(table: string, onChanges: () => void) {\n this.eventTarget.addEventListener(`table:${table}`, onChanges);\n this.eventTarget.addEventListener(\"any-table-changed\", onChanges);\n return {\n unsubscribe: () => {\n this.eventTarget.removeEventListener(`table:${table}`, onChanges);\n this.eventTarget.removeEventListener(\"any-table-changed\", onChanges);\n },\n };\n }\n\n getTablesUsed(query: string) {\n if (!this.tablesUsedStatement) {\n this.tablesUsedStatement = this.db.prepare<[string], { name: string; isWrite: boolean }>(\n \"select t.tbl_name as name, u.wr as isWrite from tables_used(?) as u inner join sqlite_master as t on t.name = u.name where u.schema = 'main'\",\n { loggerLevel: \"system\" },\n );\n }\n\n const tables = this.tablesUsedStatement.execute([query]);\n\n if (tables.length === 0 && query.toLowerCase().includes(\"delete\")) {\n // tables_used function does not work with delete queries that clear entire tables\n tables.push(...this.getClearedTables(query));\n }\n\n return tables;\n }\n\n private getClearedTables(query: string) {\n const operations = this.db.execute<{\n opcode: string;\n p1: number;\n p2: number;\n }>(`EXPLAIN ${query.split(\";\")[0]}`, { loggerLevel: \"system\" }).rows;\n\n const clearedTablesRootPages = new Set<number>();\n for (const operation of operations) {\n if (operation.opcode === \"Clear\" && operation.p2 === 0) {\n clearedTablesRootPages.add(operation.p1);\n }\n }\n\n if (clearedTablesRootPages.size === 0) {\n return [];\n }\n\n const tableNames = this.db.execute<{ name: string; isWrite: boolean }>(\n `select t.tbl_name as name, true as isWrite from sqlite_master as t where t.rootpage in (${Array.from(\n clearedTablesRootPages,\n ).join(\",\")})`,\n { loggerLevel: \"system\" },\n ).rows;\n\n return tableNames;\n }\n\n addEventListener<K extends keyof EventsMap>(type: K, listener: (event: TypedEvent<EventsMap[K]>) => void) {\n this.eventTarget.addEventListener(type, listener);\n }\n\n removeEventListener<K extends keyof EventsMap>(type: K, listener: (event: TypedEvent<EventsMap[K]>) => void) {\n this.eventTarget.removeEventListener(type, listener);\n }\n\n private notifyTableSubscribers(tables: (TableName<Database> | (string & {}))[] | Set<string> | null = null) {\n if (!tables) {\n this.eventTarget.dispatchEvent(\"any-table-changed\", undefined);\n return;\n }\n\n for (const table of tables) {\n this.eventTarget.dispatchEvent(`table:${table}`, undefined);\n }\n }\n\n private registerDbHooks() {\n let updateQueue = new Set<string>();\n\n this.sqlite3.capi.sqlite3_update_hook(\n this.db.ensureDb,\n (_ctx, _opId, _db, table) => {\n updateQueue.add(table);\n },\n 0,\n );\n\n this.sqlite3.capi.sqlite3_rollback_hook(\n this.db.ensureDb,\n () => {\n if (updateQueue.size === 0) {\n return 0;\n }\n\n updateQueue.clear();\n this.eventTarget.dispatchEvent(\"transaction-rolled-back\", undefined);\n\n return 0;\n },\n 0,\n );\n\n this.sqlite3.capi.sqlite3_commit_hook(\n this.db.ensureDb,\n () => {\n if (updateQueue.size === 0) {\n return 0;\n }\n\n const tables = updateQueue;\n updateQueue = new Set<string>();\n this.eventTarget.dispatchEvent(\"transaction-committed\", undefined);\n\n queueMicrotask(() => {\n this.notifyTableSubscribers(tables);\n });\n return 0;\n },\n 0,\n );\n }\n\n createSnapshot() {\n const perf = startPerformanceLogger(this.logger);\n const snapshot = this.sqlite3.capi.sqlite3_js_db_export(this.db.ensureDb);\n perf.logEnd(\"createSnapshot\", `snapshot size: ${snapshot.byteLength}`, \"info\");\n\n return snapshot;\n }\n\n useSnapshot(snapshot: Uint8Array<ArrayBufferLike>) {\n this.db.useSnapshot(snapshot);\n this.notifyTableSubscribers();\n }\n\n dispose() {\n this.liveQueryStatements.clear();\n if (this.tablesUsedStatement) {\n this.tablesUsedStatement.finalize();\n this.tablesUsedStatement = null;\n }\n this.db.close();\n }\n}\n\nfunction createDebouncedCallback<TArgs extends unknown[]>(callback: (...args: TArgs) => void, delay: number) {\n let timeout: unknown | null = null;\n let shouldCallWithoutDelay = true;\n\n return (...args: TArgs) => {\n if (shouldCallWithoutDelay) {\n callback(...args);\n shouldCallWithoutDelay = false;\n return;\n }\n\n const effect = () => {\n timeout = null;\n shouldCallWithoutDelay = true;\n return callback(...args);\n };\n\n if (timeout) {\n clearTimeout(timeout as any);\n }\n\n timeout = setTimeout(effect, delay);\n };\n}\n","interface BoundMapOptions<K, V> {\n maxSize: number;\n onRemove: ((key: K, value: V) => void) | undefined;\n}\n\nexport class BoundMap<K, V> {\n private map = new Map<K, V>();\n private maxSize: number;\n private onRemove: ((key: K, value: V) => void) | undefined;\n\n constructor(opts: BoundMapOptions<K, V>) {\n this.maxSize = opts.maxSize;\n this.onRemove = opts.onRemove;\n }\n\n set = (key: K, value: V) => {\n if (this.onRemove && this.map.has(key)) {\n const old = this.map.get(key) as V;\n this.map.set(key, value);\n this.onRemove(key, old);\n } else {\n this.map.set(key, value);\n }\n if (this.map.size > this.maxSize) {\n const firstKey = this.map.keys().next().value as K;\n this.delete(firstKey);\n }\n };\n\n get = (key: K): V | undefined => {\n return this.map.get(key);\n };\n\n delete = (key: K) => {\n if (this.onRemove && this.map.has(key)) {\n const value = this.map.get(key) as V;\n this.map.delete(key);\n this.onRemove(key, value);\n } else {\n this.map.delete(key);\n }\n };\n\n clear = () => {\n const onRemove = this.onRemove;\n if (onRemove) {\n this.map.forEach((value, key) => {\n onRemove(key, value);\n });\n }\n this.map.clear();\n };\n}\n","import type { ColumnType } from \"kysely\";\nimport type { Migrations } from \"../migrations/migrator\";\n\nexport type CrdtTableConfig = {\n baseTableName: string;\n crdtTableName: string;\n};\n\nexport function createSyncDbSchema({ migrations }: { migrations: Migrations }) {\n return new CrdtSchemaBuilder({ tables: [], migrations });\n}\n\nexport interface CreateCrdtSchemaOptions {\n tables: CrdtTableConfig[];\n migrations: Migrations;\n}\n\n// biome-ignore lint/complexity/noBannedTypes: required generic\nclass CrdtSchemaBuilder<ClientDB = {}, ServerDB = {}, MutationsDB = {}>\n implements SyncDbSchema<ClientDB, ServerDB, MutationsDB>\n{\n constructor(private config: CreateCrdtSchemaOptions) {}\n\n get tablesConfig() {\n return this.config.tables;\n }\n\n get migrations() {\n return this.config.migrations;\n }\n\n get \"~clientSchema\"() {\n console.warn(\"~clientSchema should not be accessed on the client\");\n return null as any;\n }\n\n get \"~serverSchema\"() {\n console.warn(\"~serverSchema should not be accessed on the server\");\n return null as any;\n }\n\n get \"~mutationsSchema\"() {\n console.warn(\"~mutationsSchema should not be accessed on the client\");\n return null as any;\n }\n\n addTable<Table extends Record<string, unknown>>() {\n const withConfig = <const CrdtTable extends string, const BaseTable extends string>({\n baseTableName,\n crdtTableName,\n }: {\n baseTableName: BaseTable;\n crdtTableName: CrdtTable;\n }) => {\n this.config.tables.push({ baseTableName, crdtTableName });\n return new CrdtSchemaBuilder<\n ClientDB & { [K in CrdtTable]: Table } & { [K in BaseTable]: ReadonlyTable<Table> },\n ServerDB & { [K in BaseTable]: ReadonlyTable<Table> },\n MutationsDB & { [K in BaseTable]: Table }\n >(this.config);\n };\n\n return { withConfig };\n }\n\n build() {\n return this as SyncDbSchema<ClientDB, ServerDB, MutationsDB>;\n }\n}\n\n// biome-ignore lint/complexity/noBannedTypes: required generic\nexport interface SyncDbSchema<ClientDB = {}, ServerDB = {}, MutationsDB = {}> {\n get tablesConfig(): CrdtTableConfig[];\n get migrations(): Migrations;\n \"~clientSchema\": ClientDB;\n \"~serverSchema\": ServerDB;\n \"~mutationsSchema\": MutationsDB;\n}\n\ntype ReadonlyTable<Table extends Record<string, unknown>> = {\n [K in keyof Table]: ColumnType<Table[K], never, never>;\n};\n","import type { CrdtStorage, OwnCrdtEvent } from \"./crdt-storage\";\n\nexport type CrdtStorageMutator<Database> = ReturnType<typeof createCrdtStorageMutator<Database>>;\n\ntype CommitEventOptions<Database, Table extends keyof Database & string> =\n | {\n type: \"item-created\";\n dataset: Table;\n item_id: string;\n payload: CreateEventPayload<Database, Table>;\n }\n | {\n type: \"item-updated\";\n dataset: Table;\n item_id: string;\n payload: UpdateEventPayload<Database, Table>;\n }\n | {\n type: \"item-deleted\";\n dataset: Table;\n item_id: string;\n };\n\ntype CreateEventPayload<Database, Table extends keyof Database> = Omit<Database[Table], \"tombstone\">;\ntype UpdateEventPayload<Database, Table extends keyof Database> = Omit<Partial<Database[Table]>, \"id\" | \"tombstone\">;\n\nexport function createCrdtStorageMutator<Database>({ storage }: { storage: CrdtStorage }) {\n const mapToStorageEvent = (event: CommitEventOptions<Database, keyof Database & string>): OwnCrdtEvent => {\n switch (event.type) {\n case \"item-created\":\n return {\n type: \"item-created\",\n dataset: event.dataset,\n item_id: event.item_id,\n payload: JSON.stringify(event.payload),\n };\n case \"item-updated\":\n return {\n type: \"item-updated\",\n dataset: event.dataset,\n item_id: event.item_id,\n payload: JSON.stringify(event.payload),\n };\n case \"item-deleted\":\n return {\n type: \"item-updated\",\n dataset: event.dataset,\n item_id: event.item_id,\n payload: JSON.stringify({ tombstone: 1 }),\n };\n }\n };\n\n const enqueueEvents = (events: CommitEventOptions<Database, keyof Database & string>[]) => {\n storage.enqueueOwnEvents(events.map(mapToStorageEvent));\n };\n\n const createEvent = <Table extends keyof Database & string>(event: CommitEventOptions<Database, Table>) => {\n return event;\n };\n\n const enqueueEvent = (event: CommitEventOptions<Database, keyof Database & string>) => {\n storage.enqueueOwnEvents([mapToStorageEvent(event)]);\n };\n\n return {\n enqueueEvents,\n createEvent,\n enqueueEvent,\n };\n}\n","import type { SQLiteReactiveDb } from \"../memory-db/sqlite-reactive-db\";\nimport type { SQLiteDbWrapper } from \"../sqlite-db-wrapper\";\nimport { quoteId } from \"../utils\";\nimport type { CrdtStorage } from \"./crdt-storage\";\n\nexport function makeCrdtTable({\n db,\n baseTableName,\n crdtTableName,\n}: {\n db: SQLiteDbWrapper<any>;\n baseTableName: string;\n crdtTableName: string;\n}) {\n const tableSchema = db.dbSchema[baseTableName];\n\n if (!tableSchema) {\n throw new Error(`Table ${baseTableName} not found`);\n }\n\n const columns = new Map(tableSchema.columns.map((c) => [c.name, c]));\n\n const idColumn = columns.get(\"id\");\n if (!idColumn) {\n throw new Error(\n `Table \"${baseTableName}\" is missing a required \"id\" column. CRDT tables must have an \"id\" column to identify items.`,\n );\n }\n if (idColumn.dataType.toUpperCase() !== \"TEXT\") {\n throw new Error(\n `Table \"${baseTableName}\": \"id\" column must be of type TEXT, got \"${idColumn.dataType}\". CRDT item IDs are stored as strings.`,\n );\n }\n\n const tombstoneColumn = columns.get(\"tombstone\");\n if (!tombstoneColumn) {\n throw new Error(\n `Table \"${baseTableName}\" is missing a required \"tombstone\" column. CRDT tables must have a \"tombstone\" INTEGER column for soft deletes.`,\n );\n }\n const tombstoneType = tombstoneColumn.dataType.toUpperCase();\n if (tombstoneType !== \"INTEGER\" && tombstoneType !== \"BOOLEAN\") {\n throw new Error(\n `Table \"${baseTableName}\": \"tombstone\" column must be of type INTEGER or BOOLEAN, got \"${tombstoneColumn.dataType}\". It is compared as 0/1 for soft deletes.`,\n );\n }\n\n db.execute(\n `\ncreate view ${quoteId(crdtTableName)} as\nselect * from ${quoteId(baseTableName)}\nwhere tombstone = 0;`,\n { loggerLevel: \"system\" },\n );\n\n const allColumnNames = tableSchema.columns.map((column) => column.name);\n\n const jsonPayload = (from: \"new\" | \"old\") =>\n `'{'||${allColumnNames.map((col) => `'\"${col}\":'||json_quote(${from}.${quoteId(col)})`).join(\"||','||\")}||'}'`;\n\n db.execute(\n `\ncreate trigger ${quoteId(`${crdtTableName}_created`)}\ninstead of insert on ${quoteId(crdtTableName)}\nfor each row\nbegin\nselect handle_item_created('${baseTableName}', ${jsonPayload(\"new\")});\nend;\n`,\n { loggerLevel: \"system\" },\n );\n\n db.execute(\n `\ncreate trigger ${quoteId(`${crdtTableName}_updated`)}\ninstead of update on ${quoteId(crdtTableName)}\nfor each row\nbegin\nselect handle_item_updated(\n '${baseTableName}',\n ${jsonPayload(\"old\")},\n ${jsonPayload(\"new\")}\n);\nend;\n`,\n { loggerLevel: \"system\" },\n );\n\n db.execute(\n `\ncreate trigger ${quoteId(`${crdtTableName}_deleted`)}\ninstead of delete on ${quoteId(crdtTableName)}\nfor each row\nwhen old.tombstone = 0\nbegin\nselect handle_item_deleted('${baseTableName}', old.id);\nend;\n`,\n { loggerLevel: \"system\" },\n );\n}\n\nexport function registerCrdtFunctions({\n reactiveDb,\n storage,\n}: {\n reactiveDb: SQLiteReactiveDb<any>;\n storage: CrdtStorage;\n}) {\n let eventApplied = false;\n\n reactiveDb.db.createScalarFunction({\n name: \"handle_item_created\",\n deterministic: false,\n directOnly: false,\n innocuous: false,\n callback: (dataset: string, payloadRaw: string) => {\n const payload = JSON.parse(payloadRaw) as { id: string };\n\n storage.applyOwnEvent(\n {\n type: \"item-created\",\n dataset,\n item_id: payload.id,\n payload: payloadRaw,\n },\n {\n wrapInTransaction: false,\n },\n );\n\n eventApplied = true;\n return undefined;\n },\n });\n\n reactiveDb.db.createScalarFunction({\n name: \"handle_item_updated\",\n deterministic: false,\n directOnly: false,\n innocuous: false,\n callback: (dataset: string, oldPayloadRaw: string, newPayloadRaw: string) => {\n if (oldPayloadRaw === newPayloadRaw) {\n return undefined;\n }\n\n const tableSchema = reactiveDb.db.dbSchema[dataset];\n\n if (!tableSchema?.columns) {\n throw new Error(`Schema not found for dataset: ${dataset}`);\n }\n\n const oldPayload = JSON.parse(oldPayloadRaw);\n const newPayload = JSON.parse(newPayloadRaw);\n\n let hasDiff = false;\n const updatePayload = {} as Record<string, unknown>;\n\n for (const column of tableSchema.columns) {\n const oldValue = oldPayload[column.name];\n const newValue = newPayload[column.name];\n if (oldValue === newValue) {\n continue;\n }\n\n if (column.name === \"id\") {\n throw new Error(\n `Cannot update the \"id\" column of an item. It is used to identify the item and must be immutable.`,\n );\n }\n\n hasDiff = true;\n updatePayload[column.name] = newValue;\n }\n\n if (!hasDiff) {\n return;\n }\n\n storage.applyOwnEvent(\n {\n type: \"item-updated\",\n dataset,\n item_id: oldPayload.id,\n payload: JSON.stringify(updatePayload),\n },\n {\n wrapInTransaction: false,\n },\n );\n\n eventApplied = true;\n return undefined;\n },\n });\n\n reactiveDb.db.createScalarFunction({\n name: \"handle_item_deleted\",\n deterministic: false,\n directOnly: false,\n innocuous: false,\n callback: (dataset: string, itemId: string) => {\n storage.applyOwnEvent(\n {\n type: \"item-updated\",\n dataset,\n item_id: itemId,\n payload: JSON.stringify({ tombstone: 1 }),\n },\n {\n wrapInTransaction: false,\n },\n );\n\n eventApplied = true;\n return undefined;\n },\n });\n\n reactiveDb.addEventListener(\"transaction-committed\", () => {\n if (eventApplied) {\n eventApplied = false;\n storage.dispatchEventsApplied();\n }\n });\n\n reactiveDb.addEventListener(\"transaction-rolled-back\", () => {\n eventApplied = false;\n });\n}\n","const dbIdRegex = /^[a-zA-Z][a-zA-Z\\-0-9]{2,63}$/;\n\nexport function validateDbId(dbId: string) {\n if (!dbIdRegex.test(dbId)) {\n throw new Error(\"Invalid dbId. Must be between 3 and 64 characters long and start with a letter.\");\n }\n}\n","import type { SyncedDb } from \"./sync-db\";\n\nconst devtoolsRegistrySymbol = Symbol.for(\"@sqlite-sync/devtools\");\n\nexport type SQLiteSyncDevtoolsInstance = {\n instanceId: string;\n dbId: string;\n createdAt: number;\n instance: SyncedDb<any>;\n};\n\nexport type SQLiteSyncDevtoolsSnapshot = {\n instances: readonly SQLiteSyncDevtoolsInstance[];\n};\n\nexport type SQLiteSyncDevtoolsRegistry = {\n version: 1;\n instances: Map<string, SQLiteSyncDevtoolsInstance>;\n subscribe(listener: () => void): () => void;\n getSnapshot(): SQLiteSyncDevtoolsSnapshot;\n register(instance: SQLiteSyncDevtoolsInstance): () => void;\n};\n\ntype RegistryGlobal = typeof globalThis & {\n [key: symbol]: SQLiteSyncDevtoolsRegistry | undefined;\n};\n\nfunction createSQLiteSyncDevtoolsRegistry(): SQLiteSyncDevtoolsRegistry {\n const instances = new Map<string, SQLiteSyncDevtoolsInstance>();\n const listeners = new Set<() => void>();\n let snapshot: SQLiteSyncDevtoolsSnapshot = {\n instances: [],\n };\n\n const notify = () => {\n for (const listener of listeners) {\n listener();\n }\n };\n\n const updateSnapshot = () => {\n snapshot = {\n instances: Array.from(instances.values()),\n };\n notify();\n };\n\n return {\n version: 1,\n instances,\n subscribe(listener) {\n listeners.add(listener);\n return () => {\n listeners.delete(listener);\n };\n },\n getSnapshot() {\n return snapshot;\n },\n register(instance) {\n instances.set(instance.instanceId, instance);\n updateSnapshot();\n\n let isUnregistered = false;\n return () => {\n if (isUnregistered) return;\n isUnregistered = true;\n\n if (!instances.delete(instance.instanceId)) return;\n updateSnapshot();\n };\n },\n };\n}\n\nexport function getOrCreateSQLiteSyncDevtoolsRegistry(): SQLiteSyncDevtoolsRegistry {\n const registryGlobal = globalThis as RegistryGlobal;\n\n registryGlobal[devtoolsRegistrySymbol] ??= createSQLiteSyncDevtoolsRegistry();\n\n return registryGlobal[devtoolsRegistrySymbol];\n}\n","import { xxhash } from \"../hash\";\nimport type { HLCCounter } from \"../hlc\";\nimport type { SyncDbMigrator } from \"../migrations/migrator\";\nimport { applyMemoryDbSchema, type MemoryDbSchema, memoryDbConfig } from \"../migrations/system-schema\";\nimport type { CrdtTableConfig } from \"../sqlite-crdt/crdt-schema\";\nimport { createCrdtStorage } from \"../sqlite-crdt/crdt-storage\";\nimport { makeCrdtTable, registerCrdtFunctions } from \"../sqlite-crdt/make-crdt-table\";\nimport type { StoredValue } from \"../sqlite-crdt/stored-value\";\nimport type { SQLiteDbWrapper } from \"../sqlite-db-wrapper\";\nimport type { SQLiteReactiveDb } from \"./sqlite-reactive-db\";\n\ntype MemoryDbOptions<Database> = {\n nodeId: string;\n migrator: SyncDbMigrator;\n reactiveDb: SQLiteReactiveDb<Database>;\n hlcCounter: HLCCounter;\n crdtTables: CrdtTableConfig[];\n initializeSchema?: boolean;\n initialSyncId?: number;\n eventHlcAccumulator?: StoredValue<string>;\n};\n\nexport async function createMemoryDb<Database>({\n nodeId,\n migrator,\n reactiveDb: _reactiveDb,\n hlcCounter,\n crdtTables,\n initializeSchema = true,\n initialSyncId,\n eventHlcAccumulator,\n}: MemoryDbOptions<Database>) {\n await xxhash.ensureLoaded();\n\n const reactiveDb = _reactiveDb as unknown as SQLiteReactiveDb<MemoryDbSchema>;\n const db = reactiveDb.db;\n\n if (initializeSchema) {\n applyMemoryDbSchema(db);\n for (const table of crdtTables) {\n makeCrdtTable({\n db,\n baseTableName: table.baseTableName,\n crdtTableName: table.crdtTableName,\n });\n }\n }\n\n const crdtStorage = createCrdtStorage({\n nodeId,\n initialLocalSyncId: initialSyncId ?? getCurrentSyncId(db),\n hlc: hlcCounter,\n migrator,\n db,\n dbConfig: memoryDbConfig,\n eventHlcAccumulator,\n });\n\n registerCrdtFunctions({\n reactiveDb,\n storage: crdtStorage,\n });\n\n return {\n crdtStorage,\n };\n}\n\nfunction getCurrentSyncId(db: SQLiteDbWrapper<MemoryDbSchema>) {\n return (\n db.execute<{ syncId: number }>(\"SELECT coalesce(max(sync_id), 0) AS syncId FROM persisted_crdt_events\", {\n loggerLevel: \"system\",\n }).rows[0]?.syncId ?? 0\n );\n}\n","import {\n createDeferredPromise,\n createTypedEventTarget,\n type DeferredPromise,\n noop,\n type TypedEventTarget,\n} from \"../utils\";\nimport type {\n AsyncRpc,\n WorkerBroadcastChannels,\n WorkerConfig,\n WorkerErrorResponseMessage,\n WorkerInitMessage,\n WorkerNotificationMessage,\n WorkerRequestMessage,\n WorkerRequestMethod,\n WorkerResponseMessage,\n WorkerRpc,\n WorkerState,\n} from \"./worker-common\";\nimport { isWorkerErrorResponseMessage, isWorkerNotificationMessage, isWorkerResponseMessage } from \"./worker-common\";\n\ntype NotificationEvents = {\n [K in WorkerNotificationMessage[\"notificationType\"]]: Extract<WorkerNotificationMessage, { notificationType: K }>;\n};\n\nexport const createWorkerDbClient = async ({\n broadcastChannels,\n worker,\n config,\n}: {\n broadcastChannels: WorkerBroadcastChannels;\n worker: Worker;\n config: WorkerConfig;\n}) => {\n const eventTarget = createTypedEventTarget<NotificationEvents>();\n const workerRequestsMap = new Map<string, DeferredPromise<unknown>>();\n let isDisposed = false;\n\n const queryWorker = <TMethod extends WorkerRequestMethod>(\n method: TMethod,\n args: Parameters<WorkerRpc[TMethod]>,\n ): Promise<Awaited<ReturnType<WorkerRpc[TMethod]>>> => {\n if (isDisposed) {\n return Promise.reject(new Error(\"Worker client disposed\"));\n }\n const requestId = crypto.randomUUID();\n const promise = createDeferredPromise<unknown>({\n timeout: 30_000,\n onTimeout: () => workerRequestsMap.delete(requestId),\n });\n workerRequestsMap.set(requestId, promise);\n\n const request: WorkerRequestMessage<TMethod> = {\n type: \"request\",\n requestId,\n method,\n args,\n };\n\n broadcastChannels.requests.postMessage(request);\n\n return promise.promise as Promise<Awaited<ReturnType<WorkerRpc[TMethod]>>>;\n };\n\n const handleWorkerResponse = (message: WorkerResponseMessage) => {\n const promise = workerRequestsMap.get(message.requestId);\n if (!promise) {\n return;\n }\n\n promise.resolve(message.data);\n workerRequestsMap.delete(message.requestId);\n };\n\n const handleWorkerError = (message: WorkerErrorResponseMessage) => {\n const promise = workerRequestsMap.get(message.requestId);\n if (!promise) {\n return;\n }\n\n promise.reject(new Error(message.error));\n workerRequestsMap.delete(message.requestId);\n };\n\n broadcastChannels.responses.onmessage = (event) => {\n const message = event.data;\n\n if (isWorkerResponseMessage(message)) {\n handleWorkerResponse(message);\n } else if (isWorkerErrorResponseMessage(message)) {\n handleWorkerError(message);\n } else if (isWorkerNotificationMessage(message)) {\n eventTarget.dispatchEvent(message.notificationType, message);\n }\n };\n\n const rpc: AsyncRpc<WorkerRpc> = {\n execute: (query) => queryWorker(\"execute\", [query]),\n getSnapshot: () => queryWorker(\"getSnapshot\", []),\n pushTabEvents: (request) => queryWorker(\"pushTabEvents\", [request]),\n pullEvents: (params) => queryWorker(\"pullEvents\", [params]),\n postState: () => queryWorker(\"postState\", []),\n goOnline: () => queryWorker(\"goOnline\", []),\n goOffline: () => queryWorker(\"goOffline\", []),\n requestReload: (options) => queryWorker(\"requestReload\", [options]),\n };\n\n const statePromise = awaitWorkerState(eventTarget);\n postWorkerConfig(worker, config);\n // On first tab, the worker may not have its BroadcastChannel listener ready yet,\n // so this request can be silently dropped and timeout. The worker will independently\n // post state at the end of init. For subsequent tabs, this triggers the already-running\n // worker to send its current state.\n rpc.postState().catch(noop);\n\n let workerState = await statePromise;\n\n eventTarget.addEventListener(\"state-changed\", (event) => {\n workerState = event.payload.state;\n });\n\n const dispose = () => {\n isDisposed = true;\n broadcastChannels.responses.onmessage = null;\n for (const [id, deferred] of workerRequestsMap) {\n deferred.reject(new Error(\"Worker client disposed\"));\n workerRequestsMap.delete(id);\n }\n };\n\n return {\n ...rpc,\n subscribe: eventTarget.addEventListener,\n getState: () => workerState,\n dispose,\n };\n};\n\nfunction awaitWorkerState(eventTarget: TypedEventTarget<NotificationEvents>) {\n const promise = createDeferredPromise<WorkerState>({ timeout: 15_000 });\n\n const subscription = eventTarget.addEventListener(\"state-changed\", (event) => {\n promise.resolve(event.payload.state);\n subscription.unsubscribe();\n });\n\n return promise.promise;\n}\n\nfunction postWorkerConfig(worker: Worker, config: WorkerConfig) {\n const configMessage: WorkerInitMessage = {\n type: \"init\",\n config,\n };\n worker.postMessage(configMessage);\n}\n","import { validateDbId } from \"./db-id\";\nimport { getOrCreateSQLiteSyncDevtoolsRegistry } from \"./devtools-registry\";\nimport { HLCCounter } from \"./hlc\";\nimport { type Logger, startPerformanceLogger } from \"./logger\";\nimport { createMemoryDb } from \"./memory-db/memory-db\";\nimport { createSQLiteReactiveDb } from \"./memory-db/sqlite-reactive-db\";\nimport type { SyncDbMigrator } from \"./migrations/migrator\";\nimport type { SyncDbSchema } from \"./sqlite-crdt/crdt-schema\";\nimport { createCrdtSyncRemoteSource } from \"./sqlite-crdt/crdt-sync-remote-source\";\nimport { createStoredValue } from \"./sqlite-crdt/stored-value\";\nimport { createDeferredPromise, generateId } from \"./utils\";\nimport { createWorkerDbClient } from \"./worker-db/db-worker-client\";\nimport { createBroadcastChannels, syncDbClientLockName } from \"./worker-db/worker-common\";\n\ntype SyncedDbOptions<Database, Props = undefined> = {\n dbId: string;\n worker: Worker;\n workerProps: Props;\n syncDbSchema: SyncDbSchema<Database>;\n};\n\nconst defaultLogger: Logger = (type, message, level = \"info\") => {\n const logMessage = `[${type}] ${message}`;\n switch (level) {\n case \"info\":\n console.log(logMessage);\n break;\n case \"warning\":\n console.warn(logMessage);\n break;\n case \"error\":\n console.error(logMessage);\n break;\n case \"trace\":\n console.trace(logMessage);\n break;\n }\n};\n\nexport async function createSyncedDb<Database, Props = undefined>(options: SyncedDbOptions<Database, Props>) {\n validateDbId(options.dbId);\n\n const perf = startPerformanceLogger(defaultLogger);\n\n const instanceId = generateId();\n const tabId = generateId();\n\n const broadcastChannels = createBroadcastChannels(options.dbId);\n\n const clientLockAcquired = createDeferredPromise<void>();\n const clientLockRelease = createDeferredPromise<void>();\n navigator.locks.request(`${syncDbClientLockName}-${options.dbId}`, { mode: \"shared\" }, () => {\n clientLockAcquired.resolve();\n return clientLockRelease.promise;\n });\n await clientLockAcquired.promise;\n\n const workerClient = await createWorkerDbClient({\n worker: options.worker,\n config: {\n clientId: generateId(),\n dbId: options.dbId,\n props: options.workerProps as never,\n },\n broadcastChannels,\n });\n\n const hlcCounter = new HLCCounter(tabId, () => Date.now());\n\n const workerClientSnapshot = await workerClient.getSnapshot();\n const reactiveDb = await createSQLiteReactiveDb<Database>({\n snapshot: workerClientSnapshot.file,\n logger: defaultLogger,\n });\n\n const memoryDbMigrator: SyncDbMigrator = {\n currentSchemaVersion: workerClientSnapshot.schemaVersion,\n latestSchemaVersion: workerClientSnapshot.schemaVersion,\n migrateDbToLatest: () => {\n throw new Error(\"Memory DB migrations are not implemented\");\n },\n migrateEvent: (event, targetVersion) => {\n if (event.schema_version === targetVersion) {\n return event;\n }\n throw new Error(\"Memory DB migrations are not implemented\");\n },\n migrateEvents: (events) => events,\n };\n const { crdtStorage } = await createMemoryDb({\n nodeId: tabId,\n migrator: memoryDbMigrator,\n reactiveDb: reactiveDb,\n hlcCounter,\n crdtTables: options.syncDbSchema.tablesConfig,\n });\n\n const pullSyncId = createStoredValue({\n initialValue: workerClientSnapshot.syncId,\n });\n const pushSyncId = createStoredValue({\n initialValue: 0,\n });\n const tabRemoteSource = createCrdtSyncRemoteSource({\n bufferSize: 500,\n pullSyncId,\n pushSyncId,\n storage: crdtStorage,\n nodeId: tabId,\n migrator: memoryDbMigrator,\n remoteFactory: ({ onEventsAvailable }) => {\n const subscription = workerClient.subscribe(\"new-event-chunk-applied\", (event) => {\n onEventsAvailable({ newSyncId: event.payload.newSyncId, remoteEventHlcSum: event.payload.eventHlcSum });\n });\n\n return {\n pullEvents: (request) => workerClient.pullEvents(request),\n pushEvents: (request) => workerClient.pushTabEvents(request),\n disconnect: () => {\n subscription.unsubscribe();\n },\n };\n },\n });\n tabRemoteSource.goOnline();\n\n const reloadRequestedSubscription = workerClient.subscribe(\"reload-requested\", () => {\n globalThis.location?.reload();\n });\n\n perf.logEnd(\"createSyncedDb\", \"initialized\", \"info\");\n\n let isDisposed = false;\n let unregisterDevtools: (() => void) | undefined;\n const dispose = async () => {\n if (isDisposed) return;\n isDisposed = true;\n\n unregisterDevtools?.();\n reloadRequestedSubscription.unsubscribe();\n clientLockRelease.resolve();\n await tabRemoteSource.dispose();\n broadcastChannels.requests.close();\n broadcastChannels.responses.close();\n workerClient.dispose();\n reactiveDb.dispose();\n };\n\n const syncedDb = {\n db: {\n execute: reactiveDb.db.execute.bind(reactiveDb.db),\n executeKysely: reactiveDb.db.executeKysely.bind(reactiveDb.db),\n executeTransaction: reactiveDb.db.executeTransaction.bind(reactiveDb.db),\n createLiveQuery: reactiveDb.createLiveQuery.bind(reactiveDb),\n },\n state: {\n getState: workerClient.getState.bind(workerClient),\n subscribe: (onChange: () => void) => {\n const { unsubscribe } = workerClient.subscribe(\"state-changed\", onChange);\n return unsubscribe;\n },\n goOnline: workerClient.goOnline.bind(workerClient),\n goOffline: workerClient.goOffline.bind(workerClient),\n },\n /**\n * Ask the elected worker to broadcast a page reload to all tabs for this dbId.\n *\n * With `clean: true` the worker durably records a reset request epoch before\n * broadcasting, so the worker elected on the next startup initializes with\n * `clearOnInit: true` and wipes the persisted DB. Destructive — use as a\n * recovery path when the durable worker DB may be de-synced.\n *\n * Pending in-memory tab events are not preserved, and the returned promise\n * may never settle in the caller — the page typically unloads first.\n */\n requestReload: async (options: { clean: boolean }) => {\n await workerClient.requestReload(options);\n // Primary path: this tab receives the worker's \"reload-requested\" broadcast\n // like every other tab. Fallback in case the broadcast is missed — in the\n // normal case the page is already unloading and this timeout never fires.\n setTimeout(() => globalThis.location?.reload(), 250);\n },\n subscribe: workerClient.subscribe,\n dispose,\n _internal: {\n executeAsync: workerClient.execute.bind(workerClient),\n getMemoryQueryTables: reactiveDb.getTablesUsed.bind(reactiveDb),\n crdtTableNames: options.syncDbSchema.tablesConfig.map((table) => table.crdtTableName),\n crdtTablesConfig: options.syncDbSchema.tablesConfig,\n schemaVersion: workerClientSnapshot.schemaVersion,\n migrationVersions: Object.keys(options.syncDbSchema.migrations)\n .map(Number)\n .sort((a, b) => a - b),\n },\n };\n\n unregisterDevtools = getOrCreateSQLiteSyncDevtoolsRegistry().register({\n instanceId,\n dbId: options.dbId,\n createdAt: Date.now(),\n instance: syncedDb,\n });\n\n return syncedDb;\n}\n\nexport type SyncedDb<Database> = Awaited<ReturnType<typeof createSyncedDb<Database>>>;\n"],"mappings":"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","names":["options"]}
|
|
1
|
+
{"version":3,"sources":["../src/memory-db/sqlite-reactive-db.ts","../src/bound-map.ts","../src/sqlite-crdt/crdt-schema.ts","../src/sqlite-crdt/crdt-storage-mutator.ts","../src/sqlite-crdt/make-crdt-table.ts","../src/db-id.ts","../src/devtools-registry.ts","../src/memory-db/memory-db.ts","../src/worker-db/db-worker-client.ts","../src/sync-db.ts"],"sourcesContent":["import sqlite3InitModule, { type Sqlite3Static } from \"@sqlite.org/sqlite-wasm\";\nimport { BoundMap } from \"../bound-map\";\nimport { type Logger, startPerformanceLogger } from \"../logger\";\nimport { type PreparedStatement, SQLiteDbWrapper } from \"../sqlite-db-wrapper\";\nimport { createTypedEventTarget, type TypedEvent } from \"../utils\";\n\nlet sqliteModule: Sqlite3Static | null = null;\n\ntype TableName<Database> = keyof Database extends string ? keyof Database : never;\n\ntype SQLiteReactiveDbOptions = {\n snapshot: Uint8Array<ArrayBufferLike>;\n logger: Logger;\n};\n\ntype EventsMap = {\n \"transaction-committed\": undefined;\n \"transaction-rolled-back\": undefined;\n \"any-table-changed\": undefined;\n} & Record<`table:${string}`, void>;\n\nexport function createSQLiteReactiveDb<Database>(opts: SQLiteReactiveDbOptions) {\n return SQLiteReactiveDb.create<Database>(opts);\n}\n\nexport class SQLiteReactiveDb<Database> {\n readonly db: SQLiteDbWrapper<Database>;\n private sqlite3: Sqlite3Static;\n\n private readonly logger: Logger;\n\n private tablesUsedStatement: PreparedStatement<[string], { name: string; isWrite: boolean }> | null = null;\n\n private eventTarget = createTypedEventTarget<EventsMap>();\n\n private constructor(sqlite3: Sqlite3Static, logger: Logger) {\n this.sqlite3 = sqlite3;\n this.logger = logger;\n\n this.db = new SQLiteDbWrapper({\n db: () => new sqlite3.oo1.DB({ filename: \":memory:\" }),\n logger: this.logger,\n loggerPrefix: \"memory\",\n sqlite3,\n });\n }\n\n static async create<Database>(opts: SQLiteReactiveDbOptions) {\n const logger = opts.logger;\n const perf = startPerformanceLogger(logger);\n if (!sqliteModule) {\n sqliteModule = await sqlite3InitModule();\n }\n\n const db = new SQLiteReactiveDb<Database>(sqliteModule, logger);\n\n if (opts.snapshot) {\n db.useSnapshot(opts.snapshot);\n }\n db.registerDbHooks();\n\n perf.logEnd(\"createSQLiteMemoryDb\", \"success\", \"system\");\n\n return db;\n }\n\n private liveQueryStatements = new BoundMap<string, PreparedStatement<any[], unknown>>({\n maxSize: 100,\n onRemove(_, value) {\n value.finalize();\n },\n });\n\n createLiveQuery<TResult>(query: { sql: string; parameters: readonly unknown[] }) {\n const fetchRows = (parameters: readonly unknown[]) => {\n let statement = this.liveQueryStatements.get(query.sql);\n if (!statement) {\n statement = this.db.prepare<any[], any>(query.sql);\n this.liveQueryStatements.set(query.sql, statement);\n }\n return statement.execute(parameters as any) as TResult[];\n };\n\n let rows: TResult[] | null = null;\n\n const getRows = () => {\n if (!rows) {\n rows = fetchRows(query.parameters);\n }\n return rows;\n };\n\n let subscriber: (() => void) | null = null;\n\n let lastParameters: readonly unknown[] = query.parameters;\n const refresh = (parameters?: readonly unknown[]) => {\n if (parameters) {\n lastParameters = parameters;\n }\n rows = fetchRows(lastParameters);\n subscriber?.();\n };\n\n const subscribe = (onchange: () => void) => {\n if (subscriber) {\n throw new Error(\"Subscriber already exists\");\n }\n\n subscriber = onchange;\n const subscription = this.subscribeToQueryChanges({\n sql: query.sql,\n onDataChange: refresh,\n });\n\n return () => {\n subscription.unsubscribe();\n subscriber = null;\n };\n };\n\n return { getRows, refresh, subscribe };\n }\n\n subscribeToQueryChanges(params: { sql: string; onDataChange: () => void }) {\n const { sql, onDataChange } = params;\n\n const tables = this.getTablesUsed(sql);\n const readTables = new Set<string>();\n for (const table of tables) {\n if (!readTables.has(table.name)) {\n readTables.add(table.name);\n } else if (table.isWrite) {\n throw new Error(\"This query writes and reads from the same table. This may cause infinite loops.\");\n }\n }\n\n const notifyDataChange = createDebouncedCallback(() => {\n onDataChange();\n }, 30);\n\n for (const table of readTables) {\n this.eventTarget.addEventListener(`table:${table}`, notifyDataChange);\n }\n this.eventTarget.addEventListener(\"any-table-changed\", notifyDataChange);\n\n return {\n unsubscribe: () => {\n for (const table of readTables) {\n this.eventTarget.removeEventListener(`table:${table}`, notifyDataChange);\n }\n this.eventTarget.removeEventListener(\"any-table-changed\", notifyDataChange);\n },\n };\n }\n\n subscribeToTableChanges(table: string, onChanges: () => void) {\n this.eventTarget.addEventListener(`table:${table}`, onChanges);\n this.eventTarget.addEventListener(\"any-table-changed\", onChanges);\n return {\n unsubscribe: () => {\n this.eventTarget.removeEventListener(`table:${table}`, onChanges);\n this.eventTarget.removeEventListener(\"any-table-changed\", onChanges);\n },\n };\n }\n\n getTablesUsed(query: string) {\n if (!this.tablesUsedStatement) {\n this.tablesUsedStatement = this.db.prepare<[string], { name: string; isWrite: boolean }>(\n \"select t.tbl_name as name, u.wr as isWrite from tables_used(?) as u inner join sqlite_master as t on t.name = u.name where u.schema = 'main'\",\n { loggerLevel: \"system\" },\n );\n }\n\n const tables = this.tablesUsedStatement.execute([query]);\n\n if (tables.length === 0 && query.toLowerCase().includes(\"delete\")) {\n // tables_used function does not work with delete queries that clear entire tables\n tables.push(...this.getClearedTables(query));\n }\n\n return tables;\n }\n\n private getClearedTables(query: string) {\n const operations = this.db.execute<{\n opcode: string;\n p1: number;\n p2: number;\n }>(`EXPLAIN ${query.split(\";\")[0]}`, { loggerLevel: \"system\" }).rows;\n\n const clearedTablesRootPages = new Set<number>();\n for (const operation of operations) {\n if (operation.opcode === \"Clear\" && operation.p2 === 0) {\n clearedTablesRootPages.add(operation.p1);\n }\n }\n\n if (clearedTablesRootPages.size === 0) {\n return [];\n }\n\n const tableNames = this.db.execute<{ name: string; isWrite: boolean }>(\n `select t.tbl_name as name, true as isWrite from sqlite_master as t where t.rootpage in (${Array.from(\n clearedTablesRootPages,\n ).join(\",\")})`,\n { loggerLevel: \"system\" },\n ).rows;\n\n return tableNames;\n }\n\n addEventListener<K extends keyof EventsMap>(type: K, listener: (event: TypedEvent<EventsMap[K]>) => void) {\n this.eventTarget.addEventListener(type, listener);\n }\n\n removeEventListener<K extends keyof EventsMap>(type: K, listener: (event: TypedEvent<EventsMap[K]>) => void) {\n this.eventTarget.removeEventListener(type, listener);\n }\n\n private notifyTableSubscribers(tables: (TableName<Database> | (string & {}))[] | Set<string> | null = null) {\n if (!tables) {\n this.eventTarget.dispatchEvent(\"any-table-changed\", undefined);\n return;\n }\n\n for (const table of tables) {\n this.eventTarget.dispatchEvent(`table:${table}`, undefined);\n }\n }\n\n private registerDbHooks() {\n let updateQueue = new Set<string>();\n\n this.sqlite3.capi.sqlite3_update_hook(\n this.db.ensureDb,\n (_ctx, _opId, _db, table) => {\n updateQueue.add(table);\n },\n 0,\n );\n\n this.sqlite3.capi.sqlite3_rollback_hook(\n this.db.ensureDb,\n () => {\n if (updateQueue.size === 0) {\n return 0;\n }\n\n updateQueue.clear();\n this.eventTarget.dispatchEvent(\"transaction-rolled-back\", undefined);\n\n return 0;\n },\n 0,\n );\n\n this.sqlite3.capi.sqlite3_commit_hook(\n this.db.ensureDb,\n () => {\n if (updateQueue.size === 0) {\n return 0;\n }\n\n const tables = updateQueue;\n updateQueue = new Set<string>();\n this.eventTarget.dispatchEvent(\"transaction-committed\", undefined);\n\n queueMicrotask(() => {\n this.notifyTableSubscribers(tables);\n });\n return 0;\n },\n 0,\n );\n }\n\n createSnapshot() {\n const perf = startPerformanceLogger(this.logger);\n const snapshot = this.sqlite3.capi.sqlite3_js_db_export(this.db.ensureDb);\n perf.logEnd(\"createSnapshot\", `snapshot size: ${snapshot.byteLength}`, \"info\");\n\n return snapshot;\n }\n\n useSnapshot(snapshot: Uint8Array<ArrayBufferLike>) {\n this.db.useSnapshot(snapshot);\n this.notifyTableSubscribers();\n }\n\n dispose() {\n this.liveQueryStatements.clear();\n if (this.tablesUsedStatement) {\n this.tablesUsedStatement.finalize();\n this.tablesUsedStatement = null;\n }\n this.db.close();\n }\n}\n\nfunction createDebouncedCallback<TArgs extends unknown[]>(callback: (...args: TArgs) => void, delay: number) {\n let timeout: unknown | null = null;\n let shouldCallWithoutDelay = true;\n\n return (...args: TArgs) => {\n if (shouldCallWithoutDelay) {\n callback(...args);\n shouldCallWithoutDelay = false;\n return;\n }\n\n const effect = () => {\n timeout = null;\n shouldCallWithoutDelay = true;\n return callback(...args);\n };\n\n if (timeout) {\n clearTimeout(timeout as any);\n }\n\n timeout = setTimeout(effect, delay);\n };\n}\n","interface BoundMapOptions<K, V> {\n maxSize: number;\n onRemove: ((key: K, value: V) => void) | undefined;\n}\n\nexport class BoundMap<K, V> {\n private map = new Map<K, V>();\n private maxSize: number;\n private onRemove: ((key: K, value: V) => void) | undefined;\n\n constructor(opts: BoundMapOptions<K, V>) {\n this.maxSize = opts.maxSize;\n this.onRemove = opts.onRemove;\n }\n\n set = (key: K, value: V) => {\n if (this.onRemove && this.map.has(key)) {\n const old = this.map.get(key) as V;\n this.map.set(key, value);\n this.onRemove(key, old);\n } else {\n this.map.set(key, value);\n }\n if (this.map.size > this.maxSize) {\n const firstKey = this.map.keys().next().value as K;\n this.delete(firstKey);\n }\n };\n\n get = (key: K): V | undefined => {\n return this.map.get(key);\n };\n\n delete = (key: K) => {\n if (this.onRemove && this.map.has(key)) {\n const value = this.map.get(key) as V;\n this.map.delete(key);\n this.onRemove(key, value);\n } else {\n this.map.delete(key);\n }\n };\n\n clear = () => {\n const onRemove = this.onRemove;\n if (onRemove) {\n this.map.forEach((value, key) => {\n onRemove(key, value);\n });\n }\n this.map.clear();\n };\n}\n","import type { ColumnType } from \"kysely\";\nimport type { Migrations } from \"../migrations/migrator\";\n\nexport type CrdtTableConfig = {\n baseTableName: string;\n crdtTableName: string;\n};\n\nexport function createSyncDbSchema({ migrations }: { migrations: Migrations }) {\n return new CrdtSchemaBuilder({ tables: [], migrations });\n}\n\nexport interface CreateCrdtSchemaOptions {\n tables: CrdtTableConfig[];\n migrations: Migrations;\n}\n\n// biome-ignore lint/complexity/noBannedTypes: required generic\nclass CrdtSchemaBuilder<ClientDB = {}, ServerDB = {}, MutationsDB = {}>\n implements SyncDbSchema<ClientDB, ServerDB, MutationsDB>\n{\n constructor(private config: CreateCrdtSchemaOptions) {}\n\n get tablesConfig() {\n return this.config.tables;\n }\n\n get migrations() {\n return this.config.migrations;\n }\n\n get \"~clientSchema\"() {\n console.warn(\"~clientSchema should not be accessed on the client\");\n return null as any;\n }\n\n get \"~serverSchema\"() {\n console.warn(\"~serverSchema should not be accessed on the server\");\n return null as any;\n }\n\n get \"~mutationsSchema\"() {\n console.warn(\"~mutationsSchema should not be accessed on the client\");\n return null as any;\n }\n\n addTable<Table extends Record<string, unknown>>() {\n const withConfig = <const CrdtTable extends string, const BaseTable extends string>({\n baseTableName,\n crdtTableName,\n }: {\n baseTableName: BaseTable;\n crdtTableName: CrdtTable;\n }) => {\n this.config.tables.push({ baseTableName, crdtTableName });\n return new CrdtSchemaBuilder<\n ClientDB & { [K in CrdtTable]: Table } & { [K in BaseTable]: ReadonlyTable<Table> },\n ServerDB & { [K in BaseTable]: ReadonlyTable<Table> },\n MutationsDB & { [K in BaseTable]: Table }\n >(this.config);\n };\n\n return { withConfig };\n }\n\n build() {\n return this as SyncDbSchema<ClientDB, ServerDB, MutationsDB>;\n }\n}\n\n// biome-ignore lint/complexity/noBannedTypes: required generic\nexport interface SyncDbSchema<ClientDB = {}, ServerDB = {}, MutationsDB = {}> {\n get tablesConfig(): CrdtTableConfig[];\n get migrations(): Migrations;\n \"~clientSchema\": ClientDB;\n \"~serverSchema\": ServerDB;\n \"~mutationsSchema\": MutationsDB;\n}\n\ntype ReadonlyTable<Table extends Record<string, unknown>> = {\n [K in keyof Table]: ColumnType<Table[K], never, never>;\n};\n","import type { CrdtStorage, OwnCrdtEvent } from \"./crdt-storage\";\n\nexport type CrdtStorageMutator<Database> = ReturnType<typeof createCrdtStorageMutator<Database>>;\n\ntype CommitEventOptions<Database, Table extends keyof Database & string> =\n | {\n type: \"item-created\";\n dataset: Table;\n item_id: string;\n payload: CreateEventPayload<Database, Table>;\n }\n | {\n type: \"item-updated\";\n dataset: Table;\n item_id: string;\n payload: UpdateEventPayload<Database, Table>;\n }\n | {\n type: \"item-deleted\";\n dataset: Table;\n item_id: string;\n };\n\ntype CreateEventPayload<Database, Table extends keyof Database> = Omit<Database[Table], \"tombstone\">;\ntype UpdateEventPayload<Database, Table extends keyof Database> = Omit<Partial<Database[Table]>, \"id\" | \"tombstone\">;\n\nexport function createCrdtStorageMutator<Database>({ storage }: { storage: CrdtStorage }) {\n const mapToStorageEvent = (event: CommitEventOptions<Database, keyof Database & string>): OwnCrdtEvent => {\n switch (event.type) {\n case \"item-created\":\n return {\n type: \"item-created\",\n dataset: event.dataset,\n item_id: event.item_id,\n payload: JSON.stringify(event.payload),\n };\n case \"item-updated\":\n return {\n type: \"item-updated\",\n dataset: event.dataset,\n item_id: event.item_id,\n payload: JSON.stringify(event.payload),\n };\n case \"item-deleted\":\n return {\n type: \"item-deleted\",\n dataset: event.dataset,\n item_id: event.item_id,\n payload: \"{}\",\n };\n }\n };\n\n const enqueueEvents = (events: CommitEventOptions<Database, keyof Database & string>[]) => {\n storage.enqueueOwnEvents(events.map(mapToStorageEvent));\n };\n\n const createEvent = <Table extends keyof Database & string>(event: CommitEventOptions<Database, Table>) => {\n return event;\n };\n\n const enqueueEvent = (event: CommitEventOptions<Database, keyof Database & string>) => {\n storage.enqueueOwnEvents([mapToStorageEvent(event)]);\n };\n\n return {\n enqueueEvents,\n createEvent,\n enqueueEvent,\n };\n}\n","import type { SQLiteReactiveDb } from \"../memory-db/sqlite-reactive-db\";\nimport type { SQLiteDbWrapper } from \"../sqlite-db-wrapper\";\nimport { quoteId } from \"../utils\";\nimport type { CrdtStorage } from \"./crdt-storage\";\n\nexport function makeCrdtTable({\n db,\n baseTableName,\n crdtTableName,\n}: {\n db: SQLiteDbWrapper<any>;\n baseTableName: string;\n crdtTableName: string;\n}) {\n const tableSchema = db.dbSchema[baseTableName];\n\n if (!tableSchema) {\n throw new Error(`Table ${baseTableName} not found`);\n }\n\n const columns = new Map(tableSchema.columns.map((c) => [c.name, c]));\n\n const idColumn = columns.get(\"id\");\n if (!idColumn) {\n throw new Error(\n `Table \"${baseTableName}\" is missing a required \"id\" column. CRDT tables must have an \"id\" column to identify items.`,\n );\n }\n if (idColumn.dataType.toUpperCase() !== \"TEXT\") {\n throw new Error(\n `Table \"${baseTableName}\": \"id\" column must be of type TEXT, got \"${idColumn.dataType}\". CRDT item IDs are stored as strings.`,\n );\n }\n\n const tombstoneColumn = columns.get(\"tombstone\");\n if (!tombstoneColumn) {\n throw new Error(\n `Table \"${baseTableName}\" is missing a required \"tombstone\" column. CRDT tables must have a \"tombstone\" INTEGER column for soft deletes.`,\n );\n }\n const tombstoneType = tombstoneColumn.dataType.toUpperCase();\n if (tombstoneType !== \"INTEGER\" && tombstoneType !== \"BOOLEAN\") {\n throw new Error(\n `Table \"${baseTableName}\": \"tombstone\" column must be of type INTEGER or BOOLEAN, got \"${tombstoneColumn.dataType}\". It is compared as 0/1 for soft deletes.`,\n );\n }\n\n db.execute(\n `\ncreate view ${quoteId(crdtTableName)} as\nselect * from ${quoteId(baseTableName)}\nwhere tombstone = 0;`,\n { loggerLevel: \"system\" },\n );\n\n const allColumnNames = tableSchema.columns.map((column) => column.name);\n\n const jsonPayload = (from: \"new\" | \"old\") =>\n `'{'||${allColumnNames.map((col) => `'\"${col}\":'||json_quote(${from}.${quoteId(col)})`).join(\"||','||\")}||'}'`;\n\n db.execute(\n `\ncreate trigger ${quoteId(`${crdtTableName}_created`)}\ninstead of insert on ${quoteId(crdtTableName)}\nfor each row\nbegin\nselect handle_item_created('${baseTableName}', ${jsonPayload(\"new\")});\nend;\n`,\n { loggerLevel: \"system\" },\n );\n\n db.execute(\n `\ncreate trigger ${quoteId(`${crdtTableName}_updated`)}\ninstead of update on ${quoteId(crdtTableName)}\nfor each row\nbegin\nselect handle_item_updated(\n '${baseTableName}',\n ${jsonPayload(\"old\")},\n ${jsonPayload(\"new\")}\n);\nend;\n`,\n { loggerLevel: \"system\" },\n );\n\n db.execute(\n `\ncreate trigger ${quoteId(`${crdtTableName}_deleted`)}\ninstead of delete on ${quoteId(crdtTableName)}\nfor each row\nwhen old.tombstone = 0\nbegin\nselect handle_item_deleted('${baseTableName}', old.id);\nend;\n`,\n { loggerLevel: \"system\" },\n );\n}\n\nexport function registerCrdtFunctions({\n reactiveDb,\n storage,\n}: {\n reactiveDb: SQLiteReactiveDb<any>;\n storage: CrdtStorage;\n}) {\n let eventApplied = false;\n\n reactiveDb.db.createScalarFunction({\n name: \"handle_item_created\",\n deterministic: false,\n directOnly: false,\n innocuous: false,\n callback: (dataset: string, payloadRaw: string) => {\n const payload = JSON.parse(payloadRaw) as { id: string };\n\n storage.applyOwnEvent(\n {\n type: \"item-created\",\n dataset,\n item_id: payload.id,\n payload: payloadRaw,\n },\n {\n wrapInTransaction: false,\n },\n );\n\n eventApplied = true;\n return undefined;\n },\n });\n\n reactiveDb.db.createScalarFunction({\n name: \"handle_item_updated\",\n deterministic: false,\n directOnly: false,\n innocuous: false,\n callback: (dataset: string, oldPayloadRaw: string, newPayloadRaw: string) => {\n if (oldPayloadRaw === newPayloadRaw) {\n return undefined;\n }\n\n const tableSchema = reactiveDb.db.dbSchema[dataset];\n\n if (!tableSchema?.columns) {\n throw new Error(`Schema not found for dataset: ${dataset}`);\n }\n\n const oldPayload = JSON.parse(oldPayloadRaw);\n const newPayload = JSON.parse(newPayloadRaw);\n\n let hasDiff = false;\n const updatePayload = {} as Record<string, unknown>;\n\n for (const column of tableSchema.columns) {\n const oldValue = oldPayload[column.name];\n const newValue = newPayload[column.name];\n if (oldValue === newValue) {\n continue;\n }\n\n if (column.name === \"id\") {\n throw new Error(\n `Cannot update the \"id\" column of an item. It is used to identify the item and must be immutable.`,\n );\n }\n\n hasDiff = true;\n updatePayload[column.name] = newValue;\n }\n\n if (!hasDiff) {\n return;\n }\n\n storage.applyOwnEvent(\n {\n type: \"item-updated\",\n dataset,\n item_id: oldPayload.id,\n payload: JSON.stringify(updatePayload),\n },\n {\n wrapInTransaction: false,\n },\n );\n\n eventApplied = true;\n return undefined;\n },\n });\n\n reactiveDb.db.createScalarFunction({\n name: \"handle_item_deleted\",\n deterministic: false,\n directOnly: false,\n innocuous: false,\n callback: (dataset: string, itemId: string) => {\n storage.applyOwnEvent(\n {\n type: \"item-deleted\",\n dataset,\n item_id: itemId,\n payload: \"{}\",\n },\n {\n wrapInTransaction: false,\n },\n );\n\n eventApplied = true;\n return undefined;\n },\n });\n\n reactiveDb.addEventListener(\"transaction-committed\", () => {\n if (eventApplied) {\n eventApplied = false;\n storage.dispatchEventsApplied();\n }\n });\n\n reactiveDb.addEventListener(\"transaction-rolled-back\", () => {\n eventApplied = false;\n });\n}\n","const dbIdRegex = /^[a-zA-Z][a-zA-Z\\-0-9]{2,63}$/;\n\nexport function validateDbId(dbId: string) {\n if (!dbIdRegex.test(dbId)) {\n throw new Error(\"Invalid dbId. Must be between 3 and 64 characters long and start with a letter.\");\n }\n}\n","import type { SyncedDb } from \"./sync-db\";\n\nconst devtoolsRegistrySymbol = Symbol.for(\"@sqlite-sync/devtools\");\n\nexport type SQLiteSyncDevtoolsInstance = {\n instanceId: string;\n dbId: string;\n createdAt: number;\n instance: SyncedDb<any>;\n};\n\nexport type SQLiteSyncDevtoolsSnapshot = {\n instances: readonly SQLiteSyncDevtoolsInstance[];\n};\n\nexport type SQLiteSyncDevtoolsRegistry = {\n version: 1;\n instances: Map<string, SQLiteSyncDevtoolsInstance>;\n subscribe(listener: () => void): () => void;\n getSnapshot(): SQLiteSyncDevtoolsSnapshot;\n register(instance: SQLiteSyncDevtoolsInstance): () => void;\n};\n\ntype RegistryGlobal = typeof globalThis & {\n [key: symbol]: SQLiteSyncDevtoolsRegistry | undefined;\n};\n\nfunction createSQLiteSyncDevtoolsRegistry(): SQLiteSyncDevtoolsRegistry {\n const instances = new Map<string, SQLiteSyncDevtoolsInstance>();\n const listeners = new Set<() => void>();\n let snapshot: SQLiteSyncDevtoolsSnapshot = {\n instances: [],\n };\n\n const notify = () => {\n for (const listener of listeners) {\n listener();\n }\n };\n\n const updateSnapshot = () => {\n snapshot = {\n instances: Array.from(instances.values()),\n };\n notify();\n };\n\n return {\n version: 1,\n instances,\n subscribe(listener) {\n listeners.add(listener);\n return () => {\n listeners.delete(listener);\n };\n },\n getSnapshot() {\n return snapshot;\n },\n register(instance) {\n instances.set(instance.instanceId, instance);\n updateSnapshot();\n\n let isUnregistered = false;\n return () => {\n if (isUnregistered) return;\n isUnregistered = true;\n\n if (!instances.delete(instance.instanceId)) return;\n updateSnapshot();\n };\n },\n };\n}\n\nexport function getOrCreateSQLiteSyncDevtoolsRegistry(): SQLiteSyncDevtoolsRegistry {\n const registryGlobal = globalThis as RegistryGlobal;\n\n registryGlobal[devtoolsRegistrySymbol] ??= createSQLiteSyncDevtoolsRegistry();\n\n return registryGlobal[devtoolsRegistrySymbol];\n}\n","import { xxhash } from \"../hash\";\nimport type { HLCCounter } from \"../hlc\";\nimport type { SyncDbMigrator } from \"../migrations/migrator\";\nimport { applyMemoryDbSchema, type MemoryDbSchema, memoryDbConfig } from \"../migrations/system-schema\";\nimport type { CrdtTableConfig } from \"../sqlite-crdt/crdt-schema\";\nimport { createCrdtStorage } from \"../sqlite-crdt/crdt-storage\";\nimport { makeCrdtTable, registerCrdtFunctions } from \"../sqlite-crdt/make-crdt-table\";\nimport type { StoredValue } from \"../sqlite-crdt/stored-value\";\nimport type { SQLiteDbWrapper } from \"../sqlite-db-wrapper\";\nimport type { SQLiteReactiveDb } from \"./sqlite-reactive-db\";\n\ntype MemoryDbOptions<Database> = {\n nodeId: string;\n migrator: SyncDbMigrator;\n reactiveDb: SQLiteReactiveDb<Database>;\n hlcCounter: HLCCounter;\n crdtTables: CrdtTableConfig[];\n initializeSchema?: boolean;\n initialSyncId?: number;\n eventHlcAccumulator?: StoredValue<string>;\n};\n\nexport async function createMemoryDb<Database>({\n nodeId,\n migrator,\n reactiveDb: _reactiveDb,\n hlcCounter,\n crdtTables,\n initializeSchema = true,\n initialSyncId,\n eventHlcAccumulator,\n}: MemoryDbOptions<Database>) {\n await xxhash.ensureLoaded();\n\n const reactiveDb = _reactiveDb as unknown as SQLiteReactiveDb<MemoryDbSchema>;\n const db = reactiveDb.db;\n\n if (initializeSchema) {\n applyMemoryDbSchema(db);\n for (const table of crdtTables) {\n makeCrdtTable({\n db,\n baseTableName: table.baseTableName,\n crdtTableName: table.crdtTableName,\n });\n }\n }\n\n const crdtStorage = createCrdtStorage({\n nodeId,\n initialLocalSyncId: initialSyncId ?? getCurrentSyncId(db),\n hlc: hlcCounter,\n migrator,\n db,\n dbConfig: memoryDbConfig,\n eventHlcAccumulator,\n });\n\n registerCrdtFunctions({\n reactiveDb,\n storage: crdtStorage,\n });\n\n return {\n crdtStorage,\n };\n}\n\nfunction getCurrentSyncId(db: SQLiteDbWrapper<MemoryDbSchema>) {\n return (\n db.execute<{ syncId: number }>(\"SELECT coalesce(max(sync_id), 0) AS syncId FROM persisted_crdt_events\", {\n loggerLevel: \"system\",\n }).rows[0]?.syncId ?? 0\n );\n}\n","import {\n createDeferredPromise,\n createTypedEventTarget,\n type DeferredPromise,\n noop,\n type TypedEventTarget,\n} from \"../utils\";\nimport type {\n AsyncRpc,\n WorkerBroadcastChannels,\n WorkerConfig,\n WorkerErrorResponseMessage,\n WorkerInitMessage,\n WorkerNotificationMessage,\n WorkerRequestMessage,\n WorkerRequestMethod,\n WorkerResponseMessage,\n WorkerRpc,\n WorkerState,\n} from \"./worker-common\";\nimport { isWorkerErrorResponseMessage, isWorkerNotificationMessage, isWorkerResponseMessage } from \"./worker-common\";\n\ntype NotificationEvents = {\n [K in WorkerNotificationMessage[\"notificationType\"]]: Extract<WorkerNotificationMessage, { notificationType: K }>;\n};\n\nexport const createWorkerDbClient = async ({\n broadcastChannels,\n worker,\n config,\n}: {\n broadcastChannels: WorkerBroadcastChannels;\n worker: Worker;\n config: WorkerConfig;\n}) => {\n const eventTarget = createTypedEventTarget<NotificationEvents>();\n const workerRequestsMap = new Map<string, DeferredPromise<unknown>>();\n let isDisposed = false;\n\n const queryWorker = <TMethod extends WorkerRequestMethod>(\n method: TMethod,\n args: Parameters<WorkerRpc[TMethod]>,\n ): Promise<Awaited<ReturnType<WorkerRpc[TMethod]>>> => {\n if (isDisposed) {\n return Promise.reject(new Error(\"Worker client disposed\"));\n }\n const requestId = crypto.randomUUID();\n const promise = createDeferredPromise<unknown>({\n timeout: 30_000,\n onTimeout: () => workerRequestsMap.delete(requestId),\n });\n workerRequestsMap.set(requestId, promise);\n\n const request: WorkerRequestMessage<TMethod> = {\n type: \"request\",\n requestId,\n method,\n args,\n };\n\n broadcastChannels.requests.postMessage(request);\n\n return promise.promise as Promise<Awaited<ReturnType<WorkerRpc[TMethod]>>>;\n };\n\n const handleWorkerResponse = (message: WorkerResponseMessage) => {\n const promise = workerRequestsMap.get(message.requestId);\n if (!promise) {\n return;\n }\n\n promise.resolve(message.data);\n workerRequestsMap.delete(message.requestId);\n };\n\n const handleWorkerError = (message: WorkerErrorResponseMessage) => {\n const promise = workerRequestsMap.get(message.requestId);\n if (!promise) {\n return;\n }\n\n promise.reject(new Error(message.error));\n workerRequestsMap.delete(message.requestId);\n };\n\n broadcastChannels.responses.onmessage = (event) => {\n const message = event.data;\n\n if (isWorkerResponseMessage(message)) {\n handleWorkerResponse(message);\n } else if (isWorkerErrorResponseMessage(message)) {\n handleWorkerError(message);\n } else if (isWorkerNotificationMessage(message)) {\n eventTarget.dispatchEvent(message.notificationType, message);\n }\n };\n\n const rpc: AsyncRpc<WorkerRpc> = {\n execute: (query) => queryWorker(\"execute\", [query]),\n getSnapshot: () => queryWorker(\"getSnapshot\", []),\n pushTabEvents: (request) => queryWorker(\"pushTabEvents\", [request]),\n pullEvents: (params) => queryWorker(\"pullEvents\", [params]),\n postState: () => queryWorker(\"postState\", []),\n goOnline: () => queryWorker(\"goOnline\", []),\n goOffline: () => queryWorker(\"goOffline\", []),\n requestReload: (options) => queryWorker(\"requestReload\", [options]),\n };\n\n const statePromise = awaitWorkerState(eventTarget);\n postWorkerConfig(worker, config);\n // On first tab, the worker may not have its BroadcastChannel listener ready yet,\n // so this request can be silently dropped and timeout. The worker will independently\n // post state at the end of init. For subsequent tabs, this triggers the already-running\n // worker to send its current state.\n rpc.postState().catch(noop);\n\n let workerState = await statePromise;\n\n eventTarget.addEventListener(\"state-changed\", (event) => {\n workerState = event.payload.state;\n });\n\n const dispose = () => {\n isDisposed = true;\n broadcastChannels.responses.onmessage = null;\n for (const [id, deferred] of workerRequestsMap) {\n deferred.reject(new Error(\"Worker client disposed\"));\n workerRequestsMap.delete(id);\n }\n };\n\n return {\n ...rpc,\n subscribe: eventTarget.addEventListener,\n getState: () => workerState,\n dispose,\n };\n};\n\nfunction awaitWorkerState(eventTarget: TypedEventTarget<NotificationEvents>) {\n const promise = createDeferredPromise<WorkerState>({ timeout: 15_000 });\n\n const subscription = eventTarget.addEventListener(\"state-changed\", (event) => {\n promise.resolve(event.payload.state);\n subscription.unsubscribe();\n });\n\n return promise.promise;\n}\n\nfunction postWorkerConfig(worker: Worker, config: WorkerConfig) {\n const configMessage: WorkerInitMessage = {\n type: \"init\",\n config,\n };\n worker.postMessage(configMessage);\n}\n","import { validateDbId } from \"./db-id\";\nimport { getOrCreateSQLiteSyncDevtoolsRegistry } from \"./devtools-registry\";\nimport { HLCCounter } from \"./hlc\";\nimport { type Logger, startPerformanceLogger } from \"./logger\";\nimport { createMemoryDb } from \"./memory-db/memory-db\";\nimport { createSQLiteReactiveDb } from \"./memory-db/sqlite-reactive-db\";\nimport type { SyncDbMigrator } from \"./migrations/migrator\";\nimport type { SyncDbSchema } from \"./sqlite-crdt/crdt-schema\";\nimport { createCrdtSyncRemoteSource } from \"./sqlite-crdt/crdt-sync-remote-source\";\nimport { createStoredValue } from \"./sqlite-crdt/stored-value\";\nimport { createDeferredPromise, generateId } from \"./utils\";\nimport { createWorkerDbClient } from \"./worker-db/db-worker-client\";\nimport { createBroadcastChannels, syncDbClientLockName } from \"./worker-db/worker-common\";\n\ntype SyncedDbOptions<Database, Props = undefined> = {\n dbId: string;\n worker: Worker;\n workerProps: Props;\n syncDbSchema: SyncDbSchema<Database>;\n};\n\nconst defaultLogger: Logger = (type, message, level = \"info\") => {\n const logMessage = `[${type}] ${message}`;\n switch (level) {\n case \"info\":\n console.log(logMessage);\n break;\n case \"warning\":\n console.warn(logMessage);\n break;\n case \"error\":\n console.error(logMessage);\n break;\n case \"trace\":\n console.trace(logMessage);\n break;\n }\n};\n\nexport async function createSyncedDb<Database, Props = undefined>(options: SyncedDbOptions<Database, Props>) {\n validateDbId(options.dbId);\n\n const perf = startPerformanceLogger(defaultLogger);\n\n const instanceId = generateId();\n const tabId = generateId();\n\n const broadcastChannels = createBroadcastChannels(options.dbId);\n\n const clientLockAcquired = createDeferredPromise<void>();\n const clientLockRelease = createDeferredPromise<void>();\n navigator.locks.request(`${syncDbClientLockName}-${options.dbId}`, { mode: \"shared\" }, () => {\n clientLockAcquired.resolve();\n return clientLockRelease.promise;\n });\n await clientLockAcquired.promise;\n\n const workerClient = await createWorkerDbClient({\n worker: options.worker,\n config: {\n clientId: generateId(),\n dbId: options.dbId,\n props: options.workerProps as never,\n },\n broadcastChannels,\n });\n\n const hlcCounter = new HLCCounter(tabId, () => Date.now());\n\n const workerClientSnapshot = await workerClient.getSnapshot();\n const reactiveDb = await createSQLiteReactiveDb<Database>({\n snapshot: workerClientSnapshot.file,\n logger: defaultLogger,\n });\n\n const memoryDbMigrator: SyncDbMigrator = {\n currentSchemaVersion: workerClientSnapshot.schemaVersion,\n latestSchemaVersion: workerClientSnapshot.schemaVersion,\n migrateDbToLatest: () => {\n throw new Error(\"Memory DB migrations are not implemented\");\n },\n migrateEvent: (event, targetVersion) => {\n if (event.schema_version === targetVersion) {\n return event;\n }\n throw new Error(\"Memory DB migrations are not implemented\");\n },\n migrateEvents: (events) => events,\n };\n const { crdtStorage } = await createMemoryDb({\n nodeId: tabId,\n migrator: memoryDbMigrator,\n reactiveDb: reactiveDb,\n hlcCounter,\n crdtTables: options.syncDbSchema.tablesConfig,\n });\n\n const pullSyncId = createStoredValue({\n initialValue: workerClientSnapshot.syncId,\n });\n const pushSyncId = createStoredValue({\n initialValue: 0,\n });\n const tabRemoteSource = createCrdtSyncRemoteSource({\n bufferSize: 500,\n pullSyncId,\n pushSyncId,\n storage: crdtStorage,\n nodeId: tabId,\n migrator: memoryDbMigrator,\n remoteFactory: ({ onEventsAvailable }) => {\n const subscription = workerClient.subscribe(\"new-event-chunk-applied\", (event) => {\n onEventsAvailable({ newSyncId: event.payload.newSyncId, remoteEventHlcSum: event.payload.eventHlcSum });\n });\n\n return {\n pullEvents: (request) => workerClient.pullEvents(request),\n pushEvents: (request) => workerClient.pushTabEvents(request),\n disconnect: () => {\n subscription.unsubscribe();\n },\n };\n },\n });\n tabRemoteSource.goOnline();\n\n const reloadRequestedSubscription = workerClient.subscribe(\"reload-requested\", () => {\n globalThis.location?.reload();\n });\n\n perf.logEnd(\"createSyncedDb\", \"initialized\", \"info\");\n\n let isDisposed = false;\n let unregisterDevtools: (() => void) | undefined;\n const dispose = async () => {\n if (isDisposed) return;\n isDisposed = true;\n\n unregisterDevtools?.();\n reloadRequestedSubscription.unsubscribe();\n clientLockRelease.resolve();\n await tabRemoteSource.dispose();\n broadcastChannels.requests.close();\n broadcastChannels.responses.close();\n workerClient.dispose();\n reactiveDb.dispose();\n };\n\n const syncedDb = {\n db: {\n execute: reactiveDb.db.execute.bind(reactiveDb.db),\n executeKysely: reactiveDb.db.executeKysely.bind(reactiveDb.db),\n executeTransaction: reactiveDb.db.executeTransaction.bind(reactiveDb.db),\n createLiveQuery: reactiveDb.createLiveQuery.bind(reactiveDb),\n },\n state: {\n getState: workerClient.getState.bind(workerClient),\n subscribe: (onChange: () => void) => {\n const { unsubscribe } = workerClient.subscribe(\"state-changed\", onChange);\n return unsubscribe;\n },\n goOnline: workerClient.goOnline.bind(workerClient),\n goOffline: workerClient.goOffline.bind(workerClient),\n },\n /**\n * Ask the elected worker to broadcast a page reload to all tabs for this dbId.\n *\n * With `clean: true` the worker durably records a reset request epoch before\n * broadcasting, so the worker elected on the next startup initializes with\n * `clearOnInit: true` and wipes the persisted DB. Destructive — use as a\n * recovery path when the durable worker DB may be de-synced.\n *\n * Pending in-memory tab events are not preserved, and the returned promise\n * may never settle in the caller — the page typically unloads first.\n */\n requestReload: async (options: { clean: boolean }) => {\n await workerClient.requestReload(options);\n // Primary path: this tab receives the worker's \"reload-requested\" broadcast\n // like every other tab. Fallback in case the broadcast is missed — in the\n // normal case the page is already unloading and this timeout never fires.\n setTimeout(() => globalThis.location?.reload(), 250);\n },\n subscribe: workerClient.subscribe,\n dispose,\n _internal: {\n executeAsync: workerClient.execute.bind(workerClient),\n getMemoryQueryTables: reactiveDb.getTablesUsed.bind(reactiveDb),\n crdtTableNames: options.syncDbSchema.tablesConfig.map((table) => table.crdtTableName),\n crdtTablesConfig: options.syncDbSchema.tablesConfig,\n schemaVersion: workerClientSnapshot.schemaVersion,\n migrationVersions: Object.keys(options.syncDbSchema.migrations)\n .map(Number)\n .sort((a, b) => a - b),\n },\n };\n\n unregisterDevtools = getOrCreateSQLiteSyncDevtoolsRegistry().register({\n instanceId,\n dbId: options.dbId,\n createdAt: Date.now(),\n instance: syncedDb,\n });\n\n return syncedDb;\n}\n\nexport type SyncedDb<Database> = Awaited<ReturnType<typeof createSyncedDb<Database>>>;\n"],"mappings":"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","names":["options"]}
|
package/dist/server.d.ts
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
|
-
import { ah as EventsPullResponse, O as EventsPushResponse } from './crdt-sync-remote-source-
|
|
2
|
-
export { a8 as jsonSafeParse } from './crdt-sync-remote-source-
|
|
1
|
+
import { ah as EventsPullResponse, O as EventsPushResponse } from './crdt-sync-remote-source-Da77s4k0.js';
|
|
2
|
+
export { a8 as jsonSafeParse } from './crdt-sync-remote-source-Da77s4k0.js';
|
|
3
3
|
import { z } from 'zod';
|
|
4
4
|
import 'kysely';
|
|
5
5
|
import '@sqlite.org/sqlite-wasm';
|
|
@@ -21,6 +21,7 @@ declare const syncServerZodSchema: {
|
|
|
21
21
|
type: z.ZodEnum<{
|
|
22
22
|
"item-created": "item-created";
|
|
23
23
|
"item-updated": "item-updated";
|
|
24
|
+
"item-deleted": "item-deleted";
|
|
24
25
|
}>;
|
|
25
26
|
dataset: z.ZodString;
|
|
26
27
|
item_id: z.ZodString;
|
|
@@ -42,6 +43,7 @@ declare const syncServerZodSchema: {
|
|
|
42
43
|
type: z.ZodEnum<{
|
|
43
44
|
"item-created": "item-created";
|
|
44
45
|
"item-updated": "item-updated";
|
|
46
|
+
"item-deleted": "item-deleted";
|
|
45
47
|
}>;
|
|
46
48
|
dataset: z.ZodString;
|
|
47
49
|
item_id: z.ZodString;
|
package/dist/server.js
CHANGED
|
@@ -18,7 +18,7 @@ var pushEventsZodSchema = z.object({
|
|
|
18
18
|
z.object({
|
|
19
19
|
schema_version: z.number(),
|
|
20
20
|
timestamp: z.string(),
|
|
21
|
-
type: z.enum(["item-created", "item-updated"]),
|
|
21
|
+
type: z.enum(["item-created", "item-updated", "item-deleted"]),
|
|
22
22
|
dataset: z.string(),
|
|
23
23
|
item_id: z.string(),
|
|
24
24
|
payload: z.string()
|
package/dist/server.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/server/server-common.ts"],"sourcesContent":["import { z } from \"zod\";\nimport type { EventsPushResponse } from \"../sqlite-crdt/crdt-sync-remote-source\";\nimport type { EventsPullResponse } from \"../worker-db/worker-common\";\n\nconst pullEventsZodSchema = z.object({\n type: z.literal(\"pull-events\"),\n requestId: z.string(),\n afterSyncId: z.number(),\n excludeNodeId: z.string().optional(),\n});\nconst pushEventsZodSchema = z.object({\n type: z.literal(\"push-events\"),\n requestId: z.string(),\n nodeId: z.string(),\n events: z.array(\n z.object({\n schema_version: z.number(),\n timestamp: z.string(),\n type: z.enum([\"item-created\", \"item-updated\"]),\n dataset: z.string(),\n item_id: z.string(),\n payload: z.string(),\n }),\n ),\n});\n\nexport const syncServerZodSchema = {\n pullEvents: pullEventsZodSchema,\n pushEvents: pushEventsZodSchema,\n request: z.discriminatedUnion(\"type\", [pullEventsZodSchema, pushEventsZodSchema]),\n};\n\nexport type SyncServerMessage =\n | {\n type: \"events-pull-response\";\n requestId: string;\n data: EventsPullResponse;\n }\n | {\n type: \"events-push-response\";\n requestId: string;\n data: EventsPushResponse;\n }\n | {\n type: \"events-applied\";\n newSyncId: number;\n /**\n * Remote's HLC checksum after applying up to `newSyncId`. Used by clients\n * to detect de-sync when they are caught up.\n */\n eventHlcSum: string | null;\n };\n\nexport type SyncServerRequest = z.infer<typeof syncServerZodSchema.request>;\n\nexport type ExtractSyncServerRequest<T extends SyncServerRequest[\"type\"]> = Extract<SyncServerRequest, { type: T }>;\n"],"mappings":";;;;;AAAA,SAAS,SAAS;AAIlB,IAAM,sBAAsB,EAAE,OAAO;AAAA,EACnC,MAAM,EAAE,QAAQ,aAAa;AAAA,EAC7B,WAAW,EAAE,OAAO;AAAA,EACpB,aAAa,EAAE,OAAO;AAAA,EACtB,eAAe,EAAE,OAAO,EAAE,SAAS;AACrC,CAAC;AACD,IAAM,sBAAsB,EAAE,OAAO;AAAA,EACnC,MAAM,EAAE,QAAQ,aAAa;AAAA,EAC7B,WAAW,EAAE,OAAO;AAAA,EACpB,QAAQ,EAAE,OAAO;AAAA,EACjB,QAAQ,EAAE;AAAA,IACR,EAAE,OAAO;AAAA,MACP,gBAAgB,EAAE,OAAO;AAAA,MACzB,WAAW,EAAE,OAAO;AAAA,MACpB,MAAM,EAAE,KAAK,CAAC,gBAAgB,cAAc,CAAC;AAAA,
|
|
1
|
+
{"version":3,"sources":["../src/server/server-common.ts"],"sourcesContent":["import { z } from \"zod\";\nimport type { EventsPushResponse } from \"../sqlite-crdt/crdt-sync-remote-source\";\nimport type { EventsPullResponse } from \"../worker-db/worker-common\";\n\nconst pullEventsZodSchema = z.object({\n type: z.literal(\"pull-events\"),\n requestId: z.string(),\n afterSyncId: z.number(),\n excludeNodeId: z.string().optional(),\n});\nconst pushEventsZodSchema = z.object({\n type: z.literal(\"push-events\"),\n requestId: z.string(),\n nodeId: z.string(),\n events: z.array(\n z.object({\n schema_version: z.number(),\n timestamp: z.string(),\n type: z.enum([\"item-created\", \"item-updated\", \"item-deleted\"]),\n dataset: z.string(),\n item_id: z.string(),\n payload: z.string(),\n }),\n ),\n});\n\nexport const syncServerZodSchema = {\n pullEvents: pullEventsZodSchema,\n pushEvents: pushEventsZodSchema,\n request: z.discriminatedUnion(\"type\", [pullEventsZodSchema, pushEventsZodSchema]),\n};\n\nexport type SyncServerMessage =\n | {\n type: \"events-pull-response\";\n requestId: string;\n data: EventsPullResponse;\n }\n | {\n type: \"events-push-response\";\n requestId: string;\n data: EventsPushResponse;\n }\n | {\n type: \"events-applied\";\n newSyncId: number;\n /**\n * Remote's HLC checksum after applying up to `newSyncId`. Used by clients\n * to detect de-sync when they are caught up.\n */\n eventHlcSum: string | null;\n };\n\nexport type SyncServerRequest = z.infer<typeof syncServerZodSchema.request>;\n\nexport type ExtractSyncServerRequest<T extends SyncServerRequest[\"type\"]> = Extract<SyncServerRequest, { type: T }>;\n"],"mappings":";;;;;AAAA,SAAS,SAAS;AAIlB,IAAM,sBAAsB,EAAE,OAAO;AAAA,EACnC,MAAM,EAAE,QAAQ,aAAa;AAAA,EAC7B,WAAW,EAAE,OAAO;AAAA,EACpB,aAAa,EAAE,OAAO;AAAA,EACtB,eAAe,EAAE,OAAO,EAAE,SAAS;AACrC,CAAC;AACD,IAAM,sBAAsB,EAAE,OAAO;AAAA,EACnC,MAAM,EAAE,QAAQ,aAAa;AAAA,EAC7B,WAAW,EAAE,OAAO;AAAA,EACpB,QAAQ,EAAE,OAAO;AAAA,EACjB,QAAQ,EAAE;AAAA,IACR,EAAE,OAAO;AAAA,MACP,gBAAgB,EAAE,OAAO;AAAA,MACzB,WAAW,EAAE,OAAO;AAAA,MACpB,MAAM,EAAE,KAAK,CAAC,gBAAgB,gBAAgB,cAAc,CAAC;AAAA,MAC7D,SAAS,EAAE,OAAO;AAAA,MAClB,SAAS,EAAE,OAAO;AAAA,MAClB,SAAS,EAAE,OAAO;AAAA,IACpB,CAAC;AAAA,EACH;AACF,CAAC;AAEM,IAAM,sBAAsB;AAAA,EACjC,YAAY;AAAA,EACZ,YAAY;AAAA,EACZ,SAAS,EAAE,mBAAmB,QAAQ,CAAC,qBAAqB,mBAAmB,CAAC;AAClF;","names":[]}
|
package/dist/worker.d.ts
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
|
-
import { af as CreateRemoteSourceFactory, ae as WorkerConfig, L as Logger } from './crdt-sync-remote-source-
|
|
2
|
-
export { J as EventsPullRequest, N as EventsPushRequest, O as EventsPushResponse, ag as GetEventsBatch } from './crdt-sync-remote-source-
|
|
3
|
-
import { S as SyncDbSchema, d as ResetStore } from './reset-state-
|
|
1
|
+
import { af as CreateRemoteSourceFactory, ae as WorkerConfig, L as Logger } from './crdt-sync-remote-source-Da77s4k0.js';
|
|
2
|
+
export { J as EventsPullRequest, N as EventsPushRequest, O as EventsPushResponse, ag as GetEventsBatch } from './crdt-sync-remote-source-Da77s4k0.js';
|
|
3
|
+
import { S as SyncDbSchema, d as ResetStore } from './reset-state-0LGwO78x.js';
|
|
4
4
|
import 'kysely';
|
|
5
5
|
import '@sqlite.org/sqlite-wasm';
|
|
6
6
|
|
package/dist/worker.js
CHANGED
package/package.json
CHANGED
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/dummy-kysely.ts","../src/hash.ts","../src/hlc.ts","../src/introspection.ts","../src/logger.ts","../src/sqlite-db-wrapper.ts","../src/sqlite-crdt/crdt-table-schema.ts","../src/migrations/migrator.ts","../src/sqlite-crdt/stored-value.ts","../src/sqlite-kv-store.ts","../src/migrations/system-schema.ts","../src/sqlite-crdt/apply-crdt-event.ts","../src/sqlite-crdt/crdt-storage.ts","../src/sqlite-crdt/event-consistency.ts","../src/sqlite-crdt/crdt-sync-producer.ts","../src/sqlite-crdt/crdt-sync-remote-source.ts","../src/worker-db/reset-state.ts","../src/worker-db/worker-common.ts"],"sourcesContent":["import { DummyDriver, Kysely, SqliteAdapter, SqliteIntrospector, SqliteQueryCompiler } from \"kysely\";\n\nexport const dummyKysely: Kysely<any> = new Kysely({\n dialect: {\n createAdapter: () => new SqliteAdapter(),\n createDriver: () => new DummyDriver(),\n createQueryCompiler: () => new SqliteQueryCompiler(),\n createIntrospector: (db) => new SqliteIntrospector(db),\n },\n});\n","import initXxhash, { type XXHashAPI } from \"xxhash-wasm\";\n\nlet loadPromise: Promise<void> | null = null;\nlet api: XXHashAPI | null = null;\n\nfunction ensureLoaded(): Promise<void> {\n if (!loadPromise) {\n loadPromise = initXxhash().then((hasher) => {\n api = hasher;\n });\n }\n return loadPromise;\n}\n\nfunction h64(input: string, seed = 0n): bigint {\n if (!api) {\n throw new Error(\"xxhash is not initialized; call xxhash.ensureLoaded() first\");\n }\n return api.h64(input, seed);\n}\n\nexport const xxhash = {\n ensureLoaded,\n h64,\n};\n","export interface HLC {\n timestamp: number;\n counter: number;\n nodeId: string;\n}\n\nconst MAX_COUNTER = 36 ** 5 - 1; // 60,466,175 — max value that fits in 5-char base36\nconst DEFAULT_MAX_DRIFT_MS = 6 * 60 * 60 * 1000; // 6 hours\n\nexport class HLCCounter {\n private timestamp: number;\n private counter: number;\n private nodeId: string;\n\n private readonly getTimestamp: () => number;\n private readonly maxDrift: number;\n\n constructor(nodeId: string, getTimestamp: () => number, maxDrift: number = DEFAULT_MAX_DRIFT_MS) {\n this.timestamp = getTimestamp();\n this.counter = 0;\n this.nodeId = nodeId;\n this.getTimestamp = getTimestamp;\n this.maxDrift = maxDrift;\n }\n\n getCurrentHLC(): HLC {\n return {\n timestamp: this.timestamp,\n counter: this.counter,\n nodeId: this.nodeId,\n };\n }\n\n getNextHLC(): HLC {\n const now = this.getTimestamp();\n\n if (now > this.timestamp) {\n this.timestamp = now;\n this.counter = 0;\n return this.getCurrentHLC();\n }\n\n this.counter++;\n if (this.counter > MAX_COUNTER) {\n throw new Error(`HLC counter overflow: exceeded max value ${MAX_COUNTER}`);\n }\n return this.getCurrentHLC();\n }\n\n mergeHLC(hlc: HLC) {\n const now = this.getTimestamp();\n if (hlc.timestamp - now > this.maxDrift) {\n console.warn(\n `HLC: ignoring far-future timestamp (remote=${hlc.timestamp}, local=${now}, drift=${hlc.timestamp - now}ms)`,\n );\n return;\n }\n\n if (this.timestamp === hlc.timestamp) {\n this.counter = Math.max(this.counter, hlc.counter) + 1;\n } else if (this.timestamp > hlc.timestamp) {\n this.counter++;\n } else {\n this.timestamp = hlc.timestamp;\n this.counter = hlc.counter + 1;\n }\n if (this.counter > MAX_COUNTER) {\n throw new Error(`HLC counter overflow: exceeded max value ${MAX_COUNTER}`);\n }\n }\n}\n\nexport function serializeHLC(hlc: HLC) {\n return `${hlc.timestamp.toString().padStart(15, \"0\")}:${hlc.counter.toString(36).padStart(5, \"0\")}:${hlc.nodeId}`;\n}\n\nexport function deserializeHLC(serialized: string): HLC {\n const parts = serialized.split(\":\");\n if (parts.length < 3) {\n throw new Error(`Invalid HLC format: expected at least 3 colon-separated segments, got ${parts.length}`);\n }\n\n const timestamp = parseInt(parts[0], 10);\n const counter = parseInt(parts[1], 36);\n\n if (Number.isNaN(timestamp) || Number.isNaN(counter)) {\n throw new Error(`Invalid HLC values: timestamp=${parts[0]}, counter=${parts[1]}`);\n }\n\n return {\n timestamp,\n counter,\n nodeId: parts.slice(2).join(\":\"),\n };\n}\n\nexport function compareHLC(one: HLC, two: HLC) {\n if (one.timestamp === two.timestamp) {\n if (one.counter === two.counter) {\n if (one.nodeId === two.nodeId) {\n return 0;\n }\n return one.nodeId < two.nodeId ? -1 : 1;\n }\n return one.counter - two.counter;\n }\n return one.timestamp - two.timestamp;\n}\n","import type { Kysely, QueryCreator } from \"kysely\";\nimport { sql } from \"kysely\";\nimport type { SQLiteDbWrapper } from \"./sqlite-db-wrapper\";\n\ninterface SqliteSystemDatabase {\n // https://www.sqlite.org/schematab.html#alternative_names\n sqlite_master: SQliteMasterTable;\n}\n\n// https://www.sqlite.org/schematab.html#interpretation_of_the_schema_table\ninterface SQliteMasterTable {\n name: string;\n rootpage: number | null;\n sql: string;\n tbl_name: string;\n type: \"index\" | \"table\" | \"trigger\" | \"view\";\n}\n\n// https://www.sqlite.org/pragma.html#pragma_table_info\ninterface PragmaTableInfo {\n cid: number;\n dflt_value: unknown;\n name: string;\n notnull: 0 | 1;\n pk: number;\n type: string;\n}\n\nfunction tablesQuery(qb: QueryCreator<SqliteSystemDatabase> | Kysely<SqliteSystemDatabase>) {\n return qb\n .selectFrom(\"sqlite_master\")\n .where(\"type\", \"in\", [\"table\", \"view\"])\n .where(\"name\", \"not like\", \"sqlite_%\")\n .select([\"name\", \"sql\", \"type\"])\n .orderBy(\"name\");\n}\n\nexport type TableMetadata = {\n name: string;\n isView: boolean;\n columns: ColumnMetadata[];\n};\n\nexport type DatabaseIntrospection = Record<string, TableMetadata>;\n\ntype ColumnMetadata = {\n name: string;\n dataType: string;\n isNullable: boolean;\n isAutoIncrementing: boolean;\n hasDefaultValue: boolean;\n comment: undefined;\n};\n\nexport function introspectDb<BaseDatabase>(_db: SQLiteDbWrapper<BaseDatabase>): DatabaseIntrospection {\n const db = _db as unknown as SQLiteDbWrapper<SqliteSystemDatabase>;\n const tables = db.executeKysely((db) => tablesQuery(db as unknown as Kysely<SqliteSystemDatabase>), {\n loggerLevel: \"system\",\n }).rows;\n\n const tablesMetadata = db.executeKysely(\n (db) =>\n db\n .with(\"table_list\", (qb) => tablesQuery(qb as unknown as Kysely<SqliteSystemDatabase>))\n .selectFrom([\"table_list as tl\", sql<PragmaTableInfo>`pragma_table_info(tl.name)`.as(\"p\")])\n .select([\"tl.name as table\", \"p.cid\", \"p.name\", \"p.type\", \"p.notnull\", \"p.dflt_value\", \"p.pk\"])\n .orderBy(\"tl.name\")\n .orderBy(\"p.cid\"),\n { loggerLevel: \"system\" },\n ).rows;\n\n const columnsByTable: Record<string, typeof tablesMetadata> = {};\n for (const row of tablesMetadata) {\n columnsByTable[row.table] ??= [];\n columnsByTable[row.table].push(row);\n }\n\n return Object.fromEntries(\n tables.map(({ name, sql, type }) => {\n // // Try to find the name of the column that has `autoincrement` 🤦\n let autoIncrementCol = sql\n ?.split(/[(),]/)\n ?.find((it) => it.toLowerCase().includes(\"autoincrement\"))\n ?.trimStart()\n ?.split(/\\s+/)?.[0]\n ?.replace(/[\"`]/g, \"\");\n\n const columns = columnsByTable[name] ?? [];\n\n // Otherwise, check for an INTEGER PRIMARY KEY\n // https://www.sqlite.org/autoinc.html\n if (!autoIncrementCol) {\n const pkCols = columns.filter((r) => r.pk > 0);\n if (pkCols.length === 1 && pkCols[0].type.toLowerCase() === \"integer\") {\n autoIncrementCol = pkCols[0].name;\n }\n }\n\n return [\n name,\n {\n name: name,\n isView: type === \"view\",\n columns: columns.map((col) => ({\n name: col.name,\n dataType: col.type,\n isNullable: !col.notnull,\n isAutoIncrementing: col.name === autoIncrementCol,\n hasDefaultValue: col.dflt_value != null,\n comment: undefined,\n })),\n },\n ];\n }),\n );\n}\n","export type LogLevel = \"info\" | \"warning\" | \"error\" | \"trace\" | \"system\";\n\nexport type Logger = (type: string, message: string, level?: LogLevel) => void;\n\nexport const startPerformanceLogger = (logger: Logger) => {\n let startTime = performance.now();\n\n return {\n restart: () => {\n startTime = performance.now();\n },\n logEnd: (type: string, message: string, level: LogLevel = \"info\") => {\n const elapsed = performance.now() - startTime;\n\n logger(type, `${elapsed.toFixed(2)}ms - ${message}`, level);\n },\n };\n};\n","import type {\n BindableValue,\n FunctionOptions,\n Database as SQLiteDatabase,\n Sqlite3Static,\n SqlValue,\n} from \"@sqlite.org/sqlite-wasm\";\nimport type { Compilable, CompiledQuery, Kysely } from \"kysely\";\nimport { dummyKysely } from \"./dummy-kysely\";\nimport { type DatabaseIntrospection, introspectDb } from \"./introspection\";\nimport { type Logger, startPerformanceLogger } from \"./logger\";\n\nexport type ExecuteParams = {\n sql: string;\n parameters: readonly unknown[];\n};\n\nexport type ExecuteResult<T> = {\n rows: T[];\n};\n\nexport type PreparedStatement<TParams extends unknown[], TResult> = {\n execute: (parameters: TParams) => TResult[];\n finalize: () => void;\n isFinalized: boolean;\n};\n\ntype ScalarFunctionOptions<TArgs extends readonly SqlValue[], TResult extends SqlValue | undefined> = {\n name: string;\n callback: (...args: TArgs) => TResult;\n} & Pick<FunctionOptions, \"deterministic\" | \"directOnly\" | \"innocuous\">;\n\ntype SqliteWrapperOptions = {\n logger?: Logger;\n loggerPrefix?: string;\n sqlite3: Sqlite3Static;\n db: () => SQLiteDatabase;\n};\n\nexport type SQLiteTransactionWrapper<TDatabase = unknown> = Pick<\n SQLiteDbWrapper<TDatabase>,\n \"sql\" | \"execute\" | \"executePrepared\" | \"executePreparedRaw\" | \"executeKysely\"\n>;\n\nexport type InternalSQLiteTransactionWrapper<TDatabase = unknown> = Pick<\n SQLiteDbWrapper<TDatabase>,\n \"executePrepared\" | \"executePreparedRaw\"\n>;\n\nexport type InternalSQLiteWrapper<TDatabase = unknown> = InternalSQLiteTransactionWrapper<TDatabase> & {\n executeTransaction: (callback: (db: InternalSQLiteTransactionWrapper<TDatabase>) => void) => void;\n};\n\ntype QueryMetaOpts = {\n loggerLevel?: \"info\" | \"system\";\n};\n\nexport class SQLiteDbWrapper<TDatabase = unknown> {\n private db: SQLiteDatabase | null = null;\n private sqlite3: Sqlite3Static;\n private logger?: Logger;\n private loggerPrefix?: string;\n\n private loadedDbSchema: DatabaseIntrospection | null = null;\n\n private readonly dataPointers = [] as number[];\n\n private preparedStatements: PreparedStatement<SqlValue[], unknown>[] = [];\n private preparedStatementsMap = new Map<string, TypedStatement<Record<string, unknown>, unknown>>();\n private preparedRawStatementsMap = new Map<string, PreparedStatement<SqlValue[], unknown>>();\n\n constructor(opts: SqliteWrapperOptions) {\n this.db = opts.db();\n this.sqlite3 = opts.sqlite3;\n this.logger = opts.logger;\n this.loggerPrefix = opts.loggerPrefix;\n }\n\n get ensureDb() {\n if (!this.db) {\n throw new Error(\"Database is already closed\");\n }\n return this.db;\n }\n\n get dbSchema() {\n if (!this.loadedDbSchema) {\n this.loadedDbSchema = introspectDb(this);\n }\n return this.loadedDbSchema;\n }\n\n execute<T = unknown>(opts: ExecuteParams | string | CompiledQuery<T>, meta?: QueryMetaOpts): ExecuteResult<T> {\n const sql = typeof opts === \"string\" ? opts : opts.sql;\n const bind = typeof opts === \"string\" ? undefined : opts.parameters;\n\n const perf = this.logger ? startPerformanceLogger(this.logger) : undefined;\n const rows = this.ensureDb.exec({\n sql,\n bind: bind as BindableValue[],\n returnValue: \"resultRows\",\n rowMode: \"object\",\n });\n perf?.logEnd(`${this.loggerPrefix ?? \"\"}:query`, sql, meta?.loggerLevel);\n\n return { rows: rows as T[] };\n }\n\n executeTransaction<T>(callback: (db: SQLiteTransactionWrapper<TDatabase>) => T): T {\n const transaction = this.beginTransaction();\n try {\n const result = callback(this);\n transaction.commit();\n return result;\n } catch (error) {\n transaction.rollback();\n throw error;\n }\n }\n\n isInTransaction() {\n // TODO: Awaiting upstream fix: https://github.com/sqlite/sqlite-wasm/pull/143\n return (this.sqlite3.capi as any).sqlite3_get_autocommit(this.ensureDb) === 0;\n }\n\n beginTransaction() {\n this.executePreparedRaw({\n key: \"$begin-transaction\",\n sql: \"begin\",\n meta: {\n loggerLevel: \"system\",\n },\n });\n\n return {\n commit: () => {\n this.executePreparedRaw({\n key: \"$commit-transaction\",\n sql: \"commit\",\n meta: {\n loggerLevel: \"system\",\n },\n });\n },\n rollback: () => {\n this.executePreparedRaw({\n key: \"$rollback-transaction\",\n sql: \"rollback\",\n meta: {\n loggerLevel: \"system\",\n },\n });\n },\n };\n }\n\n prepare<TParams extends unknown[], TResult>(sql: string, opts?: QueryMetaOpts) {\n const perf = this.logger ? startPerformanceLogger(this.logger) : undefined;\n const stmt = this.ensureDb.prepare(sql);\n perf?.logEnd(`${this.loggerPrefix ?? \"\"}:prepare`, sql, opts?.loggerLevel);\n\n let isFinalized = false;\n\n const execute = (params: TParams) => {\n if (isFinalized) {\n throw new Error(\"Statement is finalized\");\n }\n\n const perf = this.logger ? startPerformanceLogger(this.logger) : undefined;\n try {\n if (params.length > 0) {\n stmt.bind(params as SqlValue[]);\n }\n\n const results = [] as TResult[];\n while (stmt.step()) {\n results.push(stmt.get({}) as TResult);\n }\n\n return results;\n } finally {\n stmt.reset(true);\n perf?.logEnd(`${this.loggerPrefix ?? \"\"}:prepare-execute`, sql, opts?.loggerLevel);\n }\n };\n\n const finalize = () => {\n isFinalized = true;\n stmt.finalize();\n };\n\n const preparedStatement: PreparedStatement<TParams, TResult> = {\n execute,\n finalize,\n get isFinalized() {\n return isFinalized;\n },\n };\n\n this.preparedStatements.push(preparedStatement as PreparedStatement<SqlValue[], unknown>);\n\n return preparedStatement;\n }\n\n prepareKysely<TParams extends Record<string, unknown>>(opts?: QueryMetaOpts) {\n return <TQuery extends Compilable<TResult>, TResult = QueryBuilderOutput<TQuery>>(\n factory: KyselyStatementFactory<TParams, TDatabase, TQuery, TResult>,\n ): TypedStatement<TParams, TResult> => {\n const query = factory(dummyKysely, (key) => key as any).compile();\n const statement = this.prepare<SqlValue[], TResult>(query.sql, opts);\n\n return {\n execute: (parameters) => {\n const params = query.parameters.map((param) => parameters[param as keyof TParams]);\n const result = statement.execute(params as SqlValue[]);\n return result;\n },\n };\n };\n }\n\n executeKysely<TQuery extends Compilable<TResult>, TResult = QueryBuilderOutput<TQuery>>(\n factory: KyselyQueryFactory<TDatabase, TQuery, TResult>,\n meta?: QueryMetaOpts,\n ) {\n const query = factory(dummyKysely).compile();\n return this.execute(query, meta);\n }\n\n executePrepared<\n TParams extends Record<string, unknown>,\n TQuery extends Compilable<TResult>,\n TResult = QueryBuilderOutput<TQuery>,\n >(\n key: string,\n params: TParams,\n factory: KyselyStatementFactory<TParams, TDatabase, TQuery, TResult>,\n meta?: QueryMetaOpts,\n ) {\n let statement = this.preparedStatementsMap.get(key) as TypedStatement<TParams, TResult> | undefined;\n if (!statement) {\n statement = this.prepareKysely<TParams>(meta)(factory);\n this.preparedStatementsMap.set(key, statement as TypedStatement<Record<string, unknown>, unknown>);\n }\n\n return statement.execute(params);\n }\n\n executePreparedRaw<TParams extends unknown[], TResult>({\n key,\n sql,\n params,\n meta,\n }: {\n key: string;\n sql: string;\n params?: TParams;\n meta?: QueryMetaOpts;\n }) {\n let statement = this.preparedRawStatementsMap.get(key) as PreparedStatement<TParams, TResult> | undefined;\n if (!statement) {\n statement = this.prepare(sql, meta);\n this.preparedRawStatementsMap.set(key, statement as PreparedStatement<any[], unknown>);\n }\n\n return statement.execute((params ?? []) as TParams);\n }\n\n sql<T = unknown>(templateOrString: TemplateStringsArray | string, ...parameters: unknown[]) {\n if (typeof templateOrString === \"string\") {\n return this.execute<T>({\n sql: templateOrString,\n parameters,\n });\n }\n return this.execute<T>({\n sql: templateOrString.join(\"?\"),\n parameters,\n });\n }\n\n createScalarFunction<TArgs extends SqlValue[], TResult extends SqlValue | undefined>({\n name,\n callback,\n deterministic,\n directOnly,\n innocuous,\n }: ScalarFunctionOptions<TArgs, TResult>) {\n return this.ensureDb.createFunction({\n name,\n xFunc: (_, ...args) => {\n const result = callback(...(args as TArgs)) as SqlValue;\n return result;\n },\n arity: callback.length,\n deterministic,\n directOnly,\n innocuous,\n });\n }\n\n useSnapshot(snapshot: Uint8Array<ArrayBufferLike>) {\n const perf = this.logger ? startPerformanceLogger(this.logger) : undefined;\n const dataPointer = this.sqlite3.wasm.allocFromTypedArray(snapshot);\n this.dataPointers.push(dataPointer);\n\n const resultCode = this.sqlite3.capi.sqlite3_deserialize(\n this.ensureDb,\n \"main\",\n dataPointer,\n snapshot.byteLength,\n snapshot.byteLength,\n this.sqlite3.capi.SQLITE_DESERIALIZE_FREEONCLOSE | this.sqlite3.capi.SQLITE_DESERIALIZE_RESIZEABLE,\n );\n\n this.ensureDb.checkRc(resultCode);\n\n this.invalidateDbSchema();\n\n perf?.logEnd(\"useSnapshot\", \"success\", \"system\");\n }\n\n createSnapshot() {\n return this.sqlite3.capi.sqlite3_js_db_export(this.ensureDb);\n }\n\n invalidateDbSchema() {\n this.loadedDbSchema = null;\n }\n\n cleanup() {\n this.preparedStatements.forEach((stmt) => {\n stmt.finalize();\n });\n this.preparedStatements.splice(0);\n this.preparedStatementsMap.clear();\n this.preparedRawStatementsMap.clear();\n }\n\n close() {\n this.cleanup();\n\n this.db?.close();\n this.db = null;\n }\n}\n\nexport type QueryBuilderOutput<QB> = QB extends Compilable<infer O> ? O : never;\ntype ParamsGetter<TParams> = <TKey extends keyof TParams>(key: TKey) => TParams[TKey];\n\ntype TypedStatement<TParams extends Record<string, unknown>, TResult> = {\n execute: (parameters: TParams) => TResult[];\n};\nexport type KyselyStatementFactory<\n TParams extends Record<string, unknown>,\n TDatabase,\n TQuery extends Compilable<TResult>,\n TResult = QueryBuilderOutput<TQuery>,\n> = (kysely: Kysely<TDatabase>, params: ParamsGetter<TParams>) => TQuery;\nexport type KyselyQueryFactory<TDatabase, TQuery extends Compilable<TResult>, TResult = QueryBuilderOutput<TQuery>> = (\n kysely: Kysely<TDatabase>,\n) => TQuery;\n","import type { SchemaModule } from \"kysely\";\n\nexport type CrdtEventType = \"item-created\" | \"item-updated\";\n\nexport type CrdtEventStatus = \"pending\" | \"applied\" | \"failed\" | \"deduped\";\n\n/** Persisted on applied events that were accepted but did not mutate materialized state. */\nexport const CRDT_EVENT_NO_OP_PAYLOAD = \"no-op\";\n\nexport function isNoOpCrdtEventPayload(payload: string) {\n return payload === CRDT_EVENT_NO_OP_PAYLOAD;\n}\n\nexport type CrdtEventOrigin = \"remote\" | \"own\" | \"local\";\n\nexport type PersistedCrdtEvent = {\n schema_version: number;\n sync_id: number;\n status: CrdtEventStatus;\n type: CrdtEventType;\n timestamp: string;\n origin: CrdtEventOrigin;\n source_node_id: string;\n dataset: string;\n item_id: string;\n payload: string;\n};\n\nexport type CrdtUpdateLogItem = {\n dataset: string;\n item_id: string;\n payload: string;\n};\n\nexport type CrdtUpdateLogPayload = Record<string, string>;\n\nexport const crdtSchema = {\n persistedEventsTable: createPersistedEventsTable,\n crdtUpdateLogTable: createCrdtUpdateLogTableQuery,\n};\n\nfunction createPersistedEventsTable(schema: SchemaModule, tableName: string) {\n return schema\n .createTable(tableName)\n .ifNotExists()\n .addColumn(\"sync_id\", \"integer\", (col) => col.notNull().primaryKey())\n .addColumn(\"schema_version\", \"integer\", (col) => col.notNull())\n .addColumn(\"status\", \"text\", (col) => col.notNull())\n .addColumn(\"type\", \"text\", (col) => col.notNull())\n .addColumn(\"timestamp\", \"text\", (col) => col.notNull())\n .addColumn(\"origin\", \"text\", (col) => col.notNull())\n .addColumn(\"dataset\", \"text\", (col) => col.notNull())\n .addColumn(\"item_id\", \"text\", (col) => col.notNull())\n .addColumn(\"payload\", \"text\", (col) => col.notNull());\n}\n\nfunction createCrdtUpdateLogTableQuery(schema: SchemaModule, tableName: string) {\n return schema\n .createTable(tableName)\n .ifNotExists()\n .addColumn(\"dataset\", \"text\", (col) => col.notNull())\n .addColumn(\"item_id\", \"text\", (col) => col.notNull())\n .addColumn(\"payload\", \"text\", (col) => col.notNull())\n .addPrimaryKeyConstraint(`pk_${tableName}`, [\"item_id\", \"dataset\"]);\n}\n","import type {\n ColumnDataType,\n ColumnDefinitionBuilderCallback,\n Compilable,\n CreateIndexBuilder,\n CreateTableBuilder,\n Expression,\n Kysely,\n} from \"kysely\";\nimport { dummyKysely } from \"../dummy-kysely\";\nimport { type CrdtEventType, isNoOpCrdtEventPayload } from \"../sqlite-crdt/crdt-table-schema\";\nimport type { StoredValue } from \"../sqlite-crdt/stored-value\";\n\ntype CrdtEvent = {\n type: CrdtEventType;\n dataset: string;\n item_id: string;\n payload: Record<string, unknown>;\n};\n\nexport type MigratableEvent = {\n schema_version: number;\n type: CrdtEventType;\n dataset: string;\n item_id: string;\n payload: string;\n};\n\ntype CrdtEventTransformer = (event: CrdtEvent) => CrdtEvent | null;\n\ntype MigrationStepSql =\n | {\n sql: string;\n parameters?: readonly unknown[];\n }\n | Compilable\n | ((db: Kysely<unknown>) => Compilable);\n\ntype TableRename = { oldTable: string; newTable: string };\n\ntype MigrationStep = {\n sql: MigrationStepSql | MigrationStepSql[];\n eventTransformer?: MigrationEventTransformers;\n tableRenames?: TableRename[];\n tableDrops?: string[];\n};\n\ntype RawMigrationStep = {\n sql: MigrationSql[];\n eventTransformer?: CompiledMigrationEventTransformer;\n tableRenames?: TableRename[];\n tableDrops?: string[];\n};\n\ntype MigrationSql = { sql: string; parameters: readonly unknown[] };\n\ntype DataTypeExpression = ColumnDataType | Expression<any>;\n\nconst protectedColumns = [\"id\", \"tombstone\"];\n\nfunction assertColumnNotProtected(column: string, operation: string) {\n if (protectedColumns.includes(column)) {\n throw new Error(`Cannot ${operation} protected column \"${column}\"`);\n }\n}\n\nconst migrationSteps = {\n createTable: (table: string, build: (table: CreateTableBuilder<string, never>) => Compilable): MigrationStep => ({\n sql: (db) => build(db.schema.createTable(table)),\n }),\n\n dropTable: (table: string): MigrationStep => ({\n sql: (db) => db.schema.dropTable(table),\n eventTransformer: {\n [table]: () => null,\n },\n tableDrops: [table],\n }),\n\n createIndex: (indexName: string, build: (index: CreateIndexBuilder) => Compilable): MigrationStep => ({\n sql: (db) => build(db.schema.createIndex(indexName)),\n }),\n\n dropIndex: (indexName: string): MigrationStep => ({\n sql: (db) => db.schema.dropIndex(indexName),\n }),\n\n renameTable: ({ oldTable, newTable }: { oldTable: string; newTable: string }): MigrationStep => ({\n sql: (db) => db.schema.alterTable(oldTable).renameTo(newTable),\n eventTransformer: {\n [oldTable]: (event) => {\n event.dataset = newTable;\n return event;\n },\n },\n tableRenames: [{ oldTable, newTable }],\n }),\n\n renameColumn: ({\n table,\n oldColumn,\n newColumn,\n }: {\n table: string;\n oldColumn: string;\n newColumn: string;\n }): MigrationStep => {\n assertColumnNotProtected(oldColumn, \"rename\");\n assertColumnNotProtected(newColumn, \"rename to\");\n return {\n sql: (db) => db.schema.alterTable(table).renameColumn(oldColumn, newColumn),\n eventTransformer: {\n [table]: (event) => {\n if ((event.type !== \"item-updated\" && event.type !== \"item-created\") || !(oldColumn in event.payload)) {\n return event;\n }\n\n const oldVal = event.payload[oldColumn];\n delete event.payload[oldColumn];\n event.payload[newColumn] = oldVal;\n\n return event;\n },\n },\n };\n },\n\n addColumn: ({\n table,\n column,\n type,\n defaultValue,\n build = (e) => e,\n }: {\n table: string;\n column: string;\n type: DataTypeExpression;\n defaultValue: number | boolean | string | null;\n build?: ColumnDefinitionBuilderCallback;\n }): MigrationStep => ({\n sql: (db) => db.schema.alterTable(table).addColumn(column, type, (x) => build(x).defaultTo(defaultValue)),\n eventTransformer: {\n [table]: (event) => {\n if (event.type !== \"item-created\") {\n return event;\n }\n\n event.payload[column] = defaultValue;\n\n return event;\n },\n },\n }),\n\n dropColumn: ({ table, column }: { table: string; column: string }): MigrationStep => {\n assertColumnNotProtected(column, \"drop\");\n return {\n sql: (db) => db.schema.alterTable(table).dropColumn(column),\n eventTransformer: {\n [table]: (event) => {\n if (!(column in event.payload)) {\n return event;\n }\n\n delete event.payload[column];\n\n if (event.type === \"item-updated\" && Object.keys(event.payload).length === 0) {\n return null;\n }\n\n return event;\n },\n },\n };\n },\n};\n\ntype MigrationEventTransformers = Record<string, CrdtEventTransformer>;\ntype CompiledMigrationEventTransformer = (event: CrdtEvent) => CrdtEvent | null;\n\nfunction buildMigrationSql(steps: MigrationStep[]): MigrationSql[] {\n return steps\n .flatMap((step) => (Array.isArray(step.sql) ? step.sql : [step.sql]))\n .map((sql): MigrationSql => {\n if (typeof sql === \"string\") {\n return { sql, parameters: [] };\n }\n\n if (typeof sql === \"function\") {\n const query = sql(dummyKysely).compile();\n return { sql: query.sql, parameters: query.parameters };\n }\n\n if (\"compile\" in sql) {\n const query = sql.compile();\n return { sql: query.sql, parameters: query.parameters };\n }\n\n return {\n sql: sql.sql,\n parameters: sql.parameters ?? [],\n };\n });\n}\n\nfunction buildMigrationEventTransformer(steps: MigrationStep[]): CompiledMigrationEventTransformer | undefined {\n const transformers: Array<[string, CrdtEventTransformer]> = [];\n\n for (const step of steps) {\n if (step.eventTransformer) {\n transformers.push(...Object.entries(step.eventTransformer));\n }\n }\n\n if (transformers.length === 0) {\n return undefined;\n }\n\n return (event: CrdtEvent) => {\n let transformedEvent: CrdtEvent | null = event;\n\n for (const [table, transformer] of transformers) {\n if (transformedEvent === null) {\n return null;\n }\n if (transformedEvent.dataset !== table) {\n continue;\n }\n transformedEvent = transformer(transformedEvent);\n if (transformedEvent === null) {\n return null;\n }\n }\n\n return transformedEvent;\n };\n}\n\nexport function createMigrations(buildMigrations: (builder: typeof migrationSteps) => Record<number, MigrationStep[]>) {\n const migrations: Record<number, RawMigrationStep> = Object.fromEntries(\n Object.entries(buildMigrations(migrationSteps)).map(([version, steps]) => {\n const versionNumber = Number(version);\n\n if (Number.isNaN(versionNumber)) {\n throw new Error(`Invalid migration version: ${version}`);\n }\n\n if (versionNumber < 0) {\n throw new Error(`Migration version cannot be negative: ${version}`);\n }\n\n const tableRenames = steps.flatMap((s) => s.tableRenames ?? []);\n const tableDrops = steps.flatMap((s) => s.tableDrops ?? []);\n\n return [\n version,\n {\n sql: buildMigrationSql(steps),\n eventTransformer: buildMigrationEventTransformer(steps),\n ...(tableRenames.length > 0 && { tableRenames }),\n ...(tableDrops.length > 0 && { tableDrops }),\n },\n ];\n }),\n );\n\n return migrations;\n}\n\nexport type Migrations = ReturnType<typeof createMigrations>;\n\nexport type MigrationsDb = {\n startTransaction: (callback: (tx: MigrationsTransaction) => void) => void;\n};\n\ntype MigrationsTransaction = {\n execute: (sql: string, parameters: readonly unknown[]) => void;\n};\n\nexport function createMigrator({\n migrations,\n schemaVersion,\n updateLogTableName,\n}: {\n migrations: Migrations;\n schemaVersion: StoredValue<number>;\n updateLogTableName?: string;\n}) {\n const latestSchemaVersion = Math.max(...Object.keys(migrations).map(Number));\n\n // Pre-sort migrations once for efficient range lookups\n const sortedMigrations = Object.entries(migrations)\n .map(([v, m]) => [Number(v), m] as const)\n .sort((a, b) => a[0] - b[0]);\n\n const applyMigration = (db: MigrationsDb, version: number, migration: RawMigrationStep) => {\n if (version <= schemaVersion.current) {\n throw new Error(`Cannot apply migration ${version} to schema version ${schemaVersion.current}`);\n }\n\n db.startTransaction((tx) => {\n for (const statement of migration.sql) {\n tx.execute(statement.sql, statement.parameters);\n }\n if (updateLogTableName) {\n if (migration.tableRenames) {\n for (const { oldTable, newTable } of migration.tableRenames) {\n tx.execute(`UPDATE ${updateLogTableName} SET \"dataset\" = ? WHERE \"dataset\" = ?`, [newTable, oldTable]);\n }\n }\n if (migration.tableDrops) {\n for (const table of migration.tableDrops) {\n tx.execute(`DELETE FROM ${updateLogTableName} WHERE \"dataset\" = ?`, [table]);\n }\n }\n }\n schemaVersion.current = version;\n });\n };\n\n const migrateEvent = <Event extends MigratableEvent>(event: Event, targetVersion?: number): Event | null => {\n targetVersion ??= latestSchemaVersion;\n if (targetVersion > schemaVersion.current) {\n throw new Error(\n `Target schema version ${targetVersion} is greater than current schema version ${schemaVersion.current}`,\n );\n }\n\n if (isNoOpCrdtEventPayload(event.payload)) {\n if (event.schema_version < targetVersion) {\n event.schema_version = targetVersion;\n }\n return event;\n }\n\n if (event.schema_version >= targetVersion) {\n return event;\n }\n\n const fromVersion = event.schema_version;\n\n let crdtEvent: CrdtEvent | null = {\n dataset: event.dataset,\n item_id: event.item_id,\n type: event.type,\n payload: JSON.parse(event.payload),\n };\n\n for (let i = 0; i < sortedMigrations.length; i++) {\n const [version, migration] = sortedMigrations[i];\n if (version <= fromVersion) continue;\n if (version > targetVersion) break;\n\n const transformer = migration.eventTransformer;\n if (transformer) {\n crdtEvent = transformer(crdtEvent);\n if (crdtEvent === null) return null;\n }\n }\n\n if (crdtEvent === null) {\n return null;\n }\n\n event.schema_version = targetVersion;\n event.dataset = crdtEvent.dataset;\n event.item_id = crdtEvent.item_id;\n event.type = crdtEvent.type;\n event.payload = JSON.stringify(crdtEvent.payload);\n\n return event;\n };\n\n const migrateEvents = <Event extends MigratableEvent>(events: Event[], targetVersion?: number): Event[] => {\n return events\n .map((event) => migrateEvent(event, targetVersion ?? latestSchemaVersion))\n .filter((event): event is NonNullable<typeof event> => event !== null);\n };\n\n return {\n latestSchemaVersion,\n get currentSchemaVersion() {\n return schemaVersion.current;\n },\n migrateDbToLatest: (db: MigrationsDb) => {\n const currentSchemaVersion = schemaVersion.current;\n\n if (currentSchemaVersion >= latestSchemaVersion) {\n return;\n }\n\n for (let i = 0; i < sortedMigrations.length; i++) {\n const [version, migration] = sortedMigrations[i];\n if (version <= currentSchemaVersion) continue;\n applyMigration(db, version, migration);\n }\n },\n migrateEvent,\n migrateEvents,\n };\n}\n\nexport type SyncDbMigrator = ReturnType<typeof createMigrator>;\n","export type StoredValue<T> = {\n get current(): T;\n set current(newValue: T);\n};\n\nexport function createStoredValue<T>({\n initialValue,\n saveToStorage,\n}: {\n initialValue: T;\n saveToStorage?: (value: T) => void;\n}): StoredValue<T> {\n let currentValue = initialValue;\n\n return {\n get current() {\n return currentValue;\n },\n set current(newValue: T) {\n saveToStorage?.(newValue);\n currentValue = newValue;\n },\n };\n}\n","import { type SchemaModule, sql } from \"kysely\";\nimport { createStoredValue } from \"./sqlite-crdt/stored-value\";\nimport type { InternalSQLiteWrapper } from \"./sqlite-db-wrapper\";\n\nexport type KvStoreItem = {\n key: string;\n value: string;\n};\n\nexport function createKvStoreTableQuery(schema: SchemaModule, tableName: string) {\n return schema\n .createTable(tableName)\n .ifNotExists()\n .addColumn(\"key\", \"text\", (col) => col.notNull().primaryKey())\n .addColumn(\"value\", \"text\", (col) => col.notNull())\n .modifyEnd(sql`without rowid`);\n}\n\nexport function createSQLiteKvStore({ db, metaTableName }: { db: InternalSQLiteWrapper<any>; metaTableName: string }) {\n const metaDb = db as InternalSQLiteWrapper<{\n meta: KvStoreItem;\n }>;\n\n const get = (key: string): string | null => {\n const [result] = metaDb.executePrepared(\n \"get-meta-value\",\n { key },\n (db, params) =>\n db\n .selectFrom(metaTableName as \"meta\")\n .where(\"key\", \"=\", params(\"key\"))\n .select(\"value\")\n .limit((eb) => eb.lit(1)),\n { loggerLevel: \"system\" },\n );\n\n return result?.value ?? null;\n };\n\n const set = (key: string, value: string) => {\n metaDb.executePrepared(\n \"set-meta-value\",\n { key, value },\n (db, params) =>\n db\n .insertInto(metaTableName as \"meta\")\n .values({ key: params(\"key\"), value: params(\"value\") })\n .onConflict((oc) => oc.doUpdateSet({ value: params(\"value\") })),\n { loggerLevel: \"system\" },\n );\n };\n\n const remove = (key: string) => {\n metaDb.executePrepared(\n \"remove-meta-value\",\n { key },\n (db, params) => db.deleteFrom(metaTableName as \"meta\").where(\"key\", \"=\", params(\"key\")),\n { loggerLevel: \"system\" },\n );\n };\n\n const getNumberOrDefault = <T>(key: string, defaultValue: T): T | number => {\n const value = get(key);\n if (!value) return defaultValue;\n const parsedValue = Number.parseInt(value, 10);\n return Number.isNaN(parsedValue) ? defaultValue : parsedValue;\n };\n\n return {\n get,\n set,\n remove,\n createStringStoredValue: (key: string, defaultValue: string) =>\n createStoredValue<string>({\n initialValue: get(key) ?? defaultValue,\n saveToStorage: (val) => set(key, val),\n }),\n createNumberStoredValue: (key: string, defaultValue: number) =>\n createStoredValue<number>({\n initialValue: getNumberOrDefault(key, defaultValue),\n saveToStorage: (val) => set(key, val.toString()),\n }),\n };\n}\n\nexport type KvStore = ReturnType<typeof createSQLiteKvStore>;\n","import type { CrdtUpdateLogItem, PersistedCrdtEvent } from \"../sqlite-crdt/crdt-table-schema\";\nimport type { StoredValue } from \"../sqlite-crdt/stored-value\";\nimport type { SQLiteDbWrapper } from \"../sqlite-db-wrapper\";\nimport { createKvStoreTableQuery, createSQLiteKvStore, type KvStoreItem } from \"../sqlite-kv-store\";\nimport { type ParsedTableName, parseTableName } from \"../utils\";\n\nexport type SystemDbConfig = {\n eventsTable: ParsedTableName;\n updateLogTable: ParsedTableName;\n};\n\nexport function createSystemDbConfig({\n eventsTableName,\n updateLogTableName,\n}: {\n eventsTableName: string;\n updateLogTableName: string;\n}): SystemDbConfig {\n return {\n eventsTable: parseTableName(eventsTableName),\n updateLogTable: parseTableName(updateLogTableName),\n };\n}\n\nexport type WorkerDbSchema = {\n crdt_update_log: CrdtUpdateLogItem;\n \"worker.kv\": KvStoreItem;\n \"worker.crdt_events\": PersistedCrdtEvent;\n};\n\nexport const workerDbConfig = createSystemDbConfig({\n eventsTableName: \"worker.crdt_events\",\n updateLogTableName: \"crdt_update_log\",\n});\n\nexport type MemoryDbSchema = {\n crdt_update_log: CrdtUpdateLogItem;\n persisted_crdt_events: PersistedCrdtEvent;\n};\n\nexport const memoryDbConfig = createSystemDbConfig({\n eventsTableName: \"persisted_crdt_events\",\n updateLogTableName: \"crdt_update_log\",\n});\n\nexport type SystemMigrationContext = SystemDbConfig & {\n execute: (sql: string) => void;\n};\n\nexport type SystemMigration = {\n version: number;\n up: (ctx: SystemMigrationContext) => void;\n};\n\nexport const baseSystemMigrations: SystemMigration[] = [\n {\n version: 0,\n up: (ctx: SystemMigrationContext) => {\n ctx.execute(`CREATE TABLE IF NOT EXISTS ${ctx.eventsTable.fullIdentifier} (\n \"sync_id\" integer NOT NULL PRIMARY KEY,\n \"schema_version\" integer NOT NULL,\n \"status\" text NOT NULL,\n \"type\" text NOT NULL,\n \"timestamp\" text NOT NULL,\n \"origin\" text NOT NULL,\n \"dataset\" text NOT NULL,\n \"item_id\" text NOT NULL,\n \"payload\" text NOT NULL\n )`);\n ctx.execute(`CREATE TABLE IF NOT EXISTS ${ctx.updateLogTable.fullIdentifier} (\n \"dataset\" text NOT NULL,\n \"item_id\" text NOT NULL,\n \"payload\" text NOT NULL,\n PRIMARY KEY (\"item_id\", \"dataset\")\n )`);\n },\n },\n {\n version: 1,\n up: (ctx: SystemMigrationContext) => {\n ctx.execute(`ALTER TABLE ${ctx.eventsTable.fullIdentifier} ADD COLUMN \"source_node_id\" TEXT NOT NULL DEFAULT ''`);\n },\n },\n {\n version: 2,\n up: (ctx: SystemMigrationContext) => {\n const indexName = `${ctx.eventsTable.table}_status_sync_id_idx`;\n ctx.execute(\n `CREATE INDEX IF NOT EXISTS ${ctx.eventsTable.schema}.${indexName} ON ${ctx.eventsTable.table} (\"status\", \"sync_id\")`,\n );\n },\n },\n {\n version: 3,\n up: (ctx: SystemMigrationContext) => {\n const indexName = `${ctx.eventsTable.table}_timestamp_status_sync_id_idx`;\n ctx.execute(\n `CREATE INDEX IF NOT EXISTS ${ctx.eventsTable.schema}.${indexName} ON ${ctx.eventsTable.table} (\"timestamp\", \"status\", \"sync_id\")`,\n );\n },\n },\n];\n\nexport function runSystemMigrations(opts: {\n version: StoredValue<number>;\n migrations: SystemMigration[];\n dbConfig: SystemDbConfig;\n execute: (sql: string) => void;\n transaction: (callback: () => void) => void;\n}): void {\n const ctx: SystemMigrationContext = {\n ...opts.dbConfig,\n execute: opts.execute,\n };\n for (const migration of opts.migrations) {\n if (migration.version > opts.version.current) {\n opts.transaction(() => {\n migration.up(ctx);\n opts.version.current = migration.version;\n });\n }\n }\n}\n\nexport function applyWorkerDbSchema(db: SQLiteDbWrapper<any>) {\n // KV table stays separate — needed before system migrations for version tracking\n db.executeKysely((kysely) => createKvStoreTableQuery(kysely.schema, \"worker.kv\"), { loggerLevel: \"system\" });\n\n // System schema migrations (each in its own transaction)\n const kvStore = createSQLiteKvStore({ db, metaTableName: \"worker.kv\" });\n runSystemMigrations({\n migrations: baseSystemMigrations,\n version: kvStore.createNumberStoredValue(\"internal-schema-version\", -1),\n dbConfig: workerDbConfig,\n execute: (sql) => db.execute(sql, { loggerLevel: \"system\" }),\n transaction: (callback) => db.executeTransaction(callback),\n });\n\n return { kvStore };\n}\n\nexport function applyMemoryDbSchema(db: SQLiteDbWrapper<any>) {\n db.execute(\n `CREATE TABLE IF NOT EXISTS ${memoryDbConfig.eventsTable.fullIdentifier} (\n \"sync_id\" integer NOT NULL PRIMARY KEY,\n \"schema_version\" integer NOT NULL,\n \"status\" text NOT NULL,\n \"type\" text NOT NULL,\n \"timestamp\" text NOT NULL,\n \"origin\" text NOT NULL,\n \"source_node_id\" text NOT NULL DEFAULT '',\n \"dataset\" text NOT NULL,\n \"item_id\" text NOT NULL,\n \"payload\" text NOT NULL\n )`,\n { loggerLevel: \"system\" },\n );\n db.execute(\n `CREATE INDEX IF NOT EXISTS ${memoryDbConfig.eventsTable.table}_status_sync_id_idx ON ${memoryDbConfig.eventsTable.table} (\"status\", \"sync_id\")`,\n {\n loggerLevel: \"system\",\n },\n );\n db.execute(\n `CREATE INDEX IF NOT EXISTS ${memoryDbConfig.eventsTable.table}_timestamp_status_sync_id_idx ON ${memoryDbConfig.eventsTable.table} (\"timestamp\", \"status\", \"sync_id\")`,\n {\n loggerLevel: \"system\",\n },\n );\n db.execute(\n `CREATE TABLE IF NOT EXISTS ${memoryDbConfig.updateLogTable.fullIdentifier} (\n \"dataset\" text NOT NULL,\n \"item_id\" text NOT NULL,\n \"payload\" text NOT NULL,\n PRIMARY KEY (\"item_id\", \"dataset\")\n)`,\n { loggerLevel: \"system\" },\n );\n}\n","import type { SqlValue } from \"@sqlite.org/sqlite-wasm\";\nimport type { Kysely } from \"kysely\";\nimport type { SystemDbConfig } from \"../migrations/system-schema\";\nimport type { InternalSQLiteWrapper } from \"../sqlite-db-wrapper\";\nimport { quoteId } from \"../utils\";\nimport {\n type CrdtEventType,\n type CrdtUpdateLogItem,\n type CrdtUpdateLogPayload,\n isNoOpCrdtEventPayload,\n} from \"./crdt-table-schema\";\n\nexport type PendingCrdtEvent = {\n type: CrdtEventType;\n dataset: string;\n item_id: string;\n timestamp: string;\n payload: string;\n};\n\nexport const createSQLiteCrdtApplyFunction = ({\n db,\n dbConfig,\n}: {\n db: InternalSQLiteWrapper<any>;\n dbConfig: SystemDbConfig;\n}) => {\n const applyCrdtEvent = createCrdtApplyFunction({\n getCrdtUpdateLog(opts) {\n const [metaRow] = db.executePrepared(\n \"get-item-crdt-meta\",\n {\n item_id: opts.itemId,\n dataset: opts.dataset,\n },\n (db, params) => {\n return (db as unknown as Kysely<{ table: CrdtUpdateLogItem }>)\n .selectFrom(dbConfig.updateLogTable.fullIdentifier as \"table\")\n .select(\"payload\")\n .where(\"item_id\", \"=\", params(\"item_id\"))\n .where(\"dataset\", \"=\", params(\"dataset\"));\n },\n { loggerLevel: \"system\" },\n );\n const meta = metaRow ? (JSON.parse(metaRow.payload) as CrdtUpdateLogPayload) : null;\n return meta;\n },\n insertCrdtUpdateLog(opts) {\n db.executePrepared(\n \"insert-crdt-update-log\",\n {\n item_id: opts.itemId,\n dataset: opts.dataset,\n payload: opts.payload,\n },\n (db, params) =>\n (db as unknown as Kysely<{ table: CrdtUpdateLogItem }>)\n .insertInto(dbConfig.updateLogTable.fullIdentifier as \"table\")\n .values({\n item_id: params(\"item_id\"),\n dataset: params(\"dataset\"),\n payload: params(\"payload\"),\n }),\n { loggerLevel: \"system\" },\n );\n },\n updateCrdtUpdateLog(opts) {\n db.executePrepared(\n \"update-crdt-update-log\",\n {\n item_id: opts.itemId,\n dataset: opts.dataset,\n payload: opts.payload,\n },\n (db, params) =>\n (db as unknown as Kysely<{ table: CrdtUpdateLogItem }>)\n .updateTable(dbConfig.updateLogTable.fullIdentifier as \"table\")\n .set({\n payload: params(\"payload\"),\n })\n .where(\"item_id\", \"=\", params(\"item_id\"))\n .where(\"dataset\", \"=\", params(\"dataset\")),\n { loggerLevel: \"system\" },\n );\n },\n insertItem(opts) {\n const insertPayload = {} as Record<string, unknown>;\n for (const key of Object.keys(opts.payload)) {\n insertPayload[key] = key;\n }\n db.executePrepared(\n `crdt-insert-item-${opts.dataset}`,\n opts.payload,\n (db) => db.insertInto(opts.dataset).values(insertPayload),\n { loggerLevel: \"system\" },\n );\n },\n updateItem(opts) {\n const keys = Array.from(Object.keys(opts.payload));\n keys.sort();\n db.executePreparedRaw({\n key: `update-item-${opts.dataset}-${keys.join(\"-\")}`,\n sql: `update ${quoteId(opts.dataset)} set ${keys.map((key) => `${quoteId(key)} = ?`).join(\",\")} where id = ?`,\n params: [...keys.map((key) => opts.payload[key]), opts.itemId] as SqlValue[],\n meta: { loggerLevel: \"system\" },\n });\n },\n });\n\n return applyCrdtEvent;\n};\n\ntype CreateCrdtApplyOpts = {\n getCrdtUpdateLog: (opts: { itemId: string; dataset: string }) => CrdtUpdateLogPayload | null;\n insertItem: (opts: { dataset: string; payload: Record<string, unknown> }) => void;\n insertCrdtUpdateLog: (opts: { dataset: string; itemId: string; payload: string }) => void;\n updateItem: (opts: { dataset: string; itemId: string; payload: Record<string, unknown> }) => void;\n updateCrdtUpdateLog: (opts: { dataset: string; itemId: string; payload: string }) => void;\n};\n\nexport function createCrdtApplyFunction({\n getCrdtUpdateLog,\n insertItem,\n insertCrdtUpdateLog,\n updateItem,\n updateCrdtUpdateLog,\n}: CreateCrdtApplyOpts) {\n type ItemCreatedOpts = {\n event: PendingCrdtEvent;\n };\n const applyItemCreated = ({ event }: ItemCreatedOpts) => {\n const eventPayload = JSON.parse(event.payload);\n\n eventPayload.tombstone = false;\n insertItem({ dataset: event.dataset, payload: eventPayload });\n\n const newUpdateLog = {} as Record<string, string>;\n for (const key of Object.keys(eventPayload)) {\n newUpdateLog[key] = event.timestamp;\n }\n\n insertCrdtUpdateLog({\n dataset: event.dataset,\n itemId: event.item_id,\n payload: JSON.stringify(newUpdateLog),\n });\n };\n\n type ItemUpdatedOpts = {\n event: PendingCrdtEvent;\n meta: CrdtUpdateLogPayload;\n };\n const applyItemUpdated = ({ event, meta }: ItemUpdatedOpts) => {\n if (!meta) {\n throw new Error(`Item ${event.item_id} in dataset ${event.dataset} not found`);\n }\n const eventPayload = JSON.parse(event.payload);\n\n const updatePayload = {} as Record<string, unknown>;\n let hasUpdates = false;\n\n for (const [key, value] of Object.entries(eventPayload)) {\n if (key === \"id\") {\n continue;\n }\n\n const lastUpdateTimestamp = meta[key];\n const currentUpdateTimestamp = event.timestamp;\n\n if (!lastUpdateTimestamp || !currentUpdateTimestamp || currentUpdateTimestamp > lastUpdateTimestamp) {\n updatePayload[key] = value;\n meta[key] = currentUpdateTimestamp;\n hasUpdates = true;\n }\n }\n\n if (!hasUpdates) {\n return;\n }\n\n updateItem({\n dataset: event.dataset,\n itemId: event.item_id,\n payload: updatePayload,\n });\n updateCrdtUpdateLog({\n dataset: event.dataset,\n itemId: event.item_id,\n payload: JSON.stringify(meta),\n });\n };\n\n return (event: PendingCrdtEvent) => {\n if (isNoOpCrdtEventPayload(event.payload)) {\n return;\n }\n\n const meta = getCrdtUpdateLog({\n itemId: event.item_id,\n dataset: event.dataset,\n });\n\n // TODO Check primary key / unique constraints\n\n if (event.type !== \"item-created\" && event.type !== \"item-updated\") {\n throw new Error(`Unknown event type: ${event.type}`);\n }\n\n if (meta) {\n // Item already exists\n applyItemUpdated({ event, meta });\n return;\n }\n\n if (event.type === \"item-created\") {\n applyItemCreated({ event });\n return;\n }\n\n throw new Error(`Item ${event.item_id} in dataset ${event.dataset} not found`);\n };\n}\n","import { sql } from \"kysely\";\nimport { deserializeHLC, type HLCCounter, serializeHLC } from \"../hlc\";\nimport type { SyncDbMigrator } from \"../migrations/migrator\";\nimport type { SystemDbConfig } from \"../migrations/system-schema\";\nimport type { InternalSQLiteTransactionWrapper, InternalSQLiteWrapper } from \"../sqlite-db-wrapper\";\nimport { createTypedEventTarget, ensureSingletonExecution } from \"../utils\";\nimport { createSQLiteCrdtApplyFunction } from \"./apply-crdt-event\";\nimport {\n CRDT_EVENT_NO_OP_PAYLOAD,\n type CrdtEventOrigin,\n type CrdtEventStatus,\n type CrdtEventType,\n type CrdtUpdateLogItem,\n isNoOpCrdtEventPayload,\n type PersistedCrdtEvent,\n} from \"./crdt-table-schema\";\nimport { createEventHlcAccumulator } from \"./event-consistency\";\nimport type { StoredValue } from \"./stored-value\";\n\ntype LocalCrdtEvent = {\n type: CrdtEventType;\n dataset: string;\n item_id: string;\n payload: string;\n timestamp: string;\n schema_version: number;\n};\n\nexport type OwnCrdtEvent = {\n type: CrdtEventType;\n dataset: string;\n item_id: string;\n payload: string;\n timestamp?: undefined;\n schema_version?: undefined;\n};\n\ntype RemoteCrdtEvent = {\n type: CrdtEventType;\n dataset: string;\n item_id: string;\n payload: string;\n timestamp: string;\n schema_version: number;\n};\n\ntype EnqueuedCrdtEvent = LocalCrdtEvent | OwnCrdtEvent | RemoteCrdtEvent;\n\nexport type GetEventsOptions = {\n afterSyncId?: number;\n status?: CrdtEventStatus;\n excludeOrigin?: string;\n excludeNodeId?: string;\n limit?: number;\n};\n\nexport type GetEventsBatchQuery = {\n afterSyncId: number;\n status: CrdtEventStatus;\n limit: number;\n} & (\n | { excludeOrigin: CrdtEventOrigin; excludeNodeId?: undefined }\n | { excludeOrigin?: undefined; excludeNodeId: string }\n);\n\nexport type GetEventsBatch = {\n events: PersistedCrdtEvent[];\n hasMore: boolean;\n nextSyncId: number;\n};\n\nexport type EnqueueEventsResult = {\n beforeSyncId: number;\n afterSyncId: number;\n /** Resolves when the enqueued events have been processed (applied/deduped/failed). */\n processed: Promise<void>;\n};\n\nexport type EventUpdate = {\n status: CrdtEventStatus;\n schema_version: number;\n type: CrdtEventType;\n dataset: string;\n item_id: string;\n payload: string;\n};\n\ntype StorageHLC = Pick<HLCCounter, \"getNextHLC\" | \"mergeHLC\">;\n\ntype DbSyncerStorage = {\n nodeId: string;\n initialLocalSyncId: number;\n migrator: SyncDbMigrator;\n db: InternalSQLiteWrapper<any>;\n dbConfig: SystemDbConfig;\n hlc: StorageHLC;\n eventHlcAccumulator?: StoredValue<string>;\n onEventApplied?: (event: PersistedCrdtEvent) => void;\n};\n\nexport type CrdtStorage = ReturnType<typeof createCrdtStorage>;\n\ntype EventsAppliedPayload = {\n syncId: number;\n eventHlcSum: string | null;\n};\n\ntype InternalDbSchema = {\n _crdt_events: PersistedCrdtEvent;\n _crdt_update_log: CrdtUpdateLogItem;\n};\n\nexport function createCrdtStorage(storage: DbSyncerStorage) {\n let localSyncId = storage.initialLocalSyncId;\n\n const db = storage.db as InternalSQLiteWrapper<InternalDbSchema>;\n\n const crdtEventsTable = storage.dbConfig.eventsTable.fullIdentifier as \"_crdt_events\";\n\n const eventTarget = createTypedEventTarget<{\n \"events-applied\": EventsAppliedPayload;\n \"remote-event-apply-failed\": { syncId: number };\n }>();\n\n const persistEvent = (tx: InternalSQLiteTransactionWrapper<InternalDbSchema>, event: PersistedCrdtEvent) => {\n tx.executePrepared(\n \"persist-crdt-event\",\n event,\n (db, params) =>\n db.insertInto(crdtEventsTable).values({\n type: params(\"type\"),\n dataset: params(\"dataset\"),\n item_id: params(\"item_id\"),\n payload: params(\"payload\"),\n schema_version: params(\"schema_version\"),\n sync_id: params(\"sync_id\"),\n status: params(\"status\"),\n timestamp: params(\"timestamp\"),\n origin: params(\"origin\"),\n source_node_id: params(\"source_node_id\"),\n }),\n { loggerLevel: \"system\" },\n );\n };\n\n const enqueueEvents = (\n origin: CrdtEventOrigin,\n sourceNodeId: string,\n events: EnqueuedCrdtEvent[],\n ): EnqueueEventsResult => {\n const beforeSyncId = localSyncId;\n if (events.length === 0) {\n return { beforeSyncId, afterSyncId: beforeSyncId, processed: Promise.resolve() };\n }\n\n db.executeTransaction((tx) => {\n for (const event of events) {\n persistEvent(tx, {\n schema_version: event.schema_version ?? storage.migrator.currentSchemaVersion,\n timestamp: event.timestamp ?? serializeHLC(storage.hlc.getNextHLC()),\n type: event.type,\n dataset: event.dataset,\n item_id: event.item_id,\n origin: origin,\n source_node_id: sourceNodeId,\n payload: event.payload,\n sync_id: ++localSyncId,\n status: \"pending\",\n });\n }\n });\n\n return { beforeSyncId, afterSyncId: localSyncId, processed: processEnqueuedEvents() };\n };\n\n const enqueueLocalEvents = (events: LocalCrdtEvent[], sourceNodeId: string): EnqueueEventsResult => {\n return enqueueEvents(\"local\", sourceNodeId, events);\n };\n\n const enqueueOwnEvents = (events: OwnCrdtEvent[]): EnqueueEventsResult => {\n return enqueueEvents(\"own\", storage.nodeId, events);\n };\n\n const enqueueRemoteEvents = (events: RemoteCrdtEvent[]): EnqueueEventsResult => {\n return enqueueEvents(\"remote\", \"\", events);\n };\n\n const notifyEventApplied = (event: PersistedCrdtEvent) => {\n if (event.status === \"applied\") {\n storage.onEventApplied?.(event);\n }\n };\n\n const applyOwnEvent = (event: OwnCrdtEvent, { wrapInTransaction }: { wrapInTransaction?: boolean } = {}) => {\n const persistedEvent: PersistedCrdtEvent = {\n schema_version: storage.migrator.currentSchemaVersion,\n timestamp: serializeHLC(storage.hlc.getNextHLC()),\n type: event.type,\n dataset: event.dataset,\n item_id: event.item_id,\n origin: \"own\",\n source_node_id: storage.nodeId,\n payload: event.payload,\n sync_id: ++localSyncId,\n status: \"pending\",\n };\n\n if (wrapInTransaction) {\n db.executeTransaction((tx) => {\n persistEvent(tx, persistedEvent);\n processPersistedEvent(tx, persistedEvent);\n persistEventHlcAccumulator();\n });\n } else {\n persistEvent(db, persistedEvent);\n processPersistedEvent(db, persistedEvent);\n persistEventHlcAccumulator();\n }\n };\n\n const dispatchEventsApplied = (syncId = localSyncId) => {\n eventTarget.dispatchEvent(\"events-applied\", {\n syncId,\n eventHlcSum: eventHlcAccumulator?.current ?? null,\n });\n };\n\n const hasPendingEvents = (): boolean => {\n const events = db.executePrepared(\n \"has-pending-events\",\n { status: \"pending\" as const },\n (db, params) =>\n db.selectFrom(crdtEventsTable).select(\"sync_id\").where(\"status\", \"=\", params(\"status\")).limit(sql.lit(1)),\n { loggerLevel: \"system\" },\n );\n return events.length > 0;\n };\n\n const getEventsBatch = (options: GetEventsBatchQuery): GetEventsBatch => {\n const limit = options.limit ?? 50;\n\n const queryParams = {\n limit: limit + 1,\n status: options.status ?? null,\n afterSyncId: options.afterSyncId ?? null,\n excludeOrigin: options.excludeOrigin ?? null,\n excludeNodeId: options.excludeNodeId ?? null,\n };\n\n const filterKeys = [\n queryParams.excludeNodeId ? \"nodeid\" : \"no-nodeid\",\n queryParams.excludeOrigin ? \"origin\" : \"no-origin\",\n ];\n\n const events = db.executePrepared(`get-events-batch-${filterKeys.join(\"-\")}`, queryParams, (db, params) =>\n db\n .selectFrom(crdtEventsTable)\n .where(\"sync_id\", \">\", params(\"afterSyncId\"))\n .where(\"status\", \"=\", params(\"status\"))\n .$if(!!queryParams.excludeNodeId, (qb) => qb.where(\"source_node_id\", \"!=\", params(\"excludeNodeId\")))\n .$if(!!queryParams.excludeOrigin, (qb) => qb.where(\"origin\", \"!=\", params(\"excludeOrigin\")))\n .selectAll()\n .limit(params(\"limit\"))\n .orderBy(\"sync_id\", \"asc\"),\n );\n\n const hasMore = events.length > limit;\n if (hasMore) {\n events.pop();\n }\n return {\n events,\n hasMore,\n nextSyncId: events[events.length - 1]?.sync_id ?? options.afterSyncId ?? 0,\n };\n };\n\n // Storage is quiescent when there is nothing left to converge: no events waiting\n // to be applied, and no local applied events past `pushedSyncId` still waiting to\n // be pushed to the remote. When quiescent and caught up, the local and remote node\n // share the exact same set of applied events, so their HLC checksums must match.\n const checkIsQuiescent = (pushedSyncId: number): boolean => {\n if (hasPendingEvents()) {\n return false;\n }\n\n const unpushed = getEventsBatch({\n status: \"applied\",\n afterSyncId: pushedSyncId,\n excludeOrigin: \"remote\",\n limit: 1,\n });\n\n return unpushed.events.length === 0;\n };\n\n const applyCrdtEvent = createSQLiteCrdtApplyFunction({\n db,\n dbConfig: storage.dbConfig,\n });\n const eventHlcAccumulator = storage.eventHlcAccumulator\n ? createEventHlcAccumulator(storage.eventHlcAccumulator.current)\n : null;\n\n const persistEventHlcAccumulator = () => {\n if (eventHlcAccumulator && storage.eventHlcAccumulator) {\n storage.eventHlcAccumulator.current = eventHlcAccumulator.current;\n }\n };\n\n // Rebuild the accumulator from the full applied-event history when it has\n // never been computed for this storage. An empty stored value is the \"never\n // computed\" sentinel: once any event is applied the accumulator is persisted\n // as padded hex, never \"\". This recovers storages created before the\n // accumulator existed and lets us force a recompute by bumping the stored\n // value's key version. The accumulator is a commutative sum, so scan order\n // does not matter.\n const recomputeEventHlcAccumulatorIfNeeded = () => {\n if (!eventHlcAccumulator || !storage.eventHlcAccumulator) {\n return;\n }\n if (storage.eventHlcAccumulator.current !== \"\") {\n return;\n }\n\n const batchSize = 1000;\n let afterSyncId = 0;\n for (;;) {\n const rows = db.executePrepared(\n \"get-applied-event-timestamps\",\n { status: \"applied\" as const, afterSyncId, limit: batchSize },\n (db, params) =>\n db\n .selectFrom(crdtEventsTable)\n .select([\"sync_id\", \"timestamp\"])\n .where(\"status\", \"=\", params(\"status\"))\n .where(\"sync_id\", \">\", params(\"afterSyncId\"))\n .orderBy(\"sync_id\", \"asc\")\n .limit(params(\"limit\")),\n { loggerLevel: \"system\" },\n );\n\n if (rows.length === 0) {\n break;\n }\n\n for (const row of rows) {\n eventHlcAccumulator.add(row.timestamp);\n afterSyncId = row.sync_id;\n }\n\n if (rows.length < batchSize) {\n break;\n }\n }\n\n persistEventHlcAccumulator();\n };\n\n const hasAcceptedEventWithTimestamp = (\n tx: InternalSQLiteTransactionWrapper<InternalDbSchema>,\n event: PersistedCrdtEvent,\n ) => {\n const [existingEvent] = tx.executePrepared(\n \"get-accepted-crdt-event-by-timestamp\",\n {\n timestamp: event.timestamp,\n sync_id: event.sync_id,\n },\n (db, params) =>\n db\n .selectFrom(crdtEventsTable)\n .select(\"sync_id\")\n .where(\"timestamp\", \"=\", params(\"timestamp\"))\n .where(\"sync_id\", \"<\", params(\"sync_id\"))\n .where(\"status\", \"=\", sql.lit(\"applied\"))\n .limit(sql.lit(1)),\n { loggerLevel: \"system\" },\n );\n\n return existingEvent !== undefined;\n };\n\n const processPersistedEvent = (tx: InternalSQLiteTransactionWrapper<InternalDbSchema>, event: PersistedCrdtEvent) => {\n if (event.status !== \"pending\") {\n throw new Error(`Event ${event.sync_id} is not pending`);\n }\n\n try {\n if (hasAcceptedEventWithTimestamp(tx, event)) {\n event.status = \"deduped\";\n return event;\n }\n\n // Always advance HLC, even for no-op events, to maintain monotonic ordering\n if (event.origin === \"local\" || event.origin === \"remote\") {\n storage.hlc.mergeHLC(deserializeHLC(event.timestamp));\n }\n\n if (isNoOpCrdtEventPayload(event.payload)) {\n applyCrdtEvent(event);\n event.status = \"applied\";\n eventHlcAccumulator?.add(event.timestamp);\n return event;\n }\n\n // Migrate event to latest schema version\n const migratedEvent = storage.migrator.migrateEvent(event, storage.migrator.latestSchemaVersion);\n\n if (migratedEvent === null) {\n // Event was dropped during migration (e.g., table was deleted)\n event.schema_version = storage.migrator.latestSchemaVersion;\n event.payload = CRDT_EVENT_NO_OP_PAYLOAD;\n\n applyCrdtEvent(event);\n event.status = \"applied\";\n eventHlcAccumulator?.add(event.timestamp);\n return event;\n }\n\n // Update event with migrated values\n event.schema_version = migratedEvent.schema_version;\n event.type = migratedEvent.type;\n event.dataset = migratedEvent.dataset;\n event.item_id = migratedEvent.item_id;\n event.payload = migratedEvent.payload;\n\n applyCrdtEvent(event);\n event.status = \"applied\";\n eventHlcAccumulator?.add(event.timestamp);\n } catch (error) {\n console.error(\"Error applying enqueued CRDT event\", error);\n event.status = \"failed\";\n } finally {\n tx.executePrepared(\n \"update-crdt-event\",\n event,\n (db, params) =>\n db\n .updateTable(crdtEventsTable)\n .set({\n status: params(\"status\"),\n schema_version: params(\"schema_version\"),\n type: params(\"type\"),\n dataset: params(\"dataset\"),\n item_id: params(\"item_id\"),\n payload: params(\"payload\"),\n })\n .where(\"sync_id\", \"=\", params(\"sync_id\")),\n { loggerLevel: \"system\" },\n );\n }\n };\n\n const processEnqueuedEvents = ensureSingletonExecution(async () => {\n let hasMore = true;\n while (hasMore) {\n await Promise.resolve();\n\n const batchSize = 100;\n\n const events = db.executePrepared(\n \"get-enqueued-pending-events\",\n {\n status: \"pending\" as const,\n limit: batchSize + 1,\n },\n (db, params) =>\n db\n .selectFrom(crdtEventsTable)\n .selectAll()\n .where(\"status\", \"=\", params(\"status\"))\n .limit(params(\"limit\"))\n .orderBy(\"sync_id\", \"asc\"),\n );\n hasMore = events.length > batchSize;\n if (hasMore) {\n events.pop();\n }\n\n if (events.length === 0) {\n break;\n }\n\n let appliedSyncId: number | null = null;\n const failedRemoteSyncIds: number[] = [];\n\n db.executeTransaction((tx) => {\n for (const event of events) {\n processPersistedEvent(tx, event);\n notifyEventApplied(event);\n if (event.status === \"applied\") {\n appliedSyncId = event.sync_id;\n } else if (event.status === \"failed\" && event.origin === \"remote\") {\n failedRemoteSyncIds.push(event.sync_id);\n }\n }\n persistEventHlcAccumulator();\n });\n\n if (appliedSyncId !== null) {\n dispatchEventsApplied(appliedSyncId);\n }\n\n // A remote event was accepted by the server but could not be applied\n // locally, which means our local state has diverged from the server.\n for (const syncId of failedRemoteSyncIds) {\n eventTarget.dispatchEvent(\"remote-event-apply-failed\", { syncId });\n }\n }\n });\n\n recomputeEventHlcAccumulatorIfNeeded();\n\n void processEnqueuedEvents();\n\n return {\n getEventsBatch,\n enqueueLocalEvents,\n enqueueOwnEvents,\n enqueueRemoteEvents,\n applyOwnEvent,\n dispatchEventsApplied,\n checkIsQuiescent,\n getEventHlcAccumulator: () => eventHlcAccumulator?.current ?? null,\n\n addEventListener: eventTarget.addEventListener,\n removeEventListener: eventTarget.removeEventListener,\n };\n}\n","import { xxhash } from \"../hash\";\n\nconst MASK_128 = (1n << 128n) - 1n;\nconst HEX_128_PATTERN = /^[0-9a-f]{32}$/;\n\nexport function createEventHlcAccumulator(initialValue: string) {\n let current = parseHex128(initialValue);\n\n return {\n add(timestamp: string) {\n current = (current + hash128BigInt(timestamp)) & MASK_128;\n },\n get current() {\n return toHex128(current);\n },\n };\n}\n\nfunction parseHex128(value: string) {\n if (value === \"\") {\n return 0n;\n }\n const normalized = value.toLowerCase();\n if (!HEX_128_PATTERN.test(normalized)) {\n throw new Error(`Invalid event HLC accumulator value: ${value}`);\n }\n return BigInt(`0x${normalized}`);\n}\n\nfunction toHex128(value: bigint) {\n return value.toString(16).padStart(32, \"0\");\n}\n\nfunction hash128BigInt(value: string) {\n return xxhash.h64(value, 0n) | (xxhash.h64(value, 1n) << 64n);\n}\n","import type { CrdtStorage } from \"./crdt-storage\";\n\ntype CrdtSyncProducer = {\n storage: CrdtStorage;\n broadcastEvents: (request: { newSyncId: number; eventHlcSum: string | null }) => void;\n};\n\nexport const createCrdtSyncProducer = ({ storage, broadcastEvents }: CrdtSyncProducer) => {\n storage.addEventListener(\"events-applied\", (event) => {\n broadcastEvents({\n newSyncId: event.payload.syncId,\n eventHlcSum: event.payload.eventHlcSum,\n });\n });\n};\n","import retryAsPromised from \"retry-as-promised\";\nimport type { SyncDbMigrator } from \"../migrations/migrator\";\nimport { createTypedEventTarget, ensureSingletonExecution, tryCatchAsync } from \"../utils\";\nimport type { EventsPullResponse } from \"../worker-db/worker-common\";\nimport type { PendingCrdtEvent } from \"./apply-crdt-event\";\nimport type { CrdtStorage } from \"./crdt-storage\";\nimport type { StoredValue } from \"./stored-value\";\n\ntype CrdtSyncRemoteSourceConfig = {\n bufferSize: number;\n storage: CrdtStorage;\n migrator: SyncDbMigrator;\n pullSyncId: StoredValue<number>;\n pushSyncId: StoredValue<number>;\n nodeId: string;\n remoteFactory?: CreateRemoteSourceFactory;\n};\n\nexport type EventsPullRequest = {\n afterSyncId: number;\n excludeNodeId?: string;\n};\n\nexport type EventsPushRequest = {\n nodeId: string;\n events: (PendingCrdtEvent & { schema_version: number })[];\n};\nexport type EventsPushResponse = {\n ok: boolean;\n /** Remote sync_id right before the pushed events were enqueued. */\n beforeSyncId?: number;\n /** Remote sync_id right after the pushed events were enqueued. */\n afterSyncId?: number;\n};\n\nexport type CrdtSyncRemoteSource = ReturnType<typeof createCrdtSyncRemoteSource>;\n\nexport type EventsAvailable = {\n newSyncId: number;\n remoteEventHlcSum: string | null;\n};\n\nexport type CreateRemoteSourceFactory = (opts: {\n onEventsAvailable: (event: EventsAvailable) => void;\n}) => RemoteSource | Promise<RemoteSource>;\n\ntype RemoteSource = {\n pullEvents: (request: EventsPullRequest) => Promise<EventsPullResponse>;\n pushEvents: (request: EventsPushRequest) => Promise<EventsPushResponse>;\n disconnect?: () => void | Promise<void>;\n};\n\nexport class SchemaVersionMismatchError extends Error {\n constructor(\n public remoteSchemaVersion: number,\n public localSchemaVersion: number,\n ) {\n super(`Schema version mismatch: remote ${remoteSchemaVersion} != local ${localSchemaVersion}`);\n this.name = \"SchemaVersionMismatchError\";\n }\n}\n\ntype RemoteSourceState =\n | {\n type: \"pending\";\n }\n | {\n type: \"offline\";\n reason: OfflineReason;\n }\n | {\n type: \"online\";\n source: RemoteSource;\n };\n\nexport type OfflineReason =\n | \"NOT_INITIALIZED\"\n | \"INITIALIZATION_FAILED\"\n | \"REMOTE_PUSH_ERROR\"\n | \"REMOTE_PULL_ERROR\"\n | \"DISCONNECTED\";\n\nexport type DeSyncDetectedReason = \"CHECKSUM_MISMATCH\" | \"ERROR_APPLYING_REMOTE_EVENT\";\n\nexport const createCrdtSyncRemoteSource = ({\n bufferSize,\n storage,\n migrator,\n pullSyncId,\n pushSyncId,\n nodeId,\n remoteFactory,\n}: CrdtSyncRemoteSourceConfig) => {\n const eventTarget = createTypedEventTarget<{\n \"state-changed\": RemoteSourceState[\"type\"];\n \"de-sync-detected\": {\n reason: DeSyncDetectedReason;\n };\n \"remote-schema-version-mismatch\": {\n remoteSchemaVersion: number;\n localSchemaVersion: number;\n };\n }>();\n\n let remoteState: RemoteSourceState = { type: \"offline\", reason: \"NOT_INITIALIZED\" };\n\n const setRemoteState = (state: RemoteSourceState) => {\n remoteState = state;\n eventTarget.dispatchEvent(\"state-changed\", state.type);\n };\n\n const initRemote = ensureSingletonExecution(\n async () => {\n if (remoteState.type !== \"offline\") {\n throw new Error(\"Remote source is not offline\");\n }\n\n if (!remoteFactory) {\n console.warn(\"Remote source factory not provided. Going offline.\");\n setRemoteState({ type: \"offline\", reason: \"NOT_INITIALIZED\" });\n return;\n }\n\n setRemoteState({ type: \"pending\" });\n\n const factoryResult = await tryCatchAsync(async () => {\n return await remoteFactory?.({\n onEventsAvailable: ({ newSyncId, remoteEventHlcSum }) => {\n pullEvents({ remoteSyncId: newSyncId, remoteEventHlcSum, includeSelf: false });\n },\n });\n });\n\n if (!factoryResult.success) {\n setRemoteState({ type: \"offline\", reason: \"INITIALIZATION_FAILED\" });\n console.warn(\"Failed to create remote source\", factoryResult.error);\n return;\n }\n\n setRemoteState({\n type: \"online\",\n source: factoryResult.data,\n });\n },\n { queueReExecution: false },\n );\n\n const syncWithRemote = ensureSingletonExecution(\n async () => {\n if (remoteState.type !== \"online\") {\n return;\n }\n\n await pullEvents();\n await startPushingEvents();\n },\n { queueReExecution: false },\n );\n\n const goOffline = ensureSingletonExecution(\n async (reason: OfflineReason) => {\n if (remoteState.type !== \"online\") {\n return;\n }\n const source = remoteState.source;\n\n setRemoteState({ type: \"pending\" });\n\n const disconnectResult = await tryCatchAsync(async () => {\n return await source.disconnect?.();\n });\n\n if (!disconnectResult.success) {\n console.warn(\"Error while disconnecting from remote source\", disconnectResult.error);\n }\n\n setRemoteState({ type: \"offline\", reason });\n },\n { queueReExecution: false },\n );\n\n const goOnline = async () => {\n if (remoteState.type !== \"online\") {\n await initRemote();\n }\n\n if (remoteState.type === \"online\") {\n await syncWithRemote();\n }\n };\n\n let requestedPullSyncId: number | null = null;\n let pullPromise: Promise<void> | null = null;\n const pullEvents = (request?: {\n remoteSyncId?: number;\n remoteEventHlcSum?: string | null;\n includeSelf?: boolean;\n }) => {\n if (remoteState.type !== \"online\") {\n return Promise.resolve();\n }\n\n const remoteSyncId = request?.remoteSyncId;\n\n if (remoteSyncId !== undefined && remoteSyncId <= pullSyncId.current) {\n // We are already caught up to this broadcast, so there is nothing to pull.\n // This is the quiescent moment to verify we have not diverged from the\n // remote (the check is a no-op unless we are exactly aligned: remoteSyncId\n // === pullSyncId.current).\n checkRemoteConsistency(remoteSyncId, request?.remoteEventHlcSum ?? null);\n return Promise.resolve();\n }\n\n if (pullPromise) {\n if (remoteSyncId !== undefined && (!requestedPullSyncId || requestedPullSyncId < remoteSyncId)) {\n requestedPullSyncId = remoteSyncId;\n }\n return pullPromise;\n }\n\n pullPromise = pullAllEvents({\n afterSyncId: pullSyncId.current,\n excludeNodeId: request?.includeSelf ? undefined : nodeId,\n })\n .catch((error) => {\n console.error(\"Error pulling events. Going offline.\", error);\n goOffline(\"REMOTE_PULL_ERROR\");\n })\n .finally(() => {\n pullPromise = null;\n\n const nextTarget = requestedPullSyncId;\n requestedPullSyncId = null;\n\n if (nextTarget && nextTarget > pullSyncId.current) {\n pullEvents({ remoteSyncId: nextTarget });\n }\n });\n return pullPromise;\n };\n\n const pullAllEvents = async (opts: EventsPullRequest) => {\n let hasMore = true;\n let afterSyncId = opts.afterSyncId;\n while (hasMore) {\n if (remoteState.type !== \"online\") {\n return;\n }\n const source = remoteState.source;\n\n const response = await retryAsPromised(\n () =>\n source.pullEvents({\n ...opts,\n afterSyncId,\n }),\n {\n max: 3,\n backoffBase: 100,\n backoffExponent: 1.5,\n backoffJitter: 150,\n timeout: 10000,\n },\n );\n hasMore = response.hasMore;\n afterSyncId = response.nextSyncId;\n\n if (response.events) {\n storage.enqueueRemoteEvents(\n response.events.map((x) => {\n if (x.schema_version > migrator.currentSchemaVersion) {\n eventTarget.dispatchEvent(\"remote-schema-version-mismatch\", {\n remoteSchemaVersion: x.schema_version,\n localSchemaVersion: migrator.currentSchemaVersion,\n });\n throw new SchemaVersionMismatchError(x.schema_version, migrator.currentSchemaVersion);\n }\n return x;\n }),\n );\n }\n if (response.nextSyncId <= pullSyncId.current) {\n break;\n }\n if (response.nextSyncId > pullSyncId.current) {\n pullSyncId.current = response.nextSyncId;\n }\n }\n };\n\n // De-sync detection: when we are exactly caught up to the remote's broadcast\n // sync id and fully quiescent, our applied-event set must equal the remote's,\n // so our HLC checksums must match. A mismatch means the nodes have diverged.\n const checkRemoteConsistency = (remoteSyncId: number, remoteEventHlcSum: string | null) => {\n // A remote with no accumulator gives us nothing to compare against.\n if (remoteEventHlcSum === null) {\n return;\n }\n\n // Only meaningful when we are exactly caught up: if we are behind we still\n // need to pull; if we are ahead our state covers events the remote checksum\n // does not.\n if (remoteSyncId !== pullSyncId.current) {\n return;\n }\n\n // Quiescence: the accumulator only matches the remote's when nothing is left\n // to apply locally and no local applied events are still waiting to be pushed\n // (those are in our accumulator but the remote has not seen them yet).\n if (!storage.checkIsQuiescent(pushSyncId.current)) {\n return;\n }\n\n const localEventHlcSum = storage.getEventHlcAccumulator();\n if (localEventHlcSum === null) {\n return;\n }\n\n if (localEventHlcSum !== remoteEventHlcSum) {\n eventTarget.dispatchEvent(\"de-sync-detected\", { reason: \"CHECKSUM_MISMATCH\" });\n console.warn(\n `[sqlite-sync] De-sync detected at syncId ${remoteSyncId}: local HLC checksum ${localEventHlcSum} != remote ${remoteEventHlcSum}. Local and remote have diverged despite being caught up.`,\n );\n }\n };\n\n const startPushingEvents = ensureSingletonExecution(async () => {\n while (true) {\n const eventsBatch = storage.getEventsBatch({\n status: \"applied\",\n afterSyncId: pushSyncId.current,\n excludeOrigin: \"remote\",\n limit: bufferSize,\n });\n if (eventsBatch.events.length === 0) {\n break;\n }\n\n if (remoteState.type !== \"online\") {\n break;\n }\n const source = remoteState.source;\n\n let response: EventsPushResponse;\n try {\n response = await retryAsPromised(\n () =>\n source.pushEvents({\n nodeId,\n events: eventsBatch.events.map((event) => ({\n schema_version: event.schema_version,\n timestamp: event.timestamp,\n type: event.type,\n dataset: event.dataset,\n item_id: event.item_id,\n payload: event.payload,\n })),\n }),\n {\n max: 3,\n backoffBase: 100,\n backoffExponent: 1.5,\n backoffJitter: 150,\n timeout: 10000,\n },\n );\n } catch (error) {\n console.error(\"Error pushing events. Going offline.\", error);\n goOffline(\"REMOTE_PUSH_ERROR\");\n return;\n }\n\n pushSyncId.current = eventsBatch.nextSyncId;\n\n // Fast-forward the pull cursor: the remote assigns sync ids for the pushed\n // events synchronously, so (beforeSyncId, afterSyncId] contains only this\n // node's own events. If we are caught up to at least beforeSyncId, the skipped\n // range (pullSyncId, afterSyncId] contains only our own events, so there is\n // nothing to pull up to afterSyncId.\n if (\n response.ok &&\n response.beforeSyncId !== undefined &&\n response.afterSyncId !== undefined &&\n response.beforeSyncId <= pullSyncId.current &&\n response.afterSyncId > pullSyncId.current\n ) {\n pullSyncId.current = response.afterSyncId;\n }\n if (!eventsBatch.hasMore) {\n break;\n }\n }\n });\n\n const eventsAppliedSubscription = storage.addEventListener(\"events-applied\", () => {\n startPushingEvents();\n });\n\n const remoteEventApplyFailedSubscription = storage.addEventListener(\"remote-event-apply-failed\", () => {\n eventTarget.dispatchEvent(\"de-sync-detected\", { reason: \"ERROR_APPLYING_REMOTE_EVENT\" });\n });\n\n const getState = (): \"pending\" | \"offline\" | \"online\" => remoteState.type;\n\n const dispose = async () => {\n await goOffline(\"DISCONNECTED\");\n eventsAppliedSubscription.unsubscribe();\n remoteEventApplyFailedSubscription.unsubscribe();\n };\n\n return {\n goOnline,\n goOffline,\n syncWithRemote,\n getState,\n dispose,\n addEventListener: eventTarget.addEventListener,\n removeEventListener: eventTarget.removeEventListener,\n };\n};\n","import { createStore, del, get, set } from \"idb-keyval\";\nimport { generateId } from \"../utils\";\nimport type { WorkerNotificationMessage } from \"./worker-common\";\n\nexport type ResetRequest = {\n epoch: string;\n requestedAt: number;\n};\n\n/** Durable async key-value storage for reset state. Injectable for tests. */\nexport type ResetStore = {\n get: <T>(key: string) => Promise<T | undefined>;\n set: (key: string, value: unknown) => Promise<void>;\n delete: (key: string) => Promise<void>;\n};\n\n/** Default IndexedDB-backed reset store (idb-keyval over a dedicated database). */\nexport function createIdbResetStore(): ResetStore {\n const store = createStore(\"sqlite-sync\", \"kv\");\n return {\n get: (key) => get(key, store),\n set: (key, value) => set(key, value, store),\n delete: (key) => del(key, store),\n };\n}\n\n/**\n * A clean reset is a recovery action for a de-sync detected now. If the reload\n * never happens (broadcast lost, tab crashed mid-reload, browser killed the page),\n * the request must not fire on an arbitrary later cold start and silently wipe\n * local-only writes accumulated since.\n */\nexport const RESET_REQUEST_TTL_MS = 10 * 60 * 1000; // 10 minutes\n\nconst resetRequestKey = (dbId: string) => `sqlite-sync-reset-request-${dbId}`;\nconst resetAppliedKey = (dbId: string) => `sqlite-sync-reset-applied-${dbId}`;\n\ntype ResetStateStoreOptions = {\n store: ResetStore;\n dbId: string;\n now?: () => number;\n};\n\nexport type ResetStateStore = ReturnType<typeof createResetStateStore>;\n\n/**\n * Worker-owned durable reset state. The reset decision must be owned by the\n * elected worker, not by tabs during `createSyncedDb` — otherwise a later\n * worker election could repeat an already-applied wipe.\n */\nexport function createResetStateStore({ store, dbId, now = () => Date.now() }: ResetStateStoreOptions) {\n const requestKey = resetRequestKey(dbId);\n const appliedKey = resetAppliedKey(dbId);\n\n return {\n async writeResetRequest(epoch: string): Promise<ResetRequest> {\n const request: ResetRequest = { epoch, requestedAt: now() };\n await store.set(requestKey, request);\n return request;\n },\n /**\n * Read the pending reset request after winning the worker election.\n * Returns the request only when it has not been applied yet and is within\n * the TTL. Stale requests are deleted so they cannot fire on a later cold start.\n */\n async resolvePendingReset(): Promise<ResetRequest | undefined> {\n const request = await store.get<ResetRequest>(requestKey);\n if (!request) {\n return undefined;\n }\n\n if (now() - request.requestedAt > RESET_REQUEST_TTL_MS) {\n await store.delete(requestKey);\n return undefined;\n }\n\n const appliedEpoch = await store.get<string>(appliedKey);\n if (request.epoch === appliedEpoch) {\n return undefined;\n }\n\n return request;\n },\n /**\n * Record the epoch as applied. Must be called only after the worker has\n * successfully initialized with `clearOnInit: true`, so a failed init can\n * be retried by a later elected worker.\n */\n async markResetApplied(epoch: string): Promise<void> {\n await store.set(appliedKey, epoch);\n },\n };\n}\n\ntype ReloadRequestHandlerOptions = {\n resetState: ResetStateStore;\n broadcast: (message: WorkerNotificationMessage) => void;\n generateEpoch?: () => string;\n};\n\n/**\n * Worker-side `requestReload` RPC implementation. For `clean: true` the reset\n * request is durably stored before broadcasting and before the RPC resolves,\n * so the epoch survives no matter which path triggers the reload.\n */\nexport function createReloadRequestHandler({\n resetState,\n broadcast,\n generateEpoch = generateId,\n}: ReloadRequestHandlerOptions) {\n return async (options: { clean: boolean }): Promise<void> => {\n const reloadEpoch = generateEpoch();\n\n if (options.clean) {\n await resetState.writeResetRequest(reloadEpoch);\n }\n\n broadcast({\n notificationType: \"reload-requested\",\n reloadEpoch,\n clean: options.clean,\n });\n };\n}\n","import type {\n DeSyncDetectedReason,\n EventsPullRequest,\n EventsPushRequest,\n EventsPushResponse,\n} from \"../sqlite-crdt/crdt-sync-remote-source\";\nimport type { CrdtEventType } from \"../sqlite-crdt/crdt-table-schema\";\nimport type { ExecuteParams, ExecuteResult } from \"../sqlite-db-wrapper\";\nimport { TypedBroadcastChannel } from \"../utils\";\n\nexport const syncDbWorkerLockName = \"sync-db-worker-lock\";\nexport const syncDbClientLockName = \"sync-db-client-lock\";\n\nexport type WorkerNotificationMessage =\n | {\n notificationType: \"new-event-chunk-applied\";\n newSyncId: number;\n /** Worker's HLC checksum after applying up to `newSyncId`. */\n eventHlcSum: string | null;\n }\n | {\n notificationType: \"state-changed\";\n state: WorkerState;\n }\n | {\n notificationType: \"reload-requested\";\n reloadEpoch: string;\n clean: boolean;\n }\n | {\n notificationType: \"de-sync-detected\";\n reason: DeSyncDetectedReason;\n }\n | {\n notificationType: \"remote-schema-version-mismatch\";\n remoteSchemaVersion: number;\n localSchemaVersion: number;\n };\n\nexport type WorkerState = {\n remoteState: \"online\" | \"offline\" | \"pending\";\n};\n\nexport type GetSnapshotResponse = {\n file: Uint8Array<ArrayBufferLike>;\n syncId: number;\n schemaVersion: number;\n};\n\nexport type EventsPullResponse = {\n events: {\n schema_version: number;\n type: CrdtEventType;\n timestamp: string;\n dataset: string;\n item_id: string;\n payload: string;\n }[];\n hasMore: boolean;\n nextSyncId: number;\n};\n\nexport interface WorkerRpc {\n getSnapshot: () => GetSnapshotResponse;\n pushTabEvents: (request: EventsPushRequest) => EventsPushResponse;\n execute: (query: ExecuteParams) => ExecuteResult<unknown>;\n pullEvents: (params: EventsPullRequest) => EventsPullResponse;\n postState: () => void;\n goOnline: () => Promise<void>;\n goOffline: () => void;\n requestReload: (options: { clean: boolean }) => Promise<void>;\n}\n\nexport type WorkerRequestMethod = keyof WorkerRpc;\n\nexport type WorkerRequestMessage<TMethod extends WorkerRequestMethod = WorkerRequestMethod> = {\n type: \"request\";\n requestId: string;\n method: TMethod;\n args: Parameters<WorkerRpc[TMethod]>;\n};\n\nexport type WorkerResponseMessage<TMethod extends WorkerRequestMethod = WorkerRequestMethod> = {\n type: \"response\";\n requestId: string;\n data: ReturnType<WorkerRpc[TMethod]>;\n};\n\nexport type WorkerErrorResponseMessage = {\n type: \"error-response\";\n requestId: string;\n error: string;\n};\n\nexport type AsyncRpc<T> = {\n [K in keyof T]: T[K] extends (...args: infer U) => infer V ? (...args: U) => Promise<Awaited<V>> : never;\n};\n\nexport const broadcastChannelNames = {\n requests: \"sync-db-worker-requests\",\n responses: \"sync-db-worker-responses\",\n} as const;\n\nexport type WorkerBroadcastChannels = {\n requests: TypedBroadcastChannel<WorkerRequestMessage>;\n responses: TypedBroadcastChannel<WorkerResponseMessage | WorkerErrorResponseMessage | WorkerNotificationMessage>;\n};\n\nexport const createBroadcastChannels = (prefix: string): WorkerBroadcastChannels => {\n return {\n requests: new TypedBroadcastChannel(`${prefix}-${broadcastChannelNames.requests}`),\n responses: new TypedBroadcastChannel(`${prefix}-${broadcastChannelNames.responses}`),\n };\n};\n\nexport type WorkerConfig<Props = any> = {\n dbId: string;\n clientId: string;\n props: Props;\n};\n\nexport type WorkerInitMessage = {\n type: \"init\";\n config: WorkerConfig;\n};\n\nexport function isWorkerInitMessage(message: unknown): message is WorkerInitMessage {\n return typeof message === \"object\" && message !== null && \"type\" in message && message.type === \"init\";\n}\n\nexport function isWorkerRequestMessage(message: unknown): message is WorkerRequestMessage {\n return typeof message === \"object\" && message !== null && \"type\" in message && message.type === \"request\";\n}\n\nexport function isWorkerResponseMessage(message: unknown): message is WorkerResponseMessage {\n return (\n typeof message === \"object\" && message !== null && \"type\" in message && \"requestId\" in message && \"data\" in message\n );\n}\n\nexport function isWorkerErrorResponseMessage(message: unknown): message is WorkerErrorResponseMessage {\n return typeof message === \"object\" && message !== null && \"type\" in message && message.type === \"error-response\";\n}\n\nexport function isWorkerNotificationMessage(message: unknown): message is WorkerNotificationMessage {\n return typeof message === \"object\" && message !== null && \"notificationType\" in message && !!message.notificationType;\n}\n"],"mappings":"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","names":["db","sql","sql","perf","sql","sql","sql","get","db","set","sql","db","sql","db","sql"]}
|