opc-agent 1.4.0 → 1.4.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 (192) hide show
  1. package/CHANGELOG.md +69 -23
  2. package/CONTRIBUTING.md +60 -21
  3. package/README.md +358 -235
  4. package/README.zh-CN.md +415 -415
  5. package/dist/channels/slack.js +10 -93
  6. package/dist/channels/web.d.ts +0 -10
  7. package/dist/channels/web.js +2 -33
  8. package/dist/cli.js +60 -255
  9. package/dist/core/dashboard.d.ts +35 -0
  10. package/dist/core/dashboard.js +157 -0
  11. package/dist/core/fast-mode.d.ts +27 -0
  12. package/dist/core/fast-mode.js +59 -0
  13. package/dist/core/priority.d.ts +52 -0
  14. package/dist/core/priority.js +102 -0
  15. package/dist/core/runtime.d.ts +0 -4
  16. package/dist/core/runtime.js +0 -27
  17. package/dist/deploy/hermes.js +22 -22
  18. package/dist/deploy/openclaw.js +40 -31
  19. package/dist/index.d.ts +14 -3
  20. package/dist/index.js +20 -6
  21. package/dist/memory/cloud-storage.d.ts +40 -0
  22. package/dist/memory/cloud-storage.js +211 -0
  23. package/dist/providers/index.d.ts +1 -1
  24. package/dist/providers/index.js +1 -7
  25. package/dist/schema/oad.d.ts +2 -1
  26. package/dist/templates/code-reviewer.d.ts +8 -0
  27. package/dist/templates/code-reviewer.js +9 -5
  28. package/dist/templates/customer-service.d.ts +8 -0
  29. package/dist/templates/customer-service.js +6 -2
  30. package/dist/templates/data-analyst.d.ts +8 -0
  31. package/dist/templates/data-analyst.js +9 -5
  32. package/dist/templates/knowledge-base.d.ts +8 -0
  33. package/dist/templates/knowledge-base.js +6 -2
  34. package/dist/templates/sales-assistant.d.ts +8 -0
  35. package/dist/templates/sales-assistant.js +8 -4
  36. package/dist/templates/teacher.d.ts +8 -0
  37. package/dist/templates/teacher.js +10 -6
  38. package/docs/.vitepress/config.ts +103 -103
  39. package/docs/api/cli.md +48 -48
  40. package/docs/api/oad-schema.md +64 -64
  41. package/docs/api/sdk.md +80 -80
  42. package/docs/guide/concepts.md +51 -51
  43. package/docs/guide/configuration.md +79 -79
  44. package/docs/guide/deployment.md +42 -42
  45. package/docs/guide/getting-started.md +44 -44
  46. package/docs/guide/templates.md +28 -28
  47. package/docs/guide/testing.md +84 -84
  48. package/docs/index.md +27 -27
  49. package/docs/zh/api/cli.md +54 -54
  50. package/docs/zh/api/oad-schema.md +87 -87
  51. package/docs/zh/api/sdk.md +102 -102
  52. package/docs/zh/guide/concepts.md +104 -104
  53. package/docs/zh/guide/configuration.md +135 -135
  54. package/docs/zh/guide/deployment.md +81 -81
  55. package/docs/zh/guide/getting-started.md +82 -82
  56. package/docs/zh/guide/templates.md +84 -84
  57. package/docs/zh/guide/testing.md +88 -88
  58. package/docs/zh/index.md +27 -27
  59. package/examples/customer-service-demo/README.md +90 -90
  60. package/examples/customer-service-demo/oad.yaml +107 -107
  61. package/package.json +1 -1
  62. package/src/analytics/index.ts +66 -66
  63. package/src/channels/discord.ts +192 -192
  64. package/src/channels/email.ts +177 -177
  65. package/src/channels/feishu.ts +236 -236
  66. package/src/channels/index.ts +15 -15
  67. package/src/channels/slack.ts +160 -217
  68. package/src/channels/telegram.ts +90 -90
  69. package/src/channels/voice.ts +106 -106
  70. package/src/channels/web.ts +2 -38
  71. package/src/channels/webhook.ts +199 -199
  72. package/src/channels/websocket.ts +87 -87
  73. package/src/channels/wechat.ts +149 -149
  74. package/src/cli.ts +58 -282
  75. package/src/core/a2a.ts +143 -143
  76. package/src/core/agent.ts +152 -152
  77. package/src/core/analytics-engine.ts +186 -186
  78. package/src/core/auth.ts +57 -57
  79. package/src/core/cache.ts +141 -141
  80. package/src/core/compose.ts +77 -77
  81. package/src/core/config.ts +14 -14
  82. package/src/core/dashboard.ts +219 -0
  83. package/src/core/errors.ts +148 -148
  84. package/src/core/fast-mode.ts +75 -0
  85. package/src/core/hitl.ts +138 -138
  86. package/src/core/logger.ts +57 -57
  87. package/src/core/orchestrator.ts +215 -215
  88. package/src/core/performance.ts +187 -187
  89. package/src/core/priority.ts +140 -0
  90. package/src/core/rate-limiter.ts +128 -128
  91. package/src/core/room.ts +109 -109
  92. package/src/core/runtime.ts +152 -183
  93. package/src/core/sandbox.ts +101 -101
  94. package/src/core/security.ts +171 -171
  95. package/src/core/types.ts +68 -68
  96. package/src/core/versioning.ts +106 -106
  97. package/src/core/watch.ts +178 -178
  98. package/src/core/workflow.ts +235 -235
  99. package/src/deploy/hermes.ts +156 -156
  100. package/src/deploy/openclaw.ts +200 -190
  101. package/src/dtv/data.ts +29 -0
  102. package/src/dtv/trust.ts +43 -0
  103. package/src/dtv/value.ts +47 -0
  104. package/src/i18n/index.ts +216 -216
  105. package/src/index.ts +16 -3
  106. package/src/marketplace/index.ts +223 -0
  107. package/src/memory/cloud-storage.ts +217 -0
  108. package/src/memory/deepbrain.ts +108 -108
  109. package/src/memory/index.ts +34 -34
  110. package/src/plugins/index.ts +208 -208
  111. package/src/providers/index.ts +1 -9
  112. package/src/schema/oad.ts +155 -154
  113. package/src/skills/base.ts +16 -16
  114. package/src/skills/document.ts +100 -100
  115. package/src/skills/http.ts +35 -35
  116. package/src/skills/index.ts +27 -27
  117. package/src/skills/scheduler.ts +80 -80
  118. package/src/skills/webhook-trigger.ts +59 -59
  119. package/src/templates/code-reviewer.ts +34 -30
  120. package/src/templates/customer-service.ts +80 -76
  121. package/src/templates/data-analyst.ts +70 -66
  122. package/src/templates/executive-assistant.ts +71 -71
  123. package/src/templates/financial-advisor.ts +60 -60
  124. package/src/templates/knowledge-base.ts +31 -27
  125. package/src/templates/legal-assistant.ts +71 -71
  126. package/src/templates/sales-assistant.ts +79 -75
  127. package/src/templates/teacher.ts +79 -75
  128. package/src/testing/index.ts +181 -181
  129. package/src/tools/calculator.ts +73 -73
  130. package/src/tools/datetime.ts +149 -149
  131. package/src/tools/json-transform.ts +187 -187
  132. package/src/tools/mcp.ts +76 -76
  133. package/src/tools/text-analysis.ts +116 -116
  134. package/templates/Dockerfile +15 -15
  135. package/templates/code-reviewer/README.md +27 -27
  136. package/templates/code-reviewer/oad.yaml +41 -41
  137. package/templates/customer-service/README.md +22 -22
  138. package/templates/customer-service/oad.yaml +36 -36
  139. package/templates/docker-compose.yml +21 -21
  140. package/templates/ecommerce-assistant/README.md +45 -45
  141. package/templates/ecommerce-assistant/oad.yaml +47 -47
  142. package/templates/knowledge-base/README.md +28 -28
  143. package/templates/knowledge-base/oad.yaml +38 -38
  144. package/templates/sales-assistant/README.md +26 -26
  145. package/templates/sales-assistant/oad.yaml +43 -43
  146. package/templates/tech-support/README.md +43 -43
  147. package/templates/tech-support/oad.yaml +45 -45
  148. package/tests/a2a.test.ts +66 -66
  149. package/tests/agent.test.ts +72 -72
  150. package/tests/analytics.test.ts +50 -50
  151. package/tests/channel.test.ts +39 -39
  152. package/tests/e2e.test.ts +134 -134
  153. package/tests/errors.test.ts +83 -83
  154. package/tests/hitl.test.ts +71 -71
  155. package/tests/i18n.test.ts +41 -41
  156. package/tests/mcp.test.ts +54 -54
  157. package/tests/oad.test.ts +68 -68
  158. package/tests/performance.test.ts +115 -115
  159. package/tests/plugin.test.ts +74 -74
  160. package/tests/room.test.ts +106 -106
  161. package/tests/runtime.test.ts +42 -42
  162. package/tests/sandbox.test.ts +46 -46
  163. package/tests/security.test.ts +60 -60
  164. package/tests/templates.test.ts +77 -77
  165. package/tests/v070.test.ts +76 -76
  166. package/tests/versioning.test.ts +75 -75
  167. package/tests/voice.test.ts +61 -61
  168. package/tests/webhook.test.ts +29 -29
  169. package/tests/workflow.test.ts +143 -143
  170. package/tsconfig.json +19 -19
  171. package/vitest.config.ts +9 -9
  172. package/.github/ISSUE_TEMPLATE/bug_report.md +0 -20
  173. package/.github/ISSUE_TEMPLATE/feature_request.md +0 -14
  174. package/.github/PULL_REQUEST_TEMPLATE.md +0 -13
  175. package/.github/workflows/ci.yml +0 -24
  176. package/dist/traces/index.d.ts +0 -49
  177. package/dist/traces/index.js +0 -102
  178. package/examples/README.md +0 -22
  179. package/examples/basic-agent.ts +0 -90
  180. package/examples/brain-integration.ts +0 -71
  181. package/examples/multi-channel.ts +0 -74
  182. package/src/traces/index.ts +0 -132
  183. package/test-agent/Dockerfile +0 -9
  184. package/test-agent/README.md +0 -50
  185. package/test-agent/agent.yaml +0 -23
  186. package/test-agent/docker-compose.yml +0 -11
  187. package/test-agent/oad.yaml +0 -31
  188. package/test-agent/package-lock.json +0 -1492
  189. package/test-agent/package.json +0 -18
  190. package/test-agent/src/index.ts +0 -24
  191. package/test-agent/src/skills/echo.ts +0 -15
  192. package/test-agent/tsconfig.json +0 -25
