opc-agent 1.1.3 → 1.2.1

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 (156) hide show
  1. package/CHANGELOG.md +6 -0
  2. package/CONTRIBUTING.md +75 -75
  3. package/README.md +429 -429
  4. package/README.zh-CN.md +415 -415
  5. package/dist/channels/web.js +256 -256
  6. package/dist/core/streaming.d.ts +56 -0
  7. package/dist/core/streaming.js +160 -0
  8. package/dist/deploy/hermes.js +22 -22
  9. package/dist/deploy/openclaw.js +31 -31
  10. package/dist/index.d.ts +4 -0
  11. package/dist/index.js +7 -1
  12. package/dist/providers/index.d.ts +1 -1
  13. package/dist/providers/index.js +13 -148
  14. package/dist/schema/oad.d.ts +3 -3
  15. package/dist/templates/code-reviewer.js +5 -5
  16. package/dist/templates/customer-service.js +2 -2
  17. package/dist/templates/data-analyst.js +5 -5
  18. package/dist/templates/knowledge-base.js +2 -2
  19. package/dist/templates/sales-assistant.js +4 -4
  20. package/dist/templates/teacher.js +6 -6
  21. package/dist/tools/gateway.d.ts +28 -0
  22. package/dist/tools/gateway.js +177 -0
  23. package/docs/.vitepress/config.ts +103 -103
  24. package/docs/api/cli.md +48 -48
  25. package/docs/api/oad-schema.md +64 -64
  26. package/docs/api/sdk.md +80 -80
  27. package/docs/guide/concepts.md +51 -51
  28. package/docs/guide/configuration.md +79 -79
  29. package/docs/guide/deployment.md +42 -42
  30. package/docs/guide/getting-started.md +44 -44
  31. package/docs/guide/templates.md +28 -28
  32. package/docs/guide/testing.md +84 -84
  33. package/docs/index.md +27 -27
  34. package/docs/zh/api/cli.md +54 -54
  35. package/docs/zh/api/oad-schema.md +87 -87
  36. package/docs/zh/api/sdk.md +102 -102
  37. package/docs/zh/guide/concepts.md +104 -104
  38. package/docs/zh/guide/configuration.md +135 -135
  39. package/docs/zh/guide/deployment.md +81 -81
  40. package/docs/zh/guide/getting-started.md +82 -82
  41. package/docs/zh/guide/templates.md +84 -84
  42. package/docs/zh/guide/testing.md +88 -88
  43. package/docs/zh/index.md +27 -27
  44. package/examples/customer-service-demo/README.md +90 -90
  45. package/examples/customer-service-demo/oad.yaml +107 -107
  46. package/package.json +1 -1
  47. package/src/analytics/index.ts +66 -66
  48. package/src/channels/discord.ts +192 -192
  49. package/src/channels/email.ts +177 -177
  50. package/src/channels/feishu.ts +236 -236
  51. package/src/channels/index.ts +15 -15
  52. package/src/channels/slack.ts +160 -160
  53. package/src/channels/telegram.ts +90 -90
  54. package/src/channels/voice.ts +106 -106
  55. package/src/channels/web.ts +17 -17
  56. package/src/channels/webhook.ts +199 -199
  57. package/src/channels/websocket.ts +87 -87
  58. package/src/channels/wechat.ts +149 -149
  59. package/src/core/a2a.ts +143 -143
  60. package/src/core/agent.ts +152 -152
  61. package/src/core/analytics-engine.ts +186 -186
  62. package/src/core/auth.ts +57 -57
  63. package/src/core/cache.ts +141 -141
  64. package/src/core/compose.ts +77 -77
  65. package/src/core/config.ts +14 -14
  66. package/src/core/errors.ts +148 -148
  67. package/src/core/hitl.ts +138 -138
  68. package/src/core/knowledge.ts +49 -4
  69. package/src/core/logger.ts +57 -57
  70. package/src/core/orchestrator.ts +215 -215
  71. package/src/core/performance.ts +187 -187
  72. package/src/core/rate-limiter.ts +128 -128
  73. package/src/core/room.ts +109 -109
  74. package/src/core/runtime.ts +152 -152
  75. package/src/core/sandbox.ts +101 -101
  76. package/src/core/security.ts +171 -171
  77. package/src/core/streaming.ts +195 -0
  78. package/src/core/types.ts +68 -68
  79. package/src/core/versioning.ts +106 -106
  80. package/src/core/watch.ts +178 -178
  81. package/src/core/workflow.ts +235 -235
  82. package/src/deploy/hermes.ts +156 -156
  83. package/src/deploy/openclaw.ts +200 -200
  84. package/src/dtv/data.ts +29 -29
  85. package/src/dtv/trust.ts +43 -43
  86. package/src/dtv/value.ts +47 -47
  87. package/src/i18n/index.ts +216 -216
  88. package/src/index.ts +6 -0
  89. package/src/marketplace/index.ts +223 -223
  90. package/src/memory/deepbrain.ts +108 -108
  91. package/src/memory/index.ts +34 -34
  92. package/src/plugins/index.ts +208 -208
  93. package/src/providers/index.ts +12 -3
  94. package/src/schema/oad.ts +155 -155
  95. package/src/skills/base.ts +16 -16
  96. package/src/skills/document.ts +100 -100
  97. package/src/skills/http.ts +35 -35
  98. package/src/skills/index.ts +27 -27
  99. package/src/skills/scheduler.ts +80 -80
  100. package/src/skills/webhook-trigger.ts +59 -59
  101. package/src/templates/code-reviewer.ts +34 -34
  102. package/src/templates/customer-service.ts +80 -80
  103. package/src/templates/data-analyst.ts +70 -70
  104. package/src/templates/executive-assistant.ts +71 -71
  105. package/src/templates/financial-advisor.ts +60 -60
  106. package/src/templates/knowledge-base.ts +31 -31
  107. package/src/templates/legal-assistant.ts +71 -71
  108. package/src/templates/sales-assistant.ts +79 -79
  109. package/src/templates/teacher.ts +79 -79
  110. package/src/testing/index.ts +181 -181
  111. package/src/tools/calculator.ts +73 -73
  112. package/src/tools/datetime.ts +149 -149
  113. package/src/tools/gateway.ts +220 -0
  114. package/src/tools/json-transform.ts +187 -187
  115. package/src/tools/mcp.ts +76 -76
  116. package/src/tools/text-analysis.ts +116 -116
  117. package/templates/Dockerfile +15 -15
  118. package/templates/code-reviewer/README.md +27 -27
  119. package/templates/code-reviewer/oad.yaml +41 -41
  120. package/templates/customer-service/README.md +22 -22
  121. package/templates/customer-service/oad.yaml +36 -36
  122. package/templates/docker-compose.yml +21 -21
  123. package/templates/ecommerce-assistant/README.md +45 -0
  124. package/templates/ecommerce-assistant/oad.yaml +47 -0
  125. package/templates/knowledge-base/README.md +28 -28
  126. package/templates/knowledge-base/oad.yaml +38 -38
  127. package/templates/sales-assistant/README.md +26 -26
  128. package/templates/sales-assistant/oad.yaml +43 -43
  129. package/templates/tech-support/README.md +43 -0
  130. package/templates/tech-support/oad.yaml +45 -0
  131. package/tests/a2a.test.ts +66 -66
  132. package/tests/agent.test.ts +72 -72
  133. package/tests/analytics.test.ts +50 -50
  134. package/tests/channel.test.ts +39 -39
  135. package/tests/e2e.test.ts +134 -134
  136. package/tests/errors.test.ts +83 -83
  137. package/tests/gateway.test.ts +71 -0
  138. package/tests/hitl.test.ts +71 -71
  139. package/tests/i18n.test.ts +41 -41
  140. package/tests/mcp.test.ts +54 -54
  141. package/tests/oad.test.ts +68 -68
  142. package/tests/performance.test.ts +115 -115
  143. package/tests/plugin.test.ts +74 -74
  144. package/tests/room.test.ts +106 -106
  145. package/tests/runtime.test.ts +42 -42
  146. package/tests/sandbox.test.ts +46 -46
  147. package/tests/security.test.ts +60 -60
  148. package/tests/streaming.test.ts +109 -0
  149. package/tests/templates.test.ts +77 -77
  150. package/tests/v070.test.ts +76 -76
  151. package/tests/versioning.test.ts +75 -75
  152. package/tests/voice.test.ts +61 -61
  153. package/tests/webhook.test.ts +29 -29
  154. package/tests/workflow.test.ts +143 -143
  155. package/tsconfig.json +19 -19
  156. package/vitest.config.ts +9 -9
