gsd-pi 2.76.0 → 2.77.0

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 (536) hide show
  1. package/README.md +45 -25
  2. package/dist/claude-cli-check.js +32 -3
  3. package/dist/mcp-server.d.ts +7 -0
  4. package/dist/mcp-server.js +35 -1
  5. package/dist/onboarding.js +45 -0
  6. package/dist/resource-loader.d.ts +1 -1
  7. package/dist/resource-loader.js +2 -8
  8. package/dist/resources/agents/researcher.md +1 -1
  9. package/dist/resources/extensions/claude-code-cli/readiness.js +31 -8
  10. package/dist/resources/extensions/claude-code-cli/stream-adapter.js +77 -17
  11. package/dist/resources/extensions/gsd/auto/loop.js +9 -0
  12. package/dist/resources/extensions/gsd/auto/phases.js +104 -11
  13. package/dist/resources/extensions/gsd/auto/run-unit.js +38 -2
  14. package/dist/resources/extensions/gsd/auto/session.js +22 -1
  15. package/dist/resources/extensions/gsd/auto-dispatch.js +16 -3
  16. package/dist/resources/extensions/gsd/auto-model-selection.js +53 -16
  17. package/dist/resources/extensions/gsd/auto-post-unit.js +25 -2
  18. package/dist/resources/extensions/gsd/auto-prompts.js +14 -0
  19. package/dist/resources/extensions/gsd/auto-recovery.js +32 -1
  20. package/dist/resources/extensions/gsd/auto-start.js +58 -57
  21. package/dist/resources/extensions/gsd/auto-verification.js +33 -0
  22. package/dist/resources/extensions/gsd/auto-worktree.js +51 -53
  23. package/dist/resources/extensions/gsd/auto.js +70 -28
  24. package/dist/resources/extensions/gsd/blocked-models.js +68 -0
  25. package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +93 -1
  26. package/dist/resources/extensions/gsd/bootstrap/db-tools.js +39 -9
  27. package/dist/resources/extensions/gsd/bootstrap/exec-tools.js +93 -0
  28. package/dist/resources/extensions/gsd/bootstrap/memory-tools.js +3 -0
  29. package/dist/resources/extensions/gsd/bootstrap/register-extension.js +12 -0
  30. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +52 -6
  31. package/dist/resources/extensions/gsd/bootstrap/system-context.js +84 -23
  32. package/dist/resources/extensions/gsd/bootstrap/write-gate.js +34 -2
  33. package/dist/resources/extensions/gsd/clean-root-preflight.js +93 -0
  34. package/dist/resources/extensions/gsd/commands-extract-learnings.js +54 -89
  35. package/dist/resources/extensions/gsd/commands-prefs-wizard.js +968 -23
  36. package/dist/resources/extensions/gsd/compaction-snapshot.js +121 -0
  37. package/dist/resources/extensions/gsd/complexity-classifier.js +5 -3
  38. package/dist/resources/extensions/gsd/db-writer.js +88 -16
  39. package/dist/resources/extensions/gsd/doctor-git-checks.js +23 -29
  40. package/dist/resources/extensions/gsd/doctor-providers.js +51 -5
  41. package/dist/resources/extensions/gsd/ecosystem/gsd-extension-api.js +1 -0
  42. package/dist/resources/extensions/gsd/error-classifier.js +31 -3
  43. package/dist/resources/extensions/gsd/exec-history.js +120 -0
  44. package/dist/resources/extensions/gsd/exec-sandbox.js +258 -0
  45. package/dist/resources/extensions/gsd/gitignore.js +1 -0
  46. package/dist/resources/extensions/gsd/gsd-db.js +168 -23
  47. package/dist/resources/extensions/gsd/guided-flow.js +190 -1
  48. package/dist/resources/extensions/gsd/health-widget.js +4 -1
  49. package/dist/resources/extensions/gsd/hook-emitter.js +108 -0
  50. package/dist/resources/extensions/gsd/init-wizard.js +15 -1
  51. package/dist/resources/extensions/gsd/key-manager.js +28 -0
  52. package/dist/resources/extensions/gsd/memory-backfill.js +126 -0
  53. package/dist/resources/extensions/gsd/memory-store.js +19 -0
  54. package/dist/resources/extensions/gsd/model-router.js +36 -3
  55. package/dist/resources/extensions/gsd/pre-execution-checks.js +44 -9
  56. package/dist/resources/extensions/gsd/preferences-types.js +9 -0
  57. package/dist/resources/extensions/gsd/preferences-validation.js +83 -0
  58. package/dist/resources/extensions/gsd/preferences.js +17 -17
  59. package/dist/resources/extensions/gsd/prompt-loader.js +22 -7
  60. package/dist/resources/extensions/gsd/prompts/complete-milestone.md +1 -1
  61. package/dist/resources/extensions/gsd/prompts/complete-slice.md +2 -2
  62. package/dist/resources/extensions/gsd/prompts/debug-diagnose.md +2 -0
  63. package/dist/resources/extensions/gsd/prompts/discuss-headless.md +8 -0
  64. package/dist/resources/extensions/gsd/prompts/discuss.md +29 -2
  65. package/dist/resources/extensions/gsd/prompts/execute-task.md +3 -2
  66. package/dist/resources/extensions/gsd/prompts/parallel-research-slices.md +5 -2
  67. package/dist/resources/extensions/gsd/prompts/plan-slice.md +1 -0
  68. package/dist/resources/extensions/gsd/prompts/research-slice.md +1 -0
  69. package/dist/resources/extensions/gsd/safety/evidence-collector.js +96 -0
  70. package/dist/resources/extensions/gsd/safety/file-change-validator.js +13 -5
  71. package/dist/resources/extensions/gsd/safety/safety-harness.js +5 -1
  72. package/dist/resources/extensions/gsd/state.js +43 -4
  73. package/dist/resources/extensions/gsd/token-counter.js +22 -5
  74. package/dist/resources/extensions/gsd/tools/complete-milestone.js +16 -10
  75. package/dist/resources/extensions/gsd/tools/exec-search-tool.js +59 -0
  76. package/dist/resources/extensions/gsd/tools/exec-tool.js +126 -0
  77. package/dist/resources/extensions/gsd/tools/memory-tools.js +26 -1
  78. package/dist/resources/extensions/gsd/tools/resume-tool.js +23 -0
  79. package/dist/resources/extensions/gsd/uok/plan-v2.js +20 -3
  80. package/dist/resources/extensions/gsd/workflow-mcp.js +3 -0
  81. package/dist/resources/extensions/gsd/workflow-templates/spike.md +6 -0
  82. package/dist/resources/extensions/gsd/worktree-resolver.js +50 -10
  83. package/dist/resources/extensions/search-the-web/command-search-provider.js +5 -4
  84. package/dist/resources/extensions/search-the-web/native-search.js +45 -13
  85. package/dist/resources/skills/api-design/SKILL.md +190 -0
  86. package/dist/resources/skills/create-mcp-server/SKILL.md +121 -0
  87. package/dist/resources/skills/decompose-into-slices/SKILL.md +139 -0
  88. package/dist/resources/skills/dependency-upgrade/SKILL.md +158 -0
  89. package/dist/resources/skills/design-an-interface/SKILL.md +102 -0
  90. package/dist/resources/skills/forensics/SKILL.md +153 -0
  91. package/dist/resources/skills/grill-me/SKILL.md +93 -0
  92. package/dist/resources/skills/handoff/SKILL.md +121 -0
  93. package/dist/resources/skills/observability/SKILL.md +174 -0
  94. package/dist/resources/skills/security-review/SKILL.md +181 -0
  95. package/dist/resources/skills/spike-wrap-up/SKILL.md +138 -0
  96. package/dist/resources/skills/tdd/SKILL.md +112 -0
  97. package/dist/resources/skills/verify-before-complete/SKILL.md +98 -0
  98. package/dist/resources/skills/write-docs/SKILL.md +82 -0
  99. package/dist/resources/skills/write-milestone-brief/SKILL.md +135 -0
  100. package/dist/tsconfig.extensions.tsbuildinfo +1 -1
  101. package/dist/web/standalone/.next/BUILD_ID +1 -1
  102. package/dist/web/standalone/.next/app-path-routes-manifest.json +10 -10
  103. package/dist/web/standalone/.next/build-manifest.json +2 -2
  104. package/dist/web/standalone/.next/required-server-files.json +1 -1
  105. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  106. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  107. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  108. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  109. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  110. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  111. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  112. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  113. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  114. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  115. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  116. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  117. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  118. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  119. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  120. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  121. package/dist/web/standalone/.next/server/app/index.html +1 -1
  122. package/dist/web/standalone/.next/server/app/index.rsc +1 -1
  123. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  124. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
  125. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  126. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  127. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  128. package/dist/web/standalone/.next/server/app-paths-manifest.json +10 -10
  129. package/dist/web/standalone/.next/server/chunks/6897.js +2 -2
  130. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  131. package/dist/web/standalone/.next/server/middleware-manifest.json +1 -1
  132. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  133. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  134. package/dist/web/standalone/server.js +1 -1
  135. package/dist/welcome-screen.js +6 -1
  136. package/dist/wizard.js +2 -0
  137. package/package.json +1 -1
  138. package/packages/daemon/package.json +2 -2
  139. package/packages/mcp-server/dist/remote-questions.d.ts +45 -0
  140. package/packages/mcp-server/dist/remote-questions.d.ts.map +1 -0
  141. package/packages/mcp-server/dist/remote-questions.js +732 -0
  142. package/packages/mcp-server/dist/remote-questions.js.map +1 -0
  143. package/packages/mcp-server/dist/server.d.ts +7 -0
  144. package/packages/mcp-server/dist/server.d.ts.map +1 -1
  145. package/packages/mcp-server/dist/server.js +70 -8
  146. package/packages/mcp-server/dist/server.js.map +1 -1
  147. package/packages/mcp-server/dist/session-manager.d.ts +14 -0
  148. package/packages/mcp-server/dist/session-manager.d.ts.map +1 -1
  149. package/packages/mcp-server/dist/session-manager.js +49 -1
  150. package/packages/mcp-server/dist/session-manager.js.map +1 -1
  151. package/packages/mcp-server/dist/workflow-tools.d.ts +1 -1
  152. package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
  153. package/packages/mcp-server/dist/workflow-tools.js +163 -25
  154. package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
  155. package/packages/mcp-server/package.json +4 -3
  156. package/packages/mcp-server/src/mcp-server.test.ts +67 -0
  157. package/packages/mcp-server/src/remote-questions.test.ts +294 -0
  158. package/packages/mcp-server/src/remote-questions.ts +916 -0
  159. package/packages/mcp-server/src/server.ts +89 -14
  160. package/packages/mcp-server/src/session-manager.ts +43 -1
  161. package/packages/mcp-server/src/workflow-tools.test.ts +146 -1
  162. package/packages/mcp-server/src/workflow-tools.ts +215 -43
  163. package/packages/mcp-server/tsconfig.test.json +19 -0
  164. package/packages/mcp-server/tsconfig.tsbuildinfo +1 -1
  165. package/packages/native/package.json +1 -1
  166. package/packages/pi-agent-core/dist/agent-loop.js +12 -0
  167. package/packages/pi-agent-core/dist/agent-loop.js.map +1 -1
  168. package/packages/pi-agent-core/dist/types.d.ts +30 -0
  169. package/packages/pi-agent-core/dist/types.d.ts.map +1 -1
  170. package/packages/pi-agent-core/dist/types.js.map +1 -1
  171. package/packages/pi-agent-core/package.json +1 -1
  172. package/packages/pi-agent-core/src/agent-loop.ts +14 -0
  173. package/packages/pi-agent-core/src/types.ts +34 -0
  174. package/packages/pi-agent-core/tsconfig.tsbuildinfo +1 -1
  175. package/packages/pi-ai/dist/models/custom.d.ts +38 -0
  176. package/packages/pi-ai/dist/models/custom.d.ts.map +1 -1
  177. package/packages/pi-ai/dist/models/custom.js +41 -0
  178. package/packages/pi-ai/dist/models/custom.js.map +1 -1
  179. package/packages/pi-ai/dist/providers/anthropic-auth.test.js +1 -1
  180. package/packages/pi-ai/dist/providers/anthropic-auth.test.js.map +1 -1
  181. package/packages/pi-ai/dist/providers/anthropic-bearer-auth.test.d.ts +2 -0
  182. package/packages/pi-ai/dist/providers/anthropic-bearer-auth.test.d.ts.map +1 -0
  183. package/packages/pi-ai/dist/providers/anthropic-bearer-auth.test.js +13 -0
  184. package/packages/pi-ai/dist/providers/anthropic-bearer-auth.test.js.map +1 -0
  185. package/packages/pi-ai/dist/providers/anthropic-shared.d.ts.map +1 -1
  186. package/packages/pi-ai/dist/providers/anthropic-shared.js +27 -4
  187. package/packages/pi-ai/dist/providers/anthropic-shared.js.map +1 -1
  188. package/packages/pi-ai/dist/providers/anthropic.d.ts.map +1 -1
  189. package/packages/pi-ai/dist/providers/anthropic.js +13 -4
  190. package/packages/pi-ai/dist/providers/anthropic.js.map +1 -1
  191. package/packages/pi-ai/dist/providers/minimax-tool-name.test.d.ts +2 -0
  192. package/packages/pi-ai/dist/providers/minimax-tool-name.test.d.ts.map +1 -0
  193. package/packages/pi-ai/dist/providers/minimax-tool-name.test.js +80 -0
  194. package/packages/pi-ai/dist/providers/minimax-tool-name.test.js.map +1 -0
  195. package/packages/pi-ai/dist/providers/openai-completions.d.ts.map +1 -1
  196. package/packages/pi-ai/dist/providers/openai-completions.js +60 -15
  197. package/packages/pi-ai/dist/providers/openai-completions.js.map +1 -1
  198. package/packages/pi-ai/dist/providers/simple-options.d.ts +10 -0
  199. package/packages/pi-ai/dist/providers/simple-options.d.ts.map +1 -1
  200. package/packages/pi-ai/dist/providers/simple-options.js +16 -1
  201. package/packages/pi-ai/dist/providers/simple-options.js.map +1 -1
  202. package/packages/pi-ai/dist/providers/think-tag-parser.d.ts +17 -0
  203. package/packages/pi-ai/dist/providers/think-tag-parser.d.ts.map +1 -0
  204. package/packages/pi-ai/dist/providers/think-tag-parser.js +75 -0
  205. package/packages/pi-ai/dist/providers/think-tag-parser.js.map +1 -0
  206. package/packages/pi-ai/dist/providers/think-tag-parser.test.d.ts +2 -0
  207. package/packages/pi-ai/dist/providers/think-tag-parser.test.d.ts.map +1 -0
  208. package/packages/pi-ai/dist/providers/think-tag-parser.test.js +41 -0
  209. package/packages/pi-ai/dist/providers/think-tag-parser.test.js.map +1 -0
  210. package/packages/pi-ai/dist/utils/oauth/github-copilot.d.ts.map +1 -1
  211. package/packages/pi-ai/dist/utils/oauth/github-copilot.js +12 -2
  212. package/packages/pi-ai/dist/utils/oauth/github-copilot.js.map +1 -1
  213. package/packages/pi-ai/dist/utils/oauth/github-copilot.test.js +164 -14
  214. package/packages/pi-ai/dist/utils/oauth/github-copilot.test.js.map +1 -1
  215. package/packages/pi-ai/dist/utils/oauth/google-antigravity.d.ts.map +1 -1
  216. package/packages/pi-ai/dist/utils/oauth/google-antigravity.js +15 -3
  217. package/packages/pi-ai/dist/utils/oauth/google-antigravity.js.map +1 -1
  218. package/packages/pi-ai/dist/utils/oauth/google-antigravity.test.d.ts +2 -0
  219. package/packages/pi-ai/dist/utils/oauth/google-antigravity.test.d.ts.map +1 -0
  220. package/packages/pi-ai/dist/utils/oauth/google-antigravity.test.js +67 -0
  221. package/packages/pi-ai/dist/utils/oauth/google-antigravity.test.js.map +1 -0
  222. package/packages/pi-ai/dist/utils/oauth/google-gemini-cli.d.ts.map +1 -1
  223. package/packages/pi-ai/dist/utils/oauth/google-gemini-cli.js +16 -3
  224. package/packages/pi-ai/dist/utils/oauth/google-gemini-cli.js.map +1 -1
  225. package/packages/pi-ai/dist/utils/oauth/google-gemini-cli.test.d.ts +2 -0
  226. package/packages/pi-ai/dist/utils/oauth/google-gemini-cli.test.d.ts.map +1 -0
  227. package/packages/pi-ai/dist/utils/oauth/google-gemini-cli.test.js +67 -0
  228. package/packages/pi-ai/dist/utils/oauth/google-gemini-cli.test.js.map +1 -0
  229. package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.d.ts +2 -0
  230. package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.d.ts.map +1 -0
  231. package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.js +289 -0
  232. package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.js.map +1 -0
  233. package/packages/pi-ai/package.json +1 -1
  234. package/packages/pi-ai/src/models/custom.ts +42 -0
  235. package/packages/pi-ai/src/providers/anthropic-auth.test.ts +1 -1
  236. package/packages/pi-ai/src/providers/anthropic-bearer-auth.test.ts +26 -0
  237. package/packages/pi-ai/src/providers/anthropic-shared.ts +26 -5
  238. package/packages/pi-ai/src/providers/anthropic.ts +15 -4
  239. package/packages/pi-ai/src/providers/minimax-tool-name.test.ts +98 -0
  240. package/packages/pi-ai/src/providers/openai-completions.ts +57 -16
  241. package/packages/pi-ai/src/providers/simple-options.ts +17 -1
  242. package/packages/pi-ai/src/providers/think-tag-parser.test.ts +44 -0
  243. package/packages/pi-ai/src/providers/think-tag-parser.ts +94 -0
  244. package/packages/pi-ai/src/utils/oauth/github-copilot.test.ts +200 -23
  245. package/packages/pi-ai/src/utils/oauth/github-copilot.ts +12 -2
  246. package/packages/pi-ai/src/utils/oauth/google-antigravity.test.ts +84 -0
  247. package/packages/pi-ai/src/utils/oauth/google-antigravity.ts +15 -5
  248. package/packages/pi-ai/src/utils/oauth/google-gemini-cli.test.ts +84 -0
  249. package/packages/pi-ai/src/utils/oauth/google-gemini-cli.ts +16 -5
  250. package/packages/pi-ai/src/utils/oauth/oauth-providers.test.ts +363 -0
  251. package/packages/pi-ai/tsconfig.tsbuildinfo +1 -1
  252. package/packages/pi-coding-agent/dist/core/agent-session-abort-order.test.js +3 -2
  253. package/packages/pi-coding-agent/dist/core/agent-session-abort-order.test.js.map +1 -1
  254. package/packages/pi-coding-agent/dist/core/agent-session.d.ts +2 -0
  255. package/packages/pi-coding-agent/dist/core/agent-session.d.ts.map +1 -1
  256. package/packages/pi-coding-agent/dist/core/agent-session.js +32 -2
  257. package/packages/pi-coding-agent/dist/core/agent-session.js.map +1 -1
  258. package/packages/pi-coding-agent/dist/core/extensions/index.d.ts +1 -1
  259. package/packages/pi-coding-agent/dist/core/extensions/index.d.ts.map +1 -1
  260. package/packages/pi-coding-agent/dist/core/extensions/index.js.map +1 -1
  261. package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts.map +1 -1
  262. package/packages/pi-coding-agent/dist/core/extensions/loader.js +4 -0
  263. package/packages/pi-coding-agent/dist/core/extensions/loader.js.map +1 -1
  264. package/packages/pi-coding-agent/dist/core/extensions/runner.d.ts +35 -2
  265. package/packages/pi-coding-agent/dist/core/extensions/runner.d.ts.map +1 -1
  266. package/packages/pi-coding-agent/dist/core/extensions/runner.js +233 -0
  267. package/packages/pi-coding-agent/dist/core/extensions/runner.js.map +1 -1
  268. package/packages/pi-coding-agent/dist/core/extensions/types.d.ts +205 -2
  269. package/packages/pi-coding-agent/dist/core/extensions/types.d.ts.map +1 -1
  270. package/packages/pi-coding-agent/dist/core/extensions/types.js.map +1 -1
  271. package/packages/pi-coding-agent/dist/core/hooks-runner.d.ts +53 -0
  272. package/packages/pi-coding-agent/dist/core/hooks-runner.d.ts.map +1 -0
  273. package/packages/pi-coding-agent/dist/core/hooks-runner.js +337 -0
  274. package/packages/pi-coding-agent/dist/core/hooks-runner.js.map +1 -0
  275. package/packages/pi-coding-agent/dist/core/hooks-runner.test.d.ts +2 -0
  276. package/packages/pi-coding-agent/dist/core/hooks-runner.test.d.ts.map +1 -0
  277. package/packages/pi-coding-agent/dist/core/hooks-runner.test.js +234 -0
  278. package/packages/pi-coding-agent/dist/core/hooks-runner.test.js.map +1 -0
  279. package/packages/pi-coding-agent/dist/core/index.d.ts +1 -0
  280. package/packages/pi-coding-agent/dist/core/index.d.ts.map +1 -1
  281. package/packages/pi-coding-agent/dist/core/index.js +1 -0
  282. package/packages/pi-coding-agent/dist/core/index.js.map +1 -1
  283. package/packages/pi-coding-agent/dist/core/model-discovery.d.ts +3 -1
  284. package/packages/pi-coding-agent/dist/core/model-discovery.d.ts.map +1 -1
  285. package/packages/pi-coding-agent/dist/core/model-discovery.js +92 -12
  286. package/packages/pi-coding-agent/dist/core/model-discovery.js.map +1 -1
  287. package/packages/pi-coding-agent/dist/core/model-discovery.test.js +16 -1
  288. package/packages/pi-coding-agent/dist/core/model-discovery.test.js.map +1 -1
  289. package/packages/pi-coding-agent/dist/core/model-registry-auth-header.test.d.ts +2 -0
  290. package/packages/pi-coding-agent/dist/core/model-registry-auth-header.test.d.ts.map +1 -0
  291. package/packages/pi-coding-agent/dist/core/model-registry-auth-header.test.js +40 -0
  292. package/packages/pi-coding-agent/dist/core/model-registry-auth-header.test.js.map +1 -0
  293. package/packages/pi-coding-agent/dist/core/model-registry-custom-caps.test.d.ts +2 -0
  294. package/packages/pi-coding-agent/dist/core/model-registry-custom-caps.test.d.ts.map +1 -0
  295. package/packages/pi-coding-agent/dist/core/model-registry-custom-caps.test.js +203 -0
  296. package/packages/pi-coding-agent/dist/core/model-registry-custom-caps.test.js.map +1 -0
  297. package/packages/pi-coding-agent/dist/core/model-registry-discovery.test.js +61 -1
  298. package/packages/pi-coding-agent/dist/core/model-registry-discovery.test.js.map +1 -1
  299. package/packages/pi-coding-agent/dist/core/model-registry.d.ts +5 -0
  300. package/packages/pi-coding-agent/dist/core/model-registry.d.ts.map +1 -1
  301. package/packages/pi-coding-agent/dist/core/model-registry.js +90 -10
  302. package/packages/pi-coding-agent/dist/core/model-registry.js.map +1 -1
  303. package/packages/pi-coding-agent/dist/core/redact-secrets.d.ts +2 -0
  304. package/packages/pi-coding-agent/dist/core/redact-secrets.d.ts.map +1 -0
  305. package/packages/pi-coding-agent/dist/core/redact-secrets.js +49 -0
  306. package/packages/pi-coding-agent/dist/core/redact-secrets.js.map +1 -0
  307. package/packages/pi-coding-agent/dist/core/redact-secrets.test.d.ts +2 -0
  308. package/packages/pi-coding-agent/dist/core/redact-secrets.test.d.ts.map +1 -0
  309. package/packages/pi-coding-agent/dist/core/redact-secrets.test.js +67 -0
  310. package/packages/pi-coding-agent/dist/core/redact-secrets.test.js.map +1 -0
  311. package/packages/pi-coding-agent/dist/core/session-manager.d.ts.map +1 -1
  312. package/packages/pi-coding-agent/dist/core/session-manager.js +10 -6
  313. package/packages/pi-coding-agent/dist/core/session-manager.js.map +1 -1
  314. package/packages/pi-coding-agent/dist/core/session-manager.test.js +45 -1
  315. package/packages/pi-coding-agent/dist/core/session-manager.test.js.map +1 -1
  316. package/packages/pi-coding-agent/dist/core/settings-manager.d.ts +55 -0
  317. package/packages/pi-coding-agent/dist/core/settings-manager.d.ts.map +1 -1
  318. package/packages/pi-coding-agent/dist/core/settings-manager.js.map +1 -1
  319. package/packages/pi-coding-agent/dist/index.d.ts +1 -1
  320. package/packages/pi-coding-agent/dist/index.d.ts.map +1 -1
  321. package/packages/pi-coding-agent/dist/index.js.map +1 -1
  322. package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.d.ts +1 -1
  323. package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.d.ts.map +1 -1
  324. package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.js +5 -4
  325. package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.js.map +1 -1
  326. package/packages/pi-coding-agent/dist/modes/interactive/components/provider-manager.d.ts.map +1 -1
  327. package/packages/pi-coding-agent/dist/modes/interactive/components/provider-manager.js +13 -7
  328. package/packages/pi-coding-agent/dist/modes/interactive/components/provider-manager.js.map +1 -1
  329. package/packages/pi-coding-agent/dist/modes/interactive/components/skill-invocation-message.d.ts +7 -6
  330. package/packages/pi-coding-agent/dist/modes/interactive/components/skill-invocation-message.d.ts.map +1 -1
  331. package/packages/pi-coding-agent/dist/modes/interactive/components/skill-invocation-message.js +29 -21
  332. package/packages/pi-coding-agent/dist/modes/interactive/components/skill-invocation-message.js.map +1 -1
  333. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  334. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +13 -1
  335. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
  336. package/packages/pi-coding-agent/package.json +1 -1
  337. package/packages/pi-coding-agent/src/core/agent-session-abort-order.test.ts +3 -2
  338. package/packages/pi-coding-agent/src/core/agent-session.ts +38 -2
  339. package/packages/pi-coding-agent/src/core/extensions/index.ts +16 -0
  340. package/packages/pi-coding-agent/src/core/extensions/loader.ts +5 -0
  341. package/packages/pi-coding-agent/src/core/extensions/runner.ts +351 -0
  342. package/packages/pi-coding-agent/src/core/extensions/types.ts +258 -0
  343. package/packages/pi-coding-agent/src/core/hooks-runner.test.ts +269 -0
  344. package/packages/pi-coding-agent/src/core/hooks-runner.ts +460 -0
  345. package/packages/pi-coding-agent/src/core/index.ts +10 -0
  346. package/packages/pi-coding-agent/src/core/model-discovery.test.ts +19 -0
  347. package/packages/pi-coding-agent/src/core/model-discovery.ts +99 -12
  348. package/packages/pi-coding-agent/src/core/model-registry-auth-header.test.ts +44 -0
  349. package/packages/pi-coding-agent/src/core/model-registry-custom-caps.test.ts +245 -0
  350. package/packages/pi-coding-agent/src/core/model-registry-discovery.test.ts +75 -0
  351. package/packages/pi-coding-agent/src/core/model-registry.ts +102 -10
  352. package/packages/pi-coding-agent/src/core/redact-secrets.test.ts +86 -0
  353. package/packages/pi-coding-agent/src/core/redact-secrets.ts +58 -0
  354. package/packages/pi-coding-agent/src/core/session-manager.test.ts +65 -1
  355. package/packages/pi-coding-agent/src/core/session-manager.ts +10 -6
  356. package/packages/pi-coding-agent/src/core/settings-manager.ts +57 -0
  357. package/packages/pi-coding-agent/src/index.ts +16 -0
  358. package/packages/pi-coding-agent/src/modes/interactive/components/chat-frame.ts +6 -6
  359. package/packages/pi-coding-agent/src/modes/interactive/components/provider-manager.ts +16 -7
  360. package/packages/pi-coding-agent/src/modes/interactive/components/skill-invocation-message.ts +36 -22
  361. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +13 -1
  362. package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -1
  363. package/packages/pi-tui/package.json +1 -1
  364. package/packages/rpc-client/package.json +1 -1
  365. package/pkg/package.json +1 -1
  366. package/scripts/link-workspace-packages.cjs +1 -0
  367. package/src/resources/agents/researcher.md +1 -1
  368. package/src/resources/extensions/claude-code-cli/readiness.ts +32 -8
  369. package/src/resources/extensions/claude-code-cli/stream-adapter.ts +78 -17
  370. package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +149 -5
  371. package/src/resources/extensions/gsd/auto/loop-deps.ts +14 -0
  372. package/src/resources/extensions/gsd/auto/loop.ts +9 -0
  373. package/src/resources/extensions/gsd/auto/phases.ts +131 -10
  374. package/src/resources/extensions/gsd/auto/run-unit.ts +40 -2
  375. package/src/resources/extensions/gsd/auto/session.ts +35 -2
  376. package/src/resources/extensions/gsd/auto-dispatch.ts +16 -3
  377. package/src/resources/extensions/gsd/auto-model-selection.ts +71 -15
  378. package/src/resources/extensions/gsd/auto-post-unit.ts +29 -3
  379. package/src/resources/extensions/gsd/auto-prompts.ts +28 -1
  380. package/src/resources/extensions/gsd/auto-recovery.ts +26 -1
  381. package/src/resources/extensions/gsd/auto-start.ts +60 -68
  382. package/src/resources/extensions/gsd/auto-verification.ts +33 -0
  383. package/src/resources/extensions/gsd/auto-worktree.ts +62 -63
  384. package/src/resources/extensions/gsd/auto.ts +73 -28
  385. package/src/resources/extensions/gsd/blocked-models.ts +98 -0
  386. package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +120 -1
  387. package/src/resources/extensions/gsd/bootstrap/db-tools.ts +40 -9
  388. package/src/resources/extensions/gsd/bootstrap/exec-tools.ts +109 -0
  389. package/src/resources/extensions/gsd/bootstrap/memory-tools.ts +5 -0
  390. package/src/resources/extensions/gsd/bootstrap/register-extension.ts +15 -0
  391. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +54 -6
  392. package/src/resources/extensions/gsd/bootstrap/system-context.ts +89 -26
  393. package/src/resources/extensions/gsd/bootstrap/write-gate.ts +35 -2
  394. package/src/resources/extensions/gsd/clean-root-preflight.ts +111 -0
  395. package/src/resources/extensions/gsd/commands-extract-learnings.ts +55 -90
  396. package/src/resources/extensions/gsd/commands-prefs-wizard.ts +898 -32
  397. package/src/resources/extensions/gsd/compaction-snapshot.ts +165 -0
  398. package/src/resources/extensions/gsd/complexity-classifier.ts +5 -3
  399. package/src/resources/extensions/gsd/db-writer.ts +88 -17
  400. package/src/resources/extensions/gsd/doctor-git-checks.ts +23 -27
  401. package/src/resources/extensions/gsd/doctor-providers.ts +59 -6
  402. package/src/resources/extensions/gsd/ecosystem/gsd-extension-api.ts +2 -0
  403. package/src/resources/extensions/gsd/error-classifier.ts +36 -3
  404. package/src/resources/extensions/gsd/exec-history.ts +153 -0
  405. package/src/resources/extensions/gsd/exec-sandbox.ts +326 -0
  406. package/src/resources/extensions/gsd/gitignore.ts +1 -1
  407. package/src/resources/extensions/gsd/gsd-db.ts +186 -23
  408. package/src/resources/extensions/gsd/guided-flow.ts +222 -1
  409. package/src/resources/extensions/gsd/health-widget.ts +3 -1
  410. package/src/resources/extensions/gsd/hook-emitter.ts +188 -0
  411. package/src/resources/extensions/gsd/init-wizard.ts +15 -1
  412. package/src/resources/extensions/gsd/journal.ts +2 -1
  413. package/src/resources/extensions/gsd/key-manager.ts +28 -0
  414. package/src/resources/extensions/gsd/memory-backfill.ts +140 -0
  415. package/src/resources/extensions/gsd/memory-store.ts +26 -0
  416. package/src/resources/extensions/gsd/model-router.ts +42 -1
  417. package/src/resources/extensions/gsd/pre-execution-checks.ts +46 -10
  418. package/src/resources/extensions/gsd/preferences-types.ts +46 -0
  419. package/src/resources/extensions/gsd/preferences-validation.ts +79 -0
  420. package/src/resources/extensions/gsd/preferences.ts +17 -17
  421. package/src/resources/extensions/gsd/prompt-loader.ts +30 -7
  422. package/src/resources/extensions/gsd/prompts/complete-milestone.md +1 -1
  423. package/src/resources/extensions/gsd/prompts/complete-slice.md +2 -2
  424. package/src/resources/extensions/gsd/prompts/debug-diagnose.md +2 -0
  425. package/src/resources/extensions/gsd/prompts/discuss-headless.md +8 -0
  426. package/src/resources/extensions/gsd/prompts/discuss.md +29 -2
  427. package/src/resources/extensions/gsd/prompts/execute-task.md +3 -2
  428. package/src/resources/extensions/gsd/prompts/parallel-research-slices.md +5 -2
  429. package/src/resources/extensions/gsd/prompts/plan-slice.md +1 -0
  430. package/src/resources/extensions/gsd/prompts/research-slice.md +1 -0
  431. package/src/resources/extensions/gsd/safety/evidence-collector.ts +119 -0
  432. package/src/resources/extensions/gsd/safety/file-change-validator.ts +17 -4
  433. package/src/resources/extensions/gsd/safety/safety-harness.ts +9 -0
  434. package/src/resources/extensions/gsd/state.ts +45 -4
  435. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +188 -2
  436. package/src/resources/extensions/gsd/tests/auto-model-selection.test.ts +95 -1
  437. package/src/resources/extensions/gsd/tests/auto-paused-session-validation.test.ts +12 -0
  438. package/src/resources/extensions/gsd/tests/auto-recovery.test.ts +49 -0
  439. package/src/resources/extensions/gsd/tests/auto-start-bootstrap-await-3420.test.ts +141 -0
  440. package/src/resources/extensions/gsd/tests/auto-start-model-capture.test.ts +33 -3
  441. package/src/resources/extensions/gsd/tests/auto-thinking-restore.test.ts +38 -0
  442. package/src/resources/extensions/gsd/tests/auto-wrapup-inflight-guard.test.ts +23 -0
  443. package/src/resources/extensions/gsd/tests/blocked-models.test.ts +98 -0
  444. package/src/resources/extensions/gsd/tests/bundled-skill-triggers.test.ts +54 -0
  445. package/src/resources/extensions/gsd/tests/clean-root-preflight.test.ts +186 -0
  446. package/src/resources/extensions/gsd/tests/commands-extract-learnings.test.ts +68 -66
  447. package/src/resources/extensions/gsd/tests/compaction-snapshot.test.ts +123 -0
  448. package/src/resources/extensions/gsd/tests/complete-milestone.test.ts +61 -1
  449. package/src/resources/extensions/gsd/tests/complete-slice.test.ts +2 -2
  450. package/src/resources/extensions/gsd/tests/complete-task.test.ts +2 -2
  451. package/src/resources/extensions/gsd/tests/complexity-classifier.test.ts +3 -3
  452. package/src/resources/extensions/gsd/tests/custom-engine-loop-integration.test.ts +2 -0
  453. package/src/resources/extensions/gsd/tests/derive-state-db.test.ts +42 -0
  454. package/src/resources/extensions/gsd/tests/derive-state-helpers.test.ts +8 -4
  455. package/src/resources/extensions/gsd/tests/doctor-providers.test.ts +148 -3
  456. package/src/resources/extensions/gsd/tests/double-merge-guard.test.ts +1 -1
  457. package/src/resources/extensions/gsd/tests/ensure-db-open.test.ts +306 -1
  458. package/src/resources/extensions/gsd/tests/escalation.test.ts +1 -1
  459. package/src/resources/extensions/gsd/tests/exec-history.test.ts +237 -0
  460. package/src/resources/extensions/gsd/tests/exec-sandbox.test.ts +210 -0
  461. package/src/resources/extensions/gsd/tests/file-change-validator.test.ts +58 -0
  462. package/src/resources/extensions/gsd/tests/flat-rate-routing-guard.test.ts +40 -9
  463. package/src/resources/extensions/gsd/tests/freeform-decisions.test.ts +62 -0
  464. package/src/resources/extensions/gsd/tests/gsd-db.test.ts +447 -1
  465. package/src/resources/extensions/gsd/tests/init-wizard.test.ts +27 -0
  466. package/src/resources/extensions/gsd/tests/integration/doctor-git-symlink-cwd.test.ts +11 -0
  467. package/src/resources/extensions/gsd/tests/integration/doctor-git.test.ts +78 -0
  468. package/src/resources/extensions/gsd/tests/integration/git-service.test.ts +1 -0
  469. package/src/resources/extensions/gsd/tests/integration/gitignore-tracked-gsd.test.ts +1 -0
  470. package/src/resources/extensions/gsd/tests/integration/idle-recovery.test.ts +30 -0
  471. package/src/resources/extensions/gsd/tests/interactive-routing-bypass.test.ts +1 -1
  472. package/src/resources/extensions/gsd/tests/isolation-none-branch-guard.test.ts +1 -1
  473. package/src/resources/extensions/gsd/tests/issue-4540-regressions.test.ts +288 -0
  474. package/src/resources/extensions/gsd/tests/journal-integration.test.ts +37 -0
  475. package/src/resources/extensions/gsd/tests/key-manager.test.ts +9 -0
  476. package/src/resources/extensions/gsd/tests/load-memory-block.test.ts +36 -0
  477. package/src/resources/extensions/gsd/tests/md-importer.test.ts +1 -1
  478. package/src/resources/extensions/gsd/tests/memory-pressure-stuck-state.test.ts +12 -0
  479. package/src/resources/extensions/gsd/tests/memory-store.test.ts +2 -2
  480. package/src/resources/extensions/gsd/tests/parallel-research-dispatch.test.ts +19 -0
  481. package/src/resources/extensions/gsd/tests/plan-gate-failed-doctor-heal-hint.test.ts +37 -0
  482. package/src/resources/extensions/gsd/tests/pre-exec-backtick-strip.test.ts +14 -0
  483. package/src/resources/extensions/gsd/tests/pre-exec-gate-loop.test.ts +272 -0
  484. package/src/resources/extensions/gsd/tests/pre-execution-checks.test.ts +356 -0
  485. package/src/resources/extensions/gsd/tests/preferences.test.ts +110 -0
  486. package/src/resources/extensions/gsd/tests/prefs-wizard-coverage.test.ts +44 -0
  487. package/src/resources/extensions/gsd/tests/prompt-loader-extension-dir.test.ts +49 -0
  488. package/src/resources/extensions/gsd/tests/provider-errors.test.ts +103 -4
  489. package/src/resources/extensions/gsd/tests/ready-phrase-no-files-4573.test.ts +388 -0
  490. package/src/resources/extensions/gsd/tests/restore-tools-after-discuss.test.ts +9 -3
  491. package/src/resources/extensions/gsd/tests/resume-dispatch-worktree.test.ts +230 -0
  492. package/src/resources/extensions/gsd/tests/safety-harness-false-positives.test.ts +205 -0
  493. package/src/resources/extensions/gsd/tests/save-gate-result-render.test.ts +95 -0
  494. package/src/resources/extensions/gsd/tests/schema-v21-sequence.test.ts +413 -0
  495. package/src/resources/extensions/gsd/tests/session-start-footer.test.ts +32 -40
  496. package/src/resources/extensions/gsd/tests/stash-queued-context-files.test.ts +56 -0
  497. package/src/resources/extensions/gsd/tests/token-counter.test.ts +105 -1
  498. package/src/resources/extensions/gsd/tests/tool-compatibility.test.ts +107 -0
  499. package/src/resources/extensions/gsd/tests/uok-plan-v2-wiring.test.ts +23 -0
  500. package/src/resources/extensions/gsd/tests/validate-milestone.test.ts +9 -3
  501. package/src/resources/extensions/gsd/tests/workflow-tool-executors.test.ts +65 -2
  502. package/src/resources/extensions/gsd/tests/worktree-db.test.ts +35 -0
  503. package/src/resources/extensions/gsd/tests/worktree-journal-events.test.ts +6 -1
  504. package/src/resources/extensions/gsd/tests/worktree-resolver.test.ts +78 -5
  505. package/src/resources/extensions/gsd/tests/write-gate.test.ts +64 -0
  506. package/src/resources/extensions/gsd/tests/zombie-gsd-state.test.ts +3 -1
  507. package/src/resources/extensions/gsd/token-counter.ts +22 -5
  508. package/src/resources/extensions/gsd/tools/complete-milestone.ts +15 -9
  509. package/src/resources/extensions/gsd/tools/exec-search-tool.ts +81 -0
  510. package/src/resources/extensions/gsd/tools/exec-tool.ts +183 -0
  511. package/src/resources/extensions/gsd/tools/memory-tools.ts +31 -1
  512. package/src/resources/extensions/gsd/tools/resume-tool.ts +40 -0
  513. package/src/resources/extensions/gsd/uok/plan-v2.ts +26 -3
  514. package/src/resources/extensions/gsd/workflow-logger.ts +4 -1
  515. package/src/resources/extensions/gsd/workflow-mcp.ts +3 -0
  516. package/src/resources/extensions/gsd/workflow-templates/spike.md +6 -0
  517. package/src/resources/extensions/gsd/worktree-resolver.ts +54 -9
  518. package/src/resources/extensions/search-the-web/command-search-provider.ts +5 -4
  519. package/src/resources/extensions/search-the-web/native-search.ts +48 -12
  520. package/src/resources/skills/api-design/SKILL.md +190 -0
  521. package/src/resources/skills/create-mcp-server/SKILL.md +121 -0
  522. package/src/resources/skills/decompose-into-slices/SKILL.md +139 -0
  523. package/src/resources/skills/dependency-upgrade/SKILL.md +158 -0
  524. package/src/resources/skills/design-an-interface/SKILL.md +102 -0
  525. package/src/resources/skills/forensics/SKILL.md +153 -0
  526. package/src/resources/skills/grill-me/SKILL.md +93 -0
  527. package/src/resources/skills/handoff/SKILL.md +121 -0
  528. package/src/resources/skills/observability/SKILL.md +174 -0
  529. package/src/resources/skills/security-review/SKILL.md +181 -0
  530. package/src/resources/skills/spike-wrap-up/SKILL.md +138 -0
  531. package/src/resources/skills/tdd/SKILL.md +112 -0
  532. package/src/resources/skills/verify-before-complete/SKILL.md +98 -0
  533. package/src/resources/skills/write-docs/SKILL.md +82 -0
  534. package/src/resources/skills/write-milestone-brief/SKILL.md +135 -0
  535. /package/dist/web/standalone/.next/static/{ssX7BLv3Dw9Fb4CtrCGeR → pV-mPo7rYGb5JBC09C8GG}/_buildManifest.js +0 -0
  536. /package/dist/web/standalone/.next/static/{ssX7BLv3Dw9Fb4CtrCGeR → pV-mPo7rYGb5JBC09C8GG}/_ssgManifest.js +0 -0
