opc-agent 1.2.1 → 1.3.0
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.
- package/CONTRIBUTING.md +75 -75
- package/README.md +235 -358
- package/README.zh-CN.md +415 -415
- package/dist/channels/web.js +256 -256
- package/dist/core/knowledge.d.ts +5 -0
- package/dist/core/knowledge.js +39 -2
- package/dist/deploy/hermes.js +22 -22
- package/dist/deploy/openclaw.js +31 -31
- package/dist/index.d.ts +0 -4
- package/dist/index.js +1 -7
- package/dist/providers/index.d.ts +1 -1
- package/dist/providers/index.js +158 -14
- package/dist/schema/oad.d.ts +3 -3
- package/dist/templates/code-reviewer.js +5 -5
- package/dist/templates/customer-service.js +2 -2
- package/dist/templates/data-analyst.js +5 -5
- package/dist/templates/knowledge-base.js +2 -2
- package/dist/templates/sales-assistant.js +4 -4
- package/dist/templates/teacher.js +6 -6
- package/docs/.vitepress/config.ts +103 -103
- package/docs/api/cli.md +48 -48
- package/docs/api/oad-schema.md +64 -64
- package/docs/api/sdk.md +80 -80
- package/docs/guide/concepts.md +51 -51
- package/docs/guide/configuration.md +79 -79
- package/docs/guide/deployment.md +42 -42
- package/docs/guide/getting-started.md +44 -44
- package/docs/guide/templates.md +28 -28
- package/docs/guide/testing.md +84 -84
- package/docs/index.md +27 -27
- package/docs/zh/api/cli.md +54 -54
- package/docs/zh/api/oad-schema.md +87 -87
- package/docs/zh/api/sdk.md +102 -102
- package/docs/zh/guide/concepts.md +104 -104
- package/docs/zh/guide/configuration.md +135 -135
- package/docs/zh/guide/deployment.md +81 -81
- package/docs/zh/guide/getting-started.md +82 -82
- package/docs/zh/guide/templates.md +84 -84
- package/docs/zh/guide/testing.md +88 -88
- package/docs/zh/index.md +27 -27
- package/examples/customer-service-demo/README.md +90 -90
- package/examples/customer-service-demo/oad.yaml +107 -107
- package/package.json +50 -50
- package/src/analytics/index.ts +66 -66
- package/src/channels/discord.ts +192 -192
- package/src/channels/email.ts +177 -177
- package/src/channels/feishu.ts +236 -236
- package/src/channels/index.ts +15 -15
- package/src/channels/slack.ts +160 -160
- package/src/channels/telegram.ts +90 -90
- package/src/channels/voice.ts +106 -106
- package/src/channels/webhook.ts +199 -199
- package/src/channels/websocket.ts +87 -87
- package/src/channels/wechat.ts +149 -149
- package/src/cli.ts +119 -1
- package/src/core/a2a.ts +143 -143
- package/src/core/agent.ts +152 -152
- package/src/core/analytics-engine.ts +186 -186
- package/src/core/auth.ts +57 -57
- package/src/core/cache.ts +141 -141
- package/src/core/compose.ts +77 -77
- package/src/core/config.ts +14 -14
- package/src/core/errors.ts +148 -148
- package/src/core/hitl.ts +138 -138
- package/src/core/logger.ts +57 -57
- package/src/core/orchestrator.ts +215 -215
- package/src/core/performance.ts +187 -187
- package/src/core/rate-limiter.ts +128 -128
- package/src/core/room.ts +109 -109
- package/src/core/runtime.ts +152 -152
- package/src/core/sandbox.ts +101 -101
- package/src/core/security.ts +171 -171
- package/src/core/types.ts +68 -68
- package/src/core/versioning.ts +106 -106
- package/src/core/watch.ts +178 -178
- package/src/core/workflow.ts +235 -235
- package/src/deploy/hermes.ts +156 -156
- package/src/deploy/openclaw.ts +200 -200
- package/src/i18n/index.ts +216 -216
- package/src/index.ts +6 -2
- package/src/memory/deepbrain.ts +108 -108
- package/src/memory/index.ts +34 -34
- package/src/plugins/index.ts +208 -208
- package/src/schema/oad.ts +154 -155
- package/src/skills/base.ts +16 -16
- package/src/skills/document.ts +100 -100
- package/src/skills/http.ts +35 -35
- package/src/skills/index.ts +27 -27
- package/src/skills/scheduler.ts +80 -80
- package/src/skills/webhook-trigger.ts +59 -59
- package/src/templates/code-reviewer.ts +30 -34
- package/src/templates/customer-service.ts +76 -80
- package/src/templates/data-analyst.ts +66 -70
- package/src/templates/executive-assistant.ts +71 -71
- package/src/templates/financial-advisor.ts +60 -60
- package/src/templates/knowledge-base.ts +27 -31
- package/src/templates/legal-assistant.ts +71 -71
- package/src/templates/sales-assistant.ts +75 -79
- package/src/templates/teacher.ts +75 -79
- package/src/testing/index.ts +181 -181
- package/src/tools/calculator.ts +73 -73
- package/src/tools/datetime.ts +149 -149
- package/src/tools/json-transform.ts +187 -187
- package/src/tools/mcp.ts +76 -76
- package/src/tools/text-analysis.ts +116 -116
- package/src/traces/index.ts +132 -0
- package/templates/Dockerfile +15 -15
- package/templates/code-reviewer/README.md +27 -27
- package/templates/code-reviewer/oad.yaml +41 -41
- package/templates/customer-service/README.md +22 -22
- package/templates/customer-service/oad.yaml +36 -36
- package/templates/docker-compose.yml +21 -21
- package/templates/ecommerce-assistant/README.md +45 -45
- package/templates/ecommerce-assistant/oad.yaml +47 -47
- package/templates/knowledge-base/README.md +28 -28
- package/templates/knowledge-base/oad.yaml +38 -38
- package/templates/sales-assistant/README.md +26 -26
- package/templates/sales-assistant/oad.yaml +43 -43
- package/templates/tech-support/README.md +43 -43
- package/templates/tech-support/oad.yaml +45 -45
- package/tests/a2a.test.ts +66 -66
- package/tests/agent.test.ts +72 -72
- package/tests/analytics.test.ts +50 -50
- package/tests/channel.test.ts +39 -39
- package/tests/e2e.test.ts +134 -134
- package/tests/errors.test.ts +83 -83
- package/tests/hitl.test.ts +71 -71
- package/tests/i18n.test.ts +41 -41
- package/tests/mcp.test.ts +54 -54
- package/tests/oad.test.ts +68 -68
- package/tests/performance.test.ts +115 -115
- package/tests/plugin.test.ts +74 -74
- package/tests/room.test.ts +106 -106
- package/tests/runtime.test.ts +42 -42
- package/tests/sandbox.test.ts +46 -46
- package/tests/security.test.ts +60 -60
- package/tests/templates.test.ts +77 -77
- package/tests/v070.test.ts +76 -76
- package/tests/versioning.test.ts +75 -75
- package/tests/voice.test.ts +61 -61
- package/tests/webhook.test.ts +29 -29
- package/tests/workflow.test.ts +143 -143
- package/tsconfig.json +19 -19
- package/vitest.config.ts +9 -9
- package/dist/core/streaming.d.ts +0 -56
- package/dist/core/streaming.js +0 -160
- package/dist/tools/gateway.d.ts +0 -28
- package/dist/tools/gateway.js +0 -177
- package/src/dtv/data.ts +0 -29
- package/src/dtv/trust.ts +0 -43
- package/src/dtv/value.ts +0 -47
- package/src/marketplace/index.ts +0 -223
package/tests/workflow.test.ts
CHANGED
|
@@ -1,143 +1,143 @@
|
|
|
1
|
-
import { describe, it, expect, beforeEach } from 'vitest';
|
|
2
|
-
import { WorkflowEngine, WorkflowDefinition } from '../src/core/workflow';
|
|
3
|
-
import { BaseSkill } from '../src/skills/base';
|
|
4
|
-
import type { AgentContext, Message, SkillResult, MemoryStore } from '../src/core/types';
|
|
5
|
-
|
|
6
|
-
class EchoSkill extends BaseSkill {
|
|
7
|
-
name = 'echo';
|
|
8
|
-
description = 'Echo input';
|
|
9
|
-
async execute(_ctx: AgentContext, msg: Message): Promise<SkillResult> {
|
|
10
|
-
return this.match(`echo:${msg.content}`, 1.0);
|
|
11
|
-
}
|
|
12
|
-
}
|
|
13
|
-
|
|
14
|
-
class UpperSkill extends BaseSkill {
|
|
15
|
-
name = 'upper';
|
|
16
|
-
description = 'Uppercase';
|
|
17
|
-
async execute(_ctx: AgentContext, msg: Message): Promise<SkillResult> {
|
|
18
|
-
return this.match(msg.content.toUpperCase(), 1.0);
|
|
19
|
-
}
|
|
20
|
-
}
|
|
21
|
-
|
|
22
|
-
class FailSkill extends BaseSkill {
|
|
23
|
-
name = 'fail';
|
|
24
|
-
description = 'Always fails';
|
|
25
|
-
async execute(): Promise<SkillResult> {
|
|
26
|
-
throw new Error('intentional failure');
|
|
27
|
-
}
|
|
28
|
-
}
|
|
29
|
-
|
|
30
|
-
const mockMemory: MemoryStore = {
|
|
31
|
-
get: async () => null,
|
|
32
|
-
set: async () => {},
|
|
33
|
-
getConversation: async () => [],
|
|
34
|
-
addMessage: async () => {},
|
|
35
|
-
clear: async () => {},
|
|
36
|
-
};
|
|
37
|
-
|
|
38
|
-
const mockContext: AgentContext = {
|
|
39
|
-
agentName: 'test',
|
|
40
|
-
sessionId: 'test-session',
|
|
41
|
-
messages: [],
|
|
42
|
-
memory: mockMemory,
|
|
43
|
-
metadata: {},
|
|
44
|
-
};
|
|
45
|
-
|
|
46
|
-
describe('WorkflowEngine', () => {
|
|
47
|
-
let engine: WorkflowEngine;
|
|
48
|
-
|
|
49
|
-
beforeEach(() => {
|
|
50
|
-
engine = new WorkflowEngine();
|
|
51
|
-
engine.registerSkill(new EchoSkill());
|
|
52
|
-
engine.registerSkill(new UpperSkill());
|
|
53
|
-
engine.registerSkill(new FailSkill());
|
|
54
|
-
});
|
|
55
|
-
|
|
56
|
-
it('should register and list workflows', () => {
|
|
57
|
-
const wf: WorkflowDefinition = { name: 'test', steps: [] };
|
|
58
|
-
engine.registerWorkflow(wf);
|
|
59
|
-
expect(engine.listWorkflows()).toHaveLength(1);
|
|
60
|
-
expect(engine.getWorkflow('test')).toBeDefined();
|
|
61
|
-
});
|
|
62
|
-
|
|
63
|
-
it('should run sequential steps', async () => {
|
|
64
|
-
const wf: WorkflowDefinition = {
|
|
65
|
-
name: 'seq',
|
|
66
|
-
steps: [
|
|
67
|
-
{ id: 's1', type: 'skill', name: 'echo' },
|
|
68
|
-
{ id: 's2', type: 'skill', name: 'upper' },
|
|
69
|
-
],
|
|
70
|
-
};
|
|
71
|
-
engine.registerWorkflow(wf);
|
|
72
|
-
const result = await engine.run('seq', mockContext, 'hello');
|
|
73
|
-
expect(result.status).toBe('completed');
|
|
74
|
-
expect(result.steps).toHaveLength(2);
|
|
75
|
-
});
|
|
76
|
-
|
|
77
|
-
it('should handle parallel steps', async () => {
|
|
78
|
-
const wf: WorkflowDefinition = {
|
|
79
|
-
name: 'par',
|
|
80
|
-
steps: [{
|
|
81
|
-
id: 'p1', type: 'parallel', name: 'parallel',
|
|
82
|
-
parallel: [
|
|
83
|
-
{ id: 's1', type: 'skill', name: 'echo' },
|
|
84
|
-
{ id: 's2', type: 'skill', name: 'upper' },
|
|
85
|
-
],
|
|
86
|
-
}],
|
|
87
|
-
};
|
|
88
|
-
engine.registerWorkflow(wf);
|
|
89
|
-
const result = await engine.run('par', mockContext, 'test');
|
|
90
|
-
expect(result.status).toBe('completed');
|
|
91
|
-
});
|
|
92
|
-
|
|
93
|
-
it('should handle conditional branching', async () => {
|
|
94
|
-
const wf: WorkflowDefinition = {
|
|
95
|
-
name: 'cond',
|
|
96
|
-
steps: [{
|
|
97
|
-
id: 'c1', type: 'condition', name: 'check',
|
|
98
|
-
condition: 'contains:hello',
|
|
99
|
-
branches: {
|
|
100
|
-
if: [{ id: 's1', type: 'skill', name: 'echo' }],
|
|
101
|
-
else: [{ id: 's2', type: 'skill', name: 'upper' }],
|
|
102
|
-
},
|
|
103
|
-
}],
|
|
104
|
-
};
|
|
105
|
-
engine.registerWorkflow(wf);
|
|
106
|
-
const r1 = await engine.run('cond', mockContext, 'hello world');
|
|
107
|
-
expect(r1.steps.find(s => s.stepId === 's1')).toBeDefined();
|
|
108
|
-
});
|
|
109
|
-
|
|
110
|
-
it('should handle step failures with stop policy', async () => {
|
|
111
|
-
const wf: WorkflowDefinition = {
|
|
112
|
-
name: 'fail-wf',
|
|
113
|
-
onError: 'stop',
|
|
114
|
-
steps: [
|
|
115
|
-
{ id: 's1', type: 'skill', name: 'fail' },
|
|
116
|
-
{ id: 's2', type: 'skill', name: 'echo' },
|
|
117
|
-
],
|
|
118
|
-
};
|
|
119
|
-
engine.registerWorkflow(wf);
|
|
120
|
-
const result = await engine.run('fail-wf', mockContext, 'test');
|
|
121
|
-
expect(result.status).toBe('failed');
|
|
122
|
-
});
|
|
123
|
-
|
|
124
|
-
it('should retry failed steps', async () => {
|
|
125
|
-
const wf: WorkflowDefinition = {
|
|
126
|
-
name: 'retry-wf',
|
|
127
|
-
steps: [{ id: 's1', type: 'skill', name: 'fail', retries: 2 }],
|
|
128
|
-
};
|
|
129
|
-
engine.registerWorkflow(wf);
|
|
130
|
-
const result = await engine.run('retry-wf', mockContext, 'test');
|
|
131
|
-
expect(result.steps[0].status).toBe('error');
|
|
132
|
-
});
|
|
133
|
-
|
|
134
|
-
it('should throw on unknown workflow', async () => {
|
|
135
|
-
await expect(engine.run('nonexistent', mockContext)).rejects.toThrow();
|
|
136
|
-
});
|
|
137
|
-
|
|
138
|
-
it('should unregister workflows', () => {
|
|
139
|
-
engine.registerWorkflow({ name: 'temp', steps: [] });
|
|
140
|
-
engine.unregisterWorkflow('temp');
|
|
141
|
-
expect(engine.getWorkflow('temp')).toBeUndefined();
|
|
142
|
-
});
|
|
143
|
-
});
|
|
1
|
+
import { describe, it, expect, beforeEach } from 'vitest';
|
|
2
|
+
import { WorkflowEngine, WorkflowDefinition } from '../src/core/workflow';
|
|
3
|
+
import { BaseSkill } from '../src/skills/base';
|
|
4
|
+
import type { AgentContext, Message, SkillResult, MemoryStore } from '../src/core/types';
|
|
5
|
+
|
|
6
|
+
class EchoSkill extends BaseSkill {
|
|
7
|
+
name = 'echo';
|
|
8
|
+
description = 'Echo input';
|
|
9
|
+
async execute(_ctx: AgentContext, msg: Message): Promise<SkillResult> {
|
|
10
|
+
return this.match(`echo:${msg.content}`, 1.0);
|
|
11
|
+
}
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
class UpperSkill extends BaseSkill {
|
|
15
|
+
name = 'upper';
|
|
16
|
+
description = 'Uppercase';
|
|
17
|
+
async execute(_ctx: AgentContext, msg: Message): Promise<SkillResult> {
|
|
18
|
+
return this.match(msg.content.toUpperCase(), 1.0);
|
|
19
|
+
}
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
class FailSkill extends BaseSkill {
|
|
23
|
+
name = 'fail';
|
|
24
|
+
description = 'Always fails';
|
|
25
|
+
async execute(): Promise<SkillResult> {
|
|
26
|
+
throw new Error('intentional failure');
|
|
27
|
+
}
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
const mockMemory: MemoryStore = {
|
|
31
|
+
get: async () => null,
|
|
32
|
+
set: async () => {},
|
|
33
|
+
getConversation: async () => [],
|
|
34
|
+
addMessage: async () => {},
|
|
35
|
+
clear: async () => {},
|
|
36
|
+
};
|
|
37
|
+
|
|
38
|
+
const mockContext: AgentContext = {
|
|
39
|
+
agentName: 'test',
|
|
40
|
+
sessionId: 'test-session',
|
|
41
|
+
messages: [],
|
|
42
|
+
memory: mockMemory,
|
|
43
|
+
metadata: {},
|
|
44
|
+
};
|
|
45
|
+
|
|
46
|
+
describe('WorkflowEngine', () => {
|
|
47
|
+
let engine: WorkflowEngine;
|
|
48
|
+
|
|
49
|
+
beforeEach(() => {
|
|
50
|
+
engine = new WorkflowEngine();
|
|
51
|
+
engine.registerSkill(new EchoSkill());
|
|
52
|
+
engine.registerSkill(new UpperSkill());
|
|
53
|
+
engine.registerSkill(new FailSkill());
|
|
54
|
+
});
|
|
55
|
+
|
|
56
|
+
it('should register and list workflows', () => {
|
|
57
|
+
const wf: WorkflowDefinition = { name: 'test', steps: [] };
|
|
58
|
+
engine.registerWorkflow(wf);
|
|
59
|
+
expect(engine.listWorkflows()).toHaveLength(1);
|
|
60
|
+
expect(engine.getWorkflow('test')).toBeDefined();
|
|
61
|
+
});
|
|
62
|
+
|
|
63
|
+
it('should run sequential steps', async () => {
|
|
64
|
+
const wf: WorkflowDefinition = {
|
|
65
|
+
name: 'seq',
|
|
66
|
+
steps: [
|
|
67
|
+
{ id: 's1', type: 'skill', name: 'echo' },
|
|
68
|
+
{ id: 's2', type: 'skill', name: 'upper' },
|
|
69
|
+
],
|
|
70
|
+
};
|
|
71
|
+
engine.registerWorkflow(wf);
|
|
72
|
+
const result = await engine.run('seq', mockContext, 'hello');
|
|
73
|
+
expect(result.status).toBe('completed');
|
|
74
|
+
expect(result.steps).toHaveLength(2);
|
|
75
|
+
});
|
|
76
|
+
|
|
77
|
+
it('should handle parallel steps', async () => {
|
|
78
|
+
const wf: WorkflowDefinition = {
|
|
79
|
+
name: 'par',
|
|
80
|
+
steps: [{
|
|
81
|
+
id: 'p1', type: 'parallel', name: 'parallel',
|
|
82
|
+
parallel: [
|
|
83
|
+
{ id: 's1', type: 'skill', name: 'echo' },
|
|
84
|
+
{ id: 's2', type: 'skill', name: 'upper' },
|
|
85
|
+
],
|
|
86
|
+
}],
|
|
87
|
+
};
|
|
88
|
+
engine.registerWorkflow(wf);
|
|
89
|
+
const result = await engine.run('par', mockContext, 'test');
|
|
90
|
+
expect(result.status).toBe('completed');
|
|
91
|
+
});
|
|
92
|
+
|
|
93
|
+
it('should handle conditional branching', async () => {
|
|
94
|
+
const wf: WorkflowDefinition = {
|
|
95
|
+
name: 'cond',
|
|
96
|
+
steps: [{
|
|
97
|
+
id: 'c1', type: 'condition', name: 'check',
|
|
98
|
+
condition: 'contains:hello',
|
|
99
|
+
branches: {
|
|
100
|
+
if: [{ id: 's1', type: 'skill', name: 'echo' }],
|
|
101
|
+
else: [{ id: 's2', type: 'skill', name: 'upper' }],
|
|
102
|
+
},
|
|
103
|
+
}],
|
|
104
|
+
};
|
|
105
|
+
engine.registerWorkflow(wf);
|
|
106
|
+
const r1 = await engine.run('cond', mockContext, 'hello world');
|
|
107
|
+
expect(r1.steps.find(s => s.stepId === 's1')).toBeDefined();
|
|
108
|
+
});
|
|
109
|
+
|
|
110
|
+
it('should handle step failures with stop policy', async () => {
|
|
111
|
+
const wf: WorkflowDefinition = {
|
|
112
|
+
name: 'fail-wf',
|
|
113
|
+
onError: 'stop',
|
|
114
|
+
steps: [
|
|
115
|
+
{ id: 's1', type: 'skill', name: 'fail' },
|
|
116
|
+
{ id: 's2', type: 'skill', name: 'echo' },
|
|
117
|
+
],
|
|
118
|
+
};
|
|
119
|
+
engine.registerWorkflow(wf);
|
|
120
|
+
const result = await engine.run('fail-wf', mockContext, 'test');
|
|
121
|
+
expect(result.status).toBe('failed');
|
|
122
|
+
});
|
|
123
|
+
|
|
124
|
+
it('should retry failed steps', async () => {
|
|
125
|
+
const wf: WorkflowDefinition = {
|
|
126
|
+
name: 'retry-wf',
|
|
127
|
+
steps: [{ id: 's1', type: 'skill', name: 'fail', retries: 2 }],
|
|
128
|
+
};
|
|
129
|
+
engine.registerWorkflow(wf);
|
|
130
|
+
const result = await engine.run('retry-wf', mockContext, 'test');
|
|
131
|
+
expect(result.steps[0].status).toBe('error');
|
|
132
|
+
});
|
|
133
|
+
|
|
134
|
+
it('should throw on unknown workflow', async () => {
|
|
135
|
+
await expect(engine.run('nonexistent', mockContext)).rejects.toThrow();
|
|
136
|
+
});
|
|
137
|
+
|
|
138
|
+
it('should unregister workflows', () => {
|
|
139
|
+
engine.registerWorkflow({ name: 'temp', steps: [] });
|
|
140
|
+
engine.unregisterWorkflow('temp');
|
|
141
|
+
expect(engine.getWorkflow('temp')).toBeUndefined();
|
|
142
|
+
});
|
|
143
|
+
});
|
package/tsconfig.json
CHANGED
|
@@ -1,19 +1,19 @@
|
|
|
1
|
-
{
|
|
2
|
-
"compilerOptions": {
|
|
3
|
-
"target": "ES2022",
|
|
4
|
-
"module": "commonjs",
|
|
5
|
-
"lib": ["ES2022"],
|
|
6
|
-
"outDir": "dist",
|
|
7
|
-
"rootDir": "src",
|
|
8
|
-
"strict": true,
|
|
9
|
-
"esModuleInterop": true,
|
|
10
|
-
"skipLibCheck": true,
|
|
11
|
-
"forceConsistentCasingInFileNames": true,
|
|
12
|
-
"resolveJsonModule": true,
|
|
13
|
-
"declaration": true,
|
|
14
|
-
"declarationMap": true,
|
|
15
|
-
"sourceMap": true
|
|
16
|
-
},
|
|
17
|
-
"include": ["src/**/*"],
|
|
18
|
-
"exclude": ["node_modules", "dist", "tests"]
|
|
19
|
-
}
|
|
1
|
+
{
|
|
2
|
+
"compilerOptions": {
|
|
3
|
+
"target": "ES2022",
|
|
4
|
+
"module": "commonjs",
|
|
5
|
+
"lib": ["ES2022"],
|
|
6
|
+
"outDir": "dist",
|
|
7
|
+
"rootDir": "src",
|
|
8
|
+
"strict": true,
|
|
9
|
+
"esModuleInterop": true,
|
|
10
|
+
"skipLibCheck": true,
|
|
11
|
+
"forceConsistentCasingInFileNames": true,
|
|
12
|
+
"resolveJsonModule": true,
|
|
13
|
+
"declaration": true,
|
|
14
|
+
"declarationMap": true,
|
|
15
|
+
"sourceMap": true
|
|
16
|
+
},
|
|
17
|
+
"include": ["src/**/*"],
|
|
18
|
+
"exclude": ["node_modules", "dist", "tests"]
|
|
19
|
+
}
|
package/vitest.config.ts
CHANGED
|
@@ -1,9 +1,9 @@
|
|
|
1
|
-
import { defineConfig } from 'vitest/config';
|
|
2
|
-
|
|
3
|
-
export default defineConfig({
|
|
4
|
-
test: {
|
|
5
|
-
globals: true,
|
|
6
|
-
environment: 'node',
|
|
7
|
-
include: ['tests/**/*.test.ts'],
|
|
8
|
-
},
|
|
9
|
-
});
|
|
1
|
+
import { defineConfig } from 'vitest/config';
|
|
2
|
+
|
|
3
|
+
export default defineConfig({
|
|
4
|
+
test: {
|
|
5
|
+
globals: true,
|
|
6
|
+
environment: 'node',
|
|
7
|
+
include: ['tests/**/*.test.ts'],
|
|
8
|
+
},
|
|
9
|
+
});
|
package/dist/core/streaming.d.ts
DELETED
|
@@ -1,56 +0,0 @@
|
|
|
1
|
-
import { EventEmitter } from 'events';
|
|
2
|
-
export interface StreamChunk {
|
|
3
|
-
id: string;
|
|
4
|
-
type: 'text' | 'tool_call' | 'error' | 'done';
|
|
5
|
-
data: string;
|
|
6
|
-
timestamp: number;
|
|
7
|
-
metadata?: Record<string, unknown>;
|
|
8
|
-
}
|
|
9
|
-
export interface StreamOptions {
|
|
10
|
-
/** High-water mark for backpressure (default 64 chunks). */
|
|
11
|
-
highWaterMark?: number;
|
|
12
|
-
/** Heartbeat interval in ms to keep connection alive (default 15000). */
|
|
13
|
-
heartbeatInterval?: number;
|
|
14
|
-
}
|
|
15
|
-
export declare class StreamableResponse extends EventEmitter {
|
|
16
|
-
readonly id: string;
|
|
17
|
-
private chunks;
|
|
18
|
-
private ended;
|
|
19
|
-
private paused;
|
|
20
|
-
private buffer;
|
|
21
|
-
private highWaterMark;
|
|
22
|
-
constructor(id: string, options?: StreamOptions);
|
|
23
|
-
/** Push a chunk. Returns false if backpressure threshold reached. */
|
|
24
|
-
push(chunk: StreamChunk): boolean;
|
|
25
|
-
/** Resume after backpressure — flush buffered chunks. */
|
|
26
|
-
resume(): void;
|
|
27
|
-
end(): void;
|
|
28
|
-
getChunks(): StreamChunk[];
|
|
29
|
-
/** Collect all text chunks into a single string. */
|
|
30
|
-
getText(): string;
|
|
31
|
-
get isEnded(): boolean;
|
|
32
|
-
get isPaused(): boolean;
|
|
33
|
-
get length(): number;
|
|
34
|
-
}
|
|
35
|
-
export declare class StreamingManager {
|
|
36
|
-
private streams;
|
|
37
|
-
private counter;
|
|
38
|
-
/** Create a new stream. */
|
|
39
|
-
createStream(options?: StreamOptions): StreamableResponse;
|
|
40
|
-
/** Write a text chunk to a stream. */
|
|
41
|
-
writeChunk(streamId: string, data: string, metadata?: Record<string, unknown>): boolean;
|
|
42
|
-
/** End a stream. */
|
|
43
|
-
endStream(streamId: string): void;
|
|
44
|
-
/** Get an existing stream. */
|
|
45
|
-
getStream(streamId: string): StreamableResponse | undefined;
|
|
46
|
-
/** Format a chunk as an SSE event string. */
|
|
47
|
-
static formatSSE(chunk: StreamChunk): string;
|
|
48
|
-
/** Pipe a stream to an SSE-compatible HTTP response (Express-style). */
|
|
49
|
-
static pipeSSE(stream: StreamableResponse, res: {
|
|
50
|
-
write(data: string): boolean;
|
|
51
|
-
end(): void;
|
|
52
|
-
setHeader?(name: string, value: string): void;
|
|
53
|
-
}, options?: StreamOptions): void;
|
|
54
|
-
get activeCount(): number;
|
|
55
|
-
}
|
|
56
|
-
//# sourceMappingURL=streaming.d.ts.map
|
package/dist/core/streaming.js
DELETED
|
@@ -1,160 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.StreamingManager = exports.StreamableResponse = void 0;
|
|
4
|
-
const events_1 = require("events");
|
|
5
|
-
// ─── StreamableResponse ──────────────────────────────────────
|
|
6
|
-
class StreamableResponse extends events_1.EventEmitter {
|
|
7
|
-
id;
|
|
8
|
-
chunks = [];
|
|
9
|
-
ended = false;
|
|
10
|
-
paused = false;
|
|
11
|
-
buffer = [];
|
|
12
|
-
highWaterMark;
|
|
13
|
-
constructor(id, options) {
|
|
14
|
-
super();
|
|
15
|
-
this.id = id;
|
|
16
|
-
this.highWaterMark = options?.highWaterMark ?? 64;
|
|
17
|
-
}
|
|
18
|
-
/** Push a chunk. Returns false if backpressure threshold reached. */
|
|
19
|
-
push(chunk) {
|
|
20
|
-
if (this.ended)
|
|
21
|
-
return false;
|
|
22
|
-
this.chunks.push(chunk);
|
|
23
|
-
if (this.paused) {
|
|
24
|
-
this.buffer.push(chunk);
|
|
25
|
-
return this.buffer.length < this.highWaterMark;
|
|
26
|
-
}
|
|
27
|
-
this.emit('chunk', chunk);
|
|
28
|
-
if (this.chunks.length >= this.highWaterMark) {
|
|
29
|
-
this.paused = true;
|
|
30
|
-
this.emit('backpressure');
|
|
31
|
-
return false;
|
|
32
|
-
}
|
|
33
|
-
return true;
|
|
34
|
-
}
|
|
35
|
-
/** Resume after backpressure — flush buffered chunks. */
|
|
36
|
-
resume() {
|
|
37
|
-
if (!this.paused)
|
|
38
|
-
return;
|
|
39
|
-
this.paused = false;
|
|
40
|
-
const buffered = this.buffer.splice(0);
|
|
41
|
-
for (const chunk of buffered) {
|
|
42
|
-
this.emit('chunk', chunk);
|
|
43
|
-
}
|
|
44
|
-
this.emit('drain');
|
|
45
|
-
}
|
|
46
|
-
end() {
|
|
47
|
-
if (this.ended)
|
|
48
|
-
return;
|
|
49
|
-
this.ended = true;
|
|
50
|
-
if (this.paused)
|
|
51
|
-
this.resume();
|
|
52
|
-
this.emit('end');
|
|
53
|
-
}
|
|
54
|
-
getChunks() {
|
|
55
|
-
return [...this.chunks];
|
|
56
|
-
}
|
|
57
|
-
/** Collect all text chunks into a single string. */
|
|
58
|
-
getText() {
|
|
59
|
-
return this.chunks
|
|
60
|
-
.filter((c) => c.type === 'text')
|
|
61
|
-
.map((c) => c.data)
|
|
62
|
-
.join('');
|
|
63
|
-
}
|
|
64
|
-
get isEnded() {
|
|
65
|
-
return this.ended;
|
|
66
|
-
}
|
|
67
|
-
get isPaused() {
|
|
68
|
-
return this.paused;
|
|
69
|
-
}
|
|
70
|
-
get length() {
|
|
71
|
-
return this.chunks.length;
|
|
72
|
-
}
|
|
73
|
-
}
|
|
74
|
-
exports.StreamableResponse = StreamableResponse;
|
|
75
|
-
// ─── StreamingManager ────────────────────────────────────────
|
|
76
|
-
class StreamingManager {
|
|
77
|
-
streams = new Map();
|
|
78
|
-
counter = 0;
|
|
79
|
-
/** Create a new stream. */
|
|
80
|
-
createStream(options) {
|
|
81
|
-
const id = `stream_${++this.counter}_${Date.now()}`;
|
|
82
|
-
const stream = new StreamableResponse(id, options);
|
|
83
|
-
this.streams.set(id, stream);
|
|
84
|
-
stream.on('end', () => {
|
|
85
|
-
// Keep ended streams for a bit for late consumers, then clean up
|
|
86
|
-
setTimeout(() => this.streams.delete(id), 30_000);
|
|
87
|
-
});
|
|
88
|
-
return stream;
|
|
89
|
-
}
|
|
90
|
-
/** Write a text chunk to a stream. */
|
|
91
|
-
writeChunk(streamId, data, metadata) {
|
|
92
|
-
const stream = this.streams.get(streamId);
|
|
93
|
-
if (!stream)
|
|
94
|
-
return false;
|
|
95
|
-
return stream.push({
|
|
96
|
-
id: `chunk_${stream.length}`,
|
|
97
|
-
type: 'text',
|
|
98
|
-
data,
|
|
99
|
-
timestamp: Date.now(),
|
|
100
|
-
metadata,
|
|
101
|
-
});
|
|
102
|
-
}
|
|
103
|
-
/** End a stream. */
|
|
104
|
-
endStream(streamId) {
|
|
105
|
-
const stream = this.streams.get(streamId);
|
|
106
|
-
if (!stream)
|
|
107
|
-
return;
|
|
108
|
-
stream.push({
|
|
109
|
-
id: `chunk_${stream.length}`,
|
|
110
|
-
type: 'done',
|
|
111
|
-
data: '',
|
|
112
|
-
timestamp: Date.now(),
|
|
113
|
-
});
|
|
114
|
-
stream.end();
|
|
115
|
-
}
|
|
116
|
-
/** Get an existing stream. */
|
|
117
|
-
getStream(streamId) {
|
|
118
|
-
return this.streams.get(streamId);
|
|
119
|
-
}
|
|
120
|
-
/** Format a chunk as an SSE event string. */
|
|
121
|
-
static formatSSE(chunk) {
|
|
122
|
-
const lines = [];
|
|
123
|
-
lines.push(`event: ${chunk.type}`);
|
|
124
|
-
lines.push(`id: ${chunk.id}`);
|
|
125
|
-
const payload = JSON.stringify({ data: chunk.data, metadata: chunk.metadata });
|
|
126
|
-
lines.push(`data: ${payload}`);
|
|
127
|
-
lines.push('');
|
|
128
|
-
return lines.join('\n') + '\n';
|
|
129
|
-
}
|
|
130
|
-
/** Pipe a stream to an SSE-compatible HTTP response (Express-style). */
|
|
131
|
-
static pipeSSE(stream, res, options) {
|
|
132
|
-
res.setHeader?.('Content-Type', 'text/event-stream');
|
|
133
|
-
res.setHeader?.('Cache-Control', 'no-cache');
|
|
134
|
-
res.setHeader?.('Connection', 'keep-alive');
|
|
135
|
-
const heartbeatMs = options?.heartbeatInterval ?? 15_000;
|
|
136
|
-
const heartbeat = setInterval(() => {
|
|
137
|
-
res.write(': heartbeat\n\n');
|
|
138
|
-
}, heartbeatMs);
|
|
139
|
-
stream.on('chunk', (chunk) => {
|
|
140
|
-
const ok = res.write(StreamingManager.formatSSE(chunk));
|
|
141
|
-
if (!ok && stream.isPaused === false) {
|
|
142
|
-
// Downstream can't keep up — will resume on drain from stream
|
|
143
|
-
}
|
|
144
|
-
});
|
|
145
|
-
stream.on('end', () => {
|
|
146
|
-
clearInterval(heartbeat);
|
|
147
|
-
res.end();
|
|
148
|
-
});
|
|
149
|
-
}
|
|
150
|
-
get activeCount() {
|
|
151
|
-
let count = 0;
|
|
152
|
-
for (const s of this.streams.values()) {
|
|
153
|
-
if (!s.isEnded)
|
|
154
|
-
count++;
|
|
155
|
-
}
|
|
156
|
-
return count;
|
|
157
|
-
}
|
|
158
|
-
}
|
|
159
|
-
exports.StreamingManager = StreamingManager;
|
|
160
|
-
//# sourceMappingURL=streaming.js.map
|
package/dist/tools/gateway.d.ts
DELETED
|
@@ -1,28 +0,0 @@
|
|
|
1
|
-
import type { MCPTool, MCPToolDefinition, MCPToolResult } from './mcp';
|
|
2
|
-
export interface ToolGatewayConfig {
|
|
3
|
-
enabled: boolean;
|
|
4
|
-
endpoint: string;
|
|
5
|
-
apiKey: string;
|
|
6
|
-
enabledTools?: GatewayToolName[];
|
|
7
|
-
timeout?: number;
|
|
8
|
-
}
|
|
9
|
-
export type GatewayToolName = 'web-search' | 'image-gen' | 'tts' | 'browser';
|
|
10
|
-
export declare class ToolGateway {
|
|
11
|
-
private config;
|
|
12
|
-
private availableTools;
|
|
13
|
-
private connected;
|
|
14
|
-
constructor(config: ToolGatewayConfig);
|
|
15
|
-
/** Discover available tools from the gateway endpoint. */
|
|
16
|
-
connect(): Promise<void>;
|
|
17
|
-
/** Load default tool definitions (used as fallback). */
|
|
18
|
-
private loadDefaults;
|
|
19
|
-
/** Invoke a tool through the gateway. */
|
|
20
|
-
invokeTool(name: GatewayToolName, input: Record<string, unknown>): Promise<MCPToolResult>;
|
|
21
|
-
/** Get all gateway tools as MCPTool instances for registry integration. */
|
|
22
|
-
getTools(): MCPTool[];
|
|
23
|
-
/** Get tool definitions (without execute). */
|
|
24
|
-
listTools(): MCPToolDefinition[];
|
|
25
|
-
get isConnected(): boolean;
|
|
26
|
-
get toolCount(): number;
|
|
27
|
-
}
|
|
28
|
-
//# sourceMappingURL=gateway.d.ts.map
|