@dxos/functions 0.5.3-main.3456876 → 0.5.3-main.37bbd91

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 (90) hide show
  1. package/dist/lib/browser/chunk-366QG6IX.mjs +81 -0
  2. package/dist/lib/browser/chunk-366QG6IX.mjs.map +7 -0
  3. package/dist/lib/browser/index.mjs +712 -490
  4. package/dist/lib/browser/index.mjs.map +4 -4
  5. package/dist/lib/browser/meta.json +1 -1
  6. package/dist/lib/browser/types.mjs +12 -0
  7. package/dist/lib/browser/types.mjs.map +7 -0
  8. package/dist/lib/node/chunk-3VSJ57ZZ.cjs +97 -0
  9. package/dist/lib/node/chunk-3VSJ57ZZ.cjs.map +7 -0
  10. package/dist/lib/node/index.cjs +695 -483
  11. package/dist/lib/node/index.cjs.map +4 -4
  12. package/dist/lib/node/meta.json +1 -1
  13. package/dist/lib/node/types.cjs +33 -0
  14. package/dist/lib/node/types.cjs.map +7 -0
  15. package/dist/types/src/function/function-registry.d.ts +24 -0
  16. package/dist/types/src/function/function-registry.d.ts.map +1 -0
  17. package/dist/types/src/function/function-registry.test.d.ts +2 -0
  18. package/dist/types/src/function/function-registry.test.d.ts.map +1 -0
  19. package/dist/types/src/function/index.d.ts +2 -0
  20. package/dist/types/src/function/index.d.ts.map +1 -0
  21. package/dist/types/src/handler.d.ts +32 -12
  22. package/dist/types/src/handler.d.ts.map +1 -1
  23. package/dist/types/src/index.d.ts +2 -0
  24. package/dist/types/src/index.d.ts.map +1 -1
  25. package/dist/types/src/runtime/dev-server.d.ts +7 -10
  26. package/dist/types/src/runtime/dev-server.d.ts.map +1 -1
  27. package/dist/types/src/runtime/scheduler.d.ts +11 -59
  28. package/dist/types/src/runtime/scheduler.d.ts.map +1 -1
  29. package/dist/types/src/testing/functions-integration.test.d.ts +2 -0
  30. package/dist/types/src/testing/functions-integration.test.d.ts.map +1 -0
  31. package/dist/types/src/testing/index.d.ts +4 -0
  32. package/dist/types/src/testing/index.d.ts.map +1 -0
  33. package/dist/types/src/testing/setup.d.ts +5 -0
  34. package/dist/types/src/testing/setup.d.ts.map +1 -0
  35. package/dist/types/src/testing/test/handler.d.ts +1 -0
  36. package/dist/types/src/testing/test/handler.d.ts.map +1 -1
  37. package/dist/types/src/testing/types.d.ts +9 -0
  38. package/dist/types/src/testing/types.d.ts.map +1 -0
  39. package/dist/types/src/testing/util.d.ts +3 -0
  40. package/dist/types/src/testing/util.d.ts.map +1 -0
  41. package/dist/types/src/trigger/index.d.ts +2 -0
  42. package/dist/types/src/trigger/index.d.ts.map +1 -0
  43. package/dist/types/src/trigger/trigger-registry.d.ts +40 -0
  44. package/dist/types/src/trigger/trigger-registry.d.ts.map +1 -0
  45. package/dist/types/src/trigger/trigger-registry.test.d.ts +2 -0
  46. package/dist/types/src/trigger/trigger-registry.test.d.ts.map +1 -0
  47. package/dist/types/src/trigger/type/index.d.ts +5 -0
  48. package/dist/types/src/trigger/type/index.d.ts.map +1 -0
  49. package/dist/types/src/trigger/type/subscription-trigger.d.ts +4 -0
  50. package/dist/types/src/trigger/type/subscription-trigger.d.ts.map +1 -0
  51. package/dist/types/src/trigger/type/timer-trigger.d.ts +4 -0
  52. package/dist/types/src/trigger/type/timer-trigger.d.ts.map +1 -0
  53. package/dist/types/src/trigger/type/webhook-trigger.d.ts +4 -0
  54. package/dist/types/src/trigger/type/webhook-trigger.d.ts.map +1 -0
  55. package/dist/types/src/trigger/type/websocket-trigger.d.ts +13 -0
  56. package/dist/types/src/trigger/type/websocket-trigger.d.ts.map +1 -0
  57. package/dist/types/src/types.d.ts +131 -111
  58. package/dist/types/src/types.d.ts.map +1 -1
  59. package/dist/types/src/util.d.ts +15 -0
  60. package/dist/types/src/util.d.ts.map +1 -0
  61. package/dist/types/src/util.test.d.ts +2 -0
  62. package/dist/types/src/util.test.d.ts.map +1 -0
  63. package/package.json +30 -14
  64. package/schema/functions.json +140 -112
  65. package/src/function/function-registry.test.ts +105 -0
  66. package/src/function/function-registry.ts +90 -0
  67. package/src/function/index.ts +5 -0
  68. package/src/handler.ts +50 -27
  69. package/src/index.ts +2 -0
  70. package/src/runtime/dev-server.test.ts +15 -35
  71. package/src/runtime/dev-server.ts +41 -24
  72. package/src/runtime/scheduler.test.ts +54 -75
  73. package/src/runtime/scheduler.ts +75 -300
  74. package/src/testing/functions-integration.test.ts +99 -0
  75. package/src/testing/index.ts +7 -0
  76. package/src/testing/setup.ts +45 -0
  77. package/src/testing/test/handler.ts +8 -2
  78. package/src/testing/types.ts +9 -0
  79. package/src/testing/util.ts +16 -0
  80. package/src/trigger/index.ts +5 -0
  81. package/src/trigger/trigger-registry.test.ts +255 -0
  82. package/src/trigger/trigger-registry.ts +189 -0
  83. package/src/trigger/type/index.ts +8 -0
  84. package/src/trigger/type/subscription-trigger.ts +80 -0
  85. package/src/trigger/type/timer-trigger.ts +44 -0
  86. package/src/trigger/type/webhook-trigger.ts +47 -0
  87. package/src/trigger/type/websocket-trigger.ts +91 -0
  88. package/src/types.ts +58 -40
  89. package/src/util.test.ts +43 -0
  90. package/src/util.ts +48 -0
@@ -29,85 +29,205 @@ var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: tru
29
29
  var node_exports = {};
30
30
  __export(node_exports, {
31
31
  DevServer: () => DevServer,
32
- FunctionManifestSchema: () => FunctionManifestSchema,
32
+ FunctionDef: () => import_chunk_3VSJ57ZZ.FunctionDef,
33
+ FunctionManifestSchema: () => import_chunk_3VSJ57ZZ.FunctionManifestSchema,
34
+ FunctionRegistry: () => FunctionRegistry,
35
+ FunctionTrigger: () => import_chunk_3VSJ57ZZ.FunctionTrigger,
33
36
  Scheduler: () => Scheduler,
37
+ TriggerRegistry: () => TriggerRegistry,
34
38
  subscriptionHandler: () => subscriptionHandler
35
39
  });
36
40
  module.exports = __toCommonJS(node_exports);
37
- var import_client = require("@dxos/client");
41
+ var import_chunk_3VSJ57ZZ = require("./chunk-3VSJ57ZZ.cjs");
42
+ var import_async = require("@dxos/async");
43
+ var import_echo = require("@dxos/client/echo");
44
+ var import_context = require("@dxos/context");
45
+ var import_keys = require("@dxos/keys");
38
46
  var import_log = require("@dxos/log");
39
47
  var import_util = require("@dxos/util");
48
+ var import_client = require("@dxos/client");
49
+ var import_log2 = require("@dxos/log");
50
+ var import_util2 = require("@dxos/util");
40
51
  var import_express = __toESM(require("express"));