@@ -0,0 +1,245 @@
1
+ /**
2
+ * Regression tests for #4563:
3
+ * Bug 1 — custom/Anthropic-compatible models were hard-capped to 32 k output tokens
4
+ * Bug 2 — custom models in models.json could not declare capabilities.supportsXhigh
5
+ */
6
+ import assert from "node:assert/strict";
7
+ import { mkdirSync, rmSync, writeFileSync } from "node:fs";
8
+ import { tmpdir } from "node:os";
9
+ import { join } from "node:path";
10
+ import { afterEach, beforeEach, describe, it } from "node:test";
11
+ import { AuthStorage } from "./auth-storage.js";
12
+ import { ModelRegistry } from "./model-registry.js";
13
+
14
+ // ─── Helpers ─────────────────────────────────────────────────────────────────
15
+
16
+ let testDir: string;
17
+
18
+ beforeEach(() => {
19
+ testDir = join(
20
+ tmpdir(),
21
+ `model-registry-custom-caps-${Date.now()}-${Math.random().toString(36).slice(2)}`,
22
+ );
23
+ mkdirSync(testDir, { recursive: true });
24
+ });
25
+
26
+ afterEach(() => {
27
+ try {
28
+ rmSync(testDir, { recursive: true, force: true });
29
+ } catch {
30
+ // best-effort cleanup
31
+ }
32
+ });
33
+
34
+ function createRegistry(modelsJson: object): ModelRegistry {
35
+ const path = join(testDir, "models.json");
36
+ writeFileSync(path, JSON.stringify(modelsJson));
37
+ return new ModelRegistry(AuthStorage.inMemory(), path);
38
+ }
39
+
40
+ function writeModelsJson(obj: object): string {
41
+ const path = join(testDir, "models.json");
42
+ writeFileSync(path, JSON.stringify(obj));
43
+ return path;
44
+ }
45
+
46
+ // ─── Bug 1: 32 k cap must not apply to custom/OpenAI-compatible models ────────
47
+
48
+ describe("Bug 1 — maxTokens cap (#4563)", () => {
49
+ it("custom openai-completions model with maxTokens > 32 k is not capped", () => {
50
+ const registry = createRegistry({
51
+ providers: {
52
+ "kimi-custom": {
53
+ baseUrl: "https://api.example.com/v1",
54
+ apiKey: "sk-test",
55
+ api: "openai-completions",
56
+ models: [
57
+ {
58
+ id: "kimi-k2.6-code-preview",
59
+ name: "Kimi K2.6 Code Preview",
60
+ maxTokens: 131072,
61
+ contextWindow: 262144,
62
+ },
63
+ ],
64
+ },
65
+ },
66
+ });
67
+
68
+ const model = registry.getAll().find((m) => m.id === "kimi-k2.6-code-preview");
69
+ assert.ok(model, "model should be registered");
70
+ assert.equal(
71
+ model.maxTokens,
72
+ 131072,
73
+ "maxTokens must be preserved as declared — not capped to 32 000",
74
+ );
75
+ });
76
+
77
+ it("custom model with maxTokens exactly 32 k is not affected", () => {
78
+ const registry = createRegistry({
79
+ providers: {
80
+ "custom-provider": {
81
+ baseUrl: "https://api.example.com/v1",
82
+ apiKey: "sk-test",
83
+ api: "openai-completions",
84
+ models: [{ id: "model-32k", maxTokens: 32000, contextWindow: 128000 }],
85
+ },
86
+ },
87
+ });
88
+
89
+ const model = registry.getAll().find((m) => m.id === "model-32k");
90
+ assert.ok(model);
91
+ assert.equal(model.maxTokens, 32000);
92
+ });
93
+
94
+ it("custom model with maxTokens 65 k is stored at full value", () => {
95
+ const registry = createRegistry({
96
+ providers: {
97
+ "dashscope-custom": {
98
+ baseUrl: "https://dashscope-intl.aliyuncs.com/compatible-mode/v1",
99
+ apiKey: "sk-test",
100
+ api: "openai-completions",
101
+ models: [
102
+ {
103
+ id: "qwen3.5-plus",
104
+ name: "Qwen3.5 Plus",
105
+ maxTokens: 65536,
106
+ contextWindow: 1000000,
107
+ },
108
+ ],
109
+ },
110
+ },
111
+ });
112
+
113
+ const model = registry.getAll().find((m) => m.id === "qwen3.5-plus" && m.provider === "dashscope-custom");
114
+ assert.ok(model);
115
+ assert.equal(model.maxTokens, 65536);
116
+ });
117
+ });
118
+
119
+ // ─── Bug 2: capabilities.supportsXhigh must be declarable in models.json ──────
120
+
121
+ describe("Bug 2 — capabilities.supportsXhigh in models.json (#4563)", () => {
122
+ it("model with capabilities.supportsXhigh: true surfaces the flag", () => {
123
+ const registry = createRegistry({
124
+ providers: {
125
+ "kimi-custom": {
126
+ baseUrl: "https://api.example.com/v1",
127
+ apiKey: "sk-test",
128
+ api: "anthropic-messages",
129
+ models: [
130
+ {
131
+ id: "kimi-k2.6-code-preview",
132
+ name: "Kimi K2.6 Code Preview",
133
+ maxTokens: 131072,
134
+ contextWindow: 262144,
135
+ capabilities: { supportsXhigh: true },
136
+ },
137
+ ],
138
+ },
139
+ },
140
+ });
141
+
142
+ const model = registry.getAll().find((m) => m.id === "kimi-k2.6-code-preview");
143
+ assert.ok(model, "model should be registered");
144
+ assert.equal(
145
+ model.capabilities?.supportsXhigh,
146
+ true,
147
+ "supportsXhigh must be true as declared in models.json",
148
+ );
149
+ });
150
+
151
+ it("model without capabilities declaration has no supportsXhigh", () => {
152
+ const registry = createRegistry({
153
+ providers: {
154
+ "plain-provider": {
155
+ baseUrl: "https://api.example.com/v1",
156
+ apiKey: "sk-test",
157
+ api: "openai-completions",
158
+ models: [{ id: "plain-model", maxTokens: 16384, contextWindow: 128000 }],
159
+ },
160
+ },
161
+ });
162
+
163
+ const model = registry.getAll().find((m) => m.id === "plain-model");
164
+ assert.ok(model);
165
+ // supportsXhigh should be absent or explicitly false — never implicitly true
166
+ assert.ok(
167
+ !model.capabilities?.supportsXhigh,
168
+ "supportsXhigh must not be set for models that don't declare it",
169
+ );
170
+ });
171
+
172
+ it("capabilities.supportsXhigh: false is respected", () => {
173
+ const registry = createRegistry({
174
+ providers: {
175
+ "explicit-provider": {
176
+ baseUrl: "https://api.example.com/v1",
177
+ apiKey: "sk-test",
178
+ api: "openai-completions",
179
+ models: [
180
+ {
181
+ id: "no-xhigh-model",
182
+ capabilities: { supportsXhigh: false },
183
+ },
184
+ ],
185
+ },
186
+ },
187
+ });
188
+
189
+ const model = registry.getAll().find((m) => m.id === "no-xhigh-model");
190
+ assert.ok(model);
191
+ assert.equal(model.capabilities?.supportsXhigh, false);
192
+ });
193
+
194
+ it("supportsXhigh declared in models.json is not overwritten by capability patches", () => {
195
+ // The capability-patches system must not overwrite an explicit declaration in models.json.
196
+ // applyCapabilityPatches uses spread: { ...patch.caps, ...model.capabilities }
197
+ // so model.capabilities wins. This test verifies the precedence end-to-end.
198
+ const registry = createRegistry({
199
+ providers: {
200
+ "compat-provider": {
201
+ baseUrl: "https://api.example.com/v1",
202
+ apiKey: "sk-test",
203
+ api: "openai-completions",
204
+ models: [
205
+ {
206
+ id: "custom-xhigh-model",
207
+ capabilities: { supportsXhigh: true },
208
+ },
209
+ ],
210
+ },
211
+ },
212
+ });
213
+
214
+ const model = registry.getAll().find((m) => m.id === "custom-xhigh-model");
215
+ assert.ok(model);
216
+ assert.equal(model.capabilities?.supportsXhigh, true);
217
+ });
218
+
219
+ it("modelOverrides can set capabilities.supportsXhigh on built-in models", () => {
220
+ // A user-facing override in models.json should be able to add supportsXhigh
221
+ // to a built-in model that doesn't declare it.
222
+ const path = writeModelsJson({
223
+ providers: {
224
+ anthropic: {
225
+ modelOverrides: {
226
+ "claude-3-5-haiku-20241022": {
227
+ capabilities: { supportsXhigh: true },
228
+ },
229
+ },
230
+ },
231
+ },
232
+ });
233
+
234
+ const registry = new ModelRegistry(AuthStorage.inMemory(), path);
235
+ const model = registry.getAll().find(
236
+ (m) => m.provider === "anthropic" && m.id === "claude-3-5-haiku-20241022",
237
+ );
238
+ assert.ok(model, "built-in model must still be present");
239
+ assert.equal(
240
+ model.capabilities?.supportsXhigh,
241
+ true,
242
+ "modelOverrides must be able to set capabilities.supportsXhigh",
243
+ );
244
+ });
245
+ });
@@ -6,6 +6,7 @@ import { afterEach, beforeEach, describe, it } from "node:test";
6
6
  import { AuthStorage } from "./auth-storage.js";
