@unerr-ai/unerr 0.0.1 → 0.1.1

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 (556) hide show
  1. package/README.md +215 -35
  2. package/dist/__tests__/architecture-guard.test.js +122 -0
  3. package/dist/__tests__/arg-validator.test.js +205 -0
  4. package/dist/__tests__/ast-extractor.test.js +203 -0
  5. package/dist/__tests__/auto-bootstrap.test.js +280 -0
  6. package/dist/__tests__/background-indexer.test.js +228 -0
  7. package/dist/__tests__/blast-radius-engine.test.js +200 -0
  8. package/dist/__tests__/bridge-isolation.test.js +37 -0
  9. package/dist/__tests__/budget-enforcer.test.js +53 -0
  10. package/dist/__tests__/cfg-test-detection-perf.test.js +82 -0
  11. package/dist/__tests__/change-narrative.test.js +190 -0
  12. package/dist/__tests__/check-commit.test.js +258 -0
  13. package/dist/__tests__/checksum.test.js +34 -0
  14. package/dist/__tests__/commit-watcher.test.js +154 -0
  15. package/dist/__tests__/community-detection.test.js +179 -0
  16. package/dist/__tests__/community-tools.test.js +299 -0
  17. package/dist/__tests__/components.test.js +449 -0
  18. package/dist/__tests__/compression-log.test.js +174 -0
  19. package/dist/__tests__/compression-quality-monitor.test.js +40 -0
  20. package/dist/__tests__/config-healer.test.js +165 -0
  21. package/dist/__tests__/context-ledger.test.js +58 -0
  22. package/dist/__tests__/convention-detector.test.js +99 -0
  23. package/dist/__tests__/convention-learner.test.js +86 -0
  24. package/dist/__tests__/correction-detector.test.js +330 -0
  25. package/dist/__tests__/daemon-autostart-install.test.js +283 -0
  26. package/dist/__tests__/daemon-bridge.test.js +222 -0
  27. package/dist/__tests__/daemon-dashboard.test.js +202 -0
  28. package/dist/__tests__/daemon-registry.test.js +240 -0
  29. package/dist/__tests__/daemon-supervisor.test.js +318 -0
  30. package/dist/__tests__/daemon-version-check.test.js +275 -0
  31. package/dist/__tests__/decision-point-detector.test.js +98 -0
  32. package/dist/__tests__/deep-link.test.js +143 -0
  33. package/dist/__tests__/disallowed-tools.test.js +115 -0
  34. package/dist/__tests__/drift-tracker.test.js +582 -0
  35. package/dist/__tests__/durability-scorer.test.js +152 -0
  36. package/dist/__tests__/efficiency-tracker.test.js +65 -0
  37. package/dist/__tests__/enrich.test.js +144 -0
  38. package/dist/__tests__/entity-rewind.test.js +248 -0
  39. package/dist/__tests__/ephemeral.test.js +111 -0
  40. package/dist/__tests__/exploration-cost.test.js +93 -0
  41. package/dist/__tests__/fact-generator.test.js +197 -0
  42. package/dist/__tests__/file-l0-graph.test.js +244 -0
  43. package/dist/__tests__/file-logger.test.js +82 -0
  44. package/dist/__tests__/file-outline.test.js +141 -0
  45. package/dist/__tests__/file-read-protocol.test.js +188 -0
  46. package/dist/__tests__/format-encoder.test.js +233 -0
  47. package/dist/__tests__/git-attribution.test.js +259 -0
  48. package/dist/__tests__/graph-temporal-joiner.test.js +219 -0
  49. package/dist/__tests__/health-grade-enhanced.test.js +138 -0
  50. package/dist/__tests__/health-map-data.test.js +173 -0
  51. package/dist/__tests__/helpers/mcp-harness.js +45 -0
  52. package/dist/__tests__/helpers/mcp-harness.test.js +68 -0
  53. package/dist/__tests__/hook-dedup.test.js +112 -0
  54. package/dist/__tests__/hook-runner.test.js +253 -0
  55. package/dist/__tests__/indexer-cfg.test.js +185 -0
  56. package/dist/__tests__/indexer-cross-file.test.js +172 -0
  57. package/dist/__tests__/indexer-extraction.test.js +245 -0
  58. package/dist/__tests__/indexer-incremental.test.js +232 -0
  59. package/dist/__tests__/indexer-language-expansion.test.js +165 -0
  60. package/dist/__tests__/init-push.test.js +131 -0
  61. package/dist/__tests__/instruction-writer.test.js +179 -0
  62. package/dist/__tests__/intelligence-integration.test.js +217 -0
  63. package/dist/__tests__/intent-correlator.test.js +175 -0
  64. package/dist/__tests__/intent-detector.test.js +235 -0
  65. package/dist/__tests__/intent-encoder.test.js +167 -0
  66. package/dist/__tests__/java-build-tool-detection.test.js +174 -0
  67. package/dist/__tests__/layer3-sprint-q.test.js +160 -0
  68. package/dist/__tests__/layer3-sprint-r.test.js +91 -0
  69. package/dist/__tests__/layer3-sprint-s.test.js +183 -0
  70. package/dist/__tests__/layer3-sprint-t.test.js +201 -0
  71. package/dist/__tests__/layer3-sprint-u.test.js +174 -0
  72. package/dist/__tests__/layer4-sprint-ba2.test.js +354 -0
  73. package/dist/__tests__/layer4-sprint-ba4.test.js +84 -0
  74. package/dist/__tests__/layer4-sprint-vs.test.js +105 -0
  75. package/dist/__tests__/ledger-chains.test.js +162 -0
  76. package/dist/__tests__/lifecycle-machine.test.js +226 -0
  77. package/dist/__tests__/local-chat-provider.test.js +170 -0
  78. package/dist/__tests__/local-convention-detector.test.js +308 -0
  79. package/dist/__tests__/local-embeddings.test.js +422 -0
  80. package/dist/__tests__/local-graph.test.js +540 -0
  81. package/dist/__tests__/local-indexer.test.js +228 -0
  82. package/dist/__tests__/local-intelligence-l3.test.js +332 -0
  83. package/dist/__tests__/local-llm.test.js +253 -0
  84. package/dist/__tests__/local-mode-offline.test.js +187 -0
  85. package/dist/__tests__/local-mode-stats.test.js +273 -0
  86. package/dist/__tests__/local-mode-tui.test.js +343 -0
  87. package/dist/__tests__/local-parse.test.js +199 -0
  88. package/dist/__tests__/log-tailer.test.js +208 -0
  89. package/dist/__tests__/loop-breaker.test.js +276 -0
  90. package/dist/__tests__/loop-miner.test.js +226 -0
  91. package/dist/__tests__/mcp-config.test.js +126 -0
  92. package/dist/__tests__/mcp-content-json.test.js +10 -0
  93. package/dist/__tests__/mcp-envelope.test.js +124 -0
  94. package/dist/__tests__/metrics-store.test.js +223 -0
  95. package/dist/__tests__/native-watcher.test.js +191 -0
  96. package/dist/__tests__/navigation-hooks-agent-aware.test.js +145 -0
  97. package/dist/__tests__/negative-knowledge.test.js +116 -0
  98. package/dist/__tests__/network-boundary.test.js +190 -0
  99. package/dist/__tests__/network-firewall.test.js +112 -0
  100. package/dist/__tests__/nudge-invariants.test.js +160 -0
  101. package/dist/__tests__/nudge-v2.test.js +225 -0
  102. package/dist/__tests__/offline-rewind.test.js +251 -0
  103. package/dist/__tests__/open-threads.test.js +89 -0
  104. package/dist/__tests__/output-compressor.test.js +93 -0
  105. package/dist/__tests__/pending-violations.test.js +112 -0
  106. package/dist/__tests__/persistence-effectiveness.test.js +143 -0
  107. package/dist/__tests__/provider-factory.test.js +42 -0
  108. package/dist/__tests__/providers.test.js +24 -0
  109. package/dist/__tests__/proxy.test.js +314 -0
  110. package/dist/__tests__/query-router.test.js +1018 -0
  111. package/dist/__tests__/reasoning-quality-route.test.js +138 -0
  112. package/dist/__tests__/redactor.test.js +120 -0
  113. package/dist/__tests__/resource-monitor.test.js +57 -0
  114. package/dist/__tests__/response-envelope.test.js +100 -0
  115. package/dist/__tests__/risk-classifier.test.js +101 -0
  116. package/dist/__tests__/risk-signal-scope.test.js +75 -0
  117. package/dist/__tests__/rule-evaluator.test.js +280 -0
  118. package/dist/__tests__/scip-decoder.test.js +49 -0
  119. package/dist/__tests__/scip-downloader.test.js +201 -0
  120. package/dist/__tests__/scip-merger.test.js +103 -0
  121. package/dist/__tests__/search-index.test.js +422 -0
  122. package/dist/__tests__/semantic-enrichment.test.js +360 -0
  123. package/dist/__tests__/session-brief-builder.test.js +187 -0
  124. package/dist/__tests__/session-context.test.js +221 -0
  125. package/dist/__tests__/session-continuity.test.js +144 -0
  126. package/dist/__tests__/session-dedup.test.js +74 -0
  127. package/dist/__tests__/session-event-wiring.test.js +206 -0
  128. package/dist/__tests__/session-events.test.js +149 -0
  129. package/dist/__tests__/session-legend.test.js +20 -0
  130. package/dist/__tests__/session-persistence.test.js +131 -0
  131. package/dist/__tests__/session-resume-block.test.js +107 -0
  132. package/dist/__tests__/session-resume.test.js +97 -0
  133. package/dist/__tests__/session-summary-writer.test.js +134 -0
  134. package/dist/__tests__/shadow-ledger.test.js +203 -0
  135. package/dist/__tests__/shell-classifier.test.js +151 -0
  136. package/dist/__tests__/shell-compression-floor.test.js +189 -0
  137. package/dist/__tests__/shell-compression-v2.test.js +339 -0
  138. package/dist/__tests__/shell-compressor.test.js +35 -0
  139. package/dist/__tests__/shell-hooks.test.js +128 -0
  140. package/dist/__tests__/shell-strategies.test.js +644 -0
  141. package/dist/__tests__/shell-tee.test.js +133 -0
  142. package/dist/__tests__/signal-dedup.test.js +158 -0
  143. package/dist/__tests__/signal-reinforcer.test.js +77 -0
  144. package/dist/__tests__/signal-scorer.test.js +251 -0
  145. package/dist/__tests__/signal-show-store.test.js +108 -0
  146. package/dist/__tests__/smart-truncate.test.js +215 -0
  147. package/dist/__tests__/snapshot-v2.test.js +113 -0
  148. package/dist/__tests__/sprint-l1-local-mode.test.js +130 -0
  149. package/dist/__tests__/sprint-l10-boot.test.js +220 -0
  150. package/dist/__tests__/sprint-l9-offline-commands.test.js +189 -0
  151. package/dist/__tests__/sprint-q-persistent-context.test.js +198 -0
  152. package/dist/__tests__/sprint-s1-wiring.test.js +215 -0
  153. package/dist/__tests__/sprint-s2-wiring.test.js +256 -0
  154. package/dist/__tests__/sprint-s3-wiring.test.js +195 -0
  155. package/dist/__tests__/sprint-s4-wiring.test.js +213 -0
  156. package/dist/__tests__/sprint-s6-hooks.test.js +222 -0
  157. package/dist/__tests__/sprint-s7-persistent.test.js +263 -0
  158. package/dist/__tests__/sprint-s8-value.test.js +167 -0
  159. package/dist/__tests__/sprint-s9-behavioral.test.js +179 -0
  160. package/dist/__tests__/sprint3-intelligence.test.js +297 -0
  161. package/dist/__tests__/sprint5-mcp-server.test.js +136 -0
  162. package/dist/__tests__/startup-display.test.js +302 -0
  163. package/dist/__tests__/startup-log-file.test.js +97 -0
  164. package/dist/__tests__/stash-manager.test.js +229 -0
  165. package/dist/__tests__/state-detector.test.js +92 -0
  166. package/dist/__tests__/status-dashboard.test.js +142 -0
  167. package/dist/__tests__/temporal-facts.test.js +292 -0
  168. package/dist/__tests__/temporal-routes.test.js +142 -0
  169. package/dist/__tests__/test-detector.test.js +174 -0
  170. package/dist/__tests__/theme.test.js +72 -0
  171. package/dist/__tests__/timeline-agents.test.js +122 -0
  172. package/dist/__tests__/timeline-bootstrap.test.js +176 -0
  173. package/dist/__tests__/timeline-filters.test.js +193 -0
  174. package/dist/__tests__/timeline-markers.test.js +151 -0
  175. package/dist/__tests__/timeline-routes.test.js +156 -0
  176. package/dist/__tests__/timeline-store.test.js +171 -0
  177. package/dist/__tests__/token-counter.test.js +86 -0
  178. package/dist/__tests__/token-estimator.test.js +96 -0
  179. package/dist/__tests__/token-flow-api.test.js +239 -0
  180. package/dist/__tests__/token-flow-instrumentation.test.js +437 -0
  181. package/dist/__tests__/token-flow-persistence.test.js +356 -0
  182. package/dist/__tests__/token-flow-routes.test.js +199 -0
  183. package/dist/__tests__/token-flow.test.js +695 -0
  184. package/dist/__tests__/tool-clusters.test.js +177 -0
  185. package/dist/__tests__/transport-mux.test.js +283 -0
  186. package/dist/__tests__/turn-segmenter.test.js +166 -0
  187. package/dist/__tests__/uninstall.test.js +141 -0
  188. package/dist/__tests__/warm-start-policy.test.js +271 -0
  189. package/dist/__tests__/wire-cap-nudge.test.js +77 -0
  190. package/dist/__tests__/worker-pool.test.js +101 -0
  191. package/dist/behaviors/agent-llm-bridge.js +166 -0
  192. package/dist/behaviors/architecture-guard.js +256 -0
  193. package/dist/behaviors/auto-doc.js +247 -0
  194. package/dist/behaviors/cascade-guard.js +289 -0
  195. package/dist/behaviors/change-narrative.js +270 -0
  196. package/dist/behaviors/convention-drift.js +290 -0
  197. package/dist/behaviors/framework.js +235 -0
  198. package/dist/behaviors/guard-formatter.js +44 -0
  199. package/dist/behaviors/incomplete-work.js +270 -0
  200. package/dist/behaviors/loop-breaker.js +300 -0
  201. package/dist/behaviors/session-continuity.js +208 -0
  202. package/dist/cli.js +6446 -2227
  203. package/dist/commands/branches.js +97 -0
  204. package/dist/commands/check-commit.js +225 -0
  205. package/dist/commands/compress-output.js +64 -0
  206. package/dist/commands/config-verify.js +243 -0
  207. package/dist/commands/daemon.js +905 -0
  208. package/dist/commands/dashboard.js +52 -0
  209. package/dist/commands/debug.js +200 -0
  210. package/dist/commands/enrich.js +184 -0
  211. package/dist/commands/exec.js +233 -0
  212. package/dist/commands/gain.js +156 -0
  213. package/dist/commands/hook.js +88 -0
  214. package/dist/commands/index.js +88 -0
  215. package/dist/commands/init.js +74 -0
  216. package/dist/commands/install.js +505 -0
  217. package/dist/commands/learn.js +116 -0
  218. package/dist/commands/manifest.js +193 -0
  219. package/dist/commands/rewind.js +103 -0
  220. package/dist/commands/serve.js +19 -0
  221. package/dist/commands/setup-wizard.js +414 -0
  222. package/dist/commands/skills.js +64 -0
  223. package/dist/commands/stats.js +20 -0
  224. package/dist/commands/status.js +654 -0
  225. package/dist/commands/timeline.js +139 -0
  226. package/dist/commands/uninstall.js +230 -0
  227. package/dist/components/App.js +109 -0
  228. package/dist/components/Banner.js +12 -0
  229. package/dist/components/ConfirmPrompt.js +25 -0
  230. package/dist/components/DriftSummary.js +23 -0
  231. package/dist/components/GradeBadge.js +15 -0
  232. package/dist/components/HealthCard.js +18 -0
  233. package/dist/components/InkSpinner.js +22 -0
  234. package/dist/components/InputBox.js +17 -0
  235. package/dist/components/KeyValue.js +13 -0
  236. package/dist/components/MessageList.js +14 -0
  237. package/dist/components/ProgressBar.js +26 -0
  238. package/dist/components/Section.js +16 -0
  239. package/dist/components/SessionSummaryCard.js +73 -0
  240. package/dist/components/StartupDisplay.js +24 -0
  241. package/dist/components/StatusDashboard.js +57 -0
  242. package/dist/components/StatusLine.js +8 -0
  243. package/dist/components/StepLine.js +22 -0
  244. package/dist/components/Theme.js +20 -0
  245. package/dist/components/ToolProgress.js +8 -0
  246. package/dist/components/ViolationList.js +21 -0
  247. package/dist/components/render.js +13 -0
  248. package/dist/config/agent-registry.js +237 -0
  249. package/dist/config/claude-settings-hooks.js +304 -0
  250. package/dist/config/hook-installer.js +65 -0
  251. package/dist/config/instruction-writer.js +388 -0
  252. package/dist/config/mcp-config-writer.js +266 -0
  253. package/dist/config/settings.js +174 -0
  254. package/dist/config/tool-detector.js +42 -0
  255. package/dist/config/value-surfacing.js +119 -0
  256. package/dist/core/context-assembly.js +108 -0
  257. package/dist/core/conversation.js +33 -0
  258. package/dist/core/local-chat-provider.js +475 -0
  259. package/dist/core/provider-factory.js +55 -0
  260. package/dist/core/providers.js +90 -0
  261. package/dist/core/query-engine.js +174 -0
  262. package/dist/daemon/api.js +312 -0
  263. package/dist/daemon/autostart.js +119 -0
  264. package/dist/daemon/bootstrap.js +39 -0
  265. package/dist/daemon/client.js +164 -0
  266. package/dist/daemon/detect-ci.js +81 -0
  267. package/dist/daemon/platform-linux.js +146 -0
  268. package/dist/daemon/platform-macos.js +134 -0
  269. package/dist/daemon/platform-windows.js +116 -0
  270. package/dist/daemon/process-manager.js +299 -0
  271. package/dist/daemon/protocol.js +23 -0
  272. package/dist/daemon/registry.js +270 -0
  273. package/dist/daemon/settings-schema.js +72 -0
  274. package/dist/daemon/system-health.js +134 -0
  275. package/dist/daemon/version-checker.js +262 -0
  276. package/dist/daemon/warm-start.js +223 -0
  277. package/dist/entrypoints/cli.js +1043 -0
  278. package/dist/entrypoints/daemon.js +380 -0
  279. package/dist/entrypoints/repl.js +147 -0
  280. package/dist/hooks/adapters/claude-code.js +90 -0
  281. package/dist/hooks/adapters/cline.js +100 -0
  282. package/dist/hooks/adapters/cursor.js +98 -0
  283. package/dist/hooks/hook-dedup.js +79 -0
  284. package/dist/hooks/hook-runner.js +113 -0
  285. package/dist/hooks/navigation-hooks.js +175 -0
  286. package/dist/hooks/prompt-hooks.js +63 -0
  287. package/dist/hooks/shell-hooks.js +47 -0
  288. package/dist/ignore.js +111 -0
  289. package/dist/intelligence/approach-suggester.js +61 -0
  290. package/dist/intelligence/ast-extractor.js +2615 -0
  291. package/dist/intelligence/ast-worker.js +34 -0
  292. package/dist/intelligence/background-indexer.js +121 -0
  293. package/dist/intelligence/blast-radius.js +200 -0
  294. package/dist/intelligence/community-detection.js +691 -0
  295. package/dist/intelligence/community-detector.js +184 -0
  296. package/dist/intelligence/computation-scheduler.js +75 -0
  297. package/dist/intelligence/confidence-propagation.js +47 -0
  298. package/dist/intelligence/convention-detector.js +242 -0
  299. package/dist/intelligence/convention-learner.js +205 -0
  300. package/dist/intelligence/convention-matcher.js +205 -0
  301. package/dist/intelligence/cozo-schema.js +376 -0
  302. package/dist/intelligence/decision-point-detector.js +90 -0
  303. package/dist/intelligence/deep-dive-tools.js +586 -0
  304. package/dist/intelligence/durability-scorer.js +84 -0
  305. package/dist/intelligence/exploration-cost.js +204 -0
  306. package/dist/intelligence/exploration-pattern-tracker.js +61 -0
  307. package/dist/intelligence/fact-generator.js +322 -0
  308. package/dist/intelligence/facts-schema.js +90 -0
  309. package/dist/intelligence/file-intelligence.js +59 -0
  310. package/dist/intelligence/graph-holder.js +220 -0
  311. package/dist/intelligence/graph-temporal-joiner.js +238 -0
  312. package/dist/intelligence/health-grade.js +423 -0
  313. package/dist/intelligence/health-grader.js +200 -0
  314. package/dist/intelligence/health-map-data.js +259 -0
  315. package/dist/intelligence/import-symbols.js +136 -0
  316. package/dist/intelligence/incremental-indexer.js +658 -0
  317. package/dist/intelligence/indexer/centrality.js +62 -0
  318. package/dist/intelligence/indexer/cfg-context.js +95 -0
  319. package/dist/intelligence/indexer/confidence.js +34 -0
  320. package/dist/intelligence/indexer/cross-file-resolver.js +104 -0
  321. package/dist/intelligence/indexer/edge-repair.js +89 -0
  322. package/dist/intelligence/indexer/entity-key.js +17 -0
  323. package/dist/intelligence/indexer/export-map.js +132 -0
  324. package/dist/intelligence/indexer/git-cochange.js +128 -0
  325. package/dist/intelligence/indexer/graph-patch.js +147 -0
  326. package/dist/intelligence/indexer/incremental.js +78 -0
  327. package/dist/intelligence/indexer/ingest.js +160 -0
  328. package/dist/intelligence/indexer/language-detect.js +226 -0
  329. package/dist/intelligence/indexer/metadata.js +63 -0
  330. package/dist/intelligence/indexer/mutation-tracker.js +79 -0
  331. package/dist/intelligence/indexer/orchestrator.js +155 -0
  332. package/dist/intelligence/indexer/plugin-interface.js +31 -0
  333. package/dist/intelligence/indexer/plugins/csharp.js +440 -0
  334. package/dist/intelligence/indexer/plugins/go.js +335 -0
  335. package/dist/intelligence/indexer/plugins/java.js +370 -0
  336. package/dist/intelligence/indexer/plugins/python.js +358 -0
  337. package/dist/intelligence/indexer/plugins/regex-fallback.js +82 -0
  338. package/dist/intelligence/indexer/plugins/ruby.js +290 -0
  339. package/dist/intelligence/indexer/plugins/rust.js +484 -0
  340. package/dist/intelligence/indexer/plugins/tier2-generic.js +310 -0
  341. package/dist/intelligence/indexer/plugins/typescript.js +456 -0
  342. package/dist/intelligence/indexer/resource-monitor.js +93 -0
  343. package/dist/intelligence/indexer/scip/decoder.js +253 -0
  344. package/dist/intelligence/indexer/scip/detector.js +232 -0
  345. package/dist/intelligence/indexer/scip/downloader.js +427 -0
  346. package/dist/intelligence/indexer/scip/fallback.js +34 -0
  347. package/dist/intelligence/indexer/scip/merger.js +109 -0
  348. package/dist/intelligence/indexer/scip/orchestrator.js +433 -0
  349. package/dist/intelligence/indexer/scip/runner.js +98 -0
  350. package/dist/intelligence/indexer/snapshot.js +66 -0
  351. package/dist/intelligence/indexer/test-detector.js +196 -0
  352. package/dist/intelligence/indexer/watch-integration.js +61 -0
  353. package/dist/intelligence/indexer/worker.js +85 -0
  354. package/dist/intelligence/local-convention-detector.js +437 -0
  355. package/dist/intelligence/local-embeddings.js +190 -0
  356. package/dist/intelligence/local-graph.js +1946 -0
  357. package/dist/intelligence/local-indexer.js +1575 -0
  358. package/dist/intelligence/local-llm.js +163 -0
  359. package/dist/intelligence/local-rule-generator.js +154 -0
  360. package/dist/intelligence/local-snapshot.js +213 -0
  361. package/dist/intelligence/negative-knowledge.js +103 -0
  362. package/dist/intelligence/persistent-db.js +85 -0
  363. package/dist/intelligence/query-router.js +2556 -0
  364. package/dist/intelligence/risk-classifier.js +116 -0
  365. package/dist/intelligence/rule-evaluator.js +380 -0
  366. package/dist/intelligence/rule-generator.js +49 -0
  367. package/dist/intelligence/search-index.js +173 -0
  368. package/dist/intelligence/semantic/docstring-extractor.js +67 -0
  369. package/dist/intelligence/semantic/embedding-store.js +52 -0
  370. package/dist/intelligence/semantic/enrichment-orchestrator.js +48 -0
  371. package/dist/intelligence/semantic/git-message-miner.js +114 -0
  372. package/dist/intelligence/semantic/identifier-tokenizer.js +51 -0
  373. package/dist/intelligence/semantic/node2vec-embeddings.js +71 -0
  374. package/dist/intelligence/semantic/node2vec-walks.js +103 -0
  375. package/dist/intelligence/semantic/path-domain-inference.js +112 -0
  376. package/dist/intelligence/semantic/similarity-engine.js +60 -0
  377. package/dist/intelligence/semantic/tfidf-vectors.js +88 -0
  378. package/dist/intelligence/session-brief-builder.js +159 -0
  379. package/dist/intelligence/session-context.js +221 -0
  380. package/dist/intelligence/session-health-monitor.js +211 -0
  381. package/dist/intelligence/session-narrative.js +197 -0
  382. package/dist/intelligence/session-pattern-analyzer.js +218 -0
  383. package/dist/intelligence/signal-scorer.js +390 -0
  384. package/dist/intelligence/signal-show-store.js +182 -0
  385. package/dist/intelligence/smart-truncate.js +158 -0
  386. package/dist/intelligence/subgraph-cache.js +88 -0
  387. package/dist/intelligence/temporal-facts.js +494 -0
  388. package/dist/intelligence/token-estimator.js +100 -0
  389. package/dist/intelligence/tool-injector.js +87 -0
  390. package/dist/intelligence/tree-sitter-loader.js +71 -0
  391. package/dist/intelligence/worker-pool.js +116 -0
  392. package/dist/proxy/arg-validator.js +79 -0
  393. package/dist/proxy/auto-bootstrap.js +167 -0
  394. package/dist/proxy/bridge.js +147 -0
  395. package/dist/proxy/budget-enforcer.js +70 -0
  396. package/dist/proxy/compression-quality-monitor.js +160 -0
  397. package/dist/proxy/compression-stats.js +51 -0
  398. package/dist/proxy/context-rot-detector.js +137 -0
  399. package/dist/proxy/drift-detector.js +139 -0
  400. package/dist/proxy/efficiency-tracker.js +79 -0
  401. package/dist/proxy/fact-ranking.js +154 -0
  402. package/dist/proxy/format-encoder.js +266 -0
  403. package/dist/proxy/http-transport.js +90 -0
  404. package/dist/proxy/lifecycle-actor.js +55 -0
  405. package/dist/proxy/lifecycle-machine.js +187 -0
  406. package/dist/proxy/log-tailer.js +265 -0
  407. package/dist/proxy/model-pricing.js +98 -0
  408. package/dist/proxy/network-firewall.js +141 -0
  409. package/dist/proxy/nudge-state.js +93 -0
  410. package/dist/proxy/output-compressor.js +185 -0
  411. package/dist/proxy/pid-lock.js +291 -0
  412. package/dist/proxy/proxy-context.js +11 -0
  413. package/dist/proxy/proxy.js +2633 -0
  414. package/dist/proxy/response-enrichment.js +32 -0
  415. package/dist/proxy/response-envelope.js +313 -0
  416. package/dist/proxy/session-dedup.js +82 -0
  417. package/dist/proxy/session-legend.js +30 -0
  418. package/dist/proxy/session-persistence.js +210 -0
  419. package/dist/proxy/session-resume.js +94 -0
  420. package/dist/proxy/session-stats.js +513 -0
  421. package/dist/proxy/shell-classifier.js +1346 -0
  422. package/dist/proxy/shell-compression-log.js +93 -0
  423. package/dist/proxy/shell-compressor.js +390 -0
  424. package/dist/proxy/shell-graph-boost.js +202 -0
  425. package/dist/proxy/shell-monitor-map.js +18 -0
  426. package/dist/proxy/shell-stats.js +54 -0
  427. package/dist/proxy/shell-strategies/cloud.js +215 -0
  428. package/dist/proxy/shell-strategies/diff.js +159 -0
  429. package/dist/proxy/shell-strategies/error-diagnostic.js +796 -0
  430. package/dist/proxy/shell-strategies/filter-dsl.js +358 -0
  431. package/dist/proxy/shell-strategies/git-status.js +177 -0
  432. package/dist/proxy/shell-strategies/key-value.js +193 -0
  433. package/dist/proxy/shell-strategies/log-text.js +154 -0
  434. package/dist/proxy/shell-strategies/omni.js +188 -0
  435. package/dist/proxy/shell-strategies/progress.js +55 -0
  436. package/dist/proxy/shell-strategies/redact.js +76 -0
  437. package/dist/proxy/shell-strategies/structured.js +241 -0
  438. package/dist/proxy/shell-strategies/tabular.js +243 -0
  439. package/dist/proxy/shell-strategies/test-results-types.js +13 -0
  440. package/dist/proxy/shell-strategies/test-results.js +784 -0
  441. package/dist/proxy/shell-strategies/tree-paths.js +144 -0
  442. package/dist/proxy/shell-strategies/yaml.js +182 -0
  443. package/dist/proxy/shell-tee.js +111 -0
  444. package/dist/proxy/signal-dedup.js +171 -0
  445. package/dist/proxy/startup-renderer.js +158 -0
  446. package/dist/proxy/task-token-display.js +38 -0
  447. package/dist/proxy/token-counter.js +61 -0
  448. package/dist/proxy/tool-clusters.js +273 -0
  449. package/dist/proxy/tool-definitions.js +525 -0
  450. package/dist/proxy/transport-mux.js +229 -0
  451. package/dist/proxy/wire-cap.js +268 -0
  452. package/dist/schemas/api/skills.js +19 -0
  453. package/dist/schemas/common/errors.js +7 -0
  454. package/dist/schemas/common/headers.js +5 -0
  455. package/dist/schemas/entities/edge.js +25 -0
  456. package/dist/schemas/entities/entity.js +22 -0
  457. package/dist/schemas/entities/rule.js +18 -0
  458. package/dist/schemas/index.js +14 -0
  459. package/dist/server/event-bus.js +59 -0
  460. package/dist/server/http.js +156 -0
  461. package/dist/server/middleware.js +70 -0
  462. package/dist/server/routes/drift.js +97 -0
  463. package/dist/server/routes/intelligence.js +1217 -0
  464. package/dist/server/routes/reasoning-quality.js +444 -0
  465. package/dist/server/routes/session.js +86 -0
  466. package/dist/server/routes/stream.js +120 -0
  467. package/dist/server/routes/system.js +73 -0
  468. package/dist/server/routes/temporal.js +170 -0
  469. package/dist/server/routes/timeline.js +232 -0
  470. package/dist/server/routes/token-flow.js +403 -0
  471. package/dist/skills/effectiveness-tracker.js +93 -0
  472. package/dist/skills/local-pack.js +380 -0
  473. package/dist/skills/resolver.js +495 -0
  474. package/dist/state-detector.js +83 -0
  475. package/dist/timeline/intent-detector.js +263 -0
  476. package/dist/timeline/loop-miner.js +140 -0
  477. package/dist/timeline/open-threads.js +49 -0
  478. package/dist/timeline/signal-reinforcer.js +62 -0
  479. package/dist/timeline/timeline-bootstrap.js +151 -0
  480. package/dist/timeline/timeline-store.js +618 -0
  481. package/dist/tools/coding/bash.js +49 -0
  482. package/dist/tools/coding/file-edit.js +72 -0
  483. package/dist/tools/coding/file-outline.js +227 -0
  484. package/dist/tools/coding/file-read-protocol.js +425 -0
  485. package/dist/tools/coding/file-read.js +35 -0
  486. package/dist/tools/coding/file-write.js +43 -0
  487. package/dist/tools/coding/glob-tool.js +109 -0
  488. package/dist/tools/coding/grep.js +162 -0
  489. package/dist/tools/coding/index.js +27 -0
  490. package/dist/tools/intelligence/index.js +269 -0
  491. package/dist/tools/intelligence/record-fact.js +48 -0
  492. package/dist/tools/intelligence/timeline-markers.js +130 -0
  493. package/dist/tools/registry.js +47 -0
  494. package/dist/tools/types.js +8 -0
  495. package/dist/tracking/auto-snapshot-triggers.js +246 -0
  496. package/dist/tracking/branch-context.js +115 -0
  497. package/dist/tracking/branch-snapshot.js +217 -0
  498. package/dist/tracking/causal-bridge.js +317 -0
  499. package/dist/tracking/circuit-breaker.js +147 -0
  500. package/dist/tracking/commit-watcher.js +114 -0
  501. package/dist/tracking/context-ledger.js +119 -0
  502. package/dist/tracking/correction-detector.js +324 -0
  503. package/dist/tracking/drift-tracker.js +874 -0
  504. package/dist/tracking/durability-tracker.js +94 -0
  505. package/dist/tracking/entity-rewind.js +200 -0
  506. package/dist/tracking/file-hash-state.js +114 -0
  507. package/dist/tracking/git-attribution.js +132 -0
  508. package/dist/tracking/git-trailers.js +171 -0
  509. package/dist/tracking/intelligence-counter.js +46 -0
  510. package/dist/tracking/intent-correlator.js +202 -0
  511. package/dist/tracking/intent-encoder.js +52 -0
  512. package/dist/tracking/intent-token-tracker.js +159 -0
  513. package/dist/tracking/ledger-archiver.js +94 -0
  514. package/dist/tracking/ledger-chains.js +245 -0
  515. package/dist/tracking/metrics-store.js +361 -0
  516. package/dist/tracking/native-watcher.js +131 -0
  517. package/dist/tracking/offline-rewind.js +295 -0
  518. package/dist/tracking/pending-violations.js +74 -0
  519. package/dist/tracking/persistence-effectiveness.js +167 -0
  520. package/dist/tracking/prompt-durability.js +202 -0
  521. package/dist/tracking/quality-signals.js +213 -0
  522. package/dist/tracking/redactor.js +73 -0
  523. package/dist/tracking/rewind-engine.js +161 -0
  524. package/dist/tracking/session-history.js +128 -0
  525. package/dist/tracking/session-receipt.js +88 -0
  526. package/dist/tracking/session-summary-writer.js +157 -0
  527. package/dist/tracking/shadow-ledger.js +321 -0
  528. package/dist/tracking/stash-manager.js +258 -0
  529. package/dist/tracking/timeline-fork.js +213 -0
  530. package/dist/tracking/timeline.js +69 -0
  531. package/dist/tracking/token-flow.js +276 -0
  532. package/dist/tracking/turn-segmenter.js +122 -0
  533. package/dist/tracking/weekly-accumulator.js +179 -0
  534. package/dist/tracking/working-snapshots.js +188 -0
  535. package/dist/tracking/workspace-manifest.js +176 -0
  536. package/dist/transport/http.js +102 -0
  537. package/dist/ui/assets/index-7gl3mIuY.css +1 -0
  538. package/dist/ui/assets/index-BsMTQdhX.js +10 -0
  539. package/dist/ui/index.html +2 -2
  540. package/dist/utils/counterfactual.js +65 -0
  541. package/dist/utils/deep-link.js +34 -0
  542. package/dist/utils/detect.js +193 -0
  543. package/dist/utils/exec.js +73 -0
  544. package/dist/utils/file-logger.js +87 -0
  545. package/dist/utils/format-error.js +29 -0
  546. package/dist/utils/git.js +181 -0
  547. package/dist/utils/log.js +57 -0
  548. package/dist/utils/logger.js +35 -0
  549. package/dist/utils/mcp-content-json.js +8 -0
  550. package/dist/utils/session-logger.js +154 -0
  551. package/dist/utils/startup-log.js +512 -0
  552. package/dist/utils/ui.js +56 -0
  553. package/package.json +5 -3
  554. package/scripts/postinstall.mjs +299 -0
  555. package/dist/ui/assets/index-BISLlJyc.js +0 -10
  556. package/dist/ui/assets/index-BUChTv4H.css +0 -1
