@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,339 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Tests for the RTK-parity shell compression upgrades (R1, R3, R4, R6, R7, R9).
|
|
3
|
-
*
|
|
4
|
-
* The per-classifier floor test lives in shell-compression-floor.test.ts (R10).
|
|
5
|
-
*/
|
|
6
|
-
import { mkdirSync, rmSync, writeFileSync } from "node:fs";
|
|
7
|
-
import { tmpdir } from "node:os";
|
|
8
|
-
import { join } from "node:path";
|
|
9
|
-
import { afterEach, beforeEach, describe, expect, it } from "vitest";
|
|
10
|
-
import { extractFilePathCandidates } from "../proxy/shell-graph-boost.js";
|
|
11
|
-
import { tryCompressCloud } from "../proxy/shell-strategies/cloud.js";
|
|
12
|
-
import { compressErrorDiagnostic } from "../proxy/shell-strategies/error-diagnostic.js";
|
|
13
|
-
import { _clearFilterCache, applyUserFilter, } from "../proxy/shell-strategies/filter-dsl.js";
|
|
14
|
-
import { compressLogText } from "../proxy/shell-strategies/log-text.js";
|
|
15
|
-
import { getBuiltinRedactRules, redactOutput, } from "../proxy/shell-strategies/redact.js";
|
|
16
|
-
import { compressTreePaths } from "../proxy/shell-strategies/tree-paths.js";
|
|
17
|
-
describe("R6 — redact primitive", () => {
|
|
18
|
-
it("redacts the user's home directory to ~", () => {
|
|
19
|
-
const home = process.env.HOME ?? "/Users/test";
|
|
20
|
-
const input = `Editing ${home}/projects/foo.ts`;
|
|
21
|
-
const out = redactOutput(input);
|
|
22
|
-
if (home) {
|
|
23
|
-
expect(out).toContain("~/projects/foo.ts");
|
|
24
|
-
expect(out).not.toContain(home);
|
|
25
|
-
}
|
|
26
|
-
});
|
|
27
|
-
it("redacts ISO timestamps to <ts>", () => {
|
|
28
|
-
const input = "2026-05-13T14:32:01.123Z something happened";
|
|
29
|
-
expect(redactOutput(input)).toContain("<ts>");
|
|
30
|
-
});
|
|
31
|
-
it("redacts UUIDs", () => {
|
|
32
|
-
const input = "request_id=12345678-1234-1234-1234-123456789012 done";
|
|
33
|
-
expect(redactOutput(input)).toContain("<uuid>");
|
|
34
|
-
});
|
|
35
|
-
it("collapses 40-char SHA to first-7 + ellipsis", () => {
|
|
36
|
-
const sha = "abcdef1234567890abcdef1234567890abcdef12";
|
|
37
|
-
expect(redactOutput(`commit ${sha}`)).toMatch(/abcdef1[…]/);
|
|
38
|
-
});
|
|
39
|
-
it("applies extra user-supplied rules after builtins", () => {
|
|
40
|
-
const out = redactOutput("foo bar baz", [
|
|
41
|
-
{ pattern: /bar/g, replacement: "QUX" },
|
|
42
|
-
]);
|
|
43
|
-
expect(out).toBe("foo QUX baz");
|
|
44
|
-
});
|
|
45
|
-
it("exposes builtin rules read-only", () => {
|
|
46
|
-
expect(getBuiltinRedactRules().length).toBeGreaterThan(0);
|
|
47
|
-
});
|
|
48
|
-
});
|
|
49
|
-
describe("R7 — tree-paths grouping", () => {
|
|
50
|
-
it("passes through small outputs unchanged", () => {
|
|
51
|
-
const tiny = ["src/a.ts", "src/b.ts", "src/c.ts"].join("\n");
|
|
52
|
-
expect(compressTreePaths(tiny)).toBe(tiny);
|
|
53
|
-
});
|
|
54
|
-
it("groups large path lists by directory + extension", () => {
|
|
55
|
-
const lines = [];
|
|
56
|
-
for (let i = 0; i < 40; i++)
|
|
57
|
-
lines.push(`src/components/file_${i}.tsx`);
|
|
58
|
-
for (let i = 0; i < 40; i++)
|
|
59
|
-
lines.push(`src/lib/util_${i}.ts`);
|
|
60
|
-
const out = compressTreePaths(lines.join("\n"));
|
|
61
|
-
expect(out).toContain("_shell_fmt:tree_paths");
|
|
62
|
-
expect(out).toContain("src/components/");
|
|
63
|
-
expect(out).toContain(".tsx");
|
|
64
|
-
expect(out).toContain("src/lib/");
|
|
65
|
-
// Should be substantially smaller
|
|
66
|
-
expect(out.length).toBeLessThan(lines.join("\n").length * 0.6);
|
|
67
|
-
});
|
|
68
|
-
it("rolls up node_modules / .git / dist", () => {
|
|
69
|
-
const lines = [];
|
|
70
|
-
for (let i = 0; i < 50; i++)
|
|
71
|
-
lines.push(`node_modules/pkg/${i}.js`);
|
|
72
|
-
const out = compressTreePaths(lines.join("\n"));
|
|
73
|
-
expect(out).toContain("node_modules/");
|
|
74
|
-
expect(out).toContain("[collapsed]");
|
|
75
|
-
});
|
|
76
|
-
});
|
|
77
|
-
describe("R1 — success short-circuit (log_text, error_diagnostic)", () => {
|
|
78
|
-
it("collapses clean build logs to one line", () => {
|
|
79
|
-
const lines = ["Starting compile..."];
|
|
80
|
-
for (let i = 0; i < 200; i++)
|
|
81
|
-
lines.push(`compiling module_${i}.ts`);
|
|
82
|
-
lines.push("Finished release [optimized] target(s) in 12.3s");
|
|
83
|
-
const out = compressLogText(lines.join("\n"), "cargo build");
|
|
84
|
-
expect(out).toContain("build ok");
|
|
85
|
-
expect(out.split("\n").length).toBeLessThanOrEqual(2);
|
|
86
|
-
});
|
|
87
|
-
it("does NOT short-circuit when an error is present", () => {
|
|
88
|
-
const text = [
|
|
89
|
-
"compiling foo.ts",
|
|
90
|
-
"ERROR: type mismatch on line 42",
|
|
91
|
-
"compiling bar.ts",
|
|
92
|
-
]
|
|
93
|
-
.concat(Array.from({ length: 100 }, (_, i) => `line ${i}`))
|
|
94
|
-
.join("\n");
|
|
95
|
-
const out = compressLogText(text, "cargo build");
|
|
96
|
-
expect(out).not.toMatch(/^_shell_fmt:log_text\nbuild ok/);
|
|
97
|
-
});
|
|
98
|
-
it("collapses lint output when no issues found", () => {
|
|
99
|
-
const out = compressErrorDiagnostic("Checked 47 files. All checks passed.", "eslint");
|
|
100
|
-
expect(out).toContain("eslint ok");
|
|
101
|
-
});
|
|
102
|
-
it("does NOT short-circuit lint output with errors", () => {
|
|
103
|
-
const out = compressErrorDiagnostic("src/foo.ts:10:5 error: Cannot find name 'bar'", "tsc");
|
|
104
|
-
expect(out).not.toContain("ok");
|
|
105
|
-
});
|
|
106
|
-
});
|
|
107
|
-
describe("R3 — user filter DSL", () => {
|
|
108
|
-
let tmpRoot;
|
|
109
|
-
beforeEach(() => {
|
|
110
|
-
tmpRoot = join(tmpdir(), `unerr-filter-${Date.now()}-${Math.random()}`);
|
|
111
|
-
mkdirSync(join(tmpRoot, ".unerr"), { recursive: true });
|
|
112
|
-
_clearFilterCache();
|
|
113
|
-
});
|
|
114
|
-
afterEach(() => {
|
|
115
|
-
try {
|
|
116
|
-
rmSync(tmpRoot, { recursive: true, force: true });
|
|
117
|
-
}
|
|
118
|
-
catch {
|
|
119
|
-
// ignore
|
|
120
|
-
}
|
|
121
|
-
_clearFilterCache();
|
|
122
|
-
});
|
|
123
|
-
it("returns null when no filter matches", () => {
|
|
124
|
-
writeFileSync(join(tmpRoot, ".unerr", "filters.toml"), `[filters.something]\nmatch_command = "^never-runs"\nmax_lines = 5\n`);
|
|
125
|
-
expect(applyUserFilter("my-tool run", "hello", tmpRoot)).toBeNull();
|
|
126
|
-
});
|
|
127
|
-
it("applies strip_lines_matching", () => {
|
|
128
|
-
writeFileSync(join(tmpRoot, ".unerr", "filters.toml"), `[filters.mytool]\nmatch_command = "^mytool"\nstrip_lines_matching = ["^DEBUG"]\n`);
|
|
129
|
-
const r = applyUserFilter("mytool run", "INFO: ok\nDEBUG: chatter\nDONE", tmpRoot);
|
|
130
|
-
expect(r).not.toBeNull();
|
|
131
|
-
expect(r?.text).not.toContain("DEBUG");
|
|
132
|
-
expect(r?.text).toContain("INFO: ok");
|
|
133
|
-
});
|
|
134
|
-
it("short-circuits on match_output", () => {
|
|
135
|
-
writeFileSync(join(tmpRoot, ".unerr", "filters.toml"), `[filters.green]\nmatch_command = "^anything"\nmatch_output = [{ pattern = "all green", message = "ok" }]\n`);
|
|
136
|
-
const r = applyUserFilter("anything", "all green here", tmpRoot);
|
|
137
|
-
expect(r?.text).toBe("_shell_fmt:user_filter\nok");
|
|
138
|
-
});
|
|
139
|
-
it("applies replace substitutions", () => {
|
|
140
|
-
writeFileSync(join(tmpRoot, ".unerr", "filters.toml"), `[filters.scrub]\nmatch_command = "^scrub"\nreplace = [{ pattern = "secret", replacement = "***" }]\n`);
|
|
141
|
-
const r = applyUserFilter("scrub run", "the secret thing", tmpRoot);
|
|
142
|
-
expect(r?.text).toContain("***");
|
|
143
|
-
expect(r?.text).not.toContain("secret");
|
|
144
|
-
});
|
|
145
|
-
it("respects tail_lines window", () => {
|
|
146
|
-
writeFileSync(join(tmpRoot, ".unerr", "filters.toml"), `[filters.tail]\nmatch_command = "^tail"\ntail_lines = 3\n`);
|
|
147
|
-
const lines = Array.from({ length: 20 }, (_, i) => `line_${i}`).join("\n");
|
|
148
|
-
const r = applyUserFilter("tail run", lines, tmpRoot);
|
|
149
|
-
expect(r?.text).toContain("line_19");
|
|
150
|
-
expect(r?.text).toContain("earlier lines suppressed");
|
|
151
|
-
expect(r?.text).not.toContain("line_5");
|
|
152
|
-
});
|
|
153
|
-
});
|
|
154
|
-
describe("R4 — cloud parsers", () => {
|
|
155
|
-
it("compresses aws ec2 describe-instances", () => {
|
|
156
|
-
const payload = JSON.stringify({
|
|
157
|
-
Reservations: [
|
|
158
|
-
{
|
|
159
|
-
Instances: [
|
|
160
|
-
{
|
|
161
|
-
InstanceId: "i-0abc",
|
|
162
|
-
State: { Name: "running" },
|
|
163
|
-
InstanceType: "t3.micro",
|
|
164
|
-
PrivateIpAddress: "10.0.0.5",
|
|
165
|
-
Placement: { AvailabilityZone: "us-east-1a" },
|
|
166
|
-
Tags: [{ Key: "Name", Value: "web" }],
|
|
167
|
-
},
|
|
168
|
-
{
|
|
169
|
-
InstanceId: "i-0def",
|
|
170
|
-
State: { Name: "stopped" },
|
|
171
|
-
InstanceType: "t3.nano",
|
|
172
|
-
PrivateIpAddress: "10.0.0.6",
|
|
173
|
-
Placement: { AvailabilityZone: "us-east-1b" },
|
|
174
|
-
Tags: [],
|
|
175
|
-
},
|
|
176
|
-
],
|
|
177
|
-
},
|
|
178
|
-
],
|
|
179
|
-
});
|
|
180
|
-
const out = tryCompressCloud(payload, "aws ec2 describe-instances");
|
|
181
|
-
expect(out).toContain("_shell_fmt:cloud[aws-ec2]");
|
|
182
|
-
expect(out).toContain("i-0abc");
|
|
183
|
-
expect(out).toContain("running");
|
|
184
|
-
expect(out.length).toBeLessThan(payload.length);
|
|
185
|
-
});
|
|
186
|
-
it("compresses aws iam list-users", () => {
|
|
187
|
-
const payload = JSON.stringify({
|
|
188
|
-
Users: Array.from({ length: 5 }, (_, i) => ({
|
|
189
|
-
UserName: `user${i}`,
|
|
190
|
-
Arn: `arn:aws:iam::123456789012:user/user${i}`,
|
|
191
|
-
})),
|
|
192
|
-
});
|
|
193
|
-
const out = tryCompressCloud(payload, "aws iam list-users");
|
|
194
|
-
expect(out).toContain("_shell_fmt:cloud[aws-iam]");
|
|
195
|
-
expect(out).toContain("user0");
|
|
196
|
-
expect(out).toContain("arn:…:");
|
|
197
|
-
});
|
|
198
|
-
it("compresses kubectl get -o json", () => {
|
|
199
|
-
const payload = JSON.stringify({
|
|
200
|
-
kind: "PodList",
|
|
201
|
-
items: Array.from({ length: 4 }, (_, i) => ({
|
|
202
|
-
metadata: { name: `pod-${i}`, namespace: "default" },
|
|
203
|
-
status: { phase: "Running" },
|
|
204
|
-
})),
|
|
205
|
-
});
|
|
206
|
-
const out = tryCompressCloud(payload, "kubectl get pods -o json");
|
|
207
|
-
expect(out).toContain("_shell_fmt:cloud[kubectl-get]");
|
|
208
|
-
expect(out).toContain("pod-0");
|
|
209
|
-
});
|
|
210
|
-
it("compresses docker inspect", () => {
|
|
211
|
-
const payload = JSON.stringify([
|
|
212
|
-
{
|
|
213
|
-
Id: "abc123def456789",
|
|
214
|
-
State: { Status: "running" },
|
|
215
|
-
Config: { Image: "node:20" },
|
|
216
|
-
Name: "/web",
|
|
217
|
-
},
|
|
218
|
-
]);
|
|
219
|
-
const out = tryCompressCloud(payload, "docker inspect web");
|
|
220
|
-
expect(out).toContain("_shell_fmt:cloud[docker-inspect]");
|
|
221
|
-
expect(out).toContain("running");
|
|
222
|
-
expect(out).toContain("node:20");
|
|
223
|
-
});
|
|
224
|
-
it("returns null on unknown commands", () => {
|
|
225
|
-
expect(tryCompressCloud("hello", "echo hello")).toBeNull();
|
|
226
|
-
});
|
|
227
|
-
it("returns null when JSON parse fails", () => {
|
|
228
|
-
expect(tryCompressCloud("not json", "aws ec2 describe-instances")).toBeNull();
|
|
229
|
-
});
|
|
230
|
-
});
|
|
231
|
-
describe("R9 — file path extraction for graph boost", () => {
|
|
232
|
-
it("extracts code-file paths from arbitrary output", () => {
|
|
233
|
-
const text = `
|
|
234
|
-
modified: src/proxy/shell-compressor.ts
|
|
235
|
-
modified: src/intelligence/local-graph.ts
|
|
236
|
-
new file: docs/README.md
|
|
237
|
-
`;
|
|
238
|
-
const paths = extractFilePathCandidates(text);
|
|
239
|
-
expect(paths).toContain("src/proxy/shell-compressor.ts");
|
|
240
|
-
expect(paths).toContain("src/intelligence/local-graph.ts");
|
|
241
|
-
});
|
|
242
|
-
it("ignores random strings without an extension", () => {
|
|
243
|
-
const paths = extractFilePathCandidates("foo bar baz");
|
|
244
|
-
expect(paths.length).toBe(0);
|
|
245
|
-
});
|
|
246
|
-
it("caps results at 64", () => {
|
|
247
|
-
const lines = [];
|
|
248
|
-
for (let i = 0; i < 200; i++)
|
|
249
|
-
lines.push(`src/file_${i}.ts`);
|
|
250
|
-
expect(extractFilePathCandidates(lines.join("\n")).length).toBeLessThanOrEqual(64);
|
|
251
|
-
});
|
|
252
|
-
});
|
|
253
|
-
describe("F2 — git status dedicated parser", () => {
|
|
254
|
-
it("compresses a typical git status output", async () => {
|
|
255
|
-
const { compressGitStatus } = await import("../proxy/shell-strategies/git-status.js");
|
|
256
|
-
const raw = `On branch main
|
|
257
|
-
Your branch is up to date with 'origin/main'.
|
|
258
|
-
|
|
259
|
-
Changes not staged for commit:
|
|
260
|
-
(use "git add <file>..." to update what will be committed)
|
|
261
|
-
(use "git restore <file>..." to discard changes in working directory)
|
|
262
|
-
\tmodified: docs/USER_TESTING_CHECKLIST.md
|
|
263
|
-
\tmodified: package.json
|
|
264
|
-
\tmodified: src/proxy/shell-compressor.ts
|
|
265
|
-
\tmodified: src/proxy/shell-graph-boost.ts
|
|
266
|
-
\tmodified: src/proxy/shell-strategies/log-text.ts
|
|
267
|
-
|
|
268
|
-
Untracked files:
|
|
269
|
-
(use "git add <file>..." to include in what will be committed)
|
|
270
|
-
\tsrc/proxy/shell-strategies/git-status.ts
|
|
271
|
-
\tsrc/proxy/shell-strategies/redact.ts
|
|
272
|
-
|
|
273
|
-
no changes added to commit (use "git add" and/or "git commit -a")`;
|
|
274
|
-
const out = compressGitStatus(raw);
|
|
275
|
-
expect(out).not.toBeNull();
|
|
276
|
-
expect(out).toContain("_shell_fmt:git_status");
|
|
277
|
-
expect(out).toContain("branch=main");
|
|
278
|
-
expect(out).toContain("modified: 5");
|
|
279
|
-
expect(out).toContain("untracked: 2");
|
|
280
|
-
expect(out).not.toContain('(use "git add'); // boilerplate stripped
|
|
281
|
-
// Substantially smaller
|
|
282
|
-
expect(out.length).toBeLessThan(raw.length * 0.7);
|
|
283
|
-
});
|
|
284
|
-
it("returns null on non-git-status input", async () => {
|
|
285
|
-
const { compressGitStatus } = await import("../proxy/shell-strategies/git-status.js");
|
|
286
|
-
expect(compressGitStatus("some random output")).toBeNull();
|
|
287
|
-
expect(compressGitStatus("hello world")).toBeNull();
|
|
288
|
-
});
|
|
289
|
-
it("groups large file lists by directory + extension", async () => {
|
|
290
|
-
const { compressGitStatus } = await import("../proxy/shell-strategies/git-status.js");
|
|
291
|
-
const lines = ["On branch main", "", "Untracked files:"];
|
|
292
|
-
for (let i = 0; i < 25; i++)
|
|
293
|
-
lines.push(`\tsrc/components/file_${i}.tsx`);
|
|
294
|
-
for (let i = 0; i < 15; i++)
|
|
295
|
-
lines.push(`\tsrc/lib/util_${i}.ts`);
|
|
296
|
-
const out = compressGitStatus(lines.join("\n"));
|
|
297
|
-
expect(out).toContain("untracked: 40");
|
|
298
|
-
expect(out).toContain("src/components/");
|
|
299
|
-
expect(out).toContain(".tsx");
|
|
300
|
-
});
|
|
301
|
-
});
|
|
302
|
-
describe("F3 — R1 short-circuit no longer gated on build command", () => {
|
|
303
|
-
it("collapses clean wrapper-script output (was rejected before)", async () => {
|
|
304
|
-
const { compressLogText } = await import("../proxy/shell-strategies/log-text.js");
|
|
305
|
-
const lines = [];
|
|
306
|
-
for (let i = 0; i < 200; i++)
|
|
307
|
-
lines.push(`processing item_${i}`);
|
|
308
|
-
lines.push("All checks passed");
|
|
309
|
-
// No "build/cargo/gradle/..." in command — previously skipped short-circuit
|
|
310
|
-
const out = compressLogText(lines.join("\n"), "/tmp/wrapper.sh");
|
|
311
|
-
expect(out).toContain("ok — All checks passed");
|
|
312
|
-
expect(out.split("\n").length).toBeLessThanOrEqual(2);
|
|
313
|
-
});
|
|
314
|
-
it("still uses 'build ok' label for build commands", async () => {
|
|
315
|
-
const { compressLogText } = await import("../proxy/shell-strategies/log-text.js");
|
|
316
|
-
const lines = [];
|
|
317
|
-
for (let i = 0; i < 200; i++)
|
|
318
|
-
lines.push(`compiling foo_${i}.ts`);
|
|
319
|
-
lines.push("Finished release [optimized] target(s) in 12.3s");
|
|
320
|
-
const out = compressLogText(lines.join("\n"), "cargo build");
|
|
321
|
-
expect(out).toContain("build ok");
|
|
322
|
-
});
|
|
323
|
-
});
|
|
324
|
-
describe("F1 — graph boost diagnostic logging + cache", () => {
|
|
325
|
-
it("exposes _clearShellBoostCache for tests", async () => {
|
|
326
|
-
const mod = await import("../proxy/shell-graph-boost.js");
|
|
327
|
-
expect(typeof mod._clearShellBoostCache).toBe("function");
|
|
328
|
-
// Idempotent
|
|
329
|
-
mod._clearShellBoostCache();
|
|
330
|
-
mod._clearShellBoostCache();
|
|
331
|
-
});
|
|
332
|
-
it("returns null cleanly when snapshot is missing", async () => {
|
|
333
|
-
const { tryLoadGraphForShellBoost, _clearShellBoostCache } = await import("../proxy/shell-graph-boost.js");
|
|
334
|
-
_clearShellBoostCache();
|
|
335
|
-
// Random cwd guaranteed to not have a snapshot
|
|
336
|
-
const result = await tryLoadGraphForShellBoost(`/tmp/nonexistent-${Date.now()}`);
|
|
337
|
-
expect(result).toBeNull();
|
|
338
|
-
});
|
|
339
|
-
});
|
|
@@ -1,35 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Layer 6 Sprint FE-F — shell compressor + graph-aware diff hints.
|
|
3
|
-
*/
|
|
4
|
-
import { describe, expect, it, vi } from "vitest";
|
|
5
|
-
import { compressShellOutput } from "../proxy/shell-compressor.js";
|
|
6
|
-
describe("compressShellOutput graph boost (FE-F.6)", () => {
|
|
7
|
-
it("annotates diff lines when entity lookup returns high fan-in", async () => {
|
|
8
|
-
const graph = {
|
|
9
|
-
findEntityByName: vi.fn(async (name) => {
|
|
10
|
-
if (name !== "riskyFn")
|
|
11
|
-
return null;
|
|
12
|
-
return {
|
|
13
|
-
key: "k",
|
|
14
|
-
kind: "function",
|
|
15
|
-
name: "riskyFn",
|
|
16
|
-
file_path: "src/x.ts",
|
|
17
|
-
start_line: 1,
|
|
18
|
-
signature: "()",
|
|
19
|
-
body: "",
|
|
20
|
-
fan_in: 40,
|
|
21
|
-
fan_out: 0,
|
|
22
|
-
risk_level: "high",
|
|
23
|
-
community: 0,
|
|
24
|
-
};
|
|
25
|
-
}),
|
|
26
|
-
};
|
|
27
|
-
const diff = "diff --git a/x.ts b/x.ts\n@@ -1 +1 @@\n+riskyFn();\nexport function riskyFn() {}\n";
|
|
28
|
-
const r = await compressShellOutput("git diff", diff, {
|
|
29
|
-
graph,
|
|
30
|
-
persistStats: false,
|
|
31
|
-
});
|
|
32
|
-
expect(r.text).toContain("[HIGH-RISK:riskyFn");
|
|
33
|
-
expect(graph.findEntityByName).toHaveBeenCalled();
|
|
34
|
-
});
|
|
35
|
-
});
|
|
@@ -1,128 +0,0 @@
|
|
|
1
|
-
import { mkdirSync, readFileSync, rmSync } from "node:fs";
|
|
2
|
-
import { tmpdir } from "node:os";
|
|
3
|
-
import { join } from "node:path";
|
|
4
|
-
import { describe, expect, it, vi } from "vitest";
|
|
5
|
-
import { parseExecCommandLine, runExecMain } from "../commands/exec.js";
|
|
6
|
-
import { mergePreToolUseBashHook } from "../config/claude-settings-hooks.js";
|
|
7
|
-
import { runPreBashHook } from "../hooks/shell-hooks.js";
|
|
8
|
-
describe("runPreBashHook", () => {
|
|
9
|
-
it("rewrites single-line Bash command to unerr exec -- (Claude Code protocol)", () => {
|
|
10
|
-
const stdin = JSON.stringify({
|
|
11
|
-
hook_event_name: "PreToolUse",
|
|
12
|
-
tool_input: { command: "ps aux" },
|
|
13
|
-
});
|
|
14
|
-
const out = JSON.parse(runPreBashHook(stdin));
|
|
15
|
-
expect(out.hookSpecificOutput.hookEventName).toBe("PreToolUse");
|
|
16
|
-
expect(out.hookSpecificOutput.permissionDecision).toBe("allow");
|
|
17
|
-
expect(out.hookSpecificOutput.updatedInput.command).toBe("unerr exec -- ps aux");
|
|
18
|
-
});
|
|
19
|
-
it("rewrites multi-line Bash command to unerr exec --b64", () => {
|
|
20
|
-
const multiLine = 'python3 -c "import os\nprint(os.getcwd())"';
|
|
21
|
-
const stdin = JSON.stringify({
|
|
22
|
-
hook_event_name: "PreToolUse",
|
|
23
|
-
tool_input: { command: multiLine },
|
|
24
|
-
});
|
|
25
|
-
const out = JSON.parse(runPreBashHook(stdin));
|
|
26
|
-
const rewritten = out.hookSpecificOutput.updatedInput.command;
|
|
27
|
-
expect(rewritten).toMatch(/^unerr exec --b64 /);
|
|
28
|
-
const b64 = rewritten.replace("unerr exec --b64 ", "");
|
|
29
|
-
expect(Buffer.from(b64, "base64").toString("utf-8")).toBe(multiLine);
|
|
30
|
-
});
|
|
31
|
-
it("returns {} for empty stdin", () => {
|
|
32
|
-
expect(runPreBashHook("")).toBe("{}");
|
|
33
|
-
});
|
|
34
|
-
it("passthrough when already unerr exec", () => {
|
|
35
|
-
const stdin = JSON.stringify({
|
|
36
|
-
tool_input: { command: "unerr exec -- echo hi" },
|
|
37
|
-
});
|
|
38
|
-
expect(runPreBashHook(stdin)).toBe("{}");
|
|
39
|
-
});
|
|
40
|
-
});
|
|
41
|
-
describe("parseExecCommandLine", () => {
|
|
42
|
-
it("parses tokens after exec and optional --", () => {
|
|
43
|
-
expect(parseExecCommandLine(["node", "cli", "exec", "--", "echo", "a"])).toBe("echo a");
|
|
44
|
-
expect(parseExecCommandLine(["exec", "pwd"])).toBe("pwd");
|
|
45
|
-
});
|
|
46
|
-
it("decodes --b64 base64-encoded commands", () => {
|
|
47
|
-
const cmd = "python3 -c \"print('hello\\nworld')\"";
|
|
48
|
-
const b64 = Buffer.from(cmd, "utf-8").toString("base64");
|
|
49
|
-
expect(parseExecCommandLine(["exec", "--b64", b64])).toBe(cmd);
|
|
50
|
-
});
|
|
51
|
-
});
|
|
52
|
-
describe("runExecMain", () => {
|
|
53
|
-
it("runs echo via bash -lc", async () => {
|
|
54
|
-
const spy = vi
|
|
55
|
-
.spyOn(process.stdout, "write")
|
|
56
|
-
.mockImplementation(() => true);
|
|
57
|
-
const code = await runExecMain([
|
|
58
|
-
"node",
|
|
59
|
-
"cli.js",
|
|
60
|
-
"exec",
|
|
61
|
-
"--",
|
|
62
|
-
"echo",
|
|
63
|
-
"ok",
|
|
64
|
-
]);
|
|
65
|
-
expect(code).toBe(0);
|
|
66
|
-
spy.mockRestore();
|
|
67
|
-
});
|
|
68
|
-
it("falls back to raw output on shell parse error", async () => {
|
|
69
|
-
const stdoutSpy = vi
|
|
70
|
-
.spyOn(process.stdout, "write")
|
|
71
|
-
.mockImplementation(() => true);
|
|
72
|
-
const stderrSpy = vi
|
|
73
|
-
.spyOn(process.stderr, "write")
|
|
74
|
-
.mockImplementation(() => true);
|
|
75
|
-
// Unterminated quote — zsh emits "unmatched", bash emits "syntax error"
|
|
76
|
-
const code = await runExecMain([
|
|
77
|
-
"node",
|
|
78
|
-
"cli.js",
|
|
79
|
-
"exec",
|
|
80
|
-
"--b64",
|
|
81
|
-
Buffer.from('echo "unterminated', "utf-8").toString("base64"),
|
|
82
|
-
]);
|
|
83
|
-
expect(code).not.toBe(0);
|
|
84
|
-
// stdout should contain raw shell error without compression header
|
|
85
|
-
const output = stdoutSpy.mock.calls.map((c) => String(c[0])).join("");
|
|
86
|
-
expect(output).toMatch(/unmatched|parse error|syntax error|unexpected/i);
|
|
87
|
-
expect(output).not.toMatch(/^_shell_fmt:/);
|
|
88
|
-
stdoutSpy.mockRestore();
|
|
89
|
-
stderrSpy.mockRestore();
|
|
90
|
-
});
|
|
91
|
-
});
|
|
92
|
-
describe("mergePreToolUseBashHook", () => {
|
|
93
|
-
it("writes settings.json with PreToolUse entry", () => {
|
|
94
|
-
const dir = join(tmpdir(), `fe-d-st-${Date.now()}`);
|
|
95
|
-
mkdirSync(join(dir, ".claude"), { recursive: true });
|
|
96
|
-
const r = mergePreToolUseBashHook(dir);
|
|
97
|
-
expect(r.ok).toBe(true);
|
|
98
|
-
expect(r.action).toBe("merged");
|
|
99
|
-
const settings = JSON.parse(readFileSync(join(dir, ".claude", "settings.json"), "utf-8"));
|
|
100
|
-
expect(settings.hooks.PreToolUse.length).toBeGreaterThan(0);
|
|
101
|
-
rmSync(dir, { recursive: true, force: true });
|
|
102
|
-
});
|
|
103
|
-
it("rewrites hooks idempotently on subsequent merges (binary-path refresh)", () => {
|
|
104
|
-
// Behavior change: every merge now strips existing unerr hooks and re-adds
|
|
105
|
-
// them, so binary paths stay current (handles `pnpm link` upgrades from bare
|
|
106
|
-
// to absolute paths). The result is `merged` on every call but the on-disk
|
|
107
|
-
// state stays stable (same hook count, same shape).
|
|
108
|
-
const dir = join(tmpdir(), `fe-d-st2-${Date.now()}`);
|
|
109
|
-
mkdirSync(join(dir, ".claude"), { recursive: true });
|
|
110
|
-
const r1 = mergePreToolUseBashHook(dir);
|
|
111
|
-
const r2 = mergePreToolUseBashHook(dir);
|
|
112
|
-
expect(r1.ok).toBe(true);
|
|
113
|
-
expect(r2.ok).toBe(true);
|
|
114
|
-
// Idempotent in the on-disk sense: two merges produce identical settings.
|
|
115
|
-
const settings = JSON.parse(readFileSync(join(dir, ".claude", "settings.json"), "utf-8"));
|
|
116
|
-
const preTool = settings.hooks?.PreToolUse;
|
|
117
|
-
expect(Array.isArray(preTool)).toBe(true);
|
|
118
|
-
// No duplicate unerr entries.
|
|
119
|
-
// 6 PreToolUse matcher entries: Bash, Read, Grep, Glob, Write, Edit — each
|
|
120
|
-
// with one `unerr hook pre-*` command. No duplicates (key check: each
|
|
121
|
-
// matcher appears exactly once).
|
|
122
|
-
const unerrEntries = preTool.filter((entry) => entry.hooks?.some((h) => (h.command ?? "").includes("unerr")));
|
|
123
|
-
expect(unerrEntries.length).toBe(6);
|
|
124
|
-
const matchers = unerrEntries.map((e) => e.matcher).sort();
|
|
125
|
-
expect(matchers).toEqual(["Bash", "Edit", "Glob", "Grep", "Read", "Write"]);
|
|
126
|
-
rmSync(dir, { recursive: true, force: true });
|
|
127
|
-
});
|
|
128
|
-
});
|