@dxos/functions 0.5.3-main.f752aaa → 0.5.3-next.57eca40

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 (95) 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 +825 -471
  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 +805 -461
  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 +33 -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 +16 -13
  26. package/dist/types/src/runtime/dev-server.d.ts.map +1 -1
  27. package/dist/types/src/runtime/dev-server.test.d.ts +2 -0
  28. package/dist/types/src/runtime/dev-server.test.d.ts.map +1 -0
  29. package/dist/types/src/runtime/scheduler.d.ts +13 -27
  30. package/dist/types/src/runtime/scheduler.d.ts.map +1 -1
  31. package/dist/types/src/testing/functions-integration.test.d.ts +2 -0
  32. package/dist/types/src/testing/functions-integration.test.d.ts.map +1 -0
  33. package/dist/types/src/testing/index.d.ts +4 -0
  34. package/dist/types/src/testing/index.d.ts.map +1 -0
  35. package/dist/types/src/testing/setup.d.ts +5 -0
  36. package/dist/types/src/testing/setup.d.ts.map +1 -0
  37. package/dist/types/src/testing/test/handler.d.ts +4 -0
  38. package/dist/types/src/testing/test/handler.d.ts.map +1 -0
  39. package/dist/types/src/testing/test/index.d.ts +3 -0
  40. package/dist/types/src/testing/test/index.d.ts.map +1 -0
  41. package/dist/types/src/testing/types.d.ts +9 -0
  42. package/dist/types/src/testing/types.d.ts.map +1 -0
  43. package/dist/types/src/testing/util.d.ts +3 -0
  44. package/dist/types/src/testing/util.d.ts.map +1 -0
  45. package/dist/types/src/trigger/index.d.ts +2 -0
  46. package/dist/types/src/trigger/index.d.ts.map +1 -0
  47. package/dist/types/src/trigger/trigger-registry.d.ts +40 -0
  48. package/dist/types/src/trigger/trigger-registry.d.ts.map +1 -0
  49. package/dist/types/src/trigger/trigger-registry.test.d.ts +2 -0
  50. package/dist/types/src/trigger/trigger-registry.test.d.ts.map +1 -0
  51. package/dist/types/src/trigger/type/index.d.ts +5 -0
  52. package/dist/types/src/trigger/type/index.d.ts.map +1 -0
  53. package/dist/types/src/trigger/type/subscription-trigger.d.ts +4 -0
  54. package/dist/types/src/trigger/type/subscription-trigger.d.ts.map +1 -0
  55. package/dist/types/src/trigger/type/timer-trigger.d.ts +4 -0
  56. package/dist/types/src/trigger/type/timer-trigger.d.ts.map +1 -0
  57. package/dist/types/src/trigger/type/webhook-trigger.d.ts +4 -0
  58. package/dist/types/src/trigger/type/webhook-trigger.d.ts.map +1 -0
  59. package/dist/types/src/trigger/type/websocket-trigger.d.ts +13 -0
  60. package/dist/types/src/trigger/type/websocket-trigger.d.ts.map +1 -0
  61. package/dist/types/src/types.d.ts +143 -101
  62. package/dist/types/src/types.d.ts.map +1 -1
  63. package/dist/types/src/util.d.ts +15 -0
  64. package/dist/types/src/util.d.ts.map +1 -0
  65. package/dist/types/src/util.test.d.ts +2 -0
  66. package/dist/types/src/util.test.d.ts.map +1 -0
  67. package/package.json +33 -15
  68. package/schema/functions.json +140 -104
  69. package/src/function/function-registry.test.ts +105 -0
  70. package/src/function/function-registry.ts +90 -0
  71. package/src/function/index.ts +5 -0
  72. package/src/handler.ts +54 -31
  73. package/src/index.ts +2 -0
  74. package/src/runtime/dev-server.test.ts +60 -0
  75. package/src/runtime/dev-server.ts +104 -53
  76. package/src/runtime/scheduler.test.ts +56 -73
  77. package/src/runtime/scheduler.ts +87 -271
  78. package/src/testing/functions-integration.test.ts +99 -0
  79. package/src/testing/index.ts +7 -0
  80. package/src/testing/setup.ts +45 -0
  81. package/src/testing/test/handler.ts +15 -0
  82. package/src/testing/test/index.ts +7 -0
  83. package/src/testing/types.ts +9 -0
  84. package/src/testing/util.ts +16 -0
  85. package/src/trigger/index.ts +5 -0
  86. package/src/trigger/trigger-registry.test.ts +255 -0
  87. package/src/trigger/trigger-registry.ts +189 -0
  88. package/src/trigger/type/index.ts +8 -0
  89. package/src/trigger/type/subscription-trigger.ts +80 -0
  90. package/src/trigger/type/timer-trigger.ts +44 -0
  91. package/src/trigger/type/webhook-trigger.ts +47 -0
  92. package/src/trigger/type/websocket-trigger.ts +91 -0
  93. package/src/types.ts +59 -32
  94. package/src/util.test.ts +43 -0
  95. package/src/util.ts +48 -0
