2019-02-07 00:19:14 +01:00
|
|
|
export function openDatabase(name, createObjectStore, version) {
|
2019-02-06 23:06:56 +01:00
|
|
|
const req = window.indexedDB.open(name, version);
|
|
|
|
req.onupgradeneeded = (ev) => {
|
|
|
|
const db = ev.target.result;
|
|
|
|
const oldVersion = ev.oldVersion;
|
|
|
|
createObjectStore(db, oldVersion, version);
|
|
|
|
};
|
|
|
|
return reqAsPromise(req);
|
2019-01-09 11:06:09 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
export function reqAsPromise(req) {
|
|
|
|
return new Promise((resolve, reject) => {
|
2019-02-06 23:06:33 +01:00
|
|
|
req.addEventListener("success", event => resolve(event.target.result));
|
|
|
|
req.addEventListener("error", event => reject(event.target.error));
|
2019-01-09 11:06:09 +01:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
export function txnAsPromise(txn) {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
txn.addEventListener("complete", resolve);
|
|
|
|
txn.addEventListener("abort", reject);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
export function iterateCursor(cursor, processValue) {
|
2019-02-06 23:06:56 +01:00
|
|
|
// TODO: does cursor already have a value here??
|
2019-01-09 11:06:09 +01:00
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
cursor.onerror = (event) => {
|
|
|
|
reject(new Error("Query failed: " + event.target.errorCode));
|
|
|
|
};
|
|
|
|
// collect results
|
|
|
|
cursor.onsuccess = (event) => {
|
|
|
|
const cursor = event.target.result;
|
|
|
|
if (!cursor) {
|
|
|
|
resolve(false);
|
|
|
|
return; // end of results
|
|
|
|
}
|
2019-05-11 13:10:31 +02:00
|
|
|
const {done, jumpTo} = processValue(cursor.value, cursor.key);
|
|
|
|
if (done) {
|
2019-02-06 23:06:56 +01:00
|
|
|
resolve(true);
|
2019-01-09 11:06:09 +01:00
|
|
|
} else {
|
2019-05-11 13:10:31 +02:00
|
|
|
cursor.continue(jumpTo);
|
2019-01-09 11:06:09 +01:00
|
|
|
}
|
|
|
|
};
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function fetchResults(cursor, isDone) {
|
|
|
|
const results = [];
|
|
|
|
await iterateCursor(cursor, (value) => {
|
2019-02-06 23:06:56 +01:00
|
|
|
results.push(value);
|
2019-05-11 13:10:31 +02:00
|
|
|
return {done: isDone(results)};
|
2019-01-09 11:06:09 +01:00
|
|
|
});
|
|
|
|
return results;
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function select(db, storeName, toCursor, isDone) {
|
2019-02-06 23:06:56 +01:00
|
|
|
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);
|
2019-01-09 11:06:09 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
export async function updateSingletonStore(db, storeName, value) {
|
2019-02-06 23:06:56 +01:00
|
|
|
const tx = db.transaction([storeName], "readwrite");
|
|
|
|
const store = tx.objectStore(storeName);
|
|
|
|
const cursor = await reqAsPromise(store.openCursor());
|
|
|
|
if (cursor) {
|
|
|
|
return reqAsPromise(cursor.update(storeName));
|
|
|
|
} else {
|
|
|
|
return reqAsPromise(store.add(value));
|
|
|
|
}
|
2019-01-09 11:06:09 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
export async function findStoreValue(db, storeName, toCursor, matchesValue) {
|
2019-02-06 23:06:56 +01:00
|
|
|
if (!matchesValue) {
|
|
|
|
matchesValue = () => true;
|
|
|
|
}
|
|
|
|
if (!toCursor) {
|
|
|
|
toCursor = store => store.openCursor();
|
|
|
|
}
|
2019-01-09 11:06:09 +01:00
|
|
|
|
2019-02-06 23:06:56 +01:00
|
|
|
const tx = db.transaction([storeName], "readwrite");
|
|
|
|
const store = tx.objectStore(storeName);
|
|
|
|
const cursor = await reqAsPromise(toCursor(store));
|
|
|
|
let match;
|
|
|
|
const matched = await iterateCursor(cursor, (value) => {
|
|
|
|
if (matchesValue(value)) {
|
|
|
|
match = value;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
if (!matched) {
|
|
|
|
throw new Error("Value not found");
|
|
|
|
}
|
|
|
|
return match;
|
2019-05-11 13:10:31 +02:00
|
|
|
}
|