2023-10-03 12:16:33 +00:00
|
|
|
import { applyQueryNoFilterKV, evalQueryExpression } from "$sb/lib/query.ts";
|
|
|
|
import { FunctionMap, KV, KvKey, KvQuery } from "$sb/types.ts";
|
|
|
|
import { builtinFunctions } from "$sb/lib/builtin_query_functions.ts";
|
|
|
|
import { KvPrimitives } from "./kv_primitives.ts";
|
2023-09-03 19:15:17 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* This is the data store class you'll actually want to use, wrapping the primitives
|
|
|
|
* in a more user-friendly way
|
|
|
|
*/
|
|
|
|
export class DataStore {
|
2023-10-03 12:16:33 +00:00
|
|
|
constructor(
|
2023-12-06 17:44:48 +00:00
|
|
|
readonly kv: KvPrimitives,
|
2023-10-03 12:16:33 +00:00
|
|
|
private prefix: KvKey = [],
|
|
|
|
private functionMap: FunctionMap = builtinFunctions,
|
|
|
|
) {
|
|
|
|
}
|
|
|
|
|
|
|
|
prefixed(prefix: KvKey): DataStore {
|
|
|
|
return new DataStore(
|
|
|
|
this.kv,
|
|
|
|
[...this.prefix, ...prefix],
|
|
|
|
this.functionMap,
|
|
|
|
);
|
2023-09-03 19:15:17 +00:00
|
|
|
}
|
|
|
|
|
2023-10-03 12:16:33 +00:00
|
|
|
async get<T = any>(key: KvKey): Promise<T | null> {
|
|
|
|
return (await this.batchGet([key]))[0];
|
2023-09-03 19:15:17 +00:00
|
|
|
}
|
|
|
|
|
2023-10-03 12:16:33 +00:00
|
|
|
batchGet<T = any>(keys: KvKey[]): Promise<(T | null)[]> {
|
|
|
|
return this.kv.batchGet(keys.map((key) => this.applyPrefix(key)));
|
2023-09-03 19:15:17 +00:00
|
|
|
}
|
|
|
|
|
2023-10-03 12:16:33 +00:00
|
|
|
set(key: KvKey, value: any): Promise<void> {
|
|
|
|
return this.batchSet([{ key, value }]);
|
2023-09-03 19:15:17 +00:00
|
|
|
}
|
|
|
|
|
2023-10-03 12:16:33 +00:00
|
|
|
batchSet<T = any>(entries: KV<T>[]): Promise<void> {
|
|
|
|
const allKeyStrings = new Set<string>();
|
|
|
|
const uniqueEntries: KV[] = [];
|
|
|
|
for (const { key, value } of entries) {
|
|
|
|
const keyString = JSON.stringify(key);
|
|
|
|
if (allKeyStrings.has(keyString)) {
|
|
|
|
console.warn(`Duplicate key ${keyString} in batchSet, skipping`);
|
|
|
|
} else {
|
|
|
|
allKeyStrings.add(keyString);
|
|
|
|
uniqueEntries.push({ key: this.applyPrefix(key), value });
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return this.kv.batchSet(uniqueEntries);
|
2023-09-03 19:15:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
delete(key: KvKey): Promise<void> {
|
2023-10-03 12:16:33 +00:00
|
|
|
return this.batchDelete([key]);
|
2023-09-03 19:15:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
batchDelete(keys: KvKey[]): Promise<void> {
|
2023-10-03 12:16:33 +00:00
|
|
|
return this.kv.batchDelete(keys.map((key) => this.applyPrefix(key)));
|
2023-09-03 19:15:17 +00:00
|
|
|
}
|
|
|
|
|
2023-10-03 12:16:33 +00:00
|
|
|
async query<T = any>(query: KvQuery): Promise<KV<T>[]> {
|
|
|
|
const results: KV<T>[] = [];
|
2023-09-03 19:15:17 +00:00
|
|
|
let itemCount = 0;
|
2023-10-03 12:16:33 +00:00
|
|
|
// Accumulate results
|
|
|
|
let limit = Infinity;
|
|
|
|
const prefixedQuery: KvQuery = {
|
|
|
|
...query,
|
|
|
|
prefix: query.prefix ? this.applyPrefix(query.prefix) : undefined,
|
|
|
|
};
|
|
|
|
if (query.limit) {
|
|
|
|
limit = evalQueryExpression(query.limit, {}, this.functionMap);
|
|
|
|
}
|
|
|
|
for await (
|
|
|
|
const entry of this.kv.query(prefixedQuery)
|
|
|
|
) {
|
2023-09-03 19:15:17 +00:00
|
|
|
// Filter
|
2023-10-03 12:16:33 +00:00
|
|
|
if (
|
|
|
|
query.filter &&
|
|
|
|
!evalQueryExpression(query.filter, entry.value, this.functionMap)
|
|
|
|
) {
|
2023-09-03 19:15:17 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
results.push(entry);
|
|
|
|
itemCount++;
|
|
|
|
// Stop when the limit has been reached
|
2023-10-03 14:54:03 +00:00
|
|
|
if (itemCount === limit && !query.orderBy) {
|
|
|
|
// Only break when not also ordering in which case we need all results
|
2023-09-03 19:15:17 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2023-10-03 12:16:33 +00:00
|
|
|
// Apply order by, limit, and select
|
|
|
|
return applyQueryNoFilterKV(prefixedQuery, results, this.functionMap).map((
|
|
|
|
{ key, value },
|
|
|
|
) => ({ key: this.stripPrefix(key), value }));
|
|
|
|
}
|
2023-09-03 19:15:17 +00:00
|
|
|
|
2023-10-03 12:16:33 +00:00
|
|
|
async queryDelete(query: KvQuery): Promise<void> {
|
|
|
|
const keys: KvKey[] = [];
|
|
|
|
for (
|
|
|
|
const { key } of await this.query({
|
|
|
|
...query,
|
|
|
|
prefix: query.prefix ? this.applyPrefix(query.prefix) : undefined,
|
|
|
|
})
|
|
|
|
) {
|
|
|
|
keys.push(key);
|
2023-09-03 19:15:17 +00:00
|
|
|
}
|
2023-10-03 12:16:33 +00:00
|
|
|
return this.batchDelete(keys);
|
|
|
|
}
|
|
|
|
|
|
|
|
private applyPrefix(key: KvKey): KvKey {
|
|
|
|
return [...this.prefix, ...(key ? key : [])];
|
|
|
|
}
|
|
|
|
|
|
|
|
private stripPrefix(key: KvKey): KvKey {
|
|
|
|
return key.slice(this.prefix.length);
|
2023-09-03 19:15:17 +00:00
|
|
|
}
|
|
|
|
}
|