gsd-pi 2.41.0-dev.9446b20 → 2.41.0-dev.97349b1

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 (218) hide show
  1. package/README.md +69 -29
  2. package/dist/cli-web-branch.d.ts +6 -0
  3. package/dist/cli-web-branch.js +17 -0
  4. package/dist/onboarding.js +2 -1
  5. package/dist/resources/extensions/gsd/auto/loop.js +9 -1
  6. package/dist/resources/extensions/gsd/auto/phases.js +26 -8
  7. package/dist/resources/extensions/gsd/auto-dashboard.js +6 -2
  8. package/dist/resources/extensions/gsd/auto-dispatch.js +19 -2
  9. package/dist/resources/extensions/gsd/auto-post-unit.js +7 -0
  10. package/dist/resources/extensions/gsd/auto-recovery.js +12 -4
  11. package/dist/resources/extensions/gsd/auto-start.js +8 -3
  12. package/dist/resources/extensions/gsd/auto-worktree.js +147 -13
  13. package/dist/resources/extensions/gsd/auto.js +36 -1
  14. package/dist/resources/extensions/gsd/bootstrap/db-tools.js +199 -164
  15. package/dist/resources/extensions/gsd/bootstrap/journal-tools.js +62 -0
  16. package/dist/resources/extensions/gsd/bootstrap/register-extension.js +2 -0
  17. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +16 -0
  18. package/dist/resources/extensions/gsd/commands/catalog.js +8 -1
  19. package/dist/resources/extensions/gsd/commands/handlers/core.js +1 -0
  20. package/dist/resources/extensions/gsd/commands/handlers/ops.js +5 -0
  21. package/dist/resources/extensions/gsd/context-store.js +4 -3
  22. package/dist/resources/extensions/gsd/db-writer.js +5 -2
  23. package/dist/resources/extensions/gsd/detection.js +1 -1
  24. package/dist/resources/extensions/gsd/doctor.js +11 -1
  25. package/dist/resources/extensions/gsd/exit-command.js +12 -2
  26. package/dist/resources/extensions/gsd/export.js +9 -13
  27. package/dist/resources/extensions/gsd/extension-manifest.json +2 -2
  28. package/dist/resources/extensions/gsd/files.js +28 -11
  29. package/dist/resources/extensions/gsd/forensics.js +10 -3
  30. package/dist/resources/extensions/gsd/git-service.js +5 -1
  31. package/dist/resources/extensions/gsd/gsd-db.js +25 -8
  32. package/dist/resources/extensions/gsd/guided-flow-queue.js +1 -1
  33. package/dist/resources/extensions/gsd/guided-flow.js +7 -3
  34. package/dist/resources/extensions/gsd/journal.js +85 -0
  35. package/dist/resources/extensions/gsd/md-importer.js +5 -0
  36. package/dist/resources/extensions/gsd/milestone-ids.js +1 -1
  37. package/dist/resources/extensions/gsd/native-git-bridge.js +2 -2
  38. package/dist/resources/extensions/gsd/post-unit-hooks.js +24 -412
  39. package/dist/resources/extensions/gsd/preferences-types.js +1 -0
  40. package/dist/resources/extensions/gsd/preferences.js +1 -0
  41. package/dist/resources/extensions/gsd/prompt-loader.js +34 -4
  42. package/dist/resources/extensions/gsd/prompts/complete-milestone.md +11 -10
  43. package/dist/resources/extensions/gsd/prompts/discuss-headless.md +2 -2
  44. package/dist/resources/extensions/gsd/prompts/discuss.md +1 -1
  45. package/dist/resources/extensions/gsd/prompts/queue.md +1 -1
  46. package/dist/resources/extensions/gsd/repo-identity.js +46 -2
  47. package/dist/resources/extensions/gsd/rule-registry.js +489 -0
  48. package/dist/resources/extensions/gsd/rule-types.js +6 -0
  49. package/dist/resources/extensions/gsd/service-tier.js +138 -0
  50. package/dist/resources/extensions/gsd/structured-data-formatter.js +2 -1
  51. package/dist/resources/extensions/gsd/templates/decisions.md +2 -2
  52. package/dist/resources/extensions/gsd/workflow-templates.js +13 -1
  53. package/dist/resources/extensions/gsd/worktree-manager.js +20 -6
  54. package/dist/resources/extensions/gsd/worktree-resolver.js +19 -2
  55. package/dist/resources/extensions/subagent/index.js +7 -3
  56. package/dist/resources/extensions/voice/index.js +4 -4
  57. package/dist/web/standalone/.next/BUILD_ID +1 -1
  58. package/dist/web/standalone/.next/app-path-routes-manifest.json +16 -16
  59. package/dist/web/standalone/.next/build-manifest.json +3 -3
  60. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  61. package/dist/web/standalone/.next/react-loadable-manifest.json +1 -1
  62. package/dist/web/standalone/.next/server/app/_global-error.html +2 -2
  63. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  64. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  65. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  66. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  67. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  68. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  69. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  70. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  71. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  72. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  73. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  74. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  75. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  76. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  77. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  78. package/dist/web/standalone/.next/server/app/api/boot/route.js +1 -1
  79. package/dist/web/standalone/.next/server/app/api/captures/route.js +1 -1
  80. package/dist/web/standalone/.next/server/app/api/cleanup/route.js +1 -1
  81. package/dist/web/standalone/.next/server/app/api/doctor/route.js +1 -1
  82. package/dist/web/standalone/.next/server/app/api/export-data/route.js +1 -1
  83. package/dist/web/standalone/.next/server/app/api/forensics/route.js +1 -1
  84. package/dist/web/standalone/.next/server/app/api/history/route.js +1 -1
  85. package/dist/web/standalone/.next/server/app/api/hooks/route.js +1 -1
  86. package/dist/web/standalone/.next/server/app/api/recovery/route.js +1 -1
  87. package/dist/web/standalone/.next/server/app/api/settings-data/route.js +1 -1
  88. package/dist/web/standalone/.next/server/app/api/skill-health/route.js +1 -1
  89. package/dist/web/standalone/.next/server/app/api/terminal/input/route.js +1 -1
  90. package/dist/web/standalone/.next/server/app/api/terminal/resize/route.js +1 -1
  91. package/dist/web/standalone/.next/server/app/api/undo/route.js +1 -1
  92. package/dist/web/standalone/.next/server/app/api/visualizer/route.js +1 -1
  93. package/dist/web/standalone/.next/server/app/index.html +1 -1
  94. package/dist/web/standalone/.next/server/app/index.rsc +1 -1
  95. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  96. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
  97. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  98. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  99. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  100. package/dist/web/standalone/.next/server/app-paths-manifest.json +16 -16
  101. package/dist/web/standalone/.next/server/chunks/229.js +3 -3
  102. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  103. package/dist/web/standalone/.next/server/middleware-react-loadable-manifest.js +1 -1
  104. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  105. package/dist/web/standalone/.next/server/pages/500.html +2 -2
  106. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  107. package/dist/web/standalone/.next/static/chunks/4024.c195dc1fdd2adbea.js +9 -0
  108. package/dist/web/standalone/.next/static/chunks/{webpack-9afaaebf6042a1d7.js → webpack-fa307370fcf9fb2c.js} +1 -1
  109. package/dist/web-mode.d.ts +2 -0
  110. package/dist/web-mode.js +29 -7
  111. package/package.json +1 -1
  112. package/packages/native/src/__tests__/text.test.mjs +33 -0
  113. package/packages/pi-coding-agent/dist/core/discovery-cache.test.js +3 -1
  114. package/packages/pi-coding-agent/dist/core/discovery-cache.test.js.map +1 -1
  115. package/packages/pi-coding-agent/dist/modes/interactive/components/login-dialog.d.ts.map +1 -1
  116. package/packages/pi-coding-agent/dist/modes/interactive/components/login-dialog.js +10 -7
  117. package/packages/pi-coding-agent/dist/modes/interactive/components/login-dialog.js.map +1 -1
  118. package/packages/pi-coding-agent/src/core/discovery-cache.test.ts +4 -2
  119. package/packages/pi-coding-agent/src/modes/interactive/components/login-dialog.ts +11 -7
  120. package/src/resources/extensions/gsd/auto/loop-deps.ts +5 -1
  121. package/src/resources/extensions/gsd/auto/loop.ts +10 -1
  122. package/src/resources/extensions/gsd/auto/phases.ts +28 -8
  123. package/src/resources/extensions/gsd/auto/types.ts +4 -0
  124. package/src/resources/extensions/gsd/auto-dashboard.ts +7 -2
  125. package/src/resources/extensions/gsd/auto-dispatch.ts +25 -5
  126. package/src/resources/extensions/gsd/auto-post-unit.ts +8 -0
  127. package/src/resources/extensions/gsd/auto-recovery.ts +12 -4
  128. package/src/resources/extensions/gsd/auto-start.ts +8 -3
  129. package/src/resources/extensions/gsd/auto-worktree.ts +162 -18
  130. package/src/resources/extensions/gsd/auto.ts +40 -1
  131. package/src/resources/extensions/gsd/bootstrap/db-tools.ts +209 -162
  132. package/src/resources/extensions/gsd/bootstrap/journal-tools.ts +62 -0
  133. package/src/resources/extensions/gsd/bootstrap/register-extension.ts +2 -0
  134. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +13 -0
  135. package/src/resources/extensions/gsd/commands/catalog.ts +8 -1
  136. package/src/resources/extensions/gsd/commands/handlers/core.ts +1 -0
  137. package/src/resources/extensions/gsd/commands/handlers/ops.ts +5 -0
  138. package/src/resources/extensions/gsd/context-store.ts +4 -3
  139. package/src/resources/extensions/gsd/db-writer.ts +6 -2
  140. package/src/resources/extensions/gsd/detection.ts +1 -1
  141. package/src/resources/extensions/gsd/doctor.ts +12 -1
  142. package/src/resources/extensions/gsd/exit-command.ts +14 -2
  143. package/src/resources/extensions/gsd/export.ts +8 -15
  144. package/src/resources/extensions/gsd/extension-manifest.json +2 -2
  145. package/src/resources/extensions/gsd/files.ts +29 -12
  146. package/src/resources/extensions/gsd/forensics.ts +9 -3
  147. package/src/resources/extensions/gsd/git-service.ts +5 -4
  148. package/src/resources/extensions/gsd/gsd-db.ts +37 -8
  149. package/src/resources/extensions/gsd/guided-flow-queue.ts +1 -1
  150. package/src/resources/extensions/gsd/guided-flow.ts +7 -3
  151. package/src/resources/extensions/gsd/journal.ts +134 -0
  152. package/src/resources/extensions/gsd/md-importer.ts +6 -0
  153. package/src/resources/extensions/gsd/milestone-ids.ts +1 -1
  154. package/src/resources/extensions/gsd/native-git-bridge.ts +2 -2
  155. package/src/resources/extensions/gsd/post-unit-hooks.ts +24 -462
  156. package/src/resources/extensions/gsd/preferences-types.ts +3 -0
  157. package/src/resources/extensions/gsd/preferences.ts +1 -0
  158. package/src/resources/extensions/gsd/prompt-loader.ts +35 -4
  159. package/src/resources/extensions/gsd/prompts/complete-milestone.md +11 -10
  160. package/src/resources/extensions/gsd/prompts/discuss-headless.md +2 -2
  161. package/src/resources/extensions/gsd/prompts/discuss.md +1 -1
  162. package/src/resources/extensions/gsd/prompts/queue.md +1 -1
  163. package/src/resources/extensions/gsd/repo-identity.ts +47 -2
  164. package/src/resources/extensions/gsd/rule-registry.ts +599 -0
  165. package/src/resources/extensions/gsd/rule-types.ts +68 -0
  166. package/src/resources/extensions/gsd/service-tier.ts +171 -0
  167. package/src/resources/extensions/gsd/structured-data-formatter.ts +3 -1
  168. package/src/resources/extensions/gsd/templates/decisions.md +2 -2
  169. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +103 -120
  170. package/src/resources/extensions/gsd/tests/auto-recovery.test.ts +85 -0
  171. package/src/resources/extensions/gsd/tests/auto-secrets-gate.test.ts +2 -2
  172. package/src/resources/extensions/gsd/tests/auto-worktree-milestone-merge.test.ts +202 -0
  173. package/src/resources/extensions/gsd/tests/captures.test.ts +12 -1
  174. package/src/resources/extensions/gsd/tests/context-store.test.ts +10 -5
  175. package/src/resources/extensions/gsd/tests/continue-here.test.ts +20 -20
  176. package/src/resources/extensions/gsd/tests/db-writer.test.ts +10 -0
  177. package/src/resources/extensions/gsd/tests/doctor-completion-deferral.test.ts +15 -10
  178. package/src/resources/extensions/gsd/tests/doctor-fixlevel.test.ts +5 -4
  179. package/src/resources/extensions/gsd/tests/doctor-roadmap-summary-atomicity.test.ts +167 -0
  180. package/src/resources/extensions/gsd/tests/doctor-task-done-missing-summary-slice-loop.test.ts +174 -0
  181. package/src/resources/extensions/gsd/tests/exit-command.test.ts +55 -0
  182. package/src/resources/extensions/gsd/tests/gsd-db.test.ts +8 -1
  183. package/src/resources/extensions/gsd/tests/gsd-tools.test.ts +7 -7
  184. package/src/resources/extensions/gsd/tests/journal-integration.test.ts +513 -0
  185. package/src/resources/extensions/gsd/tests/journal-query-tool.test.ts +147 -0
  186. package/src/resources/extensions/gsd/tests/journal.test.ts +386 -0
  187. package/src/resources/extensions/gsd/tests/md-importer.test.ts +31 -1
  188. package/src/resources/extensions/gsd/tests/memory-store.test.ts +2 -2
  189. package/src/resources/extensions/gsd/tests/milestone-id-reservation.test.ts +1 -1
  190. package/src/resources/extensions/gsd/tests/parsers.test.ts +110 -0
  191. package/src/resources/extensions/gsd/tests/preferences.test.ts +47 -25
  192. package/src/resources/extensions/gsd/tests/prompt-db.test.ts +3 -1
  193. package/src/resources/extensions/gsd/tests/repo-identity-worktree.test.ts +61 -1
  194. package/src/resources/extensions/gsd/tests/routing-history.test.ts +11 -22
  195. package/src/resources/extensions/gsd/tests/rule-registry.test.ts +413 -0
  196. package/src/resources/extensions/gsd/tests/service-tier.test.ts +98 -0
  197. package/src/resources/extensions/gsd/tests/skill-lifecycle.test.ts +2 -2
  198. package/src/resources/extensions/gsd/tests/stalled-tool-recovery.test.ts +102 -0
  199. package/src/resources/extensions/gsd/tests/structured-data-formatter.test.ts +4 -3
  200. package/src/resources/extensions/gsd/tests/tool-naming.test.ts +117 -0
  201. package/src/resources/extensions/gsd/tests/triage-dispatch.test.ts +6 -1
  202. package/src/resources/extensions/gsd/tests/windows-path-normalization.test.ts +99 -0
  203. package/src/resources/extensions/gsd/tests/worktree-db-integration.test.ts +1 -0
  204. package/src/resources/extensions/gsd/tests/worktree-db.test.ts +4 -0
  205. package/src/resources/extensions/gsd/tests/worktree-health-dispatch.test.ts +178 -0
  206. package/src/resources/extensions/gsd/tests/worktree-manager.test.ts +195 -105
  207. package/src/resources/extensions/gsd/tests/worktree-resolver.test.ts +78 -3
  208. package/src/resources/extensions/gsd/tests/worktree-symlink-removal.test.ts +140 -0
  209. package/src/resources/extensions/gsd/tests/worktree-sync-milestones.test.ts +74 -0
  210. package/src/resources/extensions/gsd/types.ts +3 -0
  211. package/src/resources/extensions/gsd/workflow-templates.ts +12 -1
  212. package/src/resources/extensions/gsd/worktree-manager.ts +21 -6
  213. package/src/resources/extensions/gsd/worktree-resolver.ts +30 -9
  214. package/src/resources/extensions/subagent/index.ts +7 -3
  215. package/src/resources/extensions/voice/index.ts +4 -4
  216. package/dist/web/standalone/.next/static/chunks/4024.279c423e4661ece1.js +0 -9
  217. /package/dist/web/standalone/.next/static/{02cti5IXH7FycOqkbAkWL → ZrI3HOoXD7Fh84fAHZVxb}/_buildManifest.js +0 -0
  218. /package/dist/web/standalone/.next/static/{02cti5IXH7FycOqkbAkWL → ZrI3HOoXD7Fh84fAHZVxb}/_ssgManifest.js +0 -0
