@dxos/functions 0.5.2 → 0.5.3-main.056e7da
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/chunk-4D4I3YMJ.mjs +86 -0
- package/dist/lib/browser/chunk-4D4I3YMJ.mjs.map +7 -0
- package/dist/lib/browser/index.mjs +915 -297
- package/dist/lib/browser/index.mjs.map +4 -4
- package/dist/lib/browser/meta.json +1 -1
- package/dist/lib/browser/types.mjs +14 -0
- package/dist/lib/browser/types.mjs.map +7 -0
- package/dist/lib/node/chunk-3UYUR5N5.cjs +103 -0
- package/dist/lib/node/chunk-3UYUR5N5.cjs.map +7 -0
- package/dist/lib/node/index.cjs +901 -290
- package/dist/lib/node/index.cjs.map +4 -4
- package/dist/lib/node/meta.json +1 -1
- package/dist/lib/node/types.cjs +35 -0
- package/dist/lib/node/types.cjs.map +7 -0
- package/dist/types/src/browser/index.d.ts +2 -0
- package/dist/types/src/browser/index.d.ts.map +1 -0
- 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 +33 -12
- package/dist/types/src/handler.d.ts.map +1 -1
- package/dist/types/src/index.d.ts +3 -1
- package/dist/types/src/index.d.ts.map +1 -1
- package/dist/types/src/runtime/dev-server.d.ts +15 -7
- package/dist/types/src/runtime/dev-server.d.ts.map +1 -1
- package/dist/types/src/runtime/dev-server.test.d.ts +2 -0
- package/dist/types/src/runtime/dev-server.test.d.ts.map +1 -0
- package/dist/types/src/runtime/scheduler.d.ts +15 -15
- 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 +4 -0
- package/dist/types/src/testing/test/handler.d.ts.map +1 -0
- package/dist/types/src/testing/test/index.d.ts +3 -0
- package/dist/types/src/testing/test/index.d.ts.map +1 -0
- 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 +37 -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 +215 -0
- package/dist/types/src/types.d.ts.map +1 -0
- package/dist/types/tools/schema.d.ts +2 -0
- package/dist/types/tools/schema.d.ts.map +1 -0
- package/package.json +38 -13
- package/schema/functions.json +211 -0
- package/src/browser/index.ts +5 -0
- 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 +56 -26
- package/src/index.ts +3 -1
- package/src/runtime/dev-server.test.ts +60 -0
- package/src/runtime/dev-server.ts +104 -53
- package/src/runtime/scheduler.test.ts +159 -21
- package/src/runtime/scheduler.ts +87 -150
- package/src/testing/functions-integration.test.ts +100 -0
- package/src/testing/index.ts +7 -0
- package/src/testing/setup.ts +43 -0
- package/src/testing/test/handler.ts +15 -0
- package/src/testing/test/index.ts +7 -0
- 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 +272 -0
- package/src/trigger/trigger-registry.ts +211 -0
- package/src/trigger/type/index.ts +8 -0
- package/src/trigger/type/subscription-trigger.ts +86 -0
- package/src/trigger/type/timer-trigger.ts +45 -0
- package/src/trigger/type/webhook-trigger.ts +48 -0
- package/src/trigger/type/websocket-trigger.ts +92 -0
- package/src/types.ts +115 -0
- package/dist/types/src/manifest.d.ts +0 -26
- package/dist/types/src/manifest.d.ts.map +0 -1
- package/src/manifest.ts +0 -42
package/src/runtime/scheduler.ts
CHANGED
|
@@ -2,206 +2,143 @@
|
|
|
2
2
|
// Copyright 2023 DXOS.org
|
|
3
3
|
//
|
|
4
4
|
|
|
5
|
-
import
|
|
5
|
+
import path from 'node:path';
|
|
6
6
|
|
|
7
|
-
import {
|
|
8
|
-
import {
|
|
9
|
-
import { type Client, type PublicKey } from '@dxos/client';
|
|
10
|
-
import { type Space, Filter, createSubscription, type Query, getAutomergeObjectCore } from '@dxos/client/echo';
|
|
7
|
+
import { Mutex } from '@dxos/async';
|
|
8
|
+
import { type Space } from '@dxos/client/echo';
|
|
11
9
|
import { Context } from '@dxos/context';
|
|
12
|
-
import { invariant } from '@dxos/invariant';
|
|
13
10
|
import { log } from '@dxos/log';
|
|
14
|
-
import { ComplexMap } from '@dxos/util';
|
|
15
11
|
|
|
16
|
-
import { type
|
|
17
|
-
import { type
|
|
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';
|
|
18
16
|
|
|
19
|
-
type Callback = (data:
|
|
17
|
+
export type Callback = (data: any) => Promise<void | number>;
|
|
20
18
|
|
|
21
|
-
type SchedulerOptions = {
|
|
19
|
+
export type SchedulerOptions = {
|
|
22
20
|
endpoint?: string;
|
|
23
21
|
callback?: Callback;
|
|
24
22
|
};
|
|
25
23
|
|
|
26
24
|
/**
|
|
27
|
-
*
|
|
25
|
+
* The scheduler triggers function execution based on various triggers.
|
|
28
26
|
*/
|
|
29
|
-
// TODO(burdon): Create tests.
|
|
30
27
|
export class Scheduler {
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
{ ctx: Context; trigger: FunctionTrigger }
|
|
35
|
-
>(({ id, spaceKey }) => `${spaceKey.toHex()}:${id}`);
|
|
28
|
+
private _ctx = createContext();
|
|
29
|
+
|
|
30
|
+
private readonly _functionUriToCallMutex = new Map<string, Mutex>();
|
|
36
31
|
|
|
37
32
|
constructor(
|
|
38
|
-
|
|
39
|
-
|
|
33
|
+
public readonly functions: FunctionRegistry,
|
|
34
|
+
public readonly triggers: TriggerRegistry,
|
|
40
35
|
private readonly _options: SchedulerOptions = {},
|
|
41
|
-
) {
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
for (const trigger of this._manifest.triggers ?? []) {
|
|
48
|
-
await this.mount(new Context(), space, trigger);
|
|
49
|
-
}
|
|
50
|
-
}
|
|
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));
|
|
51
42
|
});
|
|
52
43
|
}
|
|
53
44
|
|
|
54
|
-
async
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
45
|
+
async start() {
|
|
46
|
+
await this._ctx.dispose();
|
|
47
|
+
this._ctx = createContext();
|
|
48
|
+
await this.functions.open(this._ctx);
|
|
49
|
+
await this.triggers.open(this._ctx);
|
|
58
50
|
}
|
|
59
51
|
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
// Currently supports only one trigger declaration per function.
|
|
66
|
-
const exists = this._mounts.get(key);
|
|
67
|
-
if (!exists) {
|
|
68
|
-
this._mounts.set(key, { ctx, trigger });
|
|
69
|
-
log('mount', { space: space.key, trigger });
|
|
70
|
-
if (ctx.disposed) {
|
|
71
|
-
return;
|
|
72
|
-
}
|
|
73
|
-
|
|
74
|
-
// Timer.
|
|
75
|
-
if (trigger.schedule) {
|
|
76
|
-
this._createTimer(ctx, space, def, trigger);
|
|
77
|
-
}
|
|
78
|
-
|
|
79
|
-
// Subscription.
|
|
80
|
-
for (const triggerSubscription of trigger.subscriptions ?? []) {
|
|
81
|
-
this._createSubscription(ctx, space, def, triggerSubscription);
|
|
82
|
-
}
|
|
83
|
-
}
|
|
52
|
+
async stop() {
|
|
53
|
+
await this._ctx.dispose();
|
|
54
|
+
await this.functions.close();
|
|
55
|
+
await this.triggers.close();
|
|
84
56
|
}
|
|
85
57
|
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
this._mounts.delete(key);
|
|
91
|
-
await ctx.dispose();
|
|
92
|
-
}
|
|
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);
|
|
93
62
|
}
|
|
94
63
|
|
|
95
|
-
private
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
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);
|
|
100
71
|
});
|
|
101
|
-
|
|
102
|
-
invariant(trigger.schedule);
|
|
103
|
-
let last = 0;
|
|
104
|
-
let run = 0;
|
|
105
|
-
// https://www.npmjs.com/package/cron#constructor
|
|
106
|
-
const job = CronJob.from({
|
|
107
|
-
cronTime: trigger.schedule,
|
|
108
|
-
runOnInit: false,
|
|
109
|
-
onTick: () => {
|
|
110
|
-
// TODO(burdon): Check greater than 30s (use cron-parser).
|
|
111
|
-
const now = Date.now();
|
|
112
|
-
const delta = last ? now - last : 0;
|
|
113
|
-
last = now;
|
|
114
|
-
|
|
115
|
-
run++;
|
|
116
|
-
log.info('tick', { space: space.key.truncate(), count: run, delta });
|
|
117
|
-
task.schedule();
|
|
118
|
-
},
|
|
119
|
-
});
|
|
120
|
-
|
|
121
|
-
job.start();
|
|
122
|
-
ctx.onDispose(() => job.stop());
|
|
72
|
+
await Promise.all(mountTasks).catch(log.catch);
|
|
123
73
|
}
|
|
124
74
|
|
|
125
|
-
private
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
objects: Array.from(objectIds),
|
|
132
|
-
});
|
|
133
|
-
});
|
|
75
|
+
private async activate(space: Space, functions: FunctionDef[], trigger: FunctionTrigger) {
|
|
76
|
+
const definition = functions.find((def) => def.uri === trigger.function);
|
|
77
|
+
if (!definition) {
|
|
78
|
+
log.info('function is not found for trigger', { trigger });
|
|
79
|
+
return;
|
|
80
|
+
}
|
|
134
81
|
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
const subscription = createSubscription(({ added, updated }) => {
|
|
139
|
-
log.info('updated', { added: added.length, updated: updated.length });
|
|
140
|
-
for (const object of added) {
|
|
141
|
-
objectIds.add(object.id);
|
|
142
|
-
}
|
|
143
|
-
for (const object of updated) {
|
|
144
|
-
objectIds.add(object.id);
|
|
145
|
-
}
|
|
82
|
+
await this.triggers.activate(space, trigger, async (args) => {
|
|
83
|
+
const mutex = this._functionUriToCallMutex.get(definition.uri) ?? new Mutex();
|
|
84
|
+
this._functionUriToCallMutex.set(definition.uri, mutex);
|
|
146
85
|
|
|
147
|
-
|
|
86
|
+
log.info('function triggered, waiting for mutex', { uri: definition.uri });
|
|
87
|
+
return mutex.executeSynchronized(() => {
|
|
88
|
+
log.info('mutex acquired', { uri: definition.uri });
|
|
89
|
+
return this._execFunction(definition, trigger, {
|
|
90
|
+
meta: trigger.meta ?? {},
|
|
91
|
+
data: { ...args, spaceKey: space.key },
|
|
92
|
+
});
|
|
93
|
+
});
|
|
148
94
|
});
|
|
149
|
-
subscriptions.push(() => subscription.unsubscribe());
|
|
150
|
-
|
|
151
|
-
// TODO(burdon): Create queue. Only allow one invocation per trigger at a time?
|
|
152
|
-
// TODO(burdon): Disable trigger if keeps failing.
|
|
153
|
-
const { type, props, deep, delay } = triggerSubscription;
|
|
154
|
-
const update = ({ objects }: Query) => {
|
|
155
|
-
subscription.update(objects);
|
|
156
|
-
|
|
157
|
-
// TODO(burdon): Hack to monitor changes to Document's text object.
|
|
158
|
-
if (deep) {
|
|
159
|
-
log.info('update', { type, deep, objects: objects.length });
|
|
160
|
-
for (const object of objects) {
|
|
161
|
-
const content = object.content;
|
|
162
|
-
if (content instanceof TextV0Type) {
|
|
163
|
-
subscriptions.push(
|
|
164
|
-
getAutomergeObjectCore(content).updates.on(debounce(() => subscription.update([object]), 1_000)),
|
|
165
|
-
);
|
|
166
|
-
}
|
|
167
|
-
}
|
|
168
|
-
}
|
|
169
|
-
};
|
|
170
|
-
|
|
171
|
-
// TODO(burdon): [Bug]: all callbacks are fired on the first mutation.
|
|
172
|
-
// TODO(burdon): [Bug]: not updated when document is deleted (either top or hierarchically).
|
|
173
|
-
const query = space.db.query(Filter.typename(type, props));
|
|
174
|
-
subscriptions.push(query.subscribe(delay ? debounce(update, delay * 1_000) : update));
|
|
175
95
|
|
|
176
|
-
|
|
177
|
-
subscriptions.forEach((unsubscribe) => unsubscribe());
|
|
178
|
-
});
|
|
96
|
+
log('activated trigger', { space: space.key, trigger });
|
|
179
97
|
}
|
|
180
98
|
|
|
181
|
-
private async _execFunction
|
|
99
|
+
private async _execFunction<TData, TMeta>(
|
|
100
|
+
def: FunctionDef,
|
|
101
|
+
trigger: FunctionTrigger,
|
|
102
|
+
{ data, meta }: { data: TData; meta?: TMeta },
|
|
103
|
+
): Promise<number> {
|
|
104
|
+
let status = 0;
|
|
182
105
|
try {
|
|
183
|
-
|
|
106
|
+
// TODO(burdon): Pass in Space key (common context)?
|
|
107
|
+
const payload = Object.assign({}, meta && ({ meta } satisfies FunctionEventMeta<TMeta>), data);
|
|
108
|
+
|
|
184
109
|
const { endpoint, callback } = this._options;
|
|
185
|
-
let status = 0;
|
|
186
110
|
if (endpoint) {
|
|
187
111
|
// TODO(burdon): Move out of scheduler (generalize as callback).
|
|
188
|
-
const
|
|
112
|
+
const url = path.join(endpoint, def.route);
|
|
113
|
+
log.info('exec', { function: def.uri, url, triggerType: trigger.spec.type });
|
|
114
|
+
const response = await fetch(url, {
|
|
189
115
|
method: 'POST',
|
|
190
116
|
headers: {
|
|
191
117
|
'Content-Type': 'application/json',
|
|
192
118
|
},
|
|
193
|
-
body: JSON.stringify(
|
|
119
|
+
body: JSON.stringify(payload),
|
|
194
120
|
});
|
|
195
121
|
|
|
196
122
|
status = response.status;
|
|
197
123
|
} else if (callback) {
|
|
198
|
-
|
|
124
|
+
log.info('exec', { function: def.uri });
|
|
125
|
+
status = (await callback(payload)) ?? 200;
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
// Check errors.
|
|
129
|
+
if (status && status >= 400) {
|
|
130
|
+
throw new Error(`Response: ${status}`);
|
|
199
131
|
}
|
|
200
132
|
|
|
201
133
|
// const result = await response.json();
|
|
202
|
-
log('
|
|
134
|
+
log.info('done', { function: def.uri, status });
|
|
203
135
|
} catch (err: any) {
|
|
204
|
-
log.error('error', { function: def.
|
|
136
|
+
log.error('error', { function: def.uri, error: err.message });
|
|
137
|
+
status = 500;
|
|
205
138
|
}
|
|
139
|
+
|
|
140
|
+
return status;
|
|
206
141
|
}
|
|
207
142
|
}
|
|
143
|
+
|
|
144
|
+
const createContext = () => new Context({ name: 'FunctionScheduler' });
|
|
@@ -0,0 +1,100 @@
|
|
|
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
|
+
enabled: true,
|
|
48
|
+
meta: triggerMeta,
|
|
49
|
+
spec: {
|
|
50
|
+
type: 'subscription',
|
|
51
|
+
filter: [{ type: TestType.typename }],
|
|
52
|
+
},
|
|
53
|
+
}),
|
|
54
|
+
);
|
|
55
|
+
|
|
56
|
+
const called = new Trigger<any>();
|
|
57
|
+
setTestCallHandler(async (args) => {
|
|
58
|
+
called.wake(args.event.data);
|
|
59
|
+
return args.response.status(200);
|
|
60
|
+
});
|
|
61
|
+
|
|
62
|
+
await waitTriggersReplicated(space, scheduler);
|
|
63
|
+
const addedObject = space.db.add(create(TestType, { title: '42' }));
|
|
64
|
+
|
|
65
|
+
const callArgs = await called.wait();
|
|
66
|
+
expect(callArgs.meta).to.deep.eq(triggerMeta);
|
|
67
|
+
expect(callArgs.objects).to.deep.eq([addedObject.id]);
|
|
68
|
+
expect(callArgs.spaceKey).to.eq(space.key.toHex());
|
|
69
|
+
});
|
|
70
|
+
|
|
71
|
+
const waitTriggersReplicated = async (space: Space, scheduler: Scheduler) => {
|
|
72
|
+
await waitForCondition({ condition: () => scheduler.triggers.getActiveTriggers(space).length > 0 });
|
|
73
|
+
};
|
|
74
|
+
|
|
75
|
+
// TODO(burdon): Factor out utils to builder pattern.
|
|
76
|
+
|
|
77
|
+
const startScheduler = async (client: Client, devServer: DevServer) => {
|
|
78
|
+
const functionRegistry = new FunctionRegistry(client);
|
|
79
|
+
const triggerRegistry = new TriggerRegistry(client);
|
|
80
|
+
const scheduler = new Scheduler(functionRegistry, triggerRegistry, { endpoint: devServer.endpoint });
|
|
81
|
+
await scheduler.start();
|
|
82
|
+
testBuilder.ctx.onDispose(() => scheduler.stop());
|
|
83
|
+
return scheduler;
|
|
84
|
+
};
|
|
85
|
+
|
|
86
|
+
const startDevServer = async (client: Client) => {
|
|
87
|
+
const functionRegistry = new FunctionRegistry(client);
|
|
88
|
+
const server = new DevServer(client, functionRegistry, {
|
|
89
|
+
baseDir: path.join(__dirname, '../testing'),
|
|
90
|
+
});
|
|
91
|
+
await server.start();
|
|
92
|
+
testBuilder.ctx.onDispose(() => server.stop());
|
|
93
|
+
return server;
|
|
94
|
+
};
|
|
95
|
+
|
|
96
|
+
const inviteMember = async (host: Space, guest: Client) => {
|
|
97
|
+
const [{ invitation: hostInvitation }] = await Promise.all(performInvitation({ host, guest: guest.spaces }));
|
|
98
|
+
expect(hostInvitation?.state).to.eq(Invitation.State.SUCCESS);
|
|
99
|
+
};
|
|
100
|
+
});
|
|
@@ -0,0 +1,43 @@
|
|
|
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): Extend/wrap 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((client) => client.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(FunctionDef, FunctionTrigger, TestType);
|
|
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);
|
|
38
|
+
const plugin = new FunctionsPlugin();
|
|
39
|
+
await plugin.initialize({ client, clientServices: client.services });
|
|
40
|
+
await plugin.open();
|
|
41
|
+
testBuilder.ctx.onDispose(() => plugin.close());
|
|
42
|
+
return client;
|
|
43
|
+
};
|
|
@@ -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,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
|
+
};
|