Skip to content

Commit 6b70241

Browse files
authoredSep 9, 2024··
Merge pull request #62 from bhargeymehta/bhargeymehta/concat-to-push
Improve InMemoryProvider getRange function from O(N^2) to O(N)
2 parents 1404aaf + 726aa2a commit 6b70241

File tree

3 files changed

+55
-20
lines changed

3 files changed

+55
-20
lines changed
 

‎index.ts

+5-1
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,11 @@ export {
33
breakAndNormalizeSearchPhrase,
44
getFullTextIndexWordsForItem,
55
} from "./src/FullTextSearchHelpers";
6-
export { InMemoryProvider, StoreData } from "./src/InMemoryProvider";
6+
export {
7+
InMemoryProvider,
8+
StoreData,
9+
ILiveConsumerConfigs,
10+
} from "./src/InMemoryProvider";
711
export {
812
ItemType,
913
KeyComponentType,

‎package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
{
22
"name": "@microsoft/objectstoreprovider",
3-
"version": "0.6.47",
3+
"version": "0.7.0",
44
"description": "A cross-browser object store library",
55
"author": "Mukundan Kavanur Kidambi <mukav@microsoft.com>",
66
"scripts": {

‎src/InMemoryProvider.ts

+49-18
Original file line numberDiff line numberDiff line change
@@ -62,6 +62,16 @@ export interface StoreData {
6262
mapType?: OrderedMapType;
6363
}
6464

65+
export interface ILiveConsumerConfigs {
66+
usePushForGetRange: boolean;
67+
}
68+
69+
export type GetLiveConsumerConfigsFn = () => ILiveConsumerConfigs;
70+
71+
const defaultLiveConsumerConfigs: ILiveConsumerConfigs = {
72+
usePushForGetRange: false,
73+
};
74+
6575
export class InMemoryProvider extends DbProvider {
6676
private _stores: Map<string, StoreData> = new Map();
6777

@@ -73,7 +83,8 @@ export class InMemoryProvider extends DbProvider {
7383
constructor(
7484
mapType?: OrderedMapType,
7585
supportsRollback = false,
76-
logger?: IObjectStoreProviderLogger
86+
logger?: IObjectStoreProviderLogger,
87+
private getLiveConfigs?: GetLiveConsumerConfigsFn
7788
) {
7889
super();
7990
this._mapType = mapType;
@@ -120,7 +131,8 @@ export class InMemoryProvider extends DbProvider {
120131
token,
121132
writeNeeded,
122133
this._supportsRollback!,
123-
this.logger
134+
this.logger,
135+
this.getLiveConfigs
124136
)
125137
);
126138
}
@@ -146,7 +158,8 @@ class InMemoryTransaction implements DbTransaction {
146158
private _transToken: TransactionToken,
147159
private _writeNeeded: boolean,
148160
private _supportsRollback: boolean,
149-
private logger: IObjectStoreProviderLogger
161+
private logger: IObjectStoreProviderLogger,
162+
private getLiveConfigs?: GetLiveConsumerConfigsFn
150163
) {
151164
// Close the transaction on the next tick. By definition, anything is completed synchronously here, so after an event tick
152165
// goes by, there can't have been anything pending.
@@ -217,7 +230,8 @@ class InMemoryTransaction implements DbTransaction {
217230
const ims = new InMemoryStore(
218231
this,
219232
store,
220-
this._writeNeeded && this._supportsRollback
233+
this._writeNeeded && this._supportsRollback,
234+
this.getLiveConfigs
221235
);
222236
this._stores.set(storeName, ims);
223237
return ims;
@@ -237,7 +251,8 @@ class InMemoryStore implements DbStore {
237251
constructor(
238252
private _trans: InMemoryTransaction,
239253
storeInfo: StoreData,
240-
private _supportsRollback: boolean
254+
private _supportsRollback: boolean,
255+
private getLiveConfigs?: GetLiveConsumerConfigsFn
241256
) {
242257
this._storeSchema = storeInfo.schema;
243258
if (this._supportsRollback) {
@@ -273,7 +288,8 @@ class InMemoryStore implements DbStore {
273288
this._mergedData,
274289
index,
275290
this._storeSchema.primaryKeyPath,
276-
this._mapType
291+
this._mapType,
292+
this.getLiveConfigs
277293
)
278294
);
279295
});
@@ -394,7 +410,8 @@ class InMemoryStore implements DbStore {
394410
this._mergedData,
395411
undefined as any,
396412
this._storeSchema.primaryKeyPath,
397-
this._mapType
413+
this._mapType,
414+
this.getLiveConfigs
398415
)
399416
);
400417
}
@@ -419,7 +436,8 @@ class InMemoryStore implements DbStore {
419436
this._mergedData,
420437
indexSchema,
421438
this._storeSchema.primaryKeyPath,
422-
this._mapType
439+
this._mapType,
440+
this.getLiveConfigs
423441
)
424442
);
425443
}
@@ -441,7 +459,8 @@ class InMemoryStore implements DbStore {
441459
this._mergedData,
442460
index,
443461
this._storeSchema.primaryKeyPath,
444-
this._mapType
462+
this._mapType,
463+
this.getLiveConfigs
445464
)
446465
);
447466
});
@@ -498,15 +517,18 @@ class InMemoryStore implements DbStore {
498517
class InMemoryIndex extends DbIndexFTSFromRangeQueries {
499518
private _indexTree: IOrderedMap<string, ItemType[]>;
500519
private _trans?: InMemoryTransaction;
520+
private getLiveConfigs: GetLiveConsumerConfigsFn;
501521
constructor(
502522
_mergedData: Map<string, ItemType>,
503523
indexSchema: IndexSchema,
504524
primaryKeyPath: KeyPathType,
505-
mapType?: OrderedMapType
525+
mapType?: OrderedMapType,
526+
getLiveConfigs?: GetLiveConsumerConfigsFn
506527
) {
507528
super(indexSchema, primaryKeyPath);
508529
this._indexTree = createOrderedMap(mapType);
509530
this.put(values(_mergedData), true);
531+
this.getLiveConfigs = getLiveConfigs ?? (() => defaultLiveConsumerConfigs);
510532
}
511533

512534
public internal_SetTransaction(trans: InMemoryTransaction) {
@@ -741,6 +763,15 @@ class InMemoryIndex extends DbIndexFTSFromRangeQueries {
741763
? this._indexTree.entriesReversed()
742764
: this._indexTree.entries();
743765
let values = [] as ItemType[];
766+
const { usePushForGetRange } = this.getLiveConfigs();
767+
const pushValues = (values: ItemType[], newValues: ItemType[]) => {
768+
newValues.forEach((v) => values.push(v));
769+
return values;
770+
};
771+
const concatValues = (values: ItemType[], newValues: ItemType[]) => {
772+
return values.concat(newValues);
773+
};
774+
const mergeFn = usePushForGetRange ? pushValues : concatValues;
744775
for (const entry of iterator) {
745776
const key = entry.key;
746777
if (key === undefined) {
@@ -769,16 +800,16 @@ class InMemoryIndex extends DbIndexFTSFromRangeQueries {
769800
}
770801

771802
if (this.isUniqueIndex()) {
772-
values = values.concat(this._indexTree.get(key) as ItemType[]);
803+
const newValues = this._indexTree.get(key) as ItemType[];
804+
values = mergeFn(values, newValues);
773805
} else {
774-
values = values.concat(
775-
this._getKeyValues(
776-
key,
777-
limit - values.length,
778-
Math.abs(offset),
779-
reverse
780-
)
806+
const newValues = this._getKeyValues(
807+
key,
808+
limit - values.length,
809+
Math.abs(offset),
810+
reverse
781811
);
812+
values = mergeFn(values, newValues);
782813

783814
if (offset < 0) {
784815
offset = 0;

0 commit comments

Comments
 (0)
Please sign in to comment.