@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
|
@@ -0,0 +1,91 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2024 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
import WebSocket from 'ws';
|
|
6
|
+
|
|
7
|
+
import { sleep, Trigger } from '@dxos/async';
|
|
8
|
+
import { type Context } from '@dxos/context';
|
|
9
|
+
import { log } from '@dxos/log';
|
|
10
|
+
|
|
11
|
+
import { type WebsocketTrigger } from '../../types';
|
|
12
|
+
import { type TriggerCallback, type TriggerContext, type TriggerFactory } from '../trigger-registry';
|
|
13
|
+
|
|
14
|
+
interface WebsocketTriggerOptions {
|
|
15
|
+
retryDelay: number;
|
|
16
|
+
maxAttempts: number;
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
/**
|
|
20
|
+
* Websocket.
|
|
21
|
+
* NOTE: The port must be unique, so the same hook cannot be used for multiple spaces.
|
|
22
|
+
*/
|
|
23
|
+
export const createWebsocketTrigger: TriggerFactory<WebsocketTrigger, WebsocketTriggerOptions> = async (
|
|
24
|
+
ctx: Context,
|
|
25
|
+
triggerCtx: TriggerContext,
|
|
26
|
+
spec: WebsocketTrigger,
|
|
27
|
+
callback: TriggerCallback,
|
|
28
|
+
options: WebsocketTriggerOptions = { retryDelay: 2, maxAttempts: 5 },
|
|
29
|
+
) => {
|
|
30
|
+
const { url, init } = spec;
|
|
31
|
+
|
|
32
|
+
let ws: WebSocket;
|
|
33
|
+
for (let attempt = 1; attempt <= options.maxAttempts; attempt++) {
|
|
34
|
+
const open = new Trigger<boolean>();
|
|
35
|
+
|
|
36
|
+
ws = new WebSocket(url);
|
|
37
|
+
Object.assign(ws, {
|
|
38
|
+
onopen: () => {
|
|
39
|
+
log.info('opened', { url });
|
|
40
|
+
if (spec.init) {
|
|
41
|
+
ws.send(new TextEncoder().encode(JSON.stringify(init)));
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
open.wake(true);
|
|
45
|
+
},
|
|
46
|
+
|
|
47
|
+
onclose: (event) => {
|
|
48
|
+
log.info('closed', { url, code: event.code });
|
|
49
|
+
// Reconnect if server closes (e.g., CF restart).
|
|
50
|
+
// https://developer.mozilla.org/en-US/docs/Web/API/CloseEvent/code
|
|
51
|
+
if (event.code === 1006) {
|
|
52
|
+
setTimeout(async () => {
|
|
53
|
+
log.info(`reconnecting in ${options.retryDelay}s...`, { url });
|
|
54
|
+
await createWebsocketTrigger(ctx, triggerCtx, spec, callback, options);
|
|
55
|
+
}, options.retryDelay * 1_000);
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
open.wake(false);
|
|
59
|
+
},
|
|
60
|
+
|
|
61
|
+
onerror: (event) => {
|
|
62
|
+
log.catch(event.error, { url });
|
|
63
|
+
},
|
|
64
|
+
|
|
65
|
+
onmessage: async (event) => {
|
|
66
|
+
try {
|
|
67
|
+
log.info('message');
|
|
68
|
+
const data = JSON.parse(new TextDecoder().decode(event.data as Uint8Array));
|
|
69
|
+
await callback({ data });
|
|
70
|
+
} catch (err) {
|
|
71
|
+
log.catch(err, { url });
|
|
72
|
+
}
|
|
73
|
+
},
|
|
74
|
+
} satisfies Partial<WebSocket>);
|
|
75
|
+
|
|
76
|
+
const isOpen = await open.wait();
|
|
77
|
+
if (isOpen) {
|
|
78
|
+
break;
|
|
79
|
+
} else {
|
|
80
|
+
const wait = Math.pow(attempt, 2) * options.retryDelay;
|
|
81
|
+
if (attempt < options.maxAttempts) {
|
|
82
|
+
log.warn(`failed to connect; trying again in ${wait}s`, { attempt });
|
|
83
|
+
await sleep(wait * 1_000);
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
ctx.onDispose(() => {
|
|
89
|
+
ws?.close();
|
|
90
|
+
});
|
|
91
|
+
};
|
package/src/types.ts
CHANGED
|
@@ -2,28 +2,19 @@
|
|
|
2
2
|
// Copyright 2023 DXOS.org
|
|
3
3
|
//
|
|
4
4
|
|
|
5
|
-
import
|
|
5
|
+
import { RawObject, S, TypedObject } from '@dxos/echo-schema';
|
|
6
6
|
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
// Assigned port.
|
|
15
|
-
port: S.optional(S.number),
|
|
16
|
-
}),
|
|
17
|
-
);
|
|
18
|
-
|
|
19
|
-
const WebsocketTriggerSchema = S.struct({
|
|
20
|
-
url: S.string,
|
|
21
|
-
init: S.optional(S.record(S.string, S.any)),
|
|
22
|
-
});
|
|
7
|
+
/**
|
|
8
|
+
* Type discriminator for TriggerSpec.
|
|
9
|
+
* Every spec has a type field of type FunctionTriggerType that we can use to understand which
|
|
10
|
+
* type we're working with.
|
|
11
|
+
* https://www.typescriptlang.org/docs/handbook/2/narrowing.html#discriminated-unions
|
|
12
|
+
*/
|
|
13
|
+
export type FunctionTriggerType = 'subscription' | 'timer' | 'webhook' | 'websocket';
|
|
23
14
|
|
|
24
15
|
const SubscriptionTriggerSchema = S.struct({
|
|
25
|
-
|
|
26
|
-
// TODO(burdon): Define query DSL.
|
|
16
|
+
type: S.literal('subscription'),
|
|
17
|
+
// TODO(burdon): Define query DSL (from ECHO).
|
|
27
18
|
filter: S.array(
|
|
28
19
|
S.struct({
|
|
29
20
|
type: S.string,
|
|
@@ -40,46 +31,73 @@ const SubscriptionTriggerSchema = S.struct({
|
|
|
40
31
|
),
|
|
41
32
|
});
|
|
42
33
|
|
|
43
|
-
|
|
44
|
-
function: S.string.pipe(S.description('Function ID/URI.')),
|
|
45
|
-
|
|
46
|
-
// Context passed to function.
|
|
47
|
-
context: S.optional(S.record(S.string, S.any)),
|
|
34
|
+
export type SubscriptionTrigger = S.Schema.Type<typeof SubscriptionTriggerSchema>;
|
|
48
35
|
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
websocket: S.optional(WebsocketTriggerSchema),
|
|
53
|
-
subscription: S.optional(SubscriptionTriggerSchema),
|
|
36
|
+
const TimerTriggerSchema = S.struct({
|
|
37
|
+
type: S.literal('timer'),
|
|
38
|
+
cron: S.string,
|
|
54
39
|
});
|
|
55
40
|
|
|
56
41
|
export type TimerTrigger = S.Schema.Type<typeof TimerTriggerSchema>;
|
|
42
|
+
|
|
43
|
+
const WebhookTriggerSchema = S.mutable(
|
|
44
|
+
S.struct({
|
|
45
|
+
type: S.literal('webhook'),
|
|
46
|
+
method: S.string,
|
|
47
|
+
// Assigned port.
|
|
48
|
+
port: S.optional(S.number),
|
|
49
|
+
}),
|
|
50
|
+
);
|
|
51
|
+
|
|
57
52
|
export type WebhookTrigger = S.Schema.Type<typeof WebhookTriggerSchema>;
|
|
53
|
+
|
|
54
|
+
const WebsocketTriggerSchema = S.struct({
|
|
55
|
+
type: S.literal('websocket'),
|
|
56
|
+
url: S.string,
|
|
57
|
+
init: S.optional(S.record(S.string, S.any)),
|
|
58
|
+
});
|
|
59
|
+
|
|
58
60
|
export type WebsocketTrigger = S.Schema.Type<typeof WebsocketTriggerSchema>;
|
|
59
|
-
|
|
60
|
-
|
|
61
|
+
|
|
62
|
+
const TriggerSpecSchema = S.union(
|
|
63
|
+
TimerTriggerSchema,
|
|
64
|
+
WebhookTriggerSchema,
|
|
65
|
+
WebsocketTriggerSchema,
|
|
66
|
+
SubscriptionTriggerSchema,
|
|
67
|
+
);
|
|
68
|
+
|
|
69
|
+
export type TriggerSpec = TimerTrigger | WebhookTrigger | WebsocketTrigger | SubscriptionTrigger;
|
|
61
70
|
|
|
62
71
|
/**
|
|
63
72
|
* Function definition.
|
|
64
73
|
*/
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
74
|
+
export class FunctionDef extends TypedObject({
|
|
75
|
+
typename: 'dxos.org/type/FunctionDef',
|
|
76
|
+
version: '0.1.0',
|
|
77
|
+
})({
|
|
78
|
+
uri: S.string,
|
|
69
79
|
description: S.optional(S.string),
|
|
70
|
-
|
|
80
|
+
route: S.string,
|
|
71
81
|
// TODO(burdon): NPM/GitHub/Docker/CF URL?
|
|
72
82
|
handler: S.string,
|
|
73
|
-
})
|
|
83
|
+
}) {}
|
|
74
84
|
|
|
75
|
-
export
|
|
85
|
+
export class FunctionTrigger extends TypedObject({
|
|
86
|
+
typename: 'dxos.org/type/FunctionTrigger',
|
|
87
|
+
version: '0.1.0',
|
|
88
|
+
})({
|
|
89
|
+
function: S.string.pipe(S.description('Function URI.')),
|
|
90
|
+
// Context is merged into the event data passed to the function.
|
|
91
|
+
meta: S.optional(S.object),
|
|
92
|
+
spec: TriggerSpecSchema,
|
|
93
|
+
}) {}
|
|
76
94
|
|
|
77
95
|
/**
|
|
78
96
|
* Function manifest file.
|
|
79
97
|
*/
|
|
80
98
|
export const FunctionManifestSchema = S.struct({
|
|
81
|
-
functions: S.mutable(S.array(
|
|
82
|
-
triggers: S.optional(S.mutable(S.array(
|
|
99
|
+
functions: S.optional(S.mutable(S.array(RawObject(FunctionDef)))),
|
|
100
|
+
triggers: S.optional(S.mutable(S.array(RawObject(FunctionTrigger)))),
|
|
83
101
|
});
|
|
84
102
|
|
|
85
103
|
export type FunctionManifest = S.Schema.Type<typeof FunctionManifestSchema>;
|
package/src/util.test.ts
ADDED
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2024 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
import { expect } from 'chai';
|
|
6
|
+
|
|
7
|
+
import { describe, test } from '@dxos/test';
|
|
8
|
+
|
|
9
|
+
import { diff, intersection } from './util';
|
|
10
|
+
|
|
11
|
+
describe('diff', () => {
|
|
12
|
+
test('returns the difference between two sets', () => {
|
|
13
|
+
{
|
|
14
|
+
const { added, updated, removed } = diff<number>([], [], (a, b) => a === b);
|
|
15
|
+
expect(added).to.deep.eq([]);
|
|
16
|
+
expect(updated).to.deep.eq([]);
|
|
17
|
+
expect(removed).to.deep.eq([]);
|
|
18
|
+
}
|
|
19
|
+
{
|
|
20
|
+
const previous = [1, 2, 3];
|
|
21
|
+
const next = [2, 3, 4];
|
|
22
|
+
const { added, updated, removed } = diff(previous, next, (a, b) => a === b);
|
|
23
|
+
expect(added).to.deep.eq([4]);
|
|
24
|
+
expect(updated).to.deep.eq([2, 3]);
|
|
25
|
+
expect(removed).to.deep.eq([1]);
|
|
26
|
+
}
|
|
27
|
+
{
|
|
28
|
+
const previous = [{ x: 1 }, { x: 2 }, { x: 3 }];
|
|
29
|
+
const next = [{ x: 2 }, { x: 3 }, { x: 4 }];
|
|
30
|
+
const { added, updated, removed } = diff(previous, next, (a, b) => a.x === b.x);
|
|
31
|
+
expect(added).to.deep.eq([{ x: 4 }]);
|
|
32
|
+
expect(updated).to.deep.eq([{ x: 2 }, { x: 3 }]);
|
|
33
|
+
expect(removed).to.deep.eq([{ x: 1 }]);
|
|
34
|
+
}
|
|
35
|
+
});
|
|
36
|
+
|
|
37
|
+
test('intersection', () => {
|
|
38
|
+
expect(intersection([1, 2, 3], [2, 3, 4], (a, b) => a === b)).to.deep.eq([2, 3]);
|
|
39
|
+
expect(
|
|
40
|
+
intersection([{ x: 1 }, { x: 2 }, { x: 3 }], [{ x: 2 }, { x: 3 }, { x: 4 }], (a, b) => a.x === b.x),
|
|
41
|
+
).to.deep.eq([{ x: 2 }, { x: 3 }]);
|
|
42
|
+
});
|
|
43
|
+
});
|
package/src/util.ts
ADDED
|
@@ -0,0 +1,48 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2024 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
export type Comparator<A, B = A> = (a: A, b: B) => boolean;
|
|
6
|
+
|
|
7
|
+
export type DiffResult<A, B = A> = {
|
|
8
|
+
added: B[];
|
|
9
|
+
updated: A[];
|
|
10
|
+
removed: A[];
|
|
11
|
+
};
|
|
12
|
+
|
|
13
|
+
/**
|
|
14
|
+
*
|
|
15
|
+
* @param previous
|
|
16
|
+
* @param next
|
|
17
|
+
* @param comparator
|
|
18
|
+
*/
|
|
19
|
+
// TODO(burdon): Factor out.
|
|
20
|
+
export const diff = <A, B = A>(
|
|
21
|
+
previous: readonly A[],
|
|
22
|
+
next: readonly B[],
|
|
23
|
+
comparator: Comparator<A, B>,
|
|
24
|
+
): DiffResult<A, B> => {
|
|
25
|
+
const remaining = [...previous];
|
|
26
|
+
const result: DiffResult<A, B> = {
|
|
27
|
+
added: [],
|
|
28
|
+
updated: [],
|
|
29
|
+
removed: remaining,
|
|
30
|
+
};
|
|
31
|
+
|
|
32
|
+
// TODO(burdon): Mark and sweep.
|
|
33
|
+
for (const object of next) {
|
|
34
|
+
const index = remaining.findIndex((item) => comparator(item, object));
|
|
35
|
+
if (index === -1) {
|
|
36
|
+
result.added.push(object);
|
|
37
|
+
} else {
|
|
38
|
+
result.updated.push(remaining[index]);
|
|
39
|
+
remaining.splice(index, 1);
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
return result;
|
|
44
|
+
};
|
|
45
|
+
|
|
46
|
+
// TODO(burdon): Factor out.
|
|
47
|
+
export const intersection = <A, B = A>(a: A[], b: B[], comparator: Comparator<A, B>): A[] =>
|
|
48
|
+
a.filter((a) => b.find((b) => comparator(a, b)) !== undefined);
|