@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.
Files changed (60) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +448 -0
  3. package/bun.lock +88 -0
  4. package/index.ts +1113 -0
  5. package/package.json +42 -0
  6. package/scripts/check-forbidden-imports.sh +33 -0
  7. package/src/background-state.ts +463 -0
  8. package/src/background.ts +964 -0
  9. package/src/commands-impl.ts +369 -0
  10. package/src/commands.ts +880 -0
  11. package/src/event-stream.ts +574 -0
  12. package/src/fingerprint.ts +120 -0
  13. package/src/handoff.ts +79 -0
  14. package/src/http-client.ts +467 -0
  15. package/src/logger.ts +144 -0
  16. package/src/loop.ts +176 -0
  17. package/src/options.ts +421 -0
  18. package/src/plan-fs.ts +323 -0
  19. package/src/report.ts +178 -0
  20. package/src/research-prompt.ts +35 -0
  21. package/src/serve.ts +476 -0
  22. package/src/settings.ts +349 -0
  23. package/src/state.ts +298 -0
  24. package/src/tools/bg-collect.ts +104 -0
  25. package/src/tools/bg-get-comments.ts +239 -0
  26. package/src/tools/bg-kill.ts +87 -0
  27. package/src/tools/bg-spawn.ts +263 -0
  28. package/src/tools/bg-status.ts +99 -0
  29. package/src/tools/plan-action.ts +767 -0
  30. package/src/tools/wait-for-feedback.ts +402 -0
  31. package/tests/attach-handler-bug.test.ts +166 -0
  32. package/tests/background-state.test.ts +277 -0
  33. package/tests/background.test.ts +402 -0
  34. package/tests/block.test.ts +193 -0
  35. package/tests/canonical-key-order.test.ts +71 -0
  36. package/tests/commands-impl.test.ts +442 -0
  37. package/tests/commands.test.ts +548 -0
  38. package/tests/config.test.ts +122 -0
  39. package/tests/dispose.test.ts +336 -0
  40. package/tests/event-stream.test.ts +409 -0
  41. package/tests/event.test.ts +262 -0
  42. package/tests/fingerprint.test.ts +161 -0
  43. package/tests/http-client.test.ts +403 -0
  44. package/tests/init-helpers.test.ts +203 -0
  45. package/tests/integration/slash-command.test.ts +348 -0
  46. package/tests/integration/tool-routing.test.ts +314 -0
  47. package/tests/loop.test.ts +397 -0
  48. package/tests/options.test.ts +274 -0
  49. package/tests/serve.test.ts +335 -0
  50. package/tests/settings.test.ts +351 -0
  51. package/tests/stall-think.test.ts +749 -0
  52. package/tests/state.test.ts +275 -0
  53. package/tests/tools/bg-collect.test.ts +337 -0
  54. package/tests/tools/bg-get-comments.test.ts +485 -0
  55. package/tests/tools/bg-kill.test.ts +231 -0
  56. package/tests/tools/bg-spawn.test.ts +311 -0
  57. package/tests/tools/bg-status.test.ts +216 -0
  58. package/tests/tools/plan-action.test.ts +599 -0
  59. package/tests/tools/wait-for-feedback.test.ts +390 -0
  60. package/tsconfig.json +29 -0
