gsd-pi 2.74.0-dev.2b524c3 → 2.74.0-dev.703eabc

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 (484) hide show
  1. package/dist/cli.js +85 -0
  2. package/dist/headless-query.js +4 -1
  3. package/dist/help-text.js +23 -0
  4. package/dist/resources/extensions/gsd/activity-log.js +16 -0
  5. package/dist/resources/extensions/gsd/auto/detect-stuck.js +11 -4
  6. package/dist/resources/extensions/gsd/auto/loop.js +147 -10
  7. package/dist/resources/extensions/gsd/auto/phases.js +158 -4
  8. package/dist/resources/extensions/gsd/auto/session.js +10 -0
  9. package/dist/resources/extensions/gsd/auto-dispatch.js +11 -1
  10. package/dist/resources/extensions/gsd/auto-model-selection.js +51 -5
  11. package/dist/resources/extensions/gsd/auto-post-unit.js +220 -17
  12. package/dist/resources/extensions/gsd/auto-prompts.js +12 -0
  13. package/dist/resources/extensions/gsd/auto-unit-closeout.js +18 -0
  14. package/dist/resources/extensions/gsd/auto-verification.js +100 -2
  15. package/dist/resources/extensions/gsd/auto.js +36 -4
  16. package/dist/resources/extensions/gsd/bootstrap/register-extension.js +30 -8
  17. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +45 -4
  18. package/dist/resources/extensions/gsd/commands/catalog.js +26 -1
  19. package/dist/resources/extensions/gsd/commands/handlers/ops.js +25 -0
  20. package/dist/resources/extensions/gsd/commands/handlers/workflow.js +68 -9
  21. package/dist/resources/extensions/gsd/commands-add-tests.js +111 -0
  22. package/dist/resources/extensions/gsd/commands-backlog.js +140 -0
  23. package/dist/resources/extensions/gsd/commands-do.js +79 -0
  24. package/dist/resources/extensions/gsd/commands-extract-learnings.js +225 -0
  25. package/dist/resources/extensions/gsd/commands-maintenance.js +6 -6
  26. package/dist/resources/extensions/gsd/commands-pr-branch.js +180 -0
  27. package/dist/resources/extensions/gsd/commands-prefs-wizard.js +1 -1
  28. package/dist/resources/extensions/gsd/commands-session-report.js +82 -0
  29. package/dist/resources/extensions/gsd/commands-ship.js +187 -0
  30. package/dist/resources/extensions/gsd/db-writer.js +3 -5
  31. package/dist/resources/extensions/gsd/docs/preferences-reference.md +14 -1
  32. package/dist/resources/extensions/gsd/ecosystem/gsd-extension-api.js +144 -0
  33. package/dist/resources/extensions/gsd/ecosystem/loader.js +145 -0
  34. package/dist/resources/extensions/gsd/git-service.js +49 -1
  35. package/dist/resources/extensions/gsd/graph-context.js +157 -0
  36. package/dist/resources/extensions/gsd/gsd-db.js +581 -2
  37. package/dist/resources/extensions/gsd/guided-flow.js +23 -0
  38. package/dist/resources/extensions/gsd/index.js +15 -2
  39. package/dist/resources/extensions/gsd/init-wizard.js +1 -0
  40. package/dist/resources/extensions/gsd/journal.js +27 -0
  41. package/dist/resources/extensions/gsd/md-importer.js +3 -4
  42. package/dist/resources/extensions/gsd/memory-store.js +19 -51
  43. package/dist/resources/extensions/gsd/metrics.js +19 -0
  44. package/dist/resources/extensions/gsd/milestone-validation-gates.js +13 -12
  45. package/dist/resources/extensions/gsd/native-git-bridge.js +7 -4
  46. package/dist/resources/extensions/gsd/parallel-orchestrator.js +33 -1
  47. package/dist/resources/extensions/gsd/preferences-models.js +20 -3
  48. package/dist/resources/extensions/gsd/preferences-types.js +1 -0
  49. package/dist/resources/extensions/gsd/preferences-validation.js +108 -2
  50. package/dist/resources/extensions/gsd/preferences.js +26 -0
  51. package/dist/resources/extensions/gsd/prompts/add-tests.md +35 -0
  52. package/dist/resources/extensions/gsd/slice-parallel-orchestrator.js +12 -2
  53. package/dist/resources/extensions/gsd/state.js +5 -1
  54. package/dist/resources/extensions/gsd/templates/PREFERENCES.md +18 -0
  55. package/dist/resources/extensions/gsd/tools/complete-slice.js +20 -0
  56. package/dist/resources/extensions/gsd/tools/validate-milestone.js +39 -4
  57. package/dist/resources/extensions/gsd/tools/workflow-tool-executors.js +3 -14
  58. package/dist/resources/extensions/gsd/triage-resolution.js +2 -5
  59. package/dist/resources/extensions/gsd/unit-ownership.js +1 -1
  60. package/dist/resources/extensions/gsd/uok/audit-toggle.js +7 -0
  61. package/dist/resources/extensions/gsd/uok/audit.js +40 -0
  62. package/dist/resources/extensions/gsd/uok/contracts.js +1 -0
  63. package/dist/resources/extensions/gsd/uok/execution-graph.js +179 -0
  64. package/dist/resources/extensions/gsd/uok/flags.js +29 -0
  65. package/dist/resources/extensions/gsd/uok/gate-runner.js +109 -0
  66. package/dist/resources/extensions/gsd/uok/gitops.js +53 -0
  67. package/dist/resources/extensions/gsd/uok/kernel.js +80 -0
  68. package/dist/resources/extensions/gsd/uok/loop-adapter.js +133 -0
  69. package/dist/resources/extensions/gsd/uok/model-policy.js +66 -0
  70. package/dist/resources/extensions/gsd/uok/plan-v2.js +132 -0
  71. package/dist/resources/extensions/gsd/workflow-logger.js +22 -0
  72. package/dist/resources/extensions/gsd/workflow-manifest.js +8 -69
  73. package/dist/resources/extensions/gsd/workflow-migration.js +21 -22
  74. package/dist/resources/extensions/gsd/workflow-projections.js +4 -1
  75. package/dist/resources/extensions/gsd/workflow-reconcile.js +14 -11
  76. package/dist/resources/extensions/ttsr/ttsr-manager.js +3 -1
  77. package/dist/tsconfig.extensions.tsbuildinfo +1 -0
  78. package/dist/web/standalone/.next/BUILD_ID +1 -1
  79. package/dist/web/standalone/.next/app-path-routes-manifest.json +12 -12
  80. package/dist/web/standalone/.next/build-manifest.json +2 -2
  81. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  82. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  83. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  84. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  85. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  86. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  87. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  88. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  89. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  90. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  91. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  92. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  93. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  94. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  95. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  96. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  97. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  98. package/dist/web/standalone/.next/server/app/api/onboarding/route.js +1 -1
  99. package/dist/web/standalone/.next/server/app/api/terminal/input/route.js +1 -1
  100. package/dist/web/standalone/.next/server/app/api/terminal/resize/route.js +1 -1
  101. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js +1 -1
  102. package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js +1 -1
  103. package/dist/web/standalone/.next/server/app/index.html +1 -1
  104. package/dist/web/standalone/.next/server/app/index.rsc +1 -1
  105. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  106. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
  107. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  108. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  109. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  110. package/dist/web/standalone/.next/server/app-paths-manifest.json +12 -12
  111. package/dist/web/standalone/.next/server/chunks/6897.js +3 -3
  112. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  113. package/dist/web/standalone/.next/server/middleware-manifest.json +5 -5
  114. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  115. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  116. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  117. package/package.json +3 -2
  118. package/packages/daemon/package.json +2 -2
  119. package/packages/mcp-server/dist/index.d.ts +3 -0
  120. package/packages/mcp-server/dist/index.d.ts.map +1 -1
  121. package/packages/mcp-server/dist/index.js +3 -0
  122. package/packages/mcp-server/dist/index.js.map +1 -1
  123. package/packages/mcp-server/dist/readers/graph.d.ts +87 -0
  124. package/packages/mcp-server/dist/readers/graph.d.ts.map +1 -0
  125. package/packages/mcp-server/dist/readers/graph.js +655 -0
  126. package/packages/mcp-server/dist/readers/graph.js.map +1 -0
  127. package/packages/mcp-server/dist/readers/index.d.ts +2 -0
  128. package/packages/mcp-server/dist/readers/index.d.ts.map +1 -1
  129. package/packages/mcp-server/dist/readers/index.js +1 -0
  130. package/packages/mcp-server/dist/readers/index.js.map +1 -1
  131. package/packages/mcp-server/dist/server.d.ts.map +1 -1
  132. package/packages/mcp-server/dist/server.js +65 -0
  133. package/packages/mcp-server/dist/server.js.map +1 -1
  134. package/packages/mcp-server/package.json +2 -2
  135. package/packages/mcp-server/src/index.ts +15 -0
  136. package/packages/mcp-server/src/readers/graph.test.ts +604 -0
  137. package/packages/mcp-server/src/readers/graph.ts +855 -0
  138. package/packages/mcp-server/src/readers/index.ts +12 -0
  139. package/packages/mcp-server/src/server.ts +83 -0
  140. package/packages/mcp-server/tsconfig.json +1 -0
  141. package/packages/mcp-server/tsconfig.tsbuildinfo +1 -0
  142. package/packages/native/package.json +2 -2
  143. package/packages/native/tsconfig.tsbuildinfo +1 -0
  144. package/packages/pi-agent-core/package.json +1 -1
  145. package/packages/pi-agent-core/tsconfig.json +1 -0
  146. package/packages/pi-agent-core/tsconfig.tsbuildinfo +1 -0
  147. package/packages/pi-ai/dist/index.d.ts +1 -9
  148. package/packages/pi-ai/dist/index.d.ts.map +1 -1
  149. package/packages/pi-ai/dist/index.js +1 -9
  150. package/packages/pi-ai/dist/index.js.map +1 -1
  151. package/packages/pi-ai/dist/models/capability-patches.d.ts +19 -0
  152. package/packages/pi-ai/dist/models/capability-patches.d.ts.map +1 -0
  153. package/packages/pi-ai/dist/models/capability-patches.js +36 -0
  154. package/packages/pi-ai/dist/models/capability-patches.js.map +1 -0
  155. package/packages/pi-ai/dist/{models.custom.d.ts → models/custom.d.ts} +1 -1
  156. package/packages/pi-ai/dist/models/custom.d.ts.map +1 -0
  157. package/packages/pi-ai/dist/{models.custom.js → models/custom.js} +4 -4
  158. package/packages/pi-ai/dist/models/custom.js.map +1 -0
  159. package/packages/pi-ai/dist/models/generated/amazon-bedrock.d.ts +1482 -0
  160. package/packages/pi-ai/dist/models/generated/amazon-bedrock.d.ts.map +1 -0
  161. package/packages/pi-ai/dist/models/generated/amazon-bedrock.js +1484 -0
  162. package/packages/pi-ai/dist/models/generated/amazon-bedrock.js.map +1 -0
  163. package/packages/pi-ai/dist/models/generated/anthropic.d.ts +377 -0
  164. package/packages/pi-ai/dist/models/generated/anthropic.d.ts.map +1 -0
  165. package/packages/pi-ai/dist/models/generated/anthropic.js +379 -0
  166. package/packages/pi-ai/dist/models/generated/anthropic.js.map +1 -0
  167. package/packages/pi-ai/dist/models/generated/azure-openai-responses.d.ts +700 -0
  168. package/packages/pi-ai/dist/models/generated/azure-openai-responses.d.ts.map +1 -0
  169. package/packages/pi-ai/dist/models/generated/azure-openai-responses.js +702 -0
  170. package/packages/pi-ai/dist/models/generated/azure-openai-responses.js.map +1 -0
  171. package/packages/pi-ai/dist/models/generated/cerebras.d.ts +71 -0
  172. package/packages/pi-ai/dist/models/generated/cerebras.d.ts.map +1 -0
  173. package/packages/pi-ai/dist/models/generated/cerebras.js +73 -0
  174. package/packages/pi-ai/dist/models/generated/cerebras.js.map +1 -0
  175. package/packages/pi-ai/dist/models/generated/github-copilot.d.ts +590 -0
  176. package/packages/pi-ai/dist/models/generated/github-copilot.d.ts.map +1 -0
  177. package/packages/pi-ai/dist/models/generated/github-copilot.js +444 -0
  178. package/packages/pi-ai/dist/models/generated/github-copilot.js.map +1 -0
  179. package/packages/pi-ai/dist/models/generated/google-antigravity.d.ts +156 -0
  180. package/packages/pi-ai/dist/models/generated/google-antigravity.d.ts.map +1 -0
  181. package/packages/pi-ai/dist/models/generated/google-antigravity.js +158 -0
  182. package/packages/pi-ai/dist/models/generated/google-antigravity.js.map +1 -0
  183. package/packages/pi-ai/dist/models/generated/google-gemini-cli.d.ts +105 -0
  184. package/packages/pi-ai/dist/models/generated/google-gemini-cli.d.ts.map +1 -0
  185. package/packages/pi-ai/dist/models/generated/google-gemini-cli.js +107 -0
  186. package/packages/pi-ai/dist/models/generated/google-gemini-cli.js.map +1 -0
  187. package/packages/pi-ai/dist/models/generated/google-vertex.d.ts +207 -0
  188. package/packages/pi-ai/dist/models/generated/google-vertex.d.ts.map +1 -0
  189. package/packages/pi-ai/dist/models/generated/google-vertex.js +209 -0
  190. package/packages/pi-ai/dist/models/generated/google-vertex.js.map +1 -0
  191. package/packages/pi-ai/dist/models/generated/google.d.ts +462 -0
  192. package/packages/pi-ai/dist/models/generated/google.d.ts.map +1 -0
  193. package/packages/pi-ai/dist/models/generated/google.js +464 -0
  194. package/packages/pi-ai/dist/models/generated/google.js.map +1 -0
  195. package/packages/pi-ai/dist/models/generated/groq.d.ts +309 -0
  196. package/packages/pi-ai/dist/models/generated/groq.d.ts.map +1 -0
  197. package/packages/pi-ai/dist/models/generated/groq.js +311 -0
  198. package/packages/pi-ai/dist/models/generated/groq.js.map +1 -0
  199. package/packages/pi-ai/dist/models/generated/huggingface.d.ts +383 -0
  200. package/packages/pi-ai/dist/models/generated/huggingface.d.ts.map +1 -0
  201. package/packages/pi-ai/dist/models/generated/huggingface.js +347 -0
  202. package/packages/pi-ai/dist/models/generated/huggingface.js.map +1 -0
  203. package/packages/pi-ai/dist/{models.generated.d.ts → models/generated/index.d.ts} +1 -1
  204. package/packages/pi-ai/dist/{models.generated.d.ts.map → models/generated/index.d.ts.map} +1 -1
  205. package/packages/pi-ai/dist/models/generated/index.js +51 -0
  206. package/packages/pi-ai/dist/models/generated/index.js.map +1 -0
  207. package/packages/pi-ai/dist/models/generated/kimi-coding.d.ts +37 -0
  208. package/packages/pi-ai/dist/models/generated/kimi-coding.d.ts.map +1 -0
  209. package/packages/pi-ai/dist/models/generated/kimi-coding.js +39 -0
  210. package/packages/pi-ai/dist/models/generated/kimi-coding.js.map +1 -0
  211. package/packages/pi-ai/dist/models/generated/minimax-cn.d.ts +105 -0
  212. package/packages/pi-ai/dist/models/generated/minimax-cn.d.ts.map +1 -0
  213. package/packages/pi-ai/dist/models/generated/minimax-cn.js +107 -0
  214. package/packages/pi-ai/dist/models/generated/minimax-cn.js.map +1 -0
  215. package/packages/pi-ai/dist/models/generated/minimax.d.ts +105 -0
  216. package/packages/pi-ai/dist/models/generated/minimax.d.ts.map +1 -0
  217. package/packages/pi-ai/dist/models/generated/minimax.js +107 -0
  218. package/packages/pi-ai/dist/models/generated/minimax.js.map +1 -0
  219. package/packages/pi-ai/dist/models/generated/mistral.d.ts +445 -0
  220. package/packages/pi-ai/dist/models/generated/mistral.d.ts.map +1 -0
  221. package/packages/pi-ai/dist/models/generated/mistral.js +447 -0
  222. package/packages/pi-ai/dist/models/generated/mistral.js.map +1 -0
  223. package/packages/pi-ai/dist/models/generated/openai-codex.d.ts +139 -0
  224. package/packages/pi-ai/dist/models/generated/openai-codex.d.ts.map +1 -0
  225. package/packages/pi-ai/dist/models/generated/openai-codex.js +141 -0
  226. package/packages/pi-ai/dist/models/generated/openai-codex.js.map +1 -0
  227. package/packages/pi-ai/dist/models/generated/openai.d.ts +700 -0
  228. package/packages/pi-ai/dist/models/generated/openai.d.ts.map +1 -0
  229. package/packages/pi-ai/dist/models/generated/openai.js +702 -0
  230. package/packages/pi-ai/dist/models/generated/openai.js.map +1 -0
  231. package/packages/pi-ai/dist/models/generated/opencode-go.d.ts +122 -0
  232. package/packages/pi-ai/dist/models/generated/opencode-go.d.ts.map +1 -0
  233. package/packages/pi-ai/dist/models/generated/opencode-go.js +124 -0
  234. package/packages/pi-ai/dist/models/generated/opencode-go.js.map +1 -0
  235. package/packages/pi-ai/dist/models/generated/opencode.d.ts +530 -0
  236. package/packages/pi-ai/dist/models/generated/opencode.d.ts.map +1 -0
  237. package/packages/pi-ai/dist/models/generated/opencode.js +532 -0
  238. package/packages/pi-ai/dist/models/generated/opencode.js.map +1 -0
  239. package/packages/pi-ai/dist/models/generated/openrouter.d.ts +4270 -0
  240. package/packages/pi-ai/dist/models/generated/openrouter.d.ts.map +1 -0
  241. package/packages/pi-ai/dist/models/generated/openrouter.js +4272 -0
  242. package/packages/pi-ai/dist/models/generated/openrouter.js.map +1 -0
  243. package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.d.ts +2604 -0
  244. package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.d.ts.map +1 -0
  245. package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.js +2606 -0
  246. package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.js.map +1 -0
  247. package/packages/pi-ai/dist/models/generated/xai.d.ts +411 -0
  248. package/packages/pi-ai/dist/models/generated/xai.d.ts.map +1 -0
  249. package/packages/pi-ai/dist/models/generated/xai.js +413 -0
  250. package/packages/pi-ai/dist/models/generated/xai.js.map +1 -0
  251. package/packages/pi-ai/dist/models/generated/zai.d.ts +276 -0
  252. package/packages/pi-ai/dist/models/generated/zai.d.ts.map +1 -0
  253. package/packages/pi-ai/dist/models/generated/zai.js +239 -0
  254. package/packages/pi-ai/dist/models/generated/zai.js.map +1 -0
  255. package/packages/pi-ai/dist/models/index.d.ts +27 -0
  256. package/packages/pi-ai/dist/models/index.d.ts.map +1 -0
  257. package/packages/pi-ai/dist/models/index.js +80 -0
  258. package/packages/pi-ai/dist/models/index.js.map +1 -0
  259. package/packages/pi-ai/dist/models.d.ts +1 -36
  260. package/packages/pi-ai/dist/models.d.ts.map +1 -1
  261. package/packages/pi-ai/dist/models.generated.test.js +1 -2
  262. package/packages/pi-ai/dist/models.generated.test.js.map +1 -1
  263. package/packages/pi-ai/dist/models.js +3 -112
  264. package/packages/pi-ai/dist/models.js.map +1 -1
  265. package/packages/pi-ai/dist/models.test.js +6 -5
  266. package/packages/pi-ai/dist/models.test.js.map +1 -1
  267. package/packages/pi-ai/package.json +1 -1
  268. package/packages/pi-ai/scripts/generate-models.ts +74 -40
  269. package/packages/pi-ai/src/index.ts +1 -9
  270. package/packages/pi-ai/src/models/capability-patches.ts +40 -0
  271. package/packages/pi-ai/src/{models.custom.ts → models/custom.ts} +4 -4
  272. package/packages/pi-ai/src/models/generated/amazon-bedrock.ts +1486 -0
  273. package/packages/pi-ai/src/models/generated/anthropic.ts +381 -0
  274. package/packages/pi-ai/src/models/generated/azure-openai-responses.ts +704 -0
  275. package/packages/pi-ai/src/models/generated/cerebras.ts +75 -0
  276. package/packages/pi-ai/src/models/generated/github-copilot.ts +446 -0
  277. package/packages/pi-ai/src/models/generated/google-antigravity.ts +160 -0
  278. package/packages/pi-ai/src/models/generated/google-gemini-cli.ts +109 -0
  279. package/packages/pi-ai/src/models/generated/google-vertex.ts +211 -0
  280. package/packages/pi-ai/src/models/generated/google.ts +466 -0
  281. package/packages/pi-ai/src/models/generated/groq.ts +313 -0
  282. package/packages/pi-ai/src/models/generated/huggingface.ts +349 -0
  283. package/packages/pi-ai/src/models/generated/index.ts +52 -0
  284. package/packages/pi-ai/src/models/generated/kimi-coding.ts +41 -0
  285. package/packages/pi-ai/src/models/generated/minimax-cn.ts +109 -0
  286. package/packages/pi-ai/src/models/generated/minimax.ts +109 -0
  287. package/packages/pi-ai/src/models/generated/mistral.ts +449 -0
  288. package/packages/pi-ai/src/models/generated/openai-codex.ts +143 -0
  289. package/packages/pi-ai/src/models/generated/openai.ts +704 -0
  290. package/packages/pi-ai/src/models/generated/opencode-go.ts +126 -0
  291. package/packages/pi-ai/src/models/generated/opencode.ts +534 -0
  292. package/packages/pi-ai/src/models/generated/openrouter.ts +4274 -0
  293. package/packages/pi-ai/src/models/generated/vercel-ai-gateway.ts +2608 -0
  294. package/packages/pi-ai/src/models/generated/xai.ts +415 -0
  295. package/packages/pi-ai/src/models/generated/zai.ts +241 -0
  296. package/packages/pi-ai/src/models/index.ts +106 -0
  297. package/packages/pi-ai/src/models.generated.test.ts +1 -2
  298. package/packages/pi-ai/src/models.test.ts +6 -5
  299. package/packages/pi-ai/src/models.ts +3 -153
  300. package/packages/pi-ai/tsconfig.json +1 -0
  301. package/packages/pi-ai/tsconfig.tsbuildinfo +1 -0
  302. package/packages/pi-coding-agent/dist/core/agent-session.d.ts.map +1 -1
  303. package/packages/pi-coding-agent/dist/core/agent-session.js +8 -2
  304. package/packages/pi-coding-agent/dist/core/agent-session.js.map +1 -1
  305. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js +472 -0
  306. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js.map +1 -1
  307. package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.d.ts +2 -0
  308. package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.d.ts.map +1 -0
  309. package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.js +52 -0
  310. package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.js.map +1 -0
  311. package/packages/pi-coding-agent/dist/core/model-registry.d.ts.map +1 -1
  312. package/packages/pi-coding-agent/dist/core/model-registry.js +2 -2
  313. package/packages/pi-coding-agent/dist/core/model-registry.js.map +1 -1
  314. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js +11 -0
  315. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js.map +1 -1
  316. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.d.ts +1 -0
  317. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.d.ts.map +1 -1
  318. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.js +23 -9
  319. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.js.map +1 -1
  320. package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.d.ts +11 -0
  321. package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.d.ts.map +1 -0
  322. package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.js +47 -0
  323. package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.js.map +1 -0
  324. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
  325. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js +51 -8
  326. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js.map +1 -1
  327. package/packages/pi-coding-agent/dist/modes/interactive/components/user-message.d.ts.map +1 -1
  328. package/packages/pi-coding-agent/dist/modes/interactive/components/user-message.js +22 -22
  329. package/packages/pi-coding-agent/dist/modes/interactive/components/user-message.js.map +1 -1
  330. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
  331. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js +232 -18
  332. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
  333. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.d.ts +2 -0
  334. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.d.ts.map +1 -0
  335. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.js +38 -0
  336. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.js.map +1 -0
  337. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts +13 -0
  338. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  339. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +53 -6
  340. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
  341. package/packages/pi-coding-agent/src/core/agent-session.ts +12 -6
  342. package/packages/pi-coding-agent/src/core/chat-controller-ordering.test.ts +612 -0
  343. package/packages/pi-coding-agent/src/core/model-registry-env-fallback.test.ts +59 -0
  344. package/packages/pi-coding-agent/src/core/model-registry.ts +2 -1
  345. package/packages/pi-coding-agent/src/modes/interactive/components/__tests__/tool-execution.test.ts +19 -0
  346. package/packages/pi-coding-agent/src/modes/interactive/components/assistant-message.ts +25 -10
  347. package/packages/pi-coding-agent/src/modes/interactive/components/chat-frame.ts +67 -0
  348. package/packages/pi-coding-agent/src/modes/interactive/components/tool-execution.ts +66 -7
  349. package/packages/pi-coding-agent/src/modes/interactive/components/user-message.ts +23 -26
  350. package/packages/pi-coding-agent/src/modes/interactive/controllers/chat-controller.ts +298 -41
  351. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode-ordering.test.ts +44 -0
  352. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +73 -6
  353. package/packages/pi-coding-agent/src/types/ambient-modules.d.ts +69 -0
  354. package/packages/pi-coding-agent/tsconfig.json +3 -2
  355. package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -0
  356. package/packages/pi-tui/package.json +1 -1
  357. package/packages/pi-tui/tsconfig.json +1 -0
  358. package/packages/pi-tui/tsconfig.tsbuildinfo +1 -0
  359. package/packages/rpc-client/package.json +1 -1
  360. package/packages/rpc-client/tsconfig.json +1 -0
  361. package/packages/rpc-client/tsconfig.tsbuildinfo +1 -0
  362. package/src/resources/extensions/gsd/activity-log.ts +21 -0
  363. package/src/resources/extensions/gsd/auto/detect-stuck.ts +12 -4
  364. package/src/resources/extensions/gsd/auto/loop-deps.ts +10 -0
  365. package/src/resources/extensions/gsd/auto/loop.ts +159 -10
  366. package/src/resources/extensions/gsd/auto/phases.ts +191 -4
  367. package/src/resources/extensions/gsd/auto/session.ts +10 -0
  368. package/src/resources/extensions/gsd/auto-dispatch.ts +16 -6
  369. package/src/resources/extensions/gsd/auto-model-selection.ts +66 -5
  370. package/src/resources/extensions/gsd/auto-post-unit.ts +238 -18
  371. package/src/resources/extensions/gsd/auto-prompts.ts +13 -0
  372. package/src/resources/extensions/gsd/auto-unit-closeout.ts +25 -1
  373. package/src/resources/extensions/gsd/auto-verification.ts +129 -2
  374. package/src/resources/extensions/gsd/auto.ts +41 -2
  375. package/src/resources/extensions/gsd/bootstrap/register-extension.ts +38 -8
  376. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +56 -3
  377. package/src/resources/extensions/gsd/commands/catalog.ts +26 -1
  378. package/src/resources/extensions/gsd/commands/handlers/ops.ts +25 -0
  379. package/src/resources/extensions/gsd/commands/handlers/workflow.ts +74 -9
  380. package/src/resources/extensions/gsd/commands-add-tests.ts +137 -0
  381. package/src/resources/extensions/gsd/commands-backlog.ts +182 -0
  382. package/src/resources/extensions/gsd/commands-do.ts +109 -0
  383. package/src/resources/extensions/gsd/commands-extract-learnings.ts +304 -0
  384. package/src/resources/extensions/gsd/commands-maintenance.ts +6 -6
  385. package/src/resources/extensions/gsd/commands-pr-branch.ts +234 -0
  386. package/src/resources/extensions/gsd/commands-prefs-wizard.ts +1 -1
  387. package/src/resources/extensions/gsd/commands-session-report.ts +101 -0
  388. package/src/resources/extensions/gsd/commands-ship.ts +219 -0
  389. package/src/resources/extensions/gsd/db-writer.ts +3 -5
  390. package/src/resources/extensions/gsd/docs/preferences-reference.md +14 -1
  391. package/src/resources/extensions/gsd/ecosystem/gsd-extension-api.ts +228 -0
  392. package/src/resources/extensions/gsd/ecosystem/loader.ts +201 -0
  393. package/src/resources/extensions/gsd/git-service.ts +68 -0
  394. package/src/resources/extensions/gsd/graph-context.ts +212 -0
  395. package/src/resources/extensions/gsd/gsd-db.ts +788 -3
  396. package/src/resources/extensions/gsd/guided-flow.ts +32 -0
  397. package/src/resources/extensions/gsd/index.ts +18 -2
  398. package/src/resources/extensions/gsd/init-wizard.ts +3 -2
  399. package/src/resources/extensions/gsd/journal.ts +30 -0
  400. package/src/resources/extensions/gsd/md-importer.ts +3 -5
  401. package/src/resources/extensions/gsd/memory-store.ts +31 -62
  402. package/src/resources/extensions/gsd/metrics.ts +26 -0
  403. package/src/resources/extensions/gsd/milestone-validation-gates.ts +13 -14
  404. package/src/resources/extensions/gsd/native-git-bridge.ts +11 -12
  405. package/src/resources/extensions/gsd/parallel-orchestrator.ts +40 -1
  406. package/src/resources/extensions/gsd/preferences-models.ts +20 -3
  407. package/src/resources/extensions/gsd/preferences-types.ts +32 -0
  408. package/src/resources/extensions/gsd/preferences-validation.ts +107 -2
  409. package/src/resources/extensions/gsd/preferences.ts +28 -0
  410. package/src/resources/extensions/gsd/prompts/add-tests.md +35 -0
  411. package/src/resources/extensions/gsd/session-lock.ts +14 -2
  412. package/src/resources/extensions/gsd/slice-parallel-orchestrator.ts +20 -1
  413. package/src/resources/extensions/gsd/state.ts +9 -2
  414. package/src/resources/extensions/gsd/templates/PREFERENCES.md +18 -0
  415. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +7 -3
  416. package/src/resources/extensions/gsd/tests/auto-model-selection.test.ts +20 -0
  417. package/src/resources/extensions/gsd/tests/auto-project-root-env.test.ts +7 -3
  418. package/src/resources/extensions/gsd/tests/cold-resume-db-reopen.test.ts +6 -2
  419. package/src/resources/extensions/gsd/tests/commands-backlog.test.ts +158 -0
  420. package/src/resources/extensions/gsd/tests/commands-do.test.ts +127 -0
  421. package/src/resources/extensions/gsd/tests/commands-extract-learnings.test.ts +340 -0
  422. package/src/resources/extensions/gsd/tests/commands-pr-branch.test.ts +68 -0
  423. package/src/resources/extensions/gsd/tests/commands-session-report.test.ts +82 -0
  424. package/src/resources/extensions/gsd/tests/commands-ship.test.ts +71 -0
  425. package/src/resources/extensions/gsd/tests/commands-workflow-custom.test.ts +14 -0
  426. package/src/resources/extensions/gsd/tests/complete-slice.test.ts +2 -2
  427. package/src/resources/extensions/gsd/tests/complete-task.test.ts +2 -2
  428. package/src/resources/extensions/gsd/tests/extension-bootstrap-isolation.test.ts +154 -0
  429. package/src/resources/extensions/gsd/tests/finalize-timeout-guard.test.ts +10 -7
  430. package/src/resources/extensions/gsd/tests/graph-context.test.ts +337 -0
  431. package/src/resources/extensions/gsd/tests/gsd-db.test.ts +1 -1
  432. package/src/resources/extensions/gsd/tests/health-widget.test.ts +1 -1
  433. package/src/resources/extensions/gsd/tests/journal-integration.test.ts +68 -1
  434. package/src/resources/extensions/gsd/tests/md-importer.test.ts +1 -2
  435. package/src/resources/extensions/gsd/tests/memory-store.test.ts +2 -3
  436. package/src/resources/extensions/gsd/tests/native-git-bridge-exec-fallback.test.ts +140 -0
  437. package/src/resources/extensions/gsd/tests/post-exec-retry-bypass.test.ts +79 -1
  438. package/src/resources/extensions/gsd/tests/post-unit-state-rebuild.test.ts +2 -1
  439. package/src/resources/extensions/gsd/tests/pre-execution-pause-wiring.test.ts +40 -1
  440. package/src/resources/extensions/gsd/tests/register-hooks-depth-verification.test.ts +1 -1
  441. package/src/resources/extensions/gsd/tests/single-writer-invariant.test.ts +180 -0
  442. package/src/resources/extensions/gsd/tests/token-profile.test.ts +8 -5
  443. package/src/resources/extensions/gsd/tests/uok-audit-unified.test.ts +101 -0
  444. package/src/resources/extensions/gsd/tests/uok-contracts.test.ts +85 -0
  445. package/src/resources/extensions/gsd/tests/uok-execution-graph.test.ts +69 -0
  446. package/src/resources/extensions/gsd/tests/uok-flags.test.ts +39 -0
  447. package/src/resources/extensions/gsd/tests/uok-gate-runner.test.ts +70 -0
  448. package/src/resources/extensions/gsd/tests/uok-gitops-turn-action.test.ts +85 -0
  449. package/src/resources/extensions/gsd/tests/uok-gitops-wiring.test.ts +35 -0
  450. package/src/resources/extensions/gsd/tests/uok-model-policy.test.ts +89 -0
  451. package/src/resources/extensions/gsd/tests/uok-plan-v2-wiring.test.ts +167 -0
  452. package/src/resources/extensions/gsd/tests/uok-preferences.test.ts +42 -0
  453. package/src/resources/extensions/gsd/tests/validate-milestone-write-order.test.ts +39 -0
  454. package/src/resources/extensions/gsd/tests/workflow-logger-wiring.test.ts +223 -0
  455. package/src/resources/extensions/gsd/tools/complete-slice.ts +26 -0
  456. package/src/resources/extensions/gsd/tools/validate-milestone.ts +48 -3
  457. package/src/resources/extensions/gsd/tools/workflow-tool-executors.ts +3 -11
  458. package/src/resources/extensions/gsd/triage-resolution.ts +2 -7
  459. package/src/resources/extensions/gsd/types.ts +14 -1
  460. package/src/resources/extensions/gsd/unit-ownership.ts +2 -2
  461. package/src/resources/extensions/gsd/uok/audit-toggle.ts +9 -0
  462. package/src/resources/extensions/gsd/uok/audit.ts +51 -0
  463. package/src/resources/extensions/gsd/uok/contracts.ts +135 -0
  464. package/src/resources/extensions/gsd/uok/execution-graph.ts +241 -0
  465. package/src/resources/extensions/gsd/uok/flags.ts +45 -0
  466. package/src/resources/extensions/gsd/uok/gate-runner.ts +146 -0
  467. package/src/resources/extensions/gsd/uok/gitops.ts +75 -0
  468. package/src/resources/extensions/gsd/uok/kernel.ts +105 -0
  469. package/src/resources/extensions/gsd/uok/loop-adapter.ts +162 -0
  470. package/src/resources/extensions/gsd/uok/model-policy.ts +112 -0
  471. package/src/resources/extensions/gsd/uok/plan-v2.ts +156 -0
  472. package/src/resources/extensions/gsd/workflow-logger.ts +27 -1
  473. package/src/resources/extensions/gsd/workflow-manifest.ts +9 -104
  474. package/src/resources/extensions/gsd/workflow-migration.ts +21 -29
  475. package/src/resources/extensions/gsd/workflow-projections.ts +8 -1
  476. package/src/resources/extensions/gsd/workflow-reconcile.ts +15 -15
  477. package/src/resources/extensions/ttsr/ttsr-manager.ts +10 -5
  478. package/packages/pi-ai/dist/models.custom.d.ts.map +0 -1
  479. package/packages/pi-ai/dist/models.custom.js.map +0 -1
  480. package/packages/pi-ai/dist/models.generated.js +0 -14343
  481. package/packages/pi-ai/dist/models.generated.js.map +0 -1
  482. package/packages/pi-ai/src/models.generated.ts +0 -14345
  483. /package/dist/web/standalone/.next/static/{YzIEI9sxJy4t5xgClF08g → 3U-oZ5FT59BM7sm2GInic}/_buildManifest.js +0 -0
  484. /package/dist/web/standalone/.next/static/{YzIEI9sxJy4t5xgClF08g → 3U-oZ5FT59BM7sm2GInic}/_ssgManifest.js +0 -0
