@warp-drive/core 5.8.0-alpha.5 → 5.8.0-alpha.7

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.
Files changed (33) hide show
  1. package/declarations/build-config.d.ts +18 -1
  2. package/declarations/index.d.ts +80 -3
  3. package/declarations/reactive/-private/document.d.ts +58 -46
  4. package/declarations/reactive/-private/schema.d.ts +1 -1
  5. package/declarations/reactive.d.ts +3 -2
  6. package/declarations/request.d.ts +25 -0
  7. package/declarations/store/-private/caches/instance-cache.d.ts +1 -1
  8. package/declarations/store/-private/default-cache-policy.d.ts +74 -73
  9. package/declarations/store/-private/managers/cache-capabilities-manager.d.ts +1 -1
  10. package/declarations/store/-private/managers/cache-manager.d.ts +6 -4
  11. package/declarations/store/-private/new-core-tmp/promise-state.d.ts +1 -0
  12. package/declarations/store/-private/store-service.d.ts +1 -1
  13. package/declarations/store/-types/q/cache-capabilities-manager.d.ts +1 -1
  14. package/declarations/store/deprecated/store.d.ts +1 -1
  15. package/declarations/types/cache.d.ts +6 -4
  16. package/declarations/types/request.d.ts +4 -6
  17. package/declarations/types/schema/fields.d.ts +11 -0
  18. package/declarations/types/symbols.d.ts +2 -2
  19. package/dist/build-config.js +1 -1
  20. package/dist/default-cache-policy-D7_u4YRH.js +572 -0
  21. package/dist/graph/-private.js +1 -1
  22. package/dist/{index-MiSBsI57.js → index-BMCk_UD5.js} +10453 -10426
  23. package/dist/index.js +5 -4
  24. package/dist/reactive.js +7 -42
  25. package/dist/{context-C_7OLieY.js → request-CN-dNlEY.js} +193 -173
  26. package/dist/request.js +1 -1
  27. package/dist/store/-private.js +1 -1
  28. package/dist/store.js +1 -570
  29. package/dist/types/-private.js +1 -1
  30. package/dist/types/request.js +3 -5
  31. package/dist/types/schema/fields.js +14 -0
  32. package/dist/types/symbols.js +2 -2
  33. package/package.json +3 -3
@@ -7,7 +7,7 @@ import type { RequestKey, ResourceKey } from "./identifier.js";
7
7
  import type { Value } from "./json/raw.js";
8
8
  import type { TypeFromInstanceOrString } from "./record.js";
9
9
  import type { RequestContext, StructuredDataDocument, StructuredDocument } from "./request.js";
10
- import type { ResourceDocument, SingleResourceDataDocument } from "./spec/document.js";
10
+ import type { CollectionResourceDataDocument, ResourceDocument, SingleResourceDataDocument } from "./spec/document.js";
11
11
  import type { ApiError } from "./spec/error.js";
12
12
  /**
13
13
  * A hash of changed attributes with the key being the attribute name and the value being an
@@ -253,7 +253,7 @@ export interface Cache {
253
253
  *
254
254
  * @public
255
255
  */
256
- willCommit(cacheKey: ResourceKey, context: RequestContext | null): void;
256
+ willCommit(cacheKey: ResourceKey | ResourceKey[], context: RequestContext | null): void;
257
257
  /**
258
258
  * [LIFECYCLE] Signals to the cache that a resource
259
259
  * was successfully updated as part of a save transaction.
@@ -262,14 +262,16 @@ export interface Cache {
262
262
  * @param the primary ResourceKey that was operated on
263
263
  * @param data - a document in the cache format containing any updated data
264
264
  */
265
- didCommit(cacheKey: ResourceKey, result: StructuredDataDocument<unknown> | null): SingleResourceDataDocument;
265
+ didCommit(cacheKey: ResourceKey, result: StructuredDataDocument<SingleResourceDataDocument> | null): SingleResourceDataDocument;
266
+ didCommit(cacheKey: ResourceKey[], result: StructuredDataDocument<SingleResourceDataDocument> | null): SingleResourceDataDocument;
267
+ didCommit(cacheKey: ResourceKey[], result: StructuredDataDocument<CollectionResourceDataDocument> | null): CollectionResourceDataDocument;
266
268
  /**
267
269
  * [LIFECYCLE] Signals to the cache that a resource
268
270
  * was update via a save transaction failed.
269
271
  *
270
272
  * @public
271
273
  */
