@osdk/client 2.5.0-beta.1 → 2.5.0-beta.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +26 -0
- package/build/browser/Client.js +1 -1
- package/build/browser/Client.js.map +1 -1
- package/build/browser/intellisense.test.helpers/orderBySuggestionIsRight.js +8 -1
- package/build/browser/intellisense.test.helpers/orderBySuggestionIsRight.js.map +1 -1
- package/build/browser/intellisense.test.js +14 -4
- package/build/browser/intellisense.test.js.map +1 -1
- package/build/browser/object/Cache.js +1 -1
- package/build/browser/object/Cache.js.map +1 -1
- package/build/browser/object/SimpleCache.js +1 -1
- package/build/browser/object/SimpleCache.js.map +1 -1
- package/build/browser/object/convertWireToOsdkObjects.js +4 -0
- package/build/browser/object/convertWireToOsdkObjects.js.map +1 -1
- package/build/browser/object/fetchPage.js +8 -4
- package/build/browser/object/fetchPage.js.map +1 -1
- package/build/browser/object/geotimeseriesreference.test.js +0 -2
- package/build/browser/object/geotimeseriesreference.test.js.map +1 -1
- package/build/browser/objectSet/createObjectSet.js +3 -1
- package/build/browser/objectSet/createObjectSet.js.map +1 -1
- package/build/browser/observable/LinkPayload.js +2 -0
- package/build/browser/observable/LinkPayload.js.map +1 -0
- package/build/browser/observable/ObjectPayload.js.map +1 -1
- package/build/browser/observable/ObservableClient/ObserveLink.js +17 -0
- package/build/browser/observable/ObservableClient/ObserveLink.js.map +1 -0
- package/build/browser/observable/ObservableClient/common.js +2 -0
- package/build/browser/observable/ObservableClient/common.js.map +1 -0
- package/build/browser/observable/ObservableClient.js +20 -0
- package/build/browser/observable/ObservableClient.js.map +1 -1
- package/build/browser/observable/Unsubscribable.js +2 -0
- package/build/browser/observable/Unsubscribable.js.map +1 -0
- package/build/browser/observable/internal/AbstractHelper.js +50 -0
- package/build/browser/observable/internal/AbstractHelper.js.map +1 -0
- package/build/browser/observable/internal/BaseCollectionQuery.js +2 -0
- package/build/browser/observable/internal/BaseCollectionQuery.js.map +1 -0
- package/build/browser/observable/internal/BulkObjectLoader.js +3 -3
- package/build/browser/observable/internal/BulkObjectLoader.js.map +1 -1
- package/build/browser/observable/internal/CacheKeys.js +17 -2
- package/build/browser/observable/internal/CacheKeys.js.map +1 -1
- package/build/browser/observable/internal/Changes.js +6 -0
- package/build/browser/observable/internal/Changes.js.map +1 -1
- package/build/browser/observable/internal/KnownCacheKey.js +2 -0
- package/build/browser/observable/internal/KnownCacheKey.js.map +1 -0
- package/build/browser/observable/internal/Layer.js.map +1 -1
- package/build/browser/observable/internal/ListCacheKey.js +2 -0
- package/build/browser/observable/internal/ListCacheKey.js.map +1 -0
- package/build/browser/observable/internal/ListQuery.js +399 -161
- package/build/browser/observable/internal/ListQuery.js.map +1 -1
- package/build/browser/observable/internal/ObjectQuery.js +27 -2
- package/build/browser/observable/internal/ObjectQuery.js.map +1 -1
- package/build/browser/observable/internal/ObservableClientImpl.js +42 -2
- package/build/browser/observable/internal/ObservableClientImpl.js.map +1 -1
- package/build/browser/observable/internal/OptimisticJob.js +12 -3
- package/build/browser/observable/internal/OptimisticJob.js.map +1 -1
- package/build/browser/observable/internal/Query.js +17 -3
- package/build/browser/observable/internal/Query.js.map +1 -1
- package/build/browser/observable/internal/QuerySubscription.js +41 -0
- package/build/browser/observable/internal/QuerySubscription.js.map +1 -0
- package/build/browser/observable/internal/Store.invalidation.test.js +481 -0
- package/build/browser/observable/internal/Store.invalidation.test.js.map +1 -0
- package/build/browser/observable/internal/Store.js +38 -91
- package/build/browser/observable/internal/Store.js.map +1 -1
- package/build/browser/observable/internal/Store.test.js +318 -63
- package/build/browser/observable/internal/Store.test.js.map +1 -1
- package/build/browser/observable/internal/UnsubscribableWrapper.js +30 -0
- package/build/browser/observable/internal/UnsubscribableWrapper.js.map +1 -0
- package/build/browser/observable/internal/createCollectionConnectable.js +50 -0
- package/build/browser/observable/internal/createCollectionConnectable.js.map +1 -0
- package/build/browser/observable/internal/createCollectionConnectable.test.js +599 -0
- package/build/browser/observable/internal/createCollectionConnectable.test.js.map +1 -0
- package/build/browser/observable/internal/getObjectTypesThatInvalidate.js +216 -0
- package/build/browser/observable/internal/getObjectTypesThatInvalidate.js.map +1 -0
- package/build/browser/observable/internal/getObjectTypesThatInvalidate.test.js +382 -0
- package/build/browser/observable/internal/getObjectTypesThatInvalidate.test.js.map +1 -0
- package/build/browser/observable/internal/isObjectInstance.js +23 -0
- package/build/browser/observable/internal/isObjectInstance.js.map +1 -0
- package/build/browser/observable/internal/links/LinksHelper.js +37 -0
- package/build/browser/observable/internal/links/LinksHelper.js.map +1 -0
- package/build/browser/observable/internal/links/SpecificLinkCacheKey.js +2 -0
- package/build/browser/observable/internal/links/SpecificLinkCacheKey.js.map +1 -0
- package/build/browser/observable/internal/links/SpecificLinkQuery.js +185 -0
- package/build/browser/observable/internal/links/SpecificLinkQuery.js.map +1 -0
- package/build/browser/observable/internal/list/ListsHelper.js +49 -0
- package/build/browser/observable/internal/list/ListsHelper.js.map +1 -0
- package/build/browser/observable/internal/object/ObjectsHelper.js +34 -0
- package/build/browser/observable/internal/object/ObjectsHelper.js.map +1 -0
- package/build/browser/observable/internal/removeDuplicates.js +36 -0
- package/build/browser/observable/internal/removeDuplicates.js.map +1 -0
- package/build/browser/observable/internal/sorting/SortingStrategy.js +78 -0
- package/build/browser/observable/internal/sorting/SortingStrategy.js.map +1 -0
- package/build/browser/observable/internal/testUtils/invalidateList.js +23 -0
- package/build/browser/observable/internal/testUtils/invalidateList.js.map +1 -0
- package/build/browser/observable/internal/testUtils/observeLink/expectStandardObserveLink.js +69 -0
- package/build/browser/observable/internal/testUtils/observeLink/expectStandardObserveLink.js.map +1 -0
- package/build/browser/observable/internal/testUtils/observeObject/expectStandardObserveObject.js +56 -0
- package/build/browser/observable/internal/testUtils/observeObject/expectStandardObserveObject.js.map +1 -0
- package/build/browser/observable/internal/testUtils.js +70 -7
- package/build/browser/observable/internal/testUtils.js.map +1 -1
- package/build/browser/public/unstable-do-not-use.js.map +1 -1
- package/build/browser/util/UserAgent.js +2 -2
- package/build/browser/util/streamutils.js +1 -1
- package/build/browser/util/streamutils.js.map +1 -1
- package/build/cjs/{chunk-URDXPIRU.cjs → chunk-N5DMNYGH.cjs} +58 -54
- package/build/cjs/chunk-N5DMNYGH.cjs.map +1 -0
- package/build/cjs/{chunk-BJYCRD5Y.cjs → chunk-NWD33DSJ.cjs} +23 -19
- package/build/cjs/chunk-NWD33DSJ.cjs.map +1 -0
- package/build/cjs/index.cjs +7 -7
- package/build/cjs/public/internal.cjs +8 -8
- package/build/cjs/public/unstable-do-not-use.cjs +779 -268
- package/build/cjs/public/unstable-do-not-use.cjs.map +1 -1
- package/build/cjs/public/unstable-do-not-use.d.cts +161 -12
- package/build/esm/Client.js +1 -1
- package/build/esm/Client.js.map +1 -1
- package/build/esm/intellisense.test.helpers/orderBySuggestionIsRight.js +8 -1
- package/build/esm/intellisense.test.helpers/orderBySuggestionIsRight.js.map +1 -1
- package/build/esm/intellisense.test.js +14 -4
- package/build/esm/intellisense.test.js.map +1 -1
- package/build/esm/object/Cache.js +1 -1
- package/build/esm/object/Cache.js.map +1 -1
- package/build/esm/object/SimpleCache.js +1 -1
- package/build/esm/object/SimpleCache.js.map +1 -1
- package/build/esm/object/convertWireToOsdkObjects.js +4 -0
- package/build/esm/object/convertWireToOsdkObjects.js.map +1 -1
- package/build/esm/object/fetchPage.js +8 -4
- package/build/esm/object/fetchPage.js.map +1 -1
- package/build/esm/object/geotimeseriesreference.test.js +0 -2
- package/build/esm/object/geotimeseriesreference.test.js.map +1 -1
- package/build/esm/objectSet/createObjectSet.js +3 -1
- package/build/esm/objectSet/createObjectSet.js.map +1 -1
- package/build/esm/observable/LinkPayload.js +2 -0
- package/build/esm/observable/LinkPayload.js.map +1 -0
- package/build/esm/observable/ObjectPayload.js.map +1 -1
- package/build/esm/observable/ObservableClient/ObserveLink.js +17 -0
- package/build/esm/observable/ObservableClient/ObserveLink.js.map +1 -0
- package/build/esm/observable/ObservableClient/common.js +2 -0
- package/build/esm/observable/ObservableClient/common.js.map +1 -0
- package/build/esm/observable/ObservableClient.js +20 -0
- package/build/esm/observable/ObservableClient.js.map +1 -1
- package/build/esm/observable/Unsubscribable.js +2 -0
- package/build/esm/observable/Unsubscribable.js.map +1 -0
- package/build/esm/observable/internal/AbstractHelper.js +50 -0
- package/build/esm/observable/internal/AbstractHelper.js.map +1 -0
- package/build/esm/observable/internal/BaseCollectionQuery.js +2 -0
- package/build/esm/observable/internal/BaseCollectionQuery.js.map +1 -0
- package/build/esm/observable/internal/BulkObjectLoader.js +3 -3
- package/build/esm/observable/internal/BulkObjectLoader.js.map +1 -1
- package/build/esm/observable/internal/CacheKeys.js +17 -2
- package/build/esm/observable/internal/CacheKeys.js.map +1 -1
- package/build/esm/observable/internal/Changes.js +6 -0
- package/build/esm/observable/internal/Changes.js.map +1 -1
- package/build/esm/observable/internal/KnownCacheKey.js +2 -0
- package/build/esm/observable/internal/KnownCacheKey.js.map +1 -0
- package/build/esm/observable/internal/Layer.js.map +1 -1
- package/build/esm/observable/internal/ListCacheKey.js +2 -0
- package/build/esm/observable/internal/ListCacheKey.js.map +1 -0
- package/build/esm/observable/internal/ListQuery.js +399 -161
- package/build/esm/observable/internal/ListQuery.js.map +1 -1
- package/build/esm/observable/internal/ObjectQuery.js +27 -2
- package/build/esm/observable/internal/ObjectQuery.js.map +1 -1
- package/build/esm/observable/internal/ObservableClientImpl.js +42 -2
- package/build/esm/observable/internal/ObservableClientImpl.js.map +1 -1
- package/build/esm/observable/internal/OptimisticJob.js +12 -3
- package/build/esm/observable/internal/OptimisticJob.js.map +1 -1
- package/build/esm/observable/internal/Query.js +17 -3
- package/build/esm/observable/internal/Query.js.map +1 -1
- package/build/esm/observable/internal/QuerySubscription.js +41 -0
- package/build/esm/observable/internal/QuerySubscription.js.map +1 -0
- package/build/esm/observable/internal/Store.invalidation.test.js +481 -0
- package/build/esm/observable/internal/Store.invalidation.test.js.map +1 -0
- package/build/esm/observable/internal/Store.js +38 -91
- package/build/esm/observable/internal/Store.js.map +1 -1
- package/build/esm/observable/internal/Store.test.js +318 -63
- package/build/esm/observable/internal/Store.test.js.map +1 -1
- package/build/esm/observable/internal/UnsubscribableWrapper.js +30 -0
- package/build/esm/observable/internal/UnsubscribableWrapper.js.map +1 -0
- package/build/esm/observable/internal/createCollectionConnectable.js +50 -0
- package/build/esm/observable/internal/createCollectionConnectable.js.map +1 -0
- package/build/esm/observable/internal/createCollectionConnectable.test.js +599 -0
- package/build/esm/observable/internal/createCollectionConnectable.test.js.map +1 -0
- package/build/esm/observable/internal/getObjectTypesThatInvalidate.js +216 -0
- package/build/esm/observable/internal/getObjectTypesThatInvalidate.js.map +1 -0
- package/build/esm/observable/internal/getObjectTypesThatInvalidate.test.js +382 -0
- package/build/esm/observable/internal/getObjectTypesThatInvalidate.test.js.map +1 -0
- package/build/esm/observable/internal/isObjectInstance.js +23 -0
- package/build/esm/observable/internal/isObjectInstance.js.map +1 -0
- package/build/esm/observable/internal/links/LinksHelper.js +37 -0
- package/build/esm/observable/internal/links/LinksHelper.js.map +1 -0
- package/build/esm/observable/internal/links/SpecificLinkCacheKey.js +2 -0
- package/build/esm/observable/internal/links/SpecificLinkCacheKey.js.map +1 -0
- package/build/esm/observable/internal/links/SpecificLinkQuery.js +185 -0
- package/build/esm/observable/internal/links/SpecificLinkQuery.js.map +1 -0
- package/build/esm/observable/internal/list/ListsHelper.js +49 -0
- package/build/esm/observable/internal/list/ListsHelper.js.map +1 -0
- package/build/esm/observable/internal/object/ObjectsHelper.js +34 -0
- package/build/esm/observable/internal/object/ObjectsHelper.js.map +1 -0
- package/build/esm/observable/internal/removeDuplicates.js +36 -0
- package/build/esm/observable/internal/removeDuplicates.js.map +1 -0
- package/build/esm/observable/internal/sorting/SortingStrategy.js +78 -0
- package/build/esm/observable/internal/sorting/SortingStrategy.js.map +1 -0
- package/build/esm/observable/internal/testUtils/invalidateList.js +23 -0
- package/build/esm/observable/internal/testUtils/invalidateList.js.map +1 -0
- package/build/esm/observable/internal/testUtils/observeLink/expectStandardObserveLink.js +69 -0
- package/build/esm/observable/internal/testUtils/observeLink/expectStandardObserveLink.js.map +1 -0
- package/build/esm/observable/internal/testUtils/observeObject/expectStandardObserveObject.js +56 -0
- package/build/esm/observable/internal/testUtils/observeObject/expectStandardObserveObject.js.map +1 -0
- package/build/esm/observable/internal/testUtils.js +70 -7
- package/build/esm/observable/internal/testUtils.js.map +1 -1
- package/build/esm/public/unstable-do-not-use.js.map +1 -1
- package/build/esm/util/UserAgent.js +2 -2
- package/build/esm/util/streamutils.js +1 -1
- package/build/esm/util/streamutils.js.map +1 -1
- package/build/types/Client.d.ts +1 -1
- package/build/types/object/fetchPage.d.ts.map +1 -1
- package/build/types/observable/LinkPayload.d.ts +9 -0
- package/build/types/observable/LinkPayload.d.ts.map +1 -0
- package/build/types/observable/ObjectPayload.d.ts +4 -0
- package/build/types/observable/ObjectPayload.d.ts.map +1 -1
- package/build/types/observable/ObservableClient/ObserveLink.d.ts +32 -0
- package/build/types/observable/ObservableClient/ObserveLink.d.ts.map +1 -0
- package/build/types/observable/ObservableClient/common.d.ts +67 -0
- package/build/types/observable/ObservableClient/common.d.ts.map +1 -0
- package/build/types/observable/ObservableClient.d.ts +82 -15
- package/build/types/observable/ObservableClient.d.ts.map +1 -1
- package/build/types/observable/Unsubscribable.d.ts +3 -0
- package/build/types/observable/Unsubscribable.d.ts.map +1 -0
- package/build/types/observable/internal/AbstractHelper.d.ts +15 -0
- package/build/types/observable/internal/AbstractHelper.d.ts.map +1 -0
- package/build/types/observable/internal/BaseCollectionQuery.d.ts +64 -0
- package/build/types/observable/internal/BaseCollectionQuery.d.ts.map +1 -0
- package/build/types/observable/internal/BulkObjectLoader.d.ts.map +1 -1
- package/build/types/observable/internal/CacheKeys.d.ts +10 -4
- package/build/types/observable/internal/CacheKeys.d.ts.map +1 -1
- package/build/types/observable/internal/Changes.d.ts +7 -4
- package/build/types/observable/internal/Changes.d.ts.map +1 -1
- package/build/types/observable/internal/KnownCacheKey.d.ts +4 -0
- package/build/types/observable/internal/KnownCacheKey.d.ts.map +1 -0
- package/build/types/observable/internal/Layer.d.ts +7 -7
- package/build/types/observable/internal/Layer.d.ts.map +1 -1
- package/build/types/observable/internal/ListCacheKey.d.ts +7 -0
- package/build/types/observable/internal/ListCacheKey.d.ts.map +1 -0
- package/build/types/observable/internal/ListQuery.d.ts +160 -21
- package/build/types/observable/internal/ListQuery.d.ts.map +1 -1
- package/build/types/observable/internal/ObjectQuery.d.ts +3 -1
- package/build/types/observable/internal/ObjectQuery.d.ts.map +1 -1
- package/build/types/observable/internal/OptimisticJob.d.ts.map +1 -1
- package/build/types/observable/internal/Query.d.ts +6 -5
- package/build/types/observable/internal/Query.d.ts.map +1 -1
- package/build/types/observable/internal/QuerySubscription.d.ts +1 -0
- package/build/types/observable/internal/QuerySubscription.d.ts.map +1 -0
- package/build/types/observable/internal/Store.d.ts +34 -31
- package/build/types/observable/internal/Store.d.ts.map +1 -1
- package/build/types/observable/internal/Store.invalidation.test.d.ts +1 -0
- package/build/types/observable/internal/Store.invalidation.test.d.ts.map +1 -0
- package/build/types/observable/internal/Store.test.d.ts +4 -1
- package/build/types/observable/internal/Store.test.d.ts.map +1 -1
- package/build/types/observable/internal/UnsubscribableWrapper.d.ts +1 -0
- package/build/types/observable/internal/UnsubscribableWrapper.d.ts.map +1 -0
- package/build/types/observable/internal/createCollectionConnectable.d.ts +38 -0
- package/build/types/observable/internal/createCollectionConnectable.d.ts.map +1 -0
- package/build/types/observable/internal/createCollectionConnectable.test.d.ts +1 -0
- package/build/types/observable/internal/createCollectionConnectable.test.d.ts.map +1 -0
- package/build/types/observable/internal/getObjectTypesThatInvalidate.d.ts +9 -0
- package/build/types/observable/internal/getObjectTypesThatInvalidate.d.ts.map +1 -0
- package/build/types/observable/internal/getObjectTypesThatInvalidate.test.d.ts +1 -0
- package/build/types/observable/internal/getObjectTypesThatInvalidate.test.d.ts.map +1 -0
- package/build/types/observable/internal/isObjectInstance.d.ts +5 -0
- package/build/types/observable/internal/isObjectInstance.d.ts.map +1 -0
- package/build/types/observable/internal/links/LinksHelper.d.ts +29 -0
- package/build/types/observable/internal/links/LinksHelper.d.ts.map +1 -0
- package/build/types/observable/internal/links/SpecificLinkCacheKey.d.ts +17 -0
- package/build/types/observable/internal/links/SpecificLinkCacheKey.d.ts.map +1 -0
- package/build/types/observable/internal/links/SpecificLinkQuery.d.ts +45 -0
- package/build/types/observable/internal/links/SpecificLinkQuery.d.ts.map +1 -0
- package/build/types/observable/internal/list/ListsHelper.d.ts +17 -0
- package/build/types/observable/internal/list/ListsHelper.d.ts.map +1 -0
- package/build/types/observable/internal/object/ObjectsHelper.d.ts +11 -0
- package/build/types/observable/internal/object/ObjectsHelper.d.ts.map +1 -0
- package/build/types/observable/internal/removeDuplicates.d.ts +11 -0
- package/build/types/observable/internal/removeDuplicates.d.ts.map +1 -0
- package/build/types/observable/internal/sorting/SortingStrategy.d.ts +41 -0
- package/build/types/observable/internal/sorting/SortingStrategy.d.ts.map +1 -0
- package/build/types/observable/internal/testUtils/invalidateList.d.ts +9 -0
- package/build/types/observable/internal/testUtils/invalidateList.d.ts.map +1 -0
- package/build/types/observable/internal/testUtils/observeLink/expectStandardObserveLink.d.ts +46 -0
- package/build/types/observable/internal/testUtils/observeLink/expectStandardObserveLink.d.ts.map +1 -0
- package/build/types/observable/internal/testUtils/observeObject/expectStandardObserveObject.d.ts +26 -0
- package/build/types/observable/internal/testUtils/observeObject/expectStandardObserveObject.d.ts.map +1 -0
- package/build/types/observable/internal/testUtils.d.ts +25 -3
- package/build/types/observable/internal/testUtils.d.ts.map +1 -1
- package/build/types/public/unstable-do-not-use.d.ts +4 -2
- package/build/types/public/unstable-do-not-use.d.ts.map +1 -1
- package/package.json +10 -10
- package/build/cjs/chunk-BJYCRD5Y.cjs.map +0 -1
- package/build/cjs/chunk-URDXPIRU.cjs.map +0 -1
|
@@ -16,44 +16,74 @@
|
|
|
16
16
|
|
|
17
17
|
import deepEqual from "fast-deep-equal";
|
|
18
18
|
import groupBy from "object.groupby";
|
|
19
|
-
import { auditTime, combineLatest, connectable, map, of, ReplaySubject, switchMap } from "rxjs";
|
|
20
19
|
import invariant from "tiny-invariant";
|
|
21
20
|
import { additionalContext } from "../../Client.js";
|
|
22
21
|
import { ObjectDefRef, UnderlyingOsdkObject } from "../../object/convertWireToOsdkObjects/InternalSymbols.js";
|
|
23
22
|
import { DEBUG_ONLY__cacheKeysToString } from "./CacheKey.js";
|
|
24
23
|
import { DEBUG_ONLY__changesToString } from "./Changes.js";
|
|
24
|
+
import { createCollectionConnectable } from "./createCollectionConnectable.js";
|
|
25
|
+
import { isObjectInstance } from "./isObjectInstance.js";
|
|
25
26
|
import { objectSortaMatchesWhereClause as objectMatchesWhereClause } from "./objectMatchesWhereClause.js";
|
|
26
27
|
import { storeOsdkInstances } from "./ObjectQuery.js";
|
|
27
28
|
import { Query } from "./Query.js";
|
|
29
|
+
import { removeDuplicates } from "./removeDuplicates.js";
|
|
30
|
+
import { NoOpSortingStrategy, OrderBySortingStrategy } from "./sorting/SortingStrategy.js";
|
|
28
31
|
export const API_NAME_IDX = 1;
|
|
29
32
|
export const TYPE_IDX = 0;
|
|
30
33
|
export const WHERE_IDX = 2;
|
|
31
34
|
export const ORDER_BY_IDX = 3;
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
35
|
+
/**
|
|
36
|
+
* Base class for collection-based queries (lists and links)
|
|
37
|
+
* Provides common functionality for working with collections of objects
|
|
38
|
+
*/
|
|
39
|
+
export class BaseListQuery extends Query {
|
|
40
|
+
/**
|
|
41
|
+
* The sorting strategy to use for this collection
|
|
42
|
+
* @protected
|
|
43
|
+
*/
|
|
44
|
+
sortingStrategy = new NoOpSortingStrategy();
|
|
45
|
+
|
|
46
|
+
// Collection-specific behavior is implemented by subclasses
|
|
47
|
+
/**
|
|
48
|
+
* Token for the next page of results
|
|
49
|
+
* @protected
|
|
50
|
+
*/
|
|
51
|
+
|
|
52
|
+
/**
|
|
53
|
+
* Promise tracking an in-progress page fetch
|
|
54
|
+
* @protected
|
|
55
|
+
*/
|
|
36
56
|
|
|
37
57
|
//
|
|
38
58
|
// Shared Implementations
|
|
39
59
|
//
|
|
40
60
|
|
|
41
61
|
/**
|
|
42
|
-
*
|
|
43
|
-
*
|
|
62
|
+
* Standard method to update a list of objects
|
|
63
|
+
* Handles common list update patterns for both ListQuery and SpecificLinkQuery
|
|
44
64
|
*
|
|
45
|
-
* @param
|
|
46
|
-
* @param
|
|
47
|
-
* @param
|
|
48
|
-
* @param
|
|
49
|
-
* @returns
|
|
65
|
+
* @param items Objects or cache keys to add to the list
|
|
66
|
+
* @param status Status to set for the list
|
|
67
|
+
* @param batch Batch context to use
|
|
68
|
+
* @param append Whether to append to the existing list or replace it
|
|
69
|
+
* @returns The updated entry
|
|
50
70
|
*/
|
|
51
|
-
_updateList(
|
|
71
|
+
_updateList(items, status, batch, append = false) {
|
|
52
72
|
if (process.env.NODE_ENV !== "production") {
|
|
53
73
|
const logger = process.env.NODE_ENV !== "production" ? this.logger?.child({
|
|
54
74
|
methodName: "updateList"
|
|
55
75
|
}) : this.logger;
|
|
56
|
-
logger?.debug(`{status: ${status}}`, JSON.stringify(
|
|
76
|
+
logger?.debug(`{status: ${status}, append: ${append}}`, JSON.stringify(items, null, 2));
|
|
77
|
+
}
|
|
78
|
+
let objectCacheKeys;
|
|
79
|
+
if (items.length === 0) {
|
|
80
|
+
objectCacheKeys = [];
|
|
81
|
+
} else if (isObjectInstance(items[0])) {
|
|
82
|
+
// Items are object instances, need to store them first
|
|
83
|
+
objectCacheKeys = this.storeObjects(items, batch);
|
|
84
|
+
} else {
|
|
85
|
+
// Items are already cache keys
|
|
86
|
+
objectCacheKeys = items;
|
|
57
87
|
}
|
|
58
88
|
objectCacheKeys = this.#retainReleaseAppend(batch, append, objectCacheKeys);
|
|
59
89
|
objectCacheKeys = this._sortCacheKeys(objectCacheKeys, batch);
|
|
@@ -62,14 +92,32 @@ class BaseListQuery extends Query {
|
|
|
62
92
|
data: objectCacheKeys
|
|
63
93
|
}, status, batch);
|
|
64
94
|
}
|
|
95
|
+
|
|
96
|
+
/**
|
|
97
|
+
* Common implementation for writing to store for collection-based queries
|
|
98
|
+
* @param data The collection data to write to the store
|
|
99
|
+
* @param status The status to set
|
|
100
|
+
* @param batch The batch context
|
|
101
|
+
*/
|
|
65
102
|
writeToStore(data, status, batch) {
|
|
66
103
|
const entry = batch.read(this.cacheKey);
|
|
67
104
|
if (entry && deepEqual(data, entry.value)) {
|
|
105
|
+
// Check if both data AND status are the same
|
|
106
|
+
if (entry.status === status) {
|
|
107
|
+
if (process.env.NODE_ENV !== "production") {
|
|
108
|
+
this.logger?.child({
|
|
109
|
+
methodName: "writeToStore"
|
|
110
|
+
}).debug(`Collection data was deep equal and status unchanged (${status}), skipping update`);
|
|
111
|
+
}
|
|
112
|
+
// Return the existing entry without writing to avoid unnecessary notifications
|
|
113
|
+
return entry;
|
|
114
|
+
}
|
|
68
115
|
if (process.env.NODE_ENV !== "production") {
|
|
69
116
|
this.logger?.child({
|
|
70
117
|
methodName: "writeToStore"
|
|
71
|
-
}).debug(`
|
|
118
|
+
}).debug(`Collection data was deep equal, just updating status from ${entry.status} to ${status}`);
|
|
72
119
|
}
|
|
120
|
+
// Keep the same value but update status and lastUpdated
|
|
73
121
|
return batch.write(this.cacheKey, entry.value, status);
|
|
74
122
|
}
|
|
75
123
|
if (process.env.NODE_ENV !== "production") {
|
|
@@ -78,9 +126,37 @@ class BaseListQuery extends Query {
|
|
|
78
126
|
}).debug(`{status: ${status}},`, DEBUG_ONLY__cacheKeysToString(data.data));
|
|
79
127
|
}
|
|
80
128
|
const ret = batch.write(this.cacheKey, data, status);
|
|
81
|
-
|
|
129
|
+
this.registerCacheChanges(batch);
|
|
82
130
|
return ret;
|
|
83
131
|
}
|
|
132
|
+
|
|
133
|
+
/**
|
|
134
|
+
* Register changes to the cache based on the specific collection type
|
|
135
|
+
* Implemented by subclasses to handle specific change registration
|
|
136
|
+
*/
|
|
137
|
+
|
|
138
|
+
/**
|
|
139
|
+
* Common method to store objects in the cache and return their cache keys
|
|
140
|
+
* Used by collection queries when storing object references
|
|
141
|
+
*
|
|
142
|
+
* @param objects Array of objects to store
|
|
143
|
+
* @param batch The batch context to use
|
|
144
|
+
* @returns Array of cache keys for the stored objects
|
|
145
|
+
*/
|
|
146
|
+
storeObjects(objects, batch) {
|
|
147
|
+
// Store the individual objects in their respective cache entries
|
|
148
|
+
return objects.length > 0 ? storeOsdkInstances(this.store, objects, batch) : [];
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
/**
|
|
152
|
+
* Common method for managing object reference counting and appending results
|
|
153
|
+
* Used by collection queries when updating object references
|
|
154
|
+
*
|
|
155
|
+
* @param batch The batch context to use
|
|
156
|
+
* @param append Whether to append to existing objects or replace them
|
|
157
|
+
* @param objectCacheKeys Array of object cache keys to process
|
|
158
|
+
* @returns The final array of object cache keys after retain/release/append
|
|
159
|
+
*/
|
|
84
160
|
#retainReleaseAppend(batch, append, objectCacheKeys) {
|
|
85
161
|
const existingList = batch.read(this.cacheKey);
|
|
86
162
|
|
|
@@ -115,73 +191,102 @@ class BaseListQuery extends Query {
|
|
|
115
191
|
}
|
|
116
192
|
});
|
|
117
193
|
}
|
|
118
|
-
}
|
|
119
|
-
export class ListQuery extends BaseListQuery {
|
|
120
|
-
// pageSize?: number; // this is the internal page size. we need to track this properly
|
|
121
|
-
|
|
122
|
-
#type;
|
|
123
|
-
#apiName;
|
|
124
|
-
#whereClause;
|
|
125
194
|
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
apiName: this.#apiName
|
|
144
|
-
}).where(this.#whereClause);
|
|
145
|
-
this.#sortFns = createOrderBySortFns(this.#orderBy);
|
|
146
|
-
}
|
|
147
|
-
get canonicalWhere() {
|
|
148
|
-
return this.#whereClause;
|
|
195
|
+
/**
|
|
196
|
+
* Creates a payload from collection parameters
|
|
197
|
+
* Default implementation that covers common fields for all collection types
|
|
198
|
+
* Subclasses may override to add type-specific fields if needed
|
|
199
|
+
*
|
|
200
|
+
* @param params Common collection parameters
|
|
201
|
+
* @returns A typed payload for the specific collection type
|
|
202
|
+
*/
|
|
203
|
+
createPayload(params) {
|
|
204
|
+
return {
|
|
205
|
+
resolvedList: params.resolvedData,
|
|
206
|
+
isOptimistic: params.isOptimistic,
|
|
207
|
+
fetchMore: this.fetchMore,
|
|
208
|
+
hasMore: this.nextPageToken != null,
|
|
209
|
+
status: params.status,
|
|
210
|
+
lastUpdated: params.lastUpdated
|
|
211
|
+
}; // Type assertion needed since we don't know exact subtype
|
|
149
212
|
}
|
|
213
|
+
|
|
214
|
+
/**
|
|
215
|
+
* Creates a connectable observable for this collection
|
|
216
|
+
* Common implementation shared by all collection types
|
|
217
|
+
*
|
|
218
|
+
* @param subject The subject to connect to
|
|
219
|
+
* @returns A connectable observable of the collection's payload type
|
|
220
|
+
*/
|
|
150
221
|
_createConnectable(subject) {
|
|
151
|
-
return
|
|
152
|
-
return combineLatest({
|
|
153
|
-
resolvedList: listEntry?.value?.data == null || listEntry.value.data.length === 0 ? of([]) : combineLatest(listEntry.value.data.map(cacheKey => this.store.getSubject(cacheKey).pipe(map(objectEntry => objectEntry?.value)))),
|
|
154
|
-
isOptimistic: of(listEntry.isOptimistic),
|
|
155
|
-
fetchMore: of(this.fetchMore),
|
|
156
|
-
hasMore: of(this.#nextPageToken != null),
|
|
157
|
-
status: of(listEntry.status),
|
|
158
|
-
lastUpdated: of(listEntry.lastUpdated)
|
|
159
|
-
});
|
|
160
|
-
}),
|
|
161
|
-
// like throttle but returns the tail
|
|
162
|
-
auditTime(0)), {
|
|
163
|
-
resetOnDisconnect: false,
|
|
164
|
-
connector: () => new ReplaySubject(1)
|
|
165
|
-
});
|
|
222
|
+
return createCollectionConnectable(subject, this.store, params => this.createPayload(params));
|
|
166
223
|
}
|
|
224
|
+
|
|
225
|
+
/**
|
|
226
|
+
* @override Reset pagination state before a fetch
|
|
227
|
+
*/
|
|
167
228
|
_preFetch() {
|
|
168
|
-
this
|
|
229
|
+
this.nextPageToken = undefined;
|
|
230
|
+
super._preFetch();
|
|
169
231
|
}
|
|
232
|
+
|
|
233
|
+
/**
|
|
234
|
+
* Common fetchMore implementation for pagination
|
|
235
|
+
* Handles pending request management and loading states
|
|
236
|
+
*/
|
|
237
|
+
fetchMore = () => {
|
|
238
|
+
if (this.pendingPageFetch) {
|
|
239
|
+
return this.pendingPageFetch;
|
|
240
|
+
}
|
|
241
|
+
if (this.pendingFetch) {
|
|
242
|
+
this.pendingPageFetch = new Promise(async res => {
|
|
243
|
+
await this.pendingFetch;
|
|
244
|
+
res(this.fetchMore());
|
|
245
|
+
});
|
|
246
|
+
return this.pendingPageFetch;
|
|
247
|
+
}
|
|
248
|
+
if (this.nextPageToken == null) {
|
|
249
|
+
return Promise.resolve(undefined);
|
|
250
|
+
}
|
|
251
|
+
this.store.batch({}, batch => {
|
|
252
|
+
this.setStatus("loading", batch);
|
|
253
|
+
});
|
|
254
|
+
this.pendingFetch = this.fetchPageAndUpdate("loaded", this.abortController?.signal).then(() => void 0).finally(() => {
|
|
255
|
+
this.pendingPageFetch = undefined;
|
|
256
|
+
});
|
|
257
|
+
return this.pendingFetch;
|
|
258
|
+
};
|
|
259
|
+
|
|
260
|
+
/**
|
|
261
|
+
* Minimum number of results to load initially
|
|
262
|
+
* May be overridden by subclasses for specific collection types
|
|
263
|
+
* @protected
|
|
264
|
+
*/
|
|
265
|
+
minResultsToLoad = 0;
|
|
266
|
+
|
|
267
|
+
/**
|
|
268
|
+
* Common _fetchAndStore implementation for pagination
|
|
269
|
+
* Uses fetchPageAndUpdate to load the initial set of data
|
|
270
|
+
* Will load multiple pages if necessary to reach minResultsToLoad
|
|
271
|
+
*/
|
|
170
272
|
async _fetchAndStore() {
|
|
171
273
|
if (process.env.NODE_ENV !== "production") {
|
|
172
274
|
this.logger?.child({
|
|
173
275
|
methodName: "_fetchAndStore"
|
|
174
276
|
}).debug("fetching pages");
|
|
175
277
|
}
|
|
278
|
+
|
|
279
|
+
// Keep fetching pages until we have the minimum number of results or no more pages
|
|
176
280
|
while (true) {
|
|
177
|
-
const entry = await this
|
|
281
|
+
const entry = await this.fetchPageAndUpdate("loading", this.abortController?.signal);
|
|
178
282
|
if (!entry) {
|
|
179
283
|
// we were aborted
|
|
180
284
|
return;
|
|
181
285
|
}
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
|
|
286
|
+
|
|
287
|
+
// Check if we have enough results or no more pages
|
|
288
|
+
const count = entry.value?.data.length || 0;
|
|
289
|
+
if (count >= this.minResultsToLoad || this.nextPageToken == null) {
|
|
185
290
|
break;
|
|
186
291
|
}
|
|
187
292
|
}
|
|
@@ -190,69 +295,237 @@ export class ListQuery extends BaseListQuery {
|
|
|
190
295
|
});
|
|
191
296
|
return Promise.resolve();
|
|
192
297
|
}
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
298
|
+
|
|
299
|
+
/**
|
|
300
|
+
* Template method for fetching a page of data and updating the store
|
|
301
|
+
* Provides common error handling and abort signal checking
|
|
302
|
+
*
|
|
303
|
+
* @param status The status to set for the entry
|
|
304
|
+
* @param signal Optional AbortSignal for cancellation
|
|
305
|
+
* @returns A promise that resolves to the updated entry or undefined if aborted
|
|
306
|
+
*/
|
|
307
|
+
async fetchPageAndUpdate(status, signal) {
|
|
308
|
+
if (process.env.NODE_ENV !== "production") {
|
|
309
|
+
this.logger?.child({
|
|
310
|
+
methodName: "fetchPageAndUpdate"
|
|
311
|
+
}).debug(`Fetching data with status: ${status}`);
|
|
203
312
|
}
|
|
204
|
-
|
|
205
|
-
|
|
313
|
+
|
|
314
|
+
// Early abort check
|
|
315
|
+
if (signal?.aborted) {
|
|
316
|
+
return undefined;
|
|
206
317
|
}
|
|
207
|
-
this.store.batch({}, batch => {
|
|
208
|
-
this.setStatus("loading", batch);
|
|
209
|
-
});
|
|
210
|
-
this.pendingFetch = this.#fetchPageAndUpdate(this.#objectSet, "loaded", this.abortController?.signal).finally(() => {
|
|
211
|
-
this.#pendingPageFetch = undefined;
|
|
212
|
-
});
|
|
213
|
-
return this.pendingFetch;
|
|
214
|
-
};
|
|
215
|
-
async #fetchPageAndUpdate(objectSet, status, signal) {
|
|
216
|
-
const append = this.#nextPageToken != null;
|
|
217
318
|
try {
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
$nextPageToken: this.#nextPageToken,
|
|
223
|
-
$pageSize: this.options.pageSize,
|
|
224
|
-
// For now this keeps the shared test code from falling apart
|
|
225
|
-
// but shouldn't be needed ideally
|
|
226
|
-
...(Object.keys(this.#orderBy).length > 0 ? {
|
|
227
|
-
$orderBy: this.#orderBy
|
|
228
|
-
} : {})
|
|
229
|
-
});
|
|
319
|
+
// Call the subclass-specific implementation to fetch data
|
|
320
|
+
const result = await this.fetchPageData(signal);
|
|
321
|
+
|
|
322
|
+
// Check for abort again after fetch
|
|
230
323
|
if (signal?.aborted) {
|
|
231
|
-
return;
|
|
324
|
+
return undefined;
|
|
232
325
|
}
|
|
233
|
-
this.#nextPageToken = nextPageToken;
|
|
234
326
|
|
|
235
|
-
//
|
|
236
|
-
// so we need to fetch them all here
|
|
237
|
-
if (this.#type === "interface") {
|
|
238
|
-
data = await reloadDataAsFullObjects(this.store.client, data);
|
|
239
|
-
}
|
|
327
|
+
// Store the fetched data using batch operations
|
|
240
328
|
const {
|
|
241
329
|
retVal
|
|
242
330
|
} = this.store.batch({}, batch => {
|
|
243
|
-
|
|
331
|
+
const append = this.nextPageToken != null;
|
|
332
|
+
const finalStatus = result.nextPageToken ? status : "loaded";
|
|
333
|
+
return this._updateList(this.storeObjects(result.data, batch), finalStatus, batch, append);
|
|
244
334
|
});
|
|
245
335
|
return retVal;
|
|
246
|
-
} catch (
|
|
247
|
-
|
|
248
|
-
|
|
336
|
+
} catch (error) {
|
|
337
|
+
// Log any errors that occur
|
|
338
|
+
if (process.env.NODE_ENV !== "production") {
|
|
339
|
+
this.logger?.child({
|
|
340
|
+
methodName: "fetchPageAndUpdate"
|
|
341
|
+
}).error("Error fetching data", error);
|
|
342
|
+
}
|
|
249
343
|
|
|
250
|
-
//
|
|
251
|
-
|
|
252
|
-
|
|
344
|
+
// For unexpected errors, write error status if not aborted
|
|
345
|
+
if (!signal?.aborted) {
|
|
346
|
+
const {
|
|
347
|
+
retVal
|
|
348
|
+
} = this.store.batch({}, batch => {
|
|
349
|
+
return this.handleFetchError(error, status, batch);
|
|
350
|
+
});
|
|
351
|
+
return retVal;
|
|
352
|
+
}
|
|
353
|
+
|
|
354
|
+
// If aborted, just return undefined
|
|
355
|
+
return undefined;
|
|
253
356
|
}
|
|
254
357
|
}
|
|
255
358
|
|
|
359
|
+
/**
|
|
360
|
+
* Abstract method that subclasses implement for their specific data fetching logic
|
|
361
|
+
*
|
|
362
|
+
* @param signal Optional AbortSignal for cancellation
|
|
363
|
+
* @returns A promise that resolves to the fetched data
|
|
364
|
+
*/
|
|
365
|
+
|
|
366
|
+
/**
|
|
367
|
+
* Handle fetch errors by setting appropriate error state
|
|
368
|
+
* Default implementation that subclasses can override
|
|
369
|
+
*
|
|
370
|
+
* @param error The error that occurred
|
|
371
|
+
* @param status The intended status if successful
|
|
372
|
+
* @param batch The batch context to use
|
|
373
|
+
* @returns The updated entry with error status
|
|
374
|
+
*/
|
|
375
|
+
handleFetchError(_error, _status, batch) {
|
|
376
|
+
// Default implementation writes an empty list with error status
|
|
377
|
+
// Most subclasses should be able to use this
|
|
378
|
+
return this.writeToStore({
|
|
379
|
+
data: []
|
|
380
|
+
}, "error", batch);
|
|
381
|
+
}
|
|
382
|
+
|
|
383
|
+
/**
|
|
384
|
+
* Sort the collection items using the configured sorting strategy
|
|
385
|
+
* @param objectCacheKeys - The cache keys to sort
|
|
386
|
+
* @param batch - The batch context
|
|
387
|
+
* @returns Sorted array of cache keys
|
|
388
|
+
*/
|
|
389
|
+
_sortCacheKeys(objectCacheKeys, batch) {
|
|
390
|
+
return this.sortingStrategy.sortCacheKeys(objectCacheKeys, batch);
|
|
391
|
+
}
|
|
392
|
+
|
|
393
|
+
/**
|
|
394
|
+
* Unified method for updating collection data in the store
|
|
395
|
+
* Handles storing, sorting, deduplication, and reference counting
|
|
396
|
+
*
|
|
397
|
+
* @param items - Either object cache keys or object instances to update
|
|
398
|
+
* @param options - Configuration options for the update
|
|
399
|
+
* @param batch - The batch context to use
|
|
400
|
+
* @returns The updated entry
|
|
401
|
+
*/
|
|
402
|
+
updateCollection(items, options, batch) {
|
|
403
|
+
if (process.env.NODE_ENV !== "production") {
|
|
404
|
+
const logger = process.env.NODE_ENV !== "production" ? this.logger?.child({
|
|
405
|
+
methodName: "updateCollection"
|
|
406
|
+
}) : this.logger;
|
|
407
|
+
logger?.debug(`{status: ${options.status}, append: ${options.append}}`, JSON.stringify(items, null, 2));
|
|
408
|
+
}
|
|
409
|
+
|
|
410
|
+
// Step 1: Convert items to object cache keys if needed
|
|
411
|
+
let objectCacheKeys;
|
|
412
|
+
if (items.length === 0) {
|
|
413
|
+
objectCacheKeys = [];
|
|
414
|
+
} else if (isObjectInstance(items[0])) {
|
|
415
|
+
// Items are object instances, need to store them first
|
|
416
|
+
objectCacheKeys = this.storeObjects(items, batch);
|
|
417
|
+
} else {
|
|
418
|
+
// Items are already cache keys
|
|
419
|
+
objectCacheKeys = items;
|
|
420
|
+
}
|
|
421
|
+
|
|
422
|
+
// Step 2: Handle retain/release/append logic
|
|
423
|
+
objectCacheKeys = this.#retainReleaseAppend(batch, options.append ?? false, objectCacheKeys);
|
|
424
|
+
|
|
425
|
+
// Step 3: Sort using the configured sorting strategy
|
|
426
|
+
objectCacheKeys = this._sortCacheKeys(objectCacheKeys, batch);
|
|
427
|
+
|
|
428
|
+
// Step 4: Remove duplicates
|
|
429
|
+
objectCacheKeys = removeDuplicates(objectCacheKeys, batch);
|
|
430
|
+
|
|
431
|
+
// Step 5: Write to store
|
|
432
|
+
return this.writeToStore({
|
|
433
|
+
data: objectCacheKeys
|
|
434
|
+
}, options.status, batch);
|
|
435
|
+
}
|
|
436
|
+
}
|
|
437
|
+
|
|
438
|
+
/**
|
|
439
|
+
* Implements filtered and sorted object collection queries.
|
|
440
|
+
* - Handles where clause filtering and orderBy sorting
|
|
441
|
+
* - Manages pagination through fetchMore
|
|
442
|
+
* - Auto-updates when matching objects change
|
|
443
|
+
* - Uses canonicalized cache keys for consistency
|
|
444
|
+
*/
|
|
445
|
+
export class ListQuery extends BaseListQuery {
|
|
446
|
+
// pageSize?: number; // this is the internal page size. we need to track this properly
|
|
447
|
+
|
|
448
|
+
#type;
|
|
449
|
+
#apiName;
|
|
450
|
+
#whereClause;
|
|
451
|
+
|
|
452
|
+
// Using base class minResultsToLoad instead of a private property
|
|
453
|
+
#orderBy;
|
|
454
|
+
#objectSet;
|
|
455
|
+
|
|
456
|
+
/**
|
|
457
|
+
* Register changes to the cache specific to ListQuery
|
|
458
|
+
*/
|
|
459
|
+
registerCacheChanges(batch) {
|
|
460
|
+
batch.changes.registerList(this.cacheKey);
|
|
461
|
+
}
|
|
462
|
+
constructor(store, subject, apiType, apiName, whereClause, orderBy, cacheKey, opts) {
|
|
463
|
+
super(store, subject, opts, cacheKey, process.env.NODE_ENV !== "production" ? store.client[additionalContext].logger?.child({}, {
|
|
464
|
+
msgPrefix: `ListQuery<${cacheKey.otherKeys.map(x => JSON.stringify(x)).join(", ")}>`
|
|
465
|
+
}) : undefined);
|
|
466
|
+
this.#type = apiType;
|
|
467
|
+
this.#apiName = apiName;
|
|
468
|
+
this.#whereClause = whereClause;
|
|
469
|
+
this.#orderBy = orderBy;
|
|
470
|
+
this.#objectSet = store.client({
|
|
471
|
+
type: this.#type,
|
|
472
|
+
apiName: this.#apiName
|
|
473
|
+
}).where(this.#whereClause);
|
|
474
|
+
// Initialize the sorting strategy
|
|
475
|
+
this.sortingStrategy = new OrderBySortingStrategy(this.#apiName, this.#orderBy);
|
|
476
|
+
// Initialize the minResultsToLoad inherited from BaseCollectionQuery
|
|
477
|
+
this.minResultsToLoad = 0;
|
|
478
|
+
}
|
|
479
|
+
get canonicalWhere() {
|
|
480
|
+
return this.#whereClause;
|
|
481
|
+
}
|
|
482
|
+
|
|
483
|
+
/**
|
|
484
|
+
* Implements fetchPageData from BaseCollectionQuery template method
|
|
485
|
+
* Fetches a page of data
|
|
486
|
+
*/
|
|
487
|
+
async fetchPageData(signal) {
|
|
488
|
+
// Fetch the data with pagination
|
|
489
|
+
const resp = await this.#objectSet.fetchPage({
|
|
490
|
+
$nextPageToken: this.nextPageToken,
|
|
491
|
+
$pageSize: this.options.pageSize,
|
|
492
|
+
// For now this keeps the shared test code from falling apart
|
|
493
|
+
// but shouldn't be needed ideally
|
|
494
|
+
...(Object.keys(this.#orderBy).length > 0 ? {
|
|
495
|
+
$orderBy: this.#orderBy
|
|
496
|
+
} : {})
|
|
497
|
+
});
|
|
498
|
+
if (signal?.aborted) {
|
|
499
|
+
throw new Error("Aborted");
|
|
500
|
+
}
|
|
501
|
+
this.nextPageToken = resp.nextPageToken;
|
|
502
|
+
let fetchedData = resp.data;
|
|
503
|
+
|
|
504
|
+
// Our caching really expects to have the full objects in the list
|
|
505
|
+
// so we need to fetch them all here
|
|
506
|
+
if (this.#type === "interface") {
|
|
507
|
+
fetchedData = await reloadDataAsFullObjects(this.store.client, fetchedData);
|
|
508
|
+
}
|
|
509
|
+
return {
|
|
510
|
+
...resp,
|
|
511
|
+
data: fetchedData
|
|
512
|
+
};
|
|
513
|
+
}
|
|
514
|
+
|
|
515
|
+
/**
|
|
516
|
+
* Handle fetch errors by setting appropriate error state and notifying subscribers
|
|
517
|
+
*/
|
|
518
|
+
handleFetchError(error, _status, batch) {
|
|
519
|
+
this.logger?.error("error", error);
|
|
520
|
+
this.store.getSubject(this.cacheKey).error(error);
|
|
521
|
+
|
|
522
|
+
// We don't call super.handleFetchError because ListQuery has special error handling
|
|
523
|
+
// but we still use writeToStore to create a properly structured Entry
|
|
524
|
+
return this.writeToStore({
|
|
525
|
+
data: []
|
|
526
|
+
}, "error", batch);
|
|
527
|
+
}
|
|
528
|
+
|
|
256
529
|
/**
|
|
257
530
|
* Will revalidate the list if its query is affected by invalidating the
|
|
258
531
|
* apiName of the object type passed in.
|
|
@@ -278,6 +551,13 @@ export class ListQuery extends BaseListQuery {
|
|
|
278
551
|
await this.revalidate(/* force */true);
|
|
279
552
|
}
|
|
280
553
|
};
|
|
554
|
+
invalidateObjectType = async (objectType, changes) => {
|
|
555
|
+
if (this.cacheKey.otherKeys[1] === objectType) {
|
|
556
|
+
// Only invalidate lists for the matching apiName
|
|
557
|
+
changes?.modified.add(this.cacheKey);
|
|
558
|
+
return this.revalidate(true);
|
|
559
|
+
}
|
|
560
|
+
};
|
|
281
561
|
|
|
282
562
|
/**
|
|
283
563
|
* Note: This method is not async because I want it to return right after it
|
|
@@ -294,6 +574,9 @@ export class ListQuery extends BaseListQuery {
|
|
|
294
574
|
this.logger?.child({
|
|
295
575
|
methodName: "maybeUpdateAndRevalidate"
|
|
296
576
|
}).debug(DEBUG_ONLY__changesToString(changes));
|
|
577
|
+
this.logger?.child({
|
|
578
|
+
methodName: "maybeUpdateAndRevalidate"
|
|
579
|
+
}).debug(`Already in changes? ${changes.modified.has(this.cacheKey)}`);
|
|
297
580
|
}
|
|
298
581
|
if (changes.modified.has(this.cacheKey)) return;
|
|
299
582
|
// mark ourselves as updated so we don't infinite recurse.
|
|
@@ -354,7 +637,7 @@ export class ListQuery extends BaseListQuery {
|
|
|
354
637
|
for (const obj of toAdd) {
|
|
355
638
|
newList.push(this.store.getCacheKey("object", obj.$objectType, obj.$primaryKey));
|
|
356
639
|
}
|
|
357
|
-
this._updateList(newList, /* append */false
|
|
640
|
+
this._updateList(newList, status, batch, /* append */false);
|
|
358
641
|
});
|
|
359
642
|
if (needsRevalidation) {
|
|
360
643
|
return this.revalidate(true);
|
|
@@ -427,20 +710,6 @@ export class ListQuery extends BaseListQuery {
|
|
|
427
710
|
}
|
|
428
711
|
};
|
|
429
712
|
}
|
|
430
|
-
_sortCacheKeys(objectCacheKeys, batch) {
|
|
431
|
-
if (Object.keys(this.#orderBy).length > 0) {
|
|
432
|
-
objectCacheKeys = objectCacheKeys.sort((a, b) => {
|
|
433
|
-
for (const sortFn of this.#sortFns) {
|
|
434
|
-
const ret = sortFn(batch.read(a)?.value?.$as(this.#apiName), batch.read(b)?.value?.$as(this.#apiName));
|
|
435
|
-
if (ret !== 0) {
|
|
436
|
-
return ret;
|
|
437
|
-
}
|
|
438
|
-
}
|
|
439
|
-
return 0;
|
|
440
|
-
});
|
|
441
|
-
}
|
|
442
|
-
return objectCacheKeys;
|
|
443
|
-
}
|
|
444
713
|
registerStreamUpdates(sub) {
|
|
445
714
|
const logger = process.env.NODE_ENV !== "production" ? this.logger?.child({
|
|
446
715
|
methodName: "registerStreamUpdates"
|
|
@@ -505,7 +774,7 @@ export class ListQuery extends BaseListQuery {
|
|
|
505
774
|
if (state === "ADDED_OR_UPDATED") {
|
|
506
775
|
const object = objOrIface.$apiName !== objOrIface.$objectType ? objOrIface.$as(objOrIface.$objectType) : objOrIface;
|
|
507
776
|
this.store.batch({}, batch => {
|
|
508
|
-
|
|
777
|
+
this.storeObjects([object], batch);
|
|
509
778
|
});
|
|
510
779
|
} else if (state === "REMOVED") {
|
|
511
780
|
this.#onOswRemoved(objOrIface, logger);
|
|
@@ -554,37 +823,6 @@ export class ListQuery extends BaseListQuery {
|
|
|
554
823
|
});
|
|
555
824
|
}
|
|
556
825
|
}
|
|
557
|
-
function removeDuplicates(objectCacheKeys, batch) {
|
|
558
|
-
const visited = new Set();
|
|
559
|
-
objectCacheKeys = objectCacheKeys.filter(key => {
|
|
560
|
-
batch.read(key);
|
|
561
|
-
if (visited.has(key)) {
|
|
562
|
-
return false;
|
|
563
|
-
}
|
|
564
|
-
visited.add(key);
|
|
565
|
-
return true;
|
|
566
|
-
});
|
|
567
|
-
return objectCacheKeys;
|
|
568
|
-
}
|
|
569
|
-
function createOrderBySortFns(orderBy) {
|
|
570
|
-
return Object.entries(orderBy).map(([key, order]) => {
|
|
571
|
-
return (a, b) => {
|
|
572
|
-
const aValue = a?.[key];
|
|
573
|
-
const bValue = b?.[key];
|
|
574
|
-
if (aValue == null && bValue == null) {
|
|
575
|
-
return 0;
|
|
576
|
-
}
|
|
577
|
-
if (aValue == null) {
|
|
578
|
-
return 1;
|
|
579
|
-
}
|
|
580
|
-
if (bValue == null) {
|
|
581
|
-
return -1;
|
|
582
|
-
}
|
|
583
|
-
const m = order === "asc" ? -1 : 1;
|
|
584
|
-
return aValue < bValue ? m : aValue > bValue ? -m : 0;
|
|
585
|
-
};
|
|
586
|
-
});
|
|
587
|
-
}
|
|
588
826
|
|
|
589
827
|
// Hopefully this can go away when we can just request the full object properties on first load
|
|
590
828
|
async function reloadDataAsFullObjects(client, data) {
|