@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,548 @@
1
+ /**
2
+ * commands.test.ts
3
+ *
4
+ * Tests for the pure `parseSlashCommand` function in src/commands.ts.
5
+ *
6
+ * The parser is a pure function — no I/O, no Date.now() — so all tests
7
+ * just feed it text and assert on the returned descriptor.
8
+ *
9
+ * Groups:
10
+ * 1. /visual-plan on/off/status
11
+ * 2. /plan new (with and without template)
12
+ * 3. /plan list
13
+ * 4. /plan open
14
+ * 5. /help / /commands
15
+ * 6. non-slash messages → null
16
+ * 7. unknown commands → error response
17
+ * 8. invalid input → error response
18
+ */
19
+
20
+ import { describe, test, expect } from "bun:test";
21
+ import { parseSlashCommand } from "../src/commands";
22
+ import {
23
+ DEFAULT_PLAN_SETTINGS,
24
+ type PlanSettings,
25
+ } from "../src/settings";
26
+
27
+ const baseCtx = {
28
+ currentSettings: DEFAULT_PLAN_SETTINGS,
29
+ availablePlanSlugs: [] as readonly string[],
30
+ defaultPort: 4321,
31
+ };
32
+
33
+ function ctxWith(
34
+ overrides: Partial<{
35
+ currentSettings: PlanSettings;
36
+ availablePlanSlugs: readonly string[];
37
+ defaultPort: number;
38
+ }> = {},
39
+ ) {
40
+ return {
41
+ currentSettings: overrides.currentSettings ?? DEFAULT_PLAN_SETTINGS,
42
+ availablePlanSlugs: overrides.availablePlanSlugs ?? [],
43
+ defaultPort: overrides.defaultPort ?? 4321,
44
+ };
45
+ }
46
+
47
+ // ===========================================================================
48
+ // Group 1 — /visual-plan on/off/status
49
+ // ===========================================================================
50
+
51
+ describe("parseSlashCommand — /visual-plan", () => {
52
+ test("'/visual-plan on' returns settingsPatch: { visualPlanEnabled: true }", () => {
53
+ const r = parseSlashCommand("/visual-plan on", baseCtx);
54
+ expect(r).not.toBeNull();
55
+ expect(r!.handled).toBe(true);
56
+ expect(r!.settingsPatch).toEqual({ visualPlanEnabled: true });
57
+ expect(r!.response).toMatch(/on/i);
58
+ });
59
+
60
+ test("'/visual-plan off' returns settingsPatch: { visualPlanEnabled: false }", () => {
61
+ const r = parseSlashCommand("/visual-plan off", baseCtx);
62
+ expect(r).not.toBeNull();
63
+ expect(r!.settingsPatch).toEqual({ visualPlanEnabled: false });
64
+ expect(r!.response).toMatch(/off/i);
65
+ });
66
+
67
+ test("'/visual-plan' (no arg) returns current state without mutation", () => {
68
+ const on = ctxWith({ currentSettings: { ...DEFAULT_PLAN_SETTINGS, visualPlanEnabled: true } });
69
+ const r = parseSlashCommand("/visual-plan", on);
70
+ expect(r).not.toBeNull();
71
+ expect(r!.settingsPatch).toBeUndefined();
72
+ expect(r!.response).toMatch(/on/);
73
+ });
74
+
75
+ test("'/visual-plan' reflects off state correctly", () => {
76
+ const off = ctxWith({ currentSettings: { ...DEFAULT_PLAN_SETTINGS, visualPlanEnabled: false } });
77
+ const r = parseSlashCommand("/visual-plan", off);
78
+ expect(r!.response).toMatch(/off/);
79
+ });
80
+
81
+ test("'/visual-plan status' returns status without mutation", () => {
82
+ const r = parseSlashCommand("/visual-plan status", baseCtx);
83
+ expect(r!.settingsPatch).toBeUndefined();
84
+ expect(r!.response).toMatch(/off/);
85
+ });
86
+
87
+ test("'/visual-plan true' is treated like 'on'", () => {
88
+ const r = parseSlashCommand("/visual-plan true", baseCtx);
89
+ expect(r!.settingsPatch).toEqual({ visualPlanEnabled: true });
90
+ });
91
+
92
+ test("'/visual-plan false' is treated like 'off'", () => {
93
+ const r = parseSlashCommand("/visual-plan false", baseCtx);
94
+ expect(r!.settingsPatch).toEqual({ visualPlanEnabled: false });
95
+ });
96
+
97
+ test("'/visual-plan nonsense' returns an error response (still handled)", () => {
98
+ const r = parseSlashCommand("/visual-plan nonsense", baseCtx);
99
+ expect(r).not.toBeNull();
100
+ expect(r!.handled).toBe(true);
101
+ expect(r!.settingsPatch).toBeUndefined();
102
+ expect(r!.response).toMatch(/Unknown argument/);
103
+ });
104
+ });
105
+
106
+ // ===========================================================================
107
+ // Group 2 — /plan new
108
+ // ===========================================================================
109
+
110
+ describe("parseSlashCommand — /plan new", () => {
111
+ test("'/plan new foo' returns success response + sideEffect=create_plan", () => {
112
+ const r = parseSlashCommand("/plan new foo", baseCtx);
113
+ expect(r).not.toBeNull();
114
+ expect(r!.handled).toBe(true);
115
+ expect(r!.sideEffect).toEqual({
116
+ kind: "create_plan",
117
+ slug: "foo",
118
+ template: null,
119
+ });
120
+ expect(r!.settingsPatch).toEqual({ lastUsedSlug: "foo" });
121
+ expect(r!.response).toMatch(/foo/);
122
+ });
123
+
124
+ test("'/plan new foo feature-design' uses the named template", () => {
125
+ const r = parseSlashCommand("/plan new foo feature-design", baseCtx);
126
+ expect(r!.sideEffect).toEqual({
127
+ kind: "create_plan",
128
+ slug: "foo",
129
+ template: "feature-design",
130
+ });
131
+ expect(r!.response).toMatch(/feature-design/);
132
+ });
133
+
134
+ test("'/plan new foo decision-record' uses the decision-record template", () => {
135
+ const r = parseSlashCommand("/plan new foo decision-record", baseCtx);
136
+ expect(r!.sideEffect).toEqual({
137
+ kind: "create_plan",
138
+ slug: "foo",
139
+ template: "decision-record",
140
+ });
141
+ });
142
+
143
+ test("'/plan new foo unknown-template' returns an error response", () => {
144
+ const r = parseSlashCommand("/plan new foo unknown-template", baseCtx);
145
+ expect(r!.handled).toBe(true);
146
+ expect(r!.sideEffect).toBeUndefined();
147
+ expect(r!.response).toMatch(/Unknown template/);
148
+ });
149
+
150
+ test("'/plan new' (no slug) returns a usage message", () => {
151
+ const r = parseSlashCommand("/plan new", baseCtx);
152
+ expect(r!.response).toMatch(/Usage/);
153
+ expect(r!.sideEffect).toBeUndefined();
154
+ });
155
+
156
+ test("'/plan new UPPERCASE' rejects invalid slug", () => {
157
+ const r = parseSlashCommand("/plan new UPPERCASE", baseCtx);
158
+ expect(r!.response).toMatch(/Invalid slug/);
159
+ expect(r!.sideEffect).toBeUndefined();
160
+ });
161
+
162
+ test("'/plan new my-feature' (with hyphen) is accepted", () => {
163
+ const r = parseSlashCommand("/plan new my-feature", baseCtx);
164
+ expect(r!.sideEffect).toEqual({
165
+ kind: "create_plan",
166
+ slug: "my-feature",
167
+ template: null,
168
+ });
169
+ });
170
+
171
+ test("'/plan new' falls back to currentSettings.defaultTemplate", () => {
172
+ const ctx = ctxWith({
173
+ currentSettings: { ...DEFAULT_PLAN_SETTINGS, defaultTemplate: "bug-investigation" },
174
+ });
175
+ const r = parseSlashCommand("/plan new foo", ctx);
176
+ expect(r!.response).toMatch(/bug-investigation/);
177
+ expect(r!.sideEffect).toEqual({
178
+ kind: "create_plan",
179
+ slug: "foo",
180
+ template: null, // null in sideEffect; the template name in the response reflects the default
181
+ });
182
+ });
183
+ });
184
+
185
+ // ===========================================================================
186
+ // Group 3 — /plan list
187
+ // ===========================================================================
188
+
189
+ describe("parseSlashCommand — /plan list", () => {
190
+ test("'/plan list' with no plans returns 'no plans found' message", () => {
191
+ const r = parseSlashCommand("/plan list", baseCtx);
192
+ expect(r!.sideEffect).toEqual({ kind: "list_plans" });
193
+ expect(r!.response).toMatch(/No plans found/);
194
+ });
195
+
196
+ test("'/plan list' with available plans lists them", () => {
197
+ const ctx = ctxWith({ availablePlanSlugs: ["alpha", "beta", "gamma"] });
198
+ const r = parseSlashCommand("/plan list", ctx);
199
+ expect(r!.sideEffect).toEqual({ kind: "list_plans" });
200
+ expect(r!.response).toContain("alpha");
201
+ expect(r!.response).toContain("beta");
202
+ expect(r!.response).toContain("gamma");
203
+ expect(r!.response).toContain("(3)");
204
+ });
205
+
206
+ test("'/plan ls' is accepted as alias for list", () => {
207
+ const r = parseSlashCommand("/plan ls", baseCtx);
208
+ expect(r!.sideEffect).toEqual({ kind: "list_plans" });
209
+ });
210
+ });
211
+
212
+ // ===========================================================================
213
+ // Group 4 — /plan open
214
+ // ===========================================================================
215
+
216
+ describe("parseSlashCommand — /plan open", () => {
217
+ test("'/plan open foo' returns the URL and updates lastUsedSlug", () => {
218
+ const r = parseSlashCommand("/plan open foo", baseCtx);
219
+ expect(r!.handled).toBe(true);
220
+ expect(r!.sideEffect).toEqual({ kind: "open_plan_url", slug: "foo" });
221
+ expect(r!.settingsPatch).toEqual({ lastUsedSlug: "foo" });
222
+ expect(r!.response).toContain("http://localhost:4321/foo/");
223
+ });
224
+
225
+ test("'/plan open' (no slug) returns a usage message", () => {
226
+ const r = parseSlashCommand("/plan open", baseCtx);
227
+ expect(r!.response).toMatch(/Usage/);
228
+ expect(r!.sideEffect).toBeUndefined();
229
+ });
230
+
231
+ test("'/plan open UPPERCASE' rejects invalid slug", () => {
232
+ const r = parseSlashCommand("/plan open UPPERCASE", baseCtx);
233
+ expect(r!.response).toMatch(/Invalid slug/);
234
+ expect(r!.sideEffect).toBeUndefined();
235
+ });
236
+
237
+ test("'/plan open' respects custom defaultPort in context", () => {
238
+ const ctx = ctxWith({ defaultPort: 5555 });
239
+ const r = parseSlashCommand("/plan open foo", ctx);
240
+ expect(r!.response).toContain("http://localhost:5555/foo/");
241
+ });
242
+ });
243
+
244
+ // ===========================================================================
245
+ // Group 5 — /help / /commands
246
+ // ===========================================================================
247
+
248
+ describe("parseSlashCommand — /help", () => {
249
+ test("'/help' returns the help text", () => {
250
+ const r = parseSlashCommand("/help", baseCtx);
251
+ expect(r!.handled).toBe(true);
252
+ expect(r!.response).toMatch(/Available commands/);
253
+ expect(r!.response).toMatch(/visual-plan/);
254
+ expect(r!.response).toMatch(/plan new/);
255
+ });
256
+
257
+ test("'/commands' is an alias for /help", () => {
258
+ const r = parseSlashCommand("/commands", baseCtx);
259
+ expect(r!.response).toMatch(/Available commands/);
260
+ });
261
+ });
262
+
263
+ // ===========================================================================
264
+ // Group 6 — non-slash messages return null
265
+ // ===========================================================================
266
+
267
+ describe("parseSlashCommand — non-slash messages", () => {
268
+ test("regular text returns null", () => {
269
+ expect(parseSlashCommand("hello world", baseCtx)).toBeNull();
270
+ });
271
+
272
+ test("empty string returns null", () => {
273
+ expect(parseSlashCommand("", baseCtx)).toBeNull();
274
+ });
275
+
276
+ test("whitespace-only returns null", () => {
277
+ expect(parseSlashCommand(" \t\n ", baseCtx)).toBeNull();
278
+ });
279
+
280
+ test("just a slash returns null", () => {
281
+ expect(parseSlashCommand("/", baseCtx)).toBeNull();
282
+ });
283
+
284
+ test("just whitespace and a slash returns null", () => {
285
+ expect(parseSlashCommand(" / ", baseCtx)).toBeNull();
286
+ });
287
+
288
+ test("non-string input returns null", () => {
289
+ expect(parseSlashCommand(null as never, baseCtx)).toBeNull();
290
+ expect(parseSlashCommand(undefined as never, baseCtx)).toBeNull();
291
+ expect(parseSlashCommand(42 as never, baseCtx)).toBeNull();
292
+ });
293
+ });
294
+
295
+ // ===========================================================================
296
+ // Group 7 — unknown commands return error response (still handled)
297
+ // ===========================================================================
298
+
299
+ describe("parseSlashCommand — unknown commands", () => {
300
+ test("'/unknown' returns error response (handled, not null)", () => {
301
+ const r = parseSlashCommand("/unknown", baseCtx);
302
+ expect(r).not.toBeNull();
303
+ expect(r!.handled).toBe(true);
304
+ expect(r!.response).toMatch(/Unknown command/);
305
+ expect(r!.response).toContain("/unknown");
306
+ });
307
+
308
+ test("'/foo bar baz' returns error response", () => {
309
+ const r = parseSlashCommand("/foo bar baz", baseCtx);
310
+ expect(r!.response).toMatch(/Unknown command/);
311
+ });
312
+
313
+ test("'/plan delete foo bar' routes to bizar_plan_action (delete_element)", () => {
314
+ const r = parseSlashCommand("/plan delete foo bar", baseCtx);
315
+ expect(r).not.toBeNull();
316
+ expect(r!.handled).toBe(true);
317
+ expect(r!.sideEffect).toEqual({
318
+ kind: "tool_invocation",
319
+ toolName: "bizar_plan_action",
320
+ args: { action: "delete_element", planSlug: "foo", elementId: "bar" },
321
+ });
322
+ expect(r!.settingsPatch).toEqual({ lastUsedSlug: "foo" });
323
+ });
324
+ });
325
+
326
+ // ===========================================================================
327
+ // Group 8 — case-insensitivity
328
+ // ===========================================================================
329
+
330
+ describe("parseSlashCommand — case-insensitivity", () => {
331
+ test("'/VISUAL-PLAN ON' (uppercase) is treated like '/visual-plan on'", () => {
332
+ const r = parseSlashCommand("/VISUAL-PLAN ON", baseCtx);
333
+ expect(r!.settingsPatch).toEqual({ visualPlanEnabled: true });
334
+ });
335
+
336
+ test("'/Help' (mixed case) is treated like '/help'", () => {
337
+ const r = parseSlashCommand("/Help", baseCtx);
338
+ expect(r!.response).toMatch(/Available commands/);
339
+ });
340
+ });
341
+
342
+ // ===========================================================================
343
+ // Group 9 — /plan (no subcommand) shows usage
344
+ // ===========================================================================
345
+
346
+ describe("parseSlashCommand — /plan usage", () => {
347
+ test("'/plan' (no subcommand) shows usage", () => {
348
+ const r = parseSlashCommand("/plan", baseCtx);
349
+ expect(r!.response).toMatch(/Plan commands:/);
350
+ expect(r!.response).toMatch(/\/plan new/);
351
+ expect(r!.response).toMatch(/\/plan list/);
352
+ });
353
+ });
354
+
355
+ // ===========================================================================
356
+ // Group 10 — v0.5.0 subcommands (R3)
357
+ // ===========================================================================
358
+
359
+ describe("parseSlashCommand — /plan get", () => {
360
+ test("'/plan get foo' routes to bizar_plan_action get_canvas", () => {
361
+ const r = parseSlashCommand("/plan get foo", baseCtx);
362
+ expect(r).not.toBeNull();
363
+ expect(r!.handled).toBe(true);
364
+ expect(r!.sideEffect).toEqual({
365
+ kind: "tool_invocation",
366
+ toolName: "bizar_plan_action",
367
+ args: { action: "get_canvas", planSlug: "foo" },
368
+ });
369
+ expect(r!.settingsPatch).toEqual({ lastUsedSlug: "foo" });
370
+ });
371
+
372
+ test("'/plan get' (no slug) returns usage", () => {
373
+ const r = parseSlashCommand("/plan get", baseCtx);
374
+ expect(r!.response).toMatch(/Usage/);
375
+ });
376
+ });
377
+
378
+ describe("parseSlashCommand — /plan add", () => {
379
+ test("'/plan add foo --title X --type task' builds add_element args", () => {
380
+ const r = parseSlashCommand("/plan add foo --title X --type task", baseCtx);
381
+ expect(r!.handled).toBe(true);
382
+ expect(r!.sideEffect).toEqual({
383
+ kind: "tool_invocation",
384
+ toolName: "bizar_plan_action",
385
+ args: {
386
+ action: "add_element",
387
+ planSlug: "foo",
388
+ element: { title: "X", type: "task" },
389
+ },
390
+ });
391
+ expect(r!.settingsPatch).toEqual({ lastUsedSlug: "foo" });
392
+ });
393
+
394
+ test("'/plan add foo --title \"Hello world\"' preserves quoted strings", () => {
395
+ const r = parseSlashCommand('/plan add foo --title "Hello world"', baseCtx);
396
+ expect(r!.sideEffect).toMatchObject({
397
+ kind: "tool_invocation",
398
+ toolName: "bizar_plan_action",
399
+ args: {
400
+ action: "add_element",
401
+ planSlug: "foo",
402
+ element: { title: "Hello world" },
403
+ },
404
+ });
405
+ });
406
+
407
+ test("'/plan add foo --x 100 --y 200' parses numeric flags", () => {
408
+ const r = parseSlashCommand("/plan add foo --x 100 --y 200", baseCtx);
409
+ expect(r!.sideEffect).toMatchObject({
410
+ args: {
411
+ action: "add_element",
412
+ planSlug: "foo",
413
+ element: { x: 100, y: 200 },
414
+ },
415
+ });
416
+ });
417
+
418
+ test("'/plan add foo' (no flags) returns usage", () => {
419
+ const r = parseSlashCommand("/plan add foo", baseCtx);
420
+ expect(r!.response).toMatch(/Usage/);
421
+ expect(r!.response).toMatch(/At least one/);
422
+ });
423
+ });
424
+
425
+ describe("parseSlashCommand — /plan update", () => {
426
+ test("'/plan update foo el_1 --x 50 --y 60' builds update_element args", () => {
427
+ const r = parseSlashCommand("/plan update foo el_1 --x 50 --y 60", baseCtx);
428
+ expect(r!.sideEffect).toEqual({
429
+ kind: "tool_invocation",
430
+ toolName: "bizar_plan_action",
431
+ args: {
432
+ action: "update_element",
433
+ planSlug: "foo",
434
+ elementId: "el_1",
435
+ element: { x: 50, y: 60 },
436
+ },
437
+ });
438
+ });
439
+
440
+ test("'/plan update foo el_1' (no flags) returns usage", () => {
441
+ const r = parseSlashCommand("/plan update foo el_1", baseCtx);
442
+ expect(r!.response).toMatch(/Usage/);
443
+ });
444
+ });
445
+
446
+ describe("parseSlashCommand — /plan comment", () => {
447
+ test("'/plan comment foo \"Make it bigger\"' builds add_comment (canvas-pinned)", () => {
448
+ const r = parseSlashCommand('/plan comment foo "Make it bigger"', baseCtx);
449
+ expect(r!.sideEffect).toEqual({
450
+ kind: "tool_invocation",
451
+ toolName: "bizar_plan_action",
452
+ args: {
453
+ action: "add_comment",
454
+ planSlug: "foo",
455
+ comment: {
456
+ elementId: null,
457
+ author: "user",
458
+ text: "Make it bigger",
459
+ },
460
+ },
461
+ });
462
+ });
463
+
464
+ test("'/plan comment foo el_1 \"text\"' pins comment to element", () => {
465
+ const r = parseSlashCommand('/plan comment foo el_1 "text"', baseCtx);
466
+ expect(r!.sideEffect).toEqual({
467
+ kind: "tool_invocation",
468
+ toolName: "bizar_plan_action",
469
+ args: {
470
+ action: "add_comment",
471
+ planSlug: "foo",
472
+ comment: { elementId: "el_1", author: "user", text: "text" },
473
+ },
474
+ });
475
+ });
476
+ });
477
+
478
+ describe("parseSlashCommand — /plan comments (list)", () => {
479
+ test("'/plan comments foo' routes to bizar_get_plan_comments", () => {
480
+ const r = parseSlashCommand("/plan comments foo", baseCtx);
481
+ expect(r!.sideEffect).toEqual({
482
+ kind: "tool_invocation",
483
+ toolName: "bizar_get_plan_comments",
484
+ args: { planSlug: "foo" },
485
+ });
486
+ });
487
+
488
+ test("'/plan comments foo el_1' filters by elementId", () => {
489
+ const r = parseSlashCommand("/plan comments foo el_1", baseCtx);
490
+ expect(r!.sideEffect).toEqual({
491
+ kind: "tool_invocation",
492
+ toolName: "bizar_get_plan_comments",
493
+ args: { planSlug: "foo", elementId: "el_1" },
494
+ });
495
+ });
496
+ });
497
+
498
+ describe("parseSlashCommand — /plan status", () => {
499
+ test("'/plan status foo approved' builds set_status args", () => {
500
+ const r = parseSlashCommand("/plan status foo approved", baseCtx);
501
+ expect(r!.sideEffect).toEqual({
502
+ kind: "tool_invocation",
503
+ toolName: "bizar_plan_action",
504
+ args: { action: "set_status", planSlug: "foo", status: "approved" },
505
+ });
506
+ });
507
+
508
+ test("'/plan status foo bogus' rejects invalid status", () => {
509
+ const r = parseSlashCommand("/plan status foo bogus", baseCtx);
510
+ expect(r!.response).toMatch(/Invalid status/);
511
+ expect(r!.sideEffect).toBeUndefined();
512
+ });
513
+ });
514
+
515
+ describe("parseSlashCommand — /plan wait (deferred)", () => {
516
+ test("'/plan wait foo' returns the deferred response (no side effect, no block)", () => {
517
+ const r = parseSlashCommand("/plan wait foo", baseCtx);
518
+ expect(r).not.toBeNull();
519
+ expect(r!.handled).toBe(true);
520
+ expect(r!.sideEffect).toBeUndefined();
521
+ expect(r!.response).toMatch(/deferred/i);
522
+ expect(r!.response).toMatch(/bizar_wait_for_feedback/);
523
+ });
524
+
525
+ test("'/plan wait' (no slug) returns usage", () => {
526
+ const r = parseSlashCommand("/plan wait", baseCtx);
527
+ expect(r!.response).toMatch(/Usage/);
528
+ });
529
+ });
530
+
531
+ describe("parseSlashCommand — /help includes new subcommands", () => {
532
+ test("'/help' lists get, add, update, delete, comment, comments, status, wait", () => {
533
+ const r = parseSlashCommand("/help", baseCtx);
534
+ const text = r!.response;
535
+ for (const cmd of [
536
+ "/plan get",
537
+ "/plan add",
538
+ "/plan update",
539
+ "/plan delete",
540
+ "/plan comment",
541
+ "/plan comments",
542
+ "/plan status",
543
+ "/plan wait",
544
+ ]) {
545
+ expect(text).toContain(cmd);
546
+ }
547
+ });
548
+ });
@@ -0,0 +1,122 @@
1
+ /**
2
+ * config.test.ts — Config drift detection tests (R4 audit).
3
+ *
4
+ * Verifies:
5
+ * 1. Every `bizar_*` tool registered in `plugins/bizar/index.ts`
6
+ * is also present in `config/opencode.json` `tools: { ... }`.
7
+ * 2. No `bizarre_*` (double-r) typos remain in `plugins/bizar/src/`.
8
+ * 3. `plugins/bizar/package.json` version is `0.5.0`.
9
+ */
10
+
11
+ import { describe, test, expect } from "bun:test";
12
+ import { readFileSync, readdirSync, statSync } from "node:fs";
13
+ import { join } from "node:path";
14
+
15
+ const BIZAR_PLUGIN_ROOT = join(__dirname, "..");
16
+ const PLUGIN_INDEX = join(BIZAR_PLUGIN_ROOT, "index.ts");
17
+ const CONFIG_OPENCODE = join(__dirname, "..", "..", "..", "config", "opencode.json");
18
+ const PKG_JSON = join(BIZAR_PLUGIN_ROOT, "package.json");
19
+ const SRC_DIR = join(BIZAR_PLUGIN_ROOT, "src");
20
+
21
+ // ---------------------------------------------------------------------------
22
+ // Helpers
23
+ // ---------------------------------------------------------------------------
24
+
25
+ /**
26
+ * Extract `bizar_*` tool registration keys from index.ts.
27
+ * Matches: `bizar_spawn_background: createBgSpawnTool(...)` etc.
28
+ */
29
+ function extractPluginToolKeys(indexContent: string): string[] {
30
+ const keys: string[] = [];
31
+ // Matches `bizar_xxx: createXxxTool(` or `bizar_xxx: createBgXxxTool(`
32
+ const toolKeyRegex = /\bbizar_([a-z_]+)\s*:\s*(?:create(?:Bg)?[A-Z]\w*Tool|createWaitForFeedbackTool)\s*\(/g;
33
+ let m: RegExpExecArray | null;
34
+ while ((m = toolKeyRegex.exec(indexContent)) !== null) {
35
+ keys.push(`bizar_${m[1]}`);
36
+ }
37
+ return [...new Set(keys)].sort();
38
+ }
39
+
40
+ /**
41
+ * Extract `tools: { ... }` keys from config/opencode.json.
42
+ */
43
+ function extractConfigToolKeys(configContent: string): string[] {
44
+ const parsed = JSON.parse(configContent) as { tools?: Record<string, unknown> };
45
+ if (!parsed.tools) return [];
46
+ return Object.keys(parsed.tools).sort();
47
+ }
48
+
49
+ // ---------------------------------------------------------------------------
50
+ // Test cases
51
+ // ---------------------------------------------------------------------------
52
+
53
+ describe("config drift detection", () => {
54
+ test("plugin_tool_keys ⊆ config_tools_keys (R4 audit)", () => {
55
+ const indexContent = readFileSync(PLUGIN_INDEX, "utf-8");
56
+ const configContent = readFileSync(CONFIG_OPENCODE, "utf-8");
57
+
58
+ const pluginKeys = extractPluginToolKeys(indexContent);
59
+ const configKeys = extractConfigToolKeys(configContent);
60
+
61
+ const missingInConfig = pluginKeys.filter((k) => !configKeys.includes(k));
62
+ expect(missingInConfig, `Plugin tools missing in config: ${JSON.stringify(missingInConfig)}`).toEqual([]);
63
+ });
64
+
65
+ test("no 'bizarre_*' (double-r) typos remain in plugins/bizar/src/", () => {
66
+ // Walk src/ directory recursively and grep for 'bizarre_'
67
+ const allSrcFiles = getAllTsFiles(SRC_DIR);
68
+ const violations: Array<{ file: string; match: string }> = [];
69
+
70
+ const doubleRRegex = /\bbizarre_[a-z_]+\b/g;
71
+
72
+ for (const file of allSrcFiles) {
73
+ const content = readFileSync(file, "utf-8");
74
+ let m: RegExpExecArray | null;
75
+ while ((m = doubleRRegex.exec(content)) !== null) {
76
+ violations.push({ file: relativeToSrc(file), match: m[0] });
77
+ }
78
+ }
79
+
80
+ expect(
81
+ violations,
82
+ `Found 'bizarre_*' typos: ${JSON.stringify(violations)}`,
83
+ ).toEqual([]);
84
+ });
85
+
86
+ test("plugins/bizar/package.json version is 0.5.4", () => {
87
+ const pkg = JSON.parse(readFileSync(PKG_JSON, "utf-8")) as { version?: string };
88
+ expect(pkg.version).toBe("0.5.4");
89
+ });
90
+ });
91
+
92
+ // ---------------------------------------------------------------------------
93
+ // Helper: recursively collect all .ts files under a directory
94
+ // ---------------------------------------------------------------------------
95
+
96
+ function getAllTsFiles(dir: string): string[] {
97
+ const results: string[] = [];
98
+ let entries: string[];
99
+ try {
100
+ entries = readdirSync(dir);
101
+ } catch {
102
+ return results;
103
+ }
104
+ for (const name of entries) {
105
+ const full = join(dir, name);
106
+ try {
107
+ const stat = statSync(full);
108
+ if (stat.isDirectory()) {
109
+ results.push(...getAllTsFiles(full));
110
+ } else if (name.endsWith(".ts") || name.endsWith(".tsx")) {
111
+ results.push(full);
112
+ }
113
+ } catch {
114
+ // skip inaccessible entries
115
+ }
116
+ }
117
+ return results;
118
+ }
119
+
120
+ function relativeToSrc(file: string): string {
121
+ return file.replace(SRC_DIR + "/", "");
122
+ }