@pugi/cli 0.1.0-beta.3 → 0.1.0-beta.31

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.
Files changed (219) hide show
  1. package/THIRD_PARTY_NOTICES.md +40 -0
  2. package/assets/pugi-mascot.ansi +15 -40
  3. package/bin/run.js +33 -1
  4. package/dist/commands/jobs-watch.js +201 -0
  5. package/dist/commands/jobs.js +15 -0
  6. package/dist/core/agent-progress/cleanup.js +134 -0
  7. package/dist/core/agent-progress/schema.js +144 -0
  8. package/dist/core/agent-progress/writer.js +101 -0
  9. package/dist/core/artifact-chain/dispatcher.js +148 -0
  10. package/dist/core/artifact-chain/exporter.js +164 -0
  11. package/dist/core/artifact-chain/state.js +243 -0
  12. package/dist/core/artifact-chain/steps.js +169 -0
  13. package/dist/core/auth/env-provider.js +238 -0
  14. package/dist/core/auto-update/channels.js +122 -0
  15. package/dist/core/auto-update/checker.js +241 -0
  16. package/dist/core/auto-update/state.js +235 -0
  17. package/dist/core/bare-mode/index.js +107 -0
  18. package/dist/core/checkpoint/resumer.js +149 -0
  19. package/dist/core/checkpoint/rewinder.js +291 -0
  20. package/dist/core/compact/auto-trigger.js +96 -0
  21. package/dist/core/compact/buffer-rewriter.js +115 -0
  22. package/dist/core/compact/summarizer.js +208 -0
  23. package/dist/core/compact/token-counter.js +108 -0
  24. package/dist/core/consensus/diff-capture.js +73 -0
  25. package/dist/core/context/index.js +7 -0
  26. package/dist/core/context/markdown-traverse.js +255 -0
  27. package/dist/core/cost/rate-card.js +129 -0
  28. package/dist/core/cost/tracker.js +221 -0
  29. package/dist/core/denial-tracking/index.js +8 -0
  30. package/dist/core/denial-tracking/state.js +264 -0
  31. package/dist/core/diagnostics/probe-runner.js +93 -0
  32. package/dist/core/diagnostics/probes/api.js +46 -0
  33. package/dist/core/diagnostics/probes/auth.js +86 -0
  34. package/dist/core/diagnostics/probes/bare-mode.js +42 -0
  35. package/dist/core/diagnostics/probes/cli-version.js +127 -0
  36. package/dist/core/diagnostics/probes/config.js +72 -0
  37. package/dist/core/diagnostics/probes/denial-tracking.js +57 -0
  38. package/dist/core/diagnostics/probes/disk.js +81 -0
  39. package/dist/core/diagnostics/probes/git.js +65 -0
  40. package/dist/core/diagnostics/probes/mcp.js +75 -0
  41. package/dist/core/diagnostics/probes/node.js +59 -0
  42. package/dist/core/diagnostics/probes/pnpm.js +36 -0
  43. package/dist/core/diagnostics/probes/pugi-md.js +89 -0
  44. package/dist/core/diagnostics/probes/session.js +74 -0
  45. package/dist/core/diagnostics/probes/status-snapshot.js +442 -0
  46. package/dist/core/diagnostics/probes/workspace.js +63 -0
  47. package/dist/core/diagnostics/types.js +70 -0
  48. package/dist/core/dispatch/cache-cleanup.js +197 -0
  49. package/dist/core/dispatch/cache-handoff.js +295 -0
  50. package/dist/core/edits/dispatch.js +218 -2
  51. package/dist/core/edits/journal.js +199 -0
  52. package/dist/core/edits/layer-d-ast.js +557 -14
  53. package/dist/core/edits/verify-hook.js +273 -0
  54. package/dist/core/edits/worktree.js +111 -18
  55. package/dist/core/engine/anvil-client.js +115 -5
  56. package/dist/core/engine/budgets.js +89 -0
  57. package/dist/core/engine/context-prefix.js +155 -0
  58. package/dist/core/engine/intent.js +260 -0
  59. package/dist/core/engine/native-pugi.js +852 -210
  60. package/dist/core/engine/prompts.js +89 -6
  61. package/dist/core/engine/strip-internal-fields.js +124 -0
  62. package/dist/core/engine/tool-bridge.js +972 -33
  63. package/dist/core/feedback/queue.js +177 -0
  64. package/dist/core/feedback/submitter.js +145 -0
  65. package/dist/core/file-cache.js +113 -1
  66. package/dist/core/hooks/events.js +44 -0
  67. package/dist/core/hooks/index.js +15 -0
  68. package/dist/core/hooks/registry.js +213 -0
  69. package/dist/core/hooks/runner.js +236 -0
  70. package/dist/core/init/scaffold.js +195 -0
  71. package/dist/core/lsp/cache.js +105 -0
  72. package/dist/core/lsp/client.js +174 -29
  73. package/dist/core/lsp/language-detect.js +66 -0
  74. package/dist/core/lsp/post-edit-diagnostics.js +171 -0
  75. package/dist/core/mcp/client.js +75 -6
  76. package/dist/core/mcp/http-server.js +553 -0
  77. package/dist/core/mcp/permission.js +190 -0
  78. package/dist/core/mcp/registry.js +24 -2
  79. package/dist/core/mcp/server-tools.js +219 -0
  80. package/dist/core/mcp/server.js +397 -0
  81. package/dist/core/memory/dual-write.js +416 -0
  82. package/dist/core/memory/dual-write.spec.js +297 -0
  83. package/dist/core/memory/phase1-kinds.js +20 -0
  84. package/dist/core/memory-sync/queue.js +158 -0
  85. package/dist/core/memory-sync/queue.spec.js +105 -0
  86. package/dist/core/onboarding/marker.js +111 -0
  87. package/dist/core/onboarding/telemetry-state.js +108 -0
  88. package/dist/core/output-style/presets.js +176 -0
  89. package/dist/core/output-style/state.js +185 -0
  90. package/dist/core/permissions/gate.js +187 -0
  91. package/dist/core/permissions/index.js +18 -0
  92. package/dist/core/permissions/mode.js +102 -0
  93. package/dist/core/permissions/state.js +215 -0
  94. package/dist/core/permissions/tool-class.js +93 -0
  95. package/dist/core/prd-check/parser.js +215 -0
  96. package/dist/core/prd-check/reporter.js +127 -0
  97. package/dist/core/prd-check/session-review.js +557 -0
  98. package/dist/core/prd-check/verifiers.js +223 -0
  99. package/dist/core/pugi-md/context-injector.js +76 -0
  100. package/dist/core/pugi-md/walk-up.js +207 -0
  101. package/dist/core/release-notes/parser.js +241 -0
  102. package/dist/core/release-notes/state.js +116 -0
  103. package/dist/core/repl/codebase-survey.js +308 -0
  104. package/dist/core/repl/history.js +11 -1
  105. package/dist/core/repl/init-interview.js +457 -0
  106. package/dist/core/repl/model-pricing.js +135 -0
  107. package/dist/core/repl/onboarding-state.js +297 -0
  108. package/dist/core/repl/session.js +1529 -30
  109. package/dist/core/repl/slash-commands.js +361 -13
  110. package/dist/core/repl/store/session-store.js +31 -2
  111. package/dist/core/repl/workspace-context.js +22 -0
  112. package/dist/core/repo-map/build.js +125 -0
  113. package/dist/core/repo-map/cache.js +185 -0
  114. package/dist/core/repo-map/extractor.js +254 -0
  115. package/dist/core/repo-map/formatter.js +145 -0
  116. package/dist/core/repo-map/scanner.js +211 -0
  117. package/dist/core/retry-budget/budget.js +284 -0
  118. package/dist/core/retry-budget/index.js +5 -0
  119. package/dist/core/session.js +44 -0
  120. package/dist/core/settings.js +80 -0
  121. package/dist/core/share/formatter.js +271 -0
  122. package/dist/core/share/redactor.js +221 -0
  123. package/dist/core/share/uploader.js +267 -0
  124. package/dist/core/skills/defaults.js +457 -0
  125. package/dist/core/subagents/dispatcher-real.js +600 -0
  126. package/dist/core/subagents/dispatcher.js +113 -24
  127. package/dist/core/subagents/index.js +18 -5
  128. package/dist/core/subagents/isolation-matrix.js +213 -0
  129. package/dist/core/subagents/spawn.js +19 -4
  130. package/dist/core/telemetry/emitter.js +229 -0
  131. package/dist/core/telemetry/queue.js +251 -0
  132. package/dist/core/theme/context.js +91 -0
  133. package/dist/core/theme/presets.js +228 -0
  134. package/dist/core/theme/state.js +181 -0
  135. package/dist/core/todos/invariant.js +10 -0
  136. package/dist/core/todos/state.js +177 -0
  137. package/dist/core/transport/version-interceptor.js +166 -0
  138. package/dist/core/vim/keymap.js +288 -0
  139. package/dist/core/vim/state.js +92 -0
  140. package/dist/index.js +28 -0
  141. package/dist/runtime/bootstrap.js +190 -0
  142. package/dist/runtime/cli.js +2603 -278
  143. package/dist/runtime/commands/chain.js +489 -0
  144. package/dist/runtime/commands/compact.js +297 -0
  145. package/dist/runtime/commands/cost.js +199 -0
  146. package/dist/runtime/commands/delegate.js +312 -0
  147. package/dist/runtime/commands/dispatch.js +126 -0
  148. package/dist/runtime/commands/doctor.js +390 -0
  149. package/dist/runtime/commands/feedback.js +184 -0
  150. package/dist/runtime/commands/hooks.js +184 -0
  151. package/dist/runtime/commands/lsp.js +212 -28
  152. package/dist/runtime/commands/mcp.js +824 -0
  153. package/dist/runtime/commands/memory.js +508 -0
  154. package/dist/runtime/commands/memory.spec.js +174 -0
  155. package/dist/runtime/commands/model.js +237 -0
  156. package/dist/runtime/commands/onboarding.js +275 -0
  157. package/dist/runtime/commands/patch.js +17 -0
  158. package/dist/runtime/commands/permissions.js +87 -0
  159. package/dist/runtime/commands/plan.js +143 -0
  160. package/dist/runtime/commands/prd-check.js +285 -0
  161. package/dist/runtime/commands/release-notes.js +229 -0
  162. package/dist/runtime/commands/repo-map.js +95 -0
  163. package/dist/runtime/commands/report.js +299 -0
  164. package/dist/runtime/commands/resume.js +118 -0
  165. package/dist/runtime/commands/review-consensus.js +17 -2
  166. package/dist/runtime/commands/rewind.js +333 -0
  167. package/dist/runtime/commands/roster.js +117 -0
  168. package/dist/runtime/commands/sessions.js +163 -0
  169. package/dist/runtime/commands/share.js +316 -0
  170. package/dist/runtime/commands/status.js +178 -0
  171. package/dist/runtime/commands/stickers.js +82 -0
  172. package/dist/runtime/commands/style.js +194 -0
  173. package/dist/runtime/commands/theme.js +196 -0
  174. package/dist/runtime/commands/update.js +289 -0
  175. package/dist/runtime/commands/vim.js +140 -0
  176. package/dist/runtime/commands/worktree.js +50 -6
  177. package/dist/runtime/headless.js +543 -0
  178. package/dist/runtime/load-hooks-or-exit.js +71 -0
  179. package/dist/runtime/plan-decompose.js +531 -0
  180. package/dist/runtime/version.js +65 -0
  181. package/dist/tools/agent-tool.js +229 -0
  182. package/dist/tools/apply-patch.js +281 -39
  183. package/dist/tools/ask-user-question.js +213 -0
  184. package/dist/tools/ask-user.js +115 -0
  185. package/dist/tools/file-tools.js +85 -14
  186. package/dist/tools/mcp-tool.js +260 -0
  187. package/dist/tools/multi-edit.js +361 -0
  188. package/dist/tools/registry.js +30 -2
  189. package/dist/tools/skill-tool.js +96 -0
  190. package/dist/tools/tasks.js +208 -0
  191. package/dist/tools/todo-write.js +184 -0
  192. package/dist/tools/web-fetch.js +147 -2
  193. package/dist/tools/web-search.js +458 -0
  194. package/dist/tui/agent-progress-card.js +111 -0
  195. package/dist/tui/agent-tree.js +10 -0
  196. package/dist/tui/ask-modal.js +2 -2
  197. package/dist/tui/ask-user-question-prompt.js +192 -0
  198. package/dist/tui/compact-banner.js +81 -0
  199. package/dist/tui/conversation-pane.js +82 -8
  200. package/dist/tui/cost-table.js +111 -0
  201. package/dist/tui/doctor-table.js +46 -0
  202. package/dist/tui/feedback-prompt.js +156 -0
  203. package/dist/tui/input-box.js +46 -2
  204. package/dist/tui/markdown-render.js +4 -4
  205. package/dist/tui/onboarding-wizard.js +240 -0
  206. package/dist/tui/repl-render.js +293 -35
  207. package/dist/tui/repl-splash.js +2 -2
  208. package/dist/tui/repl.js +45 -13
  209. package/dist/tui/splash.js +1 -1
  210. package/dist/tui/status-bar.js +94 -16
  211. package/dist/tui/status-table.js +7 -0
  212. package/dist/tui/stickers-art.js +136 -0
  213. package/dist/tui/style-table.js +28 -0
  214. package/dist/tui/theme-table.js +29 -0
  215. package/dist/tui/tool-stream-pane.js +7 -0
  216. package/dist/tui/update-banner.js +20 -2
  217. package/dist/tui/vim-input.js +267 -0
  218. package/docs/examples/codegraph.mcp.json +10 -0
  219. package/package.json +9 -6
