@mmstack/resource 20.0.0 → 20.0.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.
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import { computed, untracked, InjectionToken, inject, isDevMode, signal, effect, linkedSignal, DestroyRef } from '@angular/core';
|
|
2
2
|
import { takeUntilDestroyed, toObservable } from '@angular/core/rxjs-interop';
|
|
3
|
-
import { of, tap, map, finalize, shareReplay, interval, firstValueFrom, combineLatestWith, filter } from 'rxjs';
|
|
3
|
+
import { of, tap, map, finalize, shareReplay, interval, firstValueFrom, catchError, combineLatestWith, filter } from 'rxjs';
|
|
4
4
|
import { HttpContextToken, HttpContext, HttpResponse, HttpParams, httpResource, HttpClient } from '@angular/common/http';
|
|
5
5
|
import { mutable, toWritable } from '@mmstack/primitives';
|
|
6
6
|
import { v7 } from 'uuid';
|
|
@@ -415,6 +415,20 @@ function createCacheInterceptor(allowedMethods = ['GET', 'HEAD', 'OPTIONS']) {
|
|
|
415
415
|
};
|
|
416
416
|
}
|
|
417
417
|
|
|
418
|
+
function catchValueError(resource, fallback) {
|
|
419
|
+
return {
|
|
420
|
+
...resource,
|
|
421
|
+
value: toWritable(computed(() => {
|
|
422
|
+
try {
|
|
423
|
+
return resource.value();
|
|
424
|
+
}
|
|
425
|
+
catch {
|
|
426
|
+
return fallback;
|
|
427
|
+
}
|
|
428
|
+
}), (value) => resource.value.set(value)),
|
|
429
|
+
};
|
|
430
|
+
}
|
|
431
|
+
|
|
418
432
|
/** @internal */
|
|
419
433
|
function internalCeateCircuitBreaker(treshold = 5, resetTimeout = 30000) {
|
|
420
434
|
const halfOpen = signal(false);
|
|
@@ -805,7 +819,9 @@ function queryResource(request, options) {
|
|
|
805
819
|
return undefined;
|
|
806
820
|
return request() ?? undefined;
|
|
807
821
|
}, {
|
|
808
|
-
equal:
|
|
822
|
+
equal: options?.triggerOnSameRequest
|
|
823
|
+
? undefined
|
|
824
|
+
: createEqualRequest(options?.equal),
|
|
809
825
|
});
|
|
810
826
|
const hashFn = typeof options?.cache === 'object'
|
|
811
827
|
? (options.cache.hash ?? urlWithParams)
|
|
@@ -838,6 +854,7 @@ function queryResource(request, options) {
|
|
|
838
854
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
839
855
|
parse: options?.parse, // Not my favorite thing to do, but here it is completely safe.
|
|
840
856
|
});
|
|
857
|
+
resource = catchValueError(resource, options?.defaultValue);
|
|
841
858
|
// get full HttpResonse from Cache
|
|
842
859
|
const cachedEvent = cache.get(cacheKey);
|
|
843
860
|
const parse = options?.parse ?? ((val) => val);
|
|
@@ -1004,7 +1021,7 @@ function mutationResource(request, options = {}) {
|
|
|
1004
1021
|
? options.injector.get(DestroyRef)
|
|
1005
1022
|
: inject(DestroyRef);
|
|
1006
1023
|
const error$ = toObservable(resource.error);
|
|
1007
|
-
const value$ = toObservable(resource.value);
|
|
1024
|
+
const value$ = toObservable(resource.value).pipe(catchError(() => of(null)));
|
|
1008
1025
|
const statusSub = toObservable(resource.status)
|
|
1009
1026
|
.pipe(combineLatestWith(error$, value$), map(([status, error, value]) => {
|
|
1010
1027
|
if (status === 'error' && error) {
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"mmstack-resource.mjs","sources":["../tmp-esm2022/lib/util/cache/cache.js","../tmp-esm2022/lib/util/cache/cache.interceptor.js","../tmp-esm2022/lib/util/circuit-breaker.js","../tmp-esm2022/lib/util/dedupe.interceptor.js","../tmp-esm2022/lib/util/equality.js","../tmp-esm2022/lib/util/has-slow-connection.js","../tmp-esm2022/lib/util/persist.js","../tmp-esm2022/lib/util/refresh.js","../tmp-esm2022/lib/util/retry-on-error.js","../tmp-esm2022/lib/util/url-with-params.js","../tmp-esm2022/lib/query-resource.js","../tmp-esm2022/lib/mutation-resource.js","../tmp-esm2022/mmstack-resource.js"],"sourcesContent":["import { computed, inject, InjectionToken, isDevMode, untracked, } from '@angular/core';\nimport { mutable } from '@mmstack/primitives';\nimport { v7 } from 'uuid';\nconst ONE_DAY = 1000 * 60 * 60 * 24;\nconst ONE_HOUR = 1000 * 60 * 60;\nconst DEFAULT_CLEANUP_OPT = {\n type: 'lru',\n maxSize: 200,\n checkInterval: ONE_HOUR,\n};\n/**\n * A generic cache implementation that stores data with time-to-live (TTL) and stale-while-revalidate capabilities.\n *\n * @typeParam T - The type of data to be stored in the cache.\n */\nexport class Cache {\n ttl;\n staleTime;\n internal = mutable(new Map());\n cleanupOpt;\n /**\n * Creates a new `Cache` instance.\n *\n * @param ttl - The default Time To Live (TTL) for cache entries, in milliseconds. Defaults to one day.\n * @param staleTime - The default duration, in milliseconds, during which a cache entry is considered\n * stale but can still be used while revalidation occurs in the background. Defaults to 1 hour.\n * @param cleanupOpt - Options for configuring the cache cleanup strategy. Defaults to LRU with a\n * `maxSize` of 200 and a `checkInterval` of one hour.\n */\n constructor(ttl = ONE_DAY, staleTime = ONE_HOUR, cleanupOpt = {\n type: 'lru',\n maxSize: 1000,\n checkInterval: ONE_HOUR,\n }) {\n this.ttl = ttl;\n this.staleTime = staleTime;\n this.cleanupOpt = {\n ...DEFAULT_CLEANUP_OPT,\n ...cleanupOpt,\n };\n if (this.cleanupOpt.maxSize <= 0)\n throw new Error('maxSize must be greater than 0');\n // cleanup cache based on provided options regularly\n const cleanupInterval = setInterval(() => {\n this.cleanup();\n }, cleanupOpt.checkInterval);\n const destroyId = v7();\n // cleanup if object is garbage collected, this is because the cache can be quite large from a memory standpoint & we dont want all that floating garbage\n const registry = new FinalizationRegistry((id) => {\n if (id === destroyId) {\n clearInterval(cleanupInterval);\n }\n });\n registry.register(this, destroyId);\n }\n /** @internal */\n getInternal(key) {\n const keySignal = computed(() => key());\n return computed(() => {\n const key = keySignal();\n if (!key)\n return null;\n const found = this.internal().get(key);\n const now = Date.now();\n if (!found || found.expiresAt <= now)\n return null;\n found.useCount++;\n return {\n ...found,\n isStale: found.stale <= now,\n };\n });\n }\n /**\n * Retrieves a cache entry without affecting its usage count (for LRU). This is primarily\n * for internal use or debugging.\n * @internal\n * @param key - The key of the entry to retrieve.\n * @returns The cache entry, or `null` if not found or expired.\n */\n getUntracked(key) {\n return untracked(this.getInternal(() => key));\n }\n /**\n * Retrieves a cache entry as a signal.\n *\n * @param key - A function that returns the cache key. The key is a signal, allowing for dynamic keys. If the function returns null the value is also null.\n * @returns A signal that holds the cache entry, or `null` if not found or expired. The signal\n * updates whenever the cache entry changes (e.g., due to revalidation or expiration).\n */\n get(key) {\n return this.getInternal(key);\n }\n /**\n * Stores a value in the cache.\n *\n * @param key - The key under which to store the value.\n * @param value - The value to store.\n * @param staleTime - (Optional) The stale time for this entry, in milliseconds. Overrides the default `staleTime`.\n * @param ttl - (Optional) The TTL for this entry, in milliseconds. Overrides the default `ttl`.\n */\n store(key, value, staleTime = this.staleTime, ttl = this.ttl) {\n const entry = this.getUntracked(key);\n if (entry) {\n clearTimeout(entry.timeout); // stop invalidation\n }\n const prevCount = entry?.useCount ?? 0;\n // ttl cannot be less than staleTime\n if (ttl < staleTime)\n staleTime = ttl;\n const now = Date.now();\n this.internal.mutate((map) => {\n map.set(key, {\n value,\n created: entry?.created ?? now,\n useCount: prevCount + 1,\n stale: now + staleTime,\n expiresAt: now + ttl,\n timeout: setTimeout(() => this.invalidate(key), ttl),\n });\n return map;\n });\n }\n /**\n * Invalidates (removes) a cache entry.\n *\n * @param key - The key of the entry to invalidate.\n */\n invalidate(key) {\n const entry = this.getUntracked(key);\n if (!entry)\n return;\n clearTimeout(entry.timeout);\n this.internal.mutate((map) => {\n map.delete(key);\n return map;\n });\n }\n /** @internal */\n cleanup() {\n if (untracked(this.internal).size <= this.cleanupOpt.maxSize)\n return;\n const sorted = Array.from(untracked(this.internal).entries()).toSorted((a, b) => {\n if (this.cleanupOpt.type === 'lru') {\n return a[1].useCount - b[1].useCount; // least used first\n }\n else {\n return a[1].created - b[1].created; // oldest first\n }\n });\n const keepCount = Math.floor(this.cleanupOpt.maxSize / 2);\n const removed = sorted.slice(0, sorted.length - keepCount);\n const keep = sorted.slice(removed.length, sorted.length);\n removed.forEach(([, e]) => {\n clearTimeout(e.timeout);\n });\n this.internal.set(new Map(keep));\n }\n}\nconst CLIENT_CACHE_TOKEN = new InjectionToken('INTERNAL_CLIENT_CACHE');\n/**\n * Provides the instance of the QueryCache for queryResource. This should probably be called\n * in your application's root configuration, but can also be overriden with component/module providers.\n *\n * @param options - Optional configuration options for the cache.\n * @returns An Angular `Provider` for the cache.\n *\n * @example\n * // In your app.config.ts or AppModule providers:\n *\n * import { provideQueryCache } from './your-cache';\n *\n * export const appConfig: ApplicationConfig = {\n * providers: [\n * provideQueryCache({\n * ttl: 60000, // Default TTL of 60 seconds\n * staleTime: 30000, // Default staleTime of 30 seconds\n * }),\n * // ... other providers\n * ]\n * };\n */\nexport function provideQueryCache(opt) {\n return {\n provide: CLIENT_CACHE_TOKEN,\n useValue: new Cache(opt?.ttl, opt?.staleTime, opt?.cleanup),\n };\n}\nclass NoopCache extends Cache {\n store(_, __, ___ = super.staleTime, ____ = super.ttl) {\n // noop\n }\n}\n/**\n * Injects the `QueryCache` instance that is used within queryResource.\n * Allows for direct modification of cached data, but is mostly meant for internal use.\n *\n * @param injector - (Optional) The injector to use. If not provided, the current\n * injection context is used.\n * @returns The `QueryCache` instance.\n *\n * @example\n * // In your component or service:\n *\n * import { injectQueryCache } from './your-cache';\n *\n * constructor() {\n * const cache = injectQueryCache();\n *\n * const myData = cache.get(() => 'my-data-key');\n * if (myData() !== null) {\n * // ... use cached data ...\n * }\n * }\n */\nexport function injectQueryCache(injector) {\n const cache = injector\n ? injector.get(CLIENT_CACHE_TOKEN, null, {\n optional: true,\n })\n : inject(CLIENT_CACHE_TOKEN, {\n optional: true,\n });\n if (!cache) {\n if (isDevMode())\n throw new Error('Cache not provided, please add provideQueryCache() to providers array');\n else\n return new NoopCache();\n }\n return cache;\n}\n//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cache.js","sourceRoot":"","sources":["../../../../../../../packages/resource/src/lib/util/cache/cache.ts"],"names":[],"mappings":"AACA,OAAO,EACL,QAAQ,EACR,MAAM,EACN,cAAc,EAEd,SAAS,EAGT,SAAS,GACV,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,OAAO,EAAE,MAAM,qBAAqB,CAAC;AAC9C,OAAO,EAAE,EAAE,EAAE,MAAM,MAAM,CAAC;AAwD1B,MAAM,OAAO,GAAG,IAAI,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,CAAC;AACpC,MAAM,QAAQ,GAAG,IAAI,GAAG,EAAE,GAAG,EAAE,CAAC;AAEhC,MAAM,mBAAmB,GAAG;IAC1B,IAAI,EAAE,KAAK;IACX,OAAO,EAAE,GAAG;IACZ,aAAa,EAAE,QAAQ;CACC,CAAC;AAE3B;;;;GAIG;AACH,MAAM,OAAO,KAAK;IAcK;IACA;IAdJ,QAAQ,GAAG,OAAO,CAAC,IAAI,GAAG,EAAyB,CAAC,CAAC;IACrD,UAAU,CAAc;IAEzC;;;;;;;;OAQG;IACH,YACqB,MAAc,OAAO,EACrB,YAAoB,QAAQ,EAC/C,aAAmC;QACjC,IAAI,EAAE,KAAK;QACX,OAAO,EAAE,IAAI;QACb,aAAa,EAAE,QAAQ;KACxB;QANkB,QAAG,GAAH,GAAG,CAAkB;QACrB,cAAS,GAAT,SAAS,CAAmB;QAO/C,IAAI,CAAC,UAAU,GAAG;YAChB,GAAG,mBAAmB;YACtB,GAAG,UAAU;SACd,CAAC;QACF,IAAI,IAAI,CAAC,UAAU,CAAC,OAAO,IAAI,CAAC;YAC9B,MAAM,IAAI,KAAK,CAAC,gCAAgC,CAAC,CAAC;QAEpD,oDAAoD;QACpD,MAAM,eAAe,GAAG,WAAW,CAAC,GAAG,EAAE;YACvC,IAAI,CAAC,OAAO,EAAE,CAAC;QACjB,CAAC,EAAE,UAAU,CAAC,aAAa,CAAC,CAAC;QAE7B,MAAM,SAAS,GAAG,EAAE,EAAE,CAAC;QAEvB,yJAAyJ;QACzJ,MAAM,QAAQ,GAAG,IAAI,oBAAoB,CAAC,CAAC,EAAU,EAAE,EAAE;YACvD,IAAI,EAAE,KAAK,SAAS,EAAE,CAAC;gBACrB,aAAa,CAAC,eAAe,CAAC,CAAC;YACjC,CAAC;QACH,CAAC,CAAC,CAAC;QAEH,QAAQ,CAAC,QAAQ,CAAC,IAAI,EAAE,SAAS,CAAC,CAAC;IACrC,CAAC;IAED,gBAAgB;IACR,WAAW,CACjB,GAAwB;QAExB,MAAM,SAAS,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,GAAG,EAAE,CAAC,CAAC;QAExC,OAAO,QAAQ,CAAC,GAAG,EAAE;YACnB,MAAM,GAAG,GAAG,SAAS,EAAE,CAAC;YACxB,IAAI,CAAC,GAAG;gBAAE,OAAO,IAAI,CAAC;YACtB,MAAM,KAAK,GAAG,IAAI,CAAC,QAAQ,EAAE,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;YAEvC,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;YAEvB,IAAI,CAAC,KAAK,IAAI,KAAK,CAAC,SAAS,IAAI,GAAG;gBAAE,OAAO,IAAI,CAAC;YAClD,KAAK,CAAC,QAAQ,EAAE,CAAC;YACjB,OAAO;gBACL,GAAG,KAAK;gBACR,OAAO,EAAE,KAAK,CAAC,KAAK,IAAI,GAAG;aAC5B,CAAC;QACJ,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;;;;OAMG;IACH,YAAY,CAAC,GAAW;QACtB,OAAO,SAAS,CAAC,IAAI,CAAC,WAAW,CAAC,GAAG,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC;IAChD,CAAC;IAED;;;;;;OAMG;IACH,GAAG,CACD,GAAwB;QAExB,OAAO,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,CAAC;IAC/B,CAAC;IAED;;;;;;;OAOG;IACH,KAAK,CAAC,GAAW,EAAE,KAAQ,EAAE,SAAS,GAAG,IAAI,CAAC,SAAS,EAAE,GAAG,GAAG,IAAI,CAAC,GAAG;QACrE,MAAM,KAAK,GAAG,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC;QACrC,IAAI,KAAK,EAAE,CAAC;YACV,YAAY,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,oBAAoB;QACnD,CAAC;QAED,MAAM,SAAS,GAAG,KAAK,EAAE,QAAQ,IAAI,CAAC,CAAC;QAEvC,oCAAoC;QACpC,IAAI,GAAG,GAAG,SAAS;YAAE,SAAS,GAAG,GAAG,CAAC;QAErC,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QAEvB,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,EAAE;YAC3B,GAAG,CAAC,GAAG,CAAC,GAAG,EAAE;gBACX,KAAK;gBACL,OAAO,EAAE,KAAK,EAAE,OAAO,IAAI,GAAG;gBAC9B,QAAQ,EAAE,SAAS,GAAG,CAAC;gBACvB,KAAK,EAAE,GAAG,GAAG,SAAS;gBACtB,SAAS,EAAE,GAAG,GAAG,GAAG;gBACpB,OAAO,EAAE,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,EAAE,GAAG,CAAC;aACrD,CAAC,CAAC;YACH,OAAO,GAAG,CAAC;QACb,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;;OAIG;IACH,UAAU,CAAC,GAAW;QACpB,MAAM,KAAK,GAAG,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC;QACrC,IAAI,CAAC,KAAK;YAAE,OAAO;QACnB,YAAY,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;QAC5B,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,EAAE;YAC3B,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;YAChB,OAAO,GAAG,CAAC;QACb,CAAC,CAAC,CAAC;IACL,CAAC;IAED,gBAAgB;IACR,OAAO;QACb,IAAI,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,IAAI,IAAI,IAAI,CAAC,UAAU,CAAC,OAAO;YAAE,OAAO;QAErE,MAAM,MAAM,GAAG,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,OAAO,EAAE,CAAC,CAAC,QAAQ,CACpE,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE;YACP,IAAI,IAAI,CAAC,UAAU,CAAC,IAAI,KAAK,KAAK,EAAE,CAAC;gBACnC,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC,QAAQ,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,mBAAmB;YAC3D,CAAC;iBAAM,CAAC;gBACN,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,eAAe;YACrD,CAAC;QACH,CAAC,CACF,CAAC;QAEF,MAAM,SAAS,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,UAAU,CAAC,OAAO,GAAG,CAAC,CAAC,CAAC;QAE1D,MAAM,OAAO,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC,EAAE,MAAM,CAAC,MAAM,GAAG,SAAS,CAAC,CAAC;QAC3D,MAAM,IAAI,GAAG,MAAM,CAAC,KAAK,CAAC,OAAO,CAAC,MAAM,EAAE,MAAM,CAAC,MAAM,CAAC,CAAC;QAEzD,OAAO,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE,EAAE;YACxB,YAAY,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC;QAC1B,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,IAAI,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC;IACnC,CAAC;CACF;AAqBD,MAAM,kBAAkB,GAAG,IAAI,cAAc,CAC3C,uBAAuB,CACxB,CAAC;AAEF;;;;;;;;;;;;;;;;;;;;;GAqBG;AACH,MAAM,UAAU,iBAAiB,CAAC,GAAkB;IAClD,OAAO;QACL,OAAO,EAAE,kBAAkB;QAC3B,QAAQ,EAAE,IAAI,KAAK,CAAC,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,SAAS,EAAE,GAAG,EAAE,OAAO,CAAC;KAC5D,CAAC;AACJ,CAAC;AAED,MAAM,SAAa,SAAQ,KAAQ;IACxB,KAAK,CAAC,CAAS,EAAE,EAAK,EAAE,GAAG,GAAG,KAAK,CAAC,SAAS,EAAE,IAAI,GAAG,KAAK,CAAC,GAAG;QACtE,OAAO;IACT,CAAC;CACF;AAED;;;;;;;;;;;;;;;;;;;;;GAqBG;AACH,MAAM,UAAU,gBAAgB,CAC9B,QAAmB;IAEnB,MAAM,KAAK,GAAG,QAAQ;QACpB,CAAC,CAAC,QAAQ,CAAC,GAAG,CAAC,kBAAkB,EAAE,IAAI,EAAE;YACrC,QAAQ,EAAE,IAAI;SACf,CAAC;QACJ,CAAC,CAAC,MAAM,CAAC,kBAAkB,EAAE;YACzB,QAAQ,EAAE,IAAI;SACf,CAAC,CAAC;IAEP,IAAI,CAAC,KAAK,EAAE,CAAC;QACX,IAAI,SAAS,EAAE;YACb,MAAM,IAAI,KAAK,CACb,uEAAuE,CACxE,CAAC;;YACC,OAAO,IAAI,SAAS,EAAE,CAAC;IAC9B,CAAC;IAED,OAAO,KAAK,CAAC;AACf,CAAC","sourcesContent":["import type { HttpResponse } from '@angular/common/http';\r\nimport {\r\n  computed,\r\n  inject,\r\n  InjectionToken,\r\n  Injector,\r\n  isDevMode,\r\n  type Provider,\r\n  type Signal,\r\n  untracked,\r\n} from '@angular/core';\r\nimport { mutable } from '@mmstack/primitives';\r\nimport { v7 } from 'uuid';\r\n\r\n/**\r\n * Options for configuring the Least Recently Used (LRU) cache cleanup strategy.\r\n * @internal\r\n */\r\ntype LRUCleanupType = {\r\n  type: 'lru';\r\n  /**\r\n   * How often to check for expired or excess entries, in milliseconds.\r\n   */\r\n  checkInterval: number;\r\n  /**\r\n   * The maximum number of entries to keep in the cache.  When the cache exceeds this size,\r\n   * the least recently used entries will be removed.\r\n   */\r\n  maxSize: number;\r\n};\r\n\r\n/**\r\n * Options for configuring the \"oldest first\" cache cleanup strategy.\r\n * @internal\r\n */\r\ntype OldsetCleanupType = {\r\n  type: 'oldest';\r\n  /**\r\n   * How often to check for expired or excess entries, in milliseconds.\r\n   */\r\n  checkInterval: number;\r\n  /**\r\n   * The maximum number of entries to keep in the cache.  When the cache exceeds this size,\r\n   * the oldest entries will be removed.\r\n   */\r\n  maxSize: number;\r\n};\r\n\r\n/**\r\n * Represents an entry in the cache.\r\n * @internal\r\n */\r\ntype CacheEntry<T> = {\r\n  value: T;\r\n  created: number;\r\n  stale: number;\r\n  useCount: number;\r\n  expiresAt: number;\r\n  timeout: ReturnType<typeof setTimeout>;\r\n};\r\n\r\n/**\r\n * Defines the types of cleanup strategies available for the cache.\r\n * - `lru`: Least Recently Used.  Removes the least recently accessed entries when the cache is full.\r\n * - `oldest`: Removes the oldest entries when the cache is full.\r\n */\r\nexport type CleanupType = LRUCleanupType | OldsetCleanupType;\r\n\r\nconst ONE_DAY = 1000 * 60 * 60 * 24;\r\nconst ONE_HOUR = 1000 * 60 * 60;\r\n\r\nconst DEFAULT_CLEANUP_OPT = {\r\n  type: 'lru',\r\n  maxSize: 200,\r\n  checkInterval: ONE_HOUR,\r\n} satisfies LRUCleanupType;\r\n\r\n/**\r\n * A generic cache implementation that stores data with time-to-live (TTL) and stale-while-revalidate capabilities.\r\n *\r\n * @typeParam T - The type of data to be stored in the cache.\r\n */\r\nexport class Cache<T> {\r\n  private readonly internal = mutable(new Map<string, CacheEntry<T>>());\r\n  private readonly cleanupOpt: CleanupType;\r\n\r\n  /**\r\n   * Creates a new `Cache` instance.\r\n   *\r\n   * @param ttl - The default Time To Live (TTL) for cache entries, in milliseconds.  Defaults to one day.\r\n   * @param staleTime - The default duration, in milliseconds, during which a cache entry is considered\r\n   *                    stale but can still be used while revalidation occurs in the background. Defaults to 1 hour.\r\n   * @param cleanupOpt - Options for configuring the cache cleanup strategy.  Defaults to LRU with a\r\n   *                     `maxSize` of 200 and a `checkInterval` of one hour.\r\n   */\r\n  constructor(\r\n    protected readonly ttl: number = ONE_DAY,\r\n    protected readonly staleTime: number = ONE_HOUR,\r\n    cleanupOpt: Partial<CleanupType> = {\r\n      type: 'lru',\r\n      maxSize: 1000,\r\n      checkInterval: ONE_HOUR,\r\n    },\r\n  ) {\r\n    this.cleanupOpt = {\r\n      ...DEFAULT_CLEANUP_OPT,\r\n      ...cleanupOpt,\r\n    };\r\n    if (this.cleanupOpt.maxSize <= 0)\r\n      throw new Error('maxSize must be greater than 0');\r\n\r\n    // cleanup cache based on provided options regularly\r\n    const cleanupInterval = setInterval(() => {\r\n      this.cleanup();\r\n    }, cleanupOpt.checkInterval);\r\n\r\n    const destroyId = v7();\r\n\r\n    // cleanup if object is garbage collected, this is because the cache can be quite large from a memory standpoint & we dont want all that floating garbage\r\n    const registry = new FinalizationRegistry((id: string) => {\r\n      if (id === destroyId) {\r\n        clearInterval(cleanupInterval);\r\n      }\r\n    });\r\n\r\n    registry.register(this, destroyId);\r\n  }\r\n\r\n  /** @internal */\r\n  private getInternal(\r\n    key: () => string | null,\r\n  ): Signal<(CacheEntry<T> & { isStale: boolean }) | null> {\r\n    const keySignal = computed(() => key());\r\n\r\n    return computed(() => {\r\n      const key = keySignal();\r\n      if (!key) return null;\r\n      const found = this.internal().get(key);\r\n\r\n      const now = Date.now();\r\n\r\n      if (!found || found.expiresAt <= now) return null;\r\n      found.useCount++;\r\n      return {\r\n        ...found,\r\n        isStale: found.stale <= now,\r\n      };\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Retrieves a cache entry without affecting its usage count (for LRU).  This is primarily\r\n   * for internal use or debugging.\r\n   * @internal\r\n   * @param key - The key of the entry to retrieve.\r\n   * @returns The cache entry, or `null` if not found or expired.\r\n   */\r\n  getUntracked(key: string): (CacheEntry<T> & { isStale: boolean }) | null {\r\n    return untracked(this.getInternal(() => key));\r\n  }\r\n\r\n  /**\r\n   * Retrieves a cache entry as a signal.\r\n   *\r\n   * @param key - A function that returns the cache key. The key is a signal, allowing for dynamic keys. If the function returns null the value is also null.\r\n   * @returns A signal that holds the cache entry, or `null` if not found or expired.  The signal\r\n   *          updates whenever the cache entry changes (e.g., due to revalidation or expiration).\r\n   */\r\n  get(\r\n    key: () => string | null,\r\n  ): Signal<(CacheEntry<T> & { isStale: boolean }) | null> {\r\n    return this.getInternal(key);\r\n  }\r\n\r\n  /**\r\n   * Stores a value in the cache.\r\n   *\r\n   * @param key - The key under which to store the value.\r\n   * @param value - The value to store.\r\n   * @param staleTime - (Optional) The stale time for this entry, in milliseconds. Overrides the default `staleTime`.\r\n   * @param ttl - (Optional) The TTL for this entry, in milliseconds. Overrides the default `ttl`.\r\n   */\r\n  store(key: string, value: T, staleTime = this.staleTime, ttl = this.ttl) {\r\n    const entry = this.getUntracked(key);\r\n    if (entry) {\r\n      clearTimeout(entry.timeout); // stop invalidation\r\n    }\r\n\r\n    const prevCount = entry?.useCount ?? 0;\r\n\r\n    // ttl cannot be less than staleTime\r\n    if (ttl < staleTime) staleTime = ttl;\r\n\r\n    const now = Date.now();\r\n\r\n    this.internal.mutate((map) => {\r\n      map.set(key, {\r\n        value,\r\n        created: entry?.created ?? now,\r\n        useCount: prevCount + 1,\r\n        stale: now + staleTime,\r\n        expiresAt: now + ttl,\r\n        timeout: setTimeout(() => this.invalidate(key), ttl),\r\n      });\r\n      return map;\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Invalidates (removes) a cache entry.\r\n   *\r\n   * @param key - The key of the entry to invalidate.\r\n   */\r\n  invalidate(key: string) {\r\n    const entry = this.getUntracked(key);\r\n    if (!entry) return;\r\n    clearTimeout(entry.timeout);\r\n    this.internal.mutate((map) => {\r\n      map.delete(key);\r\n      return map;\r\n    });\r\n  }\r\n\r\n  /** @internal */\r\n  private cleanup() {\r\n    if (untracked(this.internal).size <= this.cleanupOpt.maxSize) return;\r\n\r\n    const sorted = Array.from(untracked(this.internal).entries()).toSorted(\r\n      (a, b) => {\r\n        if (this.cleanupOpt.type === 'lru') {\r\n          return a[1].useCount - b[1].useCount; // least used first\r\n        } else {\r\n          return a[1].created - b[1].created; // oldest first\r\n        }\r\n      },\r\n    );\r\n\r\n    const keepCount = Math.floor(this.cleanupOpt.maxSize / 2);\r\n\r\n    const removed = sorted.slice(0, sorted.length - keepCount);\r\n    const keep = sorted.slice(removed.length, sorted.length);\r\n\r\n    removed.forEach(([, e]) => {\r\n      clearTimeout(e.timeout);\r\n    });\r\n\r\n    this.internal.set(new Map(keep));\r\n  }\r\n}\r\n\r\n/**\r\n * Options for configuring the cache.\r\n */\r\ntype CacheOptions = {\r\n  /**\r\n   * The default Time To Live (TTL) for cache entries, in milliseconds.\r\n   */\r\n  ttl?: number;\r\n  /**\r\n   * The default duration, in milliseconds, during which a cache entry is considered\r\n   * stale but can still be used while revalidation occurs in the background.\r\n   */\r\n  staleTime?: number;\r\n  /**\r\n   * Options for configuring the cache cleanup strategy.\r\n   */\r\n  cleanup?: Partial<CleanupType>;\r\n};\r\n\r\nconst CLIENT_CACHE_TOKEN = new InjectionToken<Cache<HttpResponse<unknown>>>(\r\n  'INTERNAL_CLIENT_CACHE',\r\n);\r\n\r\n/**\r\n * Provides the instance of the QueryCache for queryResource. This should probably be called\r\n * in your application's root configuration, but can also be overriden with component/module providers.\r\n *\r\n * @param options - Optional configuration options for the cache.\r\n * @returns An Angular `Provider` for the cache.\r\n *\r\n * @example\r\n * // In your app.config.ts or AppModule providers:\r\n *\r\n * import { provideQueryCache } from './your-cache';\r\n *\r\n * export const appConfig: ApplicationConfig = {\r\n *   providers: [\r\n *     provideQueryCache({\r\n *       ttl: 60000, // Default TTL of 60 seconds\r\n *       staleTime: 30000, // Default staleTime of 30 seconds\r\n *     }),\r\n *     // ... other providers\r\n *   ]\r\n * };\r\n */\r\nexport function provideQueryCache(opt?: CacheOptions): Provider {\r\n  return {\r\n    provide: CLIENT_CACHE_TOKEN,\r\n    useValue: new Cache(opt?.ttl, opt?.staleTime, opt?.cleanup),\r\n  };\r\n}\r\n\r\nclass NoopCache<T> extends Cache<T> {\r\n  override store(_: string, __: T, ___ = super.staleTime, ____ = super.ttl) {\r\n    // noop\r\n  }\r\n}\r\n\r\n/**\r\n * Injects the `QueryCache` instance that is used within queryResource.\r\n * Allows for direct modification of cached data, but is mostly meant for internal use.\r\n *\r\n * @param injector - (Optional) The injector to use.  If not provided, the current\r\n *                   injection context is used.\r\n * @returns The `QueryCache` instance.\r\n *\r\n * @example\r\n * // In your component or service:\r\n *\r\n * import { injectQueryCache } from './your-cache';\r\n *\r\n * constructor() {\r\n *   const cache = injectQueryCache();\r\n *\r\n *   const myData = cache.get(() => 'my-data-key');\r\n *   if (myData() !== null) {\r\n *     // ... use cached data ...\r\n *   }\r\n * }\r\n */\r\nexport function injectQueryCache(\r\n  injector?: Injector,\r\n): Cache<HttpResponse<unknown>> {\r\n  const cache = injector\r\n    ? injector.get(CLIENT_CACHE_TOKEN, null, {\r\n        optional: true,\r\n      })\r\n    : inject(CLIENT_CACHE_TOKEN, {\r\n        optional: true,\r\n      });\r\n\r\n  if (!cache) {\r\n    if (isDevMode())\r\n      throw new Error(\r\n        'Cache not provided, please add provideQueryCache() to providers array',\r\n      );\r\n    else return new NoopCache();\r\n  }\r\n\r\n  return cache;\r\n}\r\n"]}","import { HttpContext, HttpContextToken, HttpResponse, } from '@angular/common/http';\nimport { map, of, tap } from 'rxjs';\nimport { injectQueryCache } from './cache';\nconst CACHE_CONTEXT = new HttpContextToken(() => ({\n cache: false,\n}));\nexport function setCacheContext(ctx = new HttpContext(), opt) {\n return ctx.set(CACHE_CONTEXT, { ...opt, cache: true });\n}\nfunction getCacheContext(ctx) {\n return ctx.get(CACHE_CONTEXT);\n}\nfunction parseCacheControlHeader(req) {\n const header = req.headers.get('Cache-Control');\n let sMaxAge = null;\n const directives = {\n noStore: false,\n noCache: false,\n mustRevalidate: false,\n immutable: false,\n maxAge: null,\n staleWhileRevalidate: null,\n };\n if (!header)\n return directives;\n const parts = header.split(',');\n for (const part of parts) {\n const [unparsedKey, value] = part.trim().split('=');\n const key = unparsedKey.trim().toLowerCase();\n switch (key) {\n case 'no-store':\n directives.noStore = true;\n break;\n case 'no-cache':\n directives.noCache = true;\n break;\n case 'must-revalidate':\n case 'proxy-revalidate':\n directives.mustRevalidate = true;\n break;\n case 'immutable':\n directives.immutable = true;\n break;\n case 'max-age': {\n if (!value)\n break;\n const parsedValue = parseInt(value, 10);\n if (!isNaN(parsedValue))\n directives.maxAge = parsedValue;\n break;\n }\n case 's-max-age': {\n if (!value)\n break;\n const parsedValue = parseInt(value, 10);\n if (!isNaN(parsedValue))\n sMaxAge = parsedValue;\n break;\n }\n case 'stale-while-revalidate': {\n if (!value)\n break;\n const parsedValue = parseInt(value, 10);\n if (!isNaN(parsedValue))\n directives.staleWhileRevalidate = parsedValue;\n break;\n }\n }\n }\n // s-max-age takes precedence over max-age\n if (sMaxAge !== null)\n directives.maxAge = sMaxAge;\n // if no store nothing else is relevant\n if (directives.noStore)\n return {\n noStore: true,\n noCache: false,\n mustRevalidate: false,\n immutable: false,\n maxAge: null,\n staleWhileRevalidate: null,\n };\n // max age does not apply to immutable resources\n if (directives.immutable)\n return {\n ...directives,\n maxAge: null,\n };\n return directives;\n}\nfunction resolveTimings(cacheControl, staleTime, ttl) {\n const timings = {\n staleTime,\n ttl,\n };\n if (cacheControl.immutable)\n return {\n staleTime: Infinity,\n ttl: Infinity,\n };\n // if no-cache is set, we must always revalidate\n if (cacheControl.noCache || cacheControl.mustRevalidate)\n timings.staleTime = 0;\n if (cacheControl.staleWhileRevalidate !== null)\n timings.staleTime = cacheControl.staleWhileRevalidate;\n if (cacheControl.maxAge !== null)\n timings.ttl = cacheControl.maxAge * 1000;\n // if stale-while-revalidate is set, we must revalidate after that time at the latest, but we can still serve the stale data\n if (cacheControl.staleWhileRevalidate !== null) {\n const ms = cacheControl.staleWhileRevalidate * 1000;\n if (timings.staleTime === undefined || timings.staleTime > ms)\n timings.staleTime = ms;\n }\n return timings;\n}\n/**\n * Creates an `HttpInterceptorFn` that implements caching for HTTP requests. This interceptor\n * checks for a caching configuration in the request's `HttpContext` (internally set by the queryResource).\n * If caching is enabled, it attempts to retrieve responses from the cache. If a cached response\n * is found and is not stale, it's returned directly. If the cached response is stale, it's returned,\n * and a background revalidation request is made. If no cached response is found, the request\n * is made to the server, and the response is cached according to the configured TTL and staleness.\n * The interceptor also respects `Cache-Control` headers from the server.\n *\n * @param allowedMethods - An array of HTTP methods for which caching should be enabled.\n * Defaults to `['GET', 'HEAD', 'OPTIONS']`.\n *\n * @returns An `HttpInterceptorFn` that implements the caching logic.\n *\n * @example\n * // In your app.config.ts or module providers:\n *\n * import { provideHttpClient, withInterceptors } from '@angular/common/http';\n * import { createCacheInterceptor } from '@mmstack/resource';\n *\n * export const appConfig: ApplicationConfig = {\n * providers: [\n * provideHttpClient(withInterceptors([createCacheInterceptor()])),\n * // ... other providers\n * ],\n * };\n */\nexport function createCacheInterceptor(allowedMethods = ['GET', 'HEAD', 'OPTIONS']) {\n const CACHE_METHODS = new Set(allowedMethods);\n return (req, next) => {\n const cache = injectQueryCache();\n if (!CACHE_METHODS.has(req.method))\n return next(req);\n const opt = getCacheContext(req.context);\n if (!opt.cache)\n return next(req);\n const key = opt.key ?? req.urlWithParams;\n const entry = cache.getUntracked(key); // null if expired or not found\n // If the entry is not stale, return it\n if (entry && !entry.isStale)\n return of(entry.value);\n // resource itself handles case of showing stale data...the request must process as this will \"refresh said data\"\n const eTag = entry?.value.headers.get('ETag');\n const lastModified = entry?.value.headers.get('Last-Modified');\n if (eTag) {\n req = req.clone({ setHeaders: { 'If-None-Match': eTag } });\n }\n if (lastModified) {\n req = req.clone({ setHeaders: { 'If-Modified-Since': lastModified } });\n }\n return next(req).pipe(tap((event) => {\n if (event instanceof HttpResponse && event.ok) {\n const cacheControl = parseCacheControlHeader(event);\n if (cacheControl.noStore)\n return;\n const { staleTime, ttl } = resolveTimings(cacheControl, opt.staleTime, opt.ttl);\n cache.store(key, event, staleTime, ttl);\n }\n }), map((event) => {\n // handle 304 responses due to eTag/last-modified\n if (event instanceof HttpResponse && event.status === 304 && entry) {\n return entry.value;\n }\n return event;\n }));\n };\n}\n//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cache.interceptor.js","sourceRoot":"","sources":["../../../../../../../packages/resource/src/lib/util/cache/cache.interceptor.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  HttpContext,\r\n  HttpContextToken,\r\n  type HttpEvent,\r\n  type HttpHandlerFn,\r\n  type HttpInterceptorFn,\r\n  type HttpRequest,\r\n  HttpResponse,\r\n} from '@angular/common/http';\r\nimport { map, Observable, of, tap } from 'rxjs';\r\nimport { injectQueryCache } from './cache';\r\n\r\ntype CacheEntryOptions = {\r\n  key?: string;\r\n  ttl?: number;\r\n  staleTime?: number;\r\n  cache: boolean;\r\n};\r\n\r\nconst CACHE_CONTEXT = new HttpContextToken<CacheEntryOptions>(() => ({\r\n  cache: false,\r\n}));\r\n\r\nexport function setCacheContext(\r\n  ctx = new HttpContext(),\r\n  opt: Omit<CacheEntryOptions, 'cache' | 'key'> & {\r\n    key: Required<CacheEntryOptions>['key'];\r\n  },\r\n) {\r\n  return ctx.set(CACHE_CONTEXT, { ...opt, cache: true });\r\n}\r\n\r\nfunction getCacheContext(ctx: HttpContext): CacheEntryOptions {\r\n  return ctx.get(CACHE_CONTEXT);\r\n}\r\n\r\ntype ResolvedCacheControl = {\r\n  noStore: boolean;\r\n  noCache: boolean;\r\n  mustRevalidate: boolean;\r\n  immutable: boolean;\r\n  maxAge: number | null;\r\n  staleWhileRevalidate: number | null;\r\n};\r\n\r\nfunction parseCacheControlHeader(\r\n  req: HttpResponse<unknown>,\r\n): ResolvedCacheControl {\r\n  const header = req.headers.get('Cache-Control');\r\n\r\n  let sMaxAge: number | null = null;\r\n  const directives: ResolvedCacheControl = {\r\n    noStore: false,\r\n    noCache: false,\r\n    mustRevalidate: false,\r\n    immutable: false,\r\n    maxAge: null,\r\n    staleWhileRevalidate: null,\r\n  };\r\n\r\n  if (!header) return directives;\r\n\r\n  const parts = header.split(',');\r\n\r\n  for (const part of parts) {\r\n    const [unparsedKey, value] = part.trim().split('=');\r\n    const key = unparsedKey.trim().toLowerCase();\r\n\r\n    switch (key) {\r\n      case 'no-store':\r\n        directives.noStore = true;\r\n        break;\r\n      case 'no-cache':\r\n        directives.noCache = true;\r\n        break;\r\n      case 'must-revalidate':\r\n      case 'proxy-revalidate':\r\n        directives.mustRevalidate = true;\r\n        break;\r\n      case 'immutable':\r\n        directives.immutable = true;\r\n        break;\r\n      case 'max-age': {\r\n        if (!value) break;\r\n        const parsedValue = parseInt(value, 10);\r\n        if (!isNaN(parsedValue)) directives.maxAge = parsedValue;\r\n        break;\r\n      }\r\n      case 's-max-age': {\r\n        if (!value) break;\r\n        const parsedValue = parseInt(value, 10);\r\n        if (!isNaN(parsedValue)) sMaxAge = parsedValue;\r\n        break;\r\n      }\r\n      case 'stale-while-revalidate': {\r\n        if (!value) break;\r\n        const parsedValue = parseInt(value, 10);\r\n        if (!isNaN(parsedValue)) directives.staleWhileRevalidate = parsedValue;\r\n        break;\r\n      }\r\n    }\r\n  }\r\n\r\n  // s-max-age takes precedence over max-age\r\n  if (sMaxAge !== null) directives.maxAge = sMaxAge;\r\n\r\n  // if no store nothing else is relevant\r\n  if (directives.noStore)\r\n    return {\r\n      noStore: true,\r\n      noCache: false,\r\n      mustRevalidate: false,\r\n      immutable: false,\r\n      maxAge: null,\r\n      staleWhileRevalidate: null,\r\n    };\r\n\r\n  // max age does not apply to immutable resources\r\n  if (directives.immutable)\r\n    return {\r\n      ...directives,\r\n      maxAge: null,\r\n    };\r\n\r\n  return directives;\r\n}\r\n\r\nfunction resolveTimings(\r\n  cacheControl: ResolvedCacheControl,\r\n  staleTime?: number,\r\n  ttl?: number,\r\n): { staleTime?: number; ttl?: number } {\r\n  const timings = {\r\n    staleTime,\r\n    ttl,\r\n  };\r\n\r\n  if (cacheControl.immutable)\r\n    return {\r\n      staleTime: Infinity,\r\n      ttl: Infinity,\r\n    };\r\n\r\n  // if no-cache is set, we must always revalidate\r\n  if (cacheControl.noCache || cacheControl.mustRevalidate)\r\n    timings.staleTime = 0;\r\n\r\n  if (cacheControl.staleWhileRevalidate !== null)\r\n    timings.staleTime = cacheControl.staleWhileRevalidate;\r\n\r\n  if (cacheControl.maxAge !== null) timings.ttl = cacheControl.maxAge * 1000;\r\n\r\n  // if stale-while-revalidate is set, we must revalidate after that time at the latest, but we can still serve the stale data\r\n  if (cacheControl.staleWhileRevalidate !== null) {\r\n    const ms = cacheControl.staleWhileRevalidate * 1000;\r\n    if (timings.staleTime === undefined || timings.staleTime > ms)\r\n      timings.staleTime = ms;\r\n  }\r\n\r\n  return timings;\r\n}\r\n\r\n/**\r\n * Creates an `HttpInterceptorFn` that implements caching for HTTP requests. This interceptor\r\n * checks for a caching configuration in the request's `HttpContext` (internally set by the queryResource).\r\n * If caching is enabled, it attempts to retrieve responses from the cache. If a cached response\r\n * is found and is not stale, it's returned directly.  If the cached response is stale, it's returned,\r\n * and a background revalidation request is made.  If no cached response is found, the request\r\n * is made to the server, and the response is cached according to the configured TTL and staleness.\r\n * The interceptor also respects `Cache-Control` headers from the server.\r\n *\r\n * @param allowedMethods - An array of HTTP methods for which caching should be enabled.\r\n *                        Defaults to `['GET', 'HEAD', 'OPTIONS']`.\r\n *\r\n * @returns An `HttpInterceptorFn` that implements the caching logic.\r\n *\r\n * @example\r\n * // In your app.config.ts or module providers:\r\n *\r\n * import { provideHttpClient, withInterceptors } from '@angular/common/http';\r\n * import { createCacheInterceptor } from '@mmstack/resource';\r\n *\r\n * export const appConfig: ApplicationConfig = {\r\n *   providers: [\r\n *     provideHttpClient(withInterceptors([createCacheInterceptor()])),\r\n *     // ... other providers\r\n *   ],\r\n * };\r\n */\r\nexport function createCacheInterceptor(\r\n  allowedMethods = ['GET', 'HEAD', 'OPTIONS'],\r\n): HttpInterceptorFn {\r\n  const CACHE_METHODS = new Set<string>(allowedMethods);\r\n\r\n  return (\r\n    req: HttpRequest<unknown>,\r\n    next: HttpHandlerFn,\r\n  ): Observable<HttpEvent<unknown>> => {\r\n    const cache = injectQueryCache();\r\n\r\n    if (!CACHE_METHODS.has(req.method)) return next(req);\r\n    const opt = getCacheContext(req.context);\r\n\r\n    if (!opt.cache) return next(req);\r\n\r\n    const key = opt.key ?? req.urlWithParams;\r\n    const entry = cache.getUntracked(key); // null if expired or not found\r\n\r\n    // If the entry is not stale, return it\r\n    if (entry && !entry.isStale) return of(entry.value);\r\n\r\n    // resource itself handles case of showing stale data...the request must process as this will \"refresh said data\"\r\n\r\n    const eTag = entry?.value.headers.get('ETag');\r\n    const lastModified = entry?.value.headers.get('Last-Modified');\r\n\r\n    if (eTag) {\r\n      req = req.clone({ setHeaders: { 'If-None-Match': eTag } });\r\n    }\r\n\r\n    if (lastModified) {\r\n      req = req.clone({ setHeaders: { 'If-Modified-Since': lastModified } });\r\n    }\r\n\r\n    return next(req).pipe(\r\n      tap((event) => {\r\n        if (event instanceof HttpResponse && event.ok) {\r\n          const cacheControl = parseCacheControlHeader(event);\r\n          if (cacheControl.noStore) return;\r\n\r\n          const { staleTime, ttl } = resolveTimings(\r\n            cacheControl,\r\n            opt.staleTime,\r\n            opt.ttl,\r\n          );\r\n\r\n          cache.store(key, event, staleTime, ttl);\r\n        }\r\n      }),\r\n      map((event) => {\r\n        // handle 304 responses due to eTag/last-modified\r\n        if (event instanceof HttpResponse && event.status === 304 && entry) {\r\n          return entry.value;\r\n        }\r\n\r\n        return event;\r\n      }),\r\n    );\r\n  };\r\n}\r\n"]}","import { computed, effect, signal, untracked } from '@angular/core';\n/** @internal */\nfunction internalCeateCircuitBreaker(treshold = 5, resetTimeout = 30000) {\n const halfOpen = signal(false);\n const failureCount = signal(0);\n const status = computed(() => {\n if (failureCount() >= treshold)\n return 'CLOSED';\n return halfOpen() ? 'HALF_OPEN' : 'OPEN';\n });\n const isClosed = computed(() => status() === 'CLOSED');\n const success = () => {\n failureCount.set(0);\n halfOpen.set(false);\n };\n const tryOnce = () => {\n if (!untracked(isClosed))\n return;\n halfOpen.set(true);\n failureCount.set(treshold - 1);\n };\n const effectRef = effect((cleanup) => {\n if (!isClosed())\n return;\n const timeout = setTimeout(tryOnce, resetTimeout);\n return cleanup(() => clearTimeout(timeout));\n });\n const fail = () => {\n failureCount.set(failureCount() + 1);\n halfOpen.set(false);\n };\n return {\n status,\n isClosed,\n fail,\n success,\n halfOpen: tryOnce,\n destroy: () => effectRef.destroy(),\n };\n}\n/** @internal */\nfunction createNeverBrokenCircuitBreaker() {\n return {\n isClosed: computed(() => false),\n status: signal('OPEN'),\n fail: () => {\n // noop\n },\n success: () => {\n // noop\n },\n halfOpen: () => {\n // noop\n },\n destroy: () => {\n // noop\n },\n };\n}\n/**\n * Creates a circuit breaker instance.\n *\n * @param options - Configuration options for the circuit breaker. Can be:\n * - `undefined`: Creates a \"no-op\" circuit breaker that is always open (never trips).\n * - `true`: Creates a circuit breaker with default settings (threshold: 5, timeout: 30000ms).\n * - `CircuitBreaker`: Reuses an existing `CircuitBreaker` instance.\n * - `{ threshold?: number; timeout?: number; }`: Creates a circuit breaker with the specified threshold and timeout.\n *\n * @returns A `CircuitBreaker` instance.\n *\n * @example\n * // Create a circuit breaker with default settings:\n * const breaker = createCircuitBreaker();\n *\n * // Create a circuit breaker with custom settings:\n * const customBreaker = createCircuitBreaker({ threshold: 10, timeout: 60000 });\n *\n * // Share a single circuit breaker instance across multiple resources:\n * const sharedBreaker = createCircuitBreaker();\n * const resource1 = queryResource(..., { circuitBreaker: sharedBreaker });\n * const resource2 = mutationResource(..., { circuitBreaker: sharedBreaker });\n */\nexport function createCircuitBreaker(opt) {\n if (opt === false)\n return createNeverBrokenCircuitBreaker();\n if (typeof opt === 'object' && 'isClosed' in opt)\n return opt;\n return internalCeateCircuitBreaker(opt?.treshold, opt?.timeout);\n}\n//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"circuit-breaker.js","sourceRoot":"","sources":["../../../../../../packages/resource/src/lib/util/circuit-breaker.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,EAAU,MAAM,EAAE,SAAS,EAAE,MAAM,eAAe,CAAC;AAqD5E,gBAAgB;AAChB,SAAS,2BAA2B,CAClC,QAAQ,GAAG,CAAC,EACZ,YAAY,GAAG,KAAK;IAEpB,MAAM,QAAQ,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;IAC/B,MAAM,YAAY,GAAG,MAAM,CAAC,CAAC,CAAC,CAAC;IAE/B,MAAM,MAAM,GAAG,QAAQ,CAAsB,GAAG,EAAE;QAChD,IAAI,YAAY,EAAE,IAAI,QAAQ;YAAE,OAAO,QAAQ,CAAC;QAChD,OAAO,QAAQ,EAAE,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC,CAAC,MAAM,CAAC;IAC3C,CAAC,CAAC,CAAC;IAEH,MAAM,QAAQ,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,MAAM,EAAE,KAAK,QAAQ,CAAC,CAAC;IAEvD,MAAM,OAAO,GAAG,GAAG,EAAE;QACnB,YAAY,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;QACpB,QAAQ,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;IACtB,CAAC,CAAC;IAEF,MAAM,OAAO,GAAG,GAAG,EAAE;QACnB,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC;YAAE,OAAO;QACjC,QAAQ,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QACnB,YAAY,CAAC,GAAG,CAAC,QAAQ,GAAG,CAAC,CAAC,CAAC;IACjC,CAAC,CAAC;IAEF,MAAM,SAAS,GAAG,MAAM,CAAC,CAAC,OAAO,EAAE,EAAE;QACnC,IAAI,CAAC,QAAQ,EAAE;YAAE,OAAO;QAExB,MAAM,OAAO,GAAG,UAAU,CAAC,OAAO,EAAE,YAAY,CAAC,CAAC;QAElD,OAAO,OAAO,CAAC,GAAG,EAAE,CAAC,YAAY,CAAC,OAAO,CAAC,CAAC,CAAC;IAC9C,CAAC,CAAC,CAAC;IAEH,MAAM,IAAI,GAAG,GAAG,EAAE;QAChB,YAAY,CAAC,GAAG,CAAC,YAAY,EAAE,GAAG,CAAC,CAAC,CAAC;QACrC,QAAQ,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;IACtB,CAAC,CAAC;IAEF,OAAO;QACL,MAAM;QACN,QAAQ;QACR,IAAI;QACJ,OAAO;QACP,QAAQ,EAAE,OAAO;QACjB,OAAO,EAAE,GAAG,EAAE,CAAC,SAAS,CAAC,OAAO,EAAE;KACnC,CAAC;AACJ,CAAC;AAED,gBAAgB;AAChB,SAAS,+BAA+B;IACtC,OAAO;QACL,QAAQ,EAAE,QAAQ,CAAC,GAAG,EAAE,CAAC,KAAK,CAAC;QAC/B,MAAM,EAAE,MAAM,CAAC,MAAM,CAAC;QACtB,IAAI,EAAE,GAAG,EAAE;YACT,OAAO;QACT,CAAC;QACD,OAAO,EAAE,GAAG,EAAE;YACZ,OAAO;QACT,CAAC;QACD,QAAQ,EAAE,GAAG,EAAE;YACb,OAAO;QACT,CAAC;QACD,OAAO,EAAE,GAAG,EAAE;YACZ,OAAO;QACT,CAAC;KACF,CAAC;AACJ,CAAC;AAED;;;;;;;;;;;;;;;;;;;;;;GAsBG;AACH,MAAM,UAAU,oBAAoB,CAClC,GAA2B;IAE3B,IAAI,GAAG,KAAK,KAAK;QAAE,OAAO,+BAA+B,EAAE,CAAC;IAE5D,IAAI,OAAO,GAAG,KAAK,QAAQ,IAAI,UAAU,IAAI,GAAG;QAAE,OAAO,GAAG,CAAC;IAE7D,OAAO,2BAA2B,CAAC,GAAG,EAAE,QAAQ,EAAE,GAAG,EAAE,OAAO,CAAC,CAAC;AAClE,CAAC","sourcesContent":["import { computed, effect, Signal, signal, untracked } from '@angular/core';\r\n\r\n/**\r\n * Represents the possible states of a circuit breaker.\r\n * - `CLOSED`: The circuit breaker is closed, and operations are allowed to proceed.\r\n * - `OPEN`: The circuit breaker is open, and operations are blocked.\r\n * - `HALF_OPEN`: The circuit breaker is in a half-open state, allowing a limited number of operations to test if the underlying issue is resolved.\r\n */\r\ntype CircuitBreakerState = 'CLOSED' | 'OPEN' | 'HALF_OPEN';\r\n\r\n/**\r\n * Represents a circuit breaker, which monitors operations and prevents failures from cascading.\r\n */\r\nexport type CircuitBreaker = {\r\n  /**\r\n   * A signal indicating whether the circuit breaker is currently closed (allowing operations).\r\n   */\r\n  isClosed: Signal<boolean>;\r\n  /**\r\n   * A signal representing the current state of the circuit breaker.\r\n   */\r\n  status: Signal<CircuitBreakerState>;\r\n  /**\r\n   * Signals a failure to the circuit breaker.  This may cause the circuit breaker to open.\r\n   */\r\n  fail: () => void;\r\n  /**\r\n   * Signals a success to the circuit breaker.  This may cause the circuit breaker to close.\r\n   */\r\n  success: () => void;\r\n  /**\r\n   * Attempts to transition the circuit breaker to the half-open state. This is typically used\r\n   * to test if the underlying issue has been resolved after the circuit breaker has been open.\r\n   */\r\n  halfOpen: () => void;\r\n  /**\r\n   * Destroys the circuit breaker & initiates related cleanup\r\n   */\r\n  destroy: () => void;\r\n};\r\n\r\n/**\r\n * Options for creating a circuit breaker.\r\n *  - `false`: Disables circuit breaker functionality (always open).\r\n *  - true: Creates a new circuit breaker with default options.\r\n *  - `CircuitBreaker`: Provides an existing `CircuitBreaker` instance to use.\r\n *  - `{ treshold?: number; timeout?: number; }`: Creates a new circuit breaker with the specified options.\r\n */\r\nexport type CircuitBreakerOptions =\r\n  | false\r\n  | CircuitBreaker\r\n  | { treshold?: number; timeout?: number };\r\n\r\n/** @internal */\r\nfunction internalCeateCircuitBreaker(\r\n  treshold = 5,\r\n  resetTimeout = 30000,\r\n): CircuitBreaker {\r\n  const halfOpen = signal(false);\r\n  const failureCount = signal(0);\r\n\r\n  const status = computed<CircuitBreakerState>(() => {\r\n    if (failureCount() >= treshold) return 'CLOSED';\r\n    return halfOpen() ? 'HALF_OPEN' : 'OPEN';\r\n  });\r\n\r\n  const isClosed = computed(() => status() === 'CLOSED');\r\n\r\n  const success = () => {\r\n    failureCount.set(0);\r\n    halfOpen.set(false);\r\n  };\r\n\r\n  const tryOnce = () => {\r\n    if (!untracked(isClosed)) return;\r\n    halfOpen.set(true);\r\n    failureCount.set(treshold - 1);\r\n  };\r\n\r\n  const effectRef = effect((cleanup) => {\r\n    if (!isClosed()) return;\r\n\r\n    const timeout = setTimeout(tryOnce, resetTimeout);\r\n\r\n    return cleanup(() => clearTimeout(timeout));\r\n  });\r\n\r\n  const fail = () => {\r\n    failureCount.set(failureCount() + 1);\r\n    halfOpen.set(false);\r\n  };\r\n\r\n  return {\r\n    status,\r\n    isClosed,\r\n    fail,\r\n    success,\r\n    halfOpen: tryOnce,\r\n    destroy: () => effectRef.destroy(),\r\n  };\r\n}\r\n\r\n/** @internal */\r\nfunction createNeverBrokenCircuitBreaker(): CircuitBreaker {\r\n  return {\r\n    isClosed: computed(() => false),\r\n    status: signal('OPEN'),\r\n    fail: () => {\r\n      // noop\r\n    },\r\n    success: () => {\r\n      // noop\r\n    },\r\n    halfOpen: () => {\r\n      // noop\r\n    },\r\n    destroy: () => {\r\n      // noop\r\n    },\r\n  };\r\n}\r\n\r\n/**\r\n * Creates a circuit breaker instance.\r\n *\r\n * @param options - Configuration options for the circuit breaker.  Can be:\r\n *   - `undefined`:  Creates a \"no-op\" circuit breaker that is always open (never trips).\r\n *   - `true`: Creates a circuit breaker with default settings (threshold: 5, timeout: 30000ms).\r\n *   - `CircuitBreaker`:  Reuses an existing `CircuitBreaker` instance.\r\n *   - `{ threshold?: number; timeout?: number; }`: Creates a circuit breaker with the specified threshold and timeout.\r\n *\r\n * @returns A `CircuitBreaker` instance.\r\n *\r\n * @example\r\n * // Create a circuit breaker with default settings:\r\n * const breaker = createCircuitBreaker();\r\n *\r\n * // Create a circuit breaker with custom settings:\r\n * const customBreaker = createCircuitBreaker({ threshold: 10, timeout: 60000 });\r\n *\r\n * // Share a single circuit breaker instance across multiple resources:\r\n * const sharedBreaker = createCircuitBreaker();\r\n * const resource1 = queryResource(..., { circuitBreaker: sharedBreaker });\r\n * const resource2 = mutationResource(..., { circuitBreaker: sharedBreaker });\r\n */\r\nexport function createCircuitBreaker(\r\n  opt?: CircuitBreakerOptions,\r\n): CircuitBreaker {\r\n  if (opt === false) return createNeverBrokenCircuitBreaker();\r\n\r\n  if (typeof opt === 'object' && 'isClosed' in opt) return opt;\r\n\r\n  return internalCeateCircuitBreaker(opt?.treshold, opt?.timeout);\r\n}\r\n"]}","// Heavily inspired by: https://dev.to/kasual1/request-deduplication-in-angular-3pd8\nimport { HttpContext, HttpContextToken, } from '@angular/common/http';\nimport { finalize, shareReplay } from 'rxjs';\nconst NO_DEDUPE = new HttpContextToken(() => false);\n/**\n * Disables request deduplication for a specific HTTP request.\n *\n * @param ctx - The `HttpContext` to modify. If not provided, a new `HttpContext` is created.\n * @returns The modified `HttpContext` with the `NO_DEDUPE` token set to `true`.\n *\n * @example\n * // Disable deduplication for a specific POST request:\n * const context = noDedupe();\n * this.http.post('/api/data', payload, { context }).subscribe(...);\n *\n * // Disable deduplication, modifying an existing context:\n * let context = new HttpContext();\n * context = noDedupe(context);\n * this.http.post('/api/data', payload, { context }).subscribe(...);\n */\nexport function noDedupe(ctx = new HttpContext()) {\n return ctx.set(NO_DEDUPE, true);\n}\n/**\n * Creates an `HttpInterceptorFn` that deduplicates identical HTTP requests.\n * If multiple identical requests (same URL and parameters) are made concurrently,\n * only the first request will be sent to the server. Subsequent requests will\n * receive the response from the first request.\n *\n * @param allowed - An array of HTTP methods for which deduplication should be enabled.\n * Defaults to `['GET', 'DELETE', 'HEAD', 'OPTIONS']`.\n *\n * @returns An `HttpInterceptorFn` that implements the request deduplication logic.\n *\n * @example\n * // In your app.config.ts or module providers:\n * import { provideHttpClient, withInterceptors } from '@angular/common/http';\n * import { createDedupeRequestsInterceptor } from './your-dedupe-interceptor';\n *\n * export const appConfig: ApplicationConfig = {\n * providers: [\n * provideHttpClient(withInterceptors([createDedupeRequestsInterceptor()])),\n * // ... other providers\n * ],\n * };\n *\n * // You can also specify which methods should be deduped\n * export const appConfig: ApplicationConfig = {\n * providers: [\n * provideHttpClient(withInterceptors([createDedupeRequestsInterceptor(['GET'])])), // only dedupe GET calls\n * // ... other providers\n * ],\n * };\n */\nexport function createDedupeRequestsInterceptor(allowed = ['GET', 'DELETE', 'HEAD', 'OPTIONS']) {\n const inFlight = new Map();\n const DEDUPE_METHODS = new Set(allowed);\n return (req, next) => {\n if (!DEDUPE_METHODS.has(req.method) || req.context.get(NO_DEDUPE))\n return next(req);\n const found = inFlight.get(req.urlWithParams);\n if (found)\n return found;\n const request = next(req).pipe(finalize(() => inFlight.delete(req.urlWithParams)), shareReplay());\n inFlight.set(req.urlWithParams, request);\n return request;\n };\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGVkdXBlLmludGVyY2VwdG9yLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcGFja2FnZXMvcmVzb3VyY2Uvc3JjL2xpYi91dGlsL2RlZHVwZS5pbnRlcmNlcHRvci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxvRkFBb0Y7QUFFcEYsT0FBTyxFQUNMLFdBQVcsRUFDWCxnQkFBZ0IsR0FLakIsTUFBTSxzQkFBc0IsQ0FBQztBQUM5QixPQUFPLEVBQUUsUUFBUSxFQUFFLFdBQVcsRUFBbUIsTUFBTSxNQUFNLENBQUM7QUFFOUQsTUFBTSxTQUFTLEdBQUcsSUFBSSxnQkFBZ0IsQ0FBVSxHQUFHLEVBQUUsQ0FBQyxLQUFLLENBQUMsQ0FBQztBQUU3RDs7Ozs7Ozs7Ozs7Ozs7O0dBZUc7QUFDSCxNQUFNLFVBQVUsUUFBUSxDQUFDLE1BQW1CLElBQUksV0FBVyxFQUFFO0lBQzNELE9BQU8sR0FBRyxDQUFDLEdBQUcsQ0FBQyxTQUFTLEVBQUUsSUFBSSxDQUFDLENBQUM7QUFDbEMsQ0FBQztBQUVEOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0E4Qkc7QUFDSCxNQUFNLFVBQVUsK0JBQStCLENBQzdDLE9BQU8sR0FBRyxDQUFDLEtBQUssRUFBRSxRQUFRLEVBQUUsTUFBTSxFQUFFLFNBQVMsQ0FBQztJQUU5QyxNQUFNLFFBQVEsR0FBRyxJQUFJLEdBQUcsRUFBMEMsQ0FBQztJQUVuRSxNQUFNLGNBQWMsR0FBRyxJQUFJLEdBQUcsQ0FBUyxPQUFPLENBQUMsQ0FBQztJQUVoRCxPQUFPLENBQ0wsR0FBeUIsRUFDekIsSUFBbUIsRUFDYSxFQUFFO1FBQ2xDLElBQUksQ0FBQyxjQUFjLENBQUMsR0FBRyxDQUFDLEdBQUcsQ0FBQyxNQUFNLENBQUMsSUFBSSxHQUFHLENBQUMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxTQUFTLENBQUM7WUFDL0QsT0FBTyxJQUFJLENBQUMsR0FBRyxDQUFDLENBQUM7UUFFbkIsTUFBTSxLQUFLLEdBQUcsUUFBUSxDQUFDLEdBQUcsQ0FBQyxHQUFHLENBQUMsYUFBYSxDQUFDLENBQUM7UUFFOUMsSUFBSSxLQUFLO1lBQUUsT0FBTyxLQUFLLENBQUM7UUFFeEIsTUFBTSxPQUFPLEdBQUcsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDLElBQUksQ0FDNUIsUUFBUSxDQUFDLEdBQUcsRUFBRSxDQUFDLFFBQVEsQ0FBQyxNQUFNLENBQUMsR0FBRyxDQUFDLGFBQWEsQ0FBQyxDQUFDLEVBQ2xELFdBQVcsRUFBRSxDQUNkLENBQUM7UUFDRixRQUFRLENBQUMsR0FBRyxDQUFDLEdBQUcsQ0FBQyxhQUFhLEVBQUUsT0FBTyxDQUFDLENBQUM7UUFFekMsT0FBTyxPQUFPLENBQUM7SUFDakIsQ0FBQyxDQUFDO0FBQ0osQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8vIEhlYXZpbHkgaW5zcGlyZWQgYnk6IGh0dHBzOi8vZGV2LnRvL2thc3VhbDEvcmVxdWVzdC1kZWR1cGxpY2F0aW9uLWluLWFuZ3VsYXItM3BkOFxyXG5cclxuaW1wb3J0IHtcclxuICBIdHRwQ29udGV4dCxcclxuICBIdHRwQ29udGV4dFRva2VuLFxyXG4gIEh0dHBJbnRlcmNlcHRvckZuLFxyXG4gIHR5cGUgSHR0cEV2ZW50LFxyXG4gIHR5cGUgSHR0cEhhbmRsZXJGbixcclxuICB0eXBlIEh0dHBSZXF1ZXN0LFxyXG59IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbi9odHRwJztcclxuaW1wb3J0IHsgZmluYWxpemUsIHNoYXJlUmVwbGF5LCB0eXBlIE9ic2VydmFibGUgfSBmcm9tICdyeGpzJztcclxuXHJcbmNvbnN0IE5PX0RFRFVQRSA9IG5ldyBIdHRwQ29udGV4dFRva2VuPGJvb2xlYW4+KCgpID0+IGZhbHNlKTtcclxuXHJcbi8qKlxyXG4gKiBEaXNhYmxlcyByZXF1ZXN0IGRlZHVwbGljYXRpb24gZm9yIGEgc3BlY2lmaWMgSFRUUCByZXF1ZXN0LlxyXG4gKlxyXG4gKiBAcGFyYW0gY3R4IC0gVGhlIGBIdHRwQ29udGV4dGAgdG8gbW9kaWZ5LiBJZiBub3QgcHJvdmlkZWQsIGEgbmV3IGBIdHRwQ29udGV4dGAgaXMgY3JlYXRlZC5cclxuICogQHJldHVybnMgVGhlIG1vZGlmaWVkIGBIdHRwQ29udGV4dGAgd2l0aCB0aGUgYE5PX0RFRFVQRWAgdG9rZW4gc2V0IHRvIGB0cnVlYC5cclxuICpcclxuICogQGV4YW1wbGVcclxuICogLy8gRGlzYWJsZSBkZWR1cGxpY2F0aW9uIGZvciBhIHNwZWNpZmljIFBPU1QgcmVxdWVzdDpcclxuICogY29uc3QgY29udGV4dCA9IG5vRGVkdXBlKCk7XHJcbiAqIHRoaXMuaHR0cC5wb3N0KCcvYXBpL2RhdGEnLCBwYXlsb2FkLCB7IGNvbnRleHQgfSkuc3Vic2NyaWJlKC4uLik7XHJcbiAqXHJcbiAqIC8vIERpc2FibGUgZGVkdXBsaWNhdGlvbiwgbW9kaWZ5aW5nIGFuIGV4aXN0aW5nIGNvbnRleHQ6XHJcbiAqIGxldCBjb250ZXh0ID0gbmV3IEh0dHBDb250ZXh0KCk7XHJcbiAqIGNvbnRleHQgPSBub0RlZHVwZShjb250ZXh0KTtcclxuICogdGhpcy5odHRwLnBvc3QoJy9hcGkvZGF0YScsIHBheWxvYWQsIHsgY29udGV4dCB9KS5zdWJzY3JpYmUoLi4uKTtcclxuICovXHJcbmV4cG9ydCBmdW5jdGlvbiBub0RlZHVwZShjdHg6IEh0dHBDb250ZXh0ID0gbmV3IEh0dHBDb250ZXh0KCkpIHtcclxuICByZXR1cm4gY3R4LnNldChOT19ERURVUEUsIHRydWUpO1xyXG59XHJcblxyXG4vKipcclxuICogQ3JlYXRlcyBhbiBgSHR0cEludGVyY2VwdG9yRm5gIHRoYXQgZGVkdXBsaWNhdGVzIGlkZW50aWNhbCBIVFRQIHJlcXVlc3RzLlxyXG4gKiBJZiBtdWx0aXBsZSBpZGVudGljYWwgcmVxdWVzdHMgKHNhbWUgVVJMIGFuZCBwYXJhbWV0ZXJzKSBhcmUgbWFkZSBjb25jdXJyZW50bHksXHJcbiAqIG9ubHkgdGhlIGZpcnN0IHJlcXVlc3Qgd2lsbCBiZSBzZW50IHRvIHRoZSBzZXJ2ZXIuIFN1YnNlcXVlbnQgcmVxdWVzdHMgd2lsbFxyXG4gKiByZWNlaXZlIHRoZSByZXNwb25zZSBmcm9tIHRoZSBmaXJzdCByZXF1ZXN0LlxyXG4gKlxyXG4gKiBAcGFyYW0gYWxsb3dlZCAtIEFuIGFycmF5IG9mIEhUVFAgbWV0aG9kcyBmb3Igd2hpY2ggZGVkdXBsaWNhdGlvbiBzaG91bGQgYmUgZW5hYmxlZC5cclxuICogICAgICAgICAgICAgICAgICBEZWZhdWx0cyB0byBgWydHRVQnLCAnREVMRVRFJywgJ0hFQUQnLCAnT1BUSU9OUyddYC5cclxuICpcclxuICogQHJldHVybnMgQW4gYEh0dHBJbnRlcmNlcHRvckZuYCB0aGF0IGltcGxlbWVudHMgdGhlIHJlcXVlc3QgZGVkdXBsaWNhdGlvbiBsb2dpYy5cclxuICpcclxuICogQGV4YW1wbGVcclxuICogLy8gSW4geW91ciBhcHAuY29uZmlnLnRzIG9yIG1vZHVsZSBwcm92aWRlcnM6XHJcbiAqIGltcG9ydCB7IHByb3ZpZGVIdHRwQ2xpZW50LCB3aXRoSW50ZXJjZXB0b3JzIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uL2h0dHAnO1xyXG4gKiBpbXBvcnQgeyBjcmVhdGVEZWR1cGVSZXF1ZXN0c0ludGVyY2VwdG9yIH0gZnJvbSAnLi95b3VyLWRlZHVwZS1pbnRlcmNlcHRvcic7XHJcbiAqXHJcbiAqIGV4cG9ydCBjb25zdCBhcHBDb25maWc6IEFwcGxpY2F0aW9uQ29uZmlnID0ge1xyXG4gKiAgIHByb3ZpZGVyczogW1xyXG4gKiAgICAgcHJvdmlkZUh0dHBDbGllbnQod2l0aEludGVyY2VwdG9ycyhbY3JlYXRlRGVkdXBlUmVxdWVzdHNJbnRlcmNlcHRvcigpXSkpLFxyXG4gKiAgICAgLy8gLi4uIG90aGVyIHByb3ZpZGVyc1xyXG4gKiAgIF0sXHJcbiAqIH07XHJcbiAqXHJcbiAqIC8vIFlvdSBjYW4gYWxzbyBzcGVjaWZ5IHdoaWNoIG1ldGhvZHMgc2hvdWxkIGJlIGRlZHVwZWRcclxuICogIGV4cG9ydCBjb25zdCBhcHBDb25maWc6IEFwcGxpY2F0aW9uQ29uZmlnID0ge1xyXG4gKiAgIHByb3ZpZGVyczogW1xyXG4gKiAgICAgcHJvdmlkZUh0dHBDbGllbnQod2l0aEludGVyY2VwdG9ycyhbY3JlYXRlRGVkdXBlUmVxdWVzdHNJbnRlcmNlcHRvcihbJ0dFVCddKV0pKSwgLy8gb25seSBkZWR1cGUgR0VUIGNhbGxzXHJcbiAqICAgICAvLyAuLi4gb3RoZXIgcHJvdmlkZXJzXHJcbiAqICAgXSxcclxuICogfTtcclxuICovXHJcbmV4cG9ydCBmdW5jdGlvbiBjcmVhdGVEZWR1cGVSZXF1ZXN0c0ludGVyY2VwdG9yKFxyXG4gIGFsbG93ZWQgPSBbJ0dFVCcsICdERUxFVEUnLCAnSEVBRCcsICdPUFRJT05TJ10sXHJcbik6IEh0dHBJbnRlcmNlcHRvckZuIHtcclxuICBjb25zdCBpbkZsaWdodCA9IG5ldyBNYXA8c3RyaW5nLCBPYnNlcnZhYmxlPEh0dHBFdmVudDx1bmtub3duPj4+KCk7XHJcblxyXG4gIGNvbnN0IERFRFVQRV9NRVRIT0RTID0gbmV3IFNldDxzdHJpbmc+KGFsbG93ZWQpO1xyXG5cclxuICByZXR1cm4gKFxyXG4gICAgcmVxOiBIdHRwUmVxdWVzdDx1bmtub3duPixcclxuICAgIG5leHQ6IEh0dHBIYW5kbGVyRm4sXHJcbiAgKTogT2JzZXJ2YWJsZTxIdHRwRXZlbnQ8dW5rbm93bj4+ID0+IHtcclxuICAgIGlmICghREVEVVBFX01FVEhPRFMuaGFzKHJlcS5tZXRob2QpIHx8IHJlcS5jb250ZXh0LmdldChOT19ERURVUEUpKVxyXG4gICAgICByZXR1cm4gbmV4dChyZXEpO1xyXG5cclxuICAgIGNvbnN0IGZvdW5kID0gaW5GbGlnaHQuZ2V0KHJlcS51cmxXaXRoUGFyYW1zKTtcclxuXHJcbiAgICBpZiAoZm91bmQpIHJldHVybiBmb3VuZDtcclxuXHJcbiAgICBjb25zdCByZXF1ZXN0ID0gbmV4dChyZXEpLnBpcGUoXHJcbiAgICAgIGZpbmFsaXplKCgpID0+IGluRmxpZ2h0LmRlbGV0ZShyZXEudXJsV2l0aFBhcmFtcykpLFxyXG4gICAgICBzaGFyZVJlcGxheSgpLFxyXG4gICAgKTtcclxuICAgIGluRmxpZ2h0LnNldChyZXEudXJsV2l0aFBhcmFtcywgcmVxdWVzdCk7XHJcblxyXG4gICAgcmV0dXJuIHJlcXVlc3Q7XHJcbiAgfTtcclxufVxyXG4iXX0=","import { hash, keys } from '@mmstack/object';\nfunction equalTransferCache(a, b) {\n if (!a && !b)\n return true;\n if (!a || !b)\n return false;\n if (typeof a !== typeof b)\n return false;\n if (typeof a === 'boolean' || typeof b === 'boolean')\n return a === b;\n if (!a.includeHeaders && !b.includeHeaders)\n return true;\n if (!a.includeHeaders || !b.includeHeaders)\n return false;\n if (a.includeHeaders.length !== b.includeHeaders.length)\n return false;\n if (a.includeHeaders.length === 0)\n return true;\n const aSet = new Set(a.includeHeaders ?? []);\n return b.includeHeaders.every((header) => aSet.has(header));\n}\nfunction equalParams(a, b) {\n if (!a && !b)\n return true;\n if (!a || !b)\n return false;\n const aKeys = keys(a);\n const bKeys = keys(b);\n if (aKeys.length !== bKeys.length)\n return false;\n return aKeys.every((key) => a[key] === b[key]);\n}\nfunction equalBody(a, b) {\n if (!a && !b)\n return true;\n if (!a || !b)\n return false;\n return hash(a) === hash(b);\n}\nfunction equalHeaders(a, b) {\n if (!a && !b)\n return true;\n if (!a || !b)\n return false;\n const aKeys = keys(a);\n const bKeys = keys(b);\n if (aKeys.length !== bKeys.length)\n return false;\n return aKeys.every((key) => a[key] === b[key]);\n}\nfunction equalContext(a, b) {\n if (!a && !b)\n return true;\n if (!a || !b)\n return false;\n const aKeys = keys(a);\n const bKeys = keys(b);\n if (aKeys.length !== bKeys.length)\n return false;\n return aKeys.every((key) => a[key] === b[key]);\n}\nexport function createEqualRequest(equalResult) {\n const eqb = equalResult ?? equalBody;\n return (a, b) => {\n if (!a && !b)\n return true;\n if (!a || !b)\n return false;\n if (a.url !== b.url)\n return false;\n if (a.method !== b.method)\n return false;\n if (!equalParams(a.params, b.params))\n return false;\n if (!equalHeaders(a.headers, b.headers))\n return false;\n if (!eqb(a.body, b.body))\n return false;\n if (!equalContext(a.context, b.context))\n return false;\n if (a.withCredentials !== b.withCredentials)\n return false;\n if (a.reportProgress !== b.reportProgress)\n return false;\n if (!equalTransferCache(a.transferCache, b.transferCache))\n return false;\n return true;\n };\n}\n//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"equality.js","sourceRoot":"","sources":["../../../../../../packages/resource/src/lib/util/equality.ts"],"names":[],"mappings":"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","sourcesContent":["import { HttpResourceRequest } from '@angular/common/http';\r\nimport { type ValueEqualityFn } from '@angular/core';\r\nimport { hash, keys } from '@mmstack/object';\r\n\r\nfunction equalTransferCache(\r\n  a: HttpResourceRequest['transferCache'],\r\n  b: HttpResourceRequest['transferCache'],\r\n): boolean {\r\n  if (!a && !b) return true;\r\n  if (!a || !b) return false;\r\n\r\n  if (typeof a !== typeof b) return false;\r\n  if (typeof a === 'boolean' || typeof b === 'boolean') return a === b;\r\n\r\n  if (!a.includeHeaders && !b.includeHeaders) return true;\r\n  if (!a.includeHeaders || !b.includeHeaders) return false;\r\n\r\n  if (a.includeHeaders.length !== b.includeHeaders.length) return false;\r\n\r\n  if (a.includeHeaders.length === 0) return true;\r\n\r\n  const aSet = new Set(a.includeHeaders ?? []);\r\n\r\n  return b.includeHeaders.every((header) => aSet.has(header));\r\n}\r\n\r\nfunction equalParams(\r\n  a: HttpResourceRequest['params'],\r\n  b: HttpResourceRequest['params'],\r\n): boolean {\r\n  if (!a && !b) return true;\r\n  if (!a || !b) return false;\r\n  const aKeys = keys(a);\r\n  const bKeys = keys(b);\r\n  if (aKeys.length !== bKeys.length) return false;\r\n\r\n  return aKeys.every((key) => a[key] === b[key]);\r\n}\r\n\r\nfunction equalBody(\r\n  a: HttpResourceRequest['body'],\r\n  b: HttpResourceRequest['body'],\r\n): boolean {\r\n  if (!a && !b) return true;\r\n  if (!a || !b) return false;\r\n  return hash(a) === hash(b);\r\n}\r\n\r\nfunction equalHeaders(\r\n  a: HttpResourceRequest['headers'],\r\n  b: HttpResourceRequest['headers'],\r\n): boolean {\r\n  if (!a && !b) return true;\r\n  if (!a || !b) return false;\r\n\r\n  const aKeys = keys(a);\r\n  const bKeys = keys(b);\r\n  if (aKeys.length !== bKeys.length) return false;\r\n  return aKeys.every((key) => a[key] === b[key]);\r\n}\r\n\r\nfunction equalContext(\r\n  a: HttpResourceRequest['context'],\r\n  b: HttpResourceRequest['context'],\r\n): boolean {\r\n  if (!a && !b) return true;\r\n  if (!a || !b) return false;\r\n\r\n  const aKeys = keys(a);\r\n  const bKeys = keys(b);\r\n  if (aKeys.length !== bKeys.length) return false;\r\n  return aKeys.every((key) => a[key] === b[key]);\r\n}\r\n\r\nexport function createEqualRequest<TResult>(\r\n  equalResult?: ValueEqualityFn<TResult>,\r\n) {\r\n  const eqb = equalResult ?? equalBody;\r\n\r\n  return (\r\n    a: Partial<HttpResourceRequest> | undefined,\r\n    b: Partial<HttpResourceRequest> | undefined,\r\n  ) => {\r\n    if (!a && !b) return true;\r\n    if (!a || !b) return false;\r\n\r\n    if (a.url !== b.url) return false;\r\n    if (a.method !== b.method) return false;\r\n    if (!equalParams(a.params, b.params)) return false;\r\n    if (!equalHeaders(a.headers, b.headers)) return false;\r\n    if (!eqb(a.body as TResult, b.body as TResult)) return false;\r\n    if (!equalContext(a.context, b.context)) return false;\r\n\r\n    if (a.withCredentials !== b.withCredentials) return false;\r\n    if (a.reportProgress !== b.reportProgress) return false;\r\n    if (!equalTransferCache(a.transferCache, b.transferCache)) return false;\r\n\r\n    return true;\r\n  };\r\n}\r\n"]}","export function hasSlowConnection() {\n if (window &&\n 'navigator' in window &&\n 'connection' in window.navigator &&\n typeof window.navigator.connection === 'object' &&\n !!window.navigator.connection &&\n 'effectiveType' in window.navigator.connection &&\n typeof window.navigator.connection.effectiveType === 'string')\n return window.navigator.connection.effectiveType.endsWith('2g');\n return false;\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaGFzLXNsb3ctY29ubmVjdGlvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3BhY2thZ2VzL3Jlc291cmNlL3NyYy9saWIvdXRpbC9oYXMtc2xvdy1jb25uZWN0aW9uLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE1BQU0sVUFBVSxpQkFBaUI7SUFDL0IsSUFDRSxNQUFNO1FBQ04sV0FBVyxJQUFJLE1BQU07UUFDckIsWUFBWSxJQUFJLE1BQU0sQ0FBQyxTQUFTO1FBQ2hDLE9BQU8sTUFBTSxDQUFDLFNBQVMsQ0FBQyxVQUFVLEtBQUssUUFBUTtRQUMvQyxDQUFDLENBQUMsTUFBTSxDQUFDLFNBQVMsQ0FBQyxVQUFVO1FBQzdCLGVBQWUsSUFBSSxNQUFNLENBQUMsU0FBUyxDQUFDLFVBQVU7UUFDOUMsT0FBTyxNQUFNLENBQUMsU0FBUyxDQUFDLFVBQVUsQ0FBQyxhQUFhLEtBQUssUUFBUTtRQUU3RCxPQUFPLE1BQU0sQ0FBQyxTQUFTLENBQUMsVUFBVSxDQUFDLGFBQWEsQ0FBQyxRQUFRLENBQUMsSUFBSSxDQUFDLENBQUM7SUFFbEUsT0FBTyxLQUFLLENBQUM7QUFDZixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGZ1bmN0aW9uIGhhc1Nsb3dDb25uZWN0aW9uKCkge1xyXG4gIGlmIChcclxuICAgIHdpbmRvdyAmJlxyXG4gICAgJ25hdmlnYXRvcicgaW4gd2luZG93ICYmXHJcbiAgICAnY29ubmVjdGlvbicgaW4gd2luZG93Lm5hdmlnYXRvciAmJlxyXG4gICAgdHlwZW9mIHdpbmRvdy5uYXZpZ2F0b3IuY29ubmVjdGlvbiA9PT0gJ29iamVjdCcgJiZcclxuICAgICEhd2luZG93Lm5hdmlnYXRvci5jb25uZWN0aW9uICYmXHJcbiAgICAnZWZmZWN0aXZlVHlwZScgaW4gd2luZG93Lm5hdmlnYXRvci5jb25uZWN0aW9uICYmXHJcbiAgICB0eXBlb2Ygd2luZG93Lm5hdmlnYXRvci5jb25uZWN0aW9uLmVmZmVjdGl2ZVR5cGUgPT09ICdzdHJpbmcnXHJcbiAgKVxyXG4gICAgcmV0dXJuIHdpbmRvdy5uYXZpZ2F0b3IuY29ubmVjdGlvbi5lZmZlY3RpdmVUeXBlLmVuZHNXaXRoKCcyZycpO1xyXG5cclxuICByZXR1cm4gZmFsc2U7XHJcbn1cclxuIl19","import { linkedSignal, } from '@angular/core';\nfunction presist(value, usePrevious, equal) {\n // linkedSignal allows us to access previous source value\n const persisted = linkedSignal({\n source: () => {\n return {\n value: value(),\n usePrevious: usePrevious(),\n };\n },\n computation: (source, prev) => {\n if (source.usePrevious && prev)\n return prev.value;\n return source.value;\n },\n equal,\n });\n // if original value was WritableSignal then override linkedSignal methods to original...angular uses linkedSignal under the hood in ResourceImpl, this applies to that.\n if ('set' in value) {\n persisted.set = value.set;\n persisted.update = value.update;\n persisted.asReadonly = value.asReadonly;\n }\n return persisted;\n}\nexport function persistResourceValues(resource, persist = false, equal) {\n if (!persist)\n return resource;\n return {\n ...resource,\n statusCode: presist(resource.statusCode, resource.isLoading),\n headers: presist(resource.headers, resource.isLoading),\n value: presist(resource.value, resource.isLoading, equal),\n };\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGVyc2lzdC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3BhY2thZ2VzL3Jlc291cmNlL3NyYy9saWIvdXRpbC9wZXJzaXN0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUNBLE9BQU8sRUFDTCxZQUFZLEdBSWIsTUFBTSxlQUFlLENBQUM7QUFjdkIsU0FBUyxPQUFPLENBQ2QsS0FBb0MsRUFDcEMsV0FBNEIsRUFDNUIsS0FBMEI7SUFFMUIseURBQXlEO0lBRXpELE1BQU0sU0FBUyxHQUFHLFlBQVksQ0FNNUI7UUFDQSxNQUFNLEVBQUUsR0FBRyxFQUFFO1lBQ1gsT0FBTztnQkFDTCxLQUFLLEVBQUUsS0FBSyxFQUFFO2dCQUNkLFdBQVcsRUFBRSxXQUFXLEVBQUU7YUFDM0IsQ0FBQztRQUNKLENBQUM7UUFDRCxXQUFXLEVBQUUsQ0FBQyxNQUFNLEVBQUUsSUFBSSxFQUFFLEVBQUU7WUFDNUIsSUFBSSxNQUFNLENBQUMsV0FBVyxJQUFJLElBQUk7Z0JBQUUsT0FBTyxJQUFJLENBQUMsS0FBSyxDQUFDO1lBRWxELE9BQU8sTUFBTSxDQUFDLEtBQUssQ0FBQztRQUN0QixDQUFDO1FBQ0QsS0FBSztLQUNOLENBQUMsQ0FBQztJQUVILHdLQUF3SztJQUN4SyxJQUFJLEtBQUssSUFBSSxLQUFLLEVBQUUsQ0FBQztRQUNuQixTQUFTLENBQUMsR0FBRyxHQUFHLEtBQUssQ0FBQyxHQUFHLENBQUM7UUFDMUIsU0FBUyxDQUFDLE1BQU0sR0FBRyxLQUFLLENBQUMsTUFBTSxDQUFDO1FBQ2hDLFNBQVMsQ0FBQyxVQUFVLEdBQUcsS0FBSyxDQUFDLFVBQVUsQ0FBQztJQUMxQyxDQUFDO0lBRUQsT0FBTyxTQUFTLENBQUM7QUFDbkIsQ0FBQztBQUVELE1BQU0sVUFBVSxxQkFBcUIsQ0FDbkMsUUFBNEIsRUFDNUIsT0FBTyxHQUFHLEtBQUssRUFDZixLQUEwQjtJQUUxQixJQUFJLENBQUMsT0FBTztRQUFFLE9BQU8sUUFBUSxDQUFDO0lBRTlCLE9BQU87UUFDTCxHQUFHLFFBQVE7UUFDWCxVQUFVLEVBQUUsT0FBTyxDQUNqQixRQUFRLENBQUMsVUFBVSxFQUNuQixRQUFRLENBQUMsU0FBUyxDQUNuQjtRQUNELE9BQU8sRUFBRSxPQUFPLENBQ2QsUUFBUSxDQUFDLE9BQU8sRUFDaEIsUUFBUSxDQUFDLFNBQVMsQ0FDbkI7UUFDRCxLQUFLLEVBQUUsT0FBTyxDQUFJLFFBQVEsQ0FBQyxLQUFLLEVBQUUsUUFBUSxDQUFDLFNBQVMsRUFBRSxLQUFLLENBQUM7S0FDN0QsQ0FBQztBQUNKLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyB0eXBlIEh0dHBIZWFkZXJzLCB0eXBlIEh0dHBSZXNvdXJjZVJlZiB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbi9odHRwJztcclxuaW1wb3J0IHtcclxuICBsaW5rZWRTaWduYWwsXHJcbiAgdHlwZSBTaWduYWwsXHJcbiAgdHlwZSBWYWx1ZUVxdWFsaXR5Rm4sXHJcbiAgdHlwZSBXcml0YWJsZVNpZ25hbCxcclxufSBmcm9tICdAYW5ndWxhci9jb3JlJztcclxuXHJcbmZ1bmN0aW9uIHByZXNpc3Q8VD4oXHJcbiAgdmFsdWU6IFdyaXRhYmxlU2lnbmFsPFQ+LFxyXG4gIHVzZVByZXZpb3VzOiBTaWduYWw8Ym9vbGVhbj4sXHJcbiAgZXF1YWw/OiBWYWx1ZUVxdWFsaXR5Rm48VD4sXHJcbik6IFdyaXRhYmxlU2lnbmFsPFQ+O1xyXG5cclxuZnVuY3Rpb24gcHJlc2lzdDxUPihcclxuICB2YWx1ZTogU2lnbmFsPFQ+LFxyXG4gIHVzZVByZXZpb3VzOiBTaWduYWw8Ym9vbGVhbj4sXHJcbiAgZXF1YWw/OiBWYWx1ZUVxdWFsaXR5Rm48VD4sXHJcbik6IFNpZ25hbDxUPjtcclxuXHJcbmZ1bmN0aW9uIHByZXNpc3Q8VD4oXHJcbiAgdmFsdWU6IFdyaXRhYmxlU2lnbmFsPFQ+IHwgU2lnbmFsPFQ+LFxyXG4gIHVzZVByZXZpb3VzOiBTaWduYWw8Ym9vbGVhbj4sXHJcbiAgZXF1YWw/OiBWYWx1ZUVxdWFsaXR5Rm48VD4sXHJcbik6IFdyaXRhYmxlU2lnbmFsPFQ+IHwgU2lnbmFsPFQ+IHtcclxuICAvLyBsaW5rZWRTaWduYWwgYWxsb3dzIHVzIHRvIGFjY2VzcyBwcmV2aW91cyBzb3VyY2UgdmFsdWVcclxuXHJcbiAgY29uc3QgcGVyc2lzdGVkID0gbGlua2VkU2lnbmFsPFxyXG4gICAge1xyXG4gICAgICB2YWx1ZTogVDtcclxuICAgICAgdXNlUHJldmlvdXM6IGJvb2xlYW47XHJcbiAgICB9LFxyXG4gICAgVFxyXG4gID4oe1xyXG4gICAgc291cmNlOiAoKSA9PiB7XHJcbiAgICAgIHJldHVybiB7XHJcbiAgICAgICAgdmFsdWU6IHZhbHVlKCksXHJcbiAgICAgICAgdXNlUHJldmlvdXM6IHVzZVByZXZpb3VzKCksXHJcbiAgICAgIH07XHJcbiAgICB9LFxyXG4gICAgY29tcHV0YXRpb246IChzb3VyY2UsIHByZXYpID0+IHtcclxuICAgICAgaWYgKHNvdXJjZS51c2VQcmV2aW91cyAmJiBwcmV2KSByZXR1cm4gcHJldi52YWx1ZTtcclxuXHJcbiAgICAgIHJldHVybiBzb3VyY2UudmFsdWU7XHJcbiAgICB9LFxyXG4gICAgZXF1YWwsXHJcbiAgfSk7XHJcblxyXG4gIC8vIGlmIG9yaWdpbmFsIHZhbHVlIHdhcyBXcml0YWJsZVNpZ25hbCB0aGVuIG92ZXJyaWRlIGxpbmtlZFNpZ25hbCBtZXRob2RzIHRvIG9yaWdpbmFsLi4uYW5ndWxhciB1c2VzIGxpbmtlZFNpZ25hbCB1bmRlciB0aGUgaG9vZCBpbiBSZXNvdXJjZUltcGwsIHRoaXMgYXBwbGllcyB0byB0aGF0LlxyXG4gIGlmICgnc2V0JyBpbiB2YWx1ZSkge1xyXG4gICAgcGVyc2lzdGVkLnNldCA9IHZhbHVlLnNldDtcclxuICAgIHBlcnNpc3RlZC51cGRhdGUgPSB2YWx1ZS51cGRhdGU7XHJcbiAgICBwZXJzaXN0ZWQuYXNSZWFkb25seSA9IHZhbHVlLmFzUmVhZG9ubHk7XHJcbiAgfVxyXG5cclxuICByZXR1cm4gcGVyc2lzdGVkO1xyXG59XHJcblxyXG5leHBvcnQgZnVuY3Rpb24gcGVyc2lzdFJlc291cmNlVmFsdWVzPFQ+KFxyXG4gIHJlc291cmNlOiBIdHRwUmVzb3VyY2VSZWY8VD4sXHJcbiAgcGVyc2lzdCA9IGZhbHNlLFxyXG4gIGVxdWFsPzogVmFsdWVFcXVhbGl0eUZuPFQ+LFxyXG4pOiBIdHRwUmVzb3VyY2VSZWY8VD4ge1xyXG4gIGlmICghcGVyc2lzdCkgcmV0dXJuIHJlc291cmNlO1xyXG5cclxuICByZXR1cm4ge1xyXG4gICAgLi4ucmVzb3VyY2UsXHJcbiAgICBzdGF0dXNDb2RlOiBwcmVzaXN0PG51bWJlciB8IHVuZGVmaW5lZD4oXHJcbiAgICAgIHJlc291cmNlLnN0YXR1c0NvZGUsXHJcbiAgICAgIHJlc291cmNlLmlzTG9hZGluZyxcclxuICAgICksXHJcbiAgICBoZWFkZXJzOiBwcmVzaXN0PEh0dHBIZWFkZXJzIHwgdW5kZWZpbmVkPihcclxuICAgICAgcmVzb3VyY2UuaGVhZGVycyxcclxuICAgICAgcmVzb3VyY2UuaXNMb2FkaW5nLFxyXG4gICAgKSxcclxuICAgIHZhbHVlOiBwcmVzaXN0PFQ+KHJlc291cmNlLnZhbHVlLCByZXNvdXJjZS5pc0xvYWRpbmcsIGVxdWFsKSxcclxuICB9O1xyXG59XHJcbiJdfQ==","import { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { interval } from 'rxjs';\n// refresh resource every n miliseconds or don't refresh if undefined provided. 0 also excluded, due to it not being a valid usecase\nexport function refresh(resource, destroyRef, refresh) {\n if (!refresh)\n return resource; // no refresh requested\n // we can use RxJs here as reloading the resource will always be a side effect & as such does not impact the reactive graph in any way.\n let sub = interval(refresh)\n .pipe(takeUntilDestroyed(destroyRef))\n .subscribe(() => resource.reload());\n const reload = () => {\n sub.unsubscribe(); // do not conflict with manual reload\n const hasReloaded = resource.reload();\n // resubscribe after manual reload\n sub = interval(refresh)\n .pipe(takeUntilDestroyed(destroyRef))\n .subscribe(() => resource.reload());\n return hasReloaded;\n };\n return {\n ...resource,\n reload,\n destroy: () => {\n sub.unsubscribe();\n resource.destroy();\n },\n };\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVmcmVzaC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3BhY2thZ2VzL3Jlc291cmNlL3NyYy9saWIvdXRpbC9yZWZyZXNoLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUVBLE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxNQUFNLDRCQUE0QixDQUFDO0FBQ2hFLE9BQU8sRUFBRSxRQUFRLEVBQUUsTUFBTSxNQUFNLENBQUM7QUFFaEMsb0lBQW9JO0FBQ3BJLE1BQU0sVUFBVSxPQUFPLENBQ3JCLFFBQTRCLEVBQzVCLFVBQXNCLEVBQ3RCLE9BQWdCO0lBRWhCLElBQUksQ0FBQyxPQUFPO1FBQUUsT0FBTyxRQUFRLENBQUMsQ0FBQyx1QkFBdUI7SUFFdEQsdUlBQXVJO0lBQ3ZJLElBQUksR0FBRyxHQUFHLFFBQVEsQ0FBQyxPQUFPLENBQUM7U0FDeEIsSUFBSSxDQUFDLGtCQUFrQixDQUFDLFVBQVUsQ0FBQyxDQUFDO1NBQ3BDLFNBQVMsQ0FBQyxHQUFHLEVBQUUsQ0FBQyxRQUFRLENBQUMsTUFBTSxFQUFFLENBQUMsQ0FBQztJQUV0QyxNQUFNLE1BQU0sR0FBRyxHQUFZLEVBQUU7UUFDM0IsR0FBRyxDQUFDLFdBQVcsRUFBRSxDQUFDLENBQUMscUNBQXFDO1FBRXhELE1BQU0sV0FBVyxHQUFHLFFBQVEsQ0FBQyxNQUFNLEVBQUUsQ0FBQztRQUV0QyxrQ0FBa0M7UUFDbEMsR0FBRyxHQUFHLFFBQVEsQ0FBQyxPQUFPLENBQUM7YUFDcEIsSUFBSSxDQUFDLGtCQUFrQixDQUFDLFVBQVUsQ0FBQyxDQUFDO2FBQ3BDLFNBQVMsQ0FBQyxHQUFHLEVBQUUsQ0FBQyxRQUFRLENBQUMsTUFBTSxFQUFFLENBQUMsQ0FBQztRQUV0QyxPQUFPLFdBQVcsQ0FBQztJQUNyQixDQUFDLENBQUM7SUFFRixPQUFPO1FBQ0wsR0FBRyxRQUFRO1FBQ1gsTUFBTTtRQUNOLE9BQU8sRUFBRSxHQUFHLEVBQUU7WUFDWixHQUFHLENBQUMsV0FBVyxFQUFFLENBQUM7WUFDbEIsUUFBUSxDQUFDLE9BQU8sRUFBRSxDQUFDO1FBQ3JCLENBQUM7S0FDRixDQUFDO0FBQ0osQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEh0dHBSZXNvdXJjZVJlZiB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbi9odHRwJztcclxuaW1wb3J0IHsgRGVzdHJveVJlZiB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xyXG5pbXBvcnQgeyB0YWtlVW50aWxEZXN0cm95ZWQgfSBmcm9tICdAYW5ndWxhci9jb3JlL3J4anMtaW50ZXJvcCc7XHJcbmltcG9ydCB7IGludGVydmFsIH0gZnJvbSAncnhqcyc7XHJcblxyXG4vLyByZWZyZXNoIHJlc291cmNlIGV2ZXJ5IG4gbWlsaXNlY29uZHMgb3IgZG9uJ3QgcmVmcmVzaCBpZiB1bmRlZmluZWQgcHJvdmlkZWQuIDAgYWxzbyBleGNsdWRlZCwgZHVlIHRvIGl0IG5vdCBiZWluZyBhIHZhbGlkIHVzZWNhc2VcclxuZXhwb3J0IGZ1bmN0aW9uIHJlZnJlc2g8VD4oXHJcbiAgcmVzb3VyY2U6IEh0dHBSZXNvdXJjZVJlZjxUPixcclxuICBkZXN0cm95UmVmOiBEZXN0cm95UmVmLFxyXG4gIHJlZnJlc2g/OiBudW1iZXIsXHJcbik6IEh0dHBSZXNvdXJjZVJlZjxUPiB7XHJcbiAgaWYgKCFyZWZyZXNoKSByZXR1cm4gcmVzb3VyY2U7IC8vIG5vIHJlZnJlc2ggcmVxdWVzdGVkXHJcblxyXG4gIC8vIHdlIGNhbiB1c2UgUnhKcyBoZXJlIGFzIHJlbG9hZGluZyB0aGUgcmVzb3VyY2Ugd2lsbCBhbHdheXMgYmUgYSBzaWRlIGVmZmVjdCAmIGFzIHN1Y2ggZG9lcyBub3QgaW1wYWN0IHRoZSByZWFjdGl2ZSBncmFwaCBpbiBhbnkgd2F5LlxyXG4gIGxldCBzdWIgPSBpbnRlcnZhbChyZWZyZXNoKVxyXG4gICAgLnBpcGUodGFrZVVudGlsRGVzdHJveWVkKGRlc3Ryb3lSZWYpKVxyXG4gICAgLnN1YnNjcmliZSgoKSA9PiByZXNvdXJjZS5yZWxvYWQoKSk7XHJcblxyXG4gIGNvbnN0IHJlbG9hZCA9ICgpOiBib29sZWFuID0+IHtcclxuICAgIHN1Yi51bnN1YnNjcmliZSgpOyAvLyBkbyBub3QgY29uZmxpY3Qgd2l0aCBtYW51YWwgcmVsb2FkXHJcblxyXG4gICAgY29uc3QgaGFzUmVsb2FkZWQgPSByZXNvdXJjZS5yZWxvYWQoKTtcclxuXHJcbiAgICAvLyByZXN1YnNjcmliZSBhZnRlciBtYW51YWwgcmVsb2FkXHJcbiAgICBzdWIgPSBpbnRlcnZhbChyZWZyZXNoKVxyXG4gICAgICAucGlwZSh0YWtlVW50aWxEZXN0cm95ZWQoZGVzdHJveVJlZikpXHJcbiAgICAgIC5zdWJzY3JpYmUoKCkgPT4gcmVzb3VyY2UucmVsb2FkKCkpO1xyXG5cclxuICAgIHJldHVybiBoYXNSZWxvYWRlZDtcclxuICB9O1xyXG5cclxuICByZXR1cm4ge1xyXG4gICAgLi4ucmVzb3VyY2UsXHJcbiAgICByZWxvYWQsXHJcbiAgICBkZXN0cm95OiAoKSA9PiB7XHJcbiAgICAgIHN1Yi51bnN1YnNjcmliZSgpO1xyXG4gICAgICByZXNvdXJjZS5kZXN0cm95KCk7XHJcbiAgICB9LFxyXG4gIH07XHJcbn1cclxuIl19","import { effect } from '@angular/core';\n// Retry on error, if number is provided it will retry that many times with exponential backoff, otherwise it will use the options provided\nexport function retryOnError(res, opt) {\n const max = opt ? (typeof opt === 'number' ? opt : (opt.max ?? 0)) : 0;\n const backoff = typeof opt === 'object' ? (opt.backoff ?? 1000) : 1000;\n let retries = 0;\n let timeout;\n const onError = () => {\n if (retries >= max)\n return;\n retries++;\n if (timeout)\n clearTimeout(timeout);\n setTimeout(() => res.reload(), retries <= 0 ? 0 : backoff * Math.pow(2, retries - 1));\n };\n const onSuccess = () => {\n if (timeout)\n clearTimeout(timeout);\n retries = 0;\n };\n const ref = effect(() => {\n switch (res.status()) {\n case 'error':\n return onError();\n case 'resolved':\n return onSuccess();\n }\n });\n return {\n ...res,\n destroy: () => {\n ref.destroy(); // cleanup on manual destroy\n res.destroy();\n },\n };\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmV0cnktb24tZXJyb3IuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9yZXNvdXJjZS9zcmMvbGliL3V0aWwvcmV0cnktb24tZXJyb3IudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQ0EsT0FBTyxFQUFFLE1BQU0sRUFBRSxNQUFNLGVBQWUsQ0FBQztBQVN2QywySUFBMkk7QUFDM0ksTUFBTSxVQUFVLFlBQVksQ0FDMUIsR0FBdUIsRUFDdkIsR0FBa0I7SUFFbEIsTUFBTSxHQUFHLEdBQUcsR0FBRyxDQUFDLENBQUMsQ0FBQyxDQUFDLE9BQU8sR0FBRyxLQUFLLFFBQVEsQ0FBQyxDQUFDLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQyxDQUFDLEdBQUcsQ0FBQyxHQUFHLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDO0lBQ3ZFLE1BQU0sT0FBTyxHQUFHLE9BQU8sR0FBRyxLQUFLLFFBQVEsQ0FBQyxDQUFDLENBQUMsQ0FBQyxHQUFHLENBQUMsT0FBTyxJQUFJLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUM7SUFFdkUsSUFBSSxPQUFPLEdBQUcsQ0FBQyxDQUFDO0lBRWhCLElBQUksT0FBa0QsQ0FBQztJQUV2RCxNQUFNLE9BQU8sR0FBRyxHQUFHLEVBQUU7UUFDbkIsSUFBSSxPQUFPLElBQUksR0FBRztZQUFFLE9BQU87UUFDM0IsT0FBTyxFQUFFLENBQUM7UUFFVixJQUFJLE9BQU87WUFBRSxZQUFZLENBQUMsT0FBTyxDQUFDLENBQUM7UUFFbkMsVUFBVSxDQUNSLEdBQUcsRUFBRSxDQUFDLEdBQUcsQ0FBQyxNQUFNLEVBQUUsRUFDbEIsT0FBTyxJQUFJLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxPQUFPLEdBQUcsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDLEVBQUUsT0FBTyxHQUFHLENBQUMsQ0FBQyxDQUN0RCxDQUFDO0lBQ0osQ0FBQyxDQUFDO0lBRUYsTUFBTSxTQUFTLEdBQUcsR0FBRyxFQUFFO1FBQ3JCLElBQUksT0FBTztZQUFFLFlBQVksQ0FBQyxPQUFPLENBQUMsQ0FBQztRQUNuQyxPQUFPLEdBQUcsQ0FBQyxDQUFDO0lBQ2QsQ0FBQyxDQUFDO0lBRUYsTUFBTSxHQUFHLEdBQUcsTUFBTSxDQUFDLEdBQUcsRUFBRTtRQUN0QixRQUFRLEdBQUcsQ0FBQyxNQUFNLEVBQUUsRUFBRSxDQUFDO1lBQ3JCLEtBQUssT0FBTztnQkFDVixPQUFPLE9BQU8sRUFBRSxDQUFDO1lBQ25CLEtBQUssVUFBVTtnQkFDYixPQUFPLFNBQVMsRUFBRSxDQUFDO1FBQ3ZCLENBQUM7SUFDSCxDQUFDLENBQUMsQ0FBQztJQUVILE9BQU87UUFDTCxHQUFHLEdBQUc7UUFDTixPQUFPLEVBQUUsR0FBRyxFQUFFO1lBQ1osR0FBRyxDQUFDLE9BQU8sRUFBRSxDQUFDLENBQUMsNEJBQTRCO1lBQzNDLEdBQUcsQ0FBQyxPQUFPLEVBQUUsQ0FBQztRQUNoQixDQUFDO0tBQ0YsQ0FBQztBQUNKLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyB0eXBlIEh0dHBSZXNvdXJjZVJlZiB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbi9odHRwJztcclxuaW1wb3J0IHsgZWZmZWN0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XHJcblxyXG5leHBvcnQgdHlwZSBSZXRyeU9wdGlvbnMgPVxyXG4gIHwgbnVtYmVyXHJcbiAgfCB7XHJcbiAgICAgIG1heD86IG51bWJlcjtcclxuICAgICAgYmFja29mZj86IG51bWJlcjtcclxuICAgIH07XHJcblxyXG4vLyBSZXRyeSBvbiBlcnJvciwgaWYgbnVtYmVyIGlzIHByb3ZpZGVkIGl0IHdpbGwgcmV0cnkgdGhhdCBtYW55IHRpbWVzIHdpdGggZXhwb25lbnRpYWwgYmFja29mZiwgb3RoZXJ3aXNlIGl0IHdpbGwgdXNlIHRoZSBvcHRpb25zIHByb3ZpZGVkXHJcbmV4cG9ydCBmdW5jdGlvbiByZXRyeU9uRXJyb3I8VD4oXHJcbiAgcmVzOiBIdHRwUmVzb3VyY2VSZWY8VD4sXHJcbiAgb3B0PzogUmV0cnlPcHRpb25zLFxyXG4pOiBIdHRwUmVzb3VyY2VSZWY8VD4ge1xyXG4gIGNvbnN0IG1heCA9IG9wdCA/ICh0eXBlb2Ygb3B0ID09PSAnbnVtYmVyJyA/IG9wdCA6IChvcHQubWF4ID8/IDApKSA6IDA7XHJcbiAgY29uc3QgYmFja29mZiA9IHR5cGVvZiBvcHQgPT09ICdvYmplY3QnID8gKG9wdC5iYWNrb2ZmID8/IDEwMDApIDogMTAwMDtcclxuXHJcbiAgbGV0IHJldHJpZXMgPSAwO1xyXG5cclxuICBsZXQgdGltZW91dDogUmV0dXJuVHlwZTx0eXBlb2Ygc2V0VGltZW91dD4gfCB1bmRlZmluZWQ7XHJcblxyXG4gIGNvbnN0IG9uRXJyb3IgPSAoKSA9PiB7XHJcbiAgICBpZiAocmV0cmllcyA+PSBtYXgpIHJldHVybjtcclxuICAgIHJldHJpZXMrKztcclxuXHJcbiAgICBpZiAodGltZW91dCkgY2xlYXJUaW1lb3V0KHRpbWVvdXQpO1xyXG5cclxuICAgIHNldFRpbWVvdXQoXHJcbiAgICAgICgpID0+IHJlcy5yZWxvYWQoKSxcclxuICAgICAgcmV0cmllcyA8PSAwID8gMCA6IGJhY2tvZmYgKiBNYXRoLnBvdygyLCByZXRyaWVzIC0gMSksXHJcbiAgICApO1xyXG4gIH07XHJcblxyXG4gIGNvbnN0IG9uU3VjY2VzcyA9ICgpID0+IHtcclxuICAgIGlmICh0aW1lb3V0KSBjbGVhclRpbWVvdXQodGltZW91dCk7XHJcbiAgICByZXRyaWVzID0gMDtcclxuICB9O1xyXG5cclxuICBjb25zdCByZWYgPSBlZmZlY3QoKCkgPT4ge1xyXG4gICAgc3dpdGNoIChyZXMuc3RhdHVzKCkpIHtcclxuICAgICAgY2FzZSAnZXJyb3InOlxyXG4gICAgICAgIHJldHVybiBvbkVycm9yKCk7XHJcbiAgICAgIGNhc2UgJ3Jlc29sdmVkJzpcclxuICAgICAgICByZXR1cm4gb25TdWNjZXNzKCk7XHJcbiAgICB9XHJcbiAgfSk7XHJcblxyXG4gIHJldHVybiB7XHJcbiAgICAuLi5yZXMsXHJcbiAgICBkZXN0cm95OiAoKSA9PiB7XHJcbiAgICAgIHJlZi5kZXN0cm95KCk7IC8vIGNsZWFudXAgb24gbWFudWFsIGRlc3Ryb3lcclxuICAgICAgcmVzLmRlc3Ryb3koKTtcclxuICAgIH0sXHJcbiAgfTtcclxufVxyXG4iXX0=","import { HttpParams } from '@angular/common/http';\nimport { entries } from '@mmstack/object';\nfunction normalizeParams(params) {\n if (params instanceof HttpParams)\n return params.toString();\n const paramMap = new Map();\n for (const [key, value] of entries(params)) {\n if (Array.isArray(value)) {\n paramMap.set(key, value.map(encodeURIComponent).join(','));\n }\n else {\n paramMap.set(key, encodeURIComponent(value.toString()));\n }\n }\n return Array.from(paramMap.entries())\n .map(([key, value]) => `${key}=${value}`)\n .join('&');\n}\nexport function urlWithParams(req) {\n if (!req.params)\n return req.url;\n return `${req.url}?${normalizeParams(req.params)}`;\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXJsLXdpdGgtcGFyYW1zLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcGFja2FnZXMvcmVzb3VyY2Uvc3JjL2xpYi91dGlsL3VybC13aXRoLXBhcmFtcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsVUFBVSxFQUF1QixNQUFNLHNCQUFzQixDQUFDO0FBQ3ZFLE9BQU8sRUFBRSxPQUFPLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUUxQyxTQUFTLGVBQWUsQ0FDdEIsTUFBK0M7SUFFL0MsSUFBSSxNQUFNLFlBQVksVUFBVTtRQUFFLE9BQU8sTUFBTSxDQUFDLFFBQVEsRUFBRSxDQUFDO0lBRTNELE1BQU0sUUFBUSxHQUFHLElBQUksR0FBRyxFQUFrQixDQUFDO0lBRTNDLEtBQUssTUFBTSxDQUFDLEdBQUcsRUFBRSxLQUFLLENBQUMsSUFBSSxPQUFPLENBQUMsTUFBTSxDQUFDLEVBQUUsQ0FBQztRQUMzQyxJQUFJLEtBQUssQ0FBQyxPQUFPLENBQUMsS0FBSyxDQUFDLEVBQUUsQ0FBQztZQUN6QixRQUFRLENBQUMsR0FBRyxDQUFDLEdBQUcsRUFBRSxLQUFLLENBQUMsR0FBRyxDQUFDLGtCQUFrQixDQUFDLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUM7UUFDN0QsQ0FBQzthQUFNLENBQUM7WUFDTixRQUFRLENBQUMsR0FBRyxDQUFDLEdBQUcsRUFBRSxrQkFBa0IsQ0FBQyxLQUFLLENBQUMsUUFBUSxFQUFFLENBQUMsQ0FBQyxDQUFDO1FBQzFELENBQUM7SUFDSCxDQUFDO0lBRUQsT0FBTyxLQUFLLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxPQUFPLEVBQUUsQ0FBQztTQUNsQyxHQUFHLENBQUMsQ0FBQyxDQUFDLEdBQUcsRUFBRSxLQUFLLENBQUMsRUFBRSxFQUFFLENBQUMsR0FBRyxHQUFHLElBQUksS0FBSyxFQUFFLENBQUM7U0FDeEMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDO0FBQ2YsQ0FBQztBQUVELE1BQU0sVUFBVSxhQUFhLENBQUMsR0FBd0I7SUFDcEQsSUFBSSxDQUFDLEdBQUcsQ0FBQyxNQUFNO1FBQUUsT0FBTyxHQUFHLENBQUMsR0FBRyxDQUFDO0lBRWhDLE9BQU8sR0FBRyxHQUFHLENBQUMsR0FBRyxJQUFJLGVBQWUsQ0FBQyxHQUFHLENBQUMsTUFBTSxDQUFDLEVBQUUsQ0FBQztBQUNyRCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSHR0cFBhcmFtcywgSHR0cFJlc291cmNlUmVxdWVzdCB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbi9odHRwJztcclxuaW1wb3J0IHsgZW50cmllcyB9IGZyb20gJ0BtbXN0YWNrL29iamVjdCc7XHJcblxyXG5mdW5jdGlvbiBub3JtYWxpemVQYXJhbXMoXHJcbiAgcGFyYW1zOiBSZXF1aXJlZDxIdHRwUmVzb3VyY2VSZXF1ZXN0PlsncGFyYW1zJ10sXHJcbik6IHN0cmluZyB7XHJcbiAgaWYgKHBhcmFtcyBpbnN0YW5jZW9mIEh0dHBQYXJhbXMpIHJldHVybiBwYXJhbXMudG9TdHJpbmcoKTtcclxuXHJcbiAgY29uc3QgcGFyYW1NYXAgPSBuZXcgTWFwPHN0cmluZywgc3RyaW5nPigpO1xyXG5cclxuICBmb3IgKGNvbnN0IFtrZXksIHZhbHVlXSBvZiBlbnRyaWVzKHBhcmFtcykpIHtcclxuICAgIGlmIChBcnJheS5pc0FycmF5KHZhbHVlKSkge1xyXG4gICAgICBwYXJhbU1hcC5zZXQoa2V5LCB2YWx1ZS5tYXAoZW5jb2RlVVJJQ29tcG9uZW50KS5qb2luKCcsJykpO1xyXG4gICAgfSBlbHNlIHtcclxuICAgICAgcGFyYW1NYXAuc2V0KGtleSwgZW5jb2RlVVJJQ29tcG9uZW50KHZhbHVlLnRvU3RyaW5nKCkpKTtcclxuICAgIH1cclxuICB9XHJcblxyXG4gIHJldHVybiBBcnJheS5mcm9tKHBhcmFtTWFwLmVudHJpZXMoKSlcclxuICAgIC5tYXAoKFtrZXksIHZhbHVlXSkgPT4gYCR7a2V5fT0ke3ZhbHVlfWApXHJcbiAgICAuam9pbignJicpO1xyXG59XHJcblxyXG5leHBvcnQgZnVuY3Rpb24gdXJsV2l0aFBhcmFtcyhyZXE6IEh0dHBSZXNvdXJjZVJlcXVlc3QpOiBzdHJpbmcge1xyXG4gIGlmICghcmVxLnBhcmFtcykgcmV0dXJuIHJlcS51cmw7XHJcblxyXG4gIHJldHVybiBgJHtyZXEudXJsfT8ke25vcm1hbGl6ZVBhcmFtcyhyZXEucGFyYW1zKX1gO1xyXG59XHJcbiJdfQ==","import { HttpClient, httpResource, HttpResponse, } from '@angular/common/http';\nimport { computed, DestroyRef, effect, inject, isDevMode, linkedSignal, untracked, } from '@angular/core';\nimport { toWritable } from '@mmstack/primitives';\nimport { firstValueFrom } from 'rxjs';\nimport { createCircuitBreaker, createEqualRequest, hasSlowConnection, injectQueryCache, persistResourceValues, refresh, retryOnError, setCacheContext, urlWithParams, } from './util';\nexport function queryResource(request, options) {\n const cache = injectQueryCache(options?.injector);\n const destroyRef = options?.injector\n ? options.injector.get(DestroyRef)\n : inject(DestroyRef);\n const cb = createCircuitBreaker(options?.circuitBreaker === true\n ? undefined\n : (options?.circuitBreaker ?? false));\n const stableRequest = computed(() => {\n if (cb.isClosed())\n return undefined;\n return request() ?? undefined;\n }, {\n equal: createEqualRequest(options?.equal),\n });\n const hashFn = typeof options?.cache === 'object'\n ? (options.cache.hash ?? urlWithParams)\n : urlWithParams;\n const staleTime = typeof options?.cache === 'object' ? options.cache.staleTime : 0;\n const ttl = typeof options?.cache === 'object' ? options.cache.ttl : undefined;\n const cacheKey = computed(() => {\n const r = stableRequest();\n if (!r)\n return null;\n return hashFn(r);\n });\n const cachedRequest = options?.cache\n ? computed(() => {\n const r = stableRequest();\n if (!r)\n return r;\n return {\n ...r,\n context: setCacheContext(r.context, {\n staleTime,\n ttl,\n key: cacheKey() ?? hashFn(r),\n }),\n };\n })\n : stableRequest;\n let resource = httpResource(cachedRequest, {\n ...options,\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n parse: options?.parse, // Not my favorite thing to do, but here it is completely safe.\n });\n // get full HttpResonse from Cache\n const cachedEvent = cache.get(cacheKey);\n const parse = options?.parse ?? ((val) => val);\n const actualCacheValue = computed(() => {\n const ce = cachedEvent();\n if (!ce || !(ce.value instanceof HttpResponse))\n return;\n return parse(ce.value.body);\n });\n // retains last cache value after it is invalidated for lifetime of resource\n const cachedValue = linkedSignal({\n source: () => actualCacheValue(),\n computation: (source, prev) => {\n if (!source && prev)\n return prev.value;\n return source;\n },\n });\n resource = refresh(resource, destroyRef, options?.refresh);\n resource = retryOnError(resource, options?.retry);\n resource = persistResourceValues(resource, options?.keepPrevious, options?.equal);\n const value = options?.cache\n ? toWritable(computed(() => {\n return cachedValue() ?? resource.value();\n }), resource.value.set, resource.value.update)\n : resource.value;\n const onError = options?.onError; // Put in own variable to ensure value remains even if options are somehow mutated in-line\n if (onError) {\n const onErrorRef = effect(() => {\n const err = resource.error();\n if (err)\n onError(err);\n });\n // cleanup on manual destroy, I'm comfortable setting these props in-line as we have yet to 'release' the object out of this lexical scope\n const destroyRest = resource.destroy;\n resource.destroy = () => {\n onErrorRef.destroy();\n destroyRest();\n };\n }\n // iterate circuit breaker state, is effect as a computed would cause a circular dependency (resource -> cb -> resource)\n const cbEffectRef = effect(() => {\n const status = resource.status();\n if (status === 'error')\n cb.fail();\n else if (status === 'resolved')\n cb.success();\n });\n const set = (value) => {\n resource.value.set(value);\n const k = untracked(cacheKey);\n if (options?.cache && k)\n cache.store(k, new HttpResponse({\n body: value,\n status: 200,\n statusText: 'OK',\n }));\n };\n const update = (updater) => {\n set(updater(untracked(resource.value)));\n };\n const client = options?.injector\n ? options.injector.get(HttpClient)\n : inject(HttpClient);\n return {\n ...resource,\n value,\n set,\n update,\n disabled: computed(() => cb.isClosed() || stableRequest() === undefined),\n reload: () => {\n cb.halfOpen(); // open the circuit for manual reload\n return resource.reload();\n },\n destroy: () => {\n cbEffectRef.destroy();\n cb.destroy();\n resource.destroy();\n },\n prefetch: async (partial) => {\n if (!options?.cache || hasSlowConnection())\n return Promise.resolve();\n const request = untracked(cachedRequest);\n if (!request)\n return Promise.resolve();\n const prefetchRequest = {\n ...request,\n ...partial,\n };\n try {\n await firstValueFrom(client.request(prefetchRequest.method ?? 'GET', prefetchRequest.url, {\n ...prefetchRequest,\n headers: prefetchRequest.headers,\n observe: 'response',\n }));\n return;\n }\n catch (err) {\n if (isDevMode())\n console.error('Prefetch failed: ', err);\n return;\n }\n },\n };\n}\n//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"query-resource.js","sourceRoot":"","sources":["../../../../../packages/resource/src/lib/query-resource.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  HttpClient,\r\n  HttpHeaders,\r\n  httpResource,\r\n  HttpResponse,\r\n  type HttpResourceOptions,\r\n  type HttpResourceRef,\r\n  type HttpResourceRequest,\r\n} from '@angular/common/http';\r\nimport {\r\n  computed,\r\n  DestroyRef,\r\n  effect,\r\n  inject,\r\n  isDevMode,\r\n  linkedSignal,\r\n  Signal,\r\n  untracked,\r\n} from '@angular/core';\r\nimport { toWritable } from '@mmstack/primitives';\r\nimport { firstValueFrom } from 'rxjs';\r\nimport {\r\n  CircuitBreakerOptions,\r\n  createCircuitBreaker,\r\n  createEqualRequest,\r\n  hasSlowConnection,\r\n  injectQueryCache,\r\n  persistResourceValues,\r\n  refresh,\r\n  retryOnError,\r\n  setCacheContext,\r\n  urlWithParams,\r\n  type RetryOptions,\r\n} from './util';\r\n\r\n/**\r\n * Options for configuring caching behavior of a `queryResource`.\r\n * - `true`: Enables caching with default settings.\r\n * - `{ ttl?: number; staleTime?: number; hash?: (req: HttpResourceRequest) => string; }`:  Configures caching with custom settings.\r\n */\r\ntype ResourceCacheOptions =\r\n  | true\r\n  | {\r\n      /**\r\n       * The Time To Live (TTL) for the cached data, in milliseconds. After this time, the cached data is\r\n       * considered expired and will be refetched.\r\n       */\r\n      ttl?: number;\r\n      /**\r\n       * The duration, in milliseconds, during which stale data can be served while a revalidation request\r\n       * is made in the background.\r\n       */\r\n      staleTime?: number;\r\n      /**\r\n       * A custom function to generate the cache key. Defaults to using the request URL with parameters.\r\n       * Provide a custom hash function if you need more control over how cache keys are generated,\r\n       * for instance, to ignore certain query parameters or to use request body for the cache key.\r\n       */\r\n      hash?: (req: HttpResourceRequest) => string;\r\n    };\r\n\r\n/**\r\n * Options for configuring a `queryResource`.\r\n */\r\nexport type QueryResourceOptions<TResult, TRaw = TResult> = HttpResourceOptions<\r\n  TResult,\r\n  TRaw\r\n> & {\r\n  /**\r\n   * Whether to keep the previous value of the resource while a refresh is in progress.\r\n   * Defaults to `false`. Also keeps status & headers while refreshing.\r\n   */\r\n  keepPrevious?: boolean;\r\n  /**\r\n   * The refresh interval, in milliseconds. If provided, the resource will automatically\r\n   * refresh its data at the specified interval.\r\n   */\r\n  refresh?: number;\r\n  /**\r\n   * Options for retrying failed requests.\r\n   */\r\n  retry?: RetryOptions;\r\n  /**\r\n   * An optional error handler callback.  This function will be called whenever the\r\n   * underlying HTTP request fails. Useful for displaying toasts or other error messages.\r\n   */\r\n  onError?: (err: unknown) => void;\r\n  /**\r\n   * Options for configuring a circuit breaker for the resource.\r\n   */\r\n  circuitBreaker?: CircuitBreakerOptions | true;\r\n  /**\r\n   * Options for enabling and configuring caching for the resource.\r\n   */\r\n  cache?: ResourceCacheOptions;\r\n};\r\n\r\n/**\r\n * Represents a resource created by `queryResource`. Extends `HttpResourceRef` with additional properties.\r\n */\r\nexport type QueryResourceRef<TResult> = HttpResourceRef<TResult> & {\r\n  /**\r\n   * A signal indicating whether the resource is currently disabled (due to circuit breaker or undefined request).\r\n   */\r\n  disabled: Signal<boolean>;\r\n  /**\r\n   * Prefetches data for the resource, populating the cache if caching is enabled.  This can be\r\n   * used to proactively load data before it's needed.  If a slow connection is detected, prefetching is skipped.\r\n   *\r\n   * @param req - Optional partial request parameters to use for the prefetch.  This allows you\r\n   *              to prefetch data with different parameters than the main resource request.\r\n   */\r\n  prefetch: (req?: Partial<HttpResourceRequest>) => Promise<void>;\r\n};\r\n\r\nexport function queryResource<TResult, TRaw = TResult>(\r\n  request: () => HttpResourceRequest | undefined | void,\r\n  options: QueryResourceOptions<TResult, TRaw> & {\r\n    defaultValue: NoInfer<TResult>;\r\n  },\r\n): QueryResourceRef<TResult>;\r\n\r\n/**\r\n * Creates an extended HTTP resource with features like caching, retries, refresh intervals,\r\n * circuit breaker, and optimistic updates. Without additional options it is equivalent to simply calling `httpResource`.\r\n *\r\n * @param request A function that returns the `HttpResourceRequest` to be made.  This function\r\n *                is called reactively, so the request can change over time.  If the function\r\n *                returns `undefined`, the resource is considered \"disabled\" and no request will be made.\r\n * @param options Configuration options for the resource.  These options extend the basic\r\n *                `HttpResourceOptions` and add features like `keepPrevious`, `refresh`, `retry`,\r\n *                `onError`, `circuitBreaker`, and `cache`.\r\n * @returns An `QueryResourceRef` instance, which extends the basic `HttpResourceRef` with additional features.\r\n */\r\nexport function queryResource<TResult, TRaw = TResult>(\r\n  request: () => HttpResourceRequest | undefined | void,\r\n  options?: QueryResourceOptions<TResult, TRaw>,\r\n): QueryResourceRef<TResult | undefined>;\r\n\r\nexport function queryResource<TResult, TRaw = TResult>(\r\n  request: () => HttpResourceRequest | undefined | void,\r\n  options?: QueryResourceOptions<TResult, TRaw>,\r\n): QueryResourceRef<TResult | undefined> {\r\n  const cache = injectQueryCache(options?.injector);\r\n\r\n  const destroyRef = options?.injector\r\n    ? options.injector.get(DestroyRef)\r\n    : inject(DestroyRef);\r\n\r\n  const cb = createCircuitBreaker(\r\n    options?.circuitBreaker === true\r\n      ? undefined\r\n      : (options?.circuitBreaker ?? false),\r\n  );\r\n\r\n  const stableRequest = computed(\r\n    () => {\r\n      if (cb.isClosed()) return undefined;\r\n      return request() ?? undefined;\r\n    },\r\n    {\r\n      equal: createEqualRequest(options?.equal),\r\n    },\r\n  );\r\n\r\n  const hashFn =\r\n    typeof options?.cache === 'object'\r\n      ? (options.cache.hash ?? urlWithParams)\r\n      : urlWithParams;\r\n\r\n  const staleTime =\r\n    typeof options?.cache === 'object' ? options.cache.staleTime : 0;\r\n  const ttl =\r\n    typeof options?.cache === 'object' ? options.cache.ttl : undefined;\r\n\r\n  const cacheKey = computed(() => {\r\n    const r = stableRequest();\r\n    if (!r) return null;\r\n    return hashFn(r);\r\n  });\r\n\r\n  const cachedRequest = options?.cache\r\n    ? computed(() => {\r\n        const r = stableRequest();\r\n        if (!r) return r;\r\n\r\n        return {\r\n          ...r,\r\n          context: setCacheContext(r.context, {\r\n            staleTime,\r\n            ttl,\r\n            key: cacheKey() ?? hashFn(r),\r\n          }),\r\n        };\r\n      })\r\n    : stableRequest;\r\n\r\n  let resource = httpResource<TResult>(cachedRequest, {\r\n    ...options,\r\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\r\n    parse: options?.parse as any, // Not my favorite thing to do, but here it is completely safe.\r\n  }) as HttpResourceRef<TResult>;\r\n\r\n  // get full HttpResonse from Cache\r\n  const cachedEvent = cache.get(cacheKey);\r\n\r\n  const parse = options?.parse ?? ((val: TRaw) => val as unknown as TResult);\r\n\r\n  const actualCacheValue = computed((): TResult | undefined => {\r\n    const ce = cachedEvent();\r\n    if (!ce || !(ce.value instanceof HttpResponse)) return;\r\n    return parse(ce.value.body as TRaw);\r\n  });\r\n\r\n  // retains last cache value after it is invalidated for lifetime of resource\r\n  const cachedValue = linkedSignal<TResult | undefined, TResult | undefined>({\r\n    source: () => actualCacheValue(),\r\n    computation: (source, prev) => {\r\n      if (!source && prev) return prev.value;\r\n      return source;\r\n    },\r\n  });\r\n\r\n  resource = refresh(resource, destroyRef, options?.refresh);\r\n  resource = retryOnError(resource, options?.retry);\r\n\r\n  resource = persistResourceValues<TResult>(\r\n    resource,\r\n    options?.keepPrevious,\r\n    options?.equal,\r\n  );\r\n\r\n  const value = options?.cache\r\n    ? toWritable(\r\n        computed((): TResult => {\r\n          return cachedValue() ?? resource.value();\r\n        }),\r\n        resource.value.set,\r\n        resource.value.update,\r\n      )\r\n    : resource.value;\r\n\r\n  const onError = options?.onError; // Put in own variable to ensure value remains even if options are somehow mutated in-line\r\n  if (onError) {\r\n    const onErrorRef = effect(() => {\r\n      const err = resource.error();\r\n      if (err) onError(err);\r\n    });\r\n\r\n    // cleanup on manual destroy, I'm comfortable setting these props in-line as we have yet to 'release' the object out of this lexical scope\r\n    const destroyRest = resource.destroy;\r\n    resource.destroy = () => {\r\n      onErrorRef.destroy();\r\n      destroyRest();\r\n    };\r\n  }\r\n\r\n  // iterate circuit breaker state, is effect as a computed would cause a circular dependency (resource -> cb -> resource)\r\n  const cbEffectRef = effect(() => {\r\n    const status = resource.status();\r\n    if (status === 'error') cb.fail();\r\n    else if (status === 'resolved') cb.success();\r\n  });\r\n\r\n  const set = (value: TResult) => {\r\n    resource.value.set(value);\r\n    const k = untracked(cacheKey);\r\n    if (options?.cache && k)\r\n      cache.store(\r\n        k,\r\n        new HttpResponse({\r\n          body: value,\r\n          status: 200,\r\n          statusText: 'OK',\r\n        }),\r\n      );\r\n  };\r\n\r\n  const update = (updater: (value: TResult) => TResult) => {\r\n    set(updater(untracked(resource.value)));\r\n  };\r\n\r\n  const client = options?.injector\r\n    ? options.injector.get(HttpClient)\r\n    : inject(HttpClient);\r\n\r\n  return {\r\n    ...resource,\r\n    value,\r\n    set,\r\n    update,\r\n    disabled: computed(() => cb.isClosed() || stableRequest() === undefined),\r\n    reload: () => {\r\n      cb.halfOpen(); // open the circuit for manual reload\r\n      return resource.reload();\r\n    },\r\n    destroy: () => {\r\n      cbEffectRef.destroy();\r\n      cb.destroy();\r\n      resource.destroy();\r\n    },\r\n    prefetch: async (partial) => {\r\n      if (!options?.cache || hasSlowConnection()) return Promise.resolve();\r\n\r\n      const request = untracked(cachedRequest);\r\n      if (!request) return Promise.resolve();\r\n\r\n      const prefetchRequest = {\r\n        ...request,\r\n        ...partial,\r\n      };\r\n\r\n      try {\r\n        await firstValueFrom(\r\n          client.request<TRaw>(\r\n            prefetchRequest.method ?? 'GET',\r\n            prefetchRequest.url,\r\n            {\r\n              ...prefetchRequest,\r\n              headers: prefetchRequest.headers as HttpHeaders,\r\n              observe: 'response',\r\n            },\r\n          ),\r\n        );\r\n\r\n        return;\r\n      } catch (err) {\r\n        if (isDevMode()) console.error('Prefetch failed: ', err);\r\n        return;\r\n      }\r\n    },\r\n  };\r\n}\r\n"]}","import { computed, DestroyRef, inject, signal, } from '@angular/core';\nimport { takeUntilDestroyed, toObservable } from '@angular/core/rxjs-interop';\nimport { combineLatestWith, filter, map } from 'rxjs';\nimport { queryResource, } from './query-resource';\nimport { createEqualRequest } from './util';\n/**\n * Creates a resource for performing mutations (e.g., POST, PUT, PATCH, DELETE requests).\n * Unlike `queryResource`, `mutationResource` is designed for one-off operations that change data.\n * It does *not* cache responses and does not provide a `value` signal. Instead, it focuses on\n * managing the mutation lifecycle (pending, error, success) and provides callbacks for handling\n * these states.\n *\n * @param request A function that returns the base `HttpResourceRequest` to be made. This\n * function is called reactively. Unlike `queryResource`, the `body` property\n * of the request is provided when `mutate` is called, *not* here. If the\n * function returns `undefined`, the mutation is considered \"disabled.\" All properties,\n * except the body, can be set here.\n * @param options Configuration options for the mutation resource. This includes callbacks\n * for `onMutate`, `onError`, `onSuccess`, and `onSettled`.\n * @typeParam TResult - The type of the expected result from the mutation.\n * @typeParam TRaw - The raw response type from the HTTP request (defaults to TResult).\n * @typeParam TCTX - The type of the context value returned by `onMutate`.\n * @returns A `MutationResourceRef` instance, which provides methods for triggering the mutation\n * and observing its status.\n */\nexport function mutationResource(request, options = {}) {\n const { onMutate, onError, onSuccess, onSettled, equal, ...rest } = options;\n const requestEqual = createEqualRequest(equal);\n const baseRequest = computed(() => request() ?? undefined, {\n equal: requestEqual,\n });\n const nextRequest = signal(null, {\n equal: (a, b) => {\n if (!a && !b)\n return true;\n if (!a || !b)\n return false;\n return requestEqual(a, b);\n },\n });\n const req = computed(() => {\n const nr = nextRequest();\n if (!nr)\n return;\n const base = baseRequest();\n const url = nr.url ?? base?.url;\n if (!url)\n return;\n const method = nr.method ?? base?.method;\n return {\n ...base,\n ...nr,\n url,\n method,\n };\n });\n const resource = queryResource(req, {\n ...rest,\n defaultValue: null, // doesnt matter since .value is not accessible\n });\n let ctx = undefined;\n const destroyRef = options.injector\n ? options.injector.get(DestroyRef)\n : inject(DestroyRef);\n const error$ = toObservable(resource.error);\n const value$ = toObservable(resource.value);\n const statusSub = toObservable(resource.status)\n .pipe(combineLatestWith(error$, value$), map(([status, error, value]) => {\n if (status === 'error' && error) {\n return {\n status: 'error',\n error,\n };\n }\n if (status === 'resolved' && value !== null) {\n return {\n status: 'resolved',\n value,\n };\n }\n return null;\n }), filter((v) => v !== null), takeUntilDestroyed(destroyRef))\n .subscribe((result) => {\n if (result.status === 'error')\n onError?.(result.error, ctx);\n else\n onSuccess?.(result.value, ctx);\n onSettled?.(ctx);\n ctx = undefined;\n nextRequest.set(null);\n });\n return {\n ...resource,\n destroy: () => {\n statusSub.unsubscribe();\n resource.destroy();\n },\n mutate: (value, ictx) => {\n ctx = onMutate?.(value.body, ictx);\n nextRequest.set(value);\n },\n current: nextRequest,\n };\n}\n//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"mutation-resource.js","sourceRoot":"","sources":["../../../../../packages/resource/src/lib/mutation-resource.ts"],"names":[],"mappings":"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","sourcesContent":["import { type HttpResourceRequest } from '@angular/common/http';\r\nimport {\r\n  computed,\r\n  DestroyRef,\r\n  inject,\r\n  Signal,\r\n  signal,\r\n  ValueEqualityFn,\r\n} from '@angular/core';\r\nimport { takeUntilDestroyed, toObservable } from '@angular/core/rxjs-interop';\r\nimport { combineLatestWith, filter, map } from 'rxjs';\r\nimport {\r\n  queryResource,\r\n  type QueryResourceOptions,\r\n  type QueryResourceRef,\r\n} from './query-resource';\r\nimport { createEqualRequest } from './util';\r\n\r\n/**\r\n * @internal\r\n * Helper type for inferring the request body type based on the HTTP method.\r\n */\r\ntype NextRequest<\r\n  TMethod extends HttpResourceRequest['method'],\r\n  TMutation,\r\n> = TMethod extends 'DELETE' | 'delete'\r\n  ? Omit<HttpResourceRequest, 'body' | 'method'> & { method?: TMethod }\r\n  : Omit<HttpResourceRequest, 'body' | 'method'> & {\r\n      body: TMutation;\r\n      method?: TMethod;\r\n    };\r\n\r\n/**\r\n * @internal\r\n * Helper type for tracking mutation status.\r\n */\r\ntype StatusResult<TResult> =\r\n  | {\r\n      status: 'error';\r\n      error: unknown;\r\n    }\r\n  | {\r\n      status: 'resolved';\r\n      value: TResult;\r\n    };\r\n\r\n/**\r\n * Options for configuring a `mutationResource`.\r\n *\r\n * @typeParam TResult - The type of the expected result from the mutation.\r\n * @typeParam TRaw - The raw response type from the HTTP request (defaults to TResult).\r\n * @typeParam TCTX - The type of the context value returned by `onMutate`.\r\n */\r\nexport type MutationResourceOptions<\r\n  TResult,\r\n  TRaw = TResult,\r\n  TMutation = TResult,\r\n  TCTX = void,\r\n  TICTX = TCTX,\r\n> = Omit<\r\n  QueryResourceOptions<TResult, TRaw>,\r\n  'equal' | 'onError' | 'keepPrevious' | 'refresh' | 'cache' // we can't keep previous values, refresh or cache mutations as they are meant to be one-off operations\r\n> & {\r\n  /**\r\n   * A callback function that is called before the mutation request is made.\r\n   * @param value The value being mutated (the `body` of the request).\r\n   * @returns An optional context value that will be passed to the `onError`, `onSuccess`, and `onSettled` callbacks. This is useful for storing\r\n   *  information needed during the mutation lifecycle, such as previous values for optimistic updates or rollback.\r\n   */\r\n  onMutate?: (value: TMutation, initialCTX?: TICTX) => TCTX;\r\n  /**\r\n   * A callback function that is called if the mutation request fails.\r\n   * @param error The error that occurred.\r\n   * @param ctx The context value returned by the `onMutate` callback (or `undefined` if `onMutate` was not provided or returned `undefined`).\r\n   */\r\n  onError?: (error: unknown, ctx: NoInfer<TCTX>) => void;\r\n  /**\r\n   * A callback function that is called if the mutation request succeeds.\r\n   * @param value The result of the mutation (the parsed response body).\r\n   * @param ctx The context value returned by the `onMutate` callback (or `undefined` if `onMutate` was not provided or returned `undefined`).\r\n   */\r\n  onSuccess?: (value: NoInfer<TResult>, ctx: NoInfer<TCTX>) => void;\r\n  /**\r\n   * A callback function that is called when the mutation request settles (either succeeds or fails).\r\n   * @param ctx The context value returned by the `onMutate` callback (or `undefined` if `onMutate` was not provided or returned `undefined`).\r\n   */\r\n  onSettled?: (ctx: NoInfer<TCTX>) => void;\r\n  equal?: ValueEqualityFn<TMutation>;\r\n};\r\n\r\n/**\r\n * Represents a mutation resource created by `mutationResource`.  Extends `QueryResourceRef`\r\n * but removes methods that don't make sense for mutations (like `prefetch`, `value`, etc.).\r\n *\r\n * @typeParam TResult - The type of the expected result from the mutation.\r\n */\r\nexport type MutationResourceRef<\r\n  TResult,\r\n  TMutation = TResult,\r\n  TICTX = void,\r\n  TMethod extends HttpResourceRequest['method'] = HttpResourceRequest['method'],\r\n> = Omit<\r\n  QueryResourceRef<TResult>,\r\n  'prefetch' | 'value' | 'hasValue' | 'set' | 'update' // we don't allow manually viewing the returned data or updating it manually, prefetching a mutation also doesn't make any sense\r\n> & {\r\n  /**\r\n   * Executes the mutation.\r\n   *\r\n   * @param value The request body and any other request parameters to use for the mutation.  The `body` property is *required*.\r\n   */\r\n  mutate: (\r\n    value: Omit<NextRequest<TMethod, TMutation>, 'url'> & { url?: string },\r\n    ctx?: TICTX,\r\n  ) => void;\r\n  /**\r\n   * A signal that holds the current mutation request, or `null` if no mutation is in progress.\r\n   * This can be useful for tracking the state of the mutation or for displaying loading indicators.\r\n   */\r\n  current: Signal<\r\n    (Omit<NextRequest<TMethod, TMutation>, 'url'> & { url?: string }) | null\r\n  >;\r\n};\r\n\r\n/**\r\n * Creates a resource for performing mutations (e.g., POST, PUT, PATCH, DELETE requests).\r\n * Unlike `queryResource`, `mutationResource` is designed for one-off operations that change data.\r\n * It does *not* cache responses and does not provide a `value` signal.  Instead, it focuses on\r\n * managing the mutation lifecycle (pending, error, success) and provides callbacks for handling\r\n * these states.\r\n *\r\n * @param request A function that returns the base `HttpResourceRequest` to be made.  This\r\n *               function is called reactively.  Unlike `queryResource`, the `body` property\r\n *               of the request is provided when `mutate` is called, *not* here.  If the\r\n *               function returns `undefined`, the mutation is considered \"disabled.\" All properties,\r\n *               except the body, can be set here.\r\n * @param options Configuration options for the mutation resource.  This includes callbacks\r\n *               for `onMutate`, `onError`, `onSuccess`, and `onSettled`.\r\n * @typeParam TResult - The type of the expected result from the mutation.\r\n * @typeParam TRaw - The raw response type from the HTTP request (defaults to TResult).\r\n * @typeParam TCTX - The type of the context value returned by `onMutate`.\r\n * @returns A `MutationResourceRef` instance, which provides methods for triggering the mutation\r\n *          and observing its status.\r\n */\r\nexport function mutationResource<\r\n  TResult,\r\n  TRaw = TResult,\r\n  TMutation = TResult,\r\n  TCTX = void,\r\n  TICTX = TCTX,\r\n  TMethod extends HttpResourceRequest['method'] = HttpResourceRequest['method'],\r\n>(\r\n  request: () =>\r\n    | Omit<NextRequest<TMethod, TMutation>, 'body'>\r\n    | undefined\r\n    | void,\r\n  options: MutationResourceOptions<TResult, TRaw, TMutation, TCTX, TICTX> = {},\r\n): MutationResourceRef<TResult, TMutation, TICTX, TMethod> {\r\n  const { onMutate, onError, onSuccess, onSettled, equal, ...rest } = options;\r\n\r\n  const requestEqual = createEqualRequest(equal);\r\n\r\n  const baseRequest = computed(() => request() ?? undefined, {\r\n    equal: requestEqual,\r\n  });\r\n\r\n  const nextRequest = signal<\r\n    (Omit<NextRequest<TMethod, TMutation>, 'url'> & { url?: string }) | null\r\n  >(null, {\r\n    equal: (a, b) => {\r\n      if (!a && !b) return true;\r\n      if (!a || !b) return false;\r\n      return requestEqual(a, b);\r\n    },\r\n  });\r\n\r\n  const req = computed((): HttpResourceRequest | undefined => {\r\n    const nr = nextRequest();\r\n    if (!nr) return;\r\n\r\n    const base = baseRequest();\r\n\r\n    const url = nr.url ?? base?.url;\r\n    if (!url) return;\r\n\r\n    const method = nr.method ?? base?.method;\r\n\r\n    return {\r\n      ...base,\r\n      ...nr,\r\n      url,\r\n      method,\r\n    };\r\n  });\r\n\r\n  const resource = queryResource<TResult, TRaw>(req, {\r\n    ...rest,\r\n    defaultValue: null as unknown as TResult, // doesnt matter since .value is not accessible\r\n  });\r\n\r\n  let ctx: TCTX = undefined as TCTX;\r\n\r\n  const destroyRef = options.injector\r\n    ? options.injector.get(DestroyRef)\r\n    : inject(DestroyRef);\r\n\r\n  const error$ = toObservable(resource.error);\r\n  const value$ = toObservable(resource.value);\r\n\r\n  const statusSub = toObservable(resource.status)\r\n    .pipe(\r\n      combineLatestWith(error$, value$),\r\n      map(([status, error, value]): StatusResult<TResult> | null => {\r\n        if (status === 'error' && error) {\r\n          return {\r\n            status: 'error',\r\n            error,\r\n          };\r\n        }\r\n\r\n        if (status === 'resolved' && value !== null) {\r\n          return {\r\n            status: 'resolved',\r\n            value,\r\n          };\r\n        }\r\n\r\n        return null;\r\n      }),\r\n      filter((v) => v !== null),\r\n      takeUntilDestroyed(destroyRef),\r\n    )\r\n    .subscribe((result) => {\r\n      if (result.status === 'error') onError?.(result.error, ctx);\r\n      else onSuccess?.(result.value, ctx);\r\n\r\n      onSettled?.(ctx);\r\n      ctx = undefined as TCTX;\r\n      nextRequest.set(null);\r\n    });\r\n\r\n  return {\r\n    ...resource,\r\n    destroy: () => {\r\n      statusSub.unsubscribe();\r\n      resource.destroy();\r\n    },\r\n    mutate: (value, ictx) => {\r\n      ctx = onMutate?.(\r\n        (value as unknown as HttpResourceRequest).body as TMutation,\r\n        ictx,\r\n      ) as TCTX;\r\n      nextRequest.set(value);\r\n    },\r\n    current: nextRequest,\r\n  };\r\n}\r\n"]}","/**\n * Generated bundle index. Do not edit.\n */\nexport * from './index';\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibW1zdGFjay1yZXNvdXJjZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3BhY2thZ2VzL3Jlc291cmNlL3NyYy9tbXN0YWNrLXJlc291cmNlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBRUgsY0FBYyxTQUFTLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEdlbmVyYXRlZCBidW5kbGUgaW5kZXguIERvIG5vdCBlZGl0LlxuICovXG5cbmV4cG9ydCAqIGZyb20gJy4vaW5kZXgnO1xuIl19"],"names":[],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"mmstack-resource.mjs","sources":["../tmp-esm2022/lib/util/cache/cache.js","../tmp-esm2022/lib/util/cache/cache.interceptor.js","../tmp-esm2022/lib/util/catch-value-error.js","../tmp-esm2022/lib/util/circuit-breaker.js","../tmp-esm2022/lib/util/dedupe.interceptor.js","../tmp-esm2022/lib/util/equality.js","../tmp-esm2022/lib/util/has-slow-connection.js","../tmp-esm2022/lib/util/persist.js","../tmp-esm2022/lib/util/refresh.js","../tmp-esm2022/lib/util/retry-on-error.js","../tmp-esm2022/lib/util/url-with-params.js","../tmp-esm2022/lib/query-resource.js","../tmp-esm2022/lib/mutation-resource.js","../tmp-esm2022/mmstack-resource.js"],"sourcesContent":["import { computed, inject, InjectionToken, isDevMode, untracked, } from '@angular/core';\nimport { mutable } from '@mmstack/primitives';\nimport { v7 } from 'uuid';\nconst ONE_DAY = 1000 * 60 * 60 * 24;\nconst ONE_HOUR = 1000 * 60 * 60;\nconst DEFAULT_CLEANUP_OPT = {\n type: 'lru',\n maxSize: 200,\n checkInterval: ONE_HOUR,\n};\n/**\n * A generic cache implementation that stores data with time-to-live (TTL) and stale-while-revalidate capabilities.\n *\n * @typeParam T - The type of data to be stored in the cache.\n */\nexport class Cache {\n ttl;\n staleTime;\n internal = mutable(new Map());\n cleanupOpt;\n /**\n * Creates a new `Cache` instance.\n *\n * @param ttl - The default Time To Live (TTL) for cache entries, in milliseconds. Defaults to one day.\n * @param staleTime - The default duration, in milliseconds, during which a cache entry is considered\n * stale but can still be used while revalidation occurs in the background. Defaults to 1 hour.\n * @param cleanupOpt - Options for configuring the cache cleanup strategy. Defaults to LRU with a\n * `maxSize` of 200 and a `checkInterval` of one hour.\n */\n constructor(ttl = ONE_DAY, staleTime = ONE_HOUR, cleanupOpt = {\n type: 'lru',\n maxSize: 1000,\n checkInterval: ONE_HOUR,\n }) {\n this.ttl = ttl;\n this.staleTime = staleTime;\n this.cleanupOpt = {\n ...DEFAULT_CLEANUP_OPT,\n ...cleanupOpt,\n };\n if (this.cleanupOpt.maxSize <= 0)\n throw new Error('maxSize must be greater than 0');\n // cleanup cache based on provided options regularly\n const cleanupInterval = setInterval(() => {\n this.cleanup();\n }, cleanupOpt.checkInterval);\n const destroyId = v7();\n // cleanup if object is garbage collected, this is because the cache can be quite large from a memory standpoint & we dont want all that floating garbage\n const registry = new FinalizationRegistry((id) => {\n if (id === destroyId) {\n clearInterval(cleanupInterval);\n }\n });\n registry.register(this, destroyId);\n }\n /** @internal */\n getInternal(key) {\n const keySignal = computed(() => key());\n return computed(() => {\n const key = keySignal();\n if (!key)\n return null;\n const found = this.internal().get(key);\n const now = Date.now();\n if (!found || found.expiresAt <= now)\n return null;\n found.useCount++;\n return {\n ...found,\n isStale: found.stale <= now,\n };\n });\n }\n /**\n * Retrieves a cache entry without affecting its usage count (for LRU). This is primarily\n * for internal use or debugging.\n * @internal\n * @param key - The key of the entry to retrieve.\n * @returns The cache entry, or `null` if not found or expired.\n */\n getUntracked(key) {\n return untracked(this.getInternal(() => key));\n }\n /**\n * Retrieves a cache entry as a signal.\n *\n * @param key - A function that returns the cache key. The key is a signal, allowing for dynamic keys. If the function returns null the value is also null.\n * @returns A signal that holds the cache entry, or `null` if not found or expired. The signal\n * updates whenever the cache entry changes (e.g., due to revalidation or expiration).\n */\n get(key) {\n return this.getInternal(key);\n }\n /**\n * Stores a value in the cache.\n *\n * @param key - The key under which to store the value.\n * @param value - The value to store.\n * @param staleTime - (Optional) The stale time for this entry, in milliseconds. Overrides the default `staleTime`.\n * @param ttl - (Optional) The TTL for this entry, in milliseconds. Overrides the default `ttl`.\n */\n store(key, value, staleTime = this.staleTime, ttl = this.ttl) {\n const entry = this.getUntracked(key);\n if (entry) {\n clearTimeout(entry.timeout); // stop invalidation\n }\n const prevCount = entry?.useCount ?? 0;\n // ttl cannot be less than staleTime\n if (ttl < staleTime)\n staleTime = ttl;\n const now = Date.now();\n this.internal.mutate((map) => {\n map.set(key, {\n value,\n created: entry?.created ?? now,\n useCount: prevCount + 1,\n stale: now + staleTime,\n expiresAt: now + ttl,\n timeout: setTimeout(() => this.invalidate(key), ttl),\n });\n return map;\n });\n }\n /**\n * Invalidates (removes) a cache entry.\n *\n * @param key - The key of the entry to invalidate.\n */\n invalidate(key) {\n const entry = this.getUntracked(key);\n if (!entry)\n return;\n clearTimeout(entry.timeout);\n this.internal.mutate((map) => {\n map.delete(key);\n return map;\n });\n }\n /** @internal */\n cleanup() {\n if (untracked(this.internal).size <= this.cleanupOpt.maxSize)\n return;\n const sorted = Array.from(untracked(this.internal).entries()).toSorted((a, b) => {\n if (this.cleanupOpt.type === 'lru') {\n return a[1].useCount - b[1].useCount; // least used first\n }\n else {\n return a[1].created - b[1].created; // oldest first\n }\n });\n const keepCount = Math.floor(this.cleanupOpt.maxSize / 2);\n const removed = sorted.slice(0, sorted.length - keepCount);\n const keep = sorted.slice(removed.length, sorted.length);\n removed.forEach(([, e]) => {\n clearTimeout(e.timeout);\n });\n this.internal.set(new Map(keep));\n }\n}\nconst CLIENT_CACHE_TOKEN = new InjectionToken('INTERNAL_CLIENT_CACHE');\n/**\n * Provides the instance of the QueryCache for queryResource. This should probably be called\n * in your application's root configuration, but can also be overriden with component/module providers.\n *\n * @param options - Optional configuration options for the cache.\n * @returns An Angular `Provider` for the cache.\n *\n * @example\n * // In your app.config.ts or AppModule providers:\n *\n * import { provideQueryCache } from './your-cache';\n *\n * export const appConfig: ApplicationConfig = {\n * providers: [\n * provideQueryCache({\n * ttl: 60000, // Default TTL of 60 seconds\n * staleTime: 30000, // Default staleTime of 30 seconds\n * }),\n * // ... other providers\n * ]\n * };\n */\nexport function provideQueryCache(opt) {\n return {\n provide: CLIENT_CACHE_TOKEN,\n useValue: new Cache(opt?.ttl, opt?.staleTime, opt?.cleanup),\n };\n}\nclass NoopCache extends Cache {\n store(_, __, ___ = super.staleTime, ____ = super.ttl) {\n // noop\n }\n}\n/**\n * Injects the `QueryCache` instance that is used within queryResource.\n * Allows for direct modification of cached data, but is mostly meant for internal use.\n *\n * @param injector - (Optional) The injector to use. If not provided, the current\n * injection context is used.\n * @returns The `QueryCache` instance.\n *\n * @example\n * // In your component or service:\n *\n * import { injectQueryCache } from './your-cache';\n *\n * constructor() {\n * const cache = injectQueryCache();\n *\n * const myData = cache.get(() => 'my-data-key');\n * if (myData() !== null) {\n * // ... use cached data ...\n * }\n * }\n */\nexport function injectQueryCache(injector) {\n const cache = injector\n ? injector.get(CLIENT_CACHE_TOKEN, null, {\n optional: true,\n })\n : inject(CLIENT_CACHE_TOKEN, {\n optional: true,\n });\n if (!cache) {\n if (isDevMode())\n throw new Error('Cache not provided, please add provideQueryCache() to providers array');\n else\n return new NoopCache();\n }\n return cache;\n}\n//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cache.js","sourceRoot":"","sources":["../../../../../../../packages/resource/src/lib/util/cache/cache.ts"],"names":[],"mappings":"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","sourcesContent":["import type { HttpResponse } from '@angular/common/http';\r\nimport {\r\n  computed,\r\n  inject,\r\n  InjectionToken,\r\n  Injector,\r\n  isDevMode,\r\n  type Provider,\r\n  type Signal,\r\n  untracked,\r\n} from '@angular/core';\r\nimport { mutable } from '@mmstack/primitives';\r\nimport { v7 } from 'uuid';\r\n\r\n/**\r\n * Options for configuring the Least Recently Used (LRU) cache cleanup strategy.\r\n * @internal\r\n */\r\ntype LRUCleanupType = {\r\n  type: 'lru';\r\n  /**\r\n   * How often to check for expired or excess entries, in milliseconds.\r\n   */\r\n  checkInterval: number;\r\n  /**\r\n   * The maximum number of entries to keep in the cache.  When the cache exceeds this size,\r\n   * the least recently used entries will be removed.\r\n   */\r\n  maxSize: number;\r\n};\r\n\r\n/**\r\n * Options for configuring the \"oldest first\" cache cleanup strategy.\r\n * @internal\r\n */\r\ntype OldsetCleanupType = {\r\n  type: 'oldest';\r\n  /**\r\n   * How often to check for expired or excess entries, in milliseconds.\r\n   */\r\n  checkInterval: number;\r\n  /**\r\n   * The maximum number of entries to keep in the cache.  When the cache exceeds this size,\r\n   * the oldest entries will be removed.\r\n   */\r\n  maxSize: number;\r\n};\r\n\r\n/**\r\n * Represents an entry in the cache.\r\n * @internal\r\n */\r\ntype CacheEntry<T> = {\r\n  value: T;\r\n  created: number;\r\n  stale: number;\r\n  useCount: number;\r\n  expiresAt: number;\r\n  timeout: ReturnType<typeof setTimeout>;\r\n};\r\n\r\n/**\r\n * Defines the types of cleanup strategies available for the cache.\r\n * - `lru`: Least Recently Used.  Removes the least recently accessed entries when the cache is full.\r\n * - `oldest`: Removes the oldest entries when the cache is full.\r\n */\r\nexport type CleanupType = LRUCleanupType | OldsetCleanupType;\r\n\r\nconst ONE_DAY = 1000 * 60 * 60 * 24;\r\nconst ONE_HOUR = 1000 * 60 * 60;\r\n\r\nconst DEFAULT_CLEANUP_OPT = {\r\n  type: 'lru',\r\n  maxSize: 200,\r\n  checkInterval: ONE_HOUR,\r\n} satisfies LRUCleanupType;\r\n\r\n/**\r\n * A generic cache implementation that stores data with time-to-live (TTL) and stale-while-revalidate capabilities.\r\n *\r\n * @typeParam T - The type of data to be stored in the cache.\r\n */\r\nexport class Cache<T> {\r\n  private readonly internal = mutable(new Map<string, CacheEntry<T>>());\r\n  private readonly cleanupOpt: CleanupType;\r\n\r\n  /**\r\n   * Creates a new `Cache` instance.\r\n   *\r\n   * @param ttl - The default Time To Live (TTL) for cache entries, in milliseconds.  Defaults to one day.\r\n   * @param staleTime - The default duration, in milliseconds, during which a cache entry is considered\r\n   *                    stale but can still be used while revalidation occurs in the background. Defaults to 1 hour.\r\n   * @param cleanupOpt - Options for configuring the cache cleanup strategy.  Defaults to LRU with a\r\n   *                     `maxSize` of 200 and a `checkInterval` of one hour.\r\n   */\r\n  constructor(\r\n    protected readonly ttl: number = ONE_DAY,\r\n    protected readonly staleTime: number = ONE_HOUR,\r\n    cleanupOpt: Partial<CleanupType> = {\r\n      type: 'lru',\r\n      maxSize: 1000,\r\n      checkInterval: ONE_HOUR,\r\n    },\r\n  ) {\r\n    this.cleanupOpt = {\r\n      ...DEFAULT_CLEANUP_OPT,\r\n      ...cleanupOpt,\r\n    };\r\n    if (this.cleanupOpt.maxSize <= 0)\r\n      throw new Error('maxSize must be greater than 0');\r\n\r\n    // cleanup cache based on provided options regularly\r\n    const cleanupInterval = setInterval(() => {\r\n      this.cleanup();\r\n    }, cleanupOpt.checkInterval);\r\n\r\n    const destroyId = v7();\r\n\r\n    // cleanup if object is garbage collected, this is because the cache can be quite large from a memory standpoint & we dont want all that floating garbage\r\n    const registry = new FinalizationRegistry((id: string) => {\r\n      if (id === destroyId) {\r\n        clearInterval(cleanupInterval);\r\n      }\r\n    });\r\n\r\n    registry.register(this, destroyId);\r\n  }\r\n\r\n  /** @internal */\r\n  private getInternal(\r\n    key: () => string | null,\r\n  ): Signal<(CacheEntry<T> & { isStale: boolean }) | null> {\r\n    const keySignal = computed(() => key());\r\n\r\n    return computed(() => {\r\n      const key = keySignal();\r\n      if (!key) return null;\r\n      const found = this.internal().get(key);\r\n\r\n      const now = Date.now();\r\n\r\n      if (!found || found.expiresAt <= now) return null;\r\n      found.useCount++;\r\n      return {\r\n        ...found,\r\n        isStale: found.stale <= now,\r\n      };\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Retrieves a cache entry without affecting its usage count (for LRU).  This is primarily\r\n   * for internal use or debugging.\r\n   * @internal\r\n   * @param key - The key of the entry to retrieve.\r\n   * @returns The cache entry, or `null` if not found or expired.\r\n   */\r\n  getUntracked(key: string): (CacheEntry<T> & { isStale: boolean }) | null {\r\n    return untracked(this.getInternal(() => key));\r\n  }\r\n\r\n  /**\r\n   * Retrieves a cache entry as a signal.\r\n   *\r\n   * @param key - A function that returns the cache key. The key is a signal, allowing for dynamic keys. If the function returns null the value is also null.\r\n   * @returns A signal that holds the cache entry, or `null` if not found or expired.  The signal\r\n   *          updates whenever the cache entry changes (e.g., due to revalidation or expiration).\r\n   */\r\n  get(\r\n    key: () => string | null,\r\n  ): Signal<(CacheEntry<T> & { isStale: boolean }) | null> {\r\n    return this.getInternal(key);\r\n  }\r\n\r\n  /**\r\n   * Stores a value in the cache.\r\n   *\r\n   * @param key - The key under which to store the value.\r\n   * @param value - The value to store.\r\n   * @param staleTime - (Optional) The stale time for this entry, in milliseconds. Overrides the default `staleTime`.\r\n   * @param ttl - (Optional) The TTL for this entry, in milliseconds. Overrides the default `ttl`.\r\n   */\r\n  store(key: string, value: T, staleTime = this.staleTime, ttl = this.ttl) {\r\n    const entry = this.getUntracked(key);\r\n    if (entry) {\r\n      clearTimeout(entry.timeout); // stop invalidation\r\n    }\r\n\r\n    const prevCount = entry?.useCount ?? 0;\r\n\r\n    // ttl cannot be less than staleTime\r\n    if (ttl < staleTime) staleTime = ttl;\r\n\r\n    const now = Date.now();\r\n\r\n    this.internal.mutate((map) => {\r\n      map.set(key, {\r\n        value,\r\n        created: entry?.created ?? now,\r\n        useCount: prevCount + 1,\r\n        stale: now + staleTime,\r\n        expiresAt: now + ttl,\r\n        timeout: setTimeout(() => this.invalidate(key), ttl),\r\n      });\r\n      return map;\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Invalidates (removes) a cache entry.\r\n   *\r\n   * @param key - The key of the entry to invalidate.\r\n   */\r\n  invalidate(key: string) {\r\n    const entry = this.getUntracked(key);\r\n    if (!entry) return;\r\n    clearTimeout(entry.timeout);\r\n    this.internal.mutate((map) => {\r\n      map.delete(key);\r\n      return map;\r\n    });\r\n  }\r\n\r\n  /** @internal */\r\n  private cleanup() {\r\n    if (untracked(this.internal).size <= this.cleanupOpt.maxSize) return;\r\n\r\n    const sorted = Array.from(untracked(this.internal).entries()).toSorted(\r\n      (a, b) => {\r\n        if (this.cleanupOpt.type === 'lru') {\r\n          return a[1].useCount - b[1].useCount; // least used first\r\n        } else {\r\n          return a[1].created - b[1].created; // oldest first\r\n        }\r\n      },\r\n    );\r\n\r\n    const keepCount = Math.floor(this.cleanupOpt.maxSize / 2);\r\n\r\n    const removed = sorted.slice(0, sorted.length - keepCount);\r\n    const keep = sorted.slice(removed.length, sorted.length);\r\n\r\n    removed.forEach(([, e]) => {\r\n      clearTimeout(e.timeout);\r\n    });\r\n\r\n    this.internal.set(new Map(keep));\r\n  }\r\n}\r\n\r\n/**\r\n * Options for configuring the cache.\r\n */\r\ntype CacheOptions = {\r\n  /**\r\n   * The default Time To Live (TTL) for cache entries, in milliseconds.\r\n   */\r\n  ttl?: number;\r\n  /**\r\n   * The default duration, in milliseconds, during which a cache entry is considered\r\n   * stale but can still be used while revalidation occurs in the background.\r\n   */\r\n  staleTime?: number;\r\n  /**\r\n   * Options for configuring the cache cleanup strategy.\r\n   */\r\n  cleanup?: Partial<CleanupType>;\r\n};\r\n\r\nconst CLIENT_CACHE_TOKEN = new InjectionToken<Cache<HttpResponse<unknown>>>(\r\n  'INTERNAL_CLIENT_CACHE',\r\n);\r\n\r\n/**\r\n * Provides the instance of the QueryCache for queryResource. This should probably be called\r\n * in your application's root configuration, but can also be overriden with component/module providers.\r\n *\r\n * @param options - Optional configuration options for the cache.\r\n * @returns An Angular `Provider` for the cache.\r\n *\r\n * @example\r\n * // In your app.config.ts or AppModule providers:\r\n *\r\n * import { provideQueryCache } from './your-cache';\r\n *\r\n * export const appConfig: ApplicationConfig = {\r\n *   providers: [\r\n *     provideQueryCache({\r\n *       ttl: 60000, // Default TTL of 60 seconds\r\n *       staleTime: 30000, // Default staleTime of 30 seconds\r\n *     }),\r\n *     // ... other providers\r\n *   ]\r\n * };\r\n */\r\nexport function provideQueryCache(opt?: CacheOptions): Provider {\r\n  return {\r\n    provide: CLIENT_CACHE_TOKEN,\r\n    useValue: new Cache(opt?.ttl, opt?.staleTime, opt?.cleanup),\r\n  };\r\n}\r\n\r\nclass NoopCache<T> extends Cache<T> {\r\n  override store(_: string, __: T, ___ = super.staleTime, ____ = super.ttl) {\r\n    // noop\r\n  }\r\n}\r\n\r\n/**\r\n * Injects the `QueryCache` instance that is used within queryResource.\r\n * Allows for direct modification of cached data, but is mostly meant for internal use.\r\n *\r\n * @param injector - (Optional) The injector to use.  If not provided, the current\r\n *                   injection context is used.\r\n * @returns The `QueryCache` instance.\r\n *\r\n * @example\r\n * // In your component or service:\r\n *\r\n * import { injectQueryCache } from './your-cache';\r\n *\r\n * constructor() {\r\n *   const cache = injectQueryCache();\r\n *\r\n *   const myData = cache.get(() => 'my-data-key');\r\n *   if (myData() !== null) {\r\n *     // ... use cached data ...\r\n *   }\r\n * }\r\n */\r\nexport function injectQueryCache(\r\n  injector?: Injector,\r\n): Cache<HttpResponse<unknown>> {\r\n  const cache = injector\r\n    ? injector.get(CLIENT_CACHE_TOKEN, null, {\r\n        optional: true,\r\n      })\r\n    : inject(CLIENT_CACHE_TOKEN, {\r\n        optional: true,\r\n      });\r\n\r\n  if (!cache) {\r\n    if (isDevMode())\r\n      throw new Error(\r\n        'Cache not provided, please add provideQueryCache() to providers array',\r\n      );\r\n    else return new NoopCache();\r\n  }\r\n\r\n  return cache;\r\n}\r\n"]}","import { HttpContext, HttpContextToken, HttpResponse, } from '@angular/common/http';\nimport { map, of, tap } from 'rxjs';\nimport { injectQueryCache } from './cache';\nconst CACHE_CONTEXT = new HttpContextToken(() => ({\n cache: false,\n}));\nexport function setCacheContext(ctx = new HttpContext(), opt) {\n return ctx.set(CACHE_CONTEXT, { ...opt, cache: true });\n}\nfunction getCacheContext(ctx) {\n return ctx.get(CACHE_CONTEXT);\n}\nfunction parseCacheControlHeader(req) {\n const header = req.headers.get('Cache-Control');\n let sMaxAge = null;\n const directives = {\n noStore: false,\n noCache: false,\n mustRevalidate: false,\n immutable: false,\n maxAge: null,\n staleWhileRevalidate: null,\n };\n if (!header)\n return directives;\n const parts = header.split(',');\n for (const part of parts) {\n const [unparsedKey, value] = part.trim().split('=');\n const key = unparsedKey.trim().toLowerCase();\n switch (key) {\n case 'no-store':\n directives.noStore = true;\n break;\n case 'no-cache':\n directives.noCache = true;\n break;\n case 'must-revalidate':\n case 'proxy-revalidate':\n directives.mustRevalidate = true;\n break;\n case 'immutable':\n directives.immutable = true;\n break;\n case 'max-age': {\n if (!value)\n break;\n const parsedValue = parseInt(value, 10);\n if (!isNaN(parsedValue))\n directives.maxAge = parsedValue;\n break;\n }\n case 's-max-age': {\n if (!value)\n break;\n const parsedValue = parseInt(value, 10);\n if (!isNaN(parsedValue))\n sMaxAge = parsedValue;\n break;\n }\n case 'stale-while-revalidate': {\n if (!value)\n break;\n const parsedValue = parseInt(value, 10);\n if (!isNaN(parsedValue))\n directives.staleWhileRevalidate = parsedValue;\n break;\n }\n }\n }\n // s-max-age takes precedence over max-age\n if (sMaxAge !== null)\n directives.maxAge = sMaxAge;\n // if no store nothing else is relevant\n if (directives.noStore)\n return {\n noStore: true,\n noCache: false,\n mustRevalidate: false,\n immutable: false,\n maxAge: null,\n staleWhileRevalidate: null,\n };\n // max age does not apply to immutable resources\n if (directives.immutable)\n return {\n ...directives,\n maxAge: null,\n };\n return directives;\n}\nfunction resolveTimings(cacheControl, staleTime, ttl) {\n const timings = {\n staleTime,\n ttl,\n };\n if (cacheControl.immutable)\n return {\n staleTime: Infinity,\n ttl: Infinity,\n };\n // if no-cache is set, we must always revalidate\n if (cacheControl.noCache || cacheControl.mustRevalidate)\n timings.staleTime = 0;\n if (cacheControl.staleWhileRevalidate !== null)\n timings.staleTime = cacheControl.staleWhileRevalidate;\n if (cacheControl.maxAge !== null)\n timings.ttl = cacheControl.maxAge * 1000;\n // if stale-while-revalidate is set, we must revalidate after that time at the latest, but we can still serve the stale data\n if (cacheControl.staleWhileRevalidate !== null) {\n const ms = cacheControl.staleWhileRevalidate * 1000;\n if (timings.staleTime === undefined || timings.staleTime > ms)\n timings.staleTime = ms;\n }\n return timings;\n}\n/**\n * Creates an `HttpInterceptorFn` that implements caching for HTTP requests. This interceptor\n * checks for a caching configuration in the request's `HttpContext` (internally set by the queryResource).\n * If caching is enabled, it attempts to retrieve responses from the cache. If a cached response\n * is found and is not stale, it's returned directly. If the cached response is stale, it's returned,\n * and a background revalidation request is made. If no cached response is found, the request\n * is made to the server, and the response is cached according to the configured TTL and staleness.\n * The interceptor also respects `Cache-Control` headers from the server.\n *\n * @param allowedMethods - An array of HTTP methods for which caching should be enabled.\n * Defaults to `['GET', 'HEAD', 'OPTIONS']`.\n *\n * @returns An `HttpInterceptorFn` that implements the caching logic.\n *\n * @example\n * // In your app.config.ts or module providers:\n *\n * import { provideHttpClient, withInterceptors } from '@angular/common/http';\n * import { createCacheInterceptor } from '@mmstack/resource';\n *\n * export const appConfig: ApplicationConfig = {\n * providers: [\n * provideHttpClient(withInterceptors([createCacheInterceptor()])),\n * // ... other providers\n * ],\n * };\n */\nexport function createCacheInterceptor(allowedMethods = ['GET', 'HEAD', 'OPTIONS']) {\n const CACHE_METHODS = new Set(allowedMethods);\n return (req, next) => {\n const cache = injectQueryCache();\n if (!CACHE_METHODS.has(req.method))\n return next(req);\n const opt = getCacheContext(req.context);\n if (!opt.cache)\n return next(req);\n const key = opt.key ?? req.urlWithParams;\n const entry = cache.getUntracked(key); // null if expired or not found\n // If the entry is not stale, return it\n if (entry && !entry.isStale)\n return of(entry.value);\n // resource itself handles case of showing stale data...the request must process as this will \"refresh said data\"\n const eTag = entry?.value.headers.get('ETag');\n const lastModified = entry?.value.headers.get('Last-Modified');\n if (eTag) {\n req = req.clone({ setHeaders: { 'If-None-Match': eTag } });\n }\n if (lastModified) {\n req = req.clone({ setHeaders: { 'If-Modified-Since': lastModified } });\n }\n return next(req).pipe(tap((event) => {\n if (event instanceof HttpResponse && event.ok) {\n const cacheControl = parseCacheControlHeader(event);\n if (cacheControl.noStore)\n return;\n const { staleTime, ttl } = resolveTimings(cacheControl, opt.staleTime, opt.ttl);\n cache.store(key, event, staleTime, ttl);\n }\n }), map((event) => {\n // handle 304 responses due to eTag/last-modified\n if (event instanceof HttpResponse && event.status === 304 && entry) {\n return entry.value;\n }\n return event;\n }));\n };\n}\n//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cache.interceptor.js","sourceRoot":"","sources":["../../../../../../../packages/resource/src/lib/util/cache/cache.interceptor.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  HttpContext,\r\n  HttpContextToken,\r\n  type HttpEvent,\r\n  type HttpHandlerFn,\r\n  type HttpInterceptorFn,\r\n  type HttpRequest,\r\n  HttpResponse,\r\n} from '@angular/common/http';\r\nimport { map, Observable, of, tap } from 'rxjs';\r\nimport { injectQueryCache } from './cache';\r\n\r\ntype CacheEntryOptions = {\r\n  key?: string;\r\n  ttl?: number;\r\n  staleTime?: number;\r\n  cache: boolean;\r\n};\r\n\r\nconst CACHE_CONTEXT = new HttpContextToken<CacheEntryOptions>(() => ({\r\n  cache: false,\r\n}));\r\n\r\nexport function setCacheContext(\r\n  ctx = new HttpContext(),\r\n  opt: Omit<CacheEntryOptions, 'cache' | 'key'> & {\r\n    key: Required<CacheEntryOptions>['key'];\r\n  },\r\n) {\r\n  return ctx.set(CACHE_CONTEXT, { ...opt, cache: true });\r\n}\r\n\r\nfunction getCacheContext(ctx: HttpContext): CacheEntryOptions {\r\n  return ctx.get(CACHE_CONTEXT);\r\n}\r\n\r\ntype ResolvedCacheControl = {\r\n  noStore: boolean;\r\n  noCache: boolean;\r\n  mustRevalidate: boolean;\r\n  immutable: boolean;\r\n  maxAge: number | null;\r\n  staleWhileRevalidate: number | null;\r\n};\r\n\r\nfunction parseCacheControlHeader(\r\n  req: HttpResponse<unknown>,\r\n): ResolvedCacheControl {\r\n  const header = req.headers.get('Cache-Control');\r\n\r\n  let sMaxAge: number | null = null;\r\n  const directives: ResolvedCacheControl = {\r\n    noStore: false,\r\n    noCache: false,\r\n    mustRevalidate: false,\r\n    immutable: false,\r\n    maxAge: null,\r\n    staleWhileRevalidate: null,\r\n  };\r\n\r\n  if (!header) return directives;\r\n\r\n  const parts = header.split(',');\r\n\r\n  for (const part of parts) {\r\n    const [unparsedKey, value] = part.trim().split('=');\r\n    const key = unparsedKey.trim().toLowerCase();\r\n\r\n    switch (key) {\r\n      case 'no-store':\r\n        directives.noStore = true;\r\n        break;\r\n      case 'no-cache':\r\n        directives.noCache = true;\r\n        break;\r\n      case 'must-revalidate':\r\n      case 'proxy-revalidate':\r\n        directives.mustRevalidate = true;\r\n        break;\r\n      case 'immutable':\r\n        directives.immutable = true;\r\n        break;\r\n      case 'max-age': {\r\n        if (!value) break;\r\n        const parsedValue = parseInt(value, 10);\r\n        if (!isNaN(parsedValue)) directives.maxAge = parsedValue;\r\n        break;\r\n      }\r\n      case 's-max-age': {\r\n        if (!value) break;\r\n        const parsedValue = parseInt(value, 10);\r\n        if (!isNaN(parsedValue)) sMaxAge = parsedValue;\r\n        break;\r\n      }\r\n      case 'stale-while-revalidate': {\r\n        if (!value) break;\r\n        const parsedValue = parseInt(value, 10);\r\n        if (!isNaN(parsedValue)) directives.staleWhileRevalidate = parsedValue;\r\n        break;\r\n      }\r\n    }\r\n  }\r\n\r\n  // s-max-age takes precedence over max-age\r\n  if (sMaxAge !== null) directives.maxAge = sMaxAge;\r\n\r\n  // if no store nothing else is relevant\r\n  if (directives.noStore)\r\n    return {\r\n      noStore: true,\r\n      noCache: false,\r\n      mustRevalidate: false,\r\n      immutable: false,\r\n      maxAge: null,\r\n      staleWhileRevalidate: null,\r\n    };\r\n\r\n  // max age does not apply to immutable resources\r\n  if (directives.immutable)\r\n    return {\r\n      ...directives,\r\n      maxAge: null,\r\n    };\r\n\r\n  return directives;\r\n}\r\n\r\nfunction resolveTimings(\r\n  cacheControl: ResolvedCacheControl,\r\n  staleTime?: number,\r\n  ttl?: number,\r\n): { staleTime?: number; ttl?: number } {\r\n  const timings = {\r\n    staleTime,\r\n    ttl,\r\n  };\r\n\r\n  if (cacheControl.immutable)\r\n    return {\r\n      staleTime: Infinity,\r\n      ttl: Infinity,\r\n    };\r\n\r\n  // if no-cache is set, we must always revalidate\r\n  if (cacheControl.noCache || cacheControl.mustRevalidate)\r\n    timings.staleTime = 0;\r\n\r\n  if (cacheControl.staleWhileRevalidate !== null)\r\n    timings.staleTime = cacheControl.staleWhileRevalidate;\r\n\r\n  if (cacheControl.maxAge !== null) timings.ttl = cacheControl.maxAge * 1000;\r\n\r\n  // if stale-while-revalidate is set, we must revalidate after that time at the latest, but we can still serve the stale data\r\n  if (cacheControl.staleWhileRevalidate !== null) {\r\n    const ms = cacheControl.staleWhileRevalidate * 1000;\r\n    if (timings.staleTime === undefined || timings.staleTime > ms)\r\n      timings.staleTime = ms;\r\n  }\r\n\r\n  return timings;\r\n}\r\n\r\n/**\r\n * Creates an `HttpInterceptorFn` that implements caching for HTTP requests. This interceptor\r\n * checks for a caching configuration in the request's `HttpContext` (internally set by the queryResource).\r\n * If caching is enabled, it attempts to retrieve responses from the cache. If a cached response\r\n * is found and is not stale, it's returned directly.  If the cached response is stale, it's returned,\r\n * and a background revalidation request is made.  If no cached response is found, the request\r\n * is made to the server, and the response is cached according to the configured TTL and staleness.\r\n * The interceptor also respects `Cache-Control` headers from the server.\r\n *\r\n * @param allowedMethods - An array of HTTP methods for which caching should be enabled.\r\n *                        Defaults to `['GET', 'HEAD', 'OPTIONS']`.\r\n *\r\n * @returns An `HttpInterceptorFn` that implements the caching logic.\r\n *\r\n * @example\r\n * // In your app.config.ts or module providers:\r\n *\r\n * import { provideHttpClient, withInterceptors } from '@angular/common/http';\r\n * import { createCacheInterceptor } from '@mmstack/resource';\r\n *\r\n * export const appConfig: ApplicationConfig = {\r\n *   providers: [\r\n *     provideHttpClient(withInterceptors([createCacheInterceptor()])),\r\n *     // ... other providers\r\n *   ],\r\n * };\r\n */\r\nexport function createCacheInterceptor(\r\n  allowedMethods = ['GET', 'HEAD', 'OPTIONS'],\r\n): HttpInterceptorFn {\r\n  const CACHE_METHODS = new Set<string>(allowedMethods);\r\n\r\n  return (\r\n    req: HttpRequest<unknown>,\r\n    next: HttpHandlerFn,\r\n  ): Observable<HttpEvent<unknown>> => {\r\n    const cache = injectQueryCache();\r\n\r\n    if (!CACHE_METHODS.has(req.method)) return next(req);\r\n    const opt = getCacheContext(req.context);\r\n\r\n    if (!opt.cache) return next(req);\r\n\r\n    const key = opt.key ?? req.urlWithParams;\r\n    const entry = cache.getUntracked(key); // null if expired or not found\r\n\r\n    // If the entry is not stale, return it\r\n    if (entry && !entry.isStale) return of(entry.value);\r\n\r\n    // resource itself handles case of showing stale data...the request must process as this will \"refresh said data\"\r\n\r\n    const eTag = entry?.value.headers.get('ETag');\r\n    const lastModified = entry?.value.headers.get('Last-Modified');\r\n\r\n    if (eTag) {\r\n      req = req.clone({ setHeaders: { 'If-None-Match': eTag } });\r\n    }\r\n\r\n    if (lastModified) {\r\n      req = req.clone({ setHeaders: { 'If-Modified-Since': lastModified } });\r\n    }\r\n\r\n    return next(req).pipe(\r\n      tap((event) => {\r\n        if (event instanceof HttpResponse && event.ok) {\r\n          const cacheControl = parseCacheControlHeader(event);\r\n          if (cacheControl.noStore) return;\r\n\r\n          const { staleTime, ttl } = resolveTimings(\r\n            cacheControl,\r\n            opt.staleTime,\r\n            opt.ttl,\r\n          );\r\n\r\n          cache.store(key, event, staleTime, ttl);\r\n        }\r\n      }),\r\n      map((event) => {\r\n        // handle 304 responses due to eTag/last-modified\r\n        if (event instanceof HttpResponse && event.status === 304 && entry) {\r\n          return entry.value;\r\n        }\r\n\r\n        return event;\r\n      }),\r\n    );\r\n  };\r\n}\r\n"]}","import { computed } from '@angular/core';\nimport { toWritable } from '@mmstack/primitives';\nexport function catchValueError(resource, fallback) {\n return {\n ...resource,\n value: toWritable(computed(() => {\n try {\n return resource.value();\n }\n catch {\n return fallback;\n }\n }), (value) => resource.value.set(value)),\n };\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2F0Y2gtdmFsdWUtZXJyb3IuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9yZXNvdXJjZS9zcmMvbGliL3V0aWwvY2F0Y2gtdmFsdWUtZXJyb3IudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQ0EsT0FBTyxFQUFFLFFBQVEsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUN6QyxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0scUJBQXFCLENBQUM7QUFFakQsTUFBTSxVQUFVLGVBQWUsQ0FDN0IsUUFBNEIsRUFDNUIsUUFBVztJQUVYLE9BQU87UUFDTCxHQUFHLFFBQVE7UUFDWCxLQUFLLEVBQUUsVUFBVSxDQUNmLFFBQVEsQ0FBQyxHQUFHLEVBQUU7WUFDWixJQUFJLENBQUM7Z0JBQ0gsT0FBTyxRQUFRLENBQUMsS0FBSyxFQUFFLENBQUM7WUFDMUIsQ0FBQztZQUFDLE1BQU0sQ0FBQztnQkFDUCxPQUFPLFFBQVEsQ0FBQztZQUNsQixDQUFDO1FBQ0gsQ0FBQyxDQUFDLEVBQ0YsQ0FBQyxLQUFLLEVBQUUsRUFBRSxDQUFDLFFBQVEsQ0FBQyxLQUFLLENBQUMsR0FBRyxDQUFDLEtBQUssQ0FBQyxDQUNyQztLQUNGLENBQUM7QUFDSixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSHR0cFJlc291cmNlUmVmIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uL2h0dHAnO1xuaW1wb3J0IHsgY29tcHV0ZWQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IHRvV3JpdGFibGUgfSBmcm9tICdAbW1zdGFjay9wcmltaXRpdmVzJztcblxuZXhwb3J0IGZ1bmN0aW9uIGNhdGNoVmFsdWVFcnJvcjxUPihcbiAgcmVzb3VyY2U6IEh0dHBSZXNvdXJjZVJlZjxUPixcbiAgZmFsbGJhY2s6IFQsXG4pOiBIdHRwUmVzb3VyY2VSZWY8VD4ge1xuICByZXR1cm4ge1xuICAgIC4uLnJlc291cmNlLFxuICAgIHZhbHVlOiB0b1dyaXRhYmxlKFxuICAgICAgY29tcHV0ZWQoKCkgPT4ge1xuICAgICAgICB0cnkge1xuICAgICAgICAgIHJldHVybiByZXNvdXJjZS52YWx1ZSgpO1xuICAgICAgICB9IGNhdGNoIHtcbiAgICAgICAgICByZXR1cm4gZmFsbGJhY2s7XG4gICAgICAgIH1cbiAgICAgIH0pLFxuICAgICAgKHZhbHVlKSA9PiByZXNvdXJjZS52YWx1ZS5zZXQodmFsdWUpLFxuICAgICksXG4gIH07XG59XG4iXX0=","import { computed, effect, signal, untracked } from '@angular/core';\n/** @internal */\nfunction internalCeateCircuitBreaker(treshold = 5, resetTimeout = 30000) {\n const halfOpen = signal(false);\n const failureCount = signal(0);\n const status = computed(() => {\n if (failureCount() >= treshold)\n return 'CLOSED';\n return halfOpen() ? 'HALF_OPEN' : 'OPEN';\n });\n const isClosed = computed(() => status() === 'CLOSED');\n const success = () => {\n failureCount.set(0);\n halfOpen.set(false);\n };\n const tryOnce = () => {\n if (!untracked(isClosed))\n return;\n halfOpen.set(true);\n failureCount.set(treshold - 1);\n };\n const effectRef = effect((cleanup) => {\n if (!isClosed())\n return;\n const timeout = setTimeout(tryOnce, resetTimeout);\n return cleanup(() => clearTimeout(timeout));\n });\n const fail = () => {\n failureCount.set(failureCount() + 1);\n halfOpen.set(false);\n };\n return {\n status,\n isClosed,\n fail,\n success,\n halfOpen: tryOnce,\n destroy: () => effectRef.destroy(),\n };\n}\n/** @internal */\nfunction createNeverBrokenCircuitBreaker() {\n return {\n isClosed: computed(() => false),\n status: signal('OPEN'),\n fail: () => {\n // noop\n },\n success: () => {\n // noop\n },\n halfOpen: () => {\n // noop\n },\n destroy: () => {\n // noop\n },\n };\n}\n/**\n * Creates a circuit breaker instance.\n *\n * @param options - Configuration options for the circuit breaker. Can be:\n * - `undefined`: Creates a \"no-op\" circuit breaker that is always open (never trips).\n * - `true`: Creates a circuit breaker with default settings (threshold: 5, timeout: 30000ms).\n * - `CircuitBreaker`: Reuses an existing `CircuitBreaker` instance.\n * - `{ threshold?: number; timeout?: number; }`: Creates a circuit breaker with the specified threshold and timeout.\n *\n * @returns A `CircuitBreaker` instance.\n *\n * @example\n * // Create a circuit breaker with default settings:\n * const breaker = createCircuitBreaker();\n *\n * // Create a circuit breaker with custom settings:\n * const customBreaker = createCircuitBreaker({ threshold: 10, timeout: 60000 });\n *\n * // Share a single circuit breaker instance across multiple resources:\n * const sharedBreaker = createCircuitBreaker();\n * const resource1 = queryResource(..., { circuitBreaker: sharedBreaker });\n * const resource2 = mutationResource(..., { circuitBreaker: sharedBreaker });\n */\nexport function createCircuitBreaker(opt) {\n if (opt === false)\n return createNeverBrokenCircuitBreaker();\n if (typeof opt === 'object' && 'isClosed' in opt)\n return opt;\n return internalCeateCircuitBreaker(opt?.treshold, opt?.timeout);\n}\n//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"circuit-breaker.js","sourceRoot":"","sources":["../../../../../../packages/resource/src/lib/util/circuit-breaker.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,EAAU,MAAM,EAAE,SAAS,EAAE,MAAM,eAAe,CAAC;AAqD5E,gBAAgB;AAChB,SAAS,2BAA2B,CAClC,QAAQ,GAAG,CAAC,EACZ,YAAY,GAAG,KAAK;IAEpB,MAAM,QAAQ,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;IAC/B,MAAM,YAAY,GAAG,MAAM,CAAC,CAAC,CAAC,CAAC;IAE/B,MAAM,MAAM,GAAG,QAAQ,CAAsB,GAAG,EAAE;QAChD,IAAI,YAAY,EAAE,IAAI,QAAQ;YAAE,OAAO,QAAQ,CAAC;QAChD,OAAO,QAAQ,EAAE,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC,CAAC,MAAM,CAAC;IAC3C,CAAC,CAAC,CAAC;IAEH,MAAM,QAAQ,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,MAAM,EAAE,KAAK,QAAQ,CAAC,CAAC;IAEvD,MAAM,OAAO,GAAG,GAAG,EAAE;QACnB,YAAY,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;QACpB,QAAQ,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;IACtB,CAAC,CAAC;IAEF,MAAM,OAAO,GAAG,GAAG,EAAE;QACnB,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC;YAAE,OAAO;QACjC,QAAQ,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QACnB,YAAY,CAAC,GAAG,CAAC,QAAQ,GAAG,CAAC,CAAC,CAAC;IACjC,CAAC,CAAC;IAEF,MAAM,SAAS,GAAG,MAAM,CAAC,CAAC,OAAO,EAAE,EAAE;QACnC,IAAI,CAAC,QAAQ,EAAE;YAAE,OAAO;QAExB,MAAM,OAAO,GAAG,UAAU,CAAC,OAAO,EAAE,YAAY,CAAC,CAAC;QAElD,OAAO,OAAO,CAAC,GAAG,EAAE,CAAC,YAAY,CAAC,OAAO,CAAC,CAAC,CAAC;IAC9C,CAAC,CAAC,CAAC;IAEH,MAAM,IAAI,GAAG,GAAG,EAAE;QAChB,YAAY,CAAC,GAAG,CAAC,YAAY,EAAE,GAAG,CAAC,CAAC,CAAC;QACrC,QAAQ,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;IACtB,CAAC,CAAC;IAEF,OAAO;QACL,MAAM;QACN,QAAQ;QACR,IAAI;QACJ,OAAO;QACP,QAAQ,EAAE,OAAO;QACjB,OAAO,EAAE,GAAG,EAAE,CAAC,SAAS,CAAC,OAAO,EAAE;KACnC,CAAC;AACJ,CAAC;AAED,gBAAgB;AAChB,SAAS,+BAA+B;IACtC,OAAO;QACL,QAAQ,EAAE,QAAQ,CAAC,GAAG,EAAE,CAAC,KAAK,CAAC;QAC/B,MAAM,EAAE,MAAM,CAAC,MAAM,CAAC;QACtB,IAAI,EAAE,GAAG,EAAE;YACT,OAAO;QACT,CAAC;QACD,OAAO,EAAE,GAAG,EAAE;YACZ,OAAO;QACT,CAAC;QACD,QAAQ,EAAE,GAAG,EAAE;YACb,OAAO;QACT,CAAC;QACD,OAAO,EAAE,GAAG,EAAE;YACZ,OAAO;QACT,CAAC;KACF,CAAC;AACJ,CAAC;AAED;;;;;;;;;;;;;;;;;;;;;;GAsBG;AACH,MAAM,UAAU,oBAAoB,CAClC,GAA2B;IAE3B,IAAI,GAAG,KAAK,KAAK;QAAE,OAAO,+BAA+B,EAAE,CAAC;IAE5D,IAAI,OAAO,GAAG,KAAK,QAAQ,IAAI,UAAU,IAAI,GAAG;QAAE,OAAO,GAAG,CAAC;IAE7D,OAAO,2BAA2B,CAAC,GAAG,EAAE,QAAQ,EAAE,GAAG,EAAE,OAAO,CAAC,CAAC;AAClE,CAAC","sourcesContent":["import { computed, effect, Signal, signal, untracked } from '@angular/core';\r\n\r\n/**\r\n * Represents the possible states of a circuit breaker.\r\n * - `CLOSED`: The circuit breaker is closed, and operations are allowed to proceed.\r\n * - `OPEN`: The circuit breaker is open, and operations are blocked.\r\n * - `HALF_OPEN`: The circuit breaker is in a half-open state, allowing a limited number of operations to test if the underlying issue is resolved.\r\n */\r\ntype CircuitBreakerState = 'CLOSED' | 'OPEN' | 'HALF_OPEN';\r\n\r\n/**\r\n * Represents a circuit breaker, which monitors operations and prevents failures from cascading.\r\n */\r\nexport type CircuitBreaker = {\r\n  /**\r\n   * A signal indicating whether the circuit breaker is currently closed (allowing operations).\r\n   */\r\n  isClosed: Signal<boolean>;\r\n  /**\r\n   * A signal representing the current state of the circuit breaker.\r\n   */\r\n  status: Signal<CircuitBreakerState>;\r\n  /**\r\n   * Signals a failure to the circuit breaker.  This may cause the circuit breaker to open.\r\n   */\r\n  fail: () => void;\r\n  /**\r\n   * Signals a success to the circuit breaker.  This may cause the circuit breaker to close.\r\n   */\r\n  success: () => void;\r\n  /**\r\n   * Attempts to transition the circuit breaker to the half-open state. This is typically used\r\n   * to test if the underlying issue has been resolved after the circuit breaker has been open.\r\n   */\r\n  halfOpen: () => void;\r\n  /**\r\n   * Destroys the circuit breaker & initiates related cleanup\r\n   */\r\n  destroy: () => void;\r\n};\r\n\r\n/**\r\n * Options for creating a circuit breaker.\r\n *  - `false`: Disables circuit breaker functionality (always open).\r\n *  - true: Creates a new circuit breaker with default options.\r\n *  - `CircuitBreaker`: Provides an existing `CircuitBreaker` instance to use.\r\n *  - `{ treshold?: number; timeout?: number; }`: Creates a new circuit breaker with the specified options.\r\n */\r\nexport type CircuitBreakerOptions =\r\n  | false\r\n  | CircuitBreaker\r\n  | { treshold?: number; timeout?: number };\r\n\r\n/** @internal */\r\nfunction internalCeateCircuitBreaker(\r\n  treshold = 5,\r\n  resetTimeout = 30000,\r\n): CircuitBreaker {\r\n  const halfOpen = signal(false);\r\n  const failureCount = signal(0);\r\n\r\n  const status = computed<CircuitBreakerState>(() => {\r\n    if (failureCount() >= treshold) return 'CLOSED';\r\n    return halfOpen() ? 'HALF_OPEN' : 'OPEN';\r\n  });\r\n\r\n  const isClosed = computed(() => status() === 'CLOSED');\r\n\r\n  const success = () => {\r\n    failureCount.set(0);\r\n    halfOpen.set(false);\r\n  };\r\n\r\n  const tryOnce = () => {\r\n    if (!untracked(isClosed)) return;\r\n    halfOpen.set(true);\r\n    failureCount.set(treshold - 1);\r\n  };\r\n\r\n  const effectRef = effect((cleanup) => {\r\n    if (!isClosed()) return;\r\n\r\n    const timeout = setTimeout(tryOnce, resetTimeout);\r\n\r\n    return cleanup(() => clearTimeout(timeout));\r\n  });\r\n\r\n  const fail = () => {\r\n    failureCount.set(failureCount() + 1);\r\n    halfOpen.set(false);\r\n  };\r\n\r\n  return {\r\n    status,\r\n    isClosed,\r\n    fail,\r\n    success,\r\n    halfOpen: tryOnce,\r\n    destroy: () => effectRef.destroy(),\r\n  };\r\n}\r\n\r\n/** @internal */\r\nfunction createNeverBrokenCircuitBreaker(): CircuitBreaker {\r\n  return {\r\n    isClosed: computed(() => false),\r\n    status: signal('OPEN'),\r\n    fail: () => {\r\n      // noop\r\n    },\r\n    success: () => {\r\n      // noop\r\n    },\r\n    halfOpen: () => {\r\n      // noop\r\n    },\r\n    destroy: () => {\r\n      // noop\r\n    },\r\n  };\r\n}\r\n\r\n/**\r\n * Creates a circuit breaker instance.\r\n *\r\n * @param options - Configuration options for the circuit breaker.  Can be:\r\n *   - `undefined`:  Creates a \"no-op\" circuit breaker that is always open (never trips).\r\n *   - `true`: Creates a circuit breaker with default settings (threshold: 5, timeout: 30000ms).\r\n *   - `CircuitBreaker`:  Reuses an existing `CircuitBreaker` instance.\r\n *   - `{ threshold?: number; timeout?: number; }`: Creates a circuit breaker with the specified threshold and timeout.\r\n *\r\n * @returns A `CircuitBreaker` instance.\r\n *\r\n * @example\r\n * // Create a circuit breaker with default settings:\r\n * const breaker = createCircuitBreaker();\r\n *\r\n * // Create a circuit breaker with custom settings:\r\n * const customBreaker = createCircuitBreaker({ threshold: 10, timeout: 60000 });\r\n *\r\n * // Share a single circuit breaker instance across multiple resources:\r\n * const sharedBreaker = createCircuitBreaker();\r\n * const resource1 = queryResource(..., { circuitBreaker: sharedBreaker });\r\n * const resource2 = mutationResource(..., { circuitBreaker: sharedBreaker });\r\n */\r\nexport function createCircuitBreaker(\r\n  opt?: CircuitBreakerOptions,\r\n): CircuitBreaker {\r\n  if (opt === false) return createNeverBrokenCircuitBreaker();\r\n\r\n  if (typeof opt === 'object' && 'isClosed' in opt) return opt;\r\n\r\n  return internalCeateCircuitBreaker(opt?.treshold, opt?.timeout);\r\n}\r\n"]}","// Heavily inspired by: https://dev.to/kasual1/request-deduplication-in-angular-3pd8\nimport { HttpContext, HttpContextToken, } from '@angular/common/http';\nimport { finalize, shareReplay } from 'rxjs';\nconst NO_DEDUPE = new HttpContextToken(() => false);\n/**\n * Disables request deduplication for a specific HTTP request.\n *\n * @param ctx - The `HttpContext` to modify. If not provided, a new `HttpContext` is created.\n * @returns The modified `HttpContext` with the `NO_DEDUPE` token set to `true`.\n *\n * @example\n * // Disable deduplication for a specific POST request:\n * const context = noDedupe();\n * this.http.post('/api/data', payload, { context }).subscribe(...);\n *\n * // Disable deduplication, modifying an existing context:\n * let context = new HttpContext();\n * context = noDedupe(context);\n * this.http.post('/api/data', payload, { context }).subscribe(...);\n */\nexport function noDedupe(ctx = new HttpContext()) {\n return ctx.set(NO_DEDUPE, true);\n}\n/**\n * Creates an `HttpInterceptorFn` that deduplicates identical HTTP requests.\n * If multiple identical requests (same URL and parameters) are made concurrently,\n * only the first request will be sent to the server. Subsequent requests will\n * receive the response from the first request.\n *\n * @param allowed - An array of HTTP methods for which deduplication should be enabled.\n * Defaults to `['GET', 'DELETE', 'HEAD', 'OPTIONS']`.\n *\n * @returns An `HttpInterceptorFn` that implements the request deduplication logic.\n *\n * @example\n * // In your app.config.ts or module providers:\n * import { provideHttpClient, withInterceptors } from '@angular/common/http';\n * import { createDedupeRequestsInterceptor } from './your-dedupe-interceptor';\n *\n * export const appConfig: ApplicationConfig = {\n * providers: [\n * provideHttpClient(withInterceptors([createDedupeRequestsInterceptor()])),\n * // ... other providers\n * ],\n * };\n *\n * // You can also specify which methods should be deduped\n * export const appConfig: ApplicationConfig = {\n * providers: [\n * provideHttpClient(withInterceptors([createDedupeRequestsInterceptor(['GET'])])), // only dedupe GET calls\n * // ... other providers\n * ],\n * };\n */\nexport function createDedupeRequestsInterceptor(allowed = ['GET', 'DELETE', 'HEAD', 'OPTIONS']) {\n const inFlight = new Map();\n const DEDUPE_METHODS = new Set(allowed);\n return (req, next) => {\n if (!DEDUPE_METHODS.has(req.method) || req.context.get(NO_DEDUPE))\n return next(req);\n const found = inFlight.get(req.urlWithParams);\n if (found)\n return found;\n const request = next(req).pipe(finalize(() => inFlight.delete(req.urlWithParams)), shareReplay());\n inFlight.set(req.urlWithParams, request);\n return request;\n };\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGVkdXBlLmludGVyY2VwdG9yLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcGFja2FnZXMvcmVzb3VyY2Uvc3JjL2xpYi91dGlsL2RlZHVwZS5pbnRlcmNlcHRvci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxvRkFBb0Y7QUFFcEYsT0FBTyxFQUNMLFdBQVcsRUFDWCxnQkFBZ0IsR0FLakIsTUFBTSxzQkFBc0IsQ0FBQztBQUM5QixPQUFPLEVBQUUsUUFBUSxFQUFFLFdBQVcsRUFBbUIsTUFBTSxNQUFNLENBQUM7QUFFOUQsTUFBTSxTQUFTLEdBQUcsSUFBSSxnQkFBZ0IsQ0FBVSxHQUFHLEVBQUUsQ0FBQyxLQUFLLENBQUMsQ0FBQztBQUU3RDs7Ozs7Ozs7Ozs7Ozs7O0dBZUc7QUFDSCxNQUFNLFVBQVUsUUFBUSxDQUFDLE1BQW1CLElBQUksV0FBVyxFQUFFO0lBQzNELE9BQU8sR0FBRyxDQUFDLEdBQUcsQ0FBQyxTQUFTLEVBQUUsSUFBSSxDQUFDLENBQUM7QUFDbEMsQ0FBQztBQUVEOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0E4Qkc7QUFDSCxNQUFNLFVBQVUsK0JBQStCLENBQzdDLE9BQU8sR0FBRyxDQUFDLEtBQUssRUFBRSxRQUFRLEVBQUUsTUFBTSxFQUFFLFNBQVMsQ0FBQztJQUU5QyxNQUFNLFFBQVEsR0FBRyxJQUFJLEdBQUcsRUFBMEMsQ0FBQztJQUVuRSxNQUFNLGNBQWMsR0FBRyxJQUFJLEdBQUcsQ0FBUyxPQUFPLENBQUMsQ0FBQztJQUVoRCxPQUFPLENBQ0wsR0FBeUIsRUFDekIsSUFBbUIsRUFDYSxFQUFFO1FBQ2xDLElBQUksQ0FBQyxjQUFjLENBQUMsR0FBRyxDQUFDLEdBQUcsQ0FBQyxNQUFNLENBQUMsSUFBSSxHQUFHLENBQUMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxTQUFTLENBQUM7WUFDL0QsT0FBTyxJQUFJLENBQUMsR0FBRyxDQUFDLENBQUM7UUFFbkIsTUFBTSxLQUFLLEdBQUcsUUFBUSxDQUFDLEdBQUcsQ0FBQyxHQUFHLENBQUMsYUFBYSxDQUFDLENBQUM7UUFFOUMsSUFBSSxLQUFLO1lBQUUsT0FBTyxLQUFLLENBQUM7UUFFeEIsTUFBTSxPQUFPLEdBQUcsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDLElBQUksQ0FDNUIsUUFBUSxDQUFDLEdBQUcsRUFBRSxDQUFDLFFBQVEsQ0FBQyxNQUFNLENBQUMsR0FBRyxDQUFDLGFBQWEsQ0FBQyxDQUFDLEVBQ2xELFdBQVcsRUFBRSxDQUNkLENBQUM7UUFDRixRQUFRLENBQUMsR0FBRyxDQUFDLEdBQUcsQ0FBQyxhQUFhLEVBQUUsT0FBTyxDQUFDLENBQUM7UUFFekMsT0FBTyxPQUFPLENBQUM7SUFDakIsQ0FBQyxDQUFDO0FBQ0osQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8vIEhlYXZpbHkgaW5zcGlyZWQgYnk6IGh0dHBzOi8vZGV2LnRvL2thc3VhbDEvcmVxdWVzdC1kZWR1cGxpY2F0aW9uLWluLWFuZ3VsYXItM3BkOFxyXG5cclxuaW1wb3J0IHtcclxuICBIdHRwQ29udGV4dCxcclxuICBIdHRwQ29udGV4dFRva2VuLFxyXG4gIEh0dHBJbnRlcmNlcHRvckZuLFxyXG4gIHR5cGUgSHR0cEV2ZW50LFxyXG4gIHR5cGUgSHR0cEhhbmRsZXJGbixcclxuICB0eXBlIEh0dHBSZXF1ZXN0LFxyXG59IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbi9odHRwJztcclxuaW1wb3J0IHsgZmluYWxpemUsIHNoYXJlUmVwbGF5LCB0eXBlIE9ic2VydmFibGUgfSBmcm9tICdyeGpzJztcclxuXHJcbmNvbnN0IE5PX0RFRFVQRSA9IG5ldyBIdHRwQ29udGV4dFRva2VuPGJvb2xlYW4+KCgpID0+IGZhbHNlKTtcclxuXHJcbi8qKlxyXG4gKiBEaXNhYmxlcyByZXF1ZXN0IGRlZHVwbGljYXRpb24gZm9yIGEgc3BlY2lmaWMgSFRUUCByZXF1ZXN0LlxyXG4gKlxyXG4gKiBAcGFyYW0gY3R4IC0gVGhlIGBIdHRwQ29udGV4dGAgdG8gbW9kaWZ5LiBJZiBub3QgcHJvdmlkZWQsIGEgbmV3IGBIdHRwQ29udGV4dGAgaXMgY3JlYXRlZC5cclxuICogQHJldHVybnMgVGhlIG1vZGlmaWVkIGBIdHRwQ29udGV4dGAgd2l0aCB0aGUgYE5PX0RFRFVQRWAgdG9rZW4gc2V0IHRvIGB0cnVlYC5cclxuICpcclxuICogQGV4YW1wbGVcclxuICogLy8gRGlzYWJsZSBkZWR1cGxpY2F0aW9uIGZvciBhIHNwZWNpZmljIFBPU1QgcmVxdWVzdDpcclxuICogY29uc3QgY29udGV4dCA9IG5vRGVkdXBlKCk7XHJcbiAqIHRoaXMuaHR0cC5wb3N0KCcvYXBpL2RhdGEnLCBwYXlsb2FkLCB7IGNvbnRleHQgfSkuc3Vic2NyaWJlKC4uLik7XHJcbiAqXHJcbiAqIC8vIERpc2FibGUgZGVkdXBsaWNhdGlvbiwgbW9kaWZ5aW5nIGFuIGV4aXN0aW5nIGNvbnRleHQ6XHJcbiAqIGxldCBjb250ZXh0ID0gbmV3IEh0dHBDb250ZXh0KCk7XHJcbiAqIGNvbnRleHQgPSBub0RlZHVwZShjb250ZXh0KTtcclxuICogdGhpcy5odHRwLnBvc3QoJy9hcGkvZGF0YScsIHBheWxvYWQsIHsgY29udGV4dCB9KS5zdWJzY3JpYmUoLi4uKTtcclxuICovXHJcbmV4cG9ydCBmdW5jdGlvbiBub0RlZHVwZShjdHg6IEh0dHBDb250ZXh0ID0gbmV3IEh0dHBDb250ZXh0KCkpIHtcclxuICByZXR1cm4gY3R4LnNldChOT19ERURVUEUsIHRydWUpO1xyXG59XHJcblxyXG4vKipcclxuICogQ3JlYXRlcyBhbiBgSHR0cEludGVyY2VwdG9yRm5gIHRoYXQgZGVkdXBsaWNhdGVzIGlkZW50aWNhbCBIVFRQIHJlcXVlc3RzLlxyXG4gKiBJZiBtdWx0aXBsZSBpZGVudGljYWwgcmVxdWVzdHMgKHNhbWUgVVJMIGFuZCBwYXJhbWV0ZXJzKSBhcmUgbWFkZSBjb25jdXJyZW50bHksXHJcbiAqIG9ubHkgdGhlIGZpcnN0IHJlcXVlc3Qgd2lsbCBiZSBzZW50IHRvIHRoZSBzZXJ2ZXIuIFN1YnNlcXVlbnQgcmVxdWVzdHMgd2lsbFxyXG4gKiByZWNlaXZlIHRoZSByZXNwb25zZSBmcm9tIHRoZSBmaXJzdCByZXF1ZXN0LlxyXG4gKlxyXG4gKiBAcGFyYW0gYWxsb3dlZCAtIEFuIGFycmF5IG9mIEhUVFAgbWV0aG9kcyBmb3Igd2hpY2ggZGVkdXBsaWNhdGlvbiBzaG91bGQgYmUgZW5hYmxlZC5cclxuICogICAgICAgICAgICAgICAgICBEZWZhdWx0cyB0byBgWydHRVQnLCAnREVMRVRFJywgJ0hFQUQnLCAnT1BUSU9OUyddYC5cclxuICpcclxuICogQHJldHVybnMgQW4gYEh0dHBJbnRlcmNlcHRvckZuYCB0aGF0IGltcGxlbWVudHMgdGhlIHJlcXVlc3QgZGVkdXBsaWNhdGlvbiBsb2dpYy5cclxuICpcclxuICogQGV4YW1wbGVcclxuICogLy8gSW4geW91ciBhcHAuY29uZmlnLnRzIG9yIG1vZHVsZSBwcm92aWRlcnM6XHJcbiAqIGltcG9ydCB7IHByb3ZpZGVIdHRwQ2xpZW50LCB3aXRoSW50ZXJjZXB0b3JzIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uL2h0dHAnO1xyXG4gKiBpbXBvcnQgeyBjcmVhdGVEZWR1cGVSZXF1ZXN0c0ludGVyY2VwdG9yIH0gZnJvbSAnLi95b3VyLWRlZHVwZS1pbnRlcmNlcHRvcic7XHJcbiAqXHJcbiAqIGV4cG9ydCBjb25zdCBhcHBDb25maWc6IEFwcGxpY2F0aW9uQ29uZmlnID0ge1xyXG4gKiAgIHByb3ZpZGVyczogW1xyXG4gKiAgICAgcHJvdmlkZUh0dHBDbGllbnQod2l0aEludGVyY2VwdG9ycyhbY3JlYXRlRGVkdXBlUmVxdWVzdHNJbnRlcmNlcHRvcigpXSkpLFxyXG4gKiAgICAgLy8gLi4uIG90aGVyIHByb3ZpZGVyc1xyXG4gKiAgIF0sXHJcbiAqIH07XHJcbiAqXHJcbiAqIC8vIFlvdSBjYW4gYWxzbyBzcGVjaWZ5IHdoaWNoIG1ldGhvZHMgc2hvdWxkIGJlIGRlZHVwZWRcclxuICogIGV4cG9ydCBjb25zdCBhcHBDb25maWc6IEFwcGxpY2F0aW9uQ29uZmlnID0ge1xyXG4gKiAgIHByb3ZpZGVyczogW1xyXG4gKiAgICAgcHJvdmlkZUh0dHBDbGllbnQod2l0aEludGVyY2VwdG9ycyhbY3JlYXRlRGVkdXBlUmVxdWVzdHNJbnRlcmNlcHRvcihbJ0dFVCddKV0pKSwgLy8gb25seSBkZWR1cGUgR0VUIGNhbGxzXHJcbiAqICAgICAvLyAuLi4gb3RoZXIgcHJvdmlkZXJzXHJcbiAqICAgXSxcclxuICogfTtcclxuICovXHJcbmV4cG9ydCBmdW5jdGlvbiBjcmVhdGVEZWR1cGVSZXF1ZXN0c0ludGVyY2VwdG9yKFxyXG4gIGFsbG93ZWQgPSBbJ0dFVCcsICdERUxFVEUnLCAnSEVBRCcsICdPUFRJT05TJ10sXHJcbik6IEh0dHBJbnRlcmNlcHRvckZuIHtcclxuICBjb25zdCBpbkZsaWdodCA9IG5ldyBNYXA8c3RyaW5nLCBPYnNlcnZhYmxlPEh0dHBFdmVudDx1bmtub3duPj4+KCk7XHJcblxyXG4gIGNvbnN0IERFRFVQRV9NRVRIT0RTID0gbmV3IFNldDxzdHJpbmc+KGFsbG93ZWQpO1xyXG5cclxuICByZXR1cm4gKFxyXG4gICAgcmVxOiBIdHRwUmVxdWVzdDx1bmtub3duPixcclxuICAgIG5leHQ6IEh0dHBIYW5kbGVyRm4sXHJcbiAgKTogT2JzZXJ2YWJsZTxIdHRwRXZlbnQ8dW5rbm93bj4+ID0+IHtcclxuICAgIGlmICghREVEVVBFX01FVEhPRFMuaGFzKHJlcS5tZXRob2QpIHx8IHJlcS5jb250ZXh0LmdldChOT19ERURVUEUpKVxyXG4gICAgICByZXR1cm4gbmV4dChyZXEpO1xyXG5cclxuICAgIGNvbnN0IGZvdW5kID0gaW5GbGlnaHQuZ2V0KHJlcS51cmxXaXRoUGFyYW1zKTtcclxuXHJcbiAgICBpZiAoZm91bmQpIHJldHVybiBmb3VuZDtcclxuXHJcbiAgICBjb25zdCByZXF1ZXN0ID0gbmV4dChyZXEpLnBpcGUoXHJcbiAgICAgIGZpbmFsaXplKCgpID0+IGluRmxpZ2h0LmRlbGV0ZShyZXEudXJsV2l0aFBhcmFtcykpLFxyXG4gICAgICBzaGFyZVJlcGxheSgpLFxyXG4gICAgKTtcclxuICAgIGluRmxpZ2h0LnNldChyZXEudXJsV2l0aFBhcmFtcywgcmVxdWVzdCk7XHJcblxyXG4gICAgcmV0dXJuIHJlcXVlc3Q7XHJcbiAgfTtcclxufVxyXG4iXX0=","import { hash, keys } from '@mmstack/object';\nfunction equalTransferCache(a, b) {\n if (!a && !b)\n return true;\n if (!a || !b)\n return false;\n if (typeof a !== typeof b)\n return false;\n if (typeof a === 'boolean' || typeof b === 'boolean')\n return a === b;\n if (!a.includeHeaders && !b.includeHeaders)\n return true;\n if (!a.includeHeaders || !b.includeHeaders)\n return false;\n if (a.includeHeaders.length !== b.includeHeaders.length)\n return false;\n if (a.includeHeaders.length === 0)\n return true;\n const aSet = new Set(a.includeHeaders ?? []);\n return b.includeHeaders.every((header) => aSet.has(header));\n}\nfunction equalParams(a, b) {\n if (!a && !b)\n return true;\n if (!a || !b)\n return false;\n const aKeys = keys(a);\n const bKeys = keys(b);\n if (aKeys.length !== bKeys.length)\n return false;\n return aKeys.every((key) => a[key] === b[key]);\n}\nfunction equalBody(a, b) {\n if (!a && !b)\n return true;\n if (!a || !b)\n return false;\n return hash(a) === hash(b);\n}\nfunction equalHeaders(a, b) {\n if (!a && !b)\n return true;\n if (!a || !b)\n return false;\n const aKeys = keys(a);\n const bKeys = keys(b);\n if (aKeys.length !== bKeys.length)\n return false;\n return aKeys.every((key) => a[key] === b[key]);\n}\nfunction equalContext(a, b) {\n if (!a && !b)\n return true;\n if (!a || !b)\n return false;\n const aKeys = keys(a);\n const bKeys = keys(b);\n if (aKeys.length !== bKeys.length)\n return false;\n return aKeys.every((key) => a[key] === b[key]);\n}\nexport function createEqualRequest(equalResult) {\n const eqb = equalResult ?? equalBody;\n return (a, b) => {\n if (!a && !b)\n return true;\n if (!a || !b)\n return false;\n if (a.url !== b.url)\n return false;\n if (a.method !== b.method)\n return false;\n if (!equalParams(a.params, b.params))\n return false;\n if (!equalHeaders(a.headers, b.headers))\n return false;\n if (!eqb(a.body, b.body))\n return false;\n if (!equalContext(a.context, b.context))\n return false;\n if (a.withCredentials !== b.withCredentials)\n return false;\n if (a.reportProgress !== b.reportProgress)\n return false;\n if (!equalTransferCache(a.transferCache, b.transferCache))\n return false;\n return true;\n };\n}\n//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"equality.js","sourceRoot":"","sources":["../../../../../../packages/resource/src/lib/util/equality.ts"],"names":[],"mappings":"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","sourcesContent":["import { HttpResourceRequest } from '@angular/common/http';\r\nimport { type ValueEqualityFn } from '@angular/core';\r\nimport { hash, keys } from '@mmstack/object';\r\n\r\nfunction equalTransferCache(\r\n  a: HttpResourceRequest['transferCache'],\r\n  b: HttpResourceRequest['transferCache'],\r\n): boolean {\r\n  if (!a && !b) return true;\r\n  if (!a || !b) return false;\r\n\r\n  if (typeof a !== typeof b) return false;\r\n  if (typeof a === 'boolean' || typeof b === 'boolean') return a === b;\r\n\r\n  if (!a.includeHeaders && !b.includeHeaders) return true;\r\n  if (!a.includeHeaders || !b.includeHeaders) return false;\r\n\r\n  if (a.includeHeaders.length !== b.includeHeaders.length) return false;\r\n\r\n  if (a.includeHeaders.length === 0) return true;\r\n\r\n  const aSet = new Set(a.includeHeaders ?? []);\r\n\r\n  return b.includeHeaders.every((header) => aSet.has(header));\r\n}\r\n\r\nfunction equalParams(\r\n  a: HttpResourceRequest['params'],\r\n  b: HttpResourceRequest['params'],\r\n): boolean {\r\n  if (!a && !b) return true;\r\n  if (!a || !b) return false;\r\n  const aKeys = keys(a);\r\n  const bKeys = keys(b);\r\n  if (aKeys.length !== bKeys.length) return false;\r\n\r\n  return aKeys.every((key) => a[key] === b[key]);\r\n}\r\n\r\nfunction equalBody(\r\n  a: HttpResourceRequest['body'],\r\n  b: HttpResourceRequest['body'],\r\n): boolean {\r\n  if (!a && !b) return true;\r\n  if (!a || !b) return false;\r\n  return hash(a) === hash(b);\r\n}\r\n\r\nfunction equalHeaders(\r\n  a: HttpResourceRequest['headers'],\r\n  b: HttpResourceRequest['headers'],\r\n): boolean {\r\n  if (!a && !b) return true;\r\n  if (!a || !b) return false;\r\n\r\n  const aKeys = keys(a);\r\n  const bKeys = keys(b);\r\n  if (aKeys.length !== bKeys.length) return false;\r\n  return aKeys.every((key) => a[key] === b[key]);\r\n}\r\n\r\nfunction equalContext(\r\n  a: HttpResourceRequest['context'],\r\n  b: HttpResourceRequest['context'],\r\n): boolean {\r\n  if (!a && !b) return true;\r\n  if (!a || !b) return false;\r\n\r\n  const aKeys = keys(a);\r\n  const bKeys = keys(b);\r\n  if (aKeys.length !== bKeys.length) return false;\r\n  return aKeys.every((key) => a[key] === b[key]);\r\n}\r\n\r\nexport function createEqualRequest<TResult>(\r\n  equalResult?: ValueEqualityFn<TResult>,\r\n) {\r\n  const eqb = equalResult ?? equalBody;\r\n\r\n  return (\r\n    a: Partial<HttpResourceRequest> | undefined,\r\n    b: Partial<HttpResourceRequest> | undefined,\r\n  ) => {\r\n    if (!a && !b) return true;\r\n    if (!a || !b) return false;\r\n\r\n    if (a.url !== b.url) return false;\r\n    if (a.method !== b.method) return false;\r\n    if (!equalParams(a.params, b.params)) return false;\r\n    if (!equalHeaders(a.headers, b.headers)) return false;\r\n    if (!eqb(a.body as TResult, b.body as TResult)) return false;\r\n    if (!equalContext(a.context, b.context)) return false;\r\n\r\n    if (a.withCredentials !== b.withCredentials) return false;\r\n    if (a.reportProgress !== b.reportProgress) return false;\r\n    if (!equalTransferCache(a.transferCache, b.transferCache)) return false;\r\n\r\n    return true;\r\n  };\r\n}\r\n"]}","export function hasSlowConnection() {\n if (window &&\n 'navigator' in window &&\n 'connection' in window.navigator &&\n typeof window.navigator.connection === 'object' &&\n !!window.navigator.connection &&\n 'effectiveType' in window.navigator.connection &&\n typeof window.navigator.connection.effectiveType === 'string')\n return window.navigator.connection.effectiveType.endsWith('2g');\n return false;\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaGFzLXNsb3ctY29ubmVjdGlvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3BhY2thZ2VzL3Jlc291cmNlL3NyYy9saWIvdXRpbC9oYXMtc2xvdy1jb25uZWN0aW9uLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE1BQU0sVUFBVSxpQkFBaUI7SUFDL0IsSUFDRSxNQUFNO1FBQ04sV0FBVyxJQUFJLE1BQU07UUFDckIsWUFBWSxJQUFJLE1BQU0sQ0FBQyxTQUFTO1FBQ2hDLE9BQU8sTUFBTSxDQUFDLFNBQVMsQ0FBQyxVQUFVLEtBQUssUUFBUTtRQUMvQyxDQUFDLENBQUMsTUFBTSxDQUFDLFNBQVMsQ0FBQyxVQUFVO1FBQzdCLGVBQWUsSUFBSSxNQUFNLENBQUMsU0FBUyxDQUFDLFVBQVU7UUFDOUMsT0FBTyxNQUFNLENBQUMsU0FBUyxDQUFDLFVBQVUsQ0FBQyxhQUFhLEtBQUssUUFBUTtRQUU3RCxPQUFPLE1BQU0sQ0FBQyxTQUFTLENBQUMsVUFBVSxDQUFDLGFBQWEsQ0FBQyxRQUFRLENBQUMsSUFBSSxDQUFDLENBQUM7SUFFbEUsT0FBTyxLQUFLLENBQUM7QUFDZixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGZ1bmN0aW9uIGhhc1Nsb3dDb25uZWN0aW9uKCkge1xyXG4gIGlmIChcclxuICAgIHdpbmRvdyAmJlxyXG4gICAgJ25hdmlnYXRvcicgaW4gd2luZG93ICYmXHJcbiAgICAnY29ubmVjdGlvbicgaW4gd2luZG93Lm5hdmlnYXRvciAmJlxyXG4gICAgdHlwZW9mIHdpbmRvdy5uYXZpZ2F0b3IuY29ubmVjdGlvbiA9PT0gJ29iamVjdCcgJiZcclxuICAgICEhd2luZG93Lm5hdmlnYXRvci5jb25uZWN0aW9uICYmXHJcbiAgICAnZWZmZWN0aXZlVHlwZScgaW4gd2luZG93Lm5hdmlnYXRvci5jb25uZWN0aW9uICYmXHJcbiAgICB0eXBlb2Ygd2luZG93Lm5hdmlnYXRvci5jb25uZWN0aW9uLmVmZmVjdGl2ZVR5cGUgPT09ICdzdHJpbmcnXHJcbiAgKVxyXG4gICAgcmV0dXJuIHdpbmRvdy5uYXZpZ2F0b3IuY29ubmVjdGlvbi5lZmZlY3RpdmVUeXBlLmVuZHNXaXRoKCcyZycpO1xyXG5cclxuICByZXR1cm4gZmFsc2U7XHJcbn1cclxuIl19","import { linkedSignal, } from '@angular/core';\nfunction presist(value, usePrevious, equal) {\n // linkedSignal allows us to access previous source value\n const persisted = linkedSignal({\n source: () => {\n return {\n value: value(),\n usePrevious: usePrevious(),\n };\n },\n computation: (source, prev) => {\n if (source.usePrevious && prev)\n return prev.value;\n return source.value;\n },\n equal,\n });\n // if original value was WritableSignal then override linkedSignal methods to original...angular uses linkedSignal under the hood in ResourceImpl, this applies to that.\n if ('set' in value) {\n persisted.set = value.set;\n persisted.update = value.update;\n persisted.asReadonly = value.asReadonly;\n }\n return persisted;\n}\nexport function persistResourceValues(resource, persist = false, equal) {\n if (!persist)\n return resource;\n return {\n ...resource,\n statusCode: presist(resource.statusCode, resource.isLoading),\n headers: presist(resource.headers, resource.isLoading),\n value: presist(resource.value, resource.isLoading, equal),\n };\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGVyc2lzdC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3BhY2thZ2VzL3Jlc291cmNlL3NyYy9saWIvdXRpbC9wZXJzaXN0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUNBLE9BQU8sRUFDTCxZQUFZLEdBSWIsTUFBTSxlQUFlLENBQUM7QUFjdkIsU0FBUyxPQUFPLENBQ2QsS0FBb0MsRUFDcEMsV0FBNEIsRUFDNUIsS0FBMEI7SUFFMUIseURBQXlEO0lBRXpELE1BQU0sU0FBUyxHQUFHLFlBQVksQ0FNNUI7UUFDQSxNQUFNLEVBQUUsR0FBRyxFQUFFO1lBQ1gsT0FBTztnQkFDTCxLQUFLLEVBQUUsS0FBSyxFQUFFO2dCQUNkLFdBQVcsRUFBRSxXQUFXLEVBQUU7YUFDM0IsQ0FBQztRQUNKLENBQUM7UUFDRCxXQUFXLEVBQUUsQ0FBQyxNQUFNLEVBQUUsSUFBSSxFQUFFLEVBQUU7WUFDNUIsSUFBSSxNQUFNLENBQUMsV0FBVyxJQUFJLElBQUk7Z0JBQUUsT0FBTyxJQUFJLENBQUMsS0FBSyxDQUFDO1lBRWxELE9BQU8sTUFBTSxDQUFDLEtBQUssQ0FBQztRQUN0QixDQUFDO1FBQ0QsS0FBSztLQUNOLENBQUMsQ0FBQztJQUVILHdLQUF3SztJQUN4SyxJQUFJLEtBQUssSUFBSSxLQUFLLEVBQUUsQ0FBQztRQUNuQixTQUFTLENBQUMsR0FBRyxHQUFHLEtBQUssQ0FBQyxHQUFHLENBQUM7UUFDMUIsU0FBUyxDQUFDLE1BQU0sR0FBRyxLQUFLLENBQUMsTUFBTSxDQUFDO1FBQ2hDLFNBQVMsQ0FBQyxVQUFVLEdBQUcsS0FBSyxDQUFDLFVBQVUsQ0FBQztJQUMxQyxDQUFDO0lBRUQsT0FBTyxTQUFTLENBQUM7QUFDbkIsQ0FBQztBQUVELE1BQU0sVUFBVSxxQkFBcUIsQ0FDbkMsUUFBNEIsRUFDNUIsT0FBTyxHQUFHLEtBQUssRUFDZixLQUEwQjtJQUUxQixJQUFJLENBQUMsT0FBTztRQUFFLE9BQU8sUUFBUSxDQUFDO0lBRTlCLE9BQU87UUFDTCxHQUFHLFFBQVE7UUFDWCxVQUFVLEVBQUUsT0FBTyxDQUNqQixRQUFRLENBQUMsVUFBVSxFQUNuQixRQUFRLENBQUMsU0FBUyxDQUNuQjtRQUNELE9BQU8sRUFBRSxPQUFPLENBQ2QsUUFBUSxDQUFDLE9BQU8sRUFDaEIsUUFBUSxDQUFDLFNBQVMsQ0FDbkI7UUFDRCxLQUFLLEVBQUUsT0FBTyxDQUFJLFFBQVEsQ0FBQyxLQUFLLEVBQUUsUUFBUSxDQUFDLFNBQVMsRUFBRSxLQUFLLENBQUM7S0FDN0QsQ0FBQztBQUNKLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyB0eXBlIEh0dHBIZWFkZXJzLCB0eXBlIEh0dHBSZXNvdXJjZVJlZiB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbi9odHRwJztcclxuaW1wb3J0IHtcclxuICBsaW5rZWRTaWduYWwsXHJcbiAgdHlwZSBTaWduYWwsXHJcbiAgdHlwZSBWYWx1ZUVxdWFsaXR5Rm4sXHJcbiAgdHlwZSBXcml0YWJsZVNpZ25hbCxcclxufSBmcm9tICdAYW5ndWxhci9jb3JlJztcclxuXHJcbmZ1bmN0aW9uIHByZXNpc3Q8VD4oXHJcbiAgdmFsdWU6IFdyaXRhYmxlU2lnbmFsPFQ+LFxyXG4gIHVzZVByZXZpb3VzOiBTaWduYWw8Ym9vbGVhbj4sXHJcbiAgZXF1YWw/OiBWYWx1ZUVxdWFsaXR5Rm48VD4sXHJcbik6IFdyaXRhYmxlU2lnbmFsPFQ+O1xyXG5cclxuZnVuY3Rpb24gcHJlc2lzdDxUPihcclxuICB2YWx1ZTogU2lnbmFsPFQ+LFxyXG4gIHVzZVByZXZpb3VzOiBTaWduYWw8Ym9vbGVhbj4sXHJcbiAgZXF1YWw/OiBWYWx1ZUVxdWFsaXR5Rm48VD4sXHJcbik6IFNpZ25hbDxUPjtcclxuXHJcbmZ1bmN0aW9uIHByZXNpc3Q8VD4oXHJcbiAgdmFsdWU6IFdyaXRhYmxlU2lnbmFsPFQ+IHwgU2lnbmFsPFQ+LFxyXG4gIHVzZVByZXZpb3VzOiBTaWduYWw8Ym9vbGVhbj4sXHJcbiAgZXF1YWw/OiBWYWx1ZUVxdWFsaXR5Rm48VD4sXHJcbik6IFdyaXRhYmxlU2lnbmFsPFQ+IHwgU2lnbmFsPFQ+IHtcclxuICAvLyBsaW5rZWRTaWduYWwgYWxsb3dzIHVzIHRvIGFjY2VzcyBwcmV2aW91cyBzb3VyY2UgdmFsdWVcclxuXHJcbiAgY29uc3QgcGVyc2lzdGVkID0gbGlua2VkU2lnbmFsPFxyXG4gICAge1xyXG4gICAgICB2YWx1ZTogVDtcclxuICAgICAgdXNlUHJldmlvdXM6IGJvb2xlYW47XHJcbiAgICB9LFxyXG4gICAgVFxyXG4gID4oe1xyXG4gICAgc291cmNlOiAoKSA9PiB7XHJcbiAgICAgIHJldHVybiB7XHJcbiAgICAgICAgdmFsdWU6IHZhbHVlKCksXHJcbiAgICAgICAgdXNlUHJldmlvdXM6IHVzZVByZXZpb3VzKCksXHJcbiAgICAgIH07XHJcbiAgICB9LFxyXG4gICAgY29tcHV0YXRpb246IChzb3VyY2UsIHByZXYpID0+IHtcclxuICAgICAgaWYgKHNvdXJjZS51c2VQcmV2aW91cyAmJiBwcmV2KSByZXR1cm4gcHJldi52YWx1ZTtcclxuXHJcbiAgICAgIHJldHVybiBzb3VyY2UudmFsdWU7XHJcbiAgICB9LFxyXG4gICAgZXF1YWwsXHJcbiAgfSk7XHJcblxyXG4gIC8vIGlmIG9yaWdpbmFsIHZhbHVlIHdhcyBXcml0YWJsZVNpZ25hbCB0aGVuIG92ZXJyaWRlIGxpbmtlZFNpZ25hbCBtZXRob2RzIHRvIG9yaWdpbmFsLi4uYW5ndWxhciB1c2VzIGxpbmtlZFNpZ25hbCB1bmRlciB0aGUgaG9vZCBpbiBSZXNvdXJjZUltcGwsIHRoaXMgYXBwbGllcyB0byB0aGF0LlxyXG4gIGlmICgnc2V0JyBpbiB2YWx1ZSkge1xyXG4gICAgcGVyc2lzdGVkLnNldCA9IHZhbHVlLnNldDtcclxuICAgIHBlcnNpc3RlZC51cGRhdGUgPSB2YWx1ZS51cGRhdGU7XHJcbiAgICBwZXJzaXN0ZWQuYXNSZWFkb25seSA9IHZhbHVlLmFzUmVhZG9ubHk7XHJcbiAgfVxyXG5cclxuICByZXR1cm4gcGVyc2lzdGVkO1xyXG59XHJcblxyXG5leHBvcnQgZnVuY3Rpb24gcGVyc2lzdFJlc291cmNlVmFsdWVzPFQ+KFxyXG4gIHJlc291cmNlOiBIdHRwUmVzb3VyY2VSZWY8VD4sXHJcbiAgcGVyc2lzdCA9IGZhbHNlLFxyXG4gIGVxdWFsPzogVmFsdWVFcXVhbGl0eUZuPFQ+LFxyXG4pOiBIdHRwUmVzb3VyY2VSZWY8VD4ge1xyXG4gIGlmICghcGVyc2lzdCkgcmV0dXJuIHJlc291cmNlO1xyXG5cclxuICByZXR1cm4ge1xyXG4gICAgLi4ucmVzb3VyY2UsXHJcbiAgICBzdGF0dXNDb2RlOiBwcmVzaXN0PG51bWJlciB8IHVuZGVmaW5lZD4oXHJcbiAgICAgIHJlc291cmNlLnN0YXR1c0NvZGUsXHJcbiAgICAgIHJlc291cmNlLmlzTG9hZGluZyxcclxuICAgICksXHJcbiAgICBoZWFkZXJzOiBwcmVzaXN0PEh0dHBIZWFkZXJzIHwgdW5kZWZpbmVkPihcclxuICAgICAgcmVzb3VyY2UuaGVhZGVycyxcclxuICAgICAgcmVzb3VyY2UuaXNMb2FkaW5nLFxyXG4gICAgKSxcclxuICAgIHZhbHVlOiBwcmVzaXN0PFQ+KHJlc291cmNlLnZhbHVlLCByZXNvdXJjZS5pc0xvYWRpbmcsIGVxdWFsKSxcclxuICB9O1xyXG59XHJcbiJdfQ==","import { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { interval } from 'rxjs';\n// refresh resource every n miliseconds or don't refresh if undefined provided. 0 also excluded, due to it not being a valid usecase\nexport function refresh(resource, destroyRef, refresh) {\n if (!refresh)\n return resource; // no refresh requested\n // we can use RxJs here as reloading the resource will always be a side effect & as such does not impact the reactive graph in any way.\n let sub = interval(refresh)\n .pipe(takeUntilDestroyed(destroyRef))\n .subscribe(() => resource.reload());\n const reload = () => {\n sub.unsubscribe(); // do not conflict with manual reload\n const hasReloaded = resource.reload();\n // resubscribe after manual reload\n sub = interval(refresh)\n .pipe(takeUntilDestroyed(destroyRef))\n .subscribe(() => resource.reload());\n return hasReloaded;\n };\n return {\n ...resource,\n reload,\n destroy: () => {\n sub.unsubscribe();\n resource.destroy();\n },\n };\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVmcmVzaC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3BhY2thZ2VzL3Jlc291cmNlL3NyYy9saWIvdXRpbC9yZWZyZXNoLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUVBLE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxNQUFNLDRCQUE0QixDQUFDO0FBQ2hFLE9BQU8sRUFBRSxRQUFRLEVBQUUsTUFBTSxNQUFNLENBQUM7QUFFaEMsb0lBQW9JO0FBQ3BJLE1BQU0sVUFBVSxPQUFPLENBQ3JCLFFBQTRCLEVBQzVCLFVBQXNCLEVBQ3RCLE9BQWdCO0lBRWhCLElBQUksQ0FBQyxPQUFPO1FBQUUsT0FBTyxRQUFRLENBQUMsQ0FBQyx1QkFBdUI7SUFFdEQsdUlBQXVJO0lBQ3ZJLElBQUksR0FBRyxHQUFHLFFBQVEsQ0FBQyxPQUFPLENBQUM7U0FDeEIsSUFBSSxDQUFDLGtCQUFrQixDQUFDLFVBQVUsQ0FBQyxDQUFDO1NBQ3BDLFNBQVMsQ0FBQyxHQUFHLEVBQUUsQ0FBQyxRQUFRLENBQUMsTUFBTSxFQUFFLENBQUMsQ0FBQztJQUV0QyxNQUFNLE1BQU0sR0FBRyxHQUFZLEVBQUU7UUFDM0IsR0FBRyxDQUFDLFdBQVcsRUFBRSxDQUFDLENBQUMscUNBQXFDO1FBRXhELE1BQU0sV0FBVyxHQUFHLFFBQVEsQ0FBQyxNQUFNLEVBQUUsQ0FBQztRQUV0QyxrQ0FBa0M7UUFDbEMsR0FBRyxHQUFHLFFBQVEsQ0FBQyxPQUFPLENBQUM7YUFDcEIsSUFBSSxDQUFDLGtCQUFrQixDQUFDLFVBQVUsQ0FBQyxDQUFDO2FBQ3BDLFNBQVMsQ0FBQyxHQUFHLEVBQUUsQ0FBQyxRQUFRLENBQUMsTUFBTSxFQUFFLENBQUMsQ0FBQztRQUV0QyxPQUFPLFdBQVcsQ0FBQztJQUNyQixDQUFDLENBQUM7SUFFRixPQUFPO1FBQ0wsR0FBRyxRQUFRO1FBQ1gsTUFBTTtRQUNOLE9BQU8sRUFBRSxHQUFHLEVBQUU7WUFDWixHQUFHLENBQUMsV0FBVyxFQUFFLENBQUM7WUFDbEIsUUFBUSxDQUFDLE9BQU8sRUFBRSxDQUFDO1FBQ3JCLENBQUM7S0FDRixDQUFDO0FBQ0osQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEh0dHBSZXNvdXJjZVJlZiB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbi9odHRwJztcclxuaW1wb3J0IHsgRGVzdHJveVJlZiB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xyXG5pbXBvcnQgeyB0YWtlVW50aWxEZXN0cm95ZWQgfSBmcm9tICdAYW5ndWxhci9jb3JlL3J4anMtaW50ZXJvcCc7XHJcbmltcG9ydCB7IGludGVydmFsIH0gZnJvbSAncnhqcyc7XHJcblxyXG4vLyByZWZyZXNoIHJlc291cmNlIGV2ZXJ5IG4gbWlsaXNlY29uZHMgb3IgZG9uJ3QgcmVmcmVzaCBpZiB1bmRlZmluZWQgcHJvdmlkZWQuIDAgYWxzbyBleGNsdWRlZCwgZHVlIHRvIGl0IG5vdCBiZWluZyBhIHZhbGlkIHVzZWNhc2VcclxuZXhwb3J0IGZ1bmN0aW9uIHJlZnJlc2g8VD4oXHJcbiAgcmVzb3VyY2U6IEh0dHBSZXNvdXJjZVJlZjxUPixcclxuICBkZXN0cm95UmVmOiBEZXN0cm95UmVmLFxyXG4gIHJlZnJlc2g/OiBudW1iZXIsXHJcbik6IEh0dHBSZXNvdXJjZVJlZjxUPiB7XHJcbiAgaWYgKCFyZWZyZXNoKSByZXR1cm4gcmVzb3VyY2U7IC8vIG5vIHJlZnJlc2ggcmVxdWVzdGVkXHJcblxyXG4gIC8vIHdlIGNhbiB1c2UgUnhKcyBoZXJlIGFzIHJlbG9hZGluZyB0aGUgcmVzb3VyY2Ugd2lsbCBhbHdheXMgYmUgYSBzaWRlIGVmZmVjdCAmIGFzIHN1Y2ggZG9lcyBub3QgaW1wYWN0IHRoZSByZWFjdGl2ZSBncmFwaCBpbiBhbnkgd2F5LlxyXG4gIGxldCBzdWIgPSBpbnRlcnZhbChyZWZyZXNoKVxyXG4gICAgLnBpcGUodGFrZVVudGlsRGVzdHJveWVkKGRlc3Ryb3lSZWYpKVxyXG4gICAgLnN1YnNjcmliZSgoKSA9PiByZXNvdXJjZS5yZWxvYWQoKSk7XHJcblxyXG4gIGNvbnN0IHJlbG9hZCA9ICgpOiBib29sZWFuID0+IHtcclxuICAgIHN1Yi51bnN1YnNjcmliZSgpOyAvLyBkbyBub3QgY29uZmxpY3Qgd2l0aCBtYW51YWwgcmVsb2FkXHJcblxyXG4gICAgY29uc3QgaGFzUmVsb2FkZWQgPSByZXNvdXJjZS5yZWxvYWQoKTtcclxuXHJcbiAgICAvLyByZXN1YnNjcmliZSBhZnRlciBtYW51YWwgcmVsb2FkXHJcbiAgICBzdWIgPSBpbnRlcnZhbChyZWZyZXNoKVxyXG4gICAgICAucGlwZSh0YWtlVW50aWxEZXN0cm95ZWQoZGVzdHJveVJlZikpXHJcbiAgICAgIC5zdWJzY3JpYmUoKCkgPT4gcmVzb3VyY2UucmVsb2FkKCkpO1xyXG5cclxuICAgIHJldHVybiBoYXNSZWxvYWRlZDtcclxuICB9O1xyXG5cclxuICByZXR1cm4ge1xyXG4gICAgLi4ucmVzb3VyY2UsXHJcbiAgICByZWxvYWQsXHJcbiAgICBkZXN0cm95OiAoKSA9PiB7XHJcbiAgICAgIHN1Yi51bnN1YnNjcmliZSgpO1xyXG4gICAgICByZXNvdXJjZS5kZXN0cm95KCk7XHJcbiAgICB9LFxyXG4gIH07XHJcbn1cclxuIl19","import { effect } from '@angular/core';\n// Retry on error, if number is provided it will retry that many times with exponential backoff, otherwise it will use the options provided\nexport function retryOnError(res, opt) {\n const max = opt ? (typeof opt === 'number' ? opt : (opt.max ?? 0)) : 0;\n const backoff = typeof opt === 'object' ? (opt.backoff ?? 1000) : 1000;\n let retries = 0;\n let timeout;\n const onError = () => {\n if (retries >= max)\n return;\n retries++;\n if (timeout)\n clearTimeout(timeout);\n setTimeout(() => res.reload(), retries <= 0 ? 0 : backoff * Math.pow(2, retries - 1));\n };\n const onSuccess = () => {\n if (timeout)\n clearTimeout(timeout);\n retries = 0;\n };\n const ref = effect(() => {\n switch (res.status()) {\n case 'error':\n return onError();\n case 'resolved':\n return onSuccess();\n }\n });\n return {\n ...res,\n destroy: () => {\n ref.destroy(); // cleanup on manual destroy\n res.destroy();\n },\n };\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmV0cnktb24tZXJyb3IuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9yZXNvdXJjZS9zcmMvbGliL3V0aWwvcmV0cnktb24tZXJyb3IudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQ0EsT0FBTyxFQUFFLE1BQU0sRUFBRSxNQUFNLGVBQWUsQ0FBQztBQVN2QywySUFBMkk7QUFDM0ksTUFBTSxVQUFVLFlBQVksQ0FDMUIsR0FBdUIsRUFDdkIsR0FBa0I7SUFFbEIsTUFBTSxHQUFHLEdBQUcsR0FBRyxDQUFDLENBQUMsQ0FBQyxDQUFDLE9BQU8sR0FBRyxLQUFLLFFBQVEsQ0FBQyxDQUFDLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQyxDQUFDLEdBQUcsQ0FBQyxHQUFHLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDO0lBQ3ZFLE1BQU0sT0FBTyxHQUFHLE9BQU8sR0FBRyxLQUFLLFFBQVEsQ0FBQyxDQUFDLENBQUMsQ0FBQyxHQUFHLENBQUMsT0FBTyxJQUFJLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUM7SUFFdkUsSUFBSSxPQUFPLEdBQUcsQ0FBQyxDQUFDO0lBRWhCLElBQUksT0FBa0QsQ0FBQztJQUV2RCxNQUFNLE9BQU8sR0FBRyxHQUFHLEVBQUU7UUFDbkIsSUFBSSxPQUFPLElBQUksR0FBRztZQUFFLE9BQU87UUFDM0IsT0FBTyxFQUFFLENBQUM7UUFFVixJQUFJLE9BQU87WUFBRSxZQUFZLENBQUMsT0FBTyxDQUFDLENBQUM7UUFFbkMsVUFBVSxDQUNSLEdBQUcsRUFBRSxDQUFDLEdBQUcsQ0FBQyxNQUFNLEVBQUUsRUFDbEIsT0FBTyxJQUFJLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxPQUFPLEdBQUcsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDLEVBQUUsT0FBTyxHQUFHLENBQUMsQ0FBQyxDQUN0RCxDQUFDO0lBQ0osQ0FBQyxDQUFDO0lBRUYsTUFBTSxTQUFTLEdBQUcsR0FBRyxFQUFFO1FBQ3JCLElBQUksT0FBTztZQUFFLFlBQVksQ0FBQyxPQUFPLENBQUMsQ0FBQztRQUNuQyxPQUFPLEdBQUcsQ0FBQyxDQUFDO0lBQ2QsQ0FBQyxDQUFDO0lBRUYsTUFBTSxHQUFHLEdBQUcsTUFBTSxDQUFDLEdBQUcsRUFBRTtRQUN0QixRQUFRLEdBQUcsQ0FBQyxNQUFNLEVBQUUsRUFBRSxDQUFDO1lBQ3JCLEtBQUssT0FBTztnQkFDVixPQUFPLE9BQU8sRUFBRSxDQUFDO1lBQ25CLEtBQUssVUFBVTtnQkFDYixPQUFPLFNBQVMsRUFBRSxDQUFDO1FBQ3ZCLENBQUM7SUFDSCxDQUFDLENBQUMsQ0FBQztJQUVILE9BQU87UUFDTCxHQUFHLEdBQUc7UUFDTixPQUFPLEVBQUUsR0FBRyxFQUFFO1lBQ1osR0FBRyxDQUFDLE9BQU8sRUFBRSxDQUFDLENBQUMsNEJBQTRCO1lBQzNDLEdBQUcsQ0FBQyxPQUFPLEVBQUUsQ0FBQztRQUNoQixDQUFDO0tBQ0YsQ0FBQztBQUNKLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyB0eXBlIEh0dHBSZXNvdXJjZVJlZiB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbi9odHRwJztcclxuaW1wb3J0IHsgZWZmZWN0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XHJcblxyXG5leHBvcnQgdHlwZSBSZXRyeU9wdGlvbnMgPVxyXG4gIHwgbnVtYmVyXHJcbiAgfCB7XHJcbiAgICAgIG1heD86IG51bWJlcjtcclxuICAgICAgYmFja29mZj86IG51bWJlcjtcclxuICAgIH07XHJcblxyXG4vLyBSZXRyeSBvbiBlcnJvciwgaWYgbnVtYmVyIGlzIHByb3ZpZGVkIGl0IHdpbGwgcmV0cnkgdGhhdCBtYW55IHRpbWVzIHdpdGggZXhwb25lbnRpYWwgYmFja29mZiwgb3RoZXJ3aXNlIGl0IHdpbGwgdXNlIHRoZSBvcHRpb25zIHByb3ZpZGVkXHJcbmV4cG9ydCBmdW5jdGlvbiByZXRyeU9uRXJyb3I8VD4oXHJcbiAgcmVzOiBIdHRwUmVzb3VyY2VSZWY8VD4sXHJcbiAgb3B0PzogUmV0cnlPcHRpb25zLFxyXG4pOiBIdHRwUmVzb3VyY2VSZWY8VD4ge1xyXG4gIGNvbnN0IG1heCA9IG9wdCA/ICh0eXBlb2Ygb3B0ID09PSAnbnVtYmVyJyA/IG9wdCA6IChvcHQubWF4ID8/IDApKSA6IDA7XHJcbiAgY29uc3QgYmFja29mZiA9IHR5cGVvZiBvcHQgPT09ICdvYmplY3QnID8gKG9wdC5iYWNrb2ZmID8/IDEwMDApIDogMTAwMDtcclxuXHJcbiAgbGV0IHJldHJpZXMgPSAwO1xyXG5cclxuICBsZXQgdGltZW91dDogUmV0dXJuVHlwZTx0eXBlb2Ygc2V0VGltZW91dD4gfCB1bmRlZmluZWQ7XHJcblxyXG4gIGNvbnN0IG9uRXJyb3IgPSAoKSA9PiB7XHJcbiAgICBpZiAocmV0cmllcyA+PSBtYXgpIHJldHVybjtcclxuICAgIHJldHJpZXMrKztcclxuXHJcbiAgICBpZiAodGltZW91dCkgY2xlYXJUaW1lb3V0KHRpbWVvdXQpO1xyXG5cclxuICAgIHNldFRpbWVvdXQoXHJcbiAgICAgICgpID0+IHJlcy5yZWxvYWQoKSxcclxuICAgICAgcmV0cmllcyA8PSAwID8gMCA6IGJhY2tvZmYgKiBNYXRoLnBvdygyLCByZXRyaWVzIC0gMSksXHJcbiAgICApO1xyXG4gIH07XHJcblxyXG4gIGNvbnN0IG9uU3VjY2VzcyA9ICgpID0+IHtcclxuICAgIGlmICh0aW1lb3V0KSBjbGVhclRpbWVvdXQodGltZW91dCk7XHJcbiAgICByZXRyaWVzID0gMDtcclxuICB9O1xyXG5cclxuICBjb25zdCByZWYgPSBlZmZlY3QoKCkgPT4ge1xyXG4gICAgc3dpdGNoIChyZXMuc3RhdHVzKCkpIHtcclxuICAgICAgY2FzZSAnZXJyb3InOlxyXG4gICAgICAgIHJldHVybiBvbkVycm9yKCk7XHJcbiAgICAgIGNhc2UgJ3Jlc29sdmVkJzpcclxuICAgICAgICByZXR1cm4gb25TdWNjZXNzKCk7XHJcbiAgICB9XHJcbiAgfSk7XHJcblxyXG4gIHJldHVybiB7XHJcbiAgICAuLi5yZXMsXHJcbiAgICBkZXN0cm95OiAoKSA9PiB7XHJcbiAgICAgIHJlZi5kZXN0cm95KCk7IC8vIGNsZWFudXAgb24gbWFudWFsIGRlc3Ryb3lcclxuICAgICAgcmVzLmRlc3Ryb3koKTtcclxuICAgIH0sXHJcbiAgfTtcclxufVxyXG4iXX0=","import { HttpParams } from '@angular/common/http';\nimport { entries } from '@mmstack/object';\nfunction normalizeParams(params) {\n if (params instanceof HttpParams)\n return params.toString();\n const paramMap = new Map();\n for (const [key, value] of entries(params)) {\n if (Array.isArray(value)) {\n paramMap.set(key, value.map(encodeURIComponent).join(','));\n }\n else {\n paramMap.set(key, encodeURIComponent(value.toString()));\n }\n }\n return Array.from(paramMap.entries())\n .map(([key, value]) => `${key}=${value}`)\n .join('&');\n}\nexport function urlWithParams(req) {\n if (!req.params)\n return req.url;\n return `${req.url}?${normalizeParams(req.params)}`;\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXJsLXdpdGgtcGFyYW1zLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcGFja2FnZXMvcmVzb3VyY2Uvc3JjL2xpYi91dGlsL3VybC13aXRoLXBhcmFtcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsVUFBVSxFQUF1QixNQUFNLHNCQUFzQixDQUFDO0FBQ3ZFLE9BQU8sRUFBRSxPQUFPLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUUxQyxTQUFTLGVBQWUsQ0FDdEIsTUFBK0M7SUFFL0MsSUFBSSxNQUFNLFlBQVksVUFBVTtRQUFFLE9BQU8sTUFBTSxDQUFDLFFBQVEsRUFBRSxDQUFDO0lBRTNELE1BQU0sUUFBUSxHQUFHLElBQUksR0FBRyxFQUFrQixDQUFDO0lBRTNDLEtBQUssTUFBTSxDQUFDLEdBQUcsRUFBRSxLQUFLLENBQUMsSUFBSSxPQUFPLENBQUMsTUFBTSxDQUFDLEVBQUUsQ0FBQztRQUMzQyxJQUFJLEtBQUssQ0FBQyxPQUFPLENBQUMsS0FBSyxDQUFDLEVBQUUsQ0FBQztZQUN6QixRQUFRLENBQUMsR0FBRyxDQUFDLEdBQUcsRUFBRSxLQUFLLENBQUMsR0FBRyxDQUFDLGtCQUFrQixDQUFDLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUM7UUFDN0QsQ0FBQzthQUFNLENBQUM7WUFDTixRQUFRLENBQUMsR0FBRyxDQUFDLEdBQUcsRUFBRSxrQkFBa0IsQ0FBQyxLQUFLLENBQUMsUUFBUSxFQUFFLENBQUMsQ0FBQyxDQUFDO1FBQzFELENBQUM7SUFDSCxDQUFDO0lBRUQsT0FBTyxLQUFLLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxPQUFPLEVBQUUsQ0FBQztTQUNsQyxHQUFHLENBQUMsQ0FBQyxDQUFDLEdBQUcsRUFBRSxLQUFLLENBQUMsRUFBRSxFQUFFLENBQUMsR0FBRyxHQUFHLElBQUksS0FBSyxFQUFFLENBQUM7U0FDeEMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDO0FBQ2YsQ0FBQztBQUVELE1BQU0sVUFBVSxhQUFhLENBQUMsR0FBd0I7SUFDcEQsSUFBSSxDQUFDLEdBQUcsQ0FBQyxNQUFNO1FBQUUsT0FBTyxHQUFHLENBQUMsR0FBRyxDQUFDO0lBRWhDLE9BQU8sR0FBRyxHQUFHLENBQUMsR0FBRyxJQUFJLGVBQWUsQ0FBQyxHQUFHLENBQUMsTUFBTSxDQUFDLEVBQUUsQ0FBQztBQUNyRCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSHR0cFBhcmFtcywgSHR0cFJlc291cmNlUmVxdWVzdCB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbi9odHRwJztcclxuaW1wb3J0IHsgZW50cmllcyB9IGZyb20gJ0BtbXN0YWNrL29iamVjdCc7XHJcblxyXG5mdW5jdGlvbiBub3JtYWxpemVQYXJhbXMoXHJcbiAgcGFyYW1zOiBSZXF1aXJlZDxIdHRwUmVzb3VyY2VSZXF1ZXN0PlsncGFyYW1zJ10sXHJcbik6IHN0cmluZyB7XHJcbiAgaWYgKHBhcmFtcyBpbnN0YW5jZW9mIEh0dHBQYXJhbXMpIHJldHVybiBwYXJhbXMudG9TdHJpbmcoKTtcclxuXHJcbiAgY29uc3QgcGFyYW1NYXAgPSBuZXcgTWFwPHN0cmluZywgc3RyaW5nPigpO1xyXG5cclxuICBmb3IgKGNvbnN0IFtrZXksIHZhbHVlXSBvZiBlbnRyaWVzKHBhcmFtcykpIHtcclxuICAgIGlmIChBcnJheS5pc0FycmF5KHZhbHVlKSkge1xyXG4gICAgICBwYXJhbU1hcC5zZXQoa2V5LCB2YWx1ZS5tYXAoZW5jb2RlVVJJQ29tcG9uZW50KS5qb2luKCcsJykpO1xyXG4gICAgfSBlbHNlIHtcclxuICAgICAgcGFyYW1NYXAuc2V0KGtleSwgZW5jb2RlVVJJQ29tcG9uZW50KHZhbHVlLnRvU3RyaW5nKCkpKTtcclxuICAgIH1cclxuICB9XHJcblxyXG4gIHJldHVybiBBcnJheS5mcm9tKHBhcmFtTWFwLmVudHJpZXMoKSlcclxuICAgIC5tYXAoKFtrZXksIHZhbHVlXSkgPT4gYCR7a2V5fT0ke3ZhbHVlfWApXHJcbiAgICAuam9pbignJicpO1xyXG59XHJcblxyXG5leHBvcnQgZnVuY3Rpb24gdXJsV2l0aFBhcmFtcyhyZXE6IEh0dHBSZXNvdXJjZVJlcXVlc3QpOiBzdHJpbmcge1xyXG4gIGlmICghcmVxLnBhcmFtcykgcmV0dXJuIHJlcS51cmw7XHJcblxyXG4gIHJldHVybiBgJHtyZXEudXJsfT8ke25vcm1hbGl6ZVBhcmFtcyhyZXEucGFyYW1zKX1gO1xyXG59XHJcbiJdfQ==","import { HttpClient, httpResource, HttpResponse, } from '@angular/common/http';\nimport { computed, DestroyRef, effect, inject, isDevMode, linkedSignal, untracked, } from '@angular/core';\nimport { toWritable } from '@mmstack/primitives';\nimport { firstValueFrom } from 'rxjs';\nimport { catchValueError, createCircuitBreaker, createEqualRequest, hasSlowConnection, injectQueryCache, persistResourceValues, refresh, retryOnError, setCacheContext, urlWithParams, } from './util';\nexport function queryResource(request, options) {\n const cache = injectQueryCache(options?.injector);\n const destroyRef = options?.injector\n ? options.injector.get(DestroyRef)\n : inject(DestroyRef);\n const cb = createCircuitBreaker(options?.circuitBreaker === true\n ? undefined\n : (options?.circuitBreaker ?? false));\n const stableRequest = computed(() => {\n if (cb.isClosed())\n return undefined;\n return request() ?? undefined;\n }, {\n equal: options?.triggerOnSameRequest\n ? undefined\n : createEqualRequest(options?.equal),\n });\n const hashFn = typeof options?.cache === 'object'\n ? (options.cache.hash ?? urlWithParams)\n : urlWithParams;\n const staleTime = typeof options?.cache === 'object' ? options.cache.staleTime : 0;\n const ttl = typeof options?.cache === 'object' ? options.cache.ttl : undefined;\n const cacheKey = computed(() => {\n const r = stableRequest();\n if (!r)\n return null;\n return hashFn(r);\n });\n const cachedRequest = options?.cache\n ? computed(() => {\n const r = stableRequest();\n if (!r)\n return r;\n return {\n ...r,\n context: setCacheContext(r.context, {\n staleTime,\n ttl,\n key: cacheKey() ?? hashFn(r),\n }),\n };\n })\n : stableRequest;\n let resource = httpResource(cachedRequest, {\n ...options,\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n parse: options?.parse, // Not my favorite thing to do, but here it is completely safe.\n });\n resource = catchValueError(resource, options?.defaultValue);\n // get full HttpResonse from Cache\n const cachedEvent = cache.get(cacheKey);\n const parse = options?.parse ?? ((val) => val);\n const actualCacheValue = computed(() => {\n const ce = cachedEvent();\n if (!ce || !(ce.value instanceof HttpResponse))\n return;\n return parse(ce.value.body);\n });\n // retains last cache value after it is invalidated for lifetime of resource\n const cachedValue = linkedSignal({\n source: () => actualCacheValue(),\n computation: (source, prev) => {\n if (!source && prev)\n return prev.value;\n return source;\n },\n });\n resource = refresh(resource, destroyRef, options?.refresh);\n resource = retryOnError(resource, options?.retry);\n resource = persistResourceValues(resource, options?.keepPrevious, options?.equal);\n const value = options?.cache\n ? toWritable(computed(() => {\n return cachedValue() ?? resource.value();\n }), resource.value.set, resource.value.update)\n : resource.value;\n const onError = options?.onError; // Put in own variable to ensure value remains even if options are somehow mutated in-line\n if (onError) {\n const onErrorRef = effect(() => {\n const err = resource.error();\n if (err)\n onError(err);\n });\n // cleanup on manual destroy, I'm comfortable setting these props in-line as we have yet to 'release' the object out of this lexical scope\n const destroyRest = resource.destroy;\n resource.destroy = () => {\n onErrorRef.destroy();\n destroyRest();\n };\n }\n // iterate circuit breaker state, is effect as a computed would cause a circular dependency (resource -> cb -> resource)\n const cbEffectRef = effect(() => {\n const status = resource.status();\n if (status === 'error')\n cb.fail();\n else if (status === 'resolved')\n cb.success();\n });\n const set = (value) => {\n resource.value.set(value);\n const k = untracked(cacheKey);\n if (options?.cache && k)\n cache.store(k, new HttpResponse({\n body: value,\n status: 200,\n statusText: 'OK',\n }));\n };\n const update = (updater) => {\n set(updater(untracked(resource.value)));\n };\n const client = options?.injector\n ? options.injector.get(HttpClient)\n : inject(HttpClient);\n return {\n ...resource,\n value,\n set,\n update,\n disabled: computed(() => cb.isClosed() || stableRequest() === undefined),\n reload: () => {\n cb.halfOpen(); // open the circuit for manual reload\n return resource.reload();\n },\n destroy: () => {\n cbEffectRef.destroy();\n cb.destroy();\n resource.destroy();\n },\n prefetch: async (partial) => {\n if (!options?.cache || hasSlowConnection())\n return Promise.resolve();\n const request = untracked(cachedRequest);\n if (!request)\n return Promise.resolve();\n const prefetchRequest = {\n ...request,\n ...partial,\n };\n try {\n await firstValueFrom(client.request(prefetchRequest.method ?? 'GET', prefetchRequest.url, {\n ...prefetchRequest,\n headers: prefetchRequest.headers,\n observe: 'response',\n }));\n return;\n }\n catch (err) {\n if (isDevMode())\n console.error('Prefetch failed: ', err);\n return;\n }\n },\n };\n}\n//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"query-resource.js","sourceRoot":"","sources":["../../../../../packages/resource/src/lib/query-resource.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  HttpClient,\r\n  HttpHeaders,\r\n  httpResource,\r\n  HttpResponse,\r\n  type HttpResourceOptions,\r\n  type HttpResourceRef,\r\n  type HttpResourceRequest,\r\n} from '@angular/common/http';\r\nimport {\r\n  computed,\r\n  DestroyRef,\r\n  effect,\r\n  inject,\r\n  isDevMode,\r\n  linkedSignal,\r\n  Signal,\r\n  untracked,\r\n} from '@angular/core';\r\nimport { toWritable } from '@mmstack/primitives';\r\nimport { firstValueFrom } from 'rxjs';\r\nimport {\r\n  catchValueError,\r\n  CircuitBreakerOptions,\r\n  createCircuitBreaker,\r\n  createEqualRequest,\r\n  hasSlowConnection,\r\n  injectQueryCache,\r\n  persistResourceValues,\r\n  refresh,\r\n  retryOnError,\r\n  setCacheContext,\r\n  urlWithParams,\r\n  type RetryOptions,\r\n} from './util';\r\n\r\n/**\r\n * Options for configuring caching behavior of a `queryResource`.\r\n * - `true`: Enables caching with default settings.\r\n * - `{ ttl?: number; staleTime?: number; hash?: (req: HttpResourceRequest) => string; }`:  Configures caching with custom settings.\r\n */\r\ntype ResourceCacheOptions =\r\n  | true\r\n  | {\r\n      /**\r\n       * The Time To Live (TTL) for the cached data, in milliseconds. After this time, the cached data is\r\n       * considered expired and will be refetched.\r\n       */\r\n      ttl?: number;\r\n      /**\r\n       * The duration, in milliseconds, during which stale data can be served while a revalidation request\r\n       * is made in the background.\r\n       */\r\n      staleTime?: number;\r\n      /**\r\n       * A custom function to generate the cache key. Defaults to using the request URL with parameters.\r\n       * Provide a custom hash function if you need more control over how cache keys are generated,\r\n       * for instance, to ignore certain query parameters or to use request body for the cache key.\r\n       */\r\n      hash?: (req: HttpResourceRequest) => string;\r\n    };\r\n\r\n/**\r\n * Options for configuring a `queryResource`.\r\n */\r\nexport type QueryResourceOptions<TResult, TRaw = TResult> = HttpResourceOptions<\r\n  TResult,\r\n  TRaw\r\n> & {\r\n  /**\r\n   * Whether to keep the previous value of the resource while a refresh is in progress.\r\n   * Defaults to `false`. Also keeps status & headers while refreshing.\r\n   */\r\n  keepPrevious?: boolean;\r\n  /**\r\n   * The refresh interval, in milliseconds. If provided, the resource will automatically\r\n   * refresh its data at the specified interval.\r\n   */\r\n  refresh?: number;\r\n  /**\r\n   * Options for retrying failed requests.\r\n   */\r\n  retry?: RetryOptions;\r\n  /**\r\n   * An optional error handler callback.  This function will be called whenever the\r\n   * underlying HTTP request fails. Useful for displaying toasts or other error messages.\r\n   */\r\n  onError?: (err: unknown) => void;\r\n  /**\r\n   * Options for configuring a circuit breaker for the resource.\r\n   */\r\n  circuitBreaker?: CircuitBreakerOptions | true;\r\n  /**\r\n   * Options for enabling and configuring caching for the resource.\r\n   */\r\n  cache?: ResourceCacheOptions;\r\n  /**\r\n   * Trigger a request every time the request function is triggered, even if the request parameters are the same.\r\n   * @default false\r\n   */\r\n  triggerOnSameRequest?: boolean;\r\n};\r\n\r\n/**\r\n * Represents a resource created by `queryResource`. Extends `HttpResourceRef` with additional properties.\r\n */\r\nexport type QueryResourceRef<TResult> = HttpResourceRef<TResult> & {\r\n  /**\r\n   * A signal indicating whether the resource is currently disabled (due to circuit breaker or undefined request).\r\n   */\r\n  disabled: Signal<boolean>;\r\n  /**\r\n   * Prefetches data for the resource, populating the cache if caching is enabled.  This can be\r\n   * used to proactively load data before it's needed.  If a slow connection is detected, prefetching is skipped.\r\n   *\r\n   * @param req - Optional partial request parameters to use for the prefetch.  This allows you\r\n   *              to prefetch data with different parameters than the main resource request.\r\n   */\r\n  prefetch: (req?: Partial<HttpResourceRequest>) => Promise<void>;\r\n};\r\n\r\nexport function queryResource<TResult, TRaw = TResult>(\r\n  request: () => HttpResourceRequest | undefined | void,\r\n  options: QueryResourceOptions<TResult, TRaw> & {\r\n    defaultValue: NoInfer<TResult>;\r\n  },\r\n): QueryResourceRef<TResult>;\r\n\r\n/**\r\n * Creates an extended HTTP resource with features like caching, retries, refresh intervals,\r\n * circuit breaker, and optimistic updates. Without additional options it is equivalent to simply calling `httpResource`.\r\n *\r\n * @param request A function that returns the `HttpResourceRequest` to be made.  This function\r\n *                is called reactively, so the request can change over time.  If the function\r\n *                returns `undefined`, the resource is considered \"disabled\" and no request will be made.\r\n * @param options Configuration options for the resource.  These options extend the basic\r\n *                `HttpResourceOptions` and add features like `keepPrevious`, `refresh`, `retry`,\r\n *                `onError`, `circuitBreaker`, and `cache`.\r\n * @returns An `QueryResourceRef` instance, which extends the basic `HttpResourceRef` with additional features.\r\n */\r\nexport function queryResource<TResult, TRaw = TResult>(\r\n  request: () => HttpResourceRequest | undefined | void,\r\n  options?: QueryResourceOptions<TResult, TRaw>,\r\n): QueryResourceRef<TResult | undefined>;\r\n\r\nexport function queryResource<TResult, TRaw = TResult>(\r\n  request: () => HttpResourceRequest | undefined | void,\r\n  options?: QueryResourceOptions<TResult, TRaw>,\r\n): QueryResourceRef<TResult | undefined> {\r\n  const cache = injectQueryCache(options?.injector);\r\n\r\n  const destroyRef = options?.injector\r\n    ? options.injector.get(DestroyRef)\r\n    : inject(DestroyRef);\r\n\r\n  const cb = createCircuitBreaker(\r\n    options?.circuitBreaker === true\r\n      ? undefined\r\n      : (options?.circuitBreaker ?? false),\r\n  );\r\n\r\n  const stableRequest = computed(\r\n    () => {\r\n      if (cb.isClosed()) return undefined;\r\n      return request() ?? undefined;\r\n    },\r\n    {\r\n      equal: options?.triggerOnSameRequest\r\n        ? undefined\r\n        : createEqualRequest(options?.equal),\r\n    },\r\n  );\r\n\r\n  const hashFn =\r\n    typeof options?.cache === 'object'\r\n      ? (options.cache.hash ?? urlWithParams)\r\n      : urlWithParams;\r\n\r\n  const staleTime =\r\n    typeof options?.cache === 'object' ? options.cache.staleTime : 0;\r\n  const ttl =\r\n    typeof options?.cache === 'object' ? options.cache.ttl : undefined;\r\n\r\n  const cacheKey = computed(() => {\r\n    const r = stableRequest();\r\n    if (!r) return null;\r\n    return hashFn(r);\r\n  });\r\n\r\n  const cachedRequest = options?.cache\r\n    ? computed(() => {\r\n        const r = stableRequest();\r\n        if (!r) return r;\r\n\r\n        return {\r\n          ...r,\r\n          context: setCacheContext(r.context, {\r\n            staleTime,\r\n            ttl,\r\n            key: cacheKey() ?? hashFn(r),\r\n          }),\r\n        };\r\n      })\r\n    : stableRequest;\r\n\r\n  let resource = httpResource<TResult>(cachedRequest, {\r\n    ...options,\r\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\r\n    parse: options?.parse as any, // Not my favorite thing to do, but here it is completely safe.\r\n  }) as HttpResourceRef<TResult>;\r\n\r\n  resource = catchValueError(resource, options?.defaultValue as TResult);\r\n\r\n  // get full HttpResonse from Cache\r\n  const cachedEvent = cache.get(cacheKey);\r\n\r\n  const parse = options?.parse ?? ((val: TRaw) => val as unknown as TResult);\r\n\r\n  const actualCacheValue = computed((): TResult | undefined => {\r\n    const ce = cachedEvent();\r\n    if (!ce || !(ce.value instanceof HttpResponse)) return;\r\n    return parse(ce.value.body as TRaw);\r\n  });\r\n\r\n  // retains last cache value after it is invalidated for lifetime of resource\r\n  const cachedValue = linkedSignal<TResult | undefined, TResult | undefined>({\r\n    source: () => actualCacheValue(),\r\n    computation: (source, prev) => {\r\n      if (!source && prev) return prev.value;\r\n      return source;\r\n    },\r\n  });\r\n\r\n  resource = refresh(resource, destroyRef, options?.refresh);\r\n  resource = retryOnError(resource, options?.retry);\r\n\r\n  resource = persistResourceValues<TResult>(\r\n    resource,\r\n    options?.keepPrevious,\r\n    options?.equal,\r\n  );\r\n\r\n  const value = options?.cache\r\n    ? toWritable(\r\n        computed((): TResult => {\r\n          return cachedValue() ?? resource.value();\r\n        }),\r\n        resource.value.set,\r\n        resource.value.update,\r\n      )\r\n    : resource.value;\r\n\r\n  const onError = options?.onError; // Put in own variable to ensure value remains even if options are somehow mutated in-line\r\n  if (onError) {\r\n    const onErrorRef = effect(() => {\r\n      const err = resource.error();\r\n      if (err) onError(err);\r\n    });\r\n\r\n    // cleanup on manual destroy, I'm comfortable setting these props in-line as we have yet to 'release' the object out of this lexical scope\r\n    const destroyRest = resource.destroy;\r\n    resource.destroy = () => {\r\n      onErrorRef.destroy();\r\n      destroyRest();\r\n    };\r\n  }\r\n\r\n  // iterate circuit breaker state, is effect as a computed would cause a circular dependency (resource -> cb -> resource)\r\n  const cbEffectRef = effect(() => {\r\n    const status = resource.status();\r\n    if (status === 'error') cb.fail();\r\n    else if (status === 'resolved') cb.success();\r\n  });\r\n\r\n  const set = (value: TResult) => {\r\n    resource.value.set(value);\r\n    const k = untracked(cacheKey);\r\n    if (options?.cache && k)\r\n      cache.store(\r\n        k,\r\n        new HttpResponse({\r\n          body: value,\r\n          status: 200,\r\n          statusText: 'OK',\r\n        }),\r\n      );\r\n  };\r\n\r\n  const update = (updater: (value: TResult) => TResult) => {\r\n    set(updater(untracked(resource.value)));\r\n  };\r\n\r\n  const client = options?.injector\r\n    ? options.injector.get(HttpClient)\r\n    : inject(HttpClient);\r\n\r\n  return {\r\n    ...resource,\r\n    value,\r\n    set,\r\n    update,\r\n    disabled: computed(() => cb.isClosed() || stableRequest() === undefined),\r\n    reload: () => {\r\n      cb.halfOpen(); // open the circuit for manual reload\r\n      return resource.reload();\r\n    },\r\n    destroy: () => {\r\n      cbEffectRef.destroy();\r\n      cb.destroy();\r\n      resource.destroy();\r\n    },\r\n    prefetch: async (partial) => {\r\n      if (!options?.cache || hasSlowConnection()) return Promise.resolve();\r\n\r\n      const request = untracked(cachedRequest);\r\n      if (!request) return Promise.resolve();\r\n\r\n      const prefetchRequest = {\r\n        ...request,\r\n        ...partial,\r\n      };\r\n\r\n      try {\r\n        await firstValueFrom(\r\n          client.request<TRaw>(\r\n            prefetchRequest.method ?? 'GET',\r\n            prefetchRequest.url,\r\n            {\r\n              ...prefetchRequest,\r\n              headers: prefetchRequest.headers as HttpHeaders,\r\n              observe: 'response',\r\n            },\r\n          ),\r\n        );\r\n\r\n        return;\r\n      } catch (err) {\r\n        if (isDevMode()) console.error('Prefetch failed: ', err);\r\n        return;\r\n      }\r\n    },\r\n  };\r\n}\r\n"]}","import { computed, DestroyRef, inject, signal, } from '@angular/core';\nimport { takeUntilDestroyed, toObservable } from '@angular/core/rxjs-interop';\nimport { catchError, combineLatestWith, filter, map, of } from 'rxjs';\nimport { queryResource, } from './query-resource';\nimport { createEqualRequest } from './util';\n/**\n * Creates a resource for performing mutations (e.g., POST, PUT, PATCH, DELETE requests).\n * Unlike `queryResource`, `mutationResource` is designed for one-off operations that change data.\n * It does *not* cache responses and does not provide a `value` signal. Instead, it focuses on\n * managing the mutation lifecycle (pending, error, success) and provides callbacks for handling\n * these states.\n *\n * @param request A function that returns the base `HttpResourceRequest` to be made. This\n * function is called reactively. Unlike `queryResource`, the `body` property\n * of the request is provided when `mutate` is called, *not* here. If the\n * function returns `undefined`, the mutation is considered \"disabled.\" All properties,\n * except the body, can be set here.\n * @param options Configuration options for the mutation resource. This includes callbacks\n * for `onMutate`, `onError`, `onSuccess`, and `onSettled`.\n * @typeParam TResult - The type of the expected result from the mutation.\n * @typeParam TRaw - The raw response type from the HTTP request (defaults to TResult).\n * @typeParam TCTX - The type of the context value returned by `onMutate`.\n * @returns A `MutationResourceRef` instance, which provides methods for triggering the mutation\n * and observing its status.\n */\nexport function mutationResource(request, options = {}) {\n const { onMutate, onError, onSuccess, onSettled, equal, ...rest } = options;\n const requestEqual = createEqualRequest(equal);\n const baseRequest = computed(() => request() ?? undefined, {\n equal: requestEqual,\n });\n const nextRequest = signal(null, {\n equal: (a, b) => {\n if (!a && !b)\n return true;\n if (!a || !b)\n return false;\n return requestEqual(a, b);\n },\n });\n const req = computed(() => {\n const nr = nextRequest();\n if (!nr)\n return;\n const base = baseRequest();\n const url = nr.url ?? base?.url;\n if (!url)\n return;\n const method = nr.method ?? base?.method;\n return {\n ...base,\n ...nr,\n url,\n method,\n };\n });\n const resource = queryResource(req, {\n ...rest,\n defaultValue: null, // doesnt matter since .value is not accessible\n });\n let ctx = undefined;\n const destroyRef = options.injector\n ? options.injector.get(DestroyRef)\n : inject(DestroyRef);\n const error$ = toObservable(resource.error);\n const value$ = toObservable(resource.value).pipe(catchError(() => of(null)));\n const statusSub = toObservable(resource.status)\n .pipe(combineLatestWith(error$, value$), map(([status, error, value]) => {\n if (status === 'error' && error) {\n return {\n status: 'error',\n error,\n };\n }\n if (status === 'resolved' && value !== null) {\n return {\n status: 'resolved',\n value,\n };\n }\n return null;\n }), filter((v) => v !== null), takeUntilDestroyed(destroyRef))\n .subscribe((result) => {\n if (result.status === 'error')\n onError?.(result.error, ctx);\n else\n onSuccess?.(result.value, ctx);\n onSettled?.(ctx);\n ctx = undefined;\n nextRequest.set(null);\n });\n return {\n ...resource,\n destroy: () => {\n statusSub.unsubscribe();\n resource.destroy();\n },\n mutate: (value, ictx) => {\n ctx = onMutate?.(value.body, ictx);\n nextRequest.set(value);\n },\n current: nextRequest,\n };\n}\n//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"mutation-resource.js","sourceRoot":"","sources":["../../../../../packages/resource/src/lib/mutation-resource.ts"],"names":[],"mappings":"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","sourcesContent":["import { type HttpResourceRequest } from '@angular/common/http';\r\nimport {\r\n  computed,\r\n  DestroyRef,\r\n  inject,\r\n  Signal,\r\n  signal,\r\n  ValueEqualityFn,\r\n} from '@angular/core';\r\nimport { takeUntilDestroyed, toObservable } from '@angular/core/rxjs-interop';\r\nimport { catchError, combineLatestWith, filter, map, of } from 'rxjs';\r\nimport {\r\n  queryResource,\r\n  type QueryResourceOptions,\r\n  type QueryResourceRef,\r\n} from './query-resource';\r\nimport { createEqualRequest } from './util';\r\n\r\n/**\r\n * @internal\r\n * Helper type for inferring the request body type based on the HTTP method.\r\n */\r\ntype NextRequest<\r\n  TMethod extends HttpResourceRequest['method'],\r\n  TMutation,\r\n> = TMethod extends 'DELETE' | 'delete'\r\n  ? Omit<HttpResourceRequest, 'body' | 'method'> & { method?: TMethod }\r\n  : Omit<HttpResourceRequest, 'body' | 'method'> & {\r\n      body: TMutation;\r\n      method?: TMethod;\r\n    };\r\n\r\n/**\r\n * @internal\r\n * Helper type for tracking mutation status.\r\n */\r\ntype StatusResult<TResult> =\r\n  | {\r\n      status: 'error';\r\n      error: unknown;\r\n    }\r\n  | {\r\n      status: 'resolved';\r\n      value: TResult;\r\n    };\r\n\r\n/**\r\n * Options for configuring a `mutationResource`.\r\n *\r\n * @typeParam TResult - The type of the expected result from the mutation.\r\n * @typeParam TRaw - The raw response type from the HTTP request (defaults to TResult).\r\n * @typeParam TCTX - The type of the context value returned by `onMutate`.\r\n */\r\nexport type MutationResourceOptions<\r\n  TResult,\r\n  TRaw = TResult,\r\n  TMutation = TResult,\r\n  TCTX = void,\r\n  TICTX = TCTX,\r\n> = Omit<\r\n  QueryResourceOptions<TResult, TRaw>,\r\n  'equal' | 'onError' | 'keepPrevious' | 'refresh' | 'cache' // we can't keep previous values, refresh or cache mutations as they are meant to be one-off operations\r\n> & {\r\n  /**\r\n   * A callback function that is called before the mutation request is made.\r\n   * @param value The value being mutated (the `body` of the request).\r\n   * @returns An optional context value that will be passed to the `onError`, `onSuccess`, and `onSettled` callbacks. This is useful for storing\r\n   *  information needed during the mutation lifecycle, such as previous values for optimistic updates or rollback.\r\n   */\r\n  onMutate?: (value: TMutation, initialCTX?: TICTX) => TCTX;\r\n  /**\r\n   * A callback function that is called if the mutation request fails.\r\n   * @param error The error that occurred.\r\n   * @param ctx The context value returned by the `onMutate` callback (or `undefined` if `onMutate` was not provided or returned `undefined`).\r\n   */\r\n  onError?: (error: unknown, ctx: NoInfer<TCTX>) => void;\r\n  /**\r\n   * A callback function that is called if the mutation request succeeds.\r\n   * @param value The result of the mutation (the parsed response body).\r\n   * @param ctx The context value returned by the `onMutate` callback (or `undefined` if `onMutate` was not provided or returned `undefined`).\r\n   */\r\n  onSuccess?: (value: NoInfer<TResult>, ctx: NoInfer<TCTX>) => void;\r\n  /**\r\n   * A callback function that is called when the mutation request settles (either succeeds or fails).\r\n   * @param ctx The context value returned by the `onMutate` callback (or `undefined` if `onMutate` was not provided or returned `undefined`).\r\n   */\r\n  onSettled?: (ctx: NoInfer<TCTX>) => void;\r\n  equal?: ValueEqualityFn<TMutation>;\r\n};\r\n\r\n/**\r\n * Represents a mutation resource created by `mutationResource`.  Extends `QueryResourceRef`\r\n * but removes methods that don't make sense for mutations (like `prefetch`, `value`, etc.).\r\n *\r\n * @typeParam TResult - The type of the expected result from the mutation.\r\n */\r\nexport type MutationResourceRef<\r\n  TResult,\r\n  TMutation = TResult,\r\n  TICTX = void,\r\n  TMethod extends HttpResourceRequest['method'] = HttpResourceRequest['method'],\r\n> = Omit<\r\n  QueryResourceRef<TResult>,\r\n  'prefetch' | 'value' | 'hasValue' | 'set' | 'update' // we don't allow manually viewing the returned data or updating it manually, prefetching a mutation also doesn't make any sense\r\n> & {\r\n  /**\r\n   * Executes the mutation.\r\n   *\r\n   * @param value The request body and any other request parameters to use for the mutation.  The `body` property is *required*.\r\n   */\r\n  mutate: (\r\n    value: Omit<NextRequest<TMethod, TMutation>, 'url'> & { url?: string },\r\n    ctx?: TICTX,\r\n  ) => void;\r\n  /**\r\n   * A signal that holds the current mutation request, or `null` if no mutation is in progress.\r\n   * This can be useful for tracking the state of the mutation or for displaying loading indicators.\r\n   */\r\n  current: Signal<\r\n    (Omit<NextRequest<TMethod, TMutation>, 'url'> & { url?: string }) | null\r\n  >;\r\n};\r\n\r\n/**\r\n * Creates a resource for performing mutations (e.g., POST, PUT, PATCH, DELETE requests).\r\n * Unlike `queryResource`, `mutationResource` is designed for one-off operations that change data.\r\n * It does *not* cache responses and does not provide a `value` signal.  Instead, it focuses on\r\n * managing the mutation lifecycle (pending, error, success) and provides callbacks for handling\r\n * these states.\r\n *\r\n * @param request A function that returns the base `HttpResourceRequest` to be made.  This\r\n *               function is called reactively.  Unlike `queryResource`, the `body` property\r\n *               of the request is provided when `mutate` is called, *not* here.  If the\r\n *               function returns `undefined`, the mutation is considered \"disabled.\" All properties,\r\n *               except the body, can be set here.\r\n * @param options Configuration options for the mutation resource.  This includes callbacks\r\n *               for `onMutate`, `onError`, `onSuccess`, and `onSettled`.\r\n * @typeParam TResult - The type of the expected result from the mutation.\r\n * @typeParam TRaw - The raw response type from the HTTP request (defaults to TResult).\r\n * @typeParam TCTX - The type of the context value returned by `onMutate`.\r\n * @returns A `MutationResourceRef` instance, which provides methods for triggering the mutation\r\n *          and observing its status.\r\n */\r\nexport function mutationResource<\r\n  TResult,\r\n  TRaw = TResult,\r\n  TMutation = TResult,\r\n  TCTX = void,\r\n  TICTX = TCTX,\r\n  TMethod extends HttpResourceRequest['method'] = HttpResourceRequest['method'],\r\n>(\r\n  request: () =>\r\n    | Omit<NextRequest<TMethod, TMutation>, 'body'>\r\n    | undefined\r\n    | void,\r\n  options: MutationResourceOptions<TResult, TRaw, TMutation, TCTX, TICTX> = {},\r\n): MutationResourceRef<TResult, TMutation, TICTX, TMethod> {\r\n  const { onMutate, onError, onSuccess, onSettled, equal, ...rest } = options;\r\n\r\n  const requestEqual = createEqualRequest(equal);\r\n\r\n  const baseRequest = computed(() => request() ?? undefined, {\r\n    equal: requestEqual,\r\n  });\r\n\r\n  const nextRequest = signal<\r\n    (Omit<NextRequest<TMethod, TMutation>, 'url'> & { url?: string }) | null\r\n  >(null, {\r\n    equal: (a, b) => {\r\n      if (!a && !b) return true;\r\n      if (!a || !b) return false;\r\n      return requestEqual(a, b);\r\n    },\r\n  });\r\n\r\n  const req = computed((): HttpResourceRequest | undefined => {\r\n    const nr = nextRequest();\r\n    if (!nr) return;\r\n\r\n    const base = baseRequest();\r\n\r\n    const url = nr.url ?? base?.url;\r\n    if (!url) return;\r\n\r\n    const method = nr.method ?? base?.method;\r\n\r\n    return {\r\n      ...base,\r\n      ...nr,\r\n      url,\r\n      method,\r\n    };\r\n  });\r\n\r\n  const resource = queryResource<TResult, TRaw>(req, {\r\n    ...rest,\r\n    defaultValue: null as unknown as TResult, // doesnt matter since .value is not accessible\r\n  });\r\n\r\n  let ctx: TCTX = undefined as TCTX;\r\n\r\n  const destroyRef = options.injector\r\n    ? options.injector.get(DestroyRef)\r\n    : inject(DestroyRef);\r\n\r\n  const error$ = toObservable(resource.error);\r\n  const value$ = toObservable(resource.value).pipe(catchError(() => of(null)));\r\n\r\n  const statusSub = toObservable(resource.status)\r\n    .pipe(\r\n      combineLatestWith(error$, value$),\r\n      map(([status, error, value]): StatusResult<TResult> | null => {\r\n        if (status === 'error' && error) {\r\n          return {\r\n            status: 'error',\r\n            error,\r\n          };\r\n        }\r\n\r\n        if (status === 'resolved' && value !== null) {\r\n          return {\r\n            status: 'resolved',\r\n            value,\r\n          };\r\n        }\r\n\r\n        return null;\r\n      }),\r\n      filter((v) => v !== null),\r\n      takeUntilDestroyed(destroyRef),\r\n    )\r\n    .subscribe((result) => {\r\n      if (result.status === 'error') onError?.(result.error, ctx);\r\n      else onSuccess?.(result.value, ctx);\r\n\r\n      onSettled?.(ctx);\r\n      ctx = undefined as TCTX;\r\n      nextRequest.set(null);\r\n    });\r\n\r\n  return {\r\n    ...resource,\r\n    destroy: () => {\r\n      statusSub.unsubscribe();\r\n      resource.destroy();\r\n    },\r\n    mutate: (value, ictx) => {\r\n      ctx = onMutate?.(\r\n        (value as unknown as HttpResourceRequest).body as TMutation,\r\n        ictx,\r\n      ) as TCTX;\r\n      nextRequest.set(value);\r\n    },\r\n    current: nextRequest,\r\n  };\r\n}\r\n"]}","/**\n * Generated bundle index. Do not edit.\n */\nexport * from './index';\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibW1zdGFjay1yZXNvdXJjZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3BhY2thZ2VzL3Jlc291cmNlL3NyYy9tbXN0YWNrLXJlc291cmNlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBRUgsY0FBYyxTQUFTLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEdlbmVyYXRlZCBidW5kbGUgaW5kZXguIERvIG5vdCBlZGl0LlxuICovXG5cbmV4cG9ydCAqIGZyb20gJy4vaW5kZXgnO1xuIl19"],"names":[],"mappings":"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"}
|
package/index.d.ts
CHANGED
|
@@ -387,6 +387,11 @@ type QueryResourceOptions<TResult, TRaw = TResult> = HttpResourceOptions<TResult
|
|
|
387
387
|
* Options for enabling and configuring caching for the resource.
|
|
388
388
|
*/
|
|
389
389
|
cache?: ResourceCacheOptions;
|
|
390
|
+
/**
|
|
391
|
+
* Trigger a request every time the request function is triggered, even if the request parameters are the same.
|
|
392
|
+
* @default false
|
|
393
|
+
*/
|
|
394
|
+
triggerOnSameRequest?: boolean;
|
|
390
395
|
};
|
|
391
396
|
/**
|
|
392
397
|
* Represents a resource created by `queryResource`. Extends `HttpResourceRef` with additional properties.
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@mmstack/resource",
|
|
3
|
-
"version": "20.0.
|
|
3
|
+
"version": "20.0.2",
|
|
4
4
|
"keywords": [
|
|
5
5
|
"angular",
|
|
6
6
|
"signals",
|
|
@@ -18,7 +18,7 @@
|
|
|
18
18
|
"homepage": "https://github.com/mihajm/mmstack/blob/master/packages/resource",
|
|
19
19
|
"dependencies": {
|
|
20
20
|
"uuid": "~11.1.0",
|
|
21
|
-
"@mmstack/primitives": "^20.0.
|
|
21
|
+
"@mmstack/primitives": "^20.0.1",
|
|
22
22
|
"@mmstack/object": "^20.0.0",
|
|
23
23
|
"tslib": "^2.3.0"
|
|
24
24
|
},
|