@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.
Files changed (218) hide show
  1. package/README.md +70 -194
  2. package/dist/cli.js +39149 -36991
  3. package/package.json +9 -2
  4. package/dist/__tests__/architecture-guard.test.js +0 -122
  5. package/dist/__tests__/arg-validator.test.js +0 -205
  6. package/dist/__tests__/ast-extractor.test.js +0 -203
  7. package/dist/__tests__/auto-bootstrap.test.js +0 -280
  8. package/dist/__tests__/background-indexer.test.js +0 -228
  9. package/dist/__tests__/blast-radius-engine.test.js +0 -200
  10. package/dist/__tests__/bridge-isolation.test.js +0 -37
  11. package/dist/__tests__/budget-enforcer.test.js +0 -53
  12. package/dist/__tests__/cfg-test-detection-perf.test.js +0 -82
  13. package/dist/__tests__/change-narrative.test.js +0 -190
  14. package/dist/__tests__/check-commit.test.js +0 -258
  15. package/dist/__tests__/checksum.test.js +0 -34
  16. package/dist/__tests__/commit-watcher.test.js +0 -154
  17. package/dist/__tests__/community-detection.test.js +0 -179
  18. package/dist/__tests__/community-tools.test.js +0 -299
  19. package/dist/__tests__/components.test.js +0 -449
  20. package/dist/__tests__/compression-log.test.js +0 -174
  21. package/dist/__tests__/compression-quality-monitor.test.js +0 -40
  22. package/dist/__tests__/config-healer.test.js +0 -165
  23. package/dist/__tests__/context-ledger.test.js +0 -58
  24. package/dist/__tests__/convention-detector.test.js +0 -99
  25. package/dist/__tests__/convention-learner.test.js +0 -86
  26. package/dist/__tests__/correction-detector.test.js +0 -330
  27. package/dist/__tests__/daemon-autostart-install.test.js +0 -283
  28. package/dist/__tests__/daemon-bridge.test.js +0 -222
  29. package/dist/__tests__/daemon-dashboard.test.js +0 -202
  30. package/dist/__tests__/daemon-registry.test.js +0 -240
  31. package/dist/__tests__/daemon-supervisor.test.js +0 -318
  32. package/dist/__tests__/daemon-version-check.test.js +0 -275
  33. package/dist/__tests__/decision-point-detector.test.js +0 -98
  34. package/dist/__tests__/deep-link.test.js +0 -143
  35. package/dist/__tests__/disallowed-tools.test.js +0 -115
  36. package/dist/__tests__/drift-tracker.test.js +0 -582
  37. package/dist/__tests__/durability-scorer.test.js +0 -152
  38. package/dist/__tests__/efficiency-tracker.test.js +0 -65
  39. package/dist/__tests__/enrich.test.js +0 -144
  40. package/dist/__tests__/entity-rewind.test.js +0 -248
  41. package/dist/__tests__/ephemeral.test.js +0 -111
  42. package/dist/__tests__/exploration-cost.test.js +0 -93
  43. package/dist/__tests__/fact-generator.test.js +0 -197
  44. package/dist/__tests__/file-l0-graph.test.js +0 -244
  45. package/dist/__tests__/file-logger.test.js +0 -82
  46. package/dist/__tests__/file-outline.test.js +0 -141
  47. package/dist/__tests__/file-read-protocol.test.js +0 -188
  48. package/dist/__tests__/format-encoder.test.js +0 -233
  49. package/dist/__tests__/git-attribution.test.js +0 -259
  50. package/dist/__tests__/graph-temporal-joiner.test.js +0 -219
  51. package/dist/__tests__/health-grade-enhanced.test.js +0 -138
  52. package/dist/__tests__/health-map-data.test.js +0 -173
  53. package/dist/__tests__/helpers/mcp-harness.js +0 -45
  54. package/dist/__tests__/helpers/mcp-harness.test.js +0 -68
  55. package/dist/__tests__/hook-dedup.test.js +0 -112
  56. package/dist/__tests__/hook-runner.test.js +0 -253
  57. package/dist/__tests__/indexer-cfg.test.js +0 -185
  58. package/dist/__tests__/indexer-cross-file.test.js +0 -172
  59. package/dist/__tests__/indexer-extraction.test.js +0 -245
  60. package/dist/__tests__/indexer-incremental.test.js +0 -232
  61. package/dist/__tests__/indexer-language-expansion.test.js +0 -165
  62. package/dist/__tests__/init-push.test.js +0 -131
  63. package/dist/__tests__/instruction-writer.test.js +0 -179
  64. package/dist/__tests__/intelligence-integration.test.js +0 -217
  65. package/dist/__tests__/intent-correlator.test.js +0 -175
  66. package/dist/__tests__/intent-detector.test.js +0 -235
  67. package/dist/__tests__/intent-encoder.test.js +0 -167
  68. package/dist/__tests__/java-build-tool-detection.test.js +0 -174
  69. package/dist/__tests__/layer3-sprint-q.test.js +0 -160
  70. package/dist/__tests__/layer3-sprint-r.test.js +0 -91
  71. package/dist/__tests__/layer3-sprint-s.test.js +0 -183
  72. package/dist/__tests__/layer3-sprint-t.test.js +0 -201
  73. package/dist/__tests__/layer3-sprint-u.test.js +0 -174
  74. package/dist/__tests__/layer4-sprint-ba2.test.js +0 -354
  75. package/dist/__tests__/layer4-sprint-ba4.test.js +0 -84
  76. package/dist/__tests__/layer4-sprint-vs.test.js +0 -105
  77. package/dist/__tests__/ledger-chains.test.js +0 -162
  78. package/dist/__tests__/lifecycle-machine.test.js +0 -226
  79. package/dist/__tests__/local-chat-provider.test.js +0 -170
  80. package/dist/__tests__/local-convention-detector.test.js +0 -308
  81. package/dist/__tests__/local-embeddings.test.js +0 -422
  82. package/dist/__tests__/local-graph.test.js +0 -540
  83. package/dist/__tests__/local-indexer.test.js +0 -228
  84. package/dist/__tests__/local-intelligence-l3.test.js +0 -332
  85. package/dist/__tests__/local-llm.test.js +0 -253
  86. package/dist/__tests__/local-mode-offline.test.js +0 -187
  87. package/dist/__tests__/local-mode-stats.test.js +0 -273
  88. package/dist/__tests__/local-mode-tui.test.js +0 -343
  89. package/dist/__tests__/local-parse.test.js +0 -199
  90. package/dist/__tests__/log-tailer.test.js +0 -208
  91. package/dist/__tests__/loop-breaker.test.js +0 -276
  92. package/dist/__tests__/loop-miner.test.js +0 -226
  93. package/dist/__tests__/mcp-config.test.js +0 -126
  94. package/dist/__tests__/mcp-content-json.test.js +0 -10
  95. package/dist/__tests__/mcp-envelope.test.js +0 -124
  96. package/dist/__tests__/metrics-store.test.js +0 -223
  97. package/dist/__tests__/native-watcher.test.js +0 -191
  98. package/dist/__tests__/navigation-hooks-agent-aware.test.js +0 -145
  99. package/dist/__tests__/negative-knowledge.test.js +0 -116
  100. package/dist/__tests__/network-boundary.test.js +0 -190
  101. package/dist/__tests__/network-firewall.test.js +0 -112
  102. package/dist/__tests__/nudge-invariants.test.js +0 -160
  103. package/dist/__tests__/nudge-v2.test.js +0 -225
  104. package/dist/__tests__/offline-rewind.test.js +0 -251
  105. package/dist/__tests__/open-threads.test.js +0 -89
  106. package/dist/__tests__/output-compressor.test.js +0 -93
  107. package/dist/__tests__/pending-violations.test.js +0 -112
  108. package/dist/__tests__/persistence-effectiveness.test.js +0 -143
  109. package/dist/__tests__/provider-factory.test.js +0 -42
  110. package/dist/__tests__/providers.test.js +0 -24
  111. package/dist/__tests__/proxy.test.js +0 -314
  112. package/dist/__tests__/query-router.test.js +0 -1018
  113. package/dist/__tests__/reasoning-quality-route.test.js +0 -138
  114. package/dist/__tests__/redactor.test.js +0 -120
  115. package/dist/__tests__/resource-monitor.test.js +0 -57
  116. package/dist/__tests__/response-envelope.test.js +0 -100
  117. package/dist/__tests__/risk-classifier.test.js +0 -101
  118. package/dist/__tests__/risk-signal-scope.test.js +0 -75
  119. package/dist/__tests__/rule-evaluator.test.js +0 -280
  120. package/dist/__tests__/scip-decoder.test.js +0 -49
  121. package/dist/__tests__/scip-downloader.test.js +0 -201
  122. package/dist/__tests__/scip-merger.test.js +0 -103
  123. package/dist/__tests__/search-index.test.js +0 -422
  124. package/dist/__tests__/semantic-enrichment.test.js +0 -360
  125. package/dist/__tests__/session-brief-builder.test.js +0 -187
  126. package/dist/__tests__/session-context.test.js +0 -221
  127. package/dist/__tests__/session-continuity.test.js +0 -144
  128. package/dist/__tests__/session-dedup.test.js +0 -74
  129. package/dist/__tests__/session-event-wiring.test.js +0 -206
  130. package/dist/__tests__/session-events.test.js +0 -149
  131. package/dist/__tests__/session-legend.test.js +0 -20
  132. package/dist/__tests__/session-persistence.test.js +0 -131
  133. package/dist/__tests__/session-resume-block.test.js +0 -107
  134. package/dist/__tests__/session-resume.test.js +0 -97
  135. package/dist/__tests__/session-summary-writer.test.js +0 -134
  136. package/dist/__tests__/shadow-ledger.test.js +0 -203
  137. package/dist/__tests__/shell-classifier.test.js +0 -151
  138. package/dist/__tests__/shell-compression-floor.test.js +0 -189
  139. package/dist/__tests__/shell-compression-v2.test.js +0 -339
  140. package/dist/__tests__/shell-compressor.test.js +0 -35
  141. package/dist/__tests__/shell-hooks.test.js +0 -128
  142. package/dist/__tests__/shell-strategies.test.js +0 -644
  143. package/dist/__tests__/shell-tee.test.js +0 -133
  144. package/dist/__tests__/signal-dedup.test.js +0 -158
  145. package/dist/__tests__/signal-reinforcer.test.js +0 -77
  146. package/dist/__tests__/signal-scorer.test.js +0 -251
  147. package/dist/__tests__/signal-show-store.test.js +0 -108
  148. package/dist/__tests__/smart-truncate.test.js +0 -215
  149. package/dist/__tests__/snapshot-v2.test.js +0 -113
  150. package/dist/__tests__/sprint-l1-local-mode.test.js +0 -130
  151. package/dist/__tests__/sprint-l10-boot.test.js +0 -220
  152. package/dist/__tests__/sprint-l9-offline-commands.test.js +0 -189
  153. package/dist/__tests__/sprint-q-persistent-context.test.js +0 -198
  154. package/dist/__tests__/sprint-s1-wiring.test.js +0 -215
  155. package/dist/__tests__/sprint-s2-wiring.test.js +0 -256
  156. package/dist/__tests__/sprint-s3-wiring.test.js +0 -195
  157. package/dist/__tests__/sprint-s4-wiring.test.js +0 -213
  158. package/dist/__tests__/sprint-s6-hooks.test.js +0 -222
  159. package/dist/__tests__/sprint-s7-persistent.test.js +0 -263
  160. package/dist/__tests__/sprint-s8-value.test.js +0 -167
  161. package/dist/__tests__/sprint-s9-behavioral.test.js +0 -179
  162. package/dist/__tests__/sprint3-intelligence.test.js +0 -297
  163. package/dist/__tests__/sprint5-mcp-server.test.js +0 -136
  164. package/dist/__tests__/startup-display.test.js +0 -302
  165. package/dist/__tests__/startup-log-file.test.js +0 -97
  166. package/dist/__tests__/stash-manager.test.js +0 -229
  167. package/dist/__tests__/state-detector.test.js +0 -92
  168. package/dist/__tests__/status-dashboard.test.js +0 -142
  169. package/dist/__tests__/temporal-facts.test.js +0 -292
  170. package/dist/__tests__/temporal-routes.test.js +0 -142
  171. package/dist/__tests__/test-detector.test.js +0 -174
  172. package/dist/__tests__/theme.test.js +0 -72
  173. package/dist/__tests__/timeline-agents.test.js +0 -122
  174. package/dist/__tests__/timeline-bootstrap.test.js +0 -176
  175. package/dist/__tests__/timeline-filters.test.js +0 -193
  176. package/dist/__tests__/timeline-markers.test.js +0 -151
  177. package/dist/__tests__/timeline-routes.test.js +0 -156
  178. package/dist/__tests__/timeline-store.test.js +0 -171
  179. package/dist/__tests__/token-counter.test.js +0 -86
  180. package/dist/__tests__/token-estimator.test.js +0 -96
  181. package/dist/__tests__/token-flow-api.test.js +0 -239
  182. package/dist/__tests__/token-flow-instrumentation.test.js +0 -437
  183. package/dist/__tests__/token-flow-persistence.test.js +0 -356
  184. package/dist/__tests__/token-flow-routes.test.js +0 -199
  185. package/dist/__tests__/token-flow.test.js +0 -695
  186. package/dist/__tests__/tool-clusters.test.js +0 -177
  187. package/dist/__tests__/transport-mux.test.js +0 -283
  188. package/dist/__tests__/turn-segmenter.test.js +0 -166
  189. package/dist/__tests__/uninstall.test.js +0 -141
  190. package/dist/__tests__/warm-start-policy.test.js +0 -271
  191. package/dist/__tests__/wire-cap-nudge.test.js +0 -77
  192. package/dist/__tests__/worker-pool.test.js +0 -101
  193. package/dist/ui/assets/index-7gl3mIuY.css +0 -1
  194. package/dist/ui/assets/index-CX4FCWGT.js +0 -10
  195. package/dist/ui/assets/rolldown-runtime-S-ySWqyJ.js +0 -1
  196. package/dist/ui/assets/vis-network-NIJHUFI3.js +0 -908
  197. package/dist/ui/fonts/jetbrains-mono-latin-400-normal.woff +0 -0
  198. package/dist/ui/icon-wordmark.png +0 -0
  199. package/dist/ui/icon-wordmark.svg +0 -30
  200. package/dist/ui/icon.png +0 -0
  201. package/dist/ui/icon.svg +0 -25
  202. package/dist/ui/index.html +0 -15
  203. package/dist/ui/prototype-sandbox/index.html +0 -257
  204. package/dist/ui/screenshots/activity.png +0 -0
  205. package/dist/ui/screenshots/code-base-intelligence.png +0 -0
  206. package/dist/ui/screenshots/dashboard.png +0 -0
  207. package/dist/ui/screenshots/project-memory.png +0 -0
  208. package/dist/ui/screenshots/reasoning-quality.png +0 -0
  209. package/dist/ui/screenshots/reasoning-session.png +0 -0
  210. package/dist/ui/screenshots/token-session.png +0 -0
  211. package/dist/ui/screenshots/token-trace-main.png +0 -0
  212. package/dist/ui/screenshots/token-turn.png +0 -0
  213. package/dist/ui/unerr-wordmark.png +0 -0
  214. package/dist/ui/unerr-wordmark.svg +0 -9
  215. package/dist/ui/unerr.png +0 -0
  216. package/dist/ui/unerr.svg +0 -25
  217. package/dist/ui/web-app-manifest-192x192.png +0 -0
  218. package/dist/ui/web-app-manifest-512x512.png +0 -0