272
- commitWasRejected(cacheKey: ResourceKey, errors?: ApiError[]): void;
274
+ commitWasRejected(cacheKey: ResourceKey | ResourceKey[], errors?: ApiError[]): void;
273
275
  /**
274
276
  * [LIFECYCLE] Signals to the cache that all data for a resource
275
277
  * should be cleared.
@@ -10,10 +10,11 @@ export declare const IS_FUTURE: "___(unique) Symbol(IS_FUTURE)";
10
10
  export declare const STRUCTURED: "___(unique) Symbol(DOC)";
11
11
  export type HTTPMethod = "QUERY" | "GET" | "OPTIONS" | "POST" | "PUT" | "PATCH" | "DELETE" | "HEAD" | "CONNECT" | "TRACE";
12
12
  /**
13
- * Use these options to adjust CacheHandler behavior for a request.
13
+ * Use these options to adjust {@link CacheHandler} behavior for a request
14
+ * via {@link RequestInfo.cacheOptions}.
14
15
  *
15
16
  */
16
- export type CacheOptions = {
17
+ export interface CacheOptions {
17
18
  /**
18
19
  * A key that uniquely identifies this request. If not present, the url wil be used
19
20
  * as the key for any GET request, while all other requests will not be cached.
@@ -57,7 +58,7 @@ export type CacheOptions = {
57
58
  *
58
59
  */
59
60
  [SkipCache]?: boolean;
60
- };
61
+ }
61
62
  export type FindRecordRequestOptions<
62
63
  RT = unknown,
63
64
  T = unknown
@@ -349,7 +350,4 @@ export interface RequestContext {
349
350
  setStream(stream: ReadableStream | Promise<ReadableStream | null>): void;
350
351
  setResponse(response: Response | ResponseInfo | null): void;
351
352
  }
352
- export declare function withBrand<T>(obj: RequestInfo): RequestInfo<T> & {
353
- [RequestSignature]: T;
354
- };
355
353
  export {};
@@ -2125,18 +2125,29 @@ export interface ObjectSchema {
2125
2125
  objectExtensions?: string[];
2126
2126
  }
2127
2127
  export type Schema = ResourceSchema | ObjectSchema;
2128
+ /**
2129
+ * A trait for use on a PolarisMode record
2130
+ */
2128
2131
  export interface PolarisTrait {
2129
2132
  name: string;
2130
2133
  mode: "polaris";
2131
2134
  fields: PolarisModeFieldSchema[];
2132
2135
  traits?: string[];
2133
2136
  }
2137
+ /**
2138
+ * A trait for use on a LegacyMode record
2139
+ */
2134
2140
  export interface LegacyTrait {
2135
2141
  name: string;
2136
2142
  mode: "legacy";
2137
2143
  fields: LegacyModeFieldSchema[];
2138
2144
  traits?: string[];
2139
2145
  }
2146
+ /**
2147
+ * A union of
2148
+ * - {@link LegacyTrait}
2149
+ * - {@link PolarisTrait}
2150
+ */
2140
2151
  export type Trait = LegacyTrait | PolarisTrait;
