import { applyQueryNoFilterKV } from "$sb/lib/query.ts"; import { FunctionMap, KV, KvKey, KvQuery } from "../../plug-api/types.ts"; import { builtinFunctions } from "$lib/builtin_query_functions.ts"; import { KvPrimitives } from "./kv_primitives.ts"; import { evalQueryExpression } from "$sb/lib/query_expression.ts"; /** * This is the data store class you'll actually want to use, wrapping the primitives * in a more user-friendly way */ export class DataStore { constructor( readonly kv: KvPrimitives, public functionMap: FunctionMap = builtinFunctions, ) { } async get<T = any>(key: KvKey): Promise<T | null> { return (await this.batchGet([key]))[0]; } batchGet<T = any>(keys: KvKey[]): Promise<(T | null)[]> { if (keys.length === 0) { return Promise.resolve([]); } return this.kv.batchGet(keys); } set(key: KvKey, value: any): Promise<void> { return this.batchSet([{ key, value }]); } batchSet<T = any>(entries: KV<T>[]): Promise<void> { if (entries.length === 0) { return Promise.resolve(); } 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, value }); } } return this.kv.batchSet(uniqueEntries); } delete(key: KvKey): Promise<void> { return this.batchDelete([key]); } batchDelete(keys: KvKey[]): Promise<void> { if (keys.length === 0) { return Promise.resolve(); } return this.kv.batchDelete(keys); } async query<T = any>( query: KvQuery, variables: Record<string, any> = {}, ): Promise<KV<T>[]> { const results: KV<T>[] = []; let itemCount = 0; // Accumulate results let limit = Infinity; if (query.limit) { limit = await evalQueryExpression( query.limit, {}, variables, this.functionMap, ); } for await ( const entry of this.kv.query(query) ) { // Filter if ( query.filter && !await evalQueryExpression( query.filter, entry.value, variables, this.functionMap, ) ) { continue; } results.push(entry); itemCount++; // Stop when the limit has been reached if (itemCount === limit && !query.orderBy) { // Only break when not also ordering in which case we need all results break; } } // Apply order by, limit, and select return applyQueryNoFilterKV( query, results, variables, this.functionMap, ); } async queryDelete( query: KvQuery, variables: Record<string, any> = {}, ): Promise<void> { const keys: KvKey[] = []; for ( const { key } of await this.query(query, variables) ) { keys.push(key); } return this.batchDelete(keys); } }