@dxos/functions 0.8.3 → 0.8.4-main.1da679c
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/bundler/index.mjs +73 -44
- package/dist/lib/browser/bundler/index.mjs.map +3 -3
- package/dist/lib/browser/chunk-D2XO7XXY.mjs +611 -0
- package/dist/lib/browser/chunk-D2XO7XXY.mjs.map +7 -0
- package/dist/lib/browser/chunk-J5LGTIGS.mjs +10 -0
- package/dist/lib/browser/chunk-J5LGTIGS.mjs.map +7 -0
- package/dist/lib/browser/edge/index.mjs +24 -10
- package/dist/lib/browser/edge/index.mjs.map +3 -3
- package/dist/lib/browser/index.mjs +981 -137
- package/dist/lib/browser/index.mjs.map +4 -4
- package/dist/lib/browser/meta.json +1 -1
- package/dist/lib/browser/testing/index.mjs +110 -9
- package/dist/lib/browser/testing/index.mjs.map +4 -4
- package/dist/lib/node-esm/bundler/index.mjs +72 -44
- package/dist/lib/node-esm/bundler/index.mjs.map +3 -3
- package/dist/lib/node-esm/chunk-HSLMI22Q.mjs +11 -0
- package/dist/lib/node-esm/chunk-HSLMI22Q.mjs.map +7 -0
- package/dist/lib/node-esm/chunk-Z4CJ62WS.mjs +613 -0
- package/dist/lib/node-esm/chunk-Z4CJ62WS.mjs.map +7 -0
- package/dist/lib/node-esm/edge/index.mjs +23 -10
- package/dist/lib/node-esm/edge/index.mjs.map +3 -3
- package/dist/lib/node-esm/index.mjs +981 -137
- package/dist/lib/node-esm/index.mjs.map +4 -4
- package/dist/lib/node-esm/meta.json +1 -1
- package/dist/lib/node-esm/testing/index.mjs +110 -9
- package/dist/lib/node-esm/testing/index.mjs.map +4 -4
- package/dist/types/src/bundler/bundler.d.ts +12 -14
- package/dist/types/src/bundler/bundler.d.ts.map +1 -1
- package/dist/types/src/edge/functions.d.ts +4 -3
- package/dist/types/src/edge/functions.d.ts.map +1 -1
- package/dist/types/src/errors.d.ts +137 -0
- package/dist/types/src/errors.d.ts.map +1 -0
- package/dist/types/src/examples/fib.d.ts +7 -0
- package/dist/types/src/examples/fib.d.ts.map +1 -0
- package/dist/types/src/examples/index.d.ts +4 -0
- package/dist/types/src/examples/index.d.ts.map +1 -0
- package/dist/types/src/examples/reply.d.ts +3 -0
- package/dist/types/src/examples/reply.d.ts.map +1 -0
- package/dist/types/src/examples/sleep.d.ts +5 -0
- package/dist/types/src/examples/sleep.d.ts.map +1 -0
- package/dist/types/src/executor/executor.d.ts +4 -1
- package/dist/types/src/executor/executor.d.ts.map +1 -1
- package/dist/types/src/handler.d.ts +46 -16
- package/dist/types/src/handler.d.ts.map +1 -1
- package/dist/types/src/index.d.ts +3 -0
- package/dist/types/src/index.d.ts.map +1 -1
- package/dist/types/src/schema.d.ts +8 -3
- package/dist/types/src/schema.d.ts.map +1 -1
- package/dist/types/src/services/credentials.d.ts +18 -4
- package/dist/types/src/services/credentials.d.ts.map +1 -1
- package/dist/types/src/services/database.d.ts +55 -3
- package/dist/types/src/services/database.d.ts.map +1 -1
- package/dist/types/src/services/event-logger.d.ts +65 -30
- package/dist/types/src/services/event-logger.d.ts.map +1 -1
- package/dist/types/src/services/index.d.ts +2 -2
- package/dist/types/src/services/index.d.ts.map +1 -1
- package/dist/types/src/services/local-function-execution.d.ts +25 -0
- package/dist/types/src/services/local-function-execution.d.ts.map +1 -0
- package/dist/types/src/services/queues.d.ts +33 -6
- package/dist/types/src/services/queues.d.ts.map +1 -1
- package/dist/types/src/services/remote-function-execution-service.d.ts +15 -0
- package/dist/types/src/services/remote-function-execution-service.d.ts.map +1 -0
- package/dist/types/src/services/service-container.d.ts +29 -17
- package/dist/types/src/services/service-container.d.ts.map +1 -1
- package/dist/types/src/services/service-registry.d.ts +29 -0
- package/dist/types/src/services/service-registry.d.ts.map +1 -0
- package/dist/types/src/services/service-registry.test.d.ts +2 -0
- package/dist/types/src/services/service-registry.test.d.ts.map +1 -0
- package/dist/types/src/services/tracing.d.ts +46 -4
- package/dist/types/src/services/tracing.d.ts.map +1 -1
- package/dist/types/src/testing/index.d.ts +1 -0
- package/dist/types/src/testing/index.d.ts.map +1 -1
- package/dist/types/src/testing/layer.d.ts +15 -0
- package/dist/types/src/testing/layer.d.ts.map +1 -0
- package/dist/types/src/testing/logger.d.ts +3 -3
- package/dist/types/src/testing/logger.d.ts.map +1 -1
- package/dist/types/src/testing/persist-database.test.d.ts +2 -0
- package/dist/types/src/testing/persist-database.test.d.ts.map +1 -0
- package/dist/types/src/testing/services.d.ts +55 -9
- package/dist/types/src/testing/services.d.ts.map +1 -1
- package/dist/types/src/trace.d.ts +20 -22
- package/dist/types/src/trace.d.ts.map +1 -1
- package/dist/types/src/translations.d.ts +9 -9
- package/dist/types/src/translations.d.ts.map +1 -1
- package/dist/types/src/triggers/index.d.ts +4 -0
- package/dist/types/src/triggers/index.d.ts.map +1 -0
- package/dist/types/src/triggers/input-builder.d.ts +3 -0
- package/dist/types/src/triggers/input-builder.d.ts.map +1 -0
- package/dist/types/src/triggers/invocation-tracer.d.ts +35 -0
- package/dist/types/src/triggers/invocation-tracer.d.ts.map +1 -0
- package/dist/types/src/triggers/trigger-dispatcher.d.ts +75 -0
- package/dist/types/src/triggers/trigger-dispatcher.d.ts.map +1 -0
- package/dist/types/src/triggers/trigger-dispatcher.test.d.ts +2 -0
- package/dist/types/src/triggers/trigger-dispatcher.test.d.ts.map +1 -0
- package/dist/types/src/triggers/trigger-state-store.d.ts +27 -0
- package/dist/types/src/triggers/trigger-state-store.d.ts.map +1 -0
- package/dist/types/src/types.d.ts +55 -245
- package/dist/types/src/types.d.ts.map +1 -1
- package/dist/types/src/url.d.ts +10 -6
- package/dist/types/src/url.d.ts.map +1 -1
- package/dist/types/tsconfig.tsbuildinfo +1 -1
- package/package.json +35 -25
- package/src/bundler/bundler.test.ts +9 -10
- package/src/bundler/bundler.ts +56 -35
- package/src/edge/functions.ts +9 -6
- package/src/errors.ts +21 -0
- package/src/examples/fib.ts +30 -0
- package/src/examples/index.ts +7 -0
- package/src/examples/reply.ts +18 -0
- package/src/examples/sleep.ts +22 -0
- package/src/executor/executor.ts +22 -15
- package/src/handler.ts +117 -27
- package/src/index.ts +3 -2
- package/src/schema.ts +11 -0
- package/src/services/credentials.ts +87 -5
- package/src/services/database.ts +146 -3
- package/src/services/event-logger.ts +68 -37
- package/src/services/index.ts +2 -2
- package/src/services/local-function-execution.ts +127 -0
- package/src/services/queues.ts +56 -11
- package/src/services/remote-function-execution-service.ts +46 -0
- package/src/services/service-container.ts +47 -42
- package/src/services/service-registry.test.ts +42 -0
- package/src/services/service-registry.ts +59 -0
- package/src/services/tracing.ts +118 -5
- package/src/testing/index.ts +1 -0
- package/src/testing/layer.ts +111 -0
- package/src/testing/logger.ts +4 -4
- package/src/testing/persist-database.test.ts +87 -0
- package/src/testing/services.ts +97 -14
- package/src/trace.ts +17 -19
- package/src/translations.ts +4 -4
- package/src/triggers/index.ts +7 -0
- package/src/triggers/input-builder.ts +35 -0
- package/src/triggers/invocation-tracer.ts +99 -0
- package/src/triggers/trigger-dispatcher.test.ts +652 -0
- package/src/triggers/trigger-dispatcher.ts +516 -0
- package/src/triggers/trigger-state-store.ts +60 -0
- package/src/types.ts +39 -36
- package/src/url.ts +13 -10
- package/dist/lib/browser/chunk-WEFZUEL2.mjs +0 -300
- package/dist/lib/browser/chunk-WEFZUEL2.mjs.map +0 -7
- package/dist/lib/node/bundler/index.cjs +0 -260
- package/dist/lib/node/bundler/index.cjs.map +0 -7
- package/dist/lib/node/chunk-IJAE7FZK.cjs +0 -320
- package/dist/lib/node/chunk-IJAE7FZK.cjs.map +0 -7
- package/dist/lib/node/edge/index.cjs +0 -94
- package/dist/lib/node/edge/index.cjs.map +0 -7
- package/dist/lib/node/index.cjs +0 -522
- package/dist/lib/node/index.cjs.map +0 -7
- package/dist/lib/node/meta.json +0 -1
- package/dist/lib/node/testing/index.cjs +0 -43
- package/dist/lib/node/testing/index.cjs.map +0 -7
- package/dist/lib/node-esm/chunk-LIYPMWNQ.mjs +0 -302
- package/dist/lib/node-esm/chunk-LIYPMWNQ.mjs.map +0 -7
- package/dist/types/src/services/ai.d.ts +0 -12
- package/dist/types/src/services/ai.d.ts.map +0 -1
- package/dist/types/src/services/function-call-service.d.ts +0 -16
- package/dist/types/src/services/function-call-service.d.ts.map +0 -1
- package/src/services/ai.ts +0 -32
- package/src/services/function-call-service.ts +0 -64
package/package.json
CHANGED
|
@@ -1,30 +1,34 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@dxos/functions",
|
|
3
|
-
"version": "0.8.
|
|
3
|
+
"version": "0.8.4-main.1da679c",
|
|
4
4
|
"description": "Functions API and runtime.",
|
|
5
5
|
"homepage": "https://dxos.org",
|
|
6
6
|
"bugs": "https://github.com/dxos/dxos/issues",
|
|
7
7
|
"license": "MIT",
|
|
8
8
|
"author": "info@dxos.org",
|
|
9
|
-
"sideEffects":
|
|
9
|
+
"sideEffects": false,
|
|
10
10
|
"type": "module",
|
|
11
11
|
"exports": {
|
|
12
12
|
".": {
|
|
13
|
+
"source": "./src/index.ts",
|
|
13
14
|
"types": "./dist/types/src/index.d.ts",
|
|
14
15
|
"browser": "./dist/lib/browser/index.mjs",
|
|
15
16
|
"node": "./dist/lib/node-esm/index.mjs"
|
|
16
17
|
},
|
|
17
18
|
"./bundler": {
|
|
19
|
+
"source": "./src/bundler/index.ts",
|
|
18
20
|
"types": "./dist/types/src/bundler/index.d.ts",
|
|
19
21
|
"browser": "./dist/lib/browser/bundler/index.mjs",
|
|
20
22
|
"node": "./dist/lib/node-esm/bundler/index.mjs"
|
|
21
23
|
},
|
|
22
24
|
"./edge": {
|
|
25
|
+
"source": "./src/edge/index.ts",
|
|
23
26
|
"types": "./dist/types/src/edge/index.d.ts",
|
|
24
27
|
"browser": "./dist/lib/browser/edge/index.mjs",
|
|
25
28
|
"node": "./dist/lib/node-esm/edge/index.mjs"
|
|
26
29
|
},
|
|
27
30
|
"./testing": {
|
|
31
|
+
"source": "./src/testing/index.ts",
|
|
28
32
|
"types": "./dist/types/src/testing/index.d.ts",
|
|
29
33
|
"browser": "./dist/lib/browser/testing/index.mjs",
|
|
30
34
|
"node": "./dist/lib/node-esm/testing/index.mjs"
|
|
@@ -47,43 +51,49 @@
|
|
|
47
51
|
"src"
|
|
48
52
|
],
|
|
49
53
|
"dependencies": {
|
|
54
|
+
"@effect/platform": "0.90.2",
|
|
50
55
|
"@preact/signals-core": "^1.9.0",
|
|
51
56
|
"cron": "^3.1.6",
|
|
52
57
|
"cron-schedule": "^5.0.4",
|
|
53
|
-
"effect": "3.
|
|
58
|
+
"effect": "3.17.7",
|
|
54
59
|
"esbuild-wasm": "^0.16.14",
|
|
55
60
|
"express": "^4.19.2",
|
|
56
61
|
"get-port-please": "^3.1.1",
|
|
62
|
+
"i18next": "^24.2.1",
|
|
57
63
|
"iso-did": "^1.6.0",
|
|
58
64
|
"ws": "^8.14.2",
|
|
59
|
-
"@dxos/ai": "0.8.
|
|
60
|
-
"@dxos/
|
|
61
|
-
"@dxos/
|
|
62
|
-
"@dxos/
|
|
63
|
-
"@dxos/
|
|
64
|
-
"@dxos/
|
|
65
|
-
"@dxos/
|
|
66
|
-
"@dxos/echo
|
|
67
|
-
"@dxos/echo-
|
|
68
|
-
"@dxos/
|
|
69
|
-
"@dxos/
|
|
70
|
-
"@dxos/
|
|
71
|
-
"@dxos/
|
|
72
|
-
"@dxos/
|
|
73
|
-
"@dxos/
|
|
74
|
-
"@dxos/
|
|
75
|
-
"@dxos/
|
|
76
|
-
"@dxos/
|
|
65
|
+
"@dxos/ai": "0.8.4-main.1da679c",
|
|
66
|
+
"@dxos/async": "0.8.4-main.1da679c",
|
|
67
|
+
"@dxos/client": "0.8.4-main.1da679c",
|
|
68
|
+
"@dxos/client-protocol": "0.8.4-main.1da679c",
|
|
69
|
+
"@dxos/context": "0.8.4-main.1da679c",
|
|
70
|
+
"@dxos/crypto": "0.8.4-main.1da679c",
|
|
71
|
+
"@dxos/debug": "0.8.4-main.1da679c",
|
|
72
|
+
"@dxos/echo": "0.8.4-main.1da679c",
|
|
73
|
+
"@dxos/echo-db": "0.8.4-main.1da679c",
|
|
74
|
+
"@dxos/echo-pipeline": "0.8.4-main.1da679c",
|
|
75
|
+
"@dxos/echo-protocol": "0.8.4-main.1da679c",
|
|
76
|
+
"@dxos/echo-schema": "0.8.4-main.1da679c",
|
|
77
|
+
"@dxos/edge-client": "0.8.4-main.1da679c",
|
|
78
|
+
"@dxos/effect": "0.8.4-main.1da679c",
|
|
79
|
+
"@dxos/errors": "0.8.4-main.1da679c",
|
|
80
|
+
"@dxos/invariant": "0.8.4-main.1da679c",
|
|
81
|
+
"@dxos/keys": "0.8.4-main.1da679c",
|
|
82
|
+
"@dxos/kv-store": "0.8.4-main.1da679c",
|
|
83
|
+
"@dxos/live-object": "0.8.4-main.1da679c",
|
|
84
|
+
"@dxos/log": "0.8.4-main.1da679c",
|
|
85
|
+
"@dxos/node-std": "0.8.4-main.1da679c",
|
|
86
|
+
"@dxos/protocols": "0.8.4-main.1da679c",
|
|
87
|
+
"@dxos/schema": "0.8.4-main.1da679c",
|
|
88
|
+
"@dxos/util": "0.8.4-main.1da679c"
|
|
77
89
|
},
|
|
78
90
|
"devDependencies": {
|
|
79
91
|
"@types/express": "^4.17.17",
|
|
80
92
|
"@types/ws": "^7.4.0",
|
|
81
|
-
"@dxos/agent": "0.8.
|
|
93
|
+
"@dxos/agent": "0.8.4-main.1da679c"
|
|
82
94
|
},
|
|
83
95
|
"publishConfig": {
|
|
84
96
|
"access": "public"
|
|
85
97
|
},
|
|
86
|
-
"scripts": {
|
|
87
|
-
"gen-schema": "ts-node ./tools/schema.ts"
|
|
88
|
-
}
|
|
98
|
+
"scripts": {}
|
|
89
99
|
}
|
|
@@ -4,7 +4,7 @@
|
|
|
4
4
|
|
|
5
5
|
// @ts-ignore
|
|
6
6
|
import wasmUrl from 'esbuild-wasm/esbuild.wasm?url';
|
|
7
|
-
import { beforeAll, describe, expect, test } from 'vitest';
|
|
7
|
+
import { assert, beforeAll, describe, expect, test } from 'vitest';
|
|
8
8
|
|
|
9
9
|
import { isNode } from '@dxos/util';
|
|
10
10
|
|
|
@@ -20,8 +20,8 @@ describe('Bundler', () => {
|
|
|
20
20
|
test('Basic', async () => {
|
|
21
21
|
const bundler = new Bundler({ platform: 'node', sandboxedModules: [], remoteModules: {} });
|
|
22
22
|
const result = await bundler.bundle({ source: 'const x = 100' }); // TODO(burdon): Test import.
|
|
23
|
-
|
|
24
|
-
expect(result.
|
|
23
|
+
assert(!('error' in result), 'error should not exist');
|
|
24
|
+
expect(result.asset).toBeDefined();
|
|
25
25
|
});
|
|
26
26
|
|
|
27
27
|
test('Import', async () => {
|
|
@@ -33,11 +33,11 @@ describe('Bundler', () => {
|
|
|
33
33
|
const query = Filter.typename('dxos.org/type/Example');
|
|
34
34
|
`,
|
|
35
35
|
});
|
|
36
|
-
|
|
37
|
-
expect(result.
|
|
36
|
+
assert(!('error' in result), 'error should not exist');
|
|
37
|
+
expect(result.asset).toBeDefined();
|
|
38
38
|
});
|
|
39
39
|
|
|
40
|
-
// TODO(dmaretskyi): Flaky on CI
|
|
40
|
+
// TODO(dmaretskyi): Flaky on CI.
|
|
41
41
|
test.skip('HTTPS Import', async () => {
|
|
42
42
|
const bundler = new Bundler({ platform: 'node', sandboxedModules: [], remoteModules: {} });
|
|
43
43
|
const result = await bundler.bundle({
|
|
@@ -46,14 +46,13 @@ describe('Bundler', () => {
|
|
|
46
46
|
invariant(true);
|
|
47
47
|
`,
|
|
48
48
|
});
|
|
49
|
-
|
|
50
|
-
expect(result.
|
|
49
|
+
assert(!('error' in result), 'error should not exist');
|
|
50
|
+
expect(result.asset).toBeDefined();
|
|
51
51
|
});
|
|
52
52
|
|
|
53
53
|
test('Error', async () => {
|
|
54
54
|
const bundler = new Bundler({ platform: 'node', sandboxedModules: [], remoteModules: {} });
|
|
55
55
|
const result = await bundler.bundle({ source: "import missing from './module'; missing();" });
|
|
56
|
-
|
|
57
|
-
expect(result.error).to.exist;
|
|
56
|
+
assert('error' in result, 'error should exist');
|
|
58
57
|
});
|
|
59
58
|
});
|
package/src/bundler/bundler.ts
CHANGED
|
@@ -2,10 +2,12 @@
|
|
|
2
2
|
// Copyright 2023 DXOS.org
|
|
3
3
|
//
|
|
4
4
|
|
|
5
|
-
import {
|
|
6
|
-
import {
|
|
5
|
+
import { FetchHttpClient, HttpClient } from '@effect/platform';
|
|
6
|
+
import { Duration, Effect, Schedule, pipe } from 'effect';
|
|
7
|
+
import { type BuildOptions, type BuildResult, type Loader, type Plugin, build, initialize } from 'esbuild-wasm';
|
|
7
8
|
|
|
8
9
|
import { subtleCrypto } from '@dxos/crypto';
|
|
10
|
+
import { runAndForwardErrors } from '@dxos/effect';
|
|
9
11
|
import { invariant } from '@dxos/invariant';
|
|
10
12
|
import { log } from '@dxos/log';
|
|
11
13
|
|
|
@@ -16,26 +18,26 @@ export type Import = {
|
|
|
16
18
|
};
|
|
17
19
|
|
|
18
20
|
export type BundleOptions = {
|
|
19
|
-
/**
|
|
20
|
-
* Path to the source file on the local file system.
|
|
21
|
-
* If provided, the path will be used instead of the `source` code.
|
|
22
|
-
*/
|
|
23
|
-
path?: string;
|
|
24
|
-
|
|
25
21
|
/**
|
|
26
22
|
* Source code to bundle.
|
|
27
|
-
* Required if `path` is not provided.
|
|
28
23
|
*/
|
|
29
|
-
source
|
|
24
|
+
source: string;
|
|
30
25
|
};
|
|
31
26
|
|
|
32
|
-
export type BundleResult =
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
27
|
+
export type BundleResult =
|
|
28
|
+
| {
|
|
29
|
+
timestamp: number;
|
|
30
|
+
sourceHash: Buffer;
|
|
31
|
+
error: unknown;
|
|
32
|
+
}
|
|
33
|
+
| {
|
|
34
|
+
timestamp: number;
|
|
35
|
+
sourceHash: Buffer;
|
|
36
|
+
imports: Import[];
|
|
37
|
+
entryPoint: string;
|
|
38
|
+
asset: Uint8Array;
|
|
39
|
+
bundle: string;
|
|
40
|
+
};
|
|
39
41
|
|
|
40
42
|
export type BundlerOptions = {
|
|
41
43
|
platform: BuildOptions['platform'];
|
|
@@ -56,16 +58,9 @@ export const initializeBundler = async (options: { wasmUrl: string }) => {
|
|
|
56
58
|
export class Bundler {
|
|
57
59
|
constructor(private readonly _options: BundlerOptions) {}
|
|
58
60
|
|
|
59
|
-
async bundle({
|
|
61
|
+
async bundle({ source }: BundleOptions): Promise<BundleResult> {
|
|
60
62
|
const { sandboxedModules: providedModules, ...options } = this._options;
|
|
61
|
-
|
|
62
|
-
const createResult = async (result?: Partial<BundleResult>) => {
|
|
63
|
-
return {
|
|
64
|
-
timestamp: Date.now(),
|
|
65
|
-
sourceHash: source ? Buffer.from(await subtleCrypto.digest('SHA-256', Buffer.from(source))) : undefined,
|
|
66
|
-
...result,
|
|
67
|
-
};
|
|
68
|
-
};
|
|
63
|
+
const sourceHash = Buffer.from(await subtleCrypto.digest('SHA-256', Buffer.from(source)));
|
|
69
64
|
|
|
70
65
|
if (this._options.platform === 'browser') {
|
|
71
66
|
invariant(initialized, 'Compiler not initialized.');
|
|
@@ -81,7 +76,10 @@ export class Bundler {
|
|
|
81
76
|
conditions: ['workerd', 'browser'],
|
|
82
77
|
metafile: true,
|
|
83
78
|
write: false,
|
|
84
|
-
entryPoints:
|
|
79
|
+
entryPoints: {
|
|
80
|
+
// Gets mapped to `userFunc.js` by esbuild.
|
|
81
|
+
userFunc: 'memory:main.tsx',
|
|
82
|
+
},
|
|
85
83
|
bundle: true,
|
|
86
84
|
format: 'esm',
|
|
87
85
|
plugins: [
|
|
@@ -134,12 +132,17 @@ export class Bundler {
|
|
|
134
132
|
|
|
135
133
|
log('compile complete', result.metafile);
|
|
136
134
|
|
|
137
|
-
|
|
135
|
+
const entryPoint = 'userFunc.js';
|
|
136
|
+
return {
|
|
137
|
+
timestamp: Date.now(),
|
|
138
|
+
sourceHash,
|
|
138
139
|
imports: this.analyzeImports(result),
|
|
140
|
+
entryPoint,
|
|
141
|
+
asset: result.outputFiles![0].contents,
|
|
139
142
|
bundle: result.outputFiles![0].text,
|
|
140
|
-
}
|
|
143
|
+
};
|
|
141
144
|
} catch (err) {
|
|
142
|
-
return
|
|
145
|
+
return { timestamp: Date.now(), sourceHash, error: err };
|
|
143
146
|
}
|
|
144
147
|
}
|
|
145
148
|
|
|
@@ -151,7 +154,6 @@ export class Bundler {
|
|
|
151
154
|
const parsedImports = allMatches(IMPORT_REGEX, result.outputFiles[0].text);
|
|
152
155
|
return Object.values(result.metafile!.outputs)[0].imports.map((entry): Import => {
|
|
153
156
|
const namedImports: string[] = [];
|
|
154
|
-
|
|
155
157
|
const parsedImport = parsedImports.find((capture) => capture?.[4] === entry.path);
|
|
156
158
|
if (parsedImport?.[2]) {
|
|
157
159
|
NAMED_IMPORTS_REGEX.lastIndex = 0;
|
|
@@ -205,10 +207,9 @@ const IMPORT_REGEX =
|
|
|
205
207
|
const NAMED_IMPORTS_REGEX = /[ \n\t]*{((?:[ \n\t]*[^ \n\t"'{}]+[ \n\t]*,?)+)}[ \n\t]*/gm;
|
|
206
208
|
|
|
207
209
|
const allMatches = (regex: RegExp, str: string) => {
|
|
208
|
-
regex.lastIndex = 0;
|
|
209
|
-
|
|
210
210
|
let match;
|
|
211
211
|
const matches = [];
|
|
212
|
+
regex.lastIndex = 0;
|
|
212
213
|
while ((match = regex.exec(str))) {
|
|
213
214
|
matches.push(match);
|
|
214
215
|
}
|
|
@@ -242,6 +243,9 @@ const analyzeSourceFileImports = (code: string): ParsedImport[] => {
|
|
|
242
243
|
});
|
|
243
244
|
};
|
|
244
245
|
|
|
246
|
+
const MAX_RETRIES = 5;
|
|
247
|
+
const INITIAL_DELAY = 1_000;
|
|
248
|
+
|
|
245
249
|
const httpPlugin: Plugin = {
|
|
246
250
|
name: 'http',
|
|
247
251
|
setup: (build) => {
|
|
@@ -263,8 +267,25 @@ const httpPlugin: Plugin = {
|
|
|
263
267
|
// When a URL is loaded, we want to actually download the content from the internet.
|
|
264
268
|
// This has just enough logic to be able to handle the example import from unpkg.com but in reality this would probably need to be more complex.
|
|
265
269
|
build.onLoad({ filter: /.*/, namespace: 'http-url' }, async (args) => {
|
|
266
|
-
|
|
267
|
-
|
|
270
|
+
return Effect.gen(function* () {
|
|
271
|
+
const response = yield* HttpClient.get(args.path);
|
|
272
|
+
if (response.status !== 200) {
|
|
273
|
+
throw new Error(`failed to fetch: ${response.status}`);
|
|
274
|
+
}
|
|
275
|
+
|
|
276
|
+
const text = yield* response.text;
|
|
277
|
+
return { contents: text, loader: 'jsx' as Loader };
|
|
278
|
+
}).pipe(
|
|
279
|
+
Effect.retry(
|
|
280
|
+
pipe(
|
|
281
|
+
Schedule.exponential(Duration.millis(INITIAL_DELAY)),
|
|
282
|
+
Schedule.jittered,
|
|
283
|
+
Schedule.intersect(Schedule.recurs(MAX_RETRIES - 1)),
|
|
284
|
+
),
|
|
285
|
+
),
|
|
286
|
+
Effect.provide(FetchHttpClient.layer),
|
|
287
|
+
runAndForwardErrors,
|
|
288
|
+
);
|
|
268
289
|
});
|
|
269
290
|
},
|
|
270
291
|
};
|
package/src/edge/functions.ts
CHANGED
|
@@ -8,27 +8,30 @@ import { type Client } from '@dxos/client';
|
|
|
8
8
|
import { createEdgeIdentity } from '@dxos/client/edge';
|
|
9
9
|
import { EdgeHttpClient } from '@dxos/edge-client';
|
|
10
10
|
import { invariant } from '@dxos/invariant';
|
|
11
|
-
import type
|
|
11
|
+
import { type PublicKey } from '@dxos/keys';
|
|
12
12
|
import { log } from '@dxos/log';
|
|
13
13
|
import { type UploadFunctionResponseBody } from '@dxos/protocols';
|
|
14
14
|
|
|
15
15
|
export type UploadWorkerArgs = {
|
|
16
16
|
client: Client;
|
|
17
|
-
source: string;
|
|
18
17
|
version: string;
|
|
19
18
|
name?: string;
|
|
20
19
|
functionId?: string;
|
|
21
20
|
ownerPublicKey: PublicKey;
|
|
21
|
+
entryPoint: string;
|
|
22
|
+
assets: Record<string, Uint8Array>;
|
|
22
23
|
};
|
|
23
24
|
|
|
24
25
|
export const uploadWorkerFunction = async ({
|
|
25
26
|
client,
|
|
26
27
|
version,
|
|
27
|
-
source,
|
|
28
28
|
name,
|
|
29
29
|
functionId,
|
|
30
30
|
ownerPublicKey,
|
|
31
|
+
entryPoint,
|
|
32
|
+
assets,
|
|
31
33
|
}: UploadWorkerArgs): Promise<UploadFunctionResponseBody> => {
|
|
34
|
+
log('uploading function', { functionId, name, version, ownerPublicKey });
|
|
32
35
|
const edgeUrl = client.config.values.runtime?.services?.edge?.url;
|
|
33
36
|
invariant(edgeUrl, 'Edge is not configured.');
|
|
34
37
|
const edgeClient = new EdgeHttpClient(edgeUrl);
|
|
@@ -36,15 +39,15 @@ export const uploadWorkerFunction = async ({
|
|
|
36
39
|
edgeClient.setIdentity(edgeIdentity);
|
|
37
40
|
const response = await edgeClient.uploadFunction(
|
|
38
41
|
{ functionId },
|
|
39
|
-
{ name, version,
|
|
42
|
+
{ name, version, ownerPublicKey: ownerPublicKey.toHex(), entryPoint, assets },
|
|
40
43
|
);
|
|
41
44
|
|
|
42
45
|
// TODO(burdon): Edge service log.
|
|
43
|
-
log
|
|
46
|
+
log('uploaded', {
|
|
44
47
|
identityKey: edgeIdentity.identityKey,
|
|
45
48
|
functionId,
|
|
46
49
|
name,
|
|
47
|
-
|
|
50
|
+
version,
|
|
48
51
|
response,
|
|
49
52
|
});
|
|
50
53
|
|
package/src/errors.ts
ADDED
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2025 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
import { BaseError, type BaseErrorOptions } from '@dxos/errors';
|
|
6
|
+
|
|
7
|
+
export class ServiceNotAvailableError extends BaseError.extend('SERVICE_NOT_AVAILABLE', 'Service not available') {
|
|
8
|
+
constructor(service: string, options?: Omit<BaseErrorOptions, 'context'>) {
|
|
9
|
+
super({ context: { service }, ...options });
|
|
10
|
+
}
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
export class FunctionNotFoundError extends BaseError.extend('FUNCTION_NOT_FOUND', 'Function not found') {
|
|
14
|
+
constructor(functionKey: string, options?: Omit<BaseErrorOptions, 'context'>) {
|
|
15
|
+
super({ context: { function: functionKey }, ...options });
|
|
16
|
+
}
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
export class FunctionError extends BaseError.extend('FUNCTION_ERROR', 'Function invocation error') {}
|
|
20
|
+
|
|
21
|
+
export class TriggerStateNotFoundError extends BaseError.extend('TRIGGER_STATE_NOT_FOUND', 'Trigger state not found') {}
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2025 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
import { Effect, Schema } from 'effect';
|
|
6
|
+
|
|
7
|
+
import { defineFunction } from '../handler';
|
|
8
|
+
|
|
9
|
+
export default defineFunction({
|
|
10
|
+
name: 'example.org/function/fib',
|
|
11
|
+
description: 'Function that calculates a Fibonacci number',
|
|
12
|
+
inputSchema: Schema.Struct({
|
|
13
|
+
iterations: Schema.optional(Schema.Number).annotations({
|
|
14
|
+
description: 'Number of iterations',
|
|
15
|
+
default: 100_000,
|
|
16
|
+
}),
|
|
17
|
+
}),
|
|
18
|
+
outputSchema: Schema.Struct({
|
|
19
|
+
result: Schema.String,
|
|
20
|
+
}),
|
|
21
|
+
handler: Effect.fn(function* ({ data: { iterations = 100_000 } }) {
|
|
22
|
+
let a = 0n;
|
|
23
|
+
let b = 1n;
|
|
24
|
+
for (let i = 0; i < iterations; i++) {
|
|
25
|
+
a += b;
|
|
26
|
+
b = a - b;
|
|
27
|
+
}
|
|
28
|
+
return { result: a.toString() };
|
|
29
|
+
}),
|
|
30
|
+
});
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2025 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
import { Console, Effect, Schema } from 'effect';
|
|
6
|
+
|
|
7
|
+
import { defineFunction } from '../handler';
|
|
8
|
+
|
|
9
|
+
export default defineFunction({
|
|
10
|
+
name: 'example.org/function/reply',
|
|
11
|
+
description: 'Function that echoes the input',
|
|
12
|
+
inputSchema: Schema.Any,
|
|
13
|
+
outputSchema: Schema.Any,
|
|
14
|
+
handler: Effect.fn(function* ({ data }) {
|
|
15
|
+
yield* Console.log('reply', { data });
|
|
16
|
+
return data;
|
|
17
|
+
}),
|
|
18
|
+
});
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2025 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
import { Effect, Schema } from 'effect';
|
|
6
|
+
|
|
7
|
+
import { defineFunction } from '../handler';
|
|
8
|
+
|
|
9
|
+
export default defineFunction({
|
|
10
|
+
name: 'example.org/function/sleep',
|
|
11
|
+
description: 'Function that sleeps for a given amount of time',
|
|
12
|
+
inputSchema: Schema.Struct({
|
|
13
|
+
duration: Schema.optional(Schema.Number).annotations({
|
|
14
|
+
description: 'Milliseconds to sleep',
|
|
15
|
+
default: 100_000,
|
|
16
|
+
}),
|
|
17
|
+
}),
|
|
18
|
+
outputSchema: Schema.Void,
|
|
19
|
+
handler: Effect.fn(function* ({ data: { duration = 100_000 } }) {
|
|
20
|
+
yield* Effect.sleep(duration);
|
|
21
|
+
}),
|
|
22
|
+
});
|
package/src/executor/executor.ts
CHANGED
|
@@ -4,44 +4,51 @@
|
|
|
4
4
|
|
|
5
5
|
import { Effect, Schema } from 'effect';
|
|
6
6
|
|
|
7
|
-
import type
|
|
7
|
+
import { type SpaceId } from '@dxos/client/echo';
|
|
8
|
+
import { runAndForwardErrors } from '@dxos/effect';
|
|
8
9
|
|
|
9
10
|
import type { FunctionContext, FunctionDefinition } from '../handler';
|
|
10
|
-
import type { ServiceContainer } from '../services';
|
|
11
|
+
import type { ServiceContainer, Services } from '../services';
|
|
11
12
|
|
|
12
13
|
export class FunctionExecutor {
|
|
13
14
|
constructor(private readonly _services: ServiceContainer) {}
|
|
14
15
|
|
|
16
|
+
/**
|
|
17
|
+
* Invoke function.
|
|
18
|
+
*/
|
|
15
19
|
// TODO(dmaretskyi): Invocation context: queue, space, etc...
|
|
16
20
|
async invoke<F extends FunctionDefinition<any, any>>(
|
|
17
|
-
|
|
21
|
+
functionDef: F,
|
|
18
22
|
input: F extends FunctionDefinition<infer I, infer _O> ? I : never,
|
|
19
23
|
): Promise<F extends FunctionDefinition<infer _I, infer O> ? O : never> {
|
|
20
24
|
// Assert input matches schema
|
|
21
|
-
const assertInput =
|
|
25
|
+
const assertInput = functionDef.inputSchema.pipe(Schema.asserts);
|
|
22
26
|
(assertInput as any)(input);
|
|
23
27
|
|
|
24
28
|
const context: FunctionContext = {
|
|
29
|
+
space: undefined,
|
|
25
30
|
getService: this._services.getService.bind(this._services),
|
|
26
31
|
getSpace: async (_spaceId: SpaceId) => {
|
|
27
32
|
throw new Error('Not available. Use the database service instead.');
|
|
28
33
|
},
|
|
29
|
-
space: undefined,
|
|
30
|
-
get ai(): never {
|
|
31
|
-
throw new Error('Not available. Use the ai service instead.');
|
|
32
|
-
},
|
|
33
34
|
};
|
|
34
35
|
|
|
35
|
-
const result =
|
|
36
|
-
|
|
37
|
-
// Assert output matches schema
|
|
38
|
-
const assertOutput = fnDef.outputSchema?.pipe(Schema.asserts);
|
|
39
|
-
(assertOutput as any)(result);
|
|
36
|
+
const result = functionDef.handler({ context, data: input });
|
|
40
37
|
|
|
38
|
+
let data: unknown;
|
|
41
39
|
if (Effect.isEffect(result)) {
|
|
42
|
-
|
|
40
|
+
data = await (result as Effect.Effect<unknown, unknown, Services>).pipe(
|
|
41
|
+
Effect.provide(this._services.createLayer()),
|
|
42
|
+
runAndForwardErrors,
|
|
43
|
+
);
|
|
44
|
+
} else {
|
|
45
|
+
data = await result;
|
|
43
46
|
}
|
|
44
47
|
|
|
45
|
-
|
|
48
|
+
// Assert output matches schema
|
|
49
|
+
const assertOutput = functionDef.outputSchema?.pipe(Schema.asserts);
|
|
50
|
+
(assertOutput as any)(data);
|
|
51
|
+
|
|
52
|
+
return data as any;
|
|
46
53
|
}
|
|
47
54
|
}
|