@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,302 @@
1
+ /**
2
+ * Tests for StartupDisplay + StartupRenderer (Task 1.3).
3
+ *
4
+ * Tests validate:
5
+ * - Three-Act structure: Banner (Act 1), HealthCard (Act 2), Invitation (Act 3)
6
+ * - Step status progression: pending → active → done
7
+ * - Health Shock display: full card on first boot, compact on subsequent
8
+ * - Invitation references specific entity from health data
9
+ * - Deep link included when repo ID available
10
+ * - Proxy mode displayed when not "full"
11
+ */
12
+ import * as fs from "node:fs";
13
+ import * as os from "node:os";
14
+ import * as path from "node:path";
15
+ import { render } from "ink-testing-library";
16
+ import React from "react";
17
+ import { afterEach, beforeEach, describe, expect, it } from "vitest";
18
+ import { StartupDisplay, } from "../components/StartupDisplay.js";
19
+ import { ThemeProvider } from "../components/Theme.js";
20
+ function renderStartup(state) {
21
+ return render(React.createElement(ThemeProvider, null, React.createElement(StartupDisplay, { state })));
22
+ }
23
+ describe("StartupDisplay (1.3)", () => {
24
+ // ── Act 1: Instant Competence ──────────────────────────────────
25
+ describe("Act 1: Banner + Steps", () => {
26
+ it("renders brand banner", () => {
27
+ const { lastFrame } = renderStartup({
28
+ localMode: false,
29
+ steps: [],
30
+ firstBoot: false,
31
+ ready: false,
32
+ });
33
+ expect(lastFrame()).toContain("unerr");
34
+ expect(lastFrame()).toContain("▸");
35
+ });
36
+ it("renders steps with correct status icons", () => {
37
+ const { lastFrame } = renderStartup({
38
+ localMode: false,
39
+ steps: [
40
+ { label: "Authenticated", value: "Org", status: "done" },
41
+ { label: "Repository", value: "acme/repo", status: "done" },
42
+ { label: "Graph loaded", status: "active" },
43
+ ],
44
+ firstBoot: false,
45
+ ready: false,
46
+ });
47
+ const frame = lastFrame() ?? "";
48
+ expect(frame).toContain("✓");
49
+ expect(frame).toContain("Authenticated");
50
+ expect(frame).toContain("Org");
51
+ expect(frame).toContain("●");
52
+ expect(frame).toContain("Graph loaded");
53
+ });
54
+ it("shows step values alongside labels", () => {
55
+ const { lastFrame } = renderStartup({
56
+ localMode: false,
57
+ steps: [
58
+ {
59
+ label: "Graph loaded",
60
+ value: "2,341 entities · 1,892 edges",
61
+ status: "done",
62
+ },
63
+ ],
64
+ firstBoot: false,
65
+ ready: false,
66
+ });
67
+ expect(lastFrame()).toContain("2,341 entities");
68
+ expect(lastFrame()).toContain("1,892 edges");
69
+ });
70
+ });
71
+ // ── Act 2: Revelation (Health Shock) ───────────────────────────
72
+ describe("Act 2: Health Shock", () => {
73
+ const health = {
74
+ grade: "C+",
75
+ totalEntities: 2341,
76
+ totalEdges: 1892,
77
+ totalRules: 12,
78
+ deadFunctionCount: 23,
79
+ highRiskEntities: [
80
+ {
81
+ name: "processPayment",
82
+ kind: "function",
83
+ file_path: "src/billing.ts",
84
+ fan_in: 14,
85
+ fan_out: 8,
86
+ },
87
+ ],
88
+ score: 62,
89
+ };
90
+ it("renders full health card on first boot", () => {
91
+ const { lastFrame } = renderStartup({
92
+ localMode: false,
93
+ steps: [],
94
+ health,
95
+ firstBoot: true,
96
+ ready: false,
97
+ });
98
+ const frame = lastFrame() ?? "";
99
+ expect(frame).toContain("First Look");
100
+ expect(frame).toContain("C+");
101
+ expect(frame).toContain("62/100");
102
+ expect(frame).toContain("23 dead functions");
103
+ expect(frame).toContain("processPayment");
104
+ });
105
+ it("renders compact health line on subsequent boots", () => {
106
+ const { lastFrame } = renderStartup({
107
+ localMode: false,
108
+ steps: [],
109
+ health,
110
+ firstBoot: false,
111
+ ready: false,
112
+ });
113
+ const frame = lastFrame() ?? "";
114
+ expect(frame).toContain("C+");
115
+ expect(frame).toContain("2341 entities");
116
+ // Should NOT show full dead function detail in compact
117
+ expect(frame).not.toContain("23 dead functions");
118
+ });
119
+ it("does not render health section when no health data", () => {
120
+ const { lastFrame } = renderStartup({
121
+ localMode: false,
122
+ steps: [],
123
+ firstBoot: false,
124
+ ready: false,
125
+ });
126
+ expect(lastFrame()).not.toContain("First Look");
127
+ });
128
+ });
129
+ // ── Act 3: Invitation ──────────────────────────────────────────
130
+ describe("Act 3: Invitation", () => {
131
+ it("renders invitation with specific entity name", () => {
132
+ const { lastFrame } = renderStartup({
133
+ localMode: false,
134
+ steps: [],
135
+ firstBoot: false,
136
+ ready: true,
137
+ invitationEntity: "processPayment",
138
+ });
139
+ const frame = lastFrame() ?? "";
140
+ expect(frame).toContain("What depends on processPayment?");
141
+ expect(frame).toContain("blast radius");
142
+ });
143
+ it("renders proxy ready message", () => {
144
+ const { lastFrame } = renderStartup({
145
+ localMode: false,
146
+ steps: [],
147
+ firstBoot: false,
148
+ ready: true,
149
+ });
150
+ expect(lastFrame()).toContain("Proxy ready");
151
+ expect(lastFrame()).toContain("MCP on stdio");
152
+ });
153
+ it("shows proxy mode when not full", () => {
154
+ const { lastFrame } = renderStartup({
155
+ localMode: false,
156
+ steps: [],
157
+ firstBoot: false,
158
+ ready: true,
159
+ proxyMode: "parse",
160
+ });
161
+ expect(lastFrame()).toContain("parse mode");
162
+ });
163
+ it("shows local mode label when proxyMode is local", () => {
164
+ const { lastFrame } = renderStartup({
165
+ localMode: false,
166
+ steps: [],
167
+ firstBoot: false,
168
+ ready: true,
169
+ proxyMode: "local",
170
+ });
171
+ expect(lastFrame()).toContain("local mode");
172
+ });
173
+ it("renders deep link when available", () => {
174
+ const { lastFrame } = renderStartup({
175
+ localMode: false,
176
+ steps: [],
177
+ firstBoot: false,
178
+ ready: true,
179
+ deepLink: "https://app.unerr.dev/r/repo_123?utm_source=cli_startup",
180
+ });
181
+ expect(lastFrame()).toContain("https://app.unerr.dev/r/repo_123");
182
+ });
183
+ it("does not render invitation when not ready", () => {
184
+ const { lastFrame } = renderStartup({
185
+ localMode: false,
186
+ steps: [],
187
+ firstBoot: false,
188
+ ready: false,
189
+ invitationEntity: "processPayment",
190
+ });
191
+ expect(lastFrame()).not.toContain("What depends on");
192
+ });
193
+ });
194
+ // ── Full Three-Act integration ─────────────────────────────────
195
+ describe("Full Three-Act integration", () => {
196
+ it("renders all three acts together", () => {
197
+ const { lastFrame } = renderStartup({
198
+ localMode: false,
199
+ steps: [
200
+ { label: "Authenticated", value: "Jaswanth's Org", status: "done" },
201
+ { label: "Repository", value: "unerr-server (main)", status: "done" },
202
+ {
203
+ label: "Graph loaded",
204
+ value: "2,341 entities · 1,892 edges",
205
+ status: "done",
206
+ },
207
+ {
208
+ label: "MCP ready",
209
+ value: "15 tools (all local)",
210
+ status: "done",
211
+ },
212
+ ],
213
+ health: {
214
+ grade: "C+",
215
+ totalEntities: 2341,
216
+ totalEdges: 1892,
217
+ totalRules: 12,
218
+ deadFunctionCount: 23,
219
+ highRiskEntities: [
220
+ {
221
+ name: "processPayment",
222
+ kind: "function",
223
+ file_path: "src/billing.ts",
224
+ fan_in: 14,
225
+ fan_out: 8,
226
+ },
227
+ ],
228
+ score: 62,
229
+ },
230
+ firstBoot: true,
231
+ ready: true,
232
+ invitationEntity: "processPayment",
233
+ deepLink: "https://app.unerr.dev/r/repo_123?utm_source=cli_startup",
234
+ });
235
+ const frame = lastFrame() ?? "";
236
+ // Act 1
237
+ expect(frame).toContain("unerr");
238
+ expect(frame).toContain("Authenticated");
239
+ expect(frame).toContain("Jaswanth's Org");
240
+ // Act 2
241
+ expect(frame).toContain("First Look");
242
+ expect(frame).toContain("C+");
243
+ expect(frame).toContain("23 dead functions");
244
+ // Act 3
245
+ expect(frame).toContain("What depends on processPayment?");
246
+ expect(frame).toContain("Proxy ready");
247
+ expect(frame).toContain("https://app.unerr.dev/r/repo_123");
248
+ });
249
+ });
250
+ });
251
+ // ── StartupRenderer unit tests ───────────────────────────────────
252
+ describe("StartupRenderer", () => {
253
+ let tmpDir;
254
+ let origCwd;
255
+ beforeEach(() => {
256
+ tmpDir = path.join(os.tmpdir(), `unerr-startup-test-${Date.now()}`);
257
+ fs.mkdirSync(path.join(tmpDir, ".unerr", "state"), { recursive: true });
258
+ origCwd = process.cwd();
259
+ process.chdir(tmpDir);
260
+ });
261
+ afterEach(() => {
262
+ process.chdir(origCwd);
263
+ fs.rmSync(tmpDir, { recursive: true, force: true });
264
+ });
265
+ it("tracks first_boot_shown flag", async () => {
266
+ const { StartupRenderer } = await import("../proxy/startup-renderer.js");
267
+ const renderer = new StartupRenderer();
268
+ // No graph_version.json → first boot
269
+ renderer.setHealth({
270
+ grade: "C+",
271
+ totalEntities: 100,
272
+ totalEdges: 50,
273
+ totalRules: 5,
274
+ deadFunctionCount: 10,
275
+ highRiskEntities: [],
276
+ score: 62,
277
+ }, "repo_test");
278
+ // Should have written first_boot_shown = true
279
+ const versionPath = path.join(tmpDir, ".unerr", "state", "graph_version.json");
280
+ const data = JSON.parse(fs.readFileSync(versionPath, "utf-8"));
281
+ expect(data.first_boot_shown).toBe(true);
282
+ });
283
+ it("detects subsequent boot after first_boot_shown is set", async () => {
284
+ const versionPath = path.join(tmpDir, ".unerr", "state", "graph_version.json");
285
+ fs.writeFileSync(versionPath, JSON.stringify({ first_boot_shown: true }));
286
+ const { StartupRenderer } = await import("../proxy/startup-renderer.js");
287
+ const renderer = new StartupRenderer();
288
+ renderer.setHealth({
289
+ grade: "A",
290
+ totalEntities: 100,
291
+ totalEdges: 50,
292
+ totalRules: 5,
293
+ deadFunctionCount: 0,
294
+ highRiskEntities: [],
295
+ score: 95,
296
+ }, "repo_test");
297
+ // firstBoot should be false since flag was already set
298
+ // We verify indirectly — the renderer won't overwrite the flag
299
+ const data = JSON.parse(fs.readFileSync(versionPath, "utf-8"));
300
+ expect(data.first_boot_shown).toBe(true);
301
+ });
302
+ });
@@ -0,0 +1,97 @@
1
+ import { existsSync, mkdirSync, readFileSync, rmSync } from "node:fs";
2
+ import os from "node:os";
3
+ import { join } from "node:path";
4
+ import { afterEach, beforeEach, describe, expect, it } from "vitest";
5
+ import { initFileLog, startupLog } from "../utils/startup-log.js";
6
+ describe("startup-log file logging", () => {
7
+ let tmpDir;
8
+ let logPath;
9
+ beforeEach(() => {
10
+ tmpDir = join(os.tmpdir(), `unerr-log-test-${Date.now()}`);
11
+ mkdirSync(tmpDir, { recursive: true });
12
+ initFileLog(tmpDir);
13
+ logPath = join(tmpDir, ".unerr", "logs", "unerr.jsonl");
14
+ });
15
+ afterEach(() => {
16
+ rmSync(tmpDir, { recursive: true, force: true });
17
+ });
18
+ it("creates .unerr/logs/unerr.jsonl on initFileLog", () => {
19
+ // initFileLog creates the directory; file created on first write
20
+ startupLog.step("test step");
21
+ expect(existsSync(logPath)).toBe(true);
22
+ });
23
+ it("step() writes a JSONL entry with level=step", () => {
24
+ startupLog.step("indexing files");
25
+ const lines = readFileSync(logPath, "utf-8").trim().split("\n");
26
+ const entry = JSON.parse(lines[lines.length - 1]);
27
+ expect(entry.level).toBe("step");
28
+ expect(entry.msg).toContain("indexing files");
29
+ expect(entry.ts).toBeDefined();
30
+ expect(entry.pid).toBe(process.pid);
31
+ });
32
+ it("done() includes ms metadata when provided", () => {
33
+ startupLog.done("graph loaded", 42);
34
+ const lines = readFileSync(logPath, "utf-8").trim().split("\n");
35
+ const entry = JSON.parse(lines[lines.length - 1]);
36
+ expect(entry.level).toBe("done");
37
+ expect(entry.ms).toBe(42);
38
+ });
39
+ it("metric() includes raw value and unit", () => {
40
+ startupLog.metric("entities", 1234, "nodes");
41
+ const lines = readFileSync(logPath, "utf-8").trim().split("\n");
42
+ const entry = JSON.parse(lines[lines.length - 1]);
43
+ expect(entry.level).toBe("metric");
44
+ expect(entry.value).toBe(1234);
45
+ expect(entry.unit).toBe("nodes");
46
+ });
47
+ it("warn() and error() write correct levels", () => {
48
+ startupLog.warn("something odd");
49
+ startupLog.error("something broke");
50
+ const lines = readFileSync(logPath, "utf-8").trim().split("\n");
51
+ const entries = lines.map((l) => JSON.parse(l));
52
+ const warn = entries.find((e) => e.level === "warn");
53
+ const err = entries.find((e) => e.level === "error");
54
+ expect(warn).toBeDefined();
55
+ expect(warn.msg).toContain("something odd");
56
+ expect(err).toBeDefined();
57
+ expect(err.msg).toContain("something broke");
58
+ });
59
+ it("graphLoaded() writes full stats object", () => {
60
+ startupLog.graphLoaded({
61
+ entities: 500,
62
+ edges: 1200,
63
+ files: 80,
64
+ communities: 5,
65
+ patterns: 12,
66
+ rules: 3,
67
+ ms: 150,
68
+ hottestFile: "src/main.ts",
69
+ hottestCount: 25,
70
+ });
71
+ const lines = readFileSync(logPath, "utf-8").trim().split("\n");
72
+ const entry = JSON.parse(lines[lines.length - 1]);
73
+ expect(entry.level).toBe("graph_loaded");
74
+ expect(entry.entities).toBe(500);
75
+ expect(entry.edges).toBe(1200);
76
+ expect(entry.files).toBe(80);
77
+ expect(entry.communities).toBe(5);
78
+ expect(entry.patterns).toBe(12);
79
+ expect(entry.rules).toBe(3);
80
+ expect(entry.ms).toBe(150);
81
+ expect(entry.hottestFile).toBe("src/main.ts");
82
+ expect(entry.hottestCount).toBe(25);
83
+ });
84
+ it("file entries have no ANSI escape codes", () => {
85
+ startupLog.step("test with colors");
86
+ const content = readFileSync(logPath, "utf-8");
87
+ expect(content).not.toContain("\x1b[");
88
+ });
89
+ it("ready() includes toolCount and mode", () => {
90
+ startupLog.ready(17, "local");
91
+ const lines = readFileSync(logPath, "utf-8").trim().split("\n");
92
+ const entry = JSON.parse(lines[lines.length - 1]);
93
+ expect(entry.level).toBe("ready");
94
+ expect(entry.toolCount).toBe(17);
95
+ expect(entry.mode).toBe("local");
96
+ });
97
+ });
@@ -0,0 +1,229 @@
1
+ /**
2
+ * Sprint 7.1: Git stash awareness tests.
3
+ */
4
+ import { mkdirSync, mkdtempSync, writeFileSync, } from "node:fs";
5
+ import { tmpdir } from "node:os";
6
+ import { join } from "node:path";
7
+ import { describe, expect, it } from "vitest";
8
+ import { StashManager } from "../tracking/stash-manager.js";
9
+ /** Minimal mock CozoGraphStore backed by a Map for drift overlay. */
10
+ function createMockGraph(driftEntities = []) {
11
+ const entities = new Map();
12
+ for (const e of driftEntities) {
13
+ entities.set(e.key, e);
14
+ }
15
+ return {
16
+ entities,
17
+ getAllDriftEntities: () => [...entities.values()],
18
+ getAllDriftEdges: () => [],
19
+ upsertDriftEntity: (entity) => {
20
+ entities.set(entity.key, entity);
21
+ },
22
+ upsertDriftEdge: () => { },
23
+ removeDriftEntity: (key) => {
24
+ entities.delete(key);
25
+ },
26
+ clearDriftOverlay: () => {
27
+ entities.clear();
28
+ },
29
+ };
30
+ }
31
+ function makeDrift(overrides) {
32
+ return {
33
+ kind: "function",
34
+ signature: "()",
35
+ body: "function test() {}",
36
+ file_path: "src/test.ts",
37
+ line_start: 1,
38
+ line_end: 3,
39
+ content_hash: "abc123",
40
+ drift_status: "modified",
41
+ intent_id: "",
42
+ modified_at: new Date().toISOString(),
43
+ origin: "human",
44
+ previous_body: "",
45
+ previous_signature: "",
46
+ ...overrides,
47
+ };
48
+ }
49
+ function setupProjectWithGit() {
50
+ const projectRoot = mkdtempSync(join(tmpdir(), "stash-"));
51
+ const unerrDir = join(projectRoot, ".unerr");
52
+ const gitDir = join(projectRoot, ".git");
53
+ mkdirSync(unerrDir, { recursive: true });
54
+ mkdirSync(join(gitDir, "refs"), { recursive: true });
55
+ mkdirSync(join(gitDir, "logs", "refs"), { recursive: true });
56
+ return { projectRoot, unerrDir, gitDir };
57
+ }
58
+ function writeStashRef(gitDir, ref) {
59
+ writeFileSync(join(gitDir, "refs", "stash"), `${ref}\n`, "utf-8");
60
+ }
61
+ function writeStashLog(gitDir, count) {
62
+ const lines = Array.from({ length: count }, (_, i) => `0000000 abcdef${i} Author <a@b.com> ${Date.now()} +0000\tstash@{${i}}: WIP`);
63
+ writeFileSync(join(gitDir, "logs", "refs", "stash"), `${lines.join("\n")}\n`, "utf-8");
64
+ }
65
+ const MOCK_FILE_HASHES = {
66
+ files: {
67
+ "src/test.ts": {
68
+ contentSha: "sha256abc",
69
+ headSha: "head123",
70
+ processedAt: new Date().toISOString(),
71
+ },
72
+ },
73
+ };
74
+ describe("StashManager", () => {
75
+ it("detects no change when stash ref unchanged", () => {
76
+ const { projectRoot, unerrDir, gitDir } = setupProjectWithGit();
77
+ writeStashRef(gitDir, "abc123def456");
78
+ writeStashLog(gitDir, 1);
79
+ const manager = new StashManager(unerrDir, projectRoot);
80
+ expect(manager.detectStashChange()).toBeNull();
81
+ });
82
+ it("detects stash push when count increases", () => {
83
+ const { projectRoot, unerrDir, gitDir } = setupProjectWithGit();
84
+ writeStashRef(gitDir, "ref1");
85
+ writeStashLog(gitDir, 1);
86
+ const manager = new StashManager(unerrDir, projectRoot);
87
+ // Simulate stash push
88
+ writeStashRef(gitDir, "ref2");
89
+ writeStashLog(gitDir, 2);
90
+ expect(manager.detectStashChange()).toBe("push");
91
+ });
92
+ it("detects stash pop when count decreases", () => {
93
+ const { projectRoot, unerrDir, gitDir } = setupProjectWithGit();
94
+ writeStashRef(gitDir, "ref1");
95
+ writeStashLog(gitDir, 2);
96
+ const manager = new StashManager(unerrDir, projectRoot);
97
+ // Simulate stash pop
98
+ writeStashRef(gitDir, "ref0");
99
+ writeStashLog(gitDir, 1);
100
+ expect(manager.detectStashChange()).toBe("pop");
101
+ });
102
+ it("saves and restores a stash snapshot", async () => {
103
+ const { projectRoot, unerrDir, gitDir } = setupProjectWithGit();
104
+ writeStashRef(gitDir, "abc123def456abc123def456abc123def456abc1");
105
+ writeStashLog(gitDir, 1);
106
+ const entities = [
107
+ makeDrift({ key: "k1", name: "fn1", drift_status: "modified" }),
108
+ makeDrift({ key: "k2", name: "fn2", drift_status: "added" }),
109
+ ];
110
+ const graph = createMockGraph(entities);
111
+ const manager = new StashManager(unerrDir, projectRoot);
112
+ // Save snapshot
113
+ const snapshotId = await manager.saveSnapshot(graph, MOCK_FILE_HASHES);
114
+ expect(snapshotId).toBe("abc123def456");
115
+ // Clear overlay to simulate stash applying
116
+ graph.entities.clear();
117
+ expect(graph.getAllDriftEntities()).toHaveLength(0);
118
+ // Restore snapshot
119
+ const restored = await manager.restoreSnapshot(graph);
120
+ expect(restored).toBe(2);
121
+ expect(graph.entities.size).toBe(2);
122
+ expect(graph.entities.get("k1")?.name).toBe("fn1");
123
+ expect(graph.entities.get("k2")?.name).toBe("fn2");
124
+ });
125
+ it("returns null when saving with no drift entities", async () => {
126
+ const { projectRoot, unerrDir, gitDir } = setupProjectWithGit();
127
+ writeStashRef(gitDir, "abc123def456");
128
+ const graph = createMockGraph([]);
129
+ const manager = new StashManager(unerrDir, projectRoot);
130
+ const snapshotId = await manager.saveSnapshot(graph, MOCK_FILE_HASHES);
131
+ expect(snapshotId).toBeNull();
132
+ });
133
+ it("returns null when saving with no stash ref", async () => {
134
+ const { projectRoot, unerrDir } = setupProjectWithGit();
135
+ const graph = createMockGraph([makeDrift({ key: "k1", name: "fn1" })]);
136
+ const manager = new StashManager(unerrDir, projectRoot);
137
+ const snapshotId = await manager.saveSnapshot(graph, MOCK_FILE_HASHES);
138
+ expect(snapshotId).toBeNull();
139
+ });
140
+ it("restores 0 when no snapshots exist", async () => {
141
+ const { projectRoot, unerrDir } = setupProjectWithGit();
142
+ const graph = createMockGraph([]);
143
+ const manager = new StashManager(unerrDir, projectRoot);
144
+ expect(await manager.restoreSnapshot(graph)).toBe(0);
145
+ });
146
+ it("drops a specific snapshot", async () => {
147
+ const { projectRoot, unerrDir, gitDir } = setupProjectWithGit();
148
+ const fullRef = "abc123def456abc123def456abc123def456abc1";
149
+ writeStashRef(gitDir, fullRef);
150
+ const graph = createMockGraph([makeDrift({ key: "k1", name: "fn1" })]);
151
+ const manager = new StashManager(unerrDir, projectRoot);
152
+ await manager.saveSnapshot(graph, MOCK_FILE_HASHES);
153
+ expect(manager.listSnapshots()).toHaveLength(1);
154
+ expect(manager.dropSnapshot(fullRef)).toBe(true);
155
+ expect(manager.listSnapshots()).toHaveLength(0);
156
+ });
157
+ it("enforces LRU cap of 10 snapshots", async () => {
158
+ const { projectRoot, unerrDir, gitDir } = setupProjectWithGit();
159
+ const graph = createMockGraph([makeDrift({ key: "k1", name: "fn1" })]);
160
+ const manager = new StashManager(unerrDir, projectRoot);
161
+ // Create 12 snapshots
162
+ for (let i = 0; i < 12; i++) {
163
+ const ref = `ref${String(i).padStart(12, "0")}aaaaaaaaaaaaaaaa`;
164
+ writeStashRef(gitDir, ref);
165
+ await manager.saveSnapshot(graph, MOCK_FILE_HASHES);
166
+ }
167
+ // Should be capped at 10
168
+ expect(manager.listSnapshots().length).toBeLessThanOrEqual(10);
169
+ });
170
+ it("retrieves file hash state from snapshot", async () => {
171
+ const { projectRoot, unerrDir, gitDir } = setupProjectWithGit();
172
+ writeStashRef(gitDir, "abc123def456abc1");
173
+ const graph = createMockGraph([makeDrift({ key: "k1", name: "fn1" })]);
174
+ const manager = new StashManager(unerrDir, projectRoot);
175
+ await manager.saveSnapshot(graph, MOCK_FILE_HASHES);
176
+ const hashes = manager.getSnapshotFileHashes();
177
+ expect(hashes).not.toBeNull();
178
+ expect(hashes?.files["src/test.ts"]?.contentSha).toBe("sha256abc");
179
+ });
180
+ it("cleans up snapshot after restore", async () => {
181
+ const { projectRoot, unerrDir, gitDir } = setupProjectWithGit();
182
+ writeStashRef(gitDir, "abc123def456abc1");
183
+ const graph = createMockGraph([makeDrift({ key: "k1", name: "fn1" })]);
184
+ const manager = new StashManager(unerrDir, projectRoot);
185
+ await manager.saveSnapshot(graph, MOCK_FILE_HASHES);
186
+ expect(manager.listSnapshots()).toHaveLength(1);
187
+ graph.entities.clear();
188
+ await manager.restoreSnapshot(graph);
189
+ // Snapshot should be cleaned up after restore
190
+ expect(manager.listSnapshots()).toHaveLength(0);
191
+ });
192
+ it("handles stash push → pop cycle preserving overlay", async () => {
193
+ const { projectRoot, unerrDir, gitDir } = setupProjectWithGit();
194
+ const originalEntities = [
195
+ makeDrift({
196
+ key: "k1",
197
+ name: "handler",
198
+ drift_status: "modified",
199
+ body: "function handler() { return 42; }",
200
+ previous_body: "function handler() { return 0; }",
201
+ }),
202
+ ];
203
+ const graph = createMockGraph(originalEntities);
204
+ // Initial state: 1 stash
205
+ writeStashRef(gitDir, "stashref1aaa");
206
+ writeStashLog(gitDir, 0);
207
+ const manager = new StashManager(unerrDir, projectRoot);
208
+ // Simulate stash push
209
+ writeStashRef(gitDir, "stashref2bbb");
210
+ writeStashLog(gitDir, 1);
211
+ expect(manager.detectStashChange()).toBe("push");
212
+ await manager.saveSnapshot(graph, MOCK_FILE_HASHES);
213
+ // Clear overlay (simulating git stash restoring working tree)
214
+ graph.entities.clear();
215
+ expect(graph.getAllDriftEntities()).toHaveLength(0);
216
+ // Simulate stash pop
217
+ writeStashRef(gitDir, "stashref1aaa");
218
+ writeStashLog(gitDir, 0);
219
+ expect(manager.detectStashChange()).toBe("pop");
220
+ const restored = await manager.restoreSnapshot(graph);
221
+ expect(restored).toBe(1);
222
+ const entity = graph.entities.get("k1");
223
+ expect(entity).toBeDefined();
224
+ expect(entity?.name).toBe("handler");
225
+ expect(entity?.drift_status).toBe("modified");
226
+ expect(entity?.body).toBe("function handler() { return 42; }");
227
+ expect(entity?.previous_body).toBe("function handler() { return 0; }");
228
+ });
229
+ });