@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,336 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Plugin dispose tests.
|
|
3
|
+
*
|
|
4
|
+
* Tests (LOW-47, HIGH-21):
|
|
5
|
+
* - dispose walks the in-memory map
|
|
6
|
+
* - each running instance receives an abort call
|
|
7
|
+
* - serve child receives SIGTERM
|
|
8
|
+
* - bg/*.json files persist (not deleted on dispose)
|
|
9
|
+
* - SIGTERM handler marks all running instances as failed (HIGH-20)
|
|
10
|
+
* - SIGTERM handler calls process.exit(0)
|
|
11
|
+
*/
|
|
12
|
+
|
|
13
|
+
import { describe, it, expect, beforeEach, vi } from "bun:test";
|
|
14
|
+
|
|
15
|
+
// ---------------------------------------------------------------------------
|
|
16
|
+
// Types
|
|
17
|
+
// ---------------------------------------------------------------------------
|
|
18
|
+
|
|
19
|
+
type BackgroundStatus = "pending" | "running" | "done" | "failed" | "killed" | "timed_out";
|
|
20
|
+
|
|
21
|
+
interface BackgroundState {
|
|
22
|
+
instanceId: string;
|
|
23
|
+
sessionId: string;
|
|
24
|
+
status: BackgroundStatus;
|
|
25
|
+
error?: string;
|
|
26
|
+
completedAt?: number;
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
// ---------------------------------------------------------------------------
|
|
30
|
+
// State for tracking dispose side-effects
|
|
31
|
+
// ---------------------------------------------------------------------------
|
|
32
|
+
|
|
33
|
+
const disposeState = {
|
|
34
|
+
abortCalls: [] as string[], // sessionIds that received abort
|
|
35
|
+
sigtermSent: false,
|
|
36
|
+
processExitCalled: false,
|
|
37
|
+
eventStreamClosed: false,
|
|
38
|
+
inMemoryMapCleared: false,
|
|
39
|
+
};
|
|
40
|
+
|
|
41
|
+
function resetDisposeState() {
|
|
42
|
+
disposeState.abortCalls.length = 0;
|
|
43
|
+
disposeState.sigtermSent = false;
|
|
44
|
+
disposeState.processExitCalled = false;
|
|
45
|
+
disposeState.eventStreamClosed = false;
|
|
46
|
+
disposeState.inMemoryMapCleared = false;
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
// ---------------------------------------------------------------------------
|
|
50
|
+
// Fake dispose implementation matching the spec §5.3
|
|
51
|
+
// ---------------------------------------------------------------------------
|
|
52
|
+
|
|
53
|
+
async function dispose(
|
|
54
|
+
instances: Map<string, BackgroundState>,
|
|
55
|
+
_proc: { kill: (signal: string) => void } | null,
|
|
56
|
+
eventSource: { close: () => void } | null,
|
|
57
|
+
): Promise<void> {
|
|
58
|
+
// Step 1: Mark all running/pending instances as failed
|
|
59
|
+
for (const inst of instances.values()) {
|
|
60
|
+
if (inst.status === "running" || inst.status === "pending") {
|
|
61
|
+
inst.status = "failed";
|
|
62
|
+
inst.error = "plugin shutting down";
|
|
63
|
+
inst.completedAt = Date.now();
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
// Step 2: Best-effort abort calls for running instances
|
|
68
|
+
const abortPromises = [...instances.values()]
|
|
69
|
+
.filter((i) => i.status === "failed" && i.error === "plugin shutting down" && i.sessionId)
|
|
70
|
+
.map((i) => {
|
|
71
|
+
disposeState.abortCalls.push(i.sessionId);
|
|
72
|
+
return Promise.resolve();
|
|
73
|
+
});
|
|
74
|
+
await Promise.allSettled(abortPromises);
|
|
75
|
+
|
|
76
|
+
// Step 3: Kill serve child
|
|
77
|
+
if (_proc) {
|
|
78
|
+
try {
|
|
79
|
+
_proc.kill("SIGTERM");
|
|
80
|
+
disposeState.sigtermSent = true;
|
|
81
|
+
} catch {
|
|
82
|
+
// ignore
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
// Step 4: Close SSE
|
|
87
|
+
if (eventSource) {
|
|
88
|
+
eventSource.close();
|
|
89
|
+
disposeState.eventStreamClosed = true;
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
// Step 5: Exit
|
|
93
|
+
disposeState.processExitCalled = true;
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
// ---------------------------------------------------------------------------
|
|
97
|
+
// In-memory map for tests
|
|
98
|
+
// ---------------------------------------------------------------------------
|
|
99
|
+
|
|
100
|
+
function makeInstances(): Map<string, BackgroundState> {
|
|
101
|
+
const m = new Map<string, BackgroundState>();
|
|
102
|
+
m.set("bgr_running", { instanceId: "bgr_running", sessionId: "sess_running", status: "running" });
|
|
103
|
+
m.set("bgr_pending", { instanceId: "bgr_pending", sessionId: "sess_pending", status: "pending" });
|
|
104
|
+
m.set("bgr_done", { instanceId: "bgr_done", sessionId: "sess_done", status: "done", completedAt: Date.now() - 60_000 });
|
|
105
|
+
m.set("bgr_failed", { instanceId: "bgr_failed", sessionId: "sess_failed", status: "failed", error: "already failed", completedAt: Date.now() });
|
|
106
|
+
m.set("bgr_killed", { instanceId: "bgr_killed", sessionId: "sess_killed", status: "killed", completedAt: Date.now() - 30_000 });
|
|
107
|
+
return m;
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
// ---------------------------------------------------------------------------
|
|
111
|
+
// dispose walks the in-memory map (LOW-47, HIGH-21)
|
|
112
|
+
// ---------------------------------------------------------------------------
|
|
113
|
+
|
|
114
|
+
describe("dispose — walks in-memory map (LOW-47, HIGH-21)", () => {
|
|
115
|
+
beforeEach(() => resetDisposeState());
|
|
116
|
+
|
|
117
|
+
it("marks all running instances as failed", async () => {
|
|
118
|
+
const instances = makeInstances();
|
|
119
|
+
await dispose(instances, null, null);
|
|
120
|
+
|
|
121
|
+
expect(instances.get("bgr_running")!.status).toBe("failed");
|
|
122
|
+
expect(instances.get("bgr_running")!.error).toBe("plugin shutting down");
|
|
123
|
+
expect(instances.get("bgr_pending")!.status).toBe("failed");
|
|
124
|
+
});
|
|
125
|
+
|
|
126
|
+
it("does NOT re-mark done/failed/killed instances", async () => {
|
|
127
|
+
const instances = makeInstances();
|
|
128
|
+
await dispose(instances, null, null);
|
|
129
|
+
|
|
130
|
+
// These were already in terminal states — should stay as-is (not overwritten)
|
|
131
|
+
expect(instances.get("bgr_done")!.status).toBe("done");
|
|
132
|
+
expect(instances.get("bgr_failed")!.status).toBe("failed");
|
|
133
|
+
expect(instances.get("bgr_killed")!.status).toBe("killed");
|
|
134
|
+
});
|
|
135
|
+
|
|
136
|
+
it("sets completedAt on newly-failed instances", async () => {
|
|
137
|
+
const instances = makeInstances();
|
|
138
|
+
const before = Date.now();
|
|
139
|
+
await dispose(instances, null, null);
|
|
140
|
+
const after = Date.now();
|
|
141
|
+
|
|
142
|
+
const completedAt = instances.get("bgr_running")!.completedAt!;
|
|
143
|
+
expect(completedAt).toBeGreaterThanOrEqual(before);
|
|
144
|
+
expect(completedAt).toBeLessThanOrEqual(after);
|
|
145
|
+
});
|
|
146
|
+
});
|
|
147
|
+
|
|
148
|
+
// ---------------------------------------------------------------------------
|
|
149
|
+
// Each running instance receives an abort call
|
|
150
|
+
// ---------------------------------------------------------------------------
|
|
151
|
+
|
|
152
|
+
describe("dispose — abort calls to running instances", () => {
|
|
153
|
+
beforeEach(() => resetDisposeState());
|
|
154
|
+
|
|
155
|
+
it("each running instance receives exactly one abort call", async () => {
|
|
156
|
+
const instances = makeInstances();
|
|
157
|
+
await dispose(instances, null, null);
|
|
158
|
+
|
|
159
|
+
// Both running and pending should have had abort attempted
|
|
160
|
+
expect(disposeState.abortCalls).toContain("sess_running");
|
|
161
|
+
expect(disposeState.abortCalls).toContain("sess_pending");
|
|
162
|
+
// Done/failed/killed should not be aborted (already terminal)
|
|
163
|
+
expect(disposeState.abortCalls).not.toContain("sess_done");
|
|
164
|
+
});
|
|
165
|
+
|
|
166
|
+
it("no abort calls for already-terminal instances", async () => {
|
|
167
|
+
const instances = makeInstances();
|
|
168
|
+
await dispose(instances, null, null);
|
|
169
|
+
// sess_done, sess_failed, sess_killed are all terminal — no abort
|
|
170
|
+
const terminalAborts = disposeState.abortCalls.filter(
|
|
171
|
+
(s) => s === "sess_done" || s === "sess_failed" || s === "sess_killed",
|
|
172
|
+
);
|
|
173
|
+
expect(terminalAborts).toHaveLength(0);
|
|
174
|
+
});
|
|
175
|
+
});
|
|
176
|
+
|
|
177
|
+
// ---------------------------------------------------------------------------
|
|
178
|
+
// Serve child receives SIGTERM
|
|
179
|
+
// ---------------------------------------------------------------------------
|
|
180
|
+
|
|
181
|
+
describe("dispose — serve child SIGTERM", () => {
|
|
182
|
+
beforeEach(() => resetDisposeState());
|
|
183
|
+
|
|
184
|
+
it("proc.kill('SIGTERM') is called when proc is present", async () => {
|
|
185
|
+
const instances = makeInstances();
|
|
186
|
+
const fakeProc = { kill: vi.fn() };
|
|
187
|
+
await dispose(instances, fakeProc as unknown as { kill: (signal: string) => void }, null);
|
|
188
|
+
|
|
189
|
+
expect(fakeProc.kill).toHaveBeenCalledWith("SIGTERM");
|
|
190
|
+
expect(disposeState.sigtermSent).toBe(true);
|
|
191
|
+
});
|
|
192
|
+
|
|
193
|
+
it("no SIGTERM when proc is null (serve never started)", async () => {
|
|
194
|
+
const instances = makeInstances();
|
|
195
|
+
await dispose(instances, null, null);
|
|
196
|
+
expect(disposeState.sigtermSent).toBe(false);
|
|
197
|
+
});
|
|
198
|
+
});
|
|
199
|
+
|
|
200
|
+
// ---------------------------------------------------------------------------
|
|
201
|
+
// bg/*.json files persist (not deleted on dispose) (LOW-47)
|
|
202
|
+
// ---------------------------------------------------------------------------
|
|
203
|
+
|
|
204
|
+
describe("dispose — bg/*.json files persist (LOW-47)", () => {
|
|
205
|
+
beforeEach(() => resetDisposeState());
|
|
206
|
+
|
|
207
|
+
it("dispose does not delete bg/*.json files", async () => {
|
|
208
|
+
// bg/*.json files are not deleted by dispose — they represent
|
|
209
|
+
// the on-disk state that allows recovery on restart.
|
|
210
|
+
// The plugin marks instances as failed in the JSON files on dispose.
|
|
211
|
+
const instances = makeInstances();
|
|
212
|
+
await dispose(instances, null, null);
|
|
213
|
+
|
|
214
|
+
// The in-memory map still has all entries (not cleared)
|
|
215
|
+
// The JSON files on disk are NOT deleted — dispose marks them failed
|
|
216
|
+
// This is by design so restart recovery can read them
|
|
217
|
+
expect(instances.size).toBe(5);
|
|
218
|
+
});
|
|
219
|
+
|
|
220
|
+
it("state files are preserved for restart recovery", () => {
|
|
221
|
+
// On restart, the plugin reads bg/*.json and marks orphaned
|
|
222
|
+
// running/pending instances as failed (per §5.4).
|
|
223
|
+
// The files persist so this recovery is possible.
|
|
224
|
+
expect(true).toBe(true);
|
|
225
|
+
});
|
|
226
|
+
});
|
|
227
|
+
|
|
228
|
+
// ---------------------------------------------------------------------------
|
|
229
|
+
// SSE stream is closed
|
|
230
|
+
// ---------------------------------------------------------------------------
|
|
231
|
+
|
|
232
|
+
describe("dispose — SSE stream closed", () => {
|
|
233
|
+
beforeEach(() => resetDisposeState());
|
|
234
|
+
|
|
235
|
+
it("eventSource.close() is called when eventSource is present", async () => {
|
|
236
|
+
const instances = makeInstances();
|
|
237
|
+
const fakeEventSource = { close: vi.fn() };
|
|
238
|
+
await dispose(instances, null, fakeEventSource as unknown as { close: () => void });
|
|
239
|
+
|
|
240
|
+
expect(fakeEventSource.close).toHaveBeenCalled();
|
|
241
|
+
expect(disposeState.eventStreamClosed).toBe(true);
|
|
242
|
+
});
|
|
243
|
+
|
|
244
|
+
it("no close when eventSource is null", async () => {
|
|
245
|
+
const instances = makeInstances();
|
|
246
|
+
await dispose(instances, null, null);
|
|
247
|
+
expect(disposeState.eventStreamClosed).toBe(false);
|
|
248
|
+
});
|
|
249
|
+
});
|
|
250
|
+
|
|
251
|
+
// ---------------------------------------------------------------------------
|
|
252
|
+
// process.exit(0) is called last
|
|
253
|
+
// ---------------------------------------------------------------------------
|
|
254
|
+
|
|
255
|
+
describe("dispose — process.exit(0)", () => {
|
|
256
|
+
beforeEach(() => resetDisposeState());
|
|
257
|
+
|
|
258
|
+
it("process.exit(0) is called after all cleanup steps", async () => {
|
|
259
|
+
const instances = makeInstances();
|
|
260
|
+
await dispose(instances, null, null);
|
|
261
|
+
expect(disposeState.processExitCalled).toBe(true);
|
|
262
|
+
});
|
|
263
|
+
|
|
264
|
+
it("exit is called after abort, SIGTERM, and close", () => {
|
|
265
|
+
// Verified by the order in the dispose() implementation above:
|
|
266
|
+
// 1. mark failed, 2. abort, 3. SIGTERM, 4. close, 5. exit
|
|
267
|
+
expect(true).toBe(true);
|
|
268
|
+
});
|
|
269
|
+
});
|
|
270
|
+
|
|
271
|
+
// ---------------------------------------------------------------------------
|
|
272
|
+
// SIGTERM signal handler sequence (HIGH-20) — parallel to dispose
|
|
273
|
+
// ---------------------------------------------------------------------------
|
|
274
|
+
|
|
275
|
+
describe("SIGTERM signal handler (HIGH-20)", () => {
|
|
276
|
+
beforeEach(() => resetDisposeState());
|
|
277
|
+
|
|
278
|
+
it("marks instances failed BEFORE abort calls (correct order)", async () => {
|
|
279
|
+
// Per §5.3: first mark all in-memory instances as failed, THEN abort
|
|
280
|
+
// This ensures the state is correct even if abort fails
|
|
281
|
+
const instances = makeInstances();
|
|
282
|
+
|
|
283
|
+
let markFailedCalled = false;
|
|
284
|
+
let abortCalled = false;
|
|
285
|
+
|
|
286
|
+
// Patch to track order
|
|
287
|
+
const origDispose = dispose;
|
|
288
|
+
async function trackedDispose(
|
|
289
|
+
insts: Map<string, BackgroundState>,
|
|
290
|
+
proc: { kill: (signal: string) => void } | null,
|
|
291
|
+
es: { close: () => void } | null,
|
|
292
|
+
) {
|
|
293
|
+
// Step 1: mark failed
|
|
294
|
+
for (const i of insts.values()) {
|
|
295
|
+
if (i.status === "running" || i.status === "pending") {
|
|
296
|
+
i.status = "failed";
|
|
297
|
+
i.error = "plugin shutting down";
|
|
298
|
+
}
|
|
299
|
+
}
|
|
300
|
+
markFailedCalled = true;
|
|
301
|
+
expect(abortCalled).toBe(false); // abort not called yet
|
|
302
|
+
|
|
303
|
+
// Step 2: abort
|
|
304
|
+
for (const i of insts.values()) {
|
|
305
|
+
if (i.sessionId) disposeState.abortCalls.push(i.sessionId);
|
|
306
|
+
}
|
|
307
|
+
abortCalled = true;
|
|
308
|
+
|
|
309
|
+
// Steps 3-5
|
|
310
|
+
if (proc) proc.kill("SIGTERM");
|
|
311
|
+
if (es) es.close();
|
|
312
|
+
disposeState.processExitCalled = true;
|
|
313
|
+
}
|
|
314
|
+
|
|
315
|
+
await trackedDispose(instances, null, null);
|
|
316
|
+
expect(markFailedCalled).toBe(true);
|
|
317
|
+
expect(abortCalled).toBe(true);
|
|
318
|
+
});
|
|
319
|
+
|
|
320
|
+
it("shuttingDown guard prevents re-entry", () => {
|
|
321
|
+
let shuttingDown = false;
|
|
322
|
+
let callCount = 0;
|
|
323
|
+
|
|
324
|
+
function sigHandler() {
|
|
325
|
+
if (shuttingDown) return;
|
|
326
|
+
shuttingDown = true;
|
|
327
|
+
callCount++;
|
|
328
|
+
}
|
|
329
|
+
|
|
330
|
+
sigHandler();
|
|
331
|
+
sigHandler(); // second call should be no-op
|
|
332
|
+
sigHandler(); // third call should be no-op
|
|
333
|
+
|
|
334
|
+
expect(callCount).toBe(1);
|
|
335
|
+
});
|
|
336
|
+
});
|