@dxos/functions 0.7.1 → 0.7.2-staging.6d26b2a

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 (53) hide show
  1. package/dist/lib/browser/{chunk-QX2SGAWR.mjs → chunk-2WV7BKNM.mjs} +28 -159
  2. package/dist/lib/browser/chunk-2WV7BKNM.mjs.map +7 -0
  3. package/dist/lib/browser/{chunk-FWIFI2MH.mjs → chunk-MKVZCFKN.mjs} +28 -15
  4. package/dist/lib/browser/chunk-MKVZCFKN.mjs.map +7 -0
  5. package/dist/lib/browser/index.mjs +5 -7
  6. package/dist/lib/browser/index.mjs.map +2 -2
  7. package/dist/lib/browser/meta.json +1 -1
  8. package/dist/lib/browser/testing/index.mjs +5 -3
  9. package/dist/lib/browser/testing/index.mjs.map +3 -3
  10. package/dist/lib/browser/types.mjs +3 -1
  11. package/dist/lib/node/{chunk-OKV7IWWB.cjs → chunk-G2EVL5ME.cjs} +45 -176
  12. package/dist/lib/node/chunk-G2EVL5ME.cjs.map +7 -0
  13. package/dist/lib/node/{chunk-GKDHAWCP.cjs → chunk-XRSOYOZ5.cjs} +31 -17
  14. package/dist/lib/node/chunk-XRSOYOZ5.cjs.map +7 -0
  15. package/dist/lib/node/index.cjs +14 -16
  16. package/dist/lib/node/index.cjs.map +2 -2
  17. package/dist/lib/node/meta.json +1 -1
  18. package/dist/lib/node/testing/index.cjs +13 -11
  19. package/dist/lib/node/testing/index.cjs.map +3 -3
  20. package/dist/lib/node/types.cjs +9 -7
  21. package/dist/lib/node/types.cjs.map +2 -2
  22. package/dist/lib/node-esm/{chunk-SPHZ5SOR.mjs → chunk-PCEHXPSI.mjs} +28 -159
  23. package/dist/lib/node-esm/chunk-PCEHXPSI.mjs.map +7 -0
  24. package/dist/lib/node-esm/{chunk-YSKH6PBN.mjs → chunk-UZVXZLDV.mjs} +28 -15
  25. package/dist/lib/node-esm/chunk-UZVXZLDV.mjs.map +7 -0
  26. package/dist/lib/node-esm/index.mjs +5 -7
  27. package/dist/lib/node-esm/index.mjs.map +2 -2
  28. package/dist/lib/node-esm/meta.json +1 -1
  29. package/dist/lib/node-esm/testing/index.mjs +5 -3
  30. package/dist/lib/node-esm/testing/index.mjs.map +3 -3
  31. package/dist/lib/node-esm/types.mjs +3 -1
  32. package/dist/types/src/handler.d.ts +2 -2
  33. package/dist/types/src/trigger/type/index.d.ts +0 -1
  34. package/dist/types/src/trigger/type/index.d.ts.map +1 -1
  35. package/dist/types/src/types.d.ts +43 -77
  36. package/dist/types/src/types.d.ts.map +1 -1
  37. package/package.json +15 -15
  38. package/src/handler.ts +3 -3
  39. package/src/runtime/scheduler.test.ts +4 -55
  40. package/src/runtime/scheduler.ts +1 -1
  41. package/src/testing/functions-integration.test.ts +2 -2
  42. package/src/trigger/trigger-registry.test.ts +4 -4
  43. package/src/trigger/type/index.ts +0 -1
  44. package/src/types.ts +26 -24
  45. package/dist/lib/browser/chunk-FWIFI2MH.mjs.map +0 -7
  46. package/dist/lib/browser/chunk-QX2SGAWR.mjs.map +0 -7
  47. package/dist/lib/node/chunk-GKDHAWCP.cjs.map +0 -7
  48. package/dist/lib/node/chunk-OKV7IWWB.cjs.map +0 -7
  49. package/dist/lib/node-esm/chunk-SPHZ5SOR.mjs.map +0 -7
  50. package/dist/lib/node-esm/chunk-YSKH6PBN.mjs.map +0 -7
  51. package/dist/types/src/trigger/type/websocket-trigger.d.ts +0 -14
  52. package/dist/types/src/trigger/type/websocket-trigger.d.ts.map +0 -1
  53. package/src/trigger/type/websocket-trigger.ts +0 -112
@@ -2,7 +2,7 @@ import "@dxos/node-std/globals";
2
2
  import {
3
3
  FunctionDef,
4
4
  FunctionTrigger
5
- } from "./chunk-FWIFI2MH.mjs";
5
+ } from "./chunk-MKVZCFKN.mjs";
6
6
 
7
7
  // packages/core/functions/src/function/function-registry.ts
8
8
  import { Event } from "@dxos/async";
@@ -217,135 +217,6 @@ var createTimerTrigger = async (ctx, space, spec, callback) => {
217
217
  ctx.onDispose(() => job.stop());
218
218
  };
219
219
 