@@ -404,6 +404,8 @@ export interface HookStatusEntry {
404
404
 
405
405
  // ─── Database Types (Decisions & Requirements) ────────────────────────────
406
406
 
407
+ export type DecisionMadeBy = "human" | "agent" | "collaborative";
408
+
407
409
  export interface Decision {
408
410
  seq: number; // auto-increment primary key
409
411
  id: string; // e.g. "D001"
@@ -413,6 +415,7 @@ export interface Decision {
413
415
  choice: string; // the specific choice made
414
416
  rationale: string; // why this choice
415
417
  revisable: string; // whether/when revisable
418
+ made_by: DecisionMadeBy; // who made the decision: human, agent, or collaborative
416
419
  superseded_by: string | null; // ID of superseding decision, or null
417
420
  }
418
421
 
@@ -8,10 +8,21 @@
8
8
  import { readFileSync, existsSync } from "node:fs";
9
9
  import { join, dirname } from "node:path";
10
10
  import { fileURLToPath } from "node:url";
11
+ import { homedir } from "node:os";
11
12
 
12
- const __extensionDir = dirname(fileURLToPath(import.meta.url));
13
+ const __extensionDir = resolveGsdExtensionDir();
13
14
  const registryPath = join(__extensionDir, "workflow-templates", "registry.json");
14
15
 
16
+ /** Resolve the GSD extension dir with fallback to ~/.gsd/agent/extensions/gsd/. */
17
+ function resolveGsdExtensionDir(): string {
18
+ const moduleDir = dirname(fileURLToPath(import.meta.url));
19
+ if (existsSync(join(moduleDir, "workflow-templates"))) return moduleDir;
20
+ const gsdHome = process.env.GSD_HOME || join(homedir(), ".gsd");
21
+ const agentGsdDir = join(gsdHome, "agent", "extensions", "gsd");
22
+ if (existsSync(join(agentGsdDir, "workflow-templates"))) return agentGsdDir;
23
+ return moduleDir;
24
+ }
25
+
15
26
  // ─── Types ───────────────────────────────────────────────────────────────────
16
27
 
17
28
  export interface TemplateEntry {
@@ -286,11 +286,26 @@ export function removeWorktree(
286
286
  name: string,
287
287
  opts: { deleteBranch?: boolean; force?: boolean; branch?: string } = {},
288
288
  ): void {
289
- const wtPath = worktreePath(basePath, name);
290
- const resolvedWtPath = existsSync(wtPath) ? realpathSync(wtPath) : wtPath;
289
+ let wtPath = worktreePath(basePath, name);
291
290
  const branch = opts.branch ?? worktreeBranchName(name);
292
291
  const { deleteBranch = true, force = true } = opts;
293
292
 
293
+ // Resolve the ACTUAL worktree path from git's worktree list.
294
+ // The computed path may differ when .gsd/ is (or was) a symlink to an
295
+ // external state directory — git resolves symlinks at worktree creation
296
+ // time, so its registered path points to the resolved external location.
297
+ // If syncStateToProjectRoot later creates a real .gsd/ directory that
298
+ // shadows the symlink, the computed path diverges from git's record.
299
+ try {
300
+ const entries = nativeWorktreeList(basePath);
301
+ const entry = entries.find(e => e.branch === branch);
302
+ if (entry?.path) {
303
+ wtPath = entry.path;
304
+ }
305
+ } catch { /* fall back to computed path */ }
306
+
307
+ const resolvedWtPath = existsSync(wtPath) ? realpathSync(wtPath) : wtPath;
308
+
294
309
  // If we're inside the worktree, move out first — git can't remove an in-use directory
295
310
  const cwd = process.cwd();
296
311
  const resolvedCwd = existsSync(cwd) ? realpathSync(cwd) : cwd;
@@ -306,12 +321,12 @@ export function removeWorktree(
306
321
  return;
307
322
  }
308
323
 
309
- // Remove worktree (force if requested, to handle dirty worktrees)
310
- try { nativeWorktreeRemove(basePath, wtPath, force); } catch { /* may fail */ }
324
+ // Remove worktree using the resolved path (force if requested, to handle dirty worktrees)
325
+ try { nativeWorktreeRemove(basePath, resolvedWtPath, force); } catch { /* may fail */ }
311
326
 
312
327
  // If the directory is still there (e.g. locked), try harder with force
313
- if (existsSync(wtPath)) {
314
- try { nativeWorktreeRemove(basePath, wtPath, true); } catch { /* may fail */ }
328
+ if (existsSync(resolvedWtPath)) {
329
+ try { nativeWorktreeRemove(basePath, resolvedWtPath, true); } catch { /* may fail */ }
315
330
  }
316
331
 
317
332
  // Prune stale entries so git knows the worktree is gone
@@ -28,7 +28,7 @@ export interface WorktreeResolverDeps {
28
28
  basePath: string,
29
29
  milestoneId: string,
30
30
  roadmapContent: string,
31
- ) => { pushed: boolean };
31
+ ) => { pushed: boolean; codeFilesChanged: boolean };
32
32
  syncWorktreeStateBack: (
33
33
  mainBasePath: string,
34
34
  worktreePath: string,
@@ -371,10 +371,23 @@ export class WorktreeResolver {
371
371
  milestoneId,
372
372
  roadmapContent,
373
373
  );
374
- ctx.notify(
375
- `Milestone ${milestoneId} merged to main.${mergeResult.pushed ? " Pushed to remote." : ""}`,
376
- "info",
377
- );
374
+ if (mergeResult.codeFilesChanged) {
375
+ ctx.notify(
376
+ `Milestone ${milestoneId} merged to main.${mergeResult.pushed ? " Pushed to remote." : ""}`,
377
+ "info",
378
+ );
379
+ } else {
380
+ // (#1906) Milestone produced only .gsd/ metadata — no actual code was
381
+ // merged. This typically means the LLM wrote planning artifacts
382
+ // (summaries, roadmaps) but never implemented the code. Surface this
383
+ // clearly so the user knows the milestone is not truly complete.
384
+ ctx.notify(
385
+ `WARNING: Milestone ${milestoneId} merged to main but contained NO code changes — only .gsd/ metadata files. ` +
386
+ `The milestone summary may describe planned work that was never implemented. ` +
387
+ `Review the milestone output and re-run if code is missing.`,
388
+ "warning",
389
+ );
390
+ }
378
391
  } else {
379
392
  // No roadmap at either location — teardown but PRESERVE the branch so
380
393
  // commits are not orphaned. The user can merge manually later (#1573).
@@ -478,10 +491,18 @@ export class WorktreeResolver {
478
491
  // Rebuild GitService after merge (branch HEAD changed)
479
492
  this.rebuildGitService();
480
493
 
481
- ctx.notify(
482
- `Milestone ${milestoneId} merged (branch mode).${mergeResult.pushed ? " Pushed to remote." : ""}`,
483
- "info",
484
- );
494
+ if (mergeResult.codeFilesChanged) {
495
+ ctx.notify(
496
+ `Milestone ${milestoneId} merged (branch mode).${mergeResult.pushed ? " Pushed to remote." : ""}`,
497
+ "info",
498
+ );
499
+ } else {
500
+ ctx.notify(
501
+ `WARNING: Milestone ${milestoneId} merged (branch mode) but contained NO code changes — only .gsd/ metadata. ` +
502
+ `Review the milestone output and re-run if code is missing.`,
503
+ "warning",
504
+ );
505
+ }
485
506
  debugLog("WorktreeResolver", {
486
507
  action: "mergeAndExit",
487
508
  milestoneId,
@@ -516,12 +516,16 @@ async function runSingleAgentInCmuxSplit(
516
516
  const bundledPaths = (process.env.GSD_BUNDLED_EXTENSION_PATHS ?? "").split(path.delimiter).map((s) => s.trim()).filter(Boolean);
517
517
  const extensionArgs = bundledPaths.flatMap((p) => ["--extension", p]);
518
518
  const processArgs = [process.env.GSD_BIN_PATH!, ...extensionArgs, ...buildSubagentProcessArgs(agent, task, tmpPromptPath)];
519
+ // Normalize all paths to forward slashes before embedding in bash strings.
520
+ // On Windows, backslashes are interpreted as escape characters by bash,
521
+ // mangling paths like C:\Users\user into C:Useruser (#1436).
522
+ const bashPath = (p: string) => shellEscape(p.replaceAll("\\", "/"));
519
523
  const innerScript = [
520
- `cd ${shellEscape(cwd ?? defaultCwd)}`,
524
+ `cd ${bashPath(cwd ?? defaultCwd)}`,
521
525
  "set -o pipefail",
522
- `${shellEscape(process.execPath)} ${processArgs.map(shellEscape).join(" ")} 2> >(tee ${shellEscape(stderrPath)} >&2) | tee ${shellEscape(stdoutPath)}`,
526
+ `${bashPath(process.execPath)} ${processArgs.map(a => bashPath(a)).join(" ")} 2> >(tee ${bashPath(stderrPath)} >&2) | tee ${bashPath(stdoutPath)}`,
523
527
  "status=${PIPESTATUS[0]}",
524
- `printf '%s' "$status" > ${shellEscape(exitPath)}`,
528
+ `printf '%s' "$status" > ${bashPath(exitPath)}`,
525
529
  ].join("; ");
526
530
 
527
531
  const sent = await cmuxClient.sendSurface(cmuxSurfaceId, `bash -lc ${shellEscape(innerScript)}`);
@@ -2,7 +2,7 @@ import type { ExtensionAPI, ExtensionContext } from "@gsd/pi-coding-agent";
2
2
  import { shortcutDesc } from "../shared/mod.js";
3
3
  import type { AssistantMessage } from "@gsd/pi-ai";
4
4
  import { isKeyRelease, Key, matchesKey, truncateToWidth, visibleWidth } from "@gsd/pi-tui";
5
- import { spawn, execSync, type ChildProcess } from "node:child_process";
5
+ import { spawn, execFileSync, type ChildProcess } from "node:child_process";
6
6
  import * as fs from "node:fs";
7
7
  import * as os from "node:os";
8
8
  import * as path from "node:path";
@@ -32,7 +32,7 @@ function linuxPython(): string {
32
32
  function ensureBinary(): boolean {
33
33
  if (fs.existsSync(RECOGNIZER_BIN)) return true;
34
34
  try {
35
- execSync(`swiftc "${SWIFT_SRC}" -o "${RECOGNIZER_BIN}" -framework Speech -framework AVFoundation`, {
35
+ execFileSync("swiftc", [SWIFT_SRC, "-o", RECOGNIZER_BIN, "-framework", "Speech", "-framework", "AVFoundation"], {
36
36
  timeout: 60000,
37
37
  });
38
38
  return true;
@@ -54,7 +54,7 @@ function ensureLinuxReady(ctx: ExtensionContext): boolean {
54
54
 
55
55
  // Check python3 exists
56
56
  try {
57
- execSync("which python3", { stdio: "pipe" });
57
+ execFileSync("which", ["python3"], { stdio: "pipe" });
58
58
  } catch {
59
59
  ctx.ui.notify("Voice: python3 not found — install with: sudo apt install python3", "error");
60
60
  return false;
@@ -63,7 +63,7 @@ function ensureLinuxReady(ctx: ExtensionContext): boolean {
63
63
  // Check that sounddevice is importable
64
64
  const py = linuxPython();
65
65
  try {
66
- execSync(`${py} -c "import sounddevice"`, {
66
+ execFileSync(py, ["-c", "import sounddevice"], {
67
67
  stdio: "pipe",
68
68
  timeout: 10000,
69
69
  });