@@ -0,0 +1,255 @@
1
+ /**
2
+ * Per-directory PUGI.md / AGENTS.md / CLAUDE.md / GEMINI.md traverse-up
3
+ * loader — β5a R4+P5.
4
+ *
5
+ * Claude Code, Codex CLI, and Gemini CLI all support a "walk up from
6
+ * cwd to the workspace root, pick up agent-context markdown at every
7
+ * level" pattern. Without this, a `pugi explain` invoked from
8
+ * `apps/admin-api/` cannot see project-local conventions encoded in
9
+ * `apps/admin-api/PUGI.md` (or the cross-CLI shim files) — the
10
+ * existing `loadMarkdownContext` only reads files at `workspaceRoot`.
11
+ *
12
+ * The β5a quality gate (≥80% win-rate vs Claude Code per CEO 2026-05-26)
13
+ * surfaces this gap repeatedly: monorepo-local conventions (NestJS
14
+ * controller style, Prisma migration name format, cabinet brand voice
15
+ * gates) live in per-app context files, and Pugi was blind to them
16
+ * pre-β5a.
17
+ *
18
+ * Contract:
19
+ *
20
+ * - Walk from `cwd` upward until we reach `workspaceRoot` OR cross
21
+ * the filesystem boundary. The workspace root file itself is
22
+ * loaded by the legacy `loadMarkdownContext` so we do NOT include
23
+ * it here (no double-load, no double-budget-charge).
24
+ *
25
+ * - At each intermediate directory, look for the four canonical
26
+ * filenames: `PUGI.md` (native), `AGENTS.md` (cross-CLI shim),
27
+ * `CLAUDE.md` (Claude Code compat), `GEMINI.md` (Gemini CLI compat).
28
+ *
29
+ * - HTML comments stripped, identical to the legacy loader.
30
+ *
31
+ * - `@import` expansion is intentionally NOT performed here — the
32
+ * per-dir surface is "drop a small file with the local
33
+ * conventions"; deep @import chains belong at workspace root.
34
+ * Keeping this surface flat means the per-dir budget cannot be
35
+ * blown out by a runaway @import in an unrelated subtree.
36
+ *
37
+ * - Aggregate budget: `MAX_TRAVERSE_BYTES` across ALL files found
38
+ * in the walk. When exhausted, remaining files are skipped with
39
+ * a `budget_exhausted` warning. Default 32 KB — half the
40
+ * workspace-root budget, because per-dir files are meant to be
41
+ * terse delta conventions, not full project briefs.
42
+ *
43
+ * - The walk is bounded: `MAX_TRAVERSE_DEPTH` levels above
44
+ * workspaceRoot are NEVER traversed (defense against being
45
+ * invoked from a malicious cwd outside the workspace; in
46
+ * practice cwd is always inside workspaceRoot but the symlink
47
+ * case demands belt + suspenders).
48
+ *
49
+ * - Order returned: shallowest-first (workspace root would be
50
+ * first if included, then each level closer to cwd). Closest-
51
+ * to-cwd files are the most specific and the context builder
52
+ * emits them LAST so the model treats them as the highest-
53
+ * priority conventions.
54
+ *
55
+ * This module is pure: no logging, no network, no fs writes. Filesystem
56
+ * reads only.
57
+ */
58
+ import { existsSync, readFileSync, realpathSync, statSync } from 'node:fs';
59
+ import { dirname, isAbsolute, relative, resolve, sep } from 'node:path';
60
+ import { stripHtmlComments } from './markdown-loader.js';
61
+ /**
62
+ * Per-traverse total byte cap. Half the workspace-root budget — these
63
+ * files are meant to be terse "in this subdir, do X". 32 KB still fits
64
+ * ~8000 words of guidance.
65
+ */
66
+ export const MAX_TRAVERSE_BYTES = 32 * 1024;
67
+ /**
68
+ * Per-file byte cap. A single per-dir file should never dominate; the
69
+ * 8 KB cap keeps any one level honest. Files larger than this are
70
+ * loaded up to the cap and flagged truncated.
71
+ */
72
+ export const MAX_TRAVERSE_PER_FILE_BYTES = 8 * 1024;
73
+ /**
74
+ * Maximum number of parent directories above workspaceRoot we will
75
+ * traverse. Zero in normal operation — cwd is always inside the
76
+ * workspace; the cap exists so a misconfigured invocation never
77
+ * walks the whole filesystem looking for AGENTS.md.
78
+ */
79
+ export const MAX_TRAVERSE_DEPTH = 0;
80
+ /**
81
+ * Filenames we look for at every level of the walk. The order here
82
+ * also defines the per-directory load order: PUGI.md first (highest
83
+ * trust), then cross-CLI compat shims. When the same directory has
84
+ * multiple files (e.g. both PUGI.md AND CLAUDE.md), all are loaded
85
+ * — operators sometimes keep both during a tool migration.
86
+ */
87
+ export const TRAVERSE_SOURCES = ['PUGI.md', 'AGENTS.md', 'CLAUDE.md', 'GEMINI.md'];
88
+ /**
89
+ * Walk from `opts.cwd` upward toward `opts.workspaceRoot`, loading
90
+ * every PUGI.md / AGENTS.md / CLAUDE.md / GEMINI.md we encounter at
91
+ * intermediate levels. The workspace root itself is NOT loaded here
92
+ * — `loadMarkdownContext(workspaceRoot)` owns that file.
93
+ *
94
+ * Returned `loaded` is sorted shallowest-first (closest-to-root) so
95
+ * the caller can emit per-dir context in increasing specificity.
96
+ *
97
+ * Safety properties (proven by spec):
98
+ *
99
+ * - Never reads outside the workspace tree (symlinks resolved via
100
+ * realpathSync; off-tree symlinks rejected as
101
+ * `import_escapes_workspace`).
102
+ * - Never reads more than `MAX_TRAVERSE_BYTES` total or more than
103
+ * `MAX_TRAVERSE_PER_FILE_BYTES` per file.
104
+ * - Never walks above the workspace root.
105
+ * - Never visits the workspace root itself (single source of truth
106
+ * for workspace-level docs stays with `loadMarkdownContext`).
107
+ */
108
+ export async function loadTraversedMarkdown(opts) {
109
+ const warnings = [];
110
+ const loaded = [];
111
+ let budgetRemaining = MAX_TRAVERSE_BYTES;
112
+ let absRoot;
113
+ let absCwd;
114
+ try {
115
+ absRoot = realpathSync(resolve(opts.workspaceRoot));
116
+ absCwd = realpathSync(resolve(opts.cwd));
117
+ }
118
+ catch (error) {
119
+ warnings.push({
120
+ kind: 'read_error',
121
+ message: `realpath failed for traverse anchor: ${error.message}`,
122
+ });
123
+ return { loaded, warnings, totalBytes: 0 };
124
+ }
125
+ // Containment guard: if cwd is not inside workspaceRoot, refuse
126
+ // to walk. Returning a clean empty result keeps the engine happy
127
+ // and surfaces nothing about the off-tree cwd to the model.
128
+ const relCwd = relative(absRoot, absCwd);
129
+ if (relCwd.startsWith('..') || isAbsolute(relCwd)) {
130
+ warnings.push({
131
+ kind: 'import_escapes_workspace',
132
+ message: `cwd is outside workspaceRoot; per-dir traverse skipped (cwd=${absCwd}, root=${absRoot})`,
133
+ path: absCwd,
134
+ });
135
+ return { loaded, warnings, totalBytes: 0 };
136
+ }
137
+ // Collect the walk: every directory from cwd UP TO (but not
138
+ // including) workspaceRoot.
139
+ const dirsToVisit = [];
140
+ let current = absCwd;
141
+ while (current !== absRoot) {
142
+ dirsToVisit.push(current);
143
+ const parent = dirname(current);
144
+ if (parent === current)
145
+ break; // hit filesystem root before workspaceRoot — defensive
146
+ current = parent;
147
+ if (dirsToVisit.length > 64)
148
+ break; // pathological depth, refuse
149
+ }
150
+ // Walk shallowest-first so we charge the budget in the order
151
+ // that matches what we return.
152
+ dirsToVisit.reverse();
153
+ for (const dir of dirsToVisit) {
154
+ if (budgetRemaining <= 0) {
155
+ warnings.push({
156
+ kind: 'budget_exhausted',
157
+ message: `per-dir traverse budget exhausted before reaching ${dir}`,
158
+ path: dir,
159
+ });
160
+ break;
161
+ }
162
+ for (const source of TRAVERSE_SOURCES) {
163
+ const candidate = resolve(dir, source);
164
+ if (!existsSync(candidate))
165
+ continue;
166
+ // Symlink guard: same realpath check as the workspace-root
167
+ // loader. A symlink inside the workspace that points outside
168
+ // the workspace must NOT be inlined.
169
+ let realCandidate;
170
+ try {
171
+ realCandidate = realpathSync(candidate);
172
+ }
173
+ catch (error) {
174
+ warnings.push({
175
+ kind: 'read_error',
176
+ message: `realpath failed for ${candidate}: ${error.message}`,
177
+ path: candidate,
178
+ });
179
+ continue;
180
+ }
181
+ const realRel = relative(absRoot, realCandidate);
182
+ if (realRel.startsWith('..') || isAbsolute(realRel)) {
183
+ warnings.push({
184
+ kind: 'import_escapes_workspace',
185
+ message: `traverse file escapes workspace via symlink: ${candidate} -> ${realCandidate}`,
186
+ path: candidate,
187
+ });
188
+ continue;
189
+ }
190
+ let raw;
191
+ let rawBytes;
192
+ try {
193
+ rawBytes = statSync(candidate).size;
194
+ raw = readFileSync(candidate, 'utf8');
195
+ }
196
+ catch (error) {
197
+ warnings.push({
198
+ kind: 'read_error',
199
+ message: `could not read ${candidate}: ${error.message}`,
200
+ path: candidate,
201
+ });
202
+ continue;
203
+ }
204
+ const stripped = stripHtmlComments(raw);
205
+ // Per-file cap first, then global budget.
206
+ const perFileCap = Math.min(MAX_TRAVERSE_PER_FILE_BYTES, budgetRemaining);
207
+ let content = stripped;
208
+ let truncated = false;
209
+ let contentBytes = Buffer.byteLength(content, 'utf8');
210
+ if (contentBytes > perFileCap) {
211
+ // Codepoint-safe slice: convert byte cap to char cap by
212
+ // taking min(byte cap, char-length-up-to-cap). We accept
213
+ // mild over-trim for safety.
214
+ content = content.slice(0, perFileCap);
215
+ truncated = true;
216
+ contentBytes = Buffer.byteLength(content, 'utf8');
217
+ }
218
+ const distance = distanceSegments(absCwd, dir);
219
+ loaded.push({
220
+ source,
221
+ resolvedPath: candidate,
222
+ dir,
223
+ distanceFromCwd: distance,
224
+ rawBytes,
225
+ loadedBytes: contentBytes,
226
+ truncated,
227
+ content,
228
+ });
229
+ budgetRemaining -= contentBytes;
230
+ if (budgetRemaining <= 0)
231
+ break;
232
+ }
233
+ }
234
+ return {
235
+ loaded,
236
+ warnings,
237
+ totalBytes: MAX_TRAVERSE_BYTES - budgetRemaining,
238
+ };
239
+ }
240
+ /**
241
+ * How many path segments separate `from` and `to`. Both must be
242
+ * absolute. `to` is assumed to be an ancestor of (or equal to)
243
+ * `from`; if not, returns -1 so callers can ignore the file.
244
+ */
245
+ function distanceSegments(from, to) {
246
+ if (from === to)
247
+ return 0;
248
+ const rel = relative(to, from);
249
+ if (rel.startsWith('..') || isAbsolute(rel))
250
+ return -1;
251
+ if (rel.length === 0)
252
+ return 0;
253
+ return rel.split(sep).length;
254
+ }
255
+ //# sourceMappingURL=markdown-traverse.js.map
@@ -0,0 +1,129 @@
1
+ /**
2
+ * Rate card for the `pugi cost` / `/cost` / `/usage` surface — L19 sprint.
3
+ *
4
+ * Distinct from `core/repl/model-pricing.ts` on purpose:
5
+ *
6
+ * - `model-pricing.ts` powers the TUI cost meter (per-turn flash, status
7
+ * row USD). Its ladder is keyed against the live Anvil model slugs and
8
+ * intentionally inflates an honest worst-case figure via the Sonnet
9
+ * fallback so an operator on a quiet model never gets billed by a
10
+ * surprise. It rounds to USD per 1M tokens at runtime.
11
+ *
12
+ * - `rate-card.ts` (this file) powers the persisted `/cost` table the
13
+ * operator reads to plan budget. It distinguishes open-weight models
14
+ * ($0 / $0 — infra cost only) from hosted closed models so the table
15
+ * does not double-charge an operator running a self-hosted Qwen or
16
+ * Kimi behind Pugi. The L19 spec calls these out by name.
17
+ *
18
+ * Both ladders intentionally agree on Anthropic Claude family pricing so
19
+ * the TUI flash and the persisted table cannot disagree on a Claude turn.
20
+ * If they diverge, the per-model-pricing ladder wins for live UI; the
21
+ * rate card here wins for the persisted `.pugi/cost.json` aggregate.
22
+ *
23
+ * Prices are USD per 1,000,000 tokens, sourced from the L19 spec
24
+ * (2026-05-27) which mirrors provider list-price pages as of that date.
25
+ */
26
+ /**
27
+ * Exact-match price ladder keyed by model slug. Slugs match the L19 task
28
+ * spec verbatim so a copy-paste from the sprint doc resolves without
29
+ * normalisation.
30
+ */
31
+ export const RATES_PER_MTOKEN = Object.freeze({
32
+ // Anthropic Claude family (hosted, billed).
33
+ 'claude-opus-4-7': { input: 15, output: 75 },
34
+ 'claude-opus-4-6': { input: 15, output: 75 },
35
+ 'claude-sonnet-4-6': { input: 3, output: 15 },
36
+ 'claude-sonnet-4-5': { input: 3, output: 15 },
37
+ 'claude-haiku-4-5-20251001': { input: 1, output: 5 },
38
+ 'claude-haiku-4-5': { input: 1, output: 5 },
39
+ // Open-weight models — infra cost only, never per-token billed. The
40
+ // note column surfaces the reason so a CFO reading the JSON envelope
41
+ // does not assume the row is broken.
42
+ 'qwen3-coder-480b-instruct-fp8': { input: 0, output: 0, note: 'open-weight' },
43
+ 'kimi-k2.6': { input: 0, output: 0, note: 'open-weight' },
44
+ 'deepseek-v4-pro': { input: 0, output: 0, note: 'open-weight' },
45
+ });
46
+ /**
47
+ * Family-prefix fallback — used only when an exact slug miss. Mirrors the
48
+ * approach in `model-pricing.ts` so a future model rebind (e.g.
49
+ * `claude-opus-4-8`) prices reasonably without a code edit.
50
+ */
51
+ const FAMILY_FALLBACKS = [
52
+ ['claude-opus-', { input: 15, output: 75 }],
53
+ ['claude-sonnet-', { input: 3, output: 15 }],
54
+ ['claude-haiku-', { input: 1, output: 5 }],
55
+ ['qwen', { input: 0, output: 0, note: 'open-weight' }],
56
+ ['kimi', { input: 0, output: 0, note: 'open-weight' }],
57
+ ['deepseek', { input: 0, output: 0, note: 'open-weight' }],
58
+ ];
59
+ /**
60
+ * Final fallback for unknown slugs. Pinned to Sonnet-tier — same posture
61
+ * as `model-pricing.ts`'s default, so an unrecognised hosted model bills
62
+ * "honestly conservative" rather than $0 (which would silently hide cost
63
+ * from the operator).
64
+ */
65
+ const DEFAULT_RATE = { input: 3, output: 15, note: 'unknown model — Sonnet-tier estimate' };
66
+ /**
67
+ * Look up the rate for a model slug.
68
+ *
69
+ * Resolution order:
70
+ * 1. Exact match in `RATES_PER_MTOKEN`.
71
+ * 2. Family-prefix match (first hit wins).
72
+ * 3. Default Sonnet-tier estimate.
73
+ *
74
+ * Pure, never throws. Called on every cost-tracker write so the hot path
75
+ * stays branch-cheap.
76
+ */
77
+ export function rateFor(model) {
78
+ if (!model || typeof model !== 'string')
79
+ return DEFAULT_RATE;
80
+ const exact = RATES_PER_MTOKEN[model];
81
+ if (exact)
82
+ return exact;
83
+ for (const [prefix, rate] of FAMILY_FALLBACKS) {
84
+ if (model.startsWith(prefix))
85
+ return rate;
86
+ }
87
+ return DEFAULT_RATE;
88
+ }
89
+ /**
90
+ * Compute the USD cost for a single (model, inputTokens, outputTokens)
91
+ * triple. Defensive against negative / NaN inputs — out-of-range values
92
+ * floor to zero so a buggy upstream cannot credit a negative cost.
93
+ */
94
+ export function estimateUsd(model, inputTokens, outputTokens) {
95
+ const rate = rateFor(model);
96
+ const safeIn = Number.isFinite(inputTokens) && inputTokens > 0 ? inputTokens : 0;
97
+ const safeOut = Number.isFinite(outputTokens) && outputTokens > 0 ? outputTokens : 0;
98
+ const usd = (safeIn * rate.input + safeOut * rate.output) / 1_000_000;
99
+ return Number.isFinite(usd) && usd > 0 ? usd : 0;
100
+ }
101
+ /**
102
+ * Format a USD figure for the `/cost` table.
103
+ *
104
+ * - `≥ $0.01` → two decimals (`$0.46`).
105
+ * - `< $0.01` but `> 0` → three decimals (`$0.003`) so fractions of a
106
+ * cent are honest instead of rounding to `$0.00`.
107
+ * - Exactly `0` or NaN → `$0.00`.
108
+ *
109
+ * Mirrors `formatCostUsd` from `model-pricing.ts` intentionally — both
110
+ * surfaces should print the same number in the same shape.
111
+ */
112
+ export function formatUsd(value) {
113
+ if (!Number.isFinite(value) || value <= 0)
114
+ return '$0.00';
115
+ if (value >= 0.01)
116
+ return `$${value.toFixed(2)}`;
117
+ return `$${value.toFixed(3)}`;
118
+ }
119
+ /**
120
+ * Format a token count for the `/cost` table. Uses comma-thousands so the
121
+ * table reads `14,300` instead of `14.3k` — distinct from the TUI status
122
+ * row which uses `k`/`m` shortening to save column width.
123
+ */
124
+ export function formatTokensWithCommas(value) {
125
+ if (!Number.isFinite(value) || value <= 0)
126
+ return '0';
127
+ return Math.floor(value).toLocaleString('en-US');
128
+ }
129
+ //# sourceMappingURL=rate-card.js.map
@@ -0,0 +1,221 @@
1
+ /**
2
+ * Persisted per-session cost tracker — L19 sprint (2026-05-27).
3
+ *
4
+ * Mission: every Anvil-mediated LLM call goes through `recordCall`, which
5
+ * aggregates per-model token + USD totals and atomically persists them
6
+ * to `.pugi/cost.json` so the operator can read `/cost` across REPL
7
+ * restarts and reconcile a 14-min session that crossed a process boundary.
8
+ *
9
+ * Why a fresh module instead of bolting onto `core/repl/session.ts`?
10
+ *
11
+ * - `session.ts` accumulates in-memory state for the live TUI status
12
+ * row, which is by-design ephemeral and cleared on REPL boot. The
13
+ * operator's "what did I spend across the project?" question needs
14
+ * a durable surface that survives a process restart.
15
+ * - L19 also has to read `--all-sessions` (last 30 days). The natural
16
+ * store for that is a per-workspace history of session aggregates,
17
+ * which is easy with the JSON file pattern below and would be
18
+ * awkward stitched into the REPL reducer.
19
+ *
20
+ * On-disk shape (single JSON file, atomic tmp+rename writes):
21
+ *
22
+ * {
23
+ * "version": 1,
24
+ * "current": { sessionId, startedAt, models: { <slug>: ModelEntry } },
25
+ * "history": [
26
+ * { sessionId, startedAt, endedAt, models: { ... } }
27
+ * ]
28
+ * }
29
+ *
30
+ * History rotation: when `recordCall` is invoked with a sessionId
31
+ * different from `current.sessionId`, the existing `current` row is
32
+ * stamped with `endedAt = now()` and pushed onto `history`, then a new
33
+ * `current` row is initialised. History is capped at 90 entries (the L19
34
+ * `--all-sessions` window is 30 days; 90 gives a generous buffer for
35
+ * operators on >1 session/day cadence without unbounded growth).
36
+ *
37
+ * The tracker is workspace-scoped — every workspace has its own
38
+ * `.pugi/cost.json`. This matches the existing `.pugi/events.jsonl` /
39
+ * `.pugi/index.json` pattern and means a multi-repo operator's costs
40
+ * are billed against the repo they were incurred in.
41
+ */
42
+ import { existsSync, mkdirSync, readFileSync, renameSync, writeFileSync, unlinkSync } from 'node:fs';
43
+ import { dirname, resolve } from 'node:path';
44
+ import { estimateUsd } from './rate-card.js';
45
+ /** On-disk schema version. Bump if the file shape changes. */
46
+ export const COST_FILE_SCHEMA_VERSION = 1;
47
+ /** Maximum number of historical sessions persisted in `.pugi/cost.json`. */
48
+ export const COST_HISTORY_CAP = 90;
49
+ export function createCostTracker(opts) {
50
+ const filePath = resolve(opts.workspaceRoot, '.pugi/cost.json');
51
+ const now = opts.now ?? Date.now;
52
+ let state = readOrInit(filePath);
53
+ function ensureCurrent(sessionId) {
54
+ if (state.current && state.current.sessionId === sessionId) {
55
+ return state.current;
56
+ }
57
+ // Session rotation: stamp the previous current with endedAt and push
58
+ // onto history. Idempotent — calling rotate twice with the same
59
+ // session id is a no-op.
60
+ if (state.current) {
61
+ const ended = {
62
+ ...state.current,
63
+ endedAt: new Date(now()).toISOString(),
64
+ };
65
+ state.history = [ended, ...state.history].slice(0, COST_HISTORY_CAP);
66
+ }
67
+ state.current = {
68
+ sessionId,
69
+ startedAt: new Date(now()).toISOString(),
70
+ models: {},
71
+ };
72
+ return state.current;
73
+ }
74
+ function persist() {
75
+ try {
76
+ mkdirSync(dirname(filePath), { recursive: true });
77
+ }
78
+ catch {
79
+ // best-effort directory create; the write below surfaces the real
80
+ // error if the parent is genuinely unwritable
81
+ }
82
+ const tmp = `${filePath}.tmp`;
83
+ writeFileSync(tmp, JSON.stringify(state, null, 2), 'utf8');
84
+ renameSync(tmp, filePath);
85
+ }
86
+ return {
87
+ recordCall(input) {
88
+ const sessionId = opts.sessionIdProvider();
89
+ if (!sessionId)
90
+ return;
91
+ const current = ensureCurrent(sessionId);
92
+ const slug = typeof input.model === 'string' && input.model.length > 0 ? input.model : 'unknown';
93
+ const safeIn = Number.isFinite(input.inputTokens) && input.inputTokens > 0 ? input.inputTokens : 0;
94
+ const safeOut = Number.isFinite(input.outputTokens) && input.outputTokens > 0 ? input.outputTokens : 0;
95
+ const existing = current.models[slug] ?? { input: 0, output: 0, callCount: 0 };
96
+ current.models[slug] = {
97
+ input: existing.input + safeIn,
98
+ output: existing.output + safeOut,
99
+ callCount: existing.callCount + 1,
100
+ };
101
+ persist();
102
+ },
103
+ current() {
104
+ return state.current;
105
+ },
106
+ history() {
107
+ return state.history;
108
+ },
109
+ aggregateWithin(withinDays) {
110
+ const cutoffMs = now() - withinDays * 24 * 60 * 60 * 1000;
111
+ const aggregate = {
112
+ sessionId: 'aggregate',
113
+ startedAt: new Date(cutoffMs).toISOString(),
114
+ endedAt: new Date(now()).toISOString(),
115
+ models: {},
116
+ };
117
+ const rows = [];
118
+ if (state.current)
119
+ rows.push(state.current);
120
+ for (const row of state.history) {
121
+ const stamp = Date.parse(row.startedAt);
122
+ if (Number.isFinite(stamp) && stamp >= cutoffMs)
123
+ rows.push(row);
124
+ }
125
+ for (const row of rows) {
126
+ for (const [slug, entry] of Object.entries(row.models)) {
127
+ const existing = aggregate.models[slug] ?? { input: 0, output: 0, callCount: 0 };
128
+ aggregate.models[slug] = {
129
+ input: existing.input + entry.input,
130
+ output: existing.output + entry.output,
131
+ callCount: existing.callCount + entry.callCount,
132
+ };
133
+ }
134
+ }
135
+ return aggregate;
136
+ },
137
+ resetCurrent() {
138
+ const wiped = state.current;
139
+ state.current = null;
140
+ persist();
141
+ return wiped;
142
+ },
143
+ flush() {
144
+ persist();
145
+ },
146
+ };
147
+ }
148
+ /**
149
+ * Compute the per-session USD total from a `SessionAggregate`. Pure —
150
+ * uses the rate card to bind a price to every model entry. Open-weight
151
+ * models contribute $0 (their entries always have $0/$0 rate).
152
+ */
153
+ export function totalUsd(aggregate) {
154
+ let total = 0;
155
+ for (const [slug, entry] of Object.entries(aggregate.models)) {
156
+ total += estimateUsd(slug, entry.input, entry.output);
157
+ }
158
+ return total;
159
+ }
160
+ /**
161
+ * Compute total input + output token sums across all models in an
162
+ * aggregate. Used by the CLI table footer.
163
+ */
164
+ export function totalTokens(aggregate) {
165
+ let input = 0;
166
+ let output = 0;
167
+ for (const entry of Object.values(aggregate.models)) {
168
+ input += entry.input;
169
+ output += entry.output;
170
+ }
171
+ return { input, output };
172
+ }
173
+ /**
174
+ * Read the persisted file (or initialise an empty one). Tolerates a
175
+ * corrupted file by returning a fresh empty state — losing one
176
+ * session's history is preferable to throwing from the boot path of
177
+ * every `pugi cost` invocation.
178
+ */
179
+ function readOrInit(filePath) {
180
+ if (!existsSync(filePath)) {
181
+ return { version: COST_FILE_SCHEMA_VERSION, current: null, history: [] };
182
+ }
183
+ try {
184
+ const raw = readFileSync(filePath, 'utf8');
185
+ const parsed = JSON.parse(raw);
186
+ if (!parsed || typeof parsed !== 'object' || Array.isArray(parsed)) {
187
+ return { version: COST_FILE_SCHEMA_VERSION, current: null, history: [] };
188
+ }
189
+ const obj = parsed;
190
+ return {
191
+ version: typeof obj.version === 'number' ? obj.version : COST_FILE_SCHEMA_VERSION,
192
+ current: isAggregate(obj.current) ? obj.current : null,
193
+ history: Array.isArray(obj.history) ? obj.history.filter(isAggregate) : [],
194
+ };
195
+ }
196
+ catch {
197
+ return { version: COST_FILE_SCHEMA_VERSION, current: null, history: [] };
198
+ }
199
+ }
200
+ function isAggregate(v) {
201
+ if (!v || typeof v !== 'object' || Array.isArray(v))
202
+ return false;
203
+ const obj = v;
204
+ if (typeof obj.sessionId !== 'string' || typeof obj.startedAt !== 'string')
205
+ return false;
206
+ if (!obj.models || typeof obj.models !== 'object')
207
+ return false;
208
+ return true;
209
+ }
210
+ /**
211
+ * Test helper — wipe the `.pugi/cost.json` file. Not exported through the
212
+ * public CostTracker surface because production code must never call
213
+ * this; an operator-facing reset goes through `resetCurrent()` which
214
+ * preserves history.
215
+ */
216
+ export function _danger_wipeCostFile_forTests(workspaceRoot) {
217
+ const filePath = resolve(workspaceRoot, '.pugi/cost.json');
218
+ if (existsSync(filePath))
219
+ unlinkSync(filePath);
220
+ }
221
+ //# sourceMappingURL=tracker.js.map
@@ -0,0 +1,8 @@
1
+ /**
2
+ * Public re-exports for the denial-tracking surface. Engine adapter
3
+ * code imports from here so we can move internals around (e.g. split
4
+ * the diagnostics probe into a sibling file) without churning every
5
+ * call site.
6
+ */
7
+ export { DenialTrackingState, buildDenialContext, canonicalArgHash, DENIAL_TRACKING_MAX_ENTRIES, DENIAL_REMINDER_THRESHOLD, DENIAL_ARGS_SUMMARY_BYTES, } from './state.js';
8
+ //# sourceMappingURL=index.js.map