220
- // packages/core/functions/src/trigger/type/websocket-trigger.ts
221
- import { sleep, Trigger } from "@dxos/async";
222
- import { log as log4 } from "@dxos/log";
223
- var __dxlog_file4 = "/home/runner/work/dxos/dxos/packages/core/functions/src/trigger/type/websocket-trigger.ts";
224
- var createWebsocketTrigger = async (ctx, space, spec, callback, options = {
225
- retryDelay: 2,
226
- maxAttempts: 5
227
- }) => {
228
- const { url, init } = spec;
229
- let wasOpen = false;
230
- let ws;
231
- for (let attempt = 1; attempt <= options.maxAttempts; attempt++) {
232
- const open = new Trigger();
233
- ws = await createWebSocket(url);
234
- Object.assign(ws, {
235
- onopen: () => {
236
- log4.info("opened", {
237
- url
238
- }, {
239
- F: __dxlog_file4,
240
- L: 39,
241
- S: void 0,
242
- C: (f, a) => f(...a)
243
- });
244
- if (spec.init) {
245
- ws.send(new TextEncoder().encode(JSON.stringify(init)));
246
- }
247
- open.wake(true);
248
- },
249
- onclose: (event) => {
250
- log4.info("closed", {
251
- url,
252
- code: event.code
253
- }, {
254
- F: __dxlog_file4,
255
- L: 48,
256
- S: void 0,
257
- C: (f, a) => f(...a)
258
- });
259
- if (event.code === 1006 && wasOpen && !ctx.disposed) {
260
- setTimeout(async () => {
261
- log4.info(`reconnecting in ${options.retryDelay}s...`, {
262
- url
263
- }, {
264
- F: __dxlog_file4,
265
- L: 53,
266
- S: void 0,
267
- C: (f, a) => f(...a)
268
- });
269
- await createWebsocketTrigger(ctx, space, spec, callback, options);
270
- }, options.retryDelay * 1e3);
271
- }
272
- open.wake(false);
273
- },
274
- onerror: (event) => {
275
- log4.catch(event.error ?? new Error("Unspecified ws error."), {
276
- url
277
- }, {
278
- F: __dxlog_file4,
279
- L: 61,
280
- S: void 0,
281
- C: (f, a) => f(...a)
282
- });
283
- open.wake(false);
284
- },
285
- onmessage: async (event) => {
286
- try {
287
- log4.info("message", void 0, {
288
- F: __dxlog_file4,
289
- L: 67,
290
- S: void 0,
291
- C: (f, a) => f(...a)
292
- });
293
- let data;
294
- if (event.data && "text" in event.data) {
295
- data = JSON.parse(await event.data.text());
296
- } else {
297
- data = JSON.parse(new TextDecoder().decode(event.data));
298
- }
299
- await callback({
300
- data
301
- });
302
- } catch (err) {
303
- log4.catch(err, {
304
- url,
305
- data: event.data
306
- }, {
307
- F: __dxlog_file4,
308
- L: 76,
309
- S: void 0,
310
- C: (f, a) => f(...a)
311
- });
312
- }
313
- }
314
- });
315
- const isOpen = await open.wait();
316
- if (ctx.disposed) {
317
- break;
318
- }
319
- if (isOpen) {
320
- wasOpen = true;
321
- break;
322
- }
323
- const wait = Math.pow(attempt, 2) * options.retryDelay;
324
- if (attempt < options.maxAttempts) {
325
- log4.warn(`failed to connect; trying again in ${wait}s`, {
326
- attempt
327
- }, {
328
- F: __dxlog_file4,
329
- L: 91,
330
- S: void 0,
331
- C: (f, a) => f(...a)
332
- });
333
- await sleep(wait * 1e3);
334
- }
335
- }
336
- ctx.onDispose(() => {
337
- ws?.close();
338
- });
339
- };
340
- var createNodeWebSocket = async (url) => {
341
- const importESM = Function("path", "return import(path)");
342
- const { default: { WebSocket: WebSocket1 } } = await importESM("ws");
343
- return new WebSocket1(url);
344
- };
345
- var createWebSocket = async (url) => {
346
- return typeof globalThis.WebSocket === "undefined" ? await createNodeWebSocket(url) : new WebSocket(url);
347
- };
348
-
349
220
  // packages/core/functions/src/trigger/trigger-registry.ts
350
221
  import { Event as Event2 } from "@dxos/async";
351
222
  import { create as create2, Filter as Filter3, getMeta } from "@dxos/client/echo";
@@ -353,9 +224,9 @@ import { Context, Resource as Resource2 } from "@dxos/context";
353
224
  import { compareForeignKeys, ECHO_ATTR_META, foreignKey } from "@dxos/echo-schema";
354
225
  import { invariant } from "@dxos/invariant";
355
226
  import { PublicKey as PublicKey2 } from "@dxos/keys";
356
- import { log as log5 } from "@dxos/log";
227
+ import { log as log4 } from "@dxos/log";
357
228
  import { ComplexMap as ComplexMap2, diff as diff2 } from "@dxos/util";
