gsd-pi 2.63.0 → 2.64.0-dev.9c14bd0

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 (448) 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/query-tools.js +85 -0
  27. package/dist/resources/extensions/gsd/bootstrap/register-extension.js +3 -0
  28. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +32 -1
  29. package/dist/resources/extensions/gsd/bootstrap/sanitize-complete-milestone.js +54 -0
  30. package/dist/resources/extensions/gsd/bootstrap/system-context.js +30 -2
  31. package/dist/resources/extensions/gsd/commands-handlers.js +9 -4
  32. package/dist/resources/extensions/gsd/constants.js +42 -0
  33. package/dist/resources/extensions/gsd/db-writer.js +72 -4
  34. package/dist/resources/extensions/gsd/forensics.js +20 -4
  35. package/dist/resources/extensions/gsd/gsd-db.js +64 -17
  36. package/dist/resources/extensions/gsd/guided-flow.js +19 -0
  37. package/dist/resources/extensions/gsd/metrics.js +27 -1
  38. package/dist/resources/extensions/gsd/native-git-bridge.js +5 -3
  39. package/dist/resources/extensions/gsd/post-execution-checks.js +407 -0
  40. package/dist/resources/extensions/gsd/pre-execution-checks.js +464 -0
  41. package/dist/resources/extensions/gsd/preferences-types.js +6 -0
  42. package/dist/resources/extensions/gsd/preferences-validation.js +33 -0
  43. package/dist/resources/extensions/gsd/preferences.js +11 -2
  44. package/dist/resources/extensions/gsd/prompt-loader.js +7 -0
  45. package/dist/resources/extensions/gsd/prompts/complete-milestone.md +2 -0
  46. package/dist/resources/extensions/gsd/prompts/complete-slice.md +2 -0
  47. package/dist/resources/extensions/gsd/prompts/doctor-heal.md +1 -0
  48. package/dist/resources/extensions/gsd/prompts/forensics.md +2 -0
  49. package/dist/resources/extensions/gsd/prompts/reassess-roadmap.md +2 -0
  50. package/dist/resources/extensions/gsd/prompts/system.md +4 -7
  51. package/dist/resources/extensions/gsd/prompts/validate-milestone.md +2 -0
  52. package/dist/resources/extensions/gsd/roadmap-mutations.js +1 -1
  53. package/dist/resources/extensions/gsd/roadmap-slices.js +9 -5
  54. package/dist/resources/extensions/gsd/safety/content-validator.js +73 -0
  55. package/dist/resources/extensions/gsd/safety/destructive-guard.js +34 -0
  56. package/dist/resources/extensions/gsd/safety/evidence-collector.js +109 -0
  57. package/dist/resources/extensions/gsd/safety/evidence-cross-ref.js +83 -0
  58. package/dist/resources/extensions/gsd/safety/file-change-validator.js +71 -0
  59. package/dist/resources/extensions/gsd/safety/git-checkpoint.js +91 -0
  60. package/dist/resources/extensions/gsd/safety/safety-harness.js +64 -0
  61. package/dist/resources/extensions/gsd/slice-parallel-conflict.js +67 -0
  62. package/dist/resources/extensions/gsd/slice-parallel-eligibility.js +51 -0
  63. package/dist/resources/extensions/gsd/slice-parallel-orchestrator.js +378 -0
  64. package/dist/resources/extensions/gsd/state.js +74 -14
  65. package/dist/resources/extensions/gsd/status-guards.js +11 -0
  66. package/dist/resources/extensions/gsd/tools/complete-milestone.js +17 -12
  67. package/dist/resources/extensions/gsd/tools/complete-slice.js +40 -26
  68. package/dist/resources/extensions/gsd/tools/complete-task.js +12 -12
  69. package/dist/resources/extensions/gsd/tools/plan-milestone.js +33 -25
  70. package/dist/resources/extensions/gsd/tools/plan-slice.js +5 -8
  71. package/dist/resources/extensions/gsd/verification-evidence.js +18 -0
  72. package/dist/resources/extensions/gsd/workflow-projections.js +21 -5
  73. package/dist/resources/extensions/gsd/worktree-manager.js +82 -29
  74. package/dist/resources/extensions/gsd/worktree-resolver.js +4 -3
  75. package/dist/resources/extensions/mcp-client/auth.js +101 -0
  76. package/dist/resources/extensions/mcp-client/index.js +10 -1
  77. package/dist/resources/extensions/ollama/index.js +28 -22
  78. package/dist/resources/extensions/ollama/model-capabilities.js +37 -34
  79. package/dist/resources/extensions/ollama/ndjson-stream.js +54 -0
  80. package/dist/resources/extensions/ollama/ollama-chat-provider.js +380 -0
  81. package/dist/resources/extensions/ollama/ollama-client.js +23 -32
  82. package/dist/resources/extensions/ollama/ollama-discovery.js +2 -7
  83. package/dist/resources/extensions/ollama/ollama-tool.js +62 -0
  84. package/dist/resources/extensions/ollama/thinking-parser.js +104 -0
  85. package/dist/update-cmd.js +4 -2
  86. package/dist/web/standalone/.next/BUILD_ID +1 -1
  87. package/dist/web/standalone/.next/app-path-routes-manifest.json +15 -15
  88. package/dist/web/standalone/.next/build-manifest.json +3 -3
  89. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  90. package/dist/web/standalone/.next/required-server-files.json +4 -4
  91. package/dist/web/standalone/.next/server/app/_global-error/page.js +3 -3
  92. package/dist/web/standalone/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
  93. package/dist/web/standalone/.next/server/app/_global-error.html +2 -2
  94. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  95. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  96. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  97. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  98. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  99. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  100. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  101. package/dist/web/standalone/.next/server/app/_not-found/page.js +2 -2
  102. package/dist/web/standalone/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
  103. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  104. package/dist/web/standalone/.next/server/app/_not-found.rsc +3 -3
  105. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +3 -3
  106. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  107. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +3 -3
  108. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  109. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  110. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  111. package/dist/web/standalone/.next/server/app/api/boot/route.js +1 -1
  112. package/dist/web/standalone/.next/server/app/api/boot/route.js.nft.json +1 -1
  113. package/dist/web/standalone/.next/server/app/api/boot/route_client-reference-manifest.js +1 -1
  114. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route.js +1 -1
  115. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route.js.nft.json +1 -1
  116. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route_client-reference-manifest.js +1 -1
  117. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route.js +1 -1
  118. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route.js.nft.json +1 -1
  119. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route_client-reference-manifest.js +1 -1
  120. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route.js +2 -2
  121. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route.js.nft.json +1 -1
  122. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route_client-reference-manifest.js +1 -1
  123. package/dist/web/standalone/.next/server/app/api/browse-directories/route.js +1 -1
  124. package/dist/web/standalone/.next/server/app/api/browse-directories/route_client-reference-manifest.js +1 -1
  125. package/dist/web/standalone/.next/server/app/api/captures/route.js +1 -1
  126. package/dist/web/standalone/.next/server/app/api/captures/route.js.nft.json +1 -1
  127. package/dist/web/standalone/.next/server/app/api/captures/route_client-reference-manifest.js +1 -1
  128. package/dist/web/standalone/.next/server/app/api/cleanup/route.js +1 -1
  129. package/dist/web/standalone/.next/server/app/api/cleanup/route.js.nft.json +1 -1
  130. package/dist/web/standalone/.next/server/app/api/cleanup/route_client-reference-manifest.js +1 -1
  131. package/dist/web/standalone/.next/server/app/api/dev-mode/route.js +1 -1
  132. package/dist/web/standalone/.next/server/app/api/dev-mode/route_client-reference-manifest.js +1 -1
  133. package/dist/web/standalone/.next/server/app/api/doctor/route.js +1 -1
  134. package/dist/web/standalone/.next/server/app/api/doctor/route.js.nft.json +1 -1
  135. package/dist/web/standalone/.next/server/app/api/doctor/route_client-reference-manifest.js +1 -1
  136. package/dist/web/standalone/.next/server/app/api/experimental/route.js +2 -2
  137. package/dist/web/standalone/.next/server/app/api/experimental/route_client-reference-manifest.js +1 -1
  138. package/dist/web/standalone/.next/server/app/api/export-data/route.js +1 -1
  139. package/dist/web/standalone/.next/server/app/api/export-data/route.js.nft.json +1 -1
  140. package/dist/web/standalone/.next/server/app/api/export-data/route_client-reference-manifest.js +1 -1
  141. package/dist/web/standalone/.next/server/app/api/files/route.js +1 -1
  142. package/dist/web/standalone/.next/server/app/api/files/route.js.nft.json +1 -1
  143. package/dist/web/standalone/.next/server/app/api/files/route_client-reference-manifest.js +1 -1
  144. package/dist/web/standalone/.next/server/app/api/forensics/route.js +1 -1
  145. package/dist/web/standalone/.next/server/app/api/forensics/route.js.nft.json +1 -1
  146. package/dist/web/standalone/.next/server/app/api/forensics/route_client-reference-manifest.js +1 -1
  147. package/dist/web/standalone/.next/server/app/api/git/route.js +1 -1
  148. package/dist/web/standalone/.next/server/app/api/git/route.js.nft.json +1 -1
  149. package/dist/web/standalone/.next/server/app/api/git/route_client-reference-manifest.js +1 -1
  150. package/dist/web/standalone/.next/server/app/api/history/route.js +1 -1
  151. package/dist/web/standalone/.next/server/app/api/history/route.js.nft.json +1 -1
  152. package/dist/web/standalone/.next/server/app/api/history/route_client-reference-manifest.js +1 -1
  153. package/dist/web/standalone/.next/server/app/api/hooks/route.js +1 -1
  154. package/dist/web/standalone/.next/server/app/api/hooks/route.js.nft.json +1 -1
  155. package/dist/web/standalone/.next/server/app/api/hooks/route_client-reference-manifest.js +1 -1
  156. package/dist/web/standalone/.next/server/app/api/inspect/route.js +1 -1
  157. package/dist/web/standalone/.next/server/app/api/inspect/route.js.nft.json +1 -1
  158. package/dist/web/standalone/.next/server/app/api/inspect/route_client-reference-manifest.js +1 -1
  159. package/dist/web/standalone/.next/server/app/api/knowledge/route.js +1 -1
  160. package/dist/web/standalone/.next/server/app/api/knowledge/route.js.nft.json +1 -1
  161. package/dist/web/standalone/.next/server/app/api/knowledge/route_client-reference-manifest.js +1 -1
  162. package/dist/web/standalone/.next/server/app/api/live-state/route.js +1 -1
  163. package/dist/web/standalone/.next/server/app/api/live-state/route.js.nft.json +1 -1
  164. package/dist/web/standalone/.next/server/app/api/live-state/route_client-reference-manifest.js +1 -1
  165. package/dist/web/standalone/.next/server/app/api/onboarding/route.js +1 -1
  166. package/dist/web/standalone/.next/server/app/api/onboarding/route.js.nft.json +1 -1
  167. package/dist/web/standalone/.next/server/app/api/onboarding/route_client-reference-manifest.js +1 -1
  168. package/dist/web/standalone/.next/server/app/api/preferences/route.js +1 -1
  169. package/dist/web/standalone/.next/server/app/api/preferences/route_client-reference-manifest.js +1 -1
  170. package/dist/web/standalone/.next/server/app/api/projects/route.js +1 -1
  171. package/dist/web/standalone/.next/server/app/api/projects/route.js.nft.json +1 -1
  172. package/dist/web/standalone/.next/server/app/api/projects/route_client-reference-manifest.js +1 -1
  173. package/dist/web/standalone/.next/server/app/api/recovery/route.js +1 -1
  174. package/dist/web/standalone/.next/server/app/api/recovery/route.js.nft.json +1 -1
  175. package/dist/web/standalone/.next/server/app/api/recovery/route_client-reference-manifest.js +1 -1
  176. package/dist/web/standalone/.next/server/app/api/remote-questions/route.js +2 -2
  177. package/dist/web/standalone/.next/server/app/api/remote-questions/route_client-reference-manifest.js +1 -1
  178. package/dist/web/standalone/.next/server/app/api/session/browser/route.js +1 -1
  179. package/dist/web/standalone/.next/server/app/api/session/browser/route.js.nft.json +1 -1
  180. package/dist/web/standalone/.next/server/app/api/session/browser/route_client-reference-manifest.js +1 -1
  181. package/dist/web/standalone/.next/server/app/api/session/command/route.js +1 -1
  182. package/dist/web/standalone/.next/server/app/api/session/command/route.js.nft.json +1 -1
  183. package/dist/web/standalone/.next/server/app/api/session/command/route_client-reference-manifest.js +1 -1
  184. package/dist/web/standalone/.next/server/app/api/session/events/route.js +2 -2
  185. package/dist/web/standalone/.next/server/app/api/session/events/route.js.nft.json +1 -1
  186. package/dist/web/standalone/.next/server/app/api/session/events/route_client-reference-manifest.js +1 -1
  187. package/dist/web/standalone/.next/server/app/api/session/manage/route.js +1 -1
  188. package/dist/web/standalone/.next/server/app/api/session/manage/route.js.nft.json +1 -1
  189. package/dist/web/standalone/.next/server/app/api/session/manage/route_client-reference-manifest.js +1 -1
  190. package/dist/web/standalone/.next/server/app/api/settings-data/route.js +1 -1
  191. package/dist/web/standalone/.next/server/app/api/settings-data/route.js.nft.json +1 -1
  192. package/dist/web/standalone/.next/server/app/api/settings-data/route_client-reference-manifest.js +1 -1
  193. package/dist/web/standalone/.next/server/app/api/shutdown/route.js +1 -1
  194. package/dist/web/standalone/.next/server/app/api/shutdown/route_client-reference-manifest.js +1 -1
  195. package/dist/web/standalone/.next/server/app/api/skill-health/route.js +1 -1
  196. package/dist/web/standalone/.next/server/app/api/skill-health/route.js.nft.json +1 -1
  197. package/dist/web/standalone/.next/server/app/api/skill-health/route_client-reference-manifest.js +1 -1
  198. package/dist/web/standalone/.next/server/app/api/steer/route.js +1 -1
  199. package/dist/web/standalone/.next/server/app/api/steer/route.js.nft.json +1 -1
  200. package/dist/web/standalone/.next/server/app/api/steer/route_client-reference-manifest.js +1 -1
  201. package/dist/web/standalone/.next/server/app/api/switch-root/route.js +1 -1
  202. package/dist/web/standalone/.next/server/app/api/switch-root/route.js.nft.json +1 -1
  203. package/dist/web/standalone/.next/server/app/api/switch-root/route_client-reference-manifest.js +1 -1
  204. package/dist/web/standalone/.next/server/app/api/terminal/input/route.js +2 -2
  205. package/dist/web/standalone/.next/server/app/api/terminal/input/route_client-reference-manifest.js +1 -1
  206. package/dist/web/standalone/.next/server/app/api/terminal/resize/route.js +2 -2
  207. package/dist/web/standalone/.next/server/app/api/terminal/resize/route_client-reference-manifest.js +1 -1
  208. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js +2 -2
  209. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js.nft.json +1 -1
  210. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route_client-reference-manifest.js +1 -1
  211. package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js +4 -4
  212. package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js.nft.json +1 -1
  213. package/dist/web/standalone/.next/server/app/api/terminal/stream/route_client-reference-manifest.js +1 -1
  214. package/dist/web/standalone/.next/server/app/api/terminal/upload/route.js +1 -1
  215. package/dist/web/standalone/.next/server/app/api/terminal/upload/route_client-reference-manifest.js +1 -1
  216. package/dist/web/standalone/.next/server/app/api/undo/route.js +1 -1
  217. package/dist/web/standalone/.next/server/app/api/undo/route.js.nft.json +1 -1
  218. package/dist/web/standalone/.next/server/app/api/undo/route_client-reference-manifest.js +1 -1
  219. package/dist/web/standalone/.next/server/app/api/update/route.js +1 -1
  220. package/dist/web/standalone/.next/server/app/api/update/route_client-reference-manifest.js +1 -1
  221. package/dist/web/standalone/.next/server/app/api/visualizer/route.js +1 -1
  222. package/dist/web/standalone/.next/server/app/api/visualizer/route.js.nft.json +1 -1
  223. package/dist/web/standalone/.next/server/app/api/visualizer/route_client-reference-manifest.js +1 -1
  224. package/dist/web/standalone/.next/server/app/index.html +1 -1
  225. package/dist/web/standalone/.next/server/app/index.rsc +4 -4
  226. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +2 -2
  227. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +4 -4
  228. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  229. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +3 -3
  230. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  231. package/dist/web/standalone/.next/server/app/page.js +2 -2
  232. package/dist/web/standalone/.next/server/app/page_client-reference-manifest.js +1 -1
  233. package/dist/web/standalone/.next/server/app-paths-manifest.json +15 -15
  234. package/dist/web/standalone/.next/server/chunks/6897.js +12 -0
  235. package/dist/web/standalone/.next/server/chunks/7471.js +3 -3
  236. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  237. package/dist/web/standalone/.next/server/middleware.js +2 -2
  238. package/dist/web/standalone/.next/server/next-font-manifest.js +1 -1
  239. package/dist/web/standalone/.next/server/next-font-manifest.json +1 -1
  240. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  241. package/dist/web/standalone/.next/server/pages/500.html +2 -2
  242. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  243. package/dist/web/standalone/.next/static/chunks/app/_not-found/{page-2f24283c162b6ab3.js → page-f2a7482d42a5614b.js} +1 -1
  244. package/dist/web/standalone/.next/static/chunks/app/{layout-9ecfd95f343793f0.js → layout-a16c7a7ecdf0c2cf.js} +1 -1
  245. package/dist/web/standalone/.next/static/chunks/app/page-0c485498795110d6.js +1 -0
  246. package/dist/web/standalone/.next/static/chunks/main-app-fdab67f7802d7832.js +1 -0
  247. package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/global-error-459824ffb8c323dd.js +1 -0
  248. package/dist/web/standalone/node_modules/node-pty/build/Makefile +2 -2
  249. package/dist/web/standalone/node_modules/node-pty/build/Release/pty.node +0 -0
  250. package/dist/web/standalone/node_modules/node-pty/build/pty.target.mk +14 -14
  251. package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api.target.mk +14 -14
  252. package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api_except.target.mk +14 -14
  253. package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api_maybe.target.mk +14 -14
  254. package/dist/web/standalone/server.js +1 -1
  255. package/dist/welcome-screen.js +1 -1
  256. package/package.json +1 -1
  257. package/packages/pi-agent-core/dist/agent-loop.d.ts +8 -0
  258. package/packages/pi-agent-core/dist/agent-loop.d.ts.map +1 -1
  259. package/packages/pi-agent-core/dist/agent-loop.js +50 -0
  260. package/packages/pi-agent-core/dist/agent-loop.js.map +1 -1
  261. package/packages/pi-agent-core/src/agent-loop.test.ts +221 -5
  262. package/packages/pi-agent-core/src/agent-loop.ts +53 -0
  263. package/packages/pi-ai/dist/types.d.ts +16 -1
  264. package/packages/pi-ai/dist/types.d.ts.map +1 -1
  265. package/packages/pi-ai/dist/types.js.map +1 -1
  266. package/packages/pi-ai/src/types.ts +18 -1
  267. package/packages/pi-coding-agent/dist/core/auth-storage.d.ts +9 -0
  268. package/packages/pi-coding-agent/dist/core/auth-storage.d.ts.map +1 -1
  269. package/packages/pi-coding-agent/dist/core/auth-storage.js +50 -1
  270. package/packages/pi-coding-agent/dist/core/auth-storage.js.map +1 -1
  271. package/packages/pi-coding-agent/dist/core/auth-storage.test.js +41 -0
  272. package/packages/pi-coding-agent/dist/core/auth-storage.test.js.map +1 -1
  273. package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts +7 -0
  274. package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts.map +1 -1
  275. package/packages/pi-coding-agent/dist/core/extensions/loader.js +31 -4
  276. package/packages/pi-coding-agent/dist/core/extensions/loader.js.map +1 -1
  277. package/packages/pi-coding-agent/dist/core/extensions/loader.test.js +28 -1
  278. package/packages/pi-coding-agent/dist/core/extensions/loader.test.js.map +1 -1
  279. package/packages/pi-coding-agent/dist/core/extensions/provider-registration.test.d.ts +2 -0
  280. package/packages/pi-coding-agent/dist/core/extensions/provider-registration.test.d.ts.map +1 -0
  281. package/packages/pi-coding-agent/dist/core/extensions/provider-registration.test.js +46 -0
  282. package/packages/pi-coding-agent/dist/core/extensions/provider-registration.test.js.map +1 -0
  283. package/packages/pi-coding-agent/dist/core/extensions/types.d.ts +2 -0
  284. package/packages/pi-coding-agent/dist/core/extensions/types.d.ts.map +1 -1
  285. package/packages/pi-coding-agent/dist/core/extensions/types.js.map +1 -1
  286. package/packages/pi-coding-agent/dist/core/model-registry.d.ts +1 -0
  287. package/packages/pi-coding-agent/dist/core/model-registry.d.ts.map +1 -1
  288. package/packages/pi-coding-agent/dist/core/model-registry.js +12 -0
  289. package/packages/pi-coding-agent/dist/core/model-registry.js.map +1 -1
  290. package/packages/pi-coding-agent/dist/core/model-resolver.js +3 -3
  291. package/packages/pi-coding-agent/dist/core/model-resolver.js.map +1 -1
  292. package/packages/pi-coding-agent/dist/core/resource-loader.d.ts +23 -1
  293. package/packages/pi-coding-agent/dist/core/resource-loader.d.ts.map +1 -1
  294. package/packages/pi-coding-agent/dist/core/resource-loader.js +80 -56
  295. package/packages/pi-coding-agent/dist/core/resource-loader.js.map +1 -1
  296. package/packages/pi-coding-agent/dist/core/sdk.d.ts.map +1 -1
  297. package/packages/pi-coding-agent/dist/core/sdk.js +9 -0
  298. package/packages/pi-coding-agent/dist/core/sdk.js.map +1 -1
  299. package/packages/pi-coding-agent/package.json +1 -1
  300. package/packages/pi-coding-agent/src/core/auth-storage.test.ts +53 -0
  301. package/packages/pi-coding-agent/src/core/auth-storage.ts +66 -1
  302. package/packages/pi-coding-agent/src/core/extensions/loader.test.ts +39 -1
  303. package/packages/pi-coding-agent/src/core/extensions/loader.ts +34 -4
  304. package/packages/pi-coding-agent/src/core/extensions/provider-registration.test.ts +81 -0
  305. package/packages/pi-coding-agent/src/core/extensions/types.ts +2 -0
  306. package/packages/pi-coding-agent/src/core/model-registry.ts +14 -0
  307. package/packages/pi-coding-agent/src/core/model-resolver.ts +3 -3
  308. package/packages/pi-coding-agent/src/core/resource-loader.ts +89 -56
  309. package/packages/pi-coding-agent/src/core/sdk.ts +10 -0
  310. package/pkg/package.json +1 -1
  311. package/src/resources/extensions/cmux/index.ts +18 -12
  312. package/src/resources/extensions/gsd/auto/detect-stuck.ts +27 -0
  313. package/src/resources/extensions/gsd/auto/finalize-timeout.ts +46 -0
  314. package/src/resources/extensions/gsd/auto/loop.ts +5 -0
  315. package/src/resources/extensions/gsd/auto/phases.ts +194 -33
  316. package/src/resources/extensions/gsd/auto/session.ts +14 -0
  317. package/src/resources/extensions/gsd/auto-dashboard.ts +16 -7
  318. package/src/resources/extensions/gsd/auto-model-selection.ts +36 -0
  319. package/src/resources/extensions/gsd/auto-post-unit.ts +263 -12
  320. package/src/resources/extensions/gsd/auto-prompts.ts +21 -0
  321. package/src/resources/extensions/gsd/auto-recovery.ts +9 -8
  322. package/src/resources/extensions/gsd/auto-start.ts +11 -20
  323. package/src/resources/extensions/gsd/auto-timers.ts +2 -1
  324. package/src/resources/extensions/gsd/auto-tool-tracking.ts +19 -0
  325. package/src/resources/extensions/gsd/auto-verification.ts +190 -2
  326. package/src/resources/extensions/gsd/auto-worktree.ts +14 -6
  327. package/src/resources/extensions/gsd/auto.ts +26 -1
  328. package/src/resources/extensions/gsd/bootstrap/db-tools.ts +160 -88
  329. package/src/resources/extensions/gsd/bootstrap/dynamic-tools.ts +15 -0
  330. package/src/resources/extensions/gsd/bootstrap/query-tools.ts +98 -0
  331. package/src/resources/extensions/gsd/bootstrap/register-extension.ts +4 -0
  332. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +36 -1
  333. package/src/resources/extensions/gsd/bootstrap/sanitize-complete-milestone.ts +57 -0
  334. package/src/resources/extensions/gsd/bootstrap/system-context.ts +31 -2
  335. package/src/resources/extensions/gsd/commands-handlers.ts +10 -4
  336. package/src/resources/extensions/gsd/constants.ts +44 -0
  337. package/src/resources/extensions/gsd/db-writer.ts +78 -4
  338. package/src/resources/extensions/gsd/forensics.ts +21 -5
  339. package/src/resources/extensions/gsd/gsd-db.ts +64 -17
  340. package/src/resources/extensions/gsd/guided-flow.ts +22 -0
  341. package/src/resources/extensions/gsd/metrics.ts +28 -1
  342. package/src/resources/extensions/gsd/native-git-bridge.ts +5 -3
  343. package/src/resources/extensions/gsd/post-execution-checks.ts +539 -0
  344. package/src/resources/extensions/gsd/pre-execution-checks.ts +573 -0
  345. package/src/resources/extensions/gsd/preferences-types.ts +44 -0
  346. package/src/resources/extensions/gsd/preferences-validation.ts +33 -0
  347. package/src/resources/extensions/gsd/preferences.ts +13 -2
  348. package/src/resources/extensions/gsd/prompt-loader.ts +8 -0
  349. package/src/resources/extensions/gsd/prompts/complete-milestone.md +2 -0
  350. package/src/resources/extensions/gsd/prompts/complete-slice.md +2 -0
  351. package/src/resources/extensions/gsd/prompts/doctor-heal.md +1 -0
  352. package/src/resources/extensions/gsd/prompts/forensics.md +2 -0
  353. package/src/resources/extensions/gsd/prompts/reassess-roadmap.md +2 -0
  354. package/src/resources/extensions/gsd/prompts/system.md +4 -7
  355. package/src/resources/extensions/gsd/prompts/validate-milestone.md +2 -0
  356. package/src/resources/extensions/gsd/roadmap-mutations.ts +1 -1
  357. package/src/resources/extensions/gsd/roadmap-slices.ts +10 -5
  358. package/src/resources/extensions/gsd/safety/content-validator.ts +98 -0
  359. package/src/resources/extensions/gsd/safety/destructive-guard.ts +49 -0
  360. package/src/resources/extensions/gsd/safety/evidence-collector.ts +151 -0
  361. package/src/resources/extensions/gsd/safety/evidence-cross-ref.ts +120 -0
  362. package/src/resources/extensions/gsd/safety/file-change-validator.ts +108 -0
  363. package/src/resources/extensions/gsd/safety/git-checkpoint.ts +106 -0
  364. package/src/resources/extensions/gsd/safety/safety-harness.ts +105 -0
  365. package/src/resources/extensions/gsd/slice-parallel-conflict.ts +86 -0
  366. package/src/resources/extensions/gsd/slice-parallel-eligibility.ts +73 -0
  367. package/src/resources/extensions/gsd/slice-parallel-orchestrator.ts +477 -0
  368. package/src/resources/extensions/gsd/state.ts +67 -12
  369. package/src/resources/extensions/gsd/status-guards.ts +13 -0
  370. package/src/resources/extensions/gsd/tests/artifact-corruption-2630.test.ts +288 -0
  371. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +34 -13
  372. package/src/resources/extensions/gsd/tests/auto-start-time-persistence.test.ts +50 -0
  373. package/src/resources/extensions/gsd/tests/cmux.test.ts +58 -0
  374. package/src/resources/extensions/gsd/tests/cold-resume-db-reopen.test.ts +51 -0
  375. package/src/resources/extensions/gsd/tests/complete-milestone.test.ts +140 -0
  376. package/src/resources/extensions/gsd/tests/complete-slice-string-coercion.test.ts +211 -0
  377. package/src/resources/extensions/gsd/tests/complete-task.test.ts +39 -0
  378. package/src/resources/extensions/gsd/tests/dashboard-model-label-ordering.test.ts +107 -0
  379. package/src/resources/extensions/gsd/tests/db-access-guardrails.test.ts +109 -0
  380. package/src/resources/extensions/gsd/tests/db-path-worktree-symlink.test.ts +13 -9
  381. package/src/resources/extensions/gsd/tests/db-writer.test.ts +134 -0
  382. package/src/resources/extensions/gsd/tests/deferred-slice-dispatch.test.ts +203 -0
  383. package/src/resources/extensions/gsd/tests/discuss-tool-scoping.test.ts +130 -0
  384. package/src/resources/extensions/gsd/tests/doctor-fix-flag.test.ts +92 -0
  385. package/src/resources/extensions/gsd/tests/enhanced-verification-integration.test.ts +526 -0
  386. package/src/resources/extensions/gsd/tests/finalize-timeout-guard.test.ts +116 -0
  387. package/src/resources/extensions/gsd/tests/flat-rate-routing-guard.test.ts +50 -0
  388. package/src/resources/extensions/gsd/tests/forensics-stuck-loops.test.ts +103 -0
  389. package/src/resources/extensions/gsd/tests/git-checkpoint.test.ts +94 -0
  390. package/src/resources/extensions/gsd/tests/insert-slice-no-wipe.test.ts +88 -0
  391. package/src/resources/extensions/gsd/tests/integration/git-service.test.ts +27 -7
  392. package/src/resources/extensions/gsd/tests/integration/idle-recovery.test.ts +34 -0
  393. package/src/resources/extensions/gsd/tests/metrics.test.ts +116 -1
  394. package/src/resources/extensions/gsd/tests/milestone-status-tool.test.ts +201 -0
  395. package/src/resources/extensions/gsd/tests/plan-milestone-title.test.ts +2 -1
  396. package/src/resources/extensions/gsd/tests/plan-milestone.test.ts +82 -18
  397. package/src/resources/extensions/gsd/tests/post-exec-retry-bypass.test.ts +312 -0
  398. package/src/resources/extensions/gsd/tests/post-execution-checks.test.ts +813 -0
  399. package/src/resources/extensions/gsd/tests/pre-execution-checks.test.ts +999 -0
  400. package/src/resources/extensions/gsd/tests/pre-execution-fail-closed.test.ts +266 -0
  401. package/src/resources/extensions/gsd/tests/pre-execution-pause-wiring.test.ts +457 -0
  402. package/src/resources/extensions/gsd/tests/preferences.test.ts +10 -0
  403. package/src/resources/extensions/gsd/tests/prompt-contracts.test.ts +25 -0
  404. package/src/resources/extensions/gsd/tests/roadmap-slices.test.ts +69 -0
  405. package/src/resources/extensions/gsd/tests/shared-wal.test.ts +30 -0
  406. package/src/resources/extensions/gsd/tests/slice-context-injection.test.ts +50 -0
  407. package/src/resources/extensions/gsd/tests/slice-parallel-conflict.test.ts +92 -0
  408. package/src/resources/extensions/gsd/tests/slice-parallel-eligibility.test.ts +95 -0
  409. package/src/resources/extensions/gsd/tests/slice-parallel-orchestrator.test.ts +83 -0
  410. package/src/resources/extensions/gsd/tests/stuck-detection-coverage.test.ts +42 -0
  411. package/src/resources/extensions/gsd/tests/tool-invocation-error-loop-break.test.ts +103 -0
  412. package/src/resources/extensions/gsd/tests/tool-param-optionality.test.ts +349 -0
  413. package/src/resources/extensions/gsd/tests/worktree-health-dispatch.test.ts +35 -2
  414. package/src/resources/extensions/gsd/tests/worktree-health-monorepo.test.ts +73 -0
  415. package/src/resources/extensions/gsd/tests/worktree-resolver.test.ts +34 -0
  416. package/src/resources/extensions/gsd/tests/worktree-submodule-safety.test.ts +1 -1
  417. package/src/resources/extensions/gsd/tests/worktree-teardown-safety.test.ts +148 -0
  418. package/src/resources/extensions/gsd/tools/complete-milestone.ts +34 -20
  419. package/src/resources/extensions/gsd/tools/complete-slice.ts +41 -26
  420. package/src/resources/extensions/gsd/tools/complete-task.ts +12 -12
  421. package/src/resources/extensions/gsd/tools/plan-milestone.ts +55 -30
  422. package/src/resources/extensions/gsd/tools/plan-slice.ts +13 -8
  423. package/src/resources/extensions/gsd/types.ts +44 -22
  424. package/src/resources/extensions/gsd/verification-evidence.ts +68 -0
  425. package/src/resources/extensions/gsd/workflow-logger.ts +2 -1
  426. package/src/resources/extensions/gsd/workflow-projections.ts +23 -5
  427. package/src/resources/extensions/gsd/worktree-manager.ts +76 -28
  428. package/src/resources/extensions/gsd/worktree-resolver.ts +4 -3
  429. package/src/resources/extensions/mcp-client/auth.ts +149 -0
  430. package/src/resources/extensions/mcp-client/index.ts +16 -1
  431. package/src/resources/extensions/ollama/index.ts +26 -25
  432. package/src/resources/extensions/ollama/model-capabilities.ts +41 -34
  433. package/src/resources/extensions/ollama/ndjson-stream.ts +63 -0
  434. package/src/resources/extensions/ollama/ollama-auth-mode.test.ts +20 -0
  435. package/src/resources/extensions/ollama/ollama-chat-provider.ts +459 -0
  436. package/src/resources/extensions/ollama/ollama-client.ts +30 -30
  437. package/src/resources/extensions/ollama/ollama-discovery.ts +5 -8
  438. package/src/resources/extensions/ollama/ollama-tool.ts +69 -0
  439. package/src/resources/extensions/ollama/tests/ollama-chat-provider-stream.test.ts +82 -0
  440. package/src/resources/extensions/ollama/tests/ollama-discovery.test.ts +0 -27
  441. package/src/resources/extensions/ollama/thinking-parser.ts +116 -0
  442. package/src/resources/extensions/ollama/types.ts +23 -0
  443. package/dist/web/standalone/.next/server/chunks/2229.js +0 -12
  444. package/dist/web/standalone/.next/static/chunks/app/page-62be3b5fa91e4c8f.js +0 -1
  445. package/dist/web/standalone/.next/static/chunks/main-app-d3d4c336195465f9.js +0 -1
  446. package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/global-error-ab5a8926e07ec673.js +0 -1
  447. /package/dist/web/standalone/.next/static/{5FLUBNdqolRyyehCyChPd → SoxM61WC_ia7R2gk4VMpJ}/_buildManifest.js +0 -0
  448. /package/dist/web/standalone/.next/static/{5FLUBNdqolRyyehCyChPd → SoxM61WC_ia7R2gk4VMpJ}/_ssgManifest.js +0 -0
