@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.
- package/CHANGELOG.md +39 -0
- package/build/browser/Client.d.ts +8 -21
- package/build/browser/Client.d.ts.map +1 -1
- package/build/browser/Client.js +2 -1
- package/build/browser/Client.js.map +1 -1
- package/build/browser/MinimalClientContext.d.ts +1 -1
- package/build/browser/MinimalClientContext.d.ts.map +1 -1
- package/build/browser/__unstable/ConjureSupport.js +4 -4
- package/build/browser/__unstable/ConjureSupport.js.map +1 -1
- package/build/browser/__unstable/createBulkLinksAsyncIterFactory.d.ts +1 -6
- package/build/browser/__unstable/createBulkLinksAsyncIterFactory.d.ts.map +1 -1
- package/build/browser/__unstable/createBulkLinksAsyncIterFactory.js +1 -1
- package/build/browser/__unstable/createBulkLinksAsyncIterFactory.js.map +1 -1
- package/build/browser/actions/applyAction.js +1 -1
- package/build/browser/actions/applyAction.js.map +1 -1
- package/build/browser/createClient.d.ts +1 -1
- package/build/browser/createClient.d.ts.map +1 -1
- package/build/browser/createClient.js +41 -19
- package/build/browser/createClient.js.map +1 -1
- package/build/browser/createGeotimeSeriesProperty.d.ts +1 -1
- package/build/browser/createGeotimeSeriesProperty.d.ts.map +1 -1
- package/build/browser/createGeotimeSeriesProperty.js +4 -1
- package/build/browser/createGeotimeSeriesProperty.js.map +1 -1
- package/build/browser/createMinimalClient.js +3 -3
- package/build/browser/createMinimalClient.js.map +1 -1
- package/build/browser/createPlatformClient.d.ts +1 -1
- package/build/browser/createPlatformClient.d.ts.map +1 -1
- package/build/browser/index.d.ts +0 -1
- package/build/browser/index.d.ts.map +1 -1
- package/build/browser/index.js +0 -1
- package/build/browser/index.js.map +1 -1
- package/build/browser/intellisense.test.js +1 -1
- package/build/browser/intellisense.test.js.map +1 -1
- package/build/browser/internal/conversions/legacyToModernSingleAggregationResult.js +1 -1
- package/build/browser/internal/conversions/legacyToModernSingleAggregationResult.js.map +1 -1
- package/build/browser/internal/conversions/modernToLegacyWhereClause.js +3 -3
- package/build/browser/internal/conversions/modernToLegacyWhereClause.js.map +1 -1
- package/build/browser/object/aggregate.js +1 -1
- package/build/browser/object/aggregate.js.map +1 -1
- package/build/browser/object/convertWireToOsdkObjects/InterfaceHolder.d.ts +1 -10
- package/build/browser/object/convertWireToOsdkObjects/InterfaceHolder.d.ts.map +1 -1
- package/build/browser/object/convertWireToOsdkObjects/InternalSymbols.d.ts +0 -7
- package/build/browser/object/convertWireToOsdkObjects/InternalSymbols.d.ts.map +1 -1
- package/build/browser/object/convertWireToOsdkObjects/InternalSymbols.js +6 -0
- package/build/browser/object/convertWireToOsdkObjects/InternalSymbols.js.map +1 -1
- package/build/browser/object/convertWireToOsdkObjects/createOsdkObject.js +7 -1
- package/build/browser/object/convertWireToOsdkObjects/createOsdkObject.js.map +1 -1
- package/build/browser/object/convertWireToOsdkObjects.js +2 -2
- package/build/browser/object/convertWireToOsdkObjects.js.map +1 -1
- package/build/browser/object/convertWireToOsdkObjects.test.js +10 -10
- package/build/browser/object/convertWireToOsdkObjects.test.js.map +1 -1
- package/build/browser/object/fetchPage.js +1 -1
- package/build/browser/object/fetchPage.js.map +1 -1
- package/build/browser/objectSet/ObjectSet.test.js +43 -47
- package/build/browser/objectSet/ObjectSet.test.js.map +1 -1
- package/build/browser/objectSet/ObjectSetListenerWebsocket.js +62 -57
- package/build/browser/objectSet/ObjectSetListenerWebsocket.js.map +1 -1
- package/build/browser/objectSet/ObjectSetListenerWebsocket.test.js +109 -21
- package/build/browser/objectSet/ObjectSetListenerWebsocket.test.js.map +1 -1
- package/build/browser/objectSet/createObjectSet.js +2 -5
- package/build/browser/objectSet/createObjectSet.js.map +1 -1
- package/build/browser/ontology/makeConjureContext.d.ts +1 -1
- package/build/browser/ontology/makeConjureContext.d.ts.map +1 -1
- package/build/browser/queries/applyQuery.js +1 -1
- package/build/browser/queries/applyQuery.js.map +1 -1
- package/build/browser/queries/queries.test.js +30 -2
- package/build/browser/queries/queries.test.js.map +1 -1
- package/build/browser/tsserver.js +1 -1
- package/build/browser/tsserver.js.map +1 -1
- package/build/browser/util/UserAgent.js +1 -1
- package/build/esm/Client.d.ts +8 -21
- package/build/esm/Client.d.ts.map +1 -1
- package/build/esm/Client.js +2 -1
- package/build/esm/Client.js.map +1 -1
- package/build/esm/MinimalClientContext.d.ts +1 -1
- package/build/esm/MinimalClientContext.d.ts.map +1 -1
- package/build/esm/__unstable/ConjureSupport.js +4 -4
- package/build/esm/__unstable/ConjureSupport.js.map +1 -1
- package/build/esm/__unstable/createBulkLinksAsyncIterFactory.d.ts +1 -6
- package/build/esm/__unstable/createBulkLinksAsyncIterFactory.d.ts.map +1 -1
- package/build/esm/__unstable/createBulkLinksAsyncIterFactory.js +1 -1
- package/build/esm/__unstable/createBulkLinksAsyncIterFactory.js.map +1 -1
- package/build/esm/actions/applyAction.js +1 -1
- package/build/esm/actions/applyAction.js.map +1 -1
- package/build/esm/createClient.d.ts +1 -1
- package/build/esm/createClient.d.ts.map +1 -1
- package/build/esm/createClient.js +41 -19
- package/build/esm/createClient.js.map +1 -1
- package/build/esm/createGeotimeSeriesProperty.d.ts +1 -1
- package/build/esm/createGeotimeSeriesProperty.d.ts.map +1 -1
- package/build/esm/createGeotimeSeriesProperty.js +4 -1
- package/build/esm/createGeotimeSeriesProperty.js.map +1 -1
- package/build/esm/createMinimalClient.js +3 -3
- package/build/esm/createMinimalClient.js.map +1 -1
- package/build/esm/createPlatformClient.d.ts +1 -1
- package/build/esm/createPlatformClient.d.ts.map +1 -1
- package/build/esm/index.d.ts +0 -1
- package/build/esm/index.d.ts.map +1 -1
- package/build/esm/index.js +0 -1
- package/build/esm/index.js.map +1 -1
- package/build/esm/intellisense.test.js +1 -1
- package/build/esm/intellisense.test.js.map +1 -1
- package/build/esm/internal/conversions/legacyToModernSingleAggregationResult.js +1 -1
- package/build/esm/internal/conversions/legacyToModernSingleAggregationResult.js.map +1 -1
- package/build/esm/internal/conversions/modernToLegacyWhereClause.js +3 -3
- package/build/esm/internal/conversions/modernToLegacyWhereClause.js.map +1 -1
- package/build/esm/object/aggregate.js +1 -1
- package/build/esm/object/aggregate.js.map +1 -1
- package/build/esm/object/convertWireToOsdkObjects/InterfaceHolder.d.ts +1 -10
- package/build/esm/object/convertWireToOsdkObjects/InterfaceHolder.d.ts.map +1 -1
- package/build/esm/object/convertWireToOsdkObjects/InternalSymbols.d.ts +0 -7
- package/build/esm/object/convertWireToOsdkObjects/InternalSymbols.d.ts.map +1 -1
- package/build/esm/object/convertWireToOsdkObjects/InternalSymbols.js +6 -0
- package/build/esm/object/convertWireToOsdkObjects/InternalSymbols.js.map +1 -1
- package/build/esm/object/convertWireToOsdkObjects/createOsdkObject.js +7 -1
- package/build/esm/object/convertWireToOsdkObjects/createOsdkObject.js.map +1 -1
- package/build/esm/object/convertWireToOsdkObjects.js +2 -2
- package/build/esm/object/convertWireToOsdkObjects.js.map +1 -1
- package/build/esm/object/convertWireToOsdkObjects.test.js +10 -10
- package/build/esm/object/convertWireToOsdkObjects.test.js.map +1 -1
- package/build/esm/object/fetchPage.js +1 -1
- package/build/esm/object/fetchPage.js.map +1 -1
- package/build/esm/objectSet/ObjectSet.test.js +43 -47
- package/build/esm/objectSet/ObjectSet.test.js.map +1 -1
- package/build/esm/objectSet/ObjectSetListenerWebsocket.js +62 -57
- package/build/esm/objectSet/ObjectSetListenerWebsocket.js.map +1 -1
- package/build/esm/objectSet/ObjectSetListenerWebsocket.test.js +109 -21
- package/build/esm/objectSet/ObjectSetListenerWebsocket.test.js.map +1 -1
- package/build/esm/objectSet/createObjectSet.js +2 -5
- package/build/esm/objectSet/createObjectSet.js.map +1 -1
- package/build/esm/ontology/makeConjureContext.d.ts +1 -1
- package/build/esm/ontology/makeConjureContext.d.ts.map +1 -1
- package/build/esm/queries/applyQuery.js +1 -1
- package/build/esm/queries/applyQuery.js.map +1 -1
- package/build/esm/queries/queries.test.js +30 -2
- package/build/esm/queries/queries.test.js.map +1 -1
- package/build/esm/tsserver.js +1 -1
- package/build/esm/tsserver.js.map +1 -1
- package/build/esm/util/UserAgent.js +1 -1
- 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
|
-
|
|
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
|
|
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
|
|
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
|
|
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
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
159
|
+
requestedProperties,
|
|
160
|
+
requestedReferenceProperties
|
|
161
|
+
}) => {
|
|
162
|
+
return {
|
|
163
|
+
objectSet: objectSet,
|
|
164
|
+
propertySet: requestedProperties,
|
|
165
|
+
referenceSet: requestedReferenceProperties
|
|
166
|
+
};
|
|
167
|
+
})
|
|
169
168
|
};
|
|
170
|
-
if (process
|
|
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
|
|
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
|
|
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
|
-
|
|
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
|
-
|
|
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
|
|
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
|
|
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
|
|
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":[]}
|