openclaw-overlay-plugin 0.7.22

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 (221) hide show
  1. package/README.md +406 -0
  2. package/SKILL.md +78 -0
  3. package/clawdbot.plugin.json +106 -0
  4. package/dist/cli-main.d.ts +7 -0
  5. package/dist/cli-main.js +192 -0
  6. package/dist/cli.d.ts +8 -0
  7. package/dist/cli.js +14 -0
  8. package/dist/core/config.d.ts +11 -0
  9. package/dist/core/config.js +13 -0
  10. package/dist/core/index.d.ts +25 -0
  11. package/dist/core/index.js +26 -0
  12. package/dist/core/payment.d.ts +16 -0
  13. package/dist/core/payment.js +94 -0
  14. package/dist/core/types.d.ts +94 -0
  15. package/dist/core/types.js +4 -0
  16. package/dist/core/verify.d.ts +28 -0
  17. package/dist/core/verify.js +104 -0
  18. package/dist/core/wallet.d.ts +99 -0
  19. package/dist/core/wallet.js +219 -0
  20. package/dist/scripts/baemail/commands.d.ts +64 -0
  21. package/dist/scripts/baemail/commands.js +258 -0
  22. package/dist/scripts/baemail/handler.d.ts +36 -0
  23. package/dist/scripts/baemail/handler.js +284 -0
  24. package/dist/scripts/baemail/index.d.ts +5 -0
  25. package/dist/scripts/baemail/index.js +5 -0
  26. package/dist/scripts/config.d.ts +48 -0
  27. package/dist/scripts/config.js +68 -0
  28. package/dist/scripts/index.d.ts +7 -0
  29. package/dist/scripts/index.js +7 -0
  30. package/dist/scripts/messaging/connect.d.ts +8 -0
  31. package/dist/scripts/messaging/connect.js +114 -0
  32. package/dist/scripts/messaging/handlers.d.ts +21 -0
  33. package/dist/scripts/messaging/handlers.js +334 -0
  34. package/dist/scripts/messaging/inbox.d.ts +11 -0
  35. package/dist/scripts/messaging/inbox.js +51 -0
  36. package/dist/scripts/messaging/index.d.ts +8 -0
  37. package/dist/scripts/messaging/index.js +8 -0
  38. package/dist/scripts/messaging/poll.d.ts +7 -0
  39. package/dist/scripts/messaging/poll.js +52 -0
  40. package/dist/scripts/messaging/send.d.ts +7 -0
  41. package/dist/scripts/messaging/send.js +43 -0
  42. package/dist/scripts/output.d.ts +12 -0
  43. package/dist/scripts/output.js +19 -0
  44. package/dist/scripts/overlay/discover.d.ts +7 -0
  45. package/dist/scripts/overlay/discover.js +72 -0
  46. package/dist/scripts/overlay/index.d.ts +7 -0
  47. package/dist/scripts/overlay/index.js +7 -0
  48. package/dist/scripts/overlay/registration.d.ts +19 -0
  49. package/dist/scripts/overlay/registration.js +176 -0
  50. package/dist/scripts/overlay/services.d.ts +29 -0
  51. package/dist/scripts/overlay/services.js +167 -0
  52. package/dist/scripts/overlay/transaction.d.ts +42 -0
  53. package/dist/scripts/overlay/transaction.js +103 -0
  54. package/dist/scripts/payment/build.d.ts +24 -0
  55. package/dist/scripts/payment/build.js +54 -0
  56. package/dist/scripts/payment/commands.d.ts +15 -0
  57. package/dist/scripts/payment/commands.js +73 -0
  58. package/dist/scripts/payment/index.d.ts +6 -0
  59. package/dist/scripts/payment/index.js +6 -0
  60. package/dist/scripts/payment/types.d.ts +56 -0
  61. package/dist/scripts/payment/types.js +4 -0
  62. package/dist/scripts/services/index.d.ts +6 -0
  63. package/dist/scripts/services/index.js +6 -0
  64. package/dist/scripts/services/queue.d.ts +11 -0
  65. package/dist/scripts/services/queue.js +28 -0
  66. package/dist/scripts/services/request.d.ts +7 -0
  67. package/dist/scripts/services/request.js +82 -0
  68. package/dist/scripts/services/respond.d.ts +11 -0
  69. package/dist/scripts/services/respond.js +132 -0
  70. package/dist/scripts/types.d.ts +107 -0
  71. package/dist/scripts/types.js +4 -0
  72. package/dist/scripts/utils/index.d.ts +6 -0
  73. package/dist/scripts/utils/index.js +6 -0
  74. package/dist/scripts/utils/merkle.d.ts +12 -0
  75. package/dist/scripts/utils/merkle.js +47 -0
  76. package/dist/scripts/utils/storage.d.ts +66 -0
  77. package/dist/scripts/utils/storage.js +211 -0
  78. package/dist/scripts/utils/woc.d.ts +26 -0
  79. package/dist/scripts/utils/woc.js +91 -0
  80. package/dist/scripts/wallet/balance.d.ts +22 -0
  81. package/dist/scripts/wallet/balance.js +240 -0
  82. package/dist/scripts/wallet/identity.d.ts +70 -0
  83. package/dist/scripts/wallet/identity.js +151 -0
  84. package/dist/scripts/wallet/index.d.ts +6 -0
  85. package/dist/scripts/wallet/index.js +6 -0
  86. package/dist/scripts/wallet/setup.d.ts +15 -0
  87. package/dist/scripts/wallet/setup.js +105 -0
  88. package/dist/scripts/x-verification/commands.d.ts +27 -0
  89. package/dist/scripts/x-verification/commands.js +222 -0
  90. package/dist/scripts/x-verification/index.d.ts +4 -0
  91. package/dist/scripts/x-verification/index.js +4 -0
  92. package/dist/services/built-in/api-proxy/index.d.ts +6 -0
  93. package/dist/services/built-in/api-proxy/index.js +23 -0
  94. package/dist/services/built-in/code-develop/index.d.ts +6 -0
  95. package/dist/services/built-in/code-develop/index.js +23 -0
  96. package/dist/services/built-in/code-review/index.d.ts +10 -0
  97. package/dist/services/built-in/code-review/index.js +51 -0
  98. package/dist/services/built-in/image-analysis/index.d.ts +6 -0
  99. package/dist/services/built-in/image-analysis/index.js +33 -0
  100. package/dist/services/built-in/memory-store/index.d.ts +6 -0
  101. package/dist/services/built-in/memory-store/index.js +22 -0
  102. package/dist/services/built-in/roulette/index.d.ts +6 -0
  103. package/dist/services/built-in/roulette/index.js +27 -0
  104. package/dist/services/built-in/summarize/index.d.ts +6 -0
  105. package/dist/services/built-in/summarize/index.js +21 -0
  106. package/dist/services/built-in/tell-joke/handler.d.ts +7 -0
  107. package/dist/services/built-in/tell-joke/handler.js +122 -0
  108. package/dist/services/built-in/tell-joke/index.d.ts +9 -0
  109. package/dist/services/built-in/tell-joke/index.js +31 -0
  110. package/dist/services/built-in/translate/index.d.ts +6 -0
  111. package/dist/services/built-in/translate/index.js +21 -0
  112. package/dist/services/built-in/web-research/index.d.ts +9 -0
  113. package/dist/services/built-in/web-research/index.js +51 -0
  114. package/dist/services/index.d.ts +13 -0
  115. package/dist/services/index.js +14 -0
  116. package/dist/services/loader.d.ts +77 -0
  117. package/dist/services/loader.js +292 -0
  118. package/dist/services/manager.d.ts +86 -0
  119. package/dist/services/manager.js +255 -0
  120. package/dist/services/registry.d.ts +98 -0
  121. package/dist/services/registry.js +204 -0
  122. package/dist/services/types.d.ts +230 -0
  123. package/dist/services/types.js +30 -0
  124. package/dist/test/cli.test.d.ts +7 -0
  125. package/dist/test/cli.test.js +329 -0
  126. package/dist/test/comprehensive-overlay.test.d.ts +13 -0
  127. package/dist/test/comprehensive-overlay.test.js +593 -0
  128. package/dist/test/key-derivation.test.d.ts +12 -0
  129. package/dist/test/key-derivation.test.js +86 -0
  130. package/dist/test/overlay-submit.test.d.ts +10 -0
  131. package/dist/test/overlay-submit.test.js +460 -0
  132. package/dist/test/request-response-flow.test.d.ts +5 -0
  133. package/dist/test/request-response-flow.test.js +209 -0
  134. package/dist/test/service-system.test.d.ts +5 -0
  135. package/dist/test/service-system.test.js +190 -0
  136. package/dist/test/utils/server-logic.d.ts +98 -0
  137. package/dist/test/utils/server-logic.js +286 -0
  138. package/dist/test/wallet.test.d.ts +7 -0
  139. package/dist/test/wallet.test.js +146 -0
  140. package/index.ts +1965 -0
  141. package/openclaw.plugin.json +106 -0
  142. package/package.json +73 -0
  143. package/src/cli-main.ts +230 -0
  144. package/src/cli.ts +16 -0
  145. package/src/core/README.md +246 -0
  146. package/src/core/config.ts +21 -0
  147. package/src/core/index.ts +42 -0
  148. package/src/core/payment.ts +111 -0
  149. package/src/core/types.ts +102 -0
  150. package/src/core/verify.ts +119 -0
  151. package/src/core/wallet.ts +282 -0
  152. package/src/scripts/baemail/commands.ts +326 -0
  153. package/src/scripts/baemail/handler.ts +338 -0
  154. package/src/scripts/baemail/index.ts +6 -0
  155. package/src/scripts/config.ts +81 -0
  156. package/src/scripts/index.ts +8 -0
  157. package/src/scripts/messaging/connect.ts +121 -0
  158. package/src/scripts/messaging/handlers.ts +394 -0
  159. package/src/scripts/messaging/inbox.ts +64 -0
  160. package/src/scripts/messaging/index.ts +9 -0
  161. package/src/scripts/messaging/poll.ts +59 -0
  162. package/src/scripts/messaging/send.ts +54 -0
  163. package/src/scripts/output.ts +21 -0
  164. package/src/scripts/overlay/discover.ts +81 -0
  165. package/src/scripts/overlay/index.ts +8 -0
  166. package/src/scripts/overlay/registration.ts +199 -0
  167. package/src/scripts/overlay/services.ts +199 -0
  168. package/src/scripts/overlay/transaction.ts +124 -0
  169. package/src/scripts/payment/build.ts +65 -0
  170. package/src/scripts/payment/commands.ts +92 -0
  171. package/src/scripts/payment/index.ts +7 -0
  172. package/src/scripts/payment/types.ts +62 -0
  173. package/src/scripts/services/index.ts +7 -0
  174. package/src/scripts/services/queue.ts +35 -0
  175. package/src/scripts/services/request.ts +98 -0
  176. package/src/scripts/services/respond.ts +149 -0
  177. package/src/scripts/types.ts +121 -0
  178. package/src/scripts/utils/index.ts +7 -0
  179. package/src/scripts/utils/merkle.ts +57 -0
  180. package/src/scripts/utils/storage.ts +231 -0
  181. package/src/scripts/utils/woc.ts +106 -0
  182. package/src/scripts/wallet/balance.ts +277 -0
  183. package/src/scripts/wallet/identity.ts +203 -0
  184. package/src/scripts/wallet/index.ts +7 -0
  185. package/src/scripts/wallet/setup.ts +121 -0
  186. package/src/scripts/x-verification/commands.ts +256 -0
  187. package/src/scripts/x-verification/index.ts +5 -0
  188. package/src/services/built-in/api-proxy/index.ts +26 -0
  189. package/src/services/built-in/api-proxy/prompt.md +26 -0
  190. package/src/services/built-in/code-develop/index.ts +26 -0
  191. package/src/services/built-in/code-develop/prompt.md +35 -0
  192. package/src/services/built-in/code-review/index.ts +54 -0
  193. package/src/services/built-in/code-review/prompt.md +105 -0
  194. package/src/services/built-in/image-analysis/index.ts +36 -0
  195. package/src/services/built-in/image-analysis/prompt.md +42 -0
  196. package/src/services/built-in/memory-store/index.ts +25 -0
  197. package/src/services/built-in/memory-store/prompt.md +45 -0
  198. package/src/services/built-in/roulette/index.ts +30 -0
  199. package/src/services/built-in/roulette/prompt.md +35 -0
  200. package/src/services/built-in/summarize/index.ts +24 -0
  201. package/src/services/built-in/summarize/prompt.md +27 -0
  202. package/src/services/built-in/tell-joke/handler.ts +134 -0
  203. package/src/services/built-in/tell-joke/index.ts +34 -0
  204. package/src/services/built-in/tell-joke/prompt.md +59 -0
  205. package/src/services/built-in/translate/index.ts +24 -0
  206. package/src/services/built-in/translate/prompt.md +23 -0
  207. package/src/services/built-in/web-research/index.ts +54 -0
  208. package/src/services/built-in/web-research/prompt.md +110 -0
  209. package/src/services/index.ts +16 -0
  210. package/src/services/loader.ts +344 -0
  211. package/src/services/manager.ts +304 -0
  212. package/src/services/registry.ts +246 -0
  213. package/src/services/types.ts +259 -0
  214. package/src/test/cli.test.ts +352 -0
  215. package/src/test/comprehensive-overlay.test.ts +729 -0
  216. package/src/test/key-derivation.test.ts +102 -0
  217. package/src/test/overlay-submit.test.ts +570 -0
  218. package/src/test/request-response-flow.test.ts +252 -0
  219. package/src/test/service-system.test.ts +241 -0
  220. package/src/test/utils/server-logic.ts +368 -0
  221. package/src/test/wallet.test.ts +166 -0
