opc-agent 4.1.0 → 4.1.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.
- package/.github/ISSUE_TEMPLATE/bug_report.md +20 -20
- package/.github/ISSUE_TEMPLATE/feature_request.md +14 -14
- package/.github/PULL_REQUEST_TEMPLATE.md +13 -13
- package/CHANGELOG.md +48 -48
- package/CONTRIBUTING.md +36 -36
- package/README.zh-CN.md +497 -497
- package/dist/channels/wechat.js +6 -6
- package/dist/deploy/index.js +56 -56
- package/dist/studio/server.js +30 -1
- package/dist/studio-ui/index.html +230 -10
- package/dist/ui/components.js +105 -105
- package/examples/README.md +22 -22
- package/examples/basic-agent.ts +90 -90
- package/examples/brain-integration.ts +71 -71
- package/examples/multi-channel.ts +74 -74
- package/fix-sidebar.mjs +188 -188
- package/install.ps1 +154 -154
- package/install.sh +164 -164
- package/package.json +1 -1
- package/scripts/install.ps1 +31 -31
- package/scripts/install.sh +40 -40
- package/serve-studio.js +13 -13
- package/serve-test.js +25 -25
- package/src/channels/dingtalk.ts +46 -46
- package/src/channels/email.ts +351 -351
- package/src/channels/feishu.ts +349 -349
- package/src/channels/googlechat.ts +42 -42
- package/src/channels/imessage.ts +31 -31
- package/src/channels/irc.ts +82 -82
- package/src/channels/line.ts +32 -32
- package/src/channels/matrix.ts +33 -33
- package/src/channels/mattermost.ts +57 -57
- package/src/channels/msteams.ts +32 -32
- package/src/channels/nostr.ts +32 -32
- package/src/channels/qq.ts +33 -33
- package/src/channels/signal.ts +32 -32
- package/src/channels/sms.ts +33 -33
- package/src/channels/telegram.ts +616 -616
- package/src/channels/twitch.ts +65 -65
- package/src/channels/voice-call.ts +100 -100
- package/src/channels/websocket.ts +399 -399
- package/src/channels/wechat.ts +329 -329
- package/src/channels/whatsapp.ts +32 -32
- package/src/cli/chat.ts +99 -99
- package/src/cli/setup.ts +314 -314
- package/src/core/agent.ts +476 -476
- package/src/core/api-server.ts +277 -277
- package/src/core/audio.ts +98 -98
- package/src/core/collaboration.ts +275 -275
- package/src/core/context-discovery.ts +85 -85
- package/src/core/context-refs.ts +140 -140
- package/src/core/gateway.ts +106 -106
- package/src/core/heartbeat.ts +51 -51
- package/src/core/hooks.ts +105 -105
- package/src/core/ide-bridge.ts +133 -133
- package/src/core/node-network.ts +86 -86
- package/src/core/profiles.ts +122 -122
- package/src/core/scheduler.ts +187 -187
- package/src/core/session-manager.ts +137 -137
- package/src/core/subagent.ts +98 -98
- package/src/core/vision.ts +180 -180
- package/src/core/workflow-graph.ts +365 -365
- package/src/daemon.ts +96 -96
- package/src/deploy/index.ts +255 -255
- package/src/doctor.ts +156 -156
- package/src/eval/index.ts +211 -211
- package/src/eval/suites/basic.json +16 -16
- package/src/eval/suites/memory.json +12 -12
- package/src/eval/suites/safety.json +14 -14
- package/src/hub/brain-seed.ts +54 -54
- package/src/hub/client.ts +60 -60
- package/src/mcp/servers/calculator-mcp.ts +65 -65
- package/src/mcp/servers/crypto-mcp.ts +73 -73
- package/src/mcp/servers/database-mcp.ts +72 -72
- package/src/mcp/servers/datetime-mcp.ts +69 -69
- package/src/mcp/servers/filesystem.ts +66 -66
- package/src/mcp/servers/github-mcp.ts +58 -58
- package/src/mcp/servers/index.ts +63 -63
- package/src/mcp/servers/json-mcp.ts +102 -102
- package/src/mcp/servers/memory-mcp.ts +56 -56
- package/src/mcp/servers/regex-mcp.ts +53 -53
- package/src/mcp/servers/web-mcp.ts +49 -49
- package/src/memory/context-compressor.ts +189 -189
- package/src/memory/seed-loader.ts +212 -212
- package/src/memory/user-profiler.ts +215 -215
- package/src/plugins/content-filter.ts +23 -23
- package/src/plugins/logger.ts +18 -18
- package/src/plugins/rate-limiter.ts +38 -38
- package/src/protocols/a2a/client.ts +132 -132
- package/src/protocols/a2a/index.ts +8 -8
- package/src/protocols/a2a/server.ts +333 -333
- package/src/protocols/a2a/types.ts +88 -88
- package/src/protocols/a2a/utils.ts +50 -50
- package/src/protocols/agui/client.ts +83 -83
- package/src/protocols/agui/index.ts +4 -4
- package/src/protocols/agui/server.ts +218 -218
- package/src/protocols/agui/types.ts +153 -153
- package/src/protocols/index.ts +2 -2
- package/src/protocols/mcp/agent-tools.ts +134 -134
- package/src/protocols/mcp/index.ts +8 -8
- package/src/protocols/mcp/server.ts +262 -262
- package/src/protocols/mcp/types.ts +69 -69
- package/src/providers/index.ts +632 -632
- package/src/publish/index.ts +376 -376
- package/src/scheduler/cron-engine.ts +191 -191
- package/src/scheduler/index.ts +2 -2
- package/src/schema/oad.ts +217 -217
- package/src/security/approval.ts +131 -131
- package/src/security/approvals.ts +143 -143
- package/src/security/elevated.ts +105 -105
- package/src/security/guardrails.ts +248 -248
- package/src/security/index.ts +9 -9
- package/src/security/keys.ts +87 -87
- package/src/security/secrets.ts +129 -129
- package/src/skills/builtin/index.ts +408 -408
- package/src/skills/marketplace.ts +113 -113
- package/src/skills/types.ts +42 -42
- package/src/studio/server.ts +31 -1
- package/src/studio/templates-data.ts +178 -178
- package/src/studio-ui/index.html +230 -10
- package/src/telemetry/index.ts +324 -324
- package/src/tools/builtin/browser.ts +299 -299
- package/src/tools/builtin/datetime.ts +41 -41
- package/src/tools/builtin/file.ts +107 -107
- package/src/tools/builtin/home-assistant.ts +116 -116
- package/src/tools/builtin/rl-tools.ts +243 -243
- package/src/tools/builtin/shell.ts +43 -43
- package/src/tools/builtin/vision.ts +64 -64
- package/src/tools/builtin/web-search.ts +126 -126
- package/src/tools/builtin/web.ts +35 -35
- package/src/tools/document-processor.ts +213 -213
- package/src/tools/image-generator.ts +150 -150
- package/src/tools/integrations/calendar.ts +73 -73
- package/src/tools/integrations/code-exec.ts +39 -39
- package/src/tools/integrations/csv-analyzer.ts +92 -92
- package/src/tools/integrations/database.ts +44 -44
- package/src/tools/integrations/email-send.ts +76 -76
- package/src/tools/integrations/git-tool.ts +42 -42
- package/src/tools/integrations/github-tool.ts +76 -76
- package/src/tools/integrations/image-gen.ts +56 -56
- package/src/tools/integrations/index.ts +92 -92
- package/src/tools/integrations/jira.ts +83 -83
- package/src/tools/integrations/notion.ts +71 -71
- package/src/tools/integrations/npm-tool.ts +48 -48
- package/src/tools/integrations/pdf-reader.ts +58 -58
- package/src/tools/integrations/slack.ts +65 -65
- package/src/tools/integrations/summarizer.ts +49 -49
- package/src/tools/integrations/translator.ts +48 -48
- package/src/tools/integrations/trello.ts +60 -60
- package/src/tools/integrations/vector-search.ts +42 -42
- package/src/tools/integrations/web-scraper.ts +47 -47
- package/src/tools/integrations/web-search.ts +58 -58
- package/src/tools/integrations/webhook.ts +38 -38
- package/src/tools/mcp-client.ts +131 -131
- package/src/tools/web-scraper.ts +179 -179
- package/src/tools/web-search.ts +180 -180
- package/src/ui/components.ts +127 -127
- package/srv-out.txt +1 -1
- package/templates/ecommerce-assistant/README.md +45 -45
- package/templates/ecommerce-assistant/oad.yaml +47 -47
- package/templates/tech-support/README.md +43 -43
- package/templates/tech-support/oad.yaml +45 -45
- package/test-agent/Dockerfile +9 -9
- package/test-agent/README.md +50 -50
- package/test-agent/agent.yaml +23 -23
- package/test-agent/docker-compose.yml +11 -11
- package/test-agent/oad.yaml +31 -31
- package/test-agent/package-lock.json +1492 -1492
- package/test-agent/package.json +17 -17
- package/test-agent/src/index.ts +24 -24
- package/test-agent/src/skills/echo.ts +15 -15
- package/test-agent/tsconfig.json +24 -24
- package/test-full.js +43 -43
- package/test-sidebar.js +22 -22
- package/test-studio3.js +75 -75
- package/test-studio4.js +41 -41
- package/tests/a2a-protocol.test.ts +285 -285
- package/tests/agui-protocol.test.ts +246 -246
- package/tests/api-server.test.ts +148 -148
- package/tests/approvals.test.ts +89 -89
- package/tests/audio.test.ts +40 -40
- package/tests/brain-seed-extended.test.ts +490 -490
- package/tests/brain-seed.test.ts +239 -239
- package/tests/browser.test.ts +179 -179
- package/tests/channels/discord.test.ts +79 -79
- package/tests/channels/email.test.ts +148 -148
- package/tests/channels/feishu.test.ts +123 -123
- package/tests/channels/telegram.test.ts +129 -129
- package/tests/channels/websocket.test.ts +53 -53
- package/tests/channels/wechat.test.ts +170 -170
- package/tests/channels-extra.test.ts +45 -45
- package/tests/chat-cli.test.ts +160 -160
- package/tests/cli.test.ts +46 -46
- package/tests/context-compressor.test.ts +172 -172
- package/tests/context-refs.test.ts +121 -121
- package/tests/cron-engine.test.ts +101 -101
- package/tests/daemon.test.ts +135 -135
- package/tests/deepbrain-wire.test.ts +234 -234
- package/tests/deploy-and-dag.test.ts +196 -196
- package/tests/doctor.test.ts +38 -38
- package/tests/document-processor.test.ts +69 -69
- package/tests/e2e-nocode.test.ts +442 -442
- package/tests/elevated.test.ts +69 -69
- package/tests/eval.test.ts +173 -173
- package/tests/gateway.test.ts +63 -63
- package/tests/guardrails.test.ts +177 -177
- package/tests/home-assistant.test.ts +40 -40
- package/tests/hooks.test.ts +79 -79
- package/tests/ide-bridge.test.ts +38 -38
- package/tests/image-generator.test.ts +84 -84
- package/tests/init-role.test.ts +124 -124
- package/tests/integrations.test.ts +249 -249
- package/tests/mcp-client.test.ts +92 -92
- package/tests/mcp-server.test.ts +178 -178
- package/tests/mcp-servers.test.ts +260 -260
- package/tests/node-network.test.ts +74 -74
- package/tests/plugin-a2a-enhanced.test.ts +230 -230
- package/tests/profiles.test.ts +61 -61
- package/tests/publish.test.ts +231 -231
- package/tests/rl-tools.test.ts +93 -93
- package/tests/sandbox-manager.test.ts +46 -46
- package/tests/scheduler.test.ts +200 -200
- package/tests/secrets.test.ts +107 -107
- package/tests/security-enhanced.test.ts +233 -233
- package/tests/settings-api.test.ts +148 -148
- package/tests/setup.test.ts +73 -73
- package/tests/subagent.test.ts +193 -193
- package/tests/telegram-discord.test.ts +60 -60
- package/tests/telemetry.test.ts +186 -186
- package/tests/user-profiler.test.ts +169 -169
- package/tests/v090-features.test.ts +254 -254
- package/tests/vision.test.ts +61 -61
- package/tests/voice-call.test.ts +47 -47
- package/tests/voice-enhanced.test.ts +169 -169
- package/tests/voice-interaction.test.ts +38 -38
- package/tests/web-search.test.ts +155 -155
- package/tests/workflow-graph.test.ts +279 -279
- package/tutorial/customer-service-agent/README.md +612 -612
- package/tutorial/customer-service-agent/SOUL.md +26 -26
- package/tutorial/customer-service-agent/agent.yaml +63 -63
- package/tutorial/customer-service-agent/package.json +19 -19
- package/tutorial/customer-service-agent/src/index.ts +69 -69
- package/tutorial/customer-service-agent/src/skills/faq.ts +27 -27
- package/tutorial/customer-service-agent/src/skills/ticket.ts +22 -22
- package/tutorial/customer-service-agent/tsconfig.json +14 -14
|
@@ -1,148 +1,148 @@
|
|
|
1
|
-
import { describe, it, expect } from 'vitest';
|
|
2
|
-
import * as fs from 'fs';
|
|
3
|
-
import * as path from 'path';
|
|
4
|
-
|
|
5
|
-
describe('Settings API & UI', () => {
|
|
6
|
-
describe('Backend - server.ts settings routes', () => {
|
|
7
|
-
const serverSrc = fs.readFileSync(path.join(__dirname, '../src/studio/server.ts'), 'utf-8');
|
|
8
|
-
|
|
9
|
-
it('has settings/models GET route', () => {
|
|
10
|
-
expect(serverSrc).toContain("route === 'settings/models' && req.method === 'GET'");
|
|
11
|
-
});
|
|
12
|
-
|
|
13
|
-
it('has settings/models PUT route', () => {
|
|
14
|
-
expect(serverSrc).toContain("route === 'settings/models' && req.method === 'PUT'");
|
|
15
|
-
});
|
|
16
|
-
|
|
17
|
-
it('has settings/models/test POST route', () => {
|
|
18
|
-
expect(serverSrc).toContain("route === 'settings/models/test' && req.method === 'POST'");
|
|
19
|
-
});
|
|
20
|
-
|
|
21
|
-
it('has settings/models/local GET route', () => {
|
|
22
|
-
expect(serverSrc).toContain("route === 'settings/models/local' && req.method === 'GET'");
|
|
23
|
-
});
|
|
24
|
-
|
|
25
|
-
it('has settings/channels GET route', () => {
|
|
26
|
-
expect(serverSrc).toContain("route === 'settings/channels' && req.method === 'GET'");
|
|
27
|
-
});
|
|
28
|
-
|
|
29
|
-
it('has settings/channels/:name PUT route', () => {
|
|
30
|
-
expect(serverSrc).toContain("settings\\/channels\\/[^/]+");
|
|
31
|
-
});
|
|
32
|
-
|
|
33
|
-
it('has settings/status GET route', () => {
|
|
34
|
-
expect(serverSrc).toContain("route === 'settings/status' && req.method === 'GET'");
|
|
35
|
-
});
|
|
36
|
-
|
|
37
|
-
it('has settings/usage GET route', () => {
|
|
38
|
-
expect(serverSrc).toContain("route === 'settings/usage' && req.method === 'GET'");
|
|
39
|
-
});
|
|
40
|
-
|
|
41
|
-
it('has detectLocalOllama method', () => {
|
|
42
|
-
expect(serverSrc).toContain('detectLocalOllama');
|
|
43
|
-
});
|
|
44
|
-
|
|
45
|
-
it('has testModelConnection method', () => {
|
|
46
|
-
expect(serverSrc).toContain('testModelConnection');
|
|
47
|
-
});
|
|
48
|
-
|
|
49
|
-
it('saves config to ~/.opc/config.json', () => {
|
|
50
|
-
expect(serverSrc).toContain("'.opc'");
|
|
51
|
-
expect(serverSrc).toContain("'config.json'");
|
|
52
|
-
});
|
|
53
|
-
});
|
|
54
|
-
|
|
55
|
-
describe('Frontend - index.html settings UI', () => {
|
|
56
|
-
const html = fs.readFileSync(path.join(__dirname, '../src/studio-ui/index.html'), 'utf-8');
|
|
57
|
-
|
|
58
|
-
it('has settings page container', () => {
|
|
59
|
-
expect(html).toContain('id="page-settings"');
|
|
60
|
-
});
|
|
61
|
-
|
|
62
|
-
it('has settings navigation items', () => {
|
|
63
|
-
expect(html).toContain('data-settings="models"');
|
|
64
|
-
expect(html).toContain('data-settings="channels"');
|
|
65
|
-
expect(html).toContain('data-settings="memory"');
|
|
66
|
-
expect(html).toContain('data-settings="role"');
|
|
67
|
-
expect(html).toContain('data-settings="status"');
|
|
68
|
-
expect(html).toContain('data-settings="usage"');
|
|
69
|
-
});
|
|
70
|
-
|
|
71
|
-
it('has models panel with local/cloud tabs', () => {
|
|
72
|
-
expect(html).toContain('id="sp-models"');
|
|
73
|
-
expect(html).toContain('id="mt-local"');
|
|
74
|
-
expect(html).toContain('id="mt-cloud"');
|
|
75
|
-
});
|
|
76
|
-
|
|
77
|
-
it('has Ollama detection UI', () => {
|
|
78
|
-
expect(html).toContain('id="ollama-status"');
|
|
79
|
-
expect(html).toContain('id="ollama-models"');
|
|
80
|
-
expect(html).toContain('id="ollama-tutorial"');
|
|
81
|
-
expect(html).toContain('ollama.com');
|
|
82
|
-
});
|
|
83
|
-
|
|
84
|
-
it('has cloud provider cards', () => {
|
|
85
|
-
expect(html).toContain('pv-openai');
|
|
86
|
-
expect(html).toContain('pv-deepseek');
|
|
87
|
-
expect(html).toContain('pv-anthropic');
|
|
88
|
-
});
|
|
89
|
-
|
|
90
|
-
it('has model assignment dropdowns', () => {
|
|
91
|
-
expect(html).toContain('id="cfg-chat-model"');
|
|
92
|
-
expect(html).toContain('id="cfg-embed-model"');
|
|
93
|
-
expect(html).toContain('qwen2.5:7b');
|
|
94
|
-
expect(html).toContain('nomic-embed-text');
|
|
95
|
-
});
|
|
96
|
-
|
|
97
|
-
it('has channels panel', () => {
|
|
98
|
-
expect(html).toContain('id="sp-channels"');
|
|
99
|
-
expect(html).toContain('id="channels-grid"');
|
|
100
|
-
});
|
|
101
|
-
|
|
102
|
-
it('has memory panel with DeepBrain iframe reference', () => {
|
|
103
|
-
expect(html).toContain('id="sp-memory"');
|
|
104
|
-
expect(html).toContain('localhost:4001');
|
|
105
|
-
expect(html).toContain('DeepBrain');
|
|
106
|
-
});
|
|
107
|
-
|
|
108
|
-
it('has role panel with Workstation iframe reference', () => {
|
|
109
|
-
expect(html).toContain('id="sp-role"');
|
|
110
|
-
expect(html).toContain('localhost:4003');
|
|
111
|
-
expect(html).toContain('Workstation');
|
|
112
|
-
});
|
|
113
|
-
|
|
114
|
-
it('has status panel with logs viewer', () => {
|
|
115
|
-
expect(html).toContain('id="sp-status"');
|
|
116
|
-
expect(html).toContain('id="status-logs"');
|
|
117
|
-
});
|
|
118
|
-
|
|
119
|
-
it('has usage panel', () => {
|
|
120
|
-
expect(html).toContain('id="sp-usage"');
|
|
121
|
-
expect(html).toContain('id="usage-stats"');
|
|
122
|
-
});
|
|
123
|
-
|
|
124
|
-
it('has provider config dialog', () => {
|
|
125
|
-
expect(html).toContain('id="provider-dialog"');
|
|
126
|
-
expect(html).toContain('id="pd-apikey"');
|
|
127
|
-
expect(html).toContain('testProvider');
|
|
128
|
-
});
|
|
129
|
-
|
|
130
|
-
it('has channel config dialog', () => {
|
|
131
|
-
expect(html).toContain('id="channel-dialog"');
|
|
132
|
-
});
|
|
133
|
-
|
|
134
|
-
it('has settings nav in sidebar', () => {
|
|
135
|
-
expect(html).toContain('data-page="settings"');
|
|
136
|
-
expect(html).toContain('Settings');
|
|
137
|
-
});
|
|
138
|
-
|
|
139
|
-
it('has Chinese-friendly copy', () => {
|
|
140
|
-
expect(html).toContain('模型配置');
|
|
141
|
-
expect(html).toContain('渠道配置');
|
|
142
|
-
expect(html).toContain('记忆管理');
|
|
143
|
-
expect(html).toContain('角色编辑');
|
|
144
|
-
expect(html).toContain('运行状态');
|
|
145
|
-
expect(html).toContain('用量统计');
|
|
146
|
-
});
|
|
147
|
-
});
|
|
148
|
-
});
|
|
1
|
+
import { describe, it, expect } from 'vitest';
|
|
2
|
+
import * as fs from 'fs';
|
|
3
|
+
import * as path from 'path';
|
|
4
|
+
|
|
5
|
+
describe('Settings API & UI', () => {
|
|
6
|
+
describe('Backend - server.ts settings routes', () => {
|
|
7
|
+
const serverSrc = fs.readFileSync(path.join(__dirname, '../src/studio/server.ts'), 'utf-8');
|
|
8
|
+
|
|
9
|
+
it('has settings/models GET route', () => {
|
|
10
|
+
expect(serverSrc).toContain("route === 'settings/models' && req.method === 'GET'");
|
|
11
|
+
});
|
|
12
|
+
|
|
13
|
+
it('has settings/models PUT route', () => {
|
|
14
|
+
expect(serverSrc).toContain("route === 'settings/models' && req.method === 'PUT'");
|
|
15
|
+
});
|
|
16
|
+
|
|
17
|
+
it('has settings/models/test POST route', () => {
|
|
18
|
+
expect(serverSrc).toContain("route === 'settings/models/test' && req.method === 'POST'");
|
|
19
|
+
});
|
|
20
|
+
|
|
21
|
+
it('has settings/models/local GET route', () => {
|
|
22
|
+
expect(serverSrc).toContain("route === 'settings/models/local' && req.method === 'GET'");
|
|
23
|
+
});
|
|
24
|
+
|
|
25
|
+
it('has settings/channels GET route', () => {
|
|
26
|
+
expect(serverSrc).toContain("route === 'settings/channels' && req.method === 'GET'");
|
|
27
|
+
});
|
|
28
|
+
|
|
29
|
+
it('has settings/channels/:name PUT route', () => {
|
|
30
|
+
expect(serverSrc).toContain("settings\\/channels\\/[^/]+");
|
|
31
|
+
});
|
|
32
|
+
|
|
33
|
+
it('has settings/status GET route', () => {
|
|
34
|
+
expect(serverSrc).toContain("route === 'settings/status' && req.method === 'GET'");
|
|
35
|
+
});
|
|
36
|
+
|
|
37
|
+
it('has settings/usage GET route', () => {
|
|
38
|
+
expect(serverSrc).toContain("route === 'settings/usage' && req.method === 'GET'");
|
|
39
|
+
});
|
|
40
|
+
|
|
41
|
+
it('has detectLocalOllama method', () => {
|
|
42
|
+
expect(serverSrc).toContain('detectLocalOllama');
|
|
43
|
+
});
|
|
44
|
+
|
|
45
|
+
it('has testModelConnection method', () => {
|
|
46
|
+
expect(serverSrc).toContain('testModelConnection');
|
|
47
|
+
});
|
|
48
|
+
|
|
49
|
+
it('saves config to ~/.opc/config.json', () => {
|
|
50
|
+
expect(serverSrc).toContain("'.opc'");
|
|
51
|
+
expect(serverSrc).toContain("'config.json'");
|
|
52
|
+
});
|
|
53
|
+
});
|
|
54
|
+
|
|
55
|
+
describe('Frontend - index.html settings UI', () => {
|
|
56
|
+
const html = fs.readFileSync(path.join(__dirname, '../src/studio-ui/index.html'), 'utf-8');
|
|
57
|
+
|
|
58
|
+
it('has settings page container', () => {
|
|
59
|
+
expect(html).toContain('id="page-settings"');
|
|
60
|
+
});
|
|
61
|
+
|
|
62
|
+
it('has settings navigation items', () => {
|
|
63
|
+
expect(html).toContain('data-settings="models"');
|
|
64
|
+
expect(html).toContain('data-settings="channels"');
|
|
65
|
+
expect(html).toContain('data-settings="memory"');
|
|
66
|
+
expect(html).toContain('data-settings="role"');
|
|
67
|
+
expect(html).toContain('data-settings="status"');
|
|
68
|
+
expect(html).toContain('data-settings="usage"');
|
|
69
|
+
});
|
|
70
|
+
|
|
71
|
+
it('has models panel with local/cloud tabs', () => {
|
|
72
|
+
expect(html).toContain('id="sp-models"');
|
|
73
|
+
expect(html).toContain('id="mt-local"');
|
|
74
|
+
expect(html).toContain('id="mt-cloud"');
|
|
75
|
+
});
|
|
76
|
+
|
|
77
|
+
it('has Ollama detection UI', () => {
|
|
78
|
+
expect(html).toContain('id="ollama-status"');
|
|
79
|
+
expect(html).toContain('id="ollama-models"');
|
|
80
|
+
expect(html).toContain('id="ollama-tutorial"');
|
|
81
|
+
expect(html).toContain('ollama.com');
|
|
82
|
+
});
|
|
83
|
+
|
|
84
|
+
it('has cloud provider cards', () => {
|
|
85
|
+
expect(html).toContain('pv-openai');
|
|
86
|
+
expect(html).toContain('pv-deepseek');
|
|
87
|
+
expect(html).toContain('pv-anthropic');
|
|
88
|
+
});
|
|
89
|
+
|
|
90
|
+
it('has model assignment dropdowns', () => {
|
|
91
|
+
expect(html).toContain('id="cfg-chat-model"');
|
|
92
|
+
expect(html).toContain('id="cfg-embed-model"');
|
|
93
|
+
expect(html).toContain('qwen2.5:7b');
|
|
94
|
+
expect(html).toContain('nomic-embed-text');
|
|
95
|
+
});
|
|
96
|
+
|
|
97
|
+
it('has channels panel', () => {
|
|
98
|
+
expect(html).toContain('id="sp-channels"');
|
|
99
|
+
expect(html).toContain('id="channels-grid"');
|
|
100
|
+
});
|
|
101
|
+
|
|
102
|
+
it('has memory panel with DeepBrain iframe reference', () => {
|
|
103
|
+
expect(html).toContain('id="sp-memory"');
|
|
104
|
+
expect(html).toContain('localhost:4001');
|
|
105
|
+
expect(html).toContain('DeepBrain');
|
|
106
|
+
});
|
|
107
|
+
|
|
108
|
+
it('has role panel with Workstation iframe reference', () => {
|
|
109
|
+
expect(html).toContain('id="sp-role"');
|
|
110
|
+
expect(html).toContain('localhost:4003');
|
|
111
|
+
expect(html).toContain('Workstation');
|
|
112
|
+
});
|
|
113
|
+
|
|
114
|
+
it('has status panel with logs viewer', () => {
|
|
115
|
+
expect(html).toContain('id="sp-status"');
|
|
116
|
+
expect(html).toContain('id="status-logs"');
|
|
117
|
+
});
|
|
118
|
+
|
|
119
|
+
it('has usage panel', () => {
|
|
120
|
+
expect(html).toContain('id="sp-usage"');
|
|
121
|
+
expect(html).toContain('id="usage-stats"');
|
|
122
|
+
});
|
|
123
|
+
|
|
124
|
+
it('has provider config dialog', () => {
|
|
125
|
+
expect(html).toContain('id="provider-dialog"');
|
|
126
|
+
expect(html).toContain('id="pd-apikey"');
|
|
127
|
+
expect(html).toContain('testProvider');
|
|
128
|
+
});
|
|
129
|
+
|
|
130
|
+
it('has channel config dialog', () => {
|
|
131
|
+
expect(html).toContain('id="channel-dialog"');
|
|
132
|
+
});
|
|
133
|
+
|
|
134
|
+
it('has settings nav in sidebar', () => {
|
|
135
|
+
expect(html).toContain('data-page="settings"');
|
|
136
|
+
expect(html).toContain('Settings');
|
|
137
|
+
});
|
|
138
|
+
|
|
139
|
+
it('has Chinese-friendly copy', () => {
|
|
140
|
+
expect(html).toContain('模型配置');
|
|
141
|
+
expect(html).toContain('渠道配置');
|
|
142
|
+
expect(html).toContain('记忆管理');
|
|
143
|
+
expect(html).toContain('角色编辑');
|
|
144
|
+
expect(html).toContain('运行状态');
|
|
145
|
+
expect(html).toContain('用量统计');
|
|
146
|
+
});
|
|
147
|
+
});
|
|
148
|
+
});
|
package/tests/setup.test.ts
CHANGED
|
@@ -1,73 +1,73 @@
|
|
|
1
|
-
import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
|
|
2
|
-
import { Readable, Writable } from 'stream';
|
|
3
|
-
import * as fs from 'fs';
|
|
4
|
-
import * as path from 'path';
|
|
5
|
-
import * as os from 'os';
|
|
6
|
-
|
|
7
|
-
// Mock http to control Ollama detection
|
|
8
|
-
vi.mock('http', async () => {
|
|
9
|
-
const actual = await vi.importActual<typeof import('http')>('http');
|
|
10
|
-
return { ...actual };
|
|
11
|
-
});
|
|
12
|
-
|
|
13
|
-
describe('setup wizard', () => {
|
|
14
|
-
const OPC_HOME = path.join(os.tmpdir(), `.opc-test-${Date.now()}`);
|
|
15
|
-
const CONFIG_PATH = path.join(OPC_HOME, 'config.json');
|
|
16
|
-
|
|
17
|
-
beforeEach(() => {
|
|
18
|
-
// Override HOME so setup writes to temp
|
|
19
|
-
vi.stubEnv('HOME', os.tmpdir());
|
|
20
|
-
fs.mkdirSync(OPC_HOME, { recursive: true });
|
|
21
|
-
});
|
|
22
|
-
|
|
23
|
-
afterEach(() => {
|
|
24
|
-
fs.rmSync(OPC_HOME, { recursive: true, force: true });
|
|
25
|
-
vi.unstubAllEnvs();
|
|
26
|
-
});
|
|
27
|
-
|
|
28
|
-
function createMockInput(lines: string[]): Readable {
|
|
29
|
-
const input = new Readable({ read() {} });
|
|
30
|
-
// Push lines async to simulate user typing
|
|
31
|
-
let i = 0;
|
|
32
|
-
const interval = setInterval(() => {
|
|
33
|
-
if (i < lines.length) {
|
|
34
|
-
input.push(lines[i] + '\n');
|
|
35
|
-
i++;
|
|
36
|
-
} else {
|
|
37
|
-
input.push(null);
|
|
38
|
-
clearInterval(interval);
|
|
39
|
-
}
|
|
40
|
-
}, 50);
|
|
41
|
-
return input;
|
|
42
|
-
}
|
|
43
|
-
|
|
44
|
-
function createMockOutput(): Writable {
|
|
45
|
-
const chunks: Buffer[] = [];
|
|
46
|
-
const output = new Writable({
|
|
47
|
-
write(chunk, _enc, cb) { chunks.push(Buffer.from(chunk)); cb(); },
|
|
48
|
-
});
|
|
49
|
-
(output as any).getOutput = () => Buffer.concat(chunks).toString();
|
|
50
|
-
return output;
|
|
51
|
-
}
|
|
52
|
-
|
|
53
|
-
it('should export runSetup function', async () => {
|
|
54
|
-
const { runSetup } = await import('../src/cli/setup');
|
|
55
|
-
expect(typeof runSetup).toBe('function');
|
|
56
|
-
});
|
|
57
|
-
|
|
58
|
-
it('should have TEMPLATES defined', async () => {
|
|
59
|
-
// Just test that the module loads without error
|
|
60
|
-
const mod = await import('../src/cli/setup');
|
|
61
|
-
expect(mod).toBeDefined();
|
|
62
|
-
});
|
|
63
|
-
|
|
64
|
-
it('createRL should return a readline interface', async () => {
|
|
65
|
-
const { createRL } = await import('../src/cli/setup');
|
|
66
|
-
const input = new Readable({ read() {} });
|
|
67
|
-
const output = createMockOutput();
|
|
68
|
-
const rl = createRL(input, output);
|
|
69
|
-
expect(rl).toBeDefined();
|
|
70
|
-
rl.close();
|
|
71
|
-
input.destroy();
|
|
72
|
-
});
|
|
73
|
-
});
|
|
1
|
+
import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
|
|
2
|
+
import { Readable, Writable } from 'stream';
|
|
3
|
+
import * as fs from 'fs';
|
|
4
|
+
import * as path from 'path';
|
|
5
|
+
import * as os from 'os';
|
|
6
|
+
|
|
7
|
+
// Mock http to control Ollama detection
|
|
8
|
+
vi.mock('http', async () => {
|
|
9
|
+
const actual = await vi.importActual<typeof import('http')>('http');
|
|
10
|
+
return { ...actual };
|
|
11
|
+
});
|
|
12
|
+
|
|
13
|
+
describe('setup wizard', () => {
|
|
14
|
+
const OPC_HOME = path.join(os.tmpdir(), `.opc-test-${Date.now()}`);
|
|
15
|
+
const CONFIG_PATH = path.join(OPC_HOME, 'config.json');
|
|
16
|
+
|
|
17
|
+
beforeEach(() => {
|
|
18
|
+
// Override HOME so setup writes to temp
|
|
19
|
+
vi.stubEnv('HOME', os.tmpdir());
|
|
20
|
+
fs.mkdirSync(OPC_HOME, { recursive: true });
|
|
21
|
+
});
|
|
22
|
+
|
|
23
|
+
afterEach(() => {
|
|
24
|
+
fs.rmSync(OPC_HOME, { recursive: true, force: true });
|
|
25
|
+
vi.unstubAllEnvs();
|
|
26
|
+
});
|
|
27
|
+
|
|
28
|
+
function createMockInput(lines: string[]): Readable {
|
|
29
|
+
const input = new Readable({ read() {} });
|
|
30
|
+
// Push lines async to simulate user typing
|
|
31
|
+
let i = 0;
|
|
32
|
+
const interval = setInterval(() => {
|
|
33
|
+
if (i < lines.length) {
|
|
34
|
+
input.push(lines[i] + '\n');
|
|
35
|
+
i++;
|
|
36
|
+
} else {
|
|
37
|
+
input.push(null);
|
|
38
|
+
clearInterval(interval);
|
|
39
|
+
}
|
|
40
|
+
}, 50);
|
|
41
|
+
return input;
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
function createMockOutput(): Writable {
|
|
45
|
+
const chunks: Buffer[] = [];
|
|
46
|
+
const output = new Writable({
|
|
47
|
+
write(chunk, _enc, cb) { chunks.push(Buffer.from(chunk)); cb(); },
|
|
48
|
+
});
|
|
49
|
+
(output as any).getOutput = () => Buffer.concat(chunks).toString();
|
|
50
|
+
return output;
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
it('should export runSetup function', async () => {
|
|
54
|
+
const { runSetup } = await import('../src/cli/setup');
|
|
55
|
+
expect(typeof runSetup).toBe('function');
|
|
56
|
+
});
|
|
57
|
+
|
|
58
|
+
it('should have TEMPLATES defined', async () => {
|
|
59
|
+
// Just test that the module loads without error
|
|
60
|
+
const mod = await import('../src/cli/setup');
|
|
61
|
+
expect(mod).toBeDefined();
|
|
62
|
+
});
|
|
63
|
+
|
|
64
|
+
it('createRL should return a readline interface', async () => {
|
|
65
|
+
const { createRL } = await import('../src/cli/setup');
|
|
66
|
+
const input = new Readable({ read() {} });
|
|
67
|
+
const output = createMockOutput();
|
|
68
|
+
const rl = createRL(input, output);
|
|
69
|
+
expect(rl).toBeDefined();
|
|
70
|
+
rl.close();
|
|
71
|
+
input.destroy();
|
|
72
|
+
});
|
|
73
|
+
});
|