7
7
  import { ModelDiscoveryCache } from "./discovery-cache.js";
8
8
  import { getDefaultTTL, getDiscoverableProviders, getDiscoveryAdapter } from "./model-discovery.js";
9
+ import { ModelRegistry } from "./model-registry.js";
9
10
 
10
11
  let testDir: string;
11
12
 
@@ -133,3 +134,77 @@ describe("Discovery TTL configuration", () => {
133
134
  assert.equal(customTTL, defaultTTL);
134
135
  });
135
136
  });
137
+
138
+ describe("ModelRegistry discovery — OpenAI-compatible custom providers", () => {
139
+ it("discovers custom OpenAI-compatible providers and maps capability metadata", async () => {
140
+ const providerName = `minimax-openai-${Date.now()}-${Math.random().toString(36).slice(2)}`;
141
+ const modelsPath = join(testDir, "models.json");
142
+ writeFileSync(
143
+ modelsPath,
144
+ JSON.stringify(
145
+ {
146
+ providers: {
147
+ [providerName]: {
148
+ baseUrl: "https://api.minimax.example",
149
+ apiKey: "minimax-test-key",
150
+ api: "openai-completions",
151
+ models: [{ id: "bootstrap-model" }],
152
+ },
153
+ },
154
+ },
155
+ null,
156
+ 2,
157
+ ),
158
+ "utf-8",
159
+ );
160
+
161
+ const prevFetch = globalThis.fetch;
162
+ let requestedUrl = "";
163
+ globalThis.fetch = (async (input: string | URL | Request) => {
164
+ requestedUrl = typeof input === "string" ? input : input instanceof URL ? input.toString() : input.url;
165
+ return new Response(
166
+ JSON.stringify({
167
+ data: [
168
+ {
169
+ id: "MiniMax-M2.7-highspeed",
170
+ name: "MiniMax M2.7 Highspeed",
171
+ context_window: 165000,
172
+ max_output_tokens: 32768,
173
+ supports_reasoning: true,
174
+ input_modalities: ["text", "image"],
175
+ },
176
+ ],
177
+ }),
178
+ {
179
+ status: 200,
180
+ headers: { "content-type": "application/json" },
181
+ },
182
+ );
183
+ }) as typeof globalThis.fetch;
184
+
185
+ try {
186
+ const registry = new ModelRegistry(AuthStorage.inMemory({}), modelsPath);
187
+ // Guard against global cache leakage from prior test runs.
188
+ registry.getDiscoveryCache().clear(providerName);
189
+ const results = await registry.discoverModels([providerName]);
190
+
191
+ const discovery = results.find((r) => r.provider === providerName);
192
+ assert.ok(discovery, "discovery result should include custom provider");
193
+ assert.equal(discovery?.error, undefined, "custom provider discovery should succeed");
194
+ assert.equal(requestedUrl, "https://api.minimax.example/v1/models");
195
+
196
+ const discovered = registry
197
+ .getAllWithDiscovered()
198
+ .find((m) => m.provider === providerName && m.id === "MiniMax-M2.7-highspeed");
199
+ assert.ok(discovered, "discovered model should be merged into model list");
200
+ assert.equal(discovered?.api, "openai-completions");
201
+ assert.equal(discovered?.baseUrl, "https://api.minimax.example");
202
+ assert.equal(discovered?.contextWindow, 165000);
203
+ assert.equal(discovered?.maxTokens, 32768);
204
+ assert.equal(discovered?.reasoning, true);
205
+ assert.deepEqual(discovered?.input, ["text", "image"]);
206
+ } finally {
207
+ globalThis.fetch = prevFetch;
208
+ }
209
+ });
210
+ });
@@ -29,7 +29,7 @@ import { getAgentDir } from "../config.js";
29
29
  import type { AuthStorage } from "./auth-storage.js";
