@unerr-ai/unerr 0.1.6 → 0.1.7
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 +70 -194
- package/dist/cli.js +39149 -36991
- package/package.json +9 -2
- package/dist/__tests__/architecture-guard.test.js +0 -122
- package/dist/__tests__/arg-validator.test.js +0 -205
- package/dist/__tests__/ast-extractor.test.js +0 -203
- package/dist/__tests__/auto-bootstrap.test.js +0 -280
- package/dist/__tests__/background-indexer.test.js +0 -228
- package/dist/__tests__/blast-radius-engine.test.js +0 -200
- package/dist/__tests__/bridge-isolation.test.js +0 -37
- package/dist/__tests__/budget-enforcer.test.js +0 -53
- package/dist/__tests__/cfg-test-detection-perf.test.js +0 -82
- package/dist/__tests__/change-narrative.test.js +0 -190
- package/dist/__tests__/check-commit.test.js +0 -258
- package/dist/__tests__/checksum.test.js +0 -34
- package/dist/__tests__/commit-watcher.test.js +0 -154
- package/dist/__tests__/community-detection.test.js +0 -179
- package/dist/__tests__/community-tools.test.js +0 -299
- package/dist/__tests__/components.test.js +0 -449
- package/dist/__tests__/compression-log.test.js +0 -174
- package/dist/__tests__/compression-quality-monitor.test.js +0 -40
- package/dist/__tests__/config-healer.test.js +0 -165
- package/dist/__tests__/context-ledger.test.js +0 -58
- package/dist/__tests__/convention-detector.test.js +0 -99
- package/dist/__tests__/convention-learner.test.js +0 -86
- package/dist/__tests__/correction-detector.test.js +0 -330
- package/dist/__tests__/daemon-autostart-install.test.js +0 -283
- package/dist/__tests__/daemon-bridge.test.js +0 -222
- package/dist/__tests__/daemon-dashboard.test.js +0 -202
- package/dist/__tests__/daemon-registry.test.js +0 -240
- package/dist/__tests__/daemon-supervisor.test.js +0 -318
- package/dist/__tests__/daemon-version-check.test.js +0 -275
- package/dist/__tests__/decision-point-detector.test.js +0 -98
- package/dist/__tests__/deep-link.test.js +0 -143
- package/dist/__tests__/disallowed-tools.test.js +0 -115
- package/dist/__tests__/drift-tracker.test.js +0 -582
- package/dist/__tests__/durability-scorer.test.js +0 -152
- package/dist/__tests__/efficiency-tracker.test.js +0 -65
- package/dist/__tests__/enrich.test.js +0 -144
- package/dist/__tests__/entity-rewind.test.js +0 -248
- package/dist/__tests__/ephemeral.test.js +0 -111
- package/dist/__tests__/exploration-cost.test.js +0 -93
- package/dist/__tests__/fact-generator.test.js +0 -197
- package/dist/__tests__/file-l0-graph.test.js +0 -244
- package/dist/__tests__/file-logger.test.js +0 -82
- package/dist/__tests__/file-outline.test.js +0 -141
- package/dist/__tests__/file-read-protocol.test.js +0 -188
- package/dist/__tests__/format-encoder.test.js +0 -233
- package/dist/__tests__/git-attribution.test.js +0 -259
- package/dist/__tests__/graph-temporal-joiner.test.js +0 -219
- package/dist/__tests__/health-grade-enhanced.test.js +0 -138
- package/dist/__tests__/health-map-data.test.js +0 -173
- package/dist/__tests__/helpers/mcp-harness.js +0 -45
- package/dist/__tests__/helpers/mcp-harness.test.js +0 -68
- package/dist/__tests__/hook-dedup.test.js +0 -112
- package/dist/__tests__/hook-runner.test.js +0 -253
- package/dist/__tests__/indexer-cfg.test.js +0 -185
- package/dist/__tests__/indexer-cross-file.test.js +0 -172
- package/dist/__tests__/indexer-extraction.test.js +0 -245
- package/dist/__tests__/indexer-incremental.test.js +0 -232
- package/dist/__tests__/indexer-language-expansion.test.js +0 -165
- package/dist/__tests__/init-push.test.js +0 -131
- package/dist/__tests__/instruction-writer.test.js +0 -179
- package/dist/__tests__/intelligence-integration.test.js +0 -217
- package/dist/__tests__/intent-correlator.test.js +0 -175
- package/dist/__tests__/intent-detector.test.js +0 -235
- package/dist/__tests__/intent-encoder.test.js +0 -167
- package/dist/__tests__/java-build-tool-detection.test.js +0 -174
- package/dist/__tests__/layer3-sprint-q.test.js +0 -160
- package/dist/__tests__/layer3-sprint-r.test.js +0 -91
- package/dist/__tests__/layer3-sprint-s.test.js +0 -183
- package/dist/__tests__/layer3-sprint-t.test.js +0 -201
- package/dist/__tests__/layer3-sprint-u.test.js +0 -174
- package/dist/__tests__/layer4-sprint-ba2.test.js +0 -354
- package/dist/__tests__/layer4-sprint-ba4.test.js +0 -84
- package/dist/__tests__/layer4-sprint-vs.test.js +0 -105
- package/dist/__tests__/ledger-chains.test.js +0 -162
- package/dist/__tests__/lifecycle-machine.test.js +0 -226
- package/dist/__tests__/local-chat-provider.test.js +0 -170
- package/dist/__tests__/local-convention-detector.test.js +0 -308
- package/dist/__tests__/local-embeddings.test.js +0 -422
- package/dist/__tests__/local-graph.test.js +0 -540
- package/dist/__tests__/local-indexer.test.js +0 -228
- package/dist/__tests__/local-intelligence-l3.test.js +0 -332
- package/dist/__tests__/local-llm.test.js +0 -253
- package/dist/__tests__/local-mode-offline.test.js +0 -187
- package/dist/__tests__/local-mode-stats.test.js +0 -273
- package/dist/__tests__/local-mode-tui.test.js +0 -343
- package/dist/__tests__/local-parse.test.js +0 -199
- package/dist/__tests__/log-tailer.test.js +0 -208
- package/dist/__tests__/loop-breaker.test.js +0 -276
- package/dist/__tests__/loop-miner.test.js +0 -226
- package/dist/__tests__/mcp-config.test.js +0 -126
- package/dist/__tests__/mcp-content-json.test.js +0 -10
- package/dist/__tests__/mcp-envelope.test.js +0 -124
- package/dist/__tests__/metrics-store.test.js +0 -223
- package/dist/__tests__/native-watcher.test.js +0 -191
- package/dist/__tests__/navigation-hooks-agent-aware.test.js +0 -145
- package/dist/__tests__/negative-knowledge.test.js +0 -116
- package/dist/__tests__/network-boundary.test.js +0 -190
- package/dist/__tests__/network-firewall.test.js +0 -112
- package/dist/__tests__/nudge-invariants.test.js +0 -160
- package/dist/__tests__/nudge-v2.test.js +0 -225
- package/dist/__tests__/offline-rewind.test.js +0 -251
- package/dist/__tests__/open-threads.test.js +0 -89
- package/dist/__tests__/output-compressor.test.js +0 -93
- package/dist/__tests__/pending-violations.test.js +0 -112
- package/dist/__tests__/persistence-effectiveness.test.js +0 -143
- package/dist/__tests__/provider-factory.test.js +0 -42
- package/dist/__tests__/providers.test.js +0 -24
- package/dist/__tests__/proxy.test.js +0 -314
- package/dist/__tests__/query-router.test.js +0 -1018
- package/dist/__tests__/reasoning-quality-route.test.js +0 -138
- package/dist/__tests__/redactor.test.js +0 -120
- package/dist/__tests__/resource-monitor.test.js +0 -57
- package/dist/__tests__/response-envelope.test.js +0 -100
- package/dist/__tests__/risk-classifier.test.js +0 -101
- package/dist/__tests__/risk-signal-scope.test.js +0 -75
- package/dist/__tests__/rule-evaluator.test.js +0 -280
- package/dist/__tests__/scip-decoder.test.js +0 -49
- package/dist/__tests__/scip-downloader.test.js +0 -201
- package/dist/__tests__/scip-merger.test.js +0 -103
- package/dist/__tests__/search-index.test.js +0 -422
- package/dist/__tests__/semantic-enrichment.test.js +0 -360
- package/dist/__tests__/session-brief-builder.test.js +0 -187
- package/dist/__tests__/session-context.test.js +0 -221
- package/dist/__tests__/session-continuity.test.js +0 -144
- package/dist/__tests__/session-dedup.test.js +0 -74
- package/dist/__tests__/session-event-wiring.test.js +0 -206
- package/dist/__tests__/session-events.test.js +0 -149
- package/dist/__tests__/session-legend.test.js +0 -20
- package/dist/__tests__/session-persistence.test.js +0 -131
- package/dist/__tests__/session-resume-block.test.js +0 -107
- package/dist/__tests__/session-resume.test.js +0 -97
- package/dist/__tests__/session-summary-writer.test.js +0 -134
- package/dist/__tests__/shadow-ledger.test.js +0 -203
- package/dist/__tests__/shell-classifier.test.js +0 -151
- package/dist/__tests__/shell-compression-floor.test.js +0 -189
- package/dist/__tests__/shell-compression-v2.test.js +0 -339
- package/dist/__tests__/shell-compressor.test.js +0 -35
- package/dist/__tests__/shell-hooks.test.js +0 -128
- package/dist/__tests__/shell-strategies.test.js +0 -644
- package/dist/__tests__/shell-tee.test.js +0 -133
- package/dist/__tests__/signal-dedup.test.js +0 -158
- package/dist/__tests__/signal-reinforcer.test.js +0 -77
- package/dist/__tests__/signal-scorer.test.js +0 -251
- package/dist/__tests__/signal-show-store.test.js +0 -108
- package/dist/__tests__/smart-truncate.test.js +0 -215
- package/dist/__tests__/snapshot-v2.test.js +0 -113
- package/dist/__tests__/sprint-l1-local-mode.test.js +0 -130
- package/dist/__tests__/sprint-l10-boot.test.js +0 -220
- package/dist/__tests__/sprint-l9-offline-commands.test.js +0 -189
- package/dist/__tests__/sprint-q-persistent-context.test.js +0 -198
- package/dist/__tests__/sprint-s1-wiring.test.js +0 -215
- package/dist/__tests__/sprint-s2-wiring.test.js +0 -256
- package/dist/__tests__/sprint-s3-wiring.test.js +0 -195
- package/dist/__tests__/sprint-s4-wiring.test.js +0 -213
- package/dist/__tests__/sprint-s6-hooks.test.js +0 -222
- package/dist/__tests__/sprint-s7-persistent.test.js +0 -263
- package/dist/__tests__/sprint-s8-value.test.js +0 -167
- package/dist/__tests__/sprint-s9-behavioral.test.js +0 -179
- package/dist/__tests__/sprint3-intelligence.test.js +0 -297
- package/dist/__tests__/sprint5-mcp-server.test.js +0 -136
- package/dist/__tests__/startup-display.test.js +0 -302
- package/dist/__tests__/startup-log-file.test.js +0 -97
- package/dist/__tests__/stash-manager.test.js +0 -229
- package/dist/__tests__/state-detector.test.js +0 -92
- package/dist/__tests__/status-dashboard.test.js +0 -142
- package/dist/__tests__/temporal-facts.test.js +0 -292
- package/dist/__tests__/temporal-routes.test.js +0 -142
- package/dist/__tests__/test-detector.test.js +0 -174
- package/dist/__tests__/theme.test.js +0 -72
- package/dist/__tests__/timeline-agents.test.js +0 -122
- package/dist/__tests__/timeline-bootstrap.test.js +0 -176
- package/dist/__tests__/timeline-filters.test.js +0 -193
- package/dist/__tests__/timeline-markers.test.js +0 -151
- package/dist/__tests__/timeline-routes.test.js +0 -156
- package/dist/__tests__/timeline-store.test.js +0 -171
- package/dist/__tests__/token-counter.test.js +0 -86
- package/dist/__tests__/token-estimator.test.js +0 -96
- package/dist/__tests__/token-flow-api.test.js +0 -239
- package/dist/__tests__/token-flow-instrumentation.test.js +0 -437
- package/dist/__tests__/token-flow-persistence.test.js +0 -356
- package/dist/__tests__/token-flow-routes.test.js +0 -199
- package/dist/__tests__/token-flow.test.js +0 -695
- package/dist/__tests__/tool-clusters.test.js +0 -177
- package/dist/__tests__/transport-mux.test.js +0 -283
- package/dist/__tests__/turn-segmenter.test.js +0 -166
- package/dist/__tests__/uninstall.test.js +0 -141
- package/dist/__tests__/warm-start-policy.test.js +0 -271
- package/dist/__tests__/wire-cap-nudge.test.js +0 -77
- package/dist/__tests__/worker-pool.test.js +0 -101
- package/dist/ui/assets/index-7gl3mIuY.css +0 -1
- package/dist/ui/assets/index-CX4FCWGT.js +0 -10
- package/dist/ui/assets/rolldown-runtime-S-ySWqyJ.js +0 -1
- package/dist/ui/assets/vis-network-NIJHUFI3.js +0 -908
- package/dist/ui/fonts/jetbrains-mono-latin-400-normal.woff +0 -0
- package/dist/ui/icon-wordmark.png +0 -0
- package/dist/ui/icon-wordmark.svg +0 -30
- package/dist/ui/icon.png +0 -0
- package/dist/ui/icon.svg +0 -25
- package/dist/ui/index.html +0 -15
- package/dist/ui/prototype-sandbox/index.html +0 -257
- package/dist/ui/screenshots/activity.png +0 -0
- package/dist/ui/screenshots/code-base-intelligence.png +0 -0
- package/dist/ui/screenshots/dashboard.png +0 -0
- package/dist/ui/screenshots/project-memory.png +0 -0
- package/dist/ui/screenshots/reasoning-quality.png +0 -0
- package/dist/ui/screenshots/reasoning-session.png +0 -0
- package/dist/ui/screenshots/token-session.png +0 -0
- package/dist/ui/screenshots/token-trace-main.png +0 -0
- package/dist/ui/screenshots/token-turn.png +0 -0
- package/dist/ui/unerr-wordmark.png +0 -0
- package/dist/ui/unerr-wordmark.svg +0 -9
- package/dist/ui/unerr.png +0 -0
- package/dist/ui/unerr.svg +0 -25
- package/dist/ui/web-app-manifest-192x192.png +0 -0
- package/dist/ui/web-app-manifest-512x512.png +0 -0
|
@@ -1,174 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Sprint R.8: Test Detector Tests
|
|
3
|
-
*
|
|
4
|
-
* Verifies isTestFile() and resolveTestSubject() work across all supported languages.
|
|
5
|
-
*/
|
|
6
|
-
import { describe, expect, it } from "vitest";
|
|
7
|
-
import { isTestFile, resolveTestSubject, } from "../intelligence/indexer/test-detector.js";
|
|
8
|
-
describe("Sprint R.8: Test File Detection", () => {
|
|
9
|
-
describe("isTestFile — TypeScript/JavaScript", () => {
|
|
10
|
-
it("detects .test.ts files", () => {
|
|
11
|
-
expect(isTestFile("src/utils/exec.test.ts")).toBe(true);
|
|
12
|
-
expect(isTestFile("src/components/App.test.tsx")).toBe(true);
|
|
13
|
-
});
|
|
14
|
-
it("detects .spec.ts files", () => {
|
|
15
|
-
expect(isTestFile("src/utils/exec.spec.ts")).toBe(true);
|
|
16
|
-
expect(isTestFile("lib/helpers.spec.js")).toBe(true);
|
|
17
|
-
});
|
|
18
|
-
it("detects __tests__ directory", () => {
|
|
19
|
-
expect(isTestFile("src/__tests__/proxy.test.ts")).toBe(true);
|
|
20
|
-
expect(isTestFile("src/__tests__/utils.ts")).toBe(true);
|
|
21
|
-
});
|
|
22
|
-
it("detects test/ directory", () => {
|
|
23
|
-
expect(isTestFile("test/integration/auth.ts")).toBe(true);
|
|
24
|
-
expect(isTestFile("tests/unit/helpers.js")).toBe(true);
|
|
25
|
-
});
|
|
26
|
-
it("rejects source files", () => {
|
|
27
|
-
expect(isTestFile("src/utils/exec.ts")).toBe(false);
|
|
28
|
-
expect(isTestFile("src/proxy/proxy.ts")).toBe(false);
|
|
29
|
-
expect(isTestFile("src/components/App.tsx")).toBe(false);
|
|
30
|
-
});
|
|
31
|
-
});
|
|
32
|
-
describe("isTestFile — Python", () => {
|
|
33
|
-
it("detects test_ prefix", () => {
|
|
34
|
-
expect(isTestFile("tests/test_models.py")).toBe(true);
|
|
35
|
-
expect(isTestFile("test_auth.py")).toBe(true);
|
|
36
|
-
});
|
|
37
|
-
it("detects _test suffix", () => {
|
|
38
|
-
expect(isTestFile("app/models/user_test.py")).toBe(true);
|
|
39
|
-
});
|
|
40
|
-
it("detects conftest.py", () => {
|
|
41
|
-
expect(isTestFile("tests/conftest.py")).toBe(true);
|
|
42
|
-
expect(isTestFile("conftest.py")).toBe(true);
|
|
43
|
-
});
|
|
44
|
-
it("rejects source files", () => {
|
|
45
|
-
expect(isTestFile("app/models/user.py")).toBe(false);
|
|
46
|
-
expect(isTestFile("app/services/auth.py")).toBe(false);
|
|
47
|
-
});
|
|
48
|
-
});
|
|
49
|
-
describe("isTestFile — Go", () => {
|
|
50
|
-
it("detects _test.go suffix", () => {
|
|
51
|
-
expect(isTestFile("pkg/handlers/auth_test.go")).toBe(true);
|
|
52
|
-
expect(isTestFile("internal/db/connection_test.go")).toBe(true);
|
|
53
|
-
});
|
|
54
|
-
it("rejects source files", () => {
|
|
55
|
-
expect(isTestFile("pkg/handlers/auth.go")).toBe(false);
|
|
56
|
-
expect(isTestFile("cmd/main.go")).toBe(false);
|
|
57
|
-
});
|
|
58
|
-
});
|
|
59
|
-
describe("isTestFile — Java", () => {
|
|
60
|
-
it("detects Test suffix", () => {
|
|
61
|
-
expect(isTestFile("src/test/java/com/example/UserServiceTest.java")).toBe(true);
|
|
62
|
-
expect(isTestFile("com/example/AuthTest.java")).toBe(true);
|
|
63
|
-
});
|
|
64
|
-
it("detects Tests suffix", () => {
|
|
65
|
-
expect(isTestFile("com/example/UserTests.java")).toBe(true);
|
|
66
|
-
});
|
|
67
|
-
it("detects IT suffix (integration test)", () => {
|
|
68
|
-
expect(isTestFile("com/example/AuthIT.java")).toBe(true);
|
|
69
|
-
});
|
|
70
|
-
it("detects src/test/ directory", () => {
|
|
71
|
-
expect(isTestFile("src/test/java/com/example/Helper.java")).toBe(true);
|
|
72
|
-
});
|
|
73
|
-
it("rejects source files", () => {
|
|
74
|
-
expect(isTestFile("src/main/java/com/example/UserService.java")).toBe(false);
|
|
75
|
-
expect(isTestFile("com/example/models/User.java")).toBe(false);
|
|
76
|
-
});
|
|
77
|
-
});
|
|
78
|
-
describe("isTestFile — Rust", () => {
|
|
79
|
-
it("detects _test.rs suffix", () => {
|
|
80
|
-
expect(isTestFile("src/auth_test.rs")).toBe(true);
|
|
81
|
-
});
|
|
82
|
-
it("detects tests/ directory", () => {
|
|
83
|
-
expect(isTestFile("tests/integration/auth.rs")).toBe(true);
|
|
84
|
-
expect(isTestFile("tests/common.rs")).toBe(true);
|
|
85
|
-
});
|
|
86
|
-
it("rejects source files", () => {
|
|
87
|
-
expect(isTestFile("src/auth.rs")).toBe(false);
|
|
88
|
-
expect(isTestFile("src/lib.rs")).toBe(false);
|
|
89
|
-
});
|
|
90
|
-
});
|
|
91
|
-
describe("isTestFile — Ruby", () => {
|
|
92
|
-
it("detects _spec.rb suffix", () => {
|
|
93
|
-
expect(isTestFile("spec/models/user_spec.rb")).toBe(true);
|
|
94
|
-
});
|
|
95
|
-
it("detects _test.rb suffix", () => {
|
|
96
|
-
expect(isTestFile("test/models/user_test.rb")).toBe(true);
|
|
97
|
-
});
|
|
98
|
-
it("detects spec/ directory", () => {
|
|
99
|
-
expect(isTestFile("spec/helpers/auth.rb")).toBe(true);
|
|
100
|
-
});
|
|
101
|
-
it("rejects source files", () => {
|
|
102
|
-
expect(isTestFile("lib/models/user.rb")).toBe(false);
|
|
103
|
-
expect(isTestFile("app/services/auth.rb")).toBe(false);
|
|
104
|
-
});
|
|
105
|
-
});
|
|
106
|
-
describe("isTestFile — C#", () => {
|
|
107
|
-
it("detects Tests suffix", () => {
|
|
108
|
-
expect(isTestFile("Tests/UserServiceTests.cs")).toBe(true);
|
|
109
|
-
});
|
|
110
|
-
it("detects Test suffix", () => {
|
|
111
|
-
expect(isTestFile("UserServiceTest.cs")).toBe(true);
|
|
112
|
-
});
|
|
113
|
-
it("rejects source files", () => {
|
|
114
|
-
expect(isTestFile("Services/UserService.cs")).toBe(false);
|
|
115
|
-
expect(isTestFile("Models/User.cs")).toBe(false);
|
|
116
|
-
});
|
|
117
|
-
});
|
|
118
|
-
describe("isTestFile — C/C++", () => {
|
|
119
|
-
it("detects _test suffix", () => {
|
|
120
|
-
expect(isTestFile("src/auth_test.cpp")).toBe(true);
|
|
121
|
-
expect(isTestFile("lib/parser_test.c")).toBe(true);
|
|
122
|
-
});
|
|
123
|
-
it("detects test_ prefix", () => {
|
|
124
|
-
expect(isTestFile("test_parser.cpp")).toBe(true);
|
|
125
|
-
});
|
|
126
|
-
it("rejects source files", () => {
|
|
127
|
-
expect(isTestFile("src/auth.cpp")).toBe(false);
|
|
128
|
-
expect(isTestFile("lib/parser.c")).toBe(false);
|
|
129
|
-
});
|
|
130
|
-
});
|
|
131
|
-
});
|
|
132
|
-
describe("Sprint R.8: resolveTestSubject", () => {
|
|
133
|
-
const projectFiles = new Set([
|
|
134
|
-
"src/utils/exec.ts",
|
|
135
|
-
"src/proxy/proxy.ts",
|
|
136
|
-
"src/exec.ts",
|
|
137
|
-
"src/__tests__/exec.test.ts",
|
|
138
|
-
"pkg/handlers/auth.go",
|
|
139
|
-
"pkg/handlers/auth_test.go",
|
|
140
|
-
"app/models/user.py",
|
|
141
|
-
"user.py",
|
|
142
|
-
"tests/test_user.py",
|
|
143
|
-
"src/main/java/com/example/UserService.java",
|
|
144
|
-
"src/test/java/com/example/UserServiceTest.java",
|
|
145
|
-
"lib/models/user.rb",
|
|
146
|
-
"models/user.rb",
|
|
147
|
-
"spec/models/user_spec.rb",
|
|
148
|
-
]);
|
|
149
|
-
it("resolves TS test → source (same dir)", () => {
|
|
150
|
-
expect(resolveTestSubject("src/utils/exec.test.ts", projectFiles)).toBe("src/utils/exec.ts");
|
|
151
|
-
});
|
|
152
|
-
it("resolves TS test → source (__tests__ dir → parent)", () => {
|
|
153
|
-
// __tests__/exec.test.ts → parent dir is src/, so resolves to src/exec.ts
|
|
154
|
-
expect(resolveTestSubject("src/__tests__/exec.test.ts", projectFiles)).toBe("src/exec.ts");
|
|
155
|
-
});
|
|
156
|
-
it("resolves Go test → source", () => {
|
|
157
|
-
expect(resolveTestSubject("pkg/handlers/auth_test.go", projectFiles)).toBe("pkg/handlers/auth.go");
|
|
158
|
-
});
|
|
159
|
-
it("resolves Python test_ prefix → source (parent dir)", () => {
|
|
160
|
-
// tests/test_user.py → tries tests/user.py then user.py (parent strip)
|
|
161
|
-
expect(resolveTestSubject("tests/test_user.py", projectFiles)).toBe("user.py");
|
|
162
|
-
});
|
|
163
|
-
it("resolves Java Test suffix → source (Maven convention)", () => {
|
|
164
|
-
// src/test/java/... → src/main/java/...
|
|
165
|
-
expect(resolveTestSubject("src/test/java/com/example/UserServiceTest.java", projectFiles)).toBe("src/main/java/com/example/UserService.java");
|
|
166
|
-
});
|
|
167
|
-
it("resolves Ruby _spec → source (spec/ → lib/)", () => {
|
|
168
|
-
// spec/models/user_spec.rb → tries models/user.rb (strip spec/)
|
|
169
|
-
expect(resolveTestSubject("spec/models/user_spec.rb", projectFiles)).toBe("models/user.rb");
|
|
170
|
-
});
|
|
171
|
-
it("returns null for unresolvable", () => {
|
|
172
|
-
expect(resolveTestSubject("test/unknown_test.go", projectFiles)).toBeNull();
|
|
173
|
-
});
|
|
174
|
-
});
|
|
@@ -1,72 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Tests for Theme.tsx (Task 1.1) — Ink foundation and theme system.
|
|
3
|
-
*
|
|
4
|
-
* Tests validate:
|
|
5
|
-
* - theme: semantic color palette with correct values
|
|
6
|
-
* - theme.grade(): letter-to-color mapping (A=green, C=yellow, F=red)
|
|
7
|
-
* - renderToStderr(): renders to stderr, not stdout
|
|
8
|
-
*/
|
|
9
|
-
import { Text } from "ink";
|
|
10
|
-
import { render } from "ink-testing-library";
|
|
11
|
-
import React from "react";
|
|
12
|
-
import { describe, expect, it } from "vitest";
|
|
13
|
-
import { ThemeProvider, theme, useTheme } from "../components/Theme.js";
|
|
14
|
-
describe("Theme System (1.1)", () => {
|
|
15
|
-
// ── theme object ───────────────────────────────────────────────
|
|
16
|
-
describe("theme", () => {
|
|
17
|
-
it("has all semantic color keys", () => {
|
|
18
|
-
expect(theme.success).toBe("green");
|
|
19
|
-
expect(theme.warning).toBe("yellow");
|
|
20
|
-
expect(theme.error).toBe("red");
|
|
21
|
-
expect(theme.info).toBe("cyan");
|
|
22
|
-
expect(theme.accent).toBe("magenta");
|
|
23
|
-
expect(theme.dim).toBe("gray");
|
|
24
|
-
expect(theme.brand).toBe("cyanBright");
|
|
25
|
-
});
|
|
26
|
-
it("grade() maps A to green", () => {
|
|
27
|
-
expect(theme.grade("A")).toBe("green");
|
|
28
|
-
});
|
|
29
|
-
it("grade() maps B+ to cyan (first char B)", () => {
|
|
30
|
-
expect(theme.grade("B+")).toBe("cyan");
|
|
31
|
-
expect(theme.grade("B")).toBe("cyan");
|
|
32
|
-
});
|
|
33
|
-
it("grade() maps C+ to yellow", () => {
|
|
34
|
-
expect(theme.grade("C+")).toBe("yellow");
|
|
35
|
-
expect(theme.grade("C")).toBe("yellow");
|
|
36
|
-
});
|
|
37
|
-
it("grade() maps D to red", () => {
|
|
38
|
-
expect(theme.grade("D")).toBe("red");
|
|
39
|
-
});
|
|
40
|
-
it("grade() maps F to red", () => {
|
|
41
|
-
expect(theme.grade("F")).toBe("red");
|
|
42
|
-
});
|
|
43
|
-
it("grade() returns gray for unknown grades", () => {
|
|
44
|
-
expect(theme.grade("X")).toBe("gray");
|
|
45
|
-
expect(theme.grade("")).toBe("gray");
|
|
46
|
-
});
|
|
47
|
-
});
|
|
48
|
-
// ── ThemeProvider + useTheme ────────────────────────────────────
|
|
49
|
-
describe("ThemeProvider + useTheme", () => {
|
|
50
|
-
function ThemeConsumer() {
|
|
51
|
-
const t = useTheme();
|
|
52
|
-
return React.createElement(Text, null, `success=${t.success}`);
|
|
53
|
-
}
|
|
54
|
-
it("provides theme to child components", () => {
|
|
55
|
-
const { lastFrame } = render(React.createElement(ThemeProvider, null, React.createElement(ThemeConsumer)));
|
|
56
|
-
expect(lastFrame()).toContain("success=green");
|
|
57
|
-
});
|
|
58
|
-
it("useTheme works without provider (uses defaults)", () => {
|
|
59
|
-
const { lastFrame } = render(React.createElement(ThemeConsumer));
|
|
60
|
-
expect(lastFrame()).toContain("success=green");
|
|
61
|
-
});
|
|
62
|
-
});
|
|
63
|
-
// ── renderToStderr ─────────────────────────────────────────────
|
|
64
|
-
describe("renderToStderr", () => {
|
|
65
|
-
it("is exported and callable", async () => {
|
|
66
|
-
// We can't easily test stderr rendering in vitest without mocking process.stderr,
|
|
67
|
-
// but we verify the function exists and has the right signature
|
|
68
|
-
const { renderToStderr } = await import("../components/render.js");
|
|
69
|
-
expect(typeof renderToStderr).toBe("function");
|
|
70
|
-
});
|
|
71
|
-
});
|
|
72
|
-
});
|
|
@@ -1,122 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* UX-1 / UX-3: session_agents store + /agents route + /sessions agent_name.
|
|
3
|
-
*/
|
|
4
|
-
import { mkdirSync, rmSync } from "node:fs";
|
|
5
|
-
import { tmpdir } from "node:os";
|
|
6
|
-
import { join } from "node:path";
|
|
7
|
-
import { afterEach, beforeEach, describe, expect, it } from "vitest";
|
|
8
|
-
import { createTimelineRoutes } from "../server/routes/timeline.js";
|
|
9
|
-
import { CozoTimelineStore } from "../timeline/timeline-store.js";
|
|
10
|
-
let tempDir;
|
|
11
|
-
let store;
|
|
12
|
-
beforeEach(async () => {
|
|
13
|
-
tempDir = join(tmpdir(), `unerr-agent-test-${Date.now()}-${Math.random().toString(36).slice(2)}`);
|
|
14
|
-
mkdirSync(join(tempDir, ".unerr"), { recursive: true });
|
|
15
|
-
store = await CozoTimelineStore.create(tempDir);
|
|
16
|
-
});
|
|
17
|
-
afterEach(() => {
|
|
18
|
-
try {
|
|
19
|
-
store.close();
|
|
20
|
-
}
|
|
21
|
-
catch {
|
|
22
|
-
/* ignore */
|
|
23
|
-
}
|
|
24
|
-
try {
|
|
25
|
-
rmSync(tempDir, { recursive: true, force: true });
|
|
26
|
-
}
|
|
27
|
-
catch {
|
|
28
|
-
/* ignore */
|
|
29
|
-
}
|
|
30
|
-
});
|
|
31
|
-
function turn(overrides) {
|
|
32
|
-
return {
|
|
33
|
-
session_id: overrides.session_id ?? "s1",
|
|
34
|
-
started_at: overrides.started_at ?? 1_000,
|
|
35
|
-
ended_at: overrides.ended_at ?? 2_000,
|
|
36
|
-
opened_by: overrides.opened_by ?? "first_call",
|
|
37
|
-
closed_reason: overrides.closed_reason ?? "session_end",
|
|
38
|
-
tool_count: overrides.tool_count ?? 3,
|
|
39
|
-
file_count: overrides.file_count ?? 2,
|
|
40
|
-
edit_count: overrides.edit_count ?? 1,
|
|
41
|
-
title: overrides.title ?? "",
|
|
42
|
-
outcome: overrides.outcome ?? "unknown",
|
|
43
|
-
...overrides,
|
|
44
|
-
};
|
|
45
|
-
}
|
|
46
|
-
describe("session_agents store", () => {
|
|
47
|
-
it("setSessionAgent / getSessionAgent round-trip", async () => {
|
|
48
|
-
await store.setSessionAgent("sA", "claude-code", 1_000);
|
|
49
|
-
expect(await store.getSessionAgent("sA")).toBe("claude-code");
|
|
50
|
-
expect(await store.getSessionAgent("sB")).toBeNull();
|
|
51
|
-
});
|
|
52
|
-
it("setSessionAgent is a no-op for empty agent name", async () => {
|
|
53
|
-
await store.setSessionAgent("sA", "", 1_000);
|
|
54
|
-
expect(await store.getSessionAgent("sA")).toBeNull();
|
|
55
|
-
});
|
|
56
|
-
it("setSessionAgent preserves first_seen across refresh", async () => {
|
|
57
|
-
await store.setSessionAgent("sA", "claude-code", 1_000);
|
|
58
|
-
await store.setSessionAgent("sA", "claude-code", 5_000);
|
|
59
|
-
const result = await store
|
|
60
|
-
.getDb()
|
|
61
|
-
.run("?[first_seen, last_seen] := *session_agents{session_id, first_seen, last_seen}, session_id = $sid", { sid: "sA" });
|
|
62
|
-
const row = result.rows[0];
|
|
63
|
-
expect(row[0]).toBe(1_000);
|
|
64
|
-
expect(row[1]).toBe(5_000);
|
|
65
|
-
});
|
|
66
|
-
it("getSessionAgents batch lookup returns only requested ids", async () => {
|
|
67
|
-
await store.setSessionAgent("sA", "claude-code", 1_000);
|
|
68
|
-
await store.setSessionAgent("sB", "cursor", 2_000);
|
|
69
|
-
await store.setSessionAgent("sC", "codex", 3_000);
|
|
70
|
-
const map = await store.getSessionAgents(["sA", "sC"]);
|
|
71
|
-
expect(map.get("sA")).toBe("claude-code");
|
|
72
|
-
expect(map.get("sC")).toBe("codex");
|
|
73
|
-
expect(map.has("sB")).toBe(false);
|
|
74
|
-
});
|
|
75
|
-
it("listAgents returns distinct agent names with session counts", async () => {
|
|
76
|
-
await store.setSessionAgent("s1", "claude-code", 1_000);
|
|
77
|
-
await store.setSessionAgent("s2", "claude-code", 2_000);
|
|
78
|
-
await store.setSessionAgent("s3", "cursor", 3_000);
|
|
79
|
-
const agents = await store.listAgents();
|
|
80
|
-
const cc = agents.find((a) => a.agent_name === "claude-code");
|
|
81
|
-
const cu = agents.find((a) => a.agent_name === "cursor");
|
|
82
|
-
expect(cc.session_count).toBe(2);
|
|
83
|
-
expect(cu.session_count).toBe(1);
|
|
84
|
-
// Ordering: most-recently-seen first.
|
|
85
|
-
expect(agents[0]?.agent_name).toBe("cursor");
|
|
86
|
-
});
|
|
87
|
-
it("listSessions joins agent_name onto each row", async () => {
|
|
88
|
-
await store.upsertTurn(turn({ turn_id: "t1", session_id: "sa" }));
|
|
89
|
-
await store.upsertTurn(turn({ turn_id: "t2", session_id: "sb", started_at: 2_000 }));
|
|
90
|
-
await store.setSessionAgent("sa", "claude-code", 1_500);
|
|
91
|
-
const list = await store.listSessions();
|
|
92
|
-
const a = list.find((s) => s.session_id === "sa");
|
|
93
|
-
const b = list.find((s) => s.session_id === "sb");
|
|
94
|
-
expect(a?.agent_name).toBe("claude-code");
|
|
95
|
-
expect(b?.agent_name).toBe("unknown");
|
|
96
|
-
});
|
|
97
|
-
});
|
|
98
|
-
describe("HTTP routes — /agents + /sessions?agent=", () => {
|
|
99
|
-
it("GET /agents returns the agent inventory", async () => {
|
|
100
|
-
await store.setSessionAgent("s1", "claude-code", 1_000);
|
|
101
|
-
await store.setSessionAgent("s2", "cursor", 2_000);
|
|
102
|
-
const app = createTimelineRoutes({ store });
|
|
103
|
-
const res = await app.request("/agents");
|
|
104
|
-
const body = (await res.json());
|
|
105
|
-
expect(body.total).toBe(2);
|
|
106
|
-
expect(body.data.map((a) => a.agent_name).sort()).toEqual([
|
|
107
|
-
"claude-code",
|
|
108
|
-
"cursor",
|
|
109
|
-
]);
|
|
110
|
-
});
|
|
111
|
-
it("GET /sessions?agent=claude-code filters by agent", async () => {
|
|
112
|
-
await store.upsertTurn(turn({ turn_id: "t1", session_id: "sa" }));
|
|
113
|
-
await store.upsertTurn(turn({ turn_id: "t2", session_id: "sb", started_at: 2_000 }));
|
|
114
|
-
await store.setSessionAgent("sa", "claude-code", 1_500);
|
|
115
|
-
await store.setSessionAgent("sb", "cursor", 2_500);
|
|
116
|
-
const app = createTimelineRoutes({ store });
|
|
117
|
-
const res = await app.request("/sessions?agent=cursor");
|
|
118
|
-
const body = (await res.json());
|
|
119
|
-
expect(body.total).toBe(1);
|
|
120
|
-
expect(body.data[0]?.session_id).toBe("sb");
|
|
121
|
-
});
|
|
122
|
-
});
|
|
@@ -1,176 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* ST-1c: Timeline bootstrap — kill-switch + turn-close → turn rollup wiring.
|
|
3
|
-
*/
|
|
4
|
-
import { existsSync, mkdirSync, rmSync } from "node:fs";
|
|
5
|
-
import { tmpdir } from "node:os";
|
|
6
|
-
import { join } from "node:path";
|
|
7
|
-
import { afterEach, beforeEach, describe, expect, it } from "vitest";
|
|
8
|
-
import { computeTurnRollup, startTimelineBootstrap, } from "../timeline/timeline-bootstrap.js";
|
|
9
|
-
import { CozoTimelineStore } from "../timeline/timeline-store.js";
|
|
10
|
-
import { ShadowLedger } from "../tracking/shadow-ledger.js";
|
|
11
|
-
let tempDir;
|
|
12
|
-
beforeEach(() => {
|
|
13
|
-
tempDir = join(tmpdir(), `unerr-tb-test-${Date.now()}-${Math.random().toString(36).slice(2)}`);
|
|
14
|
-
mkdirSync(join(tempDir, ".unerr"), { recursive: true });
|
|
15
|
-
process.env.UNERR_TIMELINE_V2 = undefined;
|
|
16
|
-
});
|
|
17
|
-
afterEach(() => {
|
|
18
|
-
try {
|
|
19
|
-
rmSync(tempDir, { recursive: true, force: true });
|
|
20
|
-
}
|
|
21
|
-
catch {
|
|
22
|
-
/* ignore */
|
|
23
|
-
}
|
|
24
|
-
process.env.UNERR_TIMELINE_V2 = undefined;
|
|
25
|
-
});
|
|
26
|
-
describe("computeTurnRollup", () => {
|
|
27
|
-
it("aggregates entries into a turn row with intent title", () => {
|
|
28
|
-
const event = {
|
|
29
|
-
turn_id: "T1",
|
|
30
|
-
session_id: "S1",
|
|
31
|
-
closed_at: 5_000,
|
|
32
|
-
reason: "idle_gap",
|
|
33
|
-
};
|
|
34
|
-
const entries = [
|
|
35
|
-
{
|
|
36
|
-
id: "1",
|
|
37
|
-
ts: "2026-05-12T10:00:00.000Z",
|
|
38
|
-
tool: "mark_intent",
|
|
39
|
-
args_summary: { text: "refactor auth" },
|
|
40
|
-
result_summary: {},
|
|
41
|
-
branch: "main",
|
|
42
|
-
head_sha: "x",
|
|
43
|
-
session_id: "S1",
|
|
44
|
-
correlation_id: null,
|
|
45
|
-
turn_id: "T1",
|
|
46
|
-
turn_confidence: "first_call",
|
|
47
|
-
},
|
|
48
|
-
{
|
|
49
|
-
id: "2",
|
|
50
|
-
ts: "2026-05-12T10:00:02.000Z",
|
|
51
|
-
tool: "file_read",
|
|
52
|
-
args_summary: { file_path: "src/auth.ts" },
|
|
53
|
-
result_summary: {},
|
|
54
|
-
branch: "main",
|
|
55
|
-
head_sha: "x",
|
|
56
|
-
session_id: "S1",
|
|
57
|
-
correlation_id: "1",
|
|
58
|
-
turn_id: "T1",
|
|
59
|
-
turn_confidence: "first_call",
|
|
60
|
-
},
|
|
61
|
-
{
|
|
62
|
-
id: "3",
|
|
63
|
-
ts: "2026-05-12T10:00:04.000Z",
|
|
64
|
-
tool: "Edit",
|
|
65
|
-
args_summary: { file_path: "src/auth.ts" },
|
|
66
|
-
result_summary: {},
|
|
67
|
-
branch: "main",
|
|
68
|
-
head_sha: "x",
|
|
69
|
-
session_id: "S1",
|
|
70
|
-
correlation_id: "1",
|
|
71
|
-
turn_id: "T1",
|
|
72
|
-
turn_confidence: "first_call",
|
|
73
|
-
},
|
|
74
|
-
];
|
|
75
|
-
const rollup = computeTurnRollup(event, entries);
|
|
76
|
-
expect(rollup.turn_id).toBe("T1");
|
|
77
|
-
expect(rollup.session_id).toBe("S1");
|
|
78
|
-
expect(rollup.tool_count).toBe(3);
|
|
79
|
-
expect(rollup.file_count).toBe(1);
|
|
80
|
-
expect(rollup.edit_count).toBe(1);
|
|
81
|
-
expect(rollup.title).toBe("refactor auth");
|
|
82
|
-
expect(rollup.opened_by).toBe("first_call");
|
|
83
|
-
expect(rollup.closed_reason).toBe("idle_gap");
|
|
84
|
-
});
|
|
85
|
-
it("falls back to basename when no mark_intent present", () => {
|
|
86
|
-
const event = {
|
|
87
|
-
turn_id: "T2",
|
|
88
|
-
session_id: "S1",
|
|
89
|
-
closed_at: 100,
|
|
90
|
-
reason: "stop_hook",
|
|
91
|
-
};
|
|
92
|
-
const entries = [
|
|
93
|
-
{
|
|
94
|
-
id: "1",
|
|
95
|
-
ts: "2026-05-12T10:00:00.000Z",
|
|
96
|
-
tool: "file_read",
|
|
97
|
-
args_summary: { file_path: "src/auth.ts" },
|
|
98
|
-
result_summary: {},
|
|
99
|
-
branch: "main",
|
|
100
|
-
head_sha: "x",
|
|
101
|
-
session_id: "S1",
|
|
102
|
-
correlation_id: null,
|
|
103
|
-
turn_id: "T2",
|
|
104
|
-
turn_confidence: "first_call",
|
|
105
|
-
},
|
|
106
|
-
];
|
|
107
|
-
const rollup = computeTurnRollup(event, entries);
|
|
108
|
-
expect(rollup.title).toBe("auth.ts");
|
|
109
|
-
});
|
|
110
|
-
});
|
|
111
|
-
describe("startTimelineBootstrap", () => {
|
|
112
|
-
it("returns null and does NOT create timeline.db when UNERR_TIMELINE_V2=0", async () => {
|
|
113
|
-
process.env.UNERR_TIMELINE_V2 = "0";
|
|
114
|
-
const ledger = new ShadowLedger(join(tempDir, ".unerr"));
|
|
115
|
-
const handle = await startTimelineBootstrap({
|
|
116
|
-
projectRoot: tempDir,
|
|
117
|
-
ledger,
|
|
118
|
-
log: () => { },
|
|
119
|
-
});
|
|
120
|
-
expect(handle).toBeNull();
|
|
121
|
-
expect(existsSync(join(tempDir, ".unerr", "timeline.db"))).toBe(false);
|
|
122
|
-
});
|
|
123
|
-
it("opens timeline.db and upserts a turn when the segmenter closes one", async () => {
|
|
124
|
-
const ledger = new ShadowLedger(join(tempDir, ".unerr"));
|
|
125
|
-
const handle = await startTimelineBootstrap({
|
|
126
|
-
projectRoot: tempDir,
|
|
127
|
-
ledger,
|
|
128
|
-
log: () => { },
|
|
129
|
-
});
|
|
130
|
-
expect(handle).not.toBeNull();
|
|
131
|
-
if (!handle)
|
|
132
|
-
return;
|
|
133
|
-
try {
|
|
134
|
-
ledger.record("file_read", { file_path: "src/a.ts" }, {}, "main", "deadbeef");
|
|
135
|
-
ledger.record("search_code", { query: "foo" }, {}, "main", "deadbeef");
|
|
136
|
-
ledger.closeTurn("session_end");
|
|
137
|
-
// Give the async upsertTurn microtask a tick to flush.
|
|
138
|
-
await new Promise((r) => setTimeout(r, 30));
|
|
139
|
-
const store = handle.store;
|
|
140
|
-
const turns = await store.listTurns();
|
|
141
|
-
expect(turns).toHaveLength(1);
|
|
142
|
-
expect(turns[0]?.session_id).toBe(ledger.getSessionId());
|
|
143
|
-
expect(turns[0]?.tool_count).toBe(2);
|
|
144
|
-
expect(turns[0]?.file_count).toBe(1);
|
|
145
|
-
expect(turns[0]?.closed_reason).toBe("session_end");
|
|
146
|
-
}
|
|
147
|
-
finally {
|
|
148
|
-
handle.stop();
|
|
149
|
-
}
|
|
150
|
-
});
|
|
151
|
-
it("stop() releases the store and detaches the listener", async () => {
|
|
152
|
-
const ledger = new ShadowLedger(join(tempDir, ".unerr"));
|
|
153
|
-
const handle = await startTimelineBootstrap({
|
|
154
|
-
projectRoot: tempDir,
|
|
155
|
-
ledger,
|
|
156
|
-
log: () => { },
|
|
157
|
-
});
|
|
158
|
-
if (!handle)
|
|
159
|
-
throw new Error("expected handle");
|
|
160
|
-
handle.stop();
|
|
161
|
-
// Subsequent close should no-op (listener already detached).
|
|
162
|
-
ledger.record("file_read", { file_path: "src/b.ts" }, {}, "m", "x");
|
|
163
|
-
ledger.closeTurn();
|
|
164
|
-
// No error should propagate; store is closed, so we just verify
|
|
165
|
-
// listTurns on a NEW store opening the same db still works (re-open path).
|
|
166
|
-
const reopened = await CozoTimelineStore.create(tempDir);
|
|
167
|
-
try {
|
|
168
|
-
// Was nothing inserted after stop (listener detached).
|
|
169
|
-
const turns = await reopened.listTurns();
|
|
170
|
-
expect(turns).toEqual([]);
|
|
171
|
-
}
|
|
172
|
-
finally {
|
|
173
|
-
reopened.close();
|
|
174
|
-
}
|
|
175
|
-
});
|
|
176
|
-
});
|