@warp-drive-mirror/experiments 0.0.1-alpha.99 → 0.0.1-beta.0

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 (31) hide show
  1. package/README.md +2 -1
  2. package/dist/data-worker.js +123 -30
  3. package/dist/data-worker.js.map +1 -1
  4. package/dist/document-storage.js +1 -243
  5. package/dist/document-storage.js.map +1 -1
  6. package/dist/image-fetch.js +78 -0
  7. package/dist/image-fetch.js.map +1 -0
  8. package/dist/image-worker.js +98 -0
  9. package/dist/image-worker.js.map +1 -0
  10. package/dist/index-Cn3o840t.js +349 -0
  11. package/dist/index-Cn3o840t.js.map +1 -0
  12. package/dist/worker-fetch.js +22 -9
  13. package/dist/worker-fetch.js.map +1 -1
  14. package/package.json +21 -15
  15. package/unstable-preview-types/data-worker/cache-handler.d.ts.map +1 -1
  16. package/unstable-preview-types/data-worker/fetch.d.ts.map +1 -1
  17. package/unstable-preview-types/data-worker/worker.d.ts +11 -1
  18. package/unstable-preview-types/data-worker/worker.d.ts.map +1 -1
  19. package/unstable-preview-types/document-storage/index.d.ts +4 -1
  20. package/unstable-preview-types/document-storage/index.d.ts.map +1 -1
  21. package/unstable-preview-types/image-fetch.d.ts +4 -0
  22. package/unstable-preview-types/image-fetch.d.ts.map +1 -0
  23. package/unstable-preview-types/image-worker/fetch.d.ts +21 -0
  24. package/unstable-preview-types/image-worker/fetch.d.ts.map +1 -0
  25. package/unstable-preview-types/image-worker/types.d.ts +24 -0
  26. package/unstable-preview-types/image-worker/types.d.ts.map +1 -0
  27. package/unstable-preview-types/image-worker/worker.d.ts +20 -0
  28. package/unstable-preview-types/image-worker/worker.d.ts.map +1 -0
  29. package/unstable-preview-types/image-worker.d.ts +4 -0
  30. package/unstable-preview-types/image-worker.d.ts.map +1 -0
  31. package/unstable-preview-types/index.d.ts +11 -6
package/README.md CHANGED
@@ -28,6 +28,7 @@ pnpm add @warp-drive-mirror/experiments
28
28
 
29
29
  ## Current Experiments
30
30
 
31
- - [PersistedCache](./src/data-worker/README.md)
31
+ - [DataWorker](./src/data-worker/README.md)
32
32
  - [DocumentStorage](./src/document-storage/README.md)
33
+ - [ImageWorker](./src/image-worker/README.md)
33
34
  - [PersistedCache](./src/persisted-cache/README.md)
@@ -1,23 +1,59 @@
1
+ import { D as DocumentStorage } from "./index-Cn3o840t";
1
2
  import { SkipCache } from '@warp-drive-mirror/core-types/request';
2
3
  import { macroCondition, getGlobalConfig } from '@embroider/macros';
