@aexol/spectral 0.7.1 → 0.7.5
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/CHANGELOG.md +5 -0
- package/dist/agent/agents.js +1 -1
- package/dist/agent/index.js +199 -184
- package/dist/commands/serve.js +0 -3
- package/dist/designer/data/systems/renault/DESIGN.md +1 -1
- package/dist/designer/philosophies.js +668 -0
- package/dist/mcp/sampling-handler.js +1 -1
- package/dist/memory/commands/status.js +1 -1
- package/dist/memory/compaction.js +2 -2
- package/dist/memory/config.js +1 -1
- package/dist/memory/debug-log.js +1 -1
- package/dist/memory/hooks/compaction-hook.js +29 -0
- package/dist/memory/index.js +2 -0
- package/dist/memory/observer.js +2 -2
- package/dist/memory/project-observations-store.js +14 -0
- package/dist/memory/tokens.js +1 -1
- package/dist/memory/tools/read-project-observations.js +82 -0
- package/dist/memory/tools/recall-observation.js +2 -2
- package/dist/pi/agent-core/agent-loop.js +501 -0
- package/dist/pi/agent-core/agent.js +401 -0
- package/dist/pi/agent-core/harness/agent-harness.js +899 -0
- package/dist/pi/agent-core/harness/compaction/branch-summarization.js +173 -0
- package/dist/pi/agent-core/harness/compaction/compaction.js +532 -0
- package/dist/pi/agent-core/harness/compaction/utils.js +130 -0
- package/dist/pi/agent-core/harness/env/nodejs.js +485 -0
- package/dist/pi/agent-core/harness/messages.js +101 -0
- package/dist/pi/agent-core/harness/prompt-templates.js +229 -0
- package/dist/pi/agent-core/harness/session/jsonl-repo.js +100 -0
- package/dist/pi/agent-core/harness/session/jsonl-storage.js +230 -0
- package/dist/pi/agent-core/harness/session/memory-repo.js +41 -0
- package/dist/pi/agent-core/harness/session/memory-storage.js +113 -0
- package/dist/pi/agent-core/harness/session/repo-utils.js +38 -0
- package/dist/pi/agent-core/harness/session/session.js +196 -0
- package/dist/pi/agent-core/harness/session/uuid.js +49 -0
- package/dist/pi/agent-core/harness/skills.js +310 -0
- package/dist/pi/agent-core/harness/system-prompt.js +29 -0
- package/dist/pi/agent-core/harness/types.js +93 -0
- package/dist/pi/agent-core/harness/utils/shell-output.js +125 -0
- package/dist/pi/agent-core/harness/utils/truncate.js +289 -0
- package/dist/pi/agent-core/index.js +24 -0
- package/dist/pi/agent-core/node.js +2 -0
- package/dist/pi/agent-core/proxy.js +277 -0
- package/dist/pi/agent-core/types.js +1 -0
- package/dist/pi/ai/api-registry.js +43 -0
- package/dist/pi/ai/cli.js +120 -0
- package/dist/pi/ai/env-api-keys.js +169 -0
- package/dist/pi/ai/image-models.generated.js +441 -0
- package/dist/pi/ai/image-models.js +22 -0
- package/dist/pi/ai/images-api-registry.js +21 -0
- package/dist/pi/ai/images.js +13 -0
- package/dist/pi/ai/index.js +18 -0
- package/dist/pi/ai/models.generated.js +16220 -0
- package/dist/pi/ai/models.js +70 -0
- package/dist/pi/ai/oauth.js +1 -0
- package/dist/pi/ai/providers/anthropic.js +945 -0
- package/dist/pi/ai/providers/faux.js +367 -0
- package/dist/pi/ai/providers/github-copilot-headers.js +28 -0
- package/dist/pi/ai/providers/openai-completions.js +945 -0
- package/dist/pi/ai/providers/openai-prompt-cache.js +9 -0
- package/dist/pi/ai/providers/register-builtins.js +97 -0
- package/dist/pi/ai/providers/simple-options.js +40 -0
- package/dist/pi/ai/providers/transform-messages.js +183 -0
- package/dist/pi/ai/session-resources.js +21 -0
- package/dist/pi/ai/stream.js +26 -0
- package/dist/pi/ai/types.js +1 -0
- package/dist/pi/ai/utils/diagnostics.js +24 -0
- package/dist/pi/ai/utils/event-stream.js +80 -0
- package/dist/pi/ai/utils/hash.js +13 -0
- package/dist/pi/ai/utils/headers.js +7 -0
- package/dist/pi/ai/utils/json-parse.js +112 -0
- package/dist/pi/ai/utils/node-http-proxy.js +96 -0
- package/dist/pi/ai/utils/oauth/anthropic.js +334 -0
- package/dist/pi/ai/utils/oauth/device-code.js +54 -0
- package/dist/pi/ai/utils/oauth/github-copilot.js +270 -0
- package/dist/pi/ai/utils/oauth/index.js +121 -0
- package/dist/pi/ai/utils/oauth/oauth-page.js +104 -0
- package/dist/pi/ai/utils/oauth/openai-codex.js +384 -0
- package/dist/pi/ai/utils/oauth/pkce.js +30 -0
- package/dist/pi/ai/utils/oauth/types.js +1 -0
- package/dist/pi/ai/utils/overflow.js +150 -0
- package/dist/pi/ai/utils/sanitize-unicode.js +25 -0
- package/dist/pi/ai/utils/typebox-helpers.js +20 -0
- package/dist/pi/ai/utils/validation.js +280 -0
- package/dist/pi/coding-agent/bun/cli.js +7 -0
- package/dist/pi/coding-agent/bun/restore-sandbox-env.js +31 -0
- package/dist/pi/coding-agent/cli/args.js +340 -0
- package/dist/pi/coding-agent/cli/file-processor.js +82 -0
- package/dist/pi/coding-agent/cli/initial-message.js +21 -0
- package/dist/pi/coding-agent/cli.js +17 -0
- package/dist/pi/coding-agent/config.js +414 -0
- package/dist/pi/coding-agent/core/agent-session-runtime.js +299 -0
- package/dist/pi/coding-agent/core/agent-session-services.js +117 -0
- package/dist/pi/coding-agent/core/agent-session.js +2498 -0
- package/dist/pi/coding-agent/core/auth-guidance.js +20 -0
- package/dist/pi/coding-agent/core/auth-storage.js +441 -0
- package/dist/pi/coding-agent/core/bash-executor.js +110 -0
- package/dist/pi/coding-agent/core/compaction/branch-summarization.js +242 -0
- package/dist/pi/coding-agent/core/compaction/compaction.js +624 -0
- package/dist/pi/coding-agent/core/compaction/index.js +6 -0
- package/dist/pi/coding-agent/core/compaction/utils.js +152 -0
- package/dist/pi/coding-agent/core/defaults.js +1 -0
- package/dist/pi/coding-agent/core/diagnostics.js +1 -0
- package/dist/pi/coding-agent/core/event-bus.js +24 -0
- package/dist/pi/coding-agent/core/exec.js +74 -0
- package/dist/pi/coding-agent/core/export-html/ansi-to-html.js +248 -0
- package/dist/pi/coding-agent/core/export-html/index.js +225 -0
- package/dist/pi/coding-agent/core/export-html/tool-renderer.js +107 -0
- package/dist/pi/coding-agent/core/extensions/index.js +8 -0
- package/dist/pi/coding-agent/core/extensions/loader.js +485 -0
- package/dist/pi/coding-agent/core/extensions/runner.js +824 -0
- package/dist/pi/coding-agent/core/extensions/types.js +44 -0
- package/dist/pi/coding-agent/core/extensions/wrapper.js +21 -0
- package/dist/pi/coding-agent/core/footer-data-provider.js +309 -0
- package/dist/pi/coding-agent/core/http-dispatcher.js +47 -0
- package/dist/pi/coding-agent/core/index.js +11 -0
- package/dist/pi/coding-agent/core/keybindings.js +294 -0
- package/dist/pi/coding-agent/core/messages.js +122 -0
- package/dist/pi/coding-agent/core/model-registry.js +728 -0
- package/dist/pi/coding-agent/core/model-resolver.js +494 -0
- package/dist/pi/coding-agent/core/output-guard.js +58 -0
- package/dist/pi/coding-agent/core/package-manager.js +2020 -0
- package/dist/pi/coding-agent/core/prompt-templates.js +237 -0
- package/dist/pi/coding-agent/core/provider-display-names.js +32 -0
- package/dist/pi/coding-agent/core/resolve-config-value.js +125 -0
- package/dist/pi/coding-agent/core/resource-loader.js +733 -0
- package/dist/pi/coding-agent/core/sdk.js +282 -0
- package/dist/pi/coding-agent/core/session-cwd.js +37 -0
- package/dist/pi/coding-agent/core/session-manager.js +1146 -0
- package/dist/pi/coding-agent/core/settings-manager.js +794 -0
- package/dist/pi/coding-agent/core/skills.js +386 -0
- package/dist/pi/coding-agent/core/slash-commands.js +24 -0
- package/dist/pi/coding-agent/core/source-info.js +18 -0
- package/dist/pi/coding-agent/core/system-prompt.js +122 -0
- package/dist/pi/coding-agent/core/telemetry.js +8 -0
- package/dist/pi/coding-agent/core/timings.js +30 -0
- package/dist/pi/coding-agent/core/tools/bash.js +341 -0
- package/dist/pi/coding-agent/core/tools/edit-diff.js +344 -0
- package/dist/pi/coding-agent/core/tools/edit.js +324 -0
- package/dist/pi/coding-agent/core/tools/file-mutation-queue.js +36 -0
- package/dist/pi/coding-agent/core/tools/find.js +297 -0
- package/dist/pi/coding-agent/core/tools/grep.js +303 -0
- package/dist/pi/coding-agent/core/tools/index.js +111 -0
- package/dist/pi/coding-agent/core/tools/ls.js +168 -0
- package/dist/pi/coding-agent/core/tools/output-accumulator.js +183 -0
- package/dist/pi/coding-agent/core/tools/path-utils.js +61 -0
- package/dist/pi/coding-agent/core/tools/read.js +288 -0
- package/dist/pi/coding-agent/core/tools/render-utils.js +48 -0
- package/dist/pi/coding-agent/core/tools/tool-definition-wrapper.js +33 -0
- package/dist/pi/coding-agent/core/tools/truncate.js +214 -0
- package/dist/pi/coding-agent/core/tools/write.js +212 -0
- package/dist/pi/coding-agent/index.js +41 -0
- package/dist/pi/coding-agent/main.js +5 -0
- package/dist/pi/coding-agent/migrations.js +280 -0
- package/dist/pi/coding-agent/modes/index.js +7 -0
- package/dist/pi/coding-agent/modes/interactive/components/diff.js +132 -0
- package/dist/pi/coding-agent/modes/interactive/components/keybinding-hints.js +35 -0
- package/dist/pi/coding-agent/modes/interactive/components/visual-truncate.js +32 -0
- package/dist/pi/coding-agent/modes/interactive/interactive-mode.js +3 -0
- package/dist/pi/coding-agent/modes/interactive/theme/theme.js +1023 -0
- package/dist/pi/coding-agent/modes/print-mode.js +130 -0
- package/dist/pi/coding-agent/modes/rpc/jsonl.js +48 -0
- package/dist/pi/coding-agent/modes/rpc/rpc-client.js +409 -0
- package/dist/pi/coding-agent/modes/rpc/rpc-mode.js +600 -0
- package/dist/pi/coding-agent/modes/rpc/rpc-types.js +7 -0
- package/dist/pi/coding-agent/utils/ansi.js +51 -0
- package/dist/pi/coding-agent/utils/changelog.js +86 -0
- package/dist/pi/coding-agent/utils/child-process.js +87 -0
- package/dist/pi/coding-agent/utils/clipboard-image.js +244 -0
- package/dist/pi/coding-agent/utils/clipboard-native.js +13 -0
- package/dist/pi/coding-agent/utils/clipboard.js +116 -0
- package/dist/pi/coding-agent/utils/exif-orientation.js +157 -0
- package/dist/pi/coding-agent/utils/frontmatter.js +25 -0
- package/dist/pi/coding-agent/utils/fs-watch.js +24 -0
- package/dist/pi/coding-agent/utils/git.js +162 -0
- package/dist/pi/coding-agent/utils/html.js +39 -0
- package/dist/pi/coding-agent/utils/image-convert.js +38 -0
- package/dist/pi/coding-agent/utils/image-resize.js +136 -0
- package/dist/pi/coding-agent/utils/mime.js +68 -0
- package/dist/pi/coding-agent/utils/paths.js +91 -0
- package/dist/pi/coding-agent/utils/photon.js +120 -0
- package/dist/pi/coding-agent/utils/pi-user-agent.js +4 -0
- package/dist/pi/coding-agent/utils/shell.js +194 -0
- package/dist/pi/coding-agent/utils/sleep.js +16 -0
- package/dist/pi/coding-agent/utils/syntax-highlight.js +117 -0
- package/dist/pi/coding-agent/utils/tools-manager.js +327 -0
- package/dist/pi/coding-agent/utils/version-check.js +81 -0
- package/dist/pi/coding-agent/utils/windows-self-update.js +76 -0
- package/dist/pi/tui/autocomplete.js +631 -0
- package/dist/pi/tui/components/box.js +103 -0
- package/dist/pi/tui/components/cancellable-loader.js +34 -0
- package/dist/pi/tui/components/editor.js +1915 -0
- package/dist/pi/tui/components/image.js +88 -0
- package/dist/pi/tui/components/input.js +425 -0
- package/dist/pi/tui/components/loader.js +68 -0
- package/dist/pi/tui/components/markdown.js +633 -0
- package/dist/pi/tui/components/select-list.js +158 -0
- package/dist/pi/tui/components/settings-list.js +184 -0
- package/dist/pi/tui/components/spacer.js +22 -0
- package/dist/pi/tui/components/text.js +88 -0
- package/dist/pi/tui/components/truncated-text.js +50 -0
- package/dist/pi/tui/editor-component.js +1 -0
- package/dist/pi/tui/fuzzy.js +109 -0
- package/dist/pi/tui/index.js +31 -0
- package/dist/pi/tui/keybindings.js +173 -0
- package/dist/pi/tui/keys.js +1172 -0
- package/dist/pi/tui/kill-ring.js +43 -0
- package/dist/pi/tui/stdin-buffer.js +360 -0
- package/dist/pi/tui/terminal-image.js +335 -0
- package/dist/pi/tui/terminal.js +324 -0
- package/dist/pi/tui/tui.js +1076 -0
- package/dist/pi/tui/undo-stack.js +24 -0
- package/dist/pi/tui/utils.js +1016 -0
- package/dist/relay/dispatcher.js +30 -0
- package/dist/server/handlers/queue.js +52 -0
- package/dist/server/pi-bridge.js +9 -1
- package/dist/server/session-stream.js +76 -111
- package/dist/server/storage.js +154 -2
- package/dist/server/title-generator.js +14 -153
- package/package.json +24 -6
|
@@ -0,0 +1,310 @@
|
|
|
1
|
+
import ignore from "ignore";
|
|
2
|
+
import { parse } from "yaml";
|
|
3
|
+
import { toError } from "./types.js";
|
|
4
|
+
const MAX_NAME_LENGTH = 64;
|
|
5
|
+
const MAX_DESCRIPTION_LENGTH = 1024;
|
|
6
|
+
const IGNORE_FILE_NAMES = [".gitignore", ".ignore", ".fdignore"];
|
|
7
|
+
/** Format a skill invocation prompt, optionally appending additional user instructions. */
|
|
8
|
+
export function formatSkillInvocation(skill, additionalInstructions) {
|
|
9
|
+
const skillBlock = `<skill name="${skill.name}" location="${skill.filePath}">\nReferences are relative to ${dirnameEnvPath(skill.filePath)}.\n\n${skill.content}\n</skill>`;
|
|
10
|
+
return additionalInstructions ? `${skillBlock}\n\n${additionalInstructions}` : skillBlock;
|
|
11
|
+
}
|
|
12
|
+
/**
|
|
13
|
+
* Load skills from one or more directories.
|
|
14
|
+
*
|
|
15
|
+
* Traverses directories recursively, loads `SKILL.md` files, loads direct root `.md` files as skills, honors ignore files,
|
|
16
|
+
* and returns diagnostics for invalid skill files. Missing input directories are skipped.
|
|
17
|
+
*/
|
|
18
|
+
export async function loadSkills(env, dirs) {
|
|
19
|
+
const skills = [];
|
|
20
|
+
const diagnostics = [];
|
|
21
|
+
for (const dir of Array.isArray(dirs) ? dirs : [dirs]) {
|
|
22
|
+
const rootInfoResult = await env.fileInfo(dir);
|
|
23
|
+
if (!rootInfoResult.ok) {
|
|
24
|
+
if (rootInfoResult.error.code !== "not_found") {
|
|
25
|
+
diagnostics.push({
|
|
26
|
+
type: "warning",
|
|
27
|
+
code: "file_info_failed",
|
|
28
|
+
message: rootInfoResult.error.message,
|
|
29
|
+
path: dir,
|
|
30
|
+
});
|
|
31
|
+
}
|
|
32
|
+
continue;
|
|
33
|
+
}
|
|
34
|
+
const rootInfo = rootInfoResult.value;
|
|
35
|
+
if ((await resolveKind(env, rootInfo, diagnostics)) !== "directory")
|
|
36
|
+
continue;
|
|
37
|
+
const result = await loadSkillsFromDirInternal(env, rootInfo.path, true, ignore(), rootInfo.path);
|
|
38
|
+
skills.push(...result.skills);
|
|
39
|
+
diagnostics.push(...result.diagnostics);
|
|
40
|
+
}
|
|
41
|
+
return { skills, diagnostics };
|
|
42
|
+
}
|
|
43
|
+
/**
|
|
44
|
+
* Load skills from source-tagged directories.
|
|
45
|
+
*
|
|
46
|
+
* Source values are preserved exactly and attached to every loaded skill and diagnostic. The agent package does not
|
|
47
|
+
* interpret source values; applications define their own provenance shape.
|
|
48
|
+
*/
|
|
49
|
+
export async function loadSourcedSkills(env, inputs, mapSkill) {
|
|
50
|
+
const skills = [];
|
|
51
|
+
const diagnostics = [];
|
|
52
|
+
for (const input of inputs) {
|
|
53
|
+
const result = await loadSkills(env, input.path);
|
|
54
|
+
for (const skill of result.skills) {
|
|
55
|
+
skills.push({ skill: mapSkill ? mapSkill(skill, input.source) : skill, source: input.source });
|
|
56
|
+
}
|
|
57
|
+
for (const diagnostic of result.diagnostics)
|
|
58
|
+
diagnostics.push({ ...diagnostic, source: input.source });
|
|
59
|
+
}
|
|
60
|
+
return { skills, diagnostics };
|
|
61
|
+
}
|
|
62
|
+
async function loadSkillsFromDirInternal(env, dir, includeRootFiles, ignoreMatcher, rootDir) {
|
|
63
|
+
const skills = [];
|
|
64
|
+
const diagnostics = [];
|
|
65
|
+
const dirInfoResult = await env.fileInfo(dir);
|
|
66
|
+
if (!dirInfoResult.ok) {
|
|
67
|
+
if (dirInfoResult.error.code !== "not_found") {
|
|
68
|
+
diagnostics.push({
|
|
69
|
+
type: "warning",
|
|
70
|
+
code: "file_info_failed",
|
|
71
|
+
message: dirInfoResult.error.message,
|
|
72
|
+
path: dir,
|
|
73
|
+
});
|
|
74
|
+
}
|
|
75
|
+
return { skills, diagnostics };
|
|
76
|
+
}
|
|
77
|
+
const dirInfo = dirInfoResult.value;
|
|
78
|
+
if ((await resolveKind(env, dirInfo, diagnostics)) !== "directory")
|
|
79
|
+
return { skills, diagnostics };
|
|
80
|
+
await addIgnoreRules(env, ignoreMatcher, dir, rootDir, diagnostics);
|
|
81
|
+
const entriesResult = await env.listDir(dir);
|
|
82
|
+
if (!entriesResult.ok) {
|
|
83
|
+
diagnostics.push({ type: "warning", code: "list_failed", message: entriesResult.error.message, path: dir });
|
|
84
|
+
return { skills, diagnostics };
|
|
85
|
+
}
|
|
86
|
+
const entries = entriesResult.value;
|
|
87
|
+
for (const entry of entries) {
|
|
88
|
+
if (entry.name !== "SKILL.md")
|
|
89
|
+
continue;
|
|
90
|
+
const fullPath = entry.path;
|
|
91
|
+
const kind = await resolveKind(env, entry, diagnostics);
|
|
92
|
+
if (kind !== "file")
|
|
93
|
+
continue;
|
|
94
|
+
const relPath = relativeEnvPath(rootDir, fullPath);
|
|
95
|
+
if (ignoreMatcher.ignores(relPath))
|
|
96
|
+
continue;
|
|
97
|
+
const result = await loadSkillFromFile(env, fullPath);
|
|
98
|
+
if (result.skill)
|
|
99
|
+
skills.push(result.skill);
|
|
100
|
+
diagnostics.push(...result.diagnostics);
|
|
101
|
+
return { skills, diagnostics };
|
|
102
|
+
}
|
|
103
|
+
for (const entry of entries.sort((a, b) => a.name.localeCompare(b.name))) {
|
|
104
|
+
if (entry.name.startsWith(".") || entry.name === "node_modules")
|
|
105
|
+
continue;
|
|
106
|
+
const fullPath = entry.path;
|
|
107
|
+
const kind = await resolveKind(env, entry, diagnostics);
|
|
108
|
+
if (!kind)
|
|
109
|
+
continue;
|
|
110
|
+
const relPath = relativeEnvPath(rootDir, fullPath);
|
|
111
|
+
const ignorePath = kind === "directory" ? `${relPath}/` : relPath;
|
|
112
|
+
if (ignoreMatcher.ignores(ignorePath))
|
|
113
|
+
continue;
|
|
114
|
+
if (kind === "directory") {
|
|
115
|
+
const result = await loadSkillsFromDirInternal(env, fullPath, false, ignoreMatcher, rootDir);
|
|
116
|
+
skills.push(...result.skills);
|
|
117
|
+
diagnostics.push(...result.diagnostics);
|
|
118
|
+
continue;
|
|
119
|
+
}
|
|
120
|
+
if (kind !== "file" || !includeRootFiles || !entry.name.endsWith(".md"))
|
|
121
|
+
continue;
|
|
122
|
+
const result = await loadSkillFromFile(env, fullPath);
|
|
123
|
+
if (result.skill)
|
|
124
|
+
skills.push(result.skill);
|
|
125
|
+
diagnostics.push(...result.diagnostics);
|
|
126
|
+
}
|
|
127
|
+
return { skills, diagnostics };
|
|
128
|
+
}
|
|
129
|
+
async function addIgnoreRules(env, ig, dir, rootDir, diagnostics) {
|
|
130
|
+
const relativeDir = relativeEnvPath(rootDir, dir);
|
|
131
|
+
const prefix = relativeDir ? `${relativeDir}/` : "";
|
|
132
|
+
for (const filename of IGNORE_FILE_NAMES) {
|
|
133
|
+
const ignorePath = joinEnvPath(dir, filename);
|
|
134
|
+
const info = await env.fileInfo(ignorePath);
|
|
135
|
+
if (!info.ok) {
|
|
136
|
+
if (info.error.code !== "not_found") {
|
|
137
|
+
diagnostics.push({
|
|
138
|
+
type: "warning",
|
|
139
|
+
code: "file_info_failed",
|
|
140
|
+
message: info.error.message,
|
|
141
|
+
path: ignorePath,
|
|
142
|
+
});
|
|
143
|
+
}
|
|
144
|
+
continue;
|
|
145
|
+
}
|
|
146
|
+
if (info.value.kind !== "file")
|
|
147
|
+
continue;
|
|
148
|
+
const content = await env.readTextFile(ignorePath);
|
|
149
|
+
if (!content.ok) {
|
|
150
|
+
diagnostics.push({ type: "warning", code: "read_failed", message: content.error.message, path: ignorePath });
|
|
151
|
+
continue;
|
|
152
|
+
}
|
|
153
|
+
const patterns = content.value
|
|
154
|
+
.split(/\r?\n/)
|
|
155
|
+
.map((line) => prefixIgnorePattern(line, prefix))
|
|
156
|
+
.filter((line) => Boolean(line));
|
|
157
|
+
if (patterns.length > 0)
|
|
158
|
+
ig.add(patterns);
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
function prefixIgnorePattern(line, prefix) {
|
|
162
|
+
const trimmed = line.trim();
|
|
163
|
+
if (!trimmed)
|
|
164
|
+
return null;
|
|
165
|
+
if (trimmed.startsWith("#") && !trimmed.startsWith("\\#"))
|
|
166
|
+
return null;
|
|
167
|
+
let pattern = line;
|
|
168
|
+
let negated = false;
|
|
169
|
+
if (pattern.startsWith("!")) {
|
|
170
|
+
negated = true;
|
|
171
|
+
pattern = pattern.slice(1);
|
|
172
|
+
}
|
|
173
|
+
else if (pattern.startsWith("\\!")) {
|
|
174
|
+
pattern = pattern.slice(1);
|
|
175
|
+
}
|
|
176
|
+
if (pattern.startsWith("/"))
|
|
177
|
+
pattern = pattern.slice(1);
|
|
178
|
+
const prefixed = prefix ? `${prefix}${pattern}` : pattern;
|
|
179
|
+
return negated ? `!${prefixed}` : prefixed;
|
|
180
|
+
}
|
|
181
|
+
async function loadSkillFromFile(env, filePath) {
|
|
182
|
+
const diagnostics = [];
|
|
183
|
+
const rawContent = await env.readTextFile(filePath);
|
|
184
|
+
if (!rawContent.ok) {
|
|
185
|
+
diagnostics.push({ type: "warning", code: "read_failed", message: rawContent.error.message, path: filePath });
|
|
186
|
+
return { skill: null, diagnostics };
|
|
187
|
+
}
|
|
188
|
+
const parsed = parseFrontmatter(rawContent.value);
|
|
189
|
+
if (!parsed.ok) {
|
|
190
|
+
diagnostics.push({ type: "warning", code: "parse_failed", message: parsed.error.message, path: filePath });
|
|
191
|
+
return { skill: null, diagnostics };
|
|
192
|
+
}
|
|
193
|
+
const { frontmatter, body } = parsed.value;
|
|
194
|
+
const skillDir = dirnameEnvPath(filePath);
|
|
195
|
+
const parentDirName = basenameEnvPath(skillDir);
|
|
196
|
+
const description = typeof frontmatter.description === "string" ? frontmatter.description : undefined;
|
|
197
|
+
for (const error of validateDescription(description)) {
|
|
198
|
+
diagnostics.push({ type: "warning", code: "invalid_metadata", message: error, path: filePath });
|
|
199
|
+
}
|
|
200
|
+
const frontmatterName = typeof frontmatter.name === "string" ? frontmatter.name : undefined;
|
|
201
|
+
const name = frontmatterName || parentDirName;
|
|
202
|
+
for (const error of validateName(name, parentDirName)) {
|
|
203
|
+
diagnostics.push({ type: "warning", code: "invalid_metadata", message: error, path: filePath });
|
|
204
|
+
}
|
|
205
|
+
if (!description || description.trim() === "") {
|
|
206
|
+
return { skill: null, diagnostics };
|
|
207
|
+
}
|
|
208
|
+
return {
|
|
209
|
+
skill: {
|
|
210
|
+
name,
|
|
211
|
+
description,
|
|
212
|
+
content: body,
|
|
213
|
+
filePath,
|
|
214
|
+
disableModelInvocation: frontmatter["disable-model-invocation"] === true,
|
|
215
|
+
},
|
|
216
|
+
diagnostics,
|
|
217
|
+
};
|
|
218
|
+
}
|
|
219
|
+
function validateName(name, parentDirName) {
|
|
220
|
+
const errors = [];
|
|
221
|
+
if (name !== parentDirName)
|
|
222
|
+
errors.push(`name "${name}" does not match parent directory "${parentDirName}"`);
|
|
223
|
+
if (name.length > MAX_NAME_LENGTH)
|
|
224
|
+
errors.push(`name exceeds ${MAX_NAME_LENGTH} characters (${name.length})`);
|
|
225
|
+
if (!/^[a-z0-9-]+$/.test(name)) {
|
|
226
|
+
errors.push("name contains invalid characters (must be lowercase a-z, 0-9, hyphens only)");
|
|
227
|
+
}
|
|
228
|
+
if (name.startsWith("-") || name.endsWith("-"))
|
|
229
|
+
errors.push("name must not start or end with a hyphen");
|
|
230
|
+
if (name.includes("--"))
|
|
231
|
+
errors.push("name must not contain consecutive hyphens");
|
|
232
|
+
return errors;
|
|
233
|
+
}
|
|
234
|
+
function validateDescription(description) {
|
|
235
|
+
const errors = [];
|
|
236
|
+
if (!description || description.trim() === "") {
|
|
237
|
+
errors.push("description is required");
|
|
238
|
+
}
|
|
239
|
+
else if (description.length > MAX_DESCRIPTION_LENGTH) {
|
|
240
|
+
errors.push(`description exceeds ${MAX_DESCRIPTION_LENGTH} characters (${description.length})`);
|
|
241
|
+
}
|
|
242
|
+
return errors;
|
|
243
|
+
}
|
|
244
|
+
function parseFrontmatter(content) {
|
|
245
|
+
try {
|
|
246
|
+
const normalized = content.replace(/\r\n/g, "\n").replace(/\r/g, "\n");
|
|
247
|
+
if (!normalized.startsWith("---"))
|
|
248
|
+
return { ok: true, value: { frontmatter: {}, body: normalized } };
|
|
249
|
+
const endIndex = normalized.indexOf("\n---", 3);
|
|
250
|
+
if (endIndex === -1)
|
|
251
|
+
return { ok: true, value: { frontmatter: {}, body: normalized } };
|
|
252
|
+
const yamlString = normalized.slice(4, endIndex);
|
|
253
|
+
const body = normalized.slice(endIndex + 4).trim();
|
|
254
|
+
return { ok: true, value: { frontmatter: (parse(yamlString) ?? {}), body } };
|
|
255
|
+
}
|
|
256
|
+
catch (error) {
|
|
257
|
+
return { ok: false, error: toError(error) };
|
|
258
|
+
}
|
|
259
|
+
}
|
|
260
|
+
async function resolveKind(env, info, diagnostics) {
|
|
261
|
+
if (info.kind === "file" || info.kind === "directory")
|
|
262
|
+
return info.kind;
|
|
263
|
+
const canonicalPath = await env.canonicalPath(info.path);
|
|
264
|
+
if (!canonicalPath.ok) {
|
|
265
|
+
if (canonicalPath.error.code !== "not_found") {
|
|
266
|
+
diagnostics.push({
|
|
267
|
+
type: "warning",
|
|
268
|
+
code: "file_info_failed",
|
|
269
|
+
message: canonicalPath.error.message,
|
|
270
|
+
path: info.path,
|
|
271
|
+
});
|
|
272
|
+
}
|
|
273
|
+
return undefined;
|
|
274
|
+
}
|
|
275
|
+
const target = await env.fileInfo(canonicalPath.value);
|
|
276
|
+
if (!target.ok) {
|
|
277
|
+
if (target.error.code !== "not_found") {
|
|
278
|
+
diagnostics.push({
|
|
279
|
+
type: "warning",
|
|
280
|
+
code: "file_info_failed",
|
|
281
|
+
message: target.error.message,
|
|
282
|
+
path: info.path,
|
|
283
|
+
});
|
|
284
|
+
}
|
|
285
|
+
return undefined;
|
|
286
|
+
}
|
|
287
|
+
return target.value.kind === "file" || target.value.kind === "directory" ? target.value.kind : undefined;
|
|
288
|
+
}
|
|
289
|
+
function joinEnvPath(base, child) {
|
|
290
|
+
return `${base.replace(/\/+$/, "")}/${child.replace(/^\/+/, "")}`;
|
|
291
|
+
}
|
|
292
|
+
function dirnameEnvPath(path) {
|
|
293
|
+
const normalized = path.replace(/\/+$/, "");
|
|
294
|
+
const slashIndex = normalized.lastIndexOf("/");
|
|
295
|
+
return slashIndex <= 0 ? "/" : normalized.slice(0, slashIndex);
|
|
296
|
+
}
|
|
297
|
+
function basenameEnvPath(path) {
|
|
298
|
+
const normalized = path.replace(/\/+$/, "");
|
|
299
|
+
const slashIndex = normalized.lastIndexOf("/");
|
|
300
|
+
return slashIndex === -1 ? normalized : normalized.slice(slashIndex + 1);
|
|
301
|
+
}
|
|
302
|
+
function relativeEnvPath(root, path) {
|
|
303
|
+
const normalizedRoot = root.replace(/\/+$/, "");
|
|
304
|
+
const normalizedPath = path.replace(/\/+$/, "");
|
|
305
|
+
if (normalizedPath === normalizedRoot)
|
|
306
|
+
return "";
|
|
307
|
+
return normalizedPath.startsWith(`${normalizedRoot}/`)
|
|
308
|
+
? normalizedPath.slice(normalizedRoot.length + 1)
|
|
309
|
+
: normalizedPath.replace(/^\/+/, "");
|
|
310
|
+
}
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
export function formatSkillsForSystemPrompt(skills) {
|
|
2
|
+
const visibleSkills = skills.filter((skill) => !skill.disableModelInvocation);
|
|
3
|
+
if (visibleSkills.length === 0)
|
|
4
|
+
return "";
|
|
5
|
+
const lines = [
|
|
6
|
+
"The following skills provide specialized instructions for specific tasks.",
|
|
7
|
+
"Read the full skill file when the task matches its description.",
|
|
8
|
+
"When a skill file references a relative path, resolve it against the skill directory (parent of SKILL.md / dirname of the path) and use that absolute path in tool commands.",
|
|
9
|
+
"",
|
|
10
|
+
"<available_skills>",
|
|
11
|
+
];
|
|
12
|
+
for (const skill of visibleSkills) {
|
|
13
|
+
lines.push(" <skill>");
|
|
14
|
+
lines.push(` <name>${escapeXml(skill.name)}</name>`);
|
|
15
|
+
lines.push(` <description>${escapeXml(skill.description)}</description>`);
|
|
16
|
+
lines.push(` <location>${escapeXml(skill.filePath)}</location>`);
|
|
17
|
+
lines.push(" </skill>");
|
|
18
|
+
}
|
|
19
|
+
lines.push("</available_skills>");
|
|
20
|
+
return lines.join("\n");
|
|
21
|
+
}
|
|
22
|
+
function escapeXml(value) {
|
|
23
|
+
return value
|
|
24
|
+
.replace(/&/g, "&")
|
|
25
|
+
.replace(/</g, "<")
|
|
26
|
+
.replace(/>/g, ">")
|
|
27
|
+
.replace(/"/g, """)
|
|
28
|
+
.replace(/'/g, "'");
|
|
29
|
+
}
|
|
@@ -0,0 +1,93 @@
|
|
|
1
|
+
/** Create a successful {@link Result}. */
|
|
2
|
+
export function ok(value) {
|
|
3
|
+
return { ok: true, value };
|
|
4
|
+
}
|
|
5
|
+
/** Create a failed {@link Result}. */
|
|
6
|
+
export function err(error) {
|
|
7
|
+
return { ok: false, error };
|
|
8
|
+
}
|
|
9
|
+
/** Return the success value or throw the failure error. Intended for tests and explicit adapter boundaries. */
|
|
10
|
+
export function getOrThrow(result) {
|
|
11
|
+
if (!result.ok)
|
|
12
|
+
throw result.error;
|
|
13
|
+
return result.value;
|
|
14
|
+
}
|
|
15
|
+
/** Return the success value or `undefined`. Only object values are allowed to avoid truthiness bugs with primitives. */
|
|
16
|
+
export function getOrUndefined(result) {
|
|
17
|
+
return result.ok ? result.value : undefined;
|
|
18
|
+
}
|
|
19
|
+
/** Normalize unknown thrown values into Error instances before using them as typed error causes. */
|
|
20
|
+
export function toError(error) {
|
|
21
|
+
if (error instanceof Error)
|
|
22
|
+
return error;
|
|
23
|
+
if (typeof error === "string")
|
|
24
|
+
return new Error(error);
|
|
25
|
+
try {
|
|
26
|
+
return new Error(JSON.stringify(error));
|
|
27
|
+
}
|
|
28
|
+
catch {
|
|
29
|
+
return new Error(String(error));
|
|
30
|
+
}
|
|
31
|
+
}
|
|
32
|
+
/** Error returned by {@link FileSystem} file operations. */
|
|
33
|
+
export class FileError extends Error {
|
|
34
|
+
/** Backend-independent error code. */
|
|
35
|
+
code;
|
|
36
|
+
/** Absolute addressed path associated with the failure, when available. */
|
|
37
|
+
path;
|
|
38
|
+
constructor(code, message, path, cause) {
|
|
39
|
+
super(message, cause === undefined ? undefined : { cause });
|
|
40
|
+
this.name = "FileError";
|
|
41
|
+
this.code = code;
|
|
42
|
+
this.path = path;
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
/** Error returned by {@link ExecutionEnv.exec}. */
|
|
46
|
+
export class ExecutionError extends Error {
|
|
47
|
+
/** Backend-independent error code. */
|
|
48
|
+
code;
|
|
49
|
+
constructor(code, message, cause) {
|
|
50
|
+
super(message, cause === undefined ? undefined : { cause });
|
|
51
|
+
this.name = "ExecutionError";
|
|
52
|
+
this.code = code;
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
/** Error returned by compaction helpers. */
|
|
56
|
+
export class CompactionError extends Error {
|
|
57
|
+
/** Backend-independent error code. */
|
|
58
|
+
code;
|
|
59
|
+
constructor(code, message, cause) {
|
|
60
|
+
super(message, cause === undefined ? undefined : { cause });
|
|
61
|
+
this.name = "CompactionError";
|
|
62
|
+
this.code = code;
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
/** Error returned by branch summarization helpers. */
|
|
66
|
+
export class BranchSummaryError extends Error {
|
|
67
|
+
/** Backend-independent error code. */
|
|
68
|
+
code;
|
|
69
|
+
constructor(code, message, cause) {
|
|
70
|
+
super(message, cause === undefined ? undefined : { cause });
|
|
71
|
+
this.name = "BranchSummaryError";
|
|
72
|
+
this.code = code;
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
/** Error thrown by session storage, repositories, and session tree operations. */
|
|
76
|
+
export class SessionError extends Error {
|
|
77
|
+
/** Session subsystem error code. */
|
|
78
|
+
code;
|
|
79
|
+
constructor(code, message, cause) {
|
|
80
|
+
super(message, cause === undefined ? undefined : { cause });
|
|
81
|
+
this.name = "SessionError";
|
|
82
|
+
this.code = code;
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
/** Public AgentHarness failure with a stable top-level classification. */
|
|
86
|
+
export class AgentHarnessError extends Error {
|
|
87
|
+
code;
|
|
88
|
+
constructor(code, message, cause) {
|
|
89
|
+
super(message, cause === undefined ? undefined : { cause });
|
|
90
|
+
this.name = "AgentHarnessError";
|
|
91
|
+
this.code = code;
|
|
92
|
+
}
|
|
93
|
+
}
|
|
@@ -0,0 +1,125 @@
|
|
|
1
|
+
import { ExecutionError, err, ok, toError, } from "../types.js";
|
|
2
|
+
import { DEFAULT_MAX_BYTES, truncateTail } from "./truncate.js";
|
|
3
|
+
function toExecutionError(error) {
|
|
4
|
+
if (error instanceof ExecutionError)
|
|
5
|
+
return error;
|
|
6
|
+
const cause = toError(error);
|
|
7
|
+
return new ExecutionError("unknown", cause.message, cause);
|
|
8
|
+
}
|
|
9
|
+
export function sanitizeBinaryOutput(str) {
|
|
10
|
+
return Array.from(str)
|
|
11
|
+
.filter((char) => {
|
|
12
|
+
const code = char.codePointAt(0);
|
|
13
|
+
if (code === undefined)
|
|
14
|
+
return false;
|
|
15
|
+
if (code === 0x09 || code === 0x0a || code === 0x0d)
|
|
16
|
+
return true;
|
|
17
|
+
if (code <= 0x1f)
|
|
18
|
+
return false;
|
|
19
|
+
if (code >= 0xfff9 && code <= 0xfffb)
|
|
20
|
+
return false;
|
|
21
|
+
return true;
|
|
22
|
+
})
|
|
23
|
+
.join("");
|
|
24
|
+
}
|
|
25
|
+
export async function executeShellWithCapture(env, command, options) {
|
|
26
|
+
const outputChunks = [];
|
|
27
|
+
let outputBytes = 0;
|
|
28
|
+
const maxOutputBytes = DEFAULT_MAX_BYTES * 2;
|
|
29
|
+
const encoder = new TextEncoder();
|
|
30
|
+
let totalBytes = 0;
|
|
31
|
+
let fullOutputPath;
|
|
32
|
+
let writeChain = Promise.resolve(ok(undefined));
|
|
33
|
+
let captureError;
|
|
34
|
+
const appendFullOutput = (text) => {
|
|
35
|
+
if (!fullOutputPath || captureError)
|
|
36
|
+
return;
|
|
37
|
+
const path = fullOutputPath;
|
|
38
|
+
writeChain = writeChain.then(async (previous) => {
|
|
39
|
+
if (!previous.ok)
|
|
40
|
+
return previous;
|
|
41
|
+
const appendResult = await env.appendFile(path, text, options?.abortSignal);
|
|
42
|
+
return appendResult.ok ? ok(undefined) : err(toExecutionError(appendResult.error));
|
|
43
|
+
});
|
|
44
|
+
};
|
|
45
|
+
const ensureFullOutputFile = (initialContent) => {
|
|
46
|
+
if (fullOutputPath || captureError)
|
|
47
|
+
return;
|
|
48
|
+
writeChain = writeChain.then(async (previous) => {
|
|
49
|
+
if (!previous.ok)
|
|
50
|
+
return previous;
|
|
51
|
+
const tempFile = await env.createTempFile({
|
|
52
|
+
prefix: "bash-",
|
|
53
|
+
suffix: ".log",
|
|
54
|
+
abortSignal: options?.abortSignal,
|
|
55
|
+
});
|
|
56
|
+
if (!tempFile.ok)
|
|
57
|
+
return err(toExecutionError(tempFile.error));
|
|
58
|
+
fullOutputPath = tempFile.value;
|
|
59
|
+
const appendResult = await env.appendFile(tempFile.value, initialContent, options?.abortSignal);
|
|
60
|
+
return appendResult.ok ? ok(undefined) : err(toExecutionError(appendResult.error));
|
|
61
|
+
});
|
|
62
|
+
};
|
|
63
|
+
const onChunk = (chunk) => {
|
|
64
|
+
try {
|
|
65
|
+
totalBytes += encoder.encode(chunk).byteLength;
|
|
66
|
+
const text = sanitizeBinaryOutput(chunk).replace(/\r/g, "");
|
|
67
|
+
if (totalBytes > DEFAULT_MAX_BYTES && !fullOutputPath) {
|
|
68
|
+
ensureFullOutputFile(outputChunks.join("") + text);
|
|
69
|
+
}
|
|
70
|
+
else {
|
|
71
|
+
appendFullOutput(text);
|
|
72
|
+
}
|
|
73
|
+
outputChunks.push(text);
|
|
74
|
+
outputBytes += text.length;
|
|
75
|
+
while (outputBytes > maxOutputBytes && outputChunks.length > 1) {
|
|
76
|
+
const removed = outputChunks.shift();
|
|
77
|
+
outputBytes -= removed.length;
|
|
78
|
+
}
|
|
79
|
+
options?.onChunk?.(text);
|
|
80
|
+
}
|
|
81
|
+
catch (error) {
|
|
82
|
+
captureError = toExecutionError(error);
|
|
83
|
+
}
|
|
84
|
+
};
|
|
85
|
+
try {
|
|
86
|
+
const result = await env.exec(command, {
|
|
87
|
+
...(options ?? {}),
|
|
88
|
+
onStdout: onChunk,
|
|
89
|
+
onStderr: onChunk,
|
|
90
|
+
});
|
|
91
|
+
const tailOutput = outputChunks.join("");
|
|
92
|
+
const truncationResult = truncateTail(tailOutput);
|
|
93
|
+
if (truncationResult.truncated && !fullOutputPath) {
|
|
94
|
+
ensureFullOutputFile(tailOutput);
|
|
95
|
+
}
|
|
96
|
+
const writeResult = await writeChain;
|
|
97
|
+
if (!writeResult.ok)
|
|
98
|
+
return err(writeResult.error);
|
|
99
|
+
if (captureError)
|
|
100
|
+
return err(captureError);
|
|
101
|
+
if (!result.ok) {
|
|
102
|
+
if (result.error.code === "aborted" || options?.abortSignal?.aborted) {
|
|
103
|
+
return ok({
|
|
104
|
+
output: truncationResult.truncated ? truncationResult.content : tailOutput,
|
|
105
|
+
exitCode: undefined,
|
|
106
|
+
cancelled: true,
|
|
107
|
+
truncated: truncationResult.truncated,
|
|
108
|
+
fullOutputPath,
|
|
109
|
+
});
|
|
110
|
+
}
|
|
111
|
+
return err(result.error);
|
|
112
|
+
}
|
|
113
|
+
const cancelled = options?.abortSignal?.aborted ?? false;
|
|
114
|
+
return ok({
|
|
115
|
+
output: truncationResult.truncated ? truncationResult.content : tailOutput,
|
|
116
|
+
exitCode: cancelled ? undefined : result.value.exitCode,
|
|
117
|
+
cancelled,
|
|
118
|
+
truncated: truncationResult.truncated,
|
|
119
|
+
fullOutputPath,
|
|
120
|
+
});
|
|
121
|
+
}
|
|
122
|
+
catch (error) {
|
|
123
|
+
return err(toExecutionError(error));
|
|
124
|
+
}
|
|
125
|
+
}
|