30
30
  import { ModelDiscoveryCache } from "./discovery-cache.js";
31
31
  import type { DiscoveredModel, DiscoveryResult } from "./model-discovery.js";
32
- import { getDefaultTTL, getDiscoverableProviders, getDiscoveryAdapter } from "./model-discovery.js";
32
+ import { getDefaultTTL, getDiscoverableProviders, getDiscoveryAdapter, supportsDiscoveryForApi } from "./model-discovery.js";
33
33
  import { clearConfigValueCache, resolveConfigValue, resolveHeaders } from "./resolve-config-value.js";
34
34
  import { isLocalModel } from "./local-model-check.js";
35
35
 
@@ -48,6 +48,14 @@ const VercelGatewayRoutingSchema = Type.Object({
48
48
  order: Type.Optional(Type.Array(Type.String())),
49
49
  });
50
50
 
51
+ // Schema for model capability declarations (mirrors ModelCapabilities in pi-ai types)
52
+ const ModelCapabilitiesSchema = Type.Object({
53
+ supportsXhigh: Type.Optional(Type.Boolean()),
54
+ requiresToolCallId: Type.Optional(Type.Boolean()),
55
+ supportsServiceTier: Type.Optional(Type.Boolean()),
56
+ charsPerToken: Type.Optional(Type.Number()),
57
+ });
58
+
51
59
  // Schema for OpenAI compatibility settings