4
+ const WorkerScope = globalThis.SharedWorkerGlobalScope;
3
5
  class DataWorker {
4
- constructor(UserStore) {
6
+ constructor(UserStore, options) {
7
+ // disable if running on main thread
8
+ if (typeof window !== 'undefined') {
9
+ return;
10
+ }
5
11
  this.store = new UserStore();
6
12
  this.threads = new Map();
7
13
  this.pending = new Map();
14
+ this.options = Object.assign({
15
+ persisted: false,
16
+ scope: ''
17
+ }, options);
18
+ this.isSharedWorker = WorkerScope && globalThis instanceof WorkerScope;
8
19
  this.initialize();
9
20
  }
10
21
  initialize() {
11
- globalThis.onmessage = event => {
12
- const {
13
- type
14
- } = event.data;
15
- switch (type) {
16
- case 'connect':
17
- this.setupThread(event.data.thread, event.ports[0]);
18
- break;
19
- }
20
- };
22
+ // enable the CacheHandler to access the worker
23
+ this.store._worker = this;
24
+ if (this.options.persisted) {
25
+ // will be accessed by the worker's CacheHandler off of store
26
+ this.storage = new DocumentStorage({
27
+ scope: this.options.scope
28
+ });
29
+ }
30
+ if (this.isSharedWorker) {
31
+ globalThis.onconnect = e => {
32
+ const port = e.ports[0];
33
+ port.onmessage = event => {
34
+ const {
35
+ type
36
+ } = event.data;
37
+ switch (type) {
38
+ case 'connect':
39
+ this.setupThread(event.data.thread, port);
40
+ break;
41
+ }
42
+ };
43
+ port.start();
44
+ };
45
+ } else {
46
+ globalThis.onmessage = event => {
47
+ const {
48
+ type
49
+ } = event.data;
50
+ switch (type) {
51
+ case 'connect':
52
+ this.setupThread(event.data.thread, event.ports[0]);
53
+ break;
54
+ }
55
+ };
56
+ }
21
57
  }
22
58
  setupThread(thread, port) {
23
59
  this.threads.set(thread, port);
@@ -35,7 +71,7 @@ class DataWorker {
35
71
  this.abortRequest(event.data);
36
72
  break;
37
73
  case 'request':
38
- void this.request(event.data);
74
+ void this.request(prepareRequest(event.data));
39
75
  break;
40
76
  }
41
77
  };
@@ -84,10 +120,7 @@ function softCloneResponse(response) {
84
120
  if (!response) return null;
85
121
  const clone = {};
86
122
  if (response.headers) {
87
- clone.headers = {};
88
- for (const [key, value] of response.headers.entries()) {
89
- clone.headers[key] = value;
90
- }
123
+ clone.headers = Array.from(response.headers.entries());
91
124
  }
92
125
  clone.ok = response.ok;
93
126
  clone.redirected = response.redirected;
@@ -107,6 +140,12 @@ function prepareResponse(result) {
107
140
  };
108
141
  return newResponse;
109
142
  }
143
+ function prepareRequest(event) {
144
+ if (event.data.headers) {
145
+ event.data.headers = new Headers(event.data.headers);
146
+ }
147
+ return event;
148
+ }
110
149
  const MUTATION_OPS = new Set(['createRecord', 'updateRecord', 'deleteRecord']);
111
150
 
112
151
  /**
@@ -150,6 +189,13 @@ function cloneError(error) {
150
189
  Object.assign(cloned, error);
151
190
  return cloned;
152
191
  }
192
+
193
+ /**
194
+ * A simplified CacheHandler that hydrates ResourceDataDocuments from the cache
195
+ * with their referenced resources.
196
+ *
197
+ * @typedoc
198
+ */
153
199
  const CacheHandler = {
154
200
  request(context, next) {
155
201
  // if we have no cache or no cache-key skip cache handling
@@ -161,24 +207,45 @@ const CacheHandler = {
161
207
  } = context.request;
162
208
  const identifier = store.identifierCache.getOrCreateDocumentIdentifier(context.request);
163
209
  const peeked = identifier ? store.cache.peekRequest(identifier) : null;
164
-
165
- // In a Worker, any time we are asked to make a request, data needs to be returned.
166
- // background requests are ergo no different than foreground requests.
167
- if (calcShouldFetch(store, context.request, !!peeked, identifier)) {
168
- return fetchContentAndHydrate(next, context, identifier);
169
- }
170
- macroCondition(getGlobalConfig().WarpDrive.env.DEBUG) ? (test => {
171
- if (!test) {
172
- throw new Error(`Expected a peeked request to be present`);
210
+ if (identifier && !peeked) {
211
+ // if we are using persisted cache, we should attempt to populate the in-memory cache now
212
+ const worker = store._worker;
213
+ if (worker?.storage) {
214
+ return worker.storage.getDocument(identifier).then(document => {
215
+ if (document) {
216
+ store.cache.put(document);
217
+ }
218
+ return completeRequest(identifier, store, context, next);
219
+ }).catch(e => {
220
+ if (macroCondition(getGlobalConfig().WarpDrive.env.DEBUG)) {
221
+ // eslint-disable-next-line no-console
222
+ console.log('Unable to retrieve document from persisted storage', e);
223
+ }
224
+ return completeRequest(identifier, store, context, next);
225
+ });
173
226
  }
174
- })(peeked) : {};
175
- context.setResponse(peeked.response);
176
- if ('error' in peeked) {
177
- throw peeked;
178
227
  }
179
- return maybeUpdateObjects(store, peeked.content);
228
+ return completeRequest(identifier, store, context, next);
180
229
  }
181
230
  };
231
+ function completeRequest(identifier, store, context, next) {
232
+ const peeked = identifier ? store.cache.peekRequest(identifier) : null;
233
+ // In a Worker, any time we are asked to make a request, data needs to be returned.
234
+ // background requests are ergo no different than foreground requests.
235
+ if (calcShouldFetch(store, context.request, !!peeked, identifier)) {
236
+ return fetchContentAndHydrate(next, context, identifier);
237
+ }
238
+ macroCondition(getGlobalConfig().WarpDrive.env.DEBUG) ? (test => {
239
+ if (!test) {
240
+ throw new Error(`Expected a peeked request to be present`);
241
+ }
242
+ })(peeked) : {};
243
+ context.setResponse(peeked.response);
244
+ if ('error' in peeked) {
245
+ throw peeked;
246
+ }
247
+ return maybeUpdateObjects(store, peeked.content);
248
+ }
182
249
  function maybeUpdateObjects(store, document) {
183
250
  if (!document) {
184
251
  return document;
@@ -197,6 +264,22 @@ function maybeUpdateObjects(store, document) {
197
264
  });
198
265
  }
199
266
  }
267
+ function maybeUpdatePersistedCache(store, document, resourceDocument) {
268
+ const worker = store._worker;
269
+ if (!worker?.storage) {
270
+ return;
271
+ }
272
+ if (!document && resourceDocument) {
273
+ // we have resources to update but not a full request to cache
274
+ void worker.storage.putResources(resourceDocument, resourceIdentifier => {
275
+ return store.cache.peek(resourceIdentifier);
276
+ });
277
+ } else if (document) {
278
+ void worker.storage.putDocument(document, resourceIdentifier => {
279
+ return store.cache.peek(resourceIdentifier);
280
+ });
281
+ }
282
+ }
200
283
  function updateCacheForSuccess(store, request, document) {
201
284
  let response = null;
202
285
  if (isMutation(request)) {
@@ -209,9 +292,15 @@ function updateCacheForSuccess(store, request, document) {
209
292
  // have no cache impact
210
293
  } else if (isCacheAffecting(document)) {
211
294
  response = store.cache.put(document);
295
+ maybeUpdatePersistedCache(store, null, response);
212
296
  }
213
297
  } else {
214
298
  response = store.cache.put(document);
299
+ if (response.lid) {
300
+ const identifier = store.identifierCache.getOrCreateDocumentIdentifier(request);
301
+ const full = store.cache.peekRequest(identifier);
302
+ maybeUpdatePersistedCache(store, full);
303
+ }
215
304
  }
216
305
  return maybeUpdateObjects(store, response);
217
306
  }
@@ -233,6 +322,10 @@ function updateCacheForError(store, request, error) {
233
322
  const record = request.data?.record || request.records?.[0];
234
323
  store.cache.commitWasRejected(record, errors);
235
324
  } else {
325
+ const identifier = store.identifierCache.getOrCreateDocumentIdentifier(request);
326
+ if (identifier) {
327
+ maybeUpdatePersistedCache(store, error);
328
+ }
236
329
  return store.cache.put(error);
237
330
  }
238
331
  }
@@ -1 +1 @@
1
- {"version":3,"file":"data-worker.js","sources":["../src/data-worker/worker.ts","../src/data-worker/utils.ts","../src/data-worker/cache-handler.ts"],"sourcesContent":["import type { Future, ResponseInfo, StructuredDataDocument } from '@ember-data-mirror/request';\nimport type Store from '@ember-data-mirror/store';\n\nimport type { AbortEventData, RequestEventData, ThreadInitEventData, WorkerThreadEvent } from './types';\n\nexport class DataWorker {\n declare store: Store;\n declare threads: Map<string, MessagePort>;\n declare pending: Map<string, Map<number, Future<unknown>>>;\n\n constructor(UserStore: typeof Store) {\n this.store = new UserStore();\n this.threads = new Map();\n this.pending = new Map();\n this.initialize();\n }\n\n initialize() {\n globalThis.onmessage = (event: MessageEvent<ThreadInitEventData>) => {\n const { type } = event.data;\n\n switch (type) {\n case 'connect':\n this.setupThread(event.data.thread, event.ports[0]);\n break;\n }\n };\n }\n\n setupThread(thread: string, port: MessagePort) {\n this.threads.set(thread, port);\n this.pending.set(thread, new Map());\n port.onmessage = (event: WorkerThreadEvent) => {\n if (event.type === 'close') {\n this.threads.delete(thread);\n return;\n }\n\n const { type } = event.data;\n switch (type) {\n case 'abort':\n this.abortRequest(event.data);\n break;\n case 'request':\n void this.request(event.data);\n break;\n }\n };\n }\n\n abortRequest(event: AbortEventData) {\n const { thread, id } = event;\n const future = this.pending.get(thread)!.get(id);\n\n if (future) {\n future.abort();\n this.pending.get(thread)!.delete(id);\n }\n }\n\n async request(event: RequestEventData) {\n const { thread, id, data } = event;\n\n try {\n const future = this.store.request(data);\n this.pending.get(thread)!.set(id, future);\n\n const result = await future;\n\n this.threads.get(thread)?.postMessage({ type: 'success-response', id, thread, data: prepareResponse(result) });\n } catch (error) {\n if (isAbortError(error)) return;\n\n this.threads.get(thread)?.postMessage({ type: 'error-response', id, thread, data: error });\n } finally {\n this.pending.get(thread)!.delete(id);\n }\n }\n}\n\ntype Mutable<T> = { -readonly [P in keyof T]: T[P] };\n\nfunction softCloneResponse(response: Response | ResponseInfo | null) {\n if (!response) return null;\n\n const clone: Partial<Mutable<Omit<Response, 'headers'>>> & { headers?: Record<string, string | number> } = {};\n\n if (response.headers) {\n clone.headers = {};\n for (const [key, value] of response.headers.entries()) {\n clone.headers[key] = value;\n }\n }\n\n clone.ok = response.ok;\n clone.redirected = response.redirected;\n clone.status = response.status;\n clone.statusText = response.statusText;\n clone.type = response.type;\n clone.url = response.url;\n\n return clone;\n}\n\nfunction isAbortError(error: unknown): error is Error {\n return error instanceof Error && error.name === 'AbortError';\n}\n\nfunction prepareResponse<T>(result: StructuredDataDocument<T>) {\n const newResponse = {\n response: softCloneResponse(result.response),\n content: result.content,\n };\n\n return newResponse;\n}\n","import type Store from '@ember-data-mirror/store';\nimport type { StableDocumentIdentifier } from '@warp-drive-mirror/core-types/identifier';\nimport type {\n ImmutableCreateRequestOptions,\n ImmutableDeleteRequestOptions,\n ImmutableRequestInfo,\n ImmutableUpdateRequestOptions,\n StructuredDataDocument,\n} from '@warp-drive-mirror/core-types/request';\nimport type { ApiError } from '@warp-drive-mirror/core-types/spec/error';\n\nexport const MUTATION_OPS = new Set(['createRecord', 'updateRecord', 'deleteRecord']);\n\n/**\n * In a Worker, any time we are asked to make a request, data needs to be returned.\n * background requests are ergo no different than foreground requests.\n * @internal\n */\nexport function calcShouldFetch(\n store: Store,\n request: ImmutableRequestInfo,\n hasCachedValue: boolean,\n identifier: StableDocumentIdentifier | null\n): boolean {\n const { cacheOptions } = request;\n return (\n (request.op && MUTATION_OPS.has(request.op)) ||\n cacheOptions?.reload ||\n cacheOptions?.backgroundReload ||\n !hasCachedValue ||\n (store.lifetimes && identifier\n ? store.lifetimes.isHardExpired(identifier, store) || store.lifetimes.isSoftExpired(identifier, store)\n : false)\n );\n}\n\nexport function isMutation(\n request: Partial<ImmutableRequestInfo>\n): request is ImmutableUpdateRequestOptions | ImmutableCreateRequestOptions | ImmutableDeleteRequestOptions {\n return Boolean(request.op && MUTATION_OPS.has(request.op));\n}\n\nexport function isCacheAffecting<T>(document: StructuredDataDocument<T>): boolean {\n if (!isMutation(document.request)) {\n return true;\n }\n // a mutation combined with a 204 has no cache impact when no known records were involved\n // a createRecord with a 201 with an empty response and no known records should similarly\n // have no cache impact\n\n if (document.request.op === 'createRecord' && document.response?.status === 201) {\n return document.content ? Object.keys(document.content).length > 0 : false;\n }\n\n return document.response?.status !== 204;\n}\n\nfunction isAggregateError(error: Error & { errors?: ApiError[] }): error is AggregateError & { errors: ApiError[] } {\n return error instanceof AggregateError || (error.name === 'AggregateError' && Array.isArray(error.errors));\n}\n\ntype RobustError = Error & { error: string | object; errors?: ApiError[]; content?: unknown };\n\n// TODO @runspired, consider if we should deep freeze errors (potentially only in debug) vs cloning them\nexport function cloneError(error: RobustError) {\n const isAggregate = isAggregateError(error);\n\n const cloned = (\n isAggregate ? new AggregateError(structuredClone(error.errors), error.message) : new Error(error.message)\n ) as RobustError;\n cloned.stack = error.stack!;\n cloned.error = error.error;\n\n // copy over enumerable properties\n Object.assign(cloned, error);\n\n return cloned;\n}\n","import type { CacheHandler as CacheHandlerType, Future, NextFn } from '@ember-data-mirror/request';\nimport type Store from '@ember-data-mirror/store';\nimport type { StoreRequestContext } from '@ember-data-mirror/store';\nimport { assert } from '@warp-drive-mirror/build-config/macros';\nimport type { StableDocumentIdentifier } from '@warp-drive-mirror/core-types/identifier';\nimport type { StructuredDataDocument, StructuredErrorDocument } from '@warp-drive-mirror/core-types/request';\nimport { SkipCache } from '@warp-drive-mirror/core-types/request';\nimport type { ResourceDataDocument, ResourceErrorDocument } from '@warp-drive-mirror/core-types/spec/document';\nimport type { ApiError } from '@warp-drive-mirror/core-types/spec/error';\n\nimport { calcShouldFetch, cloneError, isCacheAffecting, isMutation } from './utils';\n\n/**\n * A simplified CacheHandler that hydrates ResourceDataDocuments from the cache\n * with their referenced resources.\n *\n * @typedoc\n */\nexport const CacheHandler: CacheHandlerType = {\n request<T>(context: StoreRequestContext, next: NextFn<T>): Promise<T | StructuredDataDocument<T>> | Future<T> | T {\n // if we have no cache or no cache-key skip cache handling\n if (!context.request.store || context.request.cacheOptions?.[SkipCache]) {\n return next(context.request);\n }\n\n const { store } = context.request;\n const identifier = store.identifierCache.getOrCreateDocumentIdentifier(context.request);\n const peeked = identifier ? store.cache.peekRequest(identifier) : null;\n\n // In a Worker, any time we are asked to make a request, data needs to be returned.\n // background requests are ergo no different than foreground requests.\n if (calcShouldFetch(store, context.request, !!peeked, identifier)) {\n return fetchContentAndHydrate(next, context, identifier);\n }\n\n assert(`Expected a peeked request to be present`, peeked);\n context.setResponse(peeked.response);\n\n if ('error' in peeked) {\n throw peeked;\n }\n\n return maybeUpdateObjects<T>(store, peeked.content as ResourceDataDocument);\n },\n};\n\nfunction maybeUpdateObjects<T>(store: Store, document: ResourceDataDocument | null): T {\n if (!document) {\n return document as T;\n }\n\n if (Array.isArray(document.data)) {\n const data = document.data.map((identifier) => {\n return store.cache.peek(identifier);\n });\n\n return Object.assign({}, document, { data }) as T;\n } else {\n const data = (document.data ? store.cache.peek(document.data) : null) as T;\n return Object.assign({}, document, { data }) as T;\n }\n}\n\nfunction updateCacheForSuccess<T>(\n store: Store,\n request: StoreRequestContext['request'],\n document: StructuredDataDocument<T>\n) {\n let response: ResourceDataDocument | null = null;\n if (isMutation(request)) {\n const record = request.data?.record || request.records?.[0];\n if (record) {\n response = store.cache.didCommit(record, document) as ResourceDataDocument;\n\n // a mutation combined with a 204 has no cache impact when no known records were involved\n // a createRecord with a 201 with an empty response and no known records should similarly\n // have no cache impact\n } else if (isCacheAffecting(document)) {\n response = store.cache.put(document) as ResourceDataDocument;\n }\n } else {\n response = store.cache.put(document) as ResourceDataDocument;\n }\n return maybeUpdateObjects(store, response);\n}\n\nfunction handleFetchSuccess<T>(\n store: Store,\n request: StoreRequestContext['request'],\n identifier: StableDocumentIdentifier | null,\n document: StructuredDataDocument<T>\n): T {\n let response: ResourceDataDocument;\n store._join(() => {\n response = updateCacheForSuccess<T>(store, request, document) as ResourceDataDocument;\n });\n\n if (store.lifetimes?.didRequest) {\n store.lifetimes.didRequest(request, document.response, identifier, store);\n }\n\n return response! as T;\n}\n\nfunction updateCacheForError<T>(\n store: Store,\n request: StoreRequestContext['request'],\n error: StructuredErrorDocument<T>\n) {\n if (isMutation(request)) {\n // TODO similar to didCommit we should spec this to be similar to cache.put for handling full response\n // currently we let the response remain undefiend.\n const errors =\n error &&\n error.content &&\n typeof error.content === 'object' &&\n 'errors' in error.content &&\n Array.isArray(error.content.errors)\n ? (error.content.errors as ApiError[])\n : undefined;\n\n const record = request.data?.record || request.records?.[0];\n\n store.cache.commitWasRejected(record, errors);\n } else {\n return store.cache.put(error) as ResourceErrorDocument;\n }\n}\n\nfunction handleFetchError<T>(\n store: Store,\n request: StoreRequestContext['request'],\n identifier: StableDocumentIdentifier | null,\n error: StructuredErrorDocument<T>\n): never {\n if (request.signal?.aborted) {\n throw error;\n }\n let response: ResourceErrorDocument | undefined;\n store._join(() => {\n response = updateCacheForError(store, request, error);\n });\n\n if (identifier && store.lifetimes?.didRequest) {\n store.lifetimes.didRequest(request, error.response, identifier, store);\n }\n\n if (isMutation(request)) {\n throw error;\n }\n\n const newError = cloneError(error);\n newError.content = response!;\n throw newError;\n}\n\nfunction fetchContentAndHydrate<T>(\n next: NextFn<T>,\n context: StoreRequestContext,\n identifier: StableDocumentIdentifier | null\n): Promise<T> {\n const { request } = context;\n const { store } = context.request;\n\n if (isMutation(request)) {\n // TODO should we handle multiple records in request.records by iteratively calling willCommit for each\n const record = request.data?.record || request.records?.[0];\n assert(`Expected to receive a list of records included in the ${request.op} request`, record);\n if (record) {\n store.cache.willCommit(record, context);\n }\n }\n\n if (store.lifetimes?.willRequest) {\n store.lifetimes.willRequest(request, identifier, store);\n }\n\n return next(request).then(\n (document) => handleFetchSuccess(store, request, identifier, document),\n (error: StructuredErrorDocument<T>) => handleFetchError(store, request, identifier, error)\n );\n}\n"],"names":["DataWorker","constructor","UserStore","store","threads","Map","pending","initialize","globalThis","onmessage","event","type","data","setupThread","thread","ports","port","set","delete","abortRequest","request","id","future","get","abort","result","postMessage","prepareResponse","error","isAbortError","softCloneResponse","response","clone","headers","key","value","entries","ok","redirected","status","statusText","url","Error","name","newResponse","content","MUTATION_OPS","Set","calcShouldFetch","hasCachedValue","identifier","cacheOptions","op","has","reload","backgroundReload","lifetimes","isHardExpired","isSoftExpired","isMutation","Boolean","isCacheAffecting","document","Object","keys","length","isAggregateError","AggregateError","Array","isArray","errors","cloneError","isAggregate","cloned","structuredClone","message","stack","assign","CacheHandler","context","next","SkipCache","identifierCache","getOrCreateDocumentIdentifier","peeked","cache","peekRequest","fetchContentAndHydrate","macroCondition","getGlobalConfig","WarpDrive","env","DEBUG","test","setResponse","maybeUpdateObjects","map","peek","updateCacheForSuccess","record","records","didCommit","put","handleFetchSuccess","_join","didRequest","updateCacheForError","undefined","commitWasRejected","handleFetchError","signal","aborted","newError","willCommit","willRequest","then"],"mappings":"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"}
1
+ {"version":3,"file":"data-worker.js","sources":["../src/data-worker/worker.ts","../src/data-worker/utils.ts","../src/data-worker/cache-handler.ts"],"sourcesContent":["import type { Future, ResponseInfo, StructuredDataDocument } from '@ember-data-mirror/request';\nimport type Store from '@ember-data-mirror/store';\n\nimport { DocumentStorage } from '../document-storage';\nimport type { AbortEventData, RequestEventData, ThreadInitEventData, WorkerThreadEvent } from './types';\n\nconst WorkerScope = (globalThis as unknown as { SharedWorkerGlobalScope: FunctionConstructor }).SharedWorkerGlobalScope;\n\nexport class DataWorker {\n declare store: Store;\n declare threads: Map<string, MessagePort>;\n declare pending: Map<string, Map<number, Future<unknown>>>;\n declare isSharedWorker: boolean;\n declare options: { persisted: boolean; scope?: string };\n declare storage: DocumentStorage;\n\n constructor(UserStore: typeof Store, options?: { persisted: boolean; scope?: string }) {\n // disable if running on main thread\n if (typeof window !== 'undefined') {\n return;\n }\n this.store = new UserStore();\n this.threads = new Map();\n this.pending = new Map();\n this.options = Object.assign({ persisted: false, scope: '' }, options);\n this.isSharedWorker = WorkerScope && globalThis instanceof WorkerScope;\n this.initialize();\n }\n\n initialize() {\n // enable the CacheHandler to access the worker\n (this.store as unknown as { _worker: DataWorker })._worker = this;\n if (this.options.persisted) {\n // will be accessed by the worker's CacheHandler off of store\n this.storage = new DocumentStorage({ scope: this.options.scope });\n }\n if (this.isSharedWorker) {\n (globalThis as unknown as { onconnect: typeof globalThis.onmessage }).onconnect = (e) => {\n const port = e.ports[0];\n port.onmessage = (event: MessageEvent<ThreadInitEventData>) => {\n const { type } = event.data;\n\n switch (type) {\n case 'connect':\n this.setupThread(event.data.thread, port);\n break;\n }\n };\n port.start();\n };\n } else {\n globalThis.onmessage = (event: MessageEvent<ThreadInitEventData>) => {\n const { type } = event.data;\n\n switch (type) {\n case 'connect':\n this.setupThread(event.data.thread, event.ports[0]);\n break;\n }\n };\n }\n }\n\n setupThread(thread: string, port: MessagePort) {\n this.threads.set(thread, port);\n this.pending.set(thread, new Map());\n port.onmessage = (event: WorkerThreadEvent) => {\n if (event.type === 'close') {\n this.threads.delete(thread);\n return;\n }\n\n const { type } = event.data;\n switch (type) {\n case 'abort':\n this.abortRequest(event.data);\n break;\n case 'request':\n void this.request(prepareRequest(event.data));\n break;\n }\n };\n }\n\n abortRequest(event: AbortEventData) {\n const { thread, id } = event;\n const future = this.pending.get(thread)!.get(id);\n\n if (future) {\n future.abort();\n this.pending.get(thread)!.delete(id);\n }\n }\n\n async request(event: RequestEventData) {\n const { thread, id, data } = event;\n\n try {\n const future = this.store.request(data);\n this.pending.get(thread)!.set(id, future);\n\n const result = await future;\n\n this.threads.get(thread)?.postMessage({ type: 'success-response', id, thread, data: prepareResponse(result) });\n } catch (error) {\n if (isAbortError(error)) return;\n\n this.threads.get(thread)?.postMessage({ type: 'error-response', id, thread, data: error });\n } finally {\n this.pending.get(thread)!.delete(id);\n }\n }\n}\n\ntype Mutable<T> = { -readonly [P in keyof T]: T[P] };\n\nfunction softCloneResponse(response: Response | ResponseInfo | null) {\n if (!response) return null;\n\n const clone: Partial<Mutable<Response>> = {};\n\n if (response.headers) {\n clone.headers = Array.from(response.headers.entries()) as unknown as Headers;\n }\n\n clone.ok = response.ok;\n clone.redirected = response.redirected;\n clone.status = response.status;\n clone.statusText = response.statusText;\n clone.type = response.type;\n clone.url = response.url;\n\n return clone;\n}\n\nfunction isAbortError(error: unknown): error is Error {\n return error instanceof Error && error.name === 'AbortError';\n}\n\nfunction prepareResponse<T>(result: StructuredDataDocument<T>) {\n const newResponse = {\n response: softCloneResponse(result.response),\n content: result.content,\n };\n\n return newResponse;\n}\n\nfunction prepareRequest(event: RequestEventData) {\n if (event.data.headers) {\n event.data.headers = new Headers(event.data.headers);\n }\n\n return event;\n}\n","import type Store from '@ember-data-mirror/store';\nimport type { StableDocumentIdentifier } from '@warp-drive-mirror/core-types/identifier';\nimport type {\n ImmutableCreateRequestOptions,\n ImmutableDeleteRequestOptions,\n ImmutableRequestInfo,\n ImmutableUpdateRequestOptions,\n StructuredDataDocument,\n} from '@warp-drive-mirror/core-types/request';\nimport type { ApiError } from '@warp-drive-mirror/core-types/spec/error';\n\nexport const MUTATION_OPS = new Set(['createRecord', 'updateRecord', 'deleteRecord']);\n\n/**\n * In a Worker, any time we are asked to make a request, data needs to be returned.\n * background requests are ergo no different than foreground requests.\n * @internal\n */\nexport function calcShouldFetch(\n store: Store,\n request: ImmutableRequestInfo,\n hasCachedValue: boolean,\n identifier: StableDocumentIdentifier | null\n): boolean {\n const { cacheOptions } = request;\n return (\n (request.op && MUTATION_OPS.has(request.op)) ||\n cacheOptions?.reload ||\n cacheOptions?.backgroundReload ||\n !hasCachedValue ||\n (store.lifetimes && identifier\n ? store.lifetimes.isHardExpired(identifier, store) || store.lifetimes.isSoftExpired(identifier, store)\n : false)\n );\n}\n\nexport function isMutation(\n request: Partial<ImmutableRequestInfo>\n): request is ImmutableUpdateRequestOptions | ImmutableCreateRequestOptions | ImmutableDeleteRequestOptions {\n return Boolean(request.op && MUTATION_OPS.has(request.op));\n}\n\nexport function isCacheAffecting<T>(document: StructuredDataDocument<T>): boolean {\n if (!isMutation(document.request)) {\n return true;\n }\n // a mutation combined with a 204 has no cache impact when no known records were involved\n // a createRecord with a 201 with an empty response and no known records should similarly\n // have no cache impact\n\n if (document.request.op === 'createRecord' && document.response?.status === 201) {\n return document.content ? Object.keys(document.content).length > 0 : false;\n }\n\n return document.response?.status !== 204;\n}\n\nfunction isAggregateError(error: Error & { errors?: ApiError[] }): error is AggregateError & { errors: ApiError[] } {\n return error instanceof AggregateError || (error.name === 'AggregateError' && Array.isArray(error.errors));\n}\n\ntype RobustError = Error & { error: string | object; errors?: ApiError[]; content?: unknown };\n\n// TODO @runspired, consider if we should deep freeze errors (potentially only in debug) vs cloning them\nexport function cloneError(error: RobustError) {\n const isAggregate = isAggregateError(error);\n\n const cloned = (\n isAggregate ? new AggregateError(structuredClone(error.errors), error.message) : new Error(error.message)\n ) as RobustError;\n cloned.stack = error.stack!;\n cloned.error = error.error;\n\n // copy over enumerable properties\n Object.assign(cloned, error);\n\n return cloned;\n}\n","import type { CacheHandler as CacheHandlerType, Future, NextFn } from '@ember-data-mirror/request';\nimport type Store from '@ember-data-mirror/store';\nimport type { StoreRequestContext } from '@ember-data-mirror/store';\nimport { DEBUG } from '@warp-drive-mirror/build-config/env';\nimport { assert } from '@warp-drive-mirror/build-config/macros';\nimport type { ExistingRecordIdentifier, StableDocumentIdentifier } from '@warp-drive-mirror/core-types/identifier';\nimport type {\n StructuredDataDocument,\n StructuredDocument,\n StructuredErrorDocument,\n} from '@warp-drive-mirror/core-types/request';\nimport { SkipCache } from '@warp-drive-mirror/core-types/request';\nimport type {\n ResourceDataDocument,\n ResourceDocument,\n ResourceErrorDocument,\n} from '@warp-drive-mirror/core-types/spec/document';\nimport type { ApiError } from '@warp-drive-mirror/core-types/spec/error';\nimport type { ExistingResourceObject } from '@warp-drive-mirror/core-types/spec/json-api-raw';\n\nimport { calcShouldFetch, cloneError, isCacheAffecting, isMutation } from './utils';\nimport type { DataWorker } from './worker';\n\n/**\n * A simplified CacheHandler that hydrates ResourceDataDocuments from the cache\n * with their referenced resources.\n *\n * @typedoc\n */\nexport const CacheHandler: CacheHandlerType = {\n request<T>(context: StoreRequestContext, next: NextFn<T>): Promise<T | StructuredDataDocument<T>> | Future<T> | T {\n // if we have no cache or no cache-key skip cache handling\n if (!context.request.store || context.request.cacheOptions?.[SkipCache]) {\n return next(context.request);\n }\n\n const { store } = context.request;\n const identifier = store.identifierCache.getOrCreateDocumentIdentifier(context.request);\n const peeked = identifier ? store.cache.peekRequest(identifier) : null;\n\n if (identifier && !peeked) {\n // if we are using persisted cache, we should attempt to populate the in-memory cache now\n const worker = (store as unknown as { _worker: DataWorker })._worker;\n if (worker?.storage) {\n return worker.storage\n .getDocument(identifier)\n .then((document) => {\n if (document) {\n store.cache.put(document);\n }\n return completeRequest(identifier, store, context, next);\n })\n .catch((e) => {\n if (DEBUG) {\n // eslint-disable-next-line no-console\n console.log('Unable to retrieve document from persisted storage', e);\n }\n return completeRequest(identifier, store, context, next);\n });\n }\n }\n\n return completeRequest(identifier, store, context, next);\n },\n};\n\nfunction completeRequest<T>(\n identifier: StableDocumentIdentifier | null,\n store: Store,\n context: StoreRequestContext,\n next: NextFn<T>\n) {\n const peeked = identifier ? store.cache.peekRequest(identifier) : null;\n // In a Worker, any time we are asked to make a request, data needs to be returned.\n // background requests are ergo no different than foreground requests.\n if (calcShouldFetch(store, context.request, !!peeked, identifier)) {\n return fetchContentAndHydrate(next, context, identifier);\n }\n\n assert(`Expected a peeked request to be present`, peeked);\n context.setResponse(peeked.response);\n\n if ('error' in peeked) {\n throw peeked;\n }\n\n return maybeUpdateObjects<T>(store, peeked.content as ResourceDataDocument);\n}\n\nfunction maybeUpdateObjects<T>(store: Store, document: ResourceDataDocument | null): T {\n if (!document) {\n return document as T;\n }\n\n if (Array.isArray(document.data)) {\n const data = document.data.map((identifier) => {\n return store.cache.peek(identifier);\n });\n\n return Object.assign({}, document, { data }) as T;\n } else {\n const data = (document.data ? store.cache.peek(document.data) : null) as T;\n return Object.assign({}, document, { data }) as T;\n }\n}\n\nfunction maybeUpdatePersistedCache(\n store: Store,\n document: StructuredDocument<ResourceDocument<ExistingRecordIdentifier>> | null,\n resourceDocument?: ResourceDataDocument\n) {\n const worker = (store as unknown as { _worker: DataWorker })._worker;\n\n if (!worker?.storage) {\n return;\n }\n\n if (!document && resourceDocument) {\n // we have resources to update but not a full request to cache\n void worker.storage.putResources(resourceDocument, (resourceIdentifier) => {\n return store.cache.peek(resourceIdentifier) as ExistingResourceObject;\n });\n } else if (document) {\n void worker.storage.putDocument(document, (resourceIdentifier) => {\n return store.cache.peek(resourceIdentifier) as ExistingResourceObject;\n });\n }\n}\n\nfunction updateCacheForSuccess<T>(\n store: Store,\n request: StoreRequestContext['request'],\n document: StructuredDataDocument<T>\n) {\n let response: ResourceDataDocument | null = null;\n if (isMutation(request)) {\n const record = request.data?.record || request.records?.[0];\n if (record) {\n response = store.cache.didCommit(record, document) as ResourceDataDocument;\n\n // a mutation combined with a 204 has no cache impact when no known records were involved\n // a createRecord with a 201 with an empty response and no known records should similarly\n // have no cache impact\n } else if (isCacheAffecting(document)) {\n response = store.cache.put(document) as ResourceDataDocument;\n maybeUpdatePersistedCache(store, null, response);\n }\n } else {\n response = store.cache.put(document) as ResourceDataDocument;\n\n if (response.lid) {\n const identifier = store.identifierCache.getOrCreateDocumentIdentifier(request);\n const full = store.cache.peekRequest(identifier!);\n maybeUpdatePersistedCache(store, full);\n }\n }\n return maybeUpdateObjects(store, response);\n}\n\nfunction handleFetchSuccess<T>(\n store: Store,\n request: StoreRequestContext['request'],\n identifier: StableDocumentIdentifier | null,\n document: StructuredDataDocument<T>\n): T {\n let response: ResourceDataDocument;\n store._join(() => {\n response = updateCacheForSuccess<T>(store, request, document) as ResourceDataDocument;\n });\n\n if (store.lifetimes?.didRequest) {\n store.lifetimes.didRequest(request, document.response, identifier, store);\n }\n\n return response! as T;\n}\n\nfunction updateCacheForError<T>(\n store: Store,\n request: StoreRequestContext['request'],\n error: StructuredErrorDocument<T>\n) {\n if (isMutation(request)) {\n // TODO similar to didCommit we should spec this to be similar to cache.put for handling full response\n // currently we let the response remain undefiend.\n const errors =\n error &&\n error.content &&\n typeof error.content === 'object' &&\n 'errors' in error.content &&\n Array.isArray(error.content.errors)\n ? (error.content.errors as ApiError[])\n : undefined;\n\n const record = request.data?.record || request.records?.[0];\n\n store.cache.commitWasRejected(record, errors);\n } else {\n const identifier = store.identifierCache.getOrCreateDocumentIdentifier(request);\n if (identifier) {\n maybeUpdatePersistedCache(store, error as StructuredErrorDocument<ResourceErrorDocument>);\n }\n return store.cache.put(error) as ResourceErrorDocument;\n }\n}\n\nfunction handleFetchError<T>(\n store: Store,\n request: StoreRequestContext['request'],\n identifier: StableDocumentIdentifier | null,\n error: StructuredErrorDocument<T>\n): never {\n if (request.signal?.aborted) {\n throw error;\n }\n let response: ResourceErrorDocument | undefined;\n store._join(() => {\n response = updateCacheForError(store, request, error);\n });\n\n if (identifier && store.lifetimes?.didRequest) {\n store.lifetimes.didRequest(request, error.response, identifier, store);\n }\n\n if (isMutation(request)) {\n throw error;\n }\n\n const newError = cloneError(error);\n newError.content = response!;\n throw newError;\n}\n\nfunction fetchContentAndHydrate<T>(\n next: NextFn<T>,\n context: StoreRequestContext,\n identifier: StableDocumentIdentifier | null\n): Promise<T> {\n const { request } = context;\n const { store } = context.request;\n\n if (isMutation(request)) {\n // TODO should we handle multiple records in request.records by iteratively calling willCommit for each\n const record = request.data?.record || request.records?.[0];\n assert(`Expected to receive a list of records included in the ${request.op} request`, record);\n if (record) {\n store.cache.willCommit(record, context);\n }\n }\n\n if (store.lifetimes?.willRequest) {\n store.lifetimes.willRequest(request, identifier, store);\n }\n\n return next(request).then(\n (document) => handleFetchSuccess(store, request, identifier, document),\n (error: StructuredErrorDocument<T>) => handleFetchError(store, request, identifier, error)\n );\n}\n"],"names":["WorkerScope","globalThis","SharedWorkerGlobalScope","DataWorker","constructor","UserStore","options","window","store","threads","Map","pending","Object","assign","persisted","scope","isSharedWorker","initialize","_worker","storage","DocumentStorage","onconnect","e","port","ports","onmessage","event","type","data","setupThread","thread","start","set","delete","abortRequest","request","prepareRequest","id","future","get","abort","result","postMessage","prepareResponse","error","isAbortError","softCloneResponse","response","clone","headers","Array","from","entries","ok","redirected","status","statusText","url","Error","name","newResponse","content","Headers","MUTATION_OPS","Set","calcShouldFetch","hasCachedValue","identifier","cacheOptions","op","has","reload","backgroundReload","lifetimes","isHardExpired","isSoftExpired","isMutation","Boolean","isCacheAffecting","document","keys","length","isAggregateError","AggregateError","isArray","errors","cloneError","isAggregate","cloned","structuredClone","message","stack","CacheHandler","context","next","SkipCache","identifierCache","getOrCreateDocumentIdentifier","peeked","cache","peekRequest","worker","getDocument","then","put","completeRequest","catch","macroCondition","getGlobalConfig","WarpDrive","env","DEBUG","console","log","fetchContentAndHydrate","test","setResponse","maybeUpdateObjects","map","peek","maybeUpdatePersistedCache","resourceDocument","putResources","resourceIdentifier","putDocument","updateCacheForSuccess","record","records","didCommit","lid","full","handleFetchSuccess","_join","didRequest","updateCacheForError","undefined","commitWasRejected","handleFetchError","signal","aborted","newError","willCommit","willRequest"],"mappings":";;;;AAMA,MAAMA,WAAW,GAAIC,UAAU,CAAiEC,uBAAuB,CAAA;AAEhH,MAAMC,UAAU,CAAC;AAQtBC,EAAAA,WAAWA,CAACC,SAAuB,EAAEC,OAAgD,EAAE;AACrF;AACA,IAAA,IAAI,OAAOC,MAAM,KAAK,WAAW,EAAE;AACjC,MAAA,OAAA;AACF,KAAA;AACA,IAAA,IAAI,CAACC,KAAK,GAAG,IAAIH,SAAS,EAAE,CAAA;AAC5B,IAAA,IAAI,CAACI,OAAO,GAAG,IAAIC,GAAG,EAAE,CAAA;AACxB,IAAA,IAAI,CAACC,OAAO,GAAG,IAAID,GAAG,EAAE,CAAA;AACxB,IAAA,IAAI,CAACJ,OAAO,GAAGM,MAAM,CAACC,MAAM,CAAC;AAAEC,MAAAA,SAAS,EAAE,KAAK;AAAEC,MAAAA,KAAK,EAAE,EAAA;KAAI,EAAET,OAAO,CAAC,CAAA;AACtE,IAAA,IAAI,CAACU,cAAc,GAAGhB,WAAW,IAAIC,UAAU,YAAYD,WAAW,CAAA;IACtE,IAAI,CAACiB,UAAU,EAAE,CAAA;AACnB,GAAA;AAEAA,EAAAA,UAAUA,GAAG;AACX;AACC,IAAA,IAAI,CAACT,KAAK,CAAwCU,OAAO,GAAG,IAAI,CAAA;AACjE,IAAA,IAAI,IAAI,CAACZ,OAAO,CAACQ,SAAS,EAAE;AAC1B;AACA,MAAA,IAAI,CAACK,OAAO,GAAG,IAAIC,eAAe,CAAC;AAAEL,QAAAA,KAAK,EAAE,IAAI,CAACT,OAAO,CAACS,KAAAA;AAAM,OAAC,CAAC,CAAA;AACnE,KAAA;IACA,IAAI,IAAI,CAACC,cAAc,EAAE;AACtBf,MAAAA,UAAU,CAA2DoB,SAAS,GAAIC,CAAC,IAAK;AACvF,QAAA,MAAMC,IAAI,GAAGD,CAAC,CAACE,KAAK,CAAC,CAAC,CAAC,CAAA;AACvBD,QAAAA,IAAI,CAACE,SAAS,GAAIC,KAAwC,IAAK;UAC7D,MAAM;AAAEC,YAAAA,IAAAA;WAAM,GAAGD,KAAK,CAACE,IAAI,CAAA;AAE3B,UAAA,QAAQD,IAAI;AACV,YAAA,KAAK,SAAS;cACZ,IAAI,CAACE,WAAW,CAACH,KAAK,CAACE,IAAI,CAACE,MAAM,EAAEP,IAAI,CAAC,CAAA;AACzC,cAAA,MAAA;AACJ,WAAA;SACD,CAAA;QACDA,IAAI,CAACQ,KAAK,EAAE,CAAA;OACb,CAAA;AACH,KAAC,MAAM;AACL9B,MAAAA,UAAU,CAACwB,SAAS,GAAIC,KAAwC,IAAK;QACnE,MAAM;AAAEC,UAAAA,IAAAA;SAAM,GAAGD,KAAK,CAACE,IAAI,CAAA;AAE3B,QAAA,QAAQD,IAAI;AACV,UAAA,KAAK,SAAS;AACZ,YAAA,IAAI,CAACE,WAAW,CAACH,KAAK,CAACE,IAAI,CAACE,MAAM,EAAEJ,KAAK,CAACF,KAAK,CAAC,CAAC,CAAC,CAAC,CAAA;AACnD,YAAA,MAAA;AACJ,SAAA;OACD,CAAA;AACH,KAAA;AACF,GAAA;AAEAK,EAAAA,WAAWA,CAACC,MAAc,EAAEP,IAAiB,EAAE;IAC7C,IAAI,CAACd,OAAO,CAACuB,GAAG,CAACF,MAAM,EAAEP,IAAI,CAAC,CAAA;IAC9B,IAAI,CAACZ,OAAO,CAACqB,GAAG,CAACF,MAAM,EAAE,IAAIpB,GAAG,EAAE,CAAC,CAAA;AACnCa,IAAAA,IAAI,CAACE,SAAS,GAAIC,KAAwB,IAAK;AAC7C,MAAA,IAAIA,KAAK,CAACC,IAAI,KAAK,OAAO,EAAE;AAC1B,QAAA,IAAI,CAAClB,OAAO,CAACwB,MAAM,CAACH,MAAM,CAAC,CAAA;AAC3B,QAAA,OAAA;AACF,OAAA;MAEA,MAAM;AAAEH,QAAAA,IAAAA;OAAM,GAAGD,KAAK,CAACE,IAAI,CAAA;AAC3B,MAAA,QAAQD,IAAI;AACV,QAAA,KAAK,OAAO;AACV,UAAA,IAAI,CAACO,YAAY,CAACR,KAAK,CAACE,IAAI,CAAC,CAAA;AAC7B,UAAA,MAAA;AACF,QAAA,KAAK,SAAS;UACZ,KAAK,IAAI,CAACO,OAAO,CAACC,cAAc,CAACV,KAAK,CAACE,IAAI,CAAC,CAAC,CAAA;AAC7C,UAAA,MAAA;AACJ,OAAA;KACD,CAAA;AACH,GAAA;EAEAM,YAAYA,CAACR,KAAqB,EAAE;IAClC,MAAM;MAAEI,MAAM;AAAEO,MAAAA,EAAAA;AAAG,KAAC,GAAGX,KAAK,CAAA;AAC5B,IAAA,MAAMY,MAAM,GAAG,IAAI,CAAC3B,OAAO,CAAC4B,GAAG,CAACT,MAAM,CAAC,CAAES,GAAG,CAACF,EAAE,CAAC,CAAA;AAEhD,IAAA,IAAIC,MAAM,EAAE;MACVA,MAAM,CAACE,KAAK,EAAE,CAAA;MACd,IAAI,CAAC7B,OAAO,CAAC4B,GAAG,CAACT,MAAM,CAAC,CAAEG,MAAM,CAACI,EAAE,CAAC,CAAA;AACtC,KAAA;AACF,GAAA;EAEA,MAAMF,OAAOA,CAACT,KAAuB,EAAE;IACrC,MAAM;MAAEI,MAAM;MAAEO,EAAE;AAAET,MAAAA,IAAAA;AAAK,KAAC,GAAGF,KAAK,CAAA;IAElC,IAAI;MACF,MAAMY,MAAM,GAAG,IAAI,CAAC9B,KAAK,CAAC2B,OAAO,CAACP,IAAI,CAAC,CAAA;AACvC,MAAA,IAAI,CAACjB,OAAO,CAAC4B,GAAG,CAACT,MAAM,CAAC,CAAEE,GAAG,CAACK,EAAE,EAAEC,MAAM,CAAC,CAAA;MAEzC,MAAMG,MAAM,GAAG,MAAMH,MAAM,CAAA;MAE3B,IAAI,CAAC7B,OAAO,CAAC8B,GAAG,CAACT,MAAM,CAAC,EAAEY,WAAW,CAAC;AAAEf,QAAAA,IAAI,EAAE,kBAAkB;QAAEU,EAAE;QAAEP,MAAM;QAAEF,IAAI,EAAEe,eAAe,CAACF,MAAM,CAAA;AAAE,OAAC,CAAC,CAAA;KAC/G,CAAC,OAAOG,KAAK,EAAE;AACd,MAAA,IAAIC,YAAY,CAACD,KAAK,CAAC,EAAE,OAAA;MAEzB,IAAI,CAACnC,OAAO,CAAC8B,GAAG,CAACT,MAAM,CAAC,EAAEY,WAAW,CAAC;AAAEf,QAAAA,IAAI,EAAE,gBAAgB;QAAEU,EAAE;QAAEP,MAAM;AAAEF,QAAAA,IAAI,EAAEgB,KAAAA;AAAM,OAAC,CAAC,CAAA;AAC5F,KAAC,SAAS;MACR,IAAI,CAACjC,OAAO,CAAC4B,GAAG,CAACT,MAAM,CAAC,CAAEG,MAAM,CAACI,EAAE,CAAC,CAAA;AACtC,KAAA;AACF,GAAA;AACF,CAAA;AAIA,SAASS,iBAAiBA,CAACC,QAAwC,EAAE;AACnE,EAAA,IAAI,CAACA,QAAQ,EAAE,OAAO,IAAI,CAAA;EAE1B,MAAMC,KAAiC,GAAG,EAAE,CAAA;EAE5C,IAAID,QAAQ,CAACE,OAAO,EAAE;AACpBD,IAAAA,KAAK,CAACC,OAAO,GAAGC,KAAK,CAACC,IAAI,CAACJ,QAAQ,CAACE,OAAO,CAACG,OAAO,EAAE,CAAuB,CAAA;AAC9E,GAAA;AAEAJ,EAAAA,KAAK,CAACK,EAAE,GAAGN,QAAQ,CAACM,EAAE,CAAA;AACtBL,EAAAA,KAAK,CAACM,UAAU,GAAGP,QAAQ,CAACO,UAAU,CAAA;AACtCN,EAAAA,KAAK,CAACO,MAAM,GAAGR,QAAQ,CAACQ,MAAM,CAAA;AAC9BP,EAAAA,KAAK,CAACQ,UAAU,GAAGT,QAAQ,CAACS,UAAU,CAAA;AACtCR,EAAAA,KAAK,CAACrB,IAAI,GAAGoB,QAAQ,CAACpB,IAAI,CAAA;AAC1BqB,EAAAA,KAAK,CAACS,GAAG,GAAGV,QAAQ,CAACU,GAAG,CAAA;AAExB,EAAA,OAAOT,KAAK,CAAA;AACd,CAAA;AAEA,SAASH,YAAYA,CAACD,KAAc,EAAkB;EACpD,OAAOA,KAAK,YAAYc,KAAK,IAAId,KAAK,CAACe,IAAI,KAAK,YAAY,CAAA;AAC9D,CAAA;AAEA,SAAShB,eAAeA,CAAIF,MAAiC,EAAE;AAC7D,EAAA,MAAMmB,WAAW,GAAG;AAClBb,IAAAA,QAAQ,EAAED,iBAAiB,CAACL,MAAM,CAACM,QAAQ,CAAC;IAC5Cc,OAAO,EAAEpB,MAAM,CAACoB,OAAAA;GACjB,CAAA;AAED,EAAA,OAAOD,WAAW,CAAA;AACpB,CAAA;AAEA,SAASxB,cAAcA,CAACV,KAAuB,EAAE;AAC/C,EAAA,IAAIA,KAAK,CAACE,IAAI,CAACqB,OAAO,EAAE;AACtBvB,IAAAA,KAAK,CAACE,IAAI,CAACqB,OAAO,GAAG,IAAIa,OAAO,CAACpC,KAAK,CAACE,IAAI,CAACqB,OAAO,CAAC,CAAA;AACtD,GAAA;AAEA,EAAA,OAAOvB,KAAK,CAAA;AACd;;AC/IO,MAAMqC,YAAY,GAAG,IAAIC,GAAG,CAAC,CAAC,cAAc,EAAE,cAAc,EAAE,cAAc,CAAC,CAAC,CAAA;;AAErF;AACA;AACA;AACA;AACA;AACO,SAASC,eAAeA,CAC7BzD,KAAY,EACZ2B,OAA6B,EAC7B+B,cAAuB,EACvBC,UAA2C,EAClC;EACT,MAAM;AAAEC,IAAAA,YAAAA;AAAa,GAAC,GAAGjC,OAAO,CAAA;EAChC,OACGA,OAAO,CAACkC,EAAE,IAAIN,YAAY,CAACO,GAAG,CAACnC,OAAO,CAACkC,EAAE,CAAC,IAC3CD,YAAY,EAAEG,MAAM,IACpBH,YAAY,EAAEI,gBAAgB,IAC9B,CAACN,cAAc,KACd1D,KAAK,CAACiE,SAAS,IAAIN,UAAU,GAC1B3D,KAAK,CAACiE,SAAS,CAACC,aAAa,CAACP,UAAU,EAAE3D,KAAK,CAAC,IAAIA,KAAK,CAACiE,SAAS,CAACE,aAAa,CAACR,UAAU,EAAE3D,KAAK,CAAC,GACpG,KAAK,CAAC,CAAA;AAEd,CAAA;AAEO,SAASoE,UAAUA,CACxBzC,OAAsC,EACoE;AAC1G,EAAA,OAAO0C,OAAO,CAAC1C,OAAO,CAACkC,EAAE,IAAIN,YAAY,CAACO,GAAG,CAACnC,OAAO,CAACkC,EAAE,CAAC,CAAC,CAAA;AAC5D,CAAA;AAEO,SAASS,gBAAgBA,CAAIC,QAAmC,EAAW;AAChF,EAAA,IAAI,CAACH,UAAU,CAACG,QAAQ,CAAC5C,OAAO,CAAC,EAAE;AACjC,IAAA,OAAO,IAAI,CAAA;AACb,GAAA;AACA;AACA;AACA;;AAEA,EAAA,IAAI4C,QAAQ,CAAC5C,OAAO,CAACkC,EAAE,KAAK,cAAc,IAAIU,QAAQ,CAAChC,QAAQ,EAAEQ,MAAM,KAAK,GAAG,EAAE;AAC/E,IAAA,OAAOwB,QAAQ,CAAClB,OAAO,GAAGjD,MAAM,CAACoE,IAAI,CAACD,QAAQ,CAAClB,OAAO,CAAC,CAACoB,MAAM,GAAG,CAAC,GAAG,KAAK,CAAA;AAC5E,GAAA;AAEA,EAAA,OAAOF,QAAQ,CAAChC,QAAQ,EAAEQ,MAAM,KAAK,GAAG,CAAA;AAC1C,CAAA;AAEA,SAAS2B,gBAAgBA,CAACtC,KAAsC,EAAoD;AAClH,EAAA,OAAOA,KAAK,YAAYuC,cAAc,IAAKvC,KAAK,CAACe,IAAI,KAAK,gBAAgB,IAAIT,KAAK,CAACkC,OAAO,CAACxC,KAAK,CAACyC,MAAM,CAAE,CAAA;AAC5G,CAAA;AAIA;AACO,SAASC,UAAUA,CAAC1C,KAAkB,EAAE;AAC7C,EAAA,MAAM2C,WAAW,GAAGL,gBAAgB,CAACtC,KAAK,CAAC,CAAA;EAE3C,MAAM4C,MAAM,GACVD,WAAW,GAAG,IAAIJ,cAAc,CAACM,eAAe,CAAC7C,KAAK,CAACyC,MAAM,CAAC,EAAEzC,KAAK,CAAC8C,OAAO,CAAC,GAAG,IAAIhC,KAAK,CAACd,KAAK,CAAC8C,OAAO,CAC1F,CAAA;AAChBF,EAAAA,MAAM,CAACG,KAAK,GAAG/C,KAAK,CAAC+C,KAAM,CAAA;AAC3BH,EAAAA,MAAM,CAAC5C,KAAK,GAAGA,KAAK,CAACA,KAAK,CAAA;;AAE1B;AACAhC,EAAAA,MAAM,CAACC,MAAM,CAAC2E,MAAM,EAAE5C,KAAK,CAAC,CAAA;AAE5B,EAAA,OAAO4C,MAAM,CAAA;AACf;;ACtDA;AACA;AACA;AACA;AACA;AACA;AACO,MAAMI,YAA8B,GAAG;AAC5CzD,EAAAA,OAAOA,CAAI0D,OAA4B,EAAEC,IAAe,EAA0D;AAChH;AACA,IAAA,IAAI,CAACD,OAAO,CAAC1D,OAAO,CAAC3B,KAAK,IAAIqF,OAAO,CAAC1D,OAAO,CAACiC,YAAY,GAAG2B,SAAS,CAAC,EAAE;AACvE,MAAA,OAAOD,IAAI,CAACD,OAAO,CAAC1D,OAAO,CAAC,CAAA;AAC9B,KAAA;IAEA,MAAM;AAAE3B,MAAAA,KAAAA;KAAO,GAAGqF,OAAO,CAAC1D,OAAO,CAAA;IACjC,MAAMgC,UAAU,GAAG3D,KAAK,CAACwF,eAAe,CAACC,6BAA6B,CAACJ,OAAO,CAAC1D,OAAO,CAAC,CAAA;AACvF,IAAA,MAAM+D,MAAM,GAAG/B,UAAU,GAAG3D,KAAK,CAAC2F,KAAK,CAACC,WAAW,CAACjC,UAAU,CAAC,GAAG,IAAI,CAAA;AAEtE,IAAA,IAAIA,UAAU,IAAI,CAAC+B,MAAM,EAAE;AACzB;AACA,MAAA,MAAMG,MAAM,GAAI7F,KAAK,CAAwCU,OAAO,CAAA;MACpE,IAAImF,MAAM,EAAElF,OAAO,EAAE;AACnB,QAAA,OAAOkF,MAAM,CAAClF,OAAO,CAClBmF,WAAW,CAACnC,UAAU,CAAC,CACvBoC,IAAI,CAAExB,QAAQ,IAAK;AAClB,UAAA,IAAIA,QAAQ,EAAE;AACZvE,YAAAA,KAAK,CAAC2F,KAAK,CAACK,GAAG,CAACzB,QAAQ,CAAC,CAAA;AAC3B,WAAA;UACA,OAAO0B,eAAe,CAACtC,UAAU,EAAE3D,KAAK,EAAEqF,OAAO,EAAEC,IAAI,CAAC,CAAA;AAC1D,SAAC,CAAC,CACDY,KAAK,CAAEpF,CAAC,IAAK;UACZ,IAAAqF,cAAA,CAAAC,eAAA,EAAA,CAAAC,SAAA,CAAAC,GAAA,CAAAC,KAAA,CAAW,EAAA;AACT;AACAC,YAAAA,OAAO,CAACC,GAAG,CAAC,oDAAoD,EAAE3F,CAAC,CAAC,CAAA;AACtE,WAAA;UACA,OAAOmF,eAAe,CAACtC,UAAU,EAAE3D,KAAK,EAAEqF,OAAO,EAAEC,IAAI,CAAC,CAAA;AAC1D,SAAC,CAAC,CAAA;AACN,OAAA;AACF,KAAA;IAEA,OAAOW,eAAe,CAACtC,UAAU,EAAE3D,KAAK,EAAEqF,OAAO,EAAEC,IAAI,CAAC,CAAA;AAC1D,GAAA;AACF,EAAC;AAED,SAASW,eAAeA,CACtBtC,UAA2C,EAC3C3D,KAAY,EACZqF,OAA4B,EAC5BC,IAAe,EACf;AACA,EAAA,MAAMI,MAAM,GAAG/B,UAAU,GAAG3D,KAAK,CAAC2F,KAAK,CAACC,WAAW,CAACjC,UAAU,CAAC,GAAG,IAAI,CAAA;AACtE;AACA;AACA,EAAA,IAAIF,eAAe,CAACzD,KAAK,EAAEqF,OAAO,CAAC1D,OAAO,EAAE,CAAC,CAAC+D,MAAM,EAAE/B,UAAU,CAAC,EAAE;AACjE,IAAA,OAAO+C,sBAAsB,CAACpB,IAAI,EAAED,OAAO,EAAE1B,UAAU,CAAC,CAAA;AAC1D,GAAA;EAEAwC,cAAA,CAAAC,eAAA,EAAAC,CAAAA,SAAA,CAAAC,GAAA,CAAAC,KAAA,CAAA,GAAA,CAAAI,IAAA,IAAA;AAAA,IAAA,IAAA,CAAAA,IAAA,EAAA;MAAA,MAAAzD,IAAAA,KAAA,CAAQ,CAAwC,uCAAA,CAAA,CAAA,CAAA;AAAA,KAAA;AAAA,GAAA,EAAEwC,MAAM,CAAA,GAAA,EAAA,CAAA;AACxDL,EAAAA,OAAO,CAACuB,WAAW,CAAClB,MAAM,CAACnD,QAAQ,CAAC,CAAA;EAEpC,IAAI,OAAO,IAAImD,MAAM,EAAE;AACrB,IAAA,MAAMA,MAAM,CAAA;AACd,GAAA;AAEA,EAAA,OAAOmB,kBAAkB,CAAI7G,KAAK,EAAE0F,MAAM,CAACrC,OAA+B,CAAC,CAAA;AAC7E,CAAA;AAEA,SAASwD,kBAAkBA,CAAI7G,KAAY,EAAEuE,QAAqC,EAAK;EACrF,IAAI,CAACA,QAAQ,EAAE;AACb,IAAA,OAAOA,QAAQ,CAAA;AACjB,GAAA;EAEA,IAAI7B,KAAK,CAACkC,OAAO,CAACL,QAAQ,CAACnD,IAAI,CAAC,EAAE;IAChC,MAAMA,IAAI,GAAGmD,QAAQ,CAACnD,IAAI,CAAC0F,GAAG,CAAEnD,UAAU,IAAK;AAC7C,MAAA,OAAO3D,KAAK,CAAC2F,KAAK,CAACoB,IAAI,CAACpD,UAAU,CAAC,CAAA;AACrC,KAAC,CAAC,CAAA;IAEF,OAAOvD,MAAM,CAACC,MAAM,CAAC,EAAE,EAAEkE,QAAQ,EAAE;AAAEnD,MAAAA,IAAAA;AAAK,KAAC,CAAC,CAAA;AAC9C,GAAC,MAAM;AACL,IAAA,MAAMA,IAAI,GAAImD,QAAQ,CAACnD,IAAI,GAAGpB,KAAK,CAAC2F,KAAK,CAACoB,IAAI,CAACxC,QAAQ,CAACnD,IAAI,CAAC,GAAG,IAAU,CAAA;IAC1E,OAAOhB,MAAM,CAACC,MAAM,CAAC,EAAE,EAAEkE,QAAQ,EAAE;AAAEnD,MAAAA,IAAAA;AAAK,KAAC,CAAC,CAAA;AAC9C,GAAA;AACF,CAAA;AAEA,SAAS4F,yBAAyBA,CAChChH,KAAY,EACZuE,QAA+E,EAC/E0C,gBAAuC,EACvC;AACA,EAAA,MAAMpB,MAAM,GAAI7F,KAAK,CAAwCU,OAAO,CAAA;AAEpE,EAAA,IAAI,CAACmF,MAAM,EAAElF,OAAO,EAAE;AACpB,IAAA,OAAA;AACF,GAAA;AAEA,EAAA,IAAI,CAAC4D,QAAQ,IAAI0C,gBAAgB,EAAE;AACjC;IACA,KAAKpB,MAAM,CAAClF,OAAO,CAACuG,YAAY,CAACD,gBAAgB,EAAGE,kBAAkB,IAAK;AACzE,MAAA,OAAOnH,KAAK,CAAC2F,KAAK,CAACoB,IAAI,CAACI,kBAAkB,CAAC,CAAA;AAC7C,KAAC,CAAC,CAAA;GACH,MAAM,IAAI5C,QAAQ,EAAE;IACnB,KAAKsB,MAAM,CAAClF,OAAO,CAACyG,WAAW,CAAC7C,QAAQ,EAAG4C,kBAAkB,IAAK;AAChE,MAAA,OAAOnH,KAAK,CAAC2F,KAAK,CAACoB,IAAI,CAACI,kBAAkB,CAAC,CAAA;AAC7C,KAAC,CAAC,CAAA;AACJ,GAAA;AACF,CAAA;AAEA,SAASE,qBAAqBA,CAC5BrH,KAAY,EACZ2B,OAAuC,EACvC4C,QAAmC,EACnC;EACA,IAAIhC,QAAqC,GAAG,IAAI,CAAA;AAChD,EAAA,IAAI6B,UAAU,CAACzC,OAAO,CAAC,EAAE;AACvB,IAAA,MAAM2F,MAAM,GAAG3F,OAAO,CAACP,IAAI,EAAEkG,MAAM,IAAI3F,OAAO,CAAC4F,OAAO,GAAG,CAAC,CAAC,CAAA;AAC3D,IAAA,IAAID,MAAM,EAAE;MACV/E,QAAQ,GAAGvC,KAAK,CAAC2F,KAAK,CAAC6B,SAAS,CAACF,MAAM,EAAE/C,QAAQ,CAAyB,CAAA;;AAE1E;AACA;AACA;AACF,KAAC,MAAM,IAAID,gBAAgB,CAACC,QAAQ,CAAC,EAAE;MACrChC,QAAQ,GAAGvC,KAAK,CAAC2F,KAAK,CAACK,GAAG,CAACzB,QAAQ,CAAyB,CAAA;AAC5DyC,MAAAA,yBAAyB,CAAChH,KAAK,EAAE,IAAI,EAAEuC,QAAQ,CAAC,CAAA;AAClD,KAAA;AACF,GAAC,MAAM;IACLA,QAAQ,GAAGvC,KAAK,CAAC2F,KAAK,CAACK,GAAG,CAACzB,QAAQ,CAAyB,CAAA;IAE5D,IAAIhC,QAAQ,CAACkF,GAAG,EAAE;MAChB,MAAM9D,UAAU,GAAG3D,KAAK,CAACwF,eAAe,CAACC,6BAA6B,CAAC9D,OAAO,CAAC,CAAA;MAC/E,MAAM+F,IAAI,GAAG1H,KAAK,CAAC2F,KAAK,CAACC,WAAW,CAACjC,UAAW,CAAC,CAAA;AACjDqD,MAAAA,yBAAyB,CAAChH,KAAK,EAAE0H,IAAI,CAAC,CAAA;AACxC,KAAA;AACF,GAAA;AACA,EAAA,OAAOb,kBAAkB,CAAC7G,KAAK,EAAEuC,QAAQ,CAAC,CAAA;AAC5C,CAAA;AAEA,SAASoF,kBAAkBA,CACzB3H,KAAY,EACZ2B,OAAuC,EACvCgC,UAA2C,EAC3CY,QAAmC,EAChC;AACH,EAAA,IAAIhC,QAA8B,CAAA;EAClCvC,KAAK,CAAC4H,KAAK,CAAC,MAAM;IAChBrF,QAAQ,GAAG8E,qBAAqB,CAAIrH,KAAK,EAAE2B,OAAO,EAAE4C,QAAQ,CAAyB,CAAA;AACvF,GAAC,CAAC,CAAA;AAEF,EAAA,IAAIvE,KAAK,CAACiE,SAAS,EAAE4D,UAAU,EAAE;AAC/B7H,IAAAA,KAAK,CAACiE,SAAS,CAAC4D,UAAU,CAAClG,OAAO,EAAE4C,QAAQ,CAAChC,QAAQ,EAAEoB,UAAU,EAAE3D,KAAK,CAAC,CAAA;AAC3E,GAAA;AAEA,EAAA,OAAOuC,QAAQ,CAAA;AACjB,CAAA;AAEA,SAASuF,mBAAmBA,CAC1B9H,KAAY,EACZ2B,OAAuC,EACvCS,KAAiC,EACjC;AACA,EAAA,IAAIgC,UAAU,CAACzC,OAAO,CAAC,EAAE;AACvB;AACA;AACA,IAAA,MAAMkD,MAAM,GACVzC,KAAK,IACLA,KAAK,CAACiB,OAAO,IACb,OAAOjB,KAAK,CAACiB,OAAO,KAAK,QAAQ,IACjC,QAAQ,IAAIjB,KAAK,CAACiB,OAAO,IACzBX,KAAK,CAACkC,OAAO,CAACxC,KAAK,CAACiB,OAAO,CAACwB,MAAM,CAAC,GAC9BzC,KAAK,CAACiB,OAAO,CAACwB,MAAM,GACrBkD,SAAS,CAAA;AAEf,IAAA,MAAMT,MAAM,GAAG3F,OAAO,CAACP,IAAI,EAAEkG,MAAM,IAAI3F,OAAO,CAAC4F,OAAO,GAAG,CAAC,CAAC,CAAA;IAE3DvH,KAAK,CAAC2F,KAAK,CAACqC,iBAAiB,CAACV,MAAM,EAAEzC,MAAM,CAAC,CAAA;AAC/C,GAAC,MAAM;IACL,MAAMlB,UAAU,GAAG3D,KAAK,CAACwF,eAAe,CAACC,6BAA6B,CAAC9D,OAAO,CAAC,CAAA;AAC/E,IAAA,IAAIgC,UAAU,EAAE;AACdqD,MAAAA,yBAAyB,CAAChH,KAAK,EAAEoC,KAAuD,CAAC,CAAA;AAC3F,KAAA;AACA,IAAA,OAAOpC,KAAK,CAAC2F,KAAK,CAACK,GAAG,CAAC5D,KAAK,CAAC,CAAA;AAC/B,GAAA;AACF,CAAA;AAEA,SAAS6F,gBAAgBA,CACvBjI,KAAY,EACZ2B,OAAuC,EACvCgC,UAA2C,EAC3CvB,KAAiC,EAC1B;AACP,EAAA,IAAIT,OAAO,CAACuG,MAAM,EAAEC,OAAO,EAAE;AAC3B,IAAA,MAAM/F,KAAK,CAAA;AACb,GAAA;AACA,EAAA,IAAIG,QAA2C,CAAA;EAC/CvC,KAAK,CAAC4H,KAAK,CAAC,MAAM;IAChBrF,QAAQ,GAAGuF,mBAAmB,CAAC9H,KAAK,EAAE2B,OAAO,EAAES,KAAK,CAAC,CAAA;AACvD,GAAC,CAAC,CAAA;AAEF,EAAA,IAAIuB,UAAU,IAAI3D,KAAK,CAACiE,SAAS,EAAE4D,UAAU,EAAE;AAC7C7H,IAAAA,KAAK,CAACiE,SAAS,CAAC4D,UAAU,CAAClG,OAAO,EAAES,KAAK,CAACG,QAAQ,EAAEoB,UAAU,EAAE3D,KAAK,CAAC,CAAA;AACxE,GAAA;AAEA,EAAA,IAAIoE,UAAU,CAACzC,OAAO,CAAC,EAAE;AACvB,IAAA,MAAMS,KAAK,CAAA;AACb,GAAA;AAEA,EAAA,MAAMgG,QAAQ,GAAGtD,UAAU,CAAC1C,KAAK,CAAC,CAAA;EAClCgG,QAAQ,CAAC/E,OAAO,GAAGd,QAAS,CAAA;AAC5B,EAAA,MAAM6F,QAAQ,CAAA;AAChB,CAAA;AAEA,SAAS1B,sBAAsBA,CAC7BpB,IAAe,EACfD,OAA4B,EAC5B1B,UAA2C,EAC/B;EACZ,MAAM;AAAEhC,IAAAA,OAAAA;AAAQ,GAAC,GAAG0D,OAAO,CAAA;EAC3B,MAAM;AAAErF,IAAAA,KAAAA;GAAO,GAAGqF,OAAO,CAAC1D,OAAO,CAAA;AAEjC,EAAA,IAAIyC,UAAU,CAACzC,OAAO,CAAC,EAAE;AACvB;AACA,IAAA,MAAM2F,MAAM,GAAG3F,OAAO,CAACP,IAAI,EAAEkG,MAAM,IAAI3F,OAAO,CAAC4F,OAAO,GAAG,CAAC,CAAC,CAAA;IAC3DpB,cAAA,CAAAC,eAAA,EAAAC,CAAAA,SAAA,CAAAC,GAAA,CAAAC,KAAA,CAAA,GAAA,CAAAI,IAAA,IAAA;AAAA,MAAA,IAAA,CAAAA,IAAA,EAAA;AAAA,QAAA,MAAA,IAAAzD,KAAA,CAAQ,CAAA,sDAAA,EAAwDvB,OAAO,CAACkC,EAAG,CAAS,QAAA,CAAA,CAAA,CAAA;AAAA,OAAA;AAAA,KAAA,EAAEyD,MAAM,CAAA,GAAA,EAAA,CAAA;AAC5F,IAAA,IAAIA,MAAM,EAAE;MACVtH,KAAK,CAAC2F,KAAK,CAAC0C,UAAU,CAACf,MAAM,EAAEjC,OAAO,CAAC,CAAA;AACzC,KAAA;AACF,GAAA;AAEA,EAAA,IAAIrF,KAAK,CAACiE,SAAS,EAAEqE,WAAW,EAAE;IAChCtI,KAAK,CAACiE,SAAS,CAACqE,WAAW,CAAC3G,OAAO,EAAEgC,UAAU,EAAE3D,KAAK,CAAC,CAAA;AACzD,GAAA;AAEA,EAAA,OAAOsF,IAAI,CAAC3D,OAAO,CAAC,CAACoE,IAAI,CACtBxB,QAAQ,IAAKoD,kBAAkB,CAAC3H,KAAK,EAAE2B,OAAO,EAAEgC,UAAU,EAAEY,QAAQ,CAAC,EACrEnC,KAAiC,IAAK6F,gBAAgB,CAACjI,KAAK,EAAE2B,OAAO,EAAEgC,UAAU,EAAEvB,KAAK,CAC3F,CAAC,CAAA;AACH;;;;"}
@@ -1,243 +1 @@
1
- const WARP_DRIVE_STORAGE_FILE_NAME = 'warp-drive_document-storage';
2
- const WARP_DRIVE_STORAGE_VERSION = 1;
3
-
4
- /**
5
- * DocumentStorage is specifically designed around WarpDrive Cache and Request concepts.
6
- *
7
- * CacheFileDocument is a StructuredDocument (request response) whose `content` is
8
- * the ResourceDocument returned by inserting the request into a Store's Cache.
9
- */
10
-
11
- /**
12
- * A CacheDocument is a reconstructed request response that rehydrates ResourceDocument
13
- * with the associated resources based on their identifiers.
14
- */
15
-
16
- class InternalDocumentStorage {
17
- constructor(options) {
18
- this.options = options;
19
- this._lastModified = 0;
20
- this._invalidated = true;
21
- this._fileHandle = this._open(options.scope);
22
- this._channel = Object.assign(new BroadcastChannel(options.scope), {
23
- onmessage: this._onMessage.bind(this)
24
- });
25
- }
26
- _onMessage(_event) {
27
- this._invalidated = true;
28
- }
29
- async _open(scope) {
30
- const directoryHandle = await navigator.storage.getDirectory();
31
- const fileHandle = await directoryHandle.getFileHandle(scope, {
32
- create: true
33
- });
34
- return fileHandle;
35
- }
36
- async _read() {
37
- if (this._filePromise) {
38
- return this._filePromise;
39
- }
40
- if (this._invalidated) {
41
- const updateFile = async () => {
42
- const fileHandle = await this._fileHandle;
43
- const file = await fileHandle.getFile();
44
- const lastModified = file.lastModified;
45
- if (lastModified === this._lastModified && this._cache) {
46
- return this._cache;
47
- }
48
- const contents = await file.text();
49
- const cache = contents ? JSON.parse(contents) : {
50
- documents: [],
51
- resources: []
52
- };
53
- const documents = new Map(cache.documents);
54
- const resources = new Map(cache.resources);
55
- const cacheMap = {
56
- documents,
57
- resources
58
- };
59
- this._cache = cacheMap;
60
- this._invalidated = false;
61
- this._lastModified = lastModified;
62
- return cacheMap;
63
- };
64
- this._filePromise = updateFile();
65
- await this._filePromise;
66
- this._filePromise = null;
67
- }
68
- return this._cache;
69
- }
70
- async _patch(documentKey, document, updatedResources) {
71
- const fileHandle = await this._fileHandle;
72
- // secure a lock before getting latest state
73
- const writable = await fileHandle.createWritable();
74
- const cache = await this._read();
75
- cache.documents.set(documentKey, document);
76
- updatedResources.forEach((resource, key) => {
77
- cache.resources.set(key, resource);
78
- });
79
- const documents = [...cache.documents.entries()];
80
- const resources = [...cache.resources.entries()];
81
- const cacheFile = {
82
- documents,
83
- resources
84
- };
85
- await writable.write(JSON.stringify(cacheFile));
86
- await writable.close();
87
- this._channel.postMessage({
88
- type: 'patch',
89
- key: documentKey,
90
- resources: [...updatedResources.keys()]
91
- });
92
- }
93
- async getDocument(key) {
94
- const cache = await this._read();
95
- // clone the document to avoid leaking the internal cache
96
- const document = structuredClone(cache.documents.get(key.lid));
97
- if (!document) {
98
- return null;
99
- }
100
-
101
- // expand the document with the resources
102
- if (document.content) {
103
- if (docHasData(document.content)) {
104
- let data = null;
105
- if (Array.isArray(document.content.data)) {
106
- data = document.content.data.map(resourceIdentifier => {
107
- const resource = cache.resources.get(resourceIdentifier.lid);
108
- if (!resource) {
109
- throw new Error(`Resource not found for ${resourceIdentifier.lid}`);
110
- }
111
-
112
- // clone the resource to avoid leaking the internal cache
113
- return structuredClone(resource);
114
- });
115
- } else if (document.content.data) {
116
- const resource = cache.resources.get(document.content.data.lid);
117
- if (!resource) {
118
- throw new Error(`Resource not found for ${document.content.data.lid}`);
119
- }
120
-
121
- // clone the resource to avoid leaking the internal cache
122
- data = structuredClone(resource);
123
- }
124
- if (document.content.included) {
125
- const included = document.content.included.map(resourceIdentifier => {
126
- const resource = cache.resources.get(resourceIdentifier.lid);
127
- if (!resource) {
128
- throw new Error(`Resource not found for ${resourceIdentifier.lid}`);
129
- }
130
-
131
- // clone the resource to avoid leaking the internal cache
132
- return structuredClone(resource);
133
- });
134
- document.content.included = included;
135
- }
136
- document.content.data = data;
137
- }
138
- }
139
- return document;
140
- }
141
- async putDocument(document, resourceCollector) {
142
- const resources = new Map();
143
- if (!document.content) {
144
- throw new Error(`Document content is missing, only finalized documents can be stored`);
145
- }
146
- if (!document.content.lid) {
147
- throw new Error(`Document content is missing a lid, only documents with a cache-key can be stored`);
148
- }
149
- if (docHasData(document.content)) {
150
- if (Array.isArray(document.content.data)) {
151
- document.content.data.forEach(resourceIdentifier => {
152
- const resource = resourceCollector(resourceIdentifier);
153
- resources.set(resourceIdentifier.lid, structuredClone(resource));
154
- });
155
- } else if (document.content.data) {
156
- const resource = resourceCollector(document.content.data);
157
- resources.set(document.content.data.lid, structuredClone(resource));
158
- }
159
- if (document.content.included) {
160
- document.content.included.forEach(resourceIdentifier => {
161
- const resource = resourceCollector(resourceIdentifier);
162
- resources.set(resourceIdentifier.lid, structuredClone(resource));
163
- });
164
- }
165
- }
166
- await this._patch(document.content.lid, structuredClone(document), resources);
167
- }
168
- async clear(reset) {
169
- const fileHandle = await this._fileHandle;
170
- const writable = await fileHandle.createWritable();
171
- await writable.write('');
172
- await writable.close();
173
- this._invalidated = true;
174
- this._lastModified = 0;
175
- this._cache = null;
176
- this._filePromise = null;
177
- this._channel.postMessage({
178
- type: 'clear'
179
- });
180
- if (!reset) {
181
- this._channel.close();
182
- this._channel = null;
183
- if (!this.options.isolated) {
184
- Storages.delete(this.options.scope);
185
- }
186
- }
187
- }
188
- }
189
- function docHasData(doc) {
190
- return 'data' in doc;
191
- }
192
- const Storages = new Map();
193
-
194
- /**
195
- * DocumentStorage is a wrapper around the StorageManager API that provides
196
- * a simple interface for reading and updating documents and requests.
197
- *
198
- * Some goals for this experiment:
199
- *
200
- * - optimize for storing requests/documents
201
- * - optimize for storing resources
202
- * - optimize for looking up resources associated to a document
203
- * - optimize for notifying cross-tab when data is updated
204
- *
205
- * optional features:
206
- *
207
- * - support for offline mode
208
- * - ?? support for relationship based cache traversal
209
- * - a way to index records by type + another field (e.g updatedAt/createAt/name)
210
- * such that simple queries can be done without having to scan all records
211
- */
212
- class DocumentStorage {
213
- constructor(options = {}) {
214
- options.isolated = options.isolated ?? false;
215
- options.scope = options.scope ?? 'default';
216
- const fileName = `${WARP_DRIVE_STORAGE_FILE_NAME}@version_${WARP_DRIVE_STORAGE_VERSION}:${options.scope}`;
217
- if (!options.isolated && Storages.has(fileName)) {
218
- const storage = Storages.get(fileName);
219
- if (storage) {
220
- this._storage = storage.deref();
221
- return;
222
- }
223
- }
224
- const storage = new InternalDocumentStorage({
225
- scope: fileName,
226
- isolated: options.isolated
227
- });
228
- this._storage = storage;
229
- if (!options.isolated) {
230
- Storages.set(fileName, new WeakRef(storage));
231
- }
232
- }
233
- getDocument(key) {
234
- return this._storage.getDocument(key);
235
- }
236
- putDocument(document, resourceCollector) {
237
- return this._storage.putDocument(document, resourceCollector);
238
- }
239
- clear(reset) {
240
- return this._storage.clear(reset);
241
- }
242
- }
243
- export { DocumentStorage };
1
+ export { D as DocumentStorage } from "./index-Cn3o840t";