@@ -1,258 +0,0 @@
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
- });
@@ -1,34 +0,0 @@
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
- });
@@ -1,154 +0,0 @@
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
- });
@@ -1,179 +0,0 @@
1
- import { beforeEach, describe, expect, it } from "vitest";
2
- import { clearCommunityCache, detectCommunities, needsRecomputation, } from "../intelligence/community-detector.js";
3
- import { entityKey } from "../intelligence/indexer/entity-key.js";
4
- beforeEach(() => {
5
- clearCommunityCache();
6
- });
7
- function makeEntity(name, filePath) {
8
- return {
9
- key: entityKey(filePath, "function", name, ""),
10
- kind: "function",
11
- name,
12
- file_path: filePath,
13
- start_line: 1,
14
- end_line: 10,
15
- signature: `${name}()`,
16
- body_hash: "hash",
17
- exported: true,
18
- parent_key: null,
19
- language: "typescript",
20
- is_async: false,
21
- parameter_count: 0,
22
- doc: null,
23
- };
24
- }
25
- describe("Community Detection (M.1-M.3)", () => {
26
- it("detects distinct clusters in a graph", async () => {
27
- const entities = [
28
- makeEntity("authLogin", "src/auth/login.ts"),
29
- makeEntity("authLogout", "src/auth/logout.ts"),
30
- makeEntity("authValidate", "src/auth/validate.ts"),
31
- makeEntity("payProcess", "src/payments/process.ts"),
32
- makeEntity("payRefund", "src/payments/refund.ts"),
33
- makeEntity("payCharge", "src/payments/charge.ts"),
34
- ];
35
- const edges = [
36
- {
37
- from_key: entities[0].key,
38
- to_key: entities[1].key,
39
- type: "calls",
40
- file_path: "a.ts",
41
- line: 1,
42
- },
43
- {
44
- from_key: entities[0].key,
45
- to_key: entities[2].key,
46
- type: "calls",
47
- file_path: "a.ts",
48
- line: 2,
49
- },
50
- {
51
- from_key: entities[1].key,
52
- to_key: entities[2].key,
53
- type: "calls",
54
- file_path: "a.ts",
55
- line: 3,
56
- },
57
- {
58
- from_key: entities[3].key,
59
- to_key: entities[4].key,
60
- type: "calls",
61
- file_path: "b.ts",
62
- line: 1,
63
- },
64
- {
65
- from_key: entities[3].key,
66
- to_key: entities[5].key,
67
- type: "calls",
68
- file_path: "b.ts",
69
- line: 2,
70
- },
71
- {
72
- from_key: entities[4].key,
73
- to_key: entities[5].key,
74
- type: "calls",
75
- file_path: "b.ts",
76
- line: 3,
77
- },
78
- ];
79
- const result = await await detectCommunities(entities, edges);
80
- expect(result.communities.length).toBeGreaterThanOrEqual(2);
81
- expect(result.assignments.size).toBe(6);
82
- });
83
- it("produces community profiles with files and kinds", async () => {
84
- const entities = [
85
- makeEntity("fn1", "src/auth/a.ts"),
86
- makeEntity("fn2", "src/auth/b.ts"),
87
- ];
88
- const edges = [
89
- {
90
- from_key: entities[0].key,
91
- to_key: entities[1].key,
92
- type: "calls",
93
- file_path: "a.ts",
94
- line: 1,
95
- },
96
- ];
97
- const result = await await detectCommunities(entities, edges);
98
- expect(result.communities.length).toBeGreaterThanOrEqual(1);
99
- const profile = result.communities[0];
100
- expect(profile.entityCount).toBeGreaterThan(0);
101
- expect(profile.files.length).toBeGreaterThan(0);
102
- expect(profile.dominantKinds.length).toBeGreaterThan(0);
103
- });
104
- it("infers community name from directory", async () => {
105
- const entities = [
106
- makeEntity("login", "src/auth/login.ts"),
107
- makeEntity("logout", "src/auth/logout.ts"),
108
- makeEntity("validate", "src/auth/validate.ts"),
109
- ];
110
- const edges = [
111
- {
112
- from_key: entities[0].key,
113
- to_key: entities[1].key,
114
- type: "calls",
115
- file_path: "a.ts",
116
- line: 1,
117
- },
118
- {
119
- from_key: entities[1].key,
120
- to_key: entities[2].key,
121
- type: "calls",
122
- file_path: "a.ts",
123
- line: 2,
124
- },
125
- ];
126
- const result = await await detectCommunities(entities, edges);
127
- const authCommunity = result.communities.find((c) => c.name === "auth");
128
- expect(authCommunity).toBeDefined();
129
- });
130
- it("handles empty graph", async () => {
131
- const result = await detectCommunities([], []);
132
- expect(result.communities).toHaveLength(0);
133
- expect(result.modularity).toBe(0);
134
- });
135
- it("computes cohesion and coupling", async () => {
136
- const entities = [
137
- makeEntity("a", "src/mod/a.ts"),
138
- makeEntity("b", "src/mod/b.ts"),
139
- makeEntity("external", "src/other/c.ts"),
140
- ];
141
- const edges = [
142
- {
143
- from_key: entities[0].key,
144
- to_key: entities[1].key,
145
- type: "calls",
146
- file_path: "a.ts",
147
- line: 1,
148
- },
149
- {
150
- from_key: entities[0].key,
151
- to_key: entities[2].key,
152
- type: "calls",
153
- file_path: "a.ts",
154
- line: 2,
155
- },
156
- ];
157
- const result = await await detectCommunities(entities, edges);
158
- for (const community of result.communities) {
159
- expect(community.cohesion).toBeGreaterThanOrEqual(0);
160
- expect(community.coupling).toBeGreaterThanOrEqual(0);
161
- }
162
- });
163
- });
164
- describe("Community Recomputation (M.6)", () => {
165
- it("triggers recomputation when no cache", async () => {
166
- expect(needsRecomputation(100)).toBe(true);
167
- });
168
- it("skips recomputation when below threshold", async () => {
169
- const entities = Array.from({ length: 100 }, (_, i) => makeEntity(`fn${i}`, `src/f${i}.ts`));
170
- const edges = [];
171
- await detectCommunities(entities, edges);
172
- expect(needsRecomputation(105)).toBe(false);
173
- });
174
- it("triggers recomputation when >10% change", async () => {
175
- const entities = Array.from({ length: 100 }, (_, i) => makeEntity(`fn${i}`, `src/f${i}.ts`));
176
- detectCommunities(entities, []);
177
- expect(needsRecomputation(115)).toBe(true);
178
- });
179
- });