gsd-pi 2.63.0 → 2.64.0-dev.1a85e85

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 (583) hide show
  1. package/README.md +46 -134
  2. package/dist/cli.js +48 -6
  3. package/dist/headless-query.js +11 -1
  4. package/dist/help-text.js +4 -1
  5. package/dist/onboarding.js +15 -8
  6. package/dist/resource-loader.js +18 -3
  7. package/dist/resources/extensions/cmux/index.js +21 -12
  8. package/dist/resources/extensions/gsd/auto/detect-stuck.js +27 -0
  9. package/dist/resources/extensions/gsd/auto/finalize-timeout.js +40 -0
  10. package/dist/resources/extensions/gsd/auto/loop.js +4 -0
  11. package/dist/resources/extensions/gsd/auto/phases.js +157 -22
  12. package/dist/resources/extensions/gsd/auto/session.js +12 -0
  13. package/dist/resources/extensions/gsd/auto-dashboard.js +14 -8
  14. package/dist/resources/extensions/gsd/auto-model-selection.js +32 -0
  15. package/dist/resources/extensions/gsd/auto-post-unit.js +222 -11
  16. package/dist/resources/extensions/gsd/auto-prompts.js +25 -0
  17. package/dist/resources/extensions/gsd/auto-recovery.js +15 -7
  18. package/dist/resources/extensions/gsd/auto-start.js +10 -21
  19. package/dist/resources/extensions/gsd/auto-timers.js +2 -1
  20. package/dist/resources/extensions/gsd/auto-tool-tracking.js +17 -0
  21. package/dist/resources/extensions/gsd/auto-verification.js +138 -1
  22. package/dist/resources/extensions/gsd/auto-worktree.js +13 -7
  23. package/dist/resources/extensions/gsd/auto.js +24 -2
  24. package/dist/resources/extensions/gsd/bootstrap/db-tools.js +147 -75
  25. package/dist/resources/extensions/gsd/bootstrap/dynamic-tools.js +13 -0
  26. package/dist/resources/extensions/gsd/bootstrap/notify-interceptor.js +28 -0
  27. package/dist/resources/extensions/gsd/bootstrap/query-tools.js +85 -0
  28. package/dist/resources/extensions/gsd/bootstrap/register-extension.js +3 -0
  29. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +40 -1
  30. package/dist/resources/extensions/gsd/bootstrap/register-shortcuts.js +15 -0
  31. package/dist/resources/extensions/gsd/bootstrap/sanitize-complete-milestone.js +54 -0
  32. package/dist/resources/extensions/gsd/bootstrap/system-context.js +50 -2
  33. package/dist/resources/extensions/gsd/commands/catalog.js +7 -1
  34. package/dist/resources/extensions/gsd/commands/handlers/core.js +1 -0
  35. package/dist/resources/extensions/gsd/commands/handlers/notifications-handler.js +103 -0
  36. package/dist/resources/extensions/gsd/commands/handlers/ops.js +5 -0
  37. package/dist/resources/extensions/gsd/commands-handlers.js +9 -4
  38. package/dist/resources/extensions/gsd/constants.js +42 -0
  39. package/dist/resources/extensions/gsd/db-writer.js +72 -4
  40. package/dist/resources/extensions/gsd/forensics.js +20 -4
  41. package/dist/resources/extensions/gsd/gsd-db.js +64 -17
  42. package/dist/resources/extensions/gsd/guided-flow.js +19 -0
  43. package/dist/resources/extensions/gsd/metrics.js +27 -1
  44. package/dist/resources/extensions/gsd/native-git-bridge.js +5 -3
  45. package/dist/resources/extensions/gsd/notification-overlay.js +224 -0
  46. package/dist/resources/extensions/gsd/notification-store.js +268 -0
  47. package/dist/resources/extensions/gsd/notification-widget.js +56 -0
  48. package/dist/resources/extensions/gsd/post-execution-checks.js +407 -0
  49. package/dist/resources/extensions/gsd/pre-execution-checks.js +464 -0
  50. package/dist/resources/extensions/gsd/preferences-types.js +6 -0
  51. package/dist/resources/extensions/gsd/preferences-validation.js +33 -0
  52. package/dist/resources/extensions/gsd/preferences.js +11 -2
  53. package/dist/resources/extensions/gsd/prompt-loader.js +7 -0
  54. package/dist/resources/extensions/gsd/prompts/complete-milestone.md +2 -0
  55. package/dist/resources/extensions/gsd/prompts/complete-slice.md +2 -0
  56. package/dist/resources/extensions/gsd/prompts/doctor-heal.md +1 -0
  57. package/dist/resources/extensions/gsd/prompts/forensics.md +2 -0
  58. package/dist/resources/extensions/gsd/prompts/reassess-roadmap.md +2 -0
  59. package/dist/resources/extensions/gsd/prompts/system.md +4 -7
  60. package/dist/resources/extensions/gsd/prompts/validate-milestone.md +2 -0
  61. package/dist/resources/extensions/gsd/roadmap-mutations.js +1 -1
  62. package/dist/resources/extensions/gsd/roadmap-slices.js +9 -5
  63. package/dist/resources/extensions/gsd/safety/content-validator.js +73 -0
  64. package/dist/resources/extensions/gsd/safety/destructive-guard.js +34 -0
  65. package/dist/resources/extensions/gsd/safety/evidence-collector.js +109 -0
  66. package/dist/resources/extensions/gsd/safety/evidence-cross-ref.js +83 -0
  67. package/dist/resources/extensions/gsd/safety/file-change-validator.js +71 -0
  68. package/dist/resources/extensions/gsd/safety/git-checkpoint.js +91 -0
  69. package/dist/resources/extensions/gsd/safety/safety-harness.js +64 -0
  70. package/dist/resources/extensions/gsd/slice-parallel-conflict.js +67 -0
  71. package/dist/resources/extensions/gsd/slice-parallel-eligibility.js +51 -0
  72. package/dist/resources/extensions/gsd/slice-parallel-orchestrator.js +378 -0
  73. package/dist/resources/extensions/gsd/state.js +74 -14
  74. package/dist/resources/extensions/gsd/status-guards.js +11 -0
  75. package/dist/resources/extensions/gsd/tools/complete-milestone.js +17 -12
  76. package/dist/resources/extensions/gsd/tools/complete-slice.js +40 -26
  77. package/dist/resources/extensions/gsd/tools/complete-task.js +12 -12
  78. package/dist/resources/extensions/gsd/tools/plan-milestone.js +33 -25
  79. package/dist/resources/extensions/gsd/tools/plan-slice.js +5 -8
  80. package/dist/resources/extensions/gsd/verification-evidence.js +18 -0
  81. package/dist/resources/extensions/gsd/workflow-logger.js +8 -0
  82. package/dist/resources/extensions/gsd/workflow-projections.js +21 -5
  83. package/dist/resources/extensions/gsd/worktree-manager.js +82 -29
  84. package/dist/resources/extensions/gsd/worktree-resolver.js +4 -3
  85. package/dist/resources/extensions/mcp-client/auth.js +101 -0
  86. package/dist/resources/extensions/mcp-client/index.js +10 -1
  87. package/dist/resources/extensions/ollama/index.js +28 -22
  88. package/dist/resources/extensions/ollama/model-capabilities.js +37 -34
  89. package/dist/resources/extensions/ollama/ndjson-stream.js +54 -0
  90. package/dist/resources/extensions/ollama/ollama-chat-provider.js +380 -0
  91. package/dist/resources/extensions/ollama/ollama-client.js +23 -32
  92. package/dist/resources/extensions/ollama/ollama-discovery.js +2 -7
  93. package/dist/resources/extensions/ollama/ollama-tool.js +62 -0
  94. package/dist/resources/extensions/ollama/thinking-parser.js +104 -0
  95. package/dist/update-cmd.js +4 -2
  96. package/dist/web/standalone/.next/BUILD_ID +1 -1
  97. package/dist/web/standalone/.next/app-path-routes-manifest.json +19 -18
  98. package/dist/web/standalone/.next/build-manifest.json +3 -3
  99. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  100. package/dist/web/standalone/.next/required-server-files.json +4 -4
  101. package/dist/web/standalone/.next/routes-manifest.json +6 -0
  102. package/dist/web/standalone/.next/server/app/_global-error/page.js +3 -3
  103. package/dist/web/standalone/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
  104. package/dist/web/standalone/.next/server/app/_global-error.html +2 -2
  105. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  106. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  107. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  108. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  109. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  110. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  111. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  112. package/dist/web/standalone/.next/server/app/_not-found/page.js +2 -2
  113. package/dist/web/standalone/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
  114. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  115. package/dist/web/standalone/.next/server/app/_not-found.rsc +3 -3
  116. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +3 -3
  117. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  118. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +3 -3
  119. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  120. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  121. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  122. package/dist/web/standalone/.next/server/app/api/boot/route.js +1 -1
  123. package/dist/web/standalone/.next/server/app/api/boot/route.js.nft.json +1 -1
  124. package/dist/web/standalone/.next/server/app/api/boot/route_client-reference-manifest.js +1 -1
  125. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route.js +1 -1
  126. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route.js.nft.json +1 -1
  127. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route_client-reference-manifest.js +1 -1
  128. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route.js +1 -1
  129. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route.js.nft.json +1 -1
  130. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route_client-reference-manifest.js +1 -1
  131. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route.js +2 -2
  132. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route.js.nft.json +1 -1
  133. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route_client-reference-manifest.js +1 -1
  134. package/dist/web/standalone/.next/server/app/api/browse-directories/route.js +1 -1
  135. package/dist/web/standalone/.next/server/app/api/browse-directories/route_client-reference-manifest.js +1 -1
  136. package/dist/web/standalone/.next/server/app/api/captures/route.js +1 -1
  137. package/dist/web/standalone/.next/server/app/api/captures/route.js.nft.json +1 -1
  138. package/dist/web/standalone/.next/server/app/api/captures/route_client-reference-manifest.js +1 -1
  139. package/dist/web/standalone/.next/server/app/api/cleanup/route.js +1 -1
  140. package/dist/web/standalone/.next/server/app/api/cleanup/route.js.nft.json +1 -1
  141. package/dist/web/standalone/.next/server/app/api/cleanup/route_client-reference-manifest.js +1 -1
  142. package/dist/web/standalone/.next/server/app/api/dev-mode/route.js +1 -1
  143. package/dist/web/standalone/.next/server/app/api/dev-mode/route_client-reference-manifest.js +1 -1
  144. package/dist/web/standalone/.next/server/app/api/doctor/route.js +1 -1
  145. package/dist/web/standalone/.next/server/app/api/doctor/route.js.nft.json +1 -1
  146. package/dist/web/standalone/.next/server/app/api/doctor/route_client-reference-manifest.js +1 -1
  147. package/dist/web/standalone/.next/server/app/api/experimental/route.js +2 -2
  148. package/dist/web/standalone/.next/server/app/api/experimental/route_client-reference-manifest.js +1 -1
  149. package/dist/web/standalone/.next/server/app/api/export-data/route.js +1 -1
  150. package/dist/web/standalone/.next/server/app/api/export-data/route.js.nft.json +1 -1
  151. package/dist/web/standalone/.next/server/app/api/export-data/route_client-reference-manifest.js +1 -1
  152. package/dist/web/standalone/.next/server/app/api/files/route.js +1 -1
  153. package/dist/web/standalone/.next/server/app/api/files/route.js.nft.json +1 -1
  154. package/dist/web/standalone/.next/server/app/api/files/route_client-reference-manifest.js +1 -1
  155. package/dist/web/standalone/.next/server/app/api/forensics/route.js +1 -1
  156. package/dist/web/standalone/.next/server/app/api/forensics/route.js.nft.json +1 -1
  157. package/dist/web/standalone/.next/server/app/api/forensics/route_client-reference-manifest.js +1 -1
  158. package/dist/web/standalone/.next/server/app/api/git/route.js +1 -1
  159. package/dist/web/standalone/.next/server/app/api/git/route.js.nft.json +1 -1
  160. package/dist/web/standalone/.next/server/app/api/git/route_client-reference-manifest.js +1 -1
  161. package/dist/web/standalone/.next/server/app/api/history/route.js +1 -1
  162. package/dist/web/standalone/.next/server/app/api/history/route.js.nft.json +1 -1
  163. package/dist/web/standalone/.next/server/app/api/history/route_client-reference-manifest.js +1 -1
  164. package/dist/web/standalone/.next/server/app/api/hooks/route.js +1 -1
  165. package/dist/web/standalone/.next/server/app/api/hooks/route.js.nft.json +1 -1
  166. package/dist/web/standalone/.next/server/app/api/hooks/route_client-reference-manifest.js +1 -1
  167. package/dist/web/standalone/.next/server/app/api/inspect/route.js +1 -1
  168. package/dist/web/standalone/.next/server/app/api/inspect/route.js.nft.json +1 -1
  169. package/dist/web/standalone/.next/server/app/api/inspect/route_client-reference-manifest.js +1 -1
  170. package/dist/web/standalone/.next/server/app/api/knowledge/route.js +1 -1
  171. package/dist/web/standalone/.next/server/app/api/knowledge/route.js.nft.json +1 -1
  172. package/dist/web/standalone/.next/server/app/api/knowledge/route_client-reference-manifest.js +1 -1
  173. package/dist/web/standalone/.next/server/app/api/live-state/route.js +1 -1
  174. package/dist/web/standalone/.next/server/app/api/live-state/route.js.nft.json +1 -1
  175. package/dist/web/standalone/.next/server/app/api/live-state/route_client-reference-manifest.js +1 -1
  176. package/dist/web/standalone/.next/server/app/api/notifications/route.js +3 -0
  177. package/dist/web/standalone/.next/server/app/api/notifications/route.js.nft.json +1 -0
  178. package/dist/web/standalone/.next/server/app/api/notifications/route_client-reference-manifest.js +1 -0
  179. package/dist/web/standalone/.next/server/app/api/onboarding/route.js +1 -1
  180. package/dist/web/standalone/.next/server/app/api/onboarding/route.js.nft.json +1 -1
  181. package/dist/web/standalone/.next/server/app/api/onboarding/route_client-reference-manifest.js +1 -1
  182. package/dist/web/standalone/.next/server/app/api/preferences/route.js +1 -1
  183. package/dist/web/standalone/.next/server/app/api/preferences/route_client-reference-manifest.js +1 -1
  184. package/dist/web/standalone/.next/server/app/api/projects/route.js +1 -1
  185. package/dist/web/standalone/.next/server/app/api/projects/route.js.nft.json +1 -1
  186. package/dist/web/standalone/.next/server/app/api/projects/route_client-reference-manifest.js +1 -1
  187. package/dist/web/standalone/.next/server/app/api/recovery/route.js +1 -1
  188. package/dist/web/standalone/.next/server/app/api/recovery/route.js.nft.json +1 -1
  189. package/dist/web/standalone/.next/server/app/api/recovery/route_client-reference-manifest.js +1 -1
  190. package/dist/web/standalone/.next/server/app/api/remote-questions/route.js +2 -2
  191. package/dist/web/standalone/.next/server/app/api/remote-questions/route_client-reference-manifest.js +1 -1
  192. package/dist/web/standalone/.next/server/app/api/session/browser/route.js +1 -1
  193. package/dist/web/standalone/.next/server/app/api/session/browser/route.js.nft.json +1 -1
  194. package/dist/web/standalone/.next/server/app/api/session/browser/route_client-reference-manifest.js +1 -1
  195. package/dist/web/standalone/.next/server/app/api/session/command/route.js +1 -1
  196. package/dist/web/standalone/.next/server/app/api/session/command/route.js.nft.json +1 -1
  197. package/dist/web/standalone/.next/server/app/api/session/command/route_client-reference-manifest.js +1 -1
  198. package/dist/web/standalone/.next/server/app/api/session/events/route.js +2 -2
  199. package/dist/web/standalone/.next/server/app/api/session/events/route.js.nft.json +1 -1
  200. package/dist/web/standalone/.next/server/app/api/session/events/route_client-reference-manifest.js +1 -1
  201. package/dist/web/standalone/.next/server/app/api/session/manage/route.js +1 -1
  202. package/dist/web/standalone/.next/server/app/api/session/manage/route.js.nft.json +1 -1
  203. package/dist/web/standalone/.next/server/app/api/session/manage/route_client-reference-manifest.js +1 -1
  204. package/dist/web/standalone/.next/server/app/api/settings-data/route.js +1 -1
  205. package/dist/web/standalone/.next/server/app/api/settings-data/route.js.nft.json +1 -1
  206. package/dist/web/standalone/.next/server/app/api/settings-data/route_client-reference-manifest.js +1 -1
  207. package/dist/web/standalone/.next/server/app/api/shutdown/route.js +1 -1
  208. package/dist/web/standalone/.next/server/app/api/shutdown/route_client-reference-manifest.js +1 -1
  209. package/dist/web/standalone/.next/server/app/api/skill-health/route.js +1 -1
  210. package/dist/web/standalone/.next/server/app/api/skill-health/route.js.nft.json +1 -1
  211. package/dist/web/standalone/.next/server/app/api/skill-health/route_client-reference-manifest.js +1 -1
  212. package/dist/web/standalone/.next/server/app/api/steer/route.js +1 -1
  213. package/dist/web/standalone/.next/server/app/api/steer/route.js.nft.json +1 -1
  214. package/dist/web/standalone/.next/server/app/api/steer/route_client-reference-manifest.js +1 -1
  215. package/dist/web/standalone/.next/server/app/api/switch-root/route.js +1 -1
  216. package/dist/web/standalone/.next/server/app/api/switch-root/route.js.nft.json +1 -1
  217. package/dist/web/standalone/.next/server/app/api/switch-root/route_client-reference-manifest.js +1 -1
  218. package/dist/web/standalone/.next/server/app/api/terminal/input/route.js +2 -2
  219. package/dist/web/standalone/.next/server/app/api/terminal/input/route_client-reference-manifest.js +1 -1
  220. package/dist/web/standalone/.next/server/app/api/terminal/resize/route.js +2 -2
  221. package/dist/web/standalone/.next/server/app/api/terminal/resize/route_client-reference-manifest.js +1 -1
  222. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js +2 -2
  223. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js.nft.json +1 -1
  224. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route_client-reference-manifest.js +1 -1
  225. package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js +4 -4
  226. package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js.nft.json +1 -1
  227. package/dist/web/standalone/.next/server/app/api/terminal/stream/route_client-reference-manifest.js +1 -1
  228. package/dist/web/standalone/.next/server/app/api/terminal/upload/route.js +1 -1
  229. package/dist/web/standalone/.next/server/app/api/terminal/upload/route_client-reference-manifest.js +1 -1
  230. package/dist/web/standalone/.next/server/app/api/undo/route.js +1 -1
  231. package/dist/web/standalone/.next/server/app/api/undo/route.js.nft.json +1 -1
  232. package/dist/web/standalone/.next/server/app/api/undo/route_client-reference-manifest.js +1 -1
  233. package/dist/web/standalone/.next/server/app/api/update/route.js +1 -1
  234. package/dist/web/standalone/.next/server/app/api/update/route_client-reference-manifest.js +1 -1
  235. package/dist/web/standalone/.next/server/app/api/visualizer/route.js +1 -1
  236. package/dist/web/standalone/.next/server/app/api/visualizer/route.js.nft.json +1 -1
  237. package/dist/web/standalone/.next/server/app/api/visualizer/route_client-reference-manifest.js +1 -1
  238. package/dist/web/standalone/.next/server/app/index.html +1 -1
  239. package/dist/web/standalone/.next/server/app/index.rsc +4 -4
  240. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +2 -2
  241. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +4 -4
  242. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  243. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +3 -3
  244. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  245. package/dist/web/standalone/.next/server/app/page.js +2 -2
  246. package/dist/web/standalone/.next/server/app/page_client-reference-manifest.js +1 -1
  247. package/dist/web/standalone/.next/server/app-paths-manifest.json +19 -18
  248. package/dist/web/standalone/.next/server/chunks/6897.js +12 -0
  249. package/dist/web/standalone/.next/server/chunks/7471.js +3 -3
  250. package/dist/web/standalone/.next/server/functions-config-manifest.json +1 -0
  251. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  252. package/dist/web/standalone/.next/server/middleware.js +2 -2
  253. package/dist/web/standalone/.next/server/next-font-manifest.js +1 -1
  254. package/dist/web/standalone/.next/server/next-font-manifest.json +1 -1
  255. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  256. package/dist/web/standalone/.next/server/pages/500.html +2 -2
  257. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  258. package/dist/web/standalone/.next/static/chunks/app/_global-error/page-8805a20e15762c3c.js +1 -0
  259. package/dist/web/standalone/.next/static/chunks/app/_not-found/{page-2f24283c162b6ab3.js → page-f2a7482d42a5614b.js} +1 -1
  260. package/dist/web/standalone/.next/static/chunks/app/api/boot/route-8805a20e15762c3c.js +1 -0
  261. package/dist/web/standalone/.next/static/chunks/app/api/bridge-terminal/input/route-8805a20e15762c3c.js +1 -0
  262. package/dist/web/standalone/.next/static/chunks/app/api/bridge-terminal/resize/route-8805a20e15762c3c.js +1 -0
  263. package/dist/web/standalone/.next/static/chunks/app/api/bridge-terminal/stream/route-8805a20e15762c3c.js +1 -0
  264. package/dist/web/standalone/.next/static/chunks/app/api/browse-directories/route-8805a20e15762c3c.js +1 -0
  265. package/dist/web/standalone/.next/static/chunks/app/api/captures/route-8805a20e15762c3c.js +1 -0
  266. package/dist/web/standalone/.next/static/chunks/app/api/cleanup/route-8805a20e15762c3c.js +1 -0
  267. package/dist/web/standalone/.next/static/chunks/app/api/dev-mode/route-8805a20e15762c3c.js +1 -0
  268. package/dist/web/standalone/.next/static/chunks/app/api/doctor/route-8805a20e15762c3c.js +1 -0
  269. package/dist/web/standalone/.next/static/chunks/app/api/experimental/route-8805a20e15762c3c.js +1 -0
  270. package/dist/web/standalone/.next/static/chunks/app/api/export-data/route-8805a20e15762c3c.js +1 -0
  271. package/dist/web/standalone/.next/static/chunks/app/api/files/route-8805a20e15762c3c.js +1 -0
  272. package/dist/web/standalone/.next/static/chunks/app/api/forensics/route-8805a20e15762c3c.js +1 -0
  273. package/dist/web/standalone/.next/static/chunks/app/api/git/route-8805a20e15762c3c.js +1 -0
  274. package/dist/web/standalone/.next/static/chunks/app/api/history/route-8805a20e15762c3c.js +1 -0
  275. package/dist/web/standalone/.next/static/chunks/app/api/hooks/route-8805a20e15762c3c.js +1 -0
  276. package/dist/web/standalone/.next/static/chunks/app/api/inspect/route-8805a20e15762c3c.js +1 -0
  277. package/dist/web/standalone/.next/static/chunks/app/api/knowledge/route-8805a20e15762c3c.js +1 -0
  278. package/dist/web/standalone/.next/static/chunks/app/api/live-state/route-8805a20e15762c3c.js +1 -0
  279. package/dist/web/standalone/.next/static/chunks/app/api/notifications/route-8805a20e15762c3c.js +1 -0
  280. package/dist/web/standalone/.next/static/chunks/app/api/onboarding/route-8805a20e15762c3c.js +1 -0
  281. package/dist/web/standalone/.next/static/chunks/app/api/preferences/route-8805a20e15762c3c.js +1 -0
  282. package/dist/web/standalone/.next/static/chunks/app/api/projects/route-8805a20e15762c3c.js +1 -0
  283. package/dist/web/standalone/.next/static/chunks/app/api/recovery/route-8805a20e15762c3c.js +1 -0
  284. package/dist/web/standalone/.next/static/chunks/app/api/remote-questions/route-8805a20e15762c3c.js +1 -0
  285. package/dist/web/standalone/.next/static/chunks/app/api/session/browser/route-8805a20e15762c3c.js +1 -0
  286. package/dist/web/standalone/.next/static/chunks/app/api/session/command/route-8805a20e15762c3c.js +1 -0
  287. package/dist/web/standalone/.next/static/chunks/app/api/session/events/route-8805a20e15762c3c.js +1 -0
  288. package/dist/web/standalone/.next/static/chunks/app/api/session/manage/route-8805a20e15762c3c.js +1 -0
  289. package/dist/web/standalone/.next/static/chunks/app/api/settings-data/route-8805a20e15762c3c.js +1 -0
  290. package/dist/web/standalone/.next/static/chunks/app/api/shutdown/route-8805a20e15762c3c.js +1 -0
  291. package/dist/web/standalone/.next/static/chunks/app/api/skill-health/route-8805a20e15762c3c.js +1 -0
  292. package/dist/web/standalone/.next/static/chunks/app/api/steer/route-8805a20e15762c3c.js +1 -0
  293. package/dist/web/standalone/.next/static/chunks/app/api/switch-root/route-8805a20e15762c3c.js +1 -0
  294. package/dist/web/standalone/.next/static/chunks/app/api/terminal/input/route-8805a20e15762c3c.js +1 -0
  295. package/dist/web/standalone/.next/static/chunks/app/api/terminal/resize/route-8805a20e15762c3c.js +1 -0
  296. package/dist/web/standalone/.next/static/chunks/app/api/terminal/sessions/route-8805a20e15762c3c.js +1 -0
  297. package/dist/web/standalone/.next/static/chunks/app/api/terminal/stream/route-8805a20e15762c3c.js +1 -0
  298. package/dist/web/standalone/.next/static/chunks/app/api/terminal/upload/route-8805a20e15762c3c.js +1 -0
  299. package/dist/web/standalone/.next/static/chunks/app/api/undo/route-8805a20e15762c3c.js +1 -0
  300. package/dist/web/standalone/.next/static/chunks/app/api/update/route-8805a20e15762c3c.js +1 -0
  301. package/dist/web/standalone/.next/static/chunks/app/api/visualizer/route-8805a20e15762c3c.js +1 -0
  302. package/dist/web/standalone/.next/static/chunks/app/{layout-9ecfd95f343793f0.js → layout-a16c7a7ecdf0c2cf.js} +1 -1
  303. package/dist/web/standalone/.next/static/chunks/app/page-0c485498795110d6.js +1 -0
  304. package/dist/web/standalone/.next/static/chunks/main-app-fdab67f7802d7832.js +1 -0
  305. package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/app-error-8805a20e15762c3c.js +1 -0
  306. package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/forbidden-8805a20e15762c3c.js +1 -0
  307. package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/global-error-459824ffb8c323dd.js +1 -0
  308. package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/not-found-8805a20e15762c3c.js +1 -0
  309. package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/unauthorized-8805a20e15762c3c.js +1 -0
  310. package/dist/web/standalone/.next/static/ffabZXz8JdN3EzX9EKt-R/_buildManifest.js +1 -0
  311. package/dist/web/standalone/node_modules/node-pty/build/Makefile +2 -2
  312. package/dist/web/standalone/node_modules/node-pty/build/Release/pty.node +0 -0
  313. package/dist/web/standalone/node_modules/node-pty/build/pty.target.mk +14 -14
  314. package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api.target.mk +14 -14
  315. package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api_except.target.mk +14 -14
  316. package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api_maybe.target.mk +14 -14
  317. package/dist/web/standalone/server.js +1 -1
  318. package/dist/welcome-screen.js +1 -1
  319. package/package.json +1 -1
  320. package/packages/pi-agent-core/dist/agent-loop.d.ts +8 -0
  321. package/packages/pi-agent-core/dist/agent-loop.d.ts.map +1 -1
  322. package/packages/pi-agent-core/dist/agent-loop.js +70 -3
  323. package/packages/pi-agent-core/dist/agent-loop.js.map +1 -1
  324. package/packages/pi-agent-core/src/agent-loop.test.ts +317 -5
  325. package/packages/pi-agent-core/src/agent-loop.ts +90 -6
  326. package/packages/pi-ai/dist/types.d.ts +16 -1
  327. package/packages/pi-ai/dist/types.d.ts.map +1 -1
  328. package/packages/pi-ai/dist/types.js.map +1 -1
  329. package/packages/pi-ai/src/types.ts +18 -1
  330. package/packages/pi-coding-agent/dist/core/agent-session-tool-refresh.test.d.ts +2 -0
  331. package/packages/pi-coding-agent/dist/core/agent-session-tool-refresh.test.d.ts.map +1 -0
  332. package/packages/pi-coding-agent/dist/core/agent-session-tool-refresh.test.js +38 -0
  333. package/packages/pi-coding-agent/dist/core/agent-session-tool-refresh.test.js.map +1 -0
  334. package/packages/pi-coding-agent/dist/core/agent-session.d.ts.map +1 -1
  335. package/packages/pi-coding-agent/dist/core/agent-session.js +11 -0
  336. package/packages/pi-coding-agent/dist/core/agent-session.js.map +1 -1
  337. package/packages/pi-coding-agent/dist/core/auth-storage.d.ts +9 -0
  338. package/packages/pi-coding-agent/dist/core/auth-storage.d.ts.map +1 -1
  339. package/packages/pi-coding-agent/dist/core/auth-storage.js +50 -1
  340. package/packages/pi-coding-agent/dist/core/auth-storage.js.map +1 -1
  341. package/packages/pi-coding-agent/dist/core/auth-storage.test.js +41 -0
  342. package/packages/pi-coding-agent/dist/core/auth-storage.test.js.map +1 -1
  343. package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts +7 -0
  344. package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts.map +1 -1
  345. package/packages/pi-coding-agent/dist/core/extensions/loader.js +31 -4
  346. package/packages/pi-coding-agent/dist/core/extensions/loader.js.map +1 -1
  347. package/packages/pi-coding-agent/dist/core/extensions/loader.test.js +28 -1
  348. package/packages/pi-coding-agent/dist/core/extensions/loader.test.js.map +1 -1
  349. package/packages/pi-coding-agent/dist/core/extensions/provider-registration.test.d.ts +2 -0
  350. package/packages/pi-coding-agent/dist/core/extensions/provider-registration.test.d.ts.map +1 -0
  351. package/packages/pi-coding-agent/dist/core/extensions/provider-registration.test.js +46 -0
  352. package/packages/pi-coding-agent/dist/core/extensions/provider-registration.test.js.map +1 -0
  353. package/packages/pi-coding-agent/dist/core/extensions/types.d.ts +2 -0
  354. package/packages/pi-coding-agent/dist/core/extensions/types.d.ts.map +1 -1
  355. package/packages/pi-coding-agent/dist/core/extensions/types.js.map +1 -1
  356. package/packages/pi-coding-agent/dist/core/model-registry.d.ts +1 -0
  357. package/packages/pi-coding-agent/dist/core/model-registry.d.ts.map +1 -1
  358. package/packages/pi-coding-agent/dist/core/model-registry.js +12 -0
  359. package/packages/pi-coding-agent/dist/core/model-registry.js.map +1 -1
  360. package/packages/pi-coding-agent/dist/core/model-resolver.js +3 -3
  361. package/packages/pi-coding-agent/dist/core/model-resolver.js.map +1 -1
  362. package/packages/pi-coding-agent/dist/core/resource-loader-cache-reset.test.d.ts +2 -0
  363. package/packages/pi-coding-agent/dist/core/resource-loader-cache-reset.test.d.ts.map +1 -0
  364. package/packages/pi-coding-agent/dist/core/resource-loader-cache-reset.test.js +24 -0
  365. package/packages/pi-coding-agent/dist/core/resource-loader-cache-reset.test.js.map +1 -0
  366. package/packages/pi-coding-agent/dist/core/resource-loader.d.ts +23 -1
  367. package/packages/pi-coding-agent/dist/core/resource-loader.d.ts.map +1 -1
  368. package/packages/pi-coding-agent/dist/core/resource-loader.js +84 -57
  369. package/packages/pi-coding-agent/dist/core/resource-loader.js.map +1 -1
  370. package/packages/pi-coding-agent/dist/core/sdk.d.ts.map +1 -1
  371. package/packages/pi-coding-agent/dist/core/sdk.js +9 -0
  372. package/packages/pi-coding-agent/dist/core/sdk.js.map +1 -1
  373. package/packages/pi-coding-agent/package.json +1 -1
  374. package/packages/pi-coding-agent/src/core/agent-session-tool-refresh.test.ts +64 -0
  375. package/packages/pi-coding-agent/src/core/agent-session.ts +10 -0
  376. package/packages/pi-coding-agent/src/core/auth-storage.test.ts +53 -0
  377. package/packages/pi-coding-agent/src/core/auth-storage.ts +66 -1
  378. package/packages/pi-coding-agent/src/core/extensions/loader.test.ts +39 -1
  379. package/packages/pi-coding-agent/src/core/extensions/loader.ts +34 -4
  380. package/packages/pi-coding-agent/src/core/extensions/provider-registration.test.ts +81 -0
  381. package/packages/pi-coding-agent/src/core/extensions/types.ts +2 -0
  382. package/packages/pi-coding-agent/src/core/model-registry.ts +14 -0
  383. package/packages/pi-coding-agent/src/core/model-resolver.ts +3 -3
  384. package/packages/pi-coding-agent/src/core/resource-loader-cache-reset.test.ts +42 -0
  385. package/packages/pi-coding-agent/src/core/resource-loader.ts +94 -57
  386. package/packages/pi-coding-agent/src/core/sdk.ts +10 -0
  387. package/pkg/package.json +1 -1
  388. package/src/resources/extensions/cmux/index.ts +18 -12
  389. package/src/resources/extensions/gsd/auto/detect-stuck.ts +27 -0
  390. package/src/resources/extensions/gsd/auto/finalize-timeout.ts +46 -0
  391. package/src/resources/extensions/gsd/auto/loop.ts +5 -0
  392. package/src/resources/extensions/gsd/auto/phases.ts +194 -33
  393. package/src/resources/extensions/gsd/auto/session.ts +14 -0
  394. package/src/resources/extensions/gsd/auto-dashboard.ts +16 -7
  395. package/src/resources/extensions/gsd/auto-model-selection.ts +36 -0
  396. package/src/resources/extensions/gsd/auto-post-unit.ts +263 -12
  397. package/src/resources/extensions/gsd/auto-prompts.ts +21 -0
  398. package/src/resources/extensions/gsd/auto-recovery.ts +9 -8
  399. package/src/resources/extensions/gsd/auto-start.ts +11 -20
  400. package/src/resources/extensions/gsd/auto-timers.ts +2 -1
  401. package/src/resources/extensions/gsd/auto-tool-tracking.ts +19 -0
  402. package/src/resources/extensions/gsd/auto-verification.ts +190 -2
  403. package/src/resources/extensions/gsd/auto-worktree.ts +14 -6
  404. package/src/resources/extensions/gsd/auto.ts +26 -1
  405. package/src/resources/extensions/gsd/bootstrap/db-tools.ts +160 -88
  406. package/src/resources/extensions/gsd/bootstrap/dynamic-tools.ts +15 -0
  407. package/src/resources/extensions/gsd/bootstrap/notify-interceptor.ts +34 -0
  408. package/src/resources/extensions/gsd/bootstrap/query-tools.ts +98 -0
  409. package/src/resources/extensions/gsd/bootstrap/register-extension.ts +4 -0
  410. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +44 -1
  411. package/src/resources/extensions/gsd/bootstrap/register-shortcuts.ts +19 -0
  412. package/src/resources/extensions/gsd/bootstrap/sanitize-complete-milestone.ts +57 -0
  413. package/src/resources/extensions/gsd/bootstrap/system-context.ts +59 -2
  414. package/src/resources/extensions/gsd/commands/catalog.ts +7 -1
  415. package/src/resources/extensions/gsd/commands/handlers/core.ts +1 -0
  416. package/src/resources/extensions/gsd/commands/handlers/notifications-handler.ts +139 -0
  417. package/src/resources/extensions/gsd/commands/handlers/ops.ts +5 -0
  418. package/src/resources/extensions/gsd/commands-handlers.ts +10 -4
  419. package/src/resources/extensions/gsd/constants.ts +44 -0
  420. package/src/resources/extensions/gsd/db-writer.ts +78 -4
  421. package/src/resources/extensions/gsd/forensics.ts +21 -5
  422. package/src/resources/extensions/gsd/gsd-db.ts +64 -17
  423. package/src/resources/extensions/gsd/guided-flow.ts +22 -0
  424. package/src/resources/extensions/gsd/metrics.ts +28 -1
  425. package/src/resources/extensions/gsd/native-git-bridge.ts +5 -3
  426. package/src/resources/extensions/gsd/notification-overlay.ts +267 -0
  427. package/src/resources/extensions/gsd/notification-store.ts +288 -0
  428. package/src/resources/extensions/gsd/notification-widget.ts +68 -0
  429. package/src/resources/extensions/gsd/post-execution-checks.ts +539 -0
  430. package/src/resources/extensions/gsd/pre-execution-checks.ts +573 -0
  431. package/src/resources/extensions/gsd/preferences-types.ts +44 -0
  432. package/src/resources/extensions/gsd/preferences-validation.ts +33 -0
  433. package/src/resources/extensions/gsd/preferences.ts +13 -2
  434. package/src/resources/extensions/gsd/prompt-loader.ts +8 -0
  435. package/src/resources/extensions/gsd/prompts/complete-milestone.md +2 -0
  436. package/src/resources/extensions/gsd/prompts/complete-slice.md +2 -0
  437. package/src/resources/extensions/gsd/prompts/doctor-heal.md +1 -0
  438. package/src/resources/extensions/gsd/prompts/forensics.md +2 -0
  439. package/src/resources/extensions/gsd/prompts/reassess-roadmap.md +2 -0
  440. package/src/resources/extensions/gsd/prompts/system.md +4 -7
  441. package/src/resources/extensions/gsd/prompts/validate-milestone.md +2 -0
  442. package/src/resources/extensions/gsd/roadmap-mutations.ts +1 -1
  443. package/src/resources/extensions/gsd/roadmap-slices.ts +10 -5
  444. package/src/resources/extensions/gsd/safety/content-validator.ts +98 -0
  445. package/src/resources/extensions/gsd/safety/destructive-guard.ts +49 -0
  446. package/src/resources/extensions/gsd/safety/evidence-collector.ts +151 -0
  447. package/src/resources/extensions/gsd/safety/evidence-cross-ref.ts +120 -0
  448. package/src/resources/extensions/gsd/safety/file-change-validator.ts +108 -0
  449. package/src/resources/extensions/gsd/safety/git-checkpoint.ts +106 -0
  450. package/src/resources/extensions/gsd/safety/safety-harness.ts +105 -0
  451. package/src/resources/extensions/gsd/slice-parallel-conflict.ts +86 -0
  452. package/src/resources/extensions/gsd/slice-parallel-eligibility.ts +73 -0
  453. package/src/resources/extensions/gsd/slice-parallel-orchestrator.ts +477 -0
  454. package/src/resources/extensions/gsd/state.ts +67 -12
  455. package/src/resources/extensions/gsd/status-guards.ts +13 -0
  456. package/src/resources/extensions/gsd/tests/artifact-corruption-2630.test.ts +288 -0
  457. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +34 -13
  458. package/src/resources/extensions/gsd/tests/auto-start-time-persistence.test.ts +50 -0
  459. package/src/resources/extensions/gsd/tests/cmux.test.ts +58 -0
  460. package/src/resources/extensions/gsd/tests/cold-resume-db-reopen.test.ts +51 -0
  461. package/src/resources/extensions/gsd/tests/complete-milestone.test.ts +140 -0
  462. package/src/resources/extensions/gsd/tests/complete-slice-string-coercion.test.ts +211 -0
  463. package/src/resources/extensions/gsd/tests/complete-task.test.ts +39 -0
  464. package/src/resources/extensions/gsd/tests/dashboard-model-label-ordering.test.ts +107 -0
  465. package/src/resources/extensions/gsd/tests/db-access-guardrails.test.ts +109 -0
  466. package/src/resources/extensions/gsd/tests/db-path-worktree-symlink.test.ts +13 -9
  467. package/src/resources/extensions/gsd/tests/db-writer.test.ts +134 -0
  468. package/src/resources/extensions/gsd/tests/deferred-slice-dispatch.test.ts +203 -0
  469. package/src/resources/extensions/gsd/tests/discuss-tool-scope-leak.test.ts +76 -0
  470. package/src/resources/extensions/gsd/tests/discuss-tool-scoping.test.ts +130 -0
  471. package/src/resources/extensions/gsd/tests/doctor-fix-flag.test.ts +92 -0
  472. package/src/resources/extensions/gsd/tests/enhanced-verification-integration.test.ts +526 -0
  473. package/src/resources/extensions/gsd/tests/finalize-timeout-guard.test.ts +116 -0
  474. package/src/resources/extensions/gsd/tests/flat-rate-routing-guard.test.ts +50 -0
  475. package/src/resources/extensions/gsd/tests/forensics-stuck-loops.test.ts +103 -0
  476. package/src/resources/extensions/gsd/tests/git-checkpoint.test.ts +94 -0
  477. package/src/resources/extensions/gsd/tests/insert-slice-no-wipe.test.ts +88 -0
  478. package/src/resources/extensions/gsd/tests/integration/git-service.test.ts +27 -7
  479. package/src/resources/extensions/gsd/tests/integration/idle-recovery.test.ts +34 -0
  480. package/src/resources/extensions/gsd/tests/metrics.test.ts +116 -1
  481. package/src/resources/extensions/gsd/tests/milestone-status-tool.test.ts +201 -0
  482. package/src/resources/extensions/gsd/tests/notification-store.test.ts +249 -0
  483. package/src/resources/extensions/gsd/tests/plan-milestone-title.test.ts +2 -1
  484. package/src/resources/extensions/gsd/tests/plan-milestone.test.ts +82 -18
  485. package/src/resources/extensions/gsd/tests/post-exec-retry-bypass.test.ts +312 -0
  486. package/src/resources/extensions/gsd/tests/post-execution-checks.test.ts +813 -0
  487. package/src/resources/extensions/gsd/tests/pre-execution-checks.test.ts +999 -0
  488. package/src/resources/extensions/gsd/tests/pre-execution-fail-closed.test.ts +266 -0
  489. package/src/resources/extensions/gsd/tests/pre-execution-pause-wiring.test.ts +457 -0
  490. package/src/resources/extensions/gsd/tests/preferences.test.ts +10 -0
  491. package/src/resources/extensions/gsd/tests/prompt-contracts.test.ts +25 -0
  492. package/src/resources/extensions/gsd/tests/roadmap-slices.test.ts +69 -0
  493. package/src/resources/extensions/gsd/tests/shared-wal.test.ts +30 -0
  494. package/src/resources/extensions/gsd/tests/slice-context-injection.test.ts +50 -0
  495. package/src/resources/extensions/gsd/tests/slice-parallel-conflict.test.ts +92 -0
  496. package/src/resources/extensions/gsd/tests/slice-parallel-eligibility.test.ts +95 -0
  497. package/src/resources/extensions/gsd/tests/slice-parallel-orchestrator.test.ts +83 -0
  498. package/src/resources/extensions/gsd/tests/stuck-detection-coverage.test.ts +42 -0
  499. package/src/resources/extensions/gsd/tests/tool-invocation-error-loop-break.test.ts +103 -0
  500. package/src/resources/extensions/gsd/tests/tool-param-optionality.test.ts +349 -0
  501. package/src/resources/extensions/gsd/tests/unstructured-continue-context-injection.test.ts +163 -0
  502. package/src/resources/extensions/gsd/tests/worktree-health-dispatch.test.ts +35 -2
  503. package/src/resources/extensions/gsd/tests/worktree-health-monorepo.test.ts +73 -0
  504. package/src/resources/extensions/gsd/tests/worktree-resolver.test.ts +34 -0
  505. package/src/resources/extensions/gsd/tests/worktree-submodule-safety.test.ts +1 -1
  506. package/src/resources/extensions/gsd/tests/worktree-teardown-safety.test.ts +148 -0
  507. package/src/resources/extensions/gsd/tools/complete-milestone.ts +34 -20
  508. package/src/resources/extensions/gsd/tools/complete-slice.ts +41 -26
  509. package/src/resources/extensions/gsd/tools/complete-task.ts +12 -12
  510. package/src/resources/extensions/gsd/tools/plan-milestone.ts +55 -30
  511. package/src/resources/extensions/gsd/tools/plan-slice.ts +13 -8
  512. package/src/resources/extensions/gsd/types.ts +44 -22
  513. package/src/resources/extensions/gsd/verification-evidence.ts +68 -0
  514. package/src/resources/extensions/gsd/workflow-logger.ts +15 -1
  515. package/src/resources/extensions/gsd/workflow-projections.ts +23 -5
  516. package/src/resources/extensions/gsd/worktree-manager.ts +76 -28
  517. package/src/resources/extensions/gsd/worktree-resolver.ts +4 -3
  518. package/src/resources/extensions/mcp-client/auth.ts +149 -0
  519. package/src/resources/extensions/mcp-client/index.ts +16 -1
  520. package/src/resources/extensions/ollama/index.ts +26 -25
  521. package/src/resources/extensions/ollama/model-capabilities.ts +41 -34
  522. package/src/resources/extensions/ollama/ndjson-stream.ts +63 -0
  523. package/src/resources/extensions/ollama/ollama-auth-mode.test.ts +20 -0
  524. package/src/resources/extensions/ollama/ollama-chat-provider.ts +459 -0
  525. package/src/resources/extensions/ollama/ollama-client.ts +30 -30
  526. package/src/resources/extensions/ollama/ollama-discovery.ts +5 -8
  527. package/src/resources/extensions/ollama/ollama-tool.ts +69 -0
  528. package/src/resources/extensions/ollama/tests/ollama-chat-provider-stream.test.ts +82 -0
  529. package/src/resources/extensions/ollama/tests/ollama-discovery.test.ts +0 -27
  530. package/src/resources/extensions/ollama/thinking-parser.ts +116 -0
  531. package/src/resources/extensions/ollama/types.ts +23 -0
  532. package/dist/web/standalone/.next/server/chunks/2229.js +0 -12
  533. package/dist/web/standalone/.next/static/5FLUBNdqolRyyehCyChPd/_buildManifest.js +0 -1
  534. package/dist/web/standalone/.next/static/chunks/app/_global-error/page-c4cc189e7b117ea2.js +0 -1
  535. package/dist/web/standalone/.next/static/chunks/app/api/boot/route-c4cc189e7b117ea2.js +0 -1
  536. package/dist/web/standalone/.next/static/chunks/app/api/bridge-terminal/input/route-c4cc189e7b117ea2.js +0 -1
  537. package/dist/web/standalone/.next/static/chunks/app/api/bridge-terminal/resize/route-c4cc189e7b117ea2.js +0 -1
  538. package/dist/web/standalone/.next/static/chunks/app/api/bridge-terminal/stream/route-c4cc189e7b117ea2.js +0 -1
  539. package/dist/web/standalone/.next/static/chunks/app/api/browse-directories/route-c4cc189e7b117ea2.js +0 -1
  540. package/dist/web/standalone/.next/static/chunks/app/api/captures/route-c4cc189e7b117ea2.js +0 -1
  541. package/dist/web/standalone/.next/static/chunks/app/api/cleanup/route-c4cc189e7b117ea2.js +0 -1
  542. package/dist/web/standalone/.next/static/chunks/app/api/dev-mode/route-c4cc189e7b117ea2.js +0 -1
  543. package/dist/web/standalone/.next/static/chunks/app/api/doctor/route-c4cc189e7b117ea2.js +0 -1
  544. package/dist/web/standalone/.next/static/chunks/app/api/experimental/route-c4cc189e7b117ea2.js +0 -1
  545. package/dist/web/standalone/.next/static/chunks/app/api/export-data/route-c4cc189e7b117ea2.js +0 -1
  546. package/dist/web/standalone/.next/static/chunks/app/api/files/route-c4cc189e7b117ea2.js +0 -1
  547. package/dist/web/standalone/.next/static/chunks/app/api/forensics/route-c4cc189e7b117ea2.js +0 -1
  548. package/dist/web/standalone/.next/static/chunks/app/api/git/route-c4cc189e7b117ea2.js +0 -1
  549. package/dist/web/standalone/.next/static/chunks/app/api/history/route-c4cc189e7b117ea2.js +0 -1
  550. package/dist/web/standalone/.next/static/chunks/app/api/hooks/route-c4cc189e7b117ea2.js +0 -1
  551. package/dist/web/standalone/.next/static/chunks/app/api/inspect/route-c4cc189e7b117ea2.js +0 -1
  552. package/dist/web/standalone/.next/static/chunks/app/api/knowledge/route-c4cc189e7b117ea2.js +0 -1
  553. package/dist/web/standalone/.next/static/chunks/app/api/live-state/route-c4cc189e7b117ea2.js +0 -1
  554. package/dist/web/standalone/.next/static/chunks/app/api/onboarding/route-c4cc189e7b117ea2.js +0 -1
  555. package/dist/web/standalone/.next/static/chunks/app/api/preferences/route-c4cc189e7b117ea2.js +0 -1
  556. package/dist/web/standalone/.next/static/chunks/app/api/projects/route-c4cc189e7b117ea2.js +0 -1
  557. package/dist/web/standalone/.next/static/chunks/app/api/recovery/route-c4cc189e7b117ea2.js +0 -1
  558. package/dist/web/standalone/.next/static/chunks/app/api/remote-questions/route-c4cc189e7b117ea2.js +0 -1
  559. package/dist/web/standalone/.next/static/chunks/app/api/session/browser/route-c4cc189e7b117ea2.js +0 -1
  560. package/dist/web/standalone/.next/static/chunks/app/api/session/command/route-c4cc189e7b117ea2.js +0 -1
  561. package/dist/web/standalone/.next/static/chunks/app/api/session/events/route-c4cc189e7b117ea2.js +0 -1
  562. package/dist/web/standalone/.next/static/chunks/app/api/session/manage/route-c4cc189e7b117ea2.js +0 -1
  563. package/dist/web/standalone/.next/static/chunks/app/api/settings-data/route-c4cc189e7b117ea2.js +0 -1
  564. package/dist/web/standalone/.next/static/chunks/app/api/shutdown/route-c4cc189e7b117ea2.js +0 -1
  565. package/dist/web/standalone/.next/static/chunks/app/api/skill-health/route-c4cc189e7b117ea2.js +0 -1
  566. package/dist/web/standalone/.next/static/chunks/app/api/steer/route-c4cc189e7b117ea2.js +0 -1
  567. package/dist/web/standalone/.next/static/chunks/app/api/switch-root/route-c4cc189e7b117ea2.js +0 -1
  568. package/dist/web/standalone/.next/static/chunks/app/api/terminal/input/route-c4cc189e7b117ea2.js +0 -1
  569. package/dist/web/standalone/.next/static/chunks/app/api/terminal/resize/route-c4cc189e7b117ea2.js +0 -1
  570. package/dist/web/standalone/.next/static/chunks/app/api/terminal/sessions/route-c4cc189e7b117ea2.js +0 -1
  571. package/dist/web/standalone/.next/static/chunks/app/api/terminal/stream/route-c4cc189e7b117ea2.js +0 -1
  572. package/dist/web/standalone/.next/static/chunks/app/api/terminal/upload/route-c4cc189e7b117ea2.js +0 -1
  573. package/dist/web/standalone/.next/static/chunks/app/api/undo/route-c4cc189e7b117ea2.js +0 -1
  574. package/dist/web/standalone/.next/static/chunks/app/api/update/route-c4cc189e7b117ea2.js +0 -1
  575. package/dist/web/standalone/.next/static/chunks/app/api/visualizer/route-c4cc189e7b117ea2.js +0 -1
  576. package/dist/web/standalone/.next/static/chunks/app/page-62be3b5fa91e4c8f.js +0 -1
  577. package/dist/web/standalone/.next/static/chunks/main-app-d3d4c336195465f9.js +0 -1
  578. package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/app-error-c4cc189e7b117ea2.js +0 -1
  579. package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/forbidden-c4cc189e7b117ea2.js +0 -1
  580. package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/global-error-ab5a8926e07ec673.js +0 -1
  581. package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/not-found-c4cc189e7b117ea2.js +0 -1
  582. package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/unauthorized-c4cc189e7b117ea2.js +0 -1
  583. /package/dist/web/standalone/.next/static/{5FLUBNdqolRyyehCyChPd → ffabZXz8JdN3EzX9EKt-R}/_ssgManifest.js +0 -0
