@pugi/cli 0.1.0-beta.17 → 0.1.0-beta.19
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/dist/core/compact/auto-trigger.js +96 -0
- package/dist/core/compact/buffer-rewriter.js +115 -0
- package/dist/core/compact/summarizer.js +196 -0
- package/dist/core/compact/token-counter.js +108 -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/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/session.js +74 -0
- package/dist/core/diagnostics/probes/status-snapshot.js +442 -0
- package/dist/core/diagnostics/probes/workspace.js +63 -0
- package/dist/core/diagnostics/types.js +70 -0
- package/dist/core/engine/native-pugi.js +20 -0
- package/dist/core/engine/strip-internal-fields.js +124 -0
- package/dist/core/engine/tool-bridge.js +251 -49
- package/dist/core/file-cache.js +113 -1
- package/dist/core/mcp/client.js +66 -6
- package/dist/core/mcp/registry.js +24 -2
- package/dist/core/permissions/gate.js +187 -0
- package/dist/core/permissions/index.js +18 -0
- package/dist/core/permissions/mode.js +102 -0
- package/dist/core/permissions/state.js +160 -0
- package/dist/core/permissions/tool-class.js +93 -0
- package/dist/core/repl/session.js +261 -9
- package/dist/core/repl/slash-commands.js +67 -4
- package/dist/runtime/cli.js +153 -58
- package/dist/runtime/commands/compact.js +296 -0
- package/dist/runtime/commands/doctor.js +369 -0
- package/dist/runtime/commands/mcp.js +290 -3
- package/dist/runtime/commands/permissions.js +87 -0
- package/dist/runtime/commands/status.js +178 -0
- package/dist/runtime/version.js +1 -1
- package/dist/tools/agent-tool.js +18 -4
- package/dist/tools/ask-user-question.js +213 -0
- package/dist/tools/file-tools.js +57 -14
- package/dist/tools/registry.js +7 -0
- package/dist/tui/ask-user-question-prompt.js +192 -0
- package/dist/tui/compact-banner.js +54 -0
- package/dist/tui/conversation-pane.js +68 -7
- package/dist/tui/doctor-table.js +31 -0
- package/dist/tui/status-table.js +7 -0
- package/package.json +2 -2
|
@@ -0,0 +1,264 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* α7 L11 (2026-05-27) — DenialTrackingState surface (Claude Code parity).
|
|
3
|
+
*
|
|
4
|
+
* Per the openclaude / anarchic-CC leak research (`gap-analysis-3-repos`
|
|
5
|
+
* §5.2): Claude Code's `QueryEngine.ts` maintains a per-session
|
|
6
|
+
* `DenialTrackingState` that records every tool-dispatch denial.
|
|
7
|
+
* Subsequent turns receive a compact reminder so the model does not
|
|
8
|
+
* loop on the same refused operation, the operator can audit pressure
|
|
9
|
+
* points via `/permissions denials` / `/doctor`, and hooks can react
|
|
10
|
+
* to recurring patterns.
|
|
11
|
+
*
|
|
12
|
+
* Pugi pre-L11 surfaced denials only as a one-line sentinel inside the
|
|
13
|
+
* tool error (`HOOK_BLOCKED:`, `PLAN_MODE_REFUSED:`, `STALE_READ:`).
|
|
14
|
+
* The model saw each refusal in isolation; no aggregate, no count, no
|
|
15
|
+
* "do not retry" context across turns. This module closes that gap.
|
|
16
|
+
*
|
|
17
|
+
* Design contract:
|
|
18
|
+
*
|
|
19
|
+
* - Per-session in-memory state. The engine adapter creates one
|
|
20
|
+
* instance at session open and threads it through `buildExecutor`
|
|
21
|
+
* + `runEngineLoop`. No disk persistence — denial history is a
|
|
22
|
+
* turn-scoped signal, not a forensic log (the audit ledger at
|
|
23
|
+
* `.pugi/events.jsonl` already carries every refused dispatch).
|
|
24
|
+
*
|
|
25
|
+
* - Pure: no I/O, no logging. Inputs are passed by caller, output
|
|
26
|
+
* is in-memory only. The diagnostics probe + system-reminder
|
|
27
|
+
* splice are pure functions over the snapshot.
|
|
28
|
+
*
|
|
29
|
+
* - Canonical argHash: sha256 of `canonicalize(args)`. Same call
|
|
30
|
+
* to same tool with same args = same key. Reordering JSON object
|
|
31
|
+
* keys does NOT change the hash (we sort keys recursively before
|
|
32
|
+
* stringification). This makes "repeated denial" detection
|
|
33
|
+
* deterministic across model retries.
|
|
34
|
+
*
|
|
35
|
+
* - Bounded: the map caps at 256 entries per session. Once full,
|
|
36
|
+
* `recordDenial` evicts the oldest entry (lowest `lastDeniedAt`).
|
|
37
|
+
* A session with > 256 distinct denials is already pathological;
|
|
38
|
+
* the cap is defensive insurance against pathological loops
|
|
39
|
+
* blowing memory.
|
|
40
|
+
*
|
|
41
|
+
* - Reminder threshold: the reminder splice only injects when at
|
|
42
|
+
* least one record has `count >= 2`. One-off denials are normal
|
|
43
|
+
* (the operator is shaping the session); repeated denials = the
|
|
44
|
+
* model is failing to learn from the refusal and we should
|
|
45
|
+
* reinforce it.
|
|
46
|
+
*/
|
|
47
|
+
import { createHash } from 'node:crypto';
|
|
48
|
+
/** Bound on the per-session map. See module header. */
|
|
49
|
+
export const DENIAL_TRACKING_MAX_ENTRIES = 256;
|
|
50
|
+
/** Threshold above which `buildDenialContext` injects a system reminder. */
|
|
51
|
+
export const DENIAL_REMINDER_THRESHOLD = 2;
|
|
52
|
+
/** Per-denial-args summary cap surfaced to the model. */
|
|
53
|
+
export const DENIAL_ARGS_SUMMARY_BYTES = 240;
|
|
54
|
+
/**
|
|
55
|
+
* Per-session denial tracker. Instances are mutable but threaded
|
|
56
|
+
* through `buildExecutor` so every dispatcher in the loop sees the
|
|
57
|
+
* same view.
|
|
58
|
+
*/
|
|
59
|
+
export class DenialTrackingState {
|
|
60
|
+
/** Insertion-ordered map. Eviction walks oldest `lastDeniedAt`. */
|
|
61
|
+
records = new Map();
|
|
62
|
+
/**
|
|
63
|
+
* Wall clock. Defaults to `Date.now`. Injected so tests can drive
|
|
64
|
+
* deterministic timestamps.
|
|
65
|
+
*/
|
|
66
|
+
clock;
|
|
67
|
+
constructor(options = {}) {
|
|
68
|
+
this.clock = options.clock ?? Date.now;
|
|
69
|
+
}
|
|
70
|
+
/**
|
|
71
|
+
* Record a denial. Increments the count when the (tool, args) pair
|
|
72
|
+
* has been denied before; otherwise inserts a new record. The reason
|
|
73
|
+
* is always overwritten with the latest value so the system reminder
|
|
74
|
+
* surfaces the most-recent context.
|
|
75
|
+
*/
|
|
76
|
+
recordDenial(toolName, args, reason) {
|
|
77
|
+
const argHash = canonicalArgHash(args);
|
|
78
|
+
const key = `${toolName}:${argHash}`;
|
|
79
|
+
const now = new Date(this.clock());
|
|
80
|
+
const existing = this.records.get(key);
|
|
81
|
+
if (existing) {
|
|
82
|
+
existing.count += 1;
|
|
83
|
+
existing.lastDeniedAt = now;
|
|
84
|
+
existing.lastReason = truncateReason(reason);
|
|
85
|
+
// Re-insert so the iteration order tracks lastDeniedAt (LRU-ish).
|
|
86
|
+
this.records.delete(key);
|
|
87
|
+
this.records.set(key, existing);
|
|
88
|
+
return existing;
|
|
89
|
+
}
|
|
90
|
+
if (this.records.size >= DENIAL_TRACKING_MAX_ENTRIES) {
|
|
91
|
+
// Evict the oldest entry (insertion order = LRU after our delete
|
|
92
|
+
// + set pattern above). Map iteration order is insertion order
|
|
93
|
+
// in JS; the first key is the LRU one.
|
|
94
|
+
const oldestKey = this.records.keys().next().value;
|
|
95
|
+
if (oldestKey !== undefined) {
|
|
96
|
+
this.records.delete(oldestKey);
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
const record = {
|
|
100
|
+
key,
|
|
101
|
+
toolName,
|
|
102
|
+
argsSummary: summariseArgs(args),
|
|
103
|
+
firstDeniedAt: now,
|
|
104
|
+
lastDeniedAt: now,
|
|
105
|
+
count: 1,
|
|
106
|
+
lastReason: truncateReason(reason),
|
|
107
|
+
};
|
|
108
|
+
this.records.set(key, record);
|
|
109
|
+
return record;
|
|
110
|
+
}
|
|
111
|
+
/**
|
|
112
|
+
* Snapshot of every recorded denial. Returned in insertion order
|
|
113
|
+
* (LRU-ish: most-recently-touched at the tail). The caller MUST
|
|
114
|
+
* treat the array as read-only — mutating it does not affect state.
|
|
115
|
+
*/
|
|
116
|
+
getDenialHistory() {
|
|
117
|
+
return Array.from(this.records.values()).map((r) => ({ ...r }));
|
|
118
|
+
}
|
|
119
|
+
/** Lookup the record for a specific (tool, args) pair, or undefined. */
|
|
120
|
+
getPatternFor(toolName, args) {
|
|
121
|
+
const argHash = canonicalArgHash(args);
|
|
122
|
+
const key = `${toolName}:${argHash}`;
|
|
123
|
+
const record = this.records.get(key);
|
|
124
|
+
return record ? { ...record } : undefined;
|
|
125
|
+
}
|
|
126
|
+
/**
|
|
127
|
+
* Aggregate counts — handy for `/doctor`, `/permissions denials`,
|
|
128
|
+
* and the system-reminder threshold check.
|
|
129
|
+
*/
|
|
130
|
+
summary() {
|
|
131
|
+
let totalDenials = 0;
|
|
132
|
+
let repeatedPatterns = 0;
|
|
133
|
+
for (const record of this.records.values()) {
|
|
134
|
+
totalDenials += record.count;
|
|
135
|
+
if (record.count >= DENIAL_REMINDER_THRESHOLD)
|
|
136
|
+
repeatedPatterns += 1;
|
|
137
|
+
}
|
|
138
|
+
return {
|
|
139
|
+
totalDenials,
|
|
140
|
+
uniquePatterns: this.records.size,
|
|
141
|
+
repeatedPatterns,
|
|
142
|
+
};
|
|
143
|
+
}
|
|
144
|
+
/** Drop every recorded denial. Called on session end. */
|
|
145
|
+
clear() {
|
|
146
|
+
this.records.clear();
|
|
147
|
+
}
|
|
148
|
+
/** Number of distinct (tool, args) pairs currently tracked. */
|
|
149
|
+
size() {
|
|
150
|
+
return this.records.size;
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
/**
|
|
154
|
+
* Build the operator-facing system-reminder block. Returns an empty
|
|
155
|
+
* string when no record meets the threshold — the caller short-circuits
|
|
156
|
+
* the splice in that case.
|
|
157
|
+
*
|
|
158
|
+
* Rendered shape:
|
|
159
|
+
*
|
|
160
|
+
* <denial-context>
|
|
161
|
+
* - Tool `bash` denied 3 times in this session.
|
|
162
|
+
* Last reason: HOOK_BLOCKED: PreToolUse refused (exit=1).
|
|
163
|
+
* Args: {"command":"rm -rf node_modules"}
|
|
164
|
+
* Do not retry without operator intervention.
|
|
165
|
+
* - Tool `edit` denied 2 times in this session.
|
|
166
|
+
* Last reason: STALE_READ: file changed since last read.
|
|
167
|
+
* Args: {"path":"src/index.ts","oldString":"..."}
|
|
168
|
+
* Do not retry without operator intervention.
|
|
169
|
+
* </denial-context>
|
|
170
|
+
*
|
|
171
|
+
* Determinism: same snapshot always produces the same block. We sort
|
|
172
|
+
* entries by `count` descending then `lastDeniedAt` descending so the
|
|
173
|
+
* most-pressing patterns surface first.
|
|
174
|
+
*/
|
|
175
|
+
export function buildDenialContext(state) {
|
|
176
|
+
const records = state.getDenialHistory()
|
|
177
|
+
.filter((r) => r.count >= DENIAL_REMINDER_THRESHOLD)
|
|
178
|
+
.sort((a, b) => {
|
|
179
|
+
if (b.count !== a.count)
|
|
180
|
+
return b.count - a.count;
|
|
181
|
+
return b.lastDeniedAt.getTime() - a.lastDeniedAt.getTime();
|
|
182
|
+
});
|
|
183
|
+
if (records.length === 0)
|
|
184
|
+
return '';
|
|
185
|
+
const lines = ['<denial-context>'];
|
|
186
|
+
for (const record of records) {
|
|
187
|
+
lines.push(` - Tool \`${record.toolName}\` denied ${record.count} times in this session.`);
|
|
188
|
+
lines.push(` Last reason: ${record.lastReason}`);
|
|
189
|
+
lines.push(` Args: ${record.argsSummary}`);
|
|
190
|
+
lines.push(' Do not retry without operator intervention.');
|
|
191
|
+
}
|
|
192
|
+
lines.push('</denial-context>');
|
|
193
|
+
return lines.join('\n');
|
|
194
|
+
}
|
|
195
|
+
/**
|
|
196
|
+
* Canonical sha256 of the args. Object keys are sorted recursively
|
|
197
|
+
* before stringification so `{a:1,b:2}` and `{b:2,a:1}` hash to the
|
|
198
|
+
* same value. Undefined values + functions are stripped (JSON.stringify
|
|
199
|
+
* already does this); circular refs throw — the caller is expected to
|
|
200
|
+
* pass plain JSON objects (every Pugi tool argument set is a JSON
|
|
201
|
+
* object by contract).
|
|
202
|
+
*
|
|
203
|
+
* Returns the full 64-char hex digest. The denial key prepends the
|
|
204
|
+
* tool name (`<tool>:<hash>`) so a hash collision across tools cannot
|
|
205
|
+
* confuse the matcher.
|
|
206
|
+
*/
|
|
207
|
+
export function canonicalArgHash(args) {
|
|
208
|
+
const canonical = canonicalize(args);
|
|
209
|
+
const hash = createHash('sha256');
|
|
210
|
+
hash.update(canonical);
|
|
211
|
+
return hash.digest('hex');
|
|
212
|
+
}
|
|
213
|
+
function canonicalize(value) {
|
|
214
|
+
if (value === null)
|
|
215
|
+
return 'null';
|
|
216
|
+
if (value === undefined)
|
|
217
|
+
return 'null';
|
|
218
|
+
const t = typeof value;
|
|
219
|
+
if (t === 'string')
|
|
220
|
+
return JSON.stringify(value);
|
|
221
|
+
if (t === 'number' || t === 'boolean')
|
|
222
|
+
return JSON.stringify(value);
|
|
223
|
+
if (t === 'bigint')
|
|
224
|
+
return JSON.stringify(value.toString());
|
|
225
|
+
if (Array.isArray(value)) {
|
|
226
|
+
return `[${value.map((entry) => canonicalize(entry)).join(',')}]`;
|
|
227
|
+
}
|
|
228
|
+
if (t === 'object') {
|
|
229
|
+
const obj = value;
|
|
230
|
+
const keys = Object.keys(obj).sort();
|
|
231
|
+
const parts = keys.map((k) => `${JSON.stringify(k)}:${canonicalize(obj[k])}`);
|
|
232
|
+
return `{${parts.join(',')}}`;
|
|
233
|
+
}
|
|
234
|
+
// Functions / symbols — neither should reach here for tool args.
|
|
235
|
+
// Stringify defensively so the hash is stable.
|
|
236
|
+
return JSON.stringify(String(value));
|
|
237
|
+
}
|
|
238
|
+
/**
|
|
239
|
+
* Short, operator-readable preview of the args object. Cap so a
|
|
240
|
+
* 32 KB write payload does not blow up the reminder block. Falls back
|
|
241
|
+
* к `[non-JSON args]` when stringification fails (e.g. a circular ref
|
|
242
|
+
* snuck through — unlikely for tool args by contract).
|
|
243
|
+
*/
|
|
244
|
+
function summariseArgs(args) {
|
|
245
|
+
let stringified;
|
|
246
|
+
try {
|
|
247
|
+
stringified = JSON.stringify(args);
|
|
248
|
+
}
|
|
249
|
+
catch {
|
|
250
|
+
return '[non-JSON args]';
|
|
251
|
+
}
|
|
252
|
+
if (stringified === undefined)
|
|
253
|
+
return '{}';
|
|
254
|
+
if (stringified.length <= DENIAL_ARGS_SUMMARY_BYTES)
|
|
255
|
+
return stringified;
|
|
256
|
+
return `${stringified.slice(0, DENIAL_ARGS_SUMMARY_BYTES)}…`;
|
|
257
|
+
}
|
|
258
|
+
/** Cap reason at a sane length so a hook script's stdout cannot flood the reminder. */
|
|
259
|
+
function truncateReason(reason, cap = 240) {
|
|
260
|
+
if (reason.length <= cap)
|
|
261
|
+
return reason;
|
|
262
|
+
return `${reason.slice(0, cap)}…`;
|
|
263
|
+
}
|
|
264
|
+
//# sourceMappingURL=state.js.map
|
|
@@ -0,0 +1,93 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Probe runner — orchestrates a set of diagnostic probes in parallel
|
|
3
|
+
* with per-probe fail-isolation + a global wall-clock budget.
|
|
4
|
+
*
|
|
5
|
+
* Design contract:
|
|
6
|
+
*
|
|
7
|
+
* - Probes are independent: one probe's throw or timeout NEVER stops
|
|
8
|
+
* the others. The runner wraps each call in a try/catch + a
|
|
9
|
+
* `Promise.race` against a timeout sentinel.
|
|
10
|
+
*
|
|
11
|
+
* - Order preservation: the returned `probes[]` array preserves the
|
|
12
|
+
* input order so the table renderer always lists rows in the same
|
|
13
|
+
* sequence (operators rely on muscle memory: "auth is the first
|
|
14
|
+
* row, api is the second").
|
|
15
|
+
*
|
|
16
|
+
* - No I/O ownership: the runner owns ZERO file or network calls.
|
|
17
|
+
* Every external dependency is injected via the probe function
|
|
18
|
+
* itself. This keeps the test surface minimal and verifies the
|
|
19
|
+
* fail-isolation contract without spinning real subprocesses.
|
|
20
|
+
*
|
|
21
|
+
* - Crashes attributed to the probe: a probe that throws maps to a
|
|
22
|
+
* synthetic `error` ProbeResult with the probe name + the error
|
|
23
|
+
* message. A probe that exceeds the timeout becomes a synthetic
|
|
24
|
+
* `warn` (timing out is asymmetric — the host is reachable but
|
|
25
|
+
* slow).
|
|
26
|
+
*
|
|
27
|
+
* - The aggregate `DoctorReport` shape comes straight from
|
|
28
|
+
* `types.ts` so the doctor command + the Ink table renderer both
|
|
29
|
+
* consume the same struct without any glue layer.
|
|
30
|
+
*/
|
|
31
|
+
import { computeOverall, countProbes, } from './types.js';
|
|
32
|
+
/**
|
|
33
|
+
* Run a set of probes in parallel and produce a structured report.
|
|
34
|
+
* Never throws — every failure mode maps to a structured ProbeResult.
|
|
35
|
+
*/
|
|
36
|
+
export async function runProbes(probes, options = {}) {
|
|
37
|
+
const now = options.now ?? Date.now;
|
|
38
|
+
const defaultTimeoutMs = options.defaultTimeoutMs ?? 5_000;
|
|
39
|
+
const startedAt = now();
|
|
40
|
+
// Each probe is wrapped in a fail-isolation envelope.
|
|
41
|
+
const results = await Promise.all(probes.map(async (entry) => runOne(entry, defaultTimeoutMs)));
|
|
42
|
+
const overall = computeOverall(results);
|
|
43
|
+
const counts = countProbes(results);
|
|
44
|
+
const durationMs = now() - startedAt;
|
|
45
|
+
return {
|
|
46
|
+
probes: results,
|
|
47
|
+
overall,
|
|
48
|
+
counts,
|
|
49
|
+
durationMs,
|
|
50
|
+
};
|
|
51
|
+
}
|
|
52
|
+
async function runOne(entry, defaultTimeoutMs) {
|
|
53
|
+
const budget = entry.timeoutMs ?? defaultTimeoutMs;
|
|
54
|
+
try {
|
|
55
|
+
return await raceWithTimeout(entry, budget);
|
|
56
|
+
}
|
|
57
|
+
catch (error) {
|
|
58
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
59
|
+
if (error instanceof ProbeTimeoutError) {
|
|
60
|
+
return {
|
|
61
|
+
name: entry.name,
|
|
62
|
+
status: 'warn',
|
|
63
|
+
detail: `Probe timed out after ${budget}ms`,
|
|
64
|
+
remediation: 'Re-run later; the host or registry may be slow',
|
|
65
|
+
};
|
|
66
|
+
}
|
|
67
|
+
return {
|
|
68
|
+
name: entry.name,
|
|
69
|
+
status: 'error',
|
|
70
|
+
detail: `Probe crashed: ${message}`,
|
|
71
|
+
};
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
class ProbeTimeoutError extends Error {
|
|
75
|
+
constructor(name, budgetMs) {
|
|
76
|
+
super(`Probe ${name} exceeded ${budgetMs}ms`);
|
|
77
|
+
this.name = 'ProbeTimeoutError';
|
|
78
|
+
}
|
|
79
|
+
}
|
|
80
|
+
async function raceWithTimeout(entry, budgetMs) {
|
|
81
|
+
let timer;
|
|
82
|
+
const timeoutPromise = new Promise((_, reject) => {
|
|
83
|
+
timer = setTimeout(() => reject(new ProbeTimeoutError(entry.name, budgetMs)), budgetMs);
|
|
84
|
+
});
|
|
85
|
+
try {
|
|
86
|
+
return await Promise.race([entry.run(), timeoutPromise]);
|
|
87
|
+
}
|
|
88
|
+
finally {
|
|
89
|
+
if (timer !== undefined)
|
|
90
|
+
clearTimeout(timer);
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
//# sourceMappingURL=probe-runner.js.map
|
|
@@ -0,0 +1,46 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* API probe — verifies `api.pugi.io` (or the active apiUrl) is
|
|
3
|
+
* reachable WITHOUT requiring a valid auth token. The auth probe
|
|
4
|
+
* covers the credentialed path; this probe answers the orthogonal
|
|
5
|
+
* "is the network broken" question so the operator can disambiguate
|
|
6
|
+
* "I'm offline" from "my token is bad" without reading two probe
|
|
7
|
+
* details together.
|
|
8
|
+
*
|
|
9
|
+
* Success criteria: any HTTP response (including 401 unauthenticated)
|
|
10
|
+
* proves the host is reachable. A thrown fetch (DNS / TCP / TLS
|
|
11
|
+
* failure) is the only true failure mode.
|
|
12
|
+
*/
|
|
13
|
+
export async function probeApi(ctx, deps) {
|
|
14
|
+
const apiUrl = deps.resolveApiUrl(ctx.env);
|
|
15
|
+
const startedAt = deps.now();
|
|
16
|
+
try {
|
|
17
|
+
const response = await deps.fetchImpl(`${stripTrailingSlash(apiUrl)}/api/pugi/health`, {
|
|
18
|
+
method: 'GET',
|
|
19
|
+
});
|
|
20
|
+
const latencyMs = deps.now() - startedAt;
|
|
21
|
+
// Any HTTP response confirms the host is reachable. The auth probe
|
|
22
|
+
// is responsible for verdicting the 401 / 200 split — here we just
|
|
23
|
+
// confirm we can talk to the server.
|
|
24
|
+
return {
|
|
25
|
+
name: 'API',
|
|
26
|
+
status: 'ok',
|
|
27
|
+
detail: `${apiUrl} reachable (${response.status})`,
|
|
28
|
+
latencyMs,
|
|
29
|
+
};
|
|
30
|
+
}
|
|
31
|
+
catch (error) {
|
|
32
|
+
const latencyMs = deps.now() - startedAt;
|
|
33
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
34
|
+
return {
|
|
35
|
+
name: 'API',
|
|
36
|
+
status: 'error',
|
|
37
|
+
detail: `Cannot reach ${apiUrl}`,
|
|
38
|
+
latencyMs,
|
|
39
|
+
remediation: `Check network or override with PUGI_API_URL: ${message}`,
|
|
40
|
+
};
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
function stripTrailingSlash(url) {
|
|
44
|
+
return url.endsWith('/') ? url.slice(0, -1) : url;
|
|
45
|
+
}
|
|
46
|
+
//# sourceMappingURL=api.js.map
|
|
@@ -0,0 +1,86 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* AUTH probe — verifies the active credential resolves to a working
|
|
3
|
+
* Bearer token by calling `GET /api/pugi/health` with it.
|
|
4
|
+
*
|
|
5
|
+
* Failure modes (deterministic mapping):
|
|
6
|
+
* - no credential found in env or `~/.pugi/credentials.json`
|
|
7
|
+
* → status `error`, remediation = `pugi login`
|
|
8
|
+
* - credential exists but server returns 401/403
|
|
9
|
+
* → status `error`, remediation = `pugi login` (token expired/revoked)
|
|
10
|
+
* - credential exists but server returns 5xx OR network fails
|
|
11
|
+
* → status `warn` (server-side; don't blame the operator)
|
|
12
|
+
* - credential exists and server returns 200
|
|
13
|
+
* → status `ok` (latency captured)
|
|
14
|
+
*
|
|
15
|
+
* NOTE: the probe must NEVER log the token itself. Memory hit
|
|
16
|
+
* `feedback_no_claude_attribution_anywhere_hard_rule` plus the CSO
|
|
17
|
+
* sweep on bearer leaks (history: PR-AGENT-MERGE-GATE 2026-05-15)
|
|
18
|
+
* frame why this is enforced at the probe layer.
|
|
19
|
+
*/
|
|
20
|
+
export async function probeAuth(ctx, deps) {
|
|
21
|
+
const credential = deps.resolveCredential(ctx.env, ctx.home);
|
|
22
|
+
if (!credential) {
|
|
23
|
+
return {
|
|
24
|
+
name: 'AUTH',
|
|
25
|
+
status: 'error',
|
|
26
|
+
detail: 'No credential — no PUGI_API_KEY env and no ~/.pugi/credentials.json',
|
|
27
|
+
remediation: 'Run `pugi login` to authenticate',
|
|
28
|
+
};
|
|
29
|
+
}
|
|
30
|
+
const startedAt = deps.now();
|
|
31
|
+
let response;
|
|
32
|
+
try {
|
|
33
|
+
response = await deps.fetchImpl(`${stripTrailingSlash(credential.apiUrl)}/api/pugi/health`, {
|
|
34
|
+
method: 'GET',
|
|
35
|
+
headers: {
|
|
36
|
+
Authorization: `Bearer ${credential.apiKey}`,
|
|
37
|
+
},
|
|
38
|
+
});
|
|
39
|
+
}
|
|
40
|
+
catch (error) {
|
|
41
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
42
|
+
return {
|
|
43
|
+
name: 'AUTH',
|
|
44
|
+
status: 'warn',
|
|
45
|
+
detail: `Auth check skipped — network error contacting ${credential.apiUrl}`,
|
|
46
|
+
remediation: `Verify network: ${message}`,
|
|
47
|
+
};
|
|
48
|
+
}
|
|
49
|
+
const latencyMs = deps.now() - startedAt;
|
|
50
|
+
if (response.status === 401 || response.status === 403) {
|
|
51
|
+
return {
|
|
52
|
+
name: 'AUTH',
|
|
53
|
+
status: 'error',
|
|
54
|
+
detail: `Token rejected (${response.status}) by ${credential.apiUrl}`,
|
|
55
|
+
latencyMs,
|
|
56
|
+
remediation: 'Token expired or revoked — run `pugi login`',
|
|
57
|
+
};
|
|
58
|
+
}
|
|
59
|
+
if (response.status >= 500) {
|
|
60
|
+
return {
|
|
61
|
+
name: 'AUTH',
|
|
62
|
+
status: 'warn',
|
|
63
|
+
detail: `Server error ${response.status} from ${credential.apiUrl}`,
|
|
64
|
+
latencyMs,
|
|
65
|
+
remediation: 'Try again in a moment; if it persists, check api.pugi.io status',
|
|
66
|
+
};
|
|
67
|
+
}
|
|
68
|
+
if (response.status !== 200) {
|
|
69
|
+
return {
|
|
70
|
+
name: 'AUTH',
|
|
71
|
+
status: 'warn',
|
|
72
|
+
detail: `Unexpected status ${response.status} from /api/pugi/health`,
|
|
73
|
+
latencyMs,
|
|
74
|
+
};
|
|
75
|
+
}
|
|
76
|
+
return {
|
|
77
|
+
name: 'AUTH',
|
|
78
|
+
status: 'ok',
|
|
79
|
+
detail: `Authenticated against ${credential.apiUrl}`,
|
|
80
|
+
latencyMs,
|
|
81
|
+
};
|
|
82
|
+
}
|
|
83
|
+
function stripTrailingSlash(url) {
|
|
84
|
+
return url.endsWith('/') ? url.slice(0, -1) : url;
|
|
85
|
+
}
|
|
86
|
+
//# sourceMappingURL=auth.js.map
|
|
@@ -0,0 +1,127 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* CLI VERSION probe — compares the running @pugi/cli version against
|
|
3
|
+
* the npm registry's `latest` tag. Surfaces an upgrade banner when
|
|
4
|
+
* the operator is behind.
|
|
5
|
+
*
|
|
6
|
+
* Semver comparison is intentionally minimal — we only need to answer
|
|
7
|
+
* "is local strictly older than latest" for the WARN gate. Edge cases
|
|
8
|
+
* (pre-release ordering, build metadata) collapse к string equality
|
|
9
|
+
* because the publish pipeline only tags clean `X.Y.Z[-channel.N]`.
|
|
10
|
+
*
|
|
11
|
+
* Network failure is NOT an error — the operator is offline, that's
|
|
12
|
+
* a transient condition surfaced by the API probe; this probe reports
|
|
13
|
+
* `warn` so the doctor table still ships a usable verdict.
|
|
14
|
+
*/
|
|
15
|
+
const REGISTRY_URL = 'https://registry.npmjs.org/@pugi/cli/latest';
|
|
16
|
+
/**
|
|
17
|
+
* Strict-newer comparison. Returns true when `b` is strictly newer
|
|
18
|
+
* than `a`. Treats unparseable inputs as equal (no false-positive
|
|
19
|
+
* upgrade banner on a hand-edited local version).
|
|
20
|
+
*/
|
|
21
|
+
export function isNewerVersion(a, b) {
|
|
22
|
+
const left = parseSemver(a);
|
|
23
|
+
const right = parseSemver(b);
|
|
24
|
+
if (!left || !right)
|
|
25
|
+
return false;
|
|
26
|
+
if (right.major !== left.major)
|
|
27
|
+
return right.major > left.major;
|
|
28
|
+
if (right.minor !== left.minor)
|
|
29
|
+
return right.minor > left.minor;
|
|
30
|
+
if (right.patch !== left.patch)
|
|
31
|
+
return right.patch > left.patch;
|
|
32
|
+
// Same X.Y.Z — pre-release ordering: a stable release is newer
|
|
33
|
+
// than any pre-release of the same X.Y.Z; otherwise compare
|
|
34
|
+
// pre-release tokens lexicographically as a coarse heuristic
|
|
35
|
+
// sufficient for the upgrade banner.
|
|
36
|
+
if (!right.pre && left.pre)
|
|
37
|
+
return true;
|
|
38
|
+
if (right.pre && !left.pre)
|
|
39
|
+
return false;
|
|
40
|
+
if (right.pre && left.pre)
|
|
41
|
+
return right.pre > left.pre;
|
|
42
|
+
return false;
|
|
43
|
+
}
|
|
44
|
+
function parseSemver(version) {
|
|
45
|
+
const match = /^v?(\d+)\.(\d+)\.(\d+)(?:-([0-9A-Za-z.-]+))?/.exec(version);
|
|
46
|
+
if (!match)
|
|
47
|
+
return null;
|
|
48
|
+
const major = Number(match[1]);
|
|
49
|
+
const minor = Number(match[2]);
|
|
50
|
+
const patch = Number(match[3]);
|
|
51
|
+
if (!Number.isFinite(major) || !Number.isFinite(minor) || !Number.isFinite(patch)) {
|
|
52
|
+
return null;
|
|
53
|
+
}
|
|
54
|
+
return { major, minor, patch, pre: match[4] ?? '' };
|
|
55
|
+
}
|
|
56
|
+
export async function probeCliVersion(deps) {
|
|
57
|
+
const url = deps.registryUrl ?? REGISTRY_URL;
|
|
58
|
+
const timeoutMs = deps.timeoutMs ?? 3_000;
|
|
59
|
+
const controller = new AbortController();
|
|
60
|
+
const timer = setTimeout(() => controller.abort(), timeoutMs);
|
|
61
|
+
const startedAt = deps.now();
|
|
62
|
+
let response;
|
|
63
|
+
try {
|
|
64
|
+
response = await deps.fetchImpl(url, {
|
|
65
|
+
method: 'GET',
|
|
66
|
+
headers: { Accept: 'application/json' },
|
|
67
|
+
signal: controller.signal,
|
|
68
|
+
});
|
|
69
|
+
}
|
|
70
|
+
catch (error) {
|
|
71
|
+
clearTimeout(timer);
|
|
72
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
73
|
+
return {
|
|
74
|
+
name: 'CLI VERSION',
|
|
75
|
+
status: 'warn',
|
|
76
|
+
detail: `local=${deps.localVersion} — registry unreachable`,
|
|
77
|
+
remediation: `Skip-able. Network error: ${message}`,
|
|
78
|
+
};
|
|
79
|
+
}
|
|
80
|
+
clearTimeout(timer);
|
|
81
|
+
const latencyMs = deps.now() - startedAt;
|
|
82
|
+
if (!response.ok) {
|
|
83
|
+
return {
|
|
84
|
+
name: 'CLI VERSION',
|
|
85
|
+
status: 'warn',
|
|
86
|
+
detail: `local=${deps.localVersion} — registry returned ${response.status}`,
|
|
87
|
+
latencyMs,
|
|
88
|
+
};
|
|
89
|
+
}
|
|
90
|
+
let body;
|
|
91
|
+
try {
|
|
92
|
+
body = (await response.json());
|
|
93
|
+
}
|
|
94
|
+
catch {
|
|
95
|
+
return {
|
|
96
|
+
name: 'CLI VERSION',
|
|
97
|
+
status: 'warn',
|
|
98
|
+
detail: `local=${deps.localVersion} — registry JSON unparseable`,
|
|
99
|
+
latencyMs,
|
|
100
|
+
};
|
|
101
|
+
}
|
|
102
|
+
const remote = body.version;
|
|
103
|
+
if (typeof remote !== 'string' || remote.length === 0) {
|
|
104
|
+
return {
|
|
105
|
+
name: 'CLI VERSION',
|
|
106
|
+
status: 'warn',
|
|
107
|
+
detail: `local=${deps.localVersion} — registry response missing version`,
|
|
108
|
+
latencyMs,
|
|
109
|
+
};
|
|
110
|
+
}
|
|
111
|
+
if (isNewerVersion(deps.localVersion, remote)) {
|
|
112
|
+
return {
|
|
113
|
+
name: 'CLI VERSION',
|
|
114
|
+
status: 'warn',
|
|
115
|
+
detail: `local=${deps.localVersion}, latest=${remote}`,
|
|
116
|
+
latencyMs,
|
|
117
|
+
remediation: 'Run `npm i -g @pugi/cli@latest` to upgrade',
|
|
118
|
+
};
|
|
119
|
+
}
|
|
120
|
+
return {
|
|
121
|
+
name: 'CLI VERSION',
|
|
122
|
+
status: 'ok',
|
|
123
|
+
detail: `${deps.localVersion} (latest)`,
|
|
124
|
+
latencyMs,
|
|
125
|
+
};
|
|
126
|
+
}
|
|
127
|
+
//# sourceMappingURL=cli-version.js.map
|
|
@@ -0,0 +1,72 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* CONFIG probe — verifies `~/.pugi/credentials.json` exists, parses,
|
|
3
|
+
* and carries the canonical shape (a `tokens` array). Lighter-weight
|
|
4
|
+
* than the auth probe (no network); catches `corrupted JSON` /
|
|
5
|
+
* `accidentally overwrote with empty file` / `bad permissions` cases
|
|
6
|
+
* that would otherwise surface as confusing errors deeper in the
|
|
7
|
+
* auth path.
|
|
8
|
+
*
|
|
9
|
+
* Absence of the file is NOT an error here — the operator may not
|
|
10
|
+
* have run `pugi login` yet. That case is caught by the AUTH probe
|
|
11
|
+
* with a clean "run pugi login" remediation. CONFIG's job is to
|
|
12
|
+
* verify the file is sane WHEN it exists.
|
|
13
|
+
*/
|
|
14
|
+
export function probeConfig(ctx, fs) {
|
|
15
|
+
const credPath = `${ctx.home}/.pugi/credentials.json`;
|
|
16
|
+
if (!fs.existsSync(credPath)) {
|
|
17
|
+
return {
|
|
18
|
+
name: 'CONFIG',
|
|
19
|
+
status: 'skipped',
|
|
20
|
+
detail: '~/.pugi/credentials.json absent (operator has not logged in)',
|
|
21
|
+
};
|
|
22
|
+
}
|
|
23
|
+
let raw;
|
|
24
|
+
try {
|
|
25
|
+
raw = fs.readFileSync(credPath, 'utf8');
|
|
26
|
+
}
|
|
27
|
+
catch (error) {
|
|
28
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
29
|
+
return {
|
|
30
|
+
name: 'CONFIG',
|
|
31
|
+
status: 'error',
|
|
32
|
+
detail: `Cannot read ~/.pugi/credentials.json`,
|
|
33
|
+
remediation: `Fix permissions or delete and re-login: ${message}`,
|
|
34
|
+
};
|
|
35
|
+
}
|
|
36
|
+
let parsed;
|
|
37
|
+
try {
|
|
38
|
+
parsed = JSON.parse(raw);
|
|
39
|
+
}
|
|
40
|
+
catch (error) {
|
|
41
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
42
|
+
return {
|
|
43
|
+
name: 'CONFIG',
|
|
44
|
+
status: 'error',
|
|
45
|
+
detail: `~/.pugi/credentials.json is not valid JSON`,
|
|
46
|
+
remediation: `Delete and re-login: ${message}`,
|
|
47
|
+
};
|
|
48
|
+
}
|
|
49
|
+
if (!parsed || typeof parsed !== 'object') {
|
|
50
|
+
return {
|
|
51
|
+
name: 'CONFIG',
|
|
52
|
+
status: 'error',
|
|
53
|
+
detail: `credentials.json root is not an object`,
|
|
54
|
+
remediation: 'Delete and re-login',
|
|
55
|
+
};
|
|
56
|
+
}
|
|
57
|
+
const tokens = parsed.tokens;
|
|
58
|
+
if (!Array.isArray(tokens)) {
|
|
59
|
+
return {
|
|
60
|
+
name: 'CONFIG',
|
|
61
|
+
status: 'error',
|
|
62
|
+
detail: `credentials.json missing required \`tokens\` array`,
|
|
63
|
+
remediation: 'Delete and re-login',
|
|
64
|
+
};
|
|
65
|
+
}
|
|
66
|
+
return {
|
|
67
|
+
name: 'CONFIG',
|
|
68
|
+
status: 'ok',
|
|
69
|
+
detail: `~/.pugi/credentials.json valid (${tokens.length} token(s) stored)`,
|
|
70
|
+
};
|
|
71
|
+
}
|
|
72
|
+
//# sourceMappingURL=config.js.map
|