@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.
Files changed (181) hide show
  1. package/README.md +4 -6
  2. package/dist/lib/neutral/index.mjs +475 -0
  3. package/dist/lib/neutral/index.mjs.map +7 -0
  4. package/dist/lib/neutral/meta.json +1 -0
  5. package/dist/types/src/index.d.ts +3 -9
  6. package/dist/types/src/index.d.ts.map +1 -1
  7. package/dist/types/src/protocol/functions-ai-http-client.d.ts +12 -0
  8. package/dist/types/src/protocol/functions-ai-http-client.d.ts.map +1 -0
  9. package/dist/types/src/protocol/index.d.ts +2 -0
  10. package/dist/types/src/protocol/index.d.ts.map +1 -0
  11. package/dist/types/src/protocol/protocol.d.ts +7 -0
  12. package/dist/types/src/protocol/protocol.d.ts.map +1 -0
  13. package/dist/types/src/protocol/protocol.test.d.ts +2 -0
  14. package/dist/types/src/protocol/protocol.test.d.ts.map +1 -0
  15. package/dist/types/src/sdk.d.ts +10 -0
  16. package/dist/types/src/sdk.d.ts.map +1 -0
  17. package/dist/types/src/services/credentials.d.ts +22 -39
  18. package/dist/types/src/services/credentials.d.ts.map +1 -1
  19. package/dist/types/src/services/function-invocation-service.d.ts +9 -20
  20. package/dist/types/src/services/function-invocation-service.d.ts.map +1 -1
  21. package/dist/types/src/services/index.d.ts +1 -6
  22. package/dist/types/src/services/index.d.ts.map +1 -1
  23. package/dist/types/src/services/queues.d.ts +1 -44
  24. package/dist/types/src/services/queues.d.ts.map +1 -1
  25. package/dist/types/src/services/tracing.d.ts +2 -54
  26. package/dist/types/src/services/tracing.d.ts.map +1 -1
  27. package/dist/types/src/types/index.d.ts +2 -0
  28. package/dist/types/src/types/index.d.ts.map +1 -0
  29. package/dist/types/src/types/url.d.ts +13 -0
  30. package/dist/types/src/types/url.d.ts.map +1 -0
  31. package/dist/types/tsconfig.tsbuildinfo +1 -1
  32. package/package.json +25 -72
  33. package/src/index.ts +3 -9
  34. package/src/protocol/functions-ai-http-client.ts +67 -0
  35. package/src/{executor → protocol}/index.ts +1 -1
  36. package/src/protocol/protocol.test.ts +58 -0
  37. package/src/protocol/protocol.ts +292 -0
  38. package/src/sdk.ts +29 -0
  39. package/src/services/credentials.ts +89 -112
  40. package/src/services/function-invocation-service.ts +22 -71
  41. package/src/services/index.ts +1 -6
  42. package/src/services/queues.ts +1 -78
  43. package/src/services/tracing.ts +1 -134
  44. package/src/types/index.ts +5 -0
  45. package/src/types/url.ts +32 -0
  46. package/dist/lib/browser/bundler/index.mjs +0 -265
  47. package/dist/lib/browser/bundler/index.mjs.map +0 -7
  48. package/dist/lib/browser/chunk-C2Z7LCJ2.mjs +0 -649
  49. package/dist/lib/browser/chunk-C2Z7LCJ2.mjs.map +0 -7
  50. package/dist/lib/browser/chunk-J5LGTIGS.mjs +0 -10
  51. package/dist/lib/browser/chunk-J5LGTIGS.mjs.map +0 -7
  52. package/dist/lib/browser/edge/index.mjs +0 -83
  53. package/dist/lib/browser/edge/index.mjs.map +0 -7
  54. package/dist/lib/browser/index.mjs +0 -1366
  55. package/dist/lib/browser/index.mjs.map +0 -7
  56. package/dist/lib/browser/meta.json +0 -1
  57. package/dist/lib/browser/testing/index.mjs +0 -129
  58. package/dist/lib/browser/testing/index.mjs.map +0 -7
  59. package/dist/lib/node-esm/bundler/index.mjs +0 -266
  60. package/dist/lib/node-esm/bundler/index.mjs.map +0 -7
  61. package/dist/lib/node-esm/chunk-AH3AZM2U.mjs +0 -651
  62. package/dist/lib/node-esm/chunk-AH3AZM2U.mjs.map +0 -7
  63. package/dist/lib/node-esm/chunk-HSLMI22Q.mjs +0 -11
  64. package/dist/lib/node-esm/chunk-HSLMI22Q.mjs.map +0 -7
  65. package/dist/lib/node-esm/edge/index.mjs +0 -84
  66. package/dist/lib/node-esm/edge/index.mjs.map +0 -7
  67. package/dist/lib/node-esm/index.mjs +0 -1367
  68. package/dist/lib/node-esm/index.mjs.map +0 -7
  69. package/dist/lib/node-esm/meta.json +0 -1
  70. package/dist/lib/node-esm/testing/index.mjs +0 -130
  71. package/dist/lib/node-esm/testing/index.mjs.map +0 -7
  72. package/dist/types/src/bundler/bundler.d.ts +0 -49
  73. package/dist/types/src/bundler/bundler.d.ts.map +0 -1
  74. package/dist/types/src/bundler/bundler.test.d.ts +0 -2
  75. package/dist/types/src/bundler/bundler.test.d.ts.map +0 -1
  76. package/dist/types/src/bundler/index.d.ts +0 -2
  77. package/dist/types/src/bundler/index.d.ts.map +0 -1
  78. package/dist/types/src/edge/functions.d.ts +0 -17
  79. package/dist/types/src/edge/functions.d.ts.map +0 -1
  80. package/dist/types/src/edge/index.d.ts +0 -2
  81. package/dist/types/src/edge/index.d.ts.map +0 -1
  82. package/dist/types/src/errors.d.ts +0 -137
  83. package/dist/types/src/errors.d.ts.map +0 -1
  84. package/dist/types/src/examples/fib.d.ts +0 -7
  85. package/dist/types/src/examples/fib.d.ts.map +0 -1
  86. package/dist/types/src/examples/index.d.ts +0 -4
  87. package/dist/types/src/examples/index.d.ts.map +0 -1
  88. package/dist/types/src/examples/reply.d.ts +0 -3
  89. package/dist/types/src/examples/reply.d.ts.map +0 -1
  90. package/dist/types/src/examples/sleep.d.ts +0 -5
  91. package/dist/types/src/examples/sleep.d.ts.map +0 -1
  92. package/dist/types/src/executor/executor.d.ts +0 -14
  93. package/dist/types/src/executor/executor.d.ts.map +0 -1
  94. package/dist/types/src/executor/index.d.ts +0 -2
  95. package/dist/types/src/executor/index.d.ts.map +0 -1
  96. package/dist/types/src/handler.d.ts +0 -106
  97. package/dist/types/src/handler.d.ts.map +0 -1
  98. package/dist/types/src/schema.d.ts +0 -43
  99. package/dist/types/src/schema.d.ts.map +0 -1
  100. package/dist/types/src/services/database.d.ts +0 -63
  101. package/dist/types/src/services/database.d.ts.map +0 -1
  102. package/dist/types/src/services/event-logger.d.ts +0 -72
  103. package/dist/types/src/services/event-logger.d.ts.map +0 -1
  104. package/dist/types/src/services/function-invocation-service.test.d.ts +0 -2
  105. package/dist/types/src/services/function-invocation-service.test.d.ts.map +0 -1
  106. package/dist/types/src/services/local-function-execution.d.ts +0 -32
  107. package/dist/types/src/services/local-function-execution.d.ts.map +0 -1
  108. package/dist/types/src/services/remote-function-execution-service.d.ts +0 -20
  109. package/dist/types/src/services/remote-function-execution-service.d.ts.map +0 -1
  110. package/dist/types/src/services/service-container.d.ts +0 -56
  111. package/dist/types/src/services/service-container.d.ts.map +0 -1
  112. package/dist/types/src/services/service-registry.d.ts +0 -29
  113. package/dist/types/src/services/service-registry.d.ts.map +0 -1
  114. package/dist/types/src/services/service-registry.test.d.ts +0 -2
  115. package/dist/types/src/services/service-registry.test.d.ts.map +0 -1
  116. package/dist/types/src/testing/index.d.ts +0 -3
  117. package/dist/types/src/testing/index.d.ts.map +0 -1
  118. package/dist/types/src/testing/layer.d.ts +0 -17
  119. package/dist/types/src/testing/layer.d.ts.map +0 -1
  120. package/dist/types/src/testing/logger.d.ts +0 -5
  121. package/dist/types/src/testing/logger.d.ts.map +0 -1
  122. package/dist/types/src/testing/persist-database.test.d.ts +0 -2
  123. package/dist/types/src/testing/persist-database.test.d.ts.map +0 -1
  124. package/dist/types/src/testing/services.d.ts +0 -59
  125. package/dist/types/src/testing/services.d.ts.map +0 -1
  126. package/dist/types/src/trace.d.ts +0 -122
  127. package/dist/types/src/trace.d.ts.map +0 -1
  128. package/dist/types/src/translations.d.ts +0 -12
  129. package/dist/types/src/translations.d.ts.map +0 -1
  130. package/dist/types/src/triggers/index.d.ts +0 -4
  131. package/dist/types/src/triggers/index.d.ts.map +0 -1
  132. package/dist/types/src/triggers/input-builder.d.ts +0 -3
  133. package/dist/types/src/triggers/input-builder.d.ts.map +0 -1
  134. package/dist/types/src/triggers/invocation-tracer.d.ts +0 -35
  135. package/dist/types/src/triggers/invocation-tracer.d.ts.map +0 -1
  136. package/dist/types/src/triggers/trigger-dispatcher.d.ts +0 -74
  137. package/dist/types/src/triggers/trigger-dispatcher.d.ts.map +0 -1
  138. package/dist/types/src/triggers/trigger-dispatcher.test.d.ts +0 -2
  139. package/dist/types/src/triggers/trigger-dispatcher.test.d.ts.map +0 -1
  140. package/dist/types/src/triggers/trigger-state-store.d.ts +0 -27
  141. package/dist/types/src/triggers/trigger-state-store.d.ts.map +0 -1
  142. package/dist/types/src/types.d.ts +0 -221
  143. package/dist/types/src/types.d.ts.map +0 -1
  144. package/dist/types/src/url.d.ts +0 -21
  145. package/dist/types/src/url.d.ts.map +0 -1
  146. package/src/bundler/bundler.test.ts +0 -58
  147. package/src/bundler/bundler.ts +0 -291
  148. package/src/bundler/index.ts +0 -5
  149. package/src/edge/functions.ts +0 -67
  150. package/src/edge/index.ts +0 -9
  151. package/src/errors.ts +0 -21
  152. package/src/examples/fib.ts +0 -31
  153. package/src/examples/index.ts +0 -7
  154. package/src/examples/reply.ts +0 -19
  155. package/src/examples/sleep.ts +0 -23
  156. package/src/executor/executor.ts +0 -57
  157. package/src/handler.ts +0 -222
  158. package/src/schema.ts +0 -68
  159. package/src/services/database.ts +0 -171
  160. package/src/services/event-logger.ts +0 -118
  161. package/src/services/function-invocation-service.test.ts +0 -79
  162. package/src/services/local-function-execution.ts +0 -150
  163. package/src/services/remote-function-execution-service.ts +0 -61
  164. package/src/services/service-container.ts +0 -114
  165. package/src/services/service-registry.test.ts +0 -42
  166. package/src/services/service-registry.ts +0 -59
  167. package/src/testing/index.ts +0 -6
  168. package/src/testing/layer.ts +0 -112
  169. package/src/testing/logger.ts +0 -16
  170. package/src/testing/persist-database.test.ts +0 -87
  171. package/src/testing/services.ts +0 -115
  172. package/src/trace.ts +0 -178
  173. package/src/translations.ts +0 -20
  174. package/src/triggers/index.ts +0 -7
  175. package/src/triggers/input-builder.ts +0 -35
  176. package/src/triggers/invocation-tracer.ts +0 -99
  177. package/src/triggers/trigger-dispatcher.test.ts +0 -651
  178. package/src/triggers/trigger-dispatcher.ts +0 -522
  179. package/src/triggers/trigger-state-store.ts +0 -60
  180. package/src/types.ts +0 -214
  181. package/src/url.ts +0 -55
