@pugi/cli 0.1.0-beta.4 → 0.1.0-beta.41
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/THIRD_PARTY_NOTICES.md +40 -0
- package/assets/pugi-mascot.ansi +15 -25
- package/bin/run.js +33 -1
- package/dist/commands/jobs-watch.js +201 -0
- package/dist/commands/jobs.js +15 -0
- package/dist/commands/smoke.js +133 -0
- package/dist/core/agent-progress/cleanup.js +134 -0
- package/dist/core/agent-progress/schema.js +144 -0
- package/dist/core/agent-progress/writer.js +101 -0
- package/dist/core/artifact-chain/dispatcher.js +148 -0
- package/dist/core/artifact-chain/exporter.js +164 -0
- package/dist/core/artifact-chain/state.js +243 -0
- package/dist/core/artifact-chain/steps.js +169 -0
- package/dist/core/auth/ensure-authenticated.js +129 -0
- package/dist/core/auth/env-provider.js +238 -0
- package/dist/core/auto-update/channels.js +122 -0
- package/dist/core/auto-update/checker.js +241 -0
- package/dist/core/auto-update/state.js +235 -0
- package/dist/core/bare-mode/index.js +107 -0
- package/dist/core/bash-classifier.js +108 -1
- package/dist/core/checkpoint/resumer.js +149 -0
- package/dist/core/checkpoint/rewinder.js +291 -0
- package/dist/core/codegraph/decision-store.js +248 -0
- package/dist/core/codegraph/detect-repo.js +459 -0
- package/dist/core/codegraph/install.js +134 -0
- package/dist/core/codegraph/offer-hook.js +220 -0
- package/dist/core/compact/auto-trigger.js +96 -0
- package/dist/core/compact/buffer-rewriter.js +115 -0
- package/dist/core/compact/summarizer.js +208 -0
- package/dist/core/compact/token-counter.js +108 -0
- package/dist/core/consensus/diff-capture.js +73 -0
- package/dist/core/context/index.js +7 -0
- package/dist/core/context/markdown-traverse.js +255 -0
- package/dist/core/cost/rate-card.js +129 -0
- package/dist/core/cost/tracker.js +221 -0
- package/dist/core/denial-tracking/index.js +8 -0
- package/dist/core/denial-tracking/state.js +264 -0
- package/dist/core/diagnostics/probe-runner.js +93 -0
- package/dist/core/diagnostics/probes/api.js +46 -0
- package/dist/core/diagnostics/probes/auth.js +86 -0
- package/dist/core/diagnostics/probes/bare-mode.js +42 -0
- package/dist/core/diagnostics/probes/cli-version.js +127 -0
- package/dist/core/diagnostics/probes/config.js +72 -0
- package/dist/core/diagnostics/probes/denial-tracking.js +57 -0
- package/dist/core/diagnostics/probes/disk.js +81 -0
- package/dist/core/diagnostics/probes/git.js +65 -0
- package/dist/core/diagnostics/probes/mcp.js +75 -0
- package/dist/core/diagnostics/probes/node.js +59 -0
- package/dist/core/diagnostics/probes/pnpm.js +36 -0
- package/dist/core/diagnostics/probes/pugi-md.js +89 -0
- package/dist/core/diagnostics/probes/session.js +74 -0
- package/dist/core/diagnostics/probes/status-snapshot.js +488 -0
- package/dist/core/diagnostics/probes/workspace.js +63 -0
- package/dist/core/diagnostics/types.js +70 -0
- package/dist/core/dispatch/cache-cleanup.js +197 -0
- package/dist/core/dispatch/cache-handoff.js +295 -0
- package/dist/core/edits/dispatch.js +218 -2
- package/dist/core/edits/journal.js +199 -0
- package/dist/core/edits/layer-d-ast.js +557 -14
- package/dist/core/edits/verify-hook.js +273 -0
- package/dist/core/edits/worktree.js +322 -0
- package/dist/core/engine/anvil-client.js +115 -5
- package/dist/core/engine/budgets.js +98 -0
- package/dist/core/engine/context-prefix.js +155 -0
- package/dist/core/engine/intent.js +260 -0
- package/dist/core/engine/native-pugi.js +860 -211
- package/dist/core/engine/prompts.js +88 -2
- package/dist/core/engine/strip-internal-fields.js +124 -0
- package/dist/core/engine/tool-bridge.js +1045 -36
- package/dist/core/feedback/queue.js +177 -0
- package/dist/core/feedback/submitter.js +145 -0
- package/dist/core/file-cache.js +113 -1
- package/dist/core/hooks/events.js +44 -0
- package/dist/core/hooks/index.js +15 -0
- package/dist/core/hooks/registry.js +213 -0
- package/dist/core/hooks/runner.js +236 -0
- package/dist/core/hooks/v2/event-emitter.js +115 -0
- package/dist/core/hooks/v2/executor.js +282 -0
- package/dist/core/hooks/v2/index.js +25 -0
- package/dist/core/hooks/v2/lifecycle.js +104 -0
- package/dist/core/hooks/v2/loader.js +216 -0
- package/dist/core/hooks/v2/matcher.js +125 -0
- package/dist/core/hooks/v2/trust.js +143 -0
- package/dist/core/hooks/v2/types.js +86 -0
- package/dist/core/lsp/cache.js +105 -0
- package/dist/core/lsp/client.js +776 -0
- package/dist/core/lsp/language-detect.js +66 -0
- package/dist/core/lsp/post-edit-diagnostics.js +171 -0
- package/dist/core/mcp/client.js +75 -6
- package/dist/core/mcp/http-server.js +553 -0
- package/dist/core/mcp/orchestrator-tools.js +662 -0
- package/dist/core/mcp/permission.js +190 -0
- package/dist/core/mcp/registry.js +24 -2
- package/dist/core/mcp/server-tools.js +219 -0
- package/dist/core/mcp/server.js +397 -0
- package/dist/core/memory/dual-write.js +416 -0
- package/dist/core/memory/phase1-kinds.js +20 -0
- package/dist/core/memory-sync/queue.js +158 -0
- package/dist/core/onboarding/ensure-initialized.js +133 -0
- package/dist/core/onboarding/marker.js +111 -0
- package/dist/core/onboarding/telemetry-state.js +108 -0
- package/dist/core/output-style/presets.js +176 -0
- package/dist/core/output-style/state.js +185 -0
- package/dist/core/permissions/auto-classifier.js +124 -0
- package/dist/core/permissions/circuit-breaker.js +83 -0
- package/dist/core/permissions/gate.js +278 -0
- package/dist/core/permissions/index.js +20 -0
- package/dist/core/permissions/mode.js +174 -0
- package/dist/core/permissions/state.js +241 -0
- package/dist/core/permissions/tool-class.js +93 -0
- package/dist/core/prd-check/parser.js +215 -0
- package/dist/core/prd-check/reporter.js +127 -0
- package/dist/core/prd-check/session-review.js +557 -0
- package/dist/core/prd-check/verifiers.js +223 -0
- package/dist/core/pugi-md/context-injector.js +76 -0
- package/dist/core/pugi-md/walk-up.js +207 -0
- package/dist/core/release-notes/parser.js +241 -0
- package/dist/core/release-notes/state.js +116 -0
- package/dist/core/repl/history.js +11 -1
- package/dist/core/repl/model-pricing.js +135 -0
- package/dist/core/repl/session.js +1899 -38
- package/dist/core/repl/slash-commands.js +406 -21
- package/dist/core/repl/store/session-store.js +31 -2
- package/dist/core/repl/workspace-context.js +22 -0
- package/dist/core/repo-map/build.js +125 -0
- package/dist/core/repo-map/cache.js +185 -0
- package/dist/core/repo-map/extractor.js +254 -0
- package/dist/core/repo-map/formatter.js +145 -0
- package/dist/core/repo-map/scanner.js +211 -0
- package/dist/core/retry-budget/budget.js +284 -0
- package/dist/core/retry-budget/index.js +5 -0
- package/dist/core/session.js +92 -0
- package/dist/core/settings.js +80 -0
- package/dist/core/share/formatter.js +271 -0
- package/dist/core/share/redactor.js +221 -0
- package/dist/core/share/uploader.js +267 -0
- package/dist/core/skills/defaults.js +457 -0
- package/dist/core/smoke/headless-driver.js +174 -0
- package/dist/core/smoke/orchestrator.js +194 -0
- package/dist/core/smoke/runner.js +238 -0
- package/dist/core/smoke/scenario-parser.js +316 -0
- package/dist/core/subagents/dispatcher-real.js +600 -0
- package/dist/core/subagents/dispatcher.js +113 -24
- package/dist/core/subagents/index.js +18 -5
- package/dist/core/subagents/isolation-matrix.js +213 -0
- package/dist/core/subagents/spawn.js +19 -4
- package/dist/core/telemetry/emitter.js +229 -0
- package/dist/core/telemetry/queue.js +251 -0
- package/dist/core/theme/context.js +91 -0
- package/dist/core/theme/presets.js +228 -0
- package/dist/core/theme/state.js +181 -0
- package/dist/core/todos/invariant.js +10 -0
- package/dist/core/todos/state.js +177 -0
- package/dist/core/transport/version-interceptor.js +166 -0
- package/dist/core/vim/keymap.js +288 -0
- package/dist/core/vim/state.js +92 -0
- package/dist/index.js +28 -0
- package/dist/runtime/bootstrap.js +190 -0
- package/dist/runtime/cli.js +3073 -321
- package/dist/runtime/commands/cancel.js +231 -0
- package/dist/runtime/commands/chain.js +489 -0
- package/dist/runtime/commands/codegraph-status.js +227 -0
- package/dist/runtime/commands/compact.js +297 -0
- package/dist/runtime/commands/cost.js +199 -0
- package/dist/runtime/commands/delegate.js +242 -11
- package/dist/runtime/commands/dispatch.js +126 -0
- package/dist/runtime/commands/doctor.js +390 -0
- package/dist/runtime/commands/feedback.js +184 -0
- package/dist/runtime/commands/hooks.js +184 -0
- package/dist/runtime/commands/lsp.js +368 -0
- package/dist/runtime/commands/mcp.js +879 -0
- package/dist/runtime/commands/memory.js +508 -0
- package/dist/runtime/commands/model.js +237 -0
- package/dist/runtime/commands/onboarding.js +275 -0
- package/dist/runtime/commands/patch.js +128 -0
- package/dist/runtime/commands/permissions.js +112 -0
- package/dist/runtime/commands/plan.js +143 -0
- package/dist/runtime/commands/prd-check.js +285 -0
- package/dist/runtime/commands/redo-blob-store.js +92 -0
- package/dist/runtime/commands/redo.js +361 -0
- package/dist/runtime/commands/release-notes.js +229 -0
- package/dist/runtime/commands/repo-map.js +95 -0
- package/dist/runtime/commands/report.js +299 -0
- package/dist/runtime/commands/resume.js +118 -0
- package/dist/runtime/commands/review-consensus.js +17 -2
- package/dist/runtime/commands/rewind.js +333 -0
- package/dist/runtime/commands/sessions.js +163 -0
- package/dist/runtime/commands/share.js +316 -0
- package/dist/runtime/commands/status.js +186 -0
- package/dist/runtime/commands/stickers.js +82 -0
- package/dist/runtime/commands/style.js +194 -0
- package/dist/runtime/commands/theme.js +196 -0
- package/dist/runtime/commands/undo.js +32 -0
- package/dist/runtime/commands/update.js +289 -0
- package/dist/runtime/commands/vim.js +140 -0
- package/dist/runtime/commands/worktree.js +177 -0
- package/dist/runtime/headless-repl.js +195 -0
- package/dist/runtime/headless.js +543 -0
- package/dist/runtime/load-hooks-or-exit.js +71 -0
- package/dist/runtime/plan-decompose.js +531 -0
- package/dist/runtime/version.js +65 -0
- package/dist/tools/agent-tool.js +229 -0
- package/dist/tools/apply-patch.js +556 -0
- package/dist/tools/ask-user-question.js +213 -0
- package/dist/tools/ask-user.js +115 -0
- package/dist/tools/file-tools.js +85 -14
- package/dist/tools/lsp-tools.js +189 -0
- package/dist/tools/mcp-tool.js +260 -0
- package/dist/tools/multi-edit.js +361 -0
- package/dist/tools/powershell.js +156 -0
- package/dist/tools/registry.js +51 -0
- package/dist/tools/skill-tool.js +96 -0
- package/dist/tools/tasks.js +208 -0
- package/dist/tools/todo-write.js +184 -0
- package/dist/tools/web-fetch.js +147 -2
- package/dist/tools/web-search.js +458 -0
- package/dist/tui/agent-progress-card.js +111 -0
- package/dist/tui/agent-tree.js +10 -0
- package/dist/tui/ask-modal.js +2 -2
- package/dist/tui/ask-user-question-prompt.js +192 -0
- package/dist/tui/compact-banner.js +81 -0
- package/dist/tui/conversation-pane.js +82 -8
- package/dist/tui/cost-table.js +111 -0
- package/dist/tui/doctor-table.js +46 -0
- package/dist/tui/feedback-prompt.js +156 -0
- package/dist/tui/input-box.js +69 -2
- package/dist/tui/markdown-render.js +4 -4
- package/dist/tui/onboarding-wizard.js +240 -0
- package/dist/tui/permissions-picker.js +86 -0
- package/dist/tui/render.js +35 -0
- package/dist/tui/repl-render.js +303 -13
- package/dist/tui/repl-splash.js +2 -2
- package/dist/tui/repl.js +72 -14
- package/dist/tui/splash.js +1 -1
- package/dist/tui/status-bar.js +94 -16
- package/dist/tui/status-table.js +7 -0
- package/dist/tui/stickers-art.js +136 -0
- package/dist/tui/style-table.js +28 -0
- package/dist/tui/theme-table.js +29 -0
- package/dist/tui/tool-stream-pane.js +52 -3
- package/dist/tui/update-banner.js +20 -2
- package/dist/tui/vim-input.js +267 -0
- package/docs/examples/codegraph.mcp.json +10 -0
- package/package.json +12 -6
- package/test/scenarios/codegen-create-file.scenario.txt +13 -0
- package/test/scenarios/compact-force.scenario.txt +11 -0
- package/test/scenarios/identity.scenario.txt +11 -0
- package/test/scenarios/persona-handoff.scenario.txt +11 -0
- package/test/scenarios/walkback.scenario.txt +12 -0
- package/dist/core/engine/compaction-hook.js +0 -154
|
@@ -0,0 +1,156 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* PowerShell tool — leak L6 (Pugi parity catch-up 2026-05-28).
|
|
3
|
+
*
|
|
4
|
+
* Windows operators cannot run native `*.ps1` scripts via the bash tool
|
|
5
|
+
* (which spawns `/bin/sh`). This tool spawns `pwsh -NoProfile -Command`
|
|
6
|
+
* на cross-platform PowerShell 7+ binary so Windows-first workflows are
|
|
7
|
+
* first-class на Pugi.
|
|
8
|
+
*
|
|
9
|
+
* Clean-room re-implementation. Surface mirrors bashTool's permission
|
|
10
|
+
* gate, env sanitiser, output cap, timeout, and exit-code propagation;
|
|
11
|
+
* the only difference is the shell binary selection. Per-platform
|
|
12
|
+
* resolution:
|
|
13
|
+
* - All OS: try `pwsh` on $PATH first (PowerShell 7+ cross-platform).
|
|
14
|
+
* - Windows fallback: `powershell.exe` (Windows PowerShell 5.1 baked-in).
|
|
15
|
+
* - Other OS without pwsh: tool returns a clear "powershell binary
|
|
16
|
+
* not found" error so the operator can install pwsh or fall back
|
|
17
|
+
* к bash.
|
|
18
|
+
*
|
|
19
|
+
* Permission class: reuses the bash classifier — destructive patterns,
|
|
20
|
+
* sandbox detection, and additional-directories checks are command-string
|
|
21
|
+
* based and apply equally to pwsh and sh.
|
|
22
|
+
*/
|
|
23
|
+
import { spawnSync } from 'node:child_process';
|
|
24
|
+
import { evaluateBashPermission } from '../core/permission.js';
|
|
25
|
+
import { recordToolCall, recordToolResult } from '../core/session.js';
|
|
26
|
+
export const POWERSHELL_OUTPUT_CAP_BYTES = 64 * 1024;
|
|
27
|
+
export const POWERSHELL_DEFAULT_TIMEOUT_MS = 30_000;
|
|
28
|
+
export const POWERSHELL_MAX_TIMEOUT_MS = 120_000;
|
|
29
|
+
/** Cached binary path so repeated calls inside one session skip the probe. */
|
|
30
|
+
let cachedShellBinary;
|
|
31
|
+
function resolveShellBinary() {
|
|
32
|
+
if (cachedShellBinary !== undefined)
|
|
33
|
+
return cachedShellBinary;
|
|
34
|
+
// Try pwsh (cross-platform PowerShell 7+) first.
|
|
35
|
+
const pwshProbe = spawnSync('pwsh', ['-NoProfile', '-Command', 'exit 0'], {
|
|
36
|
+
encoding: 'utf8',
|
|
37
|
+
stdio: ['ignore', 'ignore', 'ignore'],
|
|
38
|
+
timeout: 3000,
|
|
39
|
+
});
|
|
40
|
+
if (pwshProbe.status === 0) {
|
|
41
|
+
cachedShellBinary = 'pwsh';
|
|
42
|
+
return 'pwsh';
|
|
43
|
+
}
|
|
44
|
+
// Windows fallback к the baked-in PowerShell 5.1.
|
|
45
|
+
if (process.platform === 'win32') {
|
|
46
|
+
const wpsProbe = spawnSync('powershell.exe', ['-NoProfile', '-Command', 'exit 0'], {
|
|
47
|
+
encoding: 'utf8',
|
|
48
|
+
stdio: ['ignore', 'ignore', 'ignore'],
|
|
49
|
+
timeout: 3000,
|
|
50
|
+
});
|
|
51
|
+
if (wpsProbe.status === 0) {
|
|
52
|
+
cachedShellBinary = 'powershell.exe';
|
|
53
|
+
return 'powershell.exe';
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
cachedShellBinary = null;
|
|
57
|
+
return null;
|
|
58
|
+
}
|
|
59
|
+
function sanitizeTimeout(value) {
|
|
60
|
+
if (value === undefined || !Number.isFinite(value) || value <= 0) {
|
|
61
|
+
return POWERSHELL_DEFAULT_TIMEOUT_MS;
|
|
62
|
+
}
|
|
63
|
+
return Math.min(value, POWERSHELL_MAX_TIMEOUT_MS);
|
|
64
|
+
}
|
|
65
|
+
function buildChildEnv() {
|
|
66
|
+
const env = { ...process.env };
|
|
67
|
+
delete env['PUGI_API_KEY'];
|
|
68
|
+
delete env['PUGI_LOGIN_TOKEN'];
|
|
69
|
+
return env;
|
|
70
|
+
}
|
|
71
|
+
/**
|
|
72
|
+
* Sync PowerShell dispatch. Mirrors bashToolSync shape so dispatchTool
|
|
73
|
+
* can call either tool with the same context shape.
|
|
74
|
+
*/
|
|
75
|
+
export function powerShellToolSync(input, ctx) {
|
|
76
|
+
const cmd = input.cmd ?? '';
|
|
77
|
+
const additionalDirectories = ctx.additionalDirectories ?? [];
|
|
78
|
+
const source = ctx.source ?? 'agent';
|
|
79
|
+
const toolCallId = recordToolCall(ctx.session, 'powershell', cmd);
|
|
80
|
+
const decision = evaluateBashPermission(cmd, ctx.settings.permissions.mode, {
|
|
81
|
+
workspaceRoot: ctx.root,
|
|
82
|
+
additionalDirectories,
|
|
83
|
+
source,
|
|
84
|
+
});
|
|
85
|
+
if (decision.decision !== 'allow') {
|
|
86
|
+
const reason = `Permission ${decision.decision}: ${decision.reason}`;
|
|
87
|
+
recordToolResult(ctx.session, toolCallId, 'error', reason);
|
|
88
|
+
return {
|
|
89
|
+
stdout: '',
|
|
90
|
+
stderr: `Permission denied: ${decision.reason}`,
|
|
91
|
+
exitCode: 126,
|
|
92
|
+
truncated: false,
|
|
93
|
+
timedOut: false,
|
|
94
|
+
shellBinary: 'unresolved',
|
|
95
|
+
};
|
|
96
|
+
}
|
|
97
|
+
const shellBinary = resolveShellBinary();
|
|
98
|
+
if (shellBinary === null) {
|
|
99
|
+
const reason = 'powershell binary not found (tried pwsh' +
|
|
100
|
+
(process.platform === 'win32' ? ', powershell.exe' : '') +
|
|
101
|
+
'). Install PowerShell 7+ from https://aka.ms/powershell or use the bash tool instead.';
|
|
102
|
+
recordToolResult(ctx.session, toolCallId, 'error', reason);
|
|
103
|
+
return {
|
|
104
|
+
stdout: '',
|
|
105
|
+
stderr: reason,
|
|
106
|
+
exitCode: 127,
|
|
107
|
+
truncated: false,
|
|
108
|
+
timedOut: false,
|
|
109
|
+
shellBinary: 'unavailable',
|
|
110
|
+
};
|
|
111
|
+
}
|
|
112
|
+
const timeoutMs = sanitizeTimeout(input.timeoutMs);
|
|
113
|
+
const childEnv = buildChildEnv();
|
|
114
|
+
const cwd = input.cwd ?? ctx.root;
|
|
115
|
+
const result = spawnSync(shellBinary, ['-NoProfile', '-Command', cmd], {
|
|
116
|
+
cwd,
|
|
117
|
+
env: childEnv,
|
|
118
|
+
encoding: 'utf8',
|
|
119
|
+
stdio: ['ignore', 'pipe', 'pipe'],
|
|
120
|
+
timeout: timeoutMs,
|
|
121
|
+
maxBuffer: 10 * 1024 * 1024,
|
|
122
|
+
});
|
|
123
|
+
const stdoutFull = (result.stdout ?? '').toString();
|
|
124
|
+
const stderrFull = (result.stderr ?? '').toString();
|
|
125
|
+
const combined = stdoutFull.length + stderrFull.length;
|
|
126
|
+
const truncated = combined > POWERSHELL_OUTPUT_CAP_BYTES;
|
|
127
|
+
let stdoutOut = stdoutFull;
|
|
128
|
+
let stderrOut = stderrFull;
|
|
129
|
+
if (truncated) {
|
|
130
|
+
const halfCap = POWERSHELL_OUTPUT_CAP_BYTES / 2;
|
|
131
|
+
stdoutOut = stdoutFull.slice(0, halfCap);
|
|
132
|
+
stderrOut = stderrFull.slice(0, halfCap);
|
|
133
|
+
}
|
|
134
|
+
const timedOut = result.error?.code === 'ETIMEDOUT' ||
|
|
135
|
+
result.signal === 'SIGTERM';
|
|
136
|
+
const exitCode = timedOut ? 124 : result.status ?? 1;
|
|
137
|
+
if (timedOut) {
|
|
138
|
+
recordToolResult(ctx.session, toolCallId, 'error', `powershell timed out after ${timeoutMs}ms`);
|
|
139
|
+
}
|
|
140
|
+
else {
|
|
141
|
+
recordToolResult(ctx.session, toolCallId, 'success', `powershell exit=${exitCode} bytes=${combined} binary=${shellBinary}`);
|
|
142
|
+
}
|
|
143
|
+
return {
|
|
144
|
+
stdout: stdoutOut,
|
|
145
|
+
stderr: stderrOut,
|
|
146
|
+
exitCode,
|
|
147
|
+
truncated,
|
|
148
|
+
timedOut,
|
|
149
|
+
shellBinary,
|
|
150
|
+
};
|
|
151
|
+
}
|
|
152
|
+
/** Visible-for-spec helper: forces a re-probe on next call. */
|
|
153
|
+
export function _resetShellBinaryCacheForSpec() {
|
|
154
|
+
cachedShellBinary = undefined;
|
|
155
|
+
}
|
|
156
|
+
//# sourceMappingURL=powershell.js.map
|
package/dist/tools/registry.js
CHANGED
|
@@ -1,8 +1,36 @@
|
|
|
1
1
|
const registry = [
|
|
2
|
+
// α7.7: unified-diff patch apply. Routes through the same security
|
|
3
|
+
// gate as Layer A/B/C, so the risk class matches `edit`/`write`
|
|
4
|
+
// (medium — writes inside the workspace, never to protected files).
|
|
5
|
+
{ name: 'apply_patch', permission: 'edit', risk: 'medium', concurrencySafe: false, m1: true },
|
|
6
|
+
// Leak L5 (2026-05-27): structured multi-choice clarifier tool. Risk =
|
|
7
|
+
// low because the dispatch is a pure UI surface — no file writes, no
|
|
8
|
+
// shell, no network. Permission = none (no workspace access required).
|
|
9
|
+
// concurrencySafe = true because the prompt-budget gate runs in the
|
|
10
|
+
// engine loop, not via tool-side mutex (one prompt per turn is enforced
|
|
11
|
+
// by the persona system prompt + the engine's tool_calls budget).
|
|
12
|
+
{ name: 'ask_user_question', permission: 'none', risk: 'low', concurrencySafe: true, m1: true },
|
|
2
13
|
{ name: 'bash', permission: 'bash', risk: 'high', concurrencySafe: false, m1: true },
|
|
3
14
|
{ name: 'edit', permission: 'edit', risk: 'medium', concurrencySafe: false, m1: true },
|
|
4
15
|
{ name: 'glob', permission: 'read', risk: 'low', concurrencySafe: true, m1: true },
|
|
5
16
|
{ name: 'grep', permission: 'read', risk: 'low', concurrencySafe: true, m1: true },
|
|
17
|
+
// α7.7: LSP read-only surface. Server runs locally, no Anvil
|
|
18
|
+
// round-trip. Concurrency-safe because every operation reads
|
|
19
|
+
// server state without mutating workspace files.
|
|
20
|
+
{ name: 'lsp_definition', permission: 'read', risk: 'low', concurrencySafe: true, m1: true },
|
|
21
|
+
{ name: 'lsp_diagnostics', permission: 'read', risk: 'low', concurrencySafe: true, m1: true },
|
|
22
|
+
{ name: 'lsp_hover', permission: 'read', risk: 'low', concurrencySafe: true, m1: true },
|
|
23
|
+
{ name: 'lsp_references', permission: 'read', risk: 'low', concurrencySafe: true, m1: true },
|
|
24
|
+
// β7 L5+T11: multi_edit dispatches an ordered batch of Layer A edits
|
|
25
|
+
// as a single transaction. Risk = medium (same chokepoints as `edit`).
|
|
26
|
+
// concurrencySafe = false because the journal serialises one dispatch
|
|
27
|
+
// per session.
|
|
28
|
+
{ name: 'multi_edit', permission: 'edit', risk: 'medium', concurrencySafe: false, m1: true },
|
|
29
|
+
// Leak L6 (2026-05-28): PowerShell tool for Windows-first workflows. Same
|
|
30
|
+
// bash permission class — destructive-pattern classification fires the
|
|
31
|
+
// same gate. concurrencySafe = false because spawn-shell child cwd /
|
|
32
|
+
// env carry-over could race across parallel agent calls.
|
|
33
|
+
{ name: 'powershell', permission: 'bash', risk: 'high', concurrencySafe: false, m1: false },
|
|
6
34
|
{ name: 'question', permission: 'none', risk: 'low', concurrencySafe: false, m1: true },
|
|
7
35
|
{ name: 'read', permission: 'read', risk: 'low', concurrencySafe: true, m1: true },
|
|
8
36
|
{ name: 'skill', permission: 'read', risk: 'low', concurrencySafe: true, m1: true },
|
|
@@ -10,7 +38,30 @@ const registry = [
|
|
|
10
38
|
{ name: 'task_get', permission: 'none', risk: 'low', concurrencySafe: true, m1: true },
|
|
11
39
|
{ name: 'task_list', permission: 'none', risk: 'low', concurrencySafe: true, m1: true },
|
|
12
40
|
{ name: 'task_update', permission: 'none', risk: 'low', concurrencySafe: false, m1: true },
|
|
41
|
+
// Leak L16 (2026-05-27): batch TodoWrite. Mirrors Claude Code's upstream
|
|
42
|
+
// surface — full board snapshot, single-in-progress invariant, atomic
|
|
43
|
+
// tmp+rename persistence to `.pugi/todos.json`. `concurrencySafe = false`
|
|
44
|
+
// because two concurrent writes could lose the loser's snapshot (the
|
|
45
|
+
// rename is atomic but the read-modify-write loop is not). Risk = low
|
|
46
|
+
// because the only filesystem mutation lands inside `.pugi/todos.json`,
|
|
47
|
+
// which is metadata, not source.
|
|
48
|
+
{ name: 'todo_write', permission: 'none', risk: 'low', concurrencySafe: false, m1: true },
|
|
13
49
|
{ name: 'web_fetch', permission: 'network', risk: 'medium', concurrencySafe: true, m1: true },
|
|
50
|
+
// α7.7: scratch worktree management. `worktree_create` writes nothing
|
|
51
|
+
// dangerous (a clone under `.pugi/worktrees/`); `worktree_promote`
|
|
52
|
+
// applies a diff back to the main tree, so it shares the `edit`
|
|
53
|
+
// risk class. `worktree_drop` is the cleanup primitive.
|
|
54
|
+
//
|
|
55
|
+
// R1 fix (2026-05-26, PR #413 r1, Fix 9): raised `worktree_create`
|
|
56
|
+
// and `worktree_drop` from `low` to `medium`. `worktree_drop` runs
|
|
57
|
+
// `rmSync` on its target — even with the new path-containment gate
|
|
58
|
+
// in `core/edits/worktree.ts::dropWorktree`, a destructive primitive
|
|
59
|
+
// belongs in `medium` so the permission FSM prompts on every call.
|
|
60
|
+
// `worktree_create` is raised for disk-pressure parity (a runaway
|
|
61
|
+
// agent loop could fill the disk with abandoned scratch worktrees).
|
|
62
|
+
{ name: 'worktree_create', permission: 'edit', risk: 'medium', concurrencySafe: false, m1: true },
|
|
63
|
+
{ name: 'worktree_drop', permission: 'edit', risk: 'medium', concurrencySafe: false, m1: true },
|
|
64
|
+
{ name: 'worktree_promote', permission: 'edit', risk: 'medium', concurrencySafe: false, m1: true },
|
|
14
65
|
{ name: 'write', permission: 'edit', risk: 'medium', concurrencySafe: false, m1: true },
|
|
15
66
|
];
|
|
16
67
|
export const toolRegistry = registry.sort((a, b) => a.name.localeCompare(b.name));
|
|
@@ -0,0 +1,96 @@
|
|
|
1
|
+
import { listSkills } from '../core/skills/loader.js';
|
|
2
|
+
import { hashSkillDir, verifyTrust } from '../core/skills/trust.js';
|
|
3
|
+
export const SKILL_BODY_CAP_BYTES = 32 * 1024;
|
|
4
|
+
export const SKILL_LIST_CAP = 100;
|
|
5
|
+
export function skillList(ctx, input) {
|
|
6
|
+
const scope = input.scope ?? 'all';
|
|
7
|
+
const all = [];
|
|
8
|
+
if (scope === 'all' || scope === 'global') {
|
|
9
|
+
all.push(...listSkills('global', ctx.workspaceRoot));
|
|
10
|
+
}
|
|
11
|
+
if (scope === 'all' || scope === 'workspace') {
|
|
12
|
+
all.push(...listSkills('workspace', ctx.workspaceRoot));
|
|
13
|
+
}
|
|
14
|
+
// Dedup by name, prefer workspace scope when both exist (workspace
|
|
15
|
+
// overrides global per skills loader convention).
|
|
16
|
+
const byName = new Map();
|
|
17
|
+
for (const skill of all) {
|
|
18
|
+
const prev = byName.get(skill.name);
|
|
19
|
+
if (!prev || skill.scope === 'workspace') {
|
|
20
|
+
byName.set(skill.name, skill);
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
return Array.from(byName.values())
|
|
24
|
+
.slice(0, SKILL_LIST_CAP)
|
|
25
|
+
.map((skill) => ({
|
|
26
|
+
name: skill.name,
|
|
27
|
+
description: skill.frontmatter.description,
|
|
28
|
+
scope: skill.scope,
|
|
29
|
+
}));
|
|
30
|
+
}
|
|
31
|
+
export async function skillInvoke(ctx, input) {
|
|
32
|
+
if (!input.name || typeof input.name !== 'string') {
|
|
33
|
+
throw new Error('skill: name is required');
|
|
34
|
+
}
|
|
35
|
+
// Defense-in-depth: skill loader already validates slugs but the
|
|
36
|
+
// tool surface is operator-controlled.
|
|
37
|
+
if (!/^[a-zA-Z0-9_-]{1,128}$/.test(input.name)) {
|
|
38
|
+
throw new Error(`skill: invalid skill name shape: "${input.name}"`);
|
|
39
|
+
}
|
|
40
|
+
// Workspace scope wins over global (operator override). Mirrors
|
|
41
|
+
// SkillLoader convention.
|
|
42
|
+
const workspace = listSkills('workspace', ctx.workspaceRoot).find((s) => s.name === input.name);
|
|
43
|
+
const global = workspace
|
|
44
|
+
? null
|
|
45
|
+
: listSkills('global', ctx.workspaceRoot).find((s) => s.name === input.name);
|
|
46
|
+
const skill = workspace ?? global;
|
|
47
|
+
if (!skill) {
|
|
48
|
+
throw new Error(`skill: not found: "${input.name}"`);
|
|
49
|
+
}
|
|
50
|
+
// β1a r1 (2026-05-26): re-verify the on-disk skill payload against
|
|
51
|
+
// the trust manifest sha256 on EVERY invoke, not just at install
|
|
52
|
+
// time. Before this fix a post-install swap (malicious npm dep that
|
|
53
|
+
// touches `~/.pugi/skills/<name>/SKILL.md` after the operator
|
|
54
|
+
// approved the install) would bypass the trust gate — `listSkills`
|
|
55
|
+
// reads the body fresh from disk and the loader does no integrity
|
|
56
|
+
// check. The skill body lands directly in the model's tool result,
|
|
57
|
+
// so a mutated body is a prompt-injection vector against the agent
|
|
58
|
+
// loop's tool surface.
|
|
59
|
+
//
|
|
60
|
+
// Posture:
|
|
61
|
+
// - `trusted` → proceed (body is hash-pinned).
|
|
62
|
+
// - `unsigned` → refuse: the operator never approved this skill.
|
|
63
|
+
// This catches the case where a skill directory was dropped in
|
|
64
|
+
// manually (no `pugi skills install`) and the loader picked it
|
|
65
|
+
// up. Refusing is fail-closed.
|
|
66
|
+
// - `mismatch` → refuse + surface the recorded vs actual hashes
|
|
67
|
+
// so the operator can decide between re-trust and revoke.
|
|
68
|
+
//
|
|
69
|
+
// Performance: `hashSkillDir` walks the skill directory on every
|
|
70
|
+
// invoke. Skills are small (median 4-8 files, <50KB total) so the
|
|
71
|
+
// cost is sub-millisecond on warm cache. The β1a r1 spec exercises
|
|
72
|
+
// a mutated-body case; the existing skill-tool.spec.ts cases for
|
|
73
|
+
// happy-path use the `recordTrust` helper to seed the registry.
|
|
74
|
+
const actualHash = hashSkillDir(skill.dir);
|
|
75
|
+
const verdict = await verifyTrust('skill', skill.scope, skill.name, actualHash);
|
|
76
|
+
if (verdict.status === 'unsigned') {
|
|
77
|
+
throw new Error(`skill: refused to invoke "${skill.name}" — no trust entry (run \`pugi skills trust ${skill.name}\` to approve)`);
|
|
78
|
+
}
|
|
79
|
+
if (verdict.status === 'mismatch') {
|
|
80
|
+
throw new Error(`skill: refused to invoke "${skill.name}" — sha256 mismatch (recorded ${verdict.recorded.slice(0, 12)}…, actual ${verdict.actual.slice(0, 12)}…). Re-trust via \`pugi skills trust ${skill.name}\`.`);
|
|
81
|
+
}
|
|
82
|
+
const body = skill.body;
|
|
83
|
+
const truncated = Buffer.byteLength(body, 'utf8') > SKILL_BODY_CAP_BYTES;
|
|
84
|
+
const cappedBody = truncated
|
|
85
|
+
? body.slice(0, SKILL_BODY_CAP_BYTES) +
|
|
86
|
+
`\n\n(... truncated at ${SKILL_BODY_CAP_BYTES} bytes — see \`pugi skills info ${skill.name}\` for full text)`
|
|
87
|
+
: body;
|
|
88
|
+
return {
|
|
89
|
+
name: skill.name,
|
|
90
|
+
scope: skill.scope,
|
|
91
|
+
description: skill.frontmatter.description,
|
|
92
|
+
body: cappedBody,
|
|
93
|
+
truncated,
|
|
94
|
+
};
|
|
95
|
+
}
|
|
96
|
+
//# sourceMappingURL=skill-tool.js.map
|
|
@@ -0,0 +1,208 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* task_* tool family — β1 T1/T6 (TodoWrite + agent task ledger).
|
|
3
|
+
*
|
|
4
|
+
* Mirrors Claude Code's TodoWrite tool surface so a model trained on
|
|
5
|
+
* the upstream tool grammar speaks Pugi's variant verbatim. Four ops:
|
|
6
|
+
*
|
|
7
|
+
* - `task_create` — append a new task to the session's todo ledger.
|
|
8
|
+
* Returns the assigned id.
|
|
9
|
+
* - `task_get` — fetch a single task by id.
|
|
10
|
+
* - `task_list` — list every task in the current session, ordered
|
|
11
|
+
* by createdAt ascending.
|
|
12
|
+
* - `task_update` — mutate status/title/notes of an existing task.
|
|
13
|
+
* Append-only journal — every mutation lands as a
|
|
14
|
+
* fresh JSONL line and the latest line per id wins
|
|
15
|
+
* on `task_list` / `task_get` reads.
|
|
16
|
+
*
|
|
17
|
+
* Persistence: append-only JSONL at
|
|
18
|
+
* `.pugi/sessions/<sessionId>/tasks.jsonl`. Append-only keeps crash
|
|
19
|
+
* recovery trivial — a partial write at the end of the file is the
|
|
20
|
+
* worst case and the parser drops the malformed tail line.
|
|
21
|
+
*
|
|
22
|
+
* Scope: this is the local-side ledger surface. Anvil-side mirror
|
|
23
|
+
* (cabinet `/projects/[id]/tasks` page) ships in β5 once the session-
|
|
24
|
+
* memory hook lands; until then the ledger is purely local.
|
|
25
|
+
*/
|
|
26
|
+
import { appendFileSync, chmodSync, existsSync, mkdirSync, readFileSync, } from 'node:fs';
|
|
27
|
+
import { dirname, join } from 'node:path';
|
|
28
|
+
import { randomUUID } from 'node:crypto';
|
|
29
|
+
function ledgerPath(ctx) {
|
|
30
|
+
// Defense-in-depth: the sessionId is supposed to be a UUID minted by
|
|
31
|
+
// openSession() but the tool surface is operator-facing. Validate the
|
|
32
|
+
// shape before composing a path — refuse anything that contains
|
|
33
|
+
// separators or shell wildcards.
|
|
34
|
+
if (!/^[a-zA-Z0-9_-]{1,128}$/.test(ctx.sessionId)) {
|
|
35
|
+
throw new Error(`task_*: invalid sessionId shape: "${ctx.sessionId}"`);
|
|
36
|
+
}
|
|
37
|
+
return join(ctx.workspaceRoot, '.pugi', 'sessions', ctx.sessionId, 'tasks.jsonl');
|
|
38
|
+
}
|
|
39
|
+
function nowIso(ctx) {
|
|
40
|
+
return (ctx.now ? ctx.now() : new Date()).toISOString();
|
|
41
|
+
}
|
|
42
|
+
function ensureDir(path) {
|
|
43
|
+
// β1a r1 (2026-05-26): switched from POSIX-only
|
|
44
|
+
// `path.slice(0, path.lastIndexOf('/'))` to `path.dirname()` so
|
|
45
|
+
// Windows path separators (`\`) work. Also chmod the per-session
|
|
46
|
+
// directory to 0o700 — the tasks ledger carries operator-confidential
|
|
47
|
+
// brief text, status notes, and timing metadata that should not be
|
|
48
|
+
// world-readable through an inherited umask.
|
|
49
|
+
const dir = dirname(path);
|
|
50
|
+
if (!existsSync(dir)) {
|
|
51
|
+
mkdirSync(dir, { recursive: true });
|
|
52
|
+
try {
|
|
53
|
+
chmodSync(dir, 0o700);
|
|
54
|
+
}
|
|
55
|
+
catch {
|
|
56
|
+
// Best-effort. POSIX permission setting is a no-op on Windows
|
|
57
|
+
// NTFS, and the dir-creation race with another concurrent task
|
|
58
|
+
// tool call is the only realistic failure case. The 0o600 mode
|
|
59
|
+
// on the JSONL file itself remains the primary guard; the dir
|
|
60
|
+
// chmod is defense in depth for tools that walk `.pugi/`.
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
function readJournal(ctx) {
|
|
65
|
+
const path = ledgerPath(ctx);
|
|
66
|
+
if (!existsSync(path))
|
|
67
|
+
return [];
|
|
68
|
+
const raw = readFileSync(path, 'utf8');
|
|
69
|
+
const out = [];
|
|
70
|
+
for (const line of raw.split('\n')) {
|
|
71
|
+
if (!line.trim())
|
|
72
|
+
continue;
|
|
73
|
+
try {
|
|
74
|
+
const parsed = JSON.parse(line);
|
|
75
|
+
if ((parsed.op === 'create' || parsed.op === 'update') &&
|
|
76
|
+
typeof parsed.id === 'string' &&
|
|
77
|
+
typeof parsed.at === 'string') {
|
|
78
|
+
out.push(parsed);
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
catch {
|
|
82
|
+
// Drop malformed line (partial-write tail or external corruption).
|
|
83
|
+
// The append-only design guarantees only the LAST line can be bad
|
|
84
|
+
// — everything before it is whole.
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
return out;
|
|
88
|
+
}
|
|
89
|
+
function fold(journal) {
|
|
90
|
+
const out = new Map();
|
|
91
|
+
for (const entry of journal) {
|
|
92
|
+
if (entry.op === 'create') {
|
|
93
|
+
if (!entry.title)
|
|
94
|
+
continue;
|
|
95
|
+
out.set(entry.id, {
|
|
96
|
+
id: entry.id,
|
|
97
|
+
title: entry.title,
|
|
98
|
+
status: entry.status ?? 'pending',
|
|
99
|
+
...(entry.notes !== undefined ? { notes: entry.notes } : {}),
|
|
100
|
+
createdAt: entry.at,
|
|
101
|
+
updatedAt: entry.at,
|
|
102
|
+
});
|
|
103
|
+
}
|
|
104
|
+
else {
|
|
105
|
+
const prev = out.get(entry.id);
|
|
106
|
+
if (!prev)
|
|
107
|
+
continue; // update before create — drop silently
|
|
108
|
+
out.set(entry.id, {
|
|
109
|
+
...prev,
|
|
110
|
+
...(entry.title !== undefined ? { title: entry.title } : {}),
|
|
111
|
+
...(entry.status !== undefined ? { status: entry.status } : {}),
|
|
112
|
+
...(entry.notes !== undefined ? { notes: entry.notes } : {}),
|
|
113
|
+
updatedAt: entry.at,
|
|
114
|
+
});
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
return out;
|
|
118
|
+
}
|
|
119
|
+
function appendEntry(ctx, entry) {
|
|
120
|
+
const path = ledgerPath(ctx);
|
|
121
|
+
ensureDir(path);
|
|
122
|
+
appendFileSync(path, `${JSON.stringify(entry)}\n`, {
|
|
123
|
+
encoding: 'utf8',
|
|
124
|
+
mode: 0o600,
|
|
125
|
+
});
|
|
126
|
+
}
|
|
127
|
+
export function taskCreate(ctx, input) {
|
|
128
|
+
const title = input.title?.trim();
|
|
129
|
+
if (!title) {
|
|
130
|
+
throw new Error('task_create: title is required');
|
|
131
|
+
}
|
|
132
|
+
if (title.length > 2_000) {
|
|
133
|
+
throw new Error('task_create: title exceeds 2000 char cap');
|
|
134
|
+
}
|
|
135
|
+
const status = input.status ?? 'pending';
|
|
136
|
+
if (!isValidStatus(status)) {
|
|
137
|
+
throw new Error(`task_create: invalid status "${status}"`);
|
|
138
|
+
}
|
|
139
|
+
const id = `task-${randomUUID()}`;
|
|
140
|
+
const at = nowIso(ctx);
|
|
141
|
+
const entry = {
|
|
142
|
+
op: 'create',
|
|
143
|
+
id,
|
|
144
|
+
title,
|
|
145
|
+
status,
|
|
146
|
+
at,
|
|
147
|
+
...(input.notes !== undefined ? { notes: input.notes } : {}),
|
|
148
|
+
};
|
|
149
|
+
appendEntry(ctx, entry);
|
|
150
|
+
return {
|
|
151
|
+
id,
|
|
152
|
+
title,
|
|
153
|
+
status,
|
|
154
|
+
...(input.notes !== undefined ? { notes: input.notes } : {}),
|
|
155
|
+
createdAt: at,
|
|
156
|
+
updatedAt: at,
|
|
157
|
+
};
|
|
158
|
+
}
|
|
159
|
+
export function taskGet(ctx, id) {
|
|
160
|
+
if (typeof id !== 'string' || id.length === 0) {
|
|
161
|
+
throw new Error('task_get: id is required');
|
|
162
|
+
}
|
|
163
|
+
const folded = fold(readJournal(ctx));
|
|
164
|
+
return folded.get(id) ?? null;
|
|
165
|
+
}
|
|
166
|
+
export function taskList(ctx) {
|
|
167
|
+
const folded = fold(readJournal(ctx));
|
|
168
|
+
return Array.from(folded.values()).sort((a, b) => a.createdAt.localeCompare(b.createdAt));
|
|
169
|
+
}
|
|
170
|
+
export function taskUpdate(ctx, input) {
|
|
171
|
+
if (!input.id)
|
|
172
|
+
throw new Error('task_update: id is required');
|
|
173
|
+
const folded = fold(readJournal(ctx));
|
|
174
|
+
const existing = folded.get(input.id);
|
|
175
|
+
if (!existing) {
|
|
176
|
+
throw new Error(`task_update: unknown id "${input.id}"`);
|
|
177
|
+
}
|
|
178
|
+
if (input.status !== undefined && !isValidStatus(input.status)) {
|
|
179
|
+
throw new Error(`task_update: invalid status "${input.status}"`);
|
|
180
|
+
}
|
|
181
|
+
if (input.title !== undefined && input.title.trim().length === 0) {
|
|
182
|
+
throw new Error('task_update: title cannot be empty');
|
|
183
|
+
}
|
|
184
|
+
const at = nowIso(ctx);
|
|
185
|
+
const entry = {
|
|
186
|
+
op: 'update',
|
|
187
|
+
id: input.id,
|
|
188
|
+
at,
|
|
189
|
+
...(input.title !== undefined ? { title: input.title } : {}),
|
|
190
|
+
...(input.status !== undefined ? { status: input.status } : {}),
|
|
191
|
+
...(input.notes !== undefined ? { notes: input.notes } : {}),
|
|
192
|
+
};
|
|
193
|
+
appendEntry(ctx, entry);
|
|
194
|
+
return {
|
|
195
|
+
...existing,
|
|
196
|
+
...(input.title !== undefined ? { title: input.title } : {}),
|
|
197
|
+
...(input.status !== undefined ? { status: input.status } : {}),
|
|
198
|
+
...(input.notes !== undefined ? { notes: input.notes } : {}),
|
|
199
|
+
updatedAt: at,
|
|
200
|
+
};
|
|
201
|
+
}
|
|
202
|
+
function isValidStatus(status) {
|
|
203
|
+
return (status === 'pending' ||
|
|
204
|
+
status === 'in_progress' ||
|
|
205
|
+
status === 'completed' ||
|
|
206
|
+
status === 'cancelled');
|
|
207
|
+
}
|
|
208
|
+
//# sourceMappingURL=tasks.js.map
|