41
52
  var import_get_port_please = require("get-port-please");
42
53
  var import_node_path = require("node:path");
43
- var import_async = require("@dxos/async");
54
+ var import_async2 = require("@dxos/async");
55
+ var import_context2 = require("@dxos/context");
44
56
  var import_invariant = require("@dxos/invariant");
45
- var import_log2 = require("@dxos/log");
57
+ var import_log3 = require("@dxos/log");
58
+ var import_node_path2 = __toESM(require("node:path"));
59
+ var import_async3 = require("@dxos/async");
60
+ var import_context3 = require("@dxos/context");
61
+ var import_log4 = require("@dxos/log");
62
+ var import_async4 = require("@dxos/async");
63
+ var import_echo2 = require("@dxos/client/echo");
64
+ var import_context4 = require("@dxos/context");
65
+ var import_echo_schema = require("@dxos/echo-schema");
66
+ var import_invariant2 = require("@dxos/invariant");
67
+ var import_keys2 = require("@dxos/keys");
68
+ var import_log5 = require("@dxos/log");
69
+ var import_util3 = require("@dxos/util");
70
+ var import_types = require("@braneframe/types");
71
+ var import_async5 = require("@dxos/async");
72
+ var import_echo_db = require("@dxos/echo-db");
73
+ var import_log6 = require("@dxos/log");
46
74
  var import_cron = require("cron");
75
+ var import_async6 = require("@dxos/async");
76
+ var import_log7 = require("@dxos/log");
47
77
  var import_get_port_please2 = require("get-port-please");
48
78
  var import_node_http = __toESM(require("node:http"));
49
- var import_node_path2 = __toESM(require("node:path"));
79
+ var import_log8 = require("@dxos/log");
50
80
  var import_ws = __toESM(require("ws"));
