skyloom 1.13.6 → 1.13.8
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.d.ts +20 -0
- package/dist/core/agent.d.ts.map +1 -1
- package/dist/core/agent.js +199 -16
- 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 +68 -68
- 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 -134
- package/src/core/agent/task.ts +100 -100
- package/src/core/agent.ts +195 -16
- 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 -75
- 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/=12 +0 -0
- package/=8 +0 -0
package/src/core/longdoc.ts
CHANGED
|
@@ -1,155 +1,155 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* 长文档处理策略 — sliding window + summary chain.
|
|
3
|
-
*
|
|
4
|
-
* When an input exceeds the agent's effective context window,
|
|
5
|
-
* split into overlapping chunks, summarize each, then chain
|
|
6
|
-
* summaries into a final digest.
|
|
7
|
-
*
|
|
8
|
-
* Architecture:
|
|
9
|
-
* Input → Chunk(sliding window) → Per-chunk Summary → Chain → Final Digest
|
|
10
|
-
*
|
|
11
|
-
* All summaries are generated by the calling agent's LLM, so quality
|
|
12
|
-
* depends on the model in use. The chunker is pure text processing
|
|
13
|
-
* and works without any LLM call.
|
|
14
|
-
*/
|
|
15
|
-
|
|
16
|
-
import type { BaseAgent } from "./agent";
|
|
17
|
-
|
|
18
|
-
/* ═══════════════════════════════════════
|
|
19
|
-
Chunker — split text into overlapping windows
|
|
20
|
-
═══════════════════════════════════════ */
|
|
21
|
-
export interface ChunkOptions {
|
|
22
|
-
/** Target chunk size in characters (default 6000) */
|
|
23
|
-
chunkSize?: number;
|
|
24
|
-
/** Overlap between consecutive chunks in characters (default 800) */
|
|
25
|
-
overlap?: number;
|
|
26
|
-
/** Minimum chunk size before we stop splitting (default 500) */
|
|
27
|
-
minChunk?: number;
|
|
28
|
-
}
|
|
29
|
-
|
|
30
|
-
export function chunkText(text: string, opts?: ChunkOptions): string[] {
|
|
31
|
-
const cs = opts?.chunkSize ?? 6000;
|
|
32
|
-
const ol = opts?.overlap ?? 800;
|
|
33
|
-
const min = opts?.minChunk ?? 500;
|
|
34
|
-
const chunks: string[] = [];
|
|
35
|
-
|
|
36
|
-
if (text.length <= cs + min) { chunks.push(text); return chunks; }
|
|
37
|
-
|
|
38
|
-
let start = 0;
|
|
39
|
-
while (start < text.length) {
|
|
40
|
-
let end = start + cs;
|
|
41
|
-
if (end >= text.length) { end = text.length; }
|
|
42
|
-
else {
|
|
43
|
-
// Try to break at paragraph boundary
|
|
44
|
-
const searchEnd = Math.min(end + 400, text.length);
|
|
45
|
-
const paraBreak = text.lastIndexOf("\n\n", searchEnd);
|
|
46
|
-
if (paraBreak > start + min) end = paraBreak;
|
|
47
|
-
else {
|
|
48
|
-
const lineBreak = text.lastIndexOf("\n", searchEnd);
|
|
49
|
-
if (lineBreak > start + min) end = lineBreak;
|
|
50
|
-
else {
|
|
51
|
-
const space = text.lastIndexOf(" ", searchEnd);
|
|
52
|
-
if (space > start + min) end = space;
|
|
53
|
-
}
|
|
54
|
-
}
|
|
55
|
-
}
|
|
56
|
-
|
|
57
|
-
chunks.push(text.slice(start, end).trim());
|
|
58
|
-
if (end >= text.length) break;
|
|
59
|
-
start = end - ol;
|
|
60
|
-
if (start < 0) start = 0;
|
|
61
|
-
}
|
|
62
|
-
|
|
63
|
-
return chunks;
|
|
64
|
-
}
|
|
65
|
-
|
|
66
|
-
/* ═══════════════════════════════════════
|
|
67
|
-
Summary chain — ask agent to summarize chunks then chain
|
|
68
|
-
═══════════════════════════════════════ */
|
|
69
|
-
export interface SummaryOptions {
|
|
70
|
-
/** Max total chars for the final digest (default 3000) */
|
|
71
|
-
maxDigestChars?: number;
|
|
72
|
-
/** Custom summarization prompt for each chunk */
|
|
73
|
-
chunkPrompt?: string;
|
|
74
|
-
/** Custom chain prompt for combining summaries */
|
|
75
|
-
chainPrompt?: string;
|
|
76
|
-
}
|
|
77
|
-
|
|
78
|
-
const DEFAULT_CHUNK_PROMPT = `Summarize the following text concisely. Keep all key facts, names, numbers, and code snippets. Output the summary directly without preamble. Limit to 300 words.
|
|
79
|
-
|
|
80
|
-
Text:
|
|
81
|
-
{text}`;
|
|
82
|
-
|
|
83
|
-
const DEFAULT_CHAIN_PROMPT = `Combine the following section summaries into a single coherent digest. Preserve all key facts, remove redundancy. Output directly without preamble.
|
|
84
|
-
|
|
85
|
-
{summaries}`;
|
|
86
|
-
|
|
87
|
-
export async function summarizeLongDoc(
|
|
88
|
-
agent: BaseAgent,
|
|
89
|
-
text: string,
|
|
90
|
-
opts?: SummaryOptions
|
|
91
|
-
): Promise<string> {
|
|
92
|
-
const maxDigest = opts?.maxDigestChars ?? 3000;
|
|
93
|
-
const chunks = chunkText(text);
|
|
94
|
-
|
|
95
|
-
// Single chunk — no summarization needed
|
|
96
|
-
if (chunks.length <= 1) {
|
|
97
|
-
if (text.length <= maxDigest) return text;
|
|
98
|
-
const prompt = (opts?.chunkPrompt || DEFAULT_CHUNK_PROMPT).replace("{text}", text);
|
|
99
|
-
return agent.chatOneshot(prompt, { maxTokens: maxDigest });
|
|
100
|
-
}
|
|
101
|
-
|
|
102
|
-
// Multi-chunk: summarize each, then chain
|
|
103
|
-
const summaries: string[] = [];
|
|
104
|
-
for (let i = 0; i < chunks.length; i++) {
|
|
105
|
-
const prompt = (opts?.chunkPrompt || DEFAULT_CHUNK_PROMPT).replace("{text}", chunks[i]);
|
|
106
|
-
try {
|
|
107
|
-
const s = await agent.chatOneshot(prompt, { maxTokens: 600 });
|
|
108
|
-
summaries.push(s);
|
|
109
|
-
} catch {
|
|
110
|
-
summaries.push(chunks[i].slice(0, 400) + "...");
|
|
111
|
-
}
|
|
112
|
-
}
|
|
113
|
-
|
|
114
|
-
// Chain summaries
|
|
115
|
-
if (summaries.length === 1) return summaries[0].slice(0, maxDigest);
|
|
116
|
-
|
|
117
|
-
const joined = summaries.map((s, i) => `## Section ${i + 1}\n${s}`).join("\n\n");
|
|
118
|
-
if (joined.length <= maxDigest) return joined;
|
|
119
|
-
|
|
120
|
-
const chainPrompt = (opts?.chainPrompt || DEFAULT_CHAIN_PROMPT).replace("{summaries}", joined);
|
|
121
|
-
const final = await agent.chatOneshot(chainPrompt, { maxTokens: maxDigest });
|
|
122
|
-
return final.slice(0, maxDigest);
|
|
123
|
-
}
|
|
124
|
-
|
|
125
|
-
/* ═══════════════════════════════════════
|
|
126
|
-
Structured data parsing helpers
|
|
127
|
-
═══════════════════════════════════════ */
|
|
128
|
-
export function parseStructuredInput(input: string): {
|
|
129
|
-
hasTable: boolean; hasJSON: boolean; hasCSV: boolean;
|
|
130
|
-
extractedJSON: string | null; extractedTable: string[][] | null;
|
|
131
|
-
} {
|
|
132
|
-
const result = { hasTable: false, hasJSON: false, hasCSV: false, extractedJSON: null as string | null, extractedTable: null as string[][] | null };
|
|
133
|
-
|
|
134
|
-
// Detect JSON
|
|
135
|
-
const jsonMatch = input.match(/\{[\s\S]*\}|\[[\s\S]*\]/);
|
|
136
|
-
if (jsonMatch) {
|
|
137
|
-
try { JSON.parse(jsonMatch[0]); result.hasJSON = true; result.extractedJSON = jsonMatch[0]; }
|
|
138
|
-
catch { /* not valid JSON */ }
|
|
139
|
-
}
|
|
140
|
-
|
|
141
|
-
// Detect markdown table
|
|
142
|
-
const tableMatch = input.match(/\|[\s\S]*?\|/);
|
|
143
|
-
if (tableMatch) {
|
|
144
|
-
result.hasTable = true;
|
|
145
|
-
const lines = input.split("\n").filter(l => l.includes("|") && !l.startsWith("|---") && !l.startsWith("| --"));
|
|
146
|
-
result.extractedTable = lines.map(l => l.split("|").filter(c => c.trim()).map(c => c.trim()));
|
|
147
|
-
}
|
|
148
|
-
|
|
149
|
-
// Detect CSV
|
|
150
|
-
if (input.includes(",") && input.split("\n").filter(l => l.includes(",")).length >= 2) {
|
|
151
|
-
result.hasCSV = true;
|
|
152
|
-
}
|
|
153
|
-
|
|
154
|
-
return result;
|
|
155
|
-
}
|
|
1
|
+
/**
|
|
2
|
+
* 长文档处理策略 — sliding window + summary chain.
|
|
3
|
+
*
|
|
4
|
+
* When an input exceeds the agent's effective context window,
|
|
5
|
+
* split into overlapping chunks, summarize each, then chain
|
|
6
|
+
* summaries into a final digest.
|
|
7
|
+
*
|
|
8
|
+
* Architecture:
|
|
9
|
+
* Input → Chunk(sliding window) → Per-chunk Summary → Chain → Final Digest
|
|
10
|
+
*
|
|
11
|
+
* All summaries are generated by the calling agent's LLM, so quality
|
|
12
|
+
* depends on the model in use. The chunker is pure text processing
|
|
13
|
+
* and works without any LLM call.
|
|
14
|
+
*/
|
|
15
|
+
|
|
16
|
+
import type { BaseAgent } from "./agent";
|
|
17
|
+
|
|
18
|
+
/* ═══════════════════════════════════════
|
|
19
|
+
Chunker — split text into overlapping windows
|
|
20
|
+
═══════════════════════════════════════ */
|
|
21
|
+
export interface ChunkOptions {
|
|
22
|
+
/** Target chunk size in characters (default 6000) */
|
|
23
|
+
chunkSize?: number;
|
|
24
|
+
/** Overlap between consecutive chunks in characters (default 800) */
|
|
25
|
+
overlap?: number;
|
|
26
|
+
/** Minimum chunk size before we stop splitting (default 500) */
|
|
27
|
+
minChunk?: number;
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
export function chunkText(text: string, opts?: ChunkOptions): string[] {
|
|
31
|
+
const cs = opts?.chunkSize ?? 6000;
|
|
32
|
+
const ol = opts?.overlap ?? 800;
|
|
33
|
+
const min = opts?.minChunk ?? 500;
|
|
34
|
+
const chunks: string[] = [];
|
|
35
|
+
|
|
36
|
+
if (text.length <= cs + min) { chunks.push(text); return chunks; }
|
|
37
|
+
|
|
38
|
+
let start = 0;
|
|
39
|
+
while (start < text.length) {
|
|
40
|
+
let end = start + cs;
|
|
41
|
+
if (end >= text.length) { end = text.length; }
|
|
42
|
+
else {
|
|
43
|
+
// Try to break at paragraph boundary
|
|
44
|
+
const searchEnd = Math.min(end + 400, text.length);
|
|
45
|
+
const paraBreak = text.lastIndexOf("\n\n", searchEnd);
|
|
46
|
+
if (paraBreak > start + min) end = paraBreak;
|
|
47
|
+
else {
|
|
48
|
+
const lineBreak = text.lastIndexOf("\n", searchEnd);
|
|
49
|
+
if (lineBreak > start + min) end = lineBreak;
|
|
50
|
+
else {
|
|
51
|
+
const space = text.lastIndexOf(" ", searchEnd);
|
|
52
|
+
if (space > start + min) end = space;
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
chunks.push(text.slice(start, end).trim());
|
|
58
|
+
if (end >= text.length) break;
|
|
59
|
+
start = end - ol;
|
|
60
|
+
if (start < 0) start = 0;
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
return chunks;
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
/* ═══════════════════════════════════════
|
|
67
|
+
Summary chain — ask agent to summarize chunks then chain
|
|
68
|
+
═══════════════════════════════════════ */
|
|
69
|
+
export interface SummaryOptions {
|
|
70
|
+
/** Max total chars for the final digest (default 3000) */
|
|
71
|
+
maxDigestChars?: number;
|
|
72
|
+
/** Custom summarization prompt for each chunk */
|
|
73
|
+
chunkPrompt?: string;
|
|
74
|
+
/** Custom chain prompt for combining summaries */
|
|
75
|
+
chainPrompt?: string;
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
const DEFAULT_CHUNK_PROMPT = `Summarize the following text concisely. Keep all key facts, names, numbers, and code snippets. Output the summary directly without preamble. Limit to 300 words.
|
|
79
|
+
|
|
80
|
+
Text:
|
|
81
|
+
{text}`;
|
|
82
|
+
|
|
83
|
+
const DEFAULT_CHAIN_PROMPT = `Combine the following section summaries into a single coherent digest. Preserve all key facts, remove redundancy. Output directly without preamble.
|
|
84
|
+
|
|
85
|
+
{summaries}`;
|
|
86
|
+
|
|
87
|
+
export async function summarizeLongDoc(
|
|
88
|
+
agent: BaseAgent,
|
|
89
|
+
text: string,
|
|
90
|
+
opts?: SummaryOptions
|
|
91
|
+
): Promise<string> {
|
|
92
|
+
const maxDigest = opts?.maxDigestChars ?? 3000;
|
|
93
|
+
const chunks = chunkText(text);
|
|
94
|
+
|
|
95
|
+
// Single chunk — no summarization needed
|
|
96
|
+
if (chunks.length <= 1) {
|
|
97
|
+
if (text.length <= maxDigest) return text;
|
|
98
|
+
const prompt = (opts?.chunkPrompt || DEFAULT_CHUNK_PROMPT).replace("{text}", text);
|
|
99
|
+
return agent.chatOneshot(prompt, { maxTokens: maxDigest });
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
// Multi-chunk: summarize each, then chain
|
|
103
|
+
const summaries: string[] = [];
|
|
104
|
+
for (let i = 0; i < chunks.length; i++) {
|
|
105
|
+
const prompt = (opts?.chunkPrompt || DEFAULT_CHUNK_PROMPT).replace("{text}", chunks[i]);
|
|
106
|
+
try {
|
|
107
|
+
const s = await agent.chatOneshot(prompt, { maxTokens: 600 });
|
|
108
|
+
summaries.push(s);
|
|
109
|
+
} catch {
|
|
110
|
+
summaries.push(chunks[i].slice(0, 400) + "...");
|
|
111
|
+
}
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
// Chain summaries
|
|
115
|
+
if (summaries.length === 1) return summaries[0].slice(0, maxDigest);
|
|
116
|
+
|
|
117
|
+
const joined = summaries.map((s, i) => `## Section ${i + 1}\n${s}`).join("\n\n");
|
|
118
|
+
if (joined.length <= maxDigest) return joined;
|
|
119
|
+
|
|
120
|
+
const chainPrompt = (opts?.chainPrompt || DEFAULT_CHAIN_PROMPT).replace("{summaries}", joined);
|
|
121
|
+
const final = await agent.chatOneshot(chainPrompt, { maxTokens: maxDigest });
|
|
122
|
+
return final.slice(0, maxDigest);
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
/* ═══════════════════════════════════════
|
|
126
|
+
Structured data parsing helpers
|
|
127
|
+
═══════════════════════════════════════ */
|
|
128
|
+
export function parseStructuredInput(input: string): {
|
|
129
|
+
hasTable: boolean; hasJSON: boolean; hasCSV: boolean;
|
|
130
|
+
extractedJSON: string | null; extractedTable: string[][] | null;
|
|
131
|
+
} {
|
|
132
|
+
const result = { hasTable: false, hasJSON: false, hasCSV: false, extractedJSON: null as string | null, extractedTable: null as string[][] | null };
|
|
133
|
+
|
|
134
|
+
// Detect JSON
|
|
135
|
+
const jsonMatch = input.match(/\{[\s\S]*\}|\[[\s\S]*\]/);
|
|
136
|
+
if (jsonMatch) {
|
|
137
|
+
try { JSON.parse(jsonMatch[0]); result.hasJSON = true; result.extractedJSON = jsonMatch[0]; }
|
|
138
|
+
catch { /* not valid JSON */ }
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
// Detect markdown table
|
|
142
|
+
const tableMatch = input.match(/\|[\s\S]*?\|/);
|
|
143
|
+
if (tableMatch) {
|
|
144
|
+
result.hasTable = true;
|
|
145
|
+
const lines = input.split("\n").filter(l => l.includes("|") && !l.startsWith("|---") && !l.startsWith("| --"));
|
|
146
|
+
result.extractedTable = lines.map(l => l.split("|").filter(c => c.trim()).map(c => c.trim()));
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
// Detect CSV
|
|
150
|
+
if (input.includes(",") && input.split("\n").filter(l => l.includes(",")).length >= 2) {
|
|
151
|
+
result.hasCSV = true;
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
return result;
|
|
155
|
+
}
|
package/src/core/mcp.ts
CHANGED
|
@@ -871,6 +871,54 @@ export class MCPManager {
|
|
|
871
871
|
}
|
|
872
872
|
}
|
|
873
873
|
|
|
874
|
+
/**
|
|
875
|
+
* Project-level .mcp.json (Claude Code standard) — drop the same file you
|
|
876
|
+
* use with Claude Code into the repo root and Skyloom picks it up:
|
|
877
|
+
*
|
|
878
|
+
* { "mcpServers": {
|
|
879
|
+
* "github": { "type": "http", "url": "https://api.example.com/mcp/" },
|
|
880
|
+
* "db": { "command": "npx", "args": ["-y", "@x/dbhub"],
|
|
881
|
+
* "env": { "DB_URL": "${DB_URL}" } } } }
|
|
882
|
+
*
|
|
883
|
+
* `${VAR}` references expand from the environment, so secrets stay out of
|
|
884
|
+
* the committed file.
|
|
885
|
+
*/
|
|
886
|
+
|
|
887
|
+
/** Expand ${VAR} environment references (Claude Code .mcp.json convention). */
|
|
888
|
+
export function expandEnvRefs(s: string): string {
|
|
889
|
+
return s.replace(/\$\{([A-Za-z0-9_]+)\}/g, (_, v) => process.env[v] ?? '');
|
|
890
|
+
}
|
|
891
|
+
|
|
892
|
+
/** Load and translate <cwd>/.mcp.json into Skyloom server configs. */
|
|
893
|
+
export function loadProjectMcpJson(cwd: string = process.cwd()): MCPServerConfig[] {
|
|
894
|
+
const file = path.join(cwd, '.mcp.json');
|
|
895
|
+
if (!fs.existsSync(file)) return [];
|
|
896
|
+
try {
|
|
897
|
+
const data = JSON.parse(fs.readFileSync(file, 'utf-8'));
|
|
898
|
+
const servers = data?.mcpServers;
|
|
899
|
+
if (!servers || typeof servers !== 'object') return [];
|
|
900
|
+
const out: MCPServerConfig[] = [];
|
|
901
|
+
for (const [name, raw] of Object.entries<any>(servers)) {
|
|
902
|
+
if (!raw || typeof raw !== 'object') continue;
|
|
903
|
+
const cfg: MCPServerConfig = { name, enabled: true };
|
|
904
|
+
if (typeof raw.command === 'string') {
|
|
905
|
+
cfg.command = expandEnvRefs(raw.command);
|
|
906
|
+
if (Array.isArray(raw.args)) cfg.args = raw.args.map((a: any) => expandEnvRefs(String(a)));
|
|
907
|
+
}
|
|
908
|
+
if (typeof raw.url === 'string') cfg.url = expandEnvRefs(raw.url);
|
|
909
|
+
if (raw.env && typeof raw.env === 'object') {
|
|
910
|
+
cfg.env = {};
|
|
911
|
+
for (const [k, v] of Object.entries(raw.env)) cfg.env[k] = expandEnvRefs(String(v));
|
|
912
|
+
}
|
|
913
|
+
if (!cfg.command && !cfg.url) continue; // unsupported transport entry
|
|
914
|
+
out.push(cfg);
|
|
915
|
+
}
|
|
916
|
+
return out;
|
|
917
|
+
} catch {
|
|
918
|
+
return [];
|
|
919
|
+
}
|
|
920
|
+
}
|
|
921
|
+
|
|
874
922
|
/**
|
|
875
923
|
* Persistence helpers for runtime-added MCP servers.
|
|
876
924
|
*/
|