@@ -2,121 +2,32 @@
2
2
  // Copyright 2025 DXOS.org
3
3
  //
4
4
 
5
- import { HttpClient, HttpClientRequest } from '@effect/platform';
6
- import { type Config, Context, Effect, Layer, Redacted } from 'effect';
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 { DataType } from '@dxos/schema';
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 API key from the credential.
41
+ * Maps the request to include the given token in the Authorization header.
131
42
  */
132
- export const withAuthorization = (query: CredentialQuery, kind?: 'Bearer' | 'Basic') =>
133
- HttpClient.mapRequestEffect(
134
- Effect.fnUntraced(function* (request) {
135
- const key = yield* CredentialsService.getApiKey(query).pipe(Effect.map(Redacted.value));
136
- const authorization = kind ? `${kind} ${key}` : key;
137
- return HttpClientRequest.setHeader(request, 'Authorization', authorization);
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 { Context, Effect, Layer } from 'effect';
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 { AiService } from '@dxos/ai';
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: FunctionDefinition<I, O>, input: I): Effect.Effect<O, never, InvocationServices>;
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 invokeFunction = Effect.serviceFunctionEffect(FunctionInvocationService, (_) => _.invokeFunction);
27
-
28
- static layer = Layer.effect(
29
- FunctionInvocationService,
30
- Effect.gen(function* () {
31
- const localExecutioner = yield* LocalFunctionExecutionService;
32
- const remoteExecutioner = yield* RemoteFunctionExecutionService;
33
-
34
- return {
35
- invokeFunction: <I, O>(
36
- functionDef: FunctionDefinition<I, O>,
37
- input: I,
38
- ): Effect.Effect<O, never, InvocationServices> =>
39
- Effect.gen(function* () {
40
- if (functionDef.meta?.deployedFunctionId) {
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
  }
@@ -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 * from './service-container';
12
- export * from './tracing';
13
- export * from './remote-function-execution-service';
8
+ export { MESSAGE_PROPERTY_TOOL_CALL_ID } from './tracing';
@@ -2,81 +2,4 @@
2
2
  // Copyright 2025 DXOS.org
3
3
  //
4
4
 
5
- import { Context, Effect, Layer } from 'effect';
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';
@@ -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 DataType.Message['properties']}
6
+ * Goes into {@link Message['properties']}
140
7
  */
141
8
  export const MESSAGE_PROPERTY_TOOL_CALL_ID = 'toolCallId' as const;
@@ -0,0 +1,5 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ export * from './url';
@@ -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
+ };