pi-vcc 0.4.0
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/LICENSE +21 -0
- package/README.md +120 -0
- package/demo.gif +0 -0
- package/flow/plans/20260515-1300/plan.md +206 -0
- package/index.ts +14 -0
- package/package.json +36 -0
- package/pi-vcc-config.schema.json +131 -0
- package/scripts/audit-sessions.ts +88 -0
- package/scripts/benchmark-real-sessions.ts +25 -0
- package/scripts/compare-before-after.ts +36 -0
- package/scripts/dump-branch-output.ts +20 -0
- package/src/commands/pi-vcc.ts +33 -0
- package/src/commands/vcc-recall.ts +65 -0
- package/src/core/brief.ts +381 -0
- package/src/core/build-sections.ts +87 -0
- package/src/core/content.ts +60 -0
- package/src/core/filter-noise.ts +42 -0
- package/src/core/format-recall.ts +27 -0
- package/src/core/format.ts +56 -0
- package/src/core/lineage.ts +26 -0
- package/src/core/load-messages.ts +63 -0
- package/src/core/normalize.ts +66 -0
- package/src/core/recall-scope.ts +14 -0
- package/src/core/render-entries.ts +68 -0
- package/src/core/report.ts +237 -0
- package/src/core/sanitize.ts +5 -0
- package/src/core/search-entries.ts +230 -0
- package/src/core/settings.ts +215 -0
- package/src/core/skill-collapse.ts +35 -0
- package/src/core/summarize.ts +159 -0
- package/src/core/tool-args.ts +14 -0
- package/src/details.ts +7 -0
- package/src/extract/commits.ts +69 -0
- package/src/extract/files.ts +80 -0
- package/src/extract/goals.ts +79 -0
- package/src/extract/preferences.ts +55 -0
- package/src/extract/references.ts +214 -0
- package/src/extract/signals.ts +145 -0
- package/src/hooks/before-compact.ts +405 -0
- package/src/sections.ts +14 -0
- package/src/tools/recall.ts +109 -0
- package/src/types.ts +14 -0
- package/tests/before-compact-hook.test.ts +181 -0
- package/tests/before-compact.test.ts +140 -0
- package/tests/brief.test.ts +206 -0
- package/tests/build-sections.test.ts +90 -0
- package/tests/compile.test.ts +110 -0
- package/tests/config-integration.test.ts +107 -0
- package/tests/content.test.ts +31 -0
- package/tests/edge-cases.test.ts +368 -0
- package/tests/extract-goals.test.ts +86 -0
- package/tests/extract-preferences.test.ts +30 -0
- package/tests/extract-references.test.ts +475 -0
- package/tests/extract-signals.test.ts +561 -0
- package/tests/filter-noise.test.ts +61 -0
- package/tests/fixtures.ts +61 -0
- package/tests/format-recall.test.ts +30 -0
- package/tests/format.test.ts +91 -0
- package/tests/lineage.test.ts +33 -0
- package/tests/load-messages.test.ts +51 -0
- package/tests/normalize.test.ts +97 -0
- package/tests/real-sessions.test.ts +38 -0
- package/tests/recall-expand.test.ts +15 -0
- package/tests/recall-scope.test.ts +32 -0
- package/tests/recall-tool-scope.test.ts +67 -0
- package/tests/render-entries.test.ts +62 -0
- package/tests/report.test.ts +44 -0
- package/tests/sanitize.test.ts +24 -0
- package/tests/search-entries.test.ts +144 -0
- package/tests/settings-scaffold.test.ts +120 -0
- package/tests/settings.test.ts +32 -0
- package/tests/support/load-session.ts +23 -0
- package/tests/support/real-sessions.ts +51 -0
- package/tsconfig.json +14 -0
- package/vitest.config.ts +7 -0
|
@@ -0,0 +1,181 @@
|
|
|
1
|
+
import { describe, test, expect, beforeEach, afterEach, beforeAll, afterAll } from "vitest";
|
|
2
|
+
import { existsSync, unlinkSync, writeFileSync, readFileSync, mkdtempSync, rmSync } from "fs";
|
|
3
|
+
import { tmpdir } from "os";
|
|
4
|
+
import { join } from "path";
|
|
5
|
+
import { registerBeforeCompactHook, PI_VCC_COMPACT_INSTRUCTION } from "../src/hooks/before-compact";
|
|
6
|
+
|
|
7
|
+
let tmpDir: string;
|
|
8
|
+
let CONFIG_PATH: string;
|
|
9
|
+
const DEBUG_PATH = "/tmp/pi-vcc-debug.json";
|
|
10
|
+
|
|
11
|
+
beforeAll(() => {
|
|
12
|
+
tmpDir = mkdtempSync(join(tmpdir(), "pi-vcc-test-"));
|
|
13
|
+
CONFIG_PATH = join(tmpDir, "pi-vcc-config.json");
|
|
14
|
+
process.env.PI_VCC_CONFIG_PATH = CONFIG_PATH;
|
|
15
|
+
});
|
|
16
|
+
|
|
17
|
+
afterAll(() => {
|
|
18
|
+
delete process.env.PI_VCC_CONFIG_PATH;
|
|
19
|
+
rmSync(tmpDir, { recursive: true, force: true });
|
|
20
|
+
});
|
|
21
|
+
|
|
22
|
+
// Minimal ExtensionAPI stub: capture handler + provide ctx with mocked ui.notify
|
|
23
|
+
function createMockPi() {
|
|
24
|
+
let handler: ((event: any, ctx: any) => any) | undefined;
|
|
25
|
+
const notifyCalls: Array<{ msg: string; level: string }> = [];
|
|
26
|
+
const ctx = {
|
|
27
|
+
hasUI: true,
|
|
28
|
+
ui: {
|
|
29
|
+
notify: (msg: string, level: string) => {
|
|
30
|
+
notifyCalls.push({ msg, level });
|
|
31
|
+
},
|
|
32
|
+
},
|
|
33
|
+
};
|
|
34
|
+
return {
|
|
35
|
+
pi: {
|
|
36
|
+
on: (eventName: string, h: (e: any, c: any) => any) => {
|
|
37
|
+
if (eventName === "session_before_compact") handler = h;
|
|
38
|
+
},
|
|
39
|
+
} as any,
|
|
40
|
+
invoke: (event: any) => handler!(event, ctx),
|
|
41
|
+
notifyCalls,
|
|
42
|
+
};
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
function setConfig(cfg: Record<string, unknown>) {
|
|
46
|
+
writeFileSync(CONFIG_PATH, JSON.stringify(cfg));
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
function makeEvent(branchEntries: any[], customInstructions?: string) {
|
|
50
|
+
return {
|
|
51
|
+
type: "session_before_compact",
|
|
52
|
+
customInstructions,
|
|
53
|
+
branchEntries,
|
|
54
|
+
preparation: {
|
|
55
|
+
previousSummary: undefined,
|
|
56
|
+
fileOps: { read: [], written: [], edited: [] },
|
|
57
|
+
tokensBefore: 1000,
|
|
58
|
+
},
|
|
59
|
+
signal: new AbortController().signal,
|
|
60
|
+
};
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
const msg = (id: string, role: "user" | "assistant" | "toolResult", content = "x") => ({
|
|
64
|
+
id,
|
|
65
|
+
type: "message",
|
|
66
|
+
message: { role, content },
|
|
67
|
+
});
|
|
68
|
+
const comp = (id: string, firstKeptEntryId?: string) => ({ id, type: "compaction", firstKeptEntryId });
|
|
69
|
+
|
|
70
|
+
describe("registerBeforeCompactHook: cancel paths", () => {
|
|
71
|
+
beforeEach(() => {
|
|
72
|
+
if (existsSync(DEBUG_PATH)) unlinkSync(DEBUG_PATH);
|
|
73
|
+
});
|
|
74
|
+
afterEach(() => {
|
|
75
|
+
if (existsSync(CONFIG_PATH)) unlinkSync(CONFIG_PATH);
|
|
76
|
+
if (existsSync(DEBUG_PATH)) unlinkSync(DEBUG_PATH);
|
|
77
|
+
});
|
|
78
|
+
|
|
79
|
+
test("/pi-vcc with too few live messages cancels and notifies warning", async () => {
|
|
80
|
+
setConfig({ debug: false, overrideDefaultCompaction: false });
|
|
81
|
+
const { pi, invoke, notifyCalls } = createMockPi();
|
|
82
|
+
registerBeforeCompactHook(pi);
|
|
83
|
+
|
|
84
|
+
const entries = [msg("m1", "user"), msg("m2", "assistant")];
|
|
85
|
+
expect(await invoke(makeEvent(entries, PI_VCC_COMPACT_INSTRUCTION))).toEqual({ cancel: true });
|
|
86
|
+
expect(notifyCalls).toHaveLength(1);
|
|
87
|
+
expect(notifyCalls[0].level).toBe("warning");
|
|
88
|
+
expect(notifyCalls[0].msg).toContain("Too few messages");
|
|
89
|
+
});
|
|
90
|
+
|
|
91
|
+
test("/pi-vcc with no user message cancels with no_user_message reason", async () => {
|
|
92
|
+
setConfig({ debug: false, overrideDefaultCompaction: false });
|
|
93
|
+
const { pi, invoke, notifyCalls } = createMockPi();
|
|
94
|
+
registerBeforeCompactHook(pi);
|
|
95
|
+
|
|
96
|
+
const entries = [msg("m1", "assistant"), msg("m2", "assistant"), msg("m3", "assistant")];
|
|
97
|
+
expect(await invoke(makeEvent(entries, PI_VCC_COMPACT_INSTRUCTION))).toEqual({ cancel: true });
|
|
98
|
+
expect(notifyCalls[0].msg).toContain("no user message");
|
|
99
|
+
});
|
|
100
|
+
|
|
101
|
+
test("/compact with override=true cancels and notifies (NEW: was silent before)", async () => {
|
|
102
|
+
setConfig({ debug: false, overrideDefaultCompaction: true });
|
|
103
|
+
const { pi, invoke, notifyCalls } = createMockPi();
|
|
104
|
+
registerBeforeCompactHook(pi);
|
|
105
|
+
|
|
106
|
+
const entries = [msg("m1", "user"), msg("m2", "assistant")];
|
|
107
|
+
expect(await invoke(makeEvent(entries, undefined))).toEqual({ cancel: true });
|
|
108
|
+
expect(notifyCalls).toHaveLength(1);
|
|
109
|
+
expect(notifyCalls[0].level).toBe("warning");
|
|
110
|
+
});
|
|
111
|
+
|
|
112
|
+
test("/compact with override=false short-circuits (no notify, returns undefined)", async () => {
|
|
113
|
+
setConfig({ debug: false, overrideDefaultCompaction: false });
|
|
114
|
+
const { pi, invoke, notifyCalls } = createMockPi();
|
|
115
|
+
registerBeforeCompactHook(pi);
|
|
116
|
+
|
|
117
|
+
const entries = [msg("m1", "user"), msg("m2", "assistant")];
|
|
118
|
+
expect(await invoke(makeEvent(entries, undefined))).toBeUndefined();
|
|
119
|
+
expect(notifyCalls).toHaveLength(0);
|
|
120
|
+
});
|
|
121
|
+
|
|
122
|
+
test("debug:true writes metrics-only snapshot with no content leakage", async () => {
|
|
123
|
+
setConfig({ debug: true, overrideDefaultCompaction: false });
|
|
124
|
+
const { pi, invoke } = createMockPi();
|
|
125
|
+
registerBeforeCompactHook(pi);
|
|
126
|
+
|
|
127
|
+
const entries = [
|
|
128
|
+
msg("m1", "assistant", "SECRET_TOKEN_abc123"),
|
|
129
|
+
msg("m2", "assistant", "sensitive response"),
|
|
130
|
+
msg("m3", "assistant", "more text"),
|
|
131
|
+
];
|
|
132
|
+
expect(await invoke(makeEvent(entries, PI_VCC_COMPACT_INSTRUCTION))).toEqual({ cancel: true });
|
|
133
|
+
|
|
134
|
+
expect(existsSync(DEBUG_PATH)).toBe(true);
|
|
135
|
+
const snapshot = JSON.parse(readFileSync(DEBUG_PATH, "utf-8"));
|
|
136
|
+
expect(snapshot.cancelled).toBe(true);
|
|
137
|
+
expect(snapshot.reason).toBe("no_user_message");
|
|
138
|
+
expect(snapshot.isPiVcc).toBe(true);
|
|
139
|
+
|
|
140
|
+
// No content leakage
|
|
141
|
+
const serialized = JSON.stringify(snapshot);
|
|
142
|
+
expect(serialized).not.toContain("SECRET_TOKEN_abc123");
|
|
143
|
+
expect(serialized).not.toContain("sensitive response");
|
|
144
|
+
});
|
|
145
|
+
|
|
146
|
+
test("debug:false does NOT write snapshot", async () => {
|
|
147
|
+
setConfig({ debug: false, overrideDefaultCompaction: false });
|
|
148
|
+
const { pi, invoke } = createMockPi();
|
|
149
|
+
registerBeforeCompactHook(pi);
|
|
150
|
+
const entries = [msg("m1", "user"), msg("m2", "assistant")];
|
|
151
|
+
expect(await invoke(makeEvent(entries, PI_VCC_COMPACT_INSTRUCTION))).toEqual({ cancel: true });
|
|
152
|
+
expect(existsSync(DEBUG_PATH)).toBe(false);
|
|
153
|
+
});
|
|
154
|
+
});
|
|
155
|
+
|
|
156
|
+
describe("registerBeforeCompactHook: compact-all path", () => {
|
|
157
|
+
beforeEach(() => {
|
|
158
|
+
if (existsSync(DEBUG_PATH)) unlinkSync(DEBUG_PATH);
|
|
159
|
+
});
|
|
160
|
+
afterEach(() => {
|
|
161
|
+
if (existsSync(CONFIG_PATH)) unlinkSync(CONFIG_PATH);
|
|
162
|
+
if (existsSync(DEBUG_PATH)) unlinkSync(DEBUG_PATH);
|
|
163
|
+
});
|
|
164
|
+
|
|
165
|
+
test("single-user + autonomous tail → zero-token guard cancels compact-all", async () => {
|
|
166
|
+
setConfig({ debug: false, overrideDefaultCompaction: false });
|
|
167
|
+
const { pi, invoke, notifyCalls } = createMockPi();
|
|
168
|
+
registerBeforeCompactHook(pi);
|
|
169
|
+
|
|
170
|
+
const entries = [
|
|
171
|
+
msg("m1", "user", "go"),
|
|
172
|
+
msg("m2", "assistant", "calling tool"),
|
|
173
|
+
msg("m3", "toolResult", "result"),
|
|
174
|
+
msg("m4", "assistant", "done"),
|
|
175
|
+
];
|
|
176
|
+
const result = await invoke(makeEvent(entries, PI_VCC_COMPACT_INSTRUCTION));
|
|
177
|
+
// Zero-token guard kicks in: compactAll with tiny summary → cancel
|
|
178
|
+
expect(result).toEqual({ cancel: true });
|
|
179
|
+
expect(notifyCalls.length).toBeGreaterThanOrEqual(1);
|
|
180
|
+
});
|
|
181
|
+
});
|
|
@@ -0,0 +1,140 @@
|
|
|
1
|
+
import { describe, test, expect } from "vitest";
|
|
2
|
+
import { buildOwnCut } from "../src/hooks/before-compact";
|
|
3
|
+
|
|
4
|
+
const msg = (id: string, role: "user" | "assistant" | "toolResult", content = "x") => ({
|
|
5
|
+
id,
|
|
6
|
+
type: "message",
|
|
7
|
+
message: { role, content },
|
|
8
|
+
});
|
|
9
|
+
|
|
10
|
+
const comp = (id: string, firstKeptEntryId?: string) => ({
|
|
11
|
+
id,
|
|
12
|
+
type: "compaction",
|
|
13
|
+
firstKeptEntryId,
|
|
14
|
+
});
|
|
15
|
+
|
|
16
|
+
describe("buildOwnCut", () => {
|
|
17
|
+
test("no prior compaction: cuts at last user message", () => {
|
|
18
|
+
const r = buildOwnCut([
|
|
19
|
+
msg("m1", "user", "a"),
|
|
20
|
+
msg("m2", "assistant", "b"),
|
|
21
|
+
msg("m3", "user", "c"),
|
|
22
|
+
msg("m4", "assistant", "d"),
|
|
23
|
+
]);
|
|
24
|
+
expect(r.ok).toBe(true);
|
|
25
|
+
if (!r.ok) return;
|
|
26
|
+
expect(r.firstKeptEntryId).toBe("m3");
|
|
27
|
+
expect(r.messages).toHaveLength(2);
|
|
28
|
+
expect(r.compactAll).toBe(false);
|
|
29
|
+
});
|
|
30
|
+
|
|
31
|
+
test("cancels with too_few_live_messages when liveMessages <= 2", () => {
|
|
32
|
+
const r = buildOwnCut([
|
|
33
|
+
comp("c1", "m1"),
|
|
34
|
+
msg("m1", "user", "x"),
|
|
35
|
+
msg("m2", "assistant", "y"),
|
|
36
|
+
]);
|
|
37
|
+
expect(r.ok).toBe(false);
|
|
38
|
+
if (r.ok) return;
|
|
39
|
+
expect(r.reason).toBe("too_few_live_messages");
|
|
40
|
+
});
|
|
41
|
+
|
|
42
|
+
test("orphan firstKeptEntryId triggers recovery (collect after compaction)", () => {
|
|
43
|
+
// Prev compaction set firstKeptEntryId to a non-existent id (e.g. "" sentinel
|
|
44
|
+
// from a previous compact-all). Recovery should collect msgs after compaction.
|
|
45
|
+
const r = buildOwnCut([
|
|
46
|
+
msg("old1", "user", "old"),
|
|
47
|
+
msg("old2", "assistant", "old"),
|
|
48
|
+
comp("c1", "ORPHAN_ID"),
|
|
49
|
+
msg("m1", "user", "a"),
|
|
50
|
+
msg("m2", "assistant", "b"),
|
|
51
|
+
msg("m3", "user", "c"),
|
|
52
|
+
msg("m4", "assistant", "d"),
|
|
53
|
+
]);
|
|
54
|
+
expect(r.ok).toBe(true);
|
|
55
|
+
if (!r.ok) return;
|
|
56
|
+
expect(r.firstKeptEntryId).toBe("m3");
|
|
57
|
+
expect(r.messages).toHaveLength(2);
|
|
58
|
+
});
|
|
59
|
+
|
|
60
|
+
test("resumes from firstKeptEntryId after prior compaction", () => {
|
|
61
|
+
const r = buildOwnCut([
|
|
62
|
+
msg("old1", "user", "old"),
|
|
63
|
+
msg("old2", "assistant", "old"),
|
|
64
|
+
comp("c1", "m1"),
|
|
65
|
+
msg("m1", "user", "a"),
|
|
66
|
+
msg("m2", "assistant", "b"),
|
|
67
|
+
msg("m3", "user", "c"),
|
|
68
|
+
msg("m4", "assistant", "d"),
|
|
69
|
+
]);
|
|
70
|
+
expect(r.ok).toBe(true);
|
|
71
|
+
if (!r.ok) return;
|
|
72
|
+
expect(r.firstKeptEntryId).toBe("m3");
|
|
73
|
+
expect(r.messages).toHaveLength(2);
|
|
74
|
+
});
|
|
75
|
+
|
|
76
|
+
test("single user prompt + autonomous tail: compact all", () => {
|
|
77
|
+
// The Discord scenario: user types 1 prompt, agent runs autonomously
|
|
78
|
+
// (assistant + toolResult interleaved). No user > idx 0.
|
|
79
|
+
const r = buildOwnCut([
|
|
80
|
+
msg("m1", "user", "go"),
|
|
81
|
+
msg("m2", "assistant", "calling tool"),
|
|
82
|
+
msg("m3", "toolResult", "result"),
|
|
83
|
+
msg("m4", "assistant", "more"),
|
|
84
|
+
msg("m5", "toolResult", "result2"),
|
|
85
|
+
msg("m6", "assistant", "done"),
|
|
86
|
+
]);
|
|
87
|
+
expect(r.ok).toBe(true);
|
|
88
|
+
if (!r.ok) return;
|
|
89
|
+
expect(r.compactAll).toBe(true);
|
|
90
|
+
expect(r.firstKeptEntryId).toBe("");
|
|
91
|
+
expect(r.messages).toHaveLength(6);
|
|
92
|
+
});
|
|
93
|
+
|
|
94
|
+
test("no_user_message when no user role at all", () => {
|
|
95
|
+
const r = buildOwnCut([
|
|
96
|
+
msg("m1", "assistant", "a"),
|
|
97
|
+
msg("m2", "assistant", "b"),
|
|
98
|
+
msg("m3", "assistant", "c"),
|
|
99
|
+
]);
|
|
100
|
+
expect(r.ok).toBe(false);
|
|
101
|
+
if (r.ok) return;
|
|
102
|
+
expect(r.reason).toBe("no_user_message");
|
|
103
|
+
});
|
|
104
|
+
|
|
105
|
+
test("compact-all then more chat: orphan recovery + normal cut", () => {
|
|
106
|
+
// After a compact-all (firstKeptEntryId=""), user chats more turns,
|
|
107
|
+
// next compaction should orphan-recover and find multiple users.
|
|
108
|
+
const r = buildOwnCut([
|
|
109
|
+
msg("o1", "user", "old"),
|
|
110
|
+
msg("o2", "assistant", "old"),
|
|
111
|
+
comp("c1", ""), // sentinel from prior compact-all
|
|
112
|
+
msg("u1", "user", "new1"),
|
|
113
|
+
msg("a1", "assistant", "reply1"),
|
|
114
|
+
msg("u2", "user", "new2"),
|
|
115
|
+
msg("a2", "assistant", "reply2"),
|
|
116
|
+
msg("u3", "user", "new3"),
|
|
117
|
+
msg("a3", "assistant", "reply3"),
|
|
118
|
+
]);
|
|
119
|
+
expect(r.ok).toBe(true);
|
|
120
|
+
if (!r.ok) return;
|
|
121
|
+
expect(r.compactAll).toBe(false);
|
|
122
|
+
expect(r.firstKeptEntryId).toBe("u3");
|
|
123
|
+
expect(r.messages).toHaveLength(4); // u1, a1, u2, a2
|
|
124
|
+
});
|
|
125
|
+
|
|
126
|
+
test("compact-all then single user msg + autonomous: compact all again", () => {
|
|
127
|
+
const r = buildOwnCut([
|
|
128
|
+
msg("o1", "user", "old"),
|
|
129
|
+
comp("c1", ""),
|
|
130
|
+
msg("u1", "user", "okay"),
|
|
131
|
+
msg("a1", "assistant", "x"),
|
|
132
|
+
msg("t1", "toolResult", "y"),
|
|
133
|
+
msg("a2", "assistant", "z"),
|
|
134
|
+
]);
|
|
135
|
+
expect(r.ok).toBe(true);
|
|
136
|
+
if (!r.ok) return;
|
|
137
|
+
expect(r.compactAll).toBe(true);
|
|
138
|
+
expect(r.firstKeptEntryId).toBe("");
|
|
139
|
+
});
|
|
140
|
+
});
|
|
@@ -0,0 +1,206 @@
|
|
|
1
|
+
import { describe, it, expect } from "vitest";
|
|
2
|
+
import { compileBrief } from "../src/core/brief";
|
|
3
|
+
import type { NormalizedBlock } from "../src/types";
|
|
4
|
+
|
|
5
|
+
describe("compileBrief", () => {
|
|
6
|
+
it("returns empty string for no blocks", () => {
|
|
7
|
+
expect(compileBrief([])).toBe("");
|
|
8
|
+
});
|
|
9
|
+
|
|
10
|
+
it("renders user and assistant text", () => {
|
|
11
|
+
const blocks: NormalizedBlock[] = [
|
|
12
|
+
{ kind: "user", text: "fix auth bug" },
|
|
13
|
+
{ kind: "assistant", text: "Let me look at the auth module." },
|
|
14
|
+
];
|
|
15
|
+
const r = compileBrief(blocks);
|
|
16
|
+
expect(r).toContain("[user]");
|
|
17
|
+
expect(r).toContain("fix auth bug");
|
|
18
|
+
expect(r).toContain("[assistant]");
|
|
19
|
+
expect(r).toContain("Let me look at the auth module.");
|
|
20
|
+
});
|
|
21
|
+
|
|
22
|
+
it("strips filler prefixes but preserves meaningful lead-ins", () => {
|
|
23
|
+
const blocks: NormalizedBlock[] = [
|
|
24
|
+
{ kind: "assistant", text: "Okay, I found the root cause." },
|
|
25
|
+
{ kind: "assistant", text: "Actually, the issue is in middleware." },
|
|
26
|
+
{ kind: "assistant", text: "Let me check the logs." },
|
|
27
|
+
];
|
|
28
|
+
const r = compileBrief(blocks);
|
|
29
|
+
expect(r).toContain("I found the root cause.");
|
|
30
|
+
expect(r).toContain("the issue is in middleware.");
|
|
31
|
+
expect(r).toContain("Let me check the logs.");
|
|
32
|
+
});
|
|
33
|
+
|
|
34
|
+
it("collapses tool calls to one-liners under [assistant]", () => {
|
|
35
|
+
const blocks: NormalizedBlock[] = [
|
|
36
|
+
{ kind: "assistant", text: "Let me check." },
|
|
37
|
+
{ kind: "tool_call", name: "Read", args: { file_path: "auth.ts" } },
|
|
38
|
+
{ kind: "tool_call", name: "Edit", args: { file_path: "auth.ts" } },
|
|
39
|
+
];
|
|
40
|
+
const r = compileBrief(blocks);
|
|
41
|
+
expect(r).toContain('* Read "auth.ts"');
|
|
42
|
+
expect(r).toContain('* Edit "auth.ts"');
|
|
43
|
+
// Should merge into single [assistant] section
|
|
44
|
+
const matches = r.match(/\[assistant\]/g);
|
|
45
|
+
expect(matches?.length).toBe(1);
|
|
46
|
+
});
|
|
47
|
+
|
|
48
|
+
it("hides non-error tool results", () => {
|
|
49
|
+
const blocks: NormalizedBlock[] = [
|
|
50
|
+
{ kind: "tool_result", name: "Read", text: "const x = 1;\nconst y = 2;\n// lots of code", isError: false },
|
|
51
|
+
];
|
|
52
|
+
const r = compileBrief(blocks);
|
|
53
|
+
expect(r).toBe("");
|
|
54
|
+
});
|
|
55
|
+
|
|
56
|
+
it("shows tool errors with first line", () => {
|
|
57
|
+
const blocks: NormalizedBlock[] = [
|
|
58
|
+
{ kind: "tool_result", name: "bash", text: "FAIL auth.test.ts\nexpected 200 got 401", isError: true },
|
|
59
|
+
];
|
|
60
|
+
const r = compileBrief(blocks);
|
|
61
|
+
expect(r).toContain("[tool_error] bash");
|
|
62
|
+
expect(r).toContain("FAIL auth.test.ts");
|
|
63
|
+
});
|
|
64
|
+
|
|
65
|
+
it("hides thinking blocks", () => {
|
|
66
|
+
const blocks: NormalizedBlock[] = [
|
|
67
|
+
{ kind: "thinking", text: "Let me think about this...", redacted: false },
|
|
68
|
+
{ kind: "assistant", text: "Here's what I found." },
|
|
69
|
+
];
|
|
70
|
+
const r = compileBrief(blocks);
|
|
71
|
+
expect(r).not.toContain("think");
|
|
72
|
+
expect(r).toContain("Here's what I found.");
|
|
73
|
+
});
|
|
74
|
+
|
|
75
|
+
it("merges adjacent assistant sections", () => {
|
|
76
|
+
const blocks: NormalizedBlock[] = [
|
|
77
|
+
{ kind: "assistant", text: "First part." },
|
|
78
|
+
{ kind: "tool_call", name: "Read", args: { file_path: "a.ts" } },
|
|
79
|
+
// No user/tool_result between these — should merge
|
|
80
|
+
{ kind: "assistant", text: "Second part." },
|
|
81
|
+
{ kind: "tool_call", name: "Read", args: { file_path: "b.ts" } },
|
|
82
|
+
];
|
|
83
|
+
const r = compileBrief(blocks);
|
|
84
|
+
const matches = r.match(/\[assistant\]/g);
|
|
85
|
+
expect(matches?.length).toBe(1);
|
|
86
|
+
});
|
|
87
|
+
|
|
88
|
+
it("does NOT merge assistant after user", () => {
|
|
89
|
+
const blocks: NormalizedBlock[] = [
|
|
90
|
+
{ kind: "assistant", text: "First." },
|
|
91
|
+
{ kind: "user", text: "Next task." },
|
|
92
|
+
{ kind: "assistant", text: "Second." },
|
|
93
|
+
];
|
|
94
|
+
const r = compileBrief(blocks);
|
|
95
|
+
const matches = r.match(/\[assistant\]/g);
|
|
96
|
+
expect(matches?.length).toBe(2);
|
|
97
|
+
});
|
|
98
|
+
|
|
99
|
+
it("truncates long user text", () => {
|
|
100
|
+
const longText = Array.from({ length: 300 }, (_, i) => `word${i}`).join(" ");
|
|
101
|
+
const blocks: NormalizedBlock[] = [
|
|
102
|
+
{ kind: "user", text: longText },
|
|
103
|
+
];
|
|
104
|
+
const r = compileBrief(blocks);
|
|
105
|
+
expect(r).toContain("(truncated)");
|
|
106
|
+
expect(r).not.toContain("word299");
|
|
107
|
+
});
|
|
108
|
+
|
|
109
|
+
it("truncates long assistant text", () => {
|
|
110
|
+
const longText = Array.from({ length: 300 }, (_, i) => `word${i}`).join(" ");
|
|
111
|
+
const blocks: NormalizedBlock[] = [
|
|
112
|
+
{ kind: "assistant", text: longText },
|
|
113
|
+
];
|
|
114
|
+
const r = compileBrief(blocks);
|
|
115
|
+
expect(r).toContain("(truncated)");
|
|
116
|
+
expect(r).not.toContain("word299");
|
|
117
|
+
});
|
|
118
|
+
|
|
119
|
+
it("renders a realistic conversation flow", () => {
|
|
120
|
+
const blocks: NormalizedBlock[] = [
|
|
121
|
+
{ kind: "user", text: "fix the login bug" },
|
|
122
|
+
{ kind: "thinking", text: "I need to check...", redacted: false },
|
|
123
|
+
{ kind: "assistant", text: "Let me investigate." },
|
|
124
|
+
{ kind: "tool_call", name: "Read", args: { file_path: "login.ts" } },
|
|
125
|
+
{ kind: "tool_result", name: "Read", text: "export function login() { ... }", isError: false },
|
|
126
|
+
{ kind: "tool_call", name: "bash", args: { command: "npm test" } },
|
|
127
|
+
{ kind: "tool_result", name: "bash", text: "FAIL: login test\nExpected true, got false", isError: true },
|
|
128
|
+
{ kind: "assistant", text: "The test is failing because..." },
|
|
129
|
+
{ kind: "tool_call", name: "Edit", args: { file_path: "login.ts" } },
|
|
130
|
+
{ kind: "tool_result", name: "Edit", text: "File edited successfully", isError: false },
|
|
131
|
+
{ kind: "user", text: "test lại đi" },
|
|
132
|
+
{ kind: "assistant", text: "Running tests again." },
|
|
133
|
+
{ kind: "tool_call", name: "bash", args: { command: "npm test" } },
|
|
134
|
+
{ kind: "tool_result", name: "bash", text: "All tests passed", isError: false },
|
|
135
|
+
];
|
|
136
|
+
const r = compileBrief(blocks);
|
|
137
|
+
|
|
138
|
+
// Check structure
|
|
139
|
+
expect(r).toContain("[user]\nfix the login bug");
|
|
140
|
+
expect(r).toContain('[assistant]\nLet me investigate.\n* Read "login.ts"');
|
|
141
|
+
expect(r).toContain("[tool_error] bash\nFAIL: login test");
|
|
142
|
+
expect(r).toContain('[assistant]\nThe test is failing because...\n* Edit "login.ts"');
|
|
143
|
+
expect(r).toContain("[user]\ntest lại đi");
|
|
144
|
+
expect(r).toContain('[assistant]\nRunning tests again.\n* bash "npm test"');
|
|
145
|
+
|
|
146
|
+
// Hidden content
|
|
147
|
+
expect(r).not.toContain("think");
|
|
148
|
+
expect(r).not.toContain("export function login");
|
|
149
|
+
expect(r).not.toContain("File edited successfully");
|
|
150
|
+
expect(r).not.toContain("All tests passed");
|
|
151
|
+
});
|
|
152
|
+
|
|
153
|
+
// ── noise filtering tests (aligned with VCC) ──
|
|
154
|
+
|
|
155
|
+
|
|
156
|
+
|
|
157
|
+
|
|
158
|
+
|
|
159
|
+
|
|
160
|
+
|
|
161
|
+
|
|
162
|
+
|
|
163
|
+
it("suppresses blank lines between consecutive tool-only sections", () => {
|
|
164
|
+
const blocks: NormalizedBlock[] = [
|
|
165
|
+
{ kind: "assistant", text: "Checking files." },
|
|
166
|
+
{ kind: "tool_call", name: "Read", args: { file_path: "a.ts" } },
|
|
167
|
+
{ kind: "tool_result", name: "Read", text: "...", isError: false },
|
|
168
|
+
// tool_result hidden → next tool_call starts new assistant section
|
|
169
|
+
// but since both are tool-only, no blank line between
|
|
170
|
+
{ kind: "tool_call", name: "Read", args: { file_path: "b.ts" } },
|
|
171
|
+
{ kind: "tool_result", name: "Read", text: "...", isError: false },
|
|
172
|
+
];
|
|
173
|
+
const r = compileBrief(blocks);
|
|
174
|
+
// The first assistant section has text + tool, so it's NOT tool-only
|
|
175
|
+
// The second would be tool-only but merges into the first (adjacent assistant)
|
|
176
|
+
// So all under one [assistant]
|
|
177
|
+
expect(r.match(/\[assistant\]/g)?.length).toBe(1);
|
|
178
|
+
});
|
|
179
|
+
|
|
180
|
+
it("caps tool calls per [assistant] turn at 8 (keep tail)", () => {
|
|
181
|
+
const blocks: NormalizedBlock[] = [
|
|
182
|
+
{ kind: "assistant", text: "Working." },
|
|
183
|
+
];
|
|
184
|
+
for (let i = 1; i <= 12; i++) {
|
|
185
|
+
blocks.push({ kind: "tool_call", name: "bash", args: { command: `echo ${i}` } });
|
|
186
|
+
}
|
|
187
|
+
const r = compileBrief(blocks);
|
|
188
|
+
expect(r).toContain("(4 earlier tool-call entries omitted)");
|
|
189
|
+
// Last 8 (5..12) kept; first 4 dropped
|
|
190
|
+
expect(r).not.toContain("echo 1\"");
|
|
191
|
+
expect(r).not.toContain("echo 4\"");
|
|
192
|
+
expect(r).toContain("echo 5");
|
|
193
|
+
expect(r).toContain("echo 12");
|
|
194
|
+
});
|
|
195
|
+
|
|
196
|
+
it("does not cap when tool calls per turn <= 8", () => {
|
|
197
|
+
const blocks: NormalizedBlock[] = [{ kind: "assistant", text: "ok" }];
|
|
198
|
+
for (let i = 1; i <= 8; i++) {
|
|
199
|
+
blocks.push({ kind: "tool_call", name: "bash", args: { command: `c${i}` } });
|
|
200
|
+
}
|
|
201
|
+
const r = compileBrief(blocks);
|
|
202
|
+
expect(r).not.toContain("entries omitted");
|
|
203
|
+
expect(r).toContain("c1");
|
|
204
|
+
expect(r).toContain("c8");
|
|
205
|
+
});
|
|
206
|
+
});
|
|
@@ -0,0 +1,90 @@
|
|
|
1
|
+
import { describe, it, expect } from "vitest";
|
|
2
|
+
import { buildSections } from "../src/core/build-sections";
|
|
3
|
+
import type { NormalizedBlock } from "../src/types";
|
|
4
|
+
|
|
5
|
+
describe("buildSections", () => {
|
|
6
|
+
it("returns all-empty for no blocks", () => {
|
|
7
|
+
const r = buildSections({ blocks: [] });
|
|
8
|
+
expect(r.sessionGoal).toEqual([]);
|
|
9
|
+
expect(r.outstandingContext).toEqual([]);
|
|
10
|
+
expect(r.briefTranscript).toBe("");
|
|
11
|
+
expect(r.references).toEqual([]);
|
|
12
|
+
});
|
|
13
|
+
|
|
14
|
+
it("populates sections from realistic blocks", () => {
|
|
15
|
+
const blocks: NormalizedBlock[] = [
|
|
16
|
+
{ kind: "user", text: "Fix the auth bug" },
|
|
17
|
+
{ kind: "tool_call", name: "Read", args: { file_path: "auth.ts" } },
|
|
18
|
+
{ kind: "tool_result", name: "Read", text: "const x = 1;", isError: false },
|
|
19
|
+
{ kind: "tool_call", name: "Edit", args: { file_path: "auth.ts" } },
|
|
20
|
+
{ kind: "tool_result", name: "Edit", text: "ok", isError: false },
|
|
21
|
+
{ kind: "assistant", text: "- run tests next" },
|
|
22
|
+
];
|
|
23
|
+
const r = buildSections({ blocks });
|
|
24
|
+
expect(r.sessionGoal).toContain("Fix the auth bug");
|
|
25
|
+
expect(r.briefTranscript).toContain('[user]');
|
|
26
|
+
expect(r.briefTranscript).toContain('* Read "auth.ts"');
|
|
27
|
+
expect(r.briefTranscript).toContain('* Edit "auth.ts"');
|
|
28
|
+
});
|
|
29
|
+
|
|
30
|
+
it("captures outstanding context from errors", () => {
|
|
31
|
+
const blocks: NormalizedBlock[] = [
|
|
32
|
+
{ kind: "tool_result", name: "bash", text: "FAIL: test broken\ndetails here", isError: true },
|
|
33
|
+
];
|
|
34
|
+
const r = buildSections({ blocks });
|
|
35
|
+
expect(r.outstandingContext.length).toBeGreaterThan(0);
|
|
36
|
+
expect(r.outstandingContext[0]).toContain("FAIL");
|
|
37
|
+
});
|
|
38
|
+
|
|
39
|
+
it("brief transcript hides tool results but shows errors", () => {
|
|
40
|
+
const blocks: NormalizedBlock[] = [
|
|
41
|
+
{ kind: "tool_result", name: "Read", text: "lots of code here ...", isError: false },
|
|
42
|
+
{ kind: "tool_result", name: "bash", text: "Command not found", isError: true },
|
|
43
|
+
];
|
|
44
|
+
const r = buildSections({ blocks });
|
|
45
|
+
expect(r.briefTranscript).not.toContain("lots of code");
|
|
46
|
+
expect(r.briefTranscript).toContain("[tool_error] bash");
|
|
47
|
+
expect(r.briefTranscript).toContain("Command not found");
|
|
48
|
+
});
|
|
49
|
+
|
|
50
|
+
it("brief transcript merges adjacent assistant sections", () => {
|
|
51
|
+
const blocks: NormalizedBlock[] = [
|
|
52
|
+
{ kind: "assistant", text: "Part one." },
|
|
53
|
+
{ kind: "tool_call", name: "Read", args: { file_path: "a.ts" } },
|
|
54
|
+
{ kind: "assistant", text: "Part two." },
|
|
55
|
+
];
|
|
56
|
+
const r = buildSections({ blocks });
|
|
57
|
+
const matches = r.briefTranscript.match(/\[assistant\]/g);
|
|
58
|
+
expect(matches?.length).toBe(1);
|
|
59
|
+
});
|
|
60
|
+
|
|
61
|
+
// ── References integration ──
|
|
62
|
+
|
|
63
|
+
it("populates references from user blocks with URLs", () => {
|
|
64
|
+
const blocks: NormalizedBlock[] = [
|
|
65
|
+
{ kind: "user", text: "Check https://docs.example.com/api for the API docs" },
|
|
66
|
+
];
|
|
67
|
+
const r = buildSections({ blocks });
|
|
68
|
+
expect(r.references.length).toBeGreaterThan(0);
|
|
69
|
+
expect(r.references[0]).toContain("URL:");
|
|
70
|
+
expect(r.references[0]).toContain("https://docs.example.com/api");
|
|
71
|
+
});
|
|
72
|
+
|
|
73
|
+
it("populates references with GitHub refs", () => {
|
|
74
|
+
const blocks: NormalizedBlock[] = [
|
|
75
|
+
{ kind: "user", text: "Fix issue #42 and merge PR #7" },
|
|
76
|
+
];
|
|
77
|
+
const r = buildSections({ blocks });
|
|
78
|
+
expect(r.references.some(ref => ref.includes("#42"))).toBe(true);
|
|
79
|
+
expect(r.references.some(ref => ref.includes("PR #7"))).toBe(true);
|
|
80
|
+
});
|
|
81
|
+
|
|
82
|
+
it("returns empty references when no user/assistant blocks have refs", () => {
|
|
83
|
+
const blocks: NormalizedBlock[] = [
|
|
84
|
+
{ kind: "user", text: "Fix the auth bug" },
|
|
85
|
+
{ kind: "tool_result", name: "bash", text: "https://example.com in output", isError: false },
|
|
86
|
+
];
|
|
87
|
+
const r = buildSections({ blocks });
|
|
88
|
+
expect(r.references).toEqual([]);
|
|
89
|
+
});
|
|
90
|
+
});
|