@kaspernj/api-maker 1.0.2151 → 1.0.2153

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -143,6 +143,24 @@ current_command.increment!
143
143
  current_command.progress = 0.5
144
144
  ```
145
145
 
146
+ ### Waiting for websocket requests to go idle
147
+
148
+ For app/test cleanup flows that need to avoid leaking in-flight websocket command work into later state, use the realtime reset helper that waits for pending websocket requests before disconnecting the realtime runtime:
149
+
150
+ ```js
151
+ import {waitForRealtimeRuntimeIdleAndReset} from "@kaspernj/api-maker/build/reset-realtime-runtime-state"
152
+
153
+ await waitForRealtimeRuntimeIdleAndReset({timeoutMs: 5000})
154
+ ```
155
+
156
+ The synchronous default export still exists for immediate resets where waiting is not required:
157
+
158
+ ```js
159
+ import resetRealtimeRuntimeState from "@kaspernj/api-maker/build/reset-realtime-runtime-state"
160
+
161
+ resetRealtimeRuntimeState()
162
+ ```
163
+
146
164
  ## Realtime model events
147
165
 
148
166
  For model-specific websocket events in React/ShapeComponent UI code, prefer `useModelEvent` over manual `ModelEvents.connect(...)` subscriptions. The hook handles subscription lifecycle automatically and avoids leaked listeners.
@@ -1,3 +1,14 @@
1
+ export default resetRealtimeRuntimeState;
2
+ /**
3
+ * Waits for websocket-backed Api Maker requests to complete before resetting the realtime runtime.
4
+ *
5
+ * @param {object} [args]
6
+ * @param {number} [args.timeoutMs]
7
+ * @returns {Promise<void>}
8
+ */
9
+ export function waitForRealtimeRuntimeIdleAndReset({ timeoutMs }?: {
10
+ timeoutMs?: number;
11
+ }): Promise<void>;
1
12
  /** @returns {void} */
2
- export default function resetRealtimeRuntimeState(): void;
13
+ declare function resetRealtimeRuntimeState(): void;
3
14
  //# sourceMappingURL=reset-realtime-runtime-state.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"reset-realtime-runtime-state.d.ts","sourceRoot":"/src/","sources":["reset-realtime-runtime-state.js"],"names":[],"mappings":"AAIA,sBAAsB;AACtB,qDADc,IAAI,CAKjB"}
1
+ {"version":3,"file":"reset-realtime-runtime-state.d.ts","sourceRoot":"/src/","sources":["reset-realtime-runtime-state.js"],"names":[],"mappings":";AAWA;;;;;;GAMG;AACH,mEAHG;IAAsB,SAAS,GAAvB,MAAM;CACd,GAAU,OAAO,CAAC,IAAI,CAAC,CAKzB;AAjBD,sBAAsB;AACtB,8CADc,IAAI,CAKjB"}
@@ -2,9 +2,22 @@ import ApiMakerCableConnectionPool from "./cable-connection-pool.js";
2
2
  import ApiMakerWebsocketRequestClient from "./websocket-request-client.js";
3
3
  import { resetChannelsConsumer } from "./channels-consumer.js";
4
4
  /** @returns {void} */
5
- export default function resetRealtimeRuntimeState() {
5
+ const resetRealtimeRuntimeState = () => {
6
6
  ApiMakerWebsocketRequestClient.resetCurrent();
7
7
  ApiMakerCableConnectionPool.resetCurrent();
8
8
  resetChannelsConsumer();
9
- }
10
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVzZXQtcmVhbHRpbWUtcnVudGltZS1zdGF0ZS5qcyIsInNvdXJjZVJvb3QiOiIvc3JjLyIsInNvdXJjZXMiOlsicmVzZXQtcmVhbHRpbWUtcnVudGltZS1zdGF0ZS5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLDJCQUEyQixNQUFNLDRCQUE0QixDQUFBO0FBQ3BFLE9BQU8sOEJBQThCLE1BQU0sK0JBQStCLENBQUE7QUFDMUUsT0FBTyxFQUFDLHFCQUFxQixFQUFDLE1BQU0sd0JBQXdCLENBQUE7QUFFNUQsc0JBQXNCO0FBQ3RCLE1BQU0sQ0FBQyxPQUFPLFVBQVUseUJBQXlCO0lBQy9DLDhCQUE4QixDQUFDLFlBQVksRUFBRSxDQUFBO0lBQzdDLDJCQUEyQixDQUFDLFlBQVksRUFBRSxDQUFBO0lBQzFDLHFCQUFxQixFQUFFLENBQUE7QUFDekIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBBcGlNYWtlckNhYmxlQ29ubmVjdGlvblBvb2wgZnJvbSBcIi4vY2FibGUtY29ubmVjdGlvbi1wb29sLmpzXCJcbmltcG9ydCBBcGlNYWtlcldlYnNvY2tldFJlcXVlc3RDbGllbnQgZnJvbSBcIi4vd2Vic29ja2V0LXJlcXVlc3QtY2xpZW50LmpzXCJcbmltcG9ydCB7cmVzZXRDaGFubmVsc0NvbnN1bWVyfSBmcm9tIFwiLi9jaGFubmVscy1jb25zdW1lci5qc1wiXG5cbi8qKiBAcmV0dXJucyB7dm9pZH0gKi9cbmV4cG9ydCBkZWZhdWx0IGZ1bmN0aW9uIHJlc2V0UmVhbHRpbWVSdW50aW1lU3RhdGUoKSB7XG4gIEFwaU1ha2VyV2Vic29ja2V0UmVxdWVzdENsaWVudC5yZXNldEN1cnJlbnQoKVxuICBBcGlNYWtlckNhYmxlQ29ubmVjdGlvblBvb2wucmVzZXRDdXJyZW50KClcbiAgcmVzZXRDaGFubmVsc0NvbnN1bWVyKClcbn1cbiJdfQ==
9
+ };
10
+ /**
11
+ * Waits for websocket-backed Api Maker requests to complete before resetting the realtime runtime.
12
+ *
13
+ * @param {object} [args]
14
+ * @param {number} [args.timeoutMs]
15
+ * @returns {Promise<void>}
16
+ */
17
+ const waitForRealtimeRuntimeIdleAndReset = async ({ timeoutMs = 5000 } = {}) => {
18
+ await ApiMakerWebsocketRequestClient.current().waitForIdle({ timeoutMs });
19
+ resetRealtimeRuntimeState();
20
+ };
21
+ export { waitForRealtimeRuntimeIdleAndReset };
22
+ export default resetRealtimeRuntimeState;
23
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVzZXQtcmVhbHRpbWUtcnVudGltZS1zdGF0ZS5qcyIsInNvdXJjZVJvb3QiOiIvc3JjLyIsInNvdXJjZXMiOlsicmVzZXQtcmVhbHRpbWUtcnVudGltZS1zdGF0ZS5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLDJCQUEyQixNQUFNLDRCQUE0QixDQUFBO0FBQ3BFLE9BQU8sOEJBQThCLE1BQU0sK0JBQStCLENBQUE7QUFDMUUsT0FBTyxFQUFDLHFCQUFxQixFQUFDLE1BQU0sd0JBQXdCLENBQUE7QUFFNUQsc0JBQXNCO0FBQ3RCLE1BQU0seUJBQXlCLEdBQUcsR0FBRyxFQUFFO0lBQ3JDLDhCQUE4QixDQUFDLFlBQVksRUFBRSxDQUFBO0lBQzdDLDJCQUEyQixDQUFDLFlBQVksRUFBRSxDQUFBO0lBQzFDLHFCQUFxQixFQUFFLENBQUE7QUFDekIsQ0FBQyxDQUFBO0FBRUQ7Ozs7OztHQU1HO0FBQ0gsTUFBTSxrQ0FBa0MsR0FBRyxLQUFLLEVBQUMsRUFBQyxTQUFTLEdBQUcsSUFBSSxFQUFDLEdBQUcsRUFBRSxFQUFFLEVBQUU7SUFDMUUsTUFBTSw4QkFBOEIsQ0FBQyxPQUFPLEVBQUUsQ0FBQyxXQUFXLENBQUMsRUFBQyxTQUFTLEVBQUMsQ0FBQyxDQUFBO0lBQ3ZFLHlCQUF5QixFQUFFLENBQUE7QUFDN0IsQ0FBQyxDQUFBO0FBRUQsT0FBTyxFQUFDLGtDQUFrQyxFQUFDLENBQUE7QUFDM0MsZUFBZSx5QkFBeUIsQ0FBQSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBBcGlNYWtlckNhYmxlQ29ubmVjdGlvblBvb2wgZnJvbSBcIi4vY2FibGUtY29ubmVjdGlvbi1wb29sLmpzXCJcbmltcG9ydCBBcGlNYWtlcldlYnNvY2tldFJlcXVlc3RDbGllbnQgZnJvbSBcIi4vd2Vic29ja2V0LXJlcXVlc3QtY2xpZW50LmpzXCJcbmltcG9ydCB7cmVzZXRDaGFubmVsc0NvbnN1bWVyfSBmcm9tIFwiLi9jaGFubmVscy1jb25zdW1lci5qc1wiXG5cbi8qKiBAcmV0dXJucyB7dm9pZH0gKi9cbmNvbnN0IHJlc2V0UmVhbHRpbWVSdW50aW1lU3RhdGUgPSAoKSA9PiB7XG4gIEFwaU1ha2VyV2Vic29ja2V0UmVxdWVzdENsaWVudC5yZXNldEN1cnJlbnQoKVxuICBBcGlNYWtlckNhYmxlQ29ubmVjdGlvblBvb2wucmVzZXRDdXJyZW50KClcbiAgcmVzZXRDaGFubmVsc0NvbnN1bWVyKClcbn1cblxuLyoqXG4gKiBXYWl0cyBmb3Igd2Vic29ja2V0LWJhY2tlZCBBcGkgTWFrZXIgcmVxdWVzdHMgdG8gY29tcGxldGUgYmVmb3JlIHJlc2V0dGluZyB0aGUgcmVhbHRpbWUgcnVudGltZS5cbiAqXG4gKiBAcGFyYW0ge29iamVjdH0gW2FyZ3NdXG4gKiBAcGFyYW0ge251bWJlcn0gW2FyZ3MudGltZW91dE1zXVxuICogQHJldHVybnMge1Byb21pc2U8dm9pZD59XG4gKi9cbmNvbnN0IHdhaXRGb3JSZWFsdGltZVJ1bnRpbWVJZGxlQW5kUmVzZXQgPSBhc3luYyh7dGltZW91dE1zID0gNTAwMH0gPSB7fSkgPT4ge1xuICBhd2FpdCBBcGlNYWtlcldlYnNvY2tldFJlcXVlc3RDbGllbnQuY3VycmVudCgpLndhaXRGb3JJZGxlKHt0aW1lb3V0TXN9KVxuICByZXNldFJlYWx0aW1lUnVudGltZVN0YXRlKClcbn1cblxuZXhwb3J0IHt3YWl0Rm9yUmVhbHRpbWVSdW50aW1lSWRsZUFuZFJlc2V0fVxuZXhwb3J0IGRlZmF1bHQgcmVzZXRSZWFsdGltZVJ1bnRpbWVTdGF0ZVxuIl19
@@ -4,12 +4,25 @@ export default class ApiMakerWebsocketRequestClient {
4
4
  static current(): ApiMakerWebsocketRequestClient;
5
5
  /** @returns {void} */
6
6
  static resetCurrent(): void;
7
+ idleWaiters: any[];
7
8
  currentRequestId: number;
8
9
  skipReconnect: boolean;
9
10
  pendingRequests: {};
10
11
  pendingRequestsByFingerprint: {};
11
12
  responseCache: {};
12
13
  subscriptionState: string;
14
+ /** @returns {number} */
15
+ pendingRequestsCount(): number;
16
+ /**
17
+ * Waits for all in-flight websocket requests to complete.
18
+ *
19
+ * @param {object} [args]
20
+ * @param {number} [args.timeoutMs]
21
+ * @returns {Promise<void>}
22
+ */
23
+ waitForIdle({ timeoutMs }?: {
24
+ timeoutMs?: number;
25
+ }): Promise<void>;
13
26
  /**
14
27
  * @param {object} args
15
28
  * @param {boolean} [args.cacheResponse]
@@ -67,6 +80,8 @@ export default class ApiMakerWebsocketRequestClient {
67
80
  */
68
81
  rejectPendingRequests(error: Error): void;
69
82
  /** @returns {void} */
83
+ resolveIdleWaitersIfIdle(): void;
84
+ /** @returns {void} */
70
85
  reset(): void;
71
86
  }
72
87
  //# sourceMappingURL=websocket-request-client.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"websocket-request-client.d.ts","sourceRoot":"/src/","sources":["websocket-request-client.js"],"names":[],"mappings":"AAOA,8EAA8E;AAC9E;IACE,gDAAgD;IAChD,kBADc,8BAA8B,CAO3C;IAED,sBAAsB;IACtB,uBADc,IAAI,CAQjB;IAIC,yBAAyB;IACzB,uBAA0B;IAC1B,oBAAyB;IACzB,iCAAsC;IACtC,kBAAuB;IACvB,0BAA8B;IAGhC;;;;;;;;;OASG;IACH,2EARG;QAAuB,aAAa,GAA5B,OAAO;QACoB,MAAM,GAAjC,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC;QACY,KAAK,GAApC,CAAC,KAAK,EAAE,MAAM,KAAK,IAAI;QACqB,UAAU,GAAtD,CAAC,KAAK,EAAE,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,KAAK,IAAI;QACQ,UAAU,GAAtD,CAAC,KAAK,EAAE,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,KAAK,IAAI;QACV,OAAO,EAAjC,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC;KAC3B,GAAU,OAAO,CAAC,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC,CAmDxC;IAED,qBAAqB;IACrB,sBADc,GAAG,CAmBhB;IAZG,kBAQC;IAML,+BAA+B;IAC/B,uBADc,OAAO,CAAC,IAAI,CAAC,CAa1B;IAED,sBAAsB;IACtB,iCADc,IAAI,CAMjB;IAJC,uCAGE;IAFA,sDAA8C;IAC9C,uDAA4C;IAIhD;;;OAGG;IACH,8BAHW,MAAM,GACJ,MAAM,CAQlB;IAED;;;OAGG;IACH,8BAHW,MAAM,GACJ,IAAI,CAqDhB;IAED,sBAAsB;IACtB,wBADc,IAAI,CAOjB;IAED,sBAAsB;IACtB,mBADc,IAAI,CAMjB;IAED,sBAAsB;IACtB,sBADc,IAAI,CAmBjB;IAED,sBAAsB;IACtB,kBADc,IAAI,CAgBjB;IAED;;;OAGG;IACH,aAAc,MAHH,MAAM,CAAC,MAAM,EAAE,GAAG,CAGX,KAFL,IAAI,CA0ChB;IAED;;;OAGG;IACH,6BAHW,KAAK,GACH,IAAI,CAWhB;IAED,sBAAsB;IACtB,SADc,IAAI,CAmBjB;CACF"}
1
+ {"version":3,"file":"websocket-request-client.d.ts","sourceRoot":"/src/","sources":["websocket-request-client.js"],"names":[],"mappings":"AAOA,8EAA8E;AAC9E;IAGE,gDAAgD;IAChD,kBADc,8BAA8B,CAO3C;IAED,sBAAsB;IACtB,uBADc,IAAI,CAQjB;IAnBD,mBAAgB;IAuBd,yBAAyB;IACzB,uBAA0B;IAC1B,oBAAyB;IACzB,iCAAsC;IACtC,kBAAuB;IACvB,0BAA8B;IAGhC,wBAAwB;IACxB,wBADc,MAAM,CAGnB;IAED;;;;;;OAMG;IACH,4BAHG;QAAsB,SAAS,GAAvB,MAAM;KACd,GAAU,OAAO,CAAC,IAAI,CAAC,CAqBzB;IAED;;;;;;;;;OASG;IACH,2EARG;QAAuB,aAAa,GAA5B,OAAO;QACoB,MAAM,GAAjC,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC;QACY,KAAK,GAApC,CAAC,KAAK,EAAE,MAAM,KAAK,IAAI;QACqB,UAAU,GAAtD,CAAC,KAAK,EAAE,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,KAAK,IAAI;QACQ,UAAU,GAAtD,CAAC,KAAK,EAAE,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,KAAK,IAAI;QACV,OAAO,EAAjC,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC;KAC3B,GAAU,OAAO,CAAC,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC,CAmDxC;IAED,qBAAqB;IACrB,sBADc,GAAG,CAmBhB;IAZG,kBAQC;IAML,+BAA+B;IAC/B,uBADc,OAAO,CAAC,IAAI,CAAC,CAa1B;IAED,sBAAsB;IACtB,iCADc,IAAI,CAMjB;IAJC,uCAGE;IAFA,sDAA8C;IAC9C,uDAA4C;IAIhD;;;OAGG;IACH,8BAHW,MAAM,GACJ,MAAM,CAQlB;IAED;;;OAGG;IACH,8BAHW,MAAM,GACJ,IAAI,CAqDhB;IAED,sBAAsB;IACtB,wBADc,IAAI,CAOjB;IAED,sBAAsB;IACtB,mBADc,IAAI,CAMjB;IAED,sBAAsB;IACtB,sBADc,IAAI,CAmBjB;IAED,sBAAsB;IACtB,kBADc,IAAI,CAgBjB;IAED;;;OAGG;IACH,aAAc,MAHH,MAAM,CAAC,MAAM,EAAE,GAAG,CAGX,KAFL,IAAI,CA6ChB;IAED;;;OAGG;IACH,6BAHW,KAAK,GACH,IAAI,CAahB;IAED,sBAAsB;IACtB,4BADc,IAAI,CAUjB;IAED,sBAAsB;IACtB,SADc,IAAI,CAmBjB;CACF"}
@@ -5,6 +5,7 @@ const logger = new Logger({ name: "ApiMaker / WebsocketRequestClient" });
5
5
  const shared = {};
6
6
  /** Shared websocket request client for ApiMaker command/service execution. */
7
7
  export default class ApiMakerWebsocketRequestClient {
8
+ idleWaiters = [];
8
9
  /** @returns {ApiMakerWebsocketRequestClient} */
9
10
  static current() {
10
11
  if (!shared.currentApiMakerWebsocketRequestClient) {
@@ -29,6 +30,35 @@ export default class ApiMakerWebsocketRequestClient {
29
30
  this.responseCache = {};
30
31
  this.subscriptionState = "new";
31
32
  }
33
+ /** @returns {number} */
34
+ pendingRequestsCount() {
35
+ return Object.keys(this.pendingRequests).length;
36
+ }
37
+ /**
38
+ * Waits for all in-flight websocket requests to complete.
39
+ *
40
+ * @param {object} [args]
41
+ * @param {number} [args.timeoutMs]
42
+ * @returns {Promise<void>}
43
+ */
44
+ waitForIdle({ timeoutMs = 5000 } = {}) {
45
+ if (this.pendingRequestsCount() == 0) {
46
+ return Promise.resolve();
47
+ }
48
+ return new Promise((resolve, reject) => {
49
+ const timeout = setTimeout(() => {
50
+ this.idleWaiters = this.idleWaiters.filter((idleWaiter) => idleWaiter.reject != reject);
51
+ reject(new Error(`Timed out while waiting for websocket requests to finish. Pending requests: ${this.pendingRequestsCount()}`));
52
+ }, timeoutMs);
53
+ this.idleWaiters.push({
54
+ reject,
55
+ resolve: () => {
56
+ clearTimeout(timeout);
57
+ resolve();
58
+ }
59
+ });
60
+ });
61
+ }
32
62
  /**
33
63
  * @param {object} args
34
64
  * @param {boolean} [args.cacheResponse]
@@ -251,14 +281,17 @@ export default class ApiMakerWebsocketRequestClient {
251
281
  this.responseCache[pendingRequest.fingerprint] = data.response;
252
282
  }
253
283
  pendingRequest.resolve(data.response);
284
+ this.resolveIdleWaitersIfIdle();
254
285
  }
255
286
  else if (data.type == "api_maker_request_error") {
256
287
  delete this.pendingRequests[data.request_id];
257
288
  pendingRequest.reject(new CustomError("Websocket request failed", { response: data.response }));
289
+ this.resolveIdleWaitersIfIdle();
258
290
  }
259
291
  else {
260
292
  delete this.pendingRequests[data.request_id];
261
293
  pendingRequest.reject(new Error(`Unknown websocket request response type: ${data.type}`));
294
+ this.resolveIdleWaitersIfIdle();
262
295
  }
263
296
  };
264
297
  /**
@@ -272,6 +305,16 @@ export default class ApiMakerWebsocketRequestClient {
272
305
  Object.values(pendingRequests).forEach((pendingRequest) => {
273
306
  queueMicrotask(() => pendingRequest.reject(error));
274
307
  });
308
+ this.resolveIdleWaitersIfIdle();
309
+ }
310
+ /** @returns {void} */
311
+ resolveIdleWaitersIfIdle() {
312
+ if (this.pendingRequestsCount() > 0 || this.idleWaiters.length == 0) {
313
+ return;
314
+ }
315
+ const idleWaiters = this.idleWaiters;
316
+ this.idleWaiters = [];
317
+ idleWaiters.forEach((idleWaiter) => idleWaiter.resolve());
275
318
  }
276
319
  /** @returns {void} */
277
320
  reset() {
@@ -291,4 +334,4 @@ export default class ApiMakerWebsocketRequestClient {
291
334
  this.subscriptionReadyPromise = undefined;
292
335
  }
293
336
  }
294
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"websocket-request-client.js","sourceRoot":"/src/","sources":["websocket-request-client.js"],"names":[],"mappings":"AAAA,OAAO,WAAW,MAAM,mBAAmB,CAAA;AAC3C,OAAO,MAAM,MAAM,aAAa,CAAA;AAChC,OAAO,gBAAgB,MAAM,wBAAwB,CAAA;AAErD,MAAM,MAAM,GAAG,IAAI,MAAM,CAAC,EAAC,IAAI,EAAE,mCAAmC,EAAC,CAAC,CAAA;AACtE,MAAM,MAAM,GAAG,EAAE,CAAA;AAEjB,8EAA8E;AAC9E,MAAM,CAAC,OAAO,OAAO,8BAA8B;IACjD,gDAAgD;IAChD,MAAM,CAAC,OAAO;QACZ,IAAI,CAAC,MAAM,CAAC,qCAAqC,EAAE,CAAC;YAClD,MAAM,CAAC,qCAAqC,GAAG,IAAI,8BAA8B,EAAE,CAAA;QACrF,CAAC;QAED,OAAO,MAAM,CAAC,qCAAqC,CAAA;IACrD,CAAC;IAED,sBAAsB;IACtB,MAAM,CAAC,YAAY;QACjB,IAAI,CAAC,MAAM,CAAC,qCAAqC,EAAE,CAAC;YAClD,OAAM;QACR,CAAC;QAED,MAAM,CAAC,qCAAqC,CAAC,KAAK,EAAE,CAAA;QACpD,OAAO,MAAM,CAAC,qCAAqC,CAAA;IACrD,CAAC;IAED,mBAAmB;IACnB;QACE,IAAI,CAAC,gBAAgB,GAAG,CAAC,CAAA;QACzB,IAAI,CAAC,aAAa,GAAG,KAAK,CAAA;QAC1B,IAAI,CAAC,eAAe,GAAG,EAAE,CAAA;QACzB,IAAI,CAAC,4BAA4B,GAAG,EAAE,CAAA;QACtC,IAAI,CAAC,aAAa,GAAG,EAAE,CAAA;QACvB,IAAI,CAAC,iBAAiB,GAAG,KAAK,CAAA;IAChC,CAAC;IAED;;;;;;;;;OASG;IACH,OAAO,CAAE,EAAC,aAAa,EAAE,MAAM,EAAE,KAAK,EAAE,UAAU,EAAE,UAAU,EAAE,OAAO,EAAC;QACtE,MAAM,WAAW,GAAG,IAAI,CAAC,SAAS,CAAC,EAAC,MAAM,EAAE,OAAO,EAAC,CAAC,CAAA;QAErD,IAAI,aAAa,IAAI,IAAI,CAAC,aAAa,CAAC,WAAW,CAAC,EAAE,CAAC;YACrD,OAAO,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,aAAa,CAAC,WAAW,CAAC,CAAC,CAAA;QACzD,CAAC;QAED,IAAI,IAAI,CAAC,4BAA4B,CAAC,WAAW,CAAC,EAAE,CAAC;YACnD,MAAM,kBAAkB,GAAG,IAAI,CAAC,4BAA4B,CAAC,WAAW,CAAC,CAAA;YACzE,MAAM,cAAc,GAAG,IAAI,CAAC,eAAe,CAAC,kBAAkB,CAAC,SAAS,CAAC,CAAA;YAEzE,IAAI,KAAK;gBAAE,cAAc,EAAE,cAAc,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;YACrD,IAAI,UAAU;gBAAE,cAAc,EAAE,mBAAmB,CAAC,IAAI,CAAC,UAAU,CAAC,CAAA;YACpE,IAAI,UAAU;gBAAE,cAAc,EAAE,mBAAmB,CAAC,IAAI,CAAC,UAAU,CAAC,CAAA;YAEpE,OAAO,kBAAkB,CAAC,OAAO,CAAA;QACnC,CAAC;QAED,MAAM,OAAO,GAAG,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;YAC9C,MAAM,SAAS,GAAG,IAAI,CAAC,gBAAgB,CAAA;YACvC,IAAI,CAAC,gBAAgB,IAAI,CAAC,CAAA;YAE1B,IAAI,CAAC,eAAe,CAAC,SAAS,CAAC,GAAG;gBAChC,aAAa;gBACb,aAAa,EAAE,QAAQ;gBACvB,WAAW;gBACX,wBAAwB,EAAE,CAAC;gBAC3B,cAAc,EAAE,KAAK,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,EAAE;gBACpC,mBAAmB,EAAE,UAAU,CAAC,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,EAAE;gBACnD,mBAAmB,EAAE,UAAU,CAAC,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,EAAE;gBACnD,MAAM;gBACN,OAAO;gBACP,UAAU,EAAE,IAAI,CAAC,kBAAkB,CAAC,SAAS,CAAC;gBAC9C,MAAM;gBACN,OAAO;aACR,CAAA;YAED,IAAI,CAAC,kBAAkB,CAAC,SAAS,CAAC,CAAA;QACpC,CAAC,CAAC,CAAA;QAEF,IAAI,CAAC,4BAA4B,CAAC,WAAW,CAAC,GAAG,EAAC,OAAO,EAAE,SAAS,EAAE,IAAI,CAAC,gBAAgB,GAAG,CAAC,EAAC,CAAA;QAEhG,OAAO,CAAC,IAAI,CAAC,GAAG,EAAE;YAChB,OAAO,IAAI,CAAC,4BAA4B,CAAC,WAAW,CAAC,CAAA;QACvD,CAAC,EAAE,GAAG,EAAE;YACN,OAAO,IAAI,CAAC,4BAA4B,CAAC,WAAW,CAAC,CAAA;QACvD,CAAC,CAAC,CAAA;QAEF,OAAO,OAAO,CAAA;IAChB,CAAC;IAED,qBAAqB;IACrB,kBAAkB;QAChB,IAAI,CAAC,IAAI,CAAC,YAAY,EAAE,CAAC;YACvB,MAAM,CAAC,KAAK,CAAC,yCAAyC,CAAC,CAAA;YACvD,IAAI,CAAC,iBAAiB,GAAG,YAAY,CAAA;YACrC,IAAI,CAAC,6BAA6B,EAAE,CAAA;YAEpC,IAAI,CAAC,YAAY,GAAG,gBAAgB,EAAE,CAAC,aAAa,CAAC,MAAM,CACzD,EAAC,OAAO,EAAE,2BAA2B,EAAC,EACtC;gBACE,SAAS,EAAE,IAAI,CAAC,WAAW;gBAC3B,YAAY,EAAE,IAAI,CAAC,cAAc;gBACjC,QAAQ,EAAE,IAAI,CAAC,UAAU;gBACzB,QAAQ,EAAE,IAAI,CAAC,UAAU;aAC1B,CACF,CAAA;QACH,CAAC;QAED,OAAO,IAAI,CAAC,YAAY,CAAA;IAC1B,CAAC;IAED,+BAA+B;IAC/B,mBAAmB;QACjB,IAAI,CAAC,kBAAkB,EAAE,CAAA;QAEzB,IAAI,IAAI,CAAC,iBAAiB,IAAI,WAAW,EAAE,CAAC;YAC1C,OAAO,OAAO,CAAC,OAAO,EAAE,CAAA;QAC1B,CAAC;QAED,IAAI,CAAC,IAAI,CAAC,wBAAwB,EAAE,CAAC;YACnC,IAAI,CAAC,6BAA6B,EAAE,CAAA;QACtC,CAAC;QAED,OAAO,IAAI,CAAC,wBAAwB,CAAA;IACtC,CAAC;IAED,sBAAsB;IACtB,6BAA6B;QAC3B,IAAI,CAAC,wBAAwB,GAAG,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;YAC9D,IAAI,CAAC,+BAA+B,GAAG,OAAO,CAAA;YAC9C,IAAI,CAAC,8BAA8B,GAAG,MAAM,CAAA;QAC9C,CAAC,CAAC,CAAA;IACJ,CAAC;IAED;;;OAGG;IACH,kBAAkB,CAAE,SAAS;QAC3B,IAAI,UAAU,CAAC,MAAM,EAAE,UAAU,EAAE,CAAC;YAClC,OAAO,UAAU,CAAC,MAAM,CAAC,UAAU,EAAE,CAAA;QACvC,CAAC;QAED,OAAO,qBAAqB,IAAI,CAAC,GAAG,EAAE,IAAI,SAAS,EAAE,CAAA;IACvD,CAAC;IAED;;;OAGG;IACH,kBAAkB,CAAE,SAAS;QAC3B,MAAM,cAAc,GAAG,IAAI,CAAC,eAAe,CAAC,SAAS,CAAC,CAAA;QAEtD,IAAI,CAAC,cAAc,EAAE,CAAC;YACpB,OAAM;QACR,CAAC;QAED,IAAI,cAAc,CAAC,aAAa,IAAI,QAAQ,EAAE,CAAC;YAC7C,OAAM;QACR,CAAC;QAED,cAAc,CAAC,aAAa,GAAG,wBAAwB,CAAA;QAEvD,IAAI,CAAC,mBAAmB,EAAE;aACvB,IAAI,CAAC,GAAG,EAAE;YACT,MAAM,oBAAoB,GAAG,IAAI,CAAC,eAAe,CAAC,SAAS,CAAC,CAAA;YAE5D,IAAI,CAAC,oBAAoB,EAAE,CAAC;gBAC1B,OAAM;YACR,CAAC;YAED,IAAI,oBAAoB,CAAC,aAAa,IAAI,wBAAwB,EAAE,CAAC;gBACnE,OAAM;YACR,CAAC;YAED,oBAAoB,CAAC,aAAa,GAAG,MAAM,CAAA;YAE3C,IAAI,CAAC,kBAAkB,EAAE,CAAC,OAAO,CAAC,SAAS,EAAE;gBAC3C,cAAc,EAAE,oBAAoB,CAAC,aAAa;gBAClD,MAAM,EAAE,oBAAoB,CAAC,MAAM;gBACnC,2BAA2B,EAAE,oBAAoB,CAAC,wBAAwB;gBAC1E,OAAO,EAAE,oBAAoB,CAAC,OAAO;gBACrC,UAAU,EAAE,SAAS;gBACrB,WAAW,EAAE,oBAAoB,CAAC,UAAU;aAC7C,CAAC,CAAA;QACJ,CAAC,CAAC;aACD,KAAK,CAAC,CAAC,KAAK,EAAE,EAAE;YACf,MAAM,oBAAoB,GAAG,IAAI,CAAC,eAAe,CAAC,SAAS,CAAC,CAAA;YAE5D,IAAI,CAAC,oBAAoB,EAAE,CAAC;gBAC1B,OAAM;YACR,CAAC;YAED,IAAI,IAAI,CAAC,iBAAiB,IAAI,UAAU,EAAE,CAAC;gBACzC,OAAO,IAAI,CAAC,eAAe,CAAC,SAAS,CAAC,CAAA;gBACtC,oBAAoB,CAAC,MAAM,CAAC,KAAK,CAAC,CAAA;gBAClC,OAAM;YACR,CAAC;YAED,oBAAoB,CAAC,aAAa,GAAG,QAAQ,CAAA;QAC/C,CAAC,CAAC,CAAA;IACN,CAAC;IAED,sBAAsB;IACtB,oBAAoB;QAClB,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,SAAS,EAAE,cAAc,CAAC,EAAE,EAAE;YAC3E,IAAI,cAAc,CAAC,aAAa,IAAI,QAAQ,EAAE,CAAC;gBAC7C,IAAI,CAAC,kBAAkB,CAAC,QAAQ,CAAC,SAAS,EAAE,EAAE,CAAC,CAAC,CAAA;YAClD,CAAC;QACH,CAAC,CAAC,CAAA;IACJ,CAAC;IAED,sBAAsB;IACtB,WAAW,GAAG,GAAG,EAAE;QACjB,MAAM,CAAC,KAAK,CAAC,0CAA0C,CAAC,CAAA;QACxD,IAAI,CAAC,iBAAiB,GAAG,WAAW,CAAA;QACpC,IAAI,CAAC,+BAA+B,EAAE,EAAE,CAAA;QACxC,IAAI,CAAC,oBAAoB,EAAE,CAAA;IAC7B,CAAC,CAAA;IAED,sBAAsB;IACtB,cAAc,GAAG,GAAG,EAAE;QACpB,MAAM,CAAC,KAAK,CAAC,6CAA6C,CAAC,CAAA;QAC3D,IAAI,CAAC,YAAY,GAAG,IAAI,CAAA;QAExB,IAAI,IAAI,CAAC,aAAa,EAAE,CAAC;YACvB,IAAI,CAAC,iBAAiB,GAAG,OAAO,CAAA;YAChC,OAAM;QACR,CAAC;QAED,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC,OAAO,CAAC,CAAC,cAAc,EAAE,EAAE;YAC7D,IAAI,cAAc,CAAC,aAAa,IAAI,WAAW,EAAE,CAAC;gBAChD,cAAc,CAAC,aAAa,GAAG,QAAQ,CAAA;YACzC,CAAC;QACH,CAAC,CAAC,CAAA;QAEF,IAAI,CAAC,iBAAiB,GAAG,cAAc,CAAA;QACvC,IAAI,CAAC,6BAA6B,EAAE,CAAA;QACpC,IAAI,CAAC,kBAAkB,EAAE,CAAA;IAC3B,CAAC,CAAA;IAED,sBAAsB;IACtB,UAAU,GAAG,GAAG,EAAE;QAChB,IAAI,IAAI,CAAC,aAAa,EAAE,CAAC;YACvB,IAAI,CAAC,iBAAiB,GAAG,OAAO,CAAA;YAChC,IAAI,CAAC,YAAY,GAAG,IAAI,CAAA;YACxB,OAAM;QACR,CAAC;QAED,MAAM,KAAK,GAAG,IAAI,KAAK,CAAC,6CAA6C,CAAC,CAAA;QAEtE,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,CAAA;QACnB,IAAI,CAAC,qBAAqB,CAAC,KAAK,CAAC,CAAA;QACjC,IAAI,CAAC,iBAAiB,GAAG,UAAU,CAAA;QACnC,IAAI,CAAC,YAAY,GAAG,IAAI,CAAA;QACxB,IAAI,CAAC,8BAA8B,EAAE,CAAC,KAAK,CAAC,CAAA;QAC5C,IAAI,CAAC,6BAA6B,EAAE,CAAA;IACtC,CAAC,CAAA;IAED;;;OAGG;IACH,UAAU,GAAG,CAAC,IAAI,EAAE,EAAE;QACpB,MAAM,cAAc,GAAG,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,UAAU,CAAC,CAAA;QAE5D,IAAI,CAAC,cAAc,EAAE,CAAC;YACpB,MAAM,CAAC,KAAK,CAAC,GAAG,EAAE,CAAC,CAAC,uDAAuD,EAAE,EAAC,IAAI,EAAC,CAAC,CAAC,CAAA;YACrF,OAAM;QACR,CAAC;QAED,IAAI,IAAI,CAAC,sBAAsB,IAAI,IAAI,CAAC,sBAAsB,GAAG,cAAc,CAAC,wBAAwB,EAAE,CAAC;YACzG,cAAc,CAAC,wBAAwB,GAAG,IAAI,CAAC,sBAAsB,CAAA;QACvE,CAAC;QAED,IAAI,IAAI,CAAC,IAAI,IAAI,uBAAuB,EAAE,CAAC;YACzC,cAAc,CAAC,cAAc,CAAC,OAAO,CAAC,CAAC,QAAQ,EAAE,EAAE,CAAC,QAAQ,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAA;QAC7E,CAAC;aAAM,IAAI,IAAI,CAAC,IAAI,IAAI,4BAA4B,EAAE,CAAC;YACrD,MAAM,YAAY,GAAG;gBACnB,KAAK,EAAE,IAAI,CAAC,KAAK;gBACjB,QAAQ,EAAE,IAAI,CAAC,QAAQ;gBACvB,KAAK,EAAE,IAAI,CAAC,KAAK;aAClB,CAAA;YAED,cAAc,CAAC,mBAAmB,CAAC,OAAO,CAAC,CAAC,QAAQ,EAAE,EAAE,CAAC,QAAQ,CAAC,YAAY,CAAC,CAAC,CAAA;QAClF,CAAC;aAAM,IAAI,IAAI,CAAC,IAAI,IAAI,4BAA4B,EAAE,CAAC;YACrD,cAAc,CAAC,aAAa,GAAG,UAAU,CAAA;YACzC,cAAc,CAAC,mBAAmB,CAAC,OAAO,CAAC,CAAC,QAAQ,EAAE,EAAE,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAA;QAC1E,CAAC;aAAM,IAAI,IAAI,CAAC,IAAI,IAAI,4BAA4B,EAAE,CAAC;YACrD,OAAO,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,UAAU,CAAC,CAAA;YAE5C,IAAI,cAAc,CAAC,aAAa,EAAE,CAAC;gBACjC,IAAI,CAAC,aAAa,CAAC,cAAc,CAAC,WAAW,CAAC,GAAG,IAAI,CAAC,QAAQ,CAAA;YAChE,CAAC;YAED,cAAc,CAAC,OAAO,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAA;QACvC,CAAC;aAAM,IAAI,IAAI,CAAC,IAAI,IAAI,yBAAyB,EAAE,CAAC;YAClD,OAAO,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,UAAU,CAAC,CAAA;YAC5C,cAAc,CAAC,MAAM,CAAC,IAAI,WAAW,CAAC,0BAA0B,EAAE,EAAC,QAAQ,EAAE,IAAI,CAAC,QAAQ,EAAC,CAAC,CAAC,CAAA;QAC/F,CAAC;aAAM,CAAC;YACN,OAAO,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,UAAU,CAAC,CAAA;YAC5C,cAAc,CAAC,MAAM,CAAC,IAAI,KAAK,CAAC,4CAA4C,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC,CAAA;QAC3F,CAAC;IACH,CAAC,CAAA;IAED;;;OAGG;IACH,qBAAqB,CAAE,KAAK;QAC1B,MAAM,eAAe,GAAG,IAAI,CAAC,eAAe,CAAA;QAE5C,IAAI,CAAC,eAAe,GAAG,EAAE,CAAA;QACzB,IAAI,CAAC,4BAA4B,GAAG,EAAE,CAAA;QAEtC,MAAM,CAAC,MAAM,CAAC,eAAe,CAAC,CAAC,OAAO,CAAC,CAAC,cAAc,EAAE,EAAE;YACxD,cAAc,CAAC,GAAG,EAAE,CAAC,cAAc,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAA;QACpD,CAAC,CAAC,CAAA;IACJ,CAAC;IAED,sBAAsB;IACtB,KAAK;QACH,MAAM,UAAU,GAAG,IAAI,KAAK,CAAC,oCAAoC,CAAC,CAAA;QAElE,IAAI,CAAC,aAAa,GAAG,IAAI,CAAA;QACzB,IAAI,CAAC,qBAAqB,CAAC,UAAU,CAAC,CAAA;QACtC,IAAI,CAAC,aAAa,GAAG,EAAE,CAAA;QACvB,IAAI,CAAC,gBAAgB,GAAG,CAAC,CAAA;QACzB,IAAI,CAAC,8BAA8B,EAAE,CAAC,UAAU,CAAC,CAAA;QAEjD,IAAI,IAAI,CAAC,YAAY,EAAE,WAAW,EAAE,CAAC;YACnC,IAAI,CAAC,YAAY,CAAC,WAAW,EAAE,CAAA;QACjC,CAAC;QAED,IAAI,CAAC,YAAY,GAAG,IAAI,CAAA;QACxB,IAAI,CAAC,iBAAiB,GAAG,KAAK,CAAA;QAC9B,IAAI,CAAC,+BAA+B,GAAG,SAAS,CAAA;QAChD,IAAI,CAAC,8BAA8B,GAAG,SAAS,CAAA;QAC/C,IAAI,CAAC,wBAAwB,GAAG,SAAS,CAAA;IAC3C,CAAC;CACF","sourcesContent":["import CustomError from \"./custom-error.js\"\nimport Logger from \"./logger.js\"\nimport channelsConsumer from \"./channels-consumer.js\"\n\nconst logger = new Logger({name: \"ApiMaker / WebsocketRequestClient\"})\nconst shared = {}\n\n/** Shared websocket request client for ApiMaker command/service execution. */\nexport default class ApiMakerWebsocketRequestClient {\n  /** @returns {ApiMakerWebsocketRequestClient} */\n  static current () {\n    if (!shared.currentApiMakerWebsocketRequestClient) {\n      shared.currentApiMakerWebsocketRequestClient = new ApiMakerWebsocketRequestClient()\n    }\n\n    return shared.currentApiMakerWebsocketRequestClient\n  }\n\n  /** @returns {void} */\n  static resetCurrent () {\n    if (!shared.currentApiMakerWebsocketRequestClient) {\n      return\n    }\n\n    shared.currentApiMakerWebsocketRequestClient.reset()\n    delete shared.currentApiMakerWebsocketRequestClient\n  }\n\n  /** Constructor. */\n  constructor () {\n    this.currentRequestId = 1\n    this.skipReconnect = false\n    this.pendingRequests = {}\n    this.pendingRequestsByFingerprint = {}\n    this.responseCache = {}\n    this.subscriptionState = \"new\"\n  }\n\n  /**\n   * @param {object} args\n   * @param {boolean} [args.cacheResponse]\n   * @param {Record<string, any>} [args.global]\n   * @param {(value: string) => void} [args.onLog]\n   * @param {(value: Record<string, any>) => void} [args.onProgress]\n   * @param {(value: Record<string, any>) => void} [args.onReceived]\n   * @param {Record<string, any>} args.request\n   * @returns {Promise<Record<string, any>>}\n   */\n  perform ({cacheResponse, global, onLog, onProgress, onReceived, request}) {\n    const fingerprint = JSON.stringify({global, request})\n\n    if (cacheResponse && this.responseCache[fingerprint]) {\n      return Promise.resolve(this.responseCache[fingerprint])\n    }\n\n    if (this.pendingRequestsByFingerprint[fingerprint]) {\n      const pendingRequestData = this.pendingRequestsByFingerprint[fingerprint]\n      const pendingRequest = this.pendingRequests[pendingRequestData.requestId]\n\n      if (onLog) pendingRequest?.onLogCallbacks.push(onLog)\n      if (onProgress) pendingRequest?.onProgressCallbacks.push(onProgress)\n      if (onReceived) pendingRequest?.onReceivedCallbacks.push(onReceived)\n\n      return pendingRequestData.promise\n    }\n\n    const promise = new Promise((resolve, reject) => {\n      const requestId = this.currentRequestId\n      this.currentRequestId += 1\n\n      this.pendingRequests[requestId] = {\n        cacheResponse,\n        deliveryState: \"queued\",\n        fingerprint,\n        lastCommandEventSequence: 0,\n        onLogCallbacks: onLog ? [onLog] : [],\n        onProgressCallbacks: onProgress ? [onProgress] : [],\n        onReceivedCallbacks: onReceived ? [onReceived] : [],\n        global,\n        request,\n        requestUid: this.generateRequestUid(requestId),\n        reject,\n        resolve\n      }\n\n      this.sendPendingRequest(requestId)\n    })\n\n    this.pendingRequestsByFingerprint[fingerprint] = {promise, requestId: this.currentRequestId - 1}\n\n    promise.then(() => {\n      delete this.pendingRequestsByFingerprint[fingerprint]\n    }, () => {\n      delete this.pendingRequestsByFingerprint[fingerprint]\n    })\n\n    return promise\n  }\n\n  /** @returns {any} */\n  ensureSubscription () {\n    if (!this.subscription) {\n      logger.debug(\"Creating websocket request subscription\")\n      this.subscriptionState = \"connecting\"\n      this.resetSubscriptionReadyPromise()\n\n      this.subscription = channelsConsumer().subscriptions.create(\n        {channel: \"ApiMaker::RequestsChannel\"},\n        {\n          connected: this.onConnected,\n          disconnected: this.onDisconnected,\n          received: this.onReceived,\n          rejected: this.onRejected\n        }\n      )\n    }\n\n    return this.subscription\n  }\n\n  /** @returns {Promise<void>} */\n  waitForSubscription () {\n    this.ensureSubscription()\n\n    if (this.subscriptionState == \"connected\") {\n      return Promise.resolve()\n    }\n\n    if (!this.subscriptionReadyPromise) {\n      this.resetSubscriptionReadyPromise()\n    }\n\n    return this.subscriptionReadyPromise\n  }\n\n  /** @returns {void} */\n  resetSubscriptionReadyPromise () {\n    this.subscriptionReadyPromise = new Promise((resolve, reject) => {\n      this.resolveSubscriptionReadyPromise = resolve\n      this.rejectSubscriptionReadyPromise = reject\n    })\n  }\n\n  /**\n   * @param {number} requestId\n   * @returns {string}\n   */\n  generateRequestUid (requestId) {\n    if (globalThis.crypto?.randomUUID) {\n      return globalThis.crypto.randomUUID()\n    }\n\n    return `api-maker-request-${Date.now()}-${requestId}`\n  }\n\n  /**\n   * @param {number} requestId\n   * @returns {void}\n   */\n  sendPendingRequest (requestId) {\n    const pendingRequest = this.pendingRequests[requestId]\n\n    if (!pendingRequest) {\n      return\n    }\n\n    if (pendingRequest.deliveryState != \"queued\") {\n      return\n    }\n\n    pendingRequest.deliveryState = \"waiting_for_connection\"\n\n    this.waitForSubscription()\n      .then(() => {\n        const latestPendingRequest = this.pendingRequests[requestId]\n\n        if (!latestPendingRequest) {\n          return\n        }\n\n        if (latestPendingRequest.deliveryState != \"waiting_for_connection\") {\n          return\n        }\n\n        latestPendingRequest.deliveryState = \"sent\"\n\n        this.ensureSubscription().perform(\"execute\", {\n          cache_response: latestPendingRequest.cacheResponse,\n          global: latestPendingRequest.global,\n          last_command_event_sequence: latestPendingRequest.lastCommandEventSequence,\n          request: latestPendingRequest.request,\n          request_id: requestId,\n          request_uid: latestPendingRequest.requestUid\n        })\n      })\n      .catch((error) => {\n        const latestPendingRequest = this.pendingRequests[requestId]\n\n        if (!latestPendingRequest) {\n          return\n        }\n\n        if (this.subscriptionState == \"rejected\") {\n          delete this.pendingRequests[requestId]\n          latestPendingRequest.reject(error)\n          return\n        }\n\n        latestPendingRequest.deliveryState = \"queued\"\n      })\n  }\n\n  /** @returns {void} */\n  resendQueuedRequests () {\n    Object.entries(this.pendingRequests).forEach(([requestId, pendingRequest]) => {\n      if (pendingRequest.deliveryState == \"queued\") {\n        this.sendPendingRequest(parseInt(requestId, 10))\n      }\n    })\n  }\n\n  /** @returns {void} */\n  onConnected = () => {\n    logger.debug(\"Websocket request subscription connected\")\n    this.subscriptionState = \"connected\"\n    this.resolveSubscriptionReadyPromise?.()\n    this.resendQueuedRequests()\n  }\n\n  /** @returns {void} */\n  onDisconnected = () => {\n    logger.debug(\"Websocket request subscription disconnected\")\n    this.subscription = null\n\n    if (this.skipReconnect) {\n      this.subscriptionState = \"reset\"\n      return\n    }\n\n    Object.values(this.pendingRequests).forEach((pendingRequest) => {\n      if (pendingRequest.deliveryState != \"completed\") {\n        pendingRequest.deliveryState = \"queued\"\n      }\n    })\n\n    this.subscriptionState = \"disconnected\"\n    this.resetSubscriptionReadyPromise()\n    this.ensureSubscription()\n  }\n\n  /** @returns {void} */\n  onRejected = () => {\n    if (this.skipReconnect) {\n      this.subscriptionState = \"reset\"\n      this.subscription = null\n      return\n    }\n\n    const error = new Error(\"Websocket request subscription was rejected\")\n\n    logger.error(error)\n    this.rejectPendingRequests(error)\n    this.subscriptionState = \"rejected\"\n    this.subscription = null\n    this.rejectSubscriptionReadyPromise?.(error)\n    this.resetSubscriptionReadyPromise()\n  }\n\n  /**\n   * @param {Record<string, any>} data\n   * @returns {void}\n   */\n  onReceived = (data) => {\n    const pendingRequest = this.pendingRequests[data.request_id]\n\n    if (!pendingRequest) {\n      logger.debug(() => [\"Ignoring websocket response without a pending request\", {data}])\n      return\n    }\n\n    if (data.command_event_sequence && data.command_event_sequence > pendingRequest.lastCommandEventSequence) {\n      pendingRequest.lastCommandEventSequence = data.command_event_sequence\n    }\n\n    if (data.type == \"api_maker_command_log\") {\n      pendingRequest.onLogCallbacks.forEach((callback) => callback(data.message))\n    } else if (data.type == \"api_maker_command_progress\") {\n      const progressData = {\n        count: data.count,\n        progress: data.progress,\n        total: data.total\n      }\n\n      pendingRequest.onProgressCallbacks.forEach((callback) => callback(progressData))\n    } else if (data.type == \"api_maker_request_received\") {\n      pendingRequest.deliveryState = \"received\"\n      pendingRequest.onReceivedCallbacks.forEach((callback) => callback(data))\n    } else if (data.type == \"api_maker_request_response\") {\n      delete this.pendingRequests[data.request_id]\n\n      if (pendingRequest.cacheResponse) {\n        this.responseCache[pendingRequest.fingerprint] = data.response\n      }\n\n      pendingRequest.resolve(data.response)\n    } else if (data.type == \"api_maker_request_error\") {\n      delete this.pendingRequests[data.request_id]\n      pendingRequest.reject(new CustomError(\"Websocket request failed\", {response: data.response}))\n    } else {\n      delete this.pendingRequests[data.request_id]\n      pendingRequest.reject(new Error(`Unknown websocket request response type: ${data.type}`))\n    }\n  }\n\n  /**\n   * @param {Error} error\n   * @returns {void}\n   */\n  rejectPendingRequests (error) {\n    const pendingRequests = this.pendingRequests\n\n    this.pendingRequests = {}\n    this.pendingRequestsByFingerprint = {}\n\n    Object.values(pendingRequests).forEach((pendingRequest) => {\n      queueMicrotask(() => pendingRequest.reject(error))\n    })\n  }\n\n  /** @returns {void} */\n  reset () {\n    const resetError = new Error(\"Websocket request client was reset\")\n\n    this.skipReconnect = true\n    this.rejectPendingRequests(resetError)\n    this.responseCache = {}\n    this.currentRequestId = 1\n    this.rejectSubscriptionReadyPromise?.(resetError)\n\n    if (this.subscription?.unsubscribe) {\n      this.subscription.unsubscribe()\n    }\n\n    this.subscription = null\n    this.subscriptionState = \"new\"\n    this.resolveSubscriptionReadyPromise = undefined\n    this.rejectSubscriptionReadyPromise = undefined\n    this.subscriptionReadyPromise = undefined\n  }\n}\n"]}
337
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"websocket-request-client.js","sourceRoot":"/src/","sources":["websocket-request-client.js"],"names":[],"mappings":"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","sourcesContent":["import CustomError from \"./custom-error.js\"\nimport Logger from \"./logger.js\"\nimport channelsConsumer from \"./channels-consumer.js\"\n\nconst logger = new Logger({name: \"ApiMaker / WebsocketRequestClient\"})\nconst shared = {}\n\n/** Shared websocket request client for ApiMaker command/service execution. */\nexport default class ApiMakerWebsocketRequestClient {\n  idleWaiters = []\n\n  /** @returns {ApiMakerWebsocketRequestClient} */\n  static current () {\n    if (!shared.currentApiMakerWebsocketRequestClient) {\n      shared.currentApiMakerWebsocketRequestClient = new ApiMakerWebsocketRequestClient()\n    }\n\n    return shared.currentApiMakerWebsocketRequestClient\n  }\n\n  /** @returns {void} */\n  static resetCurrent () {\n    if (!shared.currentApiMakerWebsocketRequestClient) {\n      return\n    }\n\n    shared.currentApiMakerWebsocketRequestClient.reset()\n    delete shared.currentApiMakerWebsocketRequestClient\n  }\n\n  /** Constructor. */\n  constructor () {\n    this.currentRequestId = 1\n    this.skipReconnect = false\n    this.pendingRequests = {}\n    this.pendingRequestsByFingerprint = {}\n    this.responseCache = {}\n    this.subscriptionState = \"new\"\n  }\n\n  /** @returns {number} */\n  pendingRequestsCount () {\n    return Object.keys(this.pendingRequests).length\n  }\n\n  /**\n   * Waits for all in-flight websocket requests to complete.\n   *\n   * @param {object} [args]\n   * @param {number} [args.timeoutMs]\n   * @returns {Promise<void>}\n   */\n  waitForIdle ({timeoutMs = 5000} = {}) {\n    if (this.pendingRequestsCount() == 0) {\n      return Promise.resolve()\n    }\n\n    return new Promise((resolve, reject) => {\n      const timeout = setTimeout(() => {\n        this.idleWaiters = this.idleWaiters.filter((idleWaiter) => idleWaiter.reject != reject)\n        reject(new Error(`Timed out while waiting for websocket requests to finish. Pending requests: ${this.pendingRequestsCount()}`))\n      }, timeoutMs)\n\n      this.idleWaiters.push({\n        reject,\n        resolve: () => {\n          clearTimeout(timeout)\n          resolve()\n        }\n      })\n    })\n  }\n\n  /**\n   * @param {object} args\n   * @param {boolean} [args.cacheResponse]\n   * @param {Record<string, any>} [args.global]\n   * @param {(value: string) => void} [args.onLog]\n   * @param {(value: Record<string, any>) => void} [args.onProgress]\n   * @param {(value: Record<string, any>) => void} [args.onReceived]\n   * @param {Record<string, any>} args.request\n   * @returns {Promise<Record<string, any>>}\n   */\n  perform ({cacheResponse, global, onLog, onProgress, onReceived, request}) {\n    const fingerprint = JSON.stringify({global, request})\n\n    if (cacheResponse && this.responseCache[fingerprint]) {\n      return Promise.resolve(this.responseCache[fingerprint])\n    }\n\n    if (this.pendingRequestsByFingerprint[fingerprint]) {\n      const pendingRequestData = this.pendingRequestsByFingerprint[fingerprint]\n      const pendingRequest = this.pendingRequests[pendingRequestData.requestId]\n\n      if (onLog) pendingRequest?.onLogCallbacks.push(onLog)\n      if (onProgress) pendingRequest?.onProgressCallbacks.push(onProgress)\n      if (onReceived) pendingRequest?.onReceivedCallbacks.push(onReceived)\n\n      return pendingRequestData.promise\n    }\n\n    const promise = new Promise((resolve, reject) => {\n      const requestId = this.currentRequestId\n      this.currentRequestId += 1\n\n      this.pendingRequests[requestId] = {\n        cacheResponse,\n        deliveryState: \"queued\",\n        fingerprint,\n        lastCommandEventSequence: 0,\n        onLogCallbacks: onLog ? [onLog] : [],\n        onProgressCallbacks: onProgress ? [onProgress] : [],\n        onReceivedCallbacks: onReceived ? [onReceived] : [],\n        global,\n        request,\n        requestUid: this.generateRequestUid(requestId),\n        reject,\n        resolve\n      }\n\n      this.sendPendingRequest(requestId)\n    })\n\n    this.pendingRequestsByFingerprint[fingerprint] = {promise, requestId: this.currentRequestId - 1}\n\n    promise.then(() => {\n      delete this.pendingRequestsByFingerprint[fingerprint]\n    }, () => {\n      delete this.pendingRequestsByFingerprint[fingerprint]\n    })\n\n    return promise\n  }\n\n  /** @returns {any} */\n  ensureSubscription () {\n    if (!this.subscription) {\n      logger.debug(\"Creating websocket request subscription\")\n      this.subscriptionState = \"connecting\"\n      this.resetSubscriptionReadyPromise()\n\n      this.subscription = channelsConsumer().subscriptions.create(\n        {channel: \"ApiMaker::RequestsChannel\"},\n        {\n          connected: this.onConnected,\n          disconnected: this.onDisconnected,\n          received: this.onReceived,\n          rejected: this.onRejected\n        }\n      )\n    }\n\n    return this.subscription\n  }\n\n  /** @returns {Promise<void>} */\n  waitForSubscription () {\n    this.ensureSubscription()\n\n    if (this.subscriptionState == \"connected\") {\n      return Promise.resolve()\n    }\n\n    if (!this.subscriptionReadyPromise) {\n      this.resetSubscriptionReadyPromise()\n    }\n\n    return this.subscriptionReadyPromise\n  }\n\n  /** @returns {void} */\n  resetSubscriptionReadyPromise () {\n    this.subscriptionReadyPromise = new Promise((resolve, reject) => {\n      this.resolveSubscriptionReadyPromise = resolve\n      this.rejectSubscriptionReadyPromise = reject\n    })\n  }\n\n  /**\n   * @param {number} requestId\n   * @returns {string}\n   */\n  generateRequestUid (requestId) {\n    if (globalThis.crypto?.randomUUID) {\n      return globalThis.crypto.randomUUID()\n    }\n\n    return `api-maker-request-${Date.now()}-${requestId}`\n  }\n\n  /**\n   * @param {number} requestId\n   * @returns {void}\n   */\n  sendPendingRequest (requestId) {\n    const pendingRequest = this.pendingRequests[requestId]\n\n    if (!pendingRequest) {\n      return\n    }\n\n    if (pendingRequest.deliveryState != \"queued\") {\n      return\n    }\n\n    pendingRequest.deliveryState = \"waiting_for_connection\"\n\n    this.waitForSubscription()\n      .then(() => {\n        const latestPendingRequest = this.pendingRequests[requestId]\n\n        if (!latestPendingRequest) {\n          return\n        }\n\n        if (latestPendingRequest.deliveryState != \"waiting_for_connection\") {\n          return\n        }\n\n        latestPendingRequest.deliveryState = \"sent\"\n\n        this.ensureSubscription().perform(\"execute\", {\n          cache_response: latestPendingRequest.cacheResponse,\n          global: latestPendingRequest.global,\n          last_command_event_sequence: latestPendingRequest.lastCommandEventSequence,\n          request: latestPendingRequest.request,\n          request_id: requestId,\n          request_uid: latestPendingRequest.requestUid\n        })\n      })\n      .catch((error) => {\n        const latestPendingRequest = this.pendingRequests[requestId]\n\n        if (!latestPendingRequest) {\n          return\n        }\n\n        if (this.subscriptionState == \"rejected\") {\n          delete this.pendingRequests[requestId]\n          latestPendingRequest.reject(error)\n          return\n        }\n\n        latestPendingRequest.deliveryState = \"queued\"\n      })\n  }\n\n  /** @returns {void} */\n  resendQueuedRequests () {\n    Object.entries(this.pendingRequests).forEach(([requestId, pendingRequest]) => {\n      if (pendingRequest.deliveryState == \"queued\") {\n        this.sendPendingRequest(parseInt(requestId, 10))\n      }\n    })\n  }\n\n  /** @returns {void} */\n  onConnected = () => {\n    logger.debug(\"Websocket request subscription connected\")\n    this.subscriptionState = \"connected\"\n    this.resolveSubscriptionReadyPromise?.()\n    this.resendQueuedRequests()\n  }\n\n  /** @returns {void} */\n  onDisconnected = () => {\n    logger.debug(\"Websocket request subscription disconnected\")\n    this.subscription = null\n\n    if (this.skipReconnect) {\n      this.subscriptionState = \"reset\"\n      return\n    }\n\n    Object.values(this.pendingRequests).forEach((pendingRequest) => {\n      if (pendingRequest.deliveryState != \"completed\") {\n        pendingRequest.deliveryState = \"queued\"\n      }\n    })\n\n    this.subscriptionState = \"disconnected\"\n    this.resetSubscriptionReadyPromise()\n    this.ensureSubscription()\n  }\n\n  /** @returns {void} */\n  onRejected = () => {\n    if (this.skipReconnect) {\n      this.subscriptionState = \"reset\"\n      this.subscription = null\n      return\n    }\n\n    const error = new Error(\"Websocket request subscription was rejected\")\n\n    logger.error(error)\n    this.rejectPendingRequests(error)\n    this.subscriptionState = \"rejected\"\n    this.subscription = null\n    this.rejectSubscriptionReadyPromise?.(error)\n    this.resetSubscriptionReadyPromise()\n  }\n\n  /**\n   * @param {Record<string, any>} data\n   * @returns {void}\n   */\n  onReceived = (data) => {\n    const pendingRequest = this.pendingRequests[data.request_id]\n\n    if (!pendingRequest) {\n      logger.debug(() => [\"Ignoring websocket response without a pending request\", {data}])\n      return\n    }\n\n    if (data.command_event_sequence && data.command_event_sequence > pendingRequest.lastCommandEventSequence) {\n      pendingRequest.lastCommandEventSequence = data.command_event_sequence\n    }\n\n    if (data.type == \"api_maker_command_log\") {\n      pendingRequest.onLogCallbacks.forEach((callback) => callback(data.message))\n    } else if (data.type == \"api_maker_command_progress\") {\n      const progressData = {\n        count: data.count,\n        progress: data.progress,\n        total: data.total\n      }\n\n      pendingRequest.onProgressCallbacks.forEach((callback) => callback(progressData))\n    } else if (data.type == \"api_maker_request_received\") {\n      pendingRequest.deliveryState = \"received\"\n      pendingRequest.onReceivedCallbacks.forEach((callback) => callback(data))\n    } else if (data.type == \"api_maker_request_response\") {\n      delete this.pendingRequests[data.request_id]\n\n      if (pendingRequest.cacheResponse) {\n        this.responseCache[pendingRequest.fingerprint] = data.response\n      }\n\n      pendingRequest.resolve(data.response)\n      this.resolveIdleWaitersIfIdle()\n    } else if (data.type == \"api_maker_request_error\") {\n      delete this.pendingRequests[data.request_id]\n      pendingRequest.reject(new CustomError(\"Websocket request failed\", {response: data.response}))\n      this.resolveIdleWaitersIfIdle()\n    } else {\n      delete this.pendingRequests[data.request_id]\n      pendingRequest.reject(new Error(`Unknown websocket request response type: ${data.type}`))\n      this.resolveIdleWaitersIfIdle()\n    }\n  }\n\n  /**\n   * @param {Error} error\n   * @returns {void}\n   */\n  rejectPendingRequests (error) {\n    const pendingRequests = this.pendingRequests\n\n    this.pendingRequests = {}\n    this.pendingRequestsByFingerprint = {}\n\n    Object.values(pendingRequests).forEach((pendingRequest) => {\n      queueMicrotask(() => pendingRequest.reject(error))\n    })\n\n    this.resolveIdleWaitersIfIdle()\n  }\n\n  /** @returns {void} */\n  resolveIdleWaitersIfIdle () {\n    if (this.pendingRequestsCount() > 0 || this.idleWaiters.length == 0) {\n      return\n    }\n\n    const idleWaiters = this.idleWaiters\n\n    this.idleWaiters = []\n    idleWaiters.forEach((idleWaiter) => idleWaiter.resolve())\n  }\n\n  /** @returns {void} */\n  reset () {\n    const resetError = new Error(\"Websocket request client was reset\")\n\n    this.skipReconnect = true\n    this.rejectPendingRequests(resetError)\n    this.responseCache = {}\n    this.currentRequestId = 1\n    this.rejectSubscriptionReadyPromise?.(resetError)\n\n    if (this.subscription?.unsubscribe) {\n      this.subscription.unsubscribe()\n    }\n\n    this.subscription = null\n    this.subscriptionState = \"new\"\n    this.resolveSubscriptionReadyPromise = undefined\n    this.rejectSubscriptionReadyPromise = undefined\n    this.subscriptionReadyPromise = undefined\n  }\n}\n"]}
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@kaspernj/api-maker",
3
3
  "type": "module",
4
- "version": "1.0.2151",
4
+ "version": "1.0.2153",
5
5
  "description": "My new module",
6
6
  "files": [
7
7
  "build/**"