@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.
Files changed (264) hide show
  1. package/THIRD_PARTY_NOTICES.md +40 -0
  2. package/assets/pugi-mascot.ansi +15 -25
  3. package/assets/pugi-prozr2-mascot.ansi +9 -0
  4. package/bin/run.js +33 -1
  5. package/dist/commands/jobs-watch.js +201 -0
  6. package/dist/commands/jobs.js +15 -0
  7. package/dist/commands/smoke.js +133 -0
  8. package/dist/core/agent-progress/cleanup.js +134 -0
  9. package/dist/core/agent-progress/schema.js +144 -0
  10. package/dist/core/agent-progress/writer.js +101 -0
  11. package/dist/core/artifact-chain/dispatcher.js +148 -0
  12. package/dist/core/artifact-chain/exporter.js +164 -0
  13. package/dist/core/artifact-chain/state.js +243 -0
  14. package/dist/core/artifact-chain/steps.js +169 -0
  15. package/dist/core/auth/ensure-authenticated.js +129 -0
  16. package/dist/core/auth/env-provider.js +238 -0
  17. package/dist/core/auto-update/channels.js +122 -0
  18. package/dist/core/auto-update/checker.js +241 -0
  19. package/dist/core/auto-update/state.js +235 -0
  20. package/dist/core/bare-mode/index.js +107 -0
  21. package/dist/core/bash-classifier.js +400 -4
  22. package/dist/core/checkpoint/resumer.js +149 -0
  23. package/dist/core/checkpoint/rewinder.js +291 -0
  24. package/dist/core/codegraph/decision-store.js +248 -0
  25. package/dist/core/codegraph/detect-repo.js +459 -0
  26. package/dist/core/codegraph/install.js +134 -0
  27. package/dist/core/codegraph/offer-hook.js +220 -0
  28. package/dist/core/compact/auto-trigger.js +96 -0
  29. package/dist/core/compact/buffer-rewriter.js +115 -0
  30. package/dist/core/compact/summarizer.js +208 -0
  31. package/dist/core/compact/token-counter.js +108 -0
  32. package/dist/core/consensus/diff-capture.js +112 -3
  33. package/dist/core/context/index.js +7 -0
  34. package/dist/core/context/markdown-traverse.js +255 -0
  35. package/dist/core/cost/rate-card.js +129 -0
  36. package/dist/core/cost/tracker.js +221 -0
  37. package/dist/core/denial-tracking/index.js +8 -0
  38. package/dist/core/denial-tracking/state.js +264 -0
  39. package/dist/core/diagnostics/probe-runner.js +93 -0
  40. package/dist/core/diagnostics/probes/api.js +46 -0
  41. package/dist/core/diagnostics/probes/auth.js +86 -0
  42. package/dist/core/diagnostics/probes/bare-mode.js +42 -0
  43. package/dist/core/diagnostics/probes/cli-version.js +127 -0
  44. package/dist/core/diagnostics/probes/config.js +72 -0
  45. package/dist/core/diagnostics/probes/denial-tracking.js +57 -0
  46. package/dist/core/diagnostics/probes/disk.js +81 -0
  47. package/dist/core/diagnostics/probes/git.js +65 -0
  48. package/dist/core/diagnostics/probes/hooks.js +118 -0
  49. package/dist/core/diagnostics/probes/mcp.js +75 -0
  50. package/dist/core/diagnostics/probes/node.js +59 -0
  51. package/dist/core/diagnostics/probes/pnpm.js +36 -0
  52. package/dist/core/diagnostics/probes/pugi-md.js +89 -0
  53. package/dist/core/diagnostics/probes/sandbox.js +40 -0
  54. package/dist/core/diagnostics/probes/session.js +74 -0
  55. package/dist/core/diagnostics/probes/status-snapshot.js +488 -0
  56. package/dist/core/diagnostics/probes/workspace.js +63 -0
  57. package/dist/core/diagnostics/types.js +70 -0
  58. package/dist/core/dispatch/cache-cleanup.js +197 -0
  59. package/dist/core/dispatch/cache-handoff.js +295 -0
  60. package/dist/core/edits/dispatch.js +218 -2
  61. package/dist/core/edits/journal.js +199 -0
  62. package/dist/core/edits/layer-d-ast.js +557 -14
  63. package/dist/core/edits/verify-hook.js +273 -0
  64. package/dist/core/edits/worktree.js +322 -0
  65. package/dist/core/engine/anvil-client.js +115 -5
  66. package/dist/core/engine/auto-compact.js +179 -0
  67. package/dist/core/engine/budgets.js +155 -0
  68. package/dist/core/engine/context-prefix.js +155 -0
  69. package/dist/core/engine/intent.js +260 -0
  70. package/dist/core/engine/native-pugi.js +897 -211
  71. package/dist/core/engine/prompts.js +88 -2
  72. package/dist/core/engine/strip-internal-fields.js +124 -0
  73. package/dist/core/engine/tool-bridge.js +1045 -36
  74. package/dist/core/feedback/queue.js +177 -0
  75. package/dist/core/feedback/submitter.js +145 -0
  76. package/dist/core/file-cache.js +113 -1
  77. package/dist/core/hooks/events.js +44 -0
  78. package/dist/core/hooks/index.js +15 -0
  79. package/dist/core/hooks/registry.js +213 -0
  80. package/dist/core/hooks/runner.js +236 -0
  81. package/dist/core/hooks/v2/event-emitter.js +115 -0
  82. package/dist/core/hooks/v2/executor.js +282 -0
  83. package/dist/core/hooks/v2/index.js +25 -0
  84. package/dist/core/hooks/v2/lifecycle.js +104 -0
  85. package/dist/core/hooks/v2/loader.js +216 -0
  86. package/dist/core/hooks/v2/matcher.js +125 -0
  87. package/dist/core/hooks/v2/trust.js +143 -0
  88. package/dist/core/hooks/v2/types.js +86 -0
  89. package/dist/core/lsp/cache.js +105 -0
  90. package/dist/core/lsp/client.js +776 -0
  91. package/dist/core/lsp/language-detect.js +66 -0
  92. package/dist/core/lsp/post-edit-diagnostics.js +171 -0
  93. package/dist/core/mcp/client.js +75 -6
  94. package/dist/core/mcp/http-server.js +553 -0
  95. package/dist/core/mcp/orchestrator-tools.js +662 -0
  96. package/dist/core/mcp/permission.js +190 -0
  97. package/dist/core/mcp/registry.js +24 -2
  98. package/dist/core/mcp/server-tools.js +219 -0
  99. package/dist/core/mcp/server.js +397 -0
  100. package/dist/core/memory/dual-write.js +416 -0
  101. package/dist/core/memory/phase1-kinds.js +20 -0
  102. package/dist/core/memory-sync/queue.js +158 -0
  103. package/dist/core/onboarding/ensure-initialized.js +133 -0
  104. package/dist/core/onboarding/marker.js +111 -0
  105. package/dist/core/onboarding/telemetry-state.js +108 -0
  106. package/dist/core/output-style/presets.js +176 -0
  107. package/dist/core/output-style/state.js +185 -0
  108. package/dist/core/path-security.js +284 -2
  109. package/dist/core/permissions/auto-classifier.js +124 -0
  110. package/dist/core/permissions/circuit-breaker.js +83 -0
  111. package/dist/core/permissions/gate.js +278 -0
  112. package/dist/core/permissions/index.js +20 -0
  113. package/dist/core/permissions/mode.js +174 -0
  114. package/dist/core/permissions/state.js +241 -0
  115. package/dist/core/permissions/tool-class.js +93 -0
  116. package/dist/core/prd-check/parser.js +215 -0
  117. package/dist/core/prd-check/reporter.js +127 -0
  118. package/dist/core/prd-check/session-review.js +557 -0
  119. package/dist/core/prd-check/verifiers.js +223 -0
  120. package/dist/core/pugi-md/context-injector.js +76 -0
  121. package/dist/core/pugi-md/walk-up.js +207 -0
  122. package/dist/core/release-notes/parser.js +241 -0
  123. package/dist/core/release-notes/state.js +116 -0
  124. package/dist/core/repl/history.js +11 -1
  125. package/dist/core/repl/model-pricing.js +135 -0
  126. package/dist/core/repl/session.js +1897 -37
  127. package/dist/core/repl/slash-commands.js +430 -15
  128. package/dist/core/repl/store/session-store.js +31 -2
  129. package/dist/core/repl/workspace-context.js +22 -0
  130. package/dist/core/repo-map/build.js +125 -0
  131. package/dist/core/repo-map/cache.js +185 -0
  132. package/dist/core/repo-map/extractor.js +254 -0
  133. package/dist/core/repo-map/formatter.js +145 -0
  134. package/dist/core/repo-map/scanner.js +211 -0
  135. package/dist/core/retry-budget/budget.js +284 -0
  136. package/dist/core/retry-budget/index.js +5 -0
  137. package/dist/core/session.js +92 -0
  138. package/dist/core/settings.js +80 -0
  139. package/dist/core/share/formatter.js +271 -0
  140. package/dist/core/share/redactor.js +221 -0
  141. package/dist/core/share/uploader.js +267 -0
  142. package/dist/core/skills/defaults.js +457 -0
  143. package/dist/core/smoke/headless-driver.js +174 -0
  144. package/dist/core/smoke/orchestrator.js +194 -0
  145. package/dist/core/smoke/runner.js +238 -0
  146. package/dist/core/smoke/scenario-parser.js +316 -0
  147. package/dist/core/subagents/dispatcher-real.js +600 -0
  148. package/dist/core/subagents/dispatcher.js +113 -24
  149. package/dist/core/subagents/index.js +18 -5
  150. package/dist/core/subagents/isolation-matrix.js +213 -0
  151. package/dist/core/subagents/spawn.js +19 -4
  152. package/dist/core/telemetry/emitter.js +229 -0
  153. package/dist/core/telemetry/queue.js +251 -0
  154. package/dist/core/theme/context.js +91 -0
  155. package/dist/core/theme/presets.js +228 -0
  156. package/dist/core/theme/state.js +181 -0
  157. package/dist/core/todos/invariant.js +10 -0
  158. package/dist/core/todos/state.js +177 -0
  159. package/dist/core/transport/version-interceptor.js +166 -0
  160. package/dist/core/vim/keymap.js +288 -0
  161. package/dist/core/vim/state.js +92 -0
  162. package/dist/core/worktree-manager/cleanup.js +123 -0
  163. package/dist/core/worktree-manager/manager.js +303 -0
  164. package/dist/index.js +28 -0
  165. package/dist/runtime/bootstrap.js +190 -0
  166. package/dist/runtime/cli.js +3241 -343
  167. package/dist/runtime/commands/cancel.js +231 -0
  168. package/dist/runtime/commands/chain.js +489 -0
  169. package/dist/runtime/commands/codegraph-status.js +227 -0
  170. package/dist/runtime/commands/compact.js +297 -0
  171. package/dist/runtime/commands/cost.js +199 -0
  172. package/dist/runtime/commands/delegate.js +242 -11
  173. package/dist/runtime/commands/dispatch.js +126 -0
  174. package/dist/runtime/commands/doctor.js +412 -0
  175. package/dist/runtime/commands/feedback.js +184 -0
  176. package/dist/runtime/commands/hooks.js +184 -0
  177. package/dist/runtime/commands/lsp.js +368 -0
  178. package/dist/runtime/commands/mcp.js +879 -0
  179. package/dist/runtime/commands/memory.js +508 -0
  180. package/dist/runtime/commands/model.js +237 -0
  181. package/dist/runtime/commands/onboarding.js +275 -0
  182. package/dist/runtime/commands/patch.js +128 -0
  183. package/dist/runtime/commands/permissions.js +112 -0
  184. package/dist/runtime/commands/plan.js +143 -0
  185. package/dist/runtime/commands/prd-check.js +285 -0
  186. package/dist/runtime/commands/redo-blob-store.js +92 -0
  187. package/dist/runtime/commands/redo.js +361 -0
  188. package/dist/runtime/commands/release-notes.js +229 -0
  189. package/dist/runtime/commands/repo-map.js +95 -0
  190. package/dist/runtime/commands/report.js +299 -0
  191. package/dist/runtime/commands/resume.js +118 -0
  192. package/dist/runtime/commands/review-consensus.js +17 -2
  193. package/dist/runtime/commands/rewind.js +333 -0
  194. package/dist/runtime/commands/sessions.js +163 -0
  195. package/dist/runtime/commands/share.js +316 -0
  196. package/dist/runtime/commands/status.js +186 -0
  197. package/dist/runtime/commands/stickers.js +82 -0
  198. package/dist/runtime/commands/style.js +194 -0
  199. package/dist/runtime/commands/theme.js +196 -0
  200. package/dist/runtime/commands/undo.js +32 -0
  201. package/dist/runtime/commands/update.js +289 -0
  202. package/dist/runtime/commands/vim.js +140 -0
  203. package/dist/runtime/commands/worktree.js +177 -0
  204. package/dist/runtime/commands/worktrees.js +155 -0
  205. package/dist/runtime/headless-repl.js +195 -0
  206. package/dist/runtime/headless.js +543 -0
  207. package/dist/runtime/load-hooks-or-exit.js +71 -0
  208. package/dist/runtime/plan-decompose.js +531 -0
  209. package/dist/runtime/version.js +65 -0
  210. package/dist/tools/agent-tool.js +229 -0
  211. package/dist/tools/apply-patch.js +556 -0
  212. package/dist/tools/ask-user-question.js +213 -0
  213. package/dist/tools/ask-user.js +115 -0
  214. package/dist/tools/bash.js +203 -4
  215. package/dist/tools/file-tools.js +85 -14
  216. package/dist/tools/lsp-tools.js +189 -0
  217. package/dist/tools/mcp-tool.js +260 -0
  218. package/dist/tools/multi-edit.js +361 -0
  219. package/dist/tools/powershell.js +268 -0
  220. package/dist/tools/registry.js +51 -0
  221. package/dist/tools/skill-tool.js +96 -0
  222. package/dist/tools/tasks.js +208 -0
  223. package/dist/tools/todo-write.js +184 -0
  224. package/dist/tools/web-fetch.js +147 -2
  225. package/dist/tools/web-search.js +458 -0
  226. package/dist/tui/agent-progress-card.js +111 -0
  227. package/dist/tui/agent-tree.js +10 -0
  228. package/dist/tui/ask-modal.js +2 -2
  229. package/dist/tui/ask-user-question-prompt.js +192 -0
  230. package/dist/tui/compact-banner.js +81 -0
  231. package/dist/tui/conversation-pane.js +82 -8
  232. package/dist/tui/cost-table.js +111 -0
  233. package/dist/tui/doctor-table.js +46 -0
  234. package/dist/tui/feedback-prompt.js +156 -0
  235. package/dist/tui/input-box.js +218 -3
  236. package/dist/tui/markdown-render.js +4 -4
  237. package/dist/tui/onboarding-wizard.js +240 -0
  238. package/dist/tui/permissions-picker.js +86 -0
  239. package/dist/tui/render.js +35 -0
  240. package/dist/tui/repl-render.js +313 -35
  241. package/dist/tui/repl-splash-art.js +1 -1
  242. package/dist/tui/repl-splash-mascot.js +32 -8
  243. package/dist/tui/repl-splash.js +2 -2
  244. package/dist/tui/repl.js +85 -5
  245. package/dist/tui/splash.js +1 -1
  246. package/dist/tui/status-bar.js +94 -16
  247. package/dist/tui/status-table.js +7 -0
  248. package/dist/tui/stickers-art.js +136 -0
  249. package/dist/tui/style-table.js +28 -0
  250. package/dist/tui/theme-table.js +29 -0
  251. package/dist/tui/thinking-spinner.js +123 -0
  252. package/dist/tui/tool-stream-pane.js +52 -3
  253. package/dist/tui/update-banner.js +27 -2
  254. package/dist/tui/vim-input.js +267 -0
  255. package/dist/tui/welcome-banner.js +107 -0
  256. package/dist/tui/welcome-data.js +293 -0
  257. package/docs/examples/codegraph.mcp.json +10 -0
  258. package/package.json +13 -7
  259. package/test/scenarios/codegen-create-file.scenario.txt +13 -0
  260. package/test/scenarios/compact-force.scenario.txt +11 -0
  261. package/test/scenarios/identity.scenario.txt +11 -0
  262. package/test/scenarios/persona-handoff.scenario.txt +11 -0
  263. package/test/scenarios/walkback.scenario.txt +12 -0
  264. package/dist/core/engine/compaction-hook.js +0 -154
