@osdk/client 2.1.0-beta.2 → 2.1.0-beta.4

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 (140) hide show
  1. package/CHANGELOG.md +39 -0
  2. package/build/browser/Client.d.ts +8 -21
  3. package/build/browser/Client.d.ts.map +1 -1
  4. package/build/browser/Client.js +2 -1
  5. package/build/browser/Client.js.map +1 -1
  6. package/build/browser/MinimalClientContext.d.ts +1 -1
  7. package/build/browser/MinimalClientContext.d.ts.map +1 -1
  8. package/build/browser/__unstable/ConjureSupport.js +4 -4
  9. package/build/browser/__unstable/ConjureSupport.js.map +1 -1
  10. package/build/browser/__unstable/createBulkLinksAsyncIterFactory.d.ts +1 -6
  11. package/build/browser/__unstable/createBulkLinksAsyncIterFactory.d.ts.map +1 -1
  12. package/build/browser/__unstable/createBulkLinksAsyncIterFactory.js +1 -1
  13. package/build/browser/__unstable/createBulkLinksAsyncIterFactory.js.map +1 -1
  14. package/build/browser/actions/applyAction.js +1 -1
  15. package/build/browser/actions/applyAction.js.map +1 -1
  16. package/build/browser/createClient.d.ts +1 -1
  17. package/build/browser/createClient.d.ts.map +1 -1
  18. package/build/browser/createClient.js +41 -19
  19. package/build/browser/createClient.js.map +1 -1
  20. package/build/browser/createGeotimeSeriesProperty.d.ts +1 -1
  21. package/build/browser/createGeotimeSeriesProperty.d.ts.map +1 -1
  22. package/build/browser/createGeotimeSeriesProperty.js +4 -1
  23. package/build/browser/createGeotimeSeriesProperty.js.map +1 -1
  24. package/build/browser/createMinimalClient.js +3 -3
  25. package/build/browser/createMinimalClient.js.map +1 -1
  26. package/build/browser/createPlatformClient.d.ts +1 -1
  27. package/build/browser/createPlatformClient.d.ts.map +1 -1
  28. package/build/browser/index.d.ts +0 -1
  29. package/build/browser/index.d.ts.map +1 -1
  30. package/build/browser/index.js +0 -1
  31. package/build/browser/index.js.map +1 -1
  32. package/build/browser/intellisense.test.js +1 -1
  33. package/build/browser/intellisense.test.js.map +1 -1
  34. package/build/browser/internal/conversions/legacyToModernSingleAggregationResult.js +1 -1
  35. package/build/browser/internal/conversions/legacyToModernSingleAggregationResult.js.map +1 -1
  36. package/build/browser/internal/conversions/modernToLegacyWhereClause.js +3 -3
  37. package/build/browser/internal/conversions/modernToLegacyWhereClause.js.map +1 -1
  38. package/build/browser/object/aggregate.js +1 -1
  39. package/build/browser/object/aggregate.js.map +1 -1
  40. package/build/browser/object/convertWireToOsdkObjects/InterfaceHolder.d.ts +1 -10
  41. package/build/browser/object/convertWireToOsdkObjects/InterfaceHolder.d.ts.map +1 -1
  42. package/build/browser/object/convertWireToOsdkObjects/InternalSymbols.d.ts +0 -7
  43. package/build/browser/object/convertWireToOsdkObjects/InternalSymbols.d.ts.map +1 -1
  44. package/build/browser/object/convertWireToOsdkObjects/InternalSymbols.js +6 -0
  45. package/build/browser/object/convertWireToOsdkObjects/InternalSymbols.js.map +1 -1
  46. package/build/browser/object/convertWireToOsdkObjects/createOsdkObject.js +7 -1
  47. package/build/browser/object/convertWireToOsdkObjects/createOsdkObject.js.map +1 -1
  48. package/build/browser/object/convertWireToOsdkObjects.js +2 -2
  49. package/build/browser/object/convertWireToOsdkObjects.js.map +1 -1
  50. package/build/browser/object/convertWireToOsdkObjects.test.js +10 -10
  51. package/build/browser/object/convertWireToOsdkObjects.test.js.map +1 -1
  52. package/build/browser/object/fetchPage.js +1 -1
  53. package/build/browser/object/fetchPage.js.map +1 -1
  54. package/build/browser/objectSet/ObjectSet.test.js +43 -47
  55. package/build/browser/objectSet/ObjectSet.test.js.map +1 -1
  56. package/build/browser/objectSet/ObjectSetListenerWebsocket.js +62 -57
  57. package/build/browser/objectSet/ObjectSetListenerWebsocket.js.map +1 -1
  58. package/build/browser/objectSet/ObjectSetListenerWebsocket.test.js +109 -21
  59. package/build/browser/objectSet/ObjectSetListenerWebsocket.test.js.map +1 -1
  60. package/build/browser/objectSet/createObjectSet.js +2 -5
  61. package/build/browser/objectSet/createObjectSet.js.map +1 -1
  62. package/build/browser/ontology/makeConjureContext.d.ts +1 -1
  63. package/build/browser/ontology/makeConjureContext.d.ts.map +1 -1
  64. package/build/browser/queries/applyQuery.js +1 -1
  65. package/build/browser/queries/applyQuery.js.map +1 -1
  66. package/build/browser/queries/queries.test.js +30 -2
  67. package/build/browser/queries/queries.test.js.map +1 -1
  68. package/build/browser/tsserver.js +1 -1
  69. package/build/browser/tsserver.js.map +1 -1
  70. package/build/browser/util/UserAgent.js +1 -1
  71. package/build/esm/Client.d.ts +8 -21
  72. package/build/esm/Client.d.ts.map +1 -1
  73. package/build/esm/Client.js +2 -1
  74. package/build/esm/Client.js.map +1 -1
  75. package/build/esm/MinimalClientContext.d.ts +1 -1
  76. package/build/esm/MinimalClientContext.d.ts.map +1 -1
  77. package/build/esm/__unstable/ConjureSupport.js +4 -4
  78. package/build/esm/__unstable/ConjureSupport.js.map +1 -1
  79. package/build/esm/__unstable/createBulkLinksAsyncIterFactory.d.ts +1 -6
  80. package/build/esm/__unstable/createBulkLinksAsyncIterFactory.d.ts.map +1 -1
  81. package/build/esm/__unstable/createBulkLinksAsyncIterFactory.js +1 -1
  82. package/build/esm/__unstable/createBulkLinksAsyncIterFactory.js.map +1 -1
  83. package/build/esm/actions/applyAction.js +1 -1
  84. package/build/esm/actions/applyAction.js.map +1 -1
  85. package/build/esm/createClient.d.ts +1 -1
  86. package/build/esm/createClient.d.ts.map +1 -1
  87. package/build/esm/createClient.js +41 -19
  88. package/build/esm/createClient.js.map +1 -1
  89. package/build/esm/createGeotimeSeriesProperty.d.ts +1 -1
  90. package/build/esm/createGeotimeSeriesProperty.d.ts.map +1 -1
  91. package/build/esm/createGeotimeSeriesProperty.js +4 -1
  92. package/build/esm/createGeotimeSeriesProperty.js.map +1 -1
  93. package/build/esm/createMinimalClient.js +3 -3
  94. package/build/esm/createMinimalClient.js.map +1 -1
  95. package/build/esm/createPlatformClient.d.ts +1 -1
  96. package/build/esm/createPlatformClient.d.ts.map +1 -1
  97. package/build/esm/index.d.ts +0 -1
  98. package/build/esm/index.d.ts.map +1 -1
  99. package/build/esm/index.js +0 -1
  100. package/build/esm/index.js.map +1 -1
  101. package/build/esm/intellisense.test.js +1 -1
  102. package/build/esm/intellisense.test.js.map +1 -1
  103. package/build/esm/internal/conversions/legacyToModernSingleAggregationResult.js +1 -1
  104. package/build/esm/internal/conversions/legacyToModernSingleAggregationResult.js.map +1 -1
  105. package/build/esm/internal/conversions/modernToLegacyWhereClause.js +3 -3
  106. package/build/esm/internal/conversions/modernToLegacyWhereClause.js.map +1 -1
  107. package/build/esm/object/aggregate.js +1 -1
  108. package/build/esm/object/aggregate.js.map +1 -1
  109. package/build/esm/object/convertWireToOsdkObjects/InterfaceHolder.d.ts +1 -10
  110. package/build/esm/object/convertWireToOsdkObjects/InterfaceHolder.d.ts.map +1 -1
  111. package/build/esm/object/convertWireToOsdkObjects/InternalSymbols.d.ts +0 -7
  112. package/build/esm/object/convertWireToOsdkObjects/InternalSymbols.d.ts.map +1 -1
  113. package/build/esm/object/convertWireToOsdkObjects/InternalSymbols.js +6 -0
  114. package/build/esm/object/convertWireToOsdkObjects/InternalSymbols.js.map +1 -1
  115. package/build/esm/object/convertWireToOsdkObjects/createOsdkObject.js +7 -1
  116. package/build/esm/object/convertWireToOsdkObjects/createOsdkObject.js.map +1 -1
  117. package/build/esm/object/convertWireToOsdkObjects.js +2 -2
  118. package/build/esm/object/convertWireToOsdkObjects.js.map +1 -1
  119. package/build/esm/object/convertWireToOsdkObjects.test.js +10 -10
  120. package/build/esm/object/convertWireToOsdkObjects.test.js.map +1 -1
  121. package/build/esm/object/fetchPage.js +1 -1
  122. package/build/esm/object/fetchPage.js.map +1 -1
  123. package/build/esm/objectSet/ObjectSet.test.js +43 -47
  124. package/build/esm/objectSet/ObjectSet.test.js.map +1 -1
  125. package/build/esm/objectSet/ObjectSetListenerWebsocket.js +62 -57
  126. package/build/esm/objectSet/ObjectSetListenerWebsocket.js.map +1 -1
  127. package/build/esm/objectSet/ObjectSetListenerWebsocket.test.js +109 -21
  128. package/build/esm/objectSet/ObjectSetListenerWebsocket.test.js.map +1 -1
  129. package/build/esm/objectSet/createObjectSet.js +2 -5
  130. package/build/esm/objectSet/createObjectSet.js.map +1 -1
  131. package/build/esm/ontology/makeConjureContext.d.ts +1 -1
  132. package/build/esm/ontology/makeConjureContext.d.ts.map +1 -1
  133. package/build/esm/queries/applyQuery.js +1 -1
  134. package/build/esm/queries/applyQuery.js.map +1 -1
  135. package/build/esm/queries/queries.test.js +30 -2
  136. package/build/esm/queries/queries.test.js.map +1 -1
  137. package/build/esm/tsserver.js +1 -1
  138. package/build/esm/tsserver.js.map +1 -1
  139. package/build/esm/util/UserAgent.js +1 -1
  140. package/package.json +14 -13
