@dxos/functions 0.8.4-main.a4bbb77 → 0.8.4-main.abd8ff62ef
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/README.md +4 -6
- package/dist/lib/neutral/index.mjs +475 -0
- package/dist/lib/neutral/index.mjs.map +7 -0
- package/dist/lib/neutral/meta.json +1 -0
- package/dist/types/src/index.d.ts +3 -9
- package/dist/types/src/index.d.ts.map +1 -1
- package/dist/types/src/protocol/functions-ai-http-client.d.ts +12 -0
- package/dist/types/src/protocol/functions-ai-http-client.d.ts.map +1 -0
- package/dist/types/src/protocol/index.d.ts +2 -0
- package/dist/types/src/protocol/index.d.ts.map +1 -0
- package/dist/types/src/protocol/protocol.d.ts +7 -0
- package/dist/types/src/protocol/protocol.d.ts.map +1 -0
- package/dist/types/src/protocol/protocol.test.d.ts +2 -0
- package/dist/types/src/protocol/protocol.test.d.ts.map +1 -0
- package/dist/types/src/sdk.d.ts +10 -0
- package/dist/types/src/sdk.d.ts.map +1 -0
- package/dist/types/src/services/credentials.d.ts +22 -39
- package/dist/types/src/services/credentials.d.ts.map +1 -1
- package/dist/types/src/services/function-invocation-service.d.ts +9 -20
- package/dist/types/src/services/function-invocation-service.d.ts.map +1 -1
- package/dist/types/src/services/index.d.ts +1 -6
- package/dist/types/src/services/index.d.ts.map +1 -1
- package/dist/types/src/services/queues.d.ts +1 -44
- package/dist/types/src/services/queues.d.ts.map +1 -1
- package/dist/types/src/services/tracing.d.ts +2 -54
- package/dist/types/src/services/tracing.d.ts.map +1 -1
- package/dist/types/src/types/index.d.ts +2 -0
- package/dist/types/src/types/index.d.ts.map +1 -0
- package/dist/types/src/types/url.d.ts +13 -0
- package/dist/types/src/types/url.d.ts.map +1 -0
- package/dist/types/tsconfig.tsbuildinfo +1 -1
- package/package.json +25 -72
- package/src/index.ts +3 -9
- package/src/protocol/functions-ai-http-client.ts +67 -0
- package/src/{executor → protocol}/index.ts +1 -1
- package/src/protocol/protocol.test.ts +58 -0
- package/src/protocol/protocol.ts +292 -0
- package/src/sdk.ts +29 -0
- package/src/services/credentials.ts +89 -112
- package/src/services/function-invocation-service.ts +22 -71
- package/src/services/index.ts +1 -6
- package/src/services/queues.ts +1 -78
- package/src/services/tracing.ts +1 -134
- package/src/types/index.ts +5 -0
- package/src/types/url.ts +32 -0
- package/dist/lib/browser/bundler/index.mjs +0 -265
- package/dist/lib/browser/bundler/index.mjs.map +0 -7
- package/dist/lib/browser/chunk-C2Z7LCJ2.mjs +0 -649
- package/dist/lib/browser/chunk-C2Z7LCJ2.mjs.map +0 -7
- package/dist/lib/browser/chunk-J5LGTIGS.mjs +0 -10
- package/dist/lib/browser/chunk-J5LGTIGS.mjs.map +0 -7
- package/dist/lib/browser/edge/index.mjs +0 -83
- package/dist/lib/browser/edge/index.mjs.map +0 -7
- package/dist/lib/browser/index.mjs +0 -1366
- package/dist/lib/browser/index.mjs.map +0 -7
- package/dist/lib/browser/meta.json +0 -1
- package/dist/lib/browser/testing/index.mjs +0 -129
- package/dist/lib/browser/testing/index.mjs.map +0 -7
- package/dist/lib/node-esm/bundler/index.mjs +0 -266
- package/dist/lib/node-esm/bundler/index.mjs.map +0 -7
- package/dist/lib/node-esm/chunk-AH3AZM2U.mjs +0 -651
- package/dist/lib/node-esm/chunk-AH3AZM2U.mjs.map +0 -7
- package/dist/lib/node-esm/chunk-HSLMI22Q.mjs +0 -11
- package/dist/lib/node-esm/chunk-HSLMI22Q.mjs.map +0 -7
- package/dist/lib/node-esm/edge/index.mjs +0 -84
- package/dist/lib/node-esm/edge/index.mjs.map +0 -7
- package/dist/lib/node-esm/index.mjs +0 -1367
- package/dist/lib/node-esm/index.mjs.map +0 -7
- package/dist/lib/node-esm/meta.json +0 -1
- package/dist/lib/node-esm/testing/index.mjs +0 -130
- package/dist/lib/node-esm/testing/index.mjs.map +0 -7
- package/dist/types/src/bundler/bundler.d.ts +0 -49
- package/dist/types/src/bundler/bundler.d.ts.map +0 -1
- package/dist/types/src/bundler/bundler.test.d.ts +0 -2
- package/dist/types/src/bundler/bundler.test.d.ts.map +0 -1
- package/dist/types/src/bundler/index.d.ts +0 -2
- package/dist/types/src/bundler/index.d.ts.map +0 -1
- package/dist/types/src/edge/functions.d.ts +0 -17
- package/dist/types/src/edge/functions.d.ts.map +0 -1
- package/dist/types/src/edge/index.d.ts +0 -2
- package/dist/types/src/edge/index.d.ts.map +0 -1
- package/dist/types/src/errors.d.ts +0 -137
- package/dist/types/src/errors.d.ts.map +0 -1
- package/dist/types/src/examples/fib.d.ts +0 -7
- package/dist/types/src/examples/fib.d.ts.map +0 -1
- package/dist/types/src/examples/index.d.ts +0 -4
- package/dist/types/src/examples/index.d.ts.map +0 -1
- package/dist/types/src/examples/reply.d.ts +0 -3
- package/dist/types/src/examples/reply.d.ts.map +0 -1
- package/dist/types/src/examples/sleep.d.ts +0 -5
- package/dist/types/src/examples/sleep.d.ts.map +0 -1
- package/dist/types/src/executor/executor.d.ts +0 -14
- package/dist/types/src/executor/executor.d.ts.map +0 -1
- package/dist/types/src/executor/index.d.ts +0 -2
- package/dist/types/src/executor/index.d.ts.map +0 -1
- package/dist/types/src/handler.d.ts +0 -106
- package/dist/types/src/handler.d.ts.map +0 -1
- package/dist/types/src/schema.d.ts +0 -43
- package/dist/types/src/schema.d.ts.map +0 -1
- package/dist/types/src/services/database.d.ts +0 -63
- package/dist/types/src/services/database.d.ts.map +0 -1
- package/dist/types/src/services/event-logger.d.ts +0 -72
- package/dist/types/src/services/event-logger.d.ts.map +0 -1
- package/dist/types/src/services/function-invocation-service.test.d.ts +0 -2
- package/dist/types/src/services/function-invocation-service.test.d.ts.map +0 -1
- package/dist/types/src/services/local-function-execution.d.ts +0 -32
- package/dist/types/src/services/local-function-execution.d.ts.map +0 -1
- package/dist/types/src/services/remote-function-execution-service.d.ts +0 -20
- package/dist/types/src/services/remote-function-execution-service.d.ts.map +0 -1
- package/dist/types/src/services/service-container.d.ts +0 -56
- package/dist/types/src/services/service-container.d.ts.map +0 -1
- package/dist/types/src/services/service-registry.d.ts +0 -29
- package/dist/types/src/services/service-registry.d.ts.map +0 -1
- package/dist/types/src/services/service-registry.test.d.ts +0 -2
- package/dist/types/src/services/service-registry.test.d.ts.map +0 -1
- package/dist/types/src/testing/index.d.ts +0 -3
- package/dist/types/src/testing/index.d.ts.map +0 -1
- package/dist/types/src/testing/layer.d.ts +0 -17
- package/dist/types/src/testing/layer.d.ts.map +0 -1
- package/dist/types/src/testing/logger.d.ts +0 -5
- package/dist/types/src/testing/logger.d.ts.map +0 -1
- package/dist/types/src/testing/persist-database.test.d.ts +0 -2
- package/dist/types/src/testing/persist-database.test.d.ts.map +0 -1
- package/dist/types/src/testing/services.d.ts +0 -59
- package/dist/types/src/testing/services.d.ts.map +0 -1
- package/dist/types/src/trace.d.ts +0 -122
- package/dist/types/src/trace.d.ts.map +0 -1
- package/dist/types/src/translations.d.ts +0 -12
- package/dist/types/src/translations.d.ts.map +0 -1
- package/dist/types/src/triggers/index.d.ts +0 -4
- package/dist/types/src/triggers/index.d.ts.map +0 -1
- package/dist/types/src/triggers/input-builder.d.ts +0 -3
- package/dist/types/src/triggers/input-builder.d.ts.map +0 -1
- package/dist/types/src/triggers/invocation-tracer.d.ts +0 -35
- package/dist/types/src/triggers/invocation-tracer.d.ts.map +0 -1
- package/dist/types/src/triggers/trigger-dispatcher.d.ts +0 -74
- package/dist/types/src/triggers/trigger-dispatcher.d.ts.map +0 -1
- package/dist/types/src/triggers/trigger-dispatcher.test.d.ts +0 -2
- package/dist/types/src/triggers/trigger-dispatcher.test.d.ts.map +0 -1
- package/dist/types/src/triggers/trigger-state-store.d.ts +0 -27
- package/dist/types/src/triggers/trigger-state-store.d.ts.map +0 -1
- package/dist/types/src/types.d.ts +0 -221
- package/dist/types/src/types.d.ts.map +0 -1
- package/dist/types/src/url.d.ts +0 -21
- package/dist/types/src/url.d.ts.map +0 -1
- package/src/bundler/bundler.test.ts +0 -58
- package/src/bundler/bundler.ts +0 -291
- package/src/bundler/index.ts +0 -5
- package/src/edge/functions.ts +0 -67
- package/src/edge/index.ts +0 -9
- package/src/errors.ts +0 -21
- package/src/examples/fib.ts +0 -31
- package/src/examples/index.ts +0 -7
- package/src/examples/reply.ts +0 -19
- package/src/examples/sleep.ts +0 -23
- package/src/executor/executor.ts +0 -57
- package/src/handler.ts +0 -222
- package/src/schema.ts +0 -68
- package/src/services/database.ts +0 -171
- package/src/services/event-logger.ts +0 -118
- package/src/services/function-invocation-service.test.ts +0 -79
- package/src/services/local-function-execution.ts +0 -150
- package/src/services/remote-function-execution-service.ts +0 -61
- package/src/services/service-container.ts +0 -114
- package/src/services/service-registry.test.ts +0 -42
- package/src/services/service-registry.ts +0 -59
- package/src/testing/index.ts +0 -6
- package/src/testing/layer.ts +0 -112
- package/src/testing/logger.ts +0 -16
- package/src/testing/persist-database.test.ts +0 -87
- package/src/testing/services.ts +0 -115
- package/src/trace.ts +0 -178
- package/src/translations.ts +0 -20
- package/src/triggers/index.ts +0 -7
- package/src/triggers/input-builder.ts +0 -35
- package/src/triggers/invocation-tracer.ts +0 -99
- package/src/triggers/trigger-dispatcher.test.ts +0 -651
- package/src/triggers/trigger-dispatcher.ts +0 -522
- package/src/triggers/trigger-state-store.ts +0 -60
- package/src/types.ts +0 -214
- package/src/url.ts +0 -55
|
@@ -2,121 +2,32 @@
|
|
|
2
2
|
// Copyright 2025 DXOS.org
|
|
3
3
|
//
|
|
4
4
|
|
|
5
|
-
import
|
|
6
|
-
import
|
|
7
|
-
|
|
5
|
+
import * as HttpClient from '@effect/platform/HttpClient';
|
|
6
|
+
import * as HttpClientRequest from '@effect/platform/HttpClientRequest';
|
|
7
|
+
import type * as Config from 'effect/Config';
|
|
8
|
+
import * as Context from 'effect/Context';
|
|
9
|
+
import * as Effect from 'effect/Effect';
|
|
10
|
+
import * as Layer from 'effect/Layer';
|
|
11
|
+
import * as Redacted from 'effect/Redacted';
|
|
12
|
+
|
|
13
|
+
import { Credential } from '@dxos/compute';
|
|
8
14
|
import { Query } from '@dxos/echo';
|
|
9
|
-
import {
|
|
10
|
-
|
|
11
|
-
import { DatabaseService } from './database';
|
|
12
|
-
|
|
13
|
-
export type CredentialQuery = {
|
|
14
|
-
service?: string;
|
|
15
|
-
};
|
|
16
|
-
|
|
17
|
-
// TODO(dmaretskyi): Unify with other apis.
|
|
18
|
-
// packages/sdk/schema/src/common/access-token.ts
|
|
19
|
-
export type ServiceCredential = {
|
|
20
|
-
service: string;
|
|
21
|
-
|
|
22
|
-
// TODO(dmaretskyi): Build out.
|
|
23
|
-
apiKey?: string;
|
|
24
|
-
};
|
|
25
|
-
|
|
26
|
-
export class CredentialsService extends Context.Tag('@dxos/functions/CredentialsService')<
|
|
27
|
-
CredentialsService,
|
|
28
|
-
{
|
|
29
|
-
/**
|
|
30
|
-
* Query all.
|
|
31
|
-
*/
|
|
32
|
-
queryCredentials: (query: CredentialQuery) => Promise<ServiceCredential[]>;
|
|
33
|
-
|
|
34
|
-
/**
|
|
35
|
-
* Get a single credential.
|
|
36
|
-
* @throws {Error} If no credential is found.
|
|
37
|
-
*/
|
|
38
|
-
getCredential: (query: CredentialQuery) => Promise<ServiceCredential>;
|
|
39
|
-
}
|
|
40
|
-
>() {
|
|
41
|
-
static getCredential = (query: CredentialQuery): Effect.Effect<ServiceCredential, never, CredentialsService> =>
|
|
42
|
-
Effect.gen(function* () {
|
|
43
|
-
const credentials = yield* CredentialsService;
|
|
44
|
-
return yield* Effect.promise(() => credentials.getCredential(query));
|
|
45
|
-
});
|
|
46
|
-
|
|
47
|
-
static getApiKey = (query: CredentialQuery): Effect.Effect<Redacted.Redacted<string>, never, CredentialsService> =>
|
|
48
|
-
Effect.gen(function* () {
|
|
49
|
-
const credential = yield* CredentialsService.getCredential(query);
|
|
50
|
-
if (!credential.apiKey) {
|
|
51
|
-
throw new Error(`API key not found for service: ${query.service}`);
|
|
52
|
-
}
|
|
53
|
-
return Redacted.make(credential.apiKey);
|
|
54
|
-
});
|
|
55
|
-
|
|
56
|
-
static configuredLayer = (credentials: ServiceCredential[]) =>
|
|
57
|
-
Layer.succeed(CredentialsService, new ConfiguredCredentialsService(credentials));
|
|
58
|
-
|
|
59
|
-
static layerConfig = (credentials: { service: string; apiKey: Config.Config<Redacted.Redacted<string>> }[]) =>
|
|
60
|
-
Layer.effect(
|
|
61
|
-
CredentialsService,
|
|
62
|
-
Effect.gen(function* () {
|
|
63
|
-
const serviceCredentials = yield* Effect.forEach(credentials, ({ service, apiKey }) =>
|
|
64
|
-
Effect.gen(function* () {
|
|
65
|
-
return {
|
|
66
|
-
service,
|
|
67
|
-
apiKey: Redacted.value(yield* apiKey),
|
|
68
|
-
};
|
|
69
|
-
}),
|
|
70
|
-
);
|
|
71
|
-
|
|
72
|
-
return new ConfiguredCredentialsService(serviceCredentials);
|
|
73
|
-
}),
|
|
74
|
-
);
|
|
75
|
-
|
|
76
|
-
static layerFromDatabase = () =>
|
|
77
|
-
Layer.effect(
|
|
78
|
-
CredentialsService,
|
|
79
|
-
Effect.gen(function* () {
|
|
80
|
-
const dbService = yield* DatabaseService;
|
|
81
|
-
const queryCredentials = async (query: CredentialQuery): Promise<ServiceCredential[]> => {
|
|
82
|
-
const { objects: accessTokens } = await dbService.db.query(Query.type(DataType.AccessToken)).run();
|
|
83
|
-
return accessTokens
|
|
84
|
-
.filter((accessToken) => accessToken.source === query.service)
|
|
85
|
-
.map((accessToken) => ({
|
|
86
|
-
service: accessToken.source,
|
|
87
|
-
apiKey: accessToken.token,
|
|
88
|
-
}));
|
|
89
|
-
};
|
|
90
|
-
return {
|
|
91
|
-
getCredential: async (query) => {
|
|
92
|
-
const credentials = await queryCredentials(query);
|
|
93
|
-
if (credentials.length === 0) {
|
|
94
|
-
throw new Error(`Credential not found for service: ${query.service}`);
|
|
95
|
-
}
|
|
96
|
-
|
|
97
|
-
return credentials[0];
|
|
98
|
-
},
|
|
99
|
-
queryCredentials: async (query) => {
|
|
100
|
-
return queryCredentials(query);
|
|
101
|
-
},
|
|
102
|
-
};
|
|
103
|
-
}),
|
|
104
|
-
);
|
|
105
|
-
}
|
|
15
|
+
import { Database } from '@dxos/echo';
|
|
16
|
+
import { AccessToken } from '@dxos/types';
|
|
106
17
|
|
|
107
|
-
export class ConfiguredCredentialsService implements Context.Tag.Service<CredentialsService> {
|
|
108
|
-
constructor(private readonly credentials: ServiceCredential[] = []) {}
|
|
18
|
+
export class ConfiguredCredentialsService implements Context.Tag.Service<Credential.CredentialsService> {
|
|
19
|
+
constructor(private readonly credentials: Credential.ServiceCredential[] = []) {}
|
|
109
20
|
|
|
110
|
-
addCredentials(credentials: ServiceCredential[]): ConfiguredCredentialsService {
|
|
21
|
+
addCredentials(credentials: Credential.ServiceCredential[]): ConfiguredCredentialsService {
|
|
111
22
|
this.credentials.push(...credentials);
|
|
112
23
|
return this;
|
|
113
24
|
}
|
|
114
25
|
|
|
115
|
-
async queryCredentials(query: CredentialQuery): Promise<ServiceCredential[]> {
|
|
26
|
+
async queryCredentials(query: Credential.CredentialQuery): Promise<Credential.ServiceCredential[]> {
|
|
116
27
|
return this.credentials.filter((credential) => credential.service === query.service);
|
|
117
28
|
}
|
|
118
29
|
|
|
119
|
-
async getCredential(query: CredentialQuery): Promise<ServiceCredential> {
|
|
30
|
+
async getCredential(query: Credential.CredentialQuery): Promise<Credential.ServiceCredential> {
|
|
120
31
|
const credential = this.credentials.find((credential) => credential.service === query.service);
|
|
121
32
|
if (!credential) {
|
|
122
33
|
throw new Error(`Credential not found for service: ${query.service}`);
|
|
@@ -127,13 +38,79 @@ export class ConfiguredCredentialsService implements Context.Tag.Service<Credent
|
|
|
127
38
|
}
|
|
128
39
|
|
|
129
40
|
/**
|
|
130
|
-
* Maps the request to include the
|
|
41
|
+
* Maps the request to include the given token in the Authorization header.
|
|
131
42
|
*/
|
|
132
|
-
export const withAuthorization = (
|
|
133
|
-
HttpClient.
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
43
|
+
export const withAuthorization = (token: string, kind?: 'Bearer' | 'Basic') =>
|
|
44
|
+
HttpClient.mapRequest((request) => {
|
|
45
|
+
const authorization = kind ? `${kind} ${token}` : token;
|
|
46
|
+
return HttpClientRequest.setHeader(request, 'Authorization', authorization);
|
|
47
|
+
});
|
|
48
|
+
|
|
49
|
+
export const configuredCredentialsLayer = (credentials: Credential.ServiceCredential[]) =>
|
|
50
|
+
Layer.succeed(Credential.CredentialsService, new ConfiguredCredentialsService(credentials));
|
|
51
|
+
|
|
52
|
+
export const credentialsLayerConfig = (
|
|
53
|
+
credentials: {
|
|
54
|
+
service: string;
|
|
55
|
+
apiKey: Config.Config<Redacted.Redacted<string>>;
|
|
56
|
+
}[],
|
|
57
|
+
) =>
|
|
58
|
+
Layer.effect(
|
|
59
|
+
Credential.CredentialsService,
|
|
60
|
+
Effect.gen(function* () {
|
|
61
|
+
const serviceCredentials = yield* Effect.forEach(credentials, ({ service, apiKey }) =>
|
|
62
|
+
Effect.gen(function* () {
|
|
63
|
+
return {
|
|
64
|
+
service,
|
|
65
|
+
apiKey: Redacted.value(yield* apiKey),
|
|
66
|
+
};
|
|
67
|
+
}),
|
|
68
|
+
);
|
|
69
|
+
|
|
70
|
+
return new ConfiguredCredentialsService(serviceCredentials);
|
|
71
|
+
}),
|
|
72
|
+
);
|
|
73
|
+
|
|
74
|
+
export const credentialsLayerFromDatabase = ({ caching = false }: { caching?: boolean } = {}) =>
|
|
75
|
+
Layer.effect(
|
|
76
|
+
Credential.CredentialsService,
|
|
77
|
+
Effect.gen(function* () {
|
|
78
|
+
const dbService = yield* Database.Service;
|
|
79
|
+
const cache = new Map<string, Credential.ServiceCredential[]>();
|
|
80
|
+
|
|
81
|
+
const queryCredentials = async (query: Credential.CredentialQuery): Promise<Credential.ServiceCredential[]> => {
|
|
82
|
+
const cacheKey = JSON.stringify(query);
|
|
83
|
+
if (caching && cache.has(cacheKey)) {
|
|
84
|
+
return cache.get(cacheKey)!;
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
const accessTokens = await dbService.db.query(Query.type(AccessToken.AccessToken)).run();
|
|
88
|
+
const credentials = accessTokens
|
|
89
|
+
.filter((accessToken) => accessToken.source === query.service)
|
|
90
|
+
.map((accessToken) => ({
|
|
91
|
+
service: accessToken.source,
|
|
92
|
+
apiKey: accessToken.token,
|
|
93
|
+
}));
|
|
94
|
+
|
|
95
|
+
if (caching) {
|
|
96
|
+
cache.set(cacheKey, credentials);
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
return credentials;
|
|
100
|
+
};
|
|
101
|
+
|
|
102
|
+
return {
|
|
103
|
+
getCredential: async (query) => {
|
|
104
|
+
const credentials = await queryCredentials(query);
|
|
105
|
+
if (credentials.length === 0) {
|
|
106
|
+
throw new Error(`Credential not found for service: ${query.service}`);
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
return credentials[0];
|
|
110
|
+
},
|
|
111
|
+
queryCredentials: async (query) => {
|
|
112
|
+
return queryCredentials(query);
|
|
113
|
+
},
|
|
114
|
+
};
|
|
138
115
|
}),
|
|
139
116
|
);
|
|
@@ -1,82 +1,33 @@
|
|
|
1
1
|
//
|
|
2
2
|
// Copyright 2025 DXOS.org
|
|
3
3
|
//
|
|
4
|
-
import
|
|
4
|
+
import * as Context from 'effect/Context';
|
|
5
|
+
import * as Effect from 'effect/Effect';
|
|
6
|
+
import * as Layer from 'effect/Layer';
|
|
5
7
|
|
|
6
|
-
import {
|
|
7
|
-
|
|
8
|
-
import { type FunctionDefinition } from '../handler';
|
|
9
|
-
|
|
10
|
-
import { CredentialsService } from './credentials';
|
|
11
|
-
import { DatabaseService } from './database';
|
|
12
|
-
import {
|
|
13
|
-
FunctionImplementationResolver,
|
|
14
|
-
type InvocationServices,
|
|
15
|
-
LocalFunctionExecutionService,
|
|
16
|
-
} from './local-function-execution';
|
|
17
|
-
import { QueueService } from './queues';
|
|
18
|
-
import { RemoteFunctionExecutionService } from './remote-function-execution-service';
|
|
8
|
+
import { type FunctionNotFoundError, Operation } from '@dxos/compute';
|
|
19
9
|
|
|
20
10
|
export class FunctionInvocationService extends Context.Tag('@dxos/functions/FunctionInvocationService')<
|
|
21
11
|
FunctionInvocationService,
|
|
22
12
|
{
|
|
23
|
-
invokeFunction<I, O>(functionDef:
|
|
13
|
+
invokeFunction<I, O>(functionDef: Operation.Definition<I, O, any>, input: I): Effect.Effect<O>;
|
|
14
|
+
|
|
15
|
+
resolveFunction(key: string): Effect.Effect<Operation.Definition.Any, FunctionNotFoundError>;
|
|
24
16
|
}
|
|
25
17
|
>() {
|
|
26
|
-
static
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
return yield* remoteExecutioner.callFunction<I, O>(functionDef.meta.deployedFunctionId, input);
|
|
42
|
-
}
|
|
43
|
-
|
|
44
|
-
return yield* localExecutioner.invokeFunction(functionDef, input);
|
|
45
|
-
}),
|
|
46
|
-
} satisfies Context.Tag.Service<FunctionInvocationService>;
|
|
47
|
-
}),
|
|
48
|
-
);
|
|
49
|
-
|
|
50
|
-
// TODO(dmaretskyi): Don't provide `FunctionImplementationResolver`.
|
|
51
|
-
static layerTest = ({
|
|
52
|
-
functions = [],
|
|
53
|
-
}: {
|
|
54
|
-
functions?: FunctionDefinition<any, any>[];
|
|
55
|
-
} = {}): Layer.Layer<
|
|
56
|
-
FunctionInvocationService,
|
|
57
|
-
never,
|
|
58
|
-
AiService.AiService | CredentialsService | DatabaseService | QueueService
|
|
59
|
-
> =>
|
|
60
|
-
FunctionInvocationService.layer.pipe(
|
|
61
|
-
Layer.provide(LocalFunctionExecutionService.layerLive),
|
|
62
|
-
Layer.provide(FunctionImplementationResolver.layerTest({ functions })),
|
|
63
|
-
Layer.provide(RemoteFunctionExecutionService.layerMock),
|
|
64
|
-
);
|
|
65
|
-
|
|
66
|
-
// TODO(dmaretskyi): This shouldn't default to all services being not available.
|
|
67
|
-
// TODO(dmaretskyi): Don't provide `FunctionImplementationResolver`.
|
|
68
|
-
/**
|
|
69
|
-
* @deprecated Use {@link layerTest} instead.
|
|
70
|
-
*/
|
|
71
|
-
static layerTestMocked = ({
|
|
72
|
-
functions,
|
|
73
|
-
}: {
|
|
74
|
-
functions: FunctionDefinition<any, any>[];
|
|
75
|
-
}): Layer.Layer<FunctionInvocationService> =>
|
|
76
|
-
FunctionInvocationService.layerTest({ functions }).pipe(
|
|
77
|
-
Layer.provide(AiService.notAvailable),
|
|
78
|
-
Layer.provide(CredentialsService.configuredLayer([])),
|
|
79
|
-
Layer.provide(DatabaseService.notAvailable),
|
|
80
|
-
Layer.provide(QueueService.notAvailable),
|
|
81
|
-
);
|
|
18
|
+
static layerNotAvailable = Layer.succeed(FunctionInvocationService, {
|
|
19
|
+
invokeFunction: () => Effect.die('FunctionInvocationService is not avaialble.'),
|
|
20
|
+
resolveFunction: () => Effect.die('FunctionInvocationService is not available.'),
|
|
21
|
+
});
|
|
22
|
+
|
|
23
|
+
static invokeFunction = <I, O>(
|
|
24
|
+
functionDef: Operation.Definition<I, O, any>,
|
|
25
|
+
input: I,
|
|
26
|
+
): Effect.Effect<O, never, FunctionInvocationService> =>
|
|
27
|
+
Effect.serviceFunctionEffect(FunctionInvocationService, (service) => service.invokeFunction)(functionDef, input);
|
|
28
|
+
|
|
29
|
+
static resolveFunction = (
|
|
30
|
+
key: string,
|
|
31
|
+
): Effect.Effect<Operation.Definition.Any, FunctionNotFoundError, FunctionInvocationService> =>
|
|
32
|
+
Effect.serviceFunctionEffect(FunctionInvocationService, (service) => service.resolveFunction)(key);
|
|
82
33
|
}
|
package/src/services/index.ts
CHANGED
|
@@ -3,11 +3,6 @@
|
|
|
3
3
|
//
|
|
4
4
|
|
|
5
5
|
export * from './credentials';
|
|
6
|
-
export * from './database';
|
|
7
|
-
export * from './event-logger';
|
|
8
6
|
export * from './function-invocation-service';
|
|
9
|
-
export * from './local-function-execution';
|
|
10
7
|
export * from './queues';
|
|
11
|
-
export
|
|
12
|
-
export * from './tracing';
|
|
13
|
-
export * from './remote-function-execution-service';
|
|
8
|
+
export { MESSAGE_PROPERTY_TOOL_CALL_ID } from './tracing';
|
package/src/services/queues.ts
CHANGED
|
@@ -2,81 +2,4 @@
|
|
|
2
2
|
// Copyright 2025 DXOS.org
|
|
3
3
|
//
|
|
4
4
|
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
import type { Obj, Relation } from '@dxos/echo';
|
|
8
|
-
import type { Queue, QueueAPI, QueueFactory } from '@dxos/echo-db';
|
|
9
|
-
import type { DXN, QueueSubspaceTag } from '@dxos/keys';
|
|
10
|
-
|
|
11
|
-
/**
|
|
12
|
-
* Gives access to all queues.
|
|
13
|
-
*/
|
|
14
|
-
export class QueueService extends Context.Tag('@dxos/functions/QueueService')<
|
|
15
|
-
QueueService,
|
|
16
|
-
{
|
|
17
|
-
/**
|
|
18
|
-
* API to access the queues.
|
|
19
|
-
*/
|
|
20
|
-
readonly queues: QueueAPI;
|
|
21
|
-
|
|
22
|
-
/**
|
|
23
|
-
* The queue that is used to store the context of the current research.
|
|
24
|
-
* @deprecated Use `ContextQueueService` instead.
|
|
25
|
-
*/
|
|
26
|
-
readonly queue: Queue | undefined;
|
|
27
|
-
}
|
|
28
|
-
>() {
|
|
29
|
-
static notAvailable = Layer.succeed(QueueService, {
|
|
30
|
-
queues: {
|
|
31
|
-
get(_dxn) {
|
|
32
|
-
throw new Error('Queues not available');
|
|
33
|
-
},
|
|
34
|
-
create() {
|
|
35
|
-
throw new Error('Queues not available');
|
|
36
|
-
},
|
|
37
|
-
},
|
|
38
|
-
queue: undefined,
|
|
39
|
-
});
|
|
40
|
-
|
|
41
|
-
static make = (queues: QueueFactory, queue?: Queue): Context.Tag.Service<QueueService> => {
|
|
42
|
-
return {
|
|
43
|
-
queues,
|
|
44
|
-
queue,
|
|
45
|
-
};
|
|
46
|
-
};
|
|
47
|
-
|
|
48
|
-
static layer = (queues: QueueFactory, queue?: Queue): Layer.Layer<QueueService> =>
|
|
49
|
-
Layer.succeed(QueueService, QueueService.make(queues, queue));
|
|
50
|
-
|
|
51
|
-
/**
|
|
52
|
-
* Gets a queue by its DXN.
|
|
53
|
-
*/
|
|
54
|
-
static getQueue = <T extends Obj.Any | Relation.Any = Obj.Any | Relation.Any>(
|
|
55
|
-
dxn: DXN,
|
|
56
|
-
): Effect.Effect<Queue<T>, never, QueueService> => QueueService.pipe(Effect.map(({ queues }) => queues.get<T>(dxn)));
|
|
57
|
-
|
|
58
|
-
/**
|
|
59
|
-
* Creates a new queue.
|
|
60
|
-
*/
|
|
61
|
-
static createQueue = <T extends Obj.Any | Relation.Any = Obj.Any | Relation.Any>(options?: {
|
|
62
|
-
subspaceTag?: QueueSubspaceTag;
|
|
63
|
-
}): Effect.Effect<Queue<T>, never, QueueService> =>
|
|
64
|
-
QueueService.pipe(Effect.map(({ queues }) => queues.create<T>(options)));
|
|
65
|
-
|
|
66
|
-
static append = <T extends Obj.Any | Relation.Any = Obj.Any | Relation.Any>(
|
|
67
|
-
queue: Queue<T>,
|
|
68
|
-
objects: T[],
|
|
69
|
-
): Effect.Effect<void> => Effect.promise(() => queue.append(objects));
|
|
70
|
-
}
|
|
71
|
-
|
|
72
|
-
/**
|
|
73
|
-
* Gives access to a specific queue passed as a context.
|
|
74
|
-
*/
|
|
75
|
-
export class ContextQueueService extends Context.Tag('@dxos/functions/ContextQueueService')<
|
|
76
|
-
ContextQueueService,
|
|
77
|
-
{
|
|
78
|
-
readonly queue: Queue;
|
|
79
|
-
}
|
|
80
|
-
>() {
|
|
81
|
-
static layer = (queue: Queue) => Layer.succeed(ContextQueueService, { queue });
|
|
82
|
-
}
|
|
5
|
+
export { ContextQueueService, QueueService, feedServiceFromQueueServiceLayer } from '@dxos/echo-db';
|
package/src/services/tracing.ts
CHANGED
|
@@ -2,140 +2,7 @@
|
|
|
2
2
|
// Copyright 2025 DXOS.org
|
|
3
3
|
//
|
|
4
4
|
|
|
5
|
-
import { Context, Effect, Layer } from 'effect';
|
|
6
|
-
|
|
7
|
-
import { AgentStatus } from '@dxos/ai';
|
|
8
|
-
import { Obj } from '@dxos/echo';
|
|
9
|
-
import type { Queue } from '@dxos/echo-db';
|
|
10
|
-
import type { ObjectId } from '@dxos/echo-schema';
|
|
11
|
-
import { log } from '@dxos/log';
|
|
12
|
-
import { DataType } from '@dxos/schema';
|
|
13
|
-
|
|
14
|
-
/**
|
|
15
|
-
* Provides a way for compute primitives (functions, workflows, tools)
|
|
16
|
-
* to emit an execution trace as a series of structured ECHO objects.
|
|
17
|
-
*/
|
|
18
|
-
export class TracingService extends Context.Tag('@dxos/functions/TracingService')<
|
|
19
|
-
TracingService,
|
|
20
|
-
{
|
|
21
|
-
/**
|
|
22
|
-
* Gets the parent message ID.
|
|
23
|
-
*/
|
|
24
|
-
getTraceContext: () => TracingService.TraceContext;
|
|
25
|
-
|
|
26
|
-
/**
|
|
27
|
-
* Write an event to the tracing queue.
|
|
28
|
-
* @param event - The event to write. Must be an a typed object.
|
|
29
|
-
*/
|
|
30
|
-
write: (event: Obj.Any) => void;
|
|
31
|
-
}
|
|
32
|
-
>() {
|
|
33
|
-
static noop: Context.Tag.Service<TracingService> = { write: () => {}, getTraceContext: () => ({}) };
|
|
34
|
-
|
|
35
|
-
static layerNoop = Layer.succeed(TracingService, TracingService.noop);
|
|
36
|
-
|
|
37
|
-
static console: Context.Tag.Service<TracingService> = {
|
|
38
|
-
write: (event) => {
|
|
39
|
-
console.log(event);
|
|
40
|
-
},
|
|
41
|
-
getTraceContext: () => ({}),
|
|
42
|
-
};
|
|
43
|
-
|
|
44
|
-
static layerConsole = Layer.succeed(TracingService, TracingService.console);
|
|
45
|
-
|
|
46
|
-
static layerLogInfo = () =>
|
|
47
|
-
Layer.succeed(TracingService, {
|
|
48
|
-
write: (event) => {
|
|
49
|
-
if (Obj.instanceOf(AgentStatus, event)) {
|
|
50
|
-
log.info('status', { message: event.message });
|
|
51
|
-
}
|
|
52
|
-
},
|
|
53
|
-
getTraceContext: () => ({}),
|
|
54
|
-
});
|
|
55
|
-
|
|
56
|
-
/**
|
|
57
|
-
* Creates a TracingService layer that emits events to the parent tracing service.
|
|
58
|
-
*/
|
|
59
|
-
static layerSubframe = (mapContext: (currentContext: TracingService.TraceContext) => TracingService.TraceContext) =>
|
|
60
|
-
Layer.effect(
|
|
61
|
-
TracingService,
|
|
62
|
-
Effect.gen(function* () {
|
|
63
|
-
const tracing = yield* TracingService;
|
|
64
|
-
const context = mapContext(tracing.getTraceContext());
|
|
65
|
-
return {
|
|
66
|
-
write: (event) => tracing.write(event),
|
|
67
|
-
getTraceContext: () => context,
|
|
68
|
-
};
|
|
69
|
-
}),
|
|
70
|
-
);
|
|
71
|
-
|
|
72
|
-
static layerQueue = (queue: Queue) =>
|
|
73
|
-
Layer.effect(
|
|
74
|
-
TracingService,
|
|
75
|
-
Effect.gen(function* () {
|
|
76
|
-
// TODO(dmaretskyi): Batching.
|
|
77
|
-
return {
|
|
78
|
-
write: (event) => queue.append([event]),
|
|
79
|
-
getTraceContext: () => ({
|
|
80
|
-
debugInfo: {
|
|
81
|
-
queue: queue.dxn.toString(),
|
|
82
|
-
},
|
|
83
|
-
}),
|
|
84
|
-
};
|
|
85
|
-
}),
|
|
86
|
-
);
|
|
87
|
-
|
|
88
|
-
/**
|
|
89
|
-
* Emit the current human-readable execution status.
|
|
90
|
-
*/
|
|
91
|
-
static emitStatus: (
|
|
92
|
-
data: Omit<Obj.MakeProps<typeof AgentStatus>, 'created'>,
|
|
93
|
-
) => Effect.Effect<void, never, TracingService> = Effect.fnUntraced(function* (data) {
|
|
94
|
-
const tracing = yield* TracingService;
|
|
95
|
-
tracing.write(
|
|
96
|
-
Obj.make(AgentStatus, {
|
|
97
|
-
parentMessage: tracing.getTraceContext().parentMessage,
|
|
98
|
-
toolCallId: tracing.getTraceContext().toolCallId,
|
|
99
|
-
created: new Date().toISOString(),
|
|
100
|
-
...data,
|
|
101
|
-
}),
|
|
102
|
-
);
|
|
103
|
-
});
|
|
104
|
-
|
|
105
|
-
static emitConverationMessage: (
|
|
106
|
-
data: Obj.MakeProps<typeof DataType.Message>,
|
|
107
|
-
) => Effect.Effect<void, never, TracingService> = Effect.fnUntraced(function* (data) {
|
|
108
|
-
const tracing = yield* TracingService;
|
|
109
|
-
tracing.write(
|
|
110
|
-
Obj.make(DataType.Message, {
|
|
111
|
-
parentMessage: tracing.getTraceContext().parentMessage,
|
|
112
|
-
...data,
|
|
113
|
-
properties: {
|
|
114
|
-
[MESSAGE_PROPERTY_TOOL_CALL_ID]: tracing.getTraceContext().toolCallId,
|
|
115
|
-
...data.properties,
|
|
116
|
-
},
|
|
117
|
-
}),
|
|
118
|
-
);
|
|
119
|
-
});
|
|
120
|
-
}
|
|
121
|
-
|
|
122
|
-
export namespace TracingService {
|
|
123
|
-
export interface TraceContext {
|
|
124
|
-
/**
|
|
125
|
-
* If this thread sprung from a tool call, this is the ID of the message containing the tool call.
|
|
126
|
-
*/
|
|
127
|
-
parentMessage?: ObjectId;
|
|
128
|
-
|
|
129
|
-
/**
|
|
130
|
-
* If the current thread is a byproduct of a tool call, this is the ID of the tool call.
|
|
131
|
-
*/
|
|
132
|
-
toolCallId?: string;
|
|
133
|
-
|
|
134
|
-
debugInfo?: unknown;
|
|
135
|
-
}
|
|
136
|
-
}
|
|
137
|
-
|
|
138
5
|
/**
|
|
139
|
-
* Goes into {@link
|
|
6
|
+
* Goes into {@link Message['properties']}
|
|
140
7
|
*/
|
|
141
8
|
export const MESSAGE_PROPERTY_TOOL_CALL_ID = 'toolCallId' as const;
|
package/src/types/url.ts
ADDED
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2025 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
import { type Obj } from '@dxos/echo';
|
|
6
|
+
|
|
7
|
+
// TODO: use URL scheme for source?
|
|
8
|
+
export const FUNCTIONS_META_KEY = 'org.dxos.service.function';
|
|
9
|
+
|
|
10
|
+
export const FUNCTIONS_PRESET_META_KEY = 'org.dxos.service.function-preset';
|
|
11
|
+
|
|
12
|
+
/**
|
|
13
|
+
* NOTE: functionId is backend ID, not ECHO object id.
|
|
14
|
+
*/
|
|
15
|
+
export const getUserFunctionIdInMetadata = (meta: Obj.ReadonlyMeta) => {
|
|
16
|
+
return meta.keys.find((key) => key.source === FUNCTIONS_META_KEY)?.id;
|
|
17
|
+
};
|
|
18
|
+
|
|
19
|
+
/**
|
|
20
|
+
* NOTE: functionId is backend ID, not ECHO object id.
|
|
21
|
+
* Must be called inside Obj.update() since it mutates the meta.
|
|
22
|
+
*/
|
|
23
|
+
export const setUserFunctionIdInMetadata = (meta: Obj.Meta, functionId: string) => {
|
|
24
|
+
const key = meta.keys.find((key) => key.source === FUNCTIONS_META_KEY);
|
|
25
|
+
if (key) {
|
|
26
|
+
if (key.id !== functionId) {
|
|
27
|
+
throw new Error('Metadata mismatch');
|
|
28
|
+
}
|
|
29
|
+
} else {
|
|
30
|
+
meta.keys.push({ source: FUNCTIONS_META_KEY, id: functionId });
|
|
31
|
+
}
|
|
32
|
+
};
|