51
- var import_types = require("@braneframe/types");
52
- var import_async2 = require("@dxos/async");
53
- var import_echo = require("@dxos/client/echo");
54
- var import_context = require("@dxos/context");
55
- var import_invariant2 = require("@dxos/invariant");
56
- var import_log3 = require("@dxos/log");
57
- var import_util2 = require("@dxos/util");
58
- var S = __toESM(require("@effect/schema/Schema"));
59
- var __require = /* @__PURE__ */ ((x) => typeof require !== "undefined" ? require : typeof Proxy !== "undefined" ? new Proxy(x, {
60
- get: (a, b) => (typeof require !== "undefined" ? require : a)[b]
61
- }) : x)(function(x) {
62
- if (typeof require !== "undefined")
63
- return require.apply(this, arguments);
64
- throw Error('Dynamic require of "' + x + '" is not supported');
65
- });
66
- var __dxlog_file = "/home/runner/work/dxos/dxos/packages/core/functions/src/handler.ts";
81
+ var import_async7 = require("@dxos/async");
82
+ var import_log9 = require("@dxos/log");
83
+ var diff = (previous, next, comparator) => {
84
+ const remaining = [
85
+ ...previous
86
+ ];
87
+ const result = {
88
+ added: [],
89
+ updated: [],
90
+ removed: remaining
91
+ };
92
+ for (const object of next) {
93
+ const index = remaining.findIndex((item) => comparator(item, object));
94
+ if (index === -1) {
95
+ result.added.push(object);
96
+ } else {
97
+ result.updated.push(remaining[index]);
98
+ remaining.splice(index, 1);
99
+ }
100
+ }
101
+ return result;
102
+ };
103
+ var intersection = (a, b, comparator) => a.filter((a2) => b.find((b2) => comparator(a2, b2)) !== void 0);
104
+ var __dxlog_file = "/home/runner/work/dxos/dxos/packages/core/functions/src/function/function-registry.ts";
105
+ var FunctionRegistry = class extends import_context.Resource {
106
+ constructor(_client) {
107
+ super();
108
+ this._client = _client;
109
+ this._functionBySpaceKey = new import_util.ComplexMap(import_keys.PublicKey.hash);
110
+ this.registered = new import_async.Event();
111
+ }
112
+ getFunctions(space) {
113
+ return this._functionBySpaceKey.get(space.key) ?? [];
114
+ }
115
+ /**
116
+ * Loads function definitions from the manifest into the space.
117
+ * We first load all the definitions from the space to deduplicate by functionId.
118
+ */
119
+ async register(space, functions) {
120
+ (0, import_log.log)("register", {
121
+ space: space.key,
122
+ functions: functions?.length ?? 0
123
+ }, {
124
+ F: __dxlog_file,
125
+ L: 39,
126
+ S: this,
127
+ C: (f, a) => f(...a)
128
+ });
129
+ if (!functions?.length) {
130
+ return;
131
+ }
132
+ if (!space.db.graph.runtimeSchemaRegistry.hasSchema(import_chunk_3VSJ57ZZ.FunctionDef)) {
133
+ space.db.graph.runtimeSchemaRegistry.registerSchema(import_chunk_3VSJ57ZZ.FunctionDef);
134
+ }
135
+ const { objects: existing } = await space.db.query(import_echo.Filter.schema(import_chunk_3VSJ57ZZ.FunctionDef)).run();
136
+ const { added, removed } = diff(existing, functions, (a, b) => a.uri === b.uri);
137
+ added.forEach((def) => space.db.add((0, import_echo.create)(import_chunk_3VSJ57ZZ.FunctionDef, def)));
138
+ removed.forEach((def) => space.db.remove(def));
139
+ }
140
+ async _open() {
141
+ const spacesSubscription = this._client.spaces.subscribe(async (spaces) => {
142
+ for (const space of spaces) {
143
+ if (this._functionBySpaceKey.has(space.key)) {
144
+ continue;
145
+ }
146
+ const registered = [];
147
+ this._functionBySpaceKey.set(space.key, registered);
148
+ await space.waitUntilReady();
149
+ if (this._ctx.disposed) {
150
+ break;
151
+ }
152
+ this._ctx.onDispose(space.db.query(import_echo.Filter.schema(import_chunk_3VSJ57ZZ.FunctionDef)).subscribe(({ objects }) => {
153
+ const { added } = diff(registered, objects, (a, b) => a.uri === b.uri);
154
+ if (added.length > 0) {
155
+ registered.push(...added);
156
+ this.registered.emit({
157
+ space,
158
+ added
159
+ });
160
+ }
161
+ }));
162
+ }
163
+ });
164
+ this._ctx.onDispose(() => spacesSubscription.unsubscribe());
165
+ }
166
+ async _close(_) {
167
+ this._functionBySpaceKey.clear();
168
+ }
169
+ };
170
+ var __dxlog_file2 = "/home/runner/work/dxos/dxos/packages/core/functions/src/handler.ts";
67
171
  var subscriptionHandler = (handler) => {
68
- return ({ event, context, ...rest }) => {
172
+ return ({ event: { data }, context, ...rest }) => {
69
173
  const { client } = context;
70
- const space = event.spaceKey ? client.spaces.get(import_client.PublicKey.from(event.spaceKey)) : void 0;
71
- const objects = space && event.objects?.map((id) => space.db.getObjectById(id)).filter(import_util.nonNullable);
72
- if (!!event.spaceKey && !space) {
73
- import_log.log.warn("invalid space", {
74
- event
174
+ const space = data.spaceKey ? client.spaces.get(import_client.PublicKey.from(data.spaceKey)) : void 0;
175
+ const objects = space ? data.objects?.map((id) => space.db.getObjectById(id)).filter(import_util2.nonNullable) : [];
176
+ if (!!data.spaceKey && !space) {
177
+ import_log2.log.warn("invalid space", {
178
+ data
75
179
  }, {
76
- F: __dxlog_file,
77
- L: 68,
180
+ F: __dxlog_file2,
181
+ L: 91,
78
182
  S: void 0,
79
183
  C: (f, a) => f(...a)
80
184
  });
81
185
  } else {
82
- import_log.log.info("handler", {
186
+ import_log2.log.info("handler", {
83
187
  space: space?.key.truncate(),
84
188
  objects: objects?.length
85
189
  }, {
86
- F: __dxlog_file,
87
- L: 70,
190
+ F: __dxlog_file2,
191
+ L: 93,
88
192
  S: void 0,
89
193
  C: (f, a) => f(...a)
90
194
  });
91
195
  }
92
196
  return handler({
93
197
  event: {
94
- space,
95
- objects
198
+ data: {
199
+ ...data,
200
+ space,
201
+ objects
202
+ }
96
203
  },
97
204
  context,
98
205
  ...rest
99
206
  });
100
207
  };
101
208
  };
102
- var __dxlog_file2 = "/home/runner/work/dxos/dxos/packages/core/functions/src/runtime/dev-server.ts";
209
+ var __dxlog_file3 = "/home/runner/work/dxos/dxos/packages/core/functions/src/runtime/dev-server.ts";
103
210
  var DevServer = class {
104
- // prettier-ignore
105
- constructor(_client, _options) {
211
+ constructor(_client, _functionsRegistry, _options) {
106
212
  this._client = _client;
213
+ this._functionsRegistry = _functionsRegistry;
107
214
  this._options = _options;
215
+ this._ctx = createContext();
108
216
  this._handlers = {};
109
217
  this._seq = 0;
110
- this.update = new import_async.Event();
218
+ this.update = new import_async2.Event();
219
+ this._functionsRegistry.registered.on(async ({ added }) => {
220
+ added.forEach((def) => this._load(def));
221
+ await this._safeUpdateRegistration();
222
+ (0, import_log3.log)("new functions loaded", {
223
+ added
224
+ }, {
225
+ F: __dxlog_file3,
226
+ L: 52,
227
+ S: this,
228
+ C: (f, a) => f(...a)
229
+ });
230
+ });
111
231
  }
112
232
  get stats() {
113
233
  return {
@@ -116,8 +236,8 @@ var DevServer = class {
116
236
  }
117
237
  get endpoint() {
118
238
  (0, import_invariant.invariant)(this._port, void 0, {
119
- F: __dxlog_file2,
120
- L: 54,
239
+ F: __dxlog_file3,
240
+ L: 63,
121
241
  S: this,
122
242
  A: [
123
243
  "this._port",
@@ -132,45 +252,32 @@ var DevServer = class {
132
252
  get functions() {
133
253
  return Object.values(this._handlers);
134
254
  }
135
- async initialize() {
136
- for (const def of this._options.manifest.functions) {
137
- try {
138
- await this._load(def);
139
- } catch (err) {
140
- import_log2.log.error("parsing function (check manifest)", err, {
141
- F: __dxlog_file2,
142
- L: 71,
143
- S: this,
144
- C: (f, a) => f(...a)
145
- });
146
- }
147
- }
148
- }
149
255
  async start() {
150
256
  (0, import_invariant.invariant)(!this._server, void 0, {
151
- F: __dxlog_file2,
152
- L: 77,
257
+ F: __dxlog_file3,
258
+ L: 76,
153
259
  S: this,
154
260
  A: [
155
261
  "!this._server",
156
262
  ""
157
263
  ]
158
264
  });
159
- import_log2.log.info("starting...", void 0, {
160
- F: __dxlog_file2,
161
- L: 78,
265
+ import_log3.log.info("starting...", void 0, {
266
+ F: __dxlog_file3,
267
+ L: 77,
162
268
  S: this,
163
269
  C: (f, a) => f(...a)
164
270
  });
271
+ this._ctx = createContext();
165
272
  const app = (0, import_express.default)();
166
273
  app.use(import_express.default.json());
167
274
  app.post("/:path", async (req, res) => {
168
275
  const { path: path2 } = req.params;
169
276
  try {
170
- import_log2.log.info("calling", {
277
+ import_log3.log.info("calling", {
171
278
  path: path2
172
279
  }, {
173
- F: __dxlog_file2,
280
+ F: __dxlog_file3,
174
281
  L: 87,
175
282
  S: this,
176
283
  C: (f, a) => f(...a)
@@ -182,8 +289,8 @@ var DevServer = class {
182
289
  res.statusCode = await this.invoke("/" + path2, req.body);
183
290
  res.end();
184
291
  } catch (err) {
185
- import_log2.log.catch(err, void 0, {
186
- F: __dxlog_file2,
292
+ import_log3.log.catch(err, void 0, {
293
+ F: __dxlog_file3,
187
294
  L: 97,
188
295
  S: this,
189
296
  C: (f, a) => f(...a)
@@ -203,64 +310,61 @@ var DevServer = class {
203
310
  this._server = app.listen(this._port);
204
311
  try {
205
312
  const { registrationId, endpoint } = await this._client.services.services.FunctionRegistryService.register({
206
- endpoint: this.endpoint,
207
- functions: this.functions.map(({ def: { id, path: path2 } }) => ({
208
- id,
209
- path: path2
210
- }))
313
+ endpoint: this.endpoint
211
314
  });
212
- import_log2.log.info("registered", {
315
+ import_log3.log.info("registered", {
213
316
  endpoint
214
317
  }, {
215
- F: __dxlog_file2,
216
- L: 113,
318
+ F: __dxlog_file3,
319
+ L: 112,
217
320
  S: this,
218
321
  C: (f, a) => f(...a)
219
322
  });
220
323
  this._proxy = endpoint;
221
324
  this._functionServiceRegistration = registrationId;
325
+ await this._functionsRegistry.open(this._ctx);
222
326
  } catch (err) {
223
327
  await this.stop();
224
328
  throw new Error("FunctionRegistryService not available (check plugin is configured).");
225
329
  }
226
- import_log2.log.info("started", {
330
+ import_log3.log.info("started", {
227
331
  port: this._port
228
332
  }, {
229
- F: __dxlog_file2,
230
- L: 121,
333
+ F: __dxlog_file3,
334
+ L: 123,
231
335
  S: this,
232
336
  C: (f, a) => f(...a)
233
337
  });
234
338
  }
235
339
  async stop() {
236
340
  (0, import_invariant.invariant)(this._server, void 0, {
237
- F: __dxlog_file2,
238
- L: 125,
341
+ F: __dxlog_file3,
342
+ L: 127,
239
343
  S: this,
240
344
  A: [
241
345
  "this._server",
242
346
  ""
243
347
  ]
244
348
  });
245
- import_log2.log.info("stopping...", void 0, {
246
- F: __dxlog_file2,
247
- L: 126,
349
+ import_log3.log.info("stopping...", void 0, {
350
+ F: __dxlog_file3,
351
+ L: 128,
248
352
  S: this,
249
353
  C: (f, a) => f(...a)
250
354
  });
251
- const trigger = new import_async.Trigger();
355
+ const trigger = new import_async2.Trigger();
252
356
  this._server.close(async () => {
253
- import_log2.log.info("server stopped", void 0, {
254
- F: __dxlog_file2,
255
- L: 130,
357
+ import_log3.log.info("server stopped", void 0, {
358
+ F: __dxlog_file3,
359
+ L: 132,
256
360
  S: this,
257
361
  C: (f, a) => f(...a)
258
362
  });
259
363
  try {
260
364
  if (this._functionServiceRegistration) {
261
365
  (0, import_invariant.invariant)(this._client.services.services.FunctionRegistryService, void 0, {
262
- F: __dxlog_file2,
263
- L: 133,
366
+ F: __dxlog_file3,
367
+ L: 135,
264
368
  S: this,
265
369
  A: [
266
370
  "this._client.services.services.FunctionRegistryService",
@@ -270,11 +374,11 @@ var DevServer = class {
270
374
  await this._client.services.services.FunctionRegistryService.unregister({
271
375
  registrationId: this._functionServiceRegistration
272
376
  });
273
- import_log2.log.info("unregistered", {
377
+ import_log3.log.info("unregistered", {
274
378
  registrationId: this._functionServiceRegistration
275
379
  }, {
276
- F: __dxlog_file2,
277
- L: 138,
380
+ F: __dxlog_file3,
381
+ L: 140,
278
382
  S: this,
279
383
  C: (f, a) => f(...a)
280
384
  });
@@ -289,9 +393,9 @@ var DevServer = class {
289
393
  await trigger.wait();
290
394
  this._port = void 0;
291
395
  this._server = void 0;
292
- import_log2.log.info("stopped", void 0, {
293
- F: __dxlog_file2,
294
- L: 152,
396
+ import_log3.log.info("stopped", void 0, {
397
+ F: __dxlog_file3,
398
+ L: 154,
295
399
  S: this,
296
400
  C: (f, a) => f(...a)
297
401
  });
@@ -299,56 +403,85 @@ var DevServer = class {
299
403
  /**
300
404
  * Load function.
301
405
  */
302
- async _load(def, force = false) {
303
- const { id, path: path2, handler } = def;
406
+ async _load(def, force) {
407
+ const { uri, route, handler } = def;
304
408
  const filePath = (0, import_node_path.join)(this._options.baseDir, handler);
305
- import_log2.log.info("loading", {
306
- id,
409
+ import_log3.log.info("loading", {
410
+ uri,
307
411
  force
308
412
  }, {
309
- F: __dxlog_file2,
310
- L: 161,
413
+ F: __dxlog_file3,
414
+ L: 163,
311
415
  S: this,
312
416
  C: (f, a) => f(...a)
313
417
  });
314
418
  if (force) {
315
- Object.keys(__require.cache).filter((key) => key.startsWith(filePath)).forEach((key) => {
316
- delete __require.cache[key];
419
+ Object.keys(import_chunk_3VSJ57ZZ.__require.cache).filter((key) => key.startsWith(filePath)).forEach((key) => {
420
+ delete import_chunk_3VSJ57ZZ.__require.cache[key];
317
421
  });
318
422
  }
319
- const module2 = __require(filePath);
423
+ const module2 = (0, import_chunk_3VSJ57ZZ.__require)(filePath);
320
424
  if (typeof module2.default !== "function") {
321
- throw new Error(`Handler must export default function: ${id}`);
425
+ throw new Error(`Handler must export default function: ${uri}`);
322
426
  }
323
- this._handlers[path2] = {
427
+ this._handlers[route] = {
324
428
  def,
325
429
  handler: module2.default
326
430
  };
327
431
  }
432
+ async _safeUpdateRegistration() {
433
+ (0, import_invariant.invariant)(this._functionServiceRegistration, void 0, {
434
+ F: __dxlog_file3,
435
+ L: 185,
436
+ S: this,
437
+ A: [
438
+ "this._functionServiceRegistration",
439
+ ""
440
+ ]
441
+ });
442
+ try {
443
+ await this._client.services.services.FunctionRegistryService.updateRegistration({
444
+ registrationId: this._functionServiceRegistration,
445
+ functions: this.functions.map(({ def: { id, route } }) => ({
446
+ id,
447
+ route
448
+ }))
449
+ });
450
+ } catch (e) {
451
+ import_log3.log.catch(e, void 0, {
452
+ F: __dxlog_file3,
453
+ L: 192,
454
+ S: this,
455
+ C: (f, a) => f(...a)
456
+ });
457
+ }
458
+ }
328
459
  /**
329
460
  * Invoke function.
330
461
  */
331
462
  async invoke(path2, data) {
332
463
  const seq = ++this._seq;
333
464
  const now = Date.now();
334
- import_log2.log.info("req", {
465
+ import_log3.log.info("req", {
335
466
  seq,
336
467
  path: path2
337
468
  }, {
338
- F: __dxlog_file2,
339
- L: 188,
469
+ F: __dxlog_file3,
470
+ L: 203,
340
471
  S: this,
341
472
  C: (f, a) => f(...a)
342
473
  });
343
- const statusCode = await this._invoke(path2, data);
344
- import_log2.log.info("res", {
474
+ const statusCode = await this._invoke(path2, {
475
+ data
476
+ });
477
+ import_log3.log.info("res", {
345
478
  seq,
346
479
  path: path2,
347
480
  statusCode,
348
481
  duration: Date.now() - now
349
482
  }, {
350
- F: __dxlog_file2,
351
- L: 191,
483
+ F: __dxlog_file3,
484
+ L: 206,
352
485
  S: this,
353
486
  C: (f, a) => f(...a)
354
487
  });
@@ -358,8 +491,8 @@ var DevServer = class {
358
491
  async _invoke(path2, event) {
359
492
  const { handler } = this._handlers[path2] ?? {};
360
493
  (0, import_invariant.invariant)(handler, `invalid path: ${path2}`, {
361
- F: __dxlog_file2,
362
- L: 198,
494
+ F: __dxlog_file3,
495
+ L: 213,
363
496
  S: this,
364
497
  A: [
365
498
  "handler",
@@ -385,109 +518,98 @@ var DevServer = class {
385
518
  return statusCode;
386
519
  }
387
520
  };
388
- var __dxlog_file3 = "/home/runner/work/dxos/dxos/packages/core/functions/src/runtime/scheduler.ts";
521
+ var createContext = () => new import_context2.Context({
522
+ name: "DevServer"
523
+ });
524
+ var __dxlog_file4 = "/home/runner/work/dxos/dxos/packages/core/functions/src/runtime/scheduler.ts";
389
525
  var Scheduler = class {
390
- constructor(_client, _manifest, _options = {}) {
391
- this._client = _client;
392
- this._manifest = _manifest;
526
+ constructor(functions, triggers, _options = {}) {
527
+ this.functions = functions;
528
+ this.triggers = triggers;
393
529
  this._options = _options;
394
- this._mounts = new import_util2.ComplexMap(({ spaceKey, id }) => `${spaceKey.toHex()}:${id}`);
395
- }
396
- get mounts() {
397
- return Array.from(this._mounts.values()).reduce((acc, { trigger }) => {
398
- acc.push(trigger);
399
- return acc;
400
- }, []);
530
+ this._ctx = createContext2();
531
+ this._callMutex = new import_async3.Mutex();
532
+ this.functions.registered.on(async ({ space, added }) => {
533
+ await this._safeActivateTriggers(space, this.triggers.getInactiveTriggers(space), added);
534
+ });
535
+ this.triggers.registered.on(async ({ space, triggers: triggers2 }) => {
536
+ await this._safeActivateTriggers(space, triggers2, this.functions.getFunctions(space));
537
+ });
401
538
  }
402
539
  async start() {
403
- this._client.spaces.subscribe(async (spaces) => {
404
- for (const space of spaces) {
405
- await space.waitUntilReady();
406
- for (const trigger of this._manifest.triggers ?? []) {
407
- await this.mount(new import_context.Context(), space, trigger);
408
- }
409
- }
410
- });
540
+ await this._ctx.dispose();
541
+ this._ctx = createContext2();
542
+ await this.functions.open(this._ctx);
543
+ await this.triggers.open(this._ctx);
411
544
  }
412
545
  async stop() {
413
- for (const { id, spaceKey } of this._mounts.keys()) {
414
- await this.unmount(id, spaceKey);
415
- }
546
+ await this._ctx.dispose();
547
+ await this.functions.close();
548
+ await this.triggers.close();
416
549
  }
417
- /**
418
- * Mount trigger.
419
- */
420
- async mount(ctx, space, trigger) {
421
- const key = {
422
- spaceKey: space.key,
423
- id: trigger.function
424
- };
425
- const def = this._manifest.functions.find((config) => config.id === trigger.function);
426
- (0, import_invariant2.invariant)(def, `Function not found: ${trigger.function}`, {
427
- F: __dxlog_file3,
428
- L: 83,
429
- S: this,
430
- A: [
431
- "def",
432
- "`Function not found: ${trigger.function}`"
433
- ]
550
+ // TODO(burdon): Remove and update registries directly.
551
+ async register(space, manifest) {
552
+ await this.functions.register(space, manifest.functions);
553
+ await this.triggers.register(space, manifest);
554
+ }
555
+ async _safeActivateTriggers(space, triggers, functions) {
556
+ const mountTasks = triggers.map((trigger) => {
557
+ return this.activate(space, functions, trigger);
434
558
  });
435
- const exists = this._mounts.get(key);
436
- if (!exists) {
437
- this._mounts.set(key, {
438
- ctx,
439
- trigger
440
- });
441
- (0, import_log3.log)("mount", {
442
- space: space.key,
443
- trigger
559
+ await Promise.all(mountTasks).catch(import_log4.log.catch);
560
+ }
561
+ async activate(space, functions, fnTrigger) {
562
+ const definition = functions.find((def) => def.uri === fnTrigger.function);
563
+ if (!definition) {
564
+ import_log4.log.info("function is not found for trigger", {
565
+ fnTrigger
444
566
  }, {
445
- F: __dxlog_file3,
446
- L: 89,
567
+ F: __dxlog_file4,
568
+ L: 78,
447
569
  S: this,
448
570
  C: (f, a) => f(...a)
449
571
  });
450
- if (ctx.disposed) {
451
- return;
452
- }
453
- if (trigger.timer) {
454
- await this._createTimer(ctx, space, def, trigger.timer);
455
- }
456
- if (trigger.webhook) {
457
- await this._createWebhook(ctx, space, def, trigger.webhook);
458
- }
459
- if (trigger.websocket) {
460
- await this._createWebsocket(ctx, space, def, trigger.websocket);
461
- }
462
- if (trigger.subscription) {
463
- await this._createSubscription(ctx, space, def, trigger.subscription);
464
- }
465
- }
466
- }
467
- async unmount(id, spaceKey) {
468
- const key = {
469
- id,
470
- spaceKey
471
- };
472
- const { ctx } = this._mounts.get(key) ?? {};
473
- if (ctx) {
474
- this._mounts.delete(key);
475
- await ctx.dispose();
572
+ return;
476
573
  }
574
+ await this.triggers.activate({
575
+ space
576
+ }, fnTrigger, async (args) => {
577
+ return this._callMutex.executeSynchronized(() => {
578
+ return this._execFunction(definition, fnTrigger, {
579
+ meta: fnTrigger.meta,
580
+ data: {
581
+ ...args,
582
+ spaceKey: space.key
583
+ }
584
+ });
585
+ });
586
+ });
587
+ (0, import_log4.log)("activated trigger", {
588
+ space: space.key,
589
+ trigger: fnTrigger
590
+ }, {
591
+ F: __dxlog_file4,
592
+ L: 91,
593
+ S: this,
594
+ C: (f, a) => f(...a)
595
+ });
477
596
  }
478
- // TODO(burdon): Pass in Space key (common context).
479
- async _execFunction(def, data) {
597
+ async _execFunction(def, trigger, { data, meta }) {
598
+ let status = 0;
480
599
  try {
481
- let status = 0;
600
+ const payload = Object.assign({}, meta && {
601
+ meta
602
+ }, data);
482
603
  const { endpoint, callback } = this._options;
483
604
  if (endpoint) {
484
- const url = import_node_path2.default.join(endpoint, def.path);
485
- import_log3.log.info("exec", {
486
- function: def.id,
487
- url
605
+ const url = import_node_path2.default.join(endpoint, def.route);
606
+ import_log4.log.info("exec", {
607
+ function: def.uri,
608
+ url,
609
+ triggerType: trigger.spec.type
488
610
  }, {
489
- F: __dxlog_file3,
490
- L: 133,
611
+ F: __dxlog_file4,
612
+ L: 108,
491
613
  S: this,
492
614
  C: (f, a) => f(...a)
493
615
  });
@@ -496,358 +618,448 @@ var Scheduler = class {
496
618
  headers: {
497
619
  "Content-Type": "application/json"
498
620
  },
499
- body: JSON.stringify(data)
621
+ body: JSON.stringify(payload)
500
622
  });
501
623
  status = response.status;
502
624
  } else if (callback) {
503
- import_log3.log.info("exec", {
504
- function: def.id
625
+ import_log4.log.info("exec", {
626
+ function: def.uri
505
627
  }, {
506
- F: __dxlog_file3,
507
- L: 144,
628
+ F: __dxlog_file4,
629
+ L: 119,
508
630
  S: this,
509
631
  C: (f, a) => f(...a)
510
632
  });
511
- status = await callback(data) ?? 200;
633
+ status = await callback(payload) ?? 200;
512
634
  }
513
635
  if (status && status >= 400) {
514
636
  throw new Error(`Response: ${status}`);
515
637
  }
516
- import_log3.log.info("done", {
517
- function: def.id,
638
+ import_log4.log.info("done", {
639
+ function: def.uri,
518
640
  status
519
641
  }, {
520
- F: __dxlog_file3,
521
- L: 154,
642
+ F: __dxlog_file4,
643
+ L: 129,
522
644
  S: this,
523
645
  C: (f, a) => f(...a)
524
646
  });
525
- return status;
526
647
  } catch (err) {
527
- import_log3.log.error("error", {
528
- function: def.id,
648
+ import_log4.log.error("error", {
649
+ function: def.uri,
529
650
  error: err.message
530
651
  }, {
531
- F: __dxlog_file3,
532
- L: 157,
652
+ F: __dxlog_file4,
653
+ L: 131,
533
654
  S: this,
534
655
  C: (f, a) => f(...a)
535
656
  });
536
- return 500;
657
+ status = 500;
537
658
  }
659
+ return status;
538
660
  }
539
- //
540
- // Triggers
541
- //
542
- /**
543
- * Cron timer.
544
- */
545
- async _createTimer(ctx, space, def, trigger) {
546
- import_log3.log.info("timer", {
547
- space: space.key,
548
- trigger
661
+ };
662
+ var createContext2 = () => new import_context3.Context({
663
+ name: "FunctionScheduler"
664
+ });
665
+ var __dxlog_file5 = "/home/runner/work/dxos/dxos/packages/core/functions/src/trigger/type/subscription-trigger.ts";
666
+ var createSubscriptionTrigger = async (ctx, triggerCtx, spec, callback) => {
667
+ const objectIds = /* @__PURE__ */ new Set();
668
+ const task = new import_async5.UpdateScheduler(ctx, async () => {
669
+ if (objectIds.size > 0) {
670
+ const objects = Array.from(objectIds);
671
+ objectIds.clear();
672
+ await callback({
673
+ objects
674
+ });
675
+ }
676
+ }, {
677
+ maxFrequency: 4
678
+ });
679
+ const subscriptions = [];
680
+ const subscription = (0, import_echo_db.createSubscription)(({ added, updated }) => {
681
+ const sizeBefore = objectIds.size;
682
+ for (const object of added) {
683
+ objectIds.add(object.id);
684
+ }
685
+ for (const object of updated) {
686
+ objectIds.add(object.id);
687
+ }
688
+ if (objectIds.size > sizeBefore) {
689
+ import_log6.log.info("updated", {
690
+ added: added.length,
691
+ updated: updated.length
692
+ }, {
693
+ F: __dxlog_file5,
694
+ L: 45,
695
+ S: void 0,
696
+ C: (f, a) => f(...a)
697
+ });
698
+ task.trigger();
699
+ }
700
+ });
701
+ subscriptions.push(() => subscription.unsubscribe());
702
+ const { filter, options: { deep, delay } = {} } = spec;
703
+ const update = ({ objects }) => {
704
+ subscription.update(objects);
705
+ if (deep) {
706
+ import_log6.log.info("update", {
707
+ objects: objects.length
708
+ }, {
709
+ F: __dxlog_file5,
710
+ L: 59,
711
+ S: void 0,
712
+ C: (f, a) => f(...a)
713
+ });
714
+ for (const object of objects) {
715
+ const content = object.content;
716
+ if (content instanceof import_types.TextV0Type) {
717
+ subscriptions.push((0, import_echo_db.getAutomergeObjectCore)(content).updates.on((0, import_async5.debounce)(() => subscription.update([
718
+ object
719
+ ]), 1e3)));
720
+ }
721
+ }
722
+ }
723
+ };
724
+ const query = triggerCtx.space.db.query(import_echo_db.Filter.or(filter.map(({ type, props }) => import_echo_db.Filter.typename(type, props))));
725
+ subscriptions.push(query.subscribe(delay ? (0, import_async5.debounce)(update, delay) : update));
726
+ ctx.onDispose(() => {
727
+ subscriptions.forEach((unsubscribe) => unsubscribe());
728
+ });
729
+ };
730
+ var __dxlog_file6 = "/home/runner/work/dxos/dxos/packages/core/functions/src/trigger/type/timer-trigger.ts";
731
+ var createTimerTrigger = async (ctx, triggerContext, spec, callback) => {
732
+ const task = new import_async6.DeferredTask(ctx, async () => {
733
+ await callback({});
734
+ });
735
+ let last = 0;
736
+ let run = 0;
737
+ const job = import_cron.CronJob.from({
738
+ cronTime: spec.cron,
739
+ runOnInit: false,
740
+ onTick: () => {
741
+ const now = Date.now();
742
+ const delta = last ? now - last : 0;
743
+ last = now;
744
+ run++;
745
+ import_log7.log.info("tick", {
746
+ space: triggerContext.space.key.truncate(),
747
+ count: run,
748
+ delta
749
+ }, {
750
+ F: __dxlog_file6,
751
+ L: 37,
752
+ S: void 0,
753
+ C: (f, a) => f(...a)
754
+ });
755
+ task.schedule();
756
+ }
757
+ });
758
+ job.start();
759
+ ctx.onDispose(() => job.stop());
760
+ };
761
+ var __dxlog_file7 = "/home/runner/work/dxos/dxos/packages/core/functions/src/trigger/type/webhook-trigger.ts";
762
+ var createWebhookTrigger = async (ctx, _, spec, callback) => {
763
+ const server = import_node_http.default.createServer(async (req, res) => {
764
+ if (req.method !== spec.method) {
765
+ res.statusCode = 405;
766
+ return res.end();
767
+ }
768
+ res.statusCode = await callback({});
769
+ res.end();
770
+ });
771
+ const port = await (0, import_get_port_please2.getPort)({
772
+ random: true
773
+ });
774
+ server.listen(port, () => {
775
+ import_log8.log.info("started webhook", {
776
+ port
549
777
  }, {
550
- F: __dxlog_file3,
551
- L: 170,
552
- S: this,
778
+ F: __dxlog_file7,
779
+ L: 40,
780
+ S: void 0,
553
781
  C: (f, a) => f(...a)
554
782
  });
555
- const { cron } = trigger;
556
- const task = new import_async2.DeferredTask(ctx, async () => {
557
- await this._execFunction(def, {
558
- spaceKey: space.key
559
- });
560
- });
561
- let last = 0;
562
- let run = 0;
563
- const job = import_cron.CronJob.from({
564
- cronTime: cron,
565
- runOnInit: false,
566
- onTick: () => {
567
- const now = Date.now();
568
- const delta = last ? now - last : 0;
569
- last = now;
570
- run++;
571
- import_log3.log.info("tick", {
572
- space: space.key.truncate(),
573
- count: run,
574
- delta
783
+ spec.port = port;
784
+ });
785
+ ctx.onDispose(() => {
786
+ server.close();
787
+ });
788
+ };
789
+ var __dxlog_file8 = "/home/runner/work/dxos/dxos/packages/core/functions/src/trigger/type/websocket-trigger.ts";
790
+ var createWebsocketTrigger = async (ctx, triggerCtx, spec, callback, options = {
791
+ retryDelay: 2,
792
+ maxAttempts: 5
793
+ }) => {
794
+ const { url, init } = spec;
795
+ let ws;
796
+ for (let attempt = 1; attempt <= options.maxAttempts; attempt++) {
797
+ const open = new import_async7.Trigger();
798
+ ws = new import_ws.default(url);
799
+ Object.assign(ws, {
800
+ onopen: () => {
801
+ import_log9.log.info("opened", {
802
+ url
575
803
  }, {
576
- F: __dxlog_file3,
577
- L: 190,
578
- S: this,
804
+ F: __dxlog_file8,
805
+ L: 39,
806
+ S: void 0,
807
+ C: (f, a) => f(...a)
808
+ });
809
+ if (spec.init) {
810
+ ws.send(new TextEncoder().encode(JSON.stringify(init)));
811
+ }
812
+ open.wake(true);
813
+ },
814
+ onclose: (event) => {
815
+ import_log9.log.info("closed", {
816
+ url,
817
+ code: event.code
818
+ }, {
819
+ F: __dxlog_file8,
820
+ L: 48,
821
+ S: void 0,
822
+ C: (f, a) => f(...a)
823
+ });
824
+ if (event.code === 1006) {
825
+ setTimeout(async () => {
826
+ import_log9.log.info(`reconnecting in ${options.retryDelay}s...`, {
827
+ url
828
+ }, {
829
+ F: __dxlog_file8,
830
+ L: 53,
831
+ S: void 0,
832
+ C: (f, a) => f(...a)
833
+ });
834
+ await createWebsocketTrigger(ctx, triggerCtx, spec, callback, options);
835
+ }, options.retryDelay * 1e3);
836
+ }
837
+ open.wake(false);
838
+ },
839
+ onerror: (event) => {
840
+ import_log9.log.catch(event.error, {
841
+ url
842
+ }, {
843
+ F: __dxlog_file8,
844
+ L: 62,
845
+ S: void 0,
579
846
  C: (f, a) => f(...a)
580
847
  });
581
- task.schedule();
848
+ },
849
+ onmessage: async (event) => {
850
+ try {
851
+ import_log9.log.info("message", void 0, {
852
+ F: __dxlog_file8,
853
+ L: 67,
854
+ S: void 0,
855
+ C: (f, a) => f(...a)
856
+ });
857
+ const data = JSON.parse(new TextDecoder().decode(event.data));
858
+ await callback({
859
+ data
860
+ });
861
+ } catch (err) {
862
+ import_log9.log.catch(err, {
863
+ url
864
+ }, {
865
+ F: __dxlog_file8,
866
+ L: 71,
867
+ S: void 0,
868
+ C: (f, a) => f(...a)
869
+ });
870
+ }
582
871
  }
583
872
  });
584
- job.start();
585
- ctx.onDispose(() => job.stop());
873
+ const isOpen = await open.wait();
874
+ if (isOpen) {
875
+ break;
876
+ } else {
877
+ const wait = Math.pow(attempt, 2) * options.retryDelay;
878
+ if (attempt < options.maxAttempts) {
879
+ import_log9.log.warn(`failed to connect; trying again in ${wait}s`, {
880
+ attempt
881
+ }, {
882
+ F: __dxlog_file8,
883
+ L: 82,
884
+ S: void 0,
885
+ C: (f, a) => f(...a)
886
+ });
887
+ await (0, import_async7.sleep)(wait * 1e3);
888
+ }
889
+ }
586
890
  }
587
- /**
588
- * Webhook.
589
- */
590
- async _createWebhook(ctx, space, def, trigger) {
591
- import_log3.log.info("webhook", {
592
- space: space.key,
891
+ ctx.onDispose(() => {
892
+ ws?.close();
893
+ });
894
+ };
895
+ var __dxlog_file9 = "/home/runner/work/dxos/dxos/packages/core/functions/src/trigger/trigger-registry.ts";
896
+ var triggerHandlers = {
897
+ subscription: createSubscriptionTrigger,
898
+ timer: createTimerTrigger,
899
+ webhook: createWebhookTrigger,
900
+ websocket: createWebsocketTrigger
901
+ };
902
+ var TriggerRegistry = class extends import_context4.Resource {
903
+ constructor(_client, _options) {
904
+ super();
905
+ this._client = _client;
906
+ this._options = _options;
907
+ this._triggersBySpaceKey = new import_util3.ComplexMap(import_keys2.PublicKey.hash);
908
+ this.registered = new import_async4.Event();
909
+ this.removed = new import_async4.Event();
910
+ }
911
+ getActiveTriggers(space) {
912
+ return this._getTriggers(space, (t) => t.activationCtx != null);
913
+ }
914
+ getInactiveTriggers(space) {
915
+ return this._getTriggers(space, (t) => t.activationCtx == null);
916
+ }
917
+ async activate(triggerCtx, trigger, callback) {
918
+ (0, import_log5.log)("activate", {
919
+ space: triggerCtx.space.key,
593
920
  trigger
594
921
  }, {
595
- F: __dxlog_file3,
596
- L: 203,
922
+ F: __dxlog_file9,
923
+ L: 75,
597
924
  S: this,
598
925
  C: (f, a) => f(...a)
599
926
  });
600
- const server = import_node_http.default.createServer(async (req, res) => {
601
- if (req.method !== trigger.method) {
602
- res.statusCode = 405;
603
- return res.end();
604
- }
605
- res.statusCode = await this._execFunction(def, {
606
- spaceKey: space.key
607
- });
608
- res.end();
609
- });
610
- const port = await (0, import_get_port_please2.getPort)({
611
- random: true
927
+ const activationCtx = new import_context4.Context({
928
+ name: `trigger_${trigger.function}`
612
929
  });
613
- server.listen(port, () => {
614
- import_log3.log.info("started webhook", {
615
- port
616
- }, {
617
- F: __dxlog_file3,
618
- L: 226,
619
- S: this,
620
- C: (f, a) => f(...a)
621
- });
622
- trigger.port = port;
623
- });
624
- ctx.onDispose(() => {
625
- server.close();
930
+ this._ctx.onDispose(() => activationCtx.dispose());
931
+ const registeredTrigger = this._triggersBySpaceKey.get(triggerCtx.space.key)?.find((reg) => reg.trigger.id === trigger.id);
932
+ (0, import_invariant2.invariant)(registeredTrigger, `Trigger is not registered: ${trigger.function}`, {
933
+ F: __dxlog_file9,
934
+ L: 81,
935
+ S: this,
936
+ A: [
937
+ "registeredTrigger",
938
+ "`Trigger is not registered: ${trigger.function}`"
939
+ ]
626
940
  });
941
+ registeredTrigger.activationCtx = activationCtx;
942
+ try {
943
+ const options = this._options?.[trigger.spec.type];
944
+ await triggerHandlers[trigger.spec.type](activationCtx, triggerCtx, trigger.spec, callback, options);
945
+ } catch (err) {
946
+ delete registeredTrigger.activationCtx;
947
+ throw err;
948
+ }
627
949
  }
628
950
  /**
629
- * Websocket.
630
- * NOTE: The port must be unique, so the same hook cannot be used for multiple spaces.
951
+ * Loads triggers from the manifest into the space.
631
952
  */
632
- async _createWebsocket(ctx, space, def, trigger, options = {
633
- retryDelay: 2,
634
- maxAttempts: 5
635
- }) {
636
- import_log3.log.info("websocket", {
637
- space: space.key,
638
- trigger
953
+ async register(space, manifest) {
954
+ (0, import_log5.log)("register", {
955
+ space: space.key
639
956
  }, {
640
- F: __dxlog_file3,
641
- L: 252,
957
+ F: __dxlog_file9,
958
+ L: 97,
642
959
  S: this,
643
960
  C: (f, a) => f(...a)
644
961
  });
645
- const { url } = trigger;
646
- let ws;
647
- for (let attempt = 1; attempt <= options.maxAttempts; attempt++) {
648
- const open = new import_async2.Trigger();
649
- ws = new import_ws.default(url);
650
- Object.assign(ws, {
651
- onopen: () => {
652
- import_log3.log.info("opened", {
653
- url
654
- }, {
655
- F: __dxlog_file3,
656
- L: 262,
657
- S: this,
658
- C: (f, a) => f(...a)
659
- });
660
- if (trigger.init) {
661
- ws.send(new TextEncoder().encode(JSON.stringify(trigger.init)));
662
- }
663
- open.wake(true);
664
- },
665
- // TODO(burdon): Config retry if server closes?
666
- onclose: (event) => {
667
- import_log3.log.info("closed", {
668
- url,
669
- code: event.code
670
- }, {
671
- F: __dxlog_file3,
672
- L: 272,
673
- S: this,
674
- C: (f, a) => f(...a)
675
- });
676
- open.wake(false);
677
- },
678
- onerror: (event) => {
679
- import_log3.log.catch(event.error, {
680
- url
681
- }, {
682
- F: __dxlog_file3,
683
- L: 277,
684
- S: this,
685
- C: (f, a) => f(...a)
686
- });
687
- },
688
- onmessage: async (event) => {
689
- try {
690
- const data = JSON.parse(new TextDecoder().decode(event.data));
691
- await this._execFunction(def, {
692
- spaceKey: space.key,
693
- data
694
- });
695
- } catch (err) {
696
- import_log3.log.catch(err, {
697
- url
698
- }, {
699
- F: __dxlog_file3,
700
- L: 285,
701
- S: this,
702
- C: (f, a) => f(...a)
703
- });
704
- }
962
+ if (!manifest.triggers?.length) {
963
+ return;
964
+ }
965
+ if (!space.db.graph.runtimeSchemaRegistry.hasSchema(import_chunk_3VSJ57ZZ.FunctionTrigger)) {
966
+ space.db.graph.runtimeSchemaRegistry.registerSchema(import_chunk_3VSJ57ZZ.FunctionTrigger);
967
+ }
968
+ const { objects: existing } = await space.db.query(import_echo2.Filter.schema(import_chunk_3VSJ57ZZ.FunctionTrigger)).run();
969
+ const { added, removed } = diff(existing, manifest.triggers, (a, b) => {
970
+ const keys = b[import_echo_schema.ECHO_ATTR_META]?.keys ?? [
971
+ (0, import_echo_schema.foreignKey)("manifest", [
972
+ b.function,
973
+ b.spec.type
974
+ ].join("-"))
975
+ ];
976
+ return intersection((0, import_echo2.getMeta)(a)?.keys ?? [], keys, import_echo_schema.foreignKeyEquals).length > 0;
977
+ });
978
+ added.forEach((trigger) => {
979
+ const { meta, object } = (0, import_echo_schema.splitMeta)(trigger);
980
+ space.db.add((0, import_echo2.create)(import_chunk_3VSJ57ZZ.FunctionTrigger, object, meta));
981
+ });
982
+ removed.forEach((trigger) => space.db.remove(trigger));
983
+ }
984
+ async _open() {
985
+ const spaceListSubscription = this._client.spaces.subscribe(async (spaces) => {
986
+ for (const space of spaces) {
987
+ if (this._triggersBySpaceKey.has(space.key)) {
988
+ continue;
705
989
  }
706
- });
707
- const isOpen = await open.wait();
708
- if (isOpen) {
709
- break;
710
- } else {
711
- const wait = Math.pow(attempt, 2) * options.retryDelay;
712
- if (attempt < options.maxAttempts) {
713
- import_log3.log.warn(`failed to connect; trying again in ${wait}s`, {
714
- attempt
715
- }, {
716
- F: __dxlog_file3,
717
- L: 296,
718
- S: this,
719
- C: (f, a) => f(...a)
720
- });
721
- await (0, import_async2.sleep)(wait * 1e3);
990
+ const registered = [];
991
+ this._triggersBySpaceKey.set(space.key, registered);
992
+ await space.waitUntilReady();
993
+ if (this._ctx.disposed) {
994
+ break;
722
995
  }
996
+ const functionsSubscription = space.db.query(import_echo2.Filter.schema(import_chunk_3VSJ57ZZ.FunctionTrigger)).subscribe(async (triggers) => {
997
+ await this._handleRemovedTriggers(space, triggers.objects, registered);
998
+ this._handleNewTriggers(space, triggers.objects, registered);
999
+ });
1000
+ this._ctx.onDispose(functionsSubscription);
723
1001
  }
724
- }
725
- ctx.onDispose(() => {
726
- ws?.close();
727
1002
  });
1003
+ this._ctx.onDispose(() => spaceListSubscription.unsubscribe());
728
1004
  }
729
- /**
730
- * ECHO subscription.
731
- */
732
- async _createSubscription(ctx, space, def, trigger) {
733
- import_log3.log.info("subscription", {
734
- space: space.key,
735
- trigger
736
- }, {
737
- F: __dxlog_file3,
738
- L: 311,
739
- S: this,
740
- C: (f, a) => f(...a)
1005
+ async _close(_) {
1006
+ this._triggersBySpaceKey.clear();
1007
+ }
1008
+ _handleNewTriggers(space, allTriggers, registered) {
1009
+ const newTriggers = allTriggers.filter((candidate) => {
1010
+ return registered.find((reg) => reg.trigger.id === candidate.id) == null;
741
1011
  });
742
- const objectIds = /* @__PURE__ */ new Set();
743
- const task = new import_async2.DeferredTask(ctx, async () => {
744
- await this._execFunction(def, {
1012
+ if (newTriggers.length > 0) {
1013
+ const newRegisteredTriggers = newTriggers.map((trigger) => ({
1014
+ trigger
1015
+ }));
1016
+ registered.push(...newRegisteredTriggers);
1017
+ (0, import_log5.log)("registered new triggers", () => ({
745
1018
  spaceKey: space.key,
746
- objects: Array.from(objectIds)
747
- });
748
- });
749
- const subscriptions = [];
750
- const subscription = (0, import_echo.createSubscription)(({ added, updated }) => {
751
- import_log3.log.info("updated", {
752
- added: added.length,
753
- updated: updated.length
754
- }, {
755
- F: __dxlog_file3,
756
- L: 321,
1019
+ functions: newTriggers.map((t) => t.function)
1020
+ }), {
1021
+ F: __dxlog_file9,
1022
+ L: 159,
757
1023
  S: this,
758
1024
  C: (f, a) => f(...a)
759
1025
  });
760
- for (const object of added) {
761
- objectIds.add(object.id);
762
- }
763
- for (const object of updated) {
764
- objectIds.add(object.id);
765
- }
766
- task.schedule();
767
- });
768
- subscriptions.push(() => subscription.unsubscribe());
769
- const { filter, options: { deep, delay } = {} } = trigger;
770
- const update = ({ objects }) => {
771
- subscription.update(objects);
772
- if (deep) {
773
- import_log3.log.info("update", {
774
- objects: objects.length
775
- }, {
776
- F: __dxlog_file3,
777
- L: 342,
778
- S: this,
779
- C: (f, a) => f(...a)
780
- });
781
- for (const object of objects) {
782
- const content = object.content;
783
- if (content instanceof import_types.TextV0Type) {
784
- subscriptions.push((0, import_echo.getAutomergeObjectCore)(content).updates.on((0, import_async2.debounce)(() => subscription.update([
785
- object
786
- ]), 1e3)));
787
- }
788
- }
1026
+ this.registered.emit({
1027
+ space,
1028
+ triggers: newTriggers
1029
+ });
1030
+ }
1031
+ }
1032
+ async _handleRemovedTriggers(space, allTriggers, registered) {
1033
+ const removed = [];
1034
+ for (let i = registered.length - 1; i >= 0; i--) {
1035
+ const wasRemoved = allTriggers.find((trigger) => trigger.id === registered[i].trigger.id) == null;
1036
+ if (wasRemoved) {
1037
+ const unregistered = registered.splice(i, 1)[0];
1038
+ await unregistered.activationCtx?.dispose();
1039
+ removed.push(unregistered.trigger);
789
1040
  }
790
- };
791
- const query = space.db.query(import_echo.Filter.or(filter.map(({ type, props }) => import_echo.Filter.typename(type, props))));
792
- subscriptions.push(query.subscribe(delay ? (0, import_async2.debounce)(update, delay) : update));
793
- ctx.onDispose(() => {
794
- subscriptions.forEach((unsubscribe) => unsubscribe());
795
- });
1041
+ }
1042
+ if (removed.length > 0) {
1043
+ this.removed.emit({
1044
+ space,
1045
+ triggers: removed
1046
+ });
1047
+ }
1048
+ }
1049
+ _getTriggers(space, predicate) {
1050
+ const allSpaceTriggers = this._triggersBySpaceKey.get(space.key) ?? [];
1051
+ return allSpaceTriggers.filter(predicate).map((trigger) => trigger.trigger);
796
1052
  }
797
1053
  };
798
- var TimerTriggerSchema = S.struct({
799
- cron: S.string
800
- });
801
- var WebhookTriggerSchema = S.mutable(S.struct({
802
- method: S.string,
803
- // Assigned port.
804
- port: S.optional(S.number)
805
- }));
806
- var WebsocketTriggerSchema = S.struct({
807
- url: S.string,
808
- init: S.optional(S.record(S.string, S.any))
809
- });
810
- var SubscriptionTriggerSchema = S.struct({
811
- spaceKey: S.optional(S.string),
812
- // TODO(burdon): Define query DSL.
813
- filter: S.array(S.struct({
814
- type: S.string,
815
- props: S.optional(S.record(S.string, S.any))
816
- })),
817
- options: S.optional(S.struct({
818
- // Watch changes to object (not just creation).
819
- deep: S.optional(S.boolean),
820
- // Debounce changes (delay in ms).
821
- delay: S.optional(S.number)
822
- }))
823
- });
824
- var FunctionTriggerSchema = S.struct({
825
- function: S.string.pipe(S.description("Function ID/URI.")),
826
- // Context passed to function.
827
- context: S.optional(S.record(S.string, S.any)),
828
- // Triggers.
829
- timer: S.optional(TimerTriggerSchema),
830
- webhook: S.optional(WebhookTriggerSchema),
831
- websocket: S.optional(WebsocketTriggerSchema),
832
- subscription: S.optional(SubscriptionTriggerSchema)
833
- });
834
- var FunctionDefSchema = S.struct({
835
- id: S.string,
836
- // name: S.string,
837
- description: S.optional(S.string),
838
- path: S.string,
839
- // TODO(burdon): NPM/GitHub/Docker/CF URL?
840
- handler: S.string
841
- });
842
- var FunctionManifestSchema = S.struct({
843
- functions: S.mutable(S.array(FunctionDefSchema)),
844
- triggers: S.optional(S.mutable(S.array(FunctionTriggerSchema)))
845
- });
846
1054
  // Annotate the CommonJS export names for ESM import in node:
847
1055
  0 && (module.exports = {
848
1056
  DevServer,
1057
+ FunctionDef,
849
1058
  FunctionManifestSchema,
1059
+ FunctionRegistry,
1060
+ FunctionTrigger,
850
1061
  Scheduler,
1062
+ TriggerRegistry,
851
1063
  subscriptionHandler
852
1064
  });
853
1065
  //# sourceMappingURL=index.cjs.map