@pugi/cli 0.1.0-beta.8 → 0.1.0-beta.87

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 (402) hide show
  1. package/CHANGELOG.md +96 -0
  2. package/THIRD_PARTY_NOTICES.md +40 -0
  3. package/assets/pugi-prozr2-mascot.ansi +9 -0
  4. package/bin/run.js +33 -1
  5. package/dist/commands/deploy.js +40 -40
  6. package/dist/commands/flatten.js +191 -0
  7. package/dist/commands/jobs-watch.js +201 -0
  8. package/dist/commands/jobs.js +42 -27
  9. package/dist/commands/smoke.js +133 -0
  10. package/dist/core/agent-progress/cleanup.js +134 -0
  11. package/dist/core/agent-progress/schema.js +144 -0
  12. package/dist/core/agent-progress/writer.js +101 -0
  13. package/dist/core/agents/adaptive-router.js +330 -0
  14. package/dist/core/agents/query-decomposer.js +297 -0
  15. package/dist/core/agents/registry.js +2 -2
  16. package/dist/core/approvals/shortcut-resolver.js +98 -0
  17. package/dist/core/artifact-chain/dispatcher.js +148 -0
  18. package/dist/core/artifact-chain/exporter.js +164 -0
  19. package/dist/core/artifact-chain/state.js +243 -0
  20. package/dist/core/artifact-chain/steps.js +169 -0
  21. package/dist/core/ask-user/question.js +92 -0
  22. package/dist/core/audit/audit-trail.js +275 -0
  23. package/dist/core/auth/ensure-authenticated.js +129 -0
  24. package/dist/core/auth/env-provider.js +238 -0
  25. package/dist/core/auto-open-browser.js +4 -4
  26. package/dist/core/auto-update/channels.js +122 -0
  27. package/dist/core/auto-update/checker.js +241 -0
  28. package/dist/core/auto-update/state.js +235 -0
  29. package/dist/core/bare-mode/index.js +107 -0
  30. package/dist/core/bash/redirect.js +281 -0
  31. package/dist/core/bash-classifier.js +436 -40
  32. package/dist/core/checkpoint/resumer.js +149 -0
  33. package/dist/core/checkpoint/rewinder.js +291 -0
  34. package/dist/core/checkpoints/shadow-git.js +670 -0
  35. package/dist/core/citations/parser.js +109 -0
  36. package/dist/core/classifier/yolo-classifier.js +88 -0
  37. package/dist/core/codegraph/decision-store.js +248 -0
  38. package/dist/core/codegraph/detect-repo.js +459 -0
  39. package/dist/core/codegraph/install.js +134 -0
  40. package/dist/core/codegraph/offer-hook.js +220 -0
  41. package/dist/core/compact/auto-trigger.js +96 -0
  42. package/dist/core/compact/buffer-rewriter.js +115 -0
  43. package/dist/core/compact/summarizer.js +208 -0
  44. package/dist/core/compact/token-counter.js +108 -0
  45. package/dist/core/consensus/anvil-fanout.js +25 -25
  46. package/dist/core/consensus/diff-capture.js +121 -12
  47. package/dist/core/consensus/rubric.js +21 -21
  48. package/dist/core/context/builder.js +6 -6
  49. package/dist/core/context/compaction-events.js +8 -8
  50. package/dist/core/context/compaction.js +31 -31
  51. package/dist/core/context/index.js +15 -8
  52. package/dist/core/context/invariants.js +51 -51
  53. package/dist/core/context/markdown-loader.js +28 -10
  54. package/dist/core/context/markdown-traverse.js +255 -0
  55. package/dist/core/context/pugiignore.js +41 -41
  56. package/dist/core/context/repo-skeleton.js +37 -37
  57. package/dist/core/context/tool-eviction.js +55 -0
  58. package/dist/core/context/watcher.js +32 -32
  59. package/dist/core/context/working-set.js +23 -23
  60. package/dist/core/coordinator/agent-tools.js +77 -0
  61. package/dist/core/coordinator/agent-toolset.js +65 -0
  62. package/dist/core/coordinator/fsm.js +73 -0
  63. package/dist/core/coordinator/mode-fsm.js +70 -0
  64. package/dist/core/cost/rate-card.js +129 -0
  65. package/dist/core/cost/tracker.js +221 -0
  66. package/dist/core/credentials.js +12 -12
  67. package/dist/core/cron/scheduler.js +138 -0
  68. package/dist/core/denial-tracking/index.js +8 -0
  69. package/dist/core/denial-tracking/state.js +264 -0
  70. package/dist/core/diagnostics/probe-runner.js +93 -0
  71. package/dist/core/diagnostics/probes/api.js +46 -0
  72. package/dist/core/diagnostics/probes/auth.js +93 -0
  73. package/dist/core/diagnostics/probes/bare-mode.js +42 -0
  74. package/dist/core/diagnostics/probes/cli-version.js +127 -0
  75. package/dist/core/diagnostics/probes/config.js +72 -0
  76. package/dist/core/diagnostics/probes/denial-tracking.js +57 -0
  77. package/dist/core/diagnostics/probes/disk.js +81 -0
  78. package/dist/core/diagnostics/probes/engine-live.js +46 -0
  79. package/dist/core/diagnostics/probes/git.js +65 -0
  80. package/dist/core/diagnostics/probes/hooks.js +118 -0
  81. package/dist/core/diagnostics/probes/mcp.js +75 -0
  82. package/dist/core/diagnostics/probes/node.js +59 -0
  83. package/dist/core/diagnostics/probes/pnpm.js +36 -0
  84. package/dist/core/diagnostics/probes/pugi-md.js +89 -0
  85. package/dist/core/diagnostics/probes/sandbox.js +40 -0
  86. package/dist/core/diagnostics/probes/session.js +74 -0
  87. package/dist/core/diagnostics/probes/status-snapshot.js +488 -0
  88. package/dist/core/diagnostics/probes/workspace.js +63 -0
  89. package/dist/core/diagnostics/types.js +70 -0
  90. package/dist/core/dispatch/cache-cleanup.js +197 -0
  91. package/dist/core/dispatch/cache-handoff.js +295 -0
  92. package/dist/core/edits/apply-patch-layer-e.js +189 -0
  93. package/dist/core/edits/dispatch.js +293 -7
  94. package/dist/core/edits/format-matrix.js +26 -0
  95. package/dist/core/edits/fuzzy-ladder.js +650 -0
  96. package/dist/core/edits/index.js +3 -1
  97. package/dist/core/edits/journal.js +199 -0
  98. package/dist/core/edits/layer-a-apply.js +15 -15
  99. package/dist/core/edits/layer-a-fuzzy-apply.js +198 -0
  100. package/dist/core/edits/layer-b-apply.js +9 -9
  101. package/dist/core/edits/layer-c-apply.js +6 -6
  102. package/dist/core/edits/layer-d-ast.js +557 -14
  103. package/dist/core/edits/marker-parser.js +12 -12
  104. package/dist/core/edits/security-gate.js +27 -27
  105. package/dist/core/edits/verify-hook.js +273 -0
  106. package/dist/core/edits/worktree.js +322 -0
  107. package/dist/core/engine/anvil-client.js +140 -26
  108. package/dist/core/engine/auto-compact.js +179 -0
  109. package/dist/core/engine/budgets.js +186 -0
  110. package/dist/core/engine/context-prefix.js +155 -0
  111. package/dist/core/engine/index.js +1 -1
  112. package/dist/core/engine/intensity.js +158 -0
  113. package/dist/core/engine/intent.js +260 -0
  114. package/dist/core/engine/native-pugi.js +1295 -227
  115. package/dist/core/engine/prompts.js +134 -16
  116. package/dist/core/engine/strip-internal-fields.js +124 -0
  117. package/dist/core/engine/tool-bridge.js +1295 -59
  118. package/dist/core/evaluation/golden-dataset.js +293 -0
  119. package/dist/core/feedback/queue.js +177 -0
  120. package/dist/core/feedback/submitter.js +145 -0
  121. package/dist/core/file-cache.js +113 -1
  122. package/dist/core/flatten/flatten-repo.js +439 -0
  123. package/dist/core/format/osc8-link.js +28 -0
  124. package/dist/core/hook-chains.js +392 -0
  125. package/dist/core/hooks/citation-verify-hook.js +138 -0
  126. package/dist/core/hooks/citation-verify.js +112 -0
  127. package/dist/core/hooks/events.js +44 -0
  128. package/dist/core/hooks/index.js +15 -0
  129. package/dist/core/hooks/registry.js +213 -0
  130. package/dist/core/hooks/runner.js +236 -0
  131. package/dist/core/hooks/v2/event-emitter.js +115 -0
  132. package/dist/core/hooks/v2/executor.js +282 -0
  133. package/dist/core/hooks/v2/index.js +25 -0
  134. package/dist/core/hooks/v2/lifecycle.js +104 -0
  135. package/dist/core/hooks/v2/loader.js +216 -0
  136. package/dist/core/hooks/v2/matcher.js +125 -0
  137. package/dist/core/hooks/v2/trust.js +143 -0
  138. package/dist/core/hooks/v2/types.js +86 -0
  139. package/dist/core/image/renderer.js +71 -0
  140. package/dist/core/init/detector.js +582 -0
  141. package/dist/core/init/template-renderer.js +242 -0
  142. package/dist/core/jobs/registry.js +18 -18
  143. package/dist/core/ledger/results-tsv.js +142 -0
  144. package/dist/core/log-discipline/stdout-redirect.js +51 -0
  145. package/dist/core/lsp/cache.js +105 -0
  146. package/dist/core/lsp/client.js +776 -0
  147. package/dist/core/lsp/language-detect.js +66 -0
  148. package/dist/core/lsp/post-edit-diagnostics.js +171 -0
  149. package/dist/core/lsp/symbol-tools.js +372 -0
  150. package/dist/core/mcp/client.js +97 -28
  151. package/dist/core/mcp/http-server.js +553 -0
  152. package/dist/core/mcp/orchestrator-tools.js +662 -0
  153. package/dist/core/mcp/permission.js +190 -0
  154. package/dist/core/mcp/registry.js +39 -17
  155. package/dist/core/mcp/server-tools.js +219 -0
  156. package/dist/core/mcp/server.js +397 -0
  157. package/dist/core/mcp/trust.js +10 -10
  158. package/dist/core/memory/dual-write.js +416 -0
  159. package/dist/core/memory/passive-extract.js +130 -0
  160. package/dist/core/memory/phase1-kinds.js +20 -0
  161. package/dist/core/memory/secret-scanner.js +304 -0
  162. package/dist/core/memory-sync/queue.js +170 -0
  163. package/dist/core/metrics/extract.js +113 -0
  164. package/dist/core/modes/roo-modes.js +68 -0
  165. package/dist/core/onboarding/ensure-initialized.js +133 -0
  166. package/dist/core/onboarding/marker.js +111 -0
  167. package/dist/core/onboarding/telemetry-state.js +108 -0
  168. package/dist/core/output-style/presets.js +176 -0
  169. package/dist/core/output-style/state.js +185 -0
  170. package/dist/core/path-security.js +287 -5
  171. package/dist/core/permission.js +82 -22
  172. package/dist/core/permissions/auto-classifier.js +124 -0
  173. package/dist/core/permissions/bash-parser.js +371 -0
  174. package/dist/core/permissions/circuit-breaker.js +83 -0
  175. package/dist/core/permissions/constrained-edit.js +91 -0
  176. package/dist/core/permissions/gate.js +278 -0
  177. package/dist/core/permissions/index.js +20 -0
  178. package/dist/core/permissions/mode.js +174 -0
  179. package/dist/core/permissions/network-egress.js +137 -0
  180. package/dist/core/permissions/state.js +241 -0
  181. package/dist/core/permissions/tool-class.js +93 -0
  182. package/dist/core/plan-mode/ui-state.js +51 -0
  183. package/dist/core/plans/plan-artifact.js +721 -0
  184. package/dist/core/policy-limits/etag-store.js +122 -0
  185. package/dist/core/prd-check/parser.js +215 -0
  186. package/dist/core/prd-check/reporter.js +127 -0
  187. package/dist/core/prd-check/session-review.js +557 -0
  188. package/dist/core/prd-check/verifiers.js +223 -0
  189. package/dist/core/prompt-cache/client-cache.js +99 -0
  190. package/dist/core/prompts/assembly.js +29 -0
  191. package/dist/core/prompts/registry.js +364 -0
  192. package/dist/core/pugi-md/cc-compat-rules.js +735 -0
  193. package/dist/core/pugi-md/context-injector.js +76 -0
  194. package/dist/core/pugi-md/walk-up.js +207 -0
  195. package/dist/core/python/uv-installer.js +270 -0
  196. package/dist/core/python/uv-resolver.js +83 -0
  197. package/dist/core/rate-limit/narrator.js +146 -0
  198. package/dist/core/recipes/cli-types.js +20 -0
  199. package/dist/core/recipes/loader.js +103 -0
  200. package/dist/core/recipes/runner.js +345 -0
  201. package/dist/core/recipes/schema.js +587 -0
  202. package/dist/core/release-notes/parser.js +241 -0
  203. package/dist/core/release-notes/state.js +116 -0
  204. package/dist/core/repl/ask.js +37 -37
  205. package/dist/core/repl/cancellation.js +26 -26
  206. package/dist/core/repl/cap-warning.js +4 -4
  207. package/dist/core/repl/clipboard-read.js +11 -11
  208. package/dist/core/repl/dispatch-fsm.js +12 -12
  209. package/dist/core/repl/history-search.js +15 -15
  210. package/dist/core/repl/history.js +28 -18
  211. package/dist/core/repl/kill-ring.js +5 -5
  212. package/dist/core/repl/model-pricing.js +135 -0
  213. package/dist/core/repl/privacy-banner.js +22 -22
  214. package/dist/core/repl/session.js +2157 -214
  215. package/dist/core/repl/slash-commands.js +533 -40
  216. package/dist/core/repl/store/index.js +1 -1
  217. package/dist/core/repl/store/jsonl-log.js +22 -22
  218. package/dist/core/repl/store/lockfile.js +10 -10
  219. package/dist/core/repl/store/session-store.js +136 -107
  220. package/dist/core/repl/store/types.js +15 -15
  221. package/dist/core/repl/store/uuid-v7.js +12 -12
  222. package/dist/core/repl/workspace-context.js +43 -21
  223. package/dist/core/repo-map/build.js +125 -0
  224. package/dist/core/repo-map/cache.js +185 -0
  225. package/dist/core/repo-map/extractor.js +254 -0
  226. package/dist/core/repo-map/formatter.js +145 -0
  227. package/dist/core/repo-map/page-rank.js +105 -0
  228. package/dist/core/repo-map/scanner.js +211 -0
  229. package/dist/core/retry-budget/budget.js +284 -0
  230. package/dist/core/retry-budget/index.js +5 -0
  231. package/dist/core/retry-budget/retry-cap.js +74 -0
  232. package/dist/core/routing/lead-worker.js +43 -0
  233. package/dist/core/routing/pre-flight-estimator.js +108 -0
  234. package/dist/core/runs/run-tree.js +103 -0
  235. package/dist/core/security/injection-scanner.js +367 -0
  236. package/dist/core/security/output-filter.js +418 -0
  237. package/dist/core/session/env-file.js +105 -0
  238. package/dist/core/session/section-budgets.js +140 -0
  239. package/dist/core/session.js +92 -0
  240. package/dist/core/settings.js +286 -5
  241. package/dist/core/share/formatter.js +271 -0
  242. package/dist/core/share/redactor.js +221 -0
  243. package/dist/core/share/uploader.js +267 -0
  244. package/dist/core/skills/defaults.js +457 -0
  245. package/dist/core/skills/loader.js +22 -22
  246. package/dist/core/skills/sources.js +27 -27
  247. package/dist/core/smoke/headless-driver.js +174 -0
  248. package/dist/core/smoke/orchestrator.js +194 -0
  249. package/dist/core/smoke/runner.js +238 -0
  250. package/dist/core/smoke/scenario-parser.js +316 -0
  251. package/dist/core/statusline.js +99 -0
  252. package/dist/core/subagents/dispatcher-real.js +600 -0
  253. package/dist/core/subagents/dispatcher.js +132 -43
  254. package/dist/core/subagents/index.js +19 -6
  255. package/dist/core/subagents/isolation-matrix.js +213 -0
  256. package/dist/core/subagents/spawn.js +19 -4
  257. package/dist/core/telemetry/emitter.js +229 -0
  258. package/dist/core/telemetry/queue.js +251 -0
  259. package/dist/core/theme/context.js +91 -0
  260. package/dist/core/theme/presets.js +228 -0
  261. package/dist/core/theme/state.js +181 -0
  262. package/dist/core/todos/invariant.js +10 -0
  263. package/dist/core/todos/state.js +177 -0
  264. package/dist/core/tool-schema/compressor.js +89 -0
  265. package/dist/core/transport/version-interceptor.js +166 -0
  266. package/dist/core/trust.js +2 -2
  267. package/dist/core/tui/thinking-block.js +64 -0
  268. package/dist/core/vim/keymap.js +288 -0
  269. package/dist/core/vim/state.js +92 -0
  270. package/dist/core/watch-markers/marker-watcher.js +133 -0
  271. package/dist/core/worktree-manager/cleanup.js +123 -0
  272. package/dist/core/worktree-manager/manager.js +303 -0
  273. package/dist/index.js +28 -0
  274. package/dist/runtime/bootstrap.js +190 -0
  275. package/dist/runtime/cli.js +4151 -489
  276. package/dist/runtime/commands/agents.js +30 -30
  277. package/dist/runtime/commands/budget.js +5 -5
  278. package/dist/runtime/commands/cancel.js +231 -0
  279. package/dist/runtime/commands/chain.js +489 -0
  280. package/dist/runtime/commands/codegraph-status.js +227 -0
  281. package/dist/runtime/commands/compact.js +297 -0
  282. package/dist/runtime/commands/config.js +32 -32
  283. package/dist/runtime/commands/cost.js +199 -0
  284. package/dist/runtime/commands/delegate.js +244 -13
  285. package/dist/runtime/commands/dispatch.js +126 -0
  286. package/dist/runtime/commands/doctor.js +579 -0
  287. package/dist/runtime/commands/feedback.js +184 -0
  288. package/dist/runtime/commands/hooks.js +184 -0
  289. package/dist/runtime/commands/init.js +254 -0
  290. package/dist/runtime/commands/lsp.js +368 -0
  291. package/dist/runtime/commands/mcp.js +879 -0
  292. package/dist/runtime/commands/memory.js +582 -0
  293. package/dist/runtime/commands/model.js +237 -0
  294. package/dist/runtime/commands/onboarding.js +275 -0
  295. package/dist/runtime/commands/patch.js +128 -0
  296. package/dist/runtime/commands/permissions.js +112 -0
  297. package/dist/runtime/commands/plan.js +143 -0
  298. package/dist/runtime/commands/prd-check.js +285 -0
  299. package/dist/runtime/commands/privacy.js +17 -17
  300. package/dist/runtime/commands/recipe.js +325 -0
  301. package/dist/runtime/commands/redo-blob-store.js +92 -0
  302. package/dist/runtime/commands/redo.js +361 -0
  303. package/dist/runtime/commands/release-notes.js +229 -0
  304. package/dist/runtime/commands/repo-map.js +95 -0
  305. package/dist/runtime/commands/report.js +299 -0
  306. package/dist/runtime/commands/resume.js +118 -0
  307. package/dist/runtime/commands/review-consensus.js +68 -53
  308. package/dist/runtime/commands/rewind.js +333 -0
  309. package/dist/runtime/commands/roster.js +14 -14
  310. package/dist/runtime/commands/sessions.js +163 -0
  311. package/dist/runtime/commands/share.js +316 -0
  312. package/dist/runtime/commands/skills.js +31 -31
  313. package/dist/runtime/commands/status.js +186 -0
  314. package/dist/runtime/commands/stickers.js +82 -0
  315. package/dist/runtime/commands/style.js +194 -0
  316. package/dist/runtime/commands/theme.js +196 -0
  317. package/dist/runtime/commands/undo.js +54 -22
  318. package/dist/runtime/commands/update.js +289 -0
  319. package/dist/runtime/commands/vim.js +140 -0
  320. package/dist/runtime/commands/worktree.js +177 -0
  321. package/dist/runtime/commands/worktrees.js +155 -0
  322. package/dist/runtime/headless-repl.js +195 -0
  323. package/dist/runtime/headless.js +543 -0
  324. package/dist/runtime/load-hooks-or-exit.js +71 -0
  325. package/dist/runtime/plan-decompose.js +531 -0
  326. package/dist/runtime/update-check.js +28 -28
  327. package/dist/runtime/version.js +65 -0
  328. package/dist/skills/bundled/batch.js +617 -0
  329. package/dist/skills/bundled/index.js +45 -0
  330. package/dist/skills/bundled/loop.js +358 -0
  331. package/dist/skills/bundled/remember.js +383 -0
  332. package/dist/skills/bundled/simplify.js +289 -0
  333. package/dist/skills/bundled/skillify.js +373 -0
  334. package/dist/skills/bundled/stuck.js +558 -0
  335. package/dist/skills/bundled/verify.js +439 -0
  336. package/dist/testing/vcr.js +486 -0
  337. package/dist/tools/agent-tool.js +229 -0
  338. package/dist/tools/apply-patch.js +556 -0
  339. package/dist/tools/ask-user-question.js +222 -0
  340. package/dist/tools/ask-user.js +115 -0
  341. package/dist/tools/bash.js +623 -45
  342. package/dist/tools/brief.js +224 -0
  343. package/dist/tools/enter-worktree.js +250 -0
  344. package/dist/tools/exit-worktree.js +147 -0
  345. package/dist/tools/file-tools.js +161 -44
  346. package/dist/tools/lsp-tools.js +189 -0
  347. package/dist/tools/mcp-tool.js +260 -0
  348. package/dist/tools/multi-edit.js +361 -0
  349. package/dist/tools/powershell.js +268 -0
  350. package/dist/tools/registry.js +85 -0
  351. package/dist/tools/skill-tool.js +96 -0
  352. package/dist/tools/sleep.js +99 -0
  353. package/dist/tools/synthetic-output.js +133 -0
  354. package/dist/tools/tasks.js +208 -0
  355. package/dist/tools/todo-write.js +184 -0
  356. package/dist/tools/verify-plan-execution.js +295 -0
  357. package/dist/tools/web-fetch-injection-scanner.js +207 -0
  358. package/dist/tools/web-fetch.js +195 -10
  359. package/dist/tools/web-search.js +458 -0
  360. package/dist/tui/agent-progress-card.js +111 -0
  361. package/dist/tui/agent-tree.js +11 -1
  362. package/dist/tui/ask-modal.js +14 -14
  363. package/dist/tui/ask-user-question-prompt.js +203 -0
  364. package/dist/tui/compact-banner.js +81 -0
  365. package/dist/tui/conversation-pane.js +85 -11
  366. package/dist/tui/cost-table.js +111 -0
  367. package/dist/tui/device-flow.js +2 -2
  368. package/dist/tui/doctor-table.js +46 -0
  369. package/dist/tui/feedback-prompt.js +156 -0
  370. package/dist/tui/input-box.js +247 -32
  371. package/dist/tui/login-picker.js +3 -3
  372. package/dist/tui/markdown-render.js +6 -6
  373. package/dist/tui/onboarding-wizard.js +240 -0
  374. package/dist/tui/permissions-picker.js +86 -0
  375. package/dist/tui/render.js +35 -0
  376. package/dist/tui/repl-render.js +332 -54
  377. package/dist/tui/repl-splash-art.js +16 -16
  378. package/dist/tui/repl-splash-mascot.js +48 -24
  379. package/dist/tui/repl-splash.js +22 -22
  380. package/dist/tui/repl.js +124 -44
  381. package/dist/tui/slash-palette.js +6 -6
  382. package/dist/tui/splash.js +2 -2
  383. package/dist/tui/status-bar.js +109 -31
  384. package/dist/tui/status-table.js +7 -0
  385. package/dist/tui/stickers-art.js +136 -0
  386. package/dist/tui/style-table.js +28 -0
  387. package/dist/tui/theme-table.js +29 -0
  388. package/dist/tui/thinking-spinner.js +123 -0
  389. package/dist/tui/tool-stream-pane.js +53 -4
  390. package/dist/tui/update-banner.js +27 -2
  391. package/dist/tui/vim-input.js +267 -0
  392. package/dist/tui/welcome-banner.js +107 -0
  393. package/dist/tui/welcome-data.js +293 -0
  394. package/dist/tui/workspace-context.js +2 -2
  395. package/docs/examples/codegraph.mcp.json +10 -0
  396. package/package.json +23 -6
  397. package/test/scenarios/codegen-create-file.scenario.txt +13 -0
  398. package/test/scenarios/compact-force.scenario.txt +11 -0
  399. package/test/scenarios/identity.scenario.txt +11 -0
  400. package/test/scenarios/persona-handoff.scenario.txt +11 -0
  401. package/test/scenarios/walkback.scenario.txt +12 -0
  402. package/dist/core/engine/compaction-hook.js +0 -154
