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
@@ -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
+ });