@@ -0,0 +1,292 @@
1
+ /**
2
+ * Service loader implementation.
3
+ *
4
+ * This dynamically loads services from directories, supporting both
5
+ * built-in services and custom user services.
6
+ */
7
+ import fs from 'node:fs';
8
+ import path from 'node:path';
9
+ import { fileURLToPath } from 'node:url';
10
+ const __filename = fileURLToPath(import.meta.url);
11
+ const __dirname = path.dirname(__filename);
12
+ /**
13
+ * Default service loader implementation.
14
+ */
15
+ export class DefaultServiceLoader {
16
+ builtInDir;
17
+ customDir;
18
+ constructor() {
19
+ // Built-in services directory
20
+ this.builtInDir = path.resolve(__dirname, 'built-in');
21
+ // Custom services directory (in user's config dir)
22
+ const homeDir = process.env.HOME || process.env.USERPROFILE || '';
23
+ this.customDir = path.join(homeDir, '.openclaw', 'services');
24
+ }
25
+ /**
26
+ * Load services from a directory.
27
+ */
28
+ async loadFromDirectory(directory) {
29
+ if (!fs.existsSync(directory)) {
30
+ return [];
31
+ }
32
+ const services = [];
33
+ const entries = fs.readdirSync(directory, { withFileTypes: true });
34
+ for (const entry of entries) {
35
+ if (!entry.isDirectory()) {
36
+ continue;
37
+ }
38
+ try {
39
+ const serviceDefinition = await this.loadServiceFromDirectory(path.join(directory, entry.name));
40
+ if (serviceDefinition) {
41
+ services.push(serviceDefinition);
42
+ }
43
+ }
44
+ catch (error) {
45
+ console.warn(`Failed to load service from ${entry.name}:`, error);
46
+ }
47
+ }
48
+ return services;
49
+ }
50
+ /**
51
+ * Load all built-in services.
52
+ */
53
+ async loadBuiltInServices() {
54
+ return this.loadFromDirectory(this.builtInDir);
55
+ }
56
+ /**
57
+ * Load custom user services.
58
+ */
59
+ async loadCustomServices() {
60
+ return this.loadFromDirectory(this.customDir);
61
+ }
62
+ /**
63
+ * Load all services (built-in + custom).
64
+ */
65
+ async loadAllServices() {
66
+ const [builtIn, custom] = await Promise.all([
67
+ this.loadBuiltInServices(),
68
+ this.loadCustomServices()
69
+ ]);
70
+ return [...builtIn, ...custom];
71
+ }
72
+ /**
73
+ * Load a service definition from a directory.
74
+ */
75
+ async loadServiceFromDirectory(serviceDir) {
76
+ const indexPath = path.join(serviceDir, 'index.ts');
77
+ const jsIndexPath = path.join(serviceDir, 'index.js');
78
+ const promptPath = path.join(serviceDir, 'prompt.md');
79
+ // Check if index file exists (TypeScript or JavaScript)
80
+ let modulePath;
81
+ if (fs.existsSync(indexPath)) {
82
+ modulePath = indexPath;
83
+ }
84
+ else if (fs.existsSync(jsIndexPath)) {
85
+ modulePath = jsIndexPath;
86
+ }
87
+ else {
88
+ throw new Error(`No index.ts or index.js found in ${serviceDir}`);
89
+ }
90
+ try {
91
+ // Dynamic import the service module
92
+ const serviceModule = await import(modulePath);
93
+ const serviceDefinition = serviceModule.default || serviceModule;
94
+ if (!serviceDefinition || typeof serviceDefinition !== 'object') {
95
+ throw new Error('Service must export a default ServiceDefinition object');
96
+ }
97
+ // Validate required fields
98
+ if (!serviceDefinition.id) {
99
+ throw new Error('Service definition must have an id');
100
+ }
101
+ // Add prompt file path if it exists
102
+ if (fs.existsSync(promptPath)) {
103
+ serviceDefinition.promptFile = promptPath;
104
+ }
105
+ return serviceDefinition;
106
+ }
107
+ catch (error) {
108
+ throw new Error(`Failed to import service from ${modulePath}: ${error}`);
109
+ }
110
+ }
111
+ /**
112
+ * Create a new custom service directory.
113
+ */
114
+ createCustomServiceDirectory(serviceId) {
115
+ const serviceDir = path.join(this.customDir, serviceId);
116
+ // Ensure custom services directory exists
117
+ fs.mkdirSync(this.customDir, { recursive: true });
118
+ // Create service directory
119
+ if (fs.existsSync(serviceDir)) {
120
+ throw new Error(`Service directory ${serviceId} already exists`);
121
+ }
122
+ fs.mkdirSync(serviceDir);
123
+ return serviceDir;
124
+ }
125
+ /**
126
+ * Create a basic service template.
127
+ */
128
+ createServiceTemplate(serviceId, options) {
129
+ const serviceDir = this.createCustomServiceDirectory(serviceId);
130
+ // Create index.ts
131
+ const indexContent = this.generateServiceTemplate(serviceId, options);
132
+ fs.writeFileSync(path.join(serviceDir, 'index.ts'), indexContent);
133
+ // Create prompt.md
134
+ const promptContent = this.generatePromptTemplate(serviceId, options);
135
+ fs.writeFileSync(path.join(serviceDir, 'prompt.md'), promptContent);
136
+ // Create handler.ts if requested
137
+ if (options.hasHandler) {
138
+ const handlerContent = this.generateHandlerTemplate(serviceId, options);
139
+ fs.writeFileSync(path.join(serviceDir, 'handler.ts'), handlerContent);
140
+ }
141
+ }
142
+ /**
143
+ * Generate service definition template.
144
+ */
145
+ generateServiceTemplate(serviceId, options) {
146
+ return `/**
147
+ * ${options.name} service definition.
148
+ */
149
+
150
+ import { ServiceDefinition${options.hasHandler ? ', ServiceHandler' : ''} } from '../../types.js';
151
+ ${options.hasHandler ? `import { ${serviceId.replace(/-/g, '')}Handler } from './handler.js';` : ''}
152
+
153
+ const ${serviceId.replace(/-/g, '')}Service: ServiceDefinition = {
154
+ id: '${serviceId}',
155
+ name: '${options.name}',
156
+ description: '${options.description}',
157
+ defaultPrice: ${options.defaultPrice},${options.category ? `\n category: '${options.category}',` : ''}
158
+ inputSchema: {
159
+ type: 'object',
160
+ properties: {
161
+ // Define your input schema here
162
+ query: {
163
+ type: 'string',
164
+ description: 'Query or input for the service'
165
+ }
166
+ },
167
+ required: ['query']
168
+ }${options.hasHandler ? `,\n handler: ${serviceId.replace(/-/g, '')}Handler` : ''}
169
+ };
170
+
171
+ export default ${serviceId.replace(/-/g, '')}Service;
172
+ `;
173
+ }
174
+ /**
175
+ * Generate prompt template.
176
+ */
177
+ generatePromptTemplate(serviceId, options) {
178
+ return `# ${options.name} Service
179
+
180
+ You are processing a request for the "${serviceId}" service.
181
+
182
+ ## Service Description
183
+ ${options.description}
184
+
185
+ ## Input
186
+ The user has provided the following input:
187
+ \`\`\`json
188
+ {{input}}
189
+ \`\`\`
190
+
191
+ ## Instructions
192
+ Process the user's request and provide a helpful response based on the service description.
193
+ Format your response as a structured result that can be easily parsed and used.
194
+
195
+ ## Response Format
196
+ Provide your response in this format:
197
+ \`\`\`json
198
+ {
199
+ "result": "your processed result here",
200
+ "metadata": {
201
+ "processingTime": "time taken",
202
+ "version": "1.0"
203
+ }
204
+ }
205
+ \`\`\`
206
+ `;
207
+ }
208
+ /**
209
+ * Generate handler template.
210
+ */
211
+ generateHandlerTemplate(serviceId, options) {
212
+ return `/**
213
+ * ${options.name} service handler.
214
+ */
215
+
216
+ import { ServiceHandler, ValidationResult, ServiceContext, ServiceResult } from '../../types.js';
217
+
218
+ export const ${serviceId.replace(/-/g, '')}Handler: ServiceHandler = {
219
+ /**
220
+ * Validate service input.
221
+ */
222
+ validate(input: any): ValidationResult {
223
+ if (!input || typeof input !== 'object') {
224
+ return { valid: false, error: 'Input must be an object' };
225
+ }
226
+
227
+ if (!input.query || typeof input.query !== 'string') {
228
+ return { valid: false, error: 'Query must be a non-empty string' };
229
+ }
230
+
231
+ // Add more validation as needed
232
+ return { valid: true, sanitized: input };
233
+ },
234
+
235
+ /**
236
+ * Process the service request.
237
+ */
238
+ async process(input: any, context: ServiceContext): Promise<ServiceResult> {
239
+ try {
240
+ const startTime = Date.now();
241
+
242
+ // Your service logic here
243
+ const result = {
244
+ query: input.query,
245
+ response: 'This is a template response. Implement your logic here.',
246
+ timestamp: new Date().toISOString()
247
+ };
248
+
249
+ const processingTime = Date.now() - startTime;
250
+
251
+ return {
252
+ success: true,
253
+ data: result,
254
+ metadata: {
255
+ processingTime,
256
+ version: '1.0'
257
+ }
258
+ };
259
+ } catch (error) {
260
+ return {
261
+ success: false,
262
+ error: error instanceof Error ? error.message : String(error)
263
+ };
264
+ }
265
+ }
266
+ };
267
+ `;
268
+ }
269
+ /**
270
+ * Get the built-in services directory.
271
+ */
272
+ getBuiltInDirectory() {
273
+ return this.builtInDir;
274
+ }
275
+ /**
276
+ * Get the custom services directory.
277
+ */
278
+ getCustomDirectory() {
279
+ return this.customDir;
280
+ }
281
+ /**
282
+ * Check if a service directory exists.
283
+ */
284
+ serviceExists(serviceId, inCustom = false) {
285
+ const baseDir = inCustom ? this.customDir : this.builtInDir;
286
+ return fs.existsSync(path.join(baseDir, serviceId));
287
+ }
288
+ }
289
+ /**
290
+ * Global service loader instance.
291
+ */
292
+ export const serviceLoader = new DefaultServiceLoader();
@@ -0,0 +1,86 @@
1
+ /**
2
+ * Service manager implementation.
3
+ *
4
+ * This orchestrates the service system, providing a high-level interface
5
+ * for service execution while keeping payment and relay logic separate.
6
+ */
7
+ import { ServiceManager, ServiceDefinition, ServiceContext, ServiceResult, ValidationResult } from './types.js';
8
+ import { DefaultServiceRegistry } from './registry.js';
9
+ import { DefaultServiceLoader } from './loader.js';
10
+ /**
11
+ * Default service manager implementation.
12
+ */
13
+ export declare class DefaultServiceManager implements ServiceManager {
14
+ readonly registry: DefaultServiceRegistry;
15
+ readonly loader: DefaultServiceLoader;
16
+ private initialized;
17
+ constructor();
18
+ /**
19
+ * Initialize the service system.
20
+ */
21
+ initialize(): Promise<void>;
22
+ /**
23
+ * Execute a service request.
24
+ */
25
+ execute(serviceId: string, input: any, context: ServiceContext): Promise<ServiceResult>;
26
+ /**
27
+ * Validate service input.
28
+ */
29
+ validate(serviceId: string, input: any): ValidationResult;
30
+ /**
31
+ * Reload all services.
32
+ */
33
+ reload(): Promise<void>;
34
+ /**
35
+ * Get service for agent-mode processing.
36
+ */
37
+ getServiceForAgentMode(serviceId: string): {
38
+ service: ServiceDefinition;
39
+ promptFile?: string;
40
+ } | null;
41
+ /**
42
+ * Check if service is available.
43
+ */
44
+ isServiceAvailable(serviceId: string): boolean;
45
+ /**
46
+ * Get service execution mode.
47
+ */
48
+ getServiceMode(serviceId: string): 'handler' | 'agent' | null;
49
+ /**
50
+ * Get all available services for discovery.
51
+ */
52
+ getAvailableServices(): Array<{
53
+ id: string;
54
+ name: string;
55
+ description: string;
56
+ defaultPrice: number;
57
+ category?: string;
58
+ mode: 'handler' | 'agent';
59
+ }>;
60
+ /**
61
+ * Validate input against JSON schema.
62
+ */
63
+ private validateAgainstSchema;
64
+ /**
65
+ * Get service statistics.
66
+ */
67
+ getStatistics(): {
68
+ totalServices: number;
69
+ handlerServices: number;
70
+ agentServices: number;
71
+ servicesByCategory: Record<string, number>;
72
+ };
73
+ }
74
+ /**
75
+ * Global service manager instance.
76
+ */
77
+ export declare const serviceManager: DefaultServiceManager;
78
+ /**
79
+ * Initialize the service system.
80
+ * This should be called once during application startup.
81
+ */
82
+ export declare function initializeServiceSystem(): Promise<void>;
83
+ /**
84
+ * Get service manager instance.
85
+ */
86
+ export declare function getServiceManager(): DefaultServiceManager;
@@ -0,0 +1,255 @@
1
+ /**
2
+ * Service manager implementation.
3
+ *
4
+ * This orchestrates the service system, providing a high-level interface
5
+ * for service execution while keeping payment and relay logic separate.
6
+ */
7
+ import { serviceRegistry } from './registry.js';
8
+ import { serviceLoader } from './loader.js';
9
+ /**
10
+ * Default service manager implementation.
11
+ */
12
+ export class DefaultServiceManager {
13
+ registry;
14
+ loader;
15
+ initialized = false;
16
+ constructor() {
17
+ this.registry = serviceRegistry;
18
+ this.loader = serviceLoader;
19
+ }
20
+ /**
21
+ * Initialize the service system.
22
+ */
23
+ async initialize() {
24
+ if (this.initialized) {
25
+ return;
26
+ }
27
+ // Load all services
28
+ const services = await this.loader.loadAllServices();
29
+ // Register all services
30
+ for (const service of services) {
31
+ try {
32
+ this.registry.register(service);
33
+ }
34
+ catch (error) {
35
+ console.warn(`Failed to register service '${service.id}':`, error);
36
+ }
37
+ }
38
+ this.initialized = true;
39
+ console.log(`Service manager initialized with ${this.registry.count()} services`);
40
+ }
41
+ /**
42
+ * Execute a service request.
43
+ */
44
+ async execute(serviceId, input, context) {
45
+ if (!this.initialized) {
46
+ throw new Error('Service manager not initialized');
47
+ }
48
+ const service = this.registry.get(serviceId);
49
+ if (!service) {
50
+ return {
51
+ success: false,
52
+ error: `Service '${serviceId}' not found`
53
+ };
54
+ }
55
+ // Validate input if service has a handler
56
+ if (service.handler) {
57
+ const validation = service.handler.validate(input);
58
+ if (!validation.valid) {
59
+ return {
60
+ success: false,
61
+ error: `Input validation failed: ${validation.error}`
62
+ };
63
+ }
64
+ // Use sanitized input if provided
65
+ input = validation.sanitized || input;
66
+ }
67
+ // Execute the service
68
+ try {
69
+ if (service.handler) {
70
+ // Use custom handler
71
+ return await service.handler.process(input, context);
72
+ }
73
+ else {
74
+ // Service uses agent mode - return success with input for agent processing
75
+ return {
76
+ success: true,
77
+ data: {
78
+ serviceId,
79
+ input,
80
+ mode: 'agent',
81
+ promptFile: service.promptFile
82
+ },
83
+ metadata: {
84
+ version: '1.0',
85
+ executionMode: 'agent'
86
+ }
87
+ };
88
+ }
89
+ }
90
+ catch (error) {
91
+ return {
92
+ success: false,
93
+ error: error instanceof Error ? error.message : String(error),
94
+ metadata: {
95
+ serviceId,
96
+ errorType: 'execution_error'
97
+ }
98
+ };
99
+ }
100
+ }
101
+ /**
102
+ * Validate service input.
103
+ */
104
+ validate(serviceId, input) {
105
+ const service = this.registry.get(serviceId);
106
+ if (!service) {
107
+ return {
108
+ valid: false,
109
+ error: `Service '${serviceId}' not found`
110
+ };
111
+ }
112
+ // Use service handler validation if available
113
+ if (service.handler) {
114
+ return service.handler.validate(input);
115
+ }
116
+ // Default validation for agent-mode services
117
+ if (service.inputSchema) {
118
+ return this.validateAgainstSchema(input, service.inputSchema);
119
+ }
120
+ // No validation required
121
+ return { valid: true };
122
+ }
123
+ /**
124
+ * Reload all services.
125
+ */
126
+ async reload() {
127
+ // Clear existing services
128
+ this.registry.clear();
129
+ this.initialized = false;
130
+ // Reinitialize
131
+ await this.initialize();
132
+ }
133
+ /**
134
+ * Get service for agent-mode processing.
135
+ */
136
+ getServiceForAgentMode(serviceId) {
137
+ const service = this.registry.get(serviceId);
138
+ if (!service) {
139
+ return null;
140
+ }
141
+ return {
142
+ service,
143
+ promptFile: service.promptFile
144
+ };
145
+ }
146
+ /**
147
+ * Check if service is available.
148
+ */
149
+ isServiceAvailable(serviceId) {
150
+ return this.registry.has(serviceId);
151
+ }
152
+ /**
153
+ * Get service execution mode.
154
+ */
155
+ getServiceMode(serviceId) {
156
+ const service = this.registry.get(serviceId);
157
+ if (!service) {
158
+ return null;
159
+ }
160
+ return service.handler ? 'handler' : 'agent';
161
+ }
162
+ /**
163
+ * Get all available services for discovery.
164
+ */
165
+ getAvailableServices() {
166
+ return this.registry.list().map(service => ({
167
+ id: service.id,
168
+ name: service.name,
169
+ description: service.description,
170
+ defaultPrice: service.defaultPrice,
171
+ category: service.category,
172
+ mode: service.handler ? 'handler' : 'agent'
173
+ }));
174
+ }
175
+ /**
176
+ * Validate input against JSON schema.
177
+ */
178
+ validateAgainstSchema(input, schema) {
179
+ // Simple schema validation - in production, you might use a library like ajv
180
+ try {
181
+ const schemaObj = schema;
182
+ if (schemaObj.type === 'object') {
183
+ if (!input || typeof input !== 'object') {
184
+ return { valid: false, error: 'Input must be an object' };
185
+ }
186
+ // Check required properties
187
+ if (schemaObj.required && Array.isArray(schemaObj.required)) {
188
+ for (const requiredProp of schemaObj.required) {
189
+ if (!(requiredProp in input)) {
190
+ return { valid: false, error: `Missing required property: ${requiredProp}` };
191
+ }
192
+ }
193
+ }
194
+ // Basic type checking for properties
195
+ if (schemaObj.properties) {
196
+ for (const [propName, propSchema] of Object.entries(schemaObj.properties)) {
197
+ if (propName in input) {
198
+ const propType = propSchema.type;
199
+ const actualType = typeof input[propName];
200
+ if (propType && propType !== actualType) {
201
+ return { valid: false, error: `Property '${propName}' must be of type ${propType}` };
202
+ }
203
+ }
204
+ }
205
+ }
206
+ }
207
+ return { valid: true, sanitized: input };
208
+ }
209
+ catch (error) {
210
+ return { valid: false, error: `Schema validation error: ${error}` };
211
+ }
212
+ }
213
+ /**
214
+ * Get service statistics.
215
+ */
216
+ getStatistics() {
217
+ const services = this.registry.list();
218
+ const stats = {
219
+ totalServices: services.length,
220
+ handlerServices: 0,
221
+ agentServices: 0,
222
+ servicesByCategory: {}
223
+ };
224
+ for (const service of services) {
225
+ // Count by execution mode
226
+ if (service.handler) {
227
+ stats.handlerServices++;
228
+ }
229
+ else {
230
+ stats.agentServices++;
231
+ }
232
+ // Count by category
233
+ const category = service.category || 'uncategorized';
234
+ stats.servicesByCategory[category] = (stats.servicesByCategory[category] || 0) + 1;
235
+ }
236
+ return stats;
237
+ }
238
+ }
239
+ /**
240
+ * Global service manager instance.
241
+ */
242
+ export const serviceManager = new DefaultServiceManager();
243
+ /**
244
+ * Initialize the service system.
245
+ * This should be called once during application startup.
246
+ */
247
+ export async function initializeServiceSystem() {
248
+ await serviceManager.initialize();
249
+ }
250
+ /**
251
+ * Get service manager instance.
252
+ */
253
+ export function getServiceManager() {
254
+ return serviceManager;
255
+ }