This repository has been archived on 2022-08-19. You can view files and clone it, but cannot push or open issues or pull requests.
hydrogen-web/src/matrix/storage/idb/utils.ts

221 lines
8.1 KiB
TypeScript
Raw Normal View History

2020-08-05 22:08:55 +05:30
/*
Copyright 2020 Bruno Windels <bruno@windels.cloud>
2020-09-28 18:58:51 +05:30
Copyright 2020 The Matrix.org Foundation C.I.C.
2020-08-05 22:08:55 +05:30
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
2020-09-29 12:47:03 +05:30
import { IDBRequestError } from "./error.js";
import { StorageError } from "../common";
let needsSyncPromise = false;
2021-08-10 02:26:20 +05:30
export const DONE = { done: true }
export const NOT_DONE = { done: false }
/* should be called on legacy platforms to see
if transactions close before draining the microtask queue (IE11 on Windows 7).
If this is the case, promises need to be resolved
synchronously from the idb request handler to prevent the transaction from closing prematurely.
*/
2021-08-10 02:26:20 +05:30
export async function checkNeedsSyncPromise(): Promise<boolean> {
// important to have it turned off while doing the test,
// otherwise reqAsPromise would not fail
needsSyncPromise = false;
const NAME = "test-idb-needs-sync-promise";
const db = await openDatabase(NAME, db => {
db.createObjectStore("test", {keyPath: "key"});
}, 1);
const txn = db.transaction("test", "readonly");
try {
await reqAsPromise(txn.objectStore("test").get(1));
await reqAsPromise(txn.objectStore("test").get(2));
} catch (err) {
// err.name would be either TransactionInactiveError or InvalidStateError,
// but let's not exclude any other failure modes
needsSyncPromise = true;
}
// we could delete the store here,
// but let's not create it on every page load on legacy platforms,
// and just keep it around
return needsSyncPromise;
}
// storage keys are defined to be unsigned 32bit numbers in KeyLimits, which is assumed by idb
2021-08-10 02:26:20 +05:30
export function encodeUint32(n: number): string {
2019-07-01 13:30:29 +05:30
const hex = n.toString(16);
return "0".repeat(8 - hex.length) + hex;
}
2021-02-12 17:34:05 +05:30
// used for logs where timestamp is part of key, which is larger than 32 bit
2021-08-10 02:26:20 +05:30
export function encodeUint64(n: number): string {
2021-02-12 17:34:05 +05:30
const hex = n.toString(16);
return "0".repeat(16 - hex.length) + hex;
}
2021-08-10 02:26:20 +05:30
export function decodeUint32(str: string): number {
2019-07-01 13:30:29 +05:30
return parseInt(str, 16);
}
2021-08-10 02:26:20 +05:30
type CreateObjectStore = (db : IDBDatabase, txn: IDBTransaction | null, oldVersion: number, version: number) => any
export function openDatabase(name: string, createObjectStore: CreateObjectStore, version: number, idbFactory: IDBFactory = window.indexedDB): Promise<IDBDatabase> {
const req = idbFactory.open(name, version);
2021-08-10 02:26:20 +05:30
req.onupgradeneeded = (ev : IDBVersionChangeEvent) => {
const req = ev.target as IDBRequest<IDBDatabase>;
const db = req.result;
const txn = req.transaction;
2019-02-07 03:36:56 +05:30
const oldVersion = ev.oldVersion;
2020-06-27 02:56:24 +05:30
createObjectStore(db, txn, oldVersion, version);
2019-02-07 03:36:56 +05:30
};
return reqAsPromise(req);
}
2021-08-10 02:26:20 +05:30
export function reqAsPromise<T>(req: IDBRequest<T>): Promise<T> {
return new Promise((resolve, reject) => {
req.addEventListener("success", event => {
2021-08-10 02:26:20 +05:30
resolve((event.target as IDBRequest<T>).result);
// @ts-ignore
needsSyncPromise && Promise._flush && Promise._flush();
});
2021-05-05 18:06:43 +05:30
req.addEventListener("error", event => {
2021-08-10 02:26:20 +05:30
const error = new IDBRequestError(event.target as IDBRequest<T>);
2021-05-05 18:06:43 +05:30
reject(error);
2021-08-10 02:26:20 +05:30
// @ts-ignore
needsSyncPromise && Promise._flush && Promise._flush();
});
});
}
2021-08-10 02:26:20 +05:30
export function txnAsPromise(txn): Promise<void> {
2021-05-05 19:32:39 +05:30
let error;
return new Promise((resolve, reject) => {
txn.addEventListener("complete", () => {
resolve();
2021-08-10 02:26:20 +05:30
// @ts-ignore
needsSyncPromise && Promise._flush && Promise._flush();
});
2021-05-05 19:32:39 +05:30
txn.addEventListener("error", event => {
const request = event.target;
// catch first error here, but don't reject yet,
// as we don't have access to the failed request in the abort event handler
if (!error && request) {
error = new IDBRequestError(request);
}
});
txn.addEventListener("abort", event => {
if (!error) {
const txn = event.target;
const dbName = txn.db.name;
const storeNames = Array.from(txn.objectStoreNames).join(", ")
error = new StorageError(`Transaction on ${dbName} with stores ${storeNames} was aborted.`);
}
reject(error);
2021-08-10 02:26:20 +05:30
// @ts-ignore
needsSyncPromise && Promise._flush && Promise._flush();
});
});
}
2021-08-10 02:26:20 +05:30
type CursorIterator<T, I extends IDBCursor> = I extends IDBCursorWithValue ?
(value: T, key: IDBValidKey, cursor: IDBCursorWithValue) => { done: boolean, jumpTo?: IDBValidKey } :
(value: undefined, key: IDBValidKey, cursor: IDBCursor) => { done: boolean, jumpTo?: IDBValidKey }
export function iterateCursor<T, I extends IDBCursor = IDBCursorWithValue>(cursorRequest: IDBRequest<I | null>, processValue: CursorIterator<T, I>): Promise<boolean> {
2019-02-07 03:36:56 +05:30
// TODO: does cursor already have a value here??
2021-08-10 02:26:20 +05:30
return new Promise<boolean>((resolve, reject) => {
cursorRequest.onerror = () => {
reject(new IDBRequestError(cursorRequest));
2021-08-10 02:26:20 +05:30
// @ts-ignore
needsSyncPromise && Promise._flush && Promise._flush();
};
// collect results
cursorRequest.onsuccess = (event) => {
2021-08-10 02:26:20 +05:30
const cursor = (event.target as IDBRequest<I>).result;
if (!cursor) {
resolve(false);
2021-08-10 02:26:20 +05:30
// @ts-ignore
needsSyncPromise && Promise._flush && Promise._flush();
return; // end of results
}
2021-08-10 02:26:20 +05:30
const result = processValue(cursor["value"], cursor.key, cursor);
// TODO: don't use object for result and assume it's jumpTo when not === true/false or undefined
2020-08-19 21:55:38 +05:30
const done = result?.done;
const jumpTo = result?.jumpTo;
2020-06-27 02:56:24 +05:30
if (done) {
2019-02-07 03:36:56 +05:30
resolve(true);
2021-08-10 02:26:20 +05:30
// @ts-ignore
needsSyncPromise && Promise._flush && Promise._flush();
} else if(jumpTo) {
cursor.continue(jumpTo);
} else {
cursor.continue();
}
};
}).catch(err => {
throw new StorageError("iterateCursor failed", err);
});
}
2021-08-10 02:26:20 +05:30
type Pred<T> = (value: T) => boolean
export async function fetchResults<T>(cursor: IDBRequest, isDone: Pred<T[]>): Promise<T[]> {
const results: T[] = [];
await iterateCursor<T>(cursor, (value) => {
2019-02-07 03:36:56 +05:30
results.push(value);
return {done: isDone(results)};
});
return results;
}
2021-08-10 02:26:20 +05:30
type ToCursor = (store: IDBObjectStore) => IDBRequest
export async function select<T>(db: IDBDatabase, storeName: string, toCursor: ToCursor, isDone: Pred<T[]>): Promise<T[]> {
2019-02-07 03:36:56 +05:30
if (!isDone) {
isDone = () => false;
}
if (!toCursor) {
toCursor = store => store.openCursor();
}
const tx = db.transaction([storeName], "readonly");
const store = tx.objectStore(storeName);
const cursor = toCursor(store);
return await fetchResults(cursor, isDone);
}
2021-08-10 02:26:20 +05:30
export async function findStoreValue<T>(db: IDBDatabase, storeName: string, toCursor: ToCursor, matchesValue: Pred<T>): Promise<T> {
2019-02-07 03:36:56 +05:30
if (!matchesValue) {
matchesValue = () => true;
}
if (!toCursor) {
toCursor = store => store.openCursor();
}
2019-02-07 03:36:56 +05:30
const tx = db.transaction([storeName], "readwrite");
const store = tx.objectStore(storeName);
const cursor = await reqAsPromise(toCursor(store));
let match;
2021-08-10 02:26:20 +05:30
const matched = await iterateCursor<T>(cursor, (value) => {
2019-02-07 03:36:56 +05:30
if (matchesValue(value)) {
match = value;
2021-08-10 07:43:32 +05:30
return DONE;
2019-02-07 03:36:56 +05:30
}
2021-08-10 07:43:32 +05:30
return NOT_DONE;
2019-02-07 03:36:56 +05:30
});
if (!matched) {
throw new StorageError("Value not found");
2019-02-07 03:36:56 +05:30
}
return match;
}