@dxos/functions 0.6.10 → 0.6.11-staging.30cf5ba

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 (43) hide show
  1. package/dist/lib/browser/chunk-IBJYIBKT.mjs +581 -0
  2. package/dist/lib/browser/chunk-IBJYIBKT.mjs.map +7 -0
  3. package/dist/lib/browser/{chunk-YSDC6YCF.mjs → chunk-XOBJR3A6.mjs} +2 -1
  4. package/dist/lib/browser/{chunk-YSDC6YCF.mjs.map → chunk-XOBJR3A6.mjs.map} +3 -3
  5. package/dist/lib/browser/index.mjs +9 -7
  6. package/dist/lib/browser/index.mjs.map +1 -1
  7. package/dist/lib/browser/meta.json +1 -1
  8. package/dist/lib/browser/testing/index.mjs +522 -11
  9. package/dist/lib/browser/testing/index.mjs.map +4 -4
  10. package/dist/lib/browser/types.mjs +1 -1
  11. package/dist/lib/node/{chunk-3E6PY6JH.cjs → chunk-GGTHSME4.cjs} +5 -4
  12. package/dist/lib/node/{chunk-3E6PY6JH.cjs.map → chunk-GGTHSME4.cjs.map} +3 -3
  13. package/dist/lib/node/chunk-V7JNSENS.cjs +604 -0
  14. package/dist/lib/node/chunk-V7JNSENS.cjs.map +7 -0
  15. package/dist/lib/node/index.cjs +14 -12
  16. package/dist/lib/node/index.cjs.map +1 -1
  17. package/dist/lib/node/meta.json +1 -1
  18. package/dist/lib/node/testing/index.cjs +521 -13
  19. package/dist/lib/node/testing/index.cjs.map +4 -4
  20. package/dist/lib/node/types.cjs +5 -5
  21. package/dist/lib/node/types.cjs.map +1 -1
  22. package/dist/types/src/index.d.ts +0 -1
  23. package/dist/types/src/index.d.ts.map +1 -1
  24. package/dist/types/src/runtime/scheduler.d.ts.map +1 -1
  25. package/dist/types/src/trigger/index.d.ts +1 -0
  26. package/dist/types/src/trigger/index.d.ts.map +1 -1
  27. package/dist/types/src/trigger/trigger-registry.d.ts +5 -4
  28. package/dist/types/src/trigger/trigger-registry.d.ts.map +1 -1
  29. package/dist/types/src/trigger/type/index.d.ts +0 -1
  30. package/dist/types/src/trigger/type/index.d.ts.map +1 -1
  31. package/dist/types/src/types.d.ts +4 -0
  32. package/dist/types/src/types.d.ts.map +1 -1
  33. package/package.json +14 -14
  34. package/src/index.ts +1 -1
  35. package/src/runtime/scheduler.ts +6 -3
  36. package/src/trigger/index.ts +1 -0
  37. package/src/trigger/trigger-registry.ts +14 -7
  38. package/src/trigger/type/index.ts +1 -1
  39. package/src/types.ts +1 -0
  40. package/dist/lib/browser/chunk-OERXFETS.mjs +0 -1120
  41. package/dist/lib/browser/chunk-OERXFETS.mjs.map +0 -7
  42. package/dist/lib/node/chunk-ITQU6E54.cjs +0 -1133
  43. package/dist/lib/node/chunk-ITQU6E54.cjs.map +0 -7