52
60
  const OpenAICompletionsCompatSchema = Type.Object({
53
61
  supportsStore: Type.Optional(Type.Boolean()),
@@ -91,6 +99,7 @@ const ModelDefinitionSchema = Type.Object({
91
99
  maxTokens: Type.Optional(Type.Number()),
92
100
  headers: Type.Optional(Type.Record(Type.String(), Type.String())),
93
101
  compat: Type.Optional(OpenAICompatSchema),
102
+ capabilities: Type.Optional(ModelCapabilitiesSchema),
94
103
  });
95
104
 
96
105
  // Schema for per-model overrides (all fields optional, merged with built-in model)
@@ -110,6 +119,7 @@ const ModelOverrideSchema = Type.Object({
110
119
  maxTokens: Type.Optional(Type.Number()),
111
120
  headers: Type.Optional(Type.Record(Type.String(), Type.String())),
112
121
  compat: Type.Optional(OpenAICompatSchema),
122
+ capabilities: Type.Optional(ModelCapabilitiesSchema),
113
123
  });
114
124
 
115
125
  type ModelOverride = Static<typeof ModelOverrideSchema>;
@@ -219,6 +229,11 @@ function applyModelOverride(model: Model<Api>, override: ModelOverride): Model<A
219
229
  // Deep merge compat
220
230
  result.compat = mergeCompat(model.compat, override.compat);
221
231
 
232
+ // Merge capabilities (override wins per-field)
233
+ if (override.capabilities) {
234
+ result.capabilities = { ...model.capabilities, ...override.capabilities };
235
+ }
236
+
222
237
  return result;
223
238
  }
224
239
 
@@ -514,6 +529,7 @@ export class ModelRegistry {
514
529
  maxTokens: modelDef.maxTokens ?? 16384,
515
530
  headers,
516
531
  compat: modelDef.compat,
532
+ capabilities: modelDef.capabilities,
517
533
  } as Model<Api>);