@@ -0,0 +1,813 @@
1
+ /**
2
+ * post-execution-checks.test.ts — Unit tests for post-execution validation checks.
3
+ *
4
+ * Tests all 3 check types:
5
+ * 1. Import resolution — verify relative imports resolve to existing files
6
+ * 2. Cross-task signatures — detect signature drift and hallucination cascades
7
+ * 3. Pattern consistency — async style drift, naming convention warnings
8
+ */
9
+
10
+ import { describe, test } from "node:test";
11
+ import assert from "node:assert/strict";
12
+ import { tmpdir } from "node:os";
13
+ import { mkdirSync, writeFileSync, rmSync } from "node:fs";
14
+ import { join } from "node:path";
15
+
16
+ import {
17
+ extractRelativeImports,
18
+ resolveImportPath,
19
+ checkImportResolution,
20
+ checkCrossTaskSignatures,
21
+ checkPatternConsistency,
22
+ runPostExecutionChecks,
23
+ type PostExecutionResult,
24
+ } from "../post-execution-checks.ts";
25
+ import type { TaskRow } from "../gsd-db.ts";
26
+
27
+ // ─── Test Fixtures ───────────────────────────────────────────────────────────
28
+
29
+ /**
30
+ * Create a minimal TaskRow for testing.
31
+ */
32
+ function createTask(overrides: Partial<TaskRow> = {}): TaskRow {
33
+ return {
34
+ milestone_id: "M001",
35
+ slice_id: "S01",
36
+ id: overrides.id ?? "T01",
37
+ title: "Test Task",
38
+ status: "complete",
39
+ one_liner: "",
40
+ narrative: "",
41
+ verification_result: "",
42
+ duration: "",
43
+ completed_at: new Date().toISOString(),
44
+ blocker_discovered: false,
45
+ deviations: "",
46
+ known_issues: "",
47
+ key_files: overrides.key_files ?? [],
48
+ key_decisions: [],
49
+ full_summary_md: "",
50
+ description: overrides.description ?? "",
51
+ estimate: "",
52
+ files: overrides.files ?? [],
53
+ verify: "",
54
+ inputs: overrides.inputs ?? [],
55
+ expected_output: overrides.expected_output ?? [],
56
+ observability_impact: "",
57
+ full_plan_md: "",
58
+ sequence: overrides.sequence ?? 0,
59
+ ...overrides,
60
+ };
61
+ }
62
+
63
+ // ─── Import Extraction Tests ─────────────────────────────────────────────────
64
+
65
+ describe("extractRelativeImports", () => {
66
+ test("extracts import ... from statements", () => {
67
+ const source = `
68
+ import { foo } from './utils';
69
+ import bar from "../helpers/bar";
70
+ `;
71
+ const imports = extractRelativeImports(source);
72
+ assert.equal(imports.length, 2);
73
+ assert.ok(imports.some((i) => i.importPath === "./utils"));
74
+ assert.ok(imports.some((i) => i.importPath === "../helpers/bar"));
75
+ });
76
+
77
+ test("extracts side-effect imports", () => {
78
+ const source = `import './polyfill';`;
79
+ const imports = extractRelativeImports(source);
80
+ assert.equal(imports.length, 1);
81
+ assert.equal(imports[0].importPath, "./polyfill");
82
+ });
83
+
84
+ test("extracts require statements", () => {
85
+ const source = `
86
+ const utils = require('./utils');
87
+ const { bar } = require("../helpers/bar");
88
+ `;
89
+ const imports = extractRelativeImports(source);
90
+ assert.equal(imports.length, 2);
91
+ assert.ok(imports.some((i) => i.importPath === "./utils"));
92
+ assert.ok(imports.some((i) => i.importPath === "../helpers/bar"));
93
+ });
94
+
95
+ test("ignores non-relative imports", () => {
96
+ const source = `
97
+ import express from 'express';
98
+ import { readFile } from 'node:fs';
99
+ const lodash = require('lodash');
100
+ `;
101
+ const imports = extractRelativeImports(source);
102
+ assert.equal(imports.length, 0);
103
+ });
104
+
105
+ test("reports correct line numbers", () => {
106
+ const source = `// comment
107
+ import { a } from './a';
108
+ // another comment
109
+ import { b } from './b';
110
+ `;
111
+ const imports = extractRelativeImports(source);
112
+ assert.equal(imports.length, 2);
113
+ const importA = imports.find((i) => i.importPath === "./a");
114
+ const importB = imports.find((i) => i.importPath === "./b");
115
+ assert.equal(importA?.lineNum, 2);
116
+ assert.equal(importB?.lineNum, 4);
117
+ });
118
+
119
+ test("handles multiple imports on same line", () => {
120
+ const source = `import a from './a'; import b from './b';`;
121
+ const imports = extractRelativeImports(source);
122
+ assert.equal(imports.length, 2);
123
+ });
124
+
125
+ test("handles empty source", () => {
126
+ const imports = extractRelativeImports("");
127
+ assert.deepEqual(imports, []);
128
+ });
129
+ });
130
+
131
+ // ─── Import Resolution Tests ─────────────────────────────────────────────────
132
+
133
+ describe("resolveImportPath", () => {
134
+ let tempDir: string;
135
+
136
+ test("resolves file with exact extension", () => {
137
+ tempDir = join(tmpdir(), `post-exec-test-${Date.now()}`);
138
+ mkdirSync(tempDir, { recursive: true });
139
+ mkdirSync(join(tempDir, "src"), { recursive: true });
140
+ writeFileSync(join(tempDir, "src", "utils.ts"), "export const a = 1;");
141
+ writeFileSync(join(tempDir, "src", "main.ts"), "import { a } from './utils';");
142
+
143
+ try {
144
+ const result = resolveImportPath("./utils", "src/main.ts", tempDir);
145
+ assert.ok(result.exists);
146
+ assert.ok(result.resolvedPath?.endsWith("utils.ts"));
147
+ } finally {
148
+ rmSync(tempDir, { recursive: true, force: true });
149
+ }
150
+ });
151
+
152
+ test("resolves file without extension", () => {
153
+ tempDir = join(tmpdir(), `post-exec-test-${Date.now()}`);
154
+ mkdirSync(tempDir, { recursive: true });
155
+ mkdirSync(join(tempDir, "src"), { recursive: true });
156
+ writeFileSync(join(tempDir, "src", "helpers.js"), "module.exports = {};");
157
+ writeFileSync(join(tempDir, "src", "index.ts"), "");
158
+
159
+ try {
160
+ const result = resolveImportPath("./helpers", "src/index.ts", tempDir);
161
+ assert.ok(result.exists);
162
+ } finally {
163
+ rmSync(tempDir, { recursive: true, force: true });
164
+ }
165
+ });
166
+
167
+ test("resolves directory index file", () => {
168
+ tempDir = join(tmpdir(), `post-exec-test-${Date.now()}`);
169
+ mkdirSync(tempDir, { recursive: true });
170
+ mkdirSync(join(tempDir, "src", "utils"), { recursive: true });
171
+ writeFileSync(join(tempDir, "src", "utils", "index.ts"), "export {};");
172
+ writeFileSync(join(tempDir, "src", "main.ts"), "");
173
+
174
+ try {
175
+ const result = resolveImportPath("./utils", "src/main.ts", tempDir);
176
+ assert.ok(result.exists);
177
+ assert.ok(result.resolvedPath?.endsWith("index.ts"));
178
+ } finally {
179
+ rmSync(tempDir, { recursive: true, force: true });
180
+ }
181
+ });
182
+
183
+ test("resolves parent directory imports", () => {
184
+ tempDir = join(tmpdir(), `post-exec-test-${Date.now()}`);
185
+ mkdirSync(tempDir, { recursive: true });
186
+ mkdirSync(join(tempDir, "src", "nested"), { recursive: true });
187
+ writeFileSync(join(tempDir, "src", "utils.ts"), "export {};");
188
+ writeFileSync(join(tempDir, "src", "nested", "child.ts"), "");
189
+
190
+ try {
191
+ const result = resolveImportPath("../utils", "src/nested/child.ts", tempDir);
192
+ assert.ok(result.exists);
193
+ } finally {
194
+ rmSync(tempDir, { recursive: true, force: true });
195
+ }
196
+ });
197
+
198
+ test("fails for non-existent file", () => {
199
+ tempDir = join(tmpdir(), `post-exec-test-${Date.now()}`);
200
+ mkdirSync(tempDir, { recursive: true });
201
+ mkdirSync(join(tempDir, "src"), { recursive: true });
202
+ writeFileSync(join(tempDir, "src", "main.ts"), "");
203
+
204
+ try {
205
+ const result = resolveImportPath("./nonexistent", "src/main.ts", tempDir);
206
+ assert.ok(!result.exists);
207
+ assert.equal(result.resolvedPath, null);
208
+ } finally {
209
+ rmSync(tempDir, { recursive: true, force: true });
210
+ }
211
+ });
212
+
213
+ test("handles explicit extension in import", () => {
214
+ tempDir = join(tmpdir(), `post-exec-test-${Date.now()}`);
215
+ mkdirSync(tempDir, { recursive: true });
216
+ mkdirSync(join(tempDir, "src"), { recursive: true });
217
+ writeFileSync(join(tempDir, "src", "data.json"), "{}");
218
+ writeFileSync(join(tempDir, "src", "main.ts"), "");
219
+
220
+ try {
221
+ const result = resolveImportPath("./data.json", "src/main.ts", tempDir);
222
+ assert.ok(result.exists);
223
+ } finally {
224
+ rmSync(tempDir, { recursive: true, force: true });
225
+ }
226
+ });
227
+ });
228
+
229
+ // ─── Import Resolution Check Tests ───────────────────────────────────────────
230
+
231
+ describe("checkImportResolution", () => {
232
+ let tempDir: string;
233
+
234
+ test("passes when all imports resolve", () => {
235
+ tempDir = join(tmpdir(), `post-exec-test-${Date.now()}`);
236
+ mkdirSync(tempDir, { recursive: true });
237
+ mkdirSync(join(tempDir, "src"), { recursive: true });
238
+ writeFileSync(join(tempDir, "src", "utils.ts"), "export const a = 1;");
239
+ writeFileSync(
240
+ join(tempDir, "src", "main.ts"),
241
+ "import { a } from './utils';"
242
+ );
243
+
244
+ try {
245
+ const task = createTask({
246
+ id: "T01",
247
+ key_files: ["src/main.ts"],
248
+ });
249
+
250
+ const results = checkImportResolution(task, [], tempDir);
251
+ assert.deepEqual(results, []);
252
+ } finally {
253
+ rmSync(tempDir, { recursive: true, force: true });
254
+ }
255
+ });
256
+
257
+ test("fails when import doesn't resolve", () => {
258
+ tempDir = join(tmpdir(), `post-exec-test-${Date.now()}`);
259
+ mkdirSync(tempDir, { recursive: true });
260
+ mkdirSync(join(tempDir, "src"), { recursive: true });
261
+ writeFileSync(
262
+ join(tempDir, "src", "main.ts"),
263
+ "import { a } from './nonexistent';"
264
+ );
265
+
266
+ try {
267
+ const task = createTask({
268
+ id: "T01",
269
+ key_files: ["src/main.ts"],
270
+ });
271
+
272
+ const results = checkImportResolution(task, [], tempDir);
273
+ assert.equal(results.length, 1);
274
+ assert.equal(results[0].category, "import");
275
+ assert.equal(results[0].passed, false);
276
+ assert.equal(results[0].blocking, true);
277
+ assert.ok(results[0].message.includes("nonexistent"));
278
+ assert.ok(results[0].target.includes("src/main.ts"));
279
+ } finally {
280
+ rmSync(tempDir, { recursive: true, force: true });
281
+ }
282
+ });
283
+
284
+ test("skips non-JS/TS files", () => {
285
+ tempDir = join(tmpdir(), `post-exec-test-${Date.now()}`);
286
+ mkdirSync(tempDir, { recursive: true });
287
+ writeFileSync(join(tempDir, "README.md"), "# Docs");
288
+
289
+ try {
290
+ const task = createTask({
291
+ id: "T01",
292
+ key_files: ["README.md"],
293
+ });
294
+
295
+ const results = checkImportResolution(task, [], tempDir);
296
+ assert.deepEqual(results, []);
297
+ } finally {
298
+ rmSync(tempDir, { recursive: true, force: true });
299
+ }
300
+ });
301
+
302
+ test("handles multiple files with multiple imports", () => {
303
+ tempDir = join(tmpdir(), `post-exec-test-${Date.now()}`);
304
+ mkdirSync(tempDir, { recursive: true });
305
+ mkdirSync(join(tempDir, "src"), { recursive: true });
306
+ writeFileSync(join(tempDir, "src", "utils.ts"), "export const a = 1;");
307
+ writeFileSync(
308
+ join(tempDir, "src", "a.ts"),
309
+ "import { a } from './utils';\nimport { b } from './missing';"
310
+ );
311
+ writeFileSync(
312
+ join(tempDir, "src", "b.ts"),
313
+ "import { x } from './also-missing';"
314
+ );
315
+
316
+ try {
317
+ const task = createTask({
318
+ id: "T01",
319
+ key_files: ["src/a.ts", "src/b.ts"],
320
+ });
321
+
322
+ const results = checkImportResolution(task, [], tempDir);
323
+ assert.equal(results.length, 2);
324
+ assert.ok(results.some((r) => r.message.includes("missing")));
325
+ assert.ok(results.some((r) => r.message.includes("also-missing")));
326
+ } finally {
327
+ rmSync(tempDir, { recursive: true, force: true });
328
+ }
329
+ });
330
+
331
+ test("skips if key_file doesn't exist", () => {
332
+ tempDir = join(tmpdir(), `post-exec-test-${Date.now()}`);
333
+ mkdirSync(tempDir, { recursive: true });
334
+
335
+ try {
336
+ const task = createTask({
337
+ id: "T01",
338
+ key_files: ["src/deleted.ts"],
339
+ });
340
+
341
+ const results = checkImportResolution(task, [], tempDir);
342
+ assert.deepEqual(results, []);
343
+ } finally {
344
+ rmSync(tempDir, { recursive: true, force: true });
345
+ }
346
+ });
347
+ });
348
+
349
+ // ─── Cross-Task Signature Tests ──────────────────────────────────────────────
350
+
351
+ describe("checkCrossTaskSignatures", () => {
352
+ let tempDir: string;
353
+
354
+ test("passes when no prior tasks exist", () => {
355
+ tempDir = join(tmpdir(), `post-exec-test-${Date.now()}`);
356
+ mkdirSync(tempDir, { recursive: true });
357
+ mkdirSync(join(tempDir, "src"), { recursive: true });
358
+ writeFileSync(
359
+ join(tempDir, "src", "api.ts"),
360
+ "export function getData(): string { return ''; }"
361
+ );
362
+
363
+ try {
364
+ const task = createTask({
365
+ id: "T02",
366
+ key_files: ["src/api.ts"],
367
+ });
368
+
369
+ const results = checkCrossTaskSignatures(task, [], tempDir);
370
+ assert.deepEqual(results, []);
371
+ } finally {
372
+ rmSync(tempDir, { recursive: true, force: true });
373
+ }
374
+ });
375
+
376
+ test("passes when signatures match", () => {
377
+ tempDir = join(tmpdir(), `post-exec-test-${Date.now()}`);
378
+ mkdirSync(tempDir, { recursive: true });
379
+ mkdirSync(join(tempDir, "src"), { recursive: true });
380
+ writeFileSync(
381
+ join(tempDir, "src", "utils.ts"),
382
+ "export function process(data: string): boolean { return true; }"
383
+ );
384
+ writeFileSync(
385
+ join(tempDir, "src", "api.ts"),
386
+ "export function process(data: string): boolean { return false; }"
387
+ );
388
+
389
+ try {
390
+ const priorTask = createTask({
391
+ id: "T01",
392
+ key_files: ["src/utils.ts"],
393
+ });
394
+ const currentTask = createTask({
395
+ id: "T02",
396
+ key_files: ["src/api.ts"],
397
+ });
398
+
399
+ const results = checkCrossTaskSignatures(currentTask, [priorTask], tempDir);
400
+ assert.deepEqual(results, []);
401
+ } finally {
402
+ rmSync(tempDir, { recursive: true, force: true });
403
+ }
404
+ });
405
+
406
+ test("warns on parameter mismatch (non-blocking)", () => {
407
+ tempDir = join(tmpdir(), `post-exec-test-${Date.now()}`);
408
+ mkdirSync(tempDir, { recursive: true });
409
+ mkdirSync(join(tempDir, "src"), { recursive: true });
410
+ writeFileSync(
411
+ join(tempDir, "src", "utils.ts"),
412
+ "export function save(name: string): void {}"
413
+ );
414
+ writeFileSync(
415
+ join(tempDir, "src", "api.ts"),
416
+ "export function save(name: string, id: number): void {}"
417
+ );
418
+
419
+ try {
420
+ const priorTask = createTask({
421
+ id: "T01",
422
+ key_files: ["src/utils.ts"],
423
+ });
424
+ const currentTask = createTask({
425
+ id: "T02",
426
+ key_files: ["src/api.ts"],
427
+ });
428
+
429
+ const results = checkCrossTaskSignatures(currentTask, [priorTask], tempDir);
430
+ assert.equal(results.length, 1);
431
+ assert.equal(results[0].category, "signature");
432
+ assert.equal(results[0].target, "save");
433
+ assert.equal(results[0].passed, false);
434
+ assert.equal(results[0].blocking, false);
435
+ assert.ok(results[0].message.includes("parameters"));
436
+ } finally {
437
+ rmSync(tempDir, { recursive: true, force: true });
438
+ }
439
+ });
440
+
441
+ test("warns on return type mismatch (non-blocking)", () => {
442
+ tempDir = join(tmpdir(), `post-exec-test-${Date.now()}`);
443
+ mkdirSync(tempDir, { recursive: true });
444
+ mkdirSync(join(tempDir, "src"), { recursive: true });
445
+ writeFileSync(
446
+ join(tempDir, "src", "utils.ts"),
447
+ "export function fetch(): string { return ''; }"
448
+ );
449
+ writeFileSync(
450
+ join(tempDir, "src", "api.ts"),
451
+ "export function fetch(): number { return 0; }"
452
+ );
453
+
454
+ try {
455
+ const priorTask = createTask({
456
+ id: "T01",
457
+ key_files: ["src/utils.ts"],
458
+ });
459
+ const currentTask = createTask({
460
+ id: "T02",
461
+ key_files: ["src/api.ts"],
462
+ });
463
+
464
+ const results = checkCrossTaskSignatures(currentTask, [priorTask], tempDir);
465
+ assert.equal(results.length, 1);
466
+ assert.ok(results[0].message.includes("return"));
467
+ } finally {
468
+ rmSync(tempDir, { recursive: true, force: true });
469
+ }
470
+ });
471
+
472
+ test("handles multiple prior tasks", () => {
473
+ tempDir = join(tmpdir(), `post-exec-test-${Date.now()}`);
474
+ mkdirSync(tempDir, { recursive: true });
475
+ mkdirSync(join(tempDir, "src"), { recursive: true });
476
+ writeFileSync(
477
+ join(tempDir, "src", "types.ts"),
478
+ "export function parse(s: string): object { return {}; }"
479
+ );
480
+ writeFileSync(
481
+ join(tempDir, "src", "utils.ts"),
482
+ "export function validate(x: object): boolean { return true; }"
483
+ );
484
+ writeFileSync(
485
+ join(tempDir, "src", "api.ts"),
486
+ `export function parse(s: number): object { return {}; }
487
+ export function validate(x: object): boolean { return true; }`
488
+ );
489
+
490
+ try {
491
+ const priorTask1 = createTask({ id: "T01", key_files: ["src/types.ts"] });
492
+ const priorTask2 = createTask({ id: "T02", key_files: ["src/utils.ts"] });
493
+ const currentTask = createTask({ id: "T03", key_files: ["src/api.ts"] });
494
+
495
+ const results = checkCrossTaskSignatures(
496
+ currentTask,
497
+ [priorTask1, priorTask2],
498
+ tempDir
499
+ );
500
+ // Should have 1 warning for parse() parameter mismatch
501
+ assert.equal(results.length, 1);
502
+ assert.ok(results[0].message.includes("parse"));
503
+ } finally {
504
+ rmSync(tempDir, { recursive: true, force: true });
505
+ }
506
+ });
507
+ });
508
+
509
+ // ─── Pattern Consistency Tests ───────────────────────────────────────────────
510
+
511
+ describe("checkPatternConsistency", () => {
512
+ let tempDir: string;
513
+
514
+ test("passes when async style is consistent (await only)", () => {
515
+ tempDir = join(tmpdir(), `post-exec-test-${Date.now()}`);
516
+ mkdirSync(tempDir, { recursive: true });
517
+ writeFileSync(
518
+ join(tempDir, "api.ts"),
519
+ `async function getData(): Promise<string> {
520
+ const result = await fetch('/api');
521
+ return await result.text();
522
+ }`
523
+ );
524
+
525
+ try {
526
+ const task = createTask({ id: "T01", key_files: ["api.ts"] });
527
+ const results = checkPatternConsistency(task, [], tempDir);
528
+ const asyncResults = results.filter((r) => r.message.includes("async"));
529
+ assert.equal(asyncResults.length, 0);
530
+ } finally {
531
+ rmSync(tempDir, { recursive: true, force: true });
532
+ }
533
+ });
534
+
535
+ test("passes when async style is consistent (.then only)", () => {
536
+ tempDir = join(tmpdir(), `post-exec-test-${Date.now()}`);
537
+ mkdirSync(tempDir, { recursive: true });
538
+ writeFileSync(
539
+ join(tempDir, "api.ts"),
540
+ `function getData(): Promise<string> {
541
+ return fetch('/api').then(r => r.text());
542
+ }`
543
+ );
544
+
545
+ try {
546
+ const task = createTask({ id: "T01", key_files: ["api.ts"] });
547
+ const results = checkPatternConsistency(task, [], tempDir);
548
+ const asyncResults = results.filter((r) => r.message.includes("async"));
549
+ assert.equal(asyncResults.length, 0);
550
+ } finally {
551
+ rmSync(tempDir, { recursive: true, force: true });
552
+ }
553
+ });
554
+
555
+ test("warns when mixing async/await with .then()", () => {
556
+ tempDir = join(tmpdir(), `post-exec-test-${Date.now()}`);
557
+ mkdirSync(tempDir, { recursive: true });
558
+ writeFileSync(
559
+ join(tempDir, "api.ts"),
560
+ `async function getData(): Promise<string> {
561
+ const result = await fetch('/api');
562
+ return result.text().then(t => t.toUpperCase());
563
+ }`
564
+ );
565
+
566
+ try {
567
+ const task = createTask({ id: "T01", key_files: ["api.ts"] });
568
+ const results = checkPatternConsistency(task, [], tempDir);
569
+ const asyncResults = results.filter((r) => r.message.includes("async"));
570
+ assert.equal(asyncResults.length, 1);
571
+ assert.equal(asyncResults[0].category, "pattern");
572
+ assert.equal(asyncResults[0].passed, true); // Warning only
573
+ assert.equal(asyncResults[0].blocking, false);
574
+ } finally {
575
+ rmSync(tempDir, { recursive: true, force: true });
576
+ }
577
+ });
578
+
579
+ test("passes when naming is consistent (camelCase only)", () => {
580
+ tempDir = join(tmpdir(), `post-exec-test-${Date.now()}`);
581
+ mkdirSync(tempDir, { recursive: true });
582
+ writeFileSync(
583
+ join(tempDir, "api.ts"),
584
+ `function getUserData() {}
585
+ const processItems = () => {};
586
+ function validateInput() {}`
587
+ );
588
+
589
+ try {
590
+ const task = createTask({ id: "T01", key_files: ["api.ts"] });
591
+ const results = checkPatternConsistency(task, [], tempDir);
592
+ const namingResults = results.filter((r) => r.message.includes("naming") || r.message.includes("Case"));
593
+ assert.equal(namingResults.length, 0);
594
+ } finally {
595
+ rmSync(tempDir, { recursive: true, force: true });
596
+ }
597
+ });
598
+
599
+ test("warns when mixing camelCase and snake_case", () => {
600
+ tempDir = join(tmpdir(), `post-exec-test-${Date.now()}`);
601
+ mkdirSync(tempDir, { recursive: true });
602
+ writeFileSync(
603
+ join(tempDir, "api.ts"),
604
+ `function getUserData() {}
605
+ function process_items() {}
606
+ const validate_input = () => {};`
607
+ );
608
+
609
+ try {
610
+ const task = createTask({ id: "T01", key_files: ["api.ts"] });
611
+ const results = checkPatternConsistency(task, [], tempDir);
612
+ const namingResults = results.filter((r) => r.message.includes("camelCase") || r.message.includes("snake_case"));
613
+ assert.equal(namingResults.length, 1);
614
+ assert.equal(namingResults[0].category, "pattern");
615
+ assert.equal(namingResults[0].blocking, false);
616
+ } finally {
617
+ rmSync(tempDir, { recursive: true, force: true });
618
+ }
619
+ });
620
+
621
+ test("skips non-JS/TS files", () => {
622
+ tempDir = join(tmpdir(), `post-exec-test-${Date.now()}`);
623
+ mkdirSync(tempDir, { recursive: true });
624
+ writeFileSync(join(tempDir, "config.json"), '{"key": "value"}');
625
+
626
+ try {
627
+ const task = createTask({ id: "T01", key_files: ["config.json"] });
628
+ const results = checkPatternConsistency(task, [], tempDir);
629
+ assert.deepEqual(results, []);
630
+ } finally {
631
+ rmSync(tempDir, { recursive: true, force: true });
632
+ }
633
+ });
634
+ });
635
+
636
+ // ─── runPostExecutionChecks Integration Tests ────────────────────────────────
637
+
638
+ describe("runPostExecutionChecks", () => {
639
+ let tempDir: string;
640
+
641
+ test("returns pass status when all checks pass", () => {
642
+ tempDir = join(tmpdir(), `post-exec-test-${Date.now()}`);
643
+ mkdirSync(tempDir, { recursive: true });
644
+ mkdirSync(join(tempDir, "src"), { recursive: true });
645
+ writeFileSync(join(tempDir, "src", "utils.ts"), "export const a = 1;");
646
+ writeFileSync(
647
+ join(tempDir, "src", "main.ts"),
648
+ `import { a } from './utils';
649
+ function processData(): void {}`
650
+ );
651
+
652
+ try {
653
+ const task = createTask({ id: "T01", key_files: ["src/main.ts"] });
654
+ const result = runPostExecutionChecks(task, [], tempDir);
655
+ assert.equal(result.status, "pass");
656
+ assert.equal(result.checks.length, 0);
657
+ assert.ok(result.durationMs >= 0);
658
+ } finally {
659
+ rmSync(tempDir, { recursive: true, force: true });
660
+ }
661
+ });
662
+
663
+ test("returns fail status when blocking failure exists", () => {
664
+ tempDir = join(tmpdir(), `post-exec-test-${Date.now()}`);
665
+ mkdirSync(tempDir, { recursive: true });
666
+ mkdirSync(join(tempDir, "src"), { recursive: true });
667
+ writeFileSync(
668
+ join(tempDir, "src", "main.ts"),
669
+ "import { a } from './nonexistent';"
670
+ );
671
+
672
+ try {
673
+ const task = createTask({ id: "T01", key_files: ["src/main.ts"] });
674
+ const result = runPostExecutionChecks(task, [], tempDir);
675
+ assert.equal(result.status, "fail");
676
+ assert.ok(result.checks.length > 0);
677
+ assert.ok(result.checks.some((c) => c.blocking === true));
678
+ } finally {
679
+ rmSync(tempDir, { recursive: true, force: true });
680
+ }
681
+ });
682
+
683
+ test("returns warn status for non-blocking issues only", () => {
684
+ tempDir = join(tmpdir(), `post-exec-test-${Date.now()}`);
685
+ mkdirSync(tempDir, { recursive: true });
686
+ mkdirSync(join(tempDir, "src"), { recursive: true });
687
+ writeFileSync(
688
+ join(tempDir, "src", "api.ts"),
689
+ `async function getData() {
690
+ const result = await fetch('/api');
691
+ return result.text().then(t => t);
692
+ }`
693
+ );
694
+
695
+ try {
696
+ const task = createTask({ id: "T01", key_files: ["src/api.ts"] });
697
+ const result = runPostExecutionChecks(task, [], tempDir);
698
+ assert.equal(result.status, "warn");
699
+ assert.ok(result.checks.some((c) => c.category === "pattern"));
700
+ } finally {
701
+ rmSync(tempDir, { recursive: true, force: true });
702
+ }
703
+ });
704
+
705
+ test("combines results from all check types", () => {
706
+ tempDir = join(tmpdir(), `post-exec-test-${Date.now()}`);
707
+ mkdirSync(tempDir, { recursive: true });
708
+ mkdirSync(join(tempDir, "src"), { recursive: true });
709
+ writeFileSync(
710
+ join(tempDir, "src", "utils.ts"),
711
+ "export function process(s: string): void {}"
712
+ );
713
+ writeFileSync(
714
+ join(tempDir, "src", "api.ts"),
715
+ `import { x } from './missing';
716
+ async function getData() {
717
+ await fetch('/api');
718
+ return fetch('/api2').then(r => r);
719
+ }
720
+ export function process(n: number): void {}`
721
+ );
722
+
723
+ try {
724
+ const priorTask = createTask({ id: "T01", key_files: ["src/utils.ts"] });
725
+ const currentTask = createTask({ id: "T02", key_files: ["src/api.ts"] });
726
+
727
+ const result = runPostExecutionChecks(currentTask, [priorTask], tempDir);
728
+ assert.equal(result.status, "fail"); // Import failure is blocking
729
+
730
+ const categories = new Set(result.checks.map((c) => c.category));
731
+ assert.ok(categories.has("import")); // From unresolved import
732
+ assert.ok(categories.has("signature")); // From signature mismatch
733
+ assert.ok(categories.has("pattern")); // From async style drift
734
+ } finally {
735
+ rmSync(tempDir, { recursive: true, force: true });
736
+ }
737
+ });
738
+
739
+ test("reports duration in milliseconds", () => {
740
+ tempDir = join(tmpdir(), `post-exec-test-${Date.now()}`);
741
+ mkdirSync(tempDir, { recursive: true });
742
+
743
+ try {
744
+ const task = createTask({ id: "T01", key_files: [] });
745
+ const result = runPostExecutionChecks(task, [], tempDir);
746
+ assert.ok(typeof result.durationMs === "number");
747
+ assert.ok(result.durationMs >= 0);
748
+ } finally {
749
+ rmSync(tempDir, { recursive: true, force: true });
750
+ }
751
+ });
752
+
753
+ test("handles empty key_files array", () => {
754
+ tempDir = join(tmpdir(), `post-exec-test-${Date.now()}`);
755
+ mkdirSync(tempDir, { recursive: true });
756
+
757
+ try {
758
+ const task = createTask({ id: "T01", key_files: [] });
759
+ const result = runPostExecutionChecks(task, [], tempDir);
760
+ assert.equal(result.status, "pass");
761
+ assert.deepEqual(result.checks, []);
762
+ } finally {
763
+ rmSync(tempDir, { recursive: true, force: true });
764
+ }
765
+ });
766
+ });
767
+
768
+ // ─── PostExecutionResult Type Tests ──────────────────────────────────────────
769
+
770
+ describe("PostExecutionResult type", () => {
771
+ test("status is one of pass, warn, fail", () => {
772
+ const tempDir = join(tmpdir(), `post-exec-test-${Date.now()}`);
773
+ mkdirSync(tempDir, { recursive: true });
774
+
775
+ try {
776
+ const task = createTask({ id: "T01", key_files: [] });
777
+ const result = runPostExecutionChecks(task, [], tempDir);
778
+ assert.ok(["pass", "warn", "fail"].includes(result.status));
779
+ } finally {
780
+ rmSync(tempDir, { recursive: true, force: true });
781
+ }
782
+ });
783
+
784
+ test("checks array matches PostExecutionCheckJSON schema", () => {
785
+ const tempDir = join(tmpdir(), `post-exec-test-${Date.now()}`);
786
+ mkdirSync(tempDir, { recursive: true });
787
+ mkdirSync(join(tempDir, "src"), { recursive: true });
788
+ writeFileSync(
789
+ join(tempDir, "src", "main.ts"),
790
+ "import { a } from './missing';"
791
+ );
792
+
793
+ try {
794
+ const task = createTask({ id: "T01", key_files: ["src/main.ts"] });
795
+ const result = runPostExecutionChecks(task, [], tempDir);
796
+
797
+ for (const check of result.checks) {
798
+ assert.ok(
799
+ ["import", "signature", "pattern"].includes(check.category),
800
+ `Invalid category: ${check.category}`
801
+ );
802
+ assert.ok(typeof check.target === "string");
803
+ assert.ok(typeof check.passed === "boolean");
804
+ assert.ok(typeof check.message === "string");
805
+ if (check.blocking !== undefined) {
806
+ assert.ok(typeof check.blocking === "boolean");
807
+ }
808
+ }
809
+ } finally {
810
+ rmSync(tempDir, { recursive: true, force: true });
811
+ }
812
+ });
813
+ });