@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.
- package/README.md +215 -35
- package/dist/__tests__/architecture-guard.test.js +122 -0
- package/dist/__tests__/arg-validator.test.js +205 -0
- package/dist/__tests__/ast-extractor.test.js +203 -0
- package/dist/__tests__/auto-bootstrap.test.js +280 -0
- package/dist/__tests__/background-indexer.test.js +228 -0
- package/dist/__tests__/blast-radius-engine.test.js +200 -0
- package/dist/__tests__/bridge-isolation.test.js +37 -0
- package/dist/__tests__/budget-enforcer.test.js +53 -0
- package/dist/__tests__/cfg-test-detection-perf.test.js +82 -0
- package/dist/__tests__/change-narrative.test.js +190 -0
- package/dist/__tests__/check-commit.test.js +258 -0
- package/dist/__tests__/checksum.test.js +34 -0
- package/dist/__tests__/commit-watcher.test.js +154 -0
- package/dist/__tests__/community-detection.test.js +179 -0
- package/dist/__tests__/community-tools.test.js +299 -0
- package/dist/__tests__/components.test.js +449 -0
- package/dist/__tests__/compression-log.test.js +174 -0
- package/dist/__tests__/compression-quality-monitor.test.js +40 -0
- package/dist/__tests__/config-healer.test.js +165 -0
- package/dist/__tests__/context-ledger.test.js +58 -0
- package/dist/__tests__/convention-detector.test.js +99 -0
- package/dist/__tests__/convention-learner.test.js +86 -0
- package/dist/__tests__/correction-detector.test.js +330 -0
- package/dist/__tests__/daemon-autostart-install.test.js +283 -0
- package/dist/__tests__/daemon-bridge.test.js +222 -0
- package/dist/__tests__/daemon-dashboard.test.js +202 -0
- package/dist/__tests__/daemon-registry.test.js +240 -0
- package/dist/__tests__/daemon-supervisor.test.js +318 -0
- package/dist/__tests__/daemon-version-check.test.js +275 -0
- package/dist/__tests__/decision-point-detector.test.js +98 -0
- package/dist/__tests__/deep-link.test.js +143 -0
- package/dist/__tests__/disallowed-tools.test.js +115 -0
- package/dist/__tests__/drift-tracker.test.js +582 -0
- package/dist/__tests__/durability-scorer.test.js +152 -0
- package/dist/__tests__/efficiency-tracker.test.js +65 -0
- package/dist/__tests__/enrich.test.js +144 -0
- package/dist/__tests__/entity-rewind.test.js +248 -0
- package/dist/__tests__/ephemeral.test.js +111 -0
- package/dist/__tests__/exploration-cost.test.js +93 -0
- package/dist/__tests__/fact-generator.test.js +197 -0
- package/dist/__tests__/file-l0-graph.test.js +244 -0
- package/dist/__tests__/file-logger.test.js +82 -0
- package/dist/__tests__/file-outline.test.js +141 -0
- package/dist/__tests__/file-read-protocol.test.js +188 -0
- package/dist/__tests__/format-encoder.test.js +233 -0
- package/dist/__tests__/git-attribution.test.js +259 -0
- package/dist/__tests__/graph-temporal-joiner.test.js +219 -0
- package/dist/__tests__/health-grade-enhanced.test.js +138 -0
- package/dist/__tests__/health-map-data.test.js +173 -0
- package/dist/__tests__/helpers/mcp-harness.js +45 -0
- package/dist/__tests__/helpers/mcp-harness.test.js +68 -0
- package/dist/__tests__/hook-dedup.test.js +112 -0
- package/dist/__tests__/hook-runner.test.js +253 -0
- package/dist/__tests__/indexer-cfg.test.js +185 -0
- package/dist/__tests__/indexer-cross-file.test.js +172 -0
- package/dist/__tests__/indexer-extraction.test.js +245 -0
- package/dist/__tests__/indexer-incremental.test.js +232 -0
- package/dist/__tests__/indexer-language-expansion.test.js +165 -0
- package/dist/__tests__/init-push.test.js +131 -0
- package/dist/__tests__/instruction-writer.test.js +179 -0
- package/dist/__tests__/intelligence-integration.test.js +217 -0
- package/dist/__tests__/intent-correlator.test.js +175 -0
- package/dist/__tests__/intent-detector.test.js +235 -0
- package/dist/__tests__/intent-encoder.test.js +167 -0
- package/dist/__tests__/java-build-tool-detection.test.js +174 -0
- package/dist/__tests__/layer3-sprint-q.test.js +160 -0
- package/dist/__tests__/layer3-sprint-r.test.js +91 -0
- package/dist/__tests__/layer3-sprint-s.test.js +183 -0
- package/dist/__tests__/layer3-sprint-t.test.js +201 -0
- package/dist/__tests__/layer3-sprint-u.test.js +174 -0
- package/dist/__tests__/layer4-sprint-ba2.test.js +354 -0
- package/dist/__tests__/layer4-sprint-ba4.test.js +84 -0
- package/dist/__tests__/layer4-sprint-vs.test.js +105 -0
- package/dist/__tests__/ledger-chains.test.js +162 -0
- package/dist/__tests__/lifecycle-machine.test.js +226 -0
- package/dist/__tests__/local-chat-provider.test.js +170 -0
- package/dist/__tests__/local-convention-detector.test.js +308 -0
- package/dist/__tests__/local-embeddings.test.js +422 -0
- package/dist/__tests__/local-graph.test.js +540 -0
- package/dist/__tests__/local-indexer.test.js +228 -0
- package/dist/__tests__/local-intelligence-l3.test.js +332 -0
- package/dist/__tests__/local-llm.test.js +253 -0
- package/dist/__tests__/local-mode-offline.test.js +187 -0
- package/dist/__tests__/local-mode-stats.test.js +273 -0
- package/dist/__tests__/local-mode-tui.test.js +343 -0
- package/dist/__tests__/local-parse.test.js +199 -0
- package/dist/__tests__/log-tailer.test.js +208 -0
- package/dist/__tests__/loop-breaker.test.js +276 -0
- package/dist/__tests__/loop-miner.test.js +226 -0
- package/dist/__tests__/mcp-config.test.js +126 -0
- package/dist/__tests__/mcp-content-json.test.js +10 -0
- package/dist/__tests__/mcp-envelope.test.js +124 -0
- package/dist/__tests__/metrics-store.test.js +223 -0
- package/dist/__tests__/native-watcher.test.js +191 -0
- package/dist/__tests__/navigation-hooks-agent-aware.test.js +145 -0
- package/dist/__tests__/negative-knowledge.test.js +116 -0
- package/dist/__tests__/network-boundary.test.js +190 -0
- package/dist/__tests__/network-firewall.test.js +112 -0
- package/dist/__tests__/nudge-invariants.test.js +160 -0
- package/dist/__tests__/nudge-v2.test.js +225 -0
- package/dist/__tests__/offline-rewind.test.js +251 -0
- package/dist/__tests__/open-threads.test.js +89 -0
- package/dist/__tests__/output-compressor.test.js +93 -0
- package/dist/__tests__/pending-violations.test.js +112 -0
- package/dist/__tests__/persistence-effectiveness.test.js +143 -0
- package/dist/__tests__/provider-factory.test.js +42 -0
- package/dist/__tests__/providers.test.js +24 -0
- package/dist/__tests__/proxy.test.js +314 -0
- package/dist/__tests__/query-router.test.js +1018 -0
- package/dist/__tests__/reasoning-quality-route.test.js +138 -0
- package/dist/__tests__/redactor.test.js +120 -0
- package/dist/__tests__/resource-monitor.test.js +57 -0
- package/dist/__tests__/response-envelope.test.js +100 -0
- package/dist/__tests__/risk-classifier.test.js +101 -0
- package/dist/__tests__/risk-signal-scope.test.js +75 -0
- package/dist/__tests__/rule-evaluator.test.js +280 -0
- package/dist/__tests__/scip-decoder.test.js +49 -0
- package/dist/__tests__/scip-downloader.test.js +201 -0
- package/dist/__tests__/scip-merger.test.js +103 -0
- package/dist/__tests__/search-index.test.js +422 -0
- package/dist/__tests__/semantic-enrichment.test.js +360 -0
- package/dist/__tests__/session-brief-builder.test.js +187 -0
- package/dist/__tests__/session-context.test.js +221 -0
- package/dist/__tests__/session-continuity.test.js +144 -0
- package/dist/__tests__/session-dedup.test.js +74 -0
- package/dist/__tests__/session-event-wiring.test.js +206 -0
- package/dist/__tests__/session-events.test.js +149 -0
- package/dist/__tests__/session-legend.test.js +20 -0
- package/dist/__tests__/session-persistence.test.js +131 -0
- package/dist/__tests__/session-resume-block.test.js +107 -0
- package/dist/__tests__/session-resume.test.js +97 -0
- package/dist/__tests__/session-summary-writer.test.js +134 -0
- package/dist/__tests__/shadow-ledger.test.js +203 -0
- package/dist/__tests__/shell-classifier.test.js +151 -0
- package/dist/__tests__/shell-compression-floor.test.js +189 -0
- package/dist/__tests__/shell-compression-v2.test.js +339 -0
- package/dist/__tests__/shell-compressor.test.js +35 -0
- package/dist/__tests__/shell-hooks.test.js +128 -0
- package/dist/__tests__/shell-strategies.test.js +644 -0
- package/dist/__tests__/shell-tee.test.js +133 -0
- package/dist/__tests__/signal-dedup.test.js +158 -0
- package/dist/__tests__/signal-reinforcer.test.js +77 -0
- package/dist/__tests__/signal-scorer.test.js +251 -0
- package/dist/__tests__/signal-show-store.test.js +108 -0
- package/dist/__tests__/smart-truncate.test.js +215 -0
- package/dist/__tests__/snapshot-v2.test.js +113 -0
- package/dist/__tests__/sprint-l1-local-mode.test.js +130 -0
- package/dist/__tests__/sprint-l10-boot.test.js +220 -0
- package/dist/__tests__/sprint-l9-offline-commands.test.js +189 -0
- package/dist/__tests__/sprint-q-persistent-context.test.js +198 -0
- package/dist/__tests__/sprint-s1-wiring.test.js +215 -0
- package/dist/__tests__/sprint-s2-wiring.test.js +256 -0
- package/dist/__tests__/sprint-s3-wiring.test.js +195 -0
- package/dist/__tests__/sprint-s4-wiring.test.js +213 -0
- package/dist/__tests__/sprint-s6-hooks.test.js +222 -0
- package/dist/__tests__/sprint-s7-persistent.test.js +263 -0
- package/dist/__tests__/sprint-s8-value.test.js +167 -0
- package/dist/__tests__/sprint-s9-behavioral.test.js +179 -0
- package/dist/__tests__/sprint3-intelligence.test.js +297 -0
- package/dist/__tests__/sprint5-mcp-server.test.js +136 -0
- package/dist/__tests__/startup-display.test.js +302 -0
- package/dist/__tests__/startup-log-file.test.js +97 -0
- package/dist/__tests__/stash-manager.test.js +229 -0
- package/dist/__tests__/state-detector.test.js +92 -0
- package/dist/__tests__/status-dashboard.test.js +142 -0
- package/dist/__tests__/temporal-facts.test.js +292 -0
- package/dist/__tests__/temporal-routes.test.js +142 -0
- package/dist/__tests__/test-detector.test.js +174 -0
- package/dist/__tests__/theme.test.js +72 -0
- package/dist/__tests__/timeline-agents.test.js +122 -0
- package/dist/__tests__/timeline-bootstrap.test.js +176 -0
- package/dist/__tests__/timeline-filters.test.js +193 -0
- package/dist/__tests__/timeline-markers.test.js +151 -0
- package/dist/__tests__/timeline-routes.test.js +156 -0
- package/dist/__tests__/timeline-store.test.js +171 -0
- package/dist/__tests__/token-counter.test.js +86 -0
- package/dist/__tests__/token-estimator.test.js +96 -0
- package/dist/__tests__/token-flow-api.test.js +239 -0
- package/dist/__tests__/token-flow-instrumentation.test.js +437 -0
- package/dist/__tests__/token-flow-persistence.test.js +356 -0
- package/dist/__tests__/token-flow-routes.test.js +199 -0
- package/dist/__tests__/token-flow.test.js +695 -0
- package/dist/__tests__/tool-clusters.test.js +177 -0
- package/dist/__tests__/transport-mux.test.js +283 -0
- package/dist/__tests__/turn-segmenter.test.js +166 -0
- package/dist/__tests__/uninstall.test.js +141 -0
- package/dist/__tests__/warm-start-policy.test.js +271 -0
- package/dist/__tests__/wire-cap-nudge.test.js +77 -0
- package/dist/__tests__/worker-pool.test.js +101 -0
- package/dist/behaviors/agent-llm-bridge.js +166 -0
- package/dist/behaviors/architecture-guard.js +256 -0
- package/dist/behaviors/auto-doc.js +247 -0
- package/dist/behaviors/cascade-guard.js +289 -0
- package/dist/behaviors/change-narrative.js +270 -0
- package/dist/behaviors/convention-drift.js +290 -0
- package/dist/behaviors/framework.js +235 -0
- package/dist/behaviors/guard-formatter.js +44 -0
- package/dist/behaviors/incomplete-work.js +270 -0
- package/dist/behaviors/loop-breaker.js +300 -0
- package/dist/behaviors/session-continuity.js +208 -0
- package/dist/cli.js +6446 -2227
- package/dist/commands/branches.js +97 -0
- package/dist/commands/check-commit.js +225 -0
- package/dist/commands/compress-output.js +64 -0
- package/dist/commands/config-verify.js +243 -0
- package/dist/commands/daemon.js +905 -0
- package/dist/commands/dashboard.js +52 -0
- package/dist/commands/debug.js +200 -0
- package/dist/commands/enrich.js +184 -0
- package/dist/commands/exec.js +233 -0
- package/dist/commands/gain.js +156 -0
- package/dist/commands/hook.js +88 -0
- package/dist/commands/index.js +88 -0
- package/dist/commands/init.js +74 -0
- package/dist/commands/install.js +505 -0
- package/dist/commands/learn.js +116 -0
- package/dist/commands/manifest.js +193 -0
- package/dist/commands/rewind.js +103 -0
- package/dist/commands/serve.js +19 -0
- package/dist/commands/setup-wizard.js +414 -0
- package/dist/commands/skills.js +64 -0
- package/dist/commands/stats.js +20 -0
- package/dist/commands/status.js +654 -0
- package/dist/commands/timeline.js +139 -0
- package/dist/commands/uninstall.js +230 -0
- package/dist/components/App.js +109 -0
- package/dist/components/Banner.js +12 -0
- package/dist/components/ConfirmPrompt.js +25 -0
- package/dist/components/DriftSummary.js +23 -0
- package/dist/components/GradeBadge.js +15 -0
- package/dist/components/HealthCard.js +18 -0
- package/dist/components/InkSpinner.js +22 -0
- package/dist/components/InputBox.js +17 -0
- package/dist/components/KeyValue.js +13 -0
- package/dist/components/MessageList.js +14 -0
- package/dist/components/ProgressBar.js +26 -0
- package/dist/components/Section.js +16 -0
- package/dist/components/SessionSummaryCard.js +73 -0
- package/dist/components/StartupDisplay.js +24 -0
- package/dist/components/StatusDashboard.js +57 -0
- package/dist/components/StatusLine.js +8 -0
- package/dist/components/StepLine.js +22 -0
- package/dist/components/Theme.js +20 -0
- package/dist/components/ToolProgress.js +8 -0
- package/dist/components/ViolationList.js +21 -0
- package/dist/components/render.js +13 -0
- package/dist/config/agent-registry.js +237 -0
- package/dist/config/claude-settings-hooks.js +304 -0
- package/dist/config/hook-installer.js +65 -0
- package/dist/config/instruction-writer.js +388 -0
- package/dist/config/mcp-config-writer.js +266 -0
- package/dist/config/settings.js +174 -0
- package/dist/config/tool-detector.js +42 -0
- package/dist/config/value-surfacing.js +119 -0
- package/dist/core/context-assembly.js +108 -0
- package/dist/core/conversation.js +33 -0
- package/dist/core/local-chat-provider.js +475 -0
- package/dist/core/provider-factory.js +55 -0
- package/dist/core/providers.js +90 -0
- package/dist/core/query-engine.js +174 -0
- package/dist/daemon/api.js +312 -0
- package/dist/daemon/autostart.js +119 -0
- package/dist/daemon/bootstrap.js +39 -0
- package/dist/daemon/client.js +164 -0
- package/dist/daemon/detect-ci.js +81 -0
- package/dist/daemon/platform-linux.js +146 -0
- package/dist/daemon/platform-macos.js +134 -0
- package/dist/daemon/platform-windows.js +116 -0
- package/dist/daemon/process-manager.js +299 -0
- package/dist/daemon/protocol.js +23 -0
- package/dist/daemon/registry.js +270 -0
- package/dist/daemon/settings-schema.js +72 -0
- package/dist/daemon/system-health.js +134 -0
- package/dist/daemon/version-checker.js +262 -0
- package/dist/daemon/warm-start.js +223 -0
- package/dist/entrypoints/cli.js +1043 -0
- package/dist/entrypoints/daemon.js +380 -0
- package/dist/entrypoints/repl.js +147 -0
- package/dist/hooks/adapters/claude-code.js +90 -0
- package/dist/hooks/adapters/cline.js +100 -0
- package/dist/hooks/adapters/cursor.js +98 -0
- package/dist/hooks/hook-dedup.js +79 -0
- package/dist/hooks/hook-runner.js +113 -0
- package/dist/hooks/navigation-hooks.js +175 -0
- package/dist/hooks/prompt-hooks.js +63 -0
- package/dist/hooks/shell-hooks.js +47 -0
- package/dist/ignore.js +111 -0
- package/dist/intelligence/approach-suggester.js +61 -0
- package/dist/intelligence/ast-extractor.js +2615 -0
- package/dist/intelligence/ast-worker.js +34 -0
- package/dist/intelligence/background-indexer.js +121 -0
- package/dist/intelligence/blast-radius.js +200 -0
- package/dist/intelligence/community-detection.js +691 -0
- package/dist/intelligence/community-detector.js +184 -0
- package/dist/intelligence/computation-scheduler.js +75 -0
- package/dist/intelligence/confidence-propagation.js +47 -0
- package/dist/intelligence/convention-detector.js +242 -0
- package/dist/intelligence/convention-learner.js +205 -0
- package/dist/intelligence/convention-matcher.js +205 -0
- package/dist/intelligence/cozo-schema.js +376 -0
- package/dist/intelligence/decision-point-detector.js +90 -0
- package/dist/intelligence/deep-dive-tools.js +586 -0
- package/dist/intelligence/durability-scorer.js +84 -0
- package/dist/intelligence/exploration-cost.js +204 -0
- package/dist/intelligence/exploration-pattern-tracker.js +61 -0
- package/dist/intelligence/fact-generator.js +322 -0
- package/dist/intelligence/facts-schema.js +90 -0
- package/dist/intelligence/file-intelligence.js +59 -0
- package/dist/intelligence/graph-holder.js +220 -0
- package/dist/intelligence/graph-temporal-joiner.js +238 -0
- package/dist/intelligence/health-grade.js +423 -0
- package/dist/intelligence/health-grader.js +200 -0
- package/dist/intelligence/health-map-data.js +259 -0
- package/dist/intelligence/import-symbols.js +136 -0
- package/dist/intelligence/incremental-indexer.js +658 -0
- package/dist/intelligence/indexer/centrality.js +62 -0
- package/dist/intelligence/indexer/cfg-context.js +95 -0
- package/dist/intelligence/indexer/confidence.js +34 -0
- package/dist/intelligence/indexer/cross-file-resolver.js +104 -0
- package/dist/intelligence/indexer/edge-repair.js +89 -0
- package/dist/intelligence/indexer/entity-key.js +17 -0
- package/dist/intelligence/indexer/export-map.js +132 -0
- package/dist/intelligence/indexer/git-cochange.js +128 -0
- package/dist/intelligence/indexer/graph-patch.js +147 -0
- package/dist/intelligence/indexer/incremental.js +78 -0
- package/dist/intelligence/indexer/ingest.js +160 -0
- package/dist/intelligence/indexer/language-detect.js +226 -0
- package/dist/intelligence/indexer/metadata.js +63 -0
- package/dist/intelligence/indexer/mutation-tracker.js +79 -0
- package/dist/intelligence/indexer/orchestrator.js +155 -0
- package/dist/intelligence/indexer/plugin-interface.js +31 -0
- package/dist/intelligence/indexer/plugins/csharp.js +440 -0
- package/dist/intelligence/indexer/plugins/go.js +335 -0
- package/dist/intelligence/indexer/plugins/java.js +370 -0
- package/dist/intelligence/indexer/plugins/python.js +358 -0
- package/dist/intelligence/indexer/plugins/regex-fallback.js +82 -0
- package/dist/intelligence/indexer/plugins/ruby.js +290 -0
- package/dist/intelligence/indexer/plugins/rust.js +484 -0
- package/dist/intelligence/indexer/plugins/tier2-generic.js +310 -0
- package/dist/intelligence/indexer/plugins/typescript.js +456 -0
- package/dist/intelligence/indexer/resource-monitor.js +93 -0
- package/dist/intelligence/indexer/scip/decoder.js +253 -0
- package/dist/intelligence/indexer/scip/detector.js +232 -0
- package/dist/intelligence/indexer/scip/downloader.js +427 -0
- package/dist/intelligence/indexer/scip/fallback.js +34 -0
- package/dist/intelligence/indexer/scip/merger.js +109 -0
- package/dist/intelligence/indexer/scip/orchestrator.js +433 -0
- package/dist/intelligence/indexer/scip/runner.js +98 -0
- package/dist/intelligence/indexer/snapshot.js +66 -0
- package/dist/intelligence/indexer/test-detector.js +196 -0
- package/dist/intelligence/indexer/watch-integration.js +61 -0
- package/dist/intelligence/indexer/worker.js +85 -0
- package/dist/intelligence/local-convention-detector.js +437 -0
- package/dist/intelligence/local-embeddings.js +190 -0
- package/dist/intelligence/local-graph.js +1946 -0
- package/dist/intelligence/local-indexer.js +1575 -0
- package/dist/intelligence/local-llm.js +163 -0
- package/dist/intelligence/local-rule-generator.js +154 -0
- package/dist/intelligence/local-snapshot.js +213 -0
- package/dist/intelligence/negative-knowledge.js +103 -0
- package/dist/intelligence/persistent-db.js +85 -0
- package/dist/intelligence/query-router.js +2556 -0
- package/dist/intelligence/risk-classifier.js +116 -0
- package/dist/intelligence/rule-evaluator.js +380 -0
- package/dist/intelligence/rule-generator.js +49 -0
- package/dist/intelligence/search-index.js +173 -0
- package/dist/intelligence/semantic/docstring-extractor.js +67 -0
- package/dist/intelligence/semantic/embedding-store.js +52 -0
- package/dist/intelligence/semantic/enrichment-orchestrator.js +48 -0
- package/dist/intelligence/semantic/git-message-miner.js +114 -0
- package/dist/intelligence/semantic/identifier-tokenizer.js +51 -0
- package/dist/intelligence/semantic/node2vec-embeddings.js +71 -0
- package/dist/intelligence/semantic/node2vec-walks.js +103 -0
- package/dist/intelligence/semantic/path-domain-inference.js +112 -0
- package/dist/intelligence/semantic/similarity-engine.js +60 -0
- package/dist/intelligence/semantic/tfidf-vectors.js +88 -0
- package/dist/intelligence/session-brief-builder.js +159 -0
- package/dist/intelligence/session-context.js +221 -0
- package/dist/intelligence/session-health-monitor.js +211 -0
- package/dist/intelligence/session-narrative.js +197 -0
- package/dist/intelligence/session-pattern-analyzer.js +218 -0
- package/dist/intelligence/signal-scorer.js +390 -0
- package/dist/intelligence/signal-show-store.js +182 -0
- package/dist/intelligence/smart-truncate.js +158 -0
- package/dist/intelligence/subgraph-cache.js +88 -0
- package/dist/intelligence/temporal-facts.js +494 -0
- package/dist/intelligence/token-estimator.js +100 -0
- package/dist/intelligence/tool-injector.js +87 -0
- package/dist/intelligence/tree-sitter-loader.js +71 -0
- package/dist/intelligence/worker-pool.js +116 -0
- package/dist/proxy/arg-validator.js +79 -0
- package/dist/proxy/auto-bootstrap.js +167 -0
- package/dist/proxy/bridge.js +147 -0
- package/dist/proxy/budget-enforcer.js +70 -0
- package/dist/proxy/compression-quality-monitor.js +160 -0
- package/dist/proxy/compression-stats.js +51 -0
- package/dist/proxy/context-rot-detector.js +137 -0
- package/dist/proxy/drift-detector.js +139 -0
- package/dist/proxy/efficiency-tracker.js +79 -0
- package/dist/proxy/fact-ranking.js +154 -0
- package/dist/proxy/format-encoder.js +266 -0
- package/dist/proxy/http-transport.js +90 -0
- package/dist/proxy/lifecycle-actor.js +55 -0
- package/dist/proxy/lifecycle-machine.js +187 -0
- package/dist/proxy/log-tailer.js +265 -0
- package/dist/proxy/model-pricing.js +98 -0
- package/dist/proxy/network-firewall.js +141 -0
- package/dist/proxy/nudge-state.js +93 -0
- package/dist/proxy/output-compressor.js +185 -0
- package/dist/proxy/pid-lock.js +291 -0
- package/dist/proxy/proxy-context.js +11 -0
- package/dist/proxy/proxy.js +2633 -0
- package/dist/proxy/response-enrichment.js +32 -0
- package/dist/proxy/response-envelope.js +313 -0
- package/dist/proxy/session-dedup.js +82 -0
- package/dist/proxy/session-legend.js +30 -0
- package/dist/proxy/session-persistence.js +210 -0
- package/dist/proxy/session-resume.js +94 -0
- package/dist/proxy/session-stats.js +513 -0
- package/dist/proxy/shell-classifier.js +1346 -0
- package/dist/proxy/shell-compression-log.js +93 -0
- package/dist/proxy/shell-compressor.js +390 -0
- package/dist/proxy/shell-graph-boost.js +202 -0
- package/dist/proxy/shell-monitor-map.js +18 -0
- package/dist/proxy/shell-stats.js +54 -0
- package/dist/proxy/shell-strategies/cloud.js +215 -0
- package/dist/proxy/shell-strategies/diff.js +159 -0
- package/dist/proxy/shell-strategies/error-diagnostic.js +796 -0
- package/dist/proxy/shell-strategies/filter-dsl.js +358 -0
- package/dist/proxy/shell-strategies/git-status.js +177 -0
- package/dist/proxy/shell-strategies/key-value.js +193 -0
- package/dist/proxy/shell-strategies/log-text.js +154 -0
- package/dist/proxy/shell-strategies/omni.js +188 -0
- package/dist/proxy/shell-strategies/progress.js +55 -0
- package/dist/proxy/shell-strategies/redact.js +76 -0
- package/dist/proxy/shell-strategies/structured.js +241 -0
- package/dist/proxy/shell-strategies/tabular.js +243 -0
- package/dist/proxy/shell-strategies/test-results-types.js +13 -0
- package/dist/proxy/shell-strategies/test-results.js +784 -0
- package/dist/proxy/shell-strategies/tree-paths.js +144 -0
- package/dist/proxy/shell-strategies/yaml.js +182 -0
- package/dist/proxy/shell-tee.js +111 -0
- package/dist/proxy/signal-dedup.js +171 -0
- package/dist/proxy/startup-renderer.js +158 -0
- package/dist/proxy/task-token-display.js +38 -0
- package/dist/proxy/token-counter.js +61 -0
- package/dist/proxy/tool-clusters.js +273 -0
- package/dist/proxy/tool-definitions.js +525 -0
- package/dist/proxy/transport-mux.js +229 -0
- package/dist/proxy/wire-cap.js +268 -0
- package/dist/schemas/api/skills.js +19 -0
- package/dist/schemas/common/errors.js +7 -0
- package/dist/schemas/common/headers.js +5 -0
- package/dist/schemas/entities/edge.js +25 -0
- package/dist/schemas/entities/entity.js +22 -0
- package/dist/schemas/entities/rule.js +18 -0
- package/dist/schemas/index.js +14 -0
- package/dist/server/event-bus.js +59 -0
- package/dist/server/http.js +156 -0
- package/dist/server/middleware.js +70 -0
- package/dist/server/routes/drift.js +97 -0
- package/dist/server/routes/intelligence.js +1217 -0
- package/dist/server/routes/reasoning-quality.js +444 -0
- package/dist/server/routes/session.js +86 -0
- package/dist/server/routes/stream.js +120 -0
- package/dist/server/routes/system.js +73 -0
- package/dist/server/routes/temporal.js +170 -0
- package/dist/server/routes/timeline.js +232 -0
- package/dist/server/routes/token-flow.js +403 -0
- package/dist/skills/effectiveness-tracker.js +93 -0
- package/dist/skills/local-pack.js +380 -0
- package/dist/skills/resolver.js +495 -0
- package/dist/state-detector.js +83 -0
- package/dist/timeline/intent-detector.js +263 -0
- package/dist/timeline/loop-miner.js +140 -0
- package/dist/timeline/open-threads.js +49 -0
- package/dist/timeline/signal-reinforcer.js +62 -0
- package/dist/timeline/timeline-bootstrap.js +151 -0
- package/dist/timeline/timeline-store.js +618 -0
- package/dist/tools/coding/bash.js +49 -0
- package/dist/tools/coding/file-edit.js +72 -0
- package/dist/tools/coding/file-outline.js +227 -0
- package/dist/tools/coding/file-read-protocol.js +425 -0
- package/dist/tools/coding/file-read.js +35 -0
- package/dist/tools/coding/file-write.js +43 -0
- package/dist/tools/coding/glob-tool.js +109 -0
- package/dist/tools/coding/grep.js +162 -0
- package/dist/tools/coding/index.js +27 -0
- package/dist/tools/intelligence/index.js +269 -0
- package/dist/tools/intelligence/record-fact.js +48 -0
- package/dist/tools/intelligence/timeline-markers.js +130 -0
- package/dist/tools/registry.js +47 -0
- package/dist/tools/types.js +8 -0
- package/dist/tracking/auto-snapshot-triggers.js +246 -0
- package/dist/tracking/branch-context.js +115 -0
- package/dist/tracking/branch-snapshot.js +217 -0
- package/dist/tracking/causal-bridge.js +317 -0
- package/dist/tracking/circuit-breaker.js +147 -0
- package/dist/tracking/commit-watcher.js +114 -0
- package/dist/tracking/context-ledger.js +119 -0
- package/dist/tracking/correction-detector.js +324 -0
- package/dist/tracking/drift-tracker.js +874 -0
- package/dist/tracking/durability-tracker.js +94 -0
- package/dist/tracking/entity-rewind.js +200 -0
- package/dist/tracking/file-hash-state.js +114 -0
- package/dist/tracking/git-attribution.js +132 -0
- package/dist/tracking/git-trailers.js +171 -0
- package/dist/tracking/intelligence-counter.js +46 -0
- package/dist/tracking/intent-correlator.js +202 -0
- package/dist/tracking/intent-encoder.js +52 -0
- package/dist/tracking/intent-token-tracker.js +159 -0
- package/dist/tracking/ledger-archiver.js +94 -0
- package/dist/tracking/ledger-chains.js +245 -0
- package/dist/tracking/metrics-store.js +361 -0
- package/dist/tracking/native-watcher.js +131 -0
- package/dist/tracking/offline-rewind.js +295 -0
- package/dist/tracking/pending-violations.js +74 -0
- package/dist/tracking/persistence-effectiveness.js +167 -0
- package/dist/tracking/prompt-durability.js +202 -0
- package/dist/tracking/quality-signals.js +213 -0
- package/dist/tracking/redactor.js +73 -0
- package/dist/tracking/rewind-engine.js +161 -0
- package/dist/tracking/session-history.js +128 -0
- package/dist/tracking/session-receipt.js +88 -0
- package/dist/tracking/session-summary-writer.js +157 -0
- package/dist/tracking/shadow-ledger.js +321 -0
- package/dist/tracking/stash-manager.js +258 -0
- package/dist/tracking/timeline-fork.js +213 -0
- package/dist/tracking/timeline.js +69 -0
- package/dist/tracking/token-flow.js +276 -0
- package/dist/tracking/turn-segmenter.js +122 -0
- package/dist/tracking/weekly-accumulator.js +179 -0
- package/dist/tracking/working-snapshots.js +188 -0
- package/dist/tracking/workspace-manifest.js +176 -0
- package/dist/transport/http.js +102 -0
- package/dist/ui/assets/index-7gl3mIuY.css +1 -0
- package/dist/ui/assets/index-BsMTQdhX.js +10 -0
- package/dist/ui/index.html +2 -2
- package/dist/utils/counterfactual.js +65 -0
- package/dist/utils/deep-link.js +34 -0
- package/dist/utils/detect.js +193 -0
- package/dist/utils/exec.js +73 -0
- package/dist/utils/file-logger.js +87 -0
- package/dist/utils/format-error.js +29 -0
- package/dist/utils/git.js +181 -0
- package/dist/utils/log.js +57 -0
- package/dist/utils/logger.js +35 -0
- package/dist/utils/mcp-content-json.js +8 -0
- package/dist/utils/session-logger.js +154 -0
- package/dist/utils/startup-log.js +512 -0
- package/dist/utils/ui.js +56 -0
- package/package.json +5 -3
- package/scripts/postinstall.mjs +299 -0
- package/dist/ui/assets/index-BISLlJyc.js +0 -10
- package/dist/ui/assets/index-BUChTv4H.css +0 -1
|
@@ -0,0 +1,190 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* BA-3.5: Change Impact Narrative + Agent-as-LLM Bridge tests.
|
|
3
|
+
*
|
|
4
|
+
* Verifies:
|
|
5
|
+
* - Narrative includes all behavior signals
|
|
6
|
+
* - Markdown well-formed
|
|
7
|
+
* - Risk level computed correctly
|
|
8
|
+
* - Counterfactual framing always present
|
|
9
|
+
* - Agent-as-LLM templates, timeout, budget, follow-through
|
|
10
|
+
*/
|
|
11
|
+
import { describe, expect, it } from "vitest";
|
|
12
|
+
import { AgentLlmBridge } from "../behaviors/agent-llm-bridge.js";
|
|
13
|
+
import { ChangeNarrativeBehavior } from "../behaviors/change-narrative.js";
|
|
14
|
+
function makeCtx() {
|
|
15
|
+
return {
|
|
16
|
+
toolName: "__session_end__",
|
|
17
|
+
args: {},
|
|
18
|
+
sessionId: "test-session",
|
|
19
|
+
};
|
|
20
|
+
}
|
|
21
|
+
// ── Change Narrative Tests ─────────────────────────────────────
|
|
22
|
+
describe("Change Impact Narrative (BA-3.2)", () => {
|
|
23
|
+
describe("Behavior Identity", () => {
|
|
24
|
+
it("has correct id and hooks", () => {
|
|
25
|
+
const narrative = new ChangeNarrativeBehavior();
|
|
26
|
+
expect(narrative.id).toBe("change_narrative");
|
|
27
|
+
expect(narrative.hooks).toContain("session_end");
|
|
28
|
+
expect(narrative.defaultLevel).toBe("suggestion");
|
|
29
|
+
});
|
|
30
|
+
});
|
|
31
|
+
describe("Empty Session", () => {
|
|
32
|
+
it("returns null when no behaviors have data", async () => {
|
|
33
|
+
const narrative = new ChangeNarrativeBehavior();
|
|
34
|
+
const output = await narrative.onSessionEnd(makeCtx());
|
|
35
|
+
expect(output).toBeNull();
|
|
36
|
+
});
|
|
37
|
+
});
|
|
38
|
+
describe("Risk Level Computation", () => {
|
|
39
|
+
it("returns low risk for clean session", async () => {
|
|
40
|
+
const narrative = new ChangeNarrativeBehavior();
|
|
41
|
+
const output = await narrative.onSessionEnd(makeCtx());
|
|
42
|
+
expect(output).toBeNull();
|
|
43
|
+
});
|
|
44
|
+
});
|
|
45
|
+
describe("Counterfactual Framing", () => {
|
|
46
|
+
it("produces counterfactual for clean sessions (no issues)", async () => {
|
|
47
|
+
const narrative = new ChangeNarrativeBehavior();
|
|
48
|
+
const output = await narrative.onSessionEnd(makeCtx());
|
|
49
|
+
if (output) {
|
|
50
|
+
const ctx = output._context?.change_narrative;
|
|
51
|
+
expect(ctx.counterfactual).toContain("Clean session");
|
|
52
|
+
}
|
|
53
|
+
});
|
|
54
|
+
});
|
|
55
|
+
describe("Markdown Output", () => {
|
|
56
|
+
it("generates well-formed markdown when behaviors have data", async () => {
|
|
57
|
+
const narrative = new ChangeNarrativeBehavior();
|
|
58
|
+
const output = await narrative.onSessionEnd(makeCtx());
|
|
59
|
+
if (output) {
|
|
60
|
+
const ctx = output._context?.change_narrative;
|
|
61
|
+
expect(ctx.markdown).toContain("## Change Impact");
|
|
62
|
+
}
|
|
63
|
+
});
|
|
64
|
+
});
|
|
65
|
+
});
|
|
66
|
+
// ── Agent-as-LLM Bridge Tests ──────────────────────────────────
|
|
67
|
+
describe("Agent-as-LLM Bridge (BA-3.3)", () => {
|
|
68
|
+
describe("Template System", () => {
|
|
69
|
+
it("lists all available templates", () => {
|
|
70
|
+
const templates = AgentLlmBridge.getTemplates();
|
|
71
|
+
expect(templates).toContain("session_resume");
|
|
72
|
+
expect(templates).toContain("doc_generation");
|
|
73
|
+
expect(templates).toContain("loop_diagnosis");
|
|
74
|
+
expect(templates).toContain("convention_fix");
|
|
75
|
+
expect(templates).toContain("cascade_fix");
|
|
76
|
+
expect(templates).toContain("architecture_alternative");
|
|
77
|
+
expect(templates.length).toBeGreaterThanOrEqual(7);
|
|
78
|
+
});
|
|
79
|
+
it("returns template definition with prefix and maxTokens", () => {
|
|
80
|
+
const def = AgentLlmBridge.getTemplateDefinition("loop_diagnosis");
|
|
81
|
+
expect(def.prefix).toContain("Analyze why");
|
|
82
|
+
expect(def.maxTokens).toBe(500);
|
|
83
|
+
expect(def.priority).toBe("high");
|
|
84
|
+
});
|
|
85
|
+
it("creates prompt with template prefix and context", () => {
|
|
86
|
+
const bridge = new AgentLlmBridge();
|
|
87
|
+
const prompt = bridge.createPrompt("doc_generation", "Function processPayment(amount: number, currency: string): Promise<Receipt>");
|
|
88
|
+
expect(prompt.template).toBe("doc_generation");
|
|
89
|
+
expect(prompt.content).toContain("Generate documentation");
|
|
90
|
+
expect(prompt.content).toContain("processPayment");
|
|
91
|
+
expect(prompt.priority).toBe("low");
|
|
92
|
+
});
|
|
93
|
+
});
|
|
94
|
+
describe("Timeout Handling", () => {
|
|
95
|
+
it("defaults to 5000ms timeout", () => {
|
|
96
|
+
const bridge = new AgentLlmBridge();
|
|
97
|
+
const prompt = bridge.createPrompt("session_resume", "test context");
|
|
98
|
+
expect(prompt.timeoutMs).toBe(5000);
|
|
99
|
+
});
|
|
100
|
+
it("accepts custom timeout", () => {
|
|
101
|
+
const bridge = new AgentLlmBridge();
|
|
102
|
+
const prompt = bridge.createPrompt("session_resume", "test context", {
|
|
103
|
+
timeoutMs: 3000,
|
|
104
|
+
});
|
|
105
|
+
expect(prompt.timeoutMs).toBe(3000);
|
|
106
|
+
});
|
|
107
|
+
});
|
|
108
|
+
describe("Fallback Handling", () => {
|
|
109
|
+
it("stores fallback content", () => {
|
|
110
|
+
const bridge = new AgentLlmBridge();
|
|
111
|
+
const prompt = bridge.createPrompt("doc_generation", "test", {
|
|
112
|
+
fallback: "AST-based docs only",
|
|
113
|
+
});
|
|
114
|
+
expect(prompt.fallbackContent).toBe("AST-based docs only");
|
|
115
|
+
});
|
|
116
|
+
it("fallback is null by default", () => {
|
|
117
|
+
const bridge = new AgentLlmBridge();
|
|
118
|
+
const prompt = bridge.createPrompt("doc_generation", "test");
|
|
119
|
+
expect(prompt.fallbackContent).toBeNull();
|
|
120
|
+
});
|
|
121
|
+
});
|
|
122
|
+
describe("Injection and Budget", () => {
|
|
123
|
+
it("injects prompt and tracks it", () => {
|
|
124
|
+
const bridge = new AgentLlmBridge();
|
|
125
|
+
const prompt = bridge.createPrompt("loop_diagnosis", "4 failed attempts on processPayment");
|
|
126
|
+
const injected = bridge.inject(prompt);
|
|
127
|
+
expect(injected).toContain("Analyze why");
|
|
128
|
+
expect(bridge.getStats().totalInjected).toBe(1);
|
|
129
|
+
});
|
|
130
|
+
it("respects per-call prompt budget", () => {
|
|
131
|
+
const bridge = new AgentLlmBridge();
|
|
132
|
+
for (let i = 0; i < 5; i++) {
|
|
133
|
+
const prompt = bridge.createPrompt("loop_diagnosis", "x".repeat(2000));
|
|
134
|
+
bridge.inject(prompt);
|
|
135
|
+
}
|
|
136
|
+
const overBudget = bridge.createPrompt("doc_generation", "x".repeat(2000));
|
|
137
|
+
const result = bridge.inject(overBudget);
|
|
138
|
+
expect(result).toBe("");
|
|
139
|
+
});
|
|
140
|
+
it("resets budget between calls", () => {
|
|
141
|
+
const bridge = new AgentLlmBridge();
|
|
142
|
+
const longContext = "x".repeat(3800);
|
|
143
|
+
const prompt1 = bridge.createPrompt("loop_diagnosis", longContext);
|
|
144
|
+
bridge.inject(prompt1);
|
|
145
|
+
bridge.resetCallBudget();
|
|
146
|
+
const prompt2 = bridge.createPrompt("doc_generation", "short context");
|
|
147
|
+
const result = bridge.inject(prompt2);
|
|
148
|
+
expect(result.length).toBeGreaterThan(0);
|
|
149
|
+
});
|
|
150
|
+
});
|
|
151
|
+
describe("Follow-Through Tracking", () => {
|
|
152
|
+
it("starts with 0% follow-through rate", () => {
|
|
153
|
+
const bridge = new AgentLlmBridge();
|
|
154
|
+
expect(bridge.getFollowThroughRate()).toBe(0);
|
|
155
|
+
});
|
|
156
|
+
it("tracks follow-through when recorded", () => {
|
|
157
|
+
const bridge = new AgentLlmBridge();
|
|
158
|
+
const prompt = bridge.createPrompt("doc_generation", "test");
|
|
159
|
+
bridge.inject(prompt);
|
|
160
|
+
const injectionId = bridge.getLastInjectionId();
|
|
161
|
+
expect(injectionId).toBeTruthy();
|
|
162
|
+
bridge.recordFollowThrough(injectionId);
|
|
163
|
+
expect(bridge.getStats().followedThrough).toBe(1);
|
|
164
|
+
});
|
|
165
|
+
it("computes follow-through rate correctly", () => {
|
|
166
|
+
const bridge = new AgentLlmBridge();
|
|
167
|
+
const p1 = bridge.createPrompt("doc_generation", "test1");
|
|
168
|
+
bridge.inject(p1);
|
|
169
|
+
const id1 = bridge.getLastInjectionId();
|
|
170
|
+
bridge.recordFollowThrough(id1);
|
|
171
|
+
bridge.resetCallBudget();
|
|
172
|
+
const p2 = bridge.createPrompt("convention_fix", "test2");
|
|
173
|
+
bridge.inject(p2);
|
|
174
|
+
expect(bridge.getFollowThroughRate()).toBe(0.5);
|
|
175
|
+
});
|
|
176
|
+
});
|
|
177
|
+
describe("Stats", () => {
|
|
178
|
+
it("tracks injections by template type", () => {
|
|
179
|
+
const bridge = new AgentLlmBridge();
|
|
180
|
+
bridge.inject(bridge.createPrompt("doc_generation", "a"));
|
|
181
|
+
bridge.resetCallBudget();
|
|
182
|
+
bridge.inject(bridge.createPrompt("doc_generation", "b"));
|
|
183
|
+
bridge.resetCallBudget();
|
|
184
|
+
bridge.inject(bridge.createPrompt("loop_diagnosis", "c"));
|
|
185
|
+
const stats = bridge.getStats();
|
|
186
|
+
expect(stats.byTemplate.doc_generation).toBe(2);
|
|
187
|
+
expect(stats.byTemplate.loop_diagnosis).toBe(1);
|
|
188
|
+
});
|
|
189
|
+
});
|
|
190
|
+
});
|
|
@@ -0,0 +1,258 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Tests for pre-commit convention check (Sprint 4, Task 4.2).
|
|
3
|
+
*
|
|
4
|
+
* Tests the blocking mode detection algorithm, config gating logic,
|
|
5
|
+
* staged file filtering, graph loading graceful degradation,
|
|
6
|
+
* and violation evaluation pipeline.
|
|
7
|
+
*/
|
|
8
|
+
import { existsSync, mkdirSync, rmSync, writeFileSync } from "node:fs";
|
|
9
|
+
import { tmpdir } from "node:os";
|
|
10
|
+
import { join } from "node:path";
|
|
11
|
+
import { afterEach, beforeEach, describe, expect, it } from "vitest";
|
|
12
|
+
describe("Check-Commit Hook", () => {
|
|
13
|
+
let tempDir;
|
|
14
|
+
beforeEach(() => {
|
|
15
|
+
tempDir = join(tmpdir(), `unerr-test-check-${Date.now()}`);
|
|
16
|
+
mkdirSync(tempDir, { recursive: true });
|
|
17
|
+
});
|
|
18
|
+
afterEach(() => {
|
|
19
|
+
rmSync(tempDir, { recursive: true, force: true });
|
|
20
|
+
});
|
|
21
|
+
it("pre-commit hook script is valid shell with shebang and non-blocking exit", () => {
|
|
22
|
+
const hookScript = `#!/bin/sh
|
|
23
|
+
# unerr pre-commit convention check
|
|
24
|
+
if command -v unerr &> /dev/null; then
|
|
25
|
+
unerr check-commit 2>/dev/null
|
|
26
|
+
fi
|
|
27
|
+
# Non-blocking — always allow commit
|
|
28
|
+
exit 0
|
|
29
|
+
`;
|
|
30
|
+
expect(hookScript.startsWith("#!/bin/sh")).toBe(true);
|
|
31
|
+
expect(hookScript.trimEnd().endsWith("exit 0")).toBe(true);
|
|
32
|
+
expect(hookScript).toContain("unerr check-commit");
|
|
33
|
+
// Stderr redirect ensures no noise in git output
|
|
34
|
+
expect(hookScript).toContain("2>/dev/null");
|
|
35
|
+
// command -v check ensures graceful degradation when unerr not installed
|
|
36
|
+
expect(hookScript).toContain("command -v unerr");
|
|
37
|
+
});
|
|
38
|
+
describe("blocking mode detection algorithm", () => {
|
|
39
|
+
it("defaults to non-blocking when no settings exist", () => {
|
|
40
|
+
const settingsPath = join(tempDir, ".unerr", "settings.json");
|
|
41
|
+
expect(existsSync(settingsPath)).toBe(false);
|
|
42
|
+
const optsBlocking = undefined;
|
|
43
|
+
let blockingMode = optsBlocking ?? false;
|
|
44
|
+
if (!blockingMode && existsSync(settingsPath)) {
|
|
45
|
+
blockingMode = true;
|
|
46
|
+
}
|
|
47
|
+
expect(blockingMode).toBe(false);
|
|
48
|
+
});
|
|
49
|
+
it("reads blocking=true from settings.json when CLI flag not set", () => {
|
|
50
|
+
const settingsDir = join(tempDir, ".unerr");
|
|
51
|
+
mkdirSync(settingsDir, { recursive: true });
|
|
52
|
+
writeFileSync(join(settingsDir, "settings.json"), JSON.stringify({ hooks: { precommit: { blocking: true } } }));
|
|
53
|
+
const settingsPath = join(settingsDir, "settings.json");
|
|
54
|
+
let blockingMode = false;
|
|
55
|
+
if (!blockingMode && existsSync(settingsPath)) {
|
|
56
|
+
const settings = JSON.parse(require("node:fs").readFileSync(settingsPath, "utf-8"));
|
|
57
|
+
blockingMode = settings.hooks?.precommit?.blocking ?? false;
|
|
58
|
+
}
|
|
59
|
+
expect(blockingMode).toBe(true);
|
|
60
|
+
});
|
|
61
|
+
it("CLI --blocking flag takes precedence over settings.json", () => {
|
|
62
|
+
const settingsDir = join(tempDir, ".unerr");
|
|
63
|
+
mkdirSync(settingsDir, { recursive: true });
|
|
64
|
+
writeFileSync(join(settingsDir, "settings.json"), JSON.stringify({ hooks: { precommit: { blocking: false } } }));
|
|
65
|
+
const optsBlocking = true;
|
|
66
|
+
let blockingMode = optsBlocking ?? false;
|
|
67
|
+
const settingsPath = join(settingsDir, "settings.json");
|
|
68
|
+
if (!blockingMode && existsSync(settingsPath)) {
|
|
69
|
+
blockingMode = false;
|
|
70
|
+
}
|
|
71
|
+
expect(blockingMode).toBe(true);
|
|
72
|
+
});
|
|
73
|
+
it("handles malformed settings.json without crashing", () => {
|
|
74
|
+
const settingsDir = join(tempDir, ".unerr");
|
|
75
|
+
mkdirSync(settingsDir, { recursive: true });
|
|
76
|
+
writeFileSync(join(settingsDir, "settings.json"), "NOT VALID JSON!!!");
|
|
77
|
+
const settingsPath = join(settingsDir, "settings.json");
|
|
78
|
+
let blockingMode = false;
|
|
79
|
+
if (!blockingMode && existsSync(settingsPath)) {
|
|
80
|
+
try {
|
|
81
|
+
const settings = JSON.parse(require("node:fs").readFileSync(settingsPath, "utf-8"));
|
|
82
|
+
blockingMode = settings.hooks?.precommit?.blocking ?? false;
|
|
83
|
+
}
|
|
84
|
+
catch {
|
|
85
|
+
// Should catch and keep blockingMode false
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
expect(blockingMode).toBe(false);
|
|
89
|
+
});
|
|
90
|
+
it("handles settings.json with missing hooks section", () => {
|
|
91
|
+
const settingsDir = join(tempDir, ".unerr");
|
|
92
|
+
mkdirSync(settingsDir, { recursive: true });
|
|
93
|
+
writeFileSync(join(settingsDir, "settings.json"), JSON.stringify({ theme: "dark" }));
|
|
94
|
+
const settingsPath = join(settingsDir, "settings.json");
|
|
95
|
+
let blockingMode = false;
|
|
96
|
+
if (!blockingMode && existsSync(settingsPath)) {
|
|
97
|
+
const settings = JSON.parse(require("node:fs").readFileSync(settingsPath, "utf-8"));
|
|
98
|
+
blockingMode = settings.hooks?.precommit?.blocking ?? false;
|
|
99
|
+
}
|
|
100
|
+
expect(blockingMode).toBe(false);
|
|
101
|
+
});
|
|
102
|
+
});
|
|
103
|
+
describe("config gating", () => {
|
|
104
|
+
it("skips when no .unerr/config.json exists", () => {
|
|
105
|
+
const configPath = join(tempDir, ".unerr", "config.json");
|
|
106
|
+
expect(existsSync(configPath)).toBe(false);
|
|
107
|
+
});
|
|
108
|
+
it("proceeds when .unerr/config.json exists with repoId", () => {
|
|
109
|
+
const unerrDir = join(tempDir, ".unerr");
|
|
110
|
+
mkdirSync(unerrDir, { recursive: true });
|
|
111
|
+
writeFileSync(join(unerrDir, "config.json"), JSON.stringify({
|
|
112
|
+
repoId: "test-repo",
|
|
113
|
+
}));
|
|
114
|
+
const configPath = join(unerrDir, "config.json");
|
|
115
|
+
expect(existsSync(configPath)).toBe(true);
|
|
116
|
+
const config = JSON.parse(require("node:fs").readFileSync(configPath, "utf-8"));
|
|
117
|
+
expect(config.repoId).toBe("test-repo");
|
|
118
|
+
});
|
|
119
|
+
it("skips when config.json has no repoId", () => {
|
|
120
|
+
const unerrDir = join(tempDir, ".unerr");
|
|
121
|
+
mkdirSync(unerrDir, { recursive: true });
|
|
122
|
+
writeFileSync(join(unerrDir, "config.json"), JSON.stringify({ orgId: "org-1" }));
|
|
123
|
+
const config = JSON.parse(require("node:fs").readFileSync(join(unerrDir, "config.json"), "utf-8"));
|
|
124
|
+
expect(config.repoId).toBeUndefined();
|
|
125
|
+
});
|
|
126
|
+
it("skips when config.json is malformed", () => {
|
|
127
|
+
const unerrDir = join(tempDir, ".unerr");
|
|
128
|
+
mkdirSync(unerrDir, { recursive: true });
|
|
129
|
+
writeFileSync(join(unerrDir, "config.json"), "{invalid json}}}");
|
|
130
|
+
let repoId;
|
|
131
|
+
try {
|
|
132
|
+
const config = JSON.parse(require("node:fs").readFileSync(join(unerrDir, "config.json"), "utf-8"));
|
|
133
|
+
repoId = config.repoId;
|
|
134
|
+
}
|
|
135
|
+
catch {
|
|
136
|
+
repoId = undefined;
|
|
137
|
+
}
|
|
138
|
+
expect(repoId).toBeUndefined();
|
|
139
|
+
});
|
|
140
|
+
});
|
|
141
|
+
describe("staged file filtering", () => {
|
|
142
|
+
const SUPPORTED_EXTENSIONS = new Set([
|
|
143
|
+
".ts",
|
|
144
|
+
".tsx",
|
|
145
|
+
".js",
|
|
146
|
+
".jsx",
|
|
147
|
+
".py",
|
|
148
|
+
".go",
|
|
149
|
+
]);
|
|
150
|
+
it("filters to supported file extensions", () => {
|
|
151
|
+
const stagedFiles = [
|
|
152
|
+
"src/index.ts",
|
|
153
|
+
"README.md",
|
|
154
|
+
"src/app.tsx",
|
|
155
|
+
"package.json",
|
|
156
|
+
"main.go",
|
|
157
|
+
"script.py",
|
|
158
|
+
"image.png",
|
|
159
|
+
"style.css",
|
|
160
|
+
];
|
|
161
|
+
const checkable = stagedFiles.filter((f) => {
|
|
162
|
+
const ext = f.slice(f.lastIndexOf("."));
|
|
163
|
+
return SUPPORTED_EXTENSIONS.has(ext);
|
|
164
|
+
});
|
|
165
|
+
expect(checkable).toEqual([
|
|
166
|
+
"src/index.ts",
|
|
167
|
+
"src/app.tsx",
|
|
168
|
+
"main.go",
|
|
169
|
+
"script.py",
|
|
170
|
+
]);
|
|
171
|
+
});
|
|
172
|
+
it("returns empty array when no supported files staged", () => {
|
|
173
|
+
const stagedFiles = ["README.md", "package.json", "image.png"];
|
|
174
|
+
const checkable = stagedFiles.filter((f) => {
|
|
175
|
+
const ext = f.slice(f.lastIndexOf("."));
|
|
176
|
+
return SUPPORTED_EXTENSIONS.has(ext);
|
|
177
|
+
});
|
|
178
|
+
expect(checkable).toHaveLength(0);
|
|
179
|
+
});
|
|
180
|
+
it("handles files with multiple dots correctly", () => {
|
|
181
|
+
const stagedFiles = ["src/app.test.ts", "config.prod.json", "lib.d.ts"];
|
|
182
|
+
const checkable = stagedFiles.filter((f) => {
|
|
183
|
+
const ext = f.slice(f.lastIndexOf("."));
|
|
184
|
+
return SUPPORTED_EXTENSIONS.has(ext);
|
|
185
|
+
});
|
|
186
|
+
expect(checkable).toEqual(["src/app.test.ts", "lib.d.ts"]);
|
|
187
|
+
});
|
|
188
|
+
});
|
|
189
|
+
describe("violation display logic", () => {
|
|
190
|
+
it("counts errors and warnings separately", () => {
|
|
191
|
+
const violations = [
|
|
192
|
+
{ severity: "error", message: "bad naming" },
|
|
193
|
+
{ severity: "warning", message: "could improve" },
|
|
194
|
+
{ severity: "error", message: "missing type" },
|
|
195
|
+
{ severity: "info", message: "suggestion" },
|
|
196
|
+
];
|
|
197
|
+
const errorCount = violations.filter((v) => v.severity === "error").length;
|
|
198
|
+
const warningCount = violations.filter((v) => v.severity === "warning").length;
|
|
199
|
+
expect(errorCount).toBe(2);
|
|
200
|
+
expect(warningCount).toBe(1);
|
|
201
|
+
});
|
|
202
|
+
it("formats violation summary correctly", () => {
|
|
203
|
+
const totalViolations = 5;
|
|
204
|
+
const errorCount = 2;
|
|
205
|
+
const warningCount = 3;
|
|
206
|
+
const errorSuffix = errorCount > 0
|
|
207
|
+
? ` (${errorCount} error${errorCount !== 1 ? "s" : ""})`
|
|
208
|
+
: "";
|
|
209
|
+
const warnSuffix = warningCount > 0
|
|
210
|
+
? ` (${warningCount} warning${warningCount !== 1 ? "s" : ""})`
|
|
211
|
+
: "";
|
|
212
|
+
const summary = `${totalViolations} violation${totalViolations !== 1 ? "s" : ""} found${errorSuffix}${warnSuffix}`;
|
|
213
|
+
expect(summary).toBe("5 violations found (2 errors) (3 warnings)");
|
|
214
|
+
});
|
|
215
|
+
it("handles singular forms correctly", () => {
|
|
216
|
+
const count = 1;
|
|
217
|
+
const errCount = 1;
|
|
218
|
+
const summary = `${count} violation${count !== 1 ? "s" : ""} found (${errCount} error${errCount !== 1 ? "s" : ""})`;
|
|
219
|
+
expect(summary).toBe("1 violation found (1 error)");
|
|
220
|
+
});
|
|
221
|
+
});
|
|
222
|
+
describe("graph loading graceful degradation", () => {
|
|
223
|
+
it("returns null when manifest does not exist", () => {
|
|
224
|
+
const manifestPath = join(tempDir, "manifests", "test-repo.json");
|
|
225
|
+
expect(existsSync(manifestPath)).toBe(false);
|
|
226
|
+
});
|
|
227
|
+
it("returns null when snapshot file does not exist", () => {
|
|
228
|
+
const manifestsDir = join(tempDir, "manifests");
|
|
229
|
+
mkdirSync(manifestsDir, { recursive: true });
|
|
230
|
+
writeFileSync(join(manifestsDir, "test-repo.json"), JSON.stringify({ entityCount: 100, edgeCount: 200 }));
|
|
231
|
+
const snapshotsDir = join(tempDir, "snapshots");
|
|
232
|
+
const gzPath = join(snapshotsDir, "test-repo.msgpack.gz");
|
|
233
|
+
const rawPath = join(snapshotsDir, "test-repo.msgpack");
|
|
234
|
+
expect(existsSync(gzPath)).toBe(false);
|
|
235
|
+
expect(existsSync(rawPath)).toBe(false);
|
|
236
|
+
});
|
|
237
|
+
});
|
|
238
|
+
describe("exit code logic", () => {
|
|
239
|
+
it("exit 0 in non-blocking mode with violations", () => {
|
|
240
|
+
const blockingMode = false;
|
|
241
|
+
const hasViolations = true;
|
|
242
|
+
const exitCode = blockingMode && hasViolations ? 1 : 0;
|
|
243
|
+
expect(exitCode).toBe(0);
|
|
244
|
+
});
|
|
245
|
+
it("exit 1 in blocking mode with violations", () => {
|
|
246
|
+
const blockingMode = true;
|
|
247
|
+
const hasViolations = true;
|
|
248
|
+
const exitCode = blockingMode && hasViolations ? 1 : 0;
|
|
249
|
+
expect(exitCode).toBe(1);
|
|
250
|
+
});
|
|
251
|
+
it("exit 0 in blocking mode without violations", () => {
|
|
252
|
+
const blockingMode = true;
|
|
253
|
+
const hasViolations = false;
|
|
254
|
+
const exitCode = blockingMode && hasViolations ? 1 : 0;
|
|
255
|
+
expect(exitCode).toBe(0);
|
|
256
|
+
});
|
|
257
|
+
});
|
|
258
|
+
});
|
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
import { createHash } from "node:crypto";
|
|
2
|
+
import { describe, expect, it } from "vitest";
|
|
3
|
+
describe("checksum verification", () => {
|
|
4
|
+
function computeChecksum(buf) {
|
|
5
|
+
return createHash("sha256").update(buf).digest("hex");
|
|
6
|
+
}
|
|
7
|
+
it("produces 64-char hex string", () => {
|
|
8
|
+
const buf = Buffer.from("hello world");
|
|
9
|
+
const checksum = computeChecksum(buf);
|
|
10
|
+
expect(checksum).toMatch(/^[0-9a-f]{64}$/);
|
|
11
|
+
});
|
|
12
|
+
it("is deterministic", () => {
|
|
13
|
+
const buf = Buffer.from("test data");
|
|
14
|
+
expect(computeChecksum(buf)).toBe(computeChecksum(buf));
|
|
15
|
+
});
|
|
16
|
+
it("detects tampered buffer", () => {
|
|
17
|
+
const original = Buffer.from("original data");
|
|
18
|
+
const originalChecksum = computeChecksum(original);
|
|
19
|
+
const tampered = Buffer.from("tampered data");
|
|
20
|
+
const tamperedChecksum = computeChecksum(tampered);
|
|
21
|
+
expect(originalChecksum).not.toBe(tamperedChecksum);
|
|
22
|
+
});
|
|
23
|
+
it("matches known SHA-256 value", () => {
|
|
24
|
+
// SHA-256 of empty string is well-known
|
|
25
|
+
const empty = Buffer.from("");
|
|
26
|
+
const checksum = computeChecksum(empty);
|
|
27
|
+
expect(checksum).toBe("e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855");
|
|
28
|
+
});
|
|
29
|
+
it("handles large buffers", () => {
|
|
30
|
+
const largeBuf = Buffer.alloc(1024 * 1024, 0x42); // 1MB of 'B'
|
|
31
|
+
const checksum = computeChecksum(largeBuf);
|
|
32
|
+
expect(checksum).toMatch(/^[0-9a-f]{64}$/);
|
|
33
|
+
});
|
|
34
|
+
});
|
|
@@ -0,0 +1,154 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* P10-TEST-06 (partial): Commit watcher tests — HEAD polling, commit association.
|
|
3
|
+
*/
|
|
4
|
+
import { execSync } from "node:child_process";
|
|
5
|
+
import { mkdirSync, rmSync, writeFileSync } from "node:fs";
|
|
6
|
+
import { tmpdir } from "node:os";
|
|
7
|
+
import { join } from "node:path";
|
|
8
|
+
import { afterEach, beforeEach, describe, expect, it } from "vitest";
|
|
9
|
+
import { CommitWatcher } from "../tracking/commit-watcher.js";
|
|
10
|
+
import { IntentCorrelator } from "../tracking/intent-correlator.js";
|
|
11
|
+
import { ShadowLedger } from "../tracking/shadow-ledger.js";
|
|
12
|
+
let tempDir;
|
|
13
|
+
let unerrDir;
|
|
14
|
+
let repoDir;
|
|
15
|
+
function initGitRepo() {
|
|
16
|
+
repoDir = join(tempDir, "repo");
|
|
17
|
+
mkdirSync(repoDir, { recursive: true });
|
|
18
|
+
execSync("git init", { cwd: repoDir, stdio: "pipe" });
|
|
19
|
+
execSync("git config user.email 'test@test.com'", {
|
|
20
|
+
cwd: repoDir,
|
|
21
|
+
stdio: "pipe",
|
|
22
|
+
});
|
|
23
|
+
execSync("git config user.name 'Test'", { cwd: repoDir, stdio: "pipe" });
|
|
24
|
+
writeFileSync(join(repoDir, "README.md"), "# Test");
|
|
25
|
+
execSync("git add -A", { cwd: repoDir, stdio: "pipe" });
|
|
26
|
+
execSync("git commit -m 'initial'", { cwd: repoDir, stdio: "pipe" });
|
|
27
|
+
}
|
|
28
|
+
beforeEach(() => {
|
|
29
|
+
tempDir = join(tmpdir(), `unerr-commit-test-${Date.now()}-${Math.random().toString(36).slice(2)}`);
|
|
30
|
+
mkdirSync(tempDir, { recursive: true });
|
|
31
|
+
initGitRepo();
|
|
32
|
+
unerrDir = join(repoDir, ".unerr");
|
|
33
|
+
mkdirSync(unerrDir, { recursive: true });
|
|
34
|
+
});
|
|
35
|
+
afterEach(() => {
|
|
36
|
+
try {
|
|
37
|
+
rmSync(tempDir, { recursive: true, force: true });
|
|
38
|
+
}
|
|
39
|
+
catch {
|
|
40
|
+
/* ignore */
|
|
41
|
+
}
|
|
42
|
+
});
|
|
43
|
+
describe("CommitWatcher", () => {
|
|
44
|
+
it("captures initial HEAD SHA on start", async () => {
|
|
45
|
+
const correlator = new IntentCorrelator(unerrDir);
|
|
46
|
+
const watcher = new CommitWatcher(correlator, { cwd: repoDir });
|
|
47
|
+
await watcher.start();
|
|
48
|
+
const headSha = execSync("git rev-parse HEAD", {
|
|
49
|
+
cwd: repoDir,
|
|
50
|
+
encoding: "utf-8",
|
|
51
|
+
}).trim();
|
|
52
|
+
expect(watcher.getLastHeadSha()).toBe(headSha);
|
|
53
|
+
watcher.stop();
|
|
54
|
+
});
|
|
55
|
+
it("detects new commit and extracts changed files", async () => {
|
|
56
|
+
const correlator = new IntentCorrelator(unerrDir);
|
|
57
|
+
const commits = [];
|
|
58
|
+
const watcher = new CommitWatcher(correlator, {
|
|
59
|
+
cwd: repoDir,
|
|
60
|
+
onCommit: (sha, files, associated) => {
|
|
61
|
+
commits.push({ sha, files, associated });
|
|
62
|
+
},
|
|
63
|
+
});
|
|
64
|
+
await watcher.start();
|
|
65
|
+
mkdirSync(join(repoDir, "src"), { recursive: true });
|
|
66
|
+
writeFileSync(join(repoDir, "src", "auth.ts"), "export function login() {}");
|
|
67
|
+
execSync("git add -A", { cwd: repoDir, stdio: "pipe" });
|
|
68
|
+
execSync("git commit -m 'add auth'", { cwd: repoDir, stdio: "pipe" });
|
|
69
|
+
await watcher.poll();
|
|
70
|
+
expect(commits).toHaveLength(1);
|
|
71
|
+
expect(commits[0]?.files).toContain("src/auth.ts");
|
|
72
|
+
const newHead = execSync("git rev-parse HEAD", {
|
|
73
|
+
cwd: repoDir,
|
|
74
|
+
encoding: "utf-8",
|
|
75
|
+
}).trim();
|
|
76
|
+
expect(commits[0]?.sha).toBe(newHead);
|
|
77
|
+
watcher.stop();
|
|
78
|
+
});
|
|
79
|
+
it("associates pending correlations on commit", async () => {
|
|
80
|
+
const ledger = new ShadowLedger(unerrDir);
|
|
81
|
+
const correlator = new IntentCorrelator(unerrDir);
|
|
82
|
+
ledger.record("get_function", { key: "abc" }, { found: true }, "main", "aaa");
|
|
83
|
+
correlator.onSyncLocalDiff(ledger, {
|
|
84
|
+
prompt: "Fix auth",
|
|
85
|
+
files: [{ path: "src/auth.ts", content: "..." }],
|
|
86
|
+
});
|
|
87
|
+
expect(correlator.getPendingCount()).toBe(1);
|
|
88
|
+
const watcher = new CommitWatcher(correlator, { cwd: repoDir });
|
|
89
|
+
await watcher.start();
|
|
90
|
+
mkdirSync(join(repoDir, "src"), { recursive: true });
|
|
91
|
+
writeFileSync(join(repoDir, "src", "auth.ts"), "export function login() {}");
|
|
92
|
+
execSync("git add -A", { cwd: repoDir, stdio: "pipe" });
|
|
93
|
+
execSync("git commit -m 'add auth'", { cwd: repoDir, stdio: "pipe" });
|
|
94
|
+
await watcher.poll();
|
|
95
|
+
expect(correlator.getPendingCount()).toBe(0);
|
|
96
|
+
expect(correlator.getCommittedUnflushed()).toHaveLength(1);
|
|
97
|
+
expect(correlator.getCommittedUnflushed()[0]?.commitSha).toBeTruthy();
|
|
98
|
+
watcher.stop();
|
|
99
|
+
});
|
|
100
|
+
it("does not fire callback when HEAD unchanged", async () => {
|
|
101
|
+
const correlator = new IntentCorrelator(unerrDir);
|
|
102
|
+
let callCount = 0;
|
|
103
|
+
const watcher = new CommitWatcher(correlator, {
|
|
104
|
+
cwd: repoDir,
|
|
105
|
+
onCommit: () => {
|
|
106
|
+
callCount++;
|
|
107
|
+
},
|
|
108
|
+
});
|
|
109
|
+
await watcher.start();
|
|
110
|
+
await watcher.poll();
|
|
111
|
+
await watcher.poll();
|
|
112
|
+
expect(callCount).toBe(0);
|
|
113
|
+
watcher.stop();
|
|
114
|
+
});
|
|
115
|
+
it("handles non-git directory gracefully", async () => {
|
|
116
|
+
const nonGitDir = join(tempDir, "not-a-repo");
|
|
117
|
+
mkdirSync(nonGitDir, { recursive: true });
|
|
118
|
+
const nonGitUnerrDir = join(nonGitDir, ".unerr");
|
|
119
|
+
mkdirSync(nonGitUnerrDir, { recursive: true });
|
|
120
|
+
const correlator = new IntentCorrelator(nonGitUnerrDir);
|
|
121
|
+
const watcher = new CommitWatcher(correlator, { cwd: nonGitDir });
|
|
122
|
+
await watcher.start();
|
|
123
|
+
await watcher.poll();
|
|
124
|
+
expect(watcher.getLastHeadSha()).toBeNull();
|
|
125
|
+
watcher.stop();
|
|
126
|
+
});
|
|
127
|
+
it("skips non-overlapping files during association", async () => {
|
|
128
|
+
const ledger = new ShadowLedger(unerrDir);
|
|
129
|
+
const correlator = new IntentCorrelator(unerrDir);
|
|
130
|
+
ledger.record("get_function", {}, {}, "main", "aaa");
|
|
131
|
+
correlator.onSyncLocalDiff(ledger, {
|
|
132
|
+
files: [{ path: "src/billing.ts", content: "..." }],
|
|
133
|
+
});
|
|
134
|
+
const watcher = new CommitWatcher(correlator, { cwd: repoDir });
|
|
135
|
+
await watcher.start();
|
|
136
|
+
writeFileSync(join(repoDir, "config.json"), "{}");
|
|
137
|
+
execSync("git add -A", { cwd: repoDir, stdio: "pipe" });
|
|
138
|
+
execSync("git commit -m 'add config'", { cwd: repoDir, stdio: "pipe" });
|
|
139
|
+
await watcher.poll();
|
|
140
|
+
expect(correlator.getPendingCount()).toBe(1);
|
|
141
|
+
expect(correlator.getCommittedUnflushed()).toHaveLength(0);
|
|
142
|
+
watcher.stop();
|
|
143
|
+
});
|
|
144
|
+
it("stop prevents further polling", async () => {
|
|
145
|
+
const correlator = new IntentCorrelator(unerrDir);
|
|
146
|
+
const watcher = new CommitWatcher(correlator, {
|
|
147
|
+
cwd: repoDir,
|
|
148
|
+
pollIntervalMs: 10,
|
|
149
|
+
});
|
|
150
|
+
await watcher.start();
|
|
151
|
+
watcher.stop();
|
|
152
|
+
expect(watcher.getLastHeadSha()).toBeTruthy();
|
|
153
|
+
});
|
|
154
|
+
});
|