@@ -0,0 +1,99 @@
1
+ //
2
+ // Copyright 2024 DXOS.org
3
+ //
4
+
5
+ import { expect } from 'chai';
6
+ import path from 'path';
7
+
8
+ import { Trigger, waitForCondition } from '@dxos/async';
9
+ import { type Client } from '@dxos/client';
10
+ import { create, type Space } from '@dxos/client/echo';
11
+ import { performInvitation, TestBuilder } from '@dxos/client/testing';
12
+ import { Invitation } from '@dxos/protocols/proto/dxos/client/services';
13
+ import { describe, test } from '@dxos/test';
14
+
15
+ import { setTestCallHandler } from './test/handler';
16
+ import { FunctionRegistry } from '../function';
17
+ import { DevServer, Scheduler } from '../runtime';
18
+ import { createFunctionRuntime, createInitializedClients, TestType } from '../testing';
19
+ import { TriggerRegistry } from '../trigger';
20
+ import { FunctionDef, FunctionTrigger } from '../types';
21
+
22
+ describe('functions e2e', () => {
23
+ let testBuilder: TestBuilder;
24
+ before(async () => {
25
+ testBuilder = new TestBuilder();
26
+ });
27
+ after(async () => {
28
+ await testBuilder.destroy();
29
+ });
30
+
31
+ test('a function gets triggered in response to another peer object creations', async () => {
32
+ // TODO(burdon): Create builder pattern.
33
+ const functionRuntime = await createFunctionRuntime(testBuilder);
34
+ const devServer = await startDevServer(functionRuntime);
35
+ const scheduler = await startScheduler(functionRuntime, devServer);
36
+
37
+ const app = (await createInitializedClients(testBuilder, 1))[0];
38
+ const space = await app.spaces.create();
39
+ await inviteMember(space, functionRuntime);
40
+
41
+ const uri = 'example.com/function/test';
42
+ space.db.add(create(FunctionDef, { uri, route: '/test', handler: 'test' }));
43
+ const triggerMeta: FunctionTrigger['meta'] = { name: 'DXOS' };
44
+ space.db.add(
45
+ create(FunctionTrigger, {
46
+ function: uri,
47
+ meta: triggerMeta,
48
+ spec: {
49
+ type: 'subscription',
50
+ filter: [{ type: TestType.typename }],
51
+ },
52
+ }),
53
+ );
54
+
55
+ const called = new Trigger<any>();
56
+ setTestCallHandler(async (args) => {
57
+ called.wake(args.event.data);
58
+ return args.response.status(200);
59
+ });
60
+
61
+ await waitTriggersReplicated(space, scheduler);
62
+ const addedObject = space.db.add(create(TestType, { title: '42' }));
63
+
64
+ const callArgs = await called.wait();
65
+ expect(callArgs.meta).to.deep.eq(triggerMeta);
66
+ expect(callArgs.objects).to.deep.eq([addedObject.id]);
67
+ expect(callArgs.spaceKey).to.eq(space.key.toHex());
68
+ });
69
+
70
+ const waitTriggersReplicated = async (space: Space, scheduler: Scheduler) => {
71
+ await waitForCondition({ condition: () => scheduler.triggers.getActiveTriggers(space).length > 0 });
72
+ };
73
+
74
+ // TODO(burdon): Factor out utils to builder pattern.
75
+
76
+ const startScheduler = async (client: Client, devServer: DevServer) => {
77
+ const functionRegistry = new FunctionRegistry(client);
78
+ const triggerRegistry = new TriggerRegistry(client);
79
+ const scheduler = new Scheduler(functionRegistry, triggerRegistry, { endpoint: devServer.endpoint });
80
+ await scheduler.start();
81
+ testBuilder.ctx.onDispose(() => scheduler.stop());
82
+ return scheduler;
83
+ };
84
+
85
+ const startDevServer = async (client: Client) => {
86
+ const functionRegistry = new FunctionRegistry(client);
87
+ const server = new DevServer(client, functionRegistry, {
88
+ baseDir: path.join(__dirname, '../testing'),
89
+ });
90
+ await server.start();
91
+ testBuilder.ctx.onDispose(() => server.stop());
92
+ return server;
93
+ };
94
+
95
+ const inviteMember = async (host: Space, guest: Client) => {
96
+ const [{ invitation: hostInvitation }] = await Promise.all(performInvitation({ host, guest: guest.spaces }));
97
+ expect(hostInvitation?.state).to.eq(Invitation.State.SUCCESS);
98
+ };
99
+ });
@@ -0,0 +1,7 @@
1
+ //
2
+ // Copyright 2024 DXOS.org
3
+ //
4
+
5
+ export * from './setup';
6
+ export * from './types';
7
+ export * from './util';
@@ -0,0 +1,45 @@
1
+ //
2
+ // Copyright 2024 DXOS.org
3
+ //
4
+
5
+ import { FunctionsPlugin } from '@dxos/agent';
6
+ import { Client, Config } from '@dxos/client';
7
+ import { type TestBuilder } from '@dxos/client/testing';
8
+ import { range } from '@dxos/util';
9
+
10
+ import { TestType } from './types';
11
+ import { FunctionDef, FunctionTrigger } from '../types';
12
+
13
+ // TODO(burdon): Create new or extend existing TestBuilder.
14
+
15
+ export const createInitializedClients = async (testBuilder: TestBuilder, count: number = 1, config?: Config) => {
16
+ const clients = range(count).map(() => new Client({ config, services: testBuilder.createLocalClientServices() }));
17
+ testBuilder.ctx.onDispose(() => Promise.all(clients.map((c) => c.destroy())));
18
+ return Promise.all(
19
+ clients.map(async (client, index) => {
20
+ await client.initialize();
21
+ await client.halo.createIdentity({ displayName: `Peer ${index}` });
22
+ client.addSchema(TestType, FunctionDef, FunctionTrigger);
23
+ return client;
24
+ }),
25
+ );
26
+ };
27
+
28
+ export const createFunctionRuntime = async (testBuilder: TestBuilder): Promise<Client> => {
29
+ const config = new Config({
30
+ runtime: {
31
+ agent: {
32
+ plugins: [{ id: 'dxos.org/agent/plugin/functions', config: { port: 8080 } }],
33
+ },
34
+ },
35
+ });
36
+
37
+ const client = (await createInitializedClients(testBuilder, 1, config))[0];
38
+
39
+ // TODO(burdon): Better way to configure plugin? (Rationalize chess.test).
40
+ const functionsPlugin = new FunctionsPlugin();
41
+ await functionsPlugin.initialize({ client, clientServices: client.services });
42
+ await functionsPlugin.open();
43
+ testBuilder.ctx.onDispose(() => functionsPlugin.close());
44
+ return client;
45
+ };
@@ -0,0 +1,15 @@
1
+ //
2
+ // Copyright 2023 DXOS.org
3
+ //
4
+
5
+ import { type FunctionHandler } from '../../handler';
6
+
7
+ let callHandler: FunctionHandler<any> = async ({ response }) => response.status(200);
8
+
9
+ export const setTestCallHandler = (handler: FunctionHandler<any>) => {
10
+ callHandler = handler;
11
+ };
12
+
13
+ export const handler: FunctionHandler<any> = async (args) => {
14
+ return callHandler(args);
15
+ };
@@ -0,0 +1,7 @@
1
+ //
2
+ // Copyright 2023 DXOS.org
3
+ //
4
+
5
+ import { handler } from './handler';
6
+
7
+ export default handler;
@@ -0,0 +1,9 @@
1
+ //
2
+ // Copyright 2024 DXOS.org
3
+ //
4
+
5
+ import { S, TypedObject } from '@dxos/echo-schema';
6
+
7
+ export class TestType extends TypedObject({ typename: 'example.com/type/Test', version: '0.1.0' })({
8
+ title: S.string,
9
+ }) {}
@@ -0,0 +1,16 @@
1
+ //
2
+ // Copyright 2024 DXOS.org
3
+ //
4
+
5
+ import { Filter, type Space } from '@dxos/client/echo';
6
+ import { invariant } from '@dxos/invariant';
7
+
8
+ import { FunctionTrigger } from '../types';
9
+
10
+ export const triggerWebhook = async (space: Space, uri: string) => {
11
+ const trigger = (
12
+ await space.db.query(Filter.schema(FunctionTrigger, (t: FunctionTrigger) => t.function === uri)).run()
13
+ ).objects[0];
14
+ invariant(trigger.spec.type === 'webhook');
15
+ void fetch(`http://localhost:${trigger.spec.port}`);
16
+ };
@@ -0,0 +1,5 @@
1
+ //
2
+ // Copyright 2024 DXOS.org
3
+ //
4
+
5
+ export * from './trigger-registry';
@@ -0,0 +1,255 @@
1
+ //
2
+ // Copyright 2023 DXOS.org
3
+ //
4
+
5
+ import chai, { expect } from 'chai';
6
+ import chaiAsPromised from 'chai-as-promised';
7
+
8
+ import { sleep, Trigger, waitForCondition } from '@dxos/async';
9
+ import { type Client } from '@dxos/client';
10
+ import { type Space } from '@dxos/client/echo';
11
+ import { TestBuilder } from '@dxos/client/testing';
12
+ import { Context } from '@dxos/context';
13
+ import { Filter } from '@dxos/echo-db';
14
+ import { create, splitMeta } from '@dxos/echo-schema';
15
+ import { describe, test } from '@dxos/test';
16
+ import { range } from '@dxos/util';
17
+
18
+ import { TriggerRegistry } from './trigger-registry';
19
+ import { createInitializedClients, TestType, triggerWebhook } from '../testing';
20
+ import { type FunctionManifest, FunctionTrigger } from '../types';
21
+
22
+ const manifest: FunctionManifest = {
23
+ triggers: [
24
+ {
25
+ '@meta': {
26
+ keys: [
27
+ {
28
+ source: 'example.com',
29
+ id: 'trigger-1',
30
+ },
31
+ ],
32
+ },
33
+ function: 'example.com/function/webhook-test',
34
+ spec: {
35
+ type: 'webhook',
36
+ method: 'GET',
37
+ },
38
+ },
39
+ {
40
+ '@meta': {
41
+ keys: [
42
+ {
43
+ source: 'example.com',
44
+ id: 'trigger-2',
45
+ },
46
+ ],
47
+ },
48
+ function: 'example.com/function/subscription-test',
49
+ spec: {
50
+ type: 'subscription',
51
+ filter: [
52
+ {
53
+ type: TestType.typename,
54
+ },
55
+ ],
56
+ },
57
+ },
58
+ ],
59
+ };
60
+
61
+ chai.use(chaiAsPromised);
62
+
63
+ describe('trigger registry', () => {
64
+ let ctx: Context;
65
+ let testBuilder: TestBuilder;
66
+ beforeEach(async () => {
67
+ ctx = new Context();
68
+ testBuilder = new TestBuilder();
69
+ });
70
+ afterEach(async () => {
71
+ await ctx.dispose();
72
+ await testBuilder.destroy();
73
+ });
74
+
75
+ describe('register', () => {
76
+ test('creates new triggers', async () => {
77
+ const client = (await createInitializedClients(testBuilder))[0];
78
+ const registry = createRegistry(client);
79
+ const space = await client.spaces.create();
80
+ await registry.register(space, manifest);
81
+ const { objects } = await space.db.query(Filter.schema(FunctionTrigger)).run();
82
+ expect(objects.length).to.eq(manifest.triggers?.length);
83
+
84
+ const expected = manifest.triggers?.map((trigger) => trigger.function).sort();
85
+ expect(objects.map((object: FunctionTrigger) => object.function).sort()).to.deep.eq(expected);
86
+ });
87
+ });
88
+
89
+ describe('activate', () => {
90
+ test('invokes the provided callback', async () => {
91
+ const client = (await createInitializedClients(testBuilder))[0];
92
+ const space = await client.spaces.create();
93
+ const registry = createRegistry(client);
94
+ await registry.register(space, manifest);
95
+ await registry.open(ctx);
96
+ await waitForInactiveTriggers(registry, space);
97
+
98
+ const callbackInvoked = new Trigger();
99
+ const { objects: allTriggers } = await space.db.query(Filter.schema(FunctionTrigger)).run();
100
+ const webhookTrigger = allTriggers.find((trigger: FunctionTrigger) => trigger.spec.type === 'webhook')!;
101
+ await registry.activate({ space }, webhookTrigger, async () => {
102
+ callbackInvoked.wake();
103
+ return 200;
104
+ });
105
+
106
+ setTimeout(() => triggerWebhook(space, webhookTrigger.function));
107
+ await callbackInvoked.wait();
108
+ });
109
+
110
+ test('removes from inactive list', async () => {
111
+ const client = (await createInitializedClients(testBuilder))[0];
112
+ const space = await client.spaces.create();
113
+ const registry = createRegistry(client);
114
+ await registry.register(space, manifest);
115
+ await registry.open(ctx);
116
+ await waitForInactiveTriggers(registry, space);
117
+
118
+ const inactiveTrigger = registry.getInactiveTriggers(space)[0];
119
+ await registry.activate({ space }, inactiveTrigger, async () => 200);
120
+
121
+ const updatedInactiveList = registry.getInactiveTriggers(space);
122
+ expect(updatedInactiveList.find((trigger: FunctionTrigger) => trigger.function === inactiveTrigger.function)).to
123
+ .be.undefined;
124
+ });
125
+ });
126
+
127
+ describe('deactivate', () => {
128
+ test('trigger object deletion deactivates a trigger', async () => {
129
+ const client = (await createInitializedClients(testBuilder))[0];
130
+ const space = await client.spaces.create();
131
+ const registry = createRegistry(client);
132
+ await registry.register(space, manifest);
133
+ await registry.open(ctx);
134
+ await waitForInactiveTriggers(registry, space);
135
+
136
+ const { objects: allTriggers } = await space.db.query(Filter.schema(FunctionTrigger)).run();
137
+ const echoTrigger = allTriggers.find((trigger: FunctionTrigger) => trigger.spec.type === 'subscription')!;
138
+ let count = 0;
139
+ await registry.activate({ space }, echoTrigger, async () => {
140
+ count++;
141
+ return 200;
142
+ });
143
+
144
+ space.db.add(create(TestType, { title: '1' }));
145
+ await sleep(20);
146
+ expect(count).to.eq(1);
147
+
148
+ space.db.remove(echoTrigger);
149
+ space.db.add(create(TestType, { title: '2' }));
150
+ await sleep(20);
151
+ expect(count).to.eq(1);
152
+ });
153
+
154
+ test('registry closing deactivates a trigger', async () => {
155
+ const client = (await createInitializedClients(testBuilder))[0];
156
+ const space = await client.spaces.create();
157
+ const registry = createRegistry(client);
158
+ await registry.register(space, manifest);
159
+ await registry.open(ctx);
160
+ await waitForInactiveTriggers(registry, space);
161
+
162
+ const { objects: allTriggers } = await space.db.query(Filter.schema(FunctionTrigger)).run();
163
+ const echoTrigger = allTriggers.find((trigger: FunctionTrigger) => trigger.spec.type === 'subscription')!;
164
+ let count = 0;
165
+ await registry.activate({ space }, echoTrigger, async () => {
166
+ count++;
167
+ return 200;
168
+ });
169
+
170
+ await registry.close();
171
+
172
+ space.db.add(create(TestType, { title: '1' }));
173
+ await sleep(20);
174
+ expect(count).to.eq(0);
175
+ });
176
+ });
177
+
178
+ describe('trigger events', () => {
179
+ test.only('event fired when all registered when opened', async () => {
180
+ const client = (await createInitializedClients(testBuilder))[0];
181
+ const registry = createRegistry(client);
182
+ const triggers = createTriggers(client.spaces.default, 3);
183
+
184
+ const triggersRegistered = new Trigger<FunctionTrigger[]>();
185
+ registry.registered.on((fn) => {
186
+ expect(fn.space.key.toHex()).to.eq(client.spaces.default.key.toHex());
187
+ triggersRegistered.wake(fn.triggers);
188
+ });
189
+
190
+ void registry.open(ctx);
191
+ const functions = await triggersRegistered.wait();
192
+ const expected = triggers.map((object) => object.id).sort();
193
+ expect(functions.map((fn) => fn.id).sort()).to.deep.eq(expected);
194
+ });
195
+
196
+ test('event fired when a new trigger is added', async () => {
197
+ const client = (await createInitializedClients(testBuilder))[0];
198
+ const registry = createRegistry(client);
199
+ const space = await client.spaces.create();
200
+
201
+ const triggerRegistered = new Trigger<FunctionTrigger>();
202
+ registry.registered.on((fn) => {
203
+ expect(fn.triggers.length).to.eq(1);
204
+ triggerRegistered.wake(fn.triggers[0]);
205
+ });
206
+ await registry.open(ctx);
207
+ await registry.register(space, { triggers: manifest?.triggers?.slice(0, 1) });
208
+ const registered = await triggerRegistered.wait();
209
+ expect(registered.function).to.eq(manifest.triggers![0].function);
210
+ });
211
+
212
+ test('event fired when a new trigger is removed', async () => {
213
+ const client = (await createInitializedClients(testBuilder))[0];
214
+ const registry = createRegistry(client);
215
+ const space = await client.spaces.create();
216
+ const triggers = createTriggers(space, 3);
217
+
218
+ const triggerLoaded = new Trigger();
219
+ registry.registered.on((fn) => triggerLoaded.wake());
220
+
221
+ const triggerRemoved = new Trigger<FunctionTrigger>();
222
+ registry.removed.on((fn) => {
223
+ expect(fn.triggers.length).to.eq(1);
224
+ triggerRemoved.wake(fn.triggers[0]);
225
+ });
226
+ await registry.register(space, manifest);
227
+ await registry.open(ctx);
228
+ await triggerLoaded.wait();
229
+
230
+ space.db.remove(triggers[0]);
231
+ const removedTrigger = await triggerRemoved.wait();
232
+ expect(removedTrigger.id).to.eq(triggers[0].id);
233
+ });
234
+ });
235
+
236
+ const createRegistry = (client: Client) => {
237
+ const registry = new TriggerRegistry(client);
238
+ ctx.onDispose(() => registry.close());
239
+ return registry;
240
+ };
241
+
242
+ const createTriggers = (space: Space, count: number) => {
243
+ const triggers = range(count, () => {
244
+ const { meta, object } = splitMeta(manifest.triggers![0]);
245
+ return create(FunctionTrigger, object, meta);
246
+ });
247
+
248
+ triggers.forEach((trigger) => space.db.add(trigger));
249
+ return triggers;
250
+ };
251
+
252
+ const waitForInactiveTriggers = async (registry: TriggerRegistry, space: Space) => {
253
+ await waitForCondition({ condition: () => registry.getInactiveTriggers(space).length > 0 });
254
+ };
255
+ });
@@ -0,0 +1,189 @@
1
+ //
2
+ // Copyright 2024 DXOS.org
3
+ //
4
+
5
+ import { Event } from '@dxos/async';
6
+ import { type Client } from '@dxos/client';
7
+ import { create, Filter, getMeta, type Space } from '@dxos/client/echo';
8
+ import { Context, Resource } from '@dxos/context';
9
+ import { ECHO_ATTR_META, foreignKey, foreignKeyEquals, splitMeta } from '@dxos/echo-schema';
10
+ import { invariant } from '@dxos/invariant';
11
+ import { PublicKey } from '@dxos/keys';
12
+ import { log } from '@dxos/log';
13
+ import { ComplexMap } from '@dxos/util';
14
+
15
+ import { createSubscriptionTrigger, createTimerTrigger, createWebhookTrigger, createWebsocketTrigger } from './type';
16
+ import { type FunctionManifest, FunctionTrigger, type FunctionTriggerType, type TriggerSpec } from '../types';
17
+ import { diff, intersection } from '../util';
18
+
19
+ type ResponseCode = number;
20
+
21
+ export type TriggerCallback = (args: object) => Promise<ResponseCode>;
22
+
23
+ export type TriggerContext = { space: Space };
24
+
25
+ // TODO(burdon): Make object?
26
+ export type TriggerFactory<Spec extends TriggerSpec, Options = any> = (
27
+ ctx: Context,
28
+ context: TriggerContext,
29
+ spec: Spec,
30
+ callback: TriggerCallback,
31
+ options?: Options,
32
+ ) => Promise<void>;
33
+
34
+ export type TriggerHandlerMap = { [type in FunctionTriggerType]: TriggerFactory<any> };
35
+
36
+ const triggerHandlers: TriggerHandlerMap = {
37
+ subscription: createSubscriptionTrigger,
38
+ timer: createTimerTrigger,
39
+ webhook: createWebhookTrigger,
40
+ websocket: createWebsocketTrigger,
41
+ };
42
+
43
+ export type TriggerEvent = {
44
+ space: Space;
45
+ triggers: FunctionTrigger[];
46
+ };
47
+
48
+ type RegisteredTrigger = {
49
+ activationCtx?: Context;
50
+ trigger: FunctionTrigger;
51
+ };
52
+
53
+ export class TriggerRegistry extends Resource {
54
+ private readonly _triggersBySpaceKey = new ComplexMap<PublicKey, RegisteredTrigger[]>(PublicKey.hash);
55
+
56
+ public readonly registered = new Event<TriggerEvent>();
57
+ public readonly removed = new Event<TriggerEvent>();
58
+
59
+ constructor(
60
+ private readonly _client: Client,
61
+ private readonly _options?: TriggerHandlerMap,
62
+ ) {
63
+ super();
64
+ }
65
+
66
+ public getActiveTriggers(space: Space): FunctionTrigger[] {
67
+ return this._getTriggers(space, (t) => t.activationCtx != null);
68
+ }
69
+
70
+ public getInactiveTriggers(space: Space): FunctionTrigger[] {
71
+ return this._getTriggers(space, (t) => t.activationCtx == null);
72
+ }
73
+
74
+ async activate(triggerCtx: TriggerContext, trigger: FunctionTrigger, callback: TriggerCallback): Promise<void> {
75
+ log('activate', { space: triggerCtx.space.key, trigger });
76
+ const activationCtx = new Context({ name: `trigger_${trigger.function}` });
77
+ this._ctx.onDispose(() => activationCtx.dispose());
78
+ const registeredTrigger = this._triggersBySpaceKey
79
+ .get(triggerCtx.space.key)
80
+ ?.find((reg) => reg.trigger.id === trigger.id);
81
+ invariant(registeredTrigger, `Trigger is not registered: ${trigger.function}`);
82
+ registeredTrigger.activationCtx = activationCtx;
83
+
84
+ try {
85
+ const options = this._options?.[trigger.spec.type];
86
+ await triggerHandlers[trigger.spec.type](activationCtx, triggerCtx, trigger.spec, callback, options);
87
+ } catch (err) {
88
+ delete registeredTrigger.activationCtx;
89
+ throw err;
90
+ }
91
+ }
92
+
93
+ /**
94
+ * Loads triggers from the manifest into the space.
95
+ */
96
+ public async register(space: Space, manifest: FunctionManifest): Promise<void> {
97
+ log('register', { space: space.key });
98
+ if (!manifest.triggers?.length) {
99
+ return;
100
+ }
101
+ if (!space.db.graph.runtimeSchemaRegistry.hasSchema(FunctionTrigger)) {
102
+ space.db.graph.runtimeSchemaRegistry.registerSchema(FunctionTrigger);
103
+ }
104
+
105
+ // Sync triggers.
106
+ const { objects: existing } = await space.db.query(Filter.schema(FunctionTrigger)).run();
107
+ const { added, removed } = diff(existing, manifest.triggers, (a, b) => {
108
+ // Create FK to enable syncing if none are set.
109
+ // TODO(burdon): Warn if not unique.
110
+ const keys = b[ECHO_ATTR_META]?.keys ?? [foreignKey('manifest', [b.function, b.spec.type].join('-'))];
111
+ return intersection(getMeta(a)?.keys ?? [], keys, foreignKeyEquals).length > 0;
112
+ });
113
+
114
+ added.forEach((trigger) => {
115
+ const { meta, object } = splitMeta(trigger);
116
+ space.db.add(create(FunctionTrigger, object, meta));
117
+ });
118
+ // TODO(burdon): Update existing triggers.
119
+ removed.forEach((trigger) => space.db.remove(trigger));
120
+ }
121
+
122
+ protected override async _open(): Promise<void> {
123
+ const spaceListSubscription = this._client.spaces.subscribe(async (spaces) => {
124
+ for (const space of spaces) {
125
+ if (this._triggersBySpaceKey.has(space.key)) {
126
+ continue;
127
+ }
128
+
129
+ const registered: RegisteredTrigger[] = [];
130
+ this._triggersBySpaceKey.set(space.key, registered);
131
+ await space.waitUntilReady();
132
+ if (this._ctx.disposed) {
133
+ break;
134
+ }
135
+ const functionsSubscription = space.db.query(Filter.schema(FunctionTrigger)).subscribe(async (triggers) => {
136
+ await this._handleRemovedTriggers(space, triggers.objects, registered);
137
+ this._handleNewTriggers(space, triggers.objects, registered);
138
+ });
139
+
140
+ this._ctx.onDispose(functionsSubscription);
141
+ }
142
+ });
143
+
144
+ this._ctx.onDispose(() => spaceListSubscription.unsubscribe());
145
+ }
146
+
147
+ protected override async _close(_: Context): Promise<void> {
148
+ this._triggersBySpaceKey.clear();
149
+ }
150
+
151
+ private _handleNewTriggers(space: Space, allTriggers: FunctionTrigger[], registered: RegisteredTrigger[]) {
152
+ const newTriggers = allTriggers.filter((candidate) => {
153
+ return registered.find((reg) => reg.trigger.id === candidate.id) == null;
154
+ });
155
+
156
+ if (newTriggers.length > 0) {
157
+ const newRegisteredTriggers: RegisteredTrigger[] = newTriggers.map((trigger) => ({ trigger }));
158
+ registered.push(...newRegisteredTriggers);
159
+ log('registered new triggers', () => ({ spaceKey: space.key, functions: newTriggers.map((t) => t.function) }));
160
+ this.registered.emit({ space, triggers: newTriggers });
161
+ }
162
+ }
163
+
164
+ private async _handleRemovedTriggers(
165
+ space: Space,
166
+ allTriggers: FunctionTrigger[],
167
+ registered: RegisteredTrigger[],
168
+ ): Promise<void> {
169
+ const removed: FunctionTrigger[] = [];
170
+ for (let i = registered.length - 1; i >= 0; i--) {
171
+ const wasRemoved =
172
+ allTriggers.find((trigger: FunctionTrigger) => trigger.id === registered[i].trigger.id) == null;
173
+ if (wasRemoved) {
174
+ const unregistered = registered.splice(i, 1)[0];
175
+ await unregistered.activationCtx?.dispose();
176
+ removed.push(unregistered.trigger);
177
+ }
178
+ }
179
+
180
+ if (removed.length > 0) {
181
+ this.removed.emit({ space, triggers: removed });
182
+ }
183
+ }
184
+
185
+ private _getTriggers(space: Space, predicate: (trigger: RegisteredTrigger) => boolean): FunctionTrigger[] {
186
+ const allSpaceTriggers = this._triggersBySpaceKey.get(space.key) ?? [];
187
+ return allSpaceTriggers.filter(predicate).map((trigger) => trigger.trigger);
188
+ }
189
+ }
@@ -0,0 +1,8 @@
1
+ //
2
+ // Copyright 2024 DXOS.org
3
+ //
4
+
5
+ export * from './subscription-trigger';
6
+ export * from './timer-trigger';
7
+ export * from './webhook-trigger';
8
+ export * from './websocket-trigger';