2141
2152
  /**
2142
2153
  * A no-op type utility that enables type-checking resource schema
@@ -9,7 +9,7 @@ export declare const RecordStore: "___(unique) Symbol(Store)";
9
9
  * record implementations to provide a typescript
10
10
  * hint for the type of the resource.
11
11
  *
12
- * When used, EmberData/WarpDrive APIs can
12
+ * When used, WarpDrive APIs can
13
13
  * take advantage of this to provide better type
14
14
  * safety and intellisense.
15
15
  *
@@ -45,7 +45,7 @@ export declare const Type: "___(unique) Symbol($type)";
45
45
  * record implementations to provide a typescript
46
46
  * hint for the type of the resource.
47
47
  *
48
- * When used, EmberData/WarpDrive APIs can
48
+ * When used, WarpDrive APIs can
49
49
  * take advantage of this to provide better type
50
50
  * safety and intellisense.
51
51
  *
@@ -1 +1 @@
1
- export { setConfig } from '@warp-drive/build-config';
1
+ export { babelPlugin, setConfig } from '@warp-drive/build-config';
@@ -0,0 +1,572 @@
1
+ import { deprecate } from '@ember/debug';
2
+ import { LRUCache } from './utils/string.js';
3
+ import { macroCondition, getGlobalConfig } from '@embroider/macros';
4
+
5
+ /**
6
+ * Interface of a parsed Cache-Control header value.
7
+ *
8
+ * - [MDN Cache-Control Reference](https://developer.mozilla.org/en-US/docs/Web/HTTP/Reference/Headers/Cache-Control)
9
+ */
10
+ const NUMERIC_KEYS = new Set(['max-age', 's-maxage', 'stale-if-error', 'stale-while-revalidate']);
11
+
12
+ /**
13
+ * Parses a string Cache-Control header value into an object with the following structure:
14
+ *
15
+ * ```ts
16
+ * interface CacheControlValue {
17
+ * immutable?: boolean;
18
+ * 'max-age'?: number;
19
+ * 'must-revalidate'?: boolean;
20
+ * 'must-understand'?: boolean;
21
+ * 'no-cache'?: boolean;
22
+ * 'no-store'?: boolean;
23
+ * 'no-transform'?: boolean;
24
+ * 'only-if-cached'?: boolean;
25
+ * private?: boolean;
26
+ * 'proxy-revalidate'?: boolean;
27
+ * public?: boolean;
28
+ * 's-maxage'?: number;
29
+ * 'stale-if-error'?: number;
30
+ * 'stale-while-revalidate'?: number;
31
+ * }
32
+ * ```
33
+ *
34
+ * See also {@link CacheControlValue} and [Response Directives](https://developer.mozilla.org/en-US/docs/Web/HTTP/Reference/Headers/Cache-Control#response_directives)
35
+ *
36
+ * @public
37
+ */
38
+ function parseCacheControl(header) {
39
+ return CACHE_CONTROL_CACHE.get(header);
40
+ }
41
+ const CACHE_CONTROL_CACHE = new LRUCache(header => {
42
+ let key = '';
43
+ let value = '';
44
+ let isParsingKey = true;
45
+ const cacheControlValue = {};
46
+ for (let i = 0; i < header.length; i++) {
47
+ const char = header.charAt(i);
48
+ if (char === ',') {
49
+ macroCondition(getGlobalConfig().WarpDrive.env.DEBUG) ? (test => {
50
+ if (!test) {
51
+ throw new Error(`Invalid Cache-Control value, expected a value`);
52
+ }
53
+ })(!isParsingKey || !NUMERIC_KEYS.has(key)) : {};
54
+ macroCondition(getGlobalConfig().WarpDrive.env.DEBUG) ? (test => {
55
+ if (!test) {
56
+ throw new Error(`Invalid Cache-Control value, expected a value after "=" but got ","`);
57
+ }
58
+ })(i === 0 || header.charAt(i - 1) !== '=') : {};
59
+ isParsingKey = true;
60
+ // @ts-expect-error TS incorrectly thinks that optional keys must have a type that includes undefined
61
+ cacheControlValue[key] = NUMERIC_KEYS.has(key) ? parseCacheControlValue(value) : true;
62
+ key = '';
63
+ value = '';
64
+ continue;
65
+ } else if (char === '=') {
66
+ macroCondition(getGlobalConfig().WarpDrive.env.DEBUG) ? (test => {
67
+ if (!test) {
68
+ throw new Error(`Invalid Cache-Control value, expected a value after "="`);
69
+ }
70
+ })(i + 1 !== header.length) : {};
71
+ isParsingKey = false;
72
+ } else if (char === ' ' || char === `\t` || char === `\n`) {
73
+ continue;
74
+ } else if (isParsingKey) {
75
+ key += char;
76
+ } else {
77
+ value += char;
78
+ }
79
+ if (i === header.length - 1) {
80
+ // @ts-expect-error TS incorrectly thinks that optional keys must have a type that includes undefined
81
+ cacheControlValue[key] = NUMERIC_KEYS.has(key) ? parseCacheControlValue(value) : true;
82
+ }
83
+ }
84
+ return cacheControlValue;
85
+ }, 200);
86
+ function parseCacheControlValue(stringToParse) {
87
+ const parsedValue = Number.parseInt(stringToParse);
88
+ macroCondition(getGlobalConfig().WarpDrive.env.DEBUG) ? (test => {
89
+ if (!test) {
90
+ throw new Error(`Invalid Cache-Control value, expected a number but got - ${stringToParse}`);
91
+ }
92
+ })(!Number.isNaN(parsedValue)) : {};
93
+ macroCondition(getGlobalConfig().WarpDrive.env.DEBUG) ? (test => {
94
+ if (!test) {
95
+ throw new Error(`Invalid Cache-Control value, expected a number greater than 0 but got - ${stringToParse}`);
96
+ }
97
+ })(parsedValue >= 0) : {};
98
+ if (Number.isNaN(parsedValue) || parsedValue < 0) {
99
+ return 0;
100
+ }
101
+ return parsedValue;
102
+ }
103
+ function isExpired(cacheKey, request, config) {
104
+ const {
105
+ constraints
106
+ } = config;
107
+ if (constraints?.isExpired) {
108
+ const result = constraints.isExpired(request);
109
+ if (result !== null) {
110
+ if (macroCondition(getGlobalConfig().WarpDrive.activeLogging.LOG_CACHE_POLICY)) {
111
+ if (getGlobalConfig().WarpDrive.debug.LOG_CACHE_POLICY || globalThis.getWarpDriveRuntimeConfig().debug.LOG_CACHE_POLICY) {
112
+ // eslint-disable-next-line no-console
113
+ console.log(`CachePolicy: ${cacheKey.lid} is ${result ? 'EXPIRED' : 'NOT expired'} because constraints.isExpired returned ${result}`);
114
+ }
115
+ }
116
+ return result;
117
+ }
118
+ }
119
+ const {
120
+ headers
121
+ } = request.response;
122
+ if (!headers) {
123
+ if (macroCondition(getGlobalConfig().WarpDrive.activeLogging.LOG_CACHE_POLICY)) {
124
+ if (getGlobalConfig().WarpDrive.debug.LOG_CACHE_POLICY || globalThis.getWarpDriveRuntimeConfig().debug.LOG_CACHE_POLICY) {
125
+ // eslint-disable-next-line no-console
126
+ console.log(`CachePolicy: ${cacheKey.lid} is EXPIRED because no headers were provided`);
127
+ }
128
+ }
129
+
130
+ // if we have no headers then both the headers based expiration
131
+ // and the time based expiration will be considered expired
132
+ return true;
133
+ }
134
+
135
+ // check for X-WarpDrive-Expires
136
+ const now = Date.now();
137
+ const date = headers.get('Date');
138
+ if (constraints?.headers) {
139
+ if (constraints.headers['X-WarpDrive-Expires']) {
140
+ const xWarpDriveExpires = headers.get('X-WarpDrive-Expires');
141
+ if (xWarpDriveExpires) {
142
+ const expirationTime = new Date(xWarpDriveExpires).getTime();
143
+ const result = now >= expirationTime;
144
+ if (macroCondition(getGlobalConfig().WarpDrive.activeLogging.LOG_CACHE_POLICY)) {
145
+ if (getGlobalConfig().WarpDrive.debug.LOG_CACHE_POLICY || globalThis.getWarpDriveRuntimeConfig().debug.LOG_CACHE_POLICY) {
146
+ // eslint-disable-next-line no-console
147
+ console.log(`CachePolicy: ${cacheKey.lid} is ${result ? 'EXPIRED' : 'NOT expired'} because the time set by X-WarpDrive-Expires header is ${result ? 'in the past' : 'in the future'}`);
148
+ }
149
+ }
150
+ return result;
151
+ }
152
+ }
153
+
154
+ // check for Cache-Control
155
+ if (constraints.headers['Cache-Control']) {
156
+ const cacheControl = headers.get('Cache-Control');
157
+ const age = headers.get('Age');
158
+ if (cacheControl && age && date) {
159
+ const cacheControlValue = parseCacheControl(cacheControl);
160
+
161
+ // max-age and s-maxage are stored in
162
+ const maxAge = cacheControlValue['max-age'] || cacheControlValue['s-maxage'];
163
+ if (maxAge) {
164
+ // age is stored in seconds
165
+ const ageValue = parseInt(age, 10);
166
+ macroCondition(getGlobalConfig().WarpDrive.env.DEBUG) ? (test => {
167
+ if (!test) {
168
+ throw new Error(`Invalid Cache-Control value, expected a number but got - ${age}`);
169
+ }
170
+ })(!Number.isNaN(ageValue)) : {};
171
+ macroCondition(getGlobalConfig().WarpDrive.env.DEBUG) ? (test => {
172
+ if (!test) {
173
+ throw new Error(`Invalid Cache-Control value, expected a number greater than 0 but got - ${age}`);
174
+ }
175
+ })(ageValue >= 0) : {};
176
+ if (!Number.isNaN(ageValue) && ageValue >= 0) {
177
+ const dateValue = new Date(date).getTime();
178
+ const expirationTime = dateValue + (maxAge - ageValue) * 1000;
179
+ const result = now >= expirationTime;
180
+ if (macroCondition(getGlobalConfig().WarpDrive.activeLogging.LOG_CACHE_POLICY)) {
181
+ if (getGlobalConfig().WarpDrive.debug.LOG_CACHE_POLICY || globalThis.getWarpDriveRuntimeConfig().debug.LOG_CACHE_POLICY) {
182
+ // eslint-disable-next-line no-console
183
+ console.log(`CachePolicy: ${cacheKey.lid} is ${result ? 'EXPIRED' : 'NOT expired'} because the time set by Cache-Control header is ${result ? 'in the past' : 'in the future'}`);
184
+ }
185
+ }
186
+ return result;
187
+ }
188
+ }
189
+ }
190
+ }
191
+
192
+ // check for Expires
193
+ if (constraints.headers.Expires) {
194
+ const expires = headers.get('Expires');
195
+ if (expires) {
196
+ const expirationTime = new Date(expires).getTime();
197
+ const result = now >= expirationTime;
198
+ if (macroCondition(getGlobalConfig().WarpDrive.activeLogging.LOG_CACHE_POLICY)) {
199
+ if (getGlobalConfig().WarpDrive.debug.LOG_CACHE_POLICY || globalThis.getWarpDriveRuntimeConfig().debug.LOG_CACHE_POLICY) {
200
+ // eslint-disable-next-line no-console
201
+ console.log(`CachePolicy: ${cacheKey.lid} is ${result ? 'EXPIRED' : 'NOT expired'} because the time set by Expires header is ${result ? 'in the past' : 'in the future'}`);
202
+ }
203
+ }
204
+ return result;
205
+ }
206
+ }
207
+ }
208
+
209
+ // check for Date
210
+ if (!date) {
211
+ if (macroCondition(getGlobalConfig().WarpDrive.activeLogging.LOG_CACHE_POLICY)) {
212
+ if (getGlobalConfig().WarpDrive.debug.LOG_CACHE_POLICY || globalThis.getWarpDriveRuntimeConfig().debug.LOG_CACHE_POLICY) {
213
+ // eslint-disable-next-line no-console
214
+ console.log(`CachePolicy: ${cacheKey.lid} is EXPIRED because no Date header was provided`);
215
+ }
216
+ }
217
+ return true;
218
+ }
219
+ let expirationTime = config.apiCacheHardExpires;
220
+ if (macroCondition(getGlobalConfig().WarpDrive.env.TESTING)) {
221
+ if (!config.disableTestOptimization) {
222
+ expirationTime = config.apiCacheSoftExpires;
223
+ }
224
+ }
225
+ const time = new Date(date).getTime();
226
+ const deadline = time + expirationTime;
227
+ const result = now >= deadline;
228
+ if (macroCondition(getGlobalConfig().WarpDrive.activeLogging.LOG_CACHE_POLICY)) {
229
+ if (getGlobalConfig().WarpDrive.debug.LOG_CACHE_POLICY || globalThis.getWarpDriveRuntimeConfig().debug.LOG_CACHE_POLICY) {
230
+ // eslint-disable-next-line no-console
231
+ console.log(`CachePolicy: ${cacheKey.lid} is ${result ? 'EXPIRED' : 'NOT expired'} because the apiCacheHardExpires time since the response's Date header is ${result ? 'in the past' : 'in the future'}`);
232
+ }
233
+ }
234
+ return result;
235
+ }
236
+
237
+ /**
238
+ * The constraint options for {@link PolicyConfig.constraints}
239
+ */
240
+
241
+ /**
242
+ * The configuration options for the {@link DefaultCachePolicy}
243
+ *
244
+ * ```ts [app/services/store.ts]
245
+ * import { Store } from '@warp-drive/core';
246
+ * import { DefaultCachePolicy } from '@warp-drive/core/store'; // [!code focus]
247
+ *
248
+ * export default class AppStore extends Store {
249
+ * lifetimes = new DefaultCachePolicy({ // [!code focus:3]
250
+ * // ... PolicyConfig Settings ... //
251
+ * });
252
+ * }
253
+ * ```
254
+ *
255
+ */
256
+
257
+ /**
258
+ * A basic {@link CachePolicy} that can be added to the Store service.
259
+ *
260
+ * ```ts [app/services/store.ts]
261
+ * import { Store } from '@warp-drive/core';
262
+ * import { DefaultCachePolicy } from '@warp-drive/core/store'; // [!code focus]
263
+ *
264
+ * export default class AppStore extends Store {
265
+ * lifetimes = new DefaultCachePolicy({ // [!code focus:5]
266
+ * apiCacheSoftExpires: 30_000,
267
+ * apiCacheHardExpires: 60_000,
268
+ * // ... Other PolicyConfig Settings ... //
269
+ * });
270
+ * }
271
+ * ```
272
+ *
273
+ * :::tip 💡 TIP
274
+ * Date headers do not have millisecond precision, so expiration times should
275
+ * generally be larger than 1000ms.
276
+ * :::
277
+ *
278
+ * See also {@link PolicyConfig} for configuration options.
279
+ *
280
+ * ### The Mechanics
281
+ *
282
+ * This policy determines staleness based on various configurable constraints falling back to a simple
283
+ * check of the time elapsed since the request was last received from the API using the `date` header
284
+ * from the last response.
285
+ *
286
+ * :::tip 💡 TIP
287
+ * The {@link Fetch} handler provided by `@warp-drive/core` will automatically
288
+ * add the `date` header to responses if it is not present.
289
+ * :::
290
+ *
291
+ * - For manual override of reload see {@link CacheOptions.reload | cacheOptions.reload}
292
+ * - For manual override of background reload see {@link CacheOptions.backgroundReload | cacheOptions.backgroundReload}
293
+ *
294
+ * In order expiration is determined by:
295
+ *
296
+ * ```md
297
+ * Is explicitly invalidated by `cacheOptions.reload`
298
+ * ↳ (if falsey) if the request has been explicitly invalidated
299
+ * since the last request (see Automatic Invalidation below)
300
+ * ↳ (if false) (If Active) isExpired function
301
+ * ↳ (if null) (If Active) X-WarpDrive-Expires header
302
+ * ↳ (if null) (If Active) Cache-Control header
303
+ * ↳ (if null) (If Active) Expires header
304
+ * ↳ (if null) Date header + apiCacheHardExpires < current time
305
+ *
306
+ * -- <if above is false, a background request is issued if> --
307
+ *
308
+ * ↳ is invalidated by `cacheOptions.backgroundReload`
309
+ * ↳ (if falsey) Date header + apiCacheSoftExpires < current time
310
+ * ```
311
+ *
312
+ * ### Automatic Invalidation / Entanglement
313
+ *
314
+ * It also invalidates any request with an {@link RequestInfo.op | OpCode} of `"query"`
315
+ * for which {@link CacheOptions.types | cacheOptions.types} was provided
316
+ * when a request with an `OpCode` of `"createRecord"` is successful and also includes
317
+ * a matching type in its own `cacheOptions.types` array.
318
+
319
+ * :::tip 💡 TIP
320
+ * Abstracting this behavior via builders is recommended to ensure consistency.
321
+ * :::
322
+ *
323
+ * ### Testing
324
+ *
325
+ * In Testing environments:
326
+ *
327
+ * - `apiCacheSoftExpires` will always be `false`
328
+ * - `apiCacheHardExpires` will use the `apiCacheSoftExpires` value.
329
+ *
330
+ * This helps reduce flakiness and produce predictably rendered results in test suites.
331
+ *
332
+ * Requests that specifically set `cacheOptions.backgroundReload = true` will
333
+ * still be background reloaded in tests.
334
+ *
335
+ * This behavior can be opted out of by setting `disableTestOptimization = true`
336
+ * in the policy config.
337
+ *
338
+ * @public
339
+ */
340
+ class DefaultCachePolicy {
341
+ /**
342
+ * @internal
343
+ */
344
+
345
+ /**
346
+ * @internal
347
+ */
348
+
349
+ /**
350
+ * @internal
351
+ */
352
+ _getStore(store) {
353
+ let set = this._stores.get(store);
354
+ if (!set) {
355
+ set = {
356
+ invalidated: new Set(),
357
+ types: new Map()
358
+ };
359
+ this._stores.set(store, set);
360
+ }
361
+ return set;
362
+ }
363
+ constructor(config) {
364
+ this._stores = new WeakMap();
365
+ const _config = arguments.length === 1 ? config : arguments[1];
366
+ deprecate(`Passing a Store to the CachePolicy is deprecated, please pass only a config instead.`, arguments.length === 1, {
367
+ id: 'ember-data:request-utils:lifetimes-service-store-arg',
368
+ since: {
369
+ enabled: '5.4',
370
+ available: '4.13'
371
+ },
372
+ for: '@ember-data/request-utils',
373
+ until: '6.0'
374
+ });
375
+ macroCondition(getGlobalConfig().WarpDrive.env.DEBUG) ? (test => {
376
+ if (!test) {
377
+ throw new Error(`You must pass a config to the CachePolicy`);
378
+ }
379
+ })(_config) : {};
380
+ macroCondition(getGlobalConfig().WarpDrive.env.DEBUG) ? (test => {
381
+ if (!test) {
382
+ throw new Error(`You must pass a apiCacheSoftExpires to the CachePolicy`);
383
+ }
384
+ })(typeof _config.apiCacheSoftExpires === 'number') : {};
385
+ macroCondition(getGlobalConfig().WarpDrive.env.DEBUG) ? (test => {
386
+ if (!test) {
387
+ throw new Error(`You must pass a apiCacheHardExpires to the CachePolicy`);
388
+ }
389
+ })(typeof _config.apiCacheHardExpires === 'number') : {};
390
+ this.config = _config;
391
+ }
392
+
393
+ /**
394
+ * Invalidate a request by its CacheKey for the given store instance.
395
+ *
396
+ * While the store argument may seem redundant, the CachePolicy
397
+ * is designed to be shared across multiple stores / forks
398
+ * of the store.
399
+ *
400
+ * ```ts
401
+ * store.lifetimes.invalidateRequest(store, cacheKey);
402
+ * ```
403
+ *
404
+ * @public
405
+ */
406
+ invalidateRequest(cacheKey, store) {
407
+ this._getStore(store).invalidated.add(cacheKey);
408
+ }
409
+
410
+ /**
411
+ * Invalidate all requests associated to a specific type
412
+ * for a given store instance.
413
+ *
414
+ * While the store argument may seem redundant, the CachePolicy
415
+ * is designed to be shared across multiple stores / forks
416
+ * of the store.
417
+ *
418
+ * This invalidation is done automatically when using this service
419
+ * for both the {@link CacheHandler} and the [NetworkHandler](/api/@warp-drive/legacy/compat/variables/LegacyNetworkHandler).
420
+ *
421
+ * ```ts
422
+ * store.lifetimes.invalidateRequestsForType(store, 'person');
423
+ * ```
424
+ *
425
+ * @public
426
+ */
427
+ invalidateRequestsForType(type, store) {
428
+ const storeCache = this._getStore(store);
429
+ const set = storeCache.types.get(type);
430
+ const notifications = store.notifications;
431
+ if (set) {
432
+ // TODO batch notifications
433
+ set.forEach(id => {
434
+ storeCache.invalidated.add(id);
435
+ // @ts-expect-error
436
+ notifications.notify(id, 'invalidated', null);
437
+ });
438
+ }
439
+ }
440
+
441
+ /**
442
+ * Invoked when a request has been fulfilled from the configured request handlers.
443
+ * This is invoked by the CacheHandler for both foreground and background requests
444
+ * once the cache has been updated.
445
+ *
446
+ * Note, this is invoked by the {@link CacheHandler} regardless of whether
447
+ * the request has a cache-key.
448
+ *
449
+ * This method should not be invoked directly by consumers.
450
+ *
451
+ * @public
452
+ */
453
+ didRequest(request, response, cacheKey, store) {
454
+ // if this is a successful createRecord request, invalidate the cacheKey for the type
455
+ if (request.op === 'createRecord') {
456
+ const statusNumber = response?.status ?? 0;
457
+ if (statusNumber >= 200 && statusNumber < 400) {
458
+ const types = new Set(request.records?.map(r => r.type));
459
+ const additionalTypes = request.cacheOptions?.types;
460
+ additionalTypes?.forEach(type => {
461
+ types.add(type);
462
+ });
463
+ types.forEach(type => {
464
+ this.invalidateRequestsForType(type, store);
465
+ });
466
+ }
467
+
468
+ // add this document's cacheKey to a map for all associated types
469
+ // it is recommended to only use this for queries
470
+ } else if (cacheKey && request.cacheOptions?.types?.length) {
471
+ const storeCache = this._getStore(store);
472
+ request.cacheOptions?.types.forEach(type => {
473
+ const set = storeCache.types.get(type);
474
+ if (set) {
475
+ set.add(cacheKey);
476
+ storeCache.invalidated.delete(cacheKey);
477
+ } else {
478
+ storeCache.types.set(type, new Set([cacheKey]));
479
+ }
480
+ });
481
+ }
482
+ }
483
+
484
+ /**
485
+ * Invoked to determine if the request may be fulfilled from cache
486
+ * if possible.
487
+ *
488
+ * Note, this is only invoked by the {@link CacheHandler} if the request has
489
+ * a cache-key.
490
+ *
491
+ * If no cache entry is found or the entry is hard expired,
492
+ * the request will be fulfilled from the configured request handlers
493
+ * and the cache will be updated before returning the response.
494
+ *
495
+ * @public
496
+ * @return true if the request is considered hard expired
497
+ */
498
+ isHardExpired(cacheKey, store) {
499
+ // if we are explicitly invalidated, we are hard expired
500
+ const storeCache = this._getStore(store);
501
+ if (storeCache.invalidated.has(cacheKey)) {
502
+ return true;
503
+ }
504
+ const cache = store.cache;
505
+ const cached = cache.peekRequest(cacheKey);
506
+ if (!cached?.response) {
507
+ if (macroCondition(getGlobalConfig().WarpDrive.activeLogging.LOG_CACHE_POLICY)) {
508
+ if (getGlobalConfig().WarpDrive.debug.LOG_CACHE_POLICY || globalThis.getWarpDriveRuntimeConfig().debug.LOG_CACHE_POLICY) {
509
+ // eslint-disable-next-line no-console
510
+ console.log(`CachePolicy: ${cacheKey.lid} is EXPIRED because no cache entry was found`);
511
+ }
512
+ }
513
+ return true;
514
+ }
515
+ return isExpired(cacheKey, cached, this.config);
516
+ }
517
+
518
+ /**
519
+ * Invoked if `isHardExpired` is false to determine if the request
520
+ * should be update behind the scenes if cache data is already available.
521
+ *
522
+ * Note, this is only invoked by the {@link CacheHandler} if the request has
523
+ * a cache-key.
524
+ *
525
+ * If true, the request will be fulfilled from cache while a backgrounded
526
+ * request is made to update the cache via the configured request handlers.
527
+ *
528
+ * @public
529
+ * @return true if the request is considered soft expired
530
+ */
531
+ isSoftExpired(cacheKey, store) {
532
+ if (macroCondition(getGlobalConfig().WarpDrive.env.TESTING)) {
533
+ if (!this.config.disableTestOptimization) {
534
+ return false;
535
+ }
536
+ }
537
+ const cache = store.cache;
538
+ const cached = cache.peekRequest(cacheKey);
539
+ if (cached?.response) {
540
+ const date = cached.response.headers.get('date');
541
+ if (!date) {
542
+ if (macroCondition(getGlobalConfig().WarpDrive.activeLogging.LOG_CACHE_POLICY)) {
543
+ if (getGlobalConfig().WarpDrive.debug.LOG_CACHE_POLICY || globalThis.getWarpDriveRuntimeConfig().debug.LOG_CACHE_POLICY) {
544
+ // eslint-disable-next-line no-console
545
+ console.log(`CachePolicy: ${cacheKey.lid} is STALE because no date header was found`);
546
+ }
547
+ }
548
+ return true;
549
+ } else {
550
+ const time = new Date(date).getTime();
551
+ const now = Date.now();
552
+ const deadline = time + this.config.apiCacheSoftExpires;
553
+ const result = now >= deadline;
554
+ if (macroCondition(getGlobalConfig().WarpDrive.activeLogging.LOG_CACHE_POLICY)) {
555
+ if (getGlobalConfig().WarpDrive.debug.LOG_CACHE_POLICY || globalThis.getWarpDriveRuntimeConfig().debug.LOG_CACHE_POLICY) {
556
+ // eslint-disable-next-line no-console
557
+ console.log(`CachePolicy: ${cacheKey.lid} is ${result ? 'STALE' : 'NOT stale'}. Expiration time: ${deadline}, now: ${now}`);
558
+ }
559
+ }
560
+ return result;
561
+ }
562
+ }
563
+ if (macroCondition(getGlobalConfig().WarpDrive.activeLogging.LOG_CACHE_POLICY)) {
564
+ if (getGlobalConfig().WarpDrive.debug.LOG_CACHE_POLICY || globalThis.getWarpDriveRuntimeConfig().debug.LOG_CACHE_POLICY) {
565
+ // eslint-disable-next-line no-console
566
+ console.log(`CachePolicy: ${cacheKey.lid} is STALE because no cache entry was found`);
567
+ }
568
+ }
569
+ return true;
570
+ }
571
+ }
572
+ export { DefaultCachePolicy as D, parseCacheControl as p };