358
- var __dxlog_file5 = "/home/runner/work/dxos/dxos/packages/core/functions/src/trigger/trigger-registry.ts";
229
+ var __dxlog_file4 = "/home/runner/work/dxos/dxos/packages/core/functions/src/trigger/trigger-registry.ts";
359
230
  var triggerFactory = {
360
231
  timer: createTimerTrigger,
361
232
  // TODO(burdon): Cannot use in browser.
@@ -381,11 +252,11 @@ var TriggerRegistry = class extends Resource2 {
381
252
  * Set callback for trigger.
382
253
  */
383
254
  async activate(space, trigger, callback) {
384
- log5("activate", {
255
+ log4("activate", {
385
256
  space: space.key,
386
257
  trigger
387
258
  }, {
388
- F: __dxlog_file5,
259
+ F: __dxlog_file4,
389
260
  L: 75,
390
261
  S: this,
391
262
  C: (f, a) => f(...a)
@@ -393,13 +264,13 @@ var TriggerRegistry = class extends Resource2 {
393
264
  const activationCtx = new Context({
394
265
  name: `FunctionTrigger-${trigger.function}`
395
266
  }, {
396
- F: __dxlog_file5,
267
+ F: __dxlog_file4,
397
268
  L: 77
398
269
  });
399
270
  this._ctx.onDispose(() => activationCtx.dispose());
400
271
  const registeredTrigger = this._triggersBySpaceKey.get(space.key)?.find((reg) => reg.trigger.id === trigger.id);
401
272
  invariant(registeredTrigger, `Trigger is not registered: ${trigger.function}`, {
402
- F: __dxlog_file5,
273
+ F: __dxlog_file4,
403
274
  L: 80,
404
275
  S: this,
405
276
  A: [
@@ -410,7 +281,7 @@ var TriggerRegistry = class extends Resource2 {
410
281
  registeredTrigger.activationCtx = activationCtx;
411
282
  try {
412
283
  invariant(trigger.spec, void 0, {
413
- F: __dxlog_file5,
284
+ F: __dxlog_file4,
414
285
  L: 85,
415
286
  S: this,
416
287
  A: [
@@ -421,7 +292,7 @@ var TriggerRegistry = class extends Resource2 {
421
292
  const options = this._options?.[trigger.spec.type];
422
293
  const createTrigger = triggerFactory[trigger.spec.type];
423
294
  invariant(createTrigger, `Trigger factory not found: ${trigger.spec.type}`, {
424
- F: __dxlog_file5,
295
+ F: __dxlog_file4,
425
296
  L: 88,
426
297
  S: this,
427
298
  A: [
@@ -439,10 +310,10 @@ var TriggerRegistry = class extends Resource2 {
439
310
  * Loads triggers from the manifest into the space.
440
311
  */
441
312
  async register(space, manifest) {
442
- log5("register", {
313
+ log4("register", {
443
314
  space: space.key
444
315
  }, {
445
- F: __dxlog_file5,
316
+ F: __dxlog_file4,
446
317
  L: 100,
447
318
  S: this,
448
319
  C: (f, a) => f(...a)
@@ -474,10 +345,10 @@ var TriggerRegistry = class extends Resource2 {
474
345
  const { added } = diff2(existing, manifestTriggers, compareForeignKeys);
475
346
  added.forEach((trigger) => {
476
347
  space.db.add(trigger);
477
- log5.info("added", {
348
+ log4.info("added", {
478
349
  meta: getMeta(trigger)
479
350
  }, {
480
- F: __dxlog_file5,
351
+ F: __dxlog_file4,
481
352
  L: 127,
482
353
  S: this,
483
354
  C: (f, a) => f(...a)
@@ -488,8 +359,8 @@ var TriggerRegistry = class extends Resource2 {
488
359
  }
489
360
  }
490
361
  async _open() {
491
- log5.info("open...", void 0, {
492
- F: __dxlog_file5,
362
+ log4.info("open...", void 0, {
363
+ F: __dxlog_file4,
493
364
  L: 136,
494
365
  S: this,
495
366
  C: (f, a) => f(...a)
@@ -506,12 +377,12 @@ var TriggerRegistry = class extends Resource2 {
506
377
  break;
507
378
  }
508
379
  this._ctx.onDispose(space.db.query(Filter3.schema(FunctionTrigger)).subscribe(async ({ objects: current }) => {
509
- log5.info("update", {
380
+ log4.info("update", {
510
381
  space: space.key,
511
382
  registered: registered.length,
512
383
  current: current.length
513
384
  }, {
514
- F: __dxlog_file5,
385
+ F: __dxlog_file4,
515
386
  L: 153,
516
387
  S: this,
517
388
  C: (f, a) => f(...a)
@@ -522,23 +393,23 @@ var TriggerRegistry = class extends Resource2 {
522
393
  }
523
394
  });
524
395
  this._ctx.onDispose(() => spaceListSubscription.unsubscribe());
525
- log5.info("opened", void 0, {
526
- F: __dxlog_file5,
396
+ log4.info("opened", void 0, {
397
+ F: __dxlog_file4,
527
398
  L: 162,
528
399
  S: this,
529
400
  C: (f, a) => f(...a)
530
401
  });
531
402
  }
532
403
  async _close(_) {
533
- log5.info("close...", void 0, {
534
- F: __dxlog_file5,
404
+ log4.info("close...", void 0, {
405
+ F: __dxlog_file4,
535
406
  L: 166,
536
407
  S: this,
537
408
  C: (f, a) => f(...a)
538
409
  });
539
410
  this._triggersBySpaceKey.clear();
540
- log5.info("closed", void 0, {
541
- F: __dxlog_file5,
411
+ log4.info("closed", void 0, {
412
+ F: __dxlog_file4,
542
413
  L: 168,
543
414
  S: this,
544
415
  C: (f, a) => f(...a)
@@ -553,11 +424,11 @@ var TriggerRegistry = class extends Resource2 {
553
424
  trigger
554
425
  }));
555
426
  registered.push(...newRegisteredTriggers);
556
- log5.info("added", () => ({
427
+ log4.info("added", () => ({
557
428
  spaceKey: space.key,
558
429
  triggers: added.map((trigger) => trigger.function)
559
430
  }), {
560
- F: __dxlog_file5,
431
+ F: __dxlog_file4,
561
432
  L: 179,
562
433
  S: this,
563
434
  C: (f, a) => f(...a)
@@ -579,11 +450,11 @@ var TriggerRegistry = class extends Resource2 {
579
450
  }
580
451
  }
581
452
  if (removed.length > 0) {
582
- log5.info("removed", () => ({
453
+ log4.info("removed", () => ({
583
454
  spaceKey: space.key,
584
455
  triggers: removed.map((trigger) => trigger.function)
585
456
  }), {
586
- F: __dxlog_file5,
457
+ F: __dxlog_file4,
587
458
  L: 205,
588
459
  S: this,
589
460
  C: (f, a) => f(...a)
@@ -604,8 +475,6 @@ export {
604
475
  FunctionRegistry,
605
476
  createSubscriptionTrigger,
606
477
  createTimerTrigger,
607
- createWebsocketTrigger,
608
- createWebSocket,
609
478
  TriggerRegistry
610
479
  };
611
- //# sourceMappingURL=chunk-QX2SGAWR.mjs.map
480
+ //# sourceMappingURL=chunk-2WV7BKNM.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/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.type) {\n const query = space.db.query(Filter.typename(filter.type, filter.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 { 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 } from './type';\nimport { type FunctionManifest, FunctionTrigger, type TriggerKind, type TriggerType } 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 TriggerType, Options = any> = (\n ctx: Context,\n space: Space,\n spec: Spec,\n callback: TriggerCallback,\n options?: Options,\n) => Promise<void>;\n\nexport type TriggerFactoryMap = Partial<Record<TriggerKind, TriggerFactory<any>>>;\n\nconst triggerFactory: TriggerFactoryMap = {\n timer: createTimerTrigger,\n // TODO(burdon): Cannot use in browser.\n // webhook: createWebhookTrigger,\n subscription: createSubscriptionTrigger,\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 invariant(trigger.spec);\n const options = this._options?.[trigger.spec.type];\n const createTrigger = triggerFactory[trigger.spec.type];\n invariant(createTrigger, `Trigger factory not found: ${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", "type", "query", "db", "typename", "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", "Event", "create", "Filter", "getMeta", "Context", "Resource", "compareForeignKeys", "ECHO_ATTR_META", "foreignKey", "invariant", "PublicKey", "log", "ComplexMap", "diff", "triggerFactory", "timer", "createTimerTrigger", "subscription", "createSubscriptionTrigger", "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", "spec", "options", "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
+ }
@@ -7,29 +7,36 @@ var __require = /* @__PURE__ */ ((x) => typeof require !== "undefined" ? require
7
7
  });
8
8
 
9
9
  // packages/core/functions/src/types.ts
10
- import { AST, RawObject, S, TypedObject } from "@dxos/echo-schema";
10
+ import { AST, OptionsAnnotationId, RawObject, S, TypedObject } from "@dxos/echo-schema";
11
+ var TriggerKind;
12
+ (function(TriggerKind2) {
13
+ TriggerKind2["Timer"] = "timer";
14
+ TriggerKind2["Webhook"] = "webhook";
15
+ TriggerKind2["Subscription"] = "subscription";
16
+ })(TriggerKind || (TriggerKind = {}));
11
17
  var typeLiteralAnnotations = {
12
18
  [AST.TitleAnnotationId]: "Type"
13
19
  };
14
20
  var TimerTriggerSchema = S.Struct({
15
21
  type: S.Literal("timer").annotations(typeLiteralAnnotations),
16
- cron: S.NonEmptyString.annotations({
22
+ cron: S.String.annotations({
17
23
  [AST.TitleAnnotationId]: "Cron",
18
- [AST.ExamplesAnnotationId]: "0 0 * * *"
24
+ [AST.ExamplesAnnotationId]: [
25
+ "0 0 * * *"
26
+ ]
19
27
  })
20
28
  }).pipe(S.mutable);
21
29
  var WebhookTriggerSchema = S.Struct({
22
30
  type: S.Literal("webhook").annotations(typeLiteralAnnotations),
23
- method: S.String,
24
- // Assigned port.
25
- port: S.optional(S.Number)
26
- }).pipe(S.mutable);
27
- var WebsocketTriggerSchema = S.Struct({
28
- type: S.Literal("websocket").annotations(typeLiteralAnnotations),
29
- url: S.String,
30
- init: S.optional(S.Record({
31
- key: S.String,
32
- value: S.Any
31
+ method: S.optional(S.String.annotations({
32
+ [AST.TitleAnnotationId]: "Method",
33
+ [OptionsAnnotationId]: [
34
+ "GET",
35
+ "POST"
36
+ ]
37
+ })),
38
+ port: S.optional(S.Number.annotations({
39
+ [AST.TitleAnnotationId]: "Port"
33
40
  }))
34
41
  }).pipe(S.mutable);
35
42
  var QuerySchema = S.Struct({
@@ -60,7 +67,12 @@ var SubscriptionTriggerSchema = S.Struct({
60
67
  [AST.TitleAnnotationId]: "Options"
61
68
  }))
62
69
  }).pipe(S.mutable);
63
- var TriggerSchema = S.Union(TimerTriggerSchema, WebhookTriggerSchema, WebsocketTriggerSchema, SubscriptionTriggerSchema).annotations({
70
+ var TriggerSchema = S.Union(
71
+ //
72
+ TimerTriggerSchema,
73
+ WebhookTriggerSchema,
74
+ SubscriptionTriggerSchema
75
+ ).annotations({
64
76
  [AST.TitleAnnotationId]: "Trigger"
65
77
  });
66
78
  var FunctionTriggerSchema = S.Struct({
@@ -106,6 +118,7 @@ var FUNCTION_TYPES = [
106
118
 
107
119
  export {
108
120
  __require,
121
+ TriggerKind,
109
122
  TriggerSchema,
110
123
  FunctionTriggerSchema,
111
124
  FunctionTrigger,
@@ -113,4 +126,4 @@ export {
113
126
  FunctionManifestSchema,
114
127
  FUNCTION_TYPES
115
128
  };
116
- //# sourceMappingURL=chunk-FWIFI2MH.mjs.map
129
+ //# sourceMappingURL=chunk-MKVZCFKN.mjs.map
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../../src/types.ts"],
4
+ "sourcesContent": ["//\n// Copyright 2023 DXOS.org\n//\n\nimport { AST, OptionsAnnotationId, RawObject, S, TypedObject } from '@dxos/echo-schema';\n\n/**\n * Type discriminator for TriggerType.\n * Every spec has a type field of type TriggerKind that we can use to understand which type we're working with.\n * https://www.typescriptlang.org/docs/handbook/2/narrowing.html#discriminated-unions\n */\nexport enum TriggerKind {\n Timer = 'timer',\n Webhook = 'webhook',\n Subscription = 'subscription',\n}\n\n// TODO(burdon): Rename prop kind.\nconst typeLiteralAnnotations = { [AST.TitleAnnotationId]: 'Type' };\n\n/**\n * Cron timer.\n */\nconst TimerTriggerSchema = S.Struct({\n type: S.Literal(TriggerKind.Timer).annotations(typeLiteralAnnotations),\n cron: S.String.annotations({\n [AST.TitleAnnotationId]: 'Cron',\n [AST.ExamplesAnnotationId]: ['0 0 * * *'],\n }),\n}).pipe(S.mutable);\n\nexport type TimerTrigger = S.Schema.Type<typeof TimerTriggerSchema>;\n\n/**\n * Webhook.\n */\nconst WebhookTriggerSchema = S.Struct({\n type: S.Literal(TriggerKind.Webhook).annotations(typeLiteralAnnotations),\n method: S.optional(\n S.String.annotations({\n [AST.TitleAnnotationId]: 'Method',\n [OptionsAnnotationId]: ['GET', 'POST'],\n }),\n ),\n port: S.optional(\n S.Number.annotations({\n [AST.TitleAnnotationId]: 'Port',\n }),\n ),\n}).pipe(S.mutable);\n\nexport type WebhookTrigger = S.Schema.Type<typeof WebhookTriggerSchema>;\n\n// TODO(burdon): Use ECHO definition (from https://github.com/dxos/dxos/pull/8233).\nconst QuerySchema = S.Struct({\n type: S.optional(S.String.annotations({ [AST.TitleAnnotationId]: 'Type' })),\n props: S.optional(S.Record({ key: S.String, value: S.Any })),\n}).annotations({ [AST.TitleAnnotationId]: 'Query' });\n\n/**\n * Subscription.\n */\nconst SubscriptionTriggerSchema = S.Struct({\n type: S.Literal(TriggerKind.Subscription).annotations(typeLiteralAnnotations),\n // TODO(burdon): Define query DSL (from ECHO). Reconcile with Table.Query.\n filter: QuerySchema,\n options: S.optional(\n S.Struct({\n // Watch changes to object (not just creation).\n deep: S.optional(S.Boolean.annotations({ [AST.TitleAnnotationId]: 'Nested' })),\n // Debounce changes (delay in ms).\n delay: S.optional(S.Number.annotations({ [AST.TitleAnnotationId]: 'Delay' })),\n }).annotations({ [AST.TitleAnnotationId]: 'Options' }),\n ),\n}).pipe(S.mutable);\n\nexport type SubscriptionTrigger = S.Schema.Type<typeof SubscriptionTriggerSchema>;\n\n/**\n * Trigger schema (discriminated union).\n */\nexport const TriggerSchema = S.Union(\n //\n TimerTriggerSchema,\n WebhookTriggerSchema,\n SubscriptionTriggerSchema,\n).annotations({\n [AST.TitleAnnotationId]: 'Trigger',\n});\n\nexport type TriggerType = S.Schema.Type<typeof TriggerSchema>;\n\n/**\n * Function trigger.\n */\nexport const FunctionTriggerSchema = S.Struct({\n // TODO(burdon): What type does this reference.\n function: S.optional(S.String.annotations({ [AST.TitleAnnotationId]: 'Function' })),\n enabled: S.optional(S.Boolean.annotations({ [AST.TitleAnnotationId]: 'Enabled' })),\n\n // TODO(burdon): Flatten?\n spec: S.optional(TriggerSchema),\n\n // TODO(burdon): Get meta from function.\n // The `meta` property is merged into the event data passed to the function.\n meta: S.optional(S.mutable(S.Record({ key: S.String, value: S.Any }))),\n});\n\nexport type FunctionTriggerType = S.Schema.Type<typeof FunctionTriggerSchema>;\n\n/**\n * Function trigger.\n */\nexport class FunctionTrigger extends TypedObject({\n typename: 'dxos.org/type/FunctionTrigger',\n version: '0.1.0',\n})(FunctionTriggerSchema.fields) {}\n\n/**\n * Function definition.\n * @deprecated (Use dxos.org/type/Function)\n */\n// TODO(burdon): Reconcile with FunctionType.\nexport class FunctionDef extends TypedObject({\n typename: 'dxos.org/type/FunctionDef',\n version: '0.1.0',\n})({\n uri: S.String,\n description: S.optional(S.String),\n route: S.String,\n handler: S.String,\n}) {}\n\n/**\n * Function manifest file.\n */\nexport const FunctionManifestSchema = S.Struct({\n functions: S.optional(S.mutable(S.Array(RawObject(FunctionDef)))),\n triggers: S.optional(S.mutable(S.Array(RawObject(FunctionTrigger)))),\n});\n\nexport type FunctionManifest = S.Schema.Type<typeof FunctionManifestSchema>;\n\nexport const FUNCTION_TYPES = [FunctionDef, FunctionTrigger];\n"],
5
+ "mappings": "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",
6
+ "names": ["AST", "OptionsAnnotationId", "RawObject", "S", "TypedObject", "TriggerKind", "typeLiteralAnnotations", "AST", "TitleAnnotationId", "TimerTriggerSchema", "S", "Struct", "type", "Literal", "annotations", "cron", "String", "ExamplesAnnotationId", "pipe", "mutable", "WebhookTriggerSchema", "method", "optional", "OptionsAnnotationId", "port", "Number", "QuerySchema", "props", "Record", "key", "value", "Any", "SubscriptionTriggerSchema", "filter", "options", "deep", "Boolean", "delay", "TriggerSchema", "Union", "FunctionTriggerSchema", "function", "enabled", "spec", "meta", "FunctionTrigger", "TypedObject", "typename", "version", "fields", "FunctionDef", "uri", "description", "route", "handler", "FunctionManifestSchema", "functions", "Array", "RawObject", "triggers", "FUNCTION_TYPES"]
7
+ }
@@ -3,18 +3,17 @@ import {
3
3
  FunctionRegistry,
4
4
  TriggerRegistry,
5
5
  createSubscriptionTrigger,
6
- createTimerTrigger,
7
- createWebSocket,
8
- createWebsocketTrigger
9
- } from "./chunk-QX2SGAWR.mjs";
6
+ createTimerTrigger
7
+ } from "./chunk-2WV7BKNM.mjs";
10
8
  import {
11
9
  FUNCTION_TYPES,
12
10
  FunctionDef,
13
11
  FunctionManifestSchema,
14
12
  FunctionTrigger,
15
13
  FunctionTriggerSchema,
14
+ TriggerKind,
16
15
  TriggerSchema
17
- } from "./chunk-FWIFI2MH.mjs";
16
+ } from "./chunk-MKVZCFKN.mjs";
18
17
 
19
18
  // packages/core/functions/src/handler.ts
20
19
  import { PublicKey } from "@dxos/client";
@@ -87,12 +86,11 @@ export {
87
86
  FunctionRegistry,
88
87
  FunctionTrigger,
89
88
  FunctionTriggerSchema,
89
+ TriggerKind,
90
90
  TriggerRegistry,
91
91
  TriggerSchema,
92
92
  createSubscriptionTrigger,
93
93
  createTimerTrigger,
94
- createWebSocket,
95
- createWebsocketTrigger,
96
94
  subscriptionHandler
97
95
  };
98
96
  //# sourceMappingURL=index.mjs.map
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../src/handler.ts"],
4
- "sourcesContent": ["//\n// Copyright 2023 DXOS.org\n//\n\nimport { type Schema as S } from '@effect/schema';\n\nimport { type Client, PublicKey } from '@dxos/client';\nimport { type Space, type SpaceId } from '@dxos/client/echo';\nimport type { CoreDatabase, EchoReactiveObject } from '@dxos/echo-db';\nimport { log } from '@dxos/log';\nimport { nonNullable } from '@dxos/util';\n\n// TODO(burdon): Model after http request. Ref Lambda/OpenFaaS.\n// https://docs.aws.amazon.com/lambda/latest/dg/typescript-handler.html\n// https://www.serverless.com/framework/docs/providers/aws/guide/serverless.yml/#functions\n// https://www.npmjs.com/package/aws-lambda\n\n/**\n * Function handler.\n */\nexport type FunctionHandler<TData = {}, TMeta = {}> = (params: {\n context: FunctionContext;\n event: FunctionEvent<TData, TMeta>;\n\n /**\n * @deprecated\n */\n response: FunctionResponse;\n}) => Promise<Response | FunctionResponse | void>;\n\n/**\n * Function context.\n */\nexport interface FunctionContext {\n getSpace: (spaceId: SpaceId) => Promise<SpaceAPI>;\n\n /**\n * Space from which the function was invoked.\n */\n space: SpaceAPI | undefined;\n\n ai: FunctionContextAi;\n\n /**\n * @deprecated\n */\n // TODO(burdon): Limit access to individual space.\n client: Client;\n /**\n * @deprecated\n */\n // TODO(burdon): Replace with storage service abstraction.\n dataDir?: string;\n}\n\nexport interface FunctionContextAi {\n // TODO(dmaretskyi): Refer to cloudflare AI docs for more comprehensive typedefs.\n run(model: string, inputs: any, options?: any): Promise<any>;\n}\n\n/**\n * Event payload.\n */\nexport type FunctionEvent<TData = {}, TMeta = {}> = {\n data: FunctionEventMeta<TMeta> & TData;\n};\n\n/**\n * Metadata from trigger.\n */\nexport type FunctionEventMeta<TMeta = {}> = {\n meta: TMeta;\n};\n\n/**\n * Function response.\n */\nexport type FunctionResponse = {\n status(code: number): FunctionResponse;\n};\n\n//\n// API.\n//\n\n/**\n * Space interface available to functions.\n */\nexport interface SpaceAPI {\n get id(): SpaceId;\n get crud(): CoreDatabase;\n}\n\nconst __assertFunctionSpaceIsCompatibleWithTheClientSpace = () => {\n // eslint-disable-next-line unused-imports/no-unused-vars\n const y: SpaceAPI = {} as Space;\n};\n\n//\n// Subscription utils.\n//\n\nexport type RawSubscriptionData = {\n spaceKey?: string;\n objects?: string[];\n};\n\nexport type SubscriptionData = {\n space?: Space;\n objects?: EchoReactiveObject<any>[];\n};\n\n/**\n * Handler wrapper for subscription events; extracts space and objects.\n *\n * To test:\n * ```\n * curl -s -X POST -H \"Content-Type: application/json\" --data '{\"space\": \"0446...1cbb\"}' http://localhost:7100/dev/email-extractor\n * ```\n *\n * NOTE: Get space key from devtools or `dx space list --json`\n */\n// TODO(burdon): Evolve into plugin definition like Composer.\nexport const subscriptionHandler = <TMeta>(\n handler: FunctionHandler<SubscriptionData, TMeta>,\n types?: S.Schema<any>[],\n): FunctionHandler<RawSubscriptionData, TMeta> => {\n return async ({ event: { data }, context, response, ...rest }) => {\n const { client } = context;\n const space = data.spaceKey ? client.spaces.get(PublicKey.from(data.spaceKey)) : undefined;\n if (!space) {\n log.error('Invalid space');\n return response.status(500);\n }\n\n registerTypes(space, types);\n const objects = space\n ? data.objects?.map<EchoReactiveObject<any> | undefined>((id) => space!.db.getObjectById(id)).filter(nonNullable)\n : [];\n\n if (!!data.spaceKey && !space) {\n log.warn('invalid space', { data });\n } else {\n log.info('handler', { space: space?.key.truncate(), objects: objects?.length });\n }\n\n return handler({ event: { data: { ...data, space, objects } }, context, response, ...rest });\n };\n};\n\n// TODO(burdon): Evolve types as part of function metadata.\nconst registerTypes = (space: Space, types: S.Schema<any>[] = []) => {\n const registry = space.db.graph.schemaRegistry;\n for (const type of types) {\n if (!registry.hasSchema(type)) {\n registry.addSchema([type]);\n }\n }\n};\n"],
5
- "mappings": ";;;;;;;;;;;;;;;;;;;AAMA,SAAsBA,iBAAiB;AAGvC,SAASC,WAAW;AACpB,SAASC,mBAAmB;;AAiHrB,IAAMC,sBAAsB,CACjCC,SACAC,UAAAA;AAEA,SAAO,OAAO,EAAEC,OAAO,EAAEC,KAAI,GAAIC,SAASC,UAAU,GAAGC,KAAAA,MAAM;AAC3D,UAAM,EAAEC,OAAM,IAAKH;AACnB,UAAMI,QAAQL,KAAKM,WAAWF,OAAOG,OAAOC,IAAIC,UAAUC,KAAKV,KAAKM,QAAQ,CAAA,IAAKK;AACjF,QAAI,CAACN,OAAO;AACVO,UAAIC,MAAM,iBAAA,QAAA;;;;;;AACV,aAAOX,SAASY,OAAO,GAAA;IACzB;AAEAC,kBAAcV,OAAOP,KAAAA;AACrB,UAAMkB,UAAUX,QACZL,KAAKgB,SAASC,IAAyC,CAACC,OAAOb,MAAOc,GAAGC,cAAcF,EAAAA,CAAAA,EAAKG,OAAOC,WAAAA,IACnG,CAAA;AAEJ,QAAI,CAAC,CAACtB,KAAKM,YAAY,CAACD,OAAO;AAC7BO,UAAIW,KAAK,iBAAiB;QAAEvB;MAAK,GAAA;;;;;;IACnC,OAAO;AACLY,UAAIY,KAAK,WAAW;QAAEnB,OAAOA,OAAOoB,IAAIC,SAAAA;QAAYV,SAASA,SAASW;MAAO,GAAA;;;;;;IAC/E;AAEA,WAAO9B,QAAQ;MAAEE,OAAO;QAAEC,MAAM;UAAE,GAAGA;UAAMK;UAAOW;QAAQ;MAAE;MAAGf;MAASC;MAAU,GAAGC;IAAK,CAAA;EAC5F;AACF;AAGA,IAAMY,gBAAgB,CAACV,OAAcP,QAAyB,CAAA,MAAE;AAC9D,QAAM8B,WAAWvB,MAAMc,GAAGU,MAAMC;AAChC,aAAWC,QAAQjC,OAAO;AACxB,QAAI,CAAC8B,SAASI,UAAUD,IAAAA,GAAO;AAC7BH,eAASK,UAAU;QAACF;OAAK;IAC3B;EACF;AACF;",
4
+ "sourcesContent": ["//\n// Copyright 2023 DXOS.org\n//\n\nimport { type Schema as S } from '@effect/schema';\n\nimport { type Client, PublicKey } from '@dxos/client';\nimport { type Space, type SpaceId } from '@dxos/client/echo';\nimport type { CoreDatabase, ReactiveEchoObject } from '@dxos/echo-db';\nimport { log } from '@dxos/log';\nimport { nonNullable } from '@dxos/util';\n\n// TODO(burdon): Model after http request. Ref Lambda/OpenFaaS.\n// https://docs.aws.amazon.com/lambda/latest/dg/typescript-handler.html\n// https://www.serverless.com/framework/docs/providers/aws/guide/serverless.yml/#functions\n// https://www.npmjs.com/package/aws-lambda\n\n/**\n * Function handler.\n */\nexport type FunctionHandler<TData = {}, TMeta = {}> = (params: {\n context: FunctionContext;\n event: FunctionEvent<TData, TMeta>;\n\n /**\n * @deprecated\n */\n response: FunctionResponse;\n}) => Promise<Response | FunctionResponse | void>;\n\n/**\n * Function context.\n */\nexport interface FunctionContext {\n getSpace: (spaceId: SpaceId) => Promise<SpaceAPI>;\n\n /**\n * Space from which the function was invoked.\n */\n space: SpaceAPI | undefined;\n\n ai: FunctionContextAi;\n\n /**\n * @deprecated\n */\n // TODO(burdon): Limit access to individual space.\n client: Client;\n /**\n * @deprecated\n */\n // TODO(burdon): Replace with storage service abstraction.\n dataDir?: string;\n}\n\nexport interface FunctionContextAi {\n // TODO(dmaretskyi): Refer to cloudflare AI docs for more comprehensive typedefs.\n run(model: string, inputs: any, options?: any): Promise<any>;\n}\n\n/**\n * Event payload.\n */\nexport type FunctionEvent<TData = {}, TMeta = {}> = {\n data: FunctionEventMeta<TMeta> & TData;\n};\n\n/**\n * Metadata from trigger.\n */\nexport type FunctionEventMeta<TMeta = {}> = {\n meta: TMeta;\n};\n\n/**\n * Function response.\n */\nexport type FunctionResponse = {\n status(code: number): FunctionResponse;\n};\n\n//\n// API.\n//\n\n/**\n * Space interface available to functions.\n */\nexport interface SpaceAPI {\n get id(): SpaceId;\n get crud(): CoreDatabase;\n}\n\nconst __assertFunctionSpaceIsCompatibleWithTheClientSpace = () => {\n // eslint-disable-next-line unused-imports/no-unused-vars\n const y: SpaceAPI = {} as Space;\n};\n\n//\n// Subscription utils.\n//\n\nexport type RawSubscriptionData = {\n spaceKey?: string;\n objects?: string[];\n};\n\nexport type SubscriptionData = {\n space?: Space;\n objects?: ReactiveEchoObject<any>[];\n};\n\n/**\n * Handler wrapper for subscription events; extracts space and objects.\n *\n * To test:\n * ```\n * curl -s -X POST -H \"Content-Type: application/json\" --data '{\"space\": \"0446...1cbb\"}' http://localhost:7100/dev/email-extractor\n * ```\n *\n * NOTE: Get space key from devtools or `dx space list --json`\n */\n// TODO(burdon): Evolve into plugin definition like Composer.\nexport const subscriptionHandler = <TMeta>(\n handler: FunctionHandler<SubscriptionData, TMeta>,\n types?: S.Schema<any>[],\n): FunctionHandler<RawSubscriptionData, TMeta> => {\n return async ({ event: { data }, context, response, ...rest }) => {\n const { client } = context;\n const space = data.spaceKey ? client.spaces.get(PublicKey.from(data.spaceKey)) : undefined;\n if (!space) {\n log.error('Invalid space');\n return response.status(500);\n }\n\n registerTypes(space, types);\n const objects = space\n ? data.objects?.map<ReactiveEchoObject<any> | undefined>((id) => space!.db.getObjectById(id)).filter(nonNullable)\n : [];\n\n if (!!data.spaceKey && !space) {\n log.warn('invalid space', { data });\n } else {\n log.info('handler', { space: space?.key.truncate(), objects: objects?.length });\n }\n\n return handler({ event: { data: { ...data, space, objects } }, context, response, ...rest });\n };\n};\n\n// TODO(burdon): Evolve types as part of function metadata.\nconst registerTypes = (space: Space, types: S.Schema<any>[] = []) => {\n const registry = space.db.graph.schemaRegistry;\n for (const type of types) {\n if (!registry.hasSchema(type)) {\n registry.addSchema([type]);\n }\n }\n};\n"],
5
+ "mappings": ";;;;;;;;;;;;;;;;;;AAMA,SAAsBA,iBAAiB;AAGvC,SAASC,WAAW;AACpB,SAASC,mBAAmB;;AAiHrB,IAAMC,sBAAsB,CACjCC,SACAC,UAAAA;AAEA,SAAO,OAAO,EAAEC,OAAO,EAAEC,KAAI,GAAIC,SAASC,UAAU,GAAGC,KAAAA,MAAM;AAC3D,UAAM,EAAEC,OAAM,IAAKH;AACnB,UAAMI,QAAQL,KAAKM,WAAWF,OAAOG,OAAOC,IAAIC,UAAUC,KAAKV,KAAKM,QAAQ,CAAA,IAAKK;AACjF,QAAI,CAACN,OAAO;AACVO,UAAIC,MAAM,iBAAA,QAAA;;;;;;AACV,aAAOX,SAASY,OAAO,GAAA;IACzB;AAEAC,kBAAcV,OAAOP,KAAAA;AACrB,UAAMkB,UAAUX,QACZL,KAAKgB,SAASC,IAAyC,CAACC,OAAOb,MAAOc,GAAGC,cAAcF,EAAAA,CAAAA,EAAKG,OAAOC,WAAAA,IACnG,CAAA;AAEJ,QAAI,CAAC,CAACtB,KAAKM,YAAY,CAACD,OAAO;AAC7BO,UAAIW,KAAK,iBAAiB;QAAEvB;MAAK,GAAA;;;;;;IACnC,OAAO;AACLY,UAAIY,KAAK,WAAW;QAAEnB,OAAOA,OAAOoB,IAAIC,SAAAA;QAAYV,SAASA,SAASW;MAAO,GAAA;;;;;;IAC/E;AAEA,WAAO9B,QAAQ;MAAEE,OAAO;QAAEC,MAAM;UAAE,GAAGA;UAAMK;UAAOW;QAAQ;MAAE;MAAGf;MAASC;MAAU,GAAGC;IAAK,CAAA;EAC5F;AACF;AAGA,IAAMY,gBAAgB,CAACV,OAAcP,QAAyB,CAAA,MAAE;AAC9D,QAAM8B,WAAWvB,MAAMc,GAAGU,MAAMC;AAChC,aAAWC,QAAQjC,OAAO;AACxB,QAAI,CAAC8B,SAASI,UAAUD,IAAAA,GAAO;AAC7BH,eAASK,UAAU;QAACF;OAAK;IAC3B;EACF;AACF;",
6
6
  "names": ["PublicKey", "log", "nonNullable", "subscriptionHandler", "handler", "types", "event", "data", "context", "response", "rest", "client", "space", "spaceKey", "spaces", "get", "PublicKey", "from", "undefined", "log", "error", "status", "registerTypes", "objects", "map", "id", "db", "getObjectById", "filter", "nonNullable", "warn", "info", "key", "truncate", "length", "registry", "graph", "schemaRegistry", "type", "hasSchema", "addSchema"]
7
7
  }