2020-08-05 18:38:55 +02:00
|
|
|
/*
|
|
|
|
Copyright 2020 Bruno Windels <bruno@windels.cloud>
|
|
|
|
|
|
|
|
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.
|
|
|
|
*/
|
|
|
|
|
2019-03-29 23:01:01 +01:00
|
|
|
import {iterateCursor, reqAsPromise} from "./utils.js";
|
2019-01-09 11:06:09 +01:00
|
|
|
|
2020-04-20 21:26:39 +02:00
|
|
|
export class QueryTarget {
|
2019-06-26 22:31:36 +02:00
|
|
|
constructor(target) {
|
|
|
|
this._target = target;
|
|
|
|
}
|
2019-02-07 00:20:27 +00:00
|
|
|
|
2019-06-26 22:02:00 +02:00
|
|
|
_openCursor(range, direction) {
|
|
|
|
if (range && direction) {
|
|
|
|
return this._target.openCursor(range, direction);
|
|
|
|
} else if (range) {
|
|
|
|
return this._target.openCursor(range);
|
|
|
|
} else if (direction) {
|
|
|
|
return this._target.openCursor(null, direction);
|
|
|
|
} else {
|
|
|
|
return this._target.openCursor();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-15 12:24:27 +02:00
|
|
|
supports(methodName) {
|
|
|
|
return this._target.supports(methodName);
|
|
|
|
}
|
|
|
|
|
2019-03-29 23:01:01 +01:00
|
|
|
get(key) {
|
|
|
|
return reqAsPromise(this._target.get(key));
|
|
|
|
}
|
|
|
|
|
2019-07-27 10:40:56 +02:00
|
|
|
getKey(key) {
|
|
|
|
return reqAsPromise(this._target.getKey(key));
|
|
|
|
}
|
|
|
|
|
2019-06-26 22:31:36 +02:00
|
|
|
reduce(range, reducer, initialValue) {
|
|
|
|
return this._reduce(range, reducer, initialValue, "next");
|
|
|
|
}
|
|
|
|
|
|
|
|
reduceReverse(range, reducer, initialValue) {
|
|
|
|
return this._reduce(range, reducer, initialValue, "prev");
|
|
|
|
}
|
|
|
|
|
|
|
|
selectLimit(range, amount) {
|
|
|
|
return this._selectLimit(range, amount, "next");
|
|
|
|
}
|
|
|
|
|
|
|
|
selectLimitReverse(range, amount) {
|
|
|
|
return this._selectLimit(range, amount, "prev");
|
|
|
|
}
|
|
|
|
|
|
|
|
selectWhile(range, predicate) {
|
2020-08-19 18:25:49 +02:00
|
|
|
return this._selectWhile(range, predicate, "next");
|
2019-06-26 22:31:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
selectWhileReverse(range, predicate) {
|
2020-08-19 18:25:49 +02:00
|
|
|
return this._selectWhile(range, predicate, "prev");
|
2019-06-26 22:31:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
async selectAll(range, direction) {
|
|
|
|
const cursor = this._openCursor(range, direction);
|
|
|
|
const results = [];
|
|
|
|
await iterateCursor(cursor, (value) => {
|
|
|
|
results.push(value);
|
|
|
|
return {done: false};
|
|
|
|
});
|
|
|
|
return results;
|
|
|
|
}
|
|
|
|
|
|
|
|
selectFirst(range) {
|
|
|
|
return this._find(range, () => true, "next");
|
|
|
|
}
|
|
|
|
|
|
|
|
selectLast(range) {
|
|
|
|
return this._find(range, () => true, "prev");
|
|
|
|
}
|
|
|
|
|
|
|
|
find(range, predicate) {
|
|
|
|
return this._find(range, predicate, "next");
|
|
|
|
}
|
|
|
|
|
|
|
|
findReverse(range, predicate) {
|
|
|
|
return this._find(range, predicate, "prev");
|
|
|
|
}
|
2019-01-09 11:06:09 +01:00
|
|
|
|
2019-07-01 10:00:29 +02:00
|
|
|
async findMaxKey(range) {
|
|
|
|
const cursor = this._target.openKeyCursor(range, "prev");
|
|
|
|
let maxKey;
|
|
|
|
await iterateCursor(cursor, (_, key) => {
|
|
|
|
maxKey = key;
|
|
|
|
return {done: true};
|
|
|
|
});
|
|
|
|
return maxKey;
|
|
|
|
}
|
|
|
|
|
2020-08-31 14:11:08 +02:00
|
|
|
async iterateKeys(range, callback) {
|
|
|
|
const cursor = this._target.openKeyCursor(range, "next");
|
|
|
|
await iterateCursor(cursor, (_, key) => {
|
|
|
|
return {done: callback(key)};
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2019-05-11 13:10:31 +02:00
|
|
|
/**
|
|
|
|
* Checks if a given set of keys exist.
|
|
|
|
* Calls `callback(key, found)` for each key in `keys`, in key sorting order (or reversed if backwards=true).
|
|
|
|
* If the callback returns true, the search is halted and callback won't be called again.
|
|
|
|
* `callback` is called with the same instances of the key as given in `keys`, so direct comparison can be used.
|
|
|
|
*/
|
|
|
|
async findExistingKeys(keys, backwards, callback) {
|
|
|
|
const direction = backwards ? "prev" : "next";
|
|
|
|
const compareKeys = (a, b) => backwards ? -indexedDB.cmp(a, b) : indexedDB.cmp(a, b);
|
|
|
|
const sortedKeys = keys.slice().sort(compareKeys);
|
|
|
|
const firstKey = backwards ? sortedKeys[sortedKeys.length - 1] : sortedKeys[0];
|
|
|
|
const lastKey = backwards ? sortedKeys[0] : sortedKeys[sortedKeys.length - 1];
|
|
|
|
const cursor = this._target.openKeyCursor(IDBKeyRange.bound(firstKey, lastKey), direction);
|
|
|
|
let i = 0;
|
|
|
|
let consumerDone = false;
|
|
|
|
await iterateCursor(cursor, (value, key) => {
|
|
|
|
// while key is larger than next key, advance and report false
|
|
|
|
while(i < sortedKeys.length && compareKeys(sortedKeys[i], key) < 0 && !consumerDone) {
|
|
|
|
consumerDone = callback(sortedKeys[i], false);
|
|
|
|
++i;
|
|
|
|
}
|
|
|
|
if (i < sortedKeys.length && compareKeys(sortedKeys[i], key) === 0 && !consumerDone) {
|
|
|
|
consumerDone = callback(sortedKeys[i], true);
|
|
|
|
++i;
|
|
|
|
}
|
|
|
|
const done = consumerDone || i >= sortedKeys.length;
|
|
|
|
const jumpTo = !done && sortedKeys[i];
|
|
|
|
return {done, jumpTo};
|
|
|
|
});
|
|
|
|
// report null for keys we didn't to at the end
|
|
|
|
while (!consumerDone && i < sortedKeys.length) {
|
|
|
|
consumerDone = callback(sortedKeys[i], false);
|
|
|
|
++i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-26 22:31:36 +02:00
|
|
|
_reduce(range, reducer, initialValue, direction) {
|
|
|
|
let reducedValue = initialValue;
|
|
|
|
const cursor = this._openCursor(range, direction);
|
|
|
|
return iterateCursor(cursor, (value) => {
|
|
|
|
reducedValue = reducer(reducedValue, value);
|
|
|
|
return {done: false};
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
_selectLimit(range, amount, direction) {
|
2020-08-19 18:25:49 +02:00
|
|
|
return this._selectUntil(range, (results) => {
|
2019-06-26 22:31:36 +02:00
|
|
|
return results.length === amount;
|
2020-08-19 18:25:49 +02:00
|
|
|
}, direction);
|
2019-06-26 22:31:36 +02:00
|
|
|
}
|
|
|
|
|
2020-08-19 18:25:49 +02:00
|
|
|
async _selectUntil(range, predicate, direction) {
|
2019-06-26 22:31:36 +02:00
|
|
|
const cursor = this._openCursor(range, direction);
|
|
|
|
const results = [];
|
|
|
|
await iterateCursor(cursor, (value) => {
|
2020-08-19 18:25:49 +02:00
|
|
|
results.push(value);
|
|
|
|
return {done: predicate(results, value)};
|
|
|
|
});
|
|
|
|
return results;
|
|
|
|
}
|
|
|
|
|
|
|
|
// allows you to fetch one too much that won't get added when the predicate fails
|
|
|
|
async _selectWhile(range, predicate, direction) {
|
|
|
|
const cursor = this._openCursor(range, direction);
|
|
|
|
const results = [];
|
|
|
|
await iterateCursor(cursor, (value) => {
|
|
|
|
const passesPredicate = predicate(value);
|
|
|
|
if (passesPredicate) {
|
2020-08-19 16:13:30 +02:00
|
|
|
results.push(value);
|
|
|
|
}
|
2020-08-19 18:25:49 +02:00
|
|
|
return {done: !passesPredicate};
|
2019-06-26 22:31:36 +02:00
|
|
|
});
|
|
|
|
return results;
|
|
|
|
}
|
|
|
|
|
2020-09-08 14:38:27 +02:00
|
|
|
async iterateWhile(range, predicate) {
|
|
|
|
const cursor = this._openCursor(range, "next");
|
|
|
|
await iterateCursor(cursor, (value) => {
|
|
|
|
const passesPredicate = predicate(value);
|
|
|
|
return {done: !passesPredicate};
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2019-06-26 22:31:36 +02:00
|
|
|
async _find(range, predicate, direction) {
|
|
|
|
const cursor = this._openCursor(range, direction);
|
|
|
|
let result;
|
|
|
|
const found = await iterateCursor(cursor, (value) => {
|
|
|
|
const found = predicate(value);
|
|
|
|
if (found) {
|
|
|
|
result = value;
|
|
|
|
}
|
|
|
|
return {done: found};
|
|
|
|
});
|
|
|
|
if (found) {
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
2019-03-21 21:36:02 +01:00
|
|
|
}
|