@@ -0,0 +1,223 @@
1
+ /**
2
+ * PRD criterion verifiers — Pugi α7 Wave 6 (`/prd-check`, 2026-05-27).
3
+ *
4
+ * Given a `ParsedCriterion` from `parser.ts`, run a set of built-in
5
+ * checks against the repository workspace and return a
6
+ * `VerifiedCriterion` with a per-criterion verdict + per-mention
7
+ * evidence trail. The module owns the file-system + grep surface
8
+ * the parser deliberately avoided.
9
+ *
10
+ * Verdict semantics (mirror the doctor probe contract so the
11
+ * reporter can render both with the same column layout):
12
+ *
13
+ * - PASS : at least one mention verified AND every attempted
14
+ * verifier passed
15
+ * - FAIL : at least one verifier failed (missing file, empty
16
+ * doc, command not in registry, etc.)
17
+ * - SKIPPED : no verifiable mentions in the criterion. The
18
+ * criterion is preserved in the report so the
19
+ * operator sees it, but it does not gate the verdict.
20
+ *
21
+ * Each `MentionResult` carries an evidence string the reporter
22
+ * shows next to the criterion. For PASS we render the resolved
23
+ * absolute path or matched line; for FAIL we render the missing
24
+ * artifact identifier so the operator can fix it directly.
25
+ *
26
+ * The verifier deps are injected (existsSync, readFileSync, …) so
27
+ * the spec can drive every branch without touching the real disk.
28
+ * The default-bound variant `runDefaultVerifiers` plugs the real
29
+ * Node fs helpers in for the CLI handler.
30
+ */
31
+ import { existsSync, readFileSync } from 'node:fs';
32
+ import { isAbsolute, resolve } from 'node:path';
33
+ /**
34
+ * Top-level verify-one-criterion entry. Walks the mention list,
35
+ * dispatches each to the right verifier, then computes the
36
+ * roll-up. Pure with respect to `deps` — no globals touched.
37
+ */
38
+ export function verifyCriterion(criterion, deps) {
39
+ if (criterion.mentions.length === 0) {
40
+ return {
41
+ criterion,
42
+ status: 'skipped',
43
+ results: [],
44
+ };
45
+ }
46
+ const results = [];
47
+ for (const mention of criterion.mentions) {
48
+ results.push(verifyMention(mention, deps));
49
+ }
50
+ const status = rollUp(results);
51
+ return { criterion, status, results };
52
+ }
53
+ /** Verify a whole PRD's worth of criteria. */
54
+ export function verifyAll(criteria, deps) {
55
+ return criteria.map((c) => verifyCriterion(c, deps));
56
+ }
57
+ function verifyMention(mention, deps) {
58
+ switch (mention.kind) {
59
+ case 'file':
60
+ return verifyFile(mention, deps);
61
+ case 'test':
62
+ return verifyTest(mention, deps);
63
+ case 'doc':
64
+ return verifyDoc(mention, deps);
65
+ case 'command':
66
+ return verifyCommand(mention, deps);
67
+ case 'route':
68
+ return verifyRoute(mention, deps);
69
+ }
70
+ }
71
+ function verifyFile(mention, deps) {
72
+ const absolute = deps.resolveWorkspacePath(mention.path);
73
+ if (deps.existsSync(absolute)) {
74
+ return {
75
+ mention,
76
+ status: 'pass',
77
+ evidence: `file present (${mention.path})`,
78
+ };
79
+ }
80
+ return {
81
+ mention,
82
+ status: 'fail',
83
+ evidence: `file missing (${mention.path})`,
84
+ };
85
+ }
86
+ function verifyTest(mention, deps) {
87
+ const absolute = deps.resolveWorkspacePath(mention.path);
88
+ if (!deps.existsSync(absolute)) {
89
+ return {
90
+ mention,
91
+ status: 'fail',
92
+ evidence: `spec missing (${mention.path})`,
93
+ };
94
+ }
95
+ let body;
96
+ try {
97
+ body = deps.readFileSync(absolute);
98
+ }
99
+ catch (error) {
100
+ const message = error instanceof Error ? error.message : String(error);
101
+ return {
102
+ mention,
103
+ status: 'fail',
104
+ evidence: `spec unreadable: ${message}`,
105
+ };
106
+ }
107
+ // Count `test(`, `it(`, and `describe(...).it(` blocks. The
108
+ // matcher is permissive — any of the three counts because the
109
+ // PRD only cares whether the spec asserts anything at all.
110
+ const matches = body.match(/\b(it|test)\s*\(/g);
111
+ if (!matches || matches.length === 0) {
112
+ return {
113
+ mention,
114
+ status: 'fail',
115
+ evidence: `spec present but has 0 test()/it() blocks`,
116
+ };
117
+ }
118
+ return {
119
+ mention,
120
+ status: 'pass',
121
+ evidence: `spec present with ${matches.length} block(s)`,
122
+ };
123
+ }
124
+ function verifyDoc(mention, deps) {
125
+ const absolute = deps.resolveWorkspacePath(mention.path);
126
+ if (!deps.existsSync(absolute)) {
127
+ return {
128
+ mention,
129
+ status: 'fail',
130
+ evidence: `doc missing (${mention.path})`,
131
+ };
132
+ }
133
+ let body;
134
+ try {
135
+ body = deps.readFileSync(absolute);
136
+ }
137
+ catch (error) {
138
+ const message = error instanceof Error ? error.message : String(error);
139
+ return {
140
+ mention,
141
+ status: 'fail',
142
+ evidence: `doc unreadable: ${message}`,
143
+ };
144
+ }
145
+ const trimmed = body.trim();
146
+ if (trimmed.length < 100) {
147
+ return {
148
+ mention,
149
+ status: 'fail',
150
+ evidence: `doc present but too short (${trimmed.length} chars, < 100)`,
151
+ };
152
+ }
153
+ return {
154
+ mention,
155
+ status: 'pass',
156
+ evidence: `doc present (${trimmed.length} chars)`,
157
+ };
158
+ }
159
+ function verifyCommand(mention, deps) {
160
+ if (deps.isKnownCommand(mention.name)) {
161
+ return {
162
+ mention,
163
+ status: 'pass',
164
+ evidence: `command \`${mention.name}\` registered`,
165
+ };
166
+ }
167
+ return {
168
+ mention,
169
+ status: 'fail',
170
+ evidence: `command \`${mention.name}\` not found in CLI registry`,
171
+ };
172
+ }
173
+ function verifyRoute(mention, deps) {
174
+ if (deps.hasRoute(mention.method, mention.path)) {
175
+ return {
176
+ mention,
177
+ status: 'pass',
178
+ evidence: `route ${mention.method} ${mention.path} registered`,
179
+ };
180
+ }
181
+ return {
182
+ mention,
183
+ status: 'fail',
184
+ evidence: `route ${mention.method} ${mention.path} not found`,
185
+ };
186
+ }
187
+ function rollUp(results) {
188
+ if (results.length === 0)
189
+ return 'skipped';
190
+ const anyFail = results.some((r) => r.status === 'fail');
191
+ if (anyFail)
192
+ return 'fail';
193
+ const anyPass = results.some((r) => r.status === 'pass');
194
+ return anyPass ? 'pass' : 'skipped';
195
+ }
196
+ /**
197
+ * Default verifier deps bound to real fs + a CLI-command predicate
198
+ * + a grep-based route locator. The CLI handler passes the
199
+ * workspace root + the known-command list at call time.
200
+ */
201
+ export function createDefaultDeps(options) {
202
+ return {
203
+ resolveWorkspacePath: (relative) => {
204
+ if (isAbsolute(relative))
205
+ return relative;
206
+ return resolve(options.workspaceRoot, relative);
207
+ },
208
+ existsSync: (path) => existsSync(path),
209
+ readFileSync: (path) => readFileSync(path, 'utf8'),
210
+ isKnownCommand: (name) => options.knownCommands.has(name),
211
+ hasRoute: () => {
212
+ // Best-effort default: the wave-6 PRD says the route verifier
213
+ // is "best-effort grep of controllers". Since the workspace
214
+ // layout varies per repo, the CLI handler injects a project
215
+ // -aware implementation when one is available; the default
216
+ // stays conservative and reports `fail` so the reporter
217
+ // surfaces the missing-verifier signal instead of silently
218
+ // passing.
219
+ return false;
220
+ },
221
+ };
222
+ }
223
+ //# sourceMappingURL=verifiers.js.map
@@ -0,0 +1,76 @@
1
+ /**
2
+ * Aggregate byte cap on the full rendered block. 96 KB = 3 files at
3
+ * the per-file cap, which is enough for cwd + parent + homedir while
4
+ * leaving plenty of prompt budget for the rest of the system prompt.
5
+ * Anything beyond is replaced with a truncation marker.
6
+ */
7
+ export const MAX_INJECT_BYTES = 96 * 1024;
8
+ /**
9
+ * Marker line emitted when the aggregate cap is hit. Visible to the
10
+ * model so it knows ambient context was clipped; visible to the
11
+ * operator via the doctor probe so they can decide whether to trim
12
+ * their `PUGI.md` hierarchy.
13
+ */
14
+ export const TRUNCATION_MARKER = '<ambient-context-truncated reason="aggregate-cap" />';
15
+ /**
16
+ * Render a HierarchyFile array into the system-prompt block. Returns
17
+ * `''` when `files` is empty. Each file becomes one
18
+ * `<ambient-context source="..." level="...">...</ambient-context>`
19
+ * stanza separated by a single newline.
20
+ *
21
+ * Determinism: same input always produces byte-identical output.
22
+ */
23
+ export function renderAmbientContext(files) {
24
+ if (files.length === 0)
25
+ return '';
26
+ const stanzas = [];
27
+ let bytes = 0;
28
+ let truncated = false;
29
+ for (const file of files) {
30
+ const stanza = renderStanza(file);
31
+ const stanzaBytes = Buffer.byteLength(stanza, 'utf8') + 1; // newline join cost
32
+ if (bytes + stanzaBytes > MAX_INJECT_BYTES) {
33
+ truncated = true;
34
+ break;
35
+ }
36
+ stanzas.push(stanza);
37
+ bytes += stanzaBytes;
38
+ }
39
+ if (truncated)
40
+ stanzas.push(TRUNCATION_MARKER);
41
+ return stanzas.join('\n');
42
+ }
43
+ /**
44
+ * Build a single `<ambient-context>` stanza for one HierarchyFile.
45
+ * The `source` attribute carries the absolute path (after realpath)
46
+ * so the model can cite which file a piece of guidance came from
47
+ * when it explains its decisions to the operator.
48
+ */
49
+ function renderStanza(file) {
50
+ const sourceAttr = escapeAttr(file.path);
51
+ const levelAttr = String(file.level);
52
+ // No trailing newline inside `content` — the join adds one between
53
+ // stanzas. Trimming the file's trailing whitespace keeps the tag
54
+ // close to the content for readability when an engineer dumps the
55
+ // assembled prompt for debugging.
56
+ const trimmed = file.content.replace(/\s+$/g, '');
57
+ return [
58
+ `<ambient-context source="${sourceAttr}" level="${levelAttr}">`,
59
+ trimmed,
60
+ `</ambient-context>`,
61
+ ].join('\n');
62
+ }
63
+ /**
64
+ * Escape an XML attribute value. We expect operator-controlled paths
65
+ * (not adversarial input) but `&`, `"` and `<` are still possible in
66
+ * symlinked / unicode paths so we escape them defensively. The model
67
+ * has been trained to read this attribute as opaque metadata.
68
+ */
69
+ function escapeAttr(value) {
70
+ return value
71
+ .replace(/&/g, '&amp;')
72
+ .replace(/"/g, '&quot;')
73
+ .replace(/</g, '&lt;')
74
+ .replace(/>/g, '&gt;');
75
+ }
76
+ //# sourceMappingURL=context-injector.js.map
@@ -0,0 +1,207 @@
1
+ /**
2
+ * Leak L32 (2026-05-27) — `PUGI.md` hierarchy walk-up to `$HOME`.
3
+ *
4
+ * Claude Code walks from `cwd` upward toward the user's homedir and
5
+ * concatenates every `CLAUDE.md` it finds at each intermediate level
6
+ * (deepest overrides shallowest). Pugi parity: same walk, looking for
7
+ * `PUGI.md` first at each level and accepting `CLAUDE.md` as a fallback
8
+ * — operators often have a leftover `~/CLAUDE.md` or a parent-dir
9
+ * `CLAUDE.md` from a previous Claude Code session and we want their
10
+ * ambient guidance picked up automatically without a migration step.
11
+ *
12
+ * Why this is a separate module from `core/context/markdown-traverse.ts`:
13
+ *
14
+ * - `markdown-traverse.ts` is the *workspace-bounded* walk (cwd → up
15
+ * to but NOT including `workspaceRoot`). It guards every read by
16
+ * `realpathSync` containment against the workspace root and
17
+ * refuses to escape — by design, because the per-dir markdown is
18
+ * part of the project's first-party context.
19
+ *
20
+ * - This module is the *home-bounded* walk (cwd → up to `homedir()`,
21
+ * OR until depth limit). It picks up the operator's personal /
22
+ * global guidance that lives ABOVE the workspace root. The two
23
+ * surfaces are complementary: workspace markdown encodes project
24
+ * conventions; this hierarchy walk encodes operator-level taste
25
+ * (preferred libraries, "always run prettier", style guides).
26
+ *
27
+ * Contract:
28
+ *
29
+ * - Walks from `cwd` upward. At each directory checks `PUGI.md`
30
+ * (preferred); when absent falls back to `CLAUDE.md`. Only ONE
31
+ * file per level is loaded — preferred wins.
32
+ * - Stops at `homedir()` INCLUSIVE — the file at `~/PUGI.md` or
33
+ * `~/CLAUDE.md` IS loaded (Claude Code parity: a `~/CLAUDE.md`
34
+ * applies to every project the operator opens).
35
+ * - Hard depth cap of `MAX_WALK_DEPTH` (20) directories regardless
36
+ * of how far cwd is from homedir; defense against symlinked or
37
+ * malicious cwd values.
38
+ * - Per-file byte cap `MAX_FILE_BYTES` (32 KB); over-cap files are
39
+ * truncated, not rejected, so a runaway `PUGI.md` does not break
40
+ * the prompt budget.
41
+ * - Returns shallow-to-deep order (cwd FIRST, homedir LAST). The
42
+ * caller is responsible for rendering precedence — Claude Code's
43
+ * rule is "deeper overrides shallower", which means the LAST
44
+ * entry in the rendered system prompt wins. Our order matches
45
+ * that convention so the context injector can splice directly.
46
+ *
47
+ * Safety:
48
+ *
49
+ * - No `realpath` on the directories themselves: the operator's
50
+ * cwd may live under a workspace symlink (common with macOS
51
+ * `/private/var/...`) and we want to honor what the operator
52
+ * sees in their shell. We DO resolve the candidate file via
53
+ * `realpathSync` before reading, but only to defeat
54
+ * symlinks-pointing-outside-homedir attacks; an off-tree symlink
55
+ * is skipped silently.
56
+ * - Catch + skip every fs error per file. The walk-up surface MUST
57
+ * NEVER break engine boot — missing read perms on a parent dir
58
+ * is the common case (e.g. `/etc` on a corp laptop) and the
59
+ * fallback is "no ambient context", not a crash.
60
+ *
61
+ * Pure module: no logging, no network, no fs writes.
62
+ */
63
+ import { existsSync, readFileSync, realpathSync, statSync } from 'node:fs';
64
+ import { dirname, resolve } from 'node:path';
65
+ /**
66
+ * Hard ceiling on parent-dir traversal depth. 20 is generous — even
67
+ * deep monorepo layouts rarely sit more than 8-10 levels below the
68
+ * homedir on a developer's laptop. The cap exists so a misconfigured
69
+ * cwd (e.g. cwd outside the user's home filesystem entirely) cannot
70
+ * cause a multi-second fs scan of unrelated directories.
71
+ */
72
+ export const MAX_WALK_DEPTH = 20;
73
+ /**
74
+ * Per-file byte cap. 32 KB matches the per-dir markdown traverse
75
+ * aggregate budget — generous enough for a fully written-out
76
+ * project / personal `PUGI.md` (~8000 words) while keeping any one
77
+ * file from blowing the prompt budget on its own.
78
+ */
79
+ export const MAX_FILE_BYTES = 32 * 1024;
80
+ /**
81
+ * Filenames consulted at each level, in lookup order. `PUGI.md` is
82
+ * preferred — when both files coexist in a directory the Pugi-native
83
+ * file wins and the Claude Code shim is ignored. This is the same
84
+ * precedence used by `markdown-traverse.ts` for workspace-bounded
85
+ * walks; keeping the two surfaces consistent removes the "why does
86
+ * Pugi sometimes read CLAUDE.md and sometimes PUGI.md?" foot-gun.
87
+ */
88
+ export const HIERARCHY_SOURCES = ['PUGI.md', 'CLAUDE.md'];
89
+ /**
90
+ * Walk from `cwd` upward, collecting ambient `PUGI.md` / `CLAUDE.md`
91
+ * files at each level until we reach the homedir (inclusive) or the
92
+ * depth cap.
93
+ *
94
+ * Returns an array ordered shallowest-first (cwd → homedir). When no
95
+ * files are found, returns `[]`. When `cwd` is OUTSIDE the homedir
96
+ * tree (e.g. the operator runs `pugi` from `/tmp`), the walk still
97
+ * proceeds upward but stops the moment we reach a filesystem root
98
+ * without ever entering the homedir — useful for ops/admin invocations
99
+ * where there is genuinely no personal context to load.
100
+ */
101
+ export function walkUpPugiMd(cwd, opts = {}) {
102
+ const limit = clampLimit(opts.limit);
103
+ const home = opts.homedir;
104
+ let absCwd;
105
+ try {
106
+ absCwd = resolve(cwd);
107
+ }
108
+ catch {
109
+ return [];
110
+ }
111
+ const absHome = home ? resolve(home) : undefined;
112
+ const results = [];
113
+ let current = absCwd;
114
+ let level = 0;
115
+ const visited = new Set();
116
+ while (level <= limit) {
117
+ if (visited.has(current))
118
+ break; // pathological symlink loop guard
119
+ visited.add(current);
120
+ const found = tryLoadDirectory(current, level);
121
+ if (found)
122
+ results.push(found);
123
+ // Inclusive home boundary: load home if we are here, then stop.
124
+ if (absHome && current === absHome)
125
+ break;
126
+ const parent = dirname(current);
127
+ if (parent === current)
128
+ break; // hit filesystem root before homedir
129
+ current = parent;
130
+ level += 1;
131
+ }
132
+ return results;
133
+ }
134
+ /**
135
+ * Pick the first matching file in `dir`, read + cap it, and produce
136
+ * a HierarchyFile row. Returns `undefined` when no file in
137
+ * `HIERARCHY_SOURCES` exists or all reads error out (perms, symlink
138
+ * escape, etc.). NEVER throws — fs errors degrade to "no file at
139
+ * this level".
140
+ */
141
+ function tryLoadDirectory(dir, level) {
142
+ for (const source of HIERARCHY_SOURCES) {
143
+ const candidate = resolve(dir, source);
144
+ if (!existsSync(candidate))
145
+ continue;
146
+ // Realpath the FILE to defeat symlink-points-elsewhere attacks.
147
+ // We do not realpath the directory itself — operators often run
148
+ // pugi from inside a workspace symlink and the walk should honor
149
+ // the path they see.
150
+ let realPath;
151
+ try {
152
+ realPath = realpathSync(candidate);
153
+ }
154
+ catch {
155
+ // Broken symlink or perms issue on the link itself. Skip this
156
+ // file and try the next source in the same directory.
157
+ continue;
158
+ }
159
+ let rawBytes;
160
+ try {
161
+ rawBytes = statSync(realPath).size;
162
+ }
163
+ catch {
164
+ continue;
165
+ }
166
+ let content;
167
+ try {
168
+ content = readFileSync(realPath, 'utf8');
169
+ }
170
+ catch {
171
+ continue;
172
+ }
173
+ let truncated = false;
174
+ if (Buffer.byteLength(content, 'utf8') > MAX_FILE_BYTES) {
175
+ // Trim by character index sized to the byte cap. Mild over-trim
176
+ // on multi-byte boundaries is acceptable — we never under-trim
177
+ // (we'd exceed the cap) and the truncation is operator-visible
178
+ // via the `truncated` flag.
179
+ content = content.slice(0, MAX_FILE_BYTES);
180
+ truncated = true;
181
+ }
182
+ return {
183
+ path: realPath,
184
+ content,
185
+ level,
186
+ source,
187
+ truncated,
188
+ rawBytes,
189
+ };
190
+ }
191
+ return undefined;
192
+ }
193
+ /**
194
+ * Bound the limit to `[0, MAX_WALK_DEPTH]`. A negative or zero value
195
+ * still permits the cwd-level file to load (level 0 is always
196
+ * considered) — passing `limit: 0` means "current directory only".
197
+ */
198
+ function clampLimit(limit) {
199
+ if (typeof limit !== 'number' || !Number.isFinite(limit))
200
+ return MAX_WALK_DEPTH;
201
+ if (limit < 0)
202
+ return 0;
203
+ if (limit > MAX_WALK_DEPTH)
204
+ return MAX_WALK_DEPTH;
205
+ return Math.floor(limit);
206
+ }
207
+ //# sourceMappingURL=walk-up.js.map