@pugi/cli 0.1.0-beta.5 → 0.1.0-beta.51
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/assets/pugi-prozr2-mascot.ansi +9 -0
- 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 +400 -4
- 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 +112 -3
- 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/hooks.js +118 -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/sandbox.js +40 -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/auto-compact.js +179 -0
- package/dist/core/engine/budgets.js +155 -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 +897 -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/path-security.js +284 -2
- 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 +1897 -37
- package/dist/core/repl/slash-commands.js +430 -15
- 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/core/worktree-manager/cleanup.js +123 -0
- package/dist/core/worktree-manager/manager.js +303 -0
- package/dist/index.js +28 -0
- package/dist/runtime/bootstrap.js +190 -0
- package/dist/runtime/cli.js +3241 -343
- 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 +412 -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/commands/worktrees.js +155 -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/bash.js +203 -4
- 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 +268 -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 +218 -3
- 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 +313 -35
- package/dist/tui/repl-splash-art.js +1 -1
- package/dist/tui/repl-splash-mascot.js +32 -8
- package/dist/tui/repl-splash.js +2 -2
- package/dist/tui/repl.js +85 -5
- 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/thinking-spinner.js +123 -0
- package/dist/tui/tool-stream-pane.js +52 -3
- package/dist/tui/update-banner.js +27 -2
- package/dist/tui/vim-input.js +267 -0
- package/dist/tui/welcome-banner.js +107 -0
- package/dist/tui/welcome-data.js +293 -0
- package/docs/examples/codegraph.mcp.json +10 -0
- package/package.json +13 -7
- 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,412 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* `pugi doctor` — environment health report (Leak L17, 2026-05-27).
|
|
3
|
+
*
|
|
4
|
+
* Parity command with Claude Code's `/doctor` (gap doc:
|
|
5
|
+
* docs/research/2026-05-27-pugi-gap-analysis-3-repos.md §6). Probes
|
|
6
|
+
* auth, API reachability, CLI version, workspace state, disk space,
|
|
7
|
+
* Node version, pnpm, git, MCP servers, config file, and session
|
|
8
|
+
* activity. Emits either a human-readable table OR a structured JSON
|
|
9
|
+
* envelope depending on `--json`.
|
|
10
|
+
*
|
|
11
|
+
* Module contract:
|
|
12
|
+
*
|
|
13
|
+
* - This file owns the WIRING from CLI flags + workspace context to
|
|
14
|
+
* the probe runner. The probes themselves live in
|
|
15
|
+
* `core/diagnostics/probes/*.ts` and have NO module-level coupling
|
|
16
|
+
* to the CLI dispatch surface.
|
|
17
|
+
*
|
|
18
|
+
* - `runDoctorCommand` is the single entry point. Both the top-level
|
|
19
|
+
* `pugi doctor` handler in `runtime/cli.ts` AND the in-REPL
|
|
20
|
+
* `/doctor` slash command call it. The function returns the
|
|
21
|
+
* `DoctorReport` so the REPL can render via the Ink table without
|
|
22
|
+
* re-running the probes.
|
|
23
|
+
*
|
|
24
|
+
* - Exit codes are derived from `exitCodeFor(overall)` in
|
|
25
|
+
* `core/diagnostics/types.ts` and bubble up via `process.exitCode`
|
|
26
|
+
* (matches the convention of every other CLI handler in cli.ts).
|
|
27
|
+
*
|
|
28
|
+
* - The MCP probe is opportunistic: if `core/mcp/registry.js` is
|
|
29
|
+
* unavailable for any reason (e.g. sibling L13 not yet landed,
|
|
30
|
+
* unexpected schema change), the probe degrades to a graceful
|
|
31
|
+
* `skipped` result so the rest of the table still renders.
|
|
32
|
+
*/
|
|
33
|
+
import { execFileSync } from 'node:child_process';
|
|
34
|
+
import { constants as fsConstants, existsSync, accessSync, readFileSync, statSync } from 'node:fs';
|
|
35
|
+
import { homedir } from 'node:os';
|
|
36
|
+
import { resolveActiveCredential } from '../../core/credentials.js';
|
|
37
|
+
import { PUGI_CLI_VERSION } from '../version.js';
|
|
38
|
+
import { runProbes, } from '../../core/diagnostics/probe-runner.js';
|
|
39
|
+
import { computeOverall, countProbes, exitCodeFor, } from '../../core/diagnostics/types.js';
|
|
40
|
+
import { probeAuth } from '../../core/diagnostics/probes/auth.js';
|
|
41
|
+
import { probeApi } from '../../core/diagnostics/probes/api.js';
|
|
42
|
+
import { probeCliVersion } from '../../core/diagnostics/probes/cli-version.js';
|
|
43
|
+
import { probeWorkspace } from '../../core/diagnostics/probes/workspace.js';
|
|
44
|
+
import { probeDisk } from '../../core/diagnostics/probes/disk.js';
|
|
45
|
+
import { probeNode } from '../../core/diagnostics/probes/node.js';
|
|
46
|
+
import { probePnpm } from '../../core/diagnostics/probes/pnpm.js';
|
|
47
|
+
import { probeGit } from '../../core/diagnostics/probes/git.js';
|
|
48
|
+
import { probeMcp } from '../../core/diagnostics/probes/mcp.js';
|
|
49
|
+
import { probeConfig } from '../../core/diagnostics/probes/config.js';
|
|
50
|
+
import { probeSession } from '../../core/diagnostics/probes/session.js';
|
|
51
|
+
import { probeDenialTracking } from '../../core/diagnostics/probes/denial-tracking.js';
|
|
52
|
+
import { probeBareMode } from '../../core/diagnostics/probes/bare-mode.js';
|
|
53
|
+
import { probePugiMdHierarchy } from '../../core/diagnostics/probes/pugi-md.js';
|
|
54
|
+
import { probeSandbox } from '../../core/diagnostics/probes/sandbox.js';
|
|
55
|
+
import { probeHooks } from '../../core/diagnostics/probes/hooks.js';
|
|
56
|
+
/**
|
|
57
|
+
* Default API URL when no PUGI_API_URL env override is set. Mirrors
|
|
58
|
+
* the constant in `core/credentials.ts` (kept local to avoid an
|
|
59
|
+
* extra named export from that module).
|
|
60
|
+
*/
|
|
61
|
+
const DEFAULT_API_URL = 'https://api.pugi.io';
|
|
62
|
+
/**
|
|
63
|
+
* Build the standard probe set with production dependencies. Exported
|
|
64
|
+
* for the spec so the test can construct the same suite with stub
|
|
65
|
+
* deps + assert per-probe ordering + fail-isolation in isolation.
|
|
66
|
+
*/
|
|
67
|
+
export function buildDefaultProbes(ctx, options = {}) {
|
|
68
|
+
const fetchImpl = ctx.fetchImpl ?? globalThis.fetch.bind(globalThis);
|
|
69
|
+
const now = Date.now;
|
|
70
|
+
const probes = [
|
|
71
|
+
{
|
|
72
|
+
name: 'AUTH',
|
|
73
|
+
run: () => probeAuth(ctx, {
|
|
74
|
+
resolveCredential: (env, home) => {
|
|
75
|
+
const credential = resolveActiveCredential(env, home);
|
|
76
|
+
if (!credential)
|
|
77
|
+
return null;
|
|
78
|
+
return { apiUrl: credential.apiUrl, apiKey: credential.apiKey };
|
|
79
|
+
},
|
|
80
|
+
fetchImpl,
|
|
81
|
+
now,
|
|
82
|
+
}),
|
|
83
|
+
timeoutMs: 4_000,
|
|
84
|
+
},
|
|
85
|
+
{
|
|
86
|
+
name: 'API',
|
|
87
|
+
run: () => probeApi(ctx, {
|
|
88
|
+
resolveApiUrl: (env) => {
|
|
89
|
+
return env.PUGI_API_URL ?? DEFAULT_API_URL;
|
|
90
|
+
},
|
|
91
|
+
fetchImpl,
|
|
92
|
+
now,
|
|
93
|
+
}),
|
|
94
|
+
timeoutMs: 4_000,
|
|
95
|
+
},
|
|
96
|
+
{
|
|
97
|
+
name: 'CLI VERSION',
|
|
98
|
+
run: () => probeCliVersion({
|
|
99
|
+
localVersion: options.localCliVersion ?? PUGI_CLI_VERSION,
|
|
100
|
+
fetchImpl,
|
|
101
|
+
now,
|
|
102
|
+
}),
|
|
103
|
+
timeoutMs: 4_000,
|
|
104
|
+
},
|
|
105
|
+
{
|
|
106
|
+
name: 'WORKSPACE',
|
|
107
|
+
run: async () => probeWorkspace(ctx, {
|
|
108
|
+
existsSync,
|
|
109
|
+
statSync,
|
|
110
|
+
accessSync,
|
|
111
|
+
W_OK: fsConstants.W_OK,
|
|
112
|
+
}),
|
|
113
|
+
},
|
|
114
|
+
{
|
|
115
|
+
name: 'DISK',
|
|
116
|
+
run: async () => probeDisk(ctx, {
|
|
117
|
+
getFreeBytes: (home) => getFreeBytesViaDf(home),
|
|
118
|
+
}),
|
|
119
|
+
},
|
|
120
|
+
{
|
|
121
|
+
name: 'NODE',
|
|
122
|
+
run: async () => probeNode({ version: process.version }),
|
|
123
|
+
},
|
|
124
|
+
{
|
|
125
|
+
name: 'PNPM',
|
|
126
|
+
run: async () => probePnpm({
|
|
127
|
+
resolveVersion: () => execFileSync('pnpm', ['--version'], {
|
|
128
|
+
encoding: 'utf8',
|
|
129
|
+
timeout: 2_000,
|
|
130
|
+
stdio: ['ignore', 'pipe', 'ignore'],
|
|
131
|
+
}).trim(),
|
|
132
|
+
}),
|
|
133
|
+
},
|
|
134
|
+
{
|
|
135
|
+
name: 'GIT',
|
|
136
|
+
run: async () => probeGit(ctx, {
|
|
137
|
+
resolveVersion: () => execFileSync('git', ['--version'], {
|
|
138
|
+
encoding: 'utf8',
|
|
139
|
+
timeout: 2_000,
|
|
140
|
+
stdio: ['ignore', 'pipe', 'ignore'],
|
|
141
|
+
}).trim(),
|
|
142
|
+
isInWorkTree: (cwd) => {
|
|
143
|
+
try {
|
|
144
|
+
const result = execFileSync('git', ['-C', cwd, 'rev-parse', '--is-inside-work-tree'], {
|
|
145
|
+
encoding: 'utf8',
|
|
146
|
+
timeout: 2_000,
|
|
147
|
+
stdio: ['ignore', 'pipe', 'ignore'],
|
|
148
|
+
}).trim();
|
|
149
|
+
return result === 'true';
|
|
150
|
+
}
|
|
151
|
+
catch {
|
|
152
|
+
return false;
|
|
153
|
+
}
|
|
154
|
+
},
|
|
155
|
+
resolveHeadSha: (cwd) => {
|
|
156
|
+
try {
|
|
157
|
+
return execFileSync('git', ['-C', cwd, 'rev-parse', 'HEAD'], {
|
|
158
|
+
encoding: 'utf8',
|
|
159
|
+
timeout: 2_000,
|
|
160
|
+
stdio: ['ignore', 'pipe', 'ignore'],
|
|
161
|
+
}).trim();
|
|
162
|
+
}
|
|
163
|
+
catch {
|
|
164
|
+
return null;
|
|
165
|
+
}
|
|
166
|
+
},
|
|
167
|
+
resolveRoot: (cwd) => {
|
|
168
|
+
try {
|
|
169
|
+
return execFileSync('git', ['-C', cwd, 'rev-parse', '--show-toplevel'], {
|
|
170
|
+
encoding: 'utf8',
|
|
171
|
+
timeout: 2_000,
|
|
172
|
+
stdio: ['ignore', 'pipe', 'ignore'],
|
|
173
|
+
}).trim();
|
|
174
|
+
}
|
|
175
|
+
catch {
|
|
176
|
+
return null;
|
|
177
|
+
}
|
|
178
|
+
},
|
|
179
|
+
}),
|
|
180
|
+
},
|
|
181
|
+
{
|
|
182
|
+
name: 'MCP SERVERS',
|
|
183
|
+
run: async () => probeMcpSafely(ctx),
|
|
184
|
+
},
|
|
185
|
+
{
|
|
186
|
+
name: 'CONFIG',
|
|
187
|
+
run: async () => probeConfig(ctx, {
|
|
188
|
+
existsSync,
|
|
189
|
+
readFileSync: (p, encoding) => readFileSync(p, encoding),
|
|
190
|
+
}),
|
|
191
|
+
},
|
|
192
|
+
{
|
|
193
|
+
name: 'SESSION',
|
|
194
|
+
run: async () => probeSession(ctx, {
|
|
195
|
+
existsSync,
|
|
196
|
+
statSync,
|
|
197
|
+
readFileSync: (p, encoding) => readFileSync(p, encoding),
|
|
198
|
+
}, {
|
|
199
|
+
now,
|
|
200
|
+
...(options.liveSessionId ? { liveSessionId: options.liveSessionId } : {}),
|
|
201
|
+
}),
|
|
202
|
+
},
|
|
203
|
+
// α7 L11 (2026-05-27): DENIAL TRACKING probe. Reports the live
|
|
204
|
+
// session's denial pressure when the REPL adapter wired the
|
|
205
|
+
// tracker through `runDoctorCommand`; degrades к `skipped` for
|
|
206
|
+
// top-level `pugi doctor` calls outside the REPL.
|
|
207
|
+
{
|
|
208
|
+
name: 'DENIAL TRACKING',
|
|
209
|
+
run: async () => probeDenialTracking({
|
|
210
|
+
...(options.denialTracking ? { tracker: options.denialTracking } : {}),
|
|
211
|
+
}),
|
|
212
|
+
},
|
|
213
|
+
// Leak L22 (2026-05-27): BARE MODE row. Always present so the JSON
|
|
214
|
+
// schema stays stable; status flips to `ok` when `--bare` or
|
|
215
|
+
// `PUGI_BARE=1` is active, otherwise `skipped`.
|
|
216
|
+
{
|
|
217
|
+
name: 'BARE MODE',
|
|
218
|
+
run: async () => probeBareMode({ env: ctx.env }),
|
|
219
|
+
},
|
|
220
|
+
// Leak L32 (2026-05-27): PUGI.md HIERARCHY row. Reports how many
|
|
221
|
+
// ambient `PUGI.md` / `CLAUDE.md` files the cwd → homedir walk
|
|
222
|
+
// discovered, and the closest path. `skipped` when bare mode is
|
|
223
|
+
// active (walk disabled) or zero files found.
|
|
224
|
+
// Leak L3 (2026-05-28): SANDBOX row. Reports the platform's available
|
|
225
|
+
// OS-level sandbox primitive (Seatbelt / Landlock / AppContainer) and
|
|
226
|
+
// surfaces a `warning` status until the bash-tool sandbox adapter (#5)
|
|
227
|
+
// is armed. Operator-trust gap visibility — better к flag "not yet
|
|
228
|
+
// jailed" loud than let operators assume it's already on.
|
|
229
|
+
{
|
|
230
|
+
name: 'SANDBOX',
|
|
231
|
+
run: async () => probeSandbox(ctx),
|
|
232
|
+
},
|
|
233
|
+
// Leak L3 (2026-05-28): HOOKS row. Validates `.pugi/hooks-mvp.json`
|
|
234
|
+
// + `.pugi/hook-chains.json` syntax + shape before the first tool
|
|
235
|
+
// dispatch fires. Absence = skipped (most workspaces don't ship
|
|
236
|
+
// hooks); bad JSON = error с remediation hint.
|
|
237
|
+
{
|
|
238
|
+
name: 'HOOKS',
|
|
239
|
+
run: async () => probeHooks(ctx, {
|
|
240
|
+
existsSync,
|
|
241
|
+
readFileSync: (p, encoding) => readFileSync(p, encoding),
|
|
242
|
+
}),
|
|
243
|
+
},
|
|
244
|
+
{
|
|
245
|
+
name: 'PUGI.md HIERARCHY',
|
|
246
|
+
run: async () => probePugiMdHierarchy({
|
|
247
|
+
cwd: ctx.cwd,
|
|
248
|
+
homedir: ctx.home,
|
|
249
|
+
env: ctx.env,
|
|
250
|
+
}),
|
|
251
|
+
},
|
|
252
|
+
];
|
|
253
|
+
return probes;
|
|
254
|
+
}
|
|
255
|
+
/**
|
|
256
|
+
* Run the full doctor sweep + emit the output via the supplied
|
|
257
|
+
* writeOutput sink. Returns the report so REPL callers can route it
|
|
258
|
+
* к the Ink renderer instead of the plain-text fallback.
|
|
259
|
+
*/
|
|
260
|
+
export async function runDoctorCommand(ctx) {
|
|
261
|
+
const probeCtx = {
|
|
262
|
+
cwd: ctx.cwd,
|
|
263
|
+
home: ctx.home,
|
|
264
|
+
env: ctx.env,
|
|
265
|
+
};
|
|
266
|
+
const probes = buildDefaultProbes(probeCtx, {
|
|
267
|
+
...(ctx.liveSessionId ? { liveSessionId: ctx.liveSessionId } : {}),
|
|
268
|
+
...(ctx.denialTracking ? { denialTracking: ctx.denialTracking } : {}),
|
|
269
|
+
});
|
|
270
|
+
const report = await runProbes(probes);
|
|
271
|
+
// Defensive recompute: even though runProbes already computed the
|
|
272
|
+
// overall + counts, recomputing here documents the invariant for the
|
|
273
|
+
// reader and gives the JSON envelope a single source of truth.
|
|
274
|
+
const overall = computeOverall(report.probes);
|
|
275
|
+
const counts = countProbes(report.probes);
|
|
276
|
+
const envelope = {
|
|
277
|
+
command: 'doctor',
|
|
278
|
+
overall,
|
|
279
|
+
counts,
|
|
280
|
+
durationMs: report.durationMs,
|
|
281
|
+
probes: report.probes,
|
|
282
|
+
meta: {
|
|
283
|
+
cliVersion: PUGI_CLI_VERSION,
|
|
284
|
+
nodeVersion: process.version,
|
|
285
|
+
cwd: ctx.cwd,
|
|
286
|
+
},
|
|
287
|
+
};
|
|
288
|
+
const text = renderDoctorTable(envelope);
|
|
289
|
+
ctx.writeOutput(envelope, text);
|
|
290
|
+
process.exitCode = exitCodeFor(overall);
|
|
291
|
+
return { ...report, overall, counts };
|
|
292
|
+
}
|
|
293
|
+
/**
|
|
294
|
+
* Plain-text table renderer. Mirrors the layout from the leak-parity
|
|
295
|
+
* spec but is intentionally column-light (3 columns: NAME / STATUS /
|
|
296
|
+
* DETAIL) so it composes well in narrow terminals without dragging
|
|
297
|
+
* a layout library into the CLI hot path. The Ink TUI renderer in
|
|
298
|
+
* `tui/doctor-table.tsx` is the colour-aware variant used inside the
|
|
299
|
+
* REPL.
|
|
300
|
+
*/
|
|
301
|
+
export function renderDoctorTable(envelope) {
|
|
302
|
+
const NAME_WIDTH = Math.max('NAME'.length, ...envelope.probes.map((row) => row.name.length));
|
|
303
|
+
const STATUS_WIDTH = Math.max('STATUS'.length, ...envelope.probes.map((row) => row.status.length));
|
|
304
|
+
const lines = [];
|
|
305
|
+
lines.push('Pugi Doctor — environment health report');
|
|
306
|
+
lines.push('='.repeat(50));
|
|
307
|
+
lines.push('');
|
|
308
|
+
for (const row of envelope.probes) {
|
|
309
|
+
const namePart = row.name.padEnd(NAME_WIDTH, ' ');
|
|
310
|
+
const statusPart = row.status.toUpperCase().padEnd(STATUS_WIDTH, ' ');
|
|
311
|
+
const latencyPart = typeof row.latencyMs === 'number' ? ` (${row.latencyMs}ms)` : '';
|
|
312
|
+
lines.push(`${namePart} ${statusPart} ${row.detail}${latencyPart}`);
|
|
313
|
+
if (row.remediation && (row.status === 'warn' || row.status === 'error')) {
|
|
314
|
+
lines.push(`${' '.repeat(NAME_WIDTH + STATUS_WIDTH + 4)}→ ${row.remediation}`);
|
|
315
|
+
}
|
|
316
|
+
}
|
|
317
|
+
lines.push('');
|
|
318
|
+
const { ok, warn, error: errorCount, skipped } = envelope.counts;
|
|
319
|
+
const summary = envelope.overall === 'healthy'
|
|
320
|
+
? 'HEALTHY'
|
|
321
|
+
: envelope.overall === 'warning'
|
|
322
|
+
? 'WARNINGS'
|
|
323
|
+
: 'ERRORS';
|
|
324
|
+
lines.push(`${errorCount} error(s), ${warn} warning(s), ${ok} ok, ${skipped} skipped. Overall: ${summary}`);
|
|
325
|
+
lines.push(`CLI ${envelope.meta.cliVersion} Node ${envelope.meta.nodeVersion} cwd ${envelope.meta.cwd}`);
|
|
326
|
+
return lines.join('\n');
|
|
327
|
+
}
|
|
328
|
+
/**
|
|
329
|
+
* Wrap the MCP probe in a dynamic import + try/catch so a missing
|
|
330
|
+
* sibling L13 surface (or a schema mismatch in `core/mcp/registry`)
|
|
331
|
+
* degrades the row к `skipped` instead of breaking the entire sweep.
|
|
332
|
+
* The probe-runner already isolates throws into `error` rows; this
|
|
333
|
+
* wrapper additionally distinguishes "feature not available" from
|
|
334
|
+
* "feature crashed".
|
|
335
|
+
*/
|
|
336
|
+
async function probeMcpSafely(ctx) {
|
|
337
|
+
try {
|
|
338
|
+
const mod = await import('../../core/mcp/registry.js');
|
|
339
|
+
if (typeof mod.loadMcpRegistry !== 'function') {
|
|
340
|
+
return {
|
|
341
|
+
name: 'MCP SERVERS',
|
|
342
|
+
status: 'skipped',
|
|
343
|
+
detail: 'MCP integration not exported by this build',
|
|
344
|
+
};
|
|
345
|
+
}
|
|
346
|
+
return await probeMcp(ctx, {
|
|
347
|
+
loadRegistry: (cwd, options) => mod.loadMcpRegistry(cwd, { connect: options.connect ?? false }),
|
|
348
|
+
});
|
|
349
|
+
}
|
|
350
|
+
catch (error) {
|
|
351
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
352
|
+
return {
|
|
353
|
+
name: 'MCP SERVERS',
|
|
354
|
+
status: 'skipped',
|
|
355
|
+
detail: 'MCP integration not available',
|
|
356
|
+
remediation: `Inspection failed: ${message}`,
|
|
357
|
+
};
|
|
358
|
+
}
|
|
359
|
+
}
|
|
360
|
+
/**
|
|
361
|
+
* Best-effort free-bytes lookup via `df -k <home>`. Parses the second
|
|
362
|
+
* line (header + one data row) and returns the `Available` column ×
|
|
363
|
+
* 1024. Throws on parse failure so the probe surfaces a `warn`
|
|
364
|
+
* instead of a misleading 0-bytes-free verdict.
|
|
365
|
+
*
|
|
366
|
+
* Exported for the spec so we can drive it through a stubbed
|
|
367
|
+
* execFileSync without spawning a real subprocess.
|
|
368
|
+
*/
|
|
369
|
+
export function getFreeBytesViaDf(home) {
|
|
370
|
+
const out = execFileSync('df', ['-k', home], {
|
|
371
|
+
encoding: 'utf8',
|
|
372
|
+
timeout: 2_000,
|
|
373
|
+
stdio: ['ignore', 'pipe', 'ignore'],
|
|
374
|
+
});
|
|
375
|
+
return parseDfOutput(out);
|
|
376
|
+
}
|
|
377
|
+
/**
|
|
378
|
+
* Parse the textual output of `df -k`. Handles both BSD and GNU
|
|
379
|
+
* variants — both emit a `Available` column at index 3 of the data
|
|
380
|
+
* row, with one quirk: long device names wrap к the next line on
|
|
381
|
+
* GNU, so we collapse whitespace + tab newlines first.
|
|
382
|
+
*/
|
|
383
|
+
export function parseDfOutput(out) {
|
|
384
|
+
// Collapse multi-line device-name wraps into a single logical row.
|
|
385
|
+
const collapsed = out.replace(/\n\s+/g, ' ');
|
|
386
|
+
const lines = collapsed
|
|
387
|
+
.split('\n')
|
|
388
|
+
.map((line) => line.trim())
|
|
389
|
+
.filter((line) => line.length > 0);
|
|
390
|
+
if (lines.length < 2) {
|
|
391
|
+
throw new Error(`df output too short: ${JSON.stringify(out.slice(0, 64))}`);
|
|
392
|
+
}
|
|
393
|
+
const data = lines[1].split(/\s+/);
|
|
394
|
+
// Schema: Filesystem 1K-blocks Used Available Capacity Mounted-on
|
|
395
|
+
const availableField = data[3];
|
|
396
|
+
if (!availableField) {
|
|
397
|
+
throw new Error(`df output missing Available column: ${JSON.stringify(lines[1])}`);
|
|
398
|
+
}
|
|
399
|
+
const value = Number(availableField);
|
|
400
|
+
if (!Number.isFinite(value) || value < 0) {
|
|
401
|
+
throw new Error(`df Available column not numeric: ${availableField}`);
|
|
402
|
+
}
|
|
403
|
+
return value * 1024;
|
|
404
|
+
}
|
|
405
|
+
/**
|
|
406
|
+
* Default home dir resolver. Centralised so the CLI handler can call
|
|
407
|
+
* `runDoctorCommand` without re-importing `os.homedir` everywhere.
|
|
408
|
+
*/
|
|
409
|
+
export function defaultHome() {
|
|
410
|
+
return homedir();
|
|
411
|
+
}
|
|
412
|
+
//# sourceMappingURL=doctor.js.map
|
|
@@ -0,0 +1,184 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* `pugi feedback` + `/feedback` slash — Leak L21 (2026-05-27).
|
|
3
|
+
*
|
|
4
|
+
* In-CLI feedback collector. Parity with Claude Code's `/feedback`
|
|
5
|
+
* built-in. The operator never has to leave the terminal to file a
|
|
6
|
+
* bug / feature / general comment / praise. The wizard collects:
|
|
7
|
+
*
|
|
8
|
+
* 1. category (bug / feature / general / praise)
|
|
9
|
+
* 2. rating (1-5)
|
|
10
|
+
* 3. comment (multi-line free text)
|
|
11
|
+
* 4. optional redacted session context (last 5 turns)
|
|
12
|
+
* 5. confirm (final y/n)
|
|
13
|
+
*
|
|
14
|
+
* # Module contract
|
|
15
|
+
*
|
|
16
|
+
* - This file owns the WIRING from the CLI surface (TTY mount,
|
|
17
|
+
* non-TTY JSON, slash dispatcher) to the queue + submitter
|
|
18
|
+
* modules. The corpus + redactor + queue persistence live in
|
|
19
|
+
* `core/feedback/{queue.ts,submitter.ts}`. The Ink prompt lives
|
|
20
|
+
* in `tui/feedback-prompt.tsx`. Both have zero coupling to the
|
|
21
|
+
* CLI dispatch surface.
|
|
22
|
+
*
|
|
23
|
+
* - `runFeedbackCommand` is the single entry point. Both the top-
|
|
24
|
+
* level `pugi feedback` handler в `runtime/cli.ts` AND the in-REPL
|
|
25
|
+
* `/feedback` slash dispatcher call it. The function returns the
|
|
26
|
+
* resolved `FeedbackRunResult` so the slash dispatcher can route
|
|
27
|
+
* the outcome message to the REPL's system pane without re-prompting.
|
|
28
|
+
*
|
|
29
|
+
* - Exit code is ALWAYS 0. Feedback is a brand surface — never a
|
|
30
|
+
* gate. Failures land as result variants; the wrapper never
|
|
31
|
+
* turns a network blip into a non-zero shell exit.
|
|
32
|
+
*
|
|
33
|
+
* - The random-source-style test seam: the run helper accepts an
|
|
34
|
+
* `interactive` flag that the spec sets to false, plus an injected
|
|
35
|
+
* `draft` so unit tests can drive the submit + queue branches
|
|
36
|
+
* without mounting Ink.
|
|
37
|
+
*/
|
|
38
|
+
import { existsSync, readFileSync } from 'node:fs';
|
|
39
|
+
import { resolve } from 'node:path';
|
|
40
|
+
import { enqueueFeedback, feedbackQueuePath, flushFeedbackQueue, } from '../../core/feedback/queue.js';
|
|
41
|
+
import { feedbackSubmitUrl, redactSessionContext, submitFeedback, } from '../../core/feedback/submitter.js';
|
|
42
|
+
/**
|
|
43
|
+
* Drive one feedback round. The function is async because of the
|
|
44
|
+
* submit round-trip, but everything else (queue write, redaction) is
|
|
45
|
+
* sync — no surprise concurrency.
|
|
46
|
+
*/
|
|
47
|
+
export async function runFeedbackCommand(ctx) {
|
|
48
|
+
if (ctx.draft == null) {
|
|
49
|
+
return { kind: 'cancelled' };
|
|
50
|
+
}
|
|
51
|
+
const envelope = {
|
|
52
|
+
category: ctx.draft.category,
|
|
53
|
+
rating: ctx.draft.rating,
|
|
54
|
+
comment: ctx.draft.comment,
|
|
55
|
+
ts: new Date().toISOString(),
|
|
56
|
+
cliVersion: ctx.cliVersion,
|
|
57
|
+
...(ctx.tier ? { tier: ctx.tier } : {}),
|
|
58
|
+
};
|
|
59
|
+
if (ctx.draft.includeSessionContext && ctx.sessionContext) {
|
|
60
|
+
const sc = ctx.sessionContext();
|
|
61
|
+
if (sc)
|
|
62
|
+
envelope.sessionContext = sc;
|
|
63
|
+
}
|
|
64
|
+
let result;
|
|
65
|
+
try {
|
|
66
|
+
result = await ctx.submit(envelope);
|
|
67
|
+
}
|
|
68
|
+
catch (err) {
|
|
69
|
+
// Defensive: a thrown submitter (should not happen — the live
|
|
70
|
+
// submitter catches everything) is treated as transient so the
|
|
71
|
+
// envelope lands in the queue.
|
|
72
|
+
const reason = err instanceof Error ? err.message : String(err);
|
|
73
|
+
result = { kind: 'transient', reason };
|
|
74
|
+
}
|
|
75
|
+
if (result.kind === 'ok') {
|
|
76
|
+
return { kind: 'submitted', envelope, httpStatus: result.httpStatus };
|
|
77
|
+
}
|
|
78
|
+
if (result.kind === 'transient') {
|
|
79
|
+
const path = enqueueFeedback(envelope, ctx.cwd);
|
|
80
|
+
return { kind: 'queued', envelope, path, reason: result.reason };
|
|
81
|
+
}
|
|
82
|
+
// permanent — log + drop
|
|
83
|
+
return {
|
|
84
|
+
kind: 'dropped',
|
|
85
|
+
envelope,
|
|
86
|
+
reason: result.reason,
|
|
87
|
+
httpStatus: result.httpStatus,
|
|
88
|
+
};
|
|
89
|
+
}
|
|
90
|
+
/**
|
|
91
|
+
* Render one human-readable toast for the operator. Centralised so the
|
|
92
|
+
* top-level `pugi feedback` shell handler + the in-REPL `/feedback`
|
|
93
|
+
* slash dispatcher agree on the copy.
|
|
94
|
+
*/
|
|
95
|
+
export function renderFeedbackToast(result) {
|
|
96
|
+
switch (result.kind) {
|
|
97
|
+
case 'submitted':
|
|
98
|
+
return 'Feedback submitted. Thank you.';
|
|
99
|
+
case 'queued':
|
|
100
|
+
return `Feedback queued locally. Will sync on next online run. (${result.path})`;
|
|
101
|
+
case 'cancelled':
|
|
102
|
+
return 'Feedback cancelled. Nothing was sent.';
|
|
103
|
+
case 'dropped':
|
|
104
|
+
return `Feedback rejected by server (${result.httpStatus}): ${result.reason}. Not queued.`;
|
|
105
|
+
case 'noop':
|
|
106
|
+
return `No feedback collected: ${result.reason}`;
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
/**
|
|
110
|
+
* Background queue flush. Invoked silently on session start so any
|
|
111
|
+
* envelopes that landed during an offline run get drained when the
|
|
112
|
+
* operator next has connectivity. The function never throws — it
|
|
113
|
+
* returns the flush stats so the caller can log them at debug level.
|
|
114
|
+
*/
|
|
115
|
+
export async function flushFeedbackQueueSilently(cwd, config) {
|
|
116
|
+
// Short-circuit when the queue file does not exist. Avoids a
|
|
117
|
+
// pointless `fs.stat` round-trip on every cold session start.
|
|
118
|
+
if (!existsSync(feedbackQueuePath(cwd))) {
|
|
119
|
+
return { attempted: 0, succeeded: 0, failed: 0 };
|
|
120
|
+
}
|
|
121
|
+
const result = await flushFeedbackQueue(cwd, async (env) => {
|
|
122
|
+
const r = await submitFeedback(env, config);
|
|
123
|
+
if (r.kind === 'ok')
|
|
124
|
+
return true;
|
|
125
|
+
if (r.kind === 'permanent') {
|
|
126
|
+
// Permanent failures are "done" from the queue's POV — they
|
|
127
|
+
// would never resolve on retry. Drop them so the queue does
|
|
128
|
+
// not grow without bound.
|
|
129
|
+
return true;
|
|
130
|
+
}
|
|
131
|
+
return false;
|
|
132
|
+
});
|
|
133
|
+
return {
|
|
134
|
+
attempted: result.attempted,
|
|
135
|
+
succeeded: result.succeeded,
|
|
136
|
+
failed: result.failed,
|
|
137
|
+
};
|
|
138
|
+
}
|
|
139
|
+
/**
|
|
140
|
+
* Re-exports — the spec imports these via the command module so the
|
|
141
|
+
* dependency graph in the test stays single-rooted.
|
|
142
|
+
*/
|
|
143
|
+
export { feedbackQueuePath, feedbackSubmitUrl, redactSessionContext, submitFeedback, };
|
|
144
|
+
/**
|
|
145
|
+
* Read the persona conversation log if present. Best-effort: returns
|
|
146
|
+
* an empty list when the file is missing or malformed. The CLI's REPL
|
|
147
|
+
* persists transcripts via the session module at a canonical relative
|
|
148
|
+
* path under `.pugi/sessions/`. The shell-level `pugi feedback` does
|
|
149
|
+
* not have access to a live session, so it tries to pick up the most
|
|
150
|
+
* recent persisted one for the `--with-context` path.
|
|
151
|
+
*
|
|
152
|
+
* Intentionally tolerant — feedback works even with no transcript.
|
|
153
|
+
*/
|
|
154
|
+
export function readMostRecentTranscript(cwd, options = {}) {
|
|
155
|
+
// The CLI may persist sessions in several places depending on the
|
|
156
|
+
// surface. We probe the conventional default; the spec drives the
|
|
157
|
+
// function via a fixture file instead of a live REPL.
|
|
158
|
+
const candidate = resolve(cwd, '.pugi', 'sessions', 'latest.jsonl');
|
|
159
|
+
if (!existsSync(candidate))
|
|
160
|
+
return [];
|
|
161
|
+
try {
|
|
162
|
+
const text = readFileSync(candidate, 'utf8');
|
|
163
|
+
const lines = text.split('\n').filter((l) => l.trim().length > 0);
|
|
164
|
+
const turns = [];
|
|
165
|
+
for (const line of lines) {
|
|
166
|
+
try {
|
|
167
|
+
const obj = JSON.parse(line);
|
|
168
|
+
if ((obj.role === 'user' || obj.role === 'assistant' || obj.role === 'system')
|
|
169
|
+
&& typeof obj.text === 'string') {
|
|
170
|
+
turns.push({ role: obj.role, text: obj.text });
|
|
171
|
+
}
|
|
172
|
+
}
|
|
173
|
+
catch {
|
|
174
|
+
// skip malformed line
|
|
175
|
+
}
|
|
176
|
+
}
|
|
177
|
+
const cap = options.maxTurns ?? 5;
|
|
178
|
+
return turns.slice(-cap);
|
|
179
|
+
}
|
|
180
|
+
catch {
|
|
181
|
+
return [];
|
|
182
|
+
}
|
|
183
|
+
}
|
|
184
|
+
//# sourceMappingURL=feedback.js.map
|