opc-agent 1.3.1 → 1.3.2

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 (160) hide show
  1. package/CHANGELOG.md +6 -0
  2. package/CONTRIBUTING.md +75 -75
  3. package/README.md +358 -235
  4. package/README.zh-CN.md +415 -415
  5. package/dist/cli.js +34 -118
  6. package/dist/core/dashboard.d.ts +35 -0
  7. package/dist/core/dashboard.js +157 -0
  8. package/dist/core/priority.d.ts +52 -0
  9. package/dist/core/priority.js +102 -0
  10. package/dist/deploy/hermes.js +22 -22
  11. package/dist/deploy/openclaw.js +40 -31
  12. package/dist/index.d.ts +10 -3
  13. package/dist/index.js +15 -6
  14. package/dist/schema/oad.d.ts +2 -1
  15. package/dist/templates/code-reviewer.d.ts +8 -0
  16. package/dist/templates/code-reviewer.js +9 -5
  17. package/dist/templates/customer-service.d.ts +8 -0
  18. package/dist/templates/customer-service.js +6 -2
  19. package/dist/templates/data-analyst.d.ts +8 -0
  20. package/dist/templates/data-analyst.js +9 -5
  21. package/dist/templates/knowledge-base.d.ts +8 -0
  22. package/dist/templates/knowledge-base.js +6 -2
  23. package/dist/templates/sales-assistant.d.ts +8 -0
  24. package/dist/templates/sales-assistant.js +8 -4
  25. package/dist/templates/teacher.d.ts +8 -0
  26. package/dist/templates/teacher.js +10 -6
  27. package/docs/.vitepress/config.ts +103 -103
  28. package/docs/api/cli.md +48 -48
  29. package/docs/api/oad-schema.md +64 -64
  30. package/docs/api/sdk.md +80 -80
  31. package/docs/guide/concepts.md +51 -51
  32. package/docs/guide/configuration.md +79 -79
  33. package/docs/guide/deployment.md +42 -42
  34. package/docs/guide/getting-started.md +44 -44
  35. package/docs/guide/templates.md +28 -28
  36. package/docs/guide/testing.md +84 -84
  37. package/docs/index.md +27 -27
  38. package/docs/zh/api/cli.md +54 -54
  39. package/docs/zh/api/oad-schema.md +87 -87
  40. package/docs/zh/api/sdk.md +102 -102
  41. package/docs/zh/guide/concepts.md +104 -104
  42. package/docs/zh/guide/configuration.md +135 -135
  43. package/docs/zh/guide/deployment.md +81 -81
  44. package/docs/zh/guide/getting-started.md +82 -82
  45. package/docs/zh/guide/templates.md +84 -84
  46. package/docs/zh/guide/testing.md +88 -88
  47. package/docs/zh/index.md +27 -27
  48. package/examples/customer-service-demo/README.md +90 -90
  49. package/examples/customer-service-demo/oad.yaml +107 -107
  50. package/package.json +1 -1
  51. package/src/analytics/index.ts +66 -66
  52. package/src/channels/discord.ts +192 -192
  53. package/src/channels/email.ts +177 -177
  54. package/src/channels/feishu.ts +236 -236
  55. package/src/channels/index.ts +15 -15
  56. package/src/channels/slack.ts +160 -160
  57. package/src/channels/telegram.ts +90 -90
  58. package/src/channels/voice.ts +106 -106
  59. package/src/channels/webhook.ts +199 -199
  60. package/src/channels/websocket.ts +87 -87
  61. package/src/channels/wechat.ts +149 -149
  62. package/src/cli.ts +32 -124
  63. package/src/core/a2a.ts +143 -143
  64. package/src/core/agent.ts +152 -152
  65. package/src/core/analytics-engine.ts +186 -186
  66. package/src/core/auth.ts +57 -57
  67. package/src/core/cache.ts +141 -141
  68. package/src/core/compose.ts +77 -77
  69. package/src/core/config.ts +14 -14
  70. package/src/core/dashboard.ts +219 -0
  71. package/src/core/errors.ts +148 -148
  72. package/src/core/hitl.ts +138 -138
  73. package/src/core/logger.ts +57 -57
  74. package/src/core/orchestrator.ts +215 -215
  75. package/src/core/performance.ts +187 -187
  76. package/src/core/priority.ts +140 -0
  77. package/src/core/rate-limiter.ts +128 -128
  78. package/src/core/room.ts +109 -109
  79. package/src/core/runtime.ts +152 -152
  80. package/src/core/sandbox.ts +101 -101
  81. package/src/core/security.ts +171 -171
  82. package/src/core/types.ts +68 -68
  83. package/src/core/versioning.ts +106 -106
  84. package/src/core/watch.ts +178 -178
  85. package/src/core/workflow.ts +235 -235
  86. package/src/deploy/hermes.ts +156 -156
  87. package/src/deploy/openclaw.ts +200 -190
  88. package/src/dtv/data.ts +29 -0
  89. package/src/dtv/trust.ts +43 -0
  90. package/src/dtv/value.ts +47 -0
  91. package/src/i18n/index.ts +216 -216
  92. package/src/index.ts +10 -3
  93. package/src/marketplace/index.ts +223 -0
  94. package/src/memory/deepbrain.ts +108 -108
  95. package/src/memory/index.ts +34 -34
  96. package/src/plugins/index.ts +208 -208
  97. package/src/schema/oad.ts +155 -154
  98. package/src/skills/base.ts +16 -16
  99. package/src/skills/document.ts +100 -100
  100. package/src/skills/http.ts +35 -35
  101. package/src/skills/index.ts +27 -27
  102. package/src/skills/scheduler.ts +80 -80
  103. package/src/skills/webhook-trigger.ts +59 -59
  104. package/src/templates/code-reviewer.ts +34 -30
  105. package/src/templates/customer-service.ts +80 -76
  106. package/src/templates/data-analyst.ts +70 -66
  107. package/src/templates/executive-assistant.ts +71 -71
  108. package/src/templates/financial-advisor.ts +60 -60
  109. package/src/templates/knowledge-base.ts +31 -27
  110. package/src/templates/legal-assistant.ts +71 -71
  111. package/src/templates/sales-assistant.ts +79 -75
  112. package/src/templates/teacher.ts +79 -75
  113. package/src/testing/index.ts +181 -181
  114. package/src/tools/calculator.ts +73 -73
  115. package/src/tools/datetime.ts +149 -149
  116. package/src/tools/json-transform.ts +187 -187
  117. package/src/tools/mcp.ts +76 -76
  118. package/src/tools/text-analysis.ts +116 -116
  119. package/templates/Dockerfile +15 -15
  120. package/templates/code-reviewer/README.md +27 -27
  121. package/templates/code-reviewer/oad.yaml +41 -41
  122. package/templates/customer-service/README.md +22 -22
  123. package/templates/customer-service/oad.yaml +36 -36
  124. package/templates/docker-compose.yml +21 -21
  125. package/templates/ecommerce-assistant/README.md +45 -45
  126. package/templates/ecommerce-assistant/oad.yaml +47 -47
  127. package/templates/knowledge-base/README.md +28 -28
  128. package/templates/knowledge-base/oad.yaml +38 -38
  129. package/templates/sales-assistant/README.md +26 -26
  130. package/templates/sales-assistant/oad.yaml +43 -43
  131. package/templates/tech-support/README.md +43 -43
  132. package/templates/tech-support/oad.yaml +45 -45
  133. package/tests/a2a.test.ts +66 -66
  134. package/tests/agent.test.ts +72 -72
  135. package/tests/analytics.test.ts +50 -50
  136. package/tests/channel.test.ts +39 -39
  137. package/tests/e2e.test.ts +134 -134
  138. package/tests/errors.test.ts +83 -83
  139. package/tests/hitl.test.ts +71 -71
  140. package/tests/i18n.test.ts +41 -41
  141. package/tests/mcp.test.ts +54 -54
  142. package/tests/oad.test.ts +68 -68
  143. package/tests/performance.test.ts +115 -115
  144. package/tests/plugin.test.ts +74 -74
  145. package/tests/room.test.ts +106 -106
  146. package/tests/runtime.test.ts +42 -42
  147. package/tests/sandbox.test.ts +46 -46
  148. package/tests/security.test.ts +60 -60
  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
  157. package/.github/workflows/ci.yml +0 -24
  158. package/dist/traces/index.d.ts +0 -49
  159. package/dist/traces/index.js +0 -102
  160. package/src/traces/index.ts +0 -132
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
+ }
@@ -0,0 +1,219 @@
1
+ import express, { Request, Response } from 'express';
2
+
3
+ // ─── Types ───────────────────────────────────────────────────
4
+
5
+ export interface DashboardConfig {
6
+ /** Enable the dashboard (default: false) */
7
+ enabled: boolean;
8
+ /** HTTP port (default: 4100) */
9
+ port?: number;
10
+ /** Bind address (default: 127.0.0.1 for security) */
11
+ host?: string;
12
+ /** Enable CORS (default: false) */
13
+ cors?: boolean;
14
+ }
15
+
16
+ interface DashboardState {
17
+ agent: { name: string; version: string; status: string; uptime: number };
18
+ sessions: SessionSummary[];
19
+ tools: ToolSummary[];
20
+ channels: ChannelSummary[];
21
+ memory: MemorySummary;
22
+ modelAuth: ModelAuthSummary;
23
+ }
24
+
25
+ interface SessionSummary {
26
+ id: string;
27
+ channel: string;
28
+ messages: number;
29
+ lastActive: number;
30
+ status: 'active' | 'idle' | 'closed';
31
+ }
32
+
33
+ interface ToolSummary {
34
+ name: string;
35
+ type: 'builtin' | 'mcp' | 'gateway';
36
+ enabled: boolean;
37
+ invocations: number;
38
+ lastUsed?: number;
39
+ }
40
+
41
+ interface ChannelSummary {
42
+ name: string;
43
+ type: string;
44
+ connected: boolean;
45
+ messageCount: number;
46
+ }
47
+
48
+ interface MemorySummary {
49
+ provider: string;
50
+ entries: number;
51
+ lastSync?: number;
52
+ }
53
+
54
+ interface ModelAuthSummary {
55
+ providers: { name: string; status: 'healthy' | 'expiring' | 'expired' | 'unconfigured'; expiresAt?: number }[];
56
+ }
57
+
58
+ // ─── Dashboard Server ────────────────────────────────────────
59
+
60
+ export class Dashboard {
61
+ private app = express();
62
+ private server: ReturnType<typeof this.app.listen> | null = null;
63
+ private config: Required<DashboardConfig>;
64
+ private startTime = Date.now();
65
+ private stats = {
66
+ sessions: new Map<string, SessionSummary>(),
67
+ toolInvocations: new Map<string, { count: number; lastUsed: number }>(),
68
+ channelStats: new Map<string, { connected: boolean; messages: number }>(),
69
+ };
70
+
71
+ constructor(config: DashboardConfig) {
72
+ this.config = {
73
+ enabled: config.enabled,
74
+ port: config.port ?? 4100,
75
+ host: config.host ?? '127.0.0.1',
76
+ cors: config.cors ?? false,
77
+ };
78
+ this.setupRoutes();
79
+ }
80
+
81
+ private setupRoutes(): void {
82
+ if (this.config.cors) {
83
+ this.app.use((_req, res, next) => {
84
+ res.header('Access-Control-Allow-Origin', '*');
85
+ res.header('Access-Control-Allow-Headers', 'Content-Type');
86
+ next();
87
+ });
88
+ }
89
+ this.app.use(express.json());
90
+
91
+ // Health check
92
+ this.app.get('/api/health', (_req: Request, res: Response) => {
93
+ res.json({ status: 'ok', uptime: Date.now() - this.startTime });
94
+ });
95
+
96
+ // Overview state
97
+ this.app.get('/api/state', (_req: Request, res: Response) => {
98
+ res.json(this.getState());
99
+ });
100
+
101
+ // Sessions
102
+ this.app.get('/api/sessions', (_req: Request, res: Response) => {
103
+ res.json([...this.stats.sessions.values()]);
104
+ });
105
+
106
+ // Tools
107
+ this.app.get('/api/tools', (_req: Request, res: Response) => {
108
+ const tools: ToolSummary[] = [];
109
+ for (const [name, stat] of this.stats.toolInvocations) {
110
+ tools.push({ name, type: 'builtin', enabled: true, invocations: stat.count, lastUsed: stat.lastUsed });
111
+ }
112
+ res.json(tools);
113
+ });
114
+
115
+ // Channels
116
+ this.app.get('/api/channels', (_req: Request, res: Response) => {
117
+ const channels: ChannelSummary[] = [];
118
+ for (const [name, stat] of this.stats.channelStats) {
119
+ channels.push({ name, type: name, connected: stat.connected, messageCount: stat.messages });
120
+ }
121
+ res.json(channels);
122
+ });
123
+
124
+ // Simple HTML dashboard
125
+ this.app.get('/', (_req: Request, res: Response) => {
126
+ res.send(this.renderHTML());
127
+ });
128
+ }
129
+
130
+ private getState(): DashboardState {
131
+ return {
132
+ agent: { name: 'opc-agent', version: '1.3.0', status: 'running', uptime: Date.now() - this.startTime },
133
+ sessions: [...this.stats.sessions.values()],
134
+ tools: [...this.stats.toolInvocations.entries()].map(([name, s]) => ({
135
+ name, type: 'builtin' as const, enabled: true, invocations: s.count, lastUsed: s.lastUsed,
136
+ })),
137
+ channels: [...this.stats.channelStats.entries()].map(([name, s]) => ({
138
+ name, type: name, connected: s.connected, messageCount: s.messages,
139
+ })),
140
+ memory: { provider: 'unknown', entries: 0 },
141
+ modelAuth: { providers: [] },
142
+ };
143
+ }
144
+
145
+ // ─── Event Tracking ──────────────────────────────────────
146
+
147
+ trackSession(session: SessionSummary): void {
148
+ this.stats.sessions.set(session.id, session);
149
+ }
150
+
151
+ trackToolCall(toolName: string): void {
152
+ const existing = this.stats.toolInvocations.get(toolName) ?? { count: 0, lastUsed: 0 };
153
+ existing.count++;
154
+ existing.lastUsed = Date.now();
155
+ this.stats.toolInvocations.set(toolName, existing);
156
+ }
157
+
158
+ trackChannel(name: string, connected: boolean, messages?: number): void {
159
+ const existing = this.stats.channelStats.get(name) ?? { connected: false, messages: 0 };
160
+ existing.connected = connected;
161
+ if (messages !== undefined) existing.messages = messages;
162
+ this.stats.channelStats.set(name, existing);
163
+ }
164
+
165
+ // ─── Lifecycle ───────────────────────────────────────────
166
+
167
+ async start(): Promise<void> {
168
+ if (!this.config.enabled) return;
169
+ return new Promise((resolve) => {
170
+ this.server = this.app.listen(this.config.port, this.config.host, () => {
171
+ console.log(`[dashboard] http://${this.config.host}:${this.config.port}`);
172
+ resolve();
173
+ });
174
+ });
175
+ }
176
+
177
+ async stop(): Promise<void> {
178
+ return new Promise((resolve) => {
179
+ if (this.server) this.server.close(() => resolve());
180
+ else resolve();
181
+ });
182
+ }
183
+
184
+ private renderHTML(): string {
185
+ return `<!DOCTYPE html>
186
+ <html><head><meta charset="utf-8"><title>OPC Agent Dashboard</title>
187
+ <meta name="viewport" content="width=device-width,initial-scale=1">
188
+ <style>
189
+ *{box-sizing:border-box;margin:0;padding:0}
190
+ body{font-family:system-ui,-apple-system,sans-serif;background:#0a0a0f;color:#e0e0e0;padding:24px}
191
+ h1{font-size:1.5rem;margin-bottom:20px;color:#7c9aff}
192
+ .grid{display:grid;grid-template-columns:repeat(auto-fill,minmax(280px,1fr));gap:16px}
193
+ .card{background:#14141f;border:1px solid #2a2a3a;border-radius:12px;padding:20px}
194
+ .card h2{font-size:0.85rem;text-transform:uppercase;letter-spacing:1px;color:#888;margin-bottom:12px}
195
+ .stat{font-size:2rem;font-weight:700;color:#7c9aff}
196
+ .sub{font-size:0.8rem;color:#666;margin-top:4px}
197
+ #data{margin-top:20px;font-family:monospace;font-size:0.75rem;color:#555;white-space:pre-wrap}
198
+ </style></head><body>
199
+ <h1>⚡ OPC Agent Dashboard</h1>
200
+ <div class="grid">
201
+ <div class="card"><h2>Status</h2><div class="stat" id="status">Loading…</div><div class="sub" id="uptime"></div></div>
202
+ <div class="card"><h2>Sessions</h2><div class="stat" id="sessions">-</div></div>
203
+ <div class="card"><h2>Tools</h2><div class="stat" id="tools">-</div></div>
204
+ <div class="card"><h2>Channels</h2><div class="stat" id="channels">-</div></div>
205
+ </div>
206
+ <div id="data"></div>
207
+ <script>
208
+ async function poll(){try{const r=await fetch('/api/state');const d=await r.json();
209
+ document.getElementById('status').textContent=d.agent.status;
210
+ document.getElementById('uptime').textContent='Uptime: '+Math.floor(d.agent.uptime/1000)+'s';
211
+ document.getElementById('sessions').textContent=d.sessions.length;
212
+ document.getElementById('tools').textContent=d.tools.length;
213
+ document.getElementById('channels').textContent=d.channels.length;
214
+ document.getElementById('data').textContent=JSON.stringify(d,null,2);
215
+ }catch(e){document.getElementById('status').textContent='offline'}}
216
+ poll();setInterval(poll,5000);
217
+ </script></body></html>`;
218
+ }
219
+ }