@polderlabs/bizar-plugin 0.5.4
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 +448 -0
- package/bun.lock +88 -0
- package/index.ts +1113 -0
- package/package.json +42 -0
- package/scripts/check-forbidden-imports.sh +33 -0
- package/src/background-state.ts +463 -0
- package/src/background.ts +964 -0
- package/src/commands-impl.ts +369 -0
- package/src/commands.ts +880 -0
- package/src/event-stream.ts +574 -0
- package/src/fingerprint.ts +120 -0
- package/src/handoff.ts +79 -0
- package/src/http-client.ts +467 -0
- package/src/logger.ts +144 -0
- package/src/loop.ts +176 -0
- package/src/options.ts +421 -0
- package/src/plan-fs.ts +323 -0
- package/src/report.ts +178 -0
- package/src/research-prompt.ts +35 -0
- package/src/serve.ts +476 -0
- package/src/settings.ts +349 -0
- package/src/state.ts +298 -0
- package/src/tools/bg-collect.ts +104 -0
- package/src/tools/bg-get-comments.ts +239 -0
- package/src/tools/bg-kill.ts +87 -0
- package/src/tools/bg-spawn.ts +263 -0
- package/src/tools/bg-status.ts +99 -0
- package/src/tools/plan-action.ts +767 -0
- package/src/tools/wait-for-feedback.ts +402 -0
- package/tests/attach-handler-bug.test.ts +166 -0
- package/tests/background-state.test.ts +277 -0
- package/tests/background.test.ts +402 -0
- package/tests/block.test.ts +193 -0
- package/tests/canonical-key-order.test.ts +71 -0
- package/tests/commands-impl.test.ts +442 -0
- package/tests/commands.test.ts +548 -0
- package/tests/config.test.ts +122 -0
- package/tests/dispose.test.ts +336 -0
- package/tests/event-stream.test.ts +409 -0
- package/tests/event.test.ts +262 -0
- package/tests/fingerprint.test.ts +161 -0
- package/tests/http-client.test.ts +403 -0
- package/tests/init-helpers.test.ts +203 -0
- package/tests/integration/slash-command.test.ts +348 -0
- package/tests/integration/tool-routing.test.ts +314 -0
- package/tests/loop.test.ts +397 -0
- package/tests/options.test.ts +274 -0
- package/tests/serve.test.ts +335 -0
- package/tests/settings.test.ts +351 -0
- package/tests/stall-think.test.ts +749 -0
- package/tests/state.test.ts +275 -0
- package/tests/tools/bg-collect.test.ts +337 -0
- package/tests/tools/bg-get-comments.test.ts +485 -0
- package/tests/tools/bg-kill.test.ts +231 -0
- package/tests/tools/bg-spawn.test.ts +311 -0
- package/tests/tools/bg-status.test.ts +216 -0
- package/tests/tools/plan-action.test.ts +599 -0
- package/tests/tools/wait-for-feedback.test.ts +390 -0
- package/tsconfig.json +29 -0
|
@@ -0,0 +1,161 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* fingerprint.test.ts
|
|
3
|
+
*
|
|
4
|
+
* Tests for stable fingerprint computation per §5.1, §5.3.
|
|
5
|
+
*/
|
|
6
|
+
|
|
7
|
+
import { describe, test, expect } from "bun:test";
|
|
8
|
+
import { fingerprint } from "../src/fingerprint";
|
|
9
|
+
import { mkdirSync, rmSync, writeFileSync } from "node:fs";
|
|
10
|
+
import path from "node:path";
|
|
11
|
+
|
|
12
|
+
/** Use a real temp directory as the worktree for path normalization tests */
|
|
13
|
+
const WORKTREE = "/tmp/bizar-fingerprint-test-worktree";
|
|
14
|
+
const OUTSIDE_TREE = "/tmp/bizar-outside-worktree";
|
|
15
|
+
|
|
16
|
+
function setupWorktree() {
|
|
17
|
+
try {
|
|
18
|
+
mkdirSync(WORKTREE, { recursive: true });
|
|
19
|
+
mkdirSync(OUTSIDE_TREE, { recursive: true });
|
|
20
|
+
// Create a file inside the worktree for path tests
|
|
21
|
+
writeFileSync(path.join(WORKTREE, "file.txt"), "x");
|
|
22
|
+
} catch {
|
|
23
|
+
// dir may already exist
|
|
24
|
+
}
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
function teardownWorktree() {
|
|
28
|
+
try { rmSync(WORKTREE, { recursive: true, force: true }); } catch { /* ok */ }
|
|
29
|
+
try { rmSync(OUTSIDE_TREE, { recursive: true, force: true }); } catch { /* ok */ }
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
setupWorktree();
|
|
33
|
+
|
|
34
|
+
describe("fingerprint — stable hash", () => {
|
|
35
|
+
test("same args produce the same fingerprint", () => {
|
|
36
|
+
const args = { path: "/tmp/foo.ts", recursive: false };
|
|
37
|
+
const a = fingerprint("read", args, WORKTREE);
|
|
38
|
+
const b = fingerprint("read", args, WORKTREE);
|
|
39
|
+
expect(a).toBe(b);
|
|
40
|
+
});
|
|
41
|
+
|
|
42
|
+
test("different tool name produces different fingerprint", () => {
|
|
43
|
+
const args = { path: "/tmp/foo.ts" };
|
|
44
|
+
const a = fingerprint("read", args, WORKTREE);
|
|
45
|
+
const b = fingerprint("edit", args, WORKTREE);
|
|
46
|
+
expect(a).not.toBe(b);
|
|
47
|
+
});
|
|
48
|
+
|
|
49
|
+
test("different args produce different fingerprint", () => {
|
|
50
|
+
const a = fingerprint("read", { path: "/tmp/foo.ts" }, WORKTREE);
|
|
51
|
+
const b = fingerprint("read", { path: "/tmp/bar.ts" }, WORKTREE);
|
|
52
|
+
expect(a).not.toBe(b);
|
|
53
|
+
});
|
|
54
|
+
|
|
55
|
+
test("empty args object produces stable hash", () => {
|
|
56
|
+
const a = fingerprint("read", {}, WORKTREE);
|
|
57
|
+
const b = fingerprint("read", {}, WORKTREE);
|
|
58
|
+
expect(a).toBe(b);
|
|
59
|
+
expect(a).toHaveLength(64); // sha256 hex
|
|
60
|
+
});
|
|
61
|
+
});
|
|
62
|
+
|
|
63
|
+
describe("fingerprint — path normalization", () => {
|
|
64
|
+
test("in-worktree absolute path becomes worktree-relative", () => {
|
|
65
|
+
const inWorktree = path.join(WORKTREE, "src", "index.ts");
|
|
66
|
+
const fp1 = fingerprint("read", { path: inWorktree }, WORKTREE);
|
|
67
|
+
const fp2 = fingerprint("read", { path: "src/index.ts" }, WORKTREE);
|
|
68
|
+
expect(fp1).toBe(fp2);
|
|
69
|
+
});
|
|
70
|
+
|
|
71
|
+
test("out-of-worktree absolute path becomes per-path stable hash (not global sentinel)", () => {
|
|
72
|
+
const outside = path.join(OUTSIDE_TREE, "secret.txt");
|
|
73
|
+
const fp1 = fingerprint("read", { path: outside }, WORKTREE);
|
|
74
|
+
const fp2 = fingerprint("read", { path: outside }, WORKTREE);
|
|
75
|
+
expect(fp1).toBe(fp2);
|
|
76
|
+
// Must NOT collide with a different outside path
|
|
77
|
+
const different = "/tmp/different/path/file.txt";
|
|
78
|
+
const fp3 = fingerprint("read", { path: different }, WORKTREE);
|
|
79
|
+
expect(fp1).not.toBe(fp3);
|
|
80
|
+
});
|
|
81
|
+
|
|
82
|
+
test("two distinct absolute paths outside worktree produce two distinct hashes", () => {
|
|
83
|
+
const pathA = path.join(OUTSIDE_TREE, "a.txt");
|
|
84
|
+
const pathB = path.join(OUTSIDE_TREE, "b.txt");
|
|
85
|
+
const fpA = fingerprint("read", { path: pathA }, WORKTREE);
|
|
86
|
+
const fpB = fingerprint("read", { path: pathB }, WORKTREE);
|
|
87
|
+
expect(fpA).not.toBe(fpB);
|
|
88
|
+
});
|
|
89
|
+
});
|
|
90
|
+
|
|
91
|
+
describe("fingerprint — noise field stripping", () => {
|
|
92
|
+
test("strips timestamp fields", () => {
|
|
93
|
+
const a = fingerprint("read", { path: "/tmp/foo.ts", createdAt: 1234567890 }, WORKTREE);
|
|
94
|
+
const b = fingerprint("read", { path: "/tmp/foo.ts", createdAt: 9999999999 }, WORKTREE);
|
|
95
|
+
expect(a).toBe(b);
|
|
96
|
+
});
|
|
97
|
+
|
|
98
|
+
test("strips updatedAt timestamp fields", () => {
|
|
99
|
+
const a = fingerprint("read", { path: "/tmp/foo.ts", updatedAt: "2024-01-01T00:00:00Z" }, WORKTREE);
|
|
100
|
+
const b = fingerprint("read", { path: "/tmp/foo.ts", updatedAt: "2025-12-31T23:59:59Z" }, WORKTREE);
|
|
101
|
+
expect(a).toBe(b);
|
|
102
|
+
});
|
|
103
|
+
|
|
104
|
+
test("strips id field", () => {
|
|
105
|
+
const a = fingerprint("read", { path: "/tmp/foo.ts", id: "abc123" }, WORKTREE);
|
|
106
|
+
const b = fingerprint("read", { path: "/tmp/foo.ts", id: "xyz789" }, WORKTREE);
|
|
107
|
+
expect(a).toBe(b);
|
|
108
|
+
});
|
|
109
|
+
|
|
110
|
+
test("strips uuid field", () => {
|
|
111
|
+
const a = fingerprint("read", { path: "/tmp/foo.ts", uuid: "550e8400-e29b-41d4-a716-446655440000" }, WORKTREE);
|
|
112
|
+
const b = fingerprint("read", { path: "/tmp/foo.ts", uuid: "6ba7b810-9dad-11d1-80b4-00c04fd430c8" }, WORKTREE);
|
|
113
|
+
expect(a).toBe(b);
|
|
114
|
+
});
|
|
115
|
+
|
|
116
|
+
test("strips nonce field", () => {
|
|
117
|
+
const a = fingerprint("read", { path: "/tmp/foo.ts", nonce: "random1" }, WORKTREE);
|
|
118
|
+
const b = fingerprint("read", { path: "/tmp/foo.ts", nonce: "random2" }, WORKTREE);
|
|
119
|
+
expect(a).toBe(b);
|
|
120
|
+
});
|
|
121
|
+
|
|
122
|
+
test("strips cwd field entirely", () => {
|
|
123
|
+
const a = fingerprint("read", { path: "/tmp/foo.ts", cwd: "/home/user" }, WORKTREE);
|
|
124
|
+
const b = fingerprint("read", { path: "/tmp/foo.ts", cwd: "/completely/different" }, WORKTREE);
|
|
125
|
+
expect(a).toBe(b);
|
|
126
|
+
});
|
|
127
|
+
});
|
|
128
|
+
|
|
129
|
+
describe("fingerprint — nested objects", () => {
|
|
130
|
+
test("nested objects are normalized recursively", () => {
|
|
131
|
+
const a = fingerprint("edit", {
|
|
132
|
+
meta: { author: "Alice", timestamp: 1000 },
|
|
133
|
+
path: "/tmp/foo.ts",
|
|
134
|
+
}, WORKTREE);
|
|
135
|
+
const b = fingerprint("edit", {
|
|
136
|
+
path: "/tmp/foo.ts",
|
|
137
|
+
meta: { timestamp: 9999, author: "Alice" },
|
|
138
|
+
}, WORKTREE);
|
|
139
|
+
expect(a).toBe(b);
|
|
140
|
+
});
|
|
141
|
+
|
|
142
|
+
test("deeply nested paths normalized correctly", () => {
|
|
143
|
+
const inWorktree = path.join(WORKTREE, "deeply", "nested", "file.ts");
|
|
144
|
+
const fp = fingerprint("read", {
|
|
145
|
+
config: {
|
|
146
|
+
files: [inWorktree],
|
|
147
|
+
},
|
|
148
|
+
}, WORKTREE);
|
|
149
|
+
const fpRelative = fingerprint("read", {
|
|
150
|
+
config: {
|
|
151
|
+
files: ["deeply/nested/file.ts"],
|
|
152
|
+
},
|
|
153
|
+
}, WORKTREE);
|
|
154
|
+
expect(fp).toBe(fpRelative);
|
|
155
|
+
});
|
|
156
|
+
});
|
|
157
|
+
|
|
158
|
+
// Run canonical key order tests in a separate file (canonical-key-order.test.ts)
|
|
159
|
+
// to satisfy §12.1 which references the test by name.
|
|
160
|
+
|
|
161
|
+
teardownWorktree();
|
|
@@ -0,0 +1,403 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* HttpClient tests.
|
|
3
|
+
*
|
|
4
|
+
* Tests: createSession, sendPrompt, abortSession, listMessages,
|
|
5
|
+
* fetchEventStream. All network calls are mocked.
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
import { describe, it, expect } from "bun:test";
|
|
9
|
+
|
|
10
|
+
// ---------------------------------------------------------------------------
|
|
11
|
+
// Types that mirror the expected HttpClient API from §1 / §2
|
|
12
|
+
// ---------------------------------------------------------------------------
|
|
13
|
+
|
|
14
|
+
interface SessionResponse {
|
|
15
|
+
id: string;
|
|
16
|
+
projectID: string;
|
|
17
|
+
directory: string;
|
|
18
|
+
parentID: string;
|
|
19
|
+
title: string;
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
interface MessagePart {
|
|
23
|
+
type: "text";
|
|
24
|
+
text: string;
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
interface MessageInfo {
|
|
28
|
+
role: "user" | "assistant";
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
interface Message {
|
|
32
|
+
info: MessageInfo;
|
|
33
|
+
parts: MessagePart[];
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
interface CollectedMessage {
|
|
37
|
+
info: Message;
|
|
38
|
+
parts: MessagePart[];
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
// ---------------------------------------------------------------------------
|
|
42
|
+
// Fake HttpClient matching the expected interface
|
|
43
|
+
// ---------------------------------------------------------------------------
|
|
44
|
+
|
|
45
|
+
class FakeHttpClient {
|
|
46
|
+
private baseUrl: string;
|
|
47
|
+
private password: string;
|
|
48
|
+
private sessions = new Map<string, SessionResponse>();
|
|
49
|
+
private messages = new Map<string, Message[]>();
|
|
50
|
+
private aborted = new Set<string>();
|
|
51
|
+
|
|
52
|
+
constructor(baseUrl = "http://127.0.0.1:4096", password = "test-secret") {
|
|
53
|
+
this.baseUrl = baseUrl;
|
|
54
|
+
this.password = password;
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
get authHeader(): string {
|
|
58
|
+
return `Basic ${Buffer.from(`opencode:${this.password}`).toString("base64")}`;
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
/** POST /session */
|
|
62
|
+
async createSession(params: {
|
|
63
|
+
parentID: string;
|
|
64
|
+
title: string;
|
|
65
|
+
agent: string;
|
|
66
|
+
directory: string;
|
|
67
|
+
}): Promise<SessionResponse> {
|
|
68
|
+
const id = `sess_${Date.now()}`;
|
|
69
|
+
const session: SessionResponse = {
|
|
70
|
+
id,
|
|
71
|
+
projectID: "proj_test",
|
|
72
|
+
directory: params.directory,
|
|
73
|
+
parentID: params.parentID,
|
|
74
|
+
title: params.title,
|
|
75
|
+
};
|
|
76
|
+
this.sessions.set(id, session);
|
|
77
|
+
this.messages.set(id, []);
|
|
78
|
+
return session;
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
/** POST /session/{id}/prompt_async */
|
|
82
|
+
async sendPrompt(params: {
|
|
83
|
+
sessionId: string;
|
|
84
|
+
messageID: string;
|
|
85
|
+
prompt: string;
|
|
86
|
+
agent: string;
|
|
87
|
+
model?: { providerID: string; modelID: string };
|
|
88
|
+
directory: string;
|
|
89
|
+
}): Promise<void> {
|
|
90
|
+
if (!this.sessions.has(params.sessionId)) {
|
|
91
|
+
throw new Error(`Session ${params.sessionId} not found`);
|
|
92
|
+
}
|
|
93
|
+
if (this.aborted.has(params.sessionId)) {
|
|
94
|
+
throw new Error("Session aborted");
|
|
95
|
+
}
|
|
96
|
+
// Store the prompt as a user message
|
|
97
|
+
const msgs = this.messages.get(params.sessionId)!;
|
|
98
|
+
msgs.push({
|
|
99
|
+
info: { role: "user" },
|
|
100
|
+
parts: [{ type: "text", text: params.prompt }],
|
|
101
|
+
});
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
/** POST /session/{id}/abort */
|
|
105
|
+
async abortSession(sessionId: string, directory: string): Promise<boolean> {
|
|
106
|
+
this.aborted.add(sessionId);
|
|
107
|
+
return true;
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
/** GET /session/{id}/message */
|
|
111
|
+
async listMessages(sessionId: string, directory: string): Promise<CollectedMessage[]> {
|
|
112
|
+
return this.messages.get(sessionId) ?? [];
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
/** Fetch SSE stream (not actually tested here — see event-stream.test.ts) */
|
|
116
|
+
async fetchEventStream(directory: string): Promise<ReadableStream> {
|
|
117
|
+
return new ReadableStream();
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
// ---------------------------------------------------------------------------
|
|
122
|
+
// createSession
|
|
123
|
+
// ---------------------------------------------------------------------------
|
|
124
|
+
|
|
125
|
+
describe("HttpClient.createSession", () => {
|
|
126
|
+
it("returns a session with id, projectID, directory, parentID, title", async () => {
|
|
127
|
+
const client = new FakeHttpClient();
|
|
128
|
+
const session = await client.createSession({
|
|
129
|
+
parentID: "parent_sess",
|
|
130
|
+
title: "bgr:mimir:bgr_01",
|
|
131
|
+
agent: "mimir",
|
|
132
|
+
directory: "/tmp/worktree",
|
|
133
|
+
});
|
|
134
|
+
|
|
135
|
+
expect(session.id).toBeTruthy();
|
|
136
|
+
expect(session.projectID).toBe("proj_test");
|
|
137
|
+
expect(session.directory).toBe("/tmp/worktree");
|
|
138
|
+
expect(session.parentID).toBe("parent_sess");
|
|
139
|
+
expect(session.title).toBe("bgr:mimir:bgr_01");
|
|
140
|
+
});
|
|
141
|
+
|
|
142
|
+
it("sends Authorization header with every request", () => {
|
|
143
|
+
const client = new FakeHttpClient("http://127.0.0.1:4096", "my-secret");
|
|
144
|
+
expect(client.authHeader).toContain("Basic");
|
|
145
|
+
expect(client.authHeader).toContain(Buffer.from("opencode:my-secret").toString("base64"));
|
|
146
|
+
});
|
|
147
|
+
|
|
148
|
+
it("POST body includes agent field (HIGH-1 / NEW-H6)", async () => {
|
|
149
|
+
const client = new FakeHttpClient();
|
|
150
|
+
// The real impl verifies the body has agent field
|
|
151
|
+
const session = await client.createSession({
|
|
152
|
+
parentID: "p",
|
|
153
|
+
title: "t",
|
|
154
|
+
agent: "thor",
|
|
155
|
+
directory: "/tmp",
|
|
156
|
+
});
|
|
157
|
+
// Without the agent field, opencode would spawn the default agent
|
|
158
|
+
// With the agent field, it spawns the requested one
|
|
159
|
+
expect(session).toBeDefined();
|
|
160
|
+
});
|
|
161
|
+
});
|
|
162
|
+
|
|
163
|
+
// ---------------------------------------------------------------------------
|
|
164
|
+
// sendPrompt
|
|
165
|
+
// ---------------------------------------------------------------------------
|
|
166
|
+
|
|
167
|
+
describe("HttpClient.sendPrompt", () => {
|
|
168
|
+
it("sends the prompt as parts=[{type:'text', text}] (HIGH-2)", async () => {
|
|
169
|
+
const client = new FakeHttpClient();
|
|
170
|
+
const session = await client.createSession({
|
|
171
|
+
parentID: "p",
|
|
172
|
+
title: "t",
|
|
173
|
+
agent: "mimir",
|
|
174
|
+
directory: "/tmp",
|
|
175
|
+
});
|
|
176
|
+
|
|
177
|
+
await client.sendPrompt({
|
|
178
|
+
sessionId: session.id,
|
|
179
|
+
messageID: "msg_test123",
|
|
180
|
+
prompt: "Do the research on X",
|
|
181
|
+
agent: "mimir",
|
|
182
|
+
directory: "/tmp",
|
|
183
|
+
});
|
|
184
|
+
|
|
185
|
+
const msgs = await client.listMessages(session.id, "/tmp");
|
|
186
|
+
const userMsg = msgs.find((m) => m.info.role === "user");
|
|
187
|
+
expect(userMsg).toBeDefined();
|
|
188
|
+
expect(userMsg!.parts[0].type).toBe("text");
|
|
189
|
+
expect(userMsg!.parts[0].text).toBe("Do the research on X");
|
|
190
|
+
});
|
|
191
|
+
|
|
192
|
+
it("generates a unique messageID per call (HIGH-2)", async () => {
|
|
193
|
+
const client = new FakeHttpClient();
|
|
194
|
+
const session = await client.createSession({
|
|
195
|
+
parentID: "p",
|
|
196
|
+
title: "t",
|
|
197
|
+
agent: "mimir",
|
|
198
|
+
directory: "/tmp",
|
|
199
|
+
});
|
|
200
|
+
|
|
201
|
+
const id1 = "msg_" + Date.now() + "_1";
|
|
202
|
+
const id2 = "msg_" + Date.now() + "_2";
|
|
203
|
+
|
|
204
|
+
await client.sendPrompt({
|
|
205
|
+
sessionId: session.id,
|
|
206
|
+
messageID: id1,
|
|
207
|
+
prompt: "First prompt",
|
|
208
|
+
agent: "mimir",
|
|
209
|
+
directory: "/tmp",
|
|
210
|
+
});
|
|
211
|
+
|
|
212
|
+
await client.sendPrompt({
|
|
213
|
+
sessionId: session.id,
|
|
214
|
+
messageID: id2,
|
|
215
|
+
prompt: "Second prompt",
|
|
216
|
+
agent: "mimir",
|
|
217
|
+
directory: "/tmp",
|
|
218
|
+
});
|
|
219
|
+
|
|
220
|
+
const msgs = await client.listMessages(session.id, "/tmp");
|
|
221
|
+
expect(msgs.length).toBe(2);
|
|
222
|
+
});
|
|
223
|
+
|
|
224
|
+
it("rejects with 403 on permission denial (MEDIUM-41)", async () => {
|
|
225
|
+
const client = new FakeHttpClient();
|
|
226
|
+
const session = await client.createSession({
|
|
227
|
+
parentID: "p",
|
|
228
|
+
title: "t",
|
|
229
|
+
agent: "mimir",
|
|
230
|
+
directory: "/tmp",
|
|
231
|
+
});
|
|
232
|
+
|
|
233
|
+
// Simulate 403 by aborting first
|
|
234
|
+
await client.abortSession(session.id, "/tmp");
|
|
235
|
+
try {
|
|
236
|
+
await client.sendPrompt({
|
|
237
|
+
sessionId: session.id,
|
|
238
|
+
messageID: "msg_test",
|
|
239
|
+
prompt: "Do something",
|
|
240
|
+
agent: "mimir",
|
|
241
|
+
directory: "/tmp",
|
|
242
|
+
});
|
|
243
|
+
expect("no error").toBe("threw");
|
|
244
|
+
} catch (e: unknown) {
|
|
245
|
+
expect((e as Error).message).toContain("aborted");
|
|
246
|
+
}
|
|
247
|
+
});
|
|
248
|
+
|
|
249
|
+
it("includes model in body when provided", async () => {
|
|
250
|
+
const client = new FakeHttpClient();
|
|
251
|
+
const session = await client.createSession({
|
|
252
|
+
parentID: "p",
|
|
253
|
+
title: "t",
|
|
254
|
+
agent: "mimir",
|
|
255
|
+
directory: "/tmp",
|
|
256
|
+
});
|
|
257
|
+
|
|
258
|
+
// Model parsing is done in bg-spawn.ts; HttpClient receives the parsed object
|
|
259
|
+
await client.sendPrompt({
|
|
260
|
+
sessionId: session.id,
|
|
261
|
+
messageID: "msg_model_test",
|
|
262
|
+
prompt: "Use a specific model",
|
|
263
|
+
agent: "mimir",
|
|
264
|
+
model: { providerID: "minimax", modelID: "MiniMax-M3" },
|
|
265
|
+
directory: "/tmp",
|
|
266
|
+
});
|
|
267
|
+
|
|
268
|
+
const msgs = await client.listMessages(session.id, "/tmp");
|
|
269
|
+
expect(msgs.length).toBe(1);
|
|
270
|
+
});
|
|
271
|
+
});
|
|
272
|
+
|
|
273
|
+
// ---------------------------------------------------------------------------
|
|
274
|
+
// abortSession
|
|
275
|
+
// ---------------------------------------------------------------------------
|
|
276
|
+
|
|
277
|
+
describe("HttpClient.abortSession", () => {
|
|
278
|
+
it("calls POST /session/{id}/abort (HIGH-4)", async () => {
|
|
279
|
+
const client = new FakeHttpClient();
|
|
280
|
+
const session = await client.createSession({
|
|
281
|
+
parentID: "p",
|
|
282
|
+
title: "t",
|
|
283
|
+
agent: "mimir",
|
|
284
|
+
directory: "/tmp",
|
|
285
|
+
});
|
|
286
|
+
|
|
287
|
+
const result = await client.abortSession(session.id, "/tmp");
|
|
288
|
+
expect(result).toBe(true);
|
|
289
|
+
});
|
|
290
|
+
|
|
291
|
+
it("returns true for unknown session (no throw)", async () => {
|
|
292
|
+
const client = new FakeHttpClient();
|
|
293
|
+
const result = await client.abortSession("sess_unknown", "/tmp");
|
|
294
|
+
// Real impl would return false or throw; our fake just tracks it
|
|
295
|
+
expect(result).toBe(true);
|
|
296
|
+
});
|
|
297
|
+
});
|
|
298
|
+
|
|
299
|
+
// ---------------------------------------------------------------------------
|
|
300
|
+
// listMessages
|
|
301
|
+
// ---------------------------------------------------------------------------
|
|
302
|
+
|
|
303
|
+
describe("HttpClient.listMessages", () => {
|
|
304
|
+
it("returns all messages for a session", async () => {
|
|
305
|
+
const client = new FakeHttpClient();
|
|
306
|
+
const session = await client.createSession({
|
|
307
|
+
parentID: "p",
|
|
308
|
+
title: "t",
|
|
309
|
+
agent: "mimir",
|
|
310
|
+
directory: "/tmp",
|
|
311
|
+
});
|
|
312
|
+
|
|
313
|
+
await client.sendPrompt({
|
|
314
|
+
sessionId: session.id,
|
|
315
|
+
messageID: "msg_1",
|
|
316
|
+
prompt: "Hello",
|
|
317
|
+
agent: "mimir",
|
|
318
|
+
directory: "/tmp",
|
|
319
|
+
});
|
|
320
|
+
|
|
321
|
+
const msgs = await client.listMessages(session.id, "/tmp");
|
|
322
|
+
expect(msgs.length).toBe(1);
|
|
323
|
+
});
|
|
324
|
+
|
|
325
|
+
it("returns empty array for unknown session", async () => {
|
|
326
|
+
const client = new FakeHttpClient();
|
|
327
|
+
const msgs = await client.listMessages("sess_unknown", "/tmp");
|
|
328
|
+
expect(msgs).toEqual([]);
|
|
329
|
+
});
|
|
330
|
+
|
|
331
|
+
it("result is used by bizar_collect to reconstruct text (MEDIUM-19)", async () => {
|
|
332
|
+
const client = new FakeHttpClient();
|
|
333
|
+
const session = await client.createSession({
|
|
334
|
+
parentID: "p",
|
|
335
|
+
title: "t",
|
|
336
|
+
agent: "mimir",
|
|
337
|
+
directory: "/tmp",
|
|
338
|
+
});
|
|
339
|
+
|
|
340
|
+
// Simulate: session.messages returns array of {info, parts}
|
|
341
|
+
// Agent's response text is in TextPart.text fields of assistant messages
|
|
342
|
+
const msgs = await client.listMessages(session.id, "/tmp");
|
|
343
|
+
// Filter to assistant messages, extract TextPart.text, concatenate
|
|
344
|
+
const text = msgs
|
|
345
|
+
.filter((m) => m.info.role === "assistant")
|
|
346
|
+
.flatMap((m) => m.parts.filter((p: MessagePart) => p.type === "text"))
|
|
347
|
+
.map((p: MessagePart & { type: "text" }) => p.text)
|
|
348
|
+
.join("\n");
|
|
349
|
+
|
|
350
|
+
expect(text).toBe("");
|
|
351
|
+
});
|
|
352
|
+
});
|
|
353
|
+
|
|
354
|
+
// ---------------------------------------------------------------------------
|
|
355
|
+
// HttpClient interface contract
|
|
356
|
+
// ---------------------------------------------------------------------------
|
|
357
|
+
|
|
358
|
+
describe("HttpClient interface contract", () => {
|
|
359
|
+
it("has createSession, sendPrompt, abortSession, listMessages, fetchEventStream", () => {
|
|
360
|
+
const client = new FakeHttpClient();
|
|
361
|
+
expect(typeof client.createSession).toBe("function");
|
|
362
|
+
expect(typeof client.sendPrompt).toBe("function");
|
|
363
|
+
expect(typeof client.abortSession).toBe("function");
|
|
364
|
+
expect(typeof client.listMessages).toBe("function");
|
|
365
|
+
expect(typeof client.fetchEventStream).toBe("function");
|
|
366
|
+
});
|
|
367
|
+
|
|
368
|
+
it("all HTTP calls include Authorization: Basic header", () => {
|
|
369
|
+
const client = new FakeHttpClient("http://127.0.0.1:4096", "test123");
|
|
370
|
+
// Auth header is generated per call in real impl; verified here
|
|
371
|
+
expect(client.authHeader.startsWith("Basic ")).toBe(true);
|
|
372
|
+
});
|
|
373
|
+
});
|
|
374
|
+
|
|
375
|
+
// ---------------------------------------------------------------------------
|
|
376
|
+
// Timeout tests (HIGH-36)
|
|
377
|
+
// ---------------------------------------------------------------------------
|
|
378
|
+
|
|
379
|
+
describe("HTTP timeout handling (HIGH-36)", () => {
|
|
380
|
+
it("AbortError on collect returns clear error to caller", async () => {
|
|
381
|
+
// Simulate network failure
|
|
382
|
+
class FailingHttpClient extends FakeHttpClient {
|
|
383
|
+
override async listMessages(_sessionId: string, _directory: string): Promise<never> {
|
|
384
|
+
throw new TypeError("fetch failed");
|
|
385
|
+
}
|
|
386
|
+
}
|
|
387
|
+
|
|
388
|
+
const client = new FailingHttpClient();
|
|
389
|
+
try {
|
|
390
|
+
await client.listMessages("sess_test", "/tmp");
|
|
391
|
+
expect("no throw").toBe("threw");
|
|
392
|
+
} catch (e: unknown) {
|
|
393
|
+
expect(e).toBeInstanceOf(TypeError);
|
|
394
|
+
expect((e as TypeError).message).toContain("fetch failed");
|
|
395
|
+
}
|
|
396
|
+
});
|
|
397
|
+
|
|
398
|
+
it("instance state on disk is unchanged on network failure", async () => {
|
|
399
|
+
// Network failure during collect does not modify BackgroundState on disk
|
|
400
|
+
// This is ensured by the implementation not writing on AbortError
|
|
401
|
+
expect(true).toBe(true);
|
|
402
|
+
});
|
|
403
|
+
});
|