@@ -16,7 +16,6 @@
16
16
  import WebSocket from "isomorphic-ws";
17
17
  import invariant from "tiny-invariant";
18
18
  import { convertWireToOsdkObjects } from "../object/convertWireToOsdkObjects.js";
19
- const ONE_DAY_MS = 24 * 60 * 60 * 1000;
20
19
  const MINIMUM_RECONNECT_DELAY_MS = 5 * 1000;
21
20
  /** Noop function to reduce conditional checks */
22
21
  function doNothing() {}
@@ -26,12 +25,14 @@ function doNothing() {}
26
25
  function fillOutListener({
27
26
  onChange = doNothing,
28
27
  onError = doNothing,
29
- onOutOfDate = doNothing
28
+ onOutOfDate = doNothing,
29
+ onSuccessfulSubscription = doNothing
30
30
  }) {
31
31
  return {
32
32
  onChange,
33
33
  onError,
34
- onOutOfDate
34
+ onOutOfDate,
35
+ onSuccessfulSubscription
35
36
  };
36
37
  }
37
38
  function isReady(sub) {
@@ -43,7 +44,6 @@ function subscriptionIsDone(sub) {
43
44
  /** @internal */
44
45
  export class ObjectSetListenerWebsocket {
45
46
  static #instances = new WeakMap();
46
- OBJECT_SET_EXPIRY_MS;
47
47
  MINIMUM_RECONNECT_DELAY_MS;
48
48
  // FIXME
49
49
  static getInstance(client) {
@@ -70,22 +70,25 @@ export class ObjectSetListenerWebsocket {
70
70
  #maybeDisconnectTimeout;
71
71
  // DO NOT CONSTRUCT DIRECTLY. ONLY EXPOSED AS A TESTING SEAM
72
72
  constructor(client, {
73
- objectSetExpiryMs = ONE_DAY_MS,
74
73
  minimumReconnectDelayMs = MINIMUM_RECONNECT_DELAY_MS
75
74
  } = {}) {
76
- this.OBJECT_SET_EXPIRY_MS = objectSetExpiryMs;
77
75
  this.MINIMUM_RECONNECT_DELAY_MS = minimumReconnectDelayMs;
78
76
  this.#client = client;
79
77
  this.#logger = client.logger?.child({}, {
80
78
  msgPrefix: "<OSW> "
81
79
  });
82
- !(client.baseUrl.startsWith("https://") || client.baseUrl.startsWith("http://")) ? invariant(false, "Stack must be a URL") : void 0;
80
+ !(client.baseUrl.startsWith("https://") || client.baseUrl.startsWith("http://")) ? process.env.NODE_ENV !== "production" ? invariant(false, "Stack must be a URL") : invariant(false) : void 0;
83
81
  }
84
- async subscribe(objectSet, listener, properties) {
82
+ async subscribe(objectType, objectSet, listener, properties) {
83
+ const objDef = await this.#client.ontologyProvider.getObjectDefinition(objectType.apiName);
84
+ const objectProperties = properties.filter(p => objDef.properties[p].type !== "geotimeSeriesReference");
85
+ const referenceProperties = properties.filter(p => objDef.properties[p].type === "geotimeSeriesReference");
85
86
  const sub = {
86
87
  listener: fillOutListener(listener),
87
88
  objectSet,
88
- requestedProperties: properties,
89
+ primaryKeyPropertyName: objDef.primaryKeyApiName,
90
+ requestedProperties: objectProperties,
91
+ requestedReferenceProperties: referenceProperties,
89
92
  status: "preparing",
90
93
  // Since we don't have a real subscription id yet but we need to keep
91
94
  // track of this reference, we can just use a random uuid.
@@ -109,16 +112,9 @@ export class ObjectSetListenerWebsocket {
109
112
  * @returns
110
113
  */
111
114
  async #initiateSubscribe(sub) {
112
- if (process?.env?.NODE_ENV !== "production") {
115
+ if (process.env.NODE_ENV !== "production") {
113
116
  this.#logger?.trace("#initiateSubscribe()");
114
117
  }
115
- if (sub.expiry) {
116
- clearTimeout(sub.expiry);
117
- }
118
- // expiry is tied to the temporary object set, which is set to `timeToLive: "ONE_DAY"`
119
- // in `#createTemporaryObjectSet`. They should be in sync
120
- sub.expiry = setTimeout(() => this.#expire(sub), this.OBJECT_SET_EXPIRY_MS);
121
- await this.#client.ontologyRid;
122
118
  try {
123
119
  await this.#ensureWebsocket();
124
120
  // the consumer may have already unsubscribed before we are ready to request a subscription
@@ -137,7 +133,7 @@ export class ObjectSetListenerWebsocket {
137
133
  }
138
134
  }
139
135
  #sendSubscribeMessage() {
140
- if (process?.env?.NODE_ENV !== "production") {
136
+ if (process.env.NODE_ENV !== "production") {
141
137
  this.#logger?.trace("#sendSubscribeMessage()");
142
138
  }
143
139
  // If two calls to `.subscribe()` happen at once (or if the connection is reset),
@@ -145,7 +141,7 @@ export class ObjectSetListenerWebsocket {
145
141
  // so we filter those out.
146
142
  const readySubs = [...this.#subscriptions.values()].filter(isReady);
147
143
  if (readySubs.length === 0) {
148
- if (process?.env?.NODE_ENV !== "production") {
144
+ if (process.env.NODE_ENV !== "production") {
149
145
  this.#logger?.trace("#sendSubscribeMessage(): aborting due to no ready subscriptions");
150
146
  }
151
147
  return;
@@ -160,31 +156,23 @@ export class ObjectSetListenerWebsocket {
160
156
  id,
161
157
  requests: readySubs.map(({
162
158
  objectSet,
163
- requestedProperties
164
- }) => ({
165
- objectSet: objectSet,
166
- propertySet: requestedProperties,
167
- referenceSet: []
168
- }))
159
+ requestedProperties,
160
+ requestedReferenceProperties
161
+ }) => {
162
+ return {
163
+ objectSet: objectSet,
164
+ propertySet: requestedProperties,
165
+ referenceSet: requestedReferenceProperties
166
+ };
167
+ })
169
168
  };
170
- if (process?.env?.NODE_ENV !== "production") {
169
+ if (process.env.NODE_ENV !== "production") {
171
170
  this.#logger?.trace({
172
171
  payload: subscribe
173
172
  }, "sending subscribe message");
174
173
  }
175
174
  this.#ws?.send(JSON.stringify(subscribe));
176
175
  }
177
- #expire(sub) {
178
- if (process?.env?.NODE_ENV !== "production") {
179
- this.#logger?.trace({
180
- subscription: sub
181
- }, "#expire()");
182
- }
183
- // the temporary ObjectSet has expired, we should re-subscribe which will cause the
184
- // listener to get an onOutOfDate message when it becomes subscribed again
185
- sub.status = "expired";
186
- this.#initiateSubscribe(sub);
187
- }
188
176
  #unsubscribe(sub, newStatus = "done") {
189
177
  if (subscriptionIsDone(sub)) {
190
178
  // if we are already done, we don't need to do anything
@@ -193,11 +181,8 @@ export class ObjectSetListenerWebsocket {
193
181
  sub.status = newStatus;
194
182
  // make sure listeners do nothing now
195
183
  sub.listener = fillOutListener({});
196
- if (sub.expiry) {
197
- clearTimeout(sub.expiry);
198
- sub.expiry = undefined;
199
- }
200
184
  this.#subscriptions.delete(sub.subscriptionId);
185
+ this.#sendSubscribeMessage();
201
186
  // If we have no more subscriptions, we can disconnect the websocket
202
187
  // however we should wait a bit to see if we get any more subscriptions.
203
188
  // For example, when switching between react views, you may unsubscribe
@@ -244,7 +229,7 @@ export class ObjectSetListenerWebsocket {
244
229
  this.#lastWsConnect = Date.now();
245
230
  // we again may have lost the race after our minimum backoff time
246
231
  if (this.#ws == null) {
247
- if (process?.env?.NODE_ENV !== "production") {
232
+ if (process.env.NODE_ENV !== "production") {
248
233
  this.#logger?.trace("Creating websocket");
249
234
  }
250
235
  this.#ws = new WebSocket(url, [`Bearer-${token}`]);
@@ -284,7 +269,7 @@ export class ObjectSetListenerWebsocket {
284
269
  };
285
270
  #onMessage = async message => {
286
271
  const data = JSON.parse(message.data.toString());
287
- if (process?.env?.NODE_ENV !== "production") {
272
+ if (process.env.NODE_ENV !== "production") {
288
273
  this.#logger?.trace({
289
274
  payload: data
290
275
  }, "received message from ws");
@@ -301,15 +286,37 @@ export class ObjectSetListenerWebsocket {
301
286
  return this.#handleMessage_subscriptionClosed(data);
302
287
  }
303
288
  default:
304
- invariant(false, "Unexpected message type");
289
+ process.env.NODE_ENV !== "production" ? invariant(false, "Unexpected message type") : invariant(false);
305
290
  }
306
291
  };
307
292
  #handleMessage_objectSetChanged = async payload => {
308
293
  const sub = this.#subscriptions.get(payload.id);
309
- !sub ? invariant(false, `Expected subscription id ${payload.id}`) : void 0;
294
+ if (sub == null) return;
310
295
  const objectUpdates = payload.updates.filter(update => update.type === "object");
311
- payload.updates.filter(update => update.type === "reference");
296
+ const referenceUpdates = payload.updates.filter(update => update.type === "reference");
297
+ const osdkObjectsWithReferenceUpdates = await Promise.all(referenceUpdates.map(async o => {
298
+ const osdkObjectArray = await convertWireToOsdkObjects(this.#client, [{
299
+ __apiName: o.objectType,
300
+ __primaryKey: o.primaryKey[sub.primaryKeyPropertyName],
301
+ ...o.primaryKey,
302
+ [o.property]: o.value
303
+ }], undefined);
304
+ const singleOsdkObject = osdkObjectArray[0] ?? undefined;
305
+ return singleOsdkObject != null ? {
306
+ object: singleOsdkObject,
307
+ state: "ADDED_OR_UPDATED"
308
+ } : undefined;
309
+ }));
310
+ for (const osdkObject of osdkObjectsWithReferenceUpdates) {
311
+ if (osdkObject != null) {
312
+ sub.listener.onChange?.(osdkObject);
313
+ }
314
+ }
312
315
  const osdkObjects = await Promise.all(objectUpdates.map(async o => {
316
+ const keysToDelete = Object.keys(o.object).filter(key => sub.requestedReferenceProperties.includes(key));
317
+ for (const key of keysToDelete) {
318
+ delete o.object[key];
319
+ }
313
320
  const osdkObjectArray = await convertWireToOsdkObjects(this.#client, [o.object], undefined);
314
321
  const singleOsdkObject = osdkObjectArray[0] ?? undefined;
315
322
  return singleOsdkObject != null ? {
@@ -319,13 +326,13 @@ export class ObjectSetListenerWebsocket {
319
326
  }));
320
327
  for (const osdkObject of osdkObjects) {
321
328
  if (osdkObject != null) {
322
- return sub.listener.onChange?.(osdkObject);
329
+ sub.listener.onChange?.(osdkObject);
323
330
  }
324
331
  }
325
332
  };
326
333
  #handleMessage_refreshObjectSet = payload => {
327
334
  const sub = this.#subscriptions.get(payload.id);
328
- !sub ? invariant(false, `Expected subscription id ${payload.id}`) : void 0;
335
+ !sub ? process.env.NODE_ENV !== "production" ? invariant(false, `Expected subscription id ${payload.id}`) : invariant(false) : void 0;
329
336
  sub.listener.onOutOfDate();
330
337
  };
331
338
  #handleMessage_subscribeResponses = payload => {
@@ -334,7 +341,7 @@ export class ObjectSetListenerWebsocket {
334
341
  responses
335
342
  } = payload;
336
343
  const subs = this.#pendingSubscriptions.get(id);
337
- !subs ? invariant(false, `should have a pending subscription for ${id}`) : void 0;
344
+ !subs ? process.env.NODE_ENV !== "production" ? invariant(false, `should have a pending subscription for ${id}`) : invariant(false) : void 0;
338
345
  this.#pendingSubscriptions.delete(id);
339
346
  for (let i = 0; i < responses.length; i++) {
340
347
  const sub = subs[i];
@@ -351,7 +358,7 @@ export class ObjectSetListenerWebsocket {
351
358
  case "success":
352
359
  // `"preparing"` should only be the status on an initial subscribe.
353
360
  const shouldFireOutOfDate = sub.status === "expired" || sub.status === "reconnecting";
354
- if (process?.env?.NODE_ENV !== "production") {
361
+ if (process.env.NODE_ENV !== "production") {
355
362
  this.#logger?.trace({
356
363
  shouldFireOutOfDate
357
364
  }, "success");
@@ -363,7 +370,7 @@ export class ObjectSetListenerWebsocket {
363
370
  sub.subscriptionId = response.id;
364
371
  this.#subscriptions.set(sub.subscriptionId, sub); // future messages come by this subId
365
372
  }
366
- if (shouldFireOutOfDate) sub.listener.onOutOfDate();
373
+ if (shouldFireOutOfDate) sub.listener.onOutOfDate();else sub.listener.onSuccessfulSubscription();
367
374
  break;
368
375
  default:
369
376
  sub.listener.onError(response);
@@ -372,12 +379,12 @@ export class ObjectSetListenerWebsocket {
372
379
  };
373
380
  #handleMessage_subscriptionClosed(payload) {
374
381
  const sub = this.#subscriptions.get(payload.id);
375
- !sub ? invariant(false, `Expected subscription id ${payload.id}`) : void 0;
382
+ !sub ? process.env.NODE_ENV !== "production" ? invariant(false, `Expected subscription id ${payload.id}`) : invariant(false) : void 0;
376
383
  sub.listener.onError([payload.cause]);
377
384
  this.#unsubscribe(sub, "error");
378
385
  }
379
386
  #onClose = event => {
380
- if (process?.env?.NODE_ENV !== "production") {
387
+ if (process.env.NODE_ENV !== "production") {
381
388
  this.#logger?.trace({
382
389
  event
383
390
  }, "Received close event from ws", event);
@@ -385,8 +392,6 @@ export class ObjectSetListenerWebsocket {
385
392
  // TODO we should probably throttle this so we don't abuse the backend
386
393
  this.#cycleWebsocket();
387
394
  };
388
- // TODO: Validate if this is needed
389
- async #enableObjectSetsWatcher() {}
390
395
  #cycleWebsocket = () => {
391
396
  if (this.#ws) {
392
397
  this.#ws.removeEventListener("open", this.#onOpen);
@@ -399,9 +404,9 @@ export class ObjectSetListenerWebsocket {
399
404
  }
400
405
  // if we have any listeners that are still depending on us, go ahead and reopen the websocket
401
406
  if (this.#subscriptions.size > 0) {
402
- if (process?.env?.NODE_ENV !== "production") {
407
+ if (process.env.NODE_ENV !== "production") {
403
408
  for (const s of this.#subscriptions.values()) {
404
- !(s.status !== "done" && s.status !== "error") ? invariant(false, "should not have done/error subscriptions still") : void 0;
409
+ !(s.status !== "done" && s.status !== "error") ? process.env.NODE_ENV !== "production" ? invariant(false, "should not have done/error subscriptions still") : invariant(false) : void 0;
405
410
  }
406
411
  }
407
412
  for (const s of this.#subscriptions.values()) {
@@ -1 +1 @@
1
- {"version":3,"file":"ObjectSetListenerWebsocket.js","names":["WebSocket","invariant","convertWireToOsdkObjects","ONE_DAY_MS","MINIMUM_RECONNECT_DELAY_MS","doNothing","fillOutListener","onChange","onError","onOutOfDate","isReady","sub","subscriptionIsDone","status","ObjectSetListenerWebsocket","instances","WeakMap","OBJECT_SET_EXPIRY_MS","getInstance","client","instance","get","clientCacheKey","set","ws","lastWsConnect","logger","pendingSubscriptions","Map","subscriptions","maybeDisconnectTimeout","constructor","objectSetExpiryMs","minimumReconnectDelayMs","child","msgPrefix","baseUrl","startsWith","subscribe","objectSet","listener","properties","requestedProperties","subscriptionId","crypto","randomUUID","initiateSubscribe","unsubscribe","#initiateSubscribe","process","env","NODE_ENV","trace","expiry","clearTimeout","setTimeout","expire","ontologyRid","ensureWebsocket","readyState","OPEN","sendSubscribeMessage","error","#sendSubscribeMessage","readySubs","values","filter","length","id","requests","map","propertySet","referenceSet","payload","send","JSON","stringify","#expire","subscription","#unsubscribe","newStatus","undefined","delete","size","cycleWebsocket","#ensureWebsocket","tokenProvider","base","URL","url","host","token","nextConnectTime","Date","now","Promise","resolve","addEventListener","onClose","onMessage","onOpen","CONNECTING","reject","cleanup","removeEventListener","open","evt","#onOpen","message","data","parse","toString","type","handleMessage_objectSetChanged","handleMessage_refreshObjectSet","handleMessage_subscribeResponses","handleMessage_subscriptionClosed","objectUpdates","updates","update","osdkObjects","all","o","osdkObjectArray","object","singleOsdkObject","state","osdkObject","responses","subs","i","response","errors","shouldFireOutOfDate","#handleMessage_subscriptionClosed","cause","event","enableObjectSetsWatcher","#enableObjectSetsWatcher","#cycleWebsocket","CLOSING","CLOSED","close","s"],"sources":["ObjectSetListenerWebsocket.js"],"sourcesContent":["/*\n * Copyright 2023 Palantir Technologies, Inc. All rights reserved.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\nimport WebSocket from \"isomorphic-ws\";\nimport invariant from \"tiny-invariant\";\nimport { convertWireToOsdkObjects } from \"../object/convertWireToOsdkObjects.js\";\nconst ONE_DAY_MS = 24 * 60 * 60 * 1000;\nconst MINIMUM_RECONNECT_DELAY_MS = 5 * 1000;\n/** Noop function to reduce conditional checks */\nfunction doNothing() { }\n/**\n * Converts an ObjectSetListener to one where all the functions are defined.\n */\nfunction fillOutListener({ onChange = doNothing, onError = doNothing, onOutOfDate = doNothing, }) {\n return { onChange, onError, onOutOfDate };\n}\nfunction isReady(sub) {\n return sub.isReady != null;\n}\nfunction subscriptionIsDone(sub) {\n return sub.status === \"done\" || sub.status === \"error\";\n}\n/** @internal */\nexport class ObjectSetListenerWebsocket {\n static #instances = new WeakMap();\n OBJECT_SET_EXPIRY_MS;\n MINIMUM_RECONNECT_DELAY_MS;\n // FIXME\n static getInstance(client) {\n let instance = ObjectSetListenerWebsocket.#instances.get(client.clientCacheKey);\n if (instance == null) {\n instance = new ObjectSetListenerWebsocket(client);\n ObjectSetListenerWebsocket.#instances.set(client.clientCacheKey, instance);\n }\n return instance;\n }\n #ws;\n #lastWsConnect = 0;\n #client;\n #logger;\n /**\n * map of requestId to all active subscriptions at the time of the request\n */\n #pendingSubscriptions = new Map();\n /**\n * Map of subscriptionId to Subscription. Note: the subscriptionId may be\n * temporary and not the actual subscriptionId from the server.\n */\n #subscriptions = new Map();\n #maybeDisconnectTimeout;\n // DO NOT CONSTRUCT DIRECTLY. ONLY EXPOSED AS A TESTING SEAM\n constructor(client, { objectSetExpiryMs = ONE_DAY_MS, minimumReconnectDelayMs = MINIMUM_RECONNECT_DELAY_MS, } = {}) {\n this.OBJECT_SET_EXPIRY_MS = objectSetExpiryMs;\n this.MINIMUM_RECONNECT_DELAY_MS = minimumReconnectDelayMs;\n this.#client = client;\n this.#logger = client.logger?.child({}, {\n msgPrefix: \"<OSW> \",\n });\n invariant(client.baseUrl.startsWith(\"https://\")\n || client.baseUrl.startsWith(\"http://\"), \"Stack must be a URL\");\n }\n async subscribe(objectSet, listener, properties) {\n if (process.env.TARGET !== \"browser\") {\n // Node 18 does not expose 'crypto' on globalThis, so we need to do it ourselves. This\n // will not be needed after our minimum version is 19 or greater.\n globalThis.crypto ??= (await import(\"node:crypto\")).webcrypto;\n }\n const sub = {\n listener: fillOutListener(listener),\n objectSet,\n requestedProperties: properties,\n status: \"preparing\",\n // Since we don't have a real subscription id yet but we need to keep\n // track of this reference, we can just use a random uuid.\n subscriptionId: `TMP-${crypto.randomUUID()}`,\n };\n this.#subscriptions.set(sub.subscriptionId, sub);\n // actually prepares the subscription, ensures the ws is ready, and sends\n // a subscribe message. We don't want to block on this.\n this.#initiateSubscribe(sub);\n return () => {\n this.#unsubscribe(sub);\n };\n }\n /**\n * Called at least once for every subscription.\n *\n * - Resets pending expiry\n * - Recreates temporary object set\n * - Triggers a full subscribe message\n *\n * @returns\n */\n async #initiateSubscribe(sub) {\n if (process?.env?.NODE_ENV !== \"production\") {\n this.#logger?.trace(\"#initiateSubscribe()\");\n }\n if (sub.expiry) {\n clearTimeout(sub.expiry);\n }\n // expiry is tied to the temporary object set, which is set to `timeToLive: \"ONE_DAY\"`\n // in `#createTemporaryObjectSet`. They should be in sync\n sub.expiry = setTimeout(() => this.#expire(sub), this.OBJECT_SET_EXPIRY_MS);\n const ontologyRid = await this.#client.ontologyRid;\n try {\n await this.#ensureWebsocket();\n // the consumer may have already unsubscribed before we are ready to request a subscription\n // so we have to acquire the pendingSubscription after the await.\n if (subscriptionIsDone(sub)) {\n return;\n }\n sub.isReady = true;\n // if we aren't open, then this happens after we #onConnect\n if (this.#ws?.readyState === WebSocket.OPEN) {\n this.#sendSubscribeMessage();\n }\n }\n catch (error) {\n this.#logger?.error(error, \"Error in #initiateSubscribe\");\n sub.listener.onError([error]);\n }\n }\n #sendSubscribeMessage() {\n if (process?.env?.NODE_ENV !== \"production\") {\n this.#logger?.trace(\"#sendSubscribeMessage()\");\n }\n // If two calls to `.subscribe()` happen at once (or if the connection is reset),\n // we may have multiple subscriptions that don't have a subscriptionId yet,\n // so we filter those out.\n const readySubs = [...this.#subscriptions.values()].filter(isReady);\n if (readySubs.length === 0) {\n if (process?.env?.NODE_ENV !== \"production\") {\n this.#logger?.trace(\"#sendSubscribeMessage(): aborting due to no ready subscriptions\");\n }\n return;\n }\n // Assumes the node 18 crypto fallback to globalThis in `subscribe` has happened.\n const id = crypto.randomUUID();\n // responses come back as an array of subIds, so we need to know the sources\n this.#pendingSubscriptions.set(id, readySubs);\n // every subscribe message \"overwrites\" the previous ones that are not\n // re-included, so we have to reconstitute the entire list of subscriptions\n const subscribe = {\n id,\n requests: readySubs.map(({ objectSet, requestedProperties }) => ({\n objectSet: objectSet,\n propertySet: requestedProperties,\n referenceSet: [],\n })),\n };\n if (process?.env?.NODE_ENV !== \"production\") {\n this.#logger?.trace({ payload: subscribe }, \"sending subscribe message\");\n }\n this.#ws?.send(JSON.stringify(subscribe));\n }\n #expire(sub) {\n if (process?.env?.NODE_ENV !== \"production\") {\n this.#logger?.trace({ subscription: sub }, \"#expire()\");\n }\n // the temporary ObjectSet has expired, we should re-subscribe which will cause the\n // listener to get an onOutOfDate message when it becomes subscribed again\n sub.status = \"expired\";\n this.#initiateSubscribe(sub);\n }\n #unsubscribe(sub, newStatus = \"done\") {\n if (subscriptionIsDone(sub)) {\n // if we are already done, we don't need to do anything\n return;\n }\n sub.status = newStatus;\n // make sure listeners do nothing now\n sub.listener = fillOutListener({});\n if (sub.expiry) {\n clearTimeout(sub.expiry);\n sub.expiry = undefined;\n }\n this.#subscriptions.delete(sub.subscriptionId);\n // If we have no more subscriptions, we can disconnect the websocket\n // however we should wait a bit to see if we get any more subscriptions.\n // For example, when switching between react views, you may unsubscribe\n // in the old view and subscribe in the new view. We don't need to re-establish\n // the websocket connection in that case.\n if (this.#maybeDisconnectTimeout) {\n // We reset the timeout on every unsubscribe so its always at least 15s from\n // the last time we are empty. E.g.:\n // - 0s: Subscribe(A)\n // - 10s: Unsubscribe(A)\n // - 11s: Subscribe(B)\n // - 20s: Unsubscribe(B)\n // If we do not clear out the timeout we would disconnect at 25s but that would only be\n // 5s after the last subscription was removed instead of at 35s for the desired 15s.\n clearTimeout(this.#maybeDisconnectTimeout);\n }\n this.#maybeDisconnectTimeout = setTimeout(() => {\n this.#maybeDisconnectTimeout = undefined;\n if (this.#subscriptions.size === 0) {\n this.#cycleWebsocket();\n }\n }, 15_000 /* ms */);\n }\n async #ensureWebsocket() {\n if (this.#ws == null) {\n const { baseUrl, tokenProvider } = this.#client;\n const base = new URL(baseUrl);\n const url = `wss://${base.host}/api/v2/ontologySubscriptions/ontologies/${this.#client.ontologyRid}/streamSubscriptions`;\n const token = await tokenProvider();\n // tokenProvider is async, there could potentially be a race to create the websocket.\n // Only the first call to reach here will find a null this.#ws, the rest will bail out\n if (this.#ws == null) {\n // TODO this can probably be exponential backoff with jitter\n // don't reconnect more quickly than MINIMUM_RECONNECT_DELAY\n const nextConnectTime = (this.#lastWsConnect ?? 0)\n + this.MINIMUM_RECONNECT_DELAY_MS;\n if (nextConnectTime > Date.now()) {\n await new Promise((resolve) => {\n setTimeout(resolve, nextConnectTime - Date.now());\n });\n }\n this.#lastWsConnect = Date.now();\n // we again may have lost the race after our minimum backoff time\n if (this.#ws == null) {\n if (process?.env?.NODE_ENV !== \"production\") {\n this.#logger?.trace(\"Creating websocket\");\n }\n this.#ws = new WebSocket(url, [`Bearer-${token}`]);\n this.#ws.addEventListener(\"close\", this.#onClose);\n this.#ws.addEventListener(\"message\", this.#onMessage);\n this.#ws.addEventListener(\"open\", this.#onOpen);\n }\n }\n // Allow await-ing the websocket open event if it isn't open already.\n // This needs to happen even for callers that didn't just create this.#ws\n if (this.#ws.readyState === WebSocket.CONNECTING) {\n const ws = this.#ws;\n return new Promise((resolve, reject) => {\n function cleanup() {\n ws.removeEventListener(\"open\", open);\n ws.removeEventListener(\"error\", error);\n ws.removeEventListener(\"close\", cleanup);\n }\n function open() {\n cleanup();\n resolve();\n }\n function error(evt) {\n cleanup();\n reject(evt);\n }\n ws.addEventListener(\"open\", open);\n ws.addEventListener(\"error\", error);\n ws.addEventListener(\"close\", cleanup);\n });\n }\n }\n }\n #onOpen = () => {\n // resubscribe all of the listeners\n this.#sendSubscribeMessage();\n };\n #onMessage = async (message) => {\n const data = JSON.parse(message.data.toString());\n if (process?.env?.NODE_ENV !== \"production\") {\n this.#logger?.trace({ payload: data }, \"received message from ws\");\n }\n switch (data.type) {\n case \"objectSetChanged\":\n return this.#handleMessage_objectSetChanged(data);\n case \"refreshObjectSet\":\n return this.#handleMessage_refreshObjectSet(data);\n case \"subscribeResponses\":\n return this.#handleMessage_subscribeResponses(data);\n case \"subscriptionClosed\": {\n return this.#handleMessage_subscriptionClosed(data);\n }\n default:\n const _ = data;\n invariant(false, \"Unexpected message type\");\n }\n };\n #handleMessage_objectSetChanged = async (payload) => {\n const sub = this.#subscriptions.get(payload.id);\n invariant(sub, `Expected subscription id ${payload.id}`);\n const objectUpdates = payload.updates.filter((update) => update.type === \"object\");\n const referenceUpdates = payload.updates.filter((update) => update.type === \"reference\");\n const osdkObjects = await Promise.all(objectUpdates.map(async (o) => {\n const osdkObjectArray = await convertWireToOsdkObjects(this.#client, [o.object], undefined);\n const singleOsdkObject = osdkObjectArray[0] ?? undefined;\n return singleOsdkObject != null\n ? {\n object: singleOsdkObject,\n state: o.state,\n }\n : undefined;\n }));\n for (const osdkObject of osdkObjects) {\n if (osdkObject != null) {\n return sub.listener.onChange?.(osdkObject);\n }\n }\n };\n #handleMessage_refreshObjectSet = (payload) => {\n const sub = this.#subscriptions.get(payload.id);\n invariant(sub, `Expected subscription id ${payload.id}`);\n sub.listener.onOutOfDate();\n };\n #handleMessage_subscribeResponses = (payload) => {\n const { id, responses } = payload;\n const subs = this.#pendingSubscriptions.get(id);\n invariant(subs, `should have a pending subscription for ${id}`);\n this.#pendingSubscriptions.delete(id);\n for (let i = 0; i < responses.length; i++) {\n const sub = subs[i];\n const response = responses[i];\n switch (response.type) {\n case \"error\":\n sub.listener.onError(response.errors);\n this.#unsubscribe(sub, \"error\");\n break;\n case \"qos\":\n // the server has requested that we tear down our websocket and reconnect to help load balance\n this.#cycleWebsocket();\n break;\n case \"success\":\n // `\"preparing\"` should only be the status on an initial subscribe.\n const shouldFireOutOfDate = sub.status === \"expired\"\n || sub.status === \"reconnecting\";\n if (process?.env?.NODE_ENV !== \"production\") {\n this.#logger?.trace({ shouldFireOutOfDate }, \"success\");\n }\n sub.status = \"subscribed\";\n if (sub.subscriptionId !== response.id) {\n // might be the temporary one\n this.#subscriptions.delete(sub.subscriptionId);\n sub.subscriptionId = response.id;\n this.#subscriptions.set(sub.subscriptionId, sub); // future messages come by this subId\n }\n if (shouldFireOutOfDate)\n sub.listener.onOutOfDate();\n break;\n default:\n const _ = response;\n sub.listener.onError(response);\n }\n }\n };\n #handleMessage_subscriptionClosed(payload) {\n const sub = this.#subscriptions.get(payload.id);\n invariant(sub, `Expected subscription id ${payload.id}`);\n sub.listener.onError([payload.cause]);\n this.#unsubscribe(sub, \"error\");\n }\n #onClose = (event) => {\n if (process?.env?.NODE_ENV !== \"production\") {\n this.#logger?.trace({ event }, \"Received close event from ws\", event);\n }\n // TODO we should probably throttle this so we don't abuse the backend\n this.#cycleWebsocket();\n };\n // TODO: Validate if this is needed\n async #enableObjectSetsWatcher(objectTypeRids) {\n // return batchEnableWatcher(this.#oswContext, {\n // requests: objectTypeRids,\n // });\n return;\n }\n #cycleWebsocket = () => {\n if (this.#ws) {\n this.#ws.removeEventListener(\"open\", this.#onOpen);\n this.#ws.removeEventListener(\"message\", this.#onMessage);\n this.#ws.removeEventListener(\"close\", this.#onClose);\n if (this.#ws.readyState !== WebSocket.CLOSING\n && this.#ws.readyState !== WebSocket.CLOSED) {\n this.#ws.close();\n }\n this.#ws = undefined;\n }\n // if we have any listeners that are still depending on us, go ahead and reopen the websocket\n if (this.#subscriptions.size > 0) {\n if (process?.env?.NODE_ENV !== \"production\") {\n for (const s of this.#subscriptions.values()) {\n invariant(s.status !== \"done\" && s.status !== \"error\", \"should not have done/error subscriptions still\");\n }\n }\n for (const s of this.#subscriptions.values()) {\n if (s.status === \"subscribed\")\n s.status = \"reconnecting\";\n }\n this.#ensureWebsocket();\n }\n };\n}\n"],"mappings":"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","ignoreList":[]}
1
+ {"version":3,"file":"ObjectSetListenerWebsocket.js","names":["WebSocket","invariant","convertWireToOsdkObjects","MINIMUM_RECONNECT_DELAY_MS","doNothing","fillOutListener","onChange","onError","onOutOfDate","onSuccessfulSubscription","isReady","sub","subscriptionIsDone","status","ObjectSetListenerWebsocket","instances","WeakMap","getInstance","client","instance","get","clientCacheKey","set","ws","lastWsConnect","logger","pendingSubscriptions","Map","subscriptions","maybeDisconnectTimeout","constructor","minimumReconnectDelayMs","child","msgPrefix","baseUrl","startsWith","process","env","NODE_ENV","subscribe","objectType","objectSet","listener","properties","objDef","ontologyProvider","getObjectDefinition","apiName","objectProperties","filter","p","type","referenceProperties","primaryKeyPropertyName","primaryKeyApiName","requestedProperties","requestedReferenceProperties","subscriptionId","crypto","randomUUID","initiateSubscribe","unsubscribe","#initiateSubscribe","trace","ensureWebsocket","readyState","OPEN","sendSubscribeMessage","error","#sendSubscribeMessage","readySubs","values","length","id","requests","map","propertySet","referenceSet","payload","send","JSON","stringify","#unsubscribe","newStatus","delete","clearTimeout","setTimeout","undefined","size","cycleWebsocket","#ensureWebsocket","tokenProvider","base","URL","url","host","ontologyRid","token","nextConnectTime","Date","now","Promise","resolve","addEventListener","onClose","onMessage","onOpen","CONNECTING","reject","cleanup","removeEventListener","open","evt","#onOpen","message","data","parse","toString","handleMessage_objectSetChanged","handleMessage_refreshObjectSet","handleMessage_subscribeResponses","handleMessage_subscriptionClosed","objectUpdates","updates","update","referenceUpdates","osdkObjectsWithReferenceUpdates","all","o","osdkObjectArray","__apiName","__primaryKey","primaryKey","property","value","singleOsdkObject","object","state","osdkObject","osdkObjects","keysToDelete","Object","keys","key","includes","responses","subs","i","response","errors","shouldFireOutOfDate","#handleMessage_subscriptionClosed","cause","event","#cycleWebsocket","CLOSING","CLOSED","close","s"],"sources":["ObjectSetListenerWebsocket.js"],"sourcesContent":["/*\n * Copyright 2023 Palantir Technologies, Inc. All rights reserved.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\nimport WebSocket from \"isomorphic-ws\";\nimport invariant from \"tiny-invariant\";\nimport { convertWireToOsdkObjects } from \"../object/convertWireToOsdkObjects.js\";\nconst MINIMUM_RECONNECT_DELAY_MS = 5 * 1000;\n/** Noop function to reduce conditional checks */\nfunction doNothing() { }\n/**\n * Converts an ObjectSetListener to one where all the functions are defined.\n */\nfunction fillOutListener({ onChange = doNothing, onError = doNothing, onOutOfDate = doNothing, onSuccessfulSubscription = doNothing, }) {\n return { onChange, onError, onOutOfDate, onSuccessfulSubscription };\n}\nfunction isReady(sub) {\n return sub.isReady != null;\n}\nfunction subscriptionIsDone(sub) {\n return sub.status === \"done\" || sub.status === \"error\";\n}\n/** @internal */\nexport class ObjectSetListenerWebsocket {\n static #instances = new WeakMap();\n MINIMUM_RECONNECT_DELAY_MS;\n // FIXME\n static getInstance(client) {\n let instance = ObjectSetListenerWebsocket.#instances.get(client.clientCacheKey);\n if (instance == null) {\n instance = new ObjectSetListenerWebsocket(client);\n ObjectSetListenerWebsocket.#instances.set(client.clientCacheKey, instance);\n }\n return instance;\n }\n #ws;\n #lastWsConnect = 0;\n #client;\n #logger;\n /**\n * map of requestId to all active subscriptions at the time of the request\n */\n #pendingSubscriptions = new Map();\n /**\n * Map of subscriptionId to Subscription. Note: the subscriptionId may be\n * temporary and not the actual subscriptionId from the server.\n */\n #subscriptions = new Map();\n #maybeDisconnectTimeout;\n // DO NOT CONSTRUCT DIRECTLY. ONLY EXPOSED AS A TESTING SEAM\n constructor(client, { minimumReconnectDelayMs = MINIMUM_RECONNECT_DELAY_MS, } = {}) {\n this.MINIMUM_RECONNECT_DELAY_MS = minimumReconnectDelayMs;\n this.#client = client;\n this.#logger = client.logger?.child({}, {\n msgPrefix: \"<OSW> \",\n });\n invariant(client.baseUrl.startsWith(\"https://\")\n || client.baseUrl.startsWith(\"http://\"), \"Stack must be a URL\");\n }\n async subscribe(objectType, objectSet, listener, properties) {\n if (process.env.TARGET !== \"browser\") {\n // Node 18 does not expose 'crypto' on globalThis, so we need to do it ourselves. This\n // will not be needed after our minimum version is 19 or greater.\n globalThis.crypto ??= (await import(\"node:crypto\")).webcrypto;\n }\n const objDef = await this.#client.ontologyProvider.getObjectDefinition(objectType.apiName);\n const objectProperties = properties.filter((p) => objDef.properties[p].type !== \"geotimeSeriesReference\");\n const referenceProperties = properties.filter((p) => objDef.properties[p].type === \"geotimeSeriesReference\");\n const sub = {\n listener: fillOutListener(listener),\n objectSet,\n primaryKeyPropertyName: objDef.primaryKeyApiName,\n requestedProperties: objectProperties,\n requestedReferenceProperties: referenceProperties,\n status: \"preparing\",\n // Since we don't have a real subscription id yet but we need to keep\n // track of this reference, we can just use a random uuid.\n subscriptionId: `TMP-${crypto.randomUUID()}`,\n };\n this.#subscriptions.set(sub.subscriptionId, sub);\n // actually prepares the subscription, ensures the ws is ready, and sends\n // a subscribe message. We don't want to block on this.\n this.#initiateSubscribe(sub);\n return () => {\n this.#unsubscribe(sub);\n };\n }\n /**\n * Called at least once for every subscription.\n *\n * - Resets pending expiry\n * - Recreates temporary object set\n * - Triggers a full subscribe message\n *\n * @returns\n */\n async #initiateSubscribe(sub) {\n if (process.env.NODE_ENV !== \"production\") {\n this.#logger?.trace(\"#initiateSubscribe()\");\n }\n try {\n await this.#ensureWebsocket();\n // the consumer may have already unsubscribed before we are ready to request a subscription\n // so we have to acquire the pendingSubscription after the await.\n if (subscriptionIsDone(sub)) {\n return;\n }\n sub.isReady = true;\n // if we aren't open, then this happens after we #onConnect\n if (this.#ws?.readyState === WebSocket.OPEN) {\n this.#sendSubscribeMessage();\n }\n }\n catch (error) {\n this.#logger?.error(error, \"Error in #initiateSubscribe\");\n sub.listener.onError([error]);\n }\n }\n #sendSubscribeMessage() {\n if (process.env.NODE_ENV !== \"production\") {\n this.#logger?.trace(\"#sendSubscribeMessage()\");\n }\n // If two calls to `.subscribe()` happen at once (or if the connection is reset),\n // we may have multiple subscriptions that don't have a subscriptionId yet,\n // so we filter those out.\n const readySubs = [...this.#subscriptions.values()].filter(isReady);\n if (readySubs.length === 0) {\n if (process.env.NODE_ENV !== \"production\") {\n this.#logger?.trace(\"#sendSubscribeMessage(): aborting due to no ready subscriptions\");\n }\n return;\n }\n // Assumes the node 18 crypto fallback to globalThis in `subscribe` has happened.\n const id = crypto.randomUUID();\n // responses come back as an array of subIds, so we need to know the sources\n this.#pendingSubscriptions.set(id, readySubs);\n // every subscribe message \"overwrites\" the previous ones that are not\n // re-included, so we have to reconstitute the entire list of subscriptions\n const subscribe = {\n id,\n requests: readySubs.map(({ objectSet, requestedProperties, requestedReferenceProperties }) => {\n return {\n objectSet: objectSet,\n propertySet: requestedProperties,\n referenceSet: requestedReferenceProperties,\n };\n }),\n };\n if (process.env.NODE_ENV !== \"production\") {\n this.#logger?.trace({ payload: subscribe }, \"sending subscribe message\");\n }\n this.#ws?.send(JSON.stringify(subscribe));\n }\n #unsubscribe(sub, newStatus = \"done\") {\n if (subscriptionIsDone(sub)) {\n // if we are already done, we don't need to do anything\n return;\n }\n sub.status = newStatus;\n // make sure listeners do nothing now\n sub.listener = fillOutListener({});\n this.#subscriptions.delete(sub.subscriptionId);\n this.#sendSubscribeMessage();\n // If we have no more subscriptions, we can disconnect the websocket\n // however we should wait a bit to see if we get any more subscriptions.\n // For example, when switching between react views, you may unsubscribe\n // in the old view and subscribe in the new view. We don't need to re-establish\n // the websocket connection in that case.\n if (this.#maybeDisconnectTimeout) {\n // We reset the timeout on every unsubscribe so its always at least 15s from\n // the last time we are empty. E.g.:\n // - 0s: Subscribe(A)\n // - 10s: Unsubscribe(A)\n // - 11s: Subscribe(B)\n // - 20s: Unsubscribe(B)\n // If we do not clear out the timeout we would disconnect at 25s but that would only be\n // 5s after the last subscription was removed instead of at 35s for the desired 15s.\n clearTimeout(this.#maybeDisconnectTimeout);\n }\n this.#maybeDisconnectTimeout = setTimeout(() => {\n this.#maybeDisconnectTimeout = undefined;\n if (this.#subscriptions.size === 0) {\n this.#cycleWebsocket();\n }\n }, 15_000 /* ms */);\n }\n async #ensureWebsocket() {\n if (this.#ws == null) {\n const { baseUrl, tokenProvider } = this.#client;\n const base = new URL(baseUrl);\n const url = `wss://${base.host}/api/v2/ontologySubscriptions/ontologies/${this.#client.ontologyRid}/streamSubscriptions`;\n const token = await tokenProvider();\n // tokenProvider is async, there could potentially be a race to create the websocket.\n // Only the first call to reach here will find a null this.#ws, the rest will bail out\n if (this.#ws == null) {\n // TODO this can probably be exponential backoff with jitter\n // don't reconnect more quickly than MINIMUM_RECONNECT_DELAY\n const nextConnectTime = (this.#lastWsConnect ?? 0)\n + this.MINIMUM_RECONNECT_DELAY_MS;\n if (nextConnectTime > Date.now()) {\n await new Promise((resolve) => {\n setTimeout(resolve, nextConnectTime - Date.now());\n });\n }\n this.#lastWsConnect = Date.now();\n // we again may have lost the race after our minimum backoff time\n if (this.#ws == null) {\n if (process.env.NODE_ENV !== \"production\") {\n this.#logger?.trace(\"Creating websocket\");\n }\n this.#ws = new WebSocket(url, [`Bearer-${token}`]);\n this.#ws.addEventListener(\"close\", this.#onClose);\n this.#ws.addEventListener(\"message\", this.#onMessage);\n this.#ws.addEventListener(\"open\", this.#onOpen);\n }\n }\n // Allow await-ing the websocket open event if it isn't open already.\n // This needs to happen even for callers that didn't just create this.#ws\n if (this.#ws.readyState === WebSocket.CONNECTING) {\n const ws = this.#ws;\n return new Promise((resolve, reject) => {\n function cleanup() {\n ws.removeEventListener(\"open\", open);\n ws.removeEventListener(\"error\", error);\n ws.removeEventListener(\"close\", cleanup);\n }\n function open() {\n cleanup();\n resolve();\n }\n function error(evt) {\n cleanup();\n reject(evt);\n }\n ws.addEventListener(\"open\", open);\n ws.addEventListener(\"error\", error);\n ws.addEventListener(\"close\", cleanup);\n });\n }\n }\n }\n #onOpen = () => {\n // resubscribe all of the listeners\n this.#sendSubscribeMessage();\n };\n #onMessage = async (message) => {\n const data = JSON.parse(message.data.toString());\n if (process.env.NODE_ENV !== \"production\") {\n this.#logger?.trace({ payload: data }, \"received message from ws\");\n }\n switch (data.type) {\n case \"objectSetChanged\":\n return this.#handleMessage_objectSetChanged(data);\n case \"refreshObjectSet\":\n return this.#handleMessage_refreshObjectSet(data);\n case \"subscribeResponses\":\n return this.#handleMessage_subscribeResponses(data);\n case \"subscriptionClosed\": {\n return this.#handleMessage_subscriptionClosed(data);\n }\n default:\n const _ = data;\n invariant(false, \"Unexpected message type\");\n }\n };\n #handleMessage_objectSetChanged = async (payload) => {\n const sub = this.#subscriptions.get(payload.id);\n if (sub == null)\n return;\n const objectUpdates = payload.updates.filter((update) => update.type === \"object\");\n const referenceUpdates = payload.updates.filter((update) => update.type === \"reference\");\n const osdkObjectsWithReferenceUpdates = await Promise.all(referenceUpdates.map(async (o) => {\n const osdkObjectArray = await convertWireToOsdkObjects(this.#client, [{\n __apiName: o.objectType,\n __primaryKey: o.primaryKey[sub.primaryKeyPropertyName],\n ...o.primaryKey,\n [o.property]: o.value,\n }], undefined);\n const singleOsdkObject = osdkObjectArray[0] ?? undefined;\n return singleOsdkObject != null\n ? {\n object: singleOsdkObject,\n state: \"ADDED_OR_UPDATED\",\n }\n : undefined;\n }));\n for (const osdkObject of osdkObjectsWithReferenceUpdates) {\n if (osdkObject != null) {\n sub.listener.onChange?.(osdkObject);\n }\n }\n const osdkObjects = await Promise.all(objectUpdates.map(async (o) => {\n const keysToDelete = Object.keys(o.object).filter((key) => sub.requestedReferenceProperties.includes(key));\n for (const key of keysToDelete) {\n delete o.object[key];\n }\n const osdkObjectArray = await convertWireToOsdkObjects(this.#client, [o.object], undefined);\n const singleOsdkObject = osdkObjectArray[0] ?? undefined;\n return singleOsdkObject != null\n ? {\n object: singleOsdkObject,\n state: o.state,\n }\n : undefined;\n }));\n for (const osdkObject of osdkObjects) {\n if (osdkObject != null) {\n sub.listener.onChange?.(osdkObject);\n }\n }\n };\n #handleMessage_refreshObjectSet = (payload) => {\n const sub = this.#subscriptions.get(payload.id);\n invariant(sub, `Expected subscription id ${payload.id}`);\n sub.listener.onOutOfDate();\n };\n #handleMessage_subscribeResponses = (payload) => {\n const { id, responses } = payload;\n const subs = this.#pendingSubscriptions.get(id);\n invariant(subs, `should have a pending subscription for ${id}`);\n this.#pendingSubscriptions.delete(id);\n for (let i = 0; i < responses.length; i++) {\n const sub = subs[i];\n const response = responses[i];\n switch (response.type) {\n case \"error\":\n sub.listener.onError(response.errors);\n this.#unsubscribe(sub, \"error\");\n break;\n case \"qos\":\n // the server has requested that we tear down our websocket and reconnect to help load balance\n this.#cycleWebsocket();\n break;\n case \"success\":\n // `\"preparing\"` should only be the status on an initial subscribe.\n const shouldFireOutOfDate = sub.status === \"expired\"\n || sub.status === \"reconnecting\";\n if (process.env.NODE_ENV !== \"production\") {\n this.#logger?.trace({ shouldFireOutOfDate }, \"success\");\n }\n sub.status = \"subscribed\";\n if (sub.subscriptionId !== response.id) {\n // might be the temporary one\n this.#subscriptions.delete(sub.subscriptionId);\n sub.subscriptionId = response.id;\n this.#subscriptions.set(sub.subscriptionId, sub); // future messages come by this subId\n }\n if (shouldFireOutOfDate)\n sub.listener.onOutOfDate();\n else\n sub.listener.onSuccessfulSubscription();\n break;\n default:\n const _ = response;\n sub.listener.onError(response);\n }\n }\n };\n #handleMessage_subscriptionClosed(payload) {\n const sub = this.#subscriptions.get(payload.id);\n invariant(sub, `Expected subscription id ${payload.id}`);\n sub.listener.onError([payload.cause]);\n this.#unsubscribe(sub, \"error\");\n }\n #onClose = (event) => {\n if (process.env.NODE_ENV !== \"production\") {\n this.#logger?.trace({ event }, \"Received close event from ws\", event);\n }\n // TODO we should probably throttle this so we don't abuse the backend\n this.#cycleWebsocket();\n };\n #cycleWebsocket = () => {\n if (this.#ws) {\n this.#ws.removeEventListener(\"open\", this.#onOpen);\n this.#ws.removeEventListener(\"message\", this.#onMessage);\n this.#ws.removeEventListener(\"close\", this.#onClose);\n if (this.#ws.readyState !== WebSocket.CLOSING\n && this.#ws.readyState !== WebSocket.CLOSED) {\n this.#ws.close();\n }\n this.#ws = undefined;\n }\n // if we have any listeners that are still depending on us, go ahead and reopen the websocket\n if (this.#subscriptions.size > 0) {\n if (process.env.NODE_ENV !== \"production\") {\n for (const s of this.#subscriptions.values()) {\n invariant(s.status !== \"done\" && s.status !== \"error\", \"should not have done/error subscriptions still\");\n }\n }\n for (const s of this.#subscriptions.values()) {\n if (s.status === \"subscribed\")\n s.status = \"reconnecting\";\n }\n this.#ensureWebsocket();\n }\n };\n}\n"],"mappings":"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","ignoreList":[]}