@@ -18,6 +18,7 @@ import { StdioClientTransport } from "@modelcontextprotocol/sdk/client/stdio.js"
18
18
  import { StreamableHTTPClientTransport } from "@modelcontextprotocol/sdk/client/streamableHttp.js";
19
19
  import { readFileSync, existsSync } from "node:fs";
20
20
  import { join } from "node:path";
21
+ import { buildHttpTransportOpts } from "./auth.js";
21
22
  // ─── Connection Manager ───────────────────────────────────────────────────────
22
23
  const connections = new Map();
23
24
  let configCache = null;
@@ -51,6 +52,8 @@ function readConfigs() {
51
52
  : hasUrl
52
53
  ? "http"
53
54
  : "unknown";
55
+ const hasHeaders = hasUrl && config.headers && typeof config.headers === "object";
56
+ const hasOAuth = hasUrl && config.oauth && typeof config.oauth === "object";
54
57
  servers.push({
55
58
  name,
56
59
  transport,
@@ -63,6 +66,8 @@ function readConfigs() {
63
66
  cwd: typeof config.cwd === "string" ? config.cwd : undefined,
64
67
  }),
65
68
  ...(hasUrl && { url: config.url }),
69
+ headers: hasHeaders ? config.headers : undefined,
70
+ oauth: hasOAuth ? config.oauth : undefined,
66
71
  });
