@dxos/functions 0.5.3-main.6f2dfea → 0.5.3-main.79e0565
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/lib/browser/index.mjs +764 -476
- package/dist/lib/browser/index.mjs.map +4 -4
- package/dist/lib/browser/meta.json +1 -1
- package/dist/lib/node/index.cjs +745 -471
- package/dist/lib/node/index.cjs.map +4 -4
- package/dist/lib/node/meta.json +1 -1
- package/dist/types/src/function/function-registry.d.ts +24 -0
- package/dist/types/src/function/function-registry.d.ts.map +1 -0
- package/dist/types/src/function/function-registry.test.d.ts +2 -0
- package/dist/types/src/function/function-registry.test.d.ts.map +1 -0
- package/dist/types/src/function/index.d.ts +2 -0
- package/dist/types/src/function/index.d.ts.map +1 -0
- package/dist/types/src/handler.d.ts +32 -12
- package/dist/types/src/handler.d.ts.map +1 -1
- package/dist/types/src/index.d.ts +2 -0
- package/dist/types/src/index.d.ts.map +1 -1
- package/dist/types/src/runtime/dev-server.d.ts +7 -10
- package/dist/types/src/runtime/dev-server.d.ts.map +1 -1
- package/dist/types/src/runtime/scheduler.d.ts +11 -59
- package/dist/types/src/runtime/scheduler.d.ts.map +1 -1
- package/dist/types/src/testing/functions-integration.test.d.ts +2 -0
- package/dist/types/src/testing/functions-integration.test.d.ts.map +1 -0
- package/dist/types/src/testing/index.d.ts +4 -0
- package/dist/types/src/testing/index.d.ts.map +1 -0
- package/dist/types/src/testing/setup.d.ts +5 -0
- package/dist/types/src/testing/setup.d.ts.map +1 -0
- package/dist/types/src/testing/test/handler.d.ts +1 -0
- package/dist/types/src/testing/test/handler.d.ts.map +1 -1
- package/dist/types/src/testing/types.d.ts +9 -0
- package/dist/types/src/testing/types.d.ts.map +1 -0
- package/dist/types/src/testing/util.d.ts +3 -0
- package/dist/types/src/testing/util.d.ts.map +1 -0
- package/dist/types/src/trigger/index.d.ts +2 -0
- package/dist/types/src/trigger/index.d.ts.map +1 -0
- package/dist/types/src/trigger/trigger-registry.d.ts +40 -0
- package/dist/types/src/trigger/trigger-registry.d.ts.map +1 -0
- package/dist/types/src/trigger/trigger-registry.test.d.ts +2 -0
- package/dist/types/src/trigger/trigger-registry.test.d.ts.map +1 -0
- package/dist/types/src/trigger/type/index.d.ts +5 -0
- package/dist/types/src/trigger/type/index.d.ts.map +1 -0
- package/dist/types/src/trigger/type/subscription-trigger.d.ts +4 -0
- package/dist/types/src/trigger/type/subscription-trigger.d.ts.map +1 -0
- package/dist/types/src/trigger/type/timer-trigger.d.ts +4 -0
- package/dist/types/src/trigger/type/timer-trigger.d.ts.map +1 -0
- package/dist/types/src/trigger/type/webhook-trigger.d.ts +4 -0
- package/dist/types/src/trigger/type/webhook-trigger.d.ts.map +1 -0
- package/dist/types/src/trigger/type/websocket-trigger.d.ts +13 -0
- package/dist/types/src/trigger/type/websocket-trigger.d.ts.map +1 -0
- package/dist/types/src/types.d.ts +131 -111
- package/dist/types/src/types.d.ts.map +1 -1
- package/dist/types/src/util.d.ts +15 -0
- package/dist/types/src/util.d.ts.map +1 -0
- package/dist/types/src/util.test.d.ts +2 -0
- package/dist/types/src/util.test.d.ts.map +1 -0
- package/package.json +14 -12
- package/schema/functions.json +140 -112
- package/src/function/function-registry.test.ts +105 -0
- package/src/function/function-registry.ts +90 -0
- package/src/function/index.ts +5 -0
- package/src/handler.ts +50 -27
- package/src/index.ts +2 -0
- package/src/runtime/dev-server.test.ts +15 -35
- package/src/runtime/dev-server.ts +40 -23
- package/src/runtime/scheduler.test.ts +54 -75
- package/src/runtime/scheduler.ts +75 -300
- package/src/testing/functions-integration.test.ts +99 -0
- package/src/testing/index.ts +7 -0
- package/src/testing/setup.ts +45 -0
- package/src/testing/test/handler.ts +8 -2
- package/src/testing/types.ts +9 -0
- package/src/testing/util.ts +16 -0
- package/src/trigger/index.ts +5 -0
- package/src/trigger/trigger-registry.test.ts +255 -0
- package/src/trigger/trigger-registry.ts +189 -0
- package/src/trigger/type/index.ts +8 -0
- package/src/trigger/type/subscription-trigger.ts +80 -0
- package/src/trigger/type/timer-trigger.ts +44 -0
- package/src/trigger/type/webhook-trigger.ts +47 -0
- package/src/trigger/type/websocket-trigger.ts +91 -0
- package/src/types.ts +58 -40
- package/src/util.test.ts +43 -0
- package/src/util.ts +48 -0
package/src/runtime/scheduler.ts
CHANGED
|
@@ -2,30 +2,17 @@
|
|
|
2
2
|
// Copyright 2023 DXOS.org
|
|
3
3
|
//
|
|
4
4
|
|
|
5
|
-
import { CronJob } from 'cron';
|
|
6
|
-
import { getPort } from 'get-port-please';
|
|
7
|
-
import http from 'node:http';
|
|
8
5
|
import path from 'node:path';
|
|
9
|
-
import WebSocket from 'ws';
|
|
10
6
|
|
|
11
|
-
import {
|
|
12
|
-
import {
|
|
13
|
-
import { type Client, type PublicKey } from '@dxos/client';
|
|
14
|
-
import { createSubscription, Filter, getAutomergeObjectCore, type Query, type Space } from '@dxos/client/echo';
|
|
7
|
+
import { Mutex } from '@dxos/async';
|
|
8
|
+
import { type Space } from '@dxos/client/echo';
|
|
15
9
|
import { Context } from '@dxos/context';
|
|
16
|
-
import { invariant } from '@dxos/invariant';
|
|
17
10
|
import { log } from '@dxos/log';
|
|
18
|
-
import { ComplexMap } from '@dxos/util';
|
|
19
11
|
|
|
20
|
-
import {
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
type SubscriptionTrigger,
|
|
25
|
-
type TimerTrigger,
|
|
26
|
-
type WebhookTrigger,
|
|
27
|
-
type WebsocketTrigger,
|
|
28
|
-
} from '../types';
|
|
12
|
+
import { type FunctionRegistry } from '../function';
|
|
13
|
+
import { type FunctionEventMeta } from '../handler';
|
|
14
|
+
import { type TriggerRegistry } from '../trigger';
|
|
15
|
+
import { type FunctionDef, type FunctionManifest, type FunctionTrigger } from '../types';
|
|
29
16
|
|
|
30
17
|
export type Callback = (data: any) => Promise<void | number>;
|
|
31
18
|
|
|
@@ -38,111 +25,99 @@ export type SchedulerOptions = {
|
|
|
38
25
|
* The scheduler triggers function execution based on various triggers.
|
|
39
26
|
*/
|
|
40
27
|
export class Scheduler {
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
{ ctx: Context; trigger: FunctionTrigger }
|
|
45
|
-
>(({ spaceKey, id }) => `${spaceKey.toHex()}:${id}`);
|
|
28
|
+
private _ctx = createContext();
|
|
29
|
+
|
|
30
|
+
private readonly _callMutex = new Mutex();
|
|
46
31
|
|
|
47
32
|
constructor(
|
|
48
|
-
|
|
49
|
-
|
|
33
|
+
public readonly functions: FunctionRegistry,
|
|
34
|
+
public readonly triggers: TriggerRegistry,
|
|
50
35
|
private readonly _options: SchedulerOptions = {},
|
|
51
|
-
) {
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
}
|
|
36
|
+
) {
|
|
37
|
+
this.functions.registered.on(async ({ space, added }) => {
|
|
38
|
+
await this._safeActivateTriggers(space, this.triggers.getInactiveTriggers(space), added);
|
|
39
|
+
});
|
|
40
|
+
this.triggers.registered.on(async ({ space, triggers }) => {
|
|
41
|
+
await this._safeActivateTriggers(space, triggers, this.functions.getFunctions(space));
|
|
42
|
+
});
|
|
58
43
|
}
|
|
59
44
|
|
|
60
45
|
async start() {
|
|
61
|
-
this.
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
await this.mount(new Context(), space, trigger);
|
|
66
|
-
}
|
|
67
|
-
}
|
|
68
|
-
});
|
|
46
|
+
await this._ctx.dispose();
|
|
47
|
+
this._ctx = createContext();
|
|
48
|
+
await this.functions.open(this._ctx);
|
|
49
|
+
await this.triggers.open(this._ctx);
|
|
69
50
|
}
|
|
70
51
|
|
|
71
52
|
async stop() {
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
53
|
+
await this._ctx.dispose();
|
|
54
|
+
await this.functions.close();
|
|
55
|
+
await this.triggers.close();
|
|
75
56
|
}
|
|
76
57
|
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
const def = this._manifest.functions.find((config) => config.id === trigger.function);
|
|
83
|
-
invariant(def, `Function not found: ${trigger.function}`);
|
|
84
|
-
|
|
85
|
-
// TODO(burdon): Currently supports only one trigger declaration per function.
|
|
86
|
-
const exists = this._mounts.get(key);
|
|
87
|
-
if (!exists) {
|
|
88
|
-
this._mounts.set(key, { ctx, trigger });
|
|
89
|
-
log('mount', { space: space.key, trigger });
|
|
90
|
-
if (ctx.disposed) {
|
|
91
|
-
return;
|
|
92
|
-
}
|
|
93
|
-
|
|
94
|
-
//
|
|
95
|
-
// Triggers types.
|
|
96
|
-
//
|
|
97
|
-
|
|
98
|
-
if (trigger.timer) {
|
|
99
|
-
await this._createTimer(ctx, space, def, trigger.timer);
|
|
100
|
-
}
|
|
101
|
-
|
|
102
|
-
if (trigger.webhook) {
|
|
103
|
-
await this._createWebhook(ctx, space, def, trigger.webhook);
|
|
104
|
-
}
|
|
105
|
-
|
|
106
|
-
if (trigger.websocket) {
|
|
107
|
-
await this._createWebsocket(ctx, space, def, trigger.websocket);
|
|
108
|
-
}
|
|
58
|
+
// TODO(burdon): Remove and update registries directly.
|
|
59
|
+
public async register(space: Space, manifest: FunctionManifest) {
|
|
60
|
+
await this.functions.register(space, manifest.functions);
|
|
61
|
+
await this.triggers.register(space, manifest);
|
|
62
|
+
}
|
|
109
63
|
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
64
|
+
private async _safeActivateTriggers(
|
|
65
|
+
space: Space,
|
|
66
|
+
triggers: FunctionTrigger[],
|
|
67
|
+
functions: FunctionDef[],
|
|
68
|
+
): Promise<void> {
|
|
69
|
+
const mountTasks = triggers.map((trigger) => {
|
|
70
|
+
return this.activate(space, functions, trigger);
|
|
71
|
+
});
|
|
72
|
+
await Promise.all(mountTasks).catch(log.catch);
|
|
114
73
|
}
|
|
115
74
|
|
|
116
|
-
private async
|
|
117
|
-
const
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
await ctx.dispose();
|
|
75
|
+
private async activate(space: Space, functions: FunctionDef[], fnTrigger: FunctionTrigger) {
|
|
76
|
+
const definition = functions.find((def) => def.uri === fnTrigger.function);
|
|
77
|
+
if (!definition) {
|
|
78
|
+
log.info('function is not found for trigger', { fnTrigger });
|
|
79
|
+
return;
|
|
122
80
|
}
|
|
81
|
+
|
|
82
|
+
await this.triggers.activate({ space }, fnTrigger, async (args) => {
|
|
83
|
+
return this._callMutex.executeSynchronized(() => {
|
|
84
|
+
return this._execFunction(definition, fnTrigger, {
|
|
85
|
+
meta: fnTrigger.meta,
|
|
86
|
+
data: { ...args, spaceKey: space.key },
|
|
87
|
+
});
|
|
88
|
+
});
|
|
89
|
+
});
|
|
90
|
+
|
|
91
|
+
log('activated trigger', { space: space.key, trigger: fnTrigger });
|
|
123
92
|
}
|
|
124
93
|
|
|
125
|
-
|
|
126
|
-
|
|
94
|
+
private async _execFunction<TData, TMeta>(
|
|
95
|
+
def: FunctionDef,
|
|
96
|
+
trigger: FunctionTrigger,
|
|
97
|
+
{ data, meta }: { data: TData; meta?: TMeta },
|
|
98
|
+
): Promise<number> {
|
|
99
|
+
let status = 0;
|
|
127
100
|
try {
|
|
128
|
-
|
|
101
|
+
// TODO(burdon): Pass in Space key (common context)?
|
|
102
|
+
const payload = Object.assign({}, meta && ({ meta } satisfies FunctionEventMeta<TMeta>), data);
|
|
103
|
+
|
|
129
104
|
const { endpoint, callback } = this._options;
|
|
130
105
|
if (endpoint) {
|
|
131
106
|
// TODO(burdon): Move out of scheduler (generalize as callback).
|
|
132
|
-
const url = path.join(endpoint, def.
|
|
133
|
-
log.info('exec', { function: def.
|
|
107
|
+
const url = path.join(endpoint, def.route);
|
|
108
|
+
log.info('exec', { function: def.uri, url, triggerType: trigger.spec.type });
|
|
134
109
|
const response = await fetch(url, {
|
|
135
110
|
method: 'POST',
|
|
136
111
|
headers: {
|
|
137
112
|
'Content-Type': 'application/json',
|
|
138
113
|
},
|
|
139
|
-
body: JSON.stringify(
|
|
114
|
+
body: JSON.stringify(payload),
|
|
140
115
|
});
|
|
141
116
|
|
|
142
117
|
status = response.status;
|
|
143
118
|
} else if (callback) {
|
|
144
|
-
log.info('exec', { function: def.
|
|
145
|
-
status = (await callback(
|
|
119
|
+
log.info('exec', { function: def.uri });
|
|
120
|
+
status = (await callback(payload)) ?? 200;
|
|
146
121
|
}
|
|
147
122
|
|
|
148
123
|
// Check errors.
|
|
@@ -151,214 +126,14 @@ export class Scheduler {
|
|
|
151
126
|
}
|
|
152
127
|
|
|
153
128
|
// const result = await response.json();
|
|
154
|
-
log.info('done', { function: def.
|
|
155
|
-
return status;
|
|
129
|
+
log.info('done', { function: def.uri, status });
|
|
156
130
|
} catch (err: any) {
|
|
157
|
-
log.error('error', { function: def.
|
|
158
|
-
|
|
159
|
-
}
|
|
160
|
-
}
|
|
161
|
-
|
|
162
|
-
//
|
|
163
|
-
// Triggers
|
|
164
|
-
//
|
|
165
|
-
|
|
166
|
-
/**
|
|
167
|
-
* Cron timer.
|
|
168
|
-
*/
|
|
169
|
-
private async _createTimer(ctx: Context, space: Space, def: FunctionDef, trigger: TimerTrigger) {
|
|
170
|
-
log.info('timer', { space: space.key, trigger });
|
|
171
|
-
const { cron } = trigger;
|
|
172
|
-
|
|
173
|
-
const task = new DeferredTask(ctx, async () => {
|
|
174
|
-
await this._execFunction(def, { spaceKey: space.key });
|
|
175
|
-
});
|
|
176
|
-
|
|
177
|
-
let last = 0;
|
|
178
|
-
let run = 0;
|
|
179
|
-
// https://www.npmjs.com/package/cron#constructor
|
|
180
|
-
const job = CronJob.from({
|
|
181
|
-
cronTime: cron,
|
|
182
|
-
runOnInit: false,
|
|
183
|
-
onTick: () => {
|
|
184
|
-
// TODO(burdon): Check greater than 30s (use cron-parser).
|
|
185
|
-
const now = Date.now();
|
|
186
|
-
const delta = last ? now - last : 0;
|
|
187
|
-
last = now;
|
|
188
|
-
|
|
189
|
-
run++;
|
|
190
|
-
log.info('tick', { space: space.key.truncate(), count: run, delta });
|
|
191
|
-
task.schedule();
|
|
192
|
-
},
|
|
193
|
-
});
|
|
194
|
-
|
|
195
|
-
job.start();
|
|
196
|
-
ctx.onDispose(() => job.stop());
|
|
197
|
-
}
|
|
198
|
-
|
|
199
|
-
/**
|
|
200
|
-
* Webhook.
|
|
201
|
-
*/
|
|
202
|
-
private async _createWebhook(ctx: Context, space: Space, def: FunctionDef, trigger: WebhookTrigger) {
|
|
203
|
-
log.info('webhook', { space: space.key, trigger });
|
|
204
|
-
|
|
205
|
-
// TODO(burdon): Enable POST hook with payload.
|
|
206
|
-
const server = http.createServer(async (req, res) => {
|
|
207
|
-
if (req.method !== trigger.method) {
|
|
208
|
-
res.statusCode = 405;
|
|
209
|
-
return res.end();
|
|
210
|
-
}
|
|
211
|
-
|
|
212
|
-
res.statusCode = await this._execFunction(def, { spaceKey: space.key });
|
|
213
|
-
res.end();
|
|
214
|
-
});
|
|
215
|
-
|
|
216
|
-
// TODO(burdon): Not used.
|
|
217
|
-
// const DEF_PORT_RANGE = { min: 7500, max: 7599 };
|
|
218
|
-
// const portRange = Object.assign({}, trigger.port, DEF_PORT_RANGE) as WebhookTrigger['port'];
|
|
219
|
-
const port = await getPort({
|
|
220
|
-
random: true,
|
|
221
|
-
// portRange: [portRange!.min, portRange!.max],
|
|
222
|
-
});
|
|
223
|
-
|
|
224
|
-
// TODO(burdon): Update trigger object with actual port.
|
|
225
|
-
server.listen(port, () => {
|
|
226
|
-
log.info('started webhook', { port });
|
|
227
|
-
trigger.port = port;
|
|
228
|
-
});
|
|
229
|
-
|
|
230
|
-
ctx.onDispose(() => {
|
|
231
|
-
server.close();
|
|
232
|
-
});
|
|
233
|
-
}
|
|
234
|
-
|
|
235
|
-
/**
|
|
236
|
-
* Websocket.
|
|
237
|
-
* NOTE: The port must be unique, so the same hook cannot be used for multiple spaces.
|
|
238
|
-
*/
|
|
239
|
-
private async _createWebsocket(
|
|
240
|
-
ctx: Context,
|
|
241
|
-
space: Space,
|
|
242
|
-
def: FunctionDef,
|
|
243
|
-
trigger: WebsocketTrigger,
|
|
244
|
-
options: {
|
|
245
|
-
retryDelay: number;
|
|
246
|
-
maxAttempts: number;
|
|
247
|
-
} = {
|
|
248
|
-
retryDelay: 2,
|
|
249
|
-
maxAttempts: 5,
|
|
250
|
-
},
|
|
251
|
-
) {
|
|
252
|
-
log.info('websocket', { space: space.key, trigger });
|
|
253
|
-
const { url } = trigger;
|
|
254
|
-
|
|
255
|
-
let ws: WebSocket;
|
|
256
|
-
for (let attempt = 1; attempt <= options.maxAttempts; attempt++) {
|
|
257
|
-
const open = new Trigger<boolean>();
|
|
258
|
-
|
|
259
|
-
ws = new WebSocket(url);
|
|
260
|
-
Object.assign(ws, {
|
|
261
|
-
onopen: () => {
|
|
262
|
-
log.info('opened', { url });
|
|
263
|
-
if (trigger.init) {
|
|
264
|
-
ws.send(new TextEncoder().encode(JSON.stringify(trigger.init)));
|
|
265
|
-
}
|
|
266
|
-
|
|
267
|
-
open.wake(true);
|
|
268
|
-
},
|
|
269
|
-
|
|
270
|
-
// TODO(burdon): Config retry if server closes?
|
|
271
|
-
onclose: (event) => {
|
|
272
|
-
log.info('closed', { url, code: event.code });
|
|
273
|
-
open.wake(false);
|
|
274
|
-
},
|
|
275
|
-
|
|
276
|
-
onerror: (event) => {
|
|
277
|
-
log.catch(event.error, { url });
|
|
278
|
-
},
|
|
279
|
-
|
|
280
|
-
onmessage: async (event) => {
|
|
281
|
-
try {
|
|
282
|
-
const data = JSON.parse(new TextDecoder().decode(event.data as Uint8Array));
|
|
283
|
-
await this._execFunction(def, { spaceKey: space.key, data });
|
|
284
|
-
} catch (err) {
|
|
285
|
-
log.catch(err, { url });
|
|
286
|
-
}
|
|
287
|
-
},
|
|
288
|
-
} satisfies Partial<WebSocket>);
|
|
289
|
-
|
|
290
|
-
const isOpen = await open.wait();
|
|
291
|
-
if (isOpen) {
|
|
292
|
-
break;
|
|
293
|
-
} else {
|
|
294
|
-
const wait = Math.pow(attempt, 2) * options.retryDelay;
|
|
295
|
-
if (attempt < options.maxAttempts) {
|
|
296
|
-
log.warn(`failed to connect; trying again in ${wait}s`, { attempt });
|
|
297
|
-
await sleep(wait * 1_000);
|
|
298
|
-
}
|
|
299
|
-
}
|
|
131
|
+
log.error('error', { function: def.uri, error: err.message });
|
|
132
|
+
status = 500;
|
|
300
133
|
}
|
|
301
134
|
|
|
302
|
-
|
|
303
|
-
ws?.close();
|
|
304
|
-
});
|
|
305
|
-
}
|
|
306
|
-
|
|
307
|
-
/**
|
|
308
|
-
* ECHO subscription.
|
|
309
|
-
*/
|
|
310
|
-
private async _createSubscription(ctx: Context, space: Space, def: FunctionDef, trigger: SubscriptionTrigger) {
|
|
311
|
-
log.info('subscription', { space: space.key, trigger });
|
|
312
|
-
const objectIds = new Set<string>();
|
|
313
|
-
const task = new DeferredTask(ctx, async () => {
|
|
314
|
-
await this._execFunction(def, { spaceKey: space.key, objects: Array.from(objectIds) });
|
|
315
|
-
});
|
|
316
|
-
|
|
317
|
-
// TODO(burdon): Don't fire initially.
|
|
318
|
-
// TODO(burdon): Subscription is called THREE times.
|
|
319
|
-
const subscriptions: (() => void)[] = [];
|
|
320
|
-
const subscription = createSubscription(({ added, updated }) => {
|
|
321
|
-
log.info('updated', { added: added.length, updated: updated.length });
|
|
322
|
-
for (const object of added) {
|
|
323
|
-
objectIds.add(object.id);
|
|
324
|
-
}
|
|
325
|
-
for (const object of updated) {
|
|
326
|
-
objectIds.add(object.id);
|
|
327
|
-
}
|
|
328
|
-
|
|
329
|
-
task.schedule();
|
|
330
|
-
});
|
|
331
|
-
|
|
332
|
-
subscriptions.push(() => subscription.unsubscribe());
|
|
333
|
-
|
|
334
|
-
// TODO(burdon): Create queue. Only allow one invocation per trigger at a time?
|
|
335
|
-
// TODO(burdon): Disable trigger if keeps failing.
|
|
336
|
-
const { filter, options: { deep, delay } = {} } = trigger;
|
|
337
|
-
const update = ({ objects }: Query) => {
|
|
338
|
-
subscription.update(objects);
|
|
339
|
-
|
|
340
|
-
// TODO(burdon): Hack to monitor changes to Document's text object.
|
|
341
|
-
if (deep) {
|
|
342
|
-
log.info('update', { objects: objects.length });
|
|
343
|
-
for (const object of objects) {
|
|
344
|
-
const content = object.content;
|
|
345
|
-
if (content instanceof TextV0Type) {
|
|
346
|
-
subscriptions.push(
|
|
347
|
-
getAutomergeObjectCore(content).updates.on(debounce(() => subscription.update([object]), 1_000)),
|
|
348
|
-
);
|
|
349
|
-
}
|
|
350
|
-
}
|
|
351
|
-
}
|
|
352
|
-
};
|
|
353
|
-
|
|
354
|
-
// TODO(burdon): Is Filter.or implemented?
|
|
355
|
-
// TODO(burdon): [Bug]: all callbacks are fired on the first mutation.
|
|
356
|
-
// TODO(burdon): [Bug]: not updated when document is deleted (either top or hierarchically).
|
|
357
|
-
const query = space.db.query(Filter.or(filter.map(({ type, props }) => Filter.typename(type, props))));
|
|
358
|
-
subscriptions.push(query.subscribe(delay ? debounce(update, delay) : update));
|
|
359
|
-
|
|
360
|
-
ctx.onDispose(() => {
|
|
361
|
-
subscriptions.forEach((unsubscribe) => unsubscribe());
|
|
362
|
-
});
|
|
135
|
+
return status;
|
|
363
136
|
}
|
|
364
137
|
}
|
|
138
|
+
|
|
139
|
+
const createContext = () => new Context({ name: 'FunctionScheduler' });
|
|
@@ -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,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
|
+
};
|
|
@@ -4,6 +4,12 @@
|
|
|
4
4
|
|
|
5
5
|
import { type FunctionHandler } from '../../handler';
|
|
6
6
|
|
|
7
|
-
|
|
8
|
-
|
|
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);
|
|
9
15
|
};
|
|
@@ -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
|
+
};
|