/* This file is part of TALER (C) 2016 GNUnet e.V. TALER is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 3, or (at your option) any later version. TALER is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with TALER; see the file COPYING. If not, see */ /** * Database query abstractions. * @module Query * @author Florian Dold */ /** * Imports. */ import { openPromise } from "./promiseUtils.js"; import { IDBRequest, IDBTransaction, IDBValidKey, IDBDatabase, IDBFactory, IDBVersionChangeEvent, IDBCursor, IDBKeyPath, } from "@gnu-taler/idb-bridge"; import { Logger } from "@gnu-taler/taler-util"; import { performanceNow } from "./timer.js"; const logger = new Logger("query.ts"); /** * Exception that should be thrown by client code to abort a transaction. */ export const TransactionAbort = Symbol("transaction_abort"); /** * Options for an index. */ export interface IndexOptions { /** * If true and the path resolves to an array, create an index entry for * each member of the array (instead of one index entry containing the full array). * * Defaults to false. */ multiEntry?: boolean; /** * Database version that this store was added in, or * undefined if added in the first version. */ versionAdded?: number; } function requestToPromise(req: IDBRequest): Promise { const stack = Error("Failed request was started here."); return new Promise((resolve, reject) => { req.onsuccess = () => { resolve(req.result); }; req.onerror = () => { console.error("error in DB request", req.error); reject(req.error); console.error("Request failed:", stack); }; }); } type CursorResult = CursorEmptyResult | CursorValueResult; interface CursorEmptyResult { hasValue: false; } interface CursorValueResult { hasValue: true; value: T; } class TransactionAbortedError extends Error { constructor(m: string) { super(m); // Set the prototype explicitly. Object.setPrototypeOf(this, TransactionAbortedError.prototype); } } class ResultStream { private currentPromise: Promise; private gotCursorEnd = false; private awaitingResult = false; constructor(private req: IDBRequest) { this.awaitingResult = true; let p = openPromise(); this.currentPromise = p.promise; req.onsuccess = () => { if (!this.awaitingResult) { throw Error("BUG: invariant violated"); } const cursor = req.result; if (cursor) { this.awaitingResult = false; p.resolve(); p = openPromise(); this.currentPromise = p.promise; } else { this.gotCursorEnd = true; p.resolve(); } }; req.onerror = () => { p.reject(req.error); }; } async toArray(): Promise { const arr: T[] = []; while (true) { const x = await this.next(); if (x.hasValue) { arr.push(x.value); } else { break; } } return arr; } async map(f: (x: T) => R): Promise { const arr: R[] = []; while (true) { const x = await this.next(); if (x.hasValue) { arr.push(f(x.value)); } else { break; } } return arr; } async mapAsync(f: (x: T) => Promise): Promise { const arr: R[] = []; while (true) { const x = await this.next(); if (x.hasValue) { arr.push(await f(x.value)); } else { break; } } return arr; } async forEachAsync(f: (x: T) => Promise): Promise { while (true) { const x = await this.next(); if (x.hasValue) { await f(x.value); } else { break; } } } async forEach(f: (x: T) => void): Promise { while (true) { const x = await this.next(); if (x.hasValue) { f(x.value); } else { break; } } } async filter(f: (x: T) => boolean): Promise { const arr: T[] = []; while (true) { const x = await this.next(); if (x.hasValue) { if (f(x.value)) { arr.push(x.value); } } else { break; } } return arr; } async next(): Promise> { if (this.gotCursorEnd) { return { hasValue: false }; } if (!this.awaitingResult) { const cursor: IDBCursor | undefined = this.req.result; if (!cursor) { throw Error("assertion failed"); } this.awaitingResult = true; cursor.continue(); } await this.currentPromise; if (this.gotCursorEnd) { return { hasValue: false }; } const cursor = this.req.result; if (!cursor) { throw Error("assertion failed"); } return { hasValue: true, value: cursor.value }; } } /** * Return a promise that resolves to the opened IndexedDB database. */ export function openDatabase( idbFactory: IDBFactory, databaseName: string, databaseVersion: number, onVersionChange: () => void, onUpgradeNeeded: ( db: IDBDatabase, oldVersion: number, newVersion: number, upgradeTransaction: IDBTransaction, ) => void, ): Promise { return new Promise((resolve, reject) => { const req = idbFactory.open(databaseName, databaseVersion); req.onerror = (e) => { logger.error("database error", e); reject(new Error("database error")); }; req.onsuccess = (e) => { req.result.onversionchange = (evt: IDBVersionChangeEvent) => { logger.info( `handling live db version change from ${evt.oldVersion} to ${evt.newVersion}`, ); req.result.close(); onVersionChange(); }; resolve(req.result); }; req.onupgradeneeded = (e) => { const db = req.result; const newVersion = e.newVersion; if (!newVersion) { throw Error("upgrade needed, but new version unknown"); } const transaction = req.transaction; if (!transaction) { throw Error("no transaction handle available in upgrade handler"); } onUpgradeNeeded(db, e.oldVersion, newVersion, transaction); }; }); } export interface IndexDescriptor { name: string; keyPath: IDBKeyPath | IDBKeyPath[]; multiEntry?: boolean; } export interface StoreDescriptor { _dummy: undefined & RecordType; name: string; keyPath?: IDBKeyPath | IDBKeyPath[]; autoIncrement?: boolean; } export interface StoreOptions { keyPath?: IDBKeyPath | IDBKeyPath[]; autoIncrement?: boolean; } export function describeContents( name: string, options: StoreOptions, ): StoreDescriptor { return { name, keyPath: options.keyPath, _dummy: undefined as any }; } export function describeIndex( name: string, keyPath: IDBKeyPath | IDBKeyPath[], options: IndexOptions = {}, ): IndexDescriptor { return { keyPath, name, multiEntry: options.multiEntry, }; } interface IndexReadOnlyAccessor { iter(query?: IDBValidKey): ResultStream; get(query: IDBValidKey): Promise; getAll(query: IDBValidKey, count?: number): Promise; } type GetIndexReadOnlyAccess = { [P in keyof IndexMap]: IndexReadOnlyAccessor; }; interface IndexReadWriteAccessor { iter(query: IDBValidKey): ResultStream; get(query: IDBValidKey): Promise; getAll(query: IDBValidKey, count?: number): Promise; } type GetIndexReadWriteAccess = { [P in keyof IndexMap]: IndexReadWriteAccessor; }; export interface StoreReadOnlyAccessor { get(key: IDBValidKey): Promise; iter(query?: IDBValidKey): ResultStream; indexes: GetIndexReadOnlyAccess; } export interface StoreReadWriteAccessor { get(key: IDBValidKey): Promise; iter(query?: IDBValidKey): ResultStream; put(r: RecordType): Promise; add(r: RecordType): Promise; delete(key: IDBValidKey): Promise; indexes: GetIndexReadWriteAccess; } export interface StoreWithIndexes< SD extends StoreDescriptor, IndexMap, > { store: SD; indexMap: IndexMap; /** * Type marker symbol, to check that the descriptor * has been created through the right function. */ mark: Symbol; } export type GetRecordType = T extends StoreDescriptor ? X : unknown; const storeWithIndexesSymbol = Symbol("StoreWithIndexesMark"); export function describeStore, IndexMap>( s: SD, m: IndexMap, ): StoreWithIndexes { return { store: s, indexMap: m, mark: storeWithIndexesSymbol, }; } export type GetReadOnlyAccess = { [P in keyof BoundStores]: BoundStores[P] extends StoreWithIndexes< infer SD, infer IM > ? StoreReadOnlyAccessor, IM> : unknown; }; export type GetReadWriteAccess = { [P in keyof BoundStores]: BoundStores[P] extends StoreWithIndexes< infer SD, infer IM > ? StoreReadWriteAccessor, IM> : unknown; }; type ReadOnlyTransactionFunction = ( t: GetReadOnlyAccess, ) => Promise; type ReadWriteTransactionFunction = ( t: GetReadWriteAccess, ) => Promise; export interface TransactionContext { runReadWrite(f: ReadWriteTransactionFunction): Promise; runReadOnly(f: ReadOnlyTransactionFunction): Promise; } type CheckDescriptor = T extends StoreWithIndexes ? StoreWithIndexes : unknown; type GetPickerType = F extends (x: SM) => infer Out ? { [P in keyof Out]: CheckDescriptor } : unknown; function runTx( tx: IDBTransaction, arg: Arg, f: (t: Arg) => Promise, ): Promise { const stack = Error("Failed transaction was started here."); return new Promise((resolve, reject) => { let funResult: any = undefined; let gotFunResult = false; let transactionException: any = undefined; tx.oncomplete = () => { // This is a fatal error: The transaction completed *before* // the transaction function returned. Likely, the transaction // function waited on a promise that is *not* resolved in the // microtask queue, thus triggering the auto-commit behavior. // Unfortunately, the auto-commit behavior of IDB can't be switched // of. There are some proposals to add this functionality in the future. if (!gotFunResult) { const msg = "BUG: transaction closed before transaction function returned"; console.error(msg); reject(Error(msg)); } resolve(funResult); }; tx.onerror = () => { logger.error("error in transaction"); logger.error(`${stack}`); }; tx.onabort = () => { let msg: string; if (tx.error) { msg = `Transaction aborted (transaction error): ${tx.error}`; } else if (transactionException !== undefined) { msg = `Transaction aborted (exception thrown): ${transactionException}`; } else { msg = "Transaction aborted (no DB error)"; } logger.error(msg); reject(new TransactionAbortedError(msg)); }; const resP = Promise.resolve().then(() => f(arg)); resP .then((result) => { gotFunResult = true; funResult = result; }) .catch((e) => { if (e == TransactionAbort) { logger.trace("aborting transaction"); } else { transactionException = e; console.error("Transaction failed:", e); console.error(stack); tx.abort(); } }) .catch((e) => { console.error("fatal: aborting transaction failed", e); }); }); } function makeReadContext( tx: IDBTransaction, storePick: { [n: string]: StoreWithIndexes }, ): any { const ctx: { [s: string]: StoreReadOnlyAccessor } = {}; for (const storeAlias in storePick) { const indexes: { [s: string]: IndexReadOnlyAccessor } = {}; const swi = storePick[storeAlias]; const storeName = swi.store.name; for (const indexAlias in storePick[storeAlias].indexMap) { const indexDescriptor: IndexDescriptor = storePick[storeAlias].indexMap[indexAlias]; const indexName = indexDescriptor.name; indexes[indexAlias] = { get(key) { const req = tx.objectStore(storeName).index(indexName).get(key); return requestToPromise(req); }, iter(query) { const req = tx .objectStore(storeName) .index(indexName) .openCursor(query); return new ResultStream(req); }, getAll(query, count) { const req = tx .objectStore(storeName) .index(indexName) .getAll(query, count); return requestToPromise(req); }, }; } ctx[storeAlias] = { indexes, get(key) { const req = tx.objectStore(storeName).get(key); return requestToPromise(req); }, iter(query) { const req = tx.objectStore(storeName).openCursor(query); return new ResultStream(req); }, }; } return ctx; } function makeWriteContext( tx: IDBTransaction, storePick: { [n: string]: StoreWithIndexes }, ): any { const ctx: { [s: string]: StoreReadWriteAccessor } = {}; for (const storeAlias in storePick) { const indexes: { [s: string]: IndexReadWriteAccessor } = {}; const swi = storePick[storeAlias]; const storeName = swi.store.name; for (const indexAlias in storePick[storeAlias].indexMap) { const indexDescriptor: IndexDescriptor = storePick[storeAlias].indexMap[indexAlias]; const indexName = indexDescriptor.name; indexes[indexAlias] = { get(key) { const req = tx.objectStore(storeName).index(indexName).get(key); return requestToPromise(req); }, iter(query) { const req = tx .objectStore(storeName) .index(indexName) .openCursor(query); return new ResultStream(req); }, getAll(query, count) { const req = tx .objectStore(storeName) .index(indexName) .getAll(query, count); return requestToPromise(req); }, }; } ctx[storeAlias] = { indexes, get(key) { const req = tx.objectStore(storeName).get(key); return requestToPromise(req); }, iter(query) { const req = tx.objectStore(storeName).openCursor(query); return new ResultStream(req); }, add(r) { const req = tx.objectStore(storeName).add(r); return requestToPromise(req); }, put(r) { const req = tx.objectStore(storeName).put(r); return requestToPromise(req); }, delete(k) { const req = tx.objectStore(storeName).delete(k); return requestToPromise(req); }, }; } return ctx; } const storeList = [ { name: "foo" as const, value: 1 as const }, { name: "bar" as const, value: 2 as const }, ]; // => { foo: { value: 1}, bar: {value: 2} } type StoreList = typeof storeList; type StoreNames = StoreList[number] extends { name: infer I } ? I : never; type H = StoreList[number] & { name: "foo"}; type Cleanup = V extends { name: infer N, value: infer X} ? {name: N, value: X} : never; type G = { [X in StoreNames]: { X: StoreList[number] & { name: X }; }; }; /** * Type-safe access to a database with a particular store map. * * A store map is the metadata that describes the store. */ export class DbAccess { constructor(private db: IDBDatabase, private stores: StoreMap) {} idbHandle(): IDBDatabase { return this.db; } mktx2< StoreNames extends keyof StoreMap, Stores extends StoreMap[StoreNames], StoreList extends Stores[], >(namePicker: (x: StoreMap) => StoreList): StoreList { return namePicker(this.stores); } mktx< PickerType extends (x: StoreMap) => unknown, BoundStores extends GetPickerType, >(f: PickerType): TransactionContext { const storePick = f(this.stores) as any; if (typeof storePick !== "object" || storePick === null) { throw Error(); } const storeNames: string[] = []; for (const storeAlias of Object.keys(storePick)) { const swi = (storePick as any)[storeAlias] as StoreWithIndexes; if (swi.mark !== storeWithIndexesSymbol) { throw Error("invalid store descriptor returned from selector function"); } storeNames.push(swi.store.name); } const runReadOnly = ( txf: ReadOnlyTransactionFunction, ): Promise => { const tx = this.db.transaction(storeNames, "readonly"); const readContext = makeReadContext(tx, storePick); return runTx(tx, readContext, txf); }; const runReadWrite = ( txf: ReadWriteTransactionFunction, ): Promise => { const tx = this.db.transaction(storeNames, "readwrite"); const writeContext = makeWriteContext(tx, storePick); return runTx(tx, writeContext, txf); }; return { runReadOnly, runReadWrite, }; } }