@@ -0,0 +1,139 @@
1
+ import * as fs from "node:fs";
2
+ import * as path from "node:path";
3
+ // ── Local Timeline from Shadow Ledger ─────────────────────────
4
+ /**
5
+ * Format a relative time string from an ISO timestamp.
6
+ */
7
+ export function formatRelativeTime(isoTs) {
8
+ const diff = Date.now() - new Date(isoTs).getTime();
9
+ const seconds = Math.floor(diff / 1000);
10
+ if (seconds < 60)
11
+ return `${seconds}s ago`;
12
+ const minutes = Math.floor(seconds / 60);
13
+ if (minutes < 60)
14
+ return `${minutes} min ago`;
15
+ const hours = Math.floor(minutes / 60);
16
+ if (hours < 24)
17
+ return `${hours}h ago`;
18
+ const days = Math.floor(hours / 24);
19
+ return `${days}d ago`;
20
+ }
21
+ /**
22
+ * Read and parse shadow ledger JSONL file.
23
+ */
24
+ export function readShadowLedger(ledgerPath) {
25
+ if (!fs.existsSync(ledgerPath))
26
+ return [];
27
+ try {
28
+ const raw = fs.readFileSync(ledgerPath, "utf-8");
29
+ const entries = [];
30
+ for (const line of raw.split("\n")) {
31
+ const trimmed = line.trim();
32
+ if (!trimmed)
33
+ continue;
34
+ try {
35
+ entries.push(JSON.parse(trimmed));
36
+ }
37
+ catch {
38
+ // Skip malformed lines
39
+ }
40
+ }
41
+ return entries;
42
+ }
43
+ catch {
44
+ return [];
45
+ }
46
+ }
47
+ /**
48
+ * Load working state snapshot SHAs for star markers.
49
+ */
50
+ function loadWorkingSnapshots(unerrDir) {
51
+ const shas = new Set();
52
+ const stateDir = path.join(unerrDir, "state");
53
+ if (!fs.existsSync(stateDir))
54
+ return shas;
55
+ try {
56
+ const entries = fs.readdirSync(stateDir, { withFileTypes: true });
57
+ for (const entry of entries) {
58
+ if (entry.name.startsWith("working-") && entry.name.endsWith(".json")) {
59
+ try {
60
+ const raw = fs.readFileSync(path.join(stateDir, entry.name), "utf-8");
61
+ const snapshot = JSON.parse(raw);
62
+ if (snapshot.commitSha)
63
+ shas.add(snapshot.commitSha);
64
+ }
65
+ catch {
66
+ // Skip corrupt files
67
+ }
68
+ }
69
+ }
70
+ }
71
+ catch {
72
+ // Non-blocking
73
+ }
74
+ return shas;
75
+ }
76
+ export function runTimelineLocal(opts) {
77
+ const cwd = process.cwd();
78
+ const unerrDir = path.join(cwd, ".unerr");
79
+ const ledgerPath = path.join(unerrDir, "ledger", "shadow.jsonl");
80
+ let entries = readShadowLedger(ledgerPath);
81
+ if (entries.length === 0) {
82
+ process.stderr.write("[unerr] No ledger entries found.\n");
83
+ process.stderr.write("[unerr] The shadow ledger is populated as you use unerr tools.\n");
84
+ return;
85
+ }
86
+ // Apply filters
87
+ if (opts.branch) {
88
+ entries = entries.filter((e) => e.branch === opts.branch);
89
+ }
90
+ if (opts.tool) {
91
+ entries = entries.filter((e) => e.tool === opts.tool);
92
+ }
93
+ // Take last N entries (most recent)
94
+ const total = entries.length;
95
+ entries = entries.slice(-opts.limit);
96
+ // Load working state markers
97
+ const workingShas = loadWorkingSnapshots(unerrDir);
98
+ process.stderr.write("\n");
99
+ for (const entry of entries) {
100
+ const timestamp = formatRelativeTime(entry.ts).padEnd(12);
101
+ const toolName = entry.tool.padEnd(22);
102
+ const branch = `[${entry.branch}]`;
103
+ const isWorking = workingShas.has(entry.head_sha);
104
+ const marker = isWorking ? " *" : "";
105
+ process.stderr.write(` ${timestamp}${toolName} ${branch}${marker}\n`);
106
+ // Show args summary if available
107
+ if (entry.args_summary && Object.keys(entry.args_summary).length > 0) {
108
+ const summary = Object.entries(entry.args_summary)
109
+ .map(([k, v]) => `${k}: ${typeof v === "string" ? v : JSON.stringify(v)}`)
110
+ .join(", ")
111
+ .slice(0, 80);
112
+ process.stderr.write(` ${summary}\n`);
113
+ }
114
+ }
115
+ const workingCount = entries.filter((e) => workingShas.has(e.head_sha)).length;
116
+ process.stderr.write(`\n ${entries.length} entries shown${total > entries.length ? ` (of ${total} total)` : ""} · ${workingCount} working states marked\n\n`);
117
+ }
118
+ export function registerTimelineCommand(program) {
119
+ program
120
+ .command("timeline")
121
+ .description("Show the prompt ledger timeline")
122
+ .option("--branch <branch>", "Filter by branch")
123
+ .option("--tool <tool>", "Filter by tool name")
124
+ .option("--limit <n>", "Number of entries to show", "50")
125
+ .action(async (opts) => {
126
+ try {
127
+ runTimelineLocal({
128
+ branch: opts.branch,
129
+ tool: opts.tool,
130
+ limit: Number.parseInt(opts.limit, 10) || 50,
131
+ });
132
+ }
133
+ catch (error) {
134
+ const message = error instanceof Error ? error.message : String(error);
135
+ console.error(`Error: ${message}`);
136
+ process.exit(1);
137
+ }
138
+ });
139
+ }
@@ -0,0 +1,230 @@
1
+ /**
2
+ * `unerr uninstall [agent]` — removes generated configs, skills, hooks, and instructions.
3
+ *
4
+ * Two modes:
5
+ * - `unerr uninstall claude` — per-agent: reverts exactly what `unerr install claude` did
6
+ * - `unerr uninstall` (no args) — removes all agents' configs
7
+ *
8
+ * Does NOT remove:
9
+ * - .unerr/ data directory (user's indexed data)
10
+ * - .gitignore entries (shared across agents)
11
+ */
12
+ import { existsSync, readFileSync, unlinkSync, writeFileSync } from "node:fs";
13
+ import { join } from "node:path";
14
+ import { AGENT_REGISTRY, getAgent, normalizeAgentName, } from "../config/agent-registry.js";
15
+ import { removeDisallowedTools, removePreToolUseBashHook, } from "../config/claude-settings-hooks.js";
16
+ import { removeClaudeHook } from "../config/hook-installer.js";
17
+ import { removeInstructionSection } from "../config/instruction-writer.js";
18
+ import { removeMcpConfig } from "../config/mcp-config-writer.js";
19
+ import { removeInstalledSkills } from "../skills/resolver.js";
20
+ export function registerUninstallCommand(program) {
21
+ program
22
+ .command("uninstall [agent]")
23
+ .description("Remove unerr configs and hooks from this project")
24
+ .option("--autostart", "Remove the platform auto-start service (launchd/systemd/schtasks)")
25
+ .action(async (agent, opts) => {
26
+ const cwd = process.cwd();
27
+ if (opts.autostart) {
28
+ try {
29
+ const { uninstallForCurrentPlatform, removeSentinel } = await import("../daemon/autostart.js");
30
+ const result = await uninstallForCurrentPlatform();
31
+ removeSentinel();
32
+ process.stderr.write(`\x1b[38;2;52;211;153m✓\x1b[0m Platform auto-start removed${result.error ? ` (note: ${result.error})` : ""}\n`);
33
+ }
34
+ catch (err) {
35
+ process.stderr.write(`\x1b[38;2;248;113;113m✗\x1b[0m Failed to remove auto-start: ${err.message}\n`);
36
+ }
37
+ return;
38
+ }
39
+ if (agent) {
40
+ const normalized = normalizeAgentName(agent);
41
+ const agentDef = getAgent(normalized);
42
+ if (!agentDef) {
43
+ process.stderr.write(`\x1b[31m✗\x1b[0m Unknown agent: "${agent}"\n`);
44
+ return;
45
+ }
46
+ const result = runUninstall(cwd, normalized);
47
+ displayUninstallResult(agentDef.name, result);
48
+ }
49
+ else {
50
+ runUninstallAll(cwd);
51
+ }
52
+ // Hint about autostart removal
53
+ try {
54
+ const { isAutostartInstalled } = await import("../daemon/autostart.js");
55
+ if (isAutostartInstalled()) {
56
+ process.stderr.write("\n \x1b[38;2;251;191;36m⚠\x1b[0m Platform auto-start is still active. Remove with: unerr uninstall --autostart\n\n");
57
+ }
58
+ }
59
+ catch {
60
+ // Non-blocking
61
+ }
62
+ });
63
+ }
64
+ /**
65
+ * Uninstall unerr for a single agent — symmetric to runInstall().
66
+ */
67
+ function runUninstall(cwd, ide) {
68
+ // 1. Remove MCP config
69
+ const mcpRemoved = removeMcpConfig(cwd, ide);
70
+ // 2. Remove skills
71
+ let skillsRemoved = 0;
72
+ try {
73
+ skillsRemoved = removeInstalledSkills(ide, cwd);
74
+ }
75
+ catch {
76
+ // Non-blocking
77
+ }
78
+ // 3. Remove hooks (agent-specific)
79
+ let hookRemoved = false;
80
+ let settingsHookRemoved = false;
81
+ if (ide === "claude-code") {
82
+ hookRemoved = removeClaudeHook(cwd);
83
+ settingsHookRemoved = removePreToolUseBashHook(cwd);
84
+ }
85
+ else if (ide === "cursor") {
86
+ hookRemoved = removeCursorHooks(cwd);
87
+ }
88
+ else if (ide === "cline") {
89
+ hookRemoved = removeClineHooks(cwd);
90
+ }
91
+ // 4. Gitignore — NOT reverted (shared across agents, .unerr/ data preserved)
92
+ // 5. Remove instructions
93
+ let instructionsRemoved = false;
94
+ try {
95
+ instructionsRemoved = removeInstructionSection(cwd, ide);
96
+ }
97
+ catch {
98
+ // Non-blocking
99
+ }
100
+ // 6. S8: Remove disallowed tool entries (Claude Code only)
101
+ let disallowedToolsRemoved = false;
102
+ if (ide === "claude-code") {
103
+ try {
104
+ disallowedToolsRemoved = removeDisallowedTools(cwd);
105
+ }
106
+ catch {
107
+ // Non-blocking
108
+ }
109
+ }
110
+ return {
111
+ mcpRemoved,
112
+ skillsRemoved,
113
+ hookRemoved,
114
+ settingsHookRemoved,
115
+ instructionsRemoved,
116
+ disallowedToolsRemoved,
117
+ };
118
+ }
119
+ /**
120
+ * Display per-agent uninstall results (matches install's ANSI style).
121
+ */
122
+ function displayUninstallResult(agentName, result) {
123
+ process.stderr.write("\n");
124
+ process.stderr.write(` \x1b[38;2;139;92;246m◆\x1b[0m \x1b[1munerr ✗ ${agentName}\x1b[0m\n`);
125
+ process.stderr.write("\n");
126
+ const nothingRemoved = !result.mcpRemoved &&
127
+ result.skillsRemoved === 0 &&
128
+ !result.hookRemoved &&
129
+ !result.settingsHookRemoved &&
130
+ !result.instructionsRemoved &&
131
+ !result.disallowedToolsRemoved;
132
+ if (nothingRemoved) {
133
+ process.stderr.write(" \x1b[38;2;161;161;170m· Nothing to uninstall\x1b[0m\n");
134
+ }
135
+ else {
136
+ if (result.mcpRemoved) {
137
+ process.stderr.write(" \x1b[38;2;52;211;153m✓\x1b[0m MCP config removed\n");
138
+ }
139
+ if (result.skillsRemoved > 0) {
140
+ process.stderr.write(` \x1b[38;2;52;211;153m✓\x1b[0m ${result.skillsRemoved} skills removed\n`);
141
+ }
142
+ if (result.hookRemoved) {
143
+ process.stderr.write(" \x1b[38;2;52;211;153m✓\x1b[0m PostToolUse hook removed\n");
144
+ }
145
+ if (result.settingsHookRemoved) {
146
+ process.stderr.write(" \x1b[38;2;52;211;153m✓\x1b[0m PreToolUse settings hook removed\n");
147
+ }
148
+ if (result.instructionsRemoved) {
149
+ process.stderr.write(" \x1b[38;2;52;211;153m✓\x1b[0m Tool preferences removed\n");
150
+ }
151
+ if (result.disallowedToolsRemoved) {
152
+ process.stderr.write(" \x1b[38;2;52;211;153m✓\x1b[0m Disallowed tools restored\n");
153
+ }
154
+ }
155
+ process.stderr.write("\n");
156
+ process.stderr.write(" \x1b[38;2;161;161;170mData in .unerr/ preserved.\x1b[0m\n");
157
+ process.stderr.write("\n");
158
+ }
159
+ /**
160
+ * Uninstall all agents — enhanced version of original behavior.
161
+ */
162
+ function runUninstallAll(cwd) {
163
+ const results = [];
164
+ for (const agent of AGENT_REGISTRY) {
165
+ const r = runUninstall(cwd, agent.id);
166
+ if (r.mcpRemoved)
167
+ results.push(`Removed MCP config for ${agent.id}`);
168
+ if (r.skillsRemoved > 0)
169
+ results.push(`Removed ${r.skillsRemoved} skills for ${agent.id}`);
170
+ if (r.hookRemoved)
171
+ results.push("Removed PostToolUse hook");
172
+ if (r.settingsHookRemoved)
173
+ results.push("Removed PreToolUse settings hook");
174
+ if (r.instructionsRemoved)
175
+ results.push(`Removed instructions for ${agent.id}`);
176
+ if (r.disallowedToolsRemoved)
177
+ results.push("Restored disallowed built-in tools");
178
+ }
179
+ if (results.length === 0) {
180
+ process.stderr.write("[unerr] Nothing to uninstall — no configs found.\n");
181
+ }
182
+ else {
183
+ for (const r of results) {
184
+ process.stderr.write(`[unerr] ${r}\n`);
185
+ }
186
+ process.stderr.write("[unerr] Uninstall complete. Data in .unerr/ preserved.\n");
187
+ }
188
+ }
189
+ /**
190
+ * Remove unerr hook entries from Cursor's `.cursor/hooks.json`.
191
+ */
192
+ function removeCursorHooks(cwd) {
193
+ const hooksPath = join(cwd, ".cursor", "hooks.json");
194
+ if (!existsSync(hooksPath))
195
+ return false;
196
+ try {
197
+ const config = JSON.parse(readFileSync(hooksPath, "utf-8"));
198
+ if (!Array.isArray(config.hooks))
199
+ return false;
200
+ const before = config.hooks.length;
201
+ config.hooks = config.hooks.filter((h) => h?.name !== "unerr-graph-tools");
202
+ if (config.hooks.length === before)
203
+ return false;
204
+ if (config.hooks.length === 0) {
205
+ unlinkSync(hooksPath);
206
+ }
207
+ else {
208
+ writeFileSync(hooksPath, `${JSON.stringify(config, null, 2)}\n`);
209
+ }
210
+ return true;
211
+ }
212
+ catch {
213
+ return false;
214
+ }
215
+ }
216
+ /**
217
+ * Remove unerr hook scripts from Cline's `.clinerules/hooks/`.
218
+ */
219
+ function removeClineHooks(cwd) {
220
+ const hookPath = join(cwd, ".clinerules", "hooks", "unerr-pre-tool.sh");
221
+ if (!existsSync(hookPath))
222
+ return false;
223
+ try {
224
+ unlinkSync(hookPath);
225
+ return true;
226
+ }
227
+ catch {
228
+ return false;
229
+ }
230
+ }
@@ -0,0 +1,109 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
2
+ /**
3
+ * App — root Ink component for the interactive REPL.
4
+ *
5
+ * Manages conversation state, dispatches user input to the QueryEngine,
6
+ * and renders the message history with streaming responses.
7
+ */
8
+ import { Box, Text, useApp, useInput } from "ink";
9
+ import { useState, useCallback } from "react";
10
+ import { executeQuery, } from "../core/query-engine.js";
11
+ import { InputBox } from "./InputBox.js";
12
+ import { MessageList } from "./MessageList.js";
13
+ import { StatusLine } from "./StatusLine.js";
14
+ import { ToolProgress } from "./ToolProgress.js";
15
+ export function App({ apiKey, chatProvider, model, systemPrompt, tools, cwd, welcomeMessage, }) {
16
+ const { exit } = useApp();
17
+ const [messages, setMessages] = useState([]);
18
+ const [streamingText, setStreamingText] = useState("");
19
+ const [isLoading, setIsLoading] = useState(false);
20
+ const [activeTools, setActiveTools] = useState([]);
21
+ const [lastUsage, setLastUsage] = useState(null);
22
+ const [error, setError] = useState(null);
23
+ // Handle Ctrl+C to exit
24
+ useInput((input, key) => {
25
+ if (key.ctrl && input === "c") {
26
+ exit();
27
+ }
28
+ });
29
+ const handleSubmit = useCallback(async (input) => {
30
+ if (!input.trim())
31
+ return;
32
+ // Handle special commands
33
+ if (input.trim() === "/exit" || input.trim() === "/quit") {
34
+ exit();
35
+ return;
36
+ }
37
+ if (input.trim() === "/clear") {
38
+ setMessages([]);
39
+ setLastUsage(null);
40
+ setError(null);
41
+ return;
42
+ }
43
+ if (input.trim() === "/cost") {
44
+ if (lastUsage) {
45
+ setError(`Session: ${lastUsage.inputTokens} in / ${lastUsage.outputTokens} out — $${lastUsage.estimatedCost.toFixed(4)}`);
46
+ }
47
+ else {
48
+ setError("No usage data yet");
49
+ }
50
+ return;
51
+ }
52
+ const userMessage = { role: "user", content: input };
53
+ const updatedMessages = [...messages, userMessage];
54
+ setMessages(updatedMessages);
55
+ setIsLoading(true);
56
+ setStreamingText("");
57
+ setActiveTools([]);
58
+ setError(null);
59
+ try {
60
+ const options = {
61
+ model,
62
+ apiKey,
63
+ chatProvider,
64
+ tools,
65
+ systemPrompt,
66
+ toolContext: { cwd },
67
+ };
68
+ const result = await executeQuery(updatedMessages, options, {
69
+ onToken: (token) => {
70
+ setStreamingText((prev) => prev + token);
71
+ },
72
+ onToolCall: (name) => {
73
+ setActiveTools((prev) => [...prev, { name, status: "running" }]);
74
+ },
75
+ onToolResult: (name) => {
76
+ setActiveTools((prev) => prev.map((t) => t.name === name && t.status === "running"
77
+ ? { ...t, status: "done" }
78
+ : t));
79
+ },
80
+ });
81
+ const assistantMessage = {
82
+ role: "assistant",
83
+ content: result.response,
84
+ toolCalls: result.toolCalls,
85
+ };
86
+ setMessages((prev) => [...prev, assistantMessage]);
87
+ setLastUsage(result.usage);
88
+ }
89
+ catch (err) {
90
+ setError(err instanceof Error ? err.message : String(err));
91
+ }
92
+ finally {
93
+ setIsLoading(false);
94
+ setStreamingText("");
95
+ setActiveTools([]);
96
+ }
97
+ }, [
98
+ messages,
99
+ model,
100
+ apiKey,
101
+ chatProvider,
102
+ tools,
103
+ systemPrompt,
104
+ cwd,
105
+ exit,
106
+ lastUsage,
107
+ ]);
108
+ return (_jsxs(Box, { flexDirection: "column", width: "100%", children: [_jsxs(Box, { marginBottom: 1, children: [_jsx(Text, { bold: true, color: "cyan", children: "unerr" }), _jsx(Text, { color: "gray", children: " \u2014 AI assistant with code intelligence" }), _jsxs(Text, { color: "gray", children: [" (", model, ")"] })] }), welcomeMessage && messages.length === 0 && (_jsx(Box, { marginBottom: 1, children: _jsx(Text, { color: "gray", children: welcomeMessage }) })), _jsx(MessageList, { messages: messages }), streamingText && (_jsxs(Box, { marginBottom: 1, children: [_jsx(Text, { color: "green", children: streamingText }), _jsx(Text, { color: "gray", children: "\u258A" })] })), activeTools.length > 0 && _jsx(ToolProgress, { tools: activeTools }), error && (_jsx(Box, { marginBottom: 1, children: _jsxs(Text, { color: "red", children: ["Error: ", error] }) })), lastUsage && _jsx(StatusLine, { usage: lastUsage }), _jsx(InputBox, { onSubmit: handleSubmit, isDisabled: isLoading })] }));
109
+ }
@@ -0,0 +1,12 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
2
+ /**
3
+ * Branded banner: unerr ▸ Code intelligence for AI agents
4
+ *
5
+ * Used in startup and setup displays. Renders to stderr via Ink.
6
+ */
7
+ import { Box, Text } from "ink";
8
+ import { useTheme } from "./Theme.js";
9
+ export function Banner() {
10
+ const t = useTheme();
11
+ return (_jsxs(Box, { marginLeft: 2, children: [_jsx(Text, { bold: true, color: t.brand, children: "unerr" }), _jsx(Text, { color: t.dim, children: " \u25B8 " }), _jsx(Text, { color: t.dim, children: "Code intelligence for your AI agents" })] }));
12
+ }
@@ -0,0 +1,25 @@
1
+ import { jsxs as _jsxs, jsx as _jsx } from "react/jsx-runtime";
2
+ /**
3
+ * Inline confirm prompt: [Y/n] with context.
4
+ *
5
+ * Uses Ink's useInput() for keyboard handling.
6
+ * Used for entity rewind, dead code removal confirmations.
7
+ */
8
+ import { Box, Text, useInput } from "ink";
9
+ import { useTheme } from "./Theme.js";
10
+ export function ConfirmPrompt({ message, onConfirm, defaultYes = true, }) {
11
+ const t = useTheme();
12
+ const hint = defaultYes ? "[Y/n]" : "[y/N]";
13
+ useInput((input, key) => {
14
+ if (input === "y" || input === "Y") {
15
+ onConfirm(true);
16
+ }
17
+ else if (input === "n" || input === "N") {
18
+ onConfirm(false);
19
+ }
20
+ else if (key.return) {
21
+ onConfirm(defaultYes);
22
+ }
23
+ });
24
+ return (_jsxs(Box, { marginLeft: 2, children: [_jsxs(Text, { children: [message, " "] }), _jsx(Text, { color: t.dim, children: hint })] }));
25
+ }
@@ -0,0 +1,23 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
2
+ /**
3
+ * Drift summary — modified/added/deleted counts.
4
+ *
5
+ * Used in status and startup displays.
6
+ */
7
+ import { Box, Text } from "ink";
8
+ import { useTheme } from "./Theme.js";
9
+ export function DriftSummary({ drift }) {
10
+ const t = useTheme();
11
+ const total = drift.modified + drift.added + drift.deleted;
12
+ if (total === 0) {
13
+ return (_jsx(Box, { marginLeft: 4, children: _jsx(Text, { color: t.dim, children: "No drift detected" }) }));
14
+ }
15
+ const parts = [];
16
+ if (drift.modified > 0)
17
+ parts.push(`${drift.modified} modified`);
18
+ if (drift.added > 0)
19
+ parts.push(`${drift.added} added`);
20
+ if (drift.deleted > 0)
21
+ parts.push(`${drift.deleted} deleted`);
22
+ return (_jsx(Box, { marginLeft: 4, children: _jsxs(Text, { color: t.warning, children: [total, " drifted entit", total !== 1 ? "ies" : "y", ": ", parts.join(", ")] }) }));
23
+ }
@@ -0,0 +1,15 @@
1
+ import { jsx as _jsx } from "react/jsx-runtime";
2
+ /**
3
+ * Colored grade badge: A, B+, C, D, F
4
+ *
5
+ * Color mapped by theme.grade() — A=green, C=yellow, F=red.
6
+ * Used in health card, status, first-connect displays.
7
+ */
8
+ import { Text } from "ink";
9
+ import { useTheme } from "./Theme.js";
10
+ export function GradeBadge({ grade, score, bold = true, }) {
11
+ const t = useTheme();
12
+ const color = t.grade(grade);
13
+ const label = score !== undefined ? `${grade} (${score}/100)` : grade;
14
+ return (_jsx(Text, { bold: bold, color: color, children: label }));
15
+ }
@@ -0,0 +1,18 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
2
+ /**
3
+ * Health grade card — bordered box with grade, progress bar, and top issues.
4
+ *
5
+ * Used in startup (Act 2: Revelation), status, and first-connect displays.
6
+ */
7
+ import { Box, Text } from "ink";
8
+ import { GradeBadge } from "./GradeBadge.js";
9
+ import { ProgressBar } from "./ProgressBar.js";
10
+ import { useTheme } from "./Theme.js";
11
+ export function HealthCard({ health, compact = false, }) {
12
+ const t = useTheme();
13
+ const gradeColor = t.grade(health.grade);
14
+ if (compact) {
15
+ return (_jsxs(Box, { marginLeft: 2, children: [_jsx(GradeBadge, { grade: health.grade, score: health.score }), _jsx(Text, { children: " " }), _jsx(ProgressBar, { value: health.score / 100, width: 16, color: gradeColor }), _jsxs(Text, { color: t.dim, children: [" ", health.totalEntities, " entities \u00B7 ", health.totalEdges, " edges"] })] }));
16
+ }
17
+ return (_jsxs(Box, { flexDirection: "column", marginLeft: 2, children: [_jsxs(Box, { children: [_jsx(Text, { children: " " }), _jsx(GradeBadge, { grade: health.grade, score: health.score }), _jsx(Text, { children: " " }), _jsx(ProgressBar, { value: health.score / 100, width: 20, color: gradeColor, showLabel: true })] }), health.deadFunctionCount > 0 && (_jsx(Box, { marginLeft: 2, children: _jsxs(Text, { color: t.warning, children: ["\u26A1 ", health.deadFunctionCount, " dead function", health.deadFunctionCount !== 1 ? "s" : "", " \u2014 your agent reads these but nothing calls them"] }) })), health.highRiskEntities.map((entity) => (_jsx(Box, { marginLeft: 2, children: _jsxs(Text, { color: t.warning, children: ["\u26A0 ", entity.name, " \u2192 ", entity.fan_in, " caller", entity.fan_in !== 1 ? "s" : "", ", ", entity.fan_out, " callee", entity.fan_out !== 1 ? "s" : "", " (chokepoint)"] }) }, entity.name)))] }));
18
+ }
@@ -0,0 +1,22 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
2
+ /**
3
+ * Animated spinner with status text.
4
+ *
5
+ * Named InkSpinner to avoid conflict with ora-based spinners.
6
+ * Uses Ink's useEffect for frame animation.
7
+ */
8
+ import { Box, Text } from "ink";
9
+ import { useEffect, useState } from "react";
10
+ import { useTheme } from "./Theme.js";
11
+ const SPINNER_FRAMES = ["⠋", "⠙", "⠹", "⠸", "⠼", "⠴", "⠦", "⠧", "⠇", "⠏"];
12
+ export function InkSpinner({ label, interval = 80, }) {
13
+ const t = useTheme();
14
+ const [frame, setFrame] = useState(0);
15
+ useEffect(() => {
16
+ const timer = setInterval(() => {
17
+ setFrame((prev) => (prev + 1) % SPINNER_FRAMES.length);
18
+ }, interval);
19
+ return () => clearInterval(timer);
20
+ }, [interval]);
21
+ return (_jsxs(Box, { marginLeft: 2, children: [_jsx(Text, { color: t.info, children: SPINNER_FRAMES[frame] }), _jsxs(Text, { children: [" ", label] })] }));
22
+ }
@@ -0,0 +1,17 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
2
+ /**
3
+ * InputBox — user input component with prompt indicator.
4
+ */
5
+ import { Box, Text } from "ink";
6
+ import TextInput from "ink-text-input";
7
+ import { useState } from "react";
8
+ export function InputBox({ onSubmit, isDisabled }) {
9
+ const [value, setValue] = useState("");
10
+ const handleSubmit = (input) => {
11
+ if (isDisabled)
12
+ return;
13
+ onSubmit(input);
14
+ setValue("");
15
+ };
16
+ return (_jsxs(Box, { children: [_jsx(Text, { color: isDisabled ? "gray" : "cyan", bold: true, children: isDisabled ? "⏳ " : "❯ " }), isDisabled ? (_jsx(Text, { color: "gray", children: "Thinking..." })) : (_jsx(TextInput, { value: value, onChange: setValue, onSubmit: handleSubmit, placeholder: "Ask anything about your code..." }))] }));
17
+ }
@@ -0,0 +1,13 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
2
+ /**
3
+ * Aligned key-value display: Label: Value
4
+ *
5
+ * Label is dimmed, value is normal. Used in status, session summary.
6
+ */
7
+ import { Box, Text } from "ink";
8
+ import { useTheme } from "./Theme.js";
9
+ export function KeyValue({ label, value, labelWidth = 20, valueColor, }) {
10
+ const t = useTheme();
11
+ const padded = label.padEnd(labelWidth);
12
+ return (_jsxs(Box, { marginLeft: 4, children: [_jsx(Text, { color: t.dim, children: padded }), _jsx(Text, { color: valueColor, children: value })] }));
13
+ }
@@ -0,0 +1,14 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
2
+ /**
3
+ * MessageList — renders conversation history.
4
+ */
5
+ import { Box, Text } from "ink";
6
+ export function MessageList({ messages }) {
7
+ return (_jsx(Box, { flexDirection: "column", children: messages.map((msg, i) => (_jsx(MessageBubble, { message: msg }, `msg-${msg.role}-${i}`))) }));
8
+ }
9
+ function MessageBubble({ message }) {
10
+ if (message.role === "user") {
11
+ return (_jsxs(Box, { marginBottom: 1, children: [_jsxs(Text, { bold: true, color: "blue", children: ["You:", " "] }), _jsx(Text, { children: message.content })] }));
12
+ }
13
+ return (_jsxs(Box, { flexDirection: "column", marginBottom: 1, children: [message.toolCalls && message.toolCalls.length > 0 && (_jsx(Box, { marginBottom: 0, children: message.toolCalls.map((tc, i) => (_jsx(Box, { marginRight: 1, children: _jsxs(Text, { color: "yellow", dimColor: true, children: ["[", tc.name, "]"] }) }, `tc-${tc.name}`))) })), _jsxs(Box, { children: [_jsxs(Text, { bold: true, color: "green", children: ["unerr:", " "] }), _jsx(Text, { children: message.content })] })] }));
14
+ }