UNPKG

@tanstack/db

Version:

A reactive client store for building super fast apps on sync

302 lines (301 loc) 9.65 kB
"use strict"; Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" }); const dbIvm = require("@tanstack/db-ivm"); const btree = require("../utils/btree.cjs"); const comparison = require("../utils/comparison.cjs"); const baseIndex = require("./base-index.cjs"); class BTreeIndex extends baseIndex.BaseIndex { constructor(id, expression, name, options) { super(id, expression, name, options); this.supportedOperations = /* @__PURE__ */ new Set([ `eq`, `gt`, `gte`, `lt`, `lte`, `in` ]); this.valueMap = /* @__PURE__ */ new Map(); this.indexedKeys = /* @__PURE__ */ new Set(); this.compareFn = comparison.defaultComparator; const baseCompareFn = options?.compareFn ?? comparison.defaultComparator; this.compareFn = (a, b) => baseCompareFn(comparison.denormalizeUndefined(a), comparison.denormalizeUndefined(b)); if (options?.compareOptions) { this.compareOptions = options.compareOptions; } this.orderedEntries = new btree.BTree(this.compareFn); } initialize(_options) { } /** * Adds a value to the index */ add(key, item) { let indexedValue; try { indexedValue = this.evaluateIndexExpression(item); } catch (error) { throw new Error( `Failed to evaluate index expression for key ${key}: ${error}` ); } const normalizedValue = comparison.normalizeForBTree(indexedValue); if (this.valueMap.has(normalizedValue)) { this.valueMap.get(normalizedValue).add(key); } else { const keySet = /* @__PURE__ */ new Set([key]); this.valueMap.set(normalizedValue, keySet); this.orderedEntries.set(normalizedValue, void 0); } this.indexedKeys.add(key); this.updateTimestamp(); } /** * Removes a value from the index */ remove(key, item) { let indexedValue; try { indexedValue = this.evaluateIndexExpression(item); } catch (error) { console.warn( `Failed to evaluate index expression for key ${key} during removal:`, error ); return; } const normalizedValue = comparison.normalizeForBTree(indexedValue); if (this.valueMap.has(normalizedValue)) { const keySet = this.valueMap.get(normalizedValue); keySet.delete(key); if (keySet.size === 0) { this.valueMap.delete(normalizedValue); this.orderedEntries.delete(normalizedValue); } } this.indexedKeys.delete(key); this.updateTimestamp(); } /** * Updates a value in the index */ update(key, oldItem, newItem) { this.remove(key, oldItem); this.add(key, newItem); } /** * Builds the index from a collection of entries */ build(entries) { this.clear(); for (const [key, item] of entries) { this.add(key, item); } } /** * Clears all data from the index */ clear() { this.orderedEntries.clear(); this.valueMap.clear(); this.indexedKeys.clear(); this.updateTimestamp(); } /** * Performs a lookup operation */ lookup(operation, value) { const startTime = performance.now(); let result; switch (operation) { case `eq`: result = this.equalityLookup(value); break; case `gt`: result = this.rangeQuery({ from: value, fromInclusive: false }); break; case `gte`: result = this.rangeQuery({ from: value, fromInclusive: true }); break; case `lt`: result = this.rangeQuery({ to: value, toInclusive: false }); break; case `lte`: result = this.rangeQuery({ to: value, toInclusive: true }); break; case `in`: result = this.inArrayLookup(value); break; default: throw new Error(`Operation ${operation} not supported by BTreeIndex`); } this.trackLookup(startTime); return result; } /** * Gets the number of indexed keys */ get keyCount() { return this.indexedKeys.size; } // Public methods for backward compatibility (used by tests) /** * Performs an equality lookup */ equalityLookup(value) { const normalizedValue = comparison.normalizeForBTree(value); return new Set(this.valueMap.get(normalizedValue) ?? []); } /** * Performs a range query with options * This is more efficient for compound queries like "WHERE a > 5 AND a < 10" */ rangeQuery(options = {}) { const { from, to, fromInclusive = true, toInclusive = true } = options; const result = /* @__PURE__ */ new Set(); const hasFrom = `from` in options; const hasTo = `to` in options; const fromKey = hasFrom ? comparison.normalizeForBTree(from) : this.orderedEntries.minKey(); const toKey = hasTo ? comparison.normalizeForBTree(to) : this.orderedEntries.maxKey(); this.orderedEntries.forRange( fromKey, toKey, toInclusive, (indexedValue, _) => { if (!fromInclusive && this.compareFn(indexedValue, from) === 0) { return; } const keys = this.valueMap.get(indexedValue); if (keys) { keys.forEach((key) => result.add(key)); } } ); return result; } /** * Performs a reversed range query */ rangeQueryReversed(options = {}) { const { from, to, fromInclusive = true, toInclusive = true } = options; const hasFrom = `from` in options; const hasTo = `to` in options; return this.rangeQuery({ from: hasTo ? to : this.orderedEntries.maxKey(), to: hasFrom ? from : this.orderedEntries.minKey(), fromInclusive: toInclusive, toInclusive: fromInclusive }); } /** * Internal method for taking items from the index. * @param n - The number of items to return * @param nextPair - Function to get the next pair from the BTree * @param from - Already normalized! undefined means "start from beginning/end", sentinel means "start from the key undefined" * @param filterFn - Optional filter function * @param reversed - Whether to reverse the order of keys within each value */ takeInternal(n, nextPair, from, filterFn, reversed = false) { const keysInResult = /* @__PURE__ */ new Set(); const result = []; let pair; let key = from; while ((pair = nextPair(key)) !== void 0 && result.length < n) { key = pair[0]; const keys = this.valueMap.get(key); if (keys && keys.size > 0) { const sorted = Array.from(keys).sort(dbIvm.compareKeys); if (reversed) sorted.reverse(); for (const ks of sorted) { if (result.length >= n) break; if (!keysInResult.has(ks) && (filterFn?.(ks) ?? true)) { result.push(ks); keysInResult.add(ks); } } } } return result; } /** * Returns the next n items after the provided item. * @param n - The number of items to return * @param from - The item to start from (exclusive). * @returns The next n items after the provided key. */ take(n, from, filterFn) { const nextPair = (k) => this.orderedEntries.nextHigherPair(k); const normalizedFrom = comparison.normalizeForBTree(from); return this.takeInternal(n, nextPair, normalizedFrom, filterFn); } /** * Returns the first n items from the beginning. * @param n - The number of items to return * @param filterFn - Optional filter function * @returns The first n items */ takeFromStart(n, filterFn) { const nextPair = (k) => this.orderedEntries.nextHigherPair(k); return this.takeInternal(n, nextPair, void 0, filterFn); } /** * Returns the next n items **before** the provided item (in descending order). * @param n - The number of items to return * @param from - The item to start from (exclusive). Required. * @returns The next n items **before** the provided key. */ takeReversed(n, from, filterFn) { const nextPair = (k) => this.orderedEntries.nextLowerPair(k); const normalizedFrom = comparison.normalizeForBTree(from); return this.takeInternal(n, nextPair, normalizedFrom, filterFn, true); } /** * Returns the last n items from the end. * @param n - The number of items to return * @param filterFn - Optional filter function * @returns The last n items */ takeReversedFromEnd(n, filterFn) { const nextPair = (k) => this.orderedEntries.nextLowerPair(k); return this.takeInternal(n, nextPair, void 0, filterFn, true); } /** * Performs an IN array lookup */ inArrayLookup(values) { const result = /* @__PURE__ */ new Set(); for (const value of values) { const normalizedValue = comparison.normalizeForBTree(value); const keys = this.valueMap.get(normalizedValue); if (keys) { keys.forEach((key) => result.add(key)); } } return result; } // Getter methods for testing compatibility get indexedKeysSet() { return this.indexedKeys; } get orderedEntriesArray() { return this.orderedEntries.keysArray().map((key) => [ comparison.denormalizeUndefined(key), this.valueMap.get(key) ?? /* @__PURE__ */ new Set() ]); } get orderedEntriesArrayReversed() { return this.takeReversedFromEnd(this.orderedEntries.size).map((key) => [ comparison.denormalizeUndefined(key), this.valueMap.get(key) ?? /* @__PURE__ */ new Set() ]); } get valueMapData() { const result = /* @__PURE__ */ new Map(); for (const [key, value] of this.valueMap) { result.set(comparison.denormalizeUndefined(key), value); } return result; } } exports.BTreeIndex = BTreeIndex; //# sourceMappingURL=btree-index.cjs.map