@@ -1,77 +1,77 @@
1
- import { describe, it, expect } from 'vitest';
2
- import { createLegalAssistantConfig, ContractReviewSkill, ComplianceCheckSkill } from '../src/templates/legal-assistant';
3
- import { createFinancialAdvisorConfig, BudgetAnalysisSkill, FinancialPlanningSkill } from '../src/templates/financial-advisor';
4
- import { createExecutiveAssistantConfig, CalendarSkill, EmailDraftSkill, MeetingPrepSkill } from '../src/templates/executive-assistant';
5
- import type { AgentContext, Message, MemoryStore } from '../src/core/types';
6
-
7
- const mockMemory: MemoryStore = {
8
- get: async () => null, set: async () => {}, getConversation: async () => [],
9
- addMessage: async () => {}, clear: async () => {},
10
- };
11
- const ctx: AgentContext = { agentName: 'test', sessionId: 's1', messages: [], memory: mockMemory, metadata: {} };
12
- const msg = (content: string): Message => ({ id: '1', role: 'user', content, timestamp: Date.now() });
13
-
14
- describe('Legal Assistant Template', () => {
15
- it('should create valid config', () => {
16
- const config = createLegalAssistantConfig();
17
- expect(config.metadata.name).toBe('legal-assistant');
18
- expect(config.spec.skills).toHaveLength(2);
19
- });
20
-
21
- it('ContractReviewSkill matches contract terms', async () => {
22
- const skill = new ContractReviewSkill();
23
- const r = await skill.execute(ctx, msg('What is force majeure?'));
24
- expect(r.handled).toBe(true);
25
- });
26
-
27
- it('ComplianceCheckSkill matches GDPR', async () => {
28
- const skill = new ComplianceCheckSkill();
29
- const r = await skill.execute(ctx, msg('GDPR requirements'));
30
- expect(r.handled).toBe(true);
31
- });
32
- });
33
-
34
- describe('Financial Advisor Template', () => {
35
- it('should create valid config', () => {
36
- const config = createFinancialAdvisorConfig();
37
- expect(config.metadata.name).toBe('financial-advisor');
38
- });
39
-
40
- it('BudgetAnalysisSkill matches budget queries', async () => {
41
- const skill = new BudgetAnalysisSkill();
42
- const r = await skill.execute(ctx, msg('Help with my budget'));
43
- expect(r.handled).toBe(true);
44
- });
45
-
46
- it('FinancialPlanningSkill matches investment queries', async () => {
47
- const skill = new FinancialPlanningSkill();
48
- const r = await skill.execute(ctx, msg('How to invest'));
49
- expect(r.handled).toBe(true);
50
- });
51
- });
52
-
53
- describe('Executive Assistant Template', () => {
54
- it('should create valid config', () => {
55
- const config = createExecutiveAssistantConfig();
56
- expect(config.metadata.name).toBe('executive-assistant');
57
- expect(config.spec.skills).toHaveLength(3);
58
- });
59
-
60
- it('CalendarSkill matches scheduling', async () => {
61
- const skill = new CalendarSkill();
62
- const r = await skill.execute(ctx, msg('Schedule a meeting'));
63
- expect(r.handled).toBe(true);
64
- });
65
-
66
- it('EmailDraftSkill matches email queries', async () => {
67
- const skill = new EmailDraftSkill();
68
- const r = await skill.execute(ctx, msg('Draft an email'));
69
- expect(r.handled).toBe(true);
70
- });
71
-
72
- it('MeetingPrepSkill matches prep queries', async () => {
73
- const skill = new MeetingPrepSkill();
74
- const r = await skill.execute(ctx, msg('Prepare the agenda'));
75
- expect(r.handled).toBe(true);
76
- });
77
- });
1
+ import { describe, it, expect } from 'vitest';
2
+ import { createLegalAssistantConfig, ContractReviewSkill, ComplianceCheckSkill } from '../src/templates/legal-assistant';
3
+ import { createFinancialAdvisorConfig, BudgetAnalysisSkill, FinancialPlanningSkill } from '../src/templates/financial-advisor';
4
+ import { createExecutiveAssistantConfig, CalendarSkill, EmailDraftSkill, MeetingPrepSkill } from '../src/templates/executive-assistant';
5
+ import type { AgentContext, Message, MemoryStore } from '../src/core/types';
6
+
7
+ const mockMemory: MemoryStore = {
8
+ get: async () => null, set: async () => {}, getConversation: async () => [],
9
+ addMessage: async () => {}, clear: async () => {},
10
+ };
11
+ const ctx: AgentContext = { agentName: 'test', sessionId: 's1', messages: [], memory: mockMemory, metadata: {} };
12
+ const msg = (content: string): Message => ({ id: '1', role: 'user', content, timestamp: Date.now() });
13
+
14
+ describe('Legal Assistant Template', () => {
15
+ it('should create valid config', () => {
16
+ const config = createLegalAssistantConfig();
17
+ expect(config.metadata.name).toBe('legal-assistant');
18
+ expect(config.spec.skills).toHaveLength(2);
19
+ });
20
+
21
+ it('ContractReviewSkill matches contract terms', async () => {
22
+ const skill = new ContractReviewSkill();
23
+ const r = await skill.execute(ctx, msg('What is force majeure?'));
24
+ expect(r.handled).toBe(true);
25
+ });
26
+
27
+ it('ComplianceCheckSkill matches GDPR', async () => {
28
+ const skill = new ComplianceCheckSkill();
29
+ const r = await skill.execute(ctx, msg('GDPR requirements'));
30
+ expect(r.handled).toBe(true);
31
+ });
32
+ });
33
+
34
+ describe('Financial Advisor Template', () => {
35
+ it('should create valid config', () => {
36
+ const config = createFinancialAdvisorConfig();
37
+ expect(config.metadata.name).toBe('financial-advisor');
38
+ });
39
+
40
+ it('BudgetAnalysisSkill matches budget queries', async () => {
41
+ const skill = new BudgetAnalysisSkill();
42
+ const r = await skill.execute(ctx, msg('Help with my budget'));
43
+ expect(r.handled).toBe(true);
44
+ });
45
+
46
+ it('FinancialPlanningSkill matches investment queries', async () => {
47
+ const skill = new FinancialPlanningSkill();
48
+ const r = await skill.execute(ctx, msg('How to invest'));
49
+ expect(r.handled).toBe(true);
50
+ });
51
+ });
52
+
53
+ describe('Executive Assistant Template', () => {
54
+ it('should create valid config', () => {
55
+ const config = createExecutiveAssistantConfig();
56
+ expect(config.metadata.name).toBe('executive-assistant');
57
+ expect(config.spec.skills).toHaveLength(3);
58
+ });
59
+
60
+ it('CalendarSkill matches scheduling', async () => {
61
+ const skill = new CalendarSkill();
62
+ const r = await skill.execute(ctx, msg('Schedule a meeting'));
63
+ expect(r.handled).toBe(true);
64
+ });
65
+
66
+ it('EmailDraftSkill matches email queries', async () => {
67
+ const skill = new EmailDraftSkill();
68
+ const r = await skill.execute(ctx, msg('Draft an email'));
69
+ expect(r.handled).toBe(true);
70
+ });
71
+
72
+ it('MeetingPrepSkill matches prep queries', async () => {
73
+ const skill = new MeetingPrepSkill();
74
+ const r = await skill.execute(ctx, msg('Prepare the agenda'));
75
+ expect(r.handled).toBe(true);
76
+ });
77
+ });
@@ -1,76 +1,76 @@
1
- import { describe, it, expect } from 'vitest';
2
- import { createAuthMiddleware, getActiveSessions } from '../src/core/auth';
3
- import { HttpSkill } from '../src/skills/http';
4
- import { WebhookTriggerSkill } from '../src/skills/webhook-trigger';
5
- import { SchedulerSkill } from '../src/skills/scheduler';
6
- import type { AgentContext, Message } from '../src/core/types';
7
-
8
- const mockContext: AgentContext = {
9
- agentName: 'test',
10
- sessionId: 'test-session',
11
- messages: [],
12
- memory: { get: async () => null, set: async () => {}, getConversation: async () => [], addMessage: async () => {}, clear: async () => {} },
13
- metadata: {},
14
- };
15
-
16
- function msg(content: string): Message {
17
- return { id: '1', role: 'user', content, timestamp: Date.now() };
18
- }
19
-
20
- describe('Auth', () => {
21
- it('should create middleware', () => {
22
- const mw = createAuthMiddleware({ apiKeys: ['test-key'] });
23
- expect(typeof mw).toBe('function');
24
- });
25
-
26
- it('should track sessions', () => {
27
- const sessions = getActiveSessions();
28
- expect(Array.isArray(sessions)).toBe(true);
29
- });
30
- });
31
-
32
- describe('HttpSkill', () => {
33
- it('should not match non-http messages', async () => {
34
- const skill = new HttpSkill();
35
- const result = await skill.execute(mockContext, msg('hello'));
36
- expect(result.handled).toBe(false);
37
- });
38
- });
39
-
40
- describe('WebhookTriggerSkill', () => {
41
- it('should report unknown webhook', async () => {
42
- const skill = new WebhookTriggerSkill();
43
- const result = await skill.execute(mockContext, msg('webhook test'));
44
- expect(result.handled).toBe(true);
45
- expect(result.response).toContain('Unknown webhook');
46
- });
47
-
48
- it('should register targets', () => {
49
- const skill = new WebhookTriggerSkill();
50
- skill.registerTarget({ name: 'slack', url: 'https://hooks.slack.com/test' });
51
- expect(skill).toBeDefined();
52
- });
53
- });
54
-
55
- describe('SchedulerSkill', () => {
56
- it('should list empty tasks', async () => {
57
- const skill = new SchedulerSkill();
58
- const result = await skill.execute(mockContext, msg('schedule list'));
59
- expect(result.handled).toBe(true);
60
- expect(result.response).toContain('No scheduled tasks');
61
- });
62
-
63
- it('should add a task', async () => {
64
- const skill = new SchedulerSkill();
65
- const result = await skill.execute(mockContext, msg('schedule add backup "every 5m" run backup'));
66
- expect(result.handled).toBe(true);
67
- expect(result.response).toContain('Task scheduled');
68
- skill.destroy();
69
- });
70
-
71
- it('should not match unrelated messages', async () => {
72
- const skill = new SchedulerSkill();
73
- const result = await skill.execute(mockContext, msg('hello'));
74
- expect(result.handled).toBe(false);
75
- });
76
- });
1
+ import { describe, it, expect } from 'vitest';
2
+ import { createAuthMiddleware, getActiveSessions } from '../src/core/auth';
3
+ import { HttpSkill } from '../src/skills/http';
4
+ import { WebhookTriggerSkill } from '../src/skills/webhook-trigger';
5
+ import { SchedulerSkill } from '../src/skills/scheduler';
6
+ import type { AgentContext, Message } from '../src/core/types';
7
+
8
+ const mockContext: AgentContext = {
9
+ agentName: 'test',
10
+ sessionId: 'test-session',
11
+ messages: [],
12
+ memory: { get: async () => null, set: async () => {}, getConversation: async () => [], addMessage: async () => {}, clear: async () => {} },
13
+ metadata: {},
14
+ };
15
+
16
+ function msg(content: string): Message {
17
+ return { id: '1', role: 'user', content, timestamp: Date.now() };
18
+ }
19
+
20
+ describe('Auth', () => {
21
+ it('should create middleware', () => {
22
+ const mw = createAuthMiddleware({ apiKeys: ['test-key'] });
23
+ expect(typeof mw).toBe('function');
24
+ });
25
+
26
+ it('should track sessions', () => {
27
+ const sessions = getActiveSessions();
28
+ expect(Array.isArray(sessions)).toBe(true);
29
+ });
30
+ });
31
+
32
+ describe('HttpSkill', () => {
33
+ it('should not match non-http messages', async () => {
34
+ const skill = new HttpSkill();
35
+ const result = await skill.execute(mockContext, msg('hello'));
36
+ expect(result.handled).toBe(false);
37
+ });
38
+ });
39
+
40
+ describe('WebhookTriggerSkill', () => {
41
+ it('should report unknown webhook', async () => {
42
+ const skill = new WebhookTriggerSkill();
43
+ const result = await skill.execute(mockContext, msg('webhook test'));
44
+ expect(result.handled).toBe(true);
45
+ expect(result.response).toContain('Unknown webhook');
46
+ });
47
+
48
+ it('should register targets', () => {
49
+ const skill = new WebhookTriggerSkill();
50
+ skill.registerTarget({ name: 'slack', url: 'https://hooks.slack.com/test' });
51
+ expect(skill).toBeDefined();
52
+ });
53
+ });
54
+
55
+ describe('SchedulerSkill', () => {
56
+ it('should list empty tasks', async () => {
57
+ const skill = new SchedulerSkill();
58
+ const result = await skill.execute(mockContext, msg('schedule list'));
59
+ expect(result.handled).toBe(true);
60
+ expect(result.response).toContain('No scheduled tasks');
61
+ });
62
+
63
+ it('should add a task', async () => {
64
+ const skill = new SchedulerSkill();
65
+ const result = await skill.execute(mockContext, msg('schedule add backup "every 5m" run backup'));
66
+ expect(result.handled).toBe(true);
67
+ expect(result.response).toContain('Task scheduled');
68
+ skill.destroy();
69
+ });
70
+
71
+ it('should not match unrelated messages', async () => {
72
+ const skill = new SchedulerSkill();
73
+ const result = await skill.execute(mockContext, msg('hello'));
74
+ expect(result.handled).toBe(false);
75
+ });
76
+ });
@@ -1,75 +1,75 @@
1
- import { describe, it, expect, beforeEach, afterEach } from 'vitest';
2
- import { VersionManager } from '../src/core/versioning';
3
- import * as fs from 'fs';
4
-
5
- const TEST_STORE = '.test-versions.json';
6
-
7
- describe('VersionManager', () => {
8
- let vm: VersionManager;
9
-
10
- beforeEach(() => {
11
- if (fs.existsSync(TEST_STORE)) fs.unlinkSync(TEST_STORE);
12
- vm = new VersionManager(TEST_STORE);
13
- });
14
-
15
- afterEach(() => {
16
- if (fs.existsSync(TEST_STORE)) fs.unlinkSync(TEST_STORE);
17
- });
18
-
19
- it('should save and list versions', () => {
20
- vm.snapshot('1.0.0', 'yaml-content-1', 'Initial');
21
- vm.snapshot('1.1.0', 'yaml-content-2', 'Update');
22
- const versions = vm.list();
23
- expect(versions).toHaveLength(2);
24
- expect(versions[0].version).toBe('1.0.0');
25
- });
26
-
27
- it('should get specific version', () => {
28
- vm.snapshot('1.0.0', 'content', 'Test');
29
- expect(vm.get('1.0.0')).toBeDefined();
30
- expect(vm.get('9.9.9')).toBeUndefined();
31
- });
32
-
33
- it('should get current version', () => {
34
- vm.snapshot('1.0.0', 'v1');
35
- vm.snapshot('2.0.0', 'v2');
36
- expect(vm.getCurrent()?.version).toBe('2.0.0');
37
- });
38
-
39
- it('should rollback to version', () => {
40
- vm.snapshot('1.0.0', 'old-yaml');
41
- vm.snapshot('2.0.0', 'new-yaml');
42
- const rolled = vm.rollback('1.0.0');
43
- expect(rolled).toBe('old-yaml');
44
- });
45
-
46
- it('should return null for unknown rollback', () => {
47
- expect(vm.rollback('9.9.9')).toBeNull();
48
- });
49
-
50
- it('should run migrations', () => {
51
- vm.registerMigration({
52
- fromVersion: '1.0.0',
53
- toVersion: '2.0.0',
54
- migrate: (oad) => ({ ...oad, upgraded: true }),
55
- });
56
- const result = vm.migrate({ data: 'test' }, '1.0.0', '2.0.0');
57
- expect((result as any).upgraded).toBe(true);
58
- });
59
-
60
- it('should throw on missing migration path', () => {
61
- expect(() => vm.migrate({}, '1.0.0', '5.0.0')).toThrow('No migration path');
62
- });
63
-
64
- it('should clear all versions', () => {
65
- vm.snapshot('1.0.0', 'content');
66
- vm.clear();
67
- expect(vm.list()).toHaveLength(0);
68
- });
69
-
70
- it('should persist to file', () => {
71
- vm.snapshot('1.0.0', 'persisted');
72
- const vm2 = new VersionManager(TEST_STORE);
73
- expect(vm2.list()).toHaveLength(1);
74
- });
75
- });
1
+ import { describe, it, expect, beforeEach, afterEach } from 'vitest';
2
+ import { VersionManager } from '../src/core/versioning';
3
+ import * as fs from 'fs';
4
+
5
+ const TEST_STORE = '.test-versions.json';
6
+
7
+ describe('VersionManager', () => {
8
+ let vm: VersionManager;
9
+
10
+ beforeEach(() => {
11
+ if (fs.existsSync(TEST_STORE)) fs.unlinkSync(TEST_STORE);
12
+ vm = new VersionManager(TEST_STORE);
13
+ });
14
+
15
+ afterEach(() => {
16
+ if (fs.existsSync(TEST_STORE)) fs.unlinkSync(TEST_STORE);
17
+ });
18
+
19
+ it('should save and list versions', () => {
20
+ vm.snapshot('1.0.0', 'yaml-content-1', 'Initial');
21
+ vm.snapshot('1.1.0', 'yaml-content-2', 'Update');
22
+ const versions = vm.list();
23
+ expect(versions).toHaveLength(2);
24
+ expect(versions[0].version).toBe('1.0.0');
25
+ });
26
+
27
+ it('should get specific version', () => {
28
+ vm.snapshot('1.0.0', 'content', 'Test');
29
+ expect(vm.get('1.0.0')).toBeDefined();
30
+ expect(vm.get('9.9.9')).toBeUndefined();
31
+ });
32
+
33
+ it('should get current version', () => {
34
+ vm.snapshot('1.0.0', 'v1');
35
+ vm.snapshot('2.0.0', 'v2');
36
+ expect(vm.getCurrent()?.version).toBe('2.0.0');
37
+ });
38
+
39
+ it('should rollback to version', () => {
40
+ vm.snapshot('1.0.0', 'old-yaml');
41
+ vm.snapshot('2.0.0', 'new-yaml');
42
+ const rolled = vm.rollback('1.0.0');
43
+ expect(rolled).toBe('old-yaml');
44
+ });
45
+
46
+ it('should return null for unknown rollback', () => {
47
+ expect(vm.rollback('9.9.9')).toBeNull();
48
+ });
49
+
50
+ it('should run migrations', () => {
51
+ vm.registerMigration({
52
+ fromVersion: '1.0.0',
53
+ toVersion: '2.0.0',
54
+ migrate: (oad) => ({ ...oad, upgraded: true }),
55
+ });
56
+ const result = vm.migrate({ data: 'test' }, '1.0.0', '2.0.0');
57
+ expect((result as any).upgraded).toBe(true);
58
+ });
59
+
60
+ it('should throw on missing migration path', () => {
61
+ expect(() => vm.migrate({}, '1.0.0', '5.0.0')).toThrow('No migration path');
62
+ });
63
+
64
+ it('should clear all versions', () => {
65
+ vm.snapshot('1.0.0', 'content');
66
+ vm.clear();
67
+ expect(vm.list()).toHaveLength(0);
68
+ });
69
+
70
+ it('should persist to file', () => {
71
+ vm.snapshot('1.0.0', 'persisted');
72
+ const vm2 = new VersionManager(TEST_STORE);
73
+ expect(vm2.list()).toHaveLength(1);
74
+ });
75
+ });
@@ -1,61 +1,61 @@
1
- import { describe, it, expect } from 'vitest';
2
- import { VoiceChannel } from '../src/channels/voice';
3
- import type { STTProvider, TTSProvider } from '../src/channels/voice';
4
-
5
- describe('VoiceChannel', () => {
6
- it('should start and stop', async () => {
7
- const channel = new VoiceChannel();
8
- await channel.start();
9
- expect(channel.isRunning()).toBe(true);
10
- await channel.stop();
11
- expect(channel.isRunning()).toBe(false);
12
- });
13
-
14
- it('should process audio with STT/TTS providers', async () => {
15
- const stt: STTProvider = {
16
- name: 'mock-stt',
17
- transcribe: async (audio) => audio.toString('utf-8'),
18
- };
19
- const tts: TTSProvider = {
20
- name: 'mock-tts',
21
- synthesize: async (text) => Buffer.from(`audio:${text}`),
22
- };
23
-
24
- const channel = new VoiceChannel({ sttProvider: stt, ttsProvider: tts });
25
- channel.onMessage(async (msg) => ({
26
- id: 'r1', role: 'assistant', content: `Reply: ${msg.content}`, timestamp: Date.now(),
27
- }));
28
- await channel.start();
29
-
30
- const result = await channel.processAudio(Buffer.from('hello'));
31
- expect(result.text).toBe('hello');
32
- expect(result.response).toBe('Reply: hello');
33
- expect(result.audioResponse?.toString()).toBe('audio:Reply: hello');
34
- });
35
-
36
- it('should work without providers (text fallback)', async () => {
37
- const channel = new VoiceChannel();
38
- channel.onMessage(async (msg) => ({
39
- id: 'r1', role: 'assistant', content: msg.content, timestamp: Date.now(),
40
- }));
41
- await channel.start();
42
-
43
- const result = await channel.processAudio(Buffer.from('test'));
44
- expect(result.text).toBe('test');
45
- expect(result.audioResponse).toBeUndefined();
46
- });
47
-
48
- it('should throw without handler', async () => {
49
- const channel = new VoiceChannel();
50
- await expect(channel.processAudio(Buffer.from('test'))).rejects.toThrow('No message handler');
51
- });
52
-
53
- it('should allow setting providers after construction', () => {
54
- const channel = new VoiceChannel();
55
- const stt: STTProvider = { name: 'stt', transcribe: async () => 'text' };
56
- const tts: TTSProvider = { name: 'tts', synthesize: async () => Buffer.from('audio') };
57
- channel.setSTTProvider(stt);
58
- channel.setTTSProvider(tts);
59
- expect(channel.type).toBe('voice');
60
- });
61
- });
1
+ import { describe, it, expect } from 'vitest';
2
+ import { VoiceChannel } from '../src/channels/voice';
3
+ import type { STTProvider, TTSProvider } from '../src/channels/voice';
4
+
5
+ describe('VoiceChannel', () => {
6
+ it('should start and stop', async () => {
7
+ const channel = new VoiceChannel();
8
+ await channel.start();
9
+ expect(channel.isRunning()).toBe(true);
10
+ await channel.stop();
11
+ expect(channel.isRunning()).toBe(false);
12
+ });
13
+
14
+ it('should process audio with STT/TTS providers', async () => {
15
+ const stt: STTProvider = {
16
+ name: 'mock-stt',
17
+ transcribe: async (audio) => audio.toString('utf-8'),
18
+ };
19
+ const tts: TTSProvider = {
20
+ name: 'mock-tts',
21
+ synthesize: async (text) => Buffer.from(`audio:${text}`),
22
+ };
23
+
24
+ const channel = new VoiceChannel({ sttProvider: stt, ttsProvider: tts });
25
+ channel.onMessage(async (msg) => ({
26
+ id: 'r1', role: 'assistant', content: `Reply: ${msg.content}`, timestamp: Date.now(),
27
+ }));
28
+ await channel.start();
29
+
30
+ const result = await channel.processAudio(Buffer.from('hello'));
31
+ expect(result.text).toBe('hello');
32
+ expect(result.response).toBe('Reply: hello');
33
+ expect(result.audioResponse?.toString()).toBe('audio:Reply: hello');
34
+ });
35
+
36
+ it('should work without providers (text fallback)', async () => {
37
+ const channel = new VoiceChannel();
38
+ channel.onMessage(async (msg) => ({
39
+ id: 'r1', role: 'assistant', content: msg.content, timestamp: Date.now(),
40
+ }));
41
+ await channel.start();
42
+
43
+ const result = await channel.processAudio(Buffer.from('test'));
44
+ expect(result.text).toBe('test');
45
+ expect(result.audioResponse).toBeUndefined();
46
+ });
47
+
48
+ it('should throw without handler', async () => {
49
+ const channel = new VoiceChannel();
50
+ await expect(channel.processAudio(Buffer.from('test'))).rejects.toThrow('No message handler');
51
+ });
52
+
53
+ it('should allow setting providers after construction', () => {
54
+ const channel = new VoiceChannel();
55
+ const stt: STTProvider = { name: 'stt', transcribe: async () => 'text' };
56
+ const tts: TTSProvider = { name: 'tts', synthesize: async () => Buffer.from('audio') };
57
+ channel.setSTTProvider(stt);
58
+ channel.setTTSProvider(tts);
59
+ expect(channel.type).toBe('voice');
60
+ });
61
+ });
@@ -1,29 +1,29 @@
1
- import { describe, it, expect } from 'vitest';
2
- import { WebhookChannel } from '../src/channels/webhook';
3
-
4
- describe('WebhookChannel', () => {
5
- it('should create with default config', () => {
6
- const channel = new WebhookChannel();
7
- expect(channel.type).toBe('webhook');
8
- });
9
-
10
- it('should verify signatures', () => {
11
- const channel = new WebhookChannel({ secret: 'test-secret' });
12
- const body = '{"event":"test"}';
13
- const sig = channel.createSignature(body, 'test-secret');
14
- expect(channel.verifySignature(body, sig, 'test-secret')).toBe(true);
15
- expect(channel.verifySignature(body, 'invalid', 'test-secret')).toBe(false);
16
- });
17
-
18
- it('should add outgoing webhooks', () => {
19
- const channel = new WebhookChannel();
20
- channel.addOutgoing({ name: 'test', url: 'http://localhost:9999', events: ['*'], secret: 'abc' });
21
- expect(channel.type).toBe('webhook');
22
- });
23
-
24
- it('should start and stop server', async () => {
25
- const channel = new WebhookChannel({ port: 0 }); // port 0 = random
26
- await channel.start();
27
- await channel.stop();
28
- });
29
- });
1
+ import { describe, it, expect } from 'vitest';
2
+ import { WebhookChannel } from '../src/channels/webhook';
3
+
4
+ describe('WebhookChannel', () => {
5
+ it('should create with default config', () => {
6
+ const channel = new WebhookChannel();
7
+ expect(channel.type).toBe('webhook');
8
+ });
9
+
10
+ it('should verify signatures', () => {
11
+ const channel = new WebhookChannel({ secret: 'test-secret' });
12
+ const body = '{"event":"test"}';
13
+ const sig = channel.createSignature(body, 'test-secret');
14
+ expect(channel.verifySignature(body, sig, 'test-secret')).toBe(true);
15
+ expect(channel.verifySignature(body, 'invalid', 'test-secret')).toBe(false);
16
+ });
17
+
18
+ it('should add outgoing webhooks', () => {
19
+ const channel = new WebhookChannel();
20
+ channel.addOutgoing({ name: 'test', url: 'http://localhost:9999', events: ['*'], secret: 'abc' });
21
+ expect(channel.type).toBe('webhook');
22
+ });
23
+
24
+ it('should start and stop server', async () => {
25
+ const channel = new WebhookChannel({ port: 0 }); // port 0 = random
26
+ await channel.start();
27
+ await channel.stop();
28
+ });
29
+ });