package/src/core/auth.ts CHANGED
@@ -1,57 +1,57 @@
1
- import type { Request, Response, NextFunction } from 'express';
2
-
3
- export interface AuthConfig {
4
- apiKeys: string[];
5
- sessionIsolation?: boolean;
6
- }
7
-
8
- export interface AuthSession {
9
- apiKey: string;
10
- userId: string;
11
- createdAt: number;
12
- }
13
-
14
- const sessions = new Map<string, AuthSession>();
15
-
16
- export function createAuthMiddleware(config: AuthConfig) {
17
- return (req: Request, res: Response, next: NextFunction): void => {
18
- // Skip auth for non-API routes and health/metrics
19
- if (!req.path.startsWith('/api/') || req.path === '/api/health' || req.path === '/api/metrics') {
20
- next();
21
- return;
22
- }
23
-
24
- const apiKey = req.headers['x-api-key'] as string
25
- ?? req.headers['authorization']?.replace(/^Bearer\s+/i, '')
26
- ?? (req.query as any).apiKey;
27
-
28
- if (!apiKey || !config.apiKeys.includes(apiKey)) {
29
- res.status(401).json({ error: 'Unauthorized. Provide a valid API key via X-API-Key header, Bearer token, or ?apiKey query.' });
30
- return;
31
- }
32
-
33
- // Derive userId from API key for session isolation
34
- const userId = `user_${hashKey(apiKey)}`;
35
- if (!sessions.has(apiKey)) {
36
- sessions.set(apiKey, { apiKey, userId, createdAt: Date.now() });
37
- }
38
-
39
- // Attach user info to request
40
- (req as any).userId = userId;
41
- (req as any).sessionPrefix = config.sessionIsolation ? `${userId}:` : '';
42
-
43
- next();
44
- };
45
- }
46
-
47
- function hashKey(key: string): string {
48
- let h = 0;
49
- for (let i = 0; i < key.length; i++) {
50
- h = ((h << 5) - h + key.charCodeAt(i)) | 0;
51
- }
52
- return Math.abs(h).toString(36);
53
- }
54
-
55
- export function getActiveSessions(): AuthSession[] {
56
- return Array.from(sessions.values());
57
- }
1
+ import type { Request, Response, NextFunction } from 'express';
2
+
3
+ export interface AuthConfig {
4
+ apiKeys: string[];
5
+ sessionIsolation?: boolean;
6
+ }
7
+
8
+ export interface AuthSession {
9
+ apiKey: string;
10
+ userId: string;
11
+ createdAt: number;
12
+ }
13
+
14
+ const sessions = new Map<string, AuthSession>();
15
+
16
+ export function createAuthMiddleware(config: AuthConfig) {
17
+ return (req: Request, res: Response, next: NextFunction): void => {
18
+ // Skip auth for non-API routes and health/metrics
19
+ if (!req.path.startsWith('/api/') || req.path === '/api/health' || req.path === '/api/metrics') {
20
+ next();
21
+ return;
22
+ }
23
+
24
+ const apiKey = req.headers['x-api-key'] as string
25
+ ?? req.headers['authorization']?.replace(/^Bearer\s+/i, '')
26
+ ?? (req.query as any).apiKey;
27
+
28
+ if (!apiKey || !config.apiKeys.includes(apiKey)) {
29
+ res.status(401).json({ error: 'Unauthorized. Provide a valid API key via X-API-Key header, Bearer token, or ?apiKey query.' });
30
+ return;
31
+ }
32
+
33
+ // Derive userId from API key for session isolation
34
+ const userId = `user_${hashKey(apiKey)}`;
35
+ if (!sessions.has(apiKey)) {
36
+ sessions.set(apiKey, { apiKey, userId, createdAt: Date.now() });
37
+ }
38
+
39
+ // Attach user info to request
40
+ (req as any).userId = userId;
41
+ (req as any).sessionPrefix = config.sessionIsolation ? `${userId}:` : '';
42
+
43
+ next();
44
+ };
45
+ }
46
+
47
+ function hashKey(key: string): string {
48
+ let h = 0;
49
+ for (let i = 0; i < key.length; i++) {
50
+ h = ((h << 5) - h + key.charCodeAt(i)) | 0;
51
+ }
52
+ return Math.abs(h).toString(36);
53
+ }
54
+
55
+ export function getActiveSessions(): AuthSession[] {
56
+ return Array.from(sessions.values());
57
+ }
package/src/core/cache.ts CHANGED
@@ -1,141 +1,141 @@
1
- /**
2
- * Caching Layer - Cache LLM responses with configurable TTL.
3
- * Hash-based key from input messages + system prompt.
4
- */
5
- import * as fs from 'fs';
6
- import * as path from 'path';
7
- import * as crypto from 'crypto';
8
-
9
- export interface CacheEntry {
10
- key: string;
11
- value: string;
12
- createdAt: number;
13
- ttlMs: number;
14
- hits: number;
15
- }
16
-
17
- export interface CacheConfig {
18
- enabled: boolean;
19
- ttlMs: number; // default TTL
20
- maxEntries: number;
21
- dataDir: string;
22
- }
23
-
24
- export class LLMCache {
25
- private cache: Map<string, CacheEntry> = new Map();
26
- private config: CacheConfig;
27
- private filePath: string;
28
- private stats = { hits: 0, misses: 0, evictions: 0 };
29
-
30
- constructor(config?: Partial<CacheConfig>) {
31
- this.config = {
32
- enabled: config?.enabled ?? true,
33
- ttlMs: config?.ttlMs ?? 3600_000, // 1 hour default
34
- maxEntries: config?.maxEntries ?? 1000,
35
- dataDir: config?.dataDir ?? '.',
36
- };
37
- this.filePath = path.join(this.config.dataDir, 'data', 'cache.json');
38
- this.load();
39
- }
40
-
41
- private load(): void {
42
- try {
43
- if (fs.existsSync(this.filePath)) {
44
- const raw = fs.readFileSync(this.filePath, 'utf-8');
45
- const entries: CacheEntry[] = JSON.parse(raw);
46
- for (const entry of entries) {
47
- if (!this.isExpired(entry)) {
48
- this.cache.set(entry.key, entry);
49
- }
50
- }
51
- }
52
- } catch {
53
- // ignore
54
- }
55
- }
56
-
57
- private save(): void {
58
- const dir = path.dirname(this.filePath);
59
- if (!fs.existsSync(dir)) fs.mkdirSync(dir, { recursive: true });
60
- const entries = Array.from(this.cache.values());
61
- fs.writeFileSync(this.filePath, JSON.stringify(entries));
62
- }
63
-
64
- private isExpired(entry: CacheEntry): boolean {
65
- return Date.now() - entry.createdAt > entry.ttlMs;
66
- }
67
-
68
- /**
69
- * Generate a cache key from messages and system prompt.
70
- */
71
- static makeKey(messages: Array<{ role: string; content: string }>, systemPrompt?: string): string {
72
- const payload = JSON.stringify({ systemPrompt, messages: messages.map(m => ({ role: m.role, content: m.content })) });
73
- return crypto.createHash('sha256').update(payload).digest('hex').slice(0, 16);
74
- }
75
-
76
- /**
77
- * Get a cached response. Returns null if not found or expired.
78
- */
79
- get(key: string): string | null {
80
- if (!this.config.enabled) return null;
81
- const entry = this.cache.get(key);
82
- if (!entry || this.isExpired(entry)) {
83
- if (entry) {
84
- this.cache.delete(key);
85
- this.stats.evictions++;
86
- }
87
- this.stats.misses++;
88
- return null;
89
- }
90
- entry.hits++;
91
- this.stats.hits++;
92
- return entry.value;
93
- }
94
-
95
- /**
96
- * Set a cached response.
97
- */
98
- set(key: string, value: string, ttlMs?: number): void {
99
- if (!this.config.enabled) return;
100
-
101
- // Evict oldest if at capacity
102
- if (this.cache.size >= this.config.maxEntries) {
103
- let oldestKey: string | null = null;
104
- let oldestTime = Infinity;
105
- for (const [k, v] of this.cache) {
106
- if (v.createdAt < oldestTime) {
107
- oldestTime = v.createdAt;
108
- oldestKey = k;
109
- }
110
- }
111
- if (oldestKey) {
112
- this.cache.delete(oldestKey);
113
- this.stats.evictions++;
114
- }
115
- }
116
-
117
- this.cache.set(key, {
118
- key,
119
- value,
120
- createdAt: Date.now(),
121
- ttlMs: ttlMs ?? this.config.ttlMs,
122
- hits: 0,
123
- });
124
- this.save();
125
- }
126
-
127
- getStats(): { hits: number; misses: number; evictions: number; size: number; hitRate: string } {
128
- const total = this.stats.hits + this.stats.misses;
129
- return {
130
- ...this.stats,
131
- size: this.cache.size,
132
- hitRate: total > 0 ? `${((this.stats.hits / total) * 100).toFixed(1)}%` : '0%',
133
- };
134
- }
135
-
136
- clear(): void {
137
- this.cache.clear();
138
- this.stats = { hits: 0, misses: 0, evictions: 0 };
139
- this.save();
140
- }
141
- }
1
+ /**
2
+ * Caching Layer - Cache LLM responses with configurable TTL.
3
+ * Hash-based key from input messages + system prompt.
4
+ */
5
+ import * as fs from 'fs';
6
+ import * as path from 'path';
7
+ import * as crypto from 'crypto';
8
+
9
+ export interface CacheEntry {
10
+ key: string;
11
+ value: string;
12
+ createdAt: number;
13
+ ttlMs: number;
14
+ hits: number;
15
+ }
16
+
17
+ export interface CacheConfig {
18
+ enabled: boolean;
19
+ ttlMs: number; // default TTL
20
+ maxEntries: number;
21
+ dataDir: string;
22
+ }
23
+
24
+ export class LLMCache {
25
+ private cache: Map<string, CacheEntry> = new Map();
26
+ private config: CacheConfig;
27
+ private filePath: string;
28
+ private stats = { hits: 0, misses: 0, evictions: 0 };
29
+
30
+ constructor(config?: Partial<CacheConfig>) {
31
+ this.config = {
32
+ enabled: config?.enabled ?? true,
33
+ ttlMs: config?.ttlMs ?? 3600_000, // 1 hour default
34
+ maxEntries: config?.maxEntries ?? 1000,
35
+ dataDir: config?.dataDir ?? '.',
36
+ };
37
+ this.filePath = path.join(this.config.dataDir, 'data', 'cache.json');
38
+ this.load();
39
+ }
40
+
41
+ private load(): void {
42
+ try {
43
+ if (fs.existsSync(this.filePath)) {
44
+ const raw = fs.readFileSync(this.filePath, 'utf-8');
45
+ const entries: CacheEntry[] = JSON.parse(raw);
46
+ for (const entry of entries) {
47
+ if (!this.isExpired(entry)) {
48
+ this.cache.set(entry.key, entry);
49
+ }
50
+ }
51
+ }
52
+ } catch {
53
+ // ignore
54
+ }
55
+ }
56
+
57
+ private save(): void {
58
+ const dir = path.dirname(this.filePath);
59
+ if (!fs.existsSync(dir)) fs.mkdirSync(dir, { recursive: true });
60
+ const entries = Array.from(this.cache.values());
61
+ fs.writeFileSync(this.filePath, JSON.stringify(entries));
62
+ }
63
+
64
+ private isExpired(entry: CacheEntry): boolean {
65
+ return Date.now() - entry.createdAt > entry.ttlMs;
66
+ }
67
+
68
+ /**
69
+ * Generate a cache key from messages and system prompt.
70
+ */
71
+ static makeKey(messages: Array<{ role: string; content: string }>, systemPrompt?: string): string {
72
+ const payload = JSON.stringify({ systemPrompt, messages: messages.map(m => ({ role: m.role, content: m.content })) });
73
+ return crypto.createHash('sha256').update(payload).digest('hex').slice(0, 16);
74
+ }
75
+
76
+ /**
77
+ * Get a cached response. Returns null if not found or expired.
78
+ */
79
+ get(key: string): string | null {
80
+ if (!this.config.enabled) return null;
81
+ const entry = this.cache.get(key);
82
+ if (!entry || this.isExpired(entry)) {
83
+ if (entry) {
84
+ this.cache.delete(key);
85
+ this.stats.evictions++;
86
+ }
87
+ this.stats.misses++;
88
+ return null;
89
+ }
90
+ entry.hits++;
91
+ this.stats.hits++;
92
+ return entry.value;
93
+ }
94
+
95
+ /**
96
+ * Set a cached response.
97
+ */
98
+ set(key: string, value: string, ttlMs?: number): void {
99
+ if (!this.config.enabled) return;
100
+
101
+ // Evict oldest if at capacity
102
+ if (this.cache.size >= this.config.maxEntries) {
103
+ let oldestKey: string | null = null;
104
+ let oldestTime = Infinity;
105
+ for (const [k, v] of this.cache) {
106
+ if (v.createdAt < oldestTime) {
107
+ oldestTime = v.createdAt;
108
+ oldestKey = k;
109
+ }
110
+ }
111
+ if (oldestKey) {
112
+ this.cache.delete(oldestKey);
113
+ this.stats.evictions++;
114
+ }
115
+ }
116
+
117
+ this.cache.set(key, {
118
+ key,
119
+ value,
120
+ createdAt: Date.now(),
121
+ ttlMs: ttlMs ?? this.config.ttlMs,
122
+ hits: 0,
123
+ });
124
+ this.save();
125
+ }
126
+
127
+ getStats(): { hits: number; misses: number; evictions: number; size: number; hitRate: string } {
128
+ const total = this.stats.hits + this.stats.misses;
129
+ return {
130
+ ...this.stats,
131
+ size: this.cache.size,
132
+ hitRate: total > 0 ? `${((this.stats.hits / total) * 100).toFixed(1)}%` : '0%',
133
+ };
134
+ }
135
+
136
+ clear(): void {
137
+ this.cache.clear();
138
+ this.stats = { hits: 0, misses: 0, evictions: 0 };
139
+ this.save();
140
+ }
141
+ }
@@ -1,77 +1,77 @@
1
- import type { AgentContext, Message } from './types';
2
-
3
- /**
4
- * Agent Composition — v0.8.0
5
- * Combine multiple agents into a pipeline: Agent A output → Agent B input.
6
- * Configurable in OAD: `compose: [agent-a, agent-b]`
7
- */
8
-
9
- export type AgentHandler = (context: AgentContext, message: Message) => Promise<Message>;
10
-
11
- export interface ComposableAgent {
12
- id: string;
13
- name: string;
14
- handler: AgentHandler;
15
- }
16
-
17
- export interface ComposeOptions {
18
- /** Stop pipeline if any agent returns empty content */
19
- stopOnEmpty?: boolean;
20
- /** Transform output between agents */
21
- transform?: (output: Message, nextAgentId: string) => Message;
22
- /** Timeout per agent in ms */
23
- timeoutMs?: number;
24
- }
25
-
26
- export class AgentPipeline {
27
- private agents: ComposableAgent[] = [];
28
- private options: ComposeOptions;
29
-
30
- constructor(agents: ComposableAgent[], options: ComposeOptions = {}) {
31
- this.agents = agents;
32
- this.options = options;
33
- }
34
-
35
- /** Run the pipeline sequentially: each agent's output becomes the next agent's input */
36
- async execute(context: AgentContext, initialMessage: Message): Promise<Message> {
37
- let currentMessage = initialMessage;
38
-
39
- for (const agent of this.agents) {
40
- if (this.options.stopOnEmpty && !currentMessage.content.trim()) {
41
- break;
42
- }
43
-
44
- // Apply transform if provided
45
- if (this.options.transform) {
46
- currentMessage = this.options.transform(currentMessage, agent.id);
47
- }
48
-
49
- if (this.options.timeoutMs) {
50
- const result = await Promise.race([
51
- agent.handler(context, currentMessage),
52
- new Promise<never>((_, reject) =>
53
- setTimeout(() => reject(new Error(`Agent ${agent.id} timed out`)), this.options.timeoutMs)
54
- ),
55
- ]);
56
- currentMessage = result;
57
- } else {
58
- currentMessage = await agent.handler(context, currentMessage);
59
- }
60
- }
61
-
62
- return currentMessage;
63
- }
64
-
65
- /** Get the pipeline agent IDs in order */
66
- getAgentIds(): string[] {
67
- return this.agents.map((a) => a.id);
68
- }
69
- }
70
-
71
- /**
72
- * Create a pipeline from an array of composable agents.
73
- * Usage in OAD: `compose: [agent-a, agent-b, agent-c]`
74
- */
75
- export function compose(agents: ComposableAgent[], options?: ComposeOptions): AgentPipeline {
76
- return new AgentPipeline(agents, options);
77
- }
1
+ import type { AgentContext, Message } from './types';
2
+
3
+ /**
4
+ * Agent Composition — v0.8.0
5
+ * Combine multiple agents into a pipeline: Agent A output → Agent B input.
6
+ * Configurable in OAD: `compose: [agent-a, agent-b]`
7
+ */
8
+
9
+ export type AgentHandler = (context: AgentContext, message: Message) => Promise<Message>;
10
+
11
+ export interface ComposableAgent {
12
+ id: string;
13
+ name: string;
14
+ handler: AgentHandler;
15
+ }
16
+
17
+ export interface ComposeOptions {
18
+ /** Stop pipeline if any agent returns empty content */
19
+ stopOnEmpty?: boolean;
20
+ /** Transform output between agents */
21
+ transform?: (output: Message, nextAgentId: string) => Message;
22
+ /** Timeout per agent in ms */
23
+ timeoutMs?: number;
24
+ }
25
+
26
+ export class AgentPipeline {
27
+ private agents: ComposableAgent[] = [];
28
+ private options: ComposeOptions;
29
+
30
+ constructor(agents: ComposableAgent[], options: ComposeOptions = {}) {
31
+ this.agents = agents;
32
+ this.options = options;
33
+ }
34
+
35
+ /** Run the pipeline sequentially: each agent's output becomes the next agent's input */
36
+ async execute(context: AgentContext, initialMessage: Message): Promise<Message> {
37
+ let currentMessage = initialMessage;
38
+
39
+ for (const agent of this.agents) {
40
+ if (this.options.stopOnEmpty && !currentMessage.content.trim()) {
41
+ break;
42
+ }
43
+
44
+ // Apply transform if provided
45
+ if (this.options.transform) {
46
+ currentMessage = this.options.transform(currentMessage, agent.id);
47
+ }
48
+
49
+ if (this.options.timeoutMs) {
50
+ const result = await Promise.race([
51
+ agent.handler(context, currentMessage),
52
+ new Promise<never>((_, reject) =>
53
+ setTimeout(() => reject(new Error(`Agent ${agent.id} timed out`)), this.options.timeoutMs)
54
+ ),
55
+ ]);
56
+ currentMessage = result;
57
+ } else {
58
+ currentMessage = await agent.handler(context, currentMessage);
59
+ }
60
+ }
61
+
62
+ return currentMessage;
63
+ }
64
+
65
+ /** Get the pipeline agent IDs in order */
66
+ getAgentIds(): string[] {
67
+ return this.agents.map((a) => a.id);
68
+ }
69
+ }
70
+
71
+ /**
72
+ * Create a pipeline from an array of composable agents.
73
+ * Usage in OAD: `compose: [agent-a, agent-b, agent-c]`
74
+ */
75
+ export function compose(agents: ComposableAgent[], options?: ComposeOptions): AgentPipeline {
76
+ return new AgentPipeline(agents, options);
77
+ }
@@ -1,14 +1,14 @@
1
- import * as fs from 'fs';
2
- import * as yaml from 'js-yaml';
3
- import { OADSchema, type OADDocument } from '../schema/oad';
4
-
5
- export function loadOAD(filePath: string): OADDocument {
6
- const raw = fs.readFileSync(filePath, 'utf-8');
7
- const ext = filePath.split('.').pop()?.toLowerCase();
8
- const data = (ext === 'yaml' || ext === 'yml') ? yaml.load(raw) : JSON.parse(raw);
9
- return OADSchema.parse(data);
10
- }
11
-
12
- export function validateOAD(data: unknown): OADDocument {
13
- return OADSchema.parse(data);
14
- }
1
+ import * as fs from 'fs';
2
+ import * as yaml from 'js-yaml';
3
+ import { OADSchema, type OADDocument } from '../schema/oad';
4
+
5
+ export function loadOAD(filePath: string): OADDocument {
6
+ const raw = fs.readFileSync(filePath, 'utf-8');
7
+ const ext = filePath.split('.').pop()?.toLowerCase();
8
+ const data = (ext === 'yaml' || ext === 'yml') ? yaml.load(raw) : JSON.parse(raw);
9
+ return OADSchema.parse(data);
10
+ }
11
+
12
+ export function validateOAD(data: unknown): OADDocument {
13
+ return OADSchema.parse(data);
14
+ }