518
534
  }
519
535
  }
@@ -788,11 +804,12 @@ export class ModelRegistry {
788
804
  * Results are cached and merged into the registry (never overrides existing models).
789
805
  */
790
806
  async discoverModels(providers?: string[]): Promise<DiscoveryResult[]> {
791
- const targetProviders = providers ?? getDiscoverableProviders();
807
+ const targetProviders = providers ?? this.getAutoDiscoverableProviders();
792
808
  const results: DiscoveryResult[] = [];
793
809
 
794
810
  for (const providerName of targetProviders) {
795
- const adapter = getDiscoveryAdapter(providerName);
811
+ const providerApis = this.getProviderApis(providerName);
812
+ const adapter = getDiscoveryAdapter(providerName, providerApis);
796
813
  if (!adapter.supportsDiscovery) continue;
797
814
 
798
815
  // Skip if cache is still fresh
@@ -812,8 +829,10 @@ export class ModelRegistry {
812
829
  const apiKey = await this.authStorage.getApiKey(providerName);
813
830
  if (!apiKey && !this.isProviderRequestReady(providerName)) continue;
814
831
 
815
- const models = await adapter.fetchModels(apiKey ?? "", undefined);
816
- this.discoveryCache.set(providerName, models);
832
+ const baseUrl = this.getProviderBaseUrl(providerName);
833
+ const models = await adapter.fetchModels(apiKey ?? "", baseUrl);
834
+ const ttlMs = this.getDiscoveryTtl(providerName, providerApis);
835
+ this.discoveryCache.set(providerName, models, ttlMs);
817
836
  results.push({
818
837
  provider: providerName,
819
838
  models,
@@ -865,24 +884,97 @@ export class ModelRegistry {
865
884
  const converted: Model<Api>[] = [];
866
885
  for (const result of results) {
867
886
  if (result.error) continue;
887
+ const providerDefaults = this.getDiscoveryProviderDefaults(result.provider);
868
888
  for (const dm of result.models) {
869
889
  converted.push({
870
890
  id: dm.id,
871
891
  name: dm.name ?? dm.id,
872
- api: "openai" as Api,
892
+ api: providerDefaults.api,
873
893
  provider: result.provider,
874
- baseUrl: "",
894
+ baseUrl: providerDefaults.baseUrl,
875
895
  reasoning: dm.reasoning ?? false,
876
- input: dm.input ?? ["text"],
896
+ input: dm.input ?? providerDefaults.input,
877
897
  cost: dm.cost ?? { input: 0, output: 0, cacheRead: 0, cacheWrite: 0 },
878
- contextWindow: dm.contextWindow ?? 128000,
879
- maxTokens: dm.maxTokens ?? 16384,
898
+ contextWindow: dm.contextWindow ?? providerDefaults.contextWindow,
899
+ maxTokens: dm.maxTokens ?? providerDefaults.maxTokens,
880
900
  } as Model<Api>);
881
901
  }
882
902
  }
883
903
  return converted;
884
904
  }
885
905
 
906
+ private getProviderApis(provider: string): Set<string> {
907
+ const apis = new Set<string>();
908
+ for (const model of this.models) {
909
+ if (model.provider === provider && typeof model.api === "string" && model.api.length > 0) {
910
+ apis.add(model.api);
911
+ }
912
+ }
913
+
914
+ const providerConfig = this.registeredProviders.get(provider);
915
+ if (providerConfig?.api) apis.add(providerConfig.api);
916
+ for (const modelDef of providerConfig?.models ?? []) {
917
+ if (modelDef.api) apis.add(modelDef.api);
918
+ }
919
+ return apis;
920
+ }
921
+
922
+ private getAutoDiscoverableProviders(): string[] {
923
+ const discoverable = new Set<string>(getDiscoverableProviders());
924
+ for (const provider of new Set(this.models.map((m) => m.provider))) {
925
+ const apis = this.getProviderApis(provider);
926
+ for (const api of apis) {
927
+ if (supportsDiscoveryForApi(api)) {
928
+ discoverable.add(provider);
929
+ break;
930
+ }
931
+ }
932
+ }
933
+ return [...discoverable];
934
+ }
935
+
936
+ private getProviderBaseUrl(provider: string): string | undefined {
937
+ const fromModels = this.models.find((m) => m.provider === provider && typeof m.baseUrl === "string" && m.baseUrl.length > 0);
938
+ if (fromModels?.baseUrl) return fromModels.baseUrl;
939
+ return this.registeredProviders.get(provider)?.baseUrl;
940
+ }
941
+
942
+ private getDiscoveryProviderDefaults(provider: string): {
943
+ api: Api;
944
+ baseUrl: string;
945
+ input: ("text" | "image")[];
946
+ contextWindow: number;
947
+ maxTokens: number;
948
+ } {
949
+ const first = this.models.find((m) => m.provider === provider);
950
+ if (first) {
951
+ return {
952
+ api: first.api,
953
+ baseUrl: first.baseUrl,
954
+ input: first.input,
955
+ contextWindow: first.contextWindow,
956
+ maxTokens: first.maxTokens,
957
+ };
958
+ }
959
+
960
+ return {
961
+ api: "openai-completions",
962
+ baseUrl: this.registeredProviders.get(provider)?.baseUrl ?? "",
963
+ input: ["text"],
964
+ contextWindow: 128000,
965
+ maxTokens: 16384,
966
+ };
967
+ }
968
+
969
+ private getDiscoveryTtl(provider: string, providerApis: Set<string>): number {
970
+ for (const api of providerApis) {
971
+ if (supportsDiscoveryForApi(api)) {
972
+ return getDefaultTTL("openai");
973
+ }
974
+ }
975
+ return getDefaultTTL(provider);
976
+ }
977
+
886
978
  /**
887
979
  * Check if a model's baseUrl points to a local endpoint.
888
980
  * Delegates to standalone isLocalModel() function.
@@ -0,0 +1,86 @@
1
+ // pi-coding-agent — unit tests for session-log secret redaction
2
+
3
+ import assert from "node:assert/strict";
4
+ import { describe, it } from "node:test";
5
+
6
+ import { redactSecrets } from "./redact-secrets.js";
7
+
8
+ describe("redactSecrets", () => {
9
+ it("is a no-op on plain text with no secret markers", () => {
10
+ const input = "Hello world — this is just some prose with numbers 12345 and dashes - - -.";
11
+ assert.equal(redactSecrets(input), input);
12
+ });
13
+
14
+ it("redacts Anthropic keys before generic openai sk- pattern", () => {
15
+ const out = redactSecrets("key=sk-ant-api03-abcDEF1234567890abcDEF1234567890");
16
+ assert.equal(out, "key=[REDACTED:anthropic]");
17
+ });
18
+
19
+ it("redacts legacy OpenAI sk- keys", () => {
20
+ const out = redactSecrets("OPENAI_API_KEY=sk-abcDEF1234567890abcDEF12");
21
+ assert.equal(out, "OPENAI_API_KEY=[REDACTED:openai]");
22
+ });
23
+
24
+ it("redacts OpenAI project sk-proj- keys with hyphens/underscores in body", () => {
25
+ const out = redactSecrets("OPENAI_API_KEY=sk-proj-AbCd_1234-EfGh_5678-IjKl_9012");
26
+ assert.equal(out, "OPENAI_API_KEY=[REDACTED:openai]");
27
+ });
28
+
29
+ it("redacts OpenAI admin sk-admin- keys", () => {
30
+ const out = redactSecrets("OPENAI_ADMIN_KEY=sk-admin-AbCd1234EfGh5678IjKl9012");
31
+ assert.equal(out, "OPENAI_ADMIN_KEY=[REDACTED:openai]");
32
+ });
33
+
34
+ it("redacts LlamaCloud llx- keys", () => {
35
+ const out = redactSecrets("LLAMA_CLOUD_API_KEY=llx-abcDEF1234567890abcDEF1234567890");
36
+ assert.equal(out, "LLAMA_CLOUD_API_KEY=[REDACTED:llamacloud]");
37
+ });
38
+
39
+ it("redacts AWS access key ids", () => {
40
+ const out = redactSecrets("aws_access_key_id = AKIAIOSFODNN7EXAMPLE");
41
+ assert.equal(out, "aws_access_key_id = [REDACTED:aws-access-key]");
42
+ });
43
+
44
+ it("redacts GitHub personal/oauth/app/server/refresh tokens", () => {
45
+ const out = redactSecrets("token=ghp_abcdefghijklmnopqrstuvwxyz0123456789");
46
+ assert.equal(out, "token=[REDACTED:github-token]");
47
+ });
48
+
49
+ it("redacts Slack tokens", () => {
50
+ const out = redactSecrets("slack=xoxb-1234567890-abcdefghij");
51
+ assert.equal(out, "slack=[REDACTED:slack-token]");
52
+ });
53
+
54
+ it("redacts Google API keys", () => {
55
+ // Google API keys are exactly AIza + 35 chars (39 total).
56
+ const out = redactSecrets("key=AIzaSyA-1234567890abcdefghijklmnopqrstu");
57
+ assert.equal(out, "key=[REDACTED:google-api-key]");
58
+ });
59
+
60
+ it("redacts PEM private key blocks across newlines", () => {
61
+ const pem = [
62
+ "-----BEGIN RSA PRIVATE KEY-----",
63
+ "MIIEowIBAAKCAQEAabcDEF...",
64
+ "morekeymaterial==",
65
+ "-----END RSA PRIVATE KEY-----",
66
+ ].join("\n");
67
+ const out = redactSecrets(`before\n${pem}\nafter`);
68
+ assert.equal(out, "before\n[REDACTED:pem-private-key]\nafter");
69
+ });
70
+
71
+ it("redacts multiple secrets in the same string", () => {
72
+ const out = redactSecrets(
73
+ "AZURE_CLIENT_SECRET: also llx-abcDEF1234567890abcDEF1234567890 and AKIAIOSFODNN7EXAMPLE",
74
+ );
75
+ assert.equal(
76
+ out,
77
+ "AZURE_CLIENT_SECRET: also [REDACTED:llamacloud] and [REDACTED:aws-access-key]",
78
+ );
79
+ });
80
+
81
+ it("does not redact short strings that merely contain sk- prose", () => {
82
+ // "sk-foo" is too short to match the openai pattern — must be 20+ chars.
83
+ const input = "the sk- prefix isn't always a secret";
84
+ assert.equal(redactSecrets(input), input);
85
+ });
86
+ });
@@ -0,0 +1,58 @@
1
+ // pi-coding-agent — secret redaction for session log persistence
2
+ //
3
+ // Called by prepareForPersistence() in session-manager.ts on every string
4
+ // before it lands in the JSONL transcript. Replaces well-known secret shapes
5
+ // with [REDACTED:<kind>] placeholders so credentials pasted by the user or
6
+ // read from .env-style files never persist to disk.
7
+ //
8
+ // Pattern selection bias: high-specificity shapes only. Loose patterns
9
+ // (e.g. FOO_SECRET=...) produce too many false positives in docs and code
10
+ // samples and are intentionally excluded.
11
+
12
+ interface SecretPattern {
13
+ kind: string;
14
+ regex: RegExp;
15
+ }
16
+
17
+ // Order matters: more-specific patterns first (sk-ant- before generic sk-).
18
+ const PATTERNS: readonly SecretPattern[] = [
19
+ { kind: "anthropic", regex: /sk-ant-[A-Za-z0-9_-]{20,}/g },
20
+ { kind: "llamacloud", regex: /llx-[A-Za-z0-9_-]{20,}/g },
21
+ // Covers all three official OpenAI key shapes: legacy `sk-…`, project `sk-proj-…`,
22
+ // and admin `sk-admin-…`. Hyphens and underscores appear inside real project keys
23
+ // so the remainder class must allow them. `sk-ant-` is matched earlier by the
24
+ // anthropic pattern and already replaced by the time this runs.
25
+ { kind: "openai", regex: /sk-(?:proj-|admin-)?[A-Za-z0-9_-]{20,}/g },
26
+ { kind: "aws-access-key", regex: /\b(?:AKIA|ASIA|AROA)[0-9A-Z]{16}\b/g },
27
+ { kind: "github-token", regex: /\bgh[pousr]_[A-Za-z0-9]{30,}\b/g },
28
+ { kind: "slack-token", regex: /\bxox[baprs]-[A-Za-z0-9-]{10,}\b/g },
29
+ { kind: "google-api-key", regex: /\bAIza[0-9A-Za-z_-]{35}\b/g },
30
+ {
31
+ kind: "pem-private-key",
32
+ regex: /-----BEGIN [A-Z ]*PRIVATE KEY-----[\s\S]*?-----END [A-Z ]*PRIVATE KEY-----/g,
33
+ },
34
+ ];
35
+
36
+ export function redactSecrets(input: string): string {
37
+ // Short-circuit: skip regex work on strings with no plausible secret markers.
38
+ // Cheap heuristic — if none of these substrings are present, no pattern can match.
39
+ if (
40
+ !input.includes("sk-") &&
41
+ !input.includes("llx-") &&
42
+ !input.includes("AKIA") &&
43
+ !input.includes("ASIA") &&
44
+ !input.includes("AROA") &&
45
+ !input.includes("gh") &&
46
+ !input.includes("xox") &&
47
+ !input.includes("AIza") &&
48
+ !input.includes("PRIVATE KEY")
49
+ ) {
50
+ return input;
51
+ }
52
+
53
+ let out = input;
54
+ for (const { kind, regex } of PATTERNS) {
55
+ out = out.replace(regex, `[REDACTED:${kind}]`);
56
+ }
57
+ return out;
58
+ }