@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,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
|
+
});
|