@@ -0,0 +1,581 @@
1
+ import "@dxos/node-std/globals";
2
+ import {
3
+ FunctionDef,
4
+ FunctionTrigger
5
+ } from "./chunk-XOBJR3A6.mjs";
6
+
7
+ // packages/core/functions/src/function/function-registry.ts
8
+ import { Event } from "@dxos/async";
9
+ import { create, Filter } from "@dxos/client/echo";
10
+ import { Resource } from "@dxos/context";
11
+ import { PublicKey } from "@dxos/keys";
12
+ import { log } from "@dxos/log";
13
+ import { ComplexMap, diff } from "@dxos/util";
14
+ var __dxlog_file = "/home/runner/work/dxos/dxos/packages/core/functions/src/function/function-registry.ts";
15
+ var FunctionRegistry = class extends Resource {
16
+ constructor(_client) {
17
+ super();
18
+ this._client = _client;
19
+ this._functionBySpaceKey = new ComplexMap(PublicKey.hash);
20
+ this.registered = new Event();
21
+ }
22
+ getFunctions(space) {
23
+ return this._functionBySpaceKey.get(space.key) ?? [];
24
+ }
25
+ getUniqueByUri() {
26
+ const uniqueByUri = [
27
+ ...this._functionBySpaceKey.values()
28
+ ].flatMap((defs) => defs).reduce((acc, v) => {
29
+ acc.set(v.uri, v);
30
+ return acc;
31
+ }, /* @__PURE__ */ new Map());
32
+ return [
33
+ ...uniqueByUri.values()
34
+ ];
35
+ }
36
+ /**
37
+ * Loads function definitions from the manifest into the space.
38
+ * We first load all the definitions from the space to deduplicate by functionId.
39
+ */
40
+ async register(space, functions) {
41
+ log("register", {
42
+ space: space.key,
43
+ functions: functions?.length ?? 0
44
+ }, {
45
+ F: __dxlog_file,
46
+ L: 48,
47
+ S: this,
48
+ C: (f, a) => f(...a)
49
+ });
50
+ if (!functions?.length) {
51
+ return;
52
+ }
53
+ if (!space.db.graph.schemaRegistry.hasSchema(FunctionDef)) {
54
+ space.db.graph.schemaRegistry.addSchema([
55
+ FunctionDef
56
+ ]);
57
+ }
58
+ const { objects: existing } = await space.db.query(Filter.schema(FunctionDef)).run();
59
+ const { added } = diff(existing, functions, (a, b) => a.uri === b.uri);
60
+ added.forEach((def) => space.db.add(create(FunctionDef, def)));
61
+ if (added.length > 0) {
62
+ await space.db.flush({
63
+ indexes: true,
64
+ updates: true
65
+ });
66
+ }
67
+ }
68
+ async _open() {
69
+ log.info("opening...", void 0, {
70
+ F: __dxlog_file,
71
+ L: 68,
72
+ S: this,
73
+ C: (f, a) => f(...a)
74
+ });
75
+ const spacesSubscription = this._client.spaces.subscribe(async (spaces) => {
76
+ for (const space of spaces) {
77
+ if (this._functionBySpaceKey.has(space.key)) {
78
+ continue;
79
+ }
80
+ const registered = [];
81
+ this._functionBySpaceKey.set(space.key, registered);
82
+ await space.waitUntilReady();
83
+ if (this._ctx.disposed) {
84
+ break;
85
+ }
86
+ this._ctx.onDispose(space.db.query(Filter.schema(FunctionDef)).subscribe(({ objects }) => {
87
+ const { added } = diff(registered, objects, (a, b) => a.uri === b.uri);
88
+ if (added.length > 0) {
89
+ registered.push(...added);
90
+ this.registered.emit({
91
+ space,
92
+ added
93
+ });
94
+ }
95
+ }));
96
+ }
97
+ });
98
+ this._ctx.onDispose(() => spacesSubscription.unsubscribe());
99
+ }
100
+ async _close(_) {
101
+ log.info("closing...", void 0, {
102
+ F: __dxlog_file,
103
+ L: 101,
104
+ S: this,
105
+ C: (f, a) => f(...a)
106
+ });
107
+ this._functionBySpaceKey.clear();
108
+ }
109
+ };
110
+
111
+ // packages/core/functions/src/trigger/type/subscription-trigger.ts
112
+ import { debounce, UpdateScheduler } from "@dxos/async";
113
+ import { Filter as Filter2 } from "@dxos/client/echo";
114
+ import { createSubscription } from "@dxos/echo-db";
115
+ import { log as log2 } from "@dxos/log";
116
+ var __dxlog_file2 = "/home/runner/work/dxos/dxos/packages/core/functions/src/trigger/type/subscription-trigger.ts";
117
+ var createSubscriptionTrigger = async (ctx, space, spec, callback) => {
118
+ const objectIds = /* @__PURE__ */ new Set();
119
+ const task = new UpdateScheduler(ctx, async () => {
120
+ if (objectIds.size > 0) {
121
+ const objects = Array.from(objectIds);
122
+ objectIds.clear();
123
+ await callback({
124
+ objects
125
+ });
126
+ }
127
+ }, {
128
+ maxFrequency: 4
129
+ });
130
+ const subscriptions = [];
131
+ const subscription = createSubscription(({ added, updated }) => {
132
+ const sizeBefore = objectIds.size;
133
+ for (const object of added) {
134
+ objectIds.add(object.id);
135
+ }
136
+ for (const object of updated) {
137
+ objectIds.add(object.id);
138
+ }
139
+ if (objectIds.size > sizeBefore) {
140
+ log2.info("updated", {
141
+ added: added.length,
142
+ updated: updated.length
143
+ }, {
144
+ F: __dxlog_file2,
145
+ L: 46,
146
+ S: void 0,
147
+ C: (f, a) => f(...a)
148
+ });
149
+ task.trigger();
150
+ }
151
+ });
152
+ subscriptions.push(() => subscription.unsubscribe());
153
+ const { filter, options: { deep, delay } = {} } = spec;
154
+ const update = ({ objects }) => {
155
+ log2.info("update", {
156
+ objects: objects.length
157
+ }, {
158
+ F: __dxlog_file2,
159
+ L: 56,
160
+ S: void 0,
161
+ C: (f, a) => f(...a)
162
+ });
163
+ subscription.update(objects);
164
+ if (deep) {
165
+ }
166
+ };
167
+ log2.info("subscription", {
168
+ filter
169
+ }, {
170
+ F: __dxlog_file2,
171
+ L: 74,
172
+ S: void 0,
173
+ C: (f, a) => f(...a)
174
+ });
175
+ if (filter) {
176
+ const query = space.db.query(Filter2.typename(filter[0].type, filter[0].props));
177
+ subscriptions.push(query.subscribe(delay ? debounce(update, delay) : update));
178
+ }
179
+ ctx.onDispose(() => {
180
+ subscriptions.forEach((unsubscribe) => unsubscribe());
181
+ });
182
+ };
183
+
184
+ // packages/core/functions/src/trigger/type/timer-trigger.ts
185
+ import { CronJob } from "cron";
186
+ import { DeferredTask } from "@dxos/async";
187
+ import { log as log3 } from "@dxos/log";
188
+ var __dxlog_file3 = "/home/runner/work/dxos/dxos/packages/core/functions/src/trigger/type/timer-trigger.ts";
189
+ var createTimerTrigger = async (ctx, space, spec, callback) => {
190
+ const task = new DeferredTask(ctx, async () => {
191
+ await callback({});
192
+ });
193
+ let last = 0;
194
+ let run = 0;
195
+ const job = CronJob.from({
196
+ cronTime: spec.cron,
197
+ runOnInit: false,
198
+ onTick: () => {
199
+ const now = Date.now();
200
+ const delta = last ? now - last : 0;
201
+ last = now;
202
+ run++;
203
+ log3.info("tick", {
204
+ space: space.key.truncate(),
205
+ count: run,
206
+ delta
207
+ }, {
208
+ F: __dxlog_file3,
209
+ L: 38,
210
+ S: void 0,
211
+ C: (f, a) => f(...a)
212
+ });
213
+ task.schedule();
214
+ }
215
+ });
216
+ job.start();
217
+ ctx.onDispose(() => job.stop());
218
+ };
219
+
220
+ // packages/core/functions/src/trigger/type/websocket-trigger.ts
221
+ import WebSocket from "ws";
222
+ import { sleep, Trigger } from "@dxos/async";
223
+ import { log as log4 } from "@dxos/log";
224
+ var __dxlog_file4 = "/home/runner/work/dxos/dxos/packages/core/functions/src/trigger/type/websocket-trigger.ts";
225
+ var createWebsocketTrigger = async (ctx, space, spec, callback, options = {
226
+ retryDelay: 2,
227
+ maxAttempts: 5
228
+ }) => {
229
+ const { url, init } = spec;
230
+ let wasOpen = false;
231
+ let ws;
232
+ for (let attempt = 1; attempt <= options.maxAttempts; attempt++) {
233
+ const open = new Trigger();
234
+ ws = new WebSocket(url);
235
+ Object.assign(ws, {
236
+ onopen: () => {
237
+ log4.info("opened", {
238
+ url
239
+ }, {
240
+ F: __dxlog_file4,
241
+ L: 41,
242
+ S: void 0,
243
+ C: (f, a) => f(...a)
244
+ });
245
+ if (spec.init) {
246
+ ws.send(new TextEncoder().encode(JSON.stringify(init)));
247
+ }
248
+ open.wake(true);
249
+ },
250
+ onclose: (event) => {
251
+ log4.info("closed", {
252
+ url,
253
+ code: event.code
254
+ }, {
255
+ F: __dxlog_file4,
256
+ L: 50,
257
+ S: void 0,
258
+ C: (f, a) => f(...a)
259
+ });
260
+ if (event.code === 1006 && wasOpen && !ctx.disposed) {
261
+ setTimeout(async () => {
262
+ log4.info(`reconnecting in ${options.retryDelay}s...`, {
263
+ url
264
+ }, {
265
+ F: __dxlog_file4,
266
+ L: 55,
267
+ S: void 0,
268
+ C: (f, a) => f(...a)
269
+ });
270
+ await createWebsocketTrigger(ctx, space, spec, callback, options);
271
+ }, options.retryDelay * 1e3);
272
+ }
273
+ open.wake(false);
274
+ },
275
+ onerror: (event) => {
276
+ log4.catch(event.error, {
277
+ url
278
+ }, {
279
+ F: __dxlog_file4,
280
+ L: 63,
281
+ S: void 0,
282
+ C: (f, a) => f(...a)
283
+ });
284
+ open.wake(false);
285
+ },
286
+ onmessage: async (event) => {
287
+ try {
288
+ log4.info("message", void 0, {
289
+ F: __dxlog_file4,
290
+ L: 69,
291
+ S: void 0,
292
+ C: (f, a) => f(...a)
293
+ });
294
+ const data = JSON.parse(new TextDecoder().decode(event.data));
295
+ await callback({
296
+ data
297
+ });
298
+ } catch (err) {
299
+ log4.catch(err, {
300
+ url
301
+ }, {
302
+ F: __dxlog_file4,
303
+ L: 73,
304
+ S: void 0,
305
+ C: (f, a) => f(...a)
306
+ });
307
+ }
308
+ }
309
+ });
310
+ const isOpen = await open.wait();
311
+ if (ctx.disposed) {
312
+ break;
313
+ }
314
+ if (isOpen) {
315
+ wasOpen = true;
316
+ break;
317
+ }
318
+ const wait = Math.pow(attempt, 2) * options.retryDelay;
319
+ if (attempt < options.maxAttempts) {
320
+ log4.warn(`failed to connect; trying again in ${wait}s`, {
321
+ attempt
322
+ }, {
323
+ F: __dxlog_file4,
324
+ L: 88,
325
+ S: void 0,
326
+ C: (f, a) => f(...a)
327
+ });
328
+ await sleep(wait * 1e3);
329
+ }
330
+ }
331
+ ctx.onDispose(() => {
332
+ ws?.close();
333
+ });
334
+ };
335
+
336
+ // packages/core/functions/src/trigger/trigger-registry.ts
337
+ import { Event as Event2 } from "@dxos/async";
338
+ import { create as create2, Filter as Filter3, getMeta } from "@dxos/client/echo";
339
+ import { Context, Resource as Resource2 } from "@dxos/context";
340
+ import { compareForeignKeys, ECHO_ATTR_META, foreignKey } from "@dxos/echo-schema";
341
+ import { invariant } from "@dxos/invariant";
342
+ import { PublicKey as PublicKey2 } from "@dxos/keys";
343
+ import { log as log5 } from "@dxos/log";
344
+ import { ComplexMap as ComplexMap2, diff as diff2 } from "@dxos/util";
345
+ var __dxlog_file5 = "/home/runner/work/dxos/dxos/packages/core/functions/src/trigger/trigger-registry.ts";
346
+ var triggerFactory = {
347
+ subscription: createSubscriptionTrigger,
348
+ timer: createTimerTrigger,
349
+ // TODO(burdon): Cannot use in browser.
350
+ // webhook: createWebhookTrigger,
351
+ webhook: null,
352
+ websocket: createWebsocketTrigger
353
+ };
354
+ var TriggerRegistry = class extends Resource2 {
355
+ constructor(_client, _options) {
356
+ super();
357
+ this._client = _client;
358
+ this._options = _options;
359
+ this._triggersBySpaceKey = new ComplexMap2(PublicKey2.hash);
360
+ this.registered = new Event2();
361
+ this.removed = new Event2();
362
+ }
363
+ getActiveTriggers(space) {
364
+ return this._getTriggers(space, (t) => t.activationCtx != null);
365
+ }
366
+ getInactiveTriggers(space) {
367
+ return this._getTriggers(space, (t) => t.activationCtx == null);
368
+ }
369
+ /**
370
+ * Set callback for trigger.
371
+ */
372
+ async activate(space, trigger, callback) {
373
+ log5("activate", {
374
+ space: space.key,
375
+ trigger
376
+ }, {
377
+ F: __dxlog_file5,
378
+ L: 77,
379
+ S: this,
380
+ C: (f, a) => f(...a)
381
+ });
382
+ const activationCtx = new Context({
383
+ name: `FunctionTrigger-${trigger.function}`
384
+ }, {
385
+ F: __dxlog_file5,
386
+ L: 79
387
+ });
388
+ this._ctx.onDispose(() => activationCtx.dispose());
389
+ const registeredTrigger = this._triggersBySpaceKey.get(space.key)?.find((reg) => reg.trigger.id === trigger.id);
390
+ invariant(registeredTrigger, `Trigger is not registered: ${trigger.function}`, {
391
+ F: __dxlog_file5,
392
+ L: 82,
393
+ S: this,
394
+ A: [
395
+ "registeredTrigger",
396
+ "`Trigger is not registered: ${trigger.function}`"
397
+ ]
398
+ });
399
+ registeredTrigger.activationCtx = activationCtx;
400
+ try {
401
+ const options = this._options?.[trigger.spec.type];
402
+ const createTrigger = triggerFactory[trigger.spec.type];
403
+ await createTrigger(activationCtx, space, trigger.spec, callback, options);
404
+ } catch (err) {
405
+ delete registeredTrigger.activationCtx;
406
+ throw err;
407
+ }
408
+ }
409
+ /**
410
+ * Loads triggers from the manifest into the space.
411
+ */
412
+ async register(space, manifest) {
413
+ log5("register", {
414
+ space: space.key
415
+ }, {
416
+ F: __dxlog_file5,
417
+ L: 100,
418
+ S: this,
419
+ C: (f, a) => f(...a)
420
+ });
421
+ if (!manifest.triggers?.length) {
422
+ return;
423
+ }
424
+ if (!space.db.graph.schemaRegistry.hasSchema(FunctionTrigger)) {
425
+ space.db.graph.schemaRegistry.addSchema([
426
+ FunctionTrigger
427
+ ]);
428
+ }
429
+ const manifestTriggers = manifest.triggers.map((trigger) => {
430
+ let keys = trigger[ECHO_ATTR_META]?.keys;
431
+ delete trigger[ECHO_ATTR_META];
432
+ if (!keys?.length) {
433
+ keys = [
434
+ foreignKey("manifest", [
435
+ trigger.function,
436
+ trigger.spec.type
437
+ ].join(":"))
438
+ ];
439
+ }
440
+ return create2(FunctionTrigger, trigger, {
441
+ keys
442
+ });
443
+ });
444
+ const { objects: existing } = await space.db.query(Filter3.schema(FunctionTrigger)).run();
445
+ const { added } = diff2(existing, manifestTriggers, compareForeignKeys);
446
+ added.forEach((trigger) => {
447
+ space.db.add(trigger);
448
+ log5.info("added", {
449
+ meta: getMeta(trigger)
450
+ }, {
451
+ F: __dxlog_file5,
452
+ L: 127,
453
+ S: this,
454
+ C: (f, a) => f(...a)
455
+ });
456
+ });
457
+ if (added.length > 0) {
458
+ await space.db.flush();
459
+ }
460
+ }
461
+ async _open() {
462
+ log5.info("open...", void 0, {
463
+ F: __dxlog_file5,
464
+ L: 136,
465
+ S: this,
466
+ C: (f, a) => f(...a)
467
+ });
468
+ const spaceListSubscription = this._client.spaces.subscribe(async (spaces) => {
469
+ for (const space of spaces) {
470
+ if (this._triggersBySpaceKey.has(space.key)) {
471
+ continue;
472
+ }
473
+ const registered = [];
474
+ this._triggersBySpaceKey.set(space.key, registered);
475
+ await space.waitUntilReady();
476
+ if (this._ctx.disposed) {
477
+ break;
478
+ }
479
+ this._ctx.onDispose(space.db.query(Filter3.schema(FunctionTrigger)).subscribe(async ({ objects: current }) => {
480
+ log5.info("update", {
481
+ space: space.key,
482
+ registered: registered.length,
483
+ current: current.length
484
+ }, {
485
+ F: __dxlog_file5,
486
+ L: 153,
487
+ S: this,
488
+ C: (f, a) => f(...a)
489
+ });
490
+ await this._handleRemovedTriggers(space, current, registered);
491
+ this._handleNewTriggers(space, current, registered);
492
+ }));
493
+ }
494
+ });
495
+ this._ctx.onDispose(() => spaceListSubscription.unsubscribe());
496
+ log5.info("opened", void 0, {
497
+ F: __dxlog_file5,
498
+ L: 162,
499
+ S: this,
500
+ C: (f, a) => f(...a)
501
+ });
502
+ }
503
+ async _close(_) {
504
+ log5.info("close...", void 0, {
505
+ F: __dxlog_file5,
506
+ L: 166,
507
+ S: this,
508
+ C: (f, a) => f(...a)
509
+ });
510
+ this._triggersBySpaceKey.clear();
511
+ log5.info("closed", void 0, {
512
+ F: __dxlog_file5,
513
+ L: 168,
514
+ S: this,
515
+ C: (f, a) => f(...a)
516
+ });
517
+ }
518
+ _handleNewTriggers(space, current, registered) {
519
+ const added = current.filter((candidate) => {
520
+ return candidate.enabled && registered.find((reg) => reg.trigger.id === candidate.id) == null;
521
+ });
522
+ if (added.length > 0) {
523
+ const newRegisteredTriggers = added.map((trigger) => ({
524
+ trigger
525
+ }));
526
+ registered.push(...newRegisteredTriggers);
527
+ log5.info("added", () => ({
528
+ spaceKey: space.key,
529
+ triggers: added.map((trigger) => trigger.function)
530
+ }), {
531
+ F: __dxlog_file5,
532
+ L: 179,
533
+ S: this,
534
+ C: (f, a) => f(...a)
535
+ });
536
+ this.registered.emit({
537
+ space,
538
+ triggers: added
539
+ });
540
+ }
541
+ }
542
+ async _handleRemovedTriggers(space, current, registered) {
543
+ const removed = [];
544
+ for (let i = registered.length - 1; i >= 0; i--) {
545
+ const wasRemoved = current.filter((trigger) => trigger.enabled).find((trigger) => trigger.id === registered[i].trigger.id) == null;
546
+ if (wasRemoved) {
547
+ const unregistered = registered.splice(i, 1)[0];
548
+ await unregistered.activationCtx?.dispose();
549
+ removed.push(unregistered.trigger);
550
+ }
551
+ }
552
+ if (removed.length > 0) {
553
+ log5.info("removed", () => ({
554
+ spaceKey: space.key,
555
+ triggers: removed.map((trigger) => trigger.function)
556
+ }), {
557
+ F: __dxlog_file5,
558
+ L: 205,
559
+ S: this,
560
+ C: (f, a) => f(...a)
561
+ });
562
+ this.removed.emit({
563
+ space,
564
+ triggers: removed
565
+ });
566
+ }
567
+ }
568
+ _getTriggers(space, predicate) {
569
+ const allSpaceTriggers = this._triggersBySpaceKey.get(space.key) ?? [];
570
+ return allSpaceTriggers.filter(predicate).map((trigger) => trigger.trigger);
571
+ }
572
+ };
573
+
574
+ export {
575
+ FunctionRegistry,
576
+ createSubscriptionTrigger,
577
+ createTimerTrigger,
578
+ createWebsocketTrigger,
579
+ TriggerRegistry
580
+ };
581
+ //# sourceMappingURL=chunk-IBJYIBKT.mjs.map
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../../src/function/function-registry.ts", "../../../src/trigger/type/subscription-trigger.ts", "../../../src/trigger/type/timer-trigger.ts", "../../../src/trigger/type/websocket-trigger.ts", "../../../src/trigger/trigger-registry.ts"],
4
+ "sourcesContent": ["//\n// Copyright 2024 DXOS.org\n//\n\nimport { Event } from '@dxos/async';\nimport { type Client } from '@dxos/client';\nimport { create, Filter, type Space } from '@dxos/client/echo';\nimport { type Context, Resource } from '@dxos/context';\nimport { PublicKey } from '@dxos/keys';\nimport { log } from '@dxos/log';\nimport { ComplexMap, diff } from '@dxos/util';\n\nimport { FunctionDef, type FunctionManifest } from '../types';\n\nexport type FunctionsRegisteredEvent = {\n space: Space;\n added: FunctionDef[];\n};\n\nexport class FunctionRegistry extends Resource {\n private readonly _functionBySpaceKey = new ComplexMap<PublicKey, FunctionDef[]>(PublicKey.hash);\n\n public readonly registered = new Event<FunctionsRegisteredEvent>();\n\n constructor(private readonly _client: Client) {\n super();\n }\n\n public getFunctions(space: Space): FunctionDef[] {\n return this._functionBySpaceKey.get(space.key) ?? [];\n }\n\n public getUniqueByUri(): FunctionDef[] {\n const uniqueByUri = [...this._functionBySpaceKey.values()]\n .flatMap((defs) => defs)\n .reduce((acc, v) => {\n acc.set(v.uri, v);\n return acc;\n }, new Map<string, FunctionDef>());\n return [...uniqueByUri.values()];\n }\n\n /**\n * Loads function definitions from the manifest into the space.\n * We first load all the definitions from the space to deduplicate by functionId.\n */\n public async register(space: Space, functions: FunctionManifest['functions']): Promise<void> {\n log('register', { space: space.key, functions: functions?.length ?? 0 });\n if (!functions?.length) {\n return;\n }\n if (!space.db.graph.schemaRegistry.hasSchema(FunctionDef)) {\n space.db.graph.schemaRegistry.addSchema([FunctionDef]);\n }\n\n // Sync definitions.\n const { objects: existing } = await space.db.query(Filter.schema(FunctionDef)).run();\n const { added } = diff(existing, functions, (a, b) => a.uri === b.uri);\n // TODO(burdon): Update existing templates.\n added.forEach((def) => space.db.add(create(FunctionDef, def)));\n\n if (added.length > 0) {\n await space.db.flush({ indexes: true, updates: true });\n }\n }\n\n protected override async _open(): Promise<void> {\n log.info('opening...');\n const spacesSubscription = this._client.spaces.subscribe(async (spaces) => {\n for (const space of spaces) {\n if (this._functionBySpaceKey.has(space.key)) {\n continue;\n }\n\n const registered: FunctionDef[] = [];\n this._functionBySpaceKey.set(space.key, registered);\n await space.waitUntilReady();\n if (this._ctx.disposed) {\n break;\n }\n\n // Subscribe to updates.\n this._ctx.onDispose(\n space.db.query(Filter.schema(FunctionDef)).subscribe(({ objects }) => {\n const { added } = diff(registered, objects, (a, b) => a.uri === b.uri);\n // TODO(burdon): Update and remove.\n if (added.length > 0) {\n registered.push(...added);\n this.registered.emit({ space, added });\n }\n }),\n );\n }\n });\n\n // TODO(burdon): API: Normalize unsubscribe methods.\n this._ctx.onDispose(() => spacesSubscription.unsubscribe());\n }\n\n protected override async _close(_: Context): Promise<void> {\n log.info('closing...');\n this._functionBySpaceKey.clear();\n }\n}\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { debounce, UpdateScheduler } from '@dxos/async';\nimport { Filter, type Space } from '@dxos/client/echo';\nimport { type Context } from '@dxos/context';\nimport { createSubscription, type Query } from '@dxos/echo-db';\nimport { log } from '@dxos/log';\n\nimport type { SubscriptionTrigger } from '../../types';\nimport { type TriggerCallback, type TriggerFactory } from '../trigger-registry';\n\nexport const createSubscriptionTrigger: TriggerFactory<SubscriptionTrigger> = async (\n ctx: Context,\n space: Space,\n spec: SubscriptionTrigger,\n callback: TriggerCallback,\n) => {\n const objectIds = new Set<string>();\n const task = new UpdateScheduler(\n ctx,\n async () => {\n if (objectIds.size > 0) {\n const objects = Array.from(objectIds);\n objectIds.clear();\n await callback({ objects });\n }\n },\n { maxFrequency: 4 },\n );\n\n // TODO(burdon): Factor out diff.\n // TODO(burdon): Don't fire initially?\n // TODO(burdon): Create queue. Only allow one invocation per trigger at a time?\n const subscriptions: (() => void)[] = [];\n const subscription = createSubscription(({ added, updated }) => {\n const sizeBefore = objectIds.size;\n for (const object of added) {\n objectIds.add(object.id);\n }\n for (const object of updated) {\n objectIds.add(object.id);\n }\n if (objectIds.size > sizeBefore) {\n log.info('updated', { added: added.length, updated: updated.length });\n task.trigger();\n }\n });\n\n subscriptions.push(() => subscription.unsubscribe());\n\n // TODO(burdon): Disable trigger if keeps failing.\n const { filter, options: { deep, delay } = {} } = spec;\n const update = ({ objects }: Query) => {\n log.info('update', { objects: objects.length });\n subscription.update(objects);\n\n // TODO(burdon): Hack to monitor changes to Document's text object.\n if (deep) {\n // TODO(dmaretskyi): Removed to not have dependency on markdown-plugin.\n // for (const object of objects) {\n // const content = object.content;\n // if (content instanceof TextType) {\n // subscriptions.push(getObjectCore(content).updates.on(debounce(() => subscription.update([object]), 1_000)));\n // }\n // }\n }\n };\n\n // TODO(burdon): OR not working.\n // TODO(burdon): [Bug]: all callbacks are fired on the first mutation.\n // TODO(burdon): [Bug]: not updated when document is deleted (either top or hierarchically).\n log.info('subscription', { filter });\n // const query = triggerCtx.space.db.query(Filter.or(filter.map(({ type, props }) => Filter.typename(type, props))));\n if (filter) {\n const query = space.db.query(Filter.typename(filter[0].type, filter[0].props));\n subscriptions.push(query.subscribe(delay ? debounce(update, delay) : update));\n }\n\n ctx.onDispose(() => {\n subscriptions.forEach((unsubscribe) => unsubscribe());\n });\n};\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { CronJob } from 'cron';\n\nimport { DeferredTask } from '@dxos/async';\nimport { type Space } from '@dxos/client/echo';\nimport { type Context } from '@dxos/context';\nimport { log } from '@dxos/log';\n\nimport type { TimerTrigger } from '../../types';\nimport { type TriggerCallback, type TriggerFactory } from '../trigger-registry';\n\nexport const createTimerTrigger: TriggerFactory<TimerTrigger> = async (\n ctx: Context,\n space: Space,\n spec: TimerTrigger,\n callback: TriggerCallback,\n) => {\n const task = new DeferredTask(ctx, async () => {\n await callback({});\n });\n\n let last = 0;\n let run = 0;\n // https://www.npmjs.com/package/cron#constructor\n const job = CronJob.from({\n cronTime: spec.cron,\n runOnInit: false,\n onTick: () => {\n // TODO(burdon): Check greater than 30s (use cron-parser).\n const now = Date.now();\n const delta = last ? now - last : 0;\n last = now;\n\n run++;\n log.info('tick', { space: space.key.truncate(), count: run, delta });\n task.schedule();\n },\n });\n\n job.start();\n ctx.onDispose(() => job.stop());\n};\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport WebSocket from 'ws';\n\nimport { sleep, Trigger } from '@dxos/async';\nimport { type Space } from '@dxos/client/echo';\nimport { type Context } from '@dxos/context';\nimport { log } from '@dxos/log';\n\nimport { type WebsocketTrigger } from '../../types';\nimport { type TriggerCallback, type TriggerFactory } from '../trigger-registry';\n\ninterface WebsocketTriggerOptions {\n retryDelay: number;\n maxAttempts: number;\n}\n\n/**\n * Websocket.\n * NOTE: The port must be unique, so the same hook cannot be used for multiple spaces.\n */\nexport const createWebsocketTrigger: TriggerFactory<WebsocketTrigger, WebsocketTriggerOptions> = async (\n ctx: Context,\n space: Space,\n spec: WebsocketTrigger,\n callback: TriggerCallback,\n options: WebsocketTriggerOptions = { retryDelay: 2, maxAttempts: 5 },\n) => {\n const { url, init } = spec;\n\n let wasOpen = false;\n let ws: WebSocket;\n for (let attempt = 1; attempt <= options.maxAttempts; attempt++) {\n const open = new Trigger<boolean>();\n\n ws = new WebSocket(url);\n Object.assign(ws, {\n onopen: () => {\n log.info('opened', { url });\n if (spec.init) {\n ws.send(new TextEncoder().encode(JSON.stringify(init)));\n }\n\n open.wake(true);\n },\n\n onclose: (event) => {\n log.info('closed', { url, code: event.code });\n // Reconnect if server closes (e.g., CF restart).\n // https://developer.mozilla.org/en-US/docs/Web/API/CloseEvent/code\n if (event.code === 1006 && wasOpen && !ctx.disposed) {\n setTimeout(async () => {\n log.info(`reconnecting in ${options.retryDelay}s...`, { url });\n await createWebsocketTrigger(ctx, space, spec, callback, options);\n }, options.retryDelay * 1_000);\n }\n open.wake(false);\n },\n\n onerror: (event) => {\n log.catch(event.error, { url });\n open.wake(false);\n },\n\n onmessage: async (event) => {\n try {\n log.info('message');\n const data = JSON.parse(new TextDecoder().decode(event.data as Uint8Array));\n await callback({ data });\n } catch (err) {\n log.catch(err, { url });\n }\n },\n } satisfies Partial<WebSocket>);\n\n const isOpen = await open.wait();\n if (ctx.disposed) {\n break;\n }\n if (isOpen) {\n wasOpen = true;\n break;\n }\n const wait = Math.pow(attempt, 2) * options.retryDelay;\n if (attempt < options.maxAttempts) {\n log.warn(`failed to connect; trying again in ${wait}s`, { attempt });\n await sleep(wait * 1_000);\n }\n }\n\n ctx.onDispose(() => {\n ws?.close();\n });\n};\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { Event } from '@dxos/async';\nimport { type Client } from '@dxos/client';\nimport { create, Filter, getMeta, type Space } from '@dxos/client/echo';\nimport { Context, Resource } from '@dxos/context';\nimport { compareForeignKeys, ECHO_ATTR_META, foreignKey } from '@dxos/echo-schema';\nimport { invariant } from '@dxos/invariant';\nimport { PublicKey } from '@dxos/keys';\nimport { log } from '@dxos/log';\nimport { ComplexMap, diff } from '@dxos/util';\n\nimport { createSubscriptionTrigger, createTimerTrigger, createWebsocketTrigger } from './type';\nimport { type FunctionManifest, FunctionTrigger, type FunctionTriggerType, type TriggerSpec } from '../types';\n\ntype ResponseCode = number;\n\nexport type TriggerCallback = (args: object) => Promise<ResponseCode>;\n\n// TODO(burdon): Make object?\nexport type TriggerFactory<Spec extends TriggerSpec, Options = any> = (\n ctx: Context,\n space: Space,\n spec: Spec,\n callback: TriggerCallback,\n options?: Options,\n) => Promise<void>;\n\nexport type TriggerFactoryMap = Record<FunctionTriggerType, TriggerFactory<any>>;\n\nconst triggerFactory: TriggerFactoryMap = {\n subscription: createSubscriptionTrigger,\n timer: createTimerTrigger,\n // TODO(burdon): Cannot use in browser.\n // webhook: createWebhookTrigger,\n webhook: null as any,\n websocket: createWebsocketTrigger,\n};\n\nexport type TriggerEvent = {\n space: Space;\n triggers: FunctionTrigger[];\n};\n\ntype RegisteredTrigger = {\n activationCtx?: Context;\n trigger: FunctionTrigger;\n};\n\nexport class TriggerRegistry extends Resource {\n private readonly _triggersBySpaceKey = new ComplexMap<PublicKey, RegisteredTrigger[]>(PublicKey.hash);\n\n public readonly registered = new Event<TriggerEvent>();\n public readonly removed = new Event<TriggerEvent>();\n\n constructor(\n private readonly _client: Client,\n private readonly _options?: TriggerFactoryMap,\n ) {\n super();\n }\n\n public getActiveTriggers(space: Space): FunctionTrigger[] {\n return this._getTriggers(space, (t) => t.activationCtx != null);\n }\n\n public getInactiveTriggers(space: Space): FunctionTrigger[] {\n return this._getTriggers(space, (t) => t.activationCtx == null);\n }\n\n /**\n * Set callback for trigger.\n */\n public async activate(space: Space, trigger: FunctionTrigger, callback: TriggerCallback): Promise<void> {\n log('activate', { space: space.key, trigger });\n\n const activationCtx = new Context({ name: `FunctionTrigger-${trigger.function}` });\n this._ctx.onDispose(() => activationCtx.dispose());\n const registeredTrigger = this._triggersBySpaceKey.get(space.key)?.find((reg) => reg.trigger.id === trigger.id);\n invariant(registeredTrigger, `Trigger is not registered: ${trigger.function}`);\n registeredTrigger.activationCtx = activationCtx;\n\n try {\n // Create trigger.\n const options = this._options?.[trigger.spec.type];\n const createTrigger = triggerFactory[trigger.spec.type];\n await createTrigger(activationCtx, space, trigger.spec, callback, options);\n } catch (err) {\n delete registeredTrigger.activationCtx;\n throw err;\n }\n }\n\n /**\n * Loads triggers from the manifest into the space.\n */\n public async register(space: Space, manifest: FunctionManifest): Promise<void> {\n log('register', { space: space.key });\n if (!manifest.triggers?.length) {\n return;\n }\n\n if (!space.db.graph.schemaRegistry.hasSchema(FunctionTrigger)) {\n space.db.graph.schemaRegistry.addSchema([FunctionTrigger]);\n }\n\n // Create FK to enable syncing if none are set (NOTE: Possible collision).\n const manifestTriggers = manifest.triggers.map((trigger) => {\n let keys = trigger[ECHO_ATTR_META]?.keys;\n delete trigger[ECHO_ATTR_META];\n if (!keys?.length) {\n keys = [foreignKey('manifest', [trigger.function, trigger.spec.type].join(':'))];\n }\n\n return create(FunctionTrigger, trigger, { keys });\n });\n\n // Sync triggers.\n const { objects: existing } = await space.db.query(Filter.schema(FunctionTrigger)).run();\n const { added } = diff(existing, manifestTriggers, compareForeignKeys);\n\n // TODO(burdon): Update existing.\n added.forEach((trigger) => {\n space.db.add(trigger);\n log.info('added', { meta: getMeta(trigger) });\n });\n\n if (added.length > 0) {\n await space.db.flush();\n }\n }\n\n protected override async _open(): Promise<void> {\n log.info('open...');\n const spaceListSubscription = this._client.spaces.subscribe(async (spaces) => {\n for (const space of spaces) {\n if (this._triggersBySpaceKey.has(space.key)) {\n continue;\n }\n\n const registered: RegisteredTrigger[] = [];\n this._triggersBySpaceKey.set(space.key, registered);\n await space.waitUntilReady();\n if (this._ctx.disposed) {\n break;\n }\n\n // Subscribe to updates.\n this._ctx.onDispose(\n space.db.query(Filter.schema(FunctionTrigger)).subscribe(async ({ objects: current }) => {\n log.info('update', { space: space.key, registered: registered.length, current: current.length });\n await this._handleRemovedTriggers(space, current, registered);\n this._handleNewTriggers(space, current, registered);\n }),\n );\n }\n });\n\n this._ctx.onDispose(() => spaceListSubscription.unsubscribe());\n log.info('opened');\n }\n\n protected override async _close(_: Context): Promise<void> {\n log.info('close...');\n this._triggersBySpaceKey.clear();\n log.info('closed');\n }\n\n private _handleNewTriggers(space: Space, current: FunctionTrigger[], registered: RegisteredTrigger[]) {\n const added = current.filter((candidate) => {\n return candidate.enabled && registered.find((reg) => reg.trigger.id === candidate.id) == null;\n });\n\n if (added.length > 0) {\n const newRegisteredTriggers: RegisteredTrigger[] = added.map((trigger) => ({ trigger }));\n registered.push(...newRegisteredTriggers);\n log.info('added', () => ({\n spaceKey: space.key,\n triggers: added.map((trigger) => trigger.function),\n }));\n\n this.registered.emit({ space, triggers: added });\n }\n }\n\n private async _handleRemovedTriggers(\n space: Space,\n current: FunctionTrigger[],\n registered: RegisteredTrigger[],\n ): Promise<void> {\n const removed: FunctionTrigger[] = [];\n for (let i = registered.length - 1; i >= 0; i--) {\n const wasRemoved =\n current.filter((trigger) => trigger.enabled).find((trigger) => trigger.id === registered[i].trigger.id) == null;\n if (wasRemoved) {\n const unregistered = registered.splice(i, 1)[0];\n await unregistered.activationCtx?.dispose();\n removed.push(unregistered.trigger);\n }\n }\n\n if (removed.length > 0) {\n log.info('removed', () => ({\n spaceKey: space.key,\n triggers: removed.map((trigger) => trigger.function),\n }));\n\n this.removed.emit({ space, triggers: removed });\n }\n }\n\n private _getTriggers(space: Space, predicate: (trigger: RegisteredTrigger) => boolean): FunctionTrigger[] {\n const allSpaceTriggers = this._triggersBySpaceKey.get(space.key) ?? [];\n return allSpaceTriggers.filter(predicate).map((trigger) => trigger.trigger);\n }\n}\n"],
5
+ "mappings": "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",
6
+ "names": ["Event", "create", "Filter", "Resource", "PublicKey", "log", "ComplexMap", "diff", "FunctionRegistry", "Resource", "constructor", "_client", "_functionBySpaceKey", "ComplexMap", "PublicKey", "hash", "registered", "Event", "getFunctions", "space", "get", "key", "getUniqueByUri", "uniqueByUri", "values", "flatMap", "defs", "reduce", "acc", "v", "set", "uri", "Map", "register", "functions", "log", "length", "db", "graph", "schemaRegistry", "hasSchema", "FunctionDef", "addSchema", "objects", "existing", "query", "Filter", "schema", "run", "added", "diff", "a", "b", "forEach", "def", "add", "create", "flush", "indexes", "updates", "_open", "info", "spacesSubscription", "spaces", "subscribe", "has", "waitUntilReady", "_ctx", "disposed", "onDispose", "push", "emit", "unsubscribe", "_close", "_", "clear", "debounce", "UpdateScheduler", "Filter", "createSubscription", "log", "createSubscriptionTrigger", "ctx", "space", "spec", "callback", "objectIds", "Set", "task", "size", "objects", "Array", "from", "clear", "maxFrequency", "subscriptions", "subscription", "added", "updated", "sizeBefore", "object", "add", "id", "info", "length", "trigger", "push", "unsubscribe", "filter", "options", "deep", "delay", "update", "query", "db", "typename", "type", "props", "subscribe", "onDispose", "forEach", "CronJob", "DeferredTask", "log", "createTimerTrigger", "ctx", "space", "spec", "callback", "task", "last", "run", "job", "from", "cronTime", "cron", "runOnInit", "onTick", "now", "Date", "delta", "info", "key", "truncate", "count", "schedule", "start", "onDispose", "stop", "WebSocket", "sleep", "Trigger", "log", "createWebsocketTrigger", "ctx", "space", "spec", "callback", "options", "retryDelay", "maxAttempts", "url", "init", "wasOpen", "ws", "attempt", "open", "Object", "assign", "onopen", "info", "send", "TextEncoder", "encode", "JSON", "stringify", "wake", "onclose", "event", "code", "disposed", "setTimeout", "onerror", "catch", "error", "onmessage", "data", "parse", "TextDecoder", "decode", "err", "isOpen", "wait", "Math", "pow", "warn", "onDispose", "close", "Event", "create", "Filter", "getMeta", "Context", "Resource", "compareForeignKeys", "ECHO_ATTR_META", "foreignKey", "invariant", "PublicKey", "log", "ComplexMap", "diff", "triggerFactory", "subscription", "createSubscriptionTrigger", "timer", "createTimerTrigger", "webhook", "websocket", "createWebsocketTrigger", "TriggerRegistry", "Resource", "constructor", "_client", "_options", "_triggersBySpaceKey", "ComplexMap", "PublicKey", "hash", "registered", "Event", "removed", "getActiveTriggers", "space", "_getTriggers", "t", "activationCtx", "getInactiveTriggers", "activate", "trigger", "callback", "log", "key", "Context", "name", "function", "_ctx", "onDispose", "dispose", "registeredTrigger", "get", "find", "reg", "id", "invariant", "options", "spec", "type", "createTrigger", "err", "register", "manifest", "triggers", "length", "db", "graph", "schemaRegistry", "hasSchema", "FunctionTrigger", "addSchema", "manifestTriggers", "map", "keys", "ECHO_ATTR_META", "foreignKey", "join", "create", "objects", "existing", "query", "Filter", "schema", "run", "added", "diff", "compareForeignKeys", "forEach", "add", "info", "meta", "getMeta", "flush", "_open", "spaceListSubscription", "spaces", "subscribe", "has", "set", "waitUntilReady", "disposed", "current", "_handleRemovedTriggers", "_handleNewTriggers", "unsubscribe", "_close", "_", "clear", "filter", "candidate", "enabled", "newRegisteredTriggers", "push", "spaceKey", "emit", "i", "wasRemoved", "unregistered", "splice", "predicate", "allSpaceTriggers"]
7
+ }
@@ -60,6 +60,7 @@ var FunctionTrigger = class extends TypedObject({
60
60
  typename: "dxos.org/type/FunctionTrigger",
61
61
  version: "0.1.0"
62
62
  })({
63
+ name: S.optional(S.String),
63
64
  enabled: S.optional(S.Boolean),
64
65
  function: S.String.pipe(S.description("Function URI.")),
65
66
  // The `meta` property is merged into the event data passed to the function.
@@ -83,4 +84,4 @@ export {
83
84
  FunctionManifestSchema,
84
85
  FUNCTION_SCHEMA
85
86
  };
86
- //# sourceMappingURL=chunk-YSDC6YCF.mjs.map
87
+ //# sourceMappingURL=chunk-XOBJR3A6.mjs.map