skyloom 1.13.5 → 1.13.7
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/workflows/ci.yml +36 -36
- package/README.md +220 -159
- package/config/providers.yaml +39 -39
- package/config/skills/api_integrator/SKILL.md +15 -15
- package/config/skills/arch_designer/SKILL.md +13 -13
- package/config/skills/ci_cd_manager/SKILL.md +14 -14
- package/config/skills/code_analysis/SKILL.md +13 -13
- package/config/skills/code_generator/SKILL.md +12 -12
- package/config/skills/code_reviewer/SKILL.md +13 -13
- package/config/skills/content_writer/SKILL.md +14 -14
- package/config/skills/data_transformer/SKILL.md +15 -15
- package/config/skills/document_analysis/SKILL.md +13 -13
- package/config/skills/emotional_companion/SKILL.md +15 -15
- package/config/skills/performance_checker/SKILL.md +14 -14
- package/config/skills/security_auditor/SKILL.md +14 -14
- package/config/skills/self_evolve/SKILL.md +13 -13
- package/config/skills/sys_operator/SKILL.md +15 -15
- package/config/skills/task_planner/SKILL.md +14 -14
- package/config/skills/web_research/SKILL.md +14 -14
- package/config/skills/workflow_designer/SKILL.md +13 -13
- package/dist/agents/dew.js +52 -52
- package/dist/agents/fair.js +84 -84
- package/dist/agents/fog.js +30 -30
- package/dist/agents/frost.js +32 -32
- package/dist/agents/rain.js +32 -32
- package/dist/agents/snow.js +68 -68
- package/dist/cli/commands_md.d.ts +41 -0
- package/dist/cli/commands_md.d.ts.map +1 -0
- package/dist/cli/commands_md.js +140 -0
- package/dist/cli/commands_md.js.map +1 -0
- package/dist/cli/input_macros.d.ts +28 -0
- package/dist/cli/input_macros.d.ts.map +1 -0
- package/dist/cli/input_macros.js +120 -0
- package/dist/cli/input_macros.js.map +1 -0
- package/dist/cli/loom.d.ts +220 -0
- package/dist/cli/loom.d.ts.map +1 -0
- package/dist/cli/loom.js +1094 -0
- package/dist/cli/loom.js.map +1 -0
- package/dist/cli/loom_chat.d.ts +20 -0
- package/dist/cli/loom_chat.d.ts.map +1 -0
- package/dist/cli/loom_chat.js +685 -0
- package/dist/cli/loom_chat.js.map +1 -0
- package/dist/cli/main.js +310 -14
- package/dist/cli/main.js.map +1 -1
- package/dist/cli/tui.d.ts.map +1 -1
- package/dist/cli/tui.js +7 -1
- package/dist/cli/tui.js.map +1 -1
- package/dist/core/agent/guard.d.ts +45 -0
- package/dist/core/agent/guard.d.ts.map +1 -0
- package/dist/core/agent/guard.js +113 -0
- package/dist/core/agent/guard.js.map +1 -0
- package/dist/core/agent.d.ts +17 -0
- package/dist/core/agent.d.ts.map +1 -1
- package/dist/core/agent.js +182 -93
- package/dist/core/agent.js.map +1 -1
- package/dist/core/factory.d.ts.map +1 -1
- package/dist/core/factory.js +34 -2
- package/dist/core/factory.js.map +1 -1
- package/dist/core/file_checkpoint.d.ts +57 -0
- package/dist/core/file_checkpoint.d.ts.map +1 -0
- package/dist/core/file_checkpoint.js +162 -0
- package/dist/core/file_checkpoint.js.map +1 -0
- package/dist/core/hooks.d.ts +43 -0
- package/dist/core/hooks.d.ts.map +1 -0
- package/dist/core/hooks.js +110 -0
- package/dist/core/hooks.js.map +1 -0
- package/dist/core/llm.d.ts.map +1 -1
- package/dist/core/llm.js +15 -9
- package/dist/core/llm.js.map +1 -1
- package/dist/core/longdoc.js +5 -5
- package/dist/core/mcp.d.ts +16 -0
- package/dist/core/mcp.d.ts.map +1 -1
- package/dist/core/mcp.js +55 -0
- package/dist/core/mcp.js.map +1 -1
- package/dist/core/model_config.d.ts +40 -0
- package/dist/core/model_config.d.ts.map +1 -0
- package/dist/core/model_config.js +191 -0
- package/dist/core/model_config.js.map +1 -0
- package/dist/core/skill.d.ts +7 -0
- package/dist/core/skill.d.ts.map +1 -1
- package/dist/core/skill.js +47 -0
- package/dist/core/skill.js.map +1 -1
- package/dist/core/skymd.d.ts +39 -0
- package/dist/core/skymd.d.ts.map +1 -0
- package/dist/core/skymd.js +177 -0
- package/dist/core/skymd.js.map +1 -0
- package/dist/core/tool.d.ts +12 -0
- package/dist/core/tool.d.ts.map +1 -1
- package/dist/core/tool.js +30 -0
- package/dist/core/tool.js.map +1 -1
- package/dist/core/verify.d.ts +27 -0
- package/dist/core/verify.d.ts.map +1 -0
- package/dist/core/verify.js +62 -0
- package/dist/core/verify.js.map +1 -0
- package/dist/skills/loader.d.ts +22 -2
- package/dist/skills/loader.d.ts.map +1 -1
- package/dist/skills/loader.js +45 -15
- package/dist/skills/loader.js.map +1 -1
- package/dist/tools/builtin.d.ts.map +1 -1
- package/dist/tools/builtin.js +13 -3
- package/dist/tools/builtin.js.map +1 -1
- package/dist/tools/model_tool.d.ts +11 -0
- package/dist/tools/model_tool.d.ts.map +1 -0
- package/dist/tools/model_tool.js +71 -0
- package/dist/tools/model_tool.js.map +1 -0
- package/dist/tools/todo.d.ts +30 -0
- package/dist/tools/todo.d.ts.map +1 -0
- package/dist/tools/todo.js +78 -0
- package/dist/tools/todo.js.map +1 -0
- package/docs/AESTHETIC_DESIGN.md +152 -144
- package/docs/OPTIMIZATION_PLAN.md +178 -178
- package/package.json +1 -1
- package/scripts/install.js +48 -48
- package/scripts/link.js +10 -10
- package/setup.bat +79 -79
- package/skill-test-ty2fOA/test.md +10 -10
- package/src/agents/dew.ts +70 -70
- package/src/agents/fair.ts +102 -102
- package/src/agents/fog.ts +48 -48
- package/src/agents/frost.ts +50 -50
- package/src/agents/rain.ts +50 -50
- package/src/agents/snow.ts +239 -239
- package/src/cli/commands_md.ts +112 -0
- package/src/cli/input_macros.ts +83 -0
- package/src/cli/loom.ts +982 -0
- package/src/cli/loom_chat.ts +598 -0
- package/src/cli/main.ts +255 -9
- package/src/cli/mode.ts +58 -58
- package/src/cli/tui.ts +228 -222
- package/src/core/agent/guard.ts +134 -0
- package/src/core/agent/task.ts +100 -100
- package/src/core/agent.ts +177 -95
- package/src/core/arbitrate.ts +162 -162
- package/src/core/catalog.ts +178 -178
- package/src/core/checkpoint.ts +94 -94
- package/src/core/estimate.ts +104 -104
- package/src/core/evolve.ts +191 -191
- package/src/core/factory.ts +31 -2
- package/src/core/file_checkpoint.ts +136 -0
- package/src/core/filter.ts +103 -103
- package/src/core/graph.ts +156 -156
- package/src/core/hooks.ts +126 -0
- package/src/core/icons.ts +53 -53
- package/src/core/index.ts +37 -37
- package/src/core/learn.ts +146 -146
- package/src/core/llm.ts +15 -9
- package/src/core/longdoc.ts +155 -155
- package/src/core/mcp.ts +48 -0
- package/src/core/mcp_server.ts +176 -176
- package/src/core/model_config.ts +157 -0
- package/src/core/profile.ts +255 -255
- package/src/core/router.ts +124 -124
- package/src/core/sandbox.ts +142 -142
- package/src/core/security.ts +243 -243
- package/src/core/skill.ts +42 -0
- package/src/core/skymd.ts +143 -0
- package/src/core/theme.ts +65 -65
- package/src/core/tool.ts +30 -0
- package/src/core/tool_router.ts +193 -193
- package/src/core/vector.ts +152 -152
- package/src/core/verify.ts +71 -0
- package/src/core/workspace.ts +150 -150
- package/src/plugins/loader.ts +66 -66
- package/src/skills/loader.ts +45 -16
- package/src/sql.js.d.ts +29 -29
- package/src/tools/builtin.ts +13 -3
- package/src/tools/computer.ts +269 -269
- package/src/tools/delegate.ts +49 -49
- package/src/tools/model_tool.ts +74 -0
- package/src/tools/todo.ts +76 -0
- package/src/web/tts.ts +93 -93
- package/tests/agent.test.ts +159 -159
- package/tests/agent_helpers.test.ts +48 -48
- package/tests/bus.test.ts +121 -121
- package/tests/catalog.test.ts +86 -86
- package/tests/checkpoint_commands.test.ts +124 -0
- package/tests/claude_compat.test.ts +110 -0
- package/tests/config.test.ts +41 -41
- package/tests/guard.test.ts +75 -0
- package/tests/icons.test.ts +45 -45
- package/tests/loom.test.ts +248 -0
- package/tests/memory.test.ts +170 -170
- package/tests/model_config.test.ts +109 -0
- package/tests/router.test.ts +86 -86
- package/tests/schemas.test.ts +51 -51
- package/tests/semantic.test.ts +83 -83
- package/tests/setup.ts +10 -10
- package/tests/skill.test.ts +172 -172
- package/tests/skymd.test.ts +146 -0
- package/tests/task.test.ts +60 -60
- package/tests/todo_toolstats.test.ts +94 -0
- package/tests/tool.test.ts +108 -108
- package/tests/tool_router.test.ts +71 -71
- package/tests/tui.test.ts +67 -67
- package/vitest.config.ts +17 -17
package/src/cli/tui.ts
CHANGED
|
@@ -1,222 +1,228 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* 天空织机 TUI — a polished *linear* terminal interface.
|
|
3
|
-
*
|
|
4
|
-
* Design note: the previous version tried to be a full-screen app, redrawing
|
|
5
|
-
* the whole screen on every keystroke while the reply streamed linearly below
|
|
6
|
-
* it — the two fought, the conversation never persisted, and hand-rolled
|
|
7
|
-
* raw-mode editing mangled CJK width. This rewrite is linear (like Claude Code
|
|
8
|
-
* / opencode): real readline line-editing + a CJK-aware wrapping stream
|
|
9
|
-
* renderer. Robust, flicker-free, and it actually reads like a conversation.
|
|
10
|
-
*/
|
|
11
|
-
|
|
12
|
-
import * as readline from "readline";
|
|
13
|
-
import chalk from "chalk";
|
|
14
|
-
import { agentTheme, PALETTE } from "../core/theme";
|
|
15
|
-
|
|
16
|
-
const TUI_VERSION = (() => { try { return require("../../package.json").version; } catch { return ""; } })();
|
|
17
|
-
|
|
18
|
-
export interface TUIContext {
|
|
19
|
-
agent: any;
|
|
20
|
-
agents: Map<string, any>;
|
|
21
|
-
model: string;
|
|
22
|
-
cost: string;
|
|
23
|
-
width: number;
|
|
24
|
-
height: number;
|
|
25
|
-
}
|
|
26
|
-
|
|
27
|
-
/* ── Slash commands (for tab-completion + the inline palette) ── */
|
|
28
|
-
export const SLASH_COMMANDS: [string, string][] = [
|
|
29
|
-
["/fog", "≋ 雾 · 探索洞察"],
|
|
30
|
-
["/rain", "⸽ 雨 · 创造产出"],
|
|
31
|
-
["/frost", "✱ 霜 · 精炼品质"],
|
|
32
|
-
["/snow", "❉ 雪 · 架构规划"],
|
|
33
|
-
["/dew", "∘ 露 · 可靠守护"],
|
|
34
|
-
["/fair", "☼ 晴 · 情感陪伴"],
|
|
35
|
-
["/help", "查看所有命令"],
|
|
36
|
-
["/setup", "配置向导"],
|
|
37
|
-
["/
|
|
38
|
-
["/
|
|
39
|
-
["/
|
|
40
|
-
["/
|
|
41
|
-
["/
|
|
42
|
-
["/
|
|
43
|
-
["/
|
|
44
|
-
["/
|
|
45
|
-
["/
|
|
46
|
-
["/
|
|
47
|
-
["/
|
|
48
|
-
["/
|
|
49
|
-
["/
|
|
50
|
-
["/
|
|
51
|
-
]
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
(cp >=
|
|
69
|
-
(cp >=
|
|
70
|
-
(cp >=
|
|
71
|
-
(cp >=
|
|
72
|
-
(cp >=
|
|
73
|
-
(cp >=
|
|
74
|
-
(cp >=
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
const
|
|
91
|
-
return
|
|
92
|
-
}
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
private
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
this.word
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
if (
|
|
146
|
-
|
|
147
|
-
this.
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
this.
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
}
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
const
|
|
178
|
-
return
|
|
179
|
-
}
|
|
180
|
-
|
|
181
|
-
/**
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
});
|
|
208
|
-
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
|
|
1
|
+
/**
|
|
2
|
+
* 天空织机 TUI — a polished *linear* terminal interface.
|
|
3
|
+
*
|
|
4
|
+
* Design note: the previous version tried to be a full-screen app, redrawing
|
|
5
|
+
* the whole screen on every keystroke while the reply streamed linearly below
|
|
6
|
+
* it — the two fought, the conversation never persisted, and hand-rolled
|
|
7
|
+
* raw-mode editing mangled CJK width. This rewrite is linear (like Claude Code
|
|
8
|
+
* / opencode): real readline line-editing + a CJK-aware wrapping stream
|
|
9
|
+
* renderer. Robust, flicker-free, and it actually reads like a conversation.
|
|
10
|
+
*/
|
|
11
|
+
|
|
12
|
+
import * as readline from "readline";
|
|
13
|
+
import chalk from "chalk";
|
|
14
|
+
import { agentTheme, PALETTE } from "../core/theme";
|
|
15
|
+
|
|
16
|
+
const TUI_VERSION = (() => { try { return require("../../package.json").version; } catch { return ""; } })();
|
|
17
|
+
|
|
18
|
+
export interface TUIContext {
|
|
19
|
+
agent: any;
|
|
20
|
+
agents: Map<string, any>;
|
|
21
|
+
model: string;
|
|
22
|
+
cost: string;
|
|
23
|
+
width: number;
|
|
24
|
+
height: number;
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
/* ── Slash commands (for tab-completion + the inline palette) ── */
|
|
28
|
+
export const SLASH_COMMANDS: [string, string][] = [
|
|
29
|
+
["/fog", "≋ 雾 · 探索洞察"],
|
|
30
|
+
["/rain", "⸽ 雨 · 创造产出"],
|
|
31
|
+
["/frost", "✱ 霜 · 精炼品质"],
|
|
32
|
+
["/snow", "❉ 雪 · 架构规划"],
|
|
33
|
+
["/dew", "∘ 露 · 可靠守护"],
|
|
34
|
+
["/fair", "☼ 晴 · 情感陪伴"],
|
|
35
|
+
["/help", "查看所有命令"],
|
|
36
|
+
["/setup", "配置向导"],
|
|
37
|
+
["/init", "扫描项目生成 SKY.md"],
|
|
38
|
+
["/plan", "切换计划模式(只读出方案)"],
|
|
39
|
+
["/verify", "运行项目验证命令"],
|
|
40
|
+
["/context", "上下文占用明细"],
|
|
41
|
+
["/rewind", "回退本轮文件改动"],
|
|
42
|
+
["/tools", "工具调用统计"],
|
|
43
|
+
["/model", "查看/切换模型(独立/统一)"],
|
|
44
|
+
["/cost", "费用统计"],
|
|
45
|
+
["/status", "状态总览"],
|
|
46
|
+
["/memory", "记忆状态"],
|
|
47
|
+
["/sessions", "会话列表"],
|
|
48
|
+
["/resume ", "恢复会话(序号/id)"],
|
|
49
|
+
["/new", "开始新会话"],
|
|
50
|
+
["/workspace", "工作空间"],
|
|
51
|
+
["/compact", "压缩上下文"],
|
|
52
|
+
["/clear", "清屏"],
|
|
53
|
+
["/task ", "多 Agent 编排"],
|
|
54
|
+
["/mcp", "MCP 服务器"],
|
|
55
|
+
["/version", "版本信息"],
|
|
56
|
+
["/quit", "退出"],
|
|
57
|
+
];
|
|
58
|
+
|
|
59
|
+
/* ════════════════════════════════════════
|
|
60
|
+
CJK-aware display width
|
|
61
|
+
════════════════════════════════════════ */
|
|
62
|
+
/** Visual columns occupied by a single code point (CJK / fullwidth = 2). */
|
|
63
|
+
export function charWidth(cp: number): number {
|
|
64
|
+
if (cp === 0) return 0;
|
|
65
|
+
if (cp < 32 || (cp >= 0x7f && cp < 0xa0)) return 0; // control
|
|
66
|
+
// East-Asian wide / fullwidth ranges
|
|
67
|
+
if (
|
|
68
|
+
(cp >= 0x1100 && cp <= 0x115f) || // Hangul Jamo
|
|
69
|
+
(cp >= 0x2e80 && cp <= 0x303e) || // CJK radicals, Kangxi, punctuation
|
|
70
|
+
(cp >= 0x3041 && cp <= 0x33ff) || // Hiragana…CJK symbols
|
|
71
|
+
(cp >= 0x3400 && cp <= 0x4dbf) || // CJK Ext A
|
|
72
|
+
(cp >= 0x4e00 && cp <= 0x9fff) || // CJK Unified
|
|
73
|
+
(cp >= 0xa000 && cp <= 0xa4cf) || // Yi
|
|
74
|
+
(cp >= 0xac00 && cp <= 0xd7a3) || // Hangul syllables
|
|
75
|
+
(cp >= 0xf900 && cp <= 0xfaff) || // CJK compat
|
|
76
|
+
(cp >= 0xfe10 && cp <= 0xfe19) ||
|
|
77
|
+
(cp >= 0xfe30 && cp <= 0xfe6f) || // CJK compat forms
|
|
78
|
+
(cp >= 0xff00 && cp <= 0xff60) || // Fullwidth forms
|
|
79
|
+
(cp >= 0xffe0 && cp <= 0xffe6) ||
|
|
80
|
+
(cp >= 0x1f300 && cp <= 0x1faff) // emoji / pictographs
|
|
81
|
+
) return 2;
|
|
82
|
+
return 1;
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
const ANSI_RE = /\x1b\[[0-9;]*m/g;
|
|
86
|
+
|
|
87
|
+
/** Visual width of a string, ignoring ANSI color codes. */
|
|
88
|
+
export function visualWidth(s: string): number {
|
|
89
|
+
let w = 0;
|
|
90
|
+
for (const ch of s.replace(ANSI_RE, "")) w += charWidth(ch.codePointAt(0) || 0);
|
|
91
|
+
return w;
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
/** Pad a string (containing ANSI) to a visual width. */
|
|
95
|
+
export function padVisual(s: string, width: number): string {
|
|
96
|
+
const diff = width - visualWidth(s);
|
|
97
|
+
return diff > 0 ? s + " ".repeat(diff) : s;
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
/* ════════════════════════════════════════
|
|
101
|
+
Streaming renderer — word-wrap aware, CJK aware
|
|
102
|
+
════════════════════════════════════════ */
|
|
103
|
+
/**
|
|
104
|
+
* Writes streamed text with a fixed left gutter, wrapping at the terminal
|
|
105
|
+
* width. English wraps on word boundaries; CJK wraps per glyph. Color is
|
|
106
|
+
* applied per flushed chunk so styling survives wrapping.
|
|
107
|
+
*/
|
|
108
|
+
export class StreamRenderer {
|
|
109
|
+
private col = 0;
|
|
110
|
+
private word = "";
|
|
111
|
+
private atLineStart = true;
|
|
112
|
+
private out: NodeJS.WriteStream;
|
|
113
|
+
private gutter: string;
|
|
114
|
+
private maxCols: number;
|
|
115
|
+
private color: (s: string) => string;
|
|
116
|
+
|
|
117
|
+
constructor(out: NodeJS.WriteStream, opts?: { gutter?: string; color?: (s: string) => string }) {
|
|
118
|
+
this.out = out;
|
|
119
|
+
this.gutter = opts?.gutter ?? " ";
|
|
120
|
+
this.color = opts?.color ?? ((s) => s);
|
|
121
|
+
const cols = out.columns || 80;
|
|
122
|
+
// content width excludes the gutter; clamp for readability
|
|
123
|
+
this.maxCols = Math.max(32, Math.min(cols - visualWidth(this.gutter) - 1, 96));
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
/** Lazily emit the left gutter at the start of each visual line. */
|
|
127
|
+
private startLine() { if (this.atLineStart) { this.out.write(this.gutter); this.atLineStart = false; } }
|
|
128
|
+
private newline() { this.out.write("\n"); this.atLineStart = true; this.col = 0; }
|
|
129
|
+
|
|
130
|
+
private flushWord() {
|
|
131
|
+
if (!this.word) return;
|
|
132
|
+
const w = visualWidth(this.word);
|
|
133
|
+
if (this.col > 0 && this.col + w > this.maxCols) this.newline();
|
|
134
|
+
this.startLine();
|
|
135
|
+
this.out.write(this.color(this.word));
|
|
136
|
+
this.col += w;
|
|
137
|
+
this.word = "";
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
/** Feed a chunk of streamed text. */
|
|
141
|
+
write(text: string) {
|
|
142
|
+
for (const ch of text) {
|
|
143
|
+
if (ch === "\r") continue; // normalize CRLF / stray CR from providers
|
|
144
|
+
if (ch === "\n") { this.flushWord(); this.newline(); continue; }
|
|
145
|
+
if (ch === " " || ch === "\t") {
|
|
146
|
+
this.flushWord();
|
|
147
|
+
if (this.col > 0 && this.col < this.maxCols) { this.startLine(); this.out.write(" "); this.col += 1; }
|
|
148
|
+
continue;
|
|
149
|
+
}
|
|
150
|
+
const cp = ch.codePointAt(0) || 0;
|
|
151
|
+
if (charWidth(cp) === 2) {
|
|
152
|
+
// CJK / wide: flush any pending latin word, then place this glyph
|
|
153
|
+
this.flushWord();
|
|
154
|
+
if (this.col > 0 && this.col + 2 > this.maxCols) this.newline();
|
|
155
|
+
this.startLine();
|
|
156
|
+
this.out.write(this.color(ch));
|
|
157
|
+
this.col += 2;
|
|
158
|
+
} else {
|
|
159
|
+
this.word += ch;
|
|
160
|
+
// very long unbroken token: hard-break to avoid overflow
|
|
161
|
+
if (visualWidth(this.word) >= this.maxCols) this.flushWord();
|
|
162
|
+
}
|
|
163
|
+
}
|
|
164
|
+
}
|
|
165
|
+
|
|
166
|
+
/** Flush any buffered word (call before switching styles / ending). */
|
|
167
|
+
flush() { this.flushWord(); }
|
|
168
|
+
}
|
|
169
|
+
|
|
170
|
+
/* ════════════════════════════════════════
|
|
171
|
+
Input — readline-based, robust line editing
|
|
172
|
+
════════════════════════════════════════ */
|
|
173
|
+
/** Tab-completer for slash commands. */
|
|
174
|
+
function slashCompleter(line: string): [string[], string] {
|
|
175
|
+
if (!line.startsWith("/")) return [[], line];
|
|
176
|
+
const names = SLASH_COMMANDS.map(([c]) => c.trimEnd());
|
|
177
|
+
const hits = names.filter((c) => c.startsWith(line));
|
|
178
|
+
return [hits.length ? hits : names, line];
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
/** The prompt string for an agent: a small mineral seal + chevron. */
|
|
182
|
+
export function promptFor(agentName: string): string {
|
|
183
|
+
const t = agentTheme(agentName);
|
|
184
|
+
return chalk.hex(t.hex)(` ${t.symbol} ${t.kanji} `) + chalk.hex(PALETTE.inkLight)("❯ ");
|
|
185
|
+
}
|
|
186
|
+
|
|
187
|
+
/** Cross-turn input history (↑/↓), shared by every per-turn reader. */
|
|
188
|
+
const inputHistory: string[] = [];
|
|
189
|
+
|
|
190
|
+
/**
|
|
191
|
+
* Read one line with the agent-themed prompt. A fresh readline interface is
|
|
192
|
+
* created and closed per call — this deliberately avoids clashing with the
|
|
193
|
+
* separate readline prompts used by the setup wizard and tool-approval flow
|
|
194
|
+
* (two live interfaces on one stdin corrupt input). History is preserved
|
|
195
|
+
* manually across turns.
|
|
196
|
+
*/
|
|
197
|
+
export function readLine(agentName: string, out: NodeJS.WriteStream = process.stdout): Promise<string> {
|
|
198
|
+
return new Promise((resolve) => {
|
|
199
|
+
const rl = readline.createInterface({
|
|
200
|
+
input: process.stdin,
|
|
201
|
+
output: out,
|
|
202
|
+
completer: slashCompleter,
|
|
203
|
+
terminal: process.stdin.isTTY ?? false,
|
|
204
|
+
history: [...inputHistory],
|
|
205
|
+
historySize: 200,
|
|
206
|
+
} as any);
|
|
207
|
+
rl.on("SIGINT", () => { out.write("\n" + chalk.dim(" 再会。\n")); rl.close(); process.exit(0); });
|
|
208
|
+
rl.question(promptFor(agentName), (answer) => {
|
|
209
|
+
const trimmed = answer.trim();
|
|
210
|
+
if (trimmed) inputHistory.unshift(trimmed);
|
|
211
|
+
rl.close();
|
|
212
|
+
resolve(trimmed);
|
|
213
|
+
});
|
|
214
|
+
});
|
|
215
|
+
}
|
|
216
|
+
|
|
217
|
+
/** Render the inline slash-command palette (printed, not full-screen). */
|
|
218
|
+
export function renderPalette(filter: string): string {
|
|
219
|
+
const f = filter.toLowerCase();
|
|
220
|
+
const matches = SLASH_COMMANDS.filter(([c]) => c.toLowerCase().startsWith(f));
|
|
221
|
+
const list = matches.length ? matches : SLASH_COMMANDS;
|
|
222
|
+
const lines = list.slice(0, 12).map(([cmd, desc]) => {
|
|
223
|
+
const isAgent = ["/fog", "/rain", "/frost", "/snow", "/dew", "/fair"].includes(cmd.trim());
|
|
224
|
+
const name = isAgent ? chalk.hex(agentTheme(cmd.trim().slice(1)).hex)(cmd.padEnd(12)) : chalk.hex(PALETTE.inkMid)(cmd.padEnd(12));
|
|
225
|
+
return " " + name + chalk.hex(PALETTE.inkLight)(desc);
|
|
226
|
+
});
|
|
227
|
+
return chalk.dim(" 命令 · Tab 补全\n") + lines.join("\n") + "\n";
|
|
228
|
+
}
|
|
@@ -0,0 +1,134 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Anti-loop guard for the agent reasoning loop.
|
|
3
|
+
*
|
|
4
|
+
* Extracted from chatStreamImpl (Phase 3). Holds the per-turn heuristic state
|
|
5
|
+
* (recent response texts, tool-call signatures, tool outcomes + once-only hint
|
|
6
|
+
* flags) and, after each round, returns a decision: zero or more system
|
|
7
|
+
* "hints" to nudge the model, and optionally a hard `stop` (assistant note +
|
|
8
|
+
* a user-visible content line). The agent loop applies the decision — the guard
|
|
9
|
+
* itself has no side effects, which makes every branch unit-testable.
|
|
10
|
+
*
|
|
11
|
+
* A fresh LoopGuard is created per turn (state must not leak across turns).
|
|
12
|
+
*/
|
|
13
|
+
|
|
14
|
+
import type { ToolCall } from '../llm';
|
|
15
|
+
import {
|
|
16
|
+
toolCallSignature,
|
|
17
|
+
textSimilarity,
|
|
18
|
+
looksLikeFailedToolResult,
|
|
19
|
+
parseToolArgs,
|
|
20
|
+
SIG_WINDOW,
|
|
21
|
+
SIG_LOOP_HINT,
|
|
22
|
+
SIG_LOOP_HARDSTOP,
|
|
23
|
+
} from '../agent_helpers';
|
|
24
|
+
|
|
25
|
+
/** A hard stop: record `note` as an assistant message, then surface `contentLine`. */
|
|
26
|
+
export interface GuardStop {
|
|
27
|
+
note: string;
|
|
28
|
+
contentLine: string;
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
/** The guard's decision for one round. `hints` apply in order; `stop` ends the turn. */
|
|
32
|
+
export interface GuardDecision {
|
|
33
|
+
hints: string[];
|
|
34
|
+
stop?: GuardStop;
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
/** Minimal shape of a tool execution result the guard inspects. */
|
|
38
|
+
export interface GuardExecResult {
|
|
39
|
+
toolName: string;
|
|
40
|
+
success: boolean;
|
|
41
|
+
result: string;
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
export class LoopGuard {
|
|
45
|
+
private recentResponseTexts: string[] = [];
|
|
46
|
+
private recentToolSigs: string[] = [];
|
|
47
|
+
private recentToolOutcomes: boolean[] = [];
|
|
48
|
+
private searchCount = 0; // cumulative search/fetch calls this turn (not window-bounded)
|
|
49
|
+
private repetitionHintInjected = false;
|
|
50
|
+
private toolLoopHintInjected = false; // shared by tool-signature loop + search-storm
|
|
51
|
+
private stuckHintInjected = false;
|
|
52
|
+
|
|
53
|
+
/**
|
|
54
|
+
* Observe one completed round. Mutates internal state and returns the
|
|
55
|
+
* hints/stop decision. Evaluation order (and the shared hint flag) mirrors
|
|
56
|
+
* the original inline logic exactly.
|
|
57
|
+
*/
|
|
58
|
+
observe(
|
|
59
|
+
roundContent: string,
|
|
60
|
+
toolCallsReceived: ToolCall[],
|
|
61
|
+
execResults: Array<GuardExecResult | null>
|
|
62
|
+
): GuardDecision {
|
|
63
|
+
const hints: string[] = [];
|
|
64
|
+
|
|
65
|
+
// 1. Narration-loop: response too similar to a recent one.
|
|
66
|
+
const normalizedRound = (roundContent || '').trim();
|
|
67
|
+
if (normalizedRound && this.recentResponseTexts.length > 0) {
|
|
68
|
+
const highSim = this.recentResponseTexts.slice(-2).some(prev => textSimilarity(normalizedRound, prev) >= 0.7);
|
|
69
|
+
if (highSim && !this.repetitionHintInjected) {
|
|
70
|
+
hints.push('[Stop narrating] Your last response is highly similar to your previous one. Stop writing prose. Either: (1) emit ONLY the next tool call, or (2) output the final deliverable.');
|
|
71
|
+
this.repetitionHintInjected = true;
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
this.recentResponseTexts.push(normalizedRound);
|
|
75
|
+
if (this.recentResponseTexts.length > 3) this.recentResponseTexts.shift();
|
|
76
|
+
|
|
77
|
+
// 2. Tool-signature loop: same call repeated within the window.
|
|
78
|
+
for (const tc of toolCallsReceived) {
|
|
79
|
+
const tName = tc.function.name;
|
|
80
|
+
if (['task_done', 'list_skills', 'use_skill'].includes(tName)) continue;
|
|
81
|
+
if (['web_search', 'fetch_page', 'http_get'].includes(tName)) this.searchCount++;
|
|
82
|
+
const rawArgs = tc.function.arguments;
|
|
83
|
+
const tArgs = typeof rawArgs === 'string' ? parseToolArgs(rawArgs) : rawArgs;
|
|
84
|
+
const sig = toolCallSignature(tName, tArgs);
|
|
85
|
+
if (sig) this.recentToolSigs.push(sig);
|
|
86
|
+
}
|
|
87
|
+
if (this.recentToolSigs.length > SIG_WINDOW) {
|
|
88
|
+
this.recentToolSigs.splice(0, this.recentToolSigs.length - SIG_WINDOW);
|
|
89
|
+
}
|
|
90
|
+
if (this.recentToolSigs.length > 0) {
|
|
91
|
+
const counts = new Map<string, number>();
|
|
92
|
+
for (const s of this.recentToolSigs) counts.set(s, (counts.get(s) || 0) + 1);
|
|
93
|
+
let topSig = '';
|
|
94
|
+
let topCount = 0;
|
|
95
|
+
for (const [s, c] of counts) { if (c > topCount) { topSig = s; topCount = c; } }
|
|
96
|
+
if (topCount >= SIG_LOOP_HINT && !this.toolLoopHintInjected) {
|
|
97
|
+
hints.push(`[Tool loop] You have called \`${topSig}\` ${topCount}x in the last ${this.recentToolSigs.length} tool calls — you are iterating without converging. STOP repeating it.`);
|
|
98
|
+
this.toolLoopHintInjected = true;
|
|
99
|
+
}
|
|
100
|
+
if (topCount >= SIG_LOOP_HARDSTOP) {
|
|
101
|
+
return { hints, stop: { note: `I have repeated \`${topSig}\` ${topCount} times without converging. Stopping.`, contentLine: `\n\n[stuck] tool \`${topSig}\` repeated ${topCount}x — stopping.` } };
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
// 3. Stuck-loop: most/all recent tool calls failed.
|
|
106
|
+
for (const r of execResults) {
|
|
107
|
+
if (!r || r.toolName === 'task_done') continue;
|
|
108
|
+
const failed = !r.success || (typeof r.result === 'string' && looksLikeFailedToolResult(r.result));
|
|
109
|
+
this.recentToolOutcomes.push(!failed);
|
|
110
|
+
// Keep 8 so the "all recent calls failed" (>=8) hard-stop below is reachable.
|
|
111
|
+
if (this.recentToolOutcomes.length > 8) this.recentToolOutcomes.shift();
|
|
112
|
+
}
|
|
113
|
+
if (!this.stuckHintInjected && this.recentToolOutcomes.length >= 5 &&
|
|
114
|
+
this.recentToolOutcomes.filter(Boolean).length <= 1) {
|
|
115
|
+
hints.push('[Recovery hint] Your last several tool calls have mostly failed. Synthesize a partial answer from what worked or ask the user for guidance.');
|
|
116
|
+
this.stuckHintInjected = true;
|
|
117
|
+
}
|
|
118
|
+
if (this.recentToolOutcomes.length >= 8 && this.recentToolOutcomes.every(x => !x)) {
|
|
119
|
+
return { hints, stop: { note: 'Every recent tool call failed. Please give me more context.', contentLine: '\n\n[stuck] every recent tool call failed — stopping.\n' } };
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
// 4. Search-storm: cumulative search/fetch calls this turn (not bounded by
|
|
123
|
+
// SIG_WINDOW, so the >=12 hard-stop is actually reachable).
|
|
124
|
+
if (this.searchCount >= 8 && !this.toolLoopHintInjected) {
|
|
125
|
+
hints.push(`[Search storm] ${this.searchCount} search calls. STOP searching and synthesize.`);
|
|
126
|
+
this.toolLoopHintInjected = true;
|
|
127
|
+
}
|
|
128
|
+
if (this.searchCount >= 12) {
|
|
129
|
+
return { hints, stop: { note: 'Too many search requests. Synthesizing best answer.', contentLine: `\n\n[stuck] excessive web searching (${this.searchCount} calls) — stopping.\n` } };
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
return { hints };
|
|
133
|
+
}
|
|
134
|
+
}
|