67
72
  }
68
73
  }
@@ -113,7 +118,11 @@ async function getOrConnect(name, signal) {
113
118
  }
114
119
  else if (config.transport === "http" && config.url) {
115
120
  const resolvedUrl = config.url.replace(/\$\{([^}]+)\}/g, (_, varName) => process.env[varName] ?? "");
116
- transport = new StreamableHTTPClientTransport(new URL(resolvedUrl));
121
+ const httpOpts = buildHttpTransportOpts({
122
+ headers: config.headers,
123
+ oauth: config.oauth,
124
+ });
125
+ transport = new StreamableHTTPClientTransport(new URL(resolvedUrl), httpOpts);
117
126
  }
118
127
  else {
119
128
  throw new Error(`Server "${config.name}" has unsupported transport: ${config.transport}`);
@@ -17,16 +17,9 @@
17
17
  */
18
18
  import { importExtensionModule } from "@gsd/pi-coding-agent";
19
19
  import * as client from "./ollama-client.js";
20
- import { discoverModels, getOllamaOpenAIBaseUrl } from "./ollama-discovery.js";
20
+ import { discoverModels } from "./ollama-discovery.js";
21
21
  import { registerOllamaCommands } from "./ollama-commands.js";
22
- /** Default compat settings for Ollama models via OpenAI-compat endpoint */
23
- const OLLAMA_COMPAT = {
24
- supportsDeveloperRole: false,
25
- supportsReasoningEffort: false,
26
- supportsUsageInStreaming: false,
27
- maxTokensField: "max_tokens",
28
- supportsStore: false,
29
- };
22
+ import { streamOllamaChat } from "./ollama-chat-provider.js";
30
23
  let toolsPromise = null;
31
24
  async function registerOllamaTools(pi) {
32
25
  if (!toolsPromise) {
@@ -58,11 +51,17 @@ async function probeAndRegister(pi) {
58
51
  const models = await discoverModels();
59
52
  if (models.length === 0)
60
53
  return true; // Running but no models pulled
61
- const baseUrl = getOllamaOpenAIBaseUrl();
54
+ const baseUrl = client.getOllamaHost();
55
+ // Use authMode "apiKey" with a dummy key (#3440).
56
+ // authMode "none" requires a custom streamSimple handler, but Ollama uses
57
+ // the standard OpenAI-compatible streaming endpoint. Ollama ignores the
58
+ // Authorization header so the dummy key is harmless.
62
59
  pi.registerProvider("ollama", {
63
- authMode: "none",
60
+ authMode: "apiKey",
61
+ apiKey: "ollama",
64
62
  baseUrl,
65
- api: "openai-completions",
63
+ api: "ollama-chat",
64
+ streamSimple: streamOllamaChat,
66
65
  isReady: () => true,
67
66
  models: models.map((m) => ({
68
67
  id: m.id,
@@ -72,7 +71,7 @@ async function probeAndRegister(pi) {
72
71
  cost: m.cost,
73
72
  contextWindow: m.contextWindow,
74
73
  maxTokens: m.maxTokens,
75
- compat: OLLAMA_COMPAT,
74
+ providerOptions: (m.ollamaOptions ?? {}),
76
75
  })),
77
76
  });
78
77
  providerRegistered = true;
@@ -91,16 +90,23 @@ export default function ollama(pi) {
91
90
  else {
92
91
  await registerOllamaTools(pi);
93
92
  }
94
- // Async probe don't block startup
95
- probeAndRegister(pi)
96
- .then((found) => {
97
- if (found && ctx.hasUI) {
98
- ctx.ui.setStatus("ollama", "Ollama");
93
+ // In headless/auto mode, await the probe so the fallback resolver can
94
+ // see Ollama before the first LLM call (#3531 race condition).
95
+ // In interactive mode, keep it async for fast startup.
96
+ if (!ctx.hasUI) {
97
+ try {
98
+ await probeAndRegister(pi);
99
99
  }
100
- })
101
- .catch(() => {
102
- // Silently ignore probe failures
103
- });
100
+ catch { /* non-fatal */ }
101
+ }
102
+ else {
103
+ probeAndRegister(pi)
104
+ .then((found) => {
105
+ if (found)
106
+ ctx.ui.setStatus("ollama", "Ollama");
107
+ })
108
+ .catch(() => { });
109
+ }
104
110
  });
105
111
  pi.on("session_shutdown", async () => {
106
112
  if (providerRegistered) {
@@ -4,50 +4,53 @@
4
4
  * Keys are matched as prefixes against the model name (before the colon/tag).
5
5
  * More specific entries should appear first.
6
6
  */
7
+ // Note: ollamaOptions.num_ctx is set for known model families where the context
8
+ // window is authoritative. For unknown/estimated models, num_ctx is NOT sent
9
+ // to avoid OOM risk — Ollama uses its own safe default instead.
7
10
  const KNOWN_MODELS = [
8
11
  // ─── Reasoning models ───────────────────────────────────────────────
9
- ["deepseek-r1", { contextWindow: 131072, reasoning: true }],
10
- ["qwq", { contextWindow: 131072, reasoning: true }],
12
+ ["deepseek-r1", { contextWindow: 131072, reasoning: true, ollamaOptions: { num_ctx: 131072 } }],
13
+ ["qwq", { contextWindow: 131072, reasoning: true, ollamaOptions: { num_ctx: 131072 } }],
11
14
  // ─── Vision models ──────────────────────────────────────────────────
12
- ["llava", { contextWindow: 4096, input: ["text", "image"] }],
13
- ["bakllava", { contextWindow: 4096, input: ["text", "image"] }],
14
- ["moondream", { contextWindow: 8192, input: ["text", "image"] }],
15
- ["llama3.2-vision", { contextWindow: 131072, input: ["text", "image"] }],
16
- ["minicpm-v", { contextWindow: 4096, input: ["text", "image"] }],
15
+ ["llava", { contextWindow: 4096, input: ["text", "image"], ollamaOptions: { num_ctx: 4096 } }],
16
+ ["bakllava", { contextWindow: 4096, input: ["text", "image"], ollamaOptions: { num_ctx: 4096 } }],
17
+ ["moondream", { contextWindow: 8192, input: ["text", "image"], ollamaOptions: { num_ctx: 8192 } }],
18
+ ["llama3.2-vision", { contextWindow: 131072, input: ["text", "image"], ollamaOptions: { num_ctx: 131072 } }],
19
+ ["minicpm-v", { contextWindow: 4096, input: ["text", "image"], ollamaOptions: { num_ctx: 4096 } }],
17
20
  // ─── Code models ────────────────────────────────────────────────────
18
- ["codestral", { contextWindow: 262144, maxTokens: 32768 }],
19
- ["qwen2.5-coder", { contextWindow: 131072, maxTokens: 32768 }],
20
- ["deepseek-coder-v2", { contextWindow: 131072, maxTokens: 16384 }],
21
- ["starcoder2", { contextWindow: 16384, maxTokens: 8192 }],
22
- ["codegemma", { contextWindow: 8192, maxTokens: 8192 }],
23
- ["codellama", { contextWindow: 16384, maxTokens: 8192 }],
24
- ["devstral", { contextWindow: 131072, maxTokens: 32768 }],
21
+ ["codestral", { contextWindow: 262144, maxTokens: 32768, ollamaOptions: { num_ctx: 262144 } }],
22
+ ["qwen2.5-coder", { contextWindow: 131072, maxTokens: 32768, ollamaOptions: { num_ctx: 131072 } }],
23
+ ["deepseek-coder-v2", { contextWindow: 131072, maxTokens: 16384, ollamaOptions: { num_ctx: 131072 } }],
24
+ ["starcoder2", { contextWindow: 16384, maxTokens: 8192, ollamaOptions: { num_ctx: 16384 } }],
25
+ ["codegemma", { contextWindow: 8192, maxTokens: 8192, ollamaOptions: { num_ctx: 8192 } }],
26
+ ["codellama", { contextWindow: 16384, maxTokens: 8192, ollamaOptions: { num_ctx: 16384 } }],
27
+ ["devstral", { contextWindow: 131072, maxTokens: 32768, ollamaOptions: { num_ctx: 131072 } }],
25
28
  // ─── Llama family ───────────────────────────────────────────────────
26
- ["llama3.3", { contextWindow: 131072, maxTokens: 16384 }],
27
- ["llama3.2", { contextWindow: 131072, maxTokens: 16384 }],
28
- ["llama3.1", { contextWindow: 131072, maxTokens: 16384 }],
29
- ["llama3", { contextWindow: 8192, maxTokens: 8192 }],
30
- ["llama2", { contextWindow: 4096, maxTokens: 4096 }],
29
+ ["llama3.3", { contextWindow: 131072, maxTokens: 16384, ollamaOptions: { num_ctx: 131072 } }],
30
+ ["llama3.2", { contextWindow: 131072, maxTokens: 16384, ollamaOptions: { num_ctx: 131072 } }],
31
+ ["llama3.1", { contextWindow: 131072, maxTokens: 16384, ollamaOptions: { num_ctx: 131072 } }],
32
+ ["llama3", { contextWindow: 8192, maxTokens: 8192, ollamaOptions: { num_ctx: 8192 } }],
33
+ ["llama2", { contextWindow: 4096, maxTokens: 4096, ollamaOptions: { num_ctx: 4096 } }],
31
34
  // ─── Qwen family ────────────────────────────────────────────────────
32
- ["qwen3", { contextWindow: 131072, maxTokens: 32768 }],
33
- ["qwen2.5", { contextWindow: 131072, maxTokens: 32768 }],
34
- ["qwen2", { contextWindow: 131072, maxTokens: 32768 }],
35
+ ["qwen3", { contextWindow: 131072, maxTokens: 32768, ollamaOptions: { num_ctx: 131072 } }],
36
+ ["qwen2.5", { contextWindow: 131072, maxTokens: 32768, ollamaOptions: { num_ctx: 131072 } }],
37
+ ["qwen2", { contextWindow: 131072, maxTokens: 32768, ollamaOptions: { num_ctx: 131072 } }],
35
38
  // ─── Gemma family ───────────────────────────────────────────────────
36
- ["gemma3", { contextWindow: 131072, maxTokens: 16384 }],
37
- ["gemma2", { contextWindow: 8192, maxTokens: 8192 }],
39
+ ["gemma3", { contextWindow: 131072, maxTokens: 16384, ollamaOptions: { num_ctx: 131072 } }],
40
+ ["gemma2", { contextWindow: 8192, maxTokens: 8192, ollamaOptions: { num_ctx: 8192 } }],
38
41
  // ─── Mistral family ─────────────────────────────────────────────────
39
- ["mistral-large", { contextWindow: 131072, maxTokens: 16384 }],
40
- ["mistral-small", { contextWindow: 131072, maxTokens: 16384 }],
41
- ["mistral-nemo", { contextWindow: 131072, maxTokens: 16384 }],
42
- ["mistral", { contextWindow: 32768, maxTokens: 8192 }],
43
- ["mixtral", { contextWindow: 32768, maxTokens: 8192 }],
42
+ ["mistral-large", { contextWindow: 131072, maxTokens: 16384, ollamaOptions: { num_ctx: 131072 } }],
43
+ ["mistral-small", { contextWindow: 131072, maxTokens: 16384, ollamaOptions: { num_ctx: 131072 } }],
44
+ ["mistral-nemo", { contextWindow: 131072, maxTokens: 16384, ollamaOptions: { num_ctx: 131072 } }],
45
+ ["mistral", { contextWindow: 32768, maxTokens: 8192, ollamaOptions: { num_ctx: 32768 } }],
46
+ ["mixtral", { contextWindow: 32768, maxTokens: 8192, ollamaOptions: { num_ctx: 32768 } }],
44
47
  // ─── Phi family ─────────────────────────────────────────────────────
45
- ["phi4", { contextWindow: 16384, maxTokens: 16384 }],
46
- ["phi3.5", { contextWindow: 131072, maxTokens: 16384 }],
47
- ["phi3", { contextWindow: 131072, maxTokens: 4096 }],
48
+ ["phi4", { contextWindow: 16384, maxTokens: 16384, ollamaOptions: { num_ctx: 16384 } }],
49
+ ["phi3.5", { contextWindow: 131072, maxTokens: 16384, ollamaOptions: { num_ctx: 131072 } }],
50
+ ["phi3", { contextWindow: 131072, maxTokens: 4096, ollamaOptions: { num_ctx: 131072 } }],
48
51
  // ─── Command R ──────────────────────────────────────────────────────
49
- ["command-r-plus", { contextWindow: 131072, maxTokens: 16384 }],
50
- ["command-r", { contextWindow: 131072, maxTokens: 16384 }],
52
+ ["command-r-plus", { contextWindow: 131072, maxTokens: 16384, ollamaOptions: { num_ctx: 131072 } }],
53
+ ["command-r", { contextWindow: 131072, maxTokens: 16384, ollamaOptions: { num_ctx: 131072 } }],
51
54
  ];
52
55
  /**
53
56
  * Look up capabilities for a model by name.
@@ -0,0 +1,54 @@
1
+ // GSD2 — Ollama Extension: NDJSON streaming parser
2
+ /**
3
+ * Parses a streaming NDJSON (newline-delimited JSON) response body into
4
+ * typed objects. Used for Ollama's /api/chat and /api/pull endpoints.
5
+ *
6
+ * @param strict When true, malformed JSON lines throw instead of being skipped.
7
+ * Use strict mode for inference streams where silent data loss is unacceptable.
8
+ * Use permissive mode (default) for progress endpoints like /api/pull.
9
+ */
10
+ export async function* parseNDJsonStream(body, signal, strict = false) {
11
+ const reader = body.getReader();
12
+ const decoder = new TextDecoder();
13
+ let buffer = "";
14
+ try {
15
+ while (true) {
16
+ if (signal?.aborted)
17
+ break;
18
+ const { done, value } = await reader.read();
19
+ if (done)
20
+ break;
21
+ buffer += decoder.decode(value, { stream: true });
22
+ const lines = buffer.split("\n");
23
+ buffer = lines.pop() ?? "";
24
+ for (const line of lines) {
25
+ const trimmed = line.trim();
26
+ if (!trimmed)
27
+ continue;
28
+ try {
29
+ yield JSON.parse(trimmed);
30
+ }
31
+ catch (err) {
32
+ if (strict) {
33
+ throw new Error(`Malformed NDJSON line from Ollama: ${trimmed.slice(0, 200)}`);
34
+ }
35
+ // Permissive mode: skip malformed lines
36
+ }
37
+ }
38
+ }
39
+ // Flush remaining buffer (skip if aborted)
40
+ if (buffer.trim() && !signal?.aborted) {
41
+ try {
42
+ yield JSON.parse(buffer.trim());
43
+ }
44
+ catch (err) {
45
+ if (strict) {
46
+ throw new Error(`Malformed NDJSON line from Ollama: ${buffer.trim().slice(0, 200)}`);
47
+ }
48
+ }
49
+ }
50
+ }
51
+ finally {
52
+ reader.releaseLock();
53
+ }
54
+ }
@@ -0,0 +1,380 @@
1
+ // GSD2 — Ollama Extension: Native /api/chat stream provider
2
+ /**
3
+ * Implements the "ollama-chat" API provider, streaming responses directly
4
+ * from Ollama's native /api/chat endpoint instead of the OpenAI compatibility
5
+ * shim. This exposes Ollama-specific options (num_ctx, keep_alive, num_gpu,
6
+ * sampling parameters) and surfaces inference performance metrics.
7
+ */
8
+ import { EventStream, } from "@gsd/pi-ai";
9
+ import { chat } from "./ollama-client.js";
10
+ import { ThinkingTagParser } from "./thinking-parser.js";
11
+ /** Create an AssistantMessageEventStream using the base EventStream class. */
12
+ function createStream() {
13
+ return new EventStream((event) => event.type === "done" || event.type === "error", (event) => {
14
+ if (event.type === "done")
15
+ return event.message;
16
+ if (event.type === "error")
17
+ return event.error;
18
+ throw new Error("Unexpected event type for final result");
19
+ });
20
+ }
21
+ // ─── Stream handler ─────────────────────────────────────────────────────────
22
+ export function streamOllamaChat(model, context, options) {
23
+ const stream = createStream();
24
+ (async () => {
25
+ const output = buildInitialOutput(model);
26
+ try {
27
+ const request = buildRequest(model, context, options);
28
+ stream.push({ type: "start", partial: output });
29
+ const useThinkingParser = model.reasoning;
30
+ const thinkParser = useThinkingParser ? new ThinkingTagParser() : null;
31
+ let contentIndex = -1;
32
+ let currentBlockType = null;
33
+ function startBlock(type) {
34
+ contentIndex++;
35
+ currentBlockType = type;
36
+ if (type === "text") {
37
+ output.content.push({ type: "text", text: "" });
38
+ stream.push({ type: "text_start", contentIndex, partial: output });
39
+ }
40
+ else {
41
+ output.content.push({ type: "thinking", thinking: "" });
42
+ stream.push({ type: "thinking_start", contentIndex, partial: output });
43
+ }
44
+ }
45
+ function endBlock() {
46
+ if (currentBlockType === null)
47
+ return;
48
+ if (currentBlockType === "text") {
49
+ const block = output.content[contentIndex];
50
+ stream.push({ type: "text_end", contentIndex, content: block.text, partial: output });
51
+ }
52
+ else {
53
+ const block = output.content[contentIndex];
54
+ stream.push({ type: "thinking_end", contentIndex, content: block.thinking, partial: output });
55
+ }
56
+ currentBlockType = null;
57
+ }
58
+ function emitDelta(type, text) {
59
+ if (!text)
60
+ return;
61
+ if (currentBlockType !== type) {
62
+ endBlock();
63
+ startBlock(type);
64
+ }
65
+ if (type === "text") {
66
+ output.content[contentIndex].text += text;
67
+ stream.push({ type: "text_delta", contentIndex, delta: text, partial: output });
68
+ }
69
+ else {
70
+ output.content[contentIndex].thinking += text;
71
+ stream.push({ type: "thinking_delta", contentIndex, delta: text, partial: output });
72
+ }
73
+ }
74
+ function processChunks(chunks) {
75
+ for (const chunk of chunks) {
76
+ emitDelta(chunk.type, chunk.text);
77
+ }
78
+ }
79
+ function processToolCalls(toolCalls) {
80
+ endBlock();
81
+ for (const tc of toolCalls) {
82
+ contentIndex++;
83
+ const toolCall = {
84
+ type: "toolCall",
85
+ id: `ollama_tc_${contentIndex}`,
86
+ name: tc.function.name,
87
+ arguments: tc.function.arguments,
88
+ };
89
+ output.content.push(toolCall);
90
+ stream.push({ type: "toolcall_start", contentIndex, partial: output });
91
+ // Emit a delta with the serialized arguments (convention: start/delta/end)
92
+ stream.push({
93
+ type: "toolcall_delta",
94
+ contentIndex,
95
+ delta: JSON.stringify(tc.function.arguments),
96
+ partial: output,
97
+ });
98
+ stream.push({
99
+ type: "toolcall_end",
100
+ contentIndex,
101
+ toolCall,
102
+ partial: output,
103
+ });
104
+ }
105
+ output.stopReason = "toolUse";
106
+ }
107
+ for await (const chunk of chat(request, options?.signal)) {
108
+ // Handle text content — process independently of tool_calls
109
+ // (a chunk may contain both content and tool_calls)
110
+ const content = chunk.message?.content ?? "";
111
+ if (content) {
112
+ if (thinkParser) {
113
+ processChunks(thinkParser.push(content));
114
+ }
115
+ else {
116
+ emitDelta("text", content);
117
+ }
118
+ }
119
+ // Handle tool calls (Ollama sends them complete, may be on done:true chunk)
120
+ if (chunk.message?.tool_calls?.length) {
121
+ processToolCalls(chunk.message.tool_calls);
122
+ }
123
+ if (chunk.done) {
124
+ // Final chunk — extract metrics and usage
125
+ if (thinkParser)
126
+ processChunks(thinkParser.flush());
127
+ endBlock();
128
+ output.usage = buildUsage(chunk);
129
+ output.inferenceMetrics = extractMetrics(chunk);
130
+ // Preserve "toolUse" if tool calls were processed
131
+ if (output.stopReason !== "toolUse") {
132
+ output.stopReason = mapStopReason(chunk.done_reason);
133
+ }
134
+ break;
135
+ }
136
+ }
137
+ assertStreamSuccess(output, options?.signal);
138
+ finalizeStream(stream, output);
139
+ }
140
+ catch (error) {
141
+ handleStreamError(stream, output, error, options?.signal);
142
+ }
143
+ })();
144
+ return stream;
145
+ }
146
+ // ─── Request building ───────────────────────────────────────────────────────
147
+ function buildRequest(model, context, options) {
148
+ const ollamaOpts = (model.providerOptions ?? {});
149
+ const request = {
150
+ model: model.id,
151
+ messages: convertMessages(context),
152
+ stream: true,
153
+ };
154
+ // Build options block with all Ollama-specific parameters
155
+ const reqOptions = {};
156
+ // Context window — only sent when explicitly configured via providerOptions.
157
+ // Sending inferred/estimated values risks OOM on constrained hosts.
158
+ // Users can set num_ctx per-model in models.json ollamaOptions or the
159
+ // capability table can provide it for known model families.
160
+ if (ollamaOpts.num_ctx !== undefined && ollamaOpts.num_ctx > 0) {
161
+ reqOptions.num_ctx = ollamaOpts.num_ctx;
162
+ }
163
+ // Max output tokens
164
+ const maxTokens = options?.maxTokens ?? model.maxTokens;
165
+ if (maxTokens > 0) {
166
+ reqOptions.num_predict = maxTokens;
167
+ }
168
+ // Temperature
169
+ if (options?.temperature !== undefined) {
170
+ reqOptions.temperature = options.temperature;
171
+ }
172
+ // Per-model sampling options from providerOptions
173
+ if (ollamaOpts.top_p !== undefined)
174
+ reqOptions.top_p = ollamaOpts.top_p;
175
+ if (ollamaOpts.top_k !== undefined)
176
+ reqOptions.top_k = ollamaOpts.top_k;
177
+ if (ollamaOpts.repeat_penalty !== undefined)
178
+ reqOptions.repeat_penalty = ollamaOpts.repeat_penalty;
179
+ if (ollamaOpts.seed !== undefined)
180
+ reqOptions.seed = ollamaOpts.seed;
181
+ if (ollamaOpts.num_gpu !== undefined)
182
+ reqOptions.num_gpu = ollamaOpts.num_gpu;
183
+ if (Object.keys(reqOptions).length > 0) {
184
+ request.options = reqOptions;
185
+ }
186
+ // Keep alive
187
+ if (ollamaOpts.keep_alive !== undefined) {
188
+ request.keep_alive = ollamaOpts.keep_alive;
189
+ }
190
+ // Tools
191
+ if (context.tools?.length) {
192
+ request.tools = convertTools(context.tools);
193
+ }
194
+ return request;
195
+ }
196
+ // ─── Message conversion ─────────────────────────────────────────────────────
197
+ function convertMessages(context) {
198
+ const messages = [];
199
+ // System prompt
200
+ if (context.systemPrompt) {
201
+ messages.push({ role: "system", content: context.systemPrompt });
202
+ }
203
+ for (const msg of context.messages) {
204
+ switch (msg.role) {
205
+ case "user":
206
+ messages.push(convertUserMessage(msg));
207
+ break;
208
+ case "assistant":
209
+ messages.push(convertAssistantMessage(msg));
210
+ break;
211
+ case "toolResult":
212
+ messages.push({
213
+ role: "tool",
214
+ content: msg.content
215
+ .filter((c) => c.type === "text")
216
+ .map((c) => c.text)
217
+ .join("\n"),
218
+ name: msg.toolName,
219
+ });
220
+ break;
221
+ }
222
+ }
223
+ return messages;
224
+ }
225
+ function convertUserMessage(msg) {
226
+ if (typeof msg.content === "string") {
227
+ return { role: "user", content: msg.content };
228
+ }
229
+ const textParts = [];
230
+ const images = [];
231
+ for (const part of msg.content) {
232
+ if (part.type === "text") {
233
+ textParts.push(part.text);
234
+ }
235
+ else if (part.type === "image") {
236
+ // Strip data URI prefix if present
237
+ let data = part.data;
238
+ const commaIdx = data.indexOf(",");
239
+ if (commaIdx !== -1 && data.startsWith("data:")) {
240
+ data = data.slice(commaIdx + 1);
241
+ }
242
+ images.push(data);
243
+ }
244
+ }
245
+ const result = {
246
+ role: "user",
247
+ content: textParts.join("\n"),
248
+ };
249
+ if (images.length > 0) {
250
+ result.images = images;
251
+ }
252
+ return result;
253
+ }
254
+ function convertAssistantMessage(msg) {
255
+ let content = "";
256
+ const toolCalls = [];
257
+ for (const block of msg.content) {
258
+ if (block.type === "thinking") {
259
+ // Serialize thinking back inline for round-trip with Ollama
260
+ content += `<think>${block.thinking}</think>`;
261
+ }
262
+ else if (block.type === "text") {
263
+ content += block.text;
264
+ }
265
+ else if (block.type === "toolCall") {
266
+ const tc = block;
267
+ toolCalls.push({
268
+ function: {
269
+ name: tc.name,
270
+ arguments: tc.arguments,
271
+ },
272
+ });
273
+ }
274
+ }
275
+ const result = { role: "assistant", content };
276
+ if (toolCalls.length > 0) {
277
+ result.tool_calls = toolCalls;
278
+ }
279
+ return result;
280
+ }
281
+ // ─── Tool conversion ────────────────────────────────────────────────────────
282
+ function convertTools(tools) {
283
+ return tools.map((tool) => {
284
+ const params = tool.parameters;
285
+ return {
286
+ type: "function",
287
+ function: {
288
+ name: tool.name,
289
+ description: tool.description,
290
+ parameters: {
291
+ type: "object",
292
+ required: params.required,
293
+ properties: params.properties ?? {},
294
+ },
295
+ },
296
+ };
297
+ });
298
+ }
299
+ // ─── Response mapping ───────────────────────────────────────────────────────
300
+ function mapStopReason(doneReason) {
301
+ switch (doneReason) {
302
+ case "stop":
303
+ return "stop";
304
+ case "length":
305
+ return "length";
306
+ default:
307
+ return "stop";
308
+ }
309
+ }
310
+ function buildUsage(chunk) {
311
+ const input = chunk.prompt_eval_count ?? 0;
312
+ const outputTokens = chunk.eval_count ?? 0;
313
+ return {
314
+ input,
315
+ output: outputTokens,
316
+ cacheRead: 0,
317
+ cacheWrite: 0,
318
+ totalTokens: input + outputTokens,
319
+ cost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0, total: 0 },
320
+ };
321
+ }
322
+ function extractMetrics(chunk) {
323
+ if (!chunk.eval_duration && !chunk.total_duration)
324
+ return undefined;
325
+ const evalCount = chunk.eval_count ?? 0;
326
+ const evalDurationNs = chunk.eval_duration ?? 0;
327
+ const evalDurationMs = evalDurationNs / 1e6;
328
+ const tokensPerSecond = evalDurationNs > 0 ? evalCount / (evalDurationNs / 1e9) : 0;
329
+ return {
330
+ tokensPerSecond,
331
+ totalDurationMs: (chunk.total_duration ?? 0) / 1e6,
332
+ evalDurationMs,
333
+ promptEvalDurationMs: (chunk.prompt_eval_duration ?? 0) / 1e6,
334
+ };
335
+ }
336
+ // ─── Stream lifecycle helpers ───────────────────────────────────────────────
337
+ // Replicated from openai-shared.ts (not exported from @gsd/pi-ai)
338
+ function buildInitialOutput(model) {
339
+ return {
340
+ role: "assistant",
341
+ content: [],
342
+ api: model.api,
343
+ provider: model.provider,
344
+ model: model.id,
345
+ usage: {
346
+ input: 0,
347
+ output: 0,
348
+ cacheRead: 0,
349
+ cacheWrite: 0,
350
+ totalTokens: 0,
351
+ cost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0, total: 0 },
352
+ },
353
+ stopReason: "stop",
354
+ timestamp: Date.now(),
355
+ };
356
+ }
357
+ function assertStreamSuccess(output, signal) {
358
+ if (signal?.aborted) {
359
+ throw new Error("Request was aborted");
360
+ }
361
+ if (output.stopReason === "aborted" || output.stopReason === "error") {
362
+ throw new Error("An unknown error occurred");
363
+ }
364
+ }
365
+ function finalizeStream(stream, output) {
366
+ stream.push({
367
+ type: "done",
368
+ reason: output.stopReason,
369
+ message: output,
370
+ });
371
+ stream.end();
372
+ }
373
+ function handleStreamError(stream, output, error, signal) {
374
+ for (const block of output.content)
375
+ delete block.index;
376
+ output.stopReason = signal?.aborted ? "aborted" : "error";
377
+ output.errorMessage = error instanceof Error ? error.message : JSON.stringify(error);
378
+ stream.push({ type: "error", reason: output.stopReason, error: output });
379
+ stream.end();
380
+ }