@@ -0,0 +1,66 @@
1
+ /**
2
+ * Language-from-extension detection — .
3
+ *
4
+ * Single source of truth for "given a file path, which `LspLanguage`
5
+ * slug do we route to". The `runtime/commands/lsp.ts` shipped its
6
+ * own inline `inferLanguage` switch; L15 (post-edit auto-diagnostics)
7
+ * needs the same lookup from `core/engine/tool-bridge.ts`, so we lift
8
+ * the table into a dedicated module to avoid a second copy drifting
9
+ * out of sync.
10
+ *
11
+ * Returning `undefined` is the calling code's signal to silently skip
12
+ * LSP — an unsupported extension is NOT an error, it just means "no
13
+ * diagnostics for this file". The tool-bridge hook treats this as a
14
+ * no-op envelope tail.
15
+ *
16
+ * Adding a new language requires THREE coordinated changes:
17
+ * 1. Add the `LspLanguage` slug + server descriptor in `client.ts`.
18
+ * 2. Map its extensions here.
19
+ * 3. Add a `lsp-language-matrix` spec row exercising the new ext.
20
+ *
21
+ * Brand voice: ASCII only, no emoji, no banned words.
22
+ */
23
+ import { extname } from 'node:path';
24
+ /**
25
+ * Lower-case extension (including the dot) → LSP language slug.
26
+ * Mirror of the switch in `runtime/commands/lsp.ts::inferLanguage`.
27
+ * The table form lets tests assert coverage and lets new languages
28
+ * land with one edit instead of two.
29
+ */
30
+ export const EXTENSION_TO_LANGUAGE = {
31
+ '.ts': 'ts',
32
+ '.tsx': 'ts',
33
+ '.mts': 'ts',
34
+ '.cts': 'ts',
35
+ '.js': 'js',
36
+ '.jsx': 'js',
37
+ '.mjs': 'js',
38
+ '.cjs': 'js',
39
+ '.py': 'py',
40
+ '.pyi': 'py',
41
+ '.go': 'go',
42
+ '.rs': 'rust',
43
+ };
44
+ /**
45
+ * Infer the `LspLanguage` for a workspace-relative or absolute path.
46
+ * Returns `undefined` for unmapped extensions — the caller decides
47
+ * whether that is silently skipped (post-edit hook) or surfaced as
48
+ * `language_unsupported` (`pugi lsp` CLI).
49
+ */
50
+ export function languageForFile(file) {
51
+ const ext = extname(file).toLowerCase();
52
+ if (!ext)
53
+ return undefined;
54
+ return EXTENSION_TO_LANGUAGE[ext];
55
+ }
56
+ /**
57
+ * Return every extension currently mapped to the given language.
58
+ * Used by the matrix spec to assert coverage without re-typing the
59
+ * extension list.
60
+ */
61
+ export function extensionsForLanguage(lang) {
62
+ return Object.entries(EXTENSION_TO_LANGUAGE)
63
+ .filter(([, value]) => value === lang)
64
+ .map(([ext]) => ext);
65
+ }
66
+ //# sourceMappingURL=language-detect.js.map
@@ -0,0 +1,171 @@
1
+ /**
2
+ * Post-edit diagnostics — .
3
+ *
4
+ * the upstream tool's leak intel surfaced this pattern: after a `FileEdit` /
5
+ * `Write` tool call lands, an LSP diagnostic pass runs on the touched
6
+ * file and the result is appended to the tool envelope before the
7
+ * model sees it. The model then self-corrects in the same turn —
8
+ * "TS2304: Cannot find name 'undef'" comes back, the model fixes the
9
+ * typo in the next tool call, no operator round-trip needed.
10
+ *
11
+ * This module is the Pugi side of that pattern:
12
+ *
13
+ * 1. The tool-bridge calls `runPostEditDiagnostics(path, ctx)` after
14
+ * a successful `edit` / `write` / `multi_edit`.
15
+ * 2. We infer the language from the extension (`language-detect`).
16
+ * Unsupported extension → `{ skip: true }` and the bridge appends
17
+ * nothing.
18
+ * 3. We borrow (or lazily start) the per-language cached client
19
+ * from `cache.ts`. A spawn failure → `{ skip: true }` and the
20
+ * envelope stays clean. Silence on failure is intentional: an
21
+ * operator who has not installed `typescript-language-server`
22
+ * should not see an LSP nag on every edit.
23
+ * 4. We pull diagnostics with a hard 5s ceiling. A timeout logs a
24
+ * warning on stderr (gated on `PUGI_LSP_DEBUG=1`) and skips —
25
+ * the envelope is never blocked on LSP.
26
+ * 5. We format the surviving diagnostics into a readable tail
27
+ * mirroring the leak format:
28
+ *
29
+ * LSP DIAGNOSTICS (typescript):
30
+ * foo.ts:42:5 error TS2304: Cannot find name 'undef'.
31
+ * foo.ts:51:1 warn TS6133: 'unused' is declared.
32
+ *
33
+ * The bridge concatenates this tail onto its existing `wrote ...` /
34
+ * `edited ...` body with a single newline separator. When there are
35
+ * zero diagnostics we return `{ skip: true }` so the existing body
36
+ * is unchanged — the "no news is good news" path stays terse.
37
+ *
38
+ * Brand voice: ASCII only, no emoji, no banned words.
39
+ */
40
+ import { isAbsolute, relative, resolve } from 'node:path';
41
+ import { getOrStartLspClient } from './cache.js';
42
+ import { languageForFile } from './language-detect.js';
43
+ const DEFAULT_TIMEOUT_MS = 5_000;
44
+ /**
45
+ * Hard cap on how many diagnostics we surface to the model. A file
46
+ * with 200 errors after a broken bulk edit would otherwise blow the
47
+ * context window; the model can re-run `pugi lsp diagnostics` if
48
+ * it needs the full list.
49
+ */
50
+ const MAX_DIAGNOSTICS = 25;
51
+ export async function runPostEditDiagnostics(filePath, opts) {
52
+ const lang = languageForFile(filePath);
53
+ if (!lang) {
54
+ return { skip: true, reason: 'unsupported_language' };
55
+ }
56
+ const timeoutMs = opts.timeoutMs ?? DEFAULT_TIMEOUT_MS;
57
+ const clientResult = await loadClient(lang, opts);
58
+ if (!clientResult.ok) {
59
+ return { skip: true, reason: mapStartFailure(clientResult.reason) };
60
+ }
61
+ // Run diagnostics with a hard timeout. The underlying LspClient has
62
+ // its own per-request timeout (5s default) but a slow handshake
63
+ // can blow past it; we belt-and-suspenders here so the agent loop
64
+ // never blocks on LSP.
65
+ const relPath = toWorkspaceRelative(filePath, opts.cwd);
66
+ const diagnosticsPromise = clientResult.client.diagnostics(relPath);
67
+ let timer;
68
+ const timeoutPromise = new Promise((resolveFn) => {
69
+ timer = setTimeout(() => resolveFn({ timedOut: true }), timeoutMs);
70
+ timer.unref();
71
+ });
72
+ const race = await Promise.race([
73
+ diagnosticsPromise.then((value) => ({ timedOut: false, value })),
74
+ timeoutPromise,
75
+ ]);
76
+ if (timer)
77
+ clearTimeout(timer);
78
+ if (race.timedOut) {
79
+ const writeFn = opts.debugWrite ?? ((line) => {
80
+ if (process.env.PUGI_LSP_DEBUG === '1')
81
+ process.stderr.write(`${line}\n`);
82
+ });
83
+ writeFn(`[pugi-lsp] post-edit diagnostics for ${relPath} timed out after ${timeoutMs}ms (lang=${lang})`);
84
+ return { skip: true, reason: 'timeout' };
85
+ }
86
+ const diag = race.value;
87
+ if (!diag.ok) {
88
+ return { skip: true, reason: 'lsp_error' };
89
+ }
90
+ if (diag.value.length === 0) {
91
+ return { skip: true, reason: 'no_diagnostics' };
92
+ }
93
+ const tail = formatDiagnosticsTail(relPath, lang, diag.value);
94
+ return { skip: false, tail, count: diag.value.length, language: lang };
95
+ }
96
+ async function loadClient(lang, opts) {
97
+ if (opts.clientLoader) {
98
+ return opts.clientLoader(lang);
99
+ }
100
+ const { cwd, timeoutMs, clientLoader: _ignoredA, debugWrite: _ignoredB, ...rest } = opts;
101
+ const result = await getOrStartLspClient(lang, { cwd, ...rest });
102
+ if (!result.ok) {
103
+ return { ok: false, reason: result.reason, detail: result.detail };
104
+ }
105
+ return { ok: true, client: result.client };
106
+ }
107
+ function mapStartFailure(reason) {
108
+ if (reason === 'lsp_unavailable' || reason === 'language_unsupported')
109
+ return 'lsp_unavailable';
110
+ if (reason === 'lsp_disabled')
111
+ return 'lsp_disabled';
112
+ return 'lsp_error';
113
+ }
114
+ /**
115
+ * Convert an absolute or workspace-relative path into the form the
116
+ * LSP client expects — same shape as `runtime/commands/lsp.ts` uses.
117
+ */
118
+ function toWorkspaceRelative(filePath, cwd) {
119
+ if (!isAbsolute(filePath))
120
+ return filePath;
121
+ const rel = relative(cwd, resolve(cwd, filePath));
122
+ return rel || filePath;
123
+ }
124
+ /**
125
+ * Format diagnostics into the leak-shaped envelope tail. Pure function
126
+ * exported for unit tests to assert the line format independent of
127
+ * any LSP plumbing.
128
+ */
129
+ export function formatDiagnosticsTail(relPath, lang, diagnostics) {
130
+ const visible = diagnostics.slice(0, MAX_DIAGNOSTICS);
131
+ const truncated = diagnostics.length > visible.length;
132
+ const lines = [`LSP DIAGNOSTICS (${LANGUAGE_LABELS[lang]}):`];
133
+ for (const diag of visible) {
134
+ const line = diag.range.start.line + 1; // LSP is zero-based; humans expect 1-based.
135
+ const col = diag.range.start.character + 1;
136
+ const severity = SEVERITY_LABELS[diag.severityLabel];
137
+ const code = diag.code !== undefined && diag.code !== '' ? ` ${diag.code}` : '';
138
+ const source = diag.source ? `${diag.source}` : '';
139
+ const head = source ? `${severity}${code} (${source}):` : `${severity}${code}:`;
140
+ lines.push(` ${relPath}:${line}:${col} ${head} ${diag.message}`);
141
+ }
142
+ if (truncated) {
143
+ lines.push(` ... ${diagnostics.length - visible.length} more diagnostic(s) — re-run pugi lsp diagnostics ${relPath} for the full list`);
144
+ }
145
+ return lines.join('\n');
146
+ }
147
+ const LANGUAGE_LABELS = {
148
+ ts: 'typescript',
149
+ js: 'javascript',
150
+ py: 'python',
151
+ go: 'go',
152
+ rust: 'rust',
153
+ };
154
+ /**
155
+ * Map LSP severity label → the short token the leak envelope uses.
156
+ * "warn" is shorter than "warning" and matches the upstream tool's leak
157
+ * verbatim; the rest mirror LSP terminology.
158
+ */
159
+ const SEVERITY_LABELS = {
160
+ error: 'error',
161
+ warning: 'warn ',
162
+ info: 'info ',
163
+ hint: 'hint ',
164
+ };
165
+ /** Test-only surface so specs can poke the pure helpers without LSP. */
166
+ export const __test__ = {
167
+ formatDiagnosticsTail,
168
+ MAX_DIAGNOSTICS,
169
+ DEFAULT_TIMEOUT_MS,
170
+ };
171
+ //# sourceMappingURL=post-edit-diagnostics.js.map
@@ -0,0 +1,372 @@
1
+ /**
2
+ * LSP `SymbolKind` enum — mirrors the official numbering from the LSP
3
+ * spec §3.17.7. We re-export the integers (not the names) because the
4
+ * Aider repomap stores numeric kinds verbatim. Callers that want a
5
+ * human label use `symbolKindLabel`.
6
+ */
7
+ export const SYMBOL_KIND = {
8
+ File: 1,
9
+ Module: 2,
10
+ Namespace: 3,
11
+ Package: 4,
12
+ Class: 5,
13
+ Method: 6,
14
+ Property: 7,
15
+ Field: 8,
16
+ Constructor: 9,
17
+ Enum: 10,
18
+ Interface: 11,
19
+ Function: 12,
20
+ Variable: 13,
21
+ Constant: 14,
22
+ String: 15,
23
+ Number: 16,
24
+ Boolean: 17,
25
+ Array: 18,
26
+ Object: 19,
27
+ Key: 20,
28
+ Null: 21,
29
+ EnumMember: 22,
30
+ Struct: 23,
31
+ Event: 24,
32
+ Operator: 25,
33
+ TypeParameter: 26,
34
+ };
35
+ const SYMBOL_KIND_LABELS = {
36
+ 1: 'file',
37
+ 2: 'module',
38
+ 3: 'namespace',
39
+ 4: 'package',
40
+ 5: 'class',
41
+ 6: 'method',
42
+ 7: 'property',
43
+ 8: 'field',
44
+ 9: 'constructor',
45
+ 10: 'enum',
46
+ 11: 'interface',
47
+ 12: 'function',
48
+ 13: 'variable',
49
+ 14: 'constant',
50
+ 15: 'string',
51
+ 16: 'number',
52
+ 17: 'boolean',
53
+ 18: 'array',
54
+ 19: 'object',
55
+ 20: 'key',
56
+ 21: 'null',
57
+ 22: 'enum-member',
58
+ 23: 'struct',
59
+ 24: 'event',
60
+ 25: 'operator',
61
+ 26: 'type-parameter',
62
+ };
63
+ /** Human-readable kind label (`'function'`, `'class'`, ...) for printing. */
64
+ export function symbolKindLabel(kind) {
65
+ if (typeof kind !== 'number')
66
+ return 'unknown';
67
+ return SYMBOL_KIND_LABELS[kind] ?? 'unknown';
68
+ }
69
+ /**
70
+ * Build a `SymbolToolsTransport` from a real `LspClient`. Production
71
+ * wire-up. `documentSymbols` / `workspaceSymbols` are omitted on
72
+ * purpose — the `LspClient` does not expose them, so the Phase 1
73
+ * surface degrades to `[]`. Phase 2 follows.
74
+ */
75
+ export function transportFromLspClient(client) {
76
+ return {
77
+ definition: (file, pos) => client.definition(file, pos),
78
+ references: (file, pos) => client.references(file, pos),
79
+ };
80
+ }
81
+ /**
82
+ * Resolve the first definition for the symbol at `(line, character)`
83
+ * in `filePath`. Returns `null` when:
84
+ *
85
+ * - `filePath` is empty / whitespace
86
+ * - the transport's `definition` call returns `ok: false`
87
+ * - the server reports an empty array
88
+ * - the transport itself throws (caught — never re-raised)
89
+ *
90
+ * Multi-hit definitions (TS overloads, polymorphic methods) collapse
91
+ * to the first location; the engine can call `findReferences` to fan
92
+ * out from there. Matches Aider's `repomap.find_definition` shape.
93
+ */
94
+ export async function findDefinition(transport, filePath, line, character) {
95
+ if (!isNonEmptyString(filePath))
96
+ return null;
97
+ if (!isFiniteNonNegativeInt(line) || !isFiniteNonNegativeInt(character)) {
98
+ return null;
99
+ }
100
+ let result;
101
+ try {
102
+ result = await transport.definition(filePath, { line, character });
103
+ }
104
+ catch {
105
+ return null;
106
+ }
107
+ if (!result.ok)
108
+ return null;
109
+ const first = result.value[0];
110
+ if (!first)
111
+ return null;
112
+ const file = pickFilePath(first);
113
+ if (!isNonEmptyString(file))
114
+ return null;
115
+ return {
116
+ file,
117
+ line: first.range.start.line,
118
+ character: first.range.start.character,
119
+ };
120
+ }
121
+ /**
122
+ * List every reference (including the declaration) for the symbol at
123
+ * `(line, character)` in `filePath`. Returns `[]` on any failure mode
124
+ * matching `findDefinition` plus the empty-array server response.
125
+ */
126
+ export async function findReferences(transport, filePath, line, character) {
127
+ if (!isNonEmptyString(filePath))
128
+ return [];
129
+ if (!isFiniteNonNegativeInt(line) || !isFiniteNonNegativeInt(character)) {
130
+ return [];
131
+ }
132
+ let result;
133
+ try {
134
+ result = await transport.references(filePath, { line, character });
135
+ }
136
+ catch {
137
+ return [];
138
+ }
139
+ if (!result.ok)
140
+ return [];
141
+ const out = [];
142
+ for (const loc of result.value) {
143
+ const file = pickFilePath(loc);
144
+ if (!isNonEmptyString(file))
145
+ continue;
146
+ out.push({
147
+ file,
148
+ line: loc.range.start.line,
149
+ character: loc.range.start.character,
150
+ });
151
+ }
152
+ return out;
153
+ }
154
+ /**
155
+ * Outline the symbols defined in `filePath`. Returns `[]` when:
156
+ *
157
+ * - `filePath` is empty
158
+ * - the transport does not implement `documentSymbols` (Phase 1
159
+ * default — the `LspClient` lacks the wire call)
160
+ * - the transport's `documentSymbols` call returns `ok: false`
161
+ * - the server returns an empty / malformed list
162
+ * - the transport throws
163
+ *
164
+ * Handles BOTH LSP response shapes: hierarchical `DocumentSymbol[]`
165
+ * (with `selectionRange` + `children`) and flat `SymbolInformation[]`
166
+ * (with `location.range`). Children are flattened into the same array
167
+ * so the engine sees one row per symbol regardless of nesting; the
168
+ * `containerName` field records the parent's name for hierarchy.
169
+ */
170
+ export async function listDocumentSymbols(transport, filePath) {
171
+ if (!isNonEmptyString(filePath))
172
+ return [];
173
+ if (!transport.documentSymbols)
174
+ return [];
175
+ let result;
176
+ try {
177
+ result = await transport.documentSymbols(filePath);
178
+ }
179
+ catch {
180
+ return [];
181
+ }
182
+ if (!result.ok)
183
+ return [];
184
+ const out = [];
185
+ for (const raw of result.value) {
186
+ walkDocumentSymbol(raw, undefined, out);
187
+ }
188
+ return out;
189
+ }
190
+ /**
191
+ * Workspace-wide symbol fuzzy search. Returns `[]` when:
192
+ *
193
+ * - `query` is empty / whitespace
194
+ * - the transport does not implement `workspaceSymbols`
195
+ * - the transport's call returns `ok: false`
196
+ * - the server returns an empty / malformed list
197
+ * - the transport throws
198
+ *
199
+ * Per LSP §3.17.13, the server is free to define its own matching
200
+ * algorithm (substring / fuzzy / prefix). We do not re-rank — the
201
+ * server's ranking is authoritative.
202
+ */
203
+ export async function findWorkspaceSymbol(transport, query) {
204
+ if (!isNonEmptyString(query))
205
+ return [];
206
+ if (!transport.workspaceSymbols)
207
+ return [];
208
+ let result;
209
+ try {
210
+ result = await transport.workspaceSymbols(query);
211
+ }
212
+ catch {
213
+ return [];
214
+ }
215
+ if (!result.ok)
216
+ return [];
217
+ const out = [];
218
+ for (const raw of result.value) {
219
+ const parsed = parseWorkspaceSymbol(raw);
220
+ if (parsed)
221
+ out.push(parsed);
222
+ }
223
+ return out;
224
+ }
225
+ // -- internals ---------------------------------------------------------------
226
+ function isNonEmptyString(value) {
227
+ return typeof value === 'string' && value.trim().length > 0;
228
+ }
229
+ function isFiniteNonNegativeInt(value) {
230
+ return typeof value === 'number' && Number.isFinite(value) && value >= 0;
231
+ }
232
+ /**
233
+ * Prefer the workspace-relative `path` from `client.ts::normalizeLocations`;
234
+ * fall back to the raw URI when the server reports an out-of-workspace
235
+ * location. Either form is a valid identifier for the caller — the
236
+ * engine prefers relative paths for prompts but URIs are still useful
237
+ * for tracing.
238
+ */
239
+ function pickFilePath(loc) {
240
+ if (isNonEmptyString(loc.path))
241
+ return loc.path;
242
+ if (isNonEmptyString(loc.uri))
243
+ return loc.uri;
244
+ return '';
245
+ }
246
+ /**
247
+ * Recursive walk over a (possibly hierarchical) `DocumentSymbol`.
248
+ * Emits one `DocumentSymbol` row per node — children inherit the
249
+ * parent's `name` as their `containerName`. Tolerates both
250
+ * `SymbolInformation` (flat, has `location.range`) and the modern
251
+ * hierarchical shape (`selectionRange` + `children`).
252
+ */
253
+ function walkDocumentSymbol(raw, parentName, out) {
254
+ if (!raw || typeof raw !== 'object')
255
+ return;
256
+ const name = typeof raw.name === 'string' ? raw.name : '';
257
+ const kind = typeof raw.kind === 'number' ? raw.kind : 0;
258
+ if (!name || !kind)
259
+ return;
260
+ const pos = pickDocumentSymbolPosition(raw);
261
+ if (!pos)
262
+ return;
263
+ const containerName = typeof raw.containerName === 'string' && raw.containerName.length > 0
264
+ ? raw.containerName
265
+ : parentName;
266
+ out.push({
267
+ name,
268
+ kind,
269
+ line: pos.line,
270
+ character: pos.character,
271
+ ...(containerName ? { containerName } : {}),
272
+ });
273
+ if (Array.isArray(raw.children)) {
274
+ for (const child of raw.children) {
275
+ walkDocumentSymbol(child, name, out);
276
+ }
277
+ }
278
+ }
279
+ /**
280
+ * LSP `DocumentSymbol` (hierarchical) carries `selectionRange.start`,
281
+ * the precise character position of the symbol name. The legacy
282
+ * `SymbolInformation` shape carries `location.range.start` instead.
283
+ * Tolerate both; prefer `selectionRange` when present (more precise).
284
+ */
285
+ function pickDocumentSymbolPosition(raw) {
286
+ const sel = pickRangeStart(raw.selectionRange);
287
+ if (sel)
288
+ return sel;
289
+ const range = pickRangeStart(raw.range);
290
+ if (range)
291
+ return range;
292
+ if (raw.location && typeof raw.location === 'object') {
293
+ const locRange = pickRangeStart(raw.location.range);
294
+ if (locRange)
295
+ return locRange;
296
+ }
297
+ return null;
298
+ }
299
+ function pickRangeStart(rawRange) {
300
+ if (!rawRange || typeof rawRange !== 'object')
301
+ return null;
302
+ const start = rawRange.start;
303
+ if (!start || typeof start !== 'object')
304
+ return null;
305
+ const line = start.line;
306
+ const character = start.character;
307
+ if (typeof line !== 'number' || typeof character !== 'number')
308
+ return null;
309
+ if (!Number.isFinite(line) || !Number.isFinite(character))
310
+ return null;
311
+ if (line < 0 || character < 0)
312
+ return null;
313
+ return { line, character };
314
+ }
315
+ function parseWorkspaceSymbol(raw) {
316
+ if (!raw || typeof raw !== 'object')
317
+ return null;
318
+ const name = typeof raw.name === 'string' ? raw.name : '';
319
+ const kind = typeof raw.kind === 'number' ? raw.kind : 0;
320
+ if (!name || !kind)
321
+ return null;
322
+ const location = raw.location;
323
+ if (!location || typeof location !== 'object')
324
+ return null;
325
+ const file = pickWorkspaceSymbolFile(location);
326
+ if (!file)
327
+ return null;
328
+ const pos = pickRangeStart(location.range);
329
+ if (!pos)
330
+ return null;
331
+ const containerName = typeof raw.containerName === 'string' && raw.containerName.length > 0
332
+ ? raw.containerName
333
+ : undefined;
334
+ return {
335
+ name,
336
+ file,
337
+ line: pos.line,
338
+ kind,
339
+ ...(containerName ? { containerName } : {}),
340
+ };
341
+ }
342
+ /**
343
+ * Workspace-symbol locations land as either a full `Location`
344
+ * (`{ uri, range }`) or LSP 3.17's `LocationLink` shape
345
+ * (`{ targetUri, targetRange, ... }`). Either form is acceptable;
346
+ * we surface the URI string verbatim and let the engine layer
347
+ * convert to a workspace-relative path when needed (mirrors how
348
+ * `client.ts::normalizeLocations` handles definition results).
349
+ */
350
+ function pickWorkspaceSymbolFile(location) {
351
+ const uri = location['uri'];
352
+ if (typeof uri === 'string' && uri.length > 0)
353
+ return uri;
354
+ const targetUri = location['targetUri'];
355
+ if (typeof targetUri === 'string' && targetUri.length > 0)
356
+ return targetUri;
357
+ return '';
358
+ }
359
+ /**
360
+ * Test-only surface — re-exports internals for direct assertion of
361
+ * parsing rules without running every primitive end-to-end. Keeping
362
+ * the seam typed (instead of using `@ts-ignore` in the spec) preserves
363
+ * `noImplicitAny` compliance project-wide.
364
+ */
365
+ export const __test__ = {
366
+ parseWorkspaceSymbol,
367
+ pickDocumentSymbolPosition,
368
+ pickRangeStart,
369
+ symbolKindLabel,
370
+ walkDocumentSymbol,
371
+ };
372
+ //# sourceMappingURL=symbol-tools.js.map