@@ -0,0 +1,187 @@
1
+ /**
2
+ * GSD Command — /gsd ship
3
+ *
4
+ * Creates a PR from milestone artifacts: generates title + body from
5
+ * roadmap, slice summaries, and metrics, then opens via `gh pr create`.
6
+ */
7
+ import { execFileSync } from "node:child_process";
8
+ import { existsSync, readFileSync, readdirSync } from "node:fs";
9
+ import { deriveState } from "./state.js";
10
+ import { resolveMilestoneFile, resolveSlicePath, resolveSliceFile } from "./paths.js";
11
+ import { getLedger, getProjectTotals, aggregateByModel, formatCost, formatTokenCount, loadLedgerFromDisk } from "./metrics.js";
12
+ import { nativeGetCurrentBranch, nativeDetectMainBranch } from "./native-git-bridge.js";
13
+ import { formatDuration } from "../shared/format-utils.js";
14
+ function git(basePath, args) {
15
+ return execFileSync("git", args, { cwd: basePath, encoding: "utf-8" }).trim();
16
+ }
17
+ function isValidRefName(name) {
18
+ try {
19
+ execFileSync("git", ["check-ref-format", "--branch", name], { stdio: "pipe" });
20
+ return true;
21
+ }
22
+ catch {
23
+ return false;
24
+ }
25
+ }
26
+ function listSliceIds(basePath, milestoneId) {
27
+ // Slices live at <milestoneDir>/slices/<sliceId>/ with canonical S\d+ IDs.
28
+ // Use resolveSlicePath with a probe to find the real slices directory root.
29
+ const probe = resolveSlicePath(basePath, milestoneId, "S01");
30
+ let slicesDir = null;
31
+ if (probe) {
32
+ // probe looks like <milestoneDir>/slices/S01 — parent is slices dir.
33
+ slicesDir = probe.replace(/[\\/][^\\/]+$/, "");
34
+ }
35
+ else {
36
+ // Fall back to scanning the milestones roadmap file's sibling slices dir.
37
+ const roadmap = resolveMilestoneFile(basePath, milestoneId, "ROADMAP");
38
+ if (roadmap) {
39
+ slicesDir = roadmap.replace(/[\\/][^\\/]+$/, "") + "/slices";
40
+ }
41
+ }
42
+ if (!slicesDir || !existsSync(slicesDir))
43
+ return [];
44
+ try {
45
+ return readdirSync(slicesDir, { withFileTypes: true })
46
+ .filter((e) => e.isDirectory() && /^S\d+$/.test(e.name))
47
+ .map((e) => e.name)
48
+ .sort();
49
+ }
50
+ catch {
51
+ return [];
52
+ }
53
+ }
54
+ function collectSliceSummaries(basePath, milestoneId) {
55
+ const summaries = [];
56
+ for (const sliceId of listSliceIds(basePath, milestoneId)) {
57
+ const summaryPath = resolveSliceFile(basePath, milestoneId, sliceId, "SUMMARY");
58
+ if (!summaryPath || !existsSync(summaryPath))
59
+ continue;
60
+ try {
61
+ const content = readFileSync(summaryPath, "utf-8").trim();
62
+ if (content)
63
+ summaries.push(`### ${sliceId}\n${content}`);
64
+ }
65
+ catch {
66
+ // non-fatal
67
+ }
68
+ }
69
+ return summaries;
70
+ }
71
+ function generatePRContent(basePath, milestoneId, milestoneTitle) {
72
+ const title = `feat: ${milestoneTitle || milestoneId}`;
73
+ const sections = [];
74
+ // TL;DR
75
+ sections.push("## TL;DR\n");
76
+ sections.push(`**What:** Ship milestone ${milestoneId} — ${milestoneTitle || "(untitled)"}`);
77
+ sections.push(`**Why:** Milestone work complete, ready for review.`);
78
+ sections.push(`**How:** See slice summaries below.\n`);
79
+ // What — slice summaries
80
+ const summaries = collectSliceSummaries(basePath, milestoneId);
81
+ if (summaries.length > 0) {
82
+ sections.push("## What\n");
83
+ sections.push(summaries.join("\n\n"));
84
+ sections.push("");
85
+ }
86
+ // Roadmap status
87
+ const roadmapPath = resolveMilestoneFile(basePath, milestoneId, "ROADMAP");
88
+ if (roadmapPath && existsSync(roadmapPath)) {
89
+ try {
90
+ const roadmap = readFileSync(roadmapPath, "utf-8");
91
+ const checkboxLines = roadmap.split("\n").filter((l) => /^\s*-\s*\[[ x]\]/.test(l));
92
+ if (checkboxLines.length > 0) {
93
+ sections.push("## Roadmap\n");
94
+ sections.push(checkboxLines.join("\n"));
95
+ sections.push("");
96
+ }
97
+ }
98
+ catch {
99
+ // non-fatal
100
+ }
101
+ }
102
+ // Metrics
103
+ const ledger = getLedger();
104
+ const units = ledger?.units ?? loadLedgerFromDisk(basePath)?.units ?? [];
105
+ if (units.length > 0) {
106
+ const totals = getProjectTotals(units);
107
+ const byModel = aggregateByModel(units);
108
+ sections.push("## Metrics\n");
109
+ sections.push(`- **Units executed:** ${units.length}`);
110
+ sections.push(`- **Total cost:** ${formatCost(totals.cost)}`);
111
+ sections.push(`- **Tokens:** ${formatTokenCount(totals.tokens.input)} input / ${formatTokenCount(totals.tokens.output)} output`);
112
+ if (totals.duration > 0) {
113
+ sections.push(`- **Duration:** ${formatDuration(totals.duration)}`);
114
+ }
115
+ if (byModel.length > 0) {
116
+ sections.push(`- **Models:** ${byModel.map((m) => `${m.model} (${m.units} units)`).join(", ")}`);
117
+ }
118
+ sections.push("");
119
+ }
120
+ // Change type checklist
121
+ sections.push("## Change type\n");
122
+ sections.push("- [x] `feat` — New feature or capability");
123
+ sections.push("- [ ] `fix` — Bug fix");
124
+ sections.push("- [ ] `refactor` — Code restructuring");
125
+ sections.push("- [ ] `test` — Adding or updating tests");
126
+ sections.push("- [ ] `docs` — Documentation only");
127
+ sections.push("- [ ] `chore` — Build, CI, or tooling changes\n");
128
+ // AI disclosure
129
+ sections.push("---\n");
130
+ sections.push("*This PR was prepared with AI assistance (GSD auto-mode).*");
131
+ return { title, body: sections.join("\n") };
132
+ }
133
+ export async function handleShip(args, ctx, _pi) {
134
+ const basePath = process.cwd();
135
+ const dryRun = args.includes("--dry-run");
136
+ const draft = args.includes("--draft");
137
+ const force = args.includes("--force");
138
+ const baseMatch = args.match(/--base\s+(\S+)/);
139
+ const base = baseMatch?.[1] ?? nativeDetectMainBranch(basePath);
140
+ if (!isValidRefName(base)) {
141
+ ctx.ui.notify(`Invalid base branch name: ${base}`, "error");
142
+ return;
143
+ }
144
+ // 1. Validate milestone state
145
+ const state = await deriveState(basePath);
146
+ if (!state.activeMilestone) {
147
+ ctx.ui.notify("No active milestone to ship. Complete milestone work first.", "warning");
148
+ return;
149
+ }
150
+ const milestoneId = state.activeMilestone.id;
151
+ const milestoneTitle = state.activeMilestone.title ?? "";
152
+ // 2. Check for incomplete work (use GSD phase as proxy — no phase field on ActiveRef)
153
+ if (state.phase !== "complete" && !force) {
154
+ ctx.ui.notify(`Milestone ${milestoneId} may not be complete (phase: ${state.phase}). Use --force to ship anyway.`, "warning");
155
+ return;
156
+ }
157
+ // 3. Generate PR content
158
+ const { title, body } = generatePRContent(basePath, milestoneId, milestoneTitle);
159
+ // 4. Dry-run — just show the PR content
160
+ if (dryRun) {
161
+ ctx.ui.notify(`--- PR Preview ---\n\nTitle: ${title}\n\n${body}`, "info");
162
+ return;
163
+ }
164
+ // 5. Check git state
165
+ const currentBranch = nativeGetCurrentBranch(basePath);
166
+ if (!isValidRefName(currentBranch)) {
167
+ ctx.ui.notify(`Current branch name is invalid for git: ${currentBranch}`, "error");
168
+ return;
169
+ }
170
+ if (currentBranch === base) {
171
+ ctx.ui.notify(`You're on ${base} — create a feature branch first.`, "warning");
172
+ return;
173
+ }
174
+ // 6. Push and create PR (all argv-safe, no shell interpolation)
175
+ try {
176
+ git(basePath, ["push", "-u", "origin", currentBranch]);
177
+ const ghArgs = ["pr", "create", "--base", base, "--title", title, "--body", body];
178
+ if (draft)
179
+ ghArgs.push("--draft");
180
+ const prUrl = execFileSync("gh", ghArgs, { cwd: basePath, encoding: "utf-8" }).trim();
181
+ ctx.ui.notify(`PR created: ${prUrl}`, "success");
182
+ }
183
+ catch (err) {
184
+ const msg = err instanceof Error ? err.message : String(err);
185
+ ctx.ui.notify(`Failed to create PR: ${msg}`, "error");
186
+ }
187
+ }
@@ -301,8 +301,7 @@ export async function saveRequirementToDb(fields, basePath) {
301
301
  catch (diskErr) {
302
302
  logError('manifest', 'disk write failed, rolling back DB row', { fn: 'saveRequirementToDb', error: String(diskErr.message) });
303
303
  try {
304
- const rollbackAdapter = db._getAdapter();
305
- rollbackAdapter?.prepare('DELETE FROM requirements WHERE id = :id').run({ ':id': id });
304
+ db.deleteRequirementById(id);
306
305
  }
307
306
  catch (rollbackErr) {
308
307
  logError('manifest', 'SPLIT BRAIN: disk write failed AND DB rollback failed — DB has orphaned row', { fn: 'saveRequirementToDb', id, error: String(rollbackErr.message) });
@@ -405,7 +404,7 @@ export async function saveDecisionToDb(fields, basePath) {
405
404
  catch (diskErr) {
406
405
  logError('manifest', 'disk write failed, rolling back DB row', { fn: 'saveDecisionToDb', error: String(diskErr.message) });
407
406
  try {
408
- adapter?.prepare('DELETE FROM decisions WHERE id = :id').run({ ':id': id });
407
+ db.deleteDecisionById(id);
409
408
  }
410
409
  catch (rollbackErr) {
411
410
  logError('manifest', 'SPLIT BRAIN: disk write failed AND DB rollback failed — DB has orphaned row', { fn: 'saveDecisionToDb', id, error: String(rollbackErr.message) });
@@ -614,8 +613,7 @@ export async function saveArtifactToDb(opts, basePath) {
614
613
  }
615
614
  catch (diskErr) {
616
615
  logError('manifest', 'disk write failed, rolling back DB row', { fn: 'saveArtifactToDb', error: String(diskErr.message) });
617
- const rollbackAdapter = db._getAdapter();
618
- rollbackAdapter?.prepare('DELETE FROM artifacts WHERE path = :path').run({ ':path': opts.path });
616
+ db.deleteArtifactByPath(opts.path);
619
617
  throw diskErr;
620
618
  }
621
619
  }
@@ -153,7 +153,7 @@ Setting `prefer_skills: []` does **not** disable skill discovery — it just mea
153
153
 
154
154
  - `context_pause_threshold`: number (0-100) — context window usage percentage at which auto-mode should pause to suggest checkpointing. Set to `0` to disable. Default: `0` (disabled).
155
155
 
156
- - `token_profile`: `"budget"`, `"balanced"`, or `"quality"` — coordinates model selection, phase skipping, and context compression. `budget` skips research/reassessment and uses cheaper models; `balanced` (default) skips research/reassessment to reduce token burn; `quality` prefers higher-quality models. See token-optimization docs.
156
+ - `token_profile`: `"budget"`, `"balanced"`, `"quality"`, or `"burn-max"` — coordinates model selection, phase skipping, and context compression. `budget` skips research/reassessment and uses cheaper models; `balanced` (default) skips research/reassessment to reduce token burn; `quality` prefers higher-quality models; `burn-max` keeps full-context defaults, disables downgrade routing, and keeps phase skips off.
157
157
 
158
158
  - `phases`: fine-grained control over which phases run. Usually set by `token_profile`, but can be overridden. Keys:
159
159
  - `skip_research`: boolean — skip milestone-level research. Default: `false`.
@@ -191,6 +191,19 @@ Setting `prefer_skills: []` does **not** disable skill discovery — it just mea
191
191
  - `hooks`: boolean — enable routing hooks. Default: `true`.
192
192
  - `capability_routing`: boolean — enable capability-profile scoring for model selection within a tier. Requires `enabled: true`. Default: `false`.
193
193
 
194
+ - `uok`: Unified Orchestration Kernel controls. Keys:
195
+ - `enabled`: boolean — enable kernel wrappers and contract observers. Default: `true`.
196
+ - `legacy_fallback.enabled`: boolean — emergency release fallback that forces legacy orchestration behavior even when `uok.enabled` is `true`. Default: `false`.
197
+ - Runtime override: set `GSD_UOK_FORCE_LEGACY=1` (or `GSD_UOK_LEGACY_FALLBACK=1`) to force legacy behavior for the current process.
198
+ - `gates.enabled`: boolean — route checks through the unified gate runner and persist `gate_runs`.
199
+ - `model_policy.enabled`: boolean — enforce policy filtering before model capability scoring.
200
+ - `execution_graph.enabled`: boolean — enable DAG scheduler facade/adapters for execution.
201
+ - `gitops.enabled`: boolean — persist turn-level git transaction records.
202
+ - `gitops.turn_action`: `"commit"` | `"snapshot"` | `"status-only"` — turn transaction mode.
203
+ - `gitops.turn_push`: boolean — whether turn transactions should include push intent metadata.
204
+ - `audit_unified.enabled`: boolean — dual-write unified audit envelope events.
205
+ - `plan_v2.enabled`: boolean — enable bounded clarify/research/draft/compile planning flow.
206
+
194
207
  - `context_management`: configures context hygiene for auto-mode sessions. Keys:
195
208
  - `observation_masking`: boolean — mask old tool results to reduce context bloat. Default: `true`.
196
209
  - `observation_mask_turns`: number — keep this many recent turns verbatim (1-50). Default: `8`.
@@ -0,0 +1,144 @@
1
+ // GSD2 — Ecosystem Extension API wrapper
2
+ // Wraps pi's ExtensionAPI to expose typed GSD context (phase + active unit)
3
+ // to extensions loaded from `./.gsd/extensions/`. The wrapper intercepts only
4
+ // `on("before_agent_start", ...)` so GSD can dispatch ecosystem handlers AFTER
5
+ // refreshing state — fixing the load-order race where third-party
6
+ // `.pi/extensions/` handlers see a stale module-level snapshot (#3338).
7
+ //
8
+ // SINGLE-SESSION INVARIANT: the module-level `_snapshot` is per-process.
9
+ // Worktree or project switches do NOT reload extensions, matching pi's
10
+ // `.pi/extensions/` behavior. Only re-launching the CLI rebinds the snapshot.
11
+ import { isGSDActive, getCurrentPhase } from "../../shared/gsd-phase-state.js";
12
+ import { logWarning } from "../workflow-logger.js";
13
+ // ─── Auto-loop phase mapping ────────────────────────────────────────────
14
+ const AUTO_LOOP_PHASE_MAP = {
15
+ "plan-milestone": "planning",
16
+ "plan-slice": "planning",
17
+ "research": "researching",
18
+ "discuss": "discussing",
19
+ "execute-task": "executing",
20
+ "verify": "verifying",
21
+ "summarize-task": "summarizing",
22
+ "summarize-slice": "summarizing",
23
+ "advance": "advancing",
24
+ "validate-milestone": "validating-milestone",
25
+ "complete-milestone": "completing-milestone",
26
+ "replan-slice": "replanning-slice",
27
+ };
28
+ /** Exposed for unit tests. Returns null for unknown keys (does NOT default). */
29
+ export function mapAutoLoopPhase(raw) {
30
+ return AUTO_LOOP_PHASE_MAP[raw] ?? null;
31
+ }
32
+ function resolvePhase(state) {
33
+ if (!state)
34
+ return null;
35
+ if (isGSDActive()) {
36
+ const raw = getCurrentPhase();
37
+ if (raw != null) {
38
+ const mapped = AUTO_LOOP_PHASE_MAP[raw];
39
+ if (mapped)
40
+ return mapped;
41
+ logWarning("ecosystem", `unknown auto-loop phase: ${raw}`);
42
+ // FALL THROUGH to state.phase rather than defaulting to "executing".
43
+ }
44
+ }
45
+ return state.phase;
46
+ }
47
+ function resolveActiveUnit(state) {
48
+ if (!state)
49
+ return null;
50
+ const m = state.activeMilestone;
51
+ const s = state.activeSlice;
52
+ const t = state.activeTask;
53
+ if (!m || !s || !t)
54
+ return null;
55
+ return {
56
+ milestoneId: m.id,
57
+ milestoneTitle: m.title,
58
+ sliceId: s.id,
59
+ sliceTitle: s.title,
60
+ taskId: t.id,
61
+ taskTitle: t.title,
62
+ };
63
+ }
64
+ let _snapshot = { phase: null, activeUnit: null };
65
+ /** Refresh the snapshot from a freshly derived GSDState (or null on failure). */
66
+ export function updateSnapshot(state) {
67
+ _snapshot = {
68
+ phase: resolvePhase(state),
69
+ activeUnit: resolveActiveUnit(state),
70
+ };
71
+ }
72
+ export function getSnapshotPhase() {
73
+ return _snapshot.phase;
74
+ }
75
+ export function getSnapshotActiveUnit() {
76
+ return _snapshot.activeUnit;
77
+ }
78
+ /** Test-only: reset the snapshot to its initial empty state. */
79
+ export function _resetSnapshot() {
80
+ _snapshot = { phase: null, activeUnit: null };
81
+ }
82
+ // ─── Wrapper factory ────────────────────────────────────────────────────
83
+ /**
84
+ * Build a GSDExtensionAPI by manually delegating every ExtensionAPI method
85
+ * to the underlying pi instance, except `on("before_agent_start", ...)`
86
+ * which is captured into `sharedHandlers` for GSD-owned dispatch.
87
+ *
88
+ * Uses `satisfies GSDExtensionAPI` (NOT `as`) so TypeScript catches drift
89
+ * when pi adds new ExtensionAPI methods.
90
+ */
91
+ export function createGSDExtensionAPI(pi, sharedHandlers) {
92
+ const wrapper = {
93
+ // ── Event subscription (single intercept point) ────────────────────
94
+ on(event, handler) {
95
+ if (event === "before_agent_start") {
96
+ sharedHandlers.push(handler);
97
+ return;
98
+ }
99
+ pi.on(event, handler);
100
+ },
101
+ // ── Event emission ─────────────────────────────────────────────────
102
+ emitBeforeModelSelect: (...args) => pi.emitBeforeModelSelect(...args),
103
+ emitAdjustToolSet: (...args) => pi.emitAdjustToolSet(...args),
104
+ // ── Tool / command / shortcut / flag registration ──────────────────
105
+ registerTool: ((tool) => pi.registerTool(tool)),
106
+ registerCommand: (...args) => pi.registerCommand(...args),
107
+ registerBeforeInstall: (...args) => pi.registerBeforeInstall(...args),
108
+ registerAfterInstall: (...args) => pi.registerAfterInstall(...args),
109
+ registerBeforeRemove: (...args) => pi.registerBeforeRemove(...args),
110
+ registerAfterRemove: (...args) => pi.registerAfterRemove(...args),
111
+ registerShortcut: (...args) => pi.registerShortcut(...args),
112
+ registerFlag: (...args) => pi.registerFlag(...args),
113
+ getFlag: (...args) => pi.getFlag(...args),
114
+ // ── Message rendering ──────────────────────────────────────────────
115
+ registerMessageRenderer: ((customType, renderer) => pi.registerMessageRenderer(customType, renderer)),
116
+ // ── Actions ────────────────────────────────────────────────────────
117
+ sendMessage: ((message, options) => pi.sendMessage(message, options)),
118
+ sendUserMessage: (...args) => pi.sendUserMessage(...args),
119
+ retryLastTurn: () => pi.retryLastTurn(),
120
+ appendEntry: ((customType, data) => pi.appendEntry(customType, data)),
121
+ // ── Session metadata ───────────────────────────────────────────────
122
+ setSessionName: (...args) => pi.setSessionName(...args),
123
+ getSessionName: () => pi.getSessionName(),
124
+ setLabel: (...args) => pi.setLabel(...args),
125
+ exec: (...args) => pi.exec(...args),
126
+ getActiveTools: () => pi.getActiveTools(),
127
+ getAllTools: () => pi.getAllTools(),
128
+ setActiveTools: (...args) => pi.setActiveTools(...args),
129
+ getCommands: () => pi.getCommands(),
130
+ // ── Model & thinking ───────────────────────────────────────────────
131
+ setModel: (...args) => pi.setModel(...args),
132
+ getThinkingLevel: () => pi.getThinkingLevel(),
133
+ setThinkingLevel: (...args) => pi.setThinkingLevel(...args),
134
+ // ── Provider registration ──────────────────────────────────────────
135
+ registerProvider: (...args) => pi.registerProvider(...args),
136
+ unregisterProvider: (...args) => pi.unregisterProvider(...args),
137
+ // ── Shared event bus (passthrough property) ────────────────────────
138
+ events: pi.events,
139
+ // ── GSD-specific additions ─────────────────────────────────────────
140
+ getPhase: () => _snapshot.phase,
141
+ getActiveUnit: () => _snapshot.activeUnit,
142
+ };
143
+ return wrapper;
144
+ }
@@ -0,0 +1,145 @@
1
+ // GSD2 — Ecosystem extension loader for ./.gsd/extensions/
2
+ // Discovers and registers single-file extensions that consume GSDExtensionAPI.
3
+ // Trust-gated (mirrors pi's `.pi/extensions/` model) and isolated from pi's
4
+ // own loader chain — handlers run in GSD's own dispatch step, not pi's.
5
+ import * as fs from "node:fs";
6
+ import * as path from "node:path";
7
+ import { pathToFileURL } from "node:url";
8
+ import { getAgentDir } from "@gsd/pi-coding-agent";
9
+ import { logWarning } from "../workflow-logger.js";
10
+ import { createGSDExtensionAPI, } from "./gsd-extension-api.js";
11
+ // ─── Trust check (inlined; pi does not export isProjectTrusted from its
12
+ // package root, and constraint forbids modifying packages/pi-coding-agent/) ─
13
+ const TRUSTED_PROJECTS_FILE = "trusted-projects.json";
14
+ function isProjectTrusted(projectPath, agentDir) {
15
+ const canonical = path.resolve(projectPath);
16
+ const trustedPath = path.join(agentDir, TRUSTED_PROJECTS_FILE);
17
+ try {
18
+ const content = fs.readFileSync(trustedPath, "utf-8");
19
+ const parsed = JSON.parse(content);
20
+ if (Array.isArray(parsed)) {
21
+ return parsed.includes(canonical);
22
+ }
23
+ }
24
+ catch {
25
+ // missing or malformed — treat as untrusted
26
+ }
27
+ return false;
28
+ }
29
+ // ─── Ready-promise singleton ────────────────────────────────────────────
30
+ let _readyPromise = null;
31
+ let _untrustedWarned = false;
32
+ /**
33
+ * Discover and register ecosystem extensions from `./.gsd/extensions/`.
34
+ * Idempotent: subsequent calls with the same arguments return the same
35
+ * pending promise (no double-load).
36
+ */
37
+ export function loadEcosystemExtensions(pi, sharedHandlers, cwd = process.cwd()) {
38
+ if (_readyPromise)
39
+ return _readyPromise;
40
+ _readyPromise = _loadEcosystemExtensionsImpl(pi, sharedHandlers, cwd);
41
+ return _readyPromise;
42
+ }
43
+ /**
44
+ * Returns a promise that resolves when ecosystem loading has completed.
45
+ * If loading was never kicked off this returns a resolved promise so the
46
+ * `before_agent_start` handler can `await` unconditionally.
47
+ */
48
+ export function getEcosystemReadyPromise() {
49
+ return _readyPromise ?? Promise.resolve();
50
+ }
51
+ /** Test-only: clear the singleton so tests can re-run loading. */
52
+ export function _resetEcosystemLoader() {
53
+ _readyPromise = null;
54
+ _untrustedWarned = false;
55
+ }
56
+ // ─── Implementation ─────────────────────────────────────────────────────
57
+ async function _loadEcosystemExtensionsImpl(pi, sharedHandlers, cwd) {
58
+ const extDir = path.join(cwd, ".gsd", "extensions");
59
+ if (!fs.existsSync(extDir))
60
+ return;
61
+ // Trust gate: refuse to load arbitrary code from untrusted project dirs.
62
+ if (!isProjectTrusted(cwd, getAgentDir())) {
63
+ if (!_untrustedWarned) {
64
+ _untrustedWarned = true;
65
+ logWarning("ecosystem", ".gsd/extensions present but project is not trusted — skipping ecosystem extensions. Run `pi trust` to opt in.");
66
+ }
67
+ return;
68
+ }
69
+ // Resolve realpath ONCE so symlink-escape detection has a stable anchor.
70
+ let realExtDir;
71
+ try {
72
+ realExtDir = fs.realpathSync(extDir);
73
+ }
74
+ catch (err) {
75
+ logWarning("ecosystem", `failed to resolve extensions dir: ${err instanceof Error ? err.message : String(err)}`);
76
+ return;
77
+ }
78
+ let entries;
79
+ try {
80
+ entries = fs
81
+ .readdirSync(extDir)
82
+ .filter((f) => f.endsWith(".js") || f.endsWith(".ts"))
83
+ .sort(); // deterministic load order
84
+ }
85
+ catch (err) {
86
+ logWarning("ecosystem", `failed to read extensions dir: ${err instanceof Error ? err.message : String(err)}`);
87
+ return;
88
+ }
89
+ // The wrapper api is built once per loader run and shared by all extensions
90
+ // so they all read from the same module-level snapshot.
91
+ const api = createGSDExtensionAPI(pi, sharedHandlers);
92
+ for (const entry of entries) {
93
+ await _loadOne(extDir, realExtDir, entry, api);
94
+ }
95
+ }
96
+ async function _loadOne(extDir, realExtDir, entry, api) {
97
+ const fullPath = path.join(extDir, entry);
98
+ // Symlink-escape guard: reject entries whose realpath is not under realExtDir.
99
+ let realFullPath;
100
+ try {
101
+ realFullPath = fs.realpathSync(fullPath);
102
+ }
103
+ catch (err) {
104
+ logWarning("ecosystem", `failed to resolve ${entry}: ${err instanceof Error ? err.message : String(err)}`);
105
+ return;
106
+ }
107
+ const realExtDirWithSep = realExtDir.endsWith(path.sep) ? realExtDir : realExtDir + path.sep;
108
+ if (realFullPath !== realExtDir &&
109
+ !realFullPath.startsWith(realExtDirWithSep)) {
110
+ logWarning("ecosystem", `rejecting ${entry}: realpath escapes extensions dir`);
111
+ return;
112
+ }
113
+ // For .ts files, require a sibling compiled .js — we do not run a TS loader
114
+ // in production. Drop mtime heuristics: if .js exists, prefer it; otherwise warn.
115
+ let importPath = realFullPath;
116
+ if (entry.endsWith(".ts")) {
117
+ const jsSibling = realFullPath.slice(0, -3) + ".js";
118
+ if (fs.existsSync(jsSibling)) {
119
+ importPath = jsSibling;
120
+ }
121
+ else {
122
+ logWarning("ecosystem", `${entry}: TypeScript source has no compiled .js sibling — compile it first`);
123
+ return;
124
+ }
125
+ }
126
+ let mod;
127
+ try {
128
+ mod = await import(pathToFileURL(importPath).href);
129
+ }
130
+ catch (err) {
131
+ logWarning("ecosystem", `failed to import ${entry}: ${err instanceof Error ? err.message : String(err)}`);
132
+ return;
133
+ }
134
+ const factory = mod?.default;
135
+ if (typeof factory !== "function") {
136
+ logWarning("ecosystem", `${entry}: default export is not a function`);
137
+ return;
138
+ }
139
+ try {
140
+ await factory(api);
141
+ }
142
+ catch (err) {
143
+ logWarning("ecosystem", `factory threw for ${entry}: ${err instanceof Error ? err.message : String(err)}`);
144
+ }
145
+ }
@@ -15,7 +15,7 @@ import { GIT_NO_PROMPT_ENV } from "./git-constants.js";
15
15
  import { loadEffectiveGSDPreferences } from "./preferences.js";
16
16
  import { detectWorktreeName, } from "./worktree.js";
17
17
  import { SLICE_BRANCH_RE, QUICK_BRANCH_RE, WORKFLOW_BRANCH_RE } from "./branch-patterns.js";
18
- import { nativeGetCurrentBranch, nativeDetectMainBranch, nativeBranchExists, nativeHasChanges, nativeAddAllWithExclusions, nativeHasStagedChanges, nativeCommit, nativeRmCached, nativeUpdateRef, nativeResetSoft, nativeCommitSubject, } from "./native-git-bridge.js";
18
+ import { nativeGetCurrentBranch, nativeDetectMainBranch, nativeBranchExists, nativeHasChanges, nativeAddAllWithExclusions, nativeHasStagedChanges, nativeCommit, nativeRmCached, nativeUpdateRef, nativeResetSoft, nativeCommitSubject, _resetHasChangesCache, } from "./native-git-bridge.js";
19
19
  import { GSDError, GSD_MERGE_CONFLICT, GSD_GIT_ERROR } from "./errors.js";
20
20
  import { getErrorMessage } from "./error-utils.js";
21
21
  export const VALID_BRANCH_NAME = /^[a-zA-Z0-9_\-\/.]+$/;
@@ -641,6 +641,54 @@ export function createGitService(basePath) {
641
641
  const gitPrefs = loadEffectiveGSDPreferences()?.preferences?.git ?? {};
642
642
  return new GitServiceImpl(basePath, gitPrefs);
643
643
  }
644
+ function buildTurnSnapshotLabel(unitType, unitId) {
645
+ const raw = `${unitType}/${unitId}`.trim();
646
+ if (!raw)
647
+ return "turn";
648
+ return raw
649
+ .replace(/[^a-zA-Z0-9._/-]/g, "-")
650
+ .replace(/\/{2,}/g, "/")
651
+ .replace(/-{2,}/g, "-")
652
+ .replace(/^[-/]+|[-/]+$/g, "") || "turn";
653
+ }
654
+ export function runTurnGitAction(args) {
655
+ try {
656
+ // Force fresh working-tree status per turn; nativeHasChanges caches briefly.
657
+ _resetHasChangesCache();
658
+ if (args.action === "status-only") {
659
+ return {
660
+ action: args.action,
661
+ status: "ok",
662
+ dirty: nativeHasChanges(args.basePath),
663
+ };
664
+ }
665
+ const git = createGitService(args.basePath);
666
+ if (args.action === "snapshot") {
667
+ const label = buildTurnSnapshotLabel(args.unitType, args.unitId);
668
+ git.createSnapshot(label);
669
+ return {
670
+ action: args.action,
671
+ status: "ok",
672
+ snapshotLabel: label,
673
+ dirty: nativeHasChanges(args.basePath),
674
+ };
675
+ }
676
+ const commitMessage = git.autoCommit(args.unitType, args.unitId, [], args.taskContext) ?? undefined;
677
+ return {
678
+ action: args.action,
679
+ status: "ok",
680
+ commitMessage,
681
+ dirty: nativeHasChanges(args.basePath),
682
+ };
683
+ }
684
+ catch (err) {
685
+ return {
686
+ action: args.action,
687
+ status: "failed",
688
+ error: getErrorMessage(err),
689
+ };
690
+ }
691
+ }
644
692
  // ─── Commit Type Inference ─────────────────────────────────────────────────
645
693
  /**
646
694
  * Infer a conventional commit type from a title (and optional one-liner).