@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,141 @@
1
+ import { existsSync, mkdirSync, readFileSync, rmSync, writeFileSync, } from "node:fs";
2
+ import { tmpdir } from "node:os";
3
+ import { join } from "node:path";
4
+ import { afterEach, beforeEach, describe, expect, it } from "vitest";
5
+ import { normalizeAgentName } from "../config/agent-registry.js";
6
+ import { mergePreToolUseBashHook, removePreToolUseBashHook, } from "../config/claude-settings-hooks.js";
7
+ import { removeInstructionSection, writeInstructionFile, } from "../config/instruction-writer.js";
8
+ import { removeMcpConfig, writeMcpConfig, } from "../config/mcp-config-writer.js";
9
+ import { removeInstalledSkills } from "../skills/resolver.js";
10
+ describe("uninstall", () => {
11
+ let tmpDir;
12
+ beforeEach(() => {
13
+ tmpDir = join(tmpdir(), `unerr-uninstall-test-${Date.now()}`);
14
+ mkdirSync(tmpDir, { recursive: true });
15
+ });
16
+ afterEach(() => {
17
+ rmSync(tmpDir, { recursive: true, force: true });
18
+ });
19
+ describe("normalizeAgentName", () => {
20
+ it("resolves 'claude' to 'claude-code'", () => {
21
+ expect(normalizeAgentName("claude")).toBe("claude-code");
22
+ });
23
+ it("resolves 'Claude' case-insensitively", () => {
24
+ expect(normalizeAgentName("Claude")).toBe("claude-code");
25
+ });
26
+ it("resolves 'gemini' to 'gemini-cli'", () => {
27
+ expect(normalizeAgentName("gemini")).toBe("gemini-cli");
28
+ });
29
+ it("passes through unknown names unchanged", () => {
30
+ expect(normalizeAgentName("cursor")).toBe("cursor");
31
+ });
32
+ });
33
+ describe("removeInstalledSkills", () => {
34
+ it("removes flat skill files for cursor", () => {
35
+ const rulesDir = join(tmpDir, ".cursor", "rules");
36
+ mkdirSync(rulesDir, { recursive: true });
37
+ writeFileSync(join(rulesDir, "unerr-graph-first.mdc"), "content");
38
+ writeFileSync(join(rulesDir, "unerr-search.mdc"), "content");
39
+ writeFileSync(join(rulesDir, "other-rule.mdc"), "should stay");
40
+ const removed = removeInstalledSkills("cursor", tmpDir);
41
+ expect(removed).toBe(2);
42
+ expect(existsSync(join(rulesDir, "unerr-graph-first.mdc"))).toBe(false);
43
+ expect(existsSync(join(rulesDir, "unerr-search.mdc"))).toBe(false);
44
+ expect(existsSync(join(rulesDir, "other-rule.mdc"))).toBe(true);
45
+ });
46
+ it("removes directory-per-skill for claude-code", () => {
47
+ const skillsDir = join(tmpDir, ".claude", "skills");
48
+ const skillDir = join(skillsDir, "unerr-graph-first");
49
+ mkdirSync(skillDir, { recursive: true });
50
+ writeFileSync(join(skillDir, "SKILL.md"), "content");
51
+ const removed = removeInstalledSkills("claude-code", tmpDir);
52
+ expect(removed).toBe(1);
53
+ expect(existsSync(skillDir)).toBe(false);
54
+ });
55
+ it("returns 0 when no skills exist", () => {
56
+ const removed = removeInstalledSkills("cursor", tmpDir);
57
+ expect(removed).toBe(0);
58
+ });
59
+ });
60
+ describe("removePreToolUseBashHook", () => {
61
+ it("removes unerr hook from settings.json", () => {
62
+ // First merge it in
63
+ mergePreToolUseBashHook(tmpDir);
64
+ const settingsPath = join(tmpDir, ".claude", "settings.json");
65
+ expect(existsSync(settingsPath)).toBe(true);
66
+ const before = JSON.parse(readFileSync(settingsPath, "utf-8"));
67
+ expect(before.hooks.PreToolUse).toHaveLength(6); // Bash, Read, Grep, Glob, Write, Edit
68
+ // Now remove
69
+ const removed = removePreToolUseBashHook(tmpDir);
70
+ expect(removed).toBe(true);
71
+ const after = JSON.parse(readFileSync(settingsPath, "utf-8"));
72
+ // PreToolUse should be cleaned up
73
+ expect(after.hooks?.PreToolUse).toBeUndefined();
74
+ });
75
+ it("preserves other hooks when removing unerr hook", () => {
76
+ const dir = join(tmpDir, ".claude");
77
+ mkdirSync(dir, { recursive: true });
78
+ writeFileSync(join(dir, "settings.json"), JSON.stringify({
79
+ hooks: {
80
+ PreToolUse: [
81
+ {
82
+ matcher: "Bash",
83
+ hooks: [{ type: "command", command: "other-tool" }],
84
+ },
85
+ {
86
+ matcher: "Bash",
87
+ hooks: [{ type: "command", command: "unerr hook pre-bash" }],
88
+ },
89
+ ],
90
+ PostToolUse: [
91
+ {
92
+ matcher: "Write",
93
+ hooks: [{ type: "command", command: "lint" }],
94
+ },
95
+ ],
96
+ },
97
+ }));
98
+ const removed = removePreToolUseBashHook(tmpDir);
99
+ expect(removed).toBe(true);
100
+ const after = JSON.parse(readFileSync(join(dir, "settings.json"), "utf-8"));
101
+ expect(after.hooks.PreToolUse).toHaveLength(1);
102
+ expect(after.hooks.PreToolUse[0].hooks[0].command).toBe("other-tool");
103
+ expect(after.hooks.PostToolUse).toHaveLength(1);
104
+ });
105
+ it("returns false when no settings.json exists", () => {
106
+ expect(removePreToolUseBashHook(tmpDir)).toBe(false);
107
+ });
108
+ it("returns false when no unerr hook present", () => {
109
+ const dir = join(tmpDir, ".claude");
110
+ mkdirSync(dir, { recursive: true });
111
+ writeFileSync(join(dir, "settings.json"), JSON.stringify({ hooks: { PreToolUse: [] } }));
112
+ expect(removePreToolUseBashHook(tmpDir)).toBe(false);
113
+ });
114
+ });
115
+ describe("per-agent uninstall symmetry", () => {
116
+ it("removeMcpConfig reverses writeMcpConfig for cursor", () => {
117
+ const result = writeMcpConfig(tmpDir, "cursor");
118
+ expect(existsSync(result.path)).toBe(true);
119
+ const removed = removeMcpConfig(tmpDir, "cursor");
120
+ expect(removed).toBe(true);
121
+ });
122
+ it("removeInstructionSection reverses writeInstructionFile", () => {
123
+ writeInstructionFile(tmpDir, "claude-code");
124
+ const claudeMd = join(tmpDir, "CLAUDE.md");
125
+ expect(existsSync(claudeMd)).toBe(true);
126
+ const removed = removeInstructionSection(tmpDir, "claude-code");
127
+ expect(removed).toBe(true);
128
+ // File should be deleted since it only had sentinel content
129
+ expect(existsSync(claudeMd)).toBe(false);
130
+ });
131
+ it("uninstall is idempotent — second call returns false/0", () => {
132
+ // Install then uninstall
133
+ writeMcpConfig(tmpDir, "cursor");
134
+ removeMcpConfig(tmpDir, "cursor");
135
+ // Second uninstall should be no-ops
136
+ expect(removeMcpConfig(tmpDir, "cursor")).toBe(false);
137
+ expect(removeInstalledSkills("cursor", tmpDir)).toBe(0);
138
+ expect(removeInstructionSection(tmpDir, "cursor")).toBe(false);
139
+ });
140
+ });
141
+ });
@@ -0,0 +1,271 @@
1
+ /**
2
+ * Tests for DM-5 warm-start policy:
3
+ * - Candidate selection respects autostart, budget, idle days
4
+ * - MRU ordering (eager first, then by lastActivity)
5
+ * - Budget enforcement (only top N warmed)
6
+ * - Battery/load abort
7
+ * - Budget=0 disables entirely
8
+ */
9
+ import { mkdirSync, rmSync } from "node:fs";
10
+ import { tmpdir } from "node:os";
11
+ import { join } from "node:path";
12
+ import { afterEach, describe, expect, it, vi } from "vitest";
13
+ import { selectCandidates, } from "../daemon/warm-start.js";
14
+ function makeRepo(name, opts = {}) {
15
+ const path = opts.path ??
16
+ join(tmpdir(), `warmtest-${name}-${Date.now()}-${Math.random().toString(36).slice(2, 8)}`);
17
+ if (opts.exists !== false) {
18
+ mkdirSync(path, { recursive: true });
19
+ }
20
+ const settings = {};
21
+ if (opts.autostart)
22
+ settings.autostart = opts.autostart;
23
+ return {
24
+ path,
25
+ addedAt: new Date().toISOString(),
26
+ lastStarted: null,
27
+ lastActivity: opts.lastActivity?.toISOString() ?? null,
28
+ idleTimeout: 1800,
29
+ label: name,
30
+ settings,
31
+ };
32
+ }
33
+ const tempPaths = [];
34
+ function trackedRepo(name, opts = {}) {
35
+ const repo = makeRepo(name, opts);
36
+ tempPaths.push(repo.path);
37
+ return repo;
38
+ }
39
+ afterEach(() => {
40
+ for (const p of tempPaths) {
41
+ try {
42
+ rmSync(p, { recursive: true, force: true });
43
+ }
44
+ catch {
45
+ /* ok */
46
+ }
47
+ }
48
+ tempPaths.length = 0;
49
+ });
50
+ const defaultConfig = {
51
+ warmStartBudget: 3,
52
+ warmStartIdleDays: 14,
53
+ warmStartDelayMs: 0,
54
+ };
55
+ describe("selectCandidates", () => {
56
+ it("selects MRU repos up to budget", () => {
57
+ const repos = [
58
+ trackedRepo("repo1", { lastActivity: new Date("2026-05-14T10:00:00Z") }),
59
+ trackedRepo("repo2", { lastActivity: new Date("2026-05-13T10:00:00Z") }),
60
+ trackedRepo("repo3", { lastActivity: new Date("2026-05-12T10:00:00Z") }),
61
+ trackedRepo("repo4", { lastActivity: new Date("2026-05-11T10:00:00Z") }),
62
+ trackedRepo("repo5", { lastActivity: new Date("2026-05-10T10:00:00Z") }),
63
+ ];
64
+ const { candidates, skipped } = selectCandidates(repos, {
65
+ ...defaultConfig,
66
+ warmStartBudget: 3,
67
+ });
68
+ // All 5 are candidates (within idle window), but budget limits to 3
69
+ expect(candidates.length).toBe(5); // selectCandidates returns all eligible; budget is enforced during run
70
+ expect(candidates[0].entry.label).toBe("repo1");
71
+ expect(candidates[1].entry.label).toBe("repo2");
72
+ expect(candidates[2].entry.label).toBe("repo3");
73
+ });
74
+ it("excludes repos with autostart=never", () => {
75
+ const repos = [
76
+ trackedRepo("active", { lastActivity: new Date("2026-05-14T10:00:00Z") }),
77
+ trackedRepo("never", {
78
+ lastActivity: new Date("2026-05-14T10:00:00Z"),
79
+ autostart: "never",
80
+ }),
81
+ ];
82
+ const { candidates, skipped } = selectCandidates(repos, defaultConfig);
83
+ expect(candidates.length).toBe(1);
84
+ expect(candidates[0].entry.label).toBe("active");
85
+ expect(skipped.length).toBe(1);
86
+ expect(skipped[0].reason).toBe("autostart=never");
87
+ });
88
+ it("excludes repos inactive beyond idle days cutoff", () => {
89
+ const repos = [
90
+ trackedRepo("recent", { lastActivity: new Date("2026-05-14T10:00:00Z") }),
91
+ trackedRepo("old", { lastActivity: new Date("2026-01-01T10:00:00Z") }),
92
+ ];
93
+ const { candidates, skipped } = selectCandidates(repos, {
94
+ ...defaultConfig,
95
+ warmStartIdleDays: 14,
96
+ });
97
+ expect(candidates.length).toBe(1);
98
+ expect(candidates[0].entry.label).toBe("recent");
99
+ expect(skipped.length).toBe(1);
100
+ expect(skipped[0].reason).toContain("inactive");
101
+ });
102
+ it("eager repos are prioritized over auto repos", () => {
103
+ const repos = [
104
+ trackedRepo("auto-recent", {
105
+ lastActivity: new Date("2026-05-14T10:00:00Z"),
106
+ autostart: "auto",
107
+ }),
108
+ trackedRepo("eager-old", {
109
+ lastActivity: new Date("2026-05-10T10:00:00Z"),
110
+ autostart: "eager",
111
+ }),
112
+ ];
113
+ const { candidates } = selectCandidates(repos, defaultConfig);
114
+ expect(candidates[0].entry.label).toBe("eager-old");
115
+ expect(candidates[1].entry.label).toBe("auto-recent");
116
+ });
117
+ it("skips repos whose directory does not exist", () => {
118
+ const repos = [
119
+ trackedRepo("good", { lastActivity: new Date("2026-05-14T10:00:00Z") }),
120
+ trackedRepo("gone", {
121
+ lastActivity: new Date("2026-05-14T10:00:00Z"),
122
+ exists: false,
123
+ }),
124
+ ];
125
+ const { candidates, skipped } = selectCandidates(repos, defaultConfig);
126
+ expect(candidates.length).toBe(1);
127
+ expect(skipped.length).toBe(1);
128
+ expect(skipped[0].reason).toBe("directory not found");
129
+ });
130
+ it("repos without lastActivity are included (never started)", () => {
131
+ const repos = [trackedRepo("new-repo", { lastActivity: null })];
132
+ const { candidates } = selectCandidates(repos, defaultConfig);
133
+ expect(candidates.length).toBe(1);
134
+ });
135
+ it("eager repos bypass idle cutoff even if old", () => {
136
+ const repos = [
137
+ trackedRepo("eager-old", {
138
+ lastActivity: new Date("2025-01-01T10:00:00Z"),
139
+ autostart: "eager",
140
+ }),
141
+ ];
142
+ const { candidates } = selectCandidates(repos, {
143
+ ...defaultConfig,
144
+ warmStartIdleDays: 7,
145
+ });
146
+ // Eager repos are never filtered by idle cutoff
147
+ expect(candidates.length).toBe(1);
148
+ });
149
+ });
150
+ describe("warm-start budget enforcement", () => {
151
+ it("budget=0 returns empty results", async () => {
152
+ vi.mock("../daemon/detect-ci.js", () => ({
153
+ isCI: vi.fn(() => false),
154
+ resetCICache: vi.fn(),
155
+ }));
156
+ const { loadWarmStartConfig } = await import("../daemon/warm-start.js");
157
+ const config = loadWarmStartConfig();
158
+ // Test the config loading works
159
+ expect(typeof config.warmStartBudget).toBe("number");
160
+ });
161
+ it("table-driven: 20 repos with budget=3 selects exactly top 3", () => {
162
+ const repos = [];
163
+ for (let i = 0; i < 20; i++) {
164
+ repos.push(trackedRepo(`repo-${i}`, {
165
+ lastActivity: new Date(Date.now() - i * 86400000),
166
+ autostart: i < 2 ? "eager" : i >= 18 ? "never" : "auto",
167
+ }));
168
+ }
169
+ const { candidates, skipped } = selectCandidates(repos, {
170
+ ...defaultConfig,
171
+ warmStartBudget: 3,
172
+ });
173
+ // 2 never repos skipped
174
+ expect(skipped.filter((s) => s.reason === "autostart=never").length).toBe(2);
175
+ // Eager repos should be first in candidates
176
+ expect(candidates[0].entry.settings.autostart).toBe("eager");
177
+ expect(candidates[1].entry.settings.autostart).toBe("eager");
178
+ });
179
+ it("table-driven: 20 repos with budget=5 selects top 5", () => {
180
+ const repos = [];
181
+ for (let i = 0; i < 20; i++) {
182
+ repos.push(trackedRepo(`repo-${i}`, {
183
+ lastActivity: new Date(Date.now() - i * 86400000),
184
+ autostart: i >= 18 ? "never" : "auto",
185
+ }));
186
+ }
187
+ const config = {
188
+ ...defaultConfig,
189
+ warmStartBudget: 5,
190
+ warmStartIdleDays: 30,
191
+ };
192
+ const { candidates } = selectCandidates(repos, config);
193
+ // All non-never repos within 30 days are candidates
194
+ expect(candidates.length).toBeLessThanOrEqual(18);
195
+ // First 5 would be the ones the scheduler picks
196
+ const topFive = candidates.slice(0, 5);
197
+ expect(topFive.length).toBe(5);
198
+ });
199
+ it("table-driven: budget=0 means selectCandidates still works but nothing warmed", () => {
200
+ const repos = [trackedRepo("repo-a", { lastActivity: new Date() })];
201
+ const config = { ...defaultConfig, warmStartBudget: 0 };
202
+ // With budget=0, the scheduler itself returns early before calling selectCandidates
203
+ // But selectCandidates itself still returns valid results
204
+ const { candidates } = selectCandidates(repos, config);
205
+ expect(candidates.length).toBe(1);
206
+ });
207
+ });
208
+ describe("system health integration", () => {
209
+ it("onBattery returns boolean", async () => {
210
+ const { onBattery, resetHealthCaches } = await import("../daemon/system-health.js");
211
+ resetHealthCaches();
212
+ const result = onBattery();
213
+ expect(typeof result).toBe("boolean");
214
+ });
215
+ it("loadAverage1 returns non-negative number", async () => {
216
+ const { loadAverage1, resetHealthCaches } = await import("../daemon/system-health.js");
217
+ resetHealthCaches();
218
+ const result = loadAverage1();
219
+ expect(typeof result).toBe("number");
220
+ expect(result).toBeGreaterThanOrEqual(0);
221
+ });
222
+ it("caches are reset correctly", async () => {
223
+ const { onBattery, loadAverage1, resetHealthCaches } = await import("../daemon/system-health.js");
224
+ const b1 = onBattery();
225
+ const l1 = loadAverage1();
226
+ resetHealthCaches();
227
+ // After reset, should still return valid values
228
+ const b2 = onBattery();
229
+ const l2 = loadAverage1();
230
+ expect(typeof b2).toBe("boolean");
231
+ expect(typeof l2).toBe("number");
232
+ });
233
+ });
234
+ describe("warm-start config", () => {
235
+ it("loadWarmStartConfig returns defaults when no config exists", async () => {
236
+ const { loadWarmStartConfig } = await import("../daemon/warm-start.js");
237
+ const config = loadWarmStartConfig();
238
+ expect(config.warmStartBudget).toBe(3);
239
+ expect(config.warmStartIdleDays).toBe(14);
240
+ expect(config.warmStartDelayMs).toBe(30_000);
241
+ });
242
+ it("saveWarmStartConfig persists values", async () => {
243
+ const { saveWarmStartConfig, loadWarmStartConfig } = await import("../daemon/warm-start.js");
244
+ // This test modifies the real config file — we accept that as it's using the real ~/.unerr/config.json
245
+ // In a real CI environment, this would need a mock
246
+ expect(typeof saveWarmStartConfig).toBe("function");
247
+ expect(typeof loadWarmStartConfig).toBe("function");
248
+ });
249
+ });
250
+ describe("WarmStartEvent types", () => {
251
+ it("events have correct shape", () => {
252
+ const event = {
253
+ type: "warm_start",
254
+ repo: "/tmp/test",
255
+ label: "test",
256
+ status: "started",
257
+ ms: 1500,
258
+ };
259
+ expect(event.type).toBe("warm_start");
260
+ expect(event.status).toBe("started");
261
+ const skipped = {
262
+ type: "warm_start",
263
+ repo: "/tmp/test2",
264
+ label: "test2",
265
+ status: "skipped",
266
+ ms: 0,
267
+ reason: "autostart=never",
268
+ };
269
+ expect(skipped.reason).toBe("autostart=never");
270
+ });
271
+ });
@@ -0,0 +1,77 @@
1
+ /**
2
+ * Verifies the wire-cap nudge interpolation is concrete (numeric values, not
3
+ * literal `N`) and context-aware (knows whether entity/limit/token_budget
4
+ * were already set).
5
+ *
6
+ * Before this change, the same generic "narrow with limit:N/entity:<name> or
7
+ * token_budget:N" hint reappeared on every retry — even when the caller had
8
+ * already passed entity: or token_budget: — producing a retry loop.
9
+ */
10
+ import { describe, expect, it } from "vitest";
11
+ import { applyWireCap } from "../proxy/wire-cap.js";
12
+ function bigString(bytes) {
13
+ return "x".repeat(bytes);
14
+ }
15
+ describe("wire-cap too_large nudge — concrete & context-aware", () => {
16
+ it("emits a numeric suggested_token_budget in the body", () => {
17
+ const oversized = bigString(20_000);
18
+ const { body, pageHint } = applyWireCap("file_read", oversized, {});
19
+ const obj = body;
20
+ expect(obj.status).toBe("too_large");
21
+ expect(typeof obj.suggested_token_budget).toBe("number");
22
+ expect(obj.needed_tokens).toBeGreaterThan(0);
23
+ expect(pageHint).toMatch(/token_budget:\d+/);
24
+ expect(pageHint).not.toMatch(/token_budget:N/);
25
+ });
26
+ it("uses `entity_too_large` reason when caller already passed entity", () => {
27
+ const oversized = bigString(20_000);
28
+ const { body, pageHint } = applyWireCap("file_read", oversized, {
29
+ entity: "indexLocalProject",
30
+ });
31
+ const obj = body;
32
+ expect(obj.reason).toBe("entity_too_large");
33
+ expect(obj.entity).toBe("indexLocalProject");
34
+ expect(pageHint).toContain("offset/limit");
35
+ // Critical: must NOT tell caller to "narrow with entity:" — they did.
36
+ expect(pageHint).not.toContain("entity:<name>");
37
+ });
38
+ it("echoes the requested_token_budget when caller already set one", () => {
39
+ const oversized = bigString(40_000);
40
+ const { body } = applyWireCap("file_read", oversized, {
41
+ token_budget: 2000,
42
+ });
43
+ const obj = body;
44
+ expect(obj.requested_token_budget).toBe(2000);
45
+ expect(obj.suggested_token_budget).toBeGreaterThan(2000);
46
+ });
47
+ it("uses `page_too_large` and shrinks the limit when paginated", () => {
48
+ const oversized = bigString(20_000);
49
+ const { body, pageHint } = applyWireCap("search_code", oversized, {
50
+ limit: 100,
51
+ });
52
+ const obj = body;
53
+ expect(obj.reason).toBe("page_too_large");
54
+ expect(pageHint).toMatch(/limit:\d+/);
55
+ expect(pageHint).not.toMatch(/limit:N/);
56
+ });
57
+ it("uses `narrow_required` when token_budget was lifted and still overflowed", () => {
58
+ // 16384 / 4 = 4096 — bump well above HARD_BYTE_CAP so a token_budget of
59
+ // 5000 lifts the cap to 20000 bytes, but the payload is bigger still.
60
+ const oversized = bigString(30_000);
61
+ const { body } = applyWireCap("file_read", oversized, {
62
+ token_budget: 5000,
63
+ });
64
+ const obj = body;
65
+ expect(obj.reason).toBe("narrow_required");
66
+ });
67
+ });
68
+ describe("wire-cap pagination hint — concrete next cursor", () => {
69
+ it("emits a numeric cursor (not the literal N)", () => {
70
+ // search_code returns a top-level array; its cursorArg is `limit`.
71
+ const arr = Array.from({ length: 50 }, (_, i) => ({ id: i }));
72
+ const { pageHint } = applyWireCap("search_code", arr, {});
73
+ // Cursor should be a real number, never the placeholder.
74
+ expect(pageHint).toMatch(/:\d+/);
75
+ expect(pageHint).not.toMatch(/:N(\s|\/|$)/);
76
+ });
77
+ });
@@ -0,0 +1,101 @@
1
+ /**
2
+ * Worker pool tests — parallel AST parsing via tinypool.
3
+ */
4
+ import { mkdtempSync, rmSync, writeFileSync } from "node:fs";
5
+ import { tmpdir } from "node:os";
6
+ import { join } from "node:path";
7
+ import { afterEach, beforeEach, describe, expect, it } from "vitest";
8
+ import { destroy, parseFiles, resetPoolState, } from "../intelligence/worker-pool.js";
9
+ const SAMPLE_TS = `
10
+ export function greet(name: string): string {
11
+ return \`Hello, \${name}\`;
12
+ }
13
+
14
+ export class Greeter {
15
+ private name: string;
16
+
17
+ constructor(name: string) {
18
+ this.name = name;
19
+ }
20
+
21
+ sayHello(): string {
22
+ return \`Hello, \${this.name}\`;
23
+ }
24
+ }
25
+
26
+ export interface GreetOptions {
27
+ loud: boolean;
28
+ prefix?: string;
29
+ }
30
+ `;
31
+ describe("worker-pool", () => {
32
+ let tempDir;
33
+ beforeEach(() => {
34
+ tempDir = mkdtempSync(join(tmpdir(), "unerr-wp-"));
35
+ });
36
+ afterEach(async () => {
37
+ await destroy();
38
+ resetPoolState();
39
+ rmSync(tempDir, { recursive: true, force: true });
40
+ });
41
+ it("initializes and parses a single file", async () => {
42
+ const filePath = join(tempDir, "sample.ts");
43
+ writeFileSync(filePath, SAMPLE_TS);
44
+ const results = await parseFiles([{ filePath, content: SAMPLE_TS }]);
45
+ expect(results).toHaveLength(1);
46
+ expect(results[0]?.filePath).toBe(filePath);
47
+ expect(results[0]?.entities.length).toBeGreaterThan(0);
48
+ expect(results[0]?.durationMs).toBeGreaterThanOrEqual(0);
49
+ const names = results[0]?.entities.map((e) => e.name);
50
+ expect(names).toContain("greet");
51
+ expect(names).toContain("Greeter");
52
+ expect(names).toContain("GreetOptions");
53
+ });
54
+ it("parses 10 files in parallel", async () => {
55
+ const files = Array.from({ length: 10 }, (_, i) => {
56
+ const filePath = join(tempDir, `file${i}.ts`);
57
+ const content = `export function fn${i}(x: number): number { return x * ${i}; }\n`;
58
+ writeFileSync(filePath, content);
59
+ return { filePath, content };
60
+ });
61
+ const results = await parseFiles(files);
62
+ expect(results).toHaveLength(10);
63
+ for (let i = 0; i < 10; i++) {
64
+ expect(results[i]?.filePath).toBe(files[i]?.filePath);
65
+ expect(results[i]?.entities.length).toBeGreaterThanOrEqual(1);
66
+ const fnEntity = results[i]?.entities.find((e) => e.name === `fn${i}`);
67
+ expect(fnEntity).toBeDefined();
68
+ expect(fnEntity?.kind).toBe("function");
69
+ }
70
+ });
71
+ it("handles empty input", async () => {
72
+ const results = await parseFiles([]);
73
+ expect(results).toEqual([]);
74
+ });
75
+ it("handles files with no extractable entities", async () => {
76
+ const filePath = join(tempDir, "data.json");
77
+ const content = '{"key": "value"}';
78
+ writeFileSync(filePath, content);
79
+ const results = await parseFiles([{ filePath, content }]);
80
+ expect(results).toHaveLength(1);
81
+ expect(results[0]?.entities).toEqual([]);
82
+ });
83
+ it("can be destroyed and recreated", async () => {
84
+ const filePath = join(tempDir, "first.ts");
85
+ const content = "export function first(): void {}\n";
86
+ writeFileSync(filePath, content);
87
+ const results1 = await parseFiles([{ filePath, content }]);
88
+ expect(results1).toHaveLength(1);
89
+ expect(results1[0]?.entities.length).toBeGreaterThan(0);
90
+ await destroy();
91
+ resetPoolState();
92
+ const filePath2 = join(tempDir, "second.ts");
93
+ const content2 = "export function second(): void {}\n";
94
+ writeFileSync(filePath2, content2);
95
+ const results2 = await parseFiles([
96
+ { filePath: filePath2, content: content2 },
97
+ ]);
98
+ expect(results2).toHaveLength(1);
99
+ expect(results2[0]?.entities.find((e) => e.name === "second")).toBeDefined();
100
+ });
101
+ });