@@ -0,0 +1,485 @@
1
+ /**
2
+ * bg-get-comments.test.ts
3
+ *
4
+ * Tests for `readPlanComments` — the pure read function extracted from
5
+ * `bg-get-comments.ts` so the file-system code path can be exercised
6
+ * without spinning up a tool framework or an opencode host.
7
+ *
8
+ * The tool wrapper (`createBgGetCommentsTool`) is a thin Zod schema
9
+ * + JSON.stringify around this function; testing the wrapper itself
10
+ * would just duplicate the same assertions on top of a JSON parser.
11
+ * We test the function directly.
12
+ *
13
+ * Groups (13 tests):
14
+ * 1. missing plan.json — returns ok:true with []
15
+ * 2. invalid slug — returns ok:false with the regex error
16
+ * 3. corrupt plan.json — returns ok:false with "Failed to read"
17
+ * 4. plan.json is null — returns ok:false with "not a v2 canvas object"
18
+ * 5. plan.json is array — returns ok:false with "not a v2 canvas object"
19
+ * 6. valid plan, no filter — returns all comments, sorted
20
+ * 7. elementId filter — only matching comments
21
+ * 8. elementId "nil"/"null" — canvas-pinned only
22
+ * 9. elementId "" — all comments (same as omitting)
23
+ * 10. chronological sort — oldest first
24
+ * 11. missing created — pushed to end
25
+ * 12. malformed comments — filtered out
26
+ * 13. thread replies — preserved
27
+ */
28
+
29
+ import { describe, it, expect, beforeAll, afterAll } from "bun:test";
30
+ import { mkdtempSync, writeFileSync, mkdirSync, rmSync } from "node:fs";
31
+ import { join } from "node:path";
32
+ import { tmpdir } from "node:os";
33
+
34
+ import { readPlanComments } from "../../src/tools/bg-get-comments.ts";
35
+
36
+ // ---------------------------------------------------------------------------
37
+ // Fixtures / helpers
38
+ // ---------------------------------------------------------------------------
39
+
40
+ /** Per-test temp worktree. Created once for the whole suite. */
41
+ let worktree: string;
42
+
43
+ beforeAll(() => {
44
+ worktree = mkdtempSync(join(tmpdir(), "bizar-comments-test-"));
45
+ });
46
+
47
+ afterAll(() => {
48
+ if (worktree) {
49
+ rmSync(worktree, { recursive: true, force: true });
50
+ }
51
+ });
52
+
53
+ /**
54
+ * Write a `plan.json` under `plans/<slug>/plan.json` inside the worktree.
55
+ * Caller controls the JSON shape (valid, null, array, malformed, etc.).
56
+ */
57
+ function writePlanJson(slug: string, data: unknown): void {
58
+ const dir = join(worktree, "plans", slug);
59
+ mkdirSync(dir, { recursive: true });
60
+ // We pass `data` through `String(...)` for the "corrupt" test, so allow
61
+ // non-stringifiable cases via JSON.stringify with the catch fallback.
62
+ const text = typeof data === "string" ? data : JSON.stringify(data);
63
+ writeFileSync(join(dir, "plan.json"), text, "utf-8");
64
+ }
65
+
66
+ /**
67
+ * Standard fixture used by most tests. Contains:
68
+ * - 2 comments on el_a (with timestamps)
69
+ * - 1 comment on el_b (with timestamp)
70
+ * - 1 canvas-pinned comment (null elementId)
71
+ * - 1 comment without a `created` timestamp
72
+ */
73
+ const SAMPLE_PLAN = {
74
+ schemaVersion: 2,
75
+ title: "Test Plan",
76
+ elements: [],
77
+ comments: [
78
+ { id: "c_1", elementId: "el_a", text: "First", created: "2026-06-18T10:00:00Z" },
79
+ { id: "c_2", elementId: "el_b", text: "Second", created: "2026-06-18T11:00:00Z" },
80
+ { id: "c_3", elementId: null, text: "Canvas pinned", created: "2026-06-18T12:00:00Z" },
81
+ { id: "c_4", elementId: "el_a", text: "No timestamp" }, // no created
82
+ ],
83
+ };
84
+
85
+ // ===========================================================================
86
+ // Group 1 — missing plan.json
87
+ // ===========================================================================
88
+
89
+ describe("readPlanComments — missing plan.json", () => {
90
+ it("returns ok:true with empty array when plan.json does not exist", () => {
91
+ const result = readPlanComments(worktree, "nonexistent-plan", undefined);
92
+ expect(result.ok).toBe(true);
93
+ if (result.ok) {
94
+ expect(result.comments).toEqual([]);
95
+ expect(result.planSlug).toBe("nonexistent-plan");
96
+ }
97
+ });
98
+
99
+ it("does not treat a missing plan as an error even with an elementId", () => {
100
+ const result = readPlanComments(worktree, "another-missing", "el_x");
101
+ expect(result.ok).toBe(true);
102
+ if (result.ok) {
103
+ expect(result.comments).toEqual([]);
104
+ }
105
+ });
106
+ });
107
+
108
+ // ===========================================================================
109
+ // Group 2 — invalid slug
110
+ // ===========================================================================
111
+
112
+ describe("readPlanComments — invalid slug", () => {
113
+ it("rejects slugs with uppercase characters", () => {
114
+ const result = readPlanComments(worktree, "UPPERCASE", undefined);
115
+ expect(result.ok).toBe(false);
116
+ if (!result.ok) {
117
+ expect(result.error).toContain("Invalid planSlug");
118
+ expect(result.error).toContain("UPPERCASE");
119
+ expect(result.error).toMatch(/a-z0-9/);
120
+ }
121
+ });
122
+
123
+ it("rejects path-traversal slugs", () => {
124
+ const result = readPlanComments(worktree, "../etc", undefined);
125
+ expect(result.ok).toBe(false);
126
+ if (!result.ok) {
127
+ expect(result.error).toContain("../etc");
128
+ expect(result.error).toContain("Invalid planSlug");
129
+ }
130
+ });
131
+
132
+ it("rejects empty slugs", () => {
133
+ const result = readPlanComments(worktree, "", undefined);
134
+ expect(result.ok).toBe(false);
135
+ if (!result.ok) {
136
+ expect(result.error).toContain("Invalid planSlug");
137
+ }
138
+ });
139
+
140
+ it("rejects slugs that start with a hyphen", () => {
141
+ const result = readPlanComments(worktree, "-starts-with-hyphen", undefined);
142
+ expect(result.ok).toBe(false);
143
+ if (!result.ok) {
144
+ expect(result.error).toContain("Invalid planSlug");
145
+ }
146
+ });
147
+
148
+ it("rejects slugs longer than 64 chars", () => {
149
+ const long = "a".repeat(65);
150
+ const result = readPlanComments(worktree, long, undefined);
151
+ expect(result.ok).toBe(false);
152
+ if (!result.ok) {
153
+ expect(result.error).toContain("Invalid planSlug");
154
+ }
155
+ });
156
+ });
157
+
158
+ // ===========================================================================
159
+ // Group 3 — corrupt plan.json
160
+ // ===========================================================================
161
+
162
+ describe("readPlanComments — corrupt plan.json", () => {
163
+ it("returns ok:false with 'Failed to read plan.json' when JSON is invalid", () => {
164
+ const slug = "corrupt-plan";
165
+ const dir = join(worktree, "plans", slug);
166
+ mkdirSync(dir, { recursive: true });
167
+ writeFileSync(join(dir, "plan.json"), "{ not valid json :: !!", "utf-8");
168
+
169
+ const result = readPlanComments(worktree, slug, undefined);
170
+ expect(result.ok).toBe(false);
171
+ if (!result.ok) {
172
+ expect(result.error.startsWith("Failed to read plan.json:")).toBe(true);
173
+ // The planSlug is always echoed back, even on error
174
+ expect(result.planSlug).toBe(slug);
175
+ }
176
+ });
177
+ });
178
+
179
+ // ===========================================================================
180
+ // Group 4 — plan.json is null
181
+ // ===========================================================================
182
+
183
+ describe("readPlanComments — plan.json is null", () => {
184
+ it("returns ok:false when the JSON literal is `null`", () => {
185
+ const slug = "null-plan";
186
+ writePlanJson(slug, null);
187
+ const result = readPlanComments(worktree, slug, undefined);
188
+ expect(result.ok).toBe(false);
189
+ if (!result.ok) {
190
+ expect(result.error).toContain("plan.json is not a v2 canvas object");
191
+ expect(result.error).toContain("object");
192
+ }
193
+ });
194
+ });
195
+
196
+ // ===========================================================================
197
+ // Group 5 — plan.json is array
198
+ // ===========================================================================
199
+
200
+ describe("readPlanComments — plan.json is array", () => {
201
+ it("returns ok:false when the JSON is a top-level array", () => {
202
+ const slug = "array-plan";
203
+ writePlanJson(slug, []);
204
+ const result = readPlanComments(worktree, slug, undefined);
205
+ expect(result.ok).toBe(false);
206
+ if (!result.ok) {
207
+ expect(result.error).toContain("plan.json is not a v2 canvas object");
208
+ expect(result.error).toContain("array");
209
+ }
210
+ });
211
+
212
+ it("returns ok:false for a non-empty array at the top level", () => {
213
+ const slug = "comments-as-array";
214
+ writePlanJson(slug, [{ id: "c_1" }]);
215
+ const result = readPlanComments(worktree, slug, undefined);
216
+ expect(result.ok).toBe(false);
217
+ if (!result.ok) {
218
+ expect(result.error).toContain("array");
219
+ }
220
+ });
221
+ });
222
+
223
+ // ===========================================================================
224
+ // Group 6 — valid plan, no elementId filter
225
+ // ===========================================================================
226
+
227
+ describe("readPlanComments — valid plan, no elementId", () => {
228
+ it("returns all comments with valid id strings, sorted by created time", () => {
229
+ const slug = "all-plan";
230
+ writePlanJson(slug, SAMPLE_PLAN);
231
+
232
+ const result = readPlanComments(worktree, slug, undefined);
233
+ expect(result.ok).toBe(true);
234
+ if (result.ok) {
235
+ expect(result.comments).toHaveLength(4);
236
+ for (const c of result.comments) {
237
+ expect(typeof c.id).toBe("string");
238
+ expect(c.id.length).toBeGreaterThan(0);
239
+ }
240
+ }
241
+ });
242
+
243
+ it("includes every comment id in the result", () => {
244
+ const slug = "all-plan-2";
245
+ writePlanJson(slug, SAMPLE_PLAN);
246
+ const result = readPlanComments(worktree, slug, undefined);
247
+ expect(result.ok).toBe(true);
248
+ if (result.ok) {
249
+ const ids = result.comments.map((c) => c.id);
250
+ expect(ids).toContain("c_1");
251
+ expect(ids).toContain("c_2");
252
+ expect(ids).toContain("c_3");
253
+ expect(ids).toContain("c_4");
254
+ }
255
+ });
256
+ });
257
+
258
+ // ===========================================================================
259
+ // Group 7 — elementId filter
260
+ // ===========================================================================
261
+
262
+ describe("readPlanComments — elementId filter", () => {
263
+ it("returns only comments pinned to the requested element", () => {
264
+ const slug = "filter-plan";
265
+ writePlanJson(slug, SAMPLE_PLAN);
266
+
267
+ const result = readPlanComments(worktree, slug, "el_a");
268
+ expect(result.ok).toBe(true);
269
+ if (result.ok) {
270
+ expect(result.comments).toHaveLength(2);
271
+ for (const c of result.comments) {
272
+ expect(c.elementId).toBe("el_a");
273
+ }
274
+ const ids = result.comments.map((c) => c.id);
275
+ expect(ids).toContain("c_1");
276
+ expect(ids).toContain("c_4");
277
+ }
278
+ });
279
+
280
+ it("returns an empty array when no comments match the requested element", () => {
281
+ const slug = "filter-plan";
282
+ writePlanJson(slug, SAMPLE_PLAN);
283
+ const result = readPlanComments(worktree, slug, "el_nonexistent");
284
+ expect(result.ok).toBe(true);
285
+ if (result.ok) {
286
+ expect(result.comments).toEqual([]);
287
+ }
288
+ });
289
+ });
290
+
291
+ // ===========================================================================
292
+ // Group 8 — elementId "nil" / "null"
293
+ // ===========================================================================
294
+
295
+ describe("readPlanComments — elementId 'nil' / 'null' (canvas-pinned only)", () => {
296
+ it("returns only canvas-pinned comments when elementId === 'nil'", () => {
297
+ const slug = "nil-plan";
298
+ writePlanJson(slug, SAMPLE_PLAN);
299
+
300
+ const result = readPlanComments(worktree, slug, "nil");
301
+ expect(result.ok).toBe(true);
302
+ if (result.ok) {
303
+ expect(result.comments).toHaveLength(1);
304
+ expect(result.comments[0]?.id).toBe("c_3");
305
+ expect(result.comments[0]?.elementId).toBeNull();
306
+ }
307
+ });
308
+
309
+ it("returns only canvas-pinned comments when elementId === 'null'", () => {
310
+ const slug = "null-string-plan";
311
+ writePlanJson(slug, SAMPLE_PLAN);
312
+
313
+ const result = readPlanComments(worktree, slug, "null");
314
+ expect(result.ok).toBe(true);
315
+ if (result.ok) {
316
+ expect(result.comments).toHaveLength(1);
317
+ expect(result.comments[0]?.id).toBe("c_3");
318
+ }
319
+ });
320
+ });
321
+
322
+ // ===========================================================================
323
+ // Group 9 — elementId ""
324
+ // ===========================================================================
325
+
326
+ describe("readPlanComments — elementId '' (empty string)", () => {
327
+ it("returns all comments (same as omitting)", () => {
328
+ const slug = "empty-element-plan";
329
+ writePlanJson(slug, SAMPLE_PLAN);
330
+
331
+ const result = readPlanComments(worktree, slug, "");
332
+ expect(result.ok).toBe(true);
333
+ if (result.ok) {
334
+ expect(result.comments).toHaveLength(4);
335
+ }
336
+ });
337
+ });
338
+
339
+ // ===========================================================================
340
+ // Group 10 — chronological sort
341
+ // ===========================================================================
342
+
343
+ describe("readPlanComments — chronological sort", () => {
344
+ it("returns comments sorted oldest first by `created`", () => {
345
+ const slug = "sort-plan";
346
+ writePlanJson(slug, {
347
+ schemaVersion: 2,
348
+ elements: [],
349
+ comments: [
350
+ { id: "c_late", elementId: "el_x", text: "Late", created: "2026-06-18T15:00:00Z" },
351
+ { id: "c_early", elementId: "el_x", text: "Early", created: "2026-06-18T09:00:00Z" },
352
+ { id: "c_mid", elementId: "el_x", text: "Mid", created: "2026-06-18T12:00:00Z" },
353
+ ],
354
+ });
355
+
356
+ const result = readPlanComments(worktree, slug, undefined);
357
+ expect(result.ok).toBe(true);
358
+ if (result.ok) {
359
+ const ids = result.comments.map((c) => c.id);
360
+ expect(ids).toEqual(["c_early", "c_mid", "c_late"]);
361
+ }
362
+ });
363
+ });
364
+
365
+ // ===========================================================================
366
+ // Group 11 — missing created timestamp
367
+ // ===========================================================================
368
+
369
+ describe("readPlanComments — missing created timestamp", () => {
370
+ it("pushes comments without `created` to the end of the sorted result", () => {
371
+ const slug = "no-timestamp-plan";
372
+ writePlanJson(slug, SAMPLE_PLAN);
373
+
374
+ const result = readPlanComments(worktree, slug, undefined);
375
+ expect(result.ok).toBe(true);
376
+ if (result.ok) {
377
+ const ids = result.comments.map((c) => c.id);
378
+ // The 3 timestamped comments come first (oldest to newest), then c_4 with no timestamp
379
+ expect(ids).toEqual(["c_1", "c_2", "c_3", "c_4"]);
380
+ }
381
+ });
382
+
383
+ it("places comments with empty-string `created` at the end too", () => {
384
+ const slug = "empty-timestamp-plan";
385
+ writePlanJson(slug, {
386
+ schemaVersion: 2,
387
+ elements: [],
388
+ comments: [
389
+ { id: "c_no_ts", elementId: "el_x", text: "No ts", created: "" },
390
+ { id: "c_a", elementId: "el_x", text: "Has ts", created: "2026-06-18T10:00:00Z" },
391
+ ],
392
+ });
393
+
394
+ const result = readPlanComments(worktree, slug, undefined);
395
+ expect(result.ok).toBe(true);
396
+ if (result.ok) {
397
+ expect(result.comments.map((c) => c.id)).toEqual(["c_a", "c_no_ts"]);
398
+ }
399
+ });
400
+ });
401
+
402
+ // ===========================================================================
403
+ // Group 12 — malformed comments filtered out
404
+ // ===========================================================================
405
+
406
+ describe("readPlanComments — malformed comments", () => {
407
+ it("filters out entries without a string `id`", () => {
408
+ const slug = "malformed-plan";
409
+ writePlanJson(slug, {
410
+ schemaVersion: 2,
411
+ elements: [],
412
+ comments: [
413
+ { id: "c_valid", elementId: "el_x", text: "Has id", created: "2026-06-18T10:00:00Z" },
414
+ // missing id entirely
415
+ { elementId: "el_x", text: "No id at all" } as unknown as { id: string },
416
+ // id is the wrong type
417
+ { id: 42, elementId: "el_x", text: "Non-string id" } as unknown as { id: string },
418
+ // null entry in the array
419
+ null,
420
+ // string entry
421
+ "string comment",
422
+ ],
423
+ });
424
+
425
+ const result = readPlanComments(worktree, slug, undefined);
426
+ expect(result.ok).toBe(true);
427
+ if (result.ok) {
428
+ expect(result.comments).toHaveLength(1);
429
+ expect(result.comments[0]?.id).toBe("c_valid");
430
+ }
431
+ });
432
+ });
433
+
434
+ // ===========================================================================
435
+ // Group 13 — thread replies preserved
436
+ // ===========================================================================
437
+
438
+ describe("readPlanComments — thread replies", () => {
439
+ it("passes the `thread` field through unchanged", () => {
440
+ const slug = "thread-plan";
441
+ const thread = [
442
+ { id: "r_1", author: "ai", text: "Done.", created: "2026-06-18T11:00:00Z" },
443
+ { id: "r_2", author: "DrB0rk", text: "Thanks!", created: "2026-06-18T12:00:00Z" },
444
+ ];
445
+ writePlanJson(slug, {
446
+ schemaVersion: 2,
447
+ elements: [],
448
+ comments: [
449
+ {
450
+ id: "c_1",
451
+ elementId: "el_x",
452
+ text: "Please address feedback",
453
+ created: "2026-06-18T10:00:00Z",
454
+ thread,
455
+ },
456
+ ],
457
+ });
458
+
459
+ const result = readPlanComments(worktree, slug, undefined);
460
+ expect(result.ok).toBe(true);
461
+ if (result.ok) {
462
+ expect(result.comments).toHaveLength(1);
463
+ // The thread array contents must be preserved (deep equality). Note:
464
+ // we do NOT assert reference identity — `JSON.parse` always produces
465
+ // fresh objects, so the returned thread is a deep-equal copy, not the
466
+ // same array reference we wrote to disk.
467
+ expect(result.comments[0]?.thread).toEqual(thread);
468
+ }
469
+ });
470
+
471
+ it("returns undefined `thread` when the comment has no thread field", () => {
472
+ const slug = "no-thread-plan";
473
+ writePlanJson(slug, {
474
+ schemaVersion: 2,
475
+ elements: [],
476
+ comments: [{ id: "c_1", elementId: "el_x", text: "Plain", created: "2026-06-18T10:00:00Z" }],
477
+ });
478
+
479
+ const result = readPlanComments(worktree, slug, undefined);
480
+ expect(result.ok).toBe(true);
481
+ if (result.ok) {
482
+ expect(result.comments[0]?.thread).toBeUndefined();
483
+ }
484
+ });
485
+ });