Rewrite SqliteWorker with Comlink
This commit is contained in:
parent
455752e656
commit
a4e7c241d0
|
@ -4,7 +4,7 @@ import path from 'node:path';
|
||||||
import { FileMigrationProvider, Kysely, Migrator, PolySqliteDialect } from '@/deps.ts';
|
import { FileMigrationProvider, Kysely, Migrator, PolySqliteDialect } from '@/deps.ts';
|
||||||
import { Conf } from '@/config.ts';
|
import { Conf } from '@/config.ts';
|
||||||
import { getPragma, setPragma } from '@/pragma.ts';
|
import { getPragma, setPragma } from '@/pragma.ts';
|
||||||
import { sqliteWorker } from '@/workers.ts';
|
import SqliteWorker from '@/workers/sqlite.ts';
|
||||||
|
|
||||||
interface DittoDB {
|
interface DittoDB {
|
||||||
events: EventRow;
|
events: EventRow;
|
||||||
|
@ -57,8 +57,8 @@ interface UnattachedMediaRow {
|
||||||
uploaded_at: Date;
|
uploaded_at: Date;
|
||||||
}
|
}
|
||||||
|
|
||||||
await sqliteWorker.ready;
|
const sqliteWorker = new SqliteWorker();
|
||||||
await sqliteWorker.open();
|
await sqliteWorker.open(Conf.dbPath);
|
||||||
|
|
||||||
const db = new Kysely<DittoDB>({
|
const db = new Kysely<DittoDB>({
|
||||||
dialect: new PolySqliteDialect({
|
dialect: new PolySqliteDialect({
|
||||||
|
|
|
@ -1,6 +0,0 @@
|
||||||
import { Conf } from '@/config.ts';
|
|
||||||
import SqliteWorker from '@/workers/sqlite.ts';
|
|
||||||
|
|
||||||
const sqliteWorker = new SqliteWorker(Conf.dbPath);
|
|
||||||
|
|
||||||
export { sqliteWorker };
|
|
|
@ -1,15 +1,18 @@
|
||||||
|
import { Comlink } from '@/deps.ts';
|
||||||
|
|
||||||
|
import type { SqliteWorker as _SqliteWorker } from './sqlite.worker.ts';
|
||||||
import type { CompiledQuery, QueryResult } from '@/deps.ts';
|
import type { CompiledQuery, QueryResult } from '@/deps.ts';
|
||||||
|
|
||||||
class SqliteWorker {
|
class SqliteWorker {
|
||||||
#path: string;
|
|
||||||
#worker: Worker;
|
#worker: Worker;
|
||||||
ready: Promise<void>;
|
#client: ReturnType<typeof Comlink.wrap<typeof _SqliteWorker>>;
|
||||||
|
#ready: Promise<void>;
|
||||||
|
|
||||||
constructor(path: string) {
|
constructor() {
|
||||||
this.#path = path;
|
|
||||||
this.#worker = new Worker(new URL('./sqlite.worker.ts', import.meta.url).href, { type: 'module' });
|
this.#worker = new Worker(new URL('./sqlite.worker.ts', import.meta.url).href, { type: 'module' });
|
||||||
|
this.#client = Comlink.wrap<typeof _SqliteWorker>(this.#worker);
|
||||||
|
|
||||||
this.ready = new Promise<void>((resolve) => {
|
this.#ready = new Promise<void>((resolve) => {
|
||||||
const handleEvent = (event: MessageEvent) => {
|
const handleEvent = (event: MessageEvent) => {
|
||||||
if (event.data[0] === 'ready') {
|
if (event.data[0] === 'ready') {
|
||||||
this.#worker.removeEventListener('message', handleEvent);
|
this.#worker.removeEventListener('message', handleEvent);
|
||||||
|
@ -20,37 +23,18 @@ class SqliteWorker {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
async open(): Promise<void> {
|
async open(path: string): Promise<void> {
|
||||||
await this.ready;
|
await this.#ready;
|
||||||
return this.#call(['open', [this.#path]]);
|
return this.#client.open(path);
|
||||||
}
|
}
|
||||||
|
|
||||||
async executeQuery<R>({ sql, parameters }: CompiledQuery): Promise<QueryResult<R>> {
|
async executeQuery<R>(query: CompiledQuery): Promise<QueryResult<R>> {
|
||||||
await this.ready;
|
await this.#ready;
|
||||||
return this.#call(['query', [sql, parameters]]);
|
return this.#client.executeQuery(query) as Promise<QueryResult<R>>;
|
||||||
}
|
}
|
||||||
|
|
||||||
#call<T>(msg: [string, unknown[]]): Promise<T> {
|
destroy(): Promise<void> {
|
||||||
const id = crypto.randomUUID();
|
return this.#client.destroy();
|
||||||
|
|
||||||
this.#worker.postMessage([id, msg]);
|
|
||||||
|
|
||||||
// TODO: use a hashmap instead of an event listener for better performance.
|
|
||||||
return new Promise((resolve) => {
|
|
||||||
const handleEvent = (event: MessageEvent<[string, T]>) => {
|
|
||||||
const [_id, result] = event.data;
|
|
||||||
if (_id === id) {
|
|
||||||
this.#worker.removeEventListener('message', handleEvent);
|
|
||||||
resolve(result);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
this.#worker.addEventListener('message', handleEvent);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
// deno-lint-ignore require-await
|
|
||||||
async destroy() {
|
|
||||||
this.#worker.terminate();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,38 +1,26 @@
|
||||||
/// <reference lib="webworker" />
|
/// <reference lib="webworker" />
|
||||||
|
|
||||||
import { DenoSqlite3 } from '@/deps.ts';
|
import { Comlink, type CompiledQuery, DenoSqlite3, type QueryResult } from '@/deps.ts';
|
||||||
|
|
||||||
let db: DenoSqlite3;
|
let db: DenoSqlite3 | undefined;
|
||||||
|
|
||||||
type Msg =
|
export const SqliteWorker = {
|
||||||
| ['open', [string]]
|
open(path: string): void {
|
||||||
| ['query', [string, unknown[]]];
|
db = new DenoSqlite3(path);
|
||||||
|
},
|
||||||
|
executeQuery<R>({ sql, parameters }: CompiledQuery): QueryResult<R> {
|
||||||
|
if (!db) throw new Error('Database not open');
|
||||||
|
return {
|
||||||
|
rows: db.prepare(sql).all(...parameters as any[]) as R[],
|
||||||
|
numAffectedRows: BigInt(db.changes),
|
||||||
|
insertId: BigInt(db.lastInsertRowId),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
destroy() {
|
||||||
|
db?.close();
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
function call([cmd, args]: Msg) {
|
Comlink.expose(SqliteWorker);
|
||||||
switch (cmd) {
|
|
||||||
case 'open':
|
|
||||||
return handleOpen(args[0]);
|
|
||||||
case 'query':
|
|
||||||
return handleQuery(args[0], args[1]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function handleOpen(path: string): void {
|
|
||||||
db = new DenoSqlite3(path);
|
|
||||||
}
|
|
||||||
|
|
||||||
function handleQuery(sql: string, params: any[] = []) {
|
|
||||||
return {
|
|
||||||
rows: db.prepare(sql).all(...params),
|
|
||||||
numAffectedRows: BigInt(db.changes),
|
|
||||||
insertId: BigInt(db.lastInsertRowId),
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
self.addEventListener('message', (event: MessageEvent<[string, Msg]>) => {
|
|
||||||
const [id, msg] = event.data;
|
|
||||||
const result = call(msg);
|
|
||||||
self.postMessage([id, result]);
|
|
||||||
});
|
|
||||||
|
|
||||||
self.postMessage(['ready']);
|
self.postMessage(['ready']);
|
||||||
|
|
Loading…
Reference in New Issue