@wahack/pi-coding-agent 15.11.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 (1152) hide show
  1. package/CHANGELOG.md +10031 -0
  2. package/README.md +36 -0
  3. package/examples/README.md +21 -0
  4. package/examples/custom-tools/README.md +104 -0
  5. package/examples/custom-tools/hello/index.ts +20 -0
  6. package/examples/extensions/README.md +142 -0
  7. package/examples/extensions/api-demo.ts +79 -0
  8. package/examples/extensions/chalk-logger.ts +25 -0
  9. package/examples/extensions/hello.ts +31 -0
  10. package/examples/extensions/pirate.ts +43 -0
  11. package/examples/extensions/plan-mode.ts +549 -0
  12. package/examples/extensions/reload-runtime.ts +38 -0
  13. package/examples/extensions/thinking-note.ts +13 -0
  14. package/examples/extensions/tools.ts +145 -0
  15. package/examples/extensions/with-deps/index.ts +36 -0
  16. package/examples/extensions/with-deps/package-lock.json +31 -0
  17. package/examples/extensions/with-deps/package.json +17 -0
  18. package/examples/hooks/README.md +56 -0
  19. package/examples/hooks/auto-commit-on-exit.ts +48 -0
  20. package/examples/hooks/confirm-destructive.ts +58 -0
  21. package/examples/hooks/custom-compaction.ts +115 -0
  22. package/examples/hooks/dirty-repo-guard.ts +51 -0
  23. package/examples/hooks/file-trigger.ts +40 -0
  24. package/examples/hooks/git-checkpoint.ts +52 -0
  25. package/examples/hooks/handoff.ts +149 -0
  26. package/examples/hooks/permission-gate.ts +33 -0
  27. package/examples/hooks/protected-paths.ts +29 -0
  28. package/examples/hooks/qna.ts +118 -0
  29. package/examples/hooks/status-line.ts +39 -0
  30. package/examples/sdk/01-minimal.ts +21 -0
  31. package/examples/sdk/02-custom-model.ts +49 -0
  32. package/examples/sdk/03-custom-prompt.ts +46 -0
  33. package/examples/sdk/04-skills.ts +43 -0
  34. package/examples/sdk/06-extensions.ts +82 -0
  35. package/examples/sdk/06-hooks.ts +61 -0
  36. package/examples/sdk/07-context-files.ts +35 -0
  37. package/examples/sdk/08-prompt-templates.ts +41 -0
  38. package/examples/sdk/08-slash-commands.ts +46 -0
  39. package/examples/sdk/09-api-keys-and-oauth.ts +54 -0
  40. package/examples/sdk/11-sessions.ts +47 -0
  41. package/examples/sdk/12-redis-sessions.ts +54 -0
  42. package/examples/sdk/13-sql-sessions.ts +61 -0
  43. package/examples/sdk/README.md +172 -0
  44. package/package.json +554 -0
  45. package/scripts/build-binary.ts +100 -0
  46. package/scripts/bundle-dist.ts +90 -0
  47. package/scripts/format-prompts.ts +68 -0
  48. package/scripts/generate-docs-index.ts +40 -0
  49. package/scripts/generate-template.ts +33 -0
  50. package/scripts/omp +42 -0
  51. package/scripts/omp.ts +19 -0
  52. package/src/async/index.ts +1 -0
  53. package/src/async/job-manager.ts +625 -0
  54. package/src/auto-thinking/classifier.ts +185 -0
  55. package/src/autoresearch/command-resume.md +14 -0
  56. package/src/autoresearch/dashboard.ts +436 -0
  57. package/src/autoresearch/git.ts +319 -0
  58. package/src/autoresearch/helpers.ts +218 -0
  59. package/src/autoresearch/index.ts +536 -0
  60. package/src/autoresearch/prompt-setup.md +43 -0
  61. package/src/autoresearch/prompt.md +103 -0
  62. package/src/autoresearch/resume-message.md +10 -0
  63. package/src/autoresearch/state.ts +273 -0
  64. package/src/autoresearch/storage.ts +699 -0
  65. package/src/autoresearch/tools/init-experiment.ts +272 -0
  66. package/src/autoresearch/tools/log-experiment.ts +524 -0
  67. package/src/autoresearch/tools/run-experiment.ts +407 -0
  68. package/src/autoresearch/tools/update-notes.ts +109 -0
  69. package/src/autoresearch/types.ts +168 -0
  70. package/src/bun-imports.d.ts +28 -0
  71. package/src/capability/context-file.ts +44 -0
  72. package/src/capability/extension-module.ts +34 -0
  73. package/src/capability/extension.ts +47 -0
  74. package/src/capability/fs.ts +117 -0
  75. package/src/capability/hook.ts +40 -0
  76. package/src/capability/index.ts +436 -0
  77. package/src/capability/instruction.ts +37 -0
  78. package/src/capability/mcp.ts +74 -0
  79. package/src/capability/prompt.ts +35 -0
  80. package/src/capability/rule-buckets.ts +66 -0
  81. package/src/capability/rule.ts +261 -0
  82. package/src/capability/settings.ts +34 -0
  83. package/src/capability/skill.ts +63 -0
  84. package/src/capability/slash-command.ts +40 -0
  85. package/src/capability/ssh.ts +41 -0
  86. package/src/capability/system-prompt.ts +34 -0
  87. package/src/capability/tool.ts +38 -0
  88. package/src/capability/types.ts +168 -0
  89. package/src/cli/agents-cli.ts +138 -0
  90. package/src/cli/args.ts +340 -0
  91. package/src/cli/auth-broker-cli.ts +895 -0
  92. package/src/cli/auth-gateway-cli.ts +611 -0
  93. package/src/cli/classify-install-target.ts +76 -0
  94. package/src/cli/claude-trace-cli.ts +795 -0
  95. package/src/cli/commands/init-xdg.ts +27 -0
  96. package/src/cli/completion-gen.ts +550 -0
  97. package/src/cli/config-cli.ts +418 -0
  98. package/src/cli/dry-balance-cli.ts +856 -0
  99. package/src/cli/extension-flags.ts +48 -0
  100. package/src/cli/file-processor.ts +133 -0
  101. package/src/cli/gallery-cli.ts +230 -0
  102. package/src/cli/gallery-fixtures/agentic.ts +407 -0
  103. package/src/cli/gallery-fixtures/codeintel.ts +187 -0
  104. package/src/cli/gallery-fixtures/edit.ts +194 -0
  105. package/src/cli/gallery-fixtures/fs.ts +220 -0
  106. package/src/cli/gallery-fixtures/index.ts +40 -0
  107. package/src/cli/gallery-fixtures/interaction.ts +49 -0
  108. package/src/cli/gallery-fixtures/memory.ts +81 -0
  109. package/src/cli/gallery-fixtures/misc.ts +250 -0
  110. package/src/cli/gallery-fixtures/search.ts +213 -0
  111. package/src/cli/gallery-fixtures/shell.ts +167 -0
  112. package/src/cli/gallery-fixtures/types.ts +57 -0
  113. package/src/cli/gallery-fixtures/web.ts +158 -0
  114. package/src/cli/gallery-screenshot.ts +279 -0
  115. package/src/cli/grep-cli.ts +160 -0
  116. package/src/cli/grievances-cli.ts +256 -0
  117. package/src/cli/initial-message.ts +58 -0
  118. package/src/cli/list-models.ts +194 -0
  119. package/src/cli/plugin-cli.ts +996 -0
  120. package/src/cli/read-cli.ts +57 -0
  121. package/src/cli/session-picker.ts +79 -0
  122. package/src/cli/setup-cli.ts +231 -0
  123. package/src/cli/shell-cli.ts +176 -0
  124. package/src/cli/ssh-cli.ts +179 -0
  125. package/src/cli/startup-cwd.ts +68 -0
  126. package/src/cli/stats-cli.ts +238 -0
  127. package/src/cli/tiny-models-cli.ts +127 -0
  128. package/src/cli/update-cli.ts +611 -0
  129. package/src/cli/usage-cli.ts +603 -0
  130. package/src/cli/web-search-cli.ts +132 -0
  131. package/src/cli/worktree-cli.ts +291 -0
  132. package/src/cli-commands.ts +79 -0
  133. package/src/cli.ts +200 -0
  134. package/src/commands/acp.ts +24 -0
  135. package/src/commands/agents.ts +57 -0
  136. package/src/commands/auth-broker.ts +99 -0
  137. package/src/commands/auth-gateway.ts +69 -0
  138. package/src/commands/commit.ts +46 -0
  139. package/src/commands/complete.ts +66 -0
  140. package/src/commands/completions.ts +60 -0
  141. package/src/commands/config.ts +51 -0
  142. package/src/commands/dry-balance.ts +43 -0
  143. package/src/commands/gallery.ts +52 -0
  144. package/src/commands/grep.ts +48 -0
  145. package/src/commands/grievances.ts +51 -0
  146. package/src/commands/install.ts +107 -0
  147. package/src/commands/launch.ts +169 -0
  148. package/src/commands/plugin.ts +78 -0
  149. package/src/commands/read.ts +38 -0
  150. package/src/commands/setup.ts +67 -0
  151. package/src/commands/shell.ts +29 -0
  152. package/src/commands/ssh.ts +60 -0
  153. package/src/commands/stats.ts +29 -0
  154. package/src/commands/tiny-models.ts +36 -0
  155. package/src/commands/update.ts +21 -0
  156. package/src/commands/usage.ts +35 -0
  157. package/src/commands/web-search.ts +42 -0
  158. package/src/commands/worktree.ts +56 -0
  159. package/src/commit/agentic/agent.ts +317 -0
  160. package/src/commit/agentic/fallback.ts +96 -0
  161. package/src/commit/agentic/index.ts +355 -0
  162. package/src/commit/agentic/prompts/analyze-file.md +22 -0
  163. package/src/commit/agentic/prompts/session-user.md +25 -0
  164. package/src/commit/agentic/prompts/split-confirm.md +1 -0
  165. package/src/commit/agentic/prompts/system.md +38 -0
  166. package/src/commit/agentic/state.ts +60 -0
  167. package/src/commit/agentic/tools/analyze-file.ts +146 -0
  168. package/src/commit/agentic/tools/git-file-diff.ts +191 -0
  169. package/src/commit/agentic/tools/git-hunk.ts +50 -0
  170. package/src/commit/agentic/tools/git-overview.ts +81 -0
  171. package/src/commit/agentic/tools/index.ts +54 -0
  172. package/src/commit/agentic/tools/propose-changelog.ts +144 -0
  173. package/src/commit/agentic/tools/propose-commit.ts +109 -0
  174. package/src/commit/agentic/tools/recent-commits.ts +81 -0
  175. package/src/commit/agentic/tools/schemas.ts +23 -0
  176. package/src/commit/agentic/tools/split-commit.ts +245 -0
  177. package/src/commit/agentic/topo-sort.ts +44 -0
  178. package/src/commit/agentic/trivial.ts +51 -0
  179. package/src/commit/agentic/validation.ts +183 -0
  180. package/src/commit/analysis/conventional.ts +64 -0
  181. package/src/commit/analysis/index.ts +4 -0
  182. package/src/commit/analysis/scope.ts +242 -0
  183. package/src/commit/analysis/summary.ts +105 -0
  184. package/src/commit/analysis/validation.ts +66 -0
  185. package/src/commit/changelog/detect.ts +40 -0
  186. package/src/commit/changelog/generate.ts +97 -0
  187. package/src/commit/changelog/index.ts +234 -0
  188. package/src/commit/changelog/parse.ts +44 -0
  189. package/src/commit/cli.ts +85 -0
  190. package/src/commit/git/diff.ts +148 -0
  191. package/src/commit/index.ts +5 -0
  192. package/src/commit/map-reduce/index.ts +69 -0
  193. package/src/commit/map-reduce/map-phase.ts +193 -0
  194. package/src/commit/map-reduce/reduce-phase.ts +49 -0
  195. package/src/commit/map-reduce/utils.ts +9 -0
  196. package/src/commit/message.ts +11 -0
  197. package/src/commit/model-selection.ts +92 -0
  198. package/src/commit/pipeline.ts +243 -0
  199. package/src/commit/prompts/analysis-system.md +148 -0
  200. package/src/commit/prompts/analysis-user.md +38 -0
  201. package/src/commit/prompts/changelog-system.md +50 -0
  202. package/src/commit/prompts/changelog-user.md +18 -0
  203. package/src/commit/prompts/file-observer-system.md +24 -0
  204. package/src/commit/prompts/file-observer-user.md +8 -0
  205. package/src/commit/prompts/reduce-system.md +50 -0
  206. package/src/commit/prompts/reduce-user.md +17 -0
  207. package/src/commit/prompts/summary-retry.md +3 -0
  208. package/src/commit/prompts/summary-system.md +38 -0
  209. package/src/commit/prompts/summary-user.md +13 -0
  210. package/src/commit/prompts/types-description.md +2 -0
  211. package/src/commit/shared-llm.ts +77 -0
  212. package/src/commit/types.ts +118 -0
  213. package/src/commit/utils/exclusions.ts +42 -0
  214. package/src/commit/utils.ts +58 -0
  215. package/src/config/api-key-resolver.ts +60 -0
  216. package/src/config/append-only-context-mode.ts +31 -0
  217. package/src/config/config-file.ts +317 -0
  218. package/src/config/file-lock.ts +164 -0
  219. package/src/config/keybindings.ts +628 -0
  220. package/src/config/mcp-schema.json +230 -0
  221. package/src/config/model-discovery.ts +554 -0
  222. package/src/config/model-registry.ts +2090 -0
  223. package/src/config/model-resolver.ts +1502 -0
  224. package/src/config/model-roles.ts +74 -0
  225. package/src/config/models-config-schema.ts +226 -0
  226. package/src/config/models-config.ts +129 -0
  227. package/src/config/prompt-templates.ts +185 -0
  228. package/src/config/resolve-config-value.ts +94 -0
  229. package/src/config/settings-schema.ts +3530 -0
  230. package/src/config/settings.ts +1178 -0
  231. package/src/config.ts +242 -0
  232. package/src/cursor.ts +340 -0
  233. package/src/dap/client.ts +760 -0
  234. package/src/dap/config.ts +189 -0
  235. package/src/dap/defaults.json +212 -0
  236. package/src/dap/index.ts +4 -0
  237. package/src/dap/session.ts +1441 -0
  238. package/src/dap/types.ts +610 -0
  239. package/src/debug/index.ts +515 -0
  240. package/src/debug/log-formatting.ts +58 -0
  241. package/src/debug/log-viewer.ts +908 -0
  242. package/src/debug/profiler.ts +162 -0
  243. package/src/debug/protocol-probe.ts +267 -0
  244. package/src/debug/raw-sse-buffer.ts +273 -0
  245. package/src/debug/raw-sse.ts +292 -0
  246. package/src/debug/report-bundle.ts +374 -0
  247. package/src/debug/system-info.ts +111 -0
  248. package/src/debug/terminal-info.ts +124 -0
  249. package/src/discovery/agents-md.ts +67 -0
  250. package/src/discovery/agents.ts +230 -0
  251. package/src/discovery/at-imports.ts +273 -0
  252. package/src/discovery/builtin-defaults.ts +39 -0
  253. package/src/discovery/builtin-rules/index.ts +54 -0
  254. package/src/discovery/builtin-rules/rs-box-leak.md +48 -0
  255. package/src/discovery/builtin-rules/rs-future-prelude.md +23 -0
  256. package/src/discovery/builtin-rules/rs-lazylock.md +51 -0
  257. package/src/discovery/builtin-rules/rs-match-ergonomics.md +67 -0
  258. package/src/discovery/builtin-rules/rs-parking-lot.md +44 -0
  259. package/src/discovery/builtin-rules/rs-result-type.md +19 -0
  260. package/src/discovery/builtin-rules/ts-bare-catch.md +38 -0
  261. package/src/discovery/builtin-rules/ts-import-type.md +42 -0
  262. package/src/discovery/builtin-rules/ts-no-any.md +56 -0
  263. package/src/discovery/builtin-rules/ts-no-deprecated-leftovers.md +44 -0
  264. package/src/discovery/builtin-rules/ts-no-dynamic-import.md +39 -0
  265. package/src/discovery/builtin-rules/ts-no-return-type.md +45 -0
  266. package/src/discovery/builtin-rules/ts-no-test-timers.md +55 -0
  267. package/src/discovery/builtin-rules/ts-no-tiny-functions.md +51 -0
  268. package/src/discovery/builtin-rules/ts-promise-with-resolvers.md +65 -0
  269. package/src/discovery/builtin-rules/ts-redundant-clear-guard.md +75 -0
  270. package/src/discovery/builtin-rules/ts-set-map.md +28 -0
  271. package/src/discovery/builtin.ts +906 -0
  272. package/src/discovery/claude-plugins.ts +386 -0
  273. package/src/discovery/claude.ts +584 -0
  274. package/src/discovery/cline.ts +83 -0
  275. package/src/discovery/codex.ts +522 -0
  276. package/src/discovery/cursor.ts +220 -0
  277. package/src/discovery/gemini.ts +383 -0
  278. package/src/discovery/github.ts +154 -0
  279. package/src/discovery/helpers.ts +1016 -0
  280. package/src/discovery/index.ts +81 -0
  281. package/src/discovery/mcp-json.ts +171 -0
  282. package/src/discovery/omp-extension-roots.ts +190 -0
  283. package/src/discovery/omp-plugins.ts +383 -0
  284. package/src/discovery/opencode.ts +398 -0
  285. package/src/discovery/plugin-dir-roots.ts +28 -0
  286. package/src/discovery/ssh.ts +153 -0
  287. package/src/discovery/substitute-plugin-root.ts +29 -0
  288. package/src/discovery/vscode.ts +105 -0
  289. package/src/discovery/windsurf.ts +147 -0
  290. package/src/edit/apply-patch/index.ts +87 -0
  291. package/src/edit/apply-patch/parser.ts +174 -0
  292. package/src/edit/diff.ts +999 -0
  293. package/src/edit/file-snapshot-store.ts +91 -0
  294. package/src/edit/hashline/block-resolver.ts +33 -0
  295. package/src/edit/hashline/diff.ts +290 -0
  296. package/src/edit/hashline/execute.ts +242 -0
  297. package/src/edit/hashline/filesystem.ts +130 -0
  298. package/src/edit/hashline/index.ts +5 -0
  299. package/src/edit/hashline/noop-loop-guard.ts +99 -0
  300. package/src/edit/hashline/params.ts +18 -0
  301. package/src/edit/index.ts +571 -0
  302. package/src/edit/modes/apply-patch.lark +19 -0
  303. package/src/edit/modes/apply-patch.ts +53 -0
  304. package/src/edit/modes/patch.ts +1891 -0
  305. package/src/edit/modes/replace.ts +1137 -0
  306. package/src/edit/normalize.ts +345 -0
  307. package/src/edit/notebook.ts +242 -0
  308. package/src/edit/read-file.ts +25 -0
  309. package/src/edit/renderer.ts +769 -0
  310. package/src/edit/streaming.ts +517 -0
  311. package/src/eval/__tests__/agent-bridge.test.ts +708 -0
  312. package/src/eval/__tests__/bridge-timeout.test.ts +64 -0
  313. package/src/eval/__tests__/budget-bridge.test.ts +69 -0
  314. package/src/eval/__tests__/completion-bridge.test.ts +412 -0
  315. package/src/eval/__tests__/helpers-local-roots.test.ts +58 -0
  316. package/src/eval/__tests__/idle-timeout.test.ts +80 -0
  317. package/src/eval/__tests__/js-context-manager.test.ts +241 -0
  318. package/src/eval/__tests__/kernel-spawn.test.ts +103 -0
  319. package/src/eval/agent-bridge.ts +319 -0
  320. package/src/eval/backend.ts +71 -0
  321. package/src/eval/bridge-timeout.ts +44 -0
  322. package/src/eval/budget-bridge.ts +48 -0
  323. package/src/eval/completion-bridge.ts +207 -0
  324. package/src/eval/concurrency-bridge.ts +34 -0
  325. package/src/eval/idle-timeout.ts +91 -0
  326. package/src/eval/index.ts +4 -0
  327. package/src/eval/js/context-manager.ts +502 -0
  328. package/src/eval/js/executor.ts +173 -0
  329. package/src/eval/js/index.ts +51 -0
  330. package/src/eval/js/shared/helpers.ts +283 -0
  331. package/src/eval/js/shared/indirect-eval.ts +30 -0
  332. package/src/eval/js/shared/local-module-loader.ts +342 -0
  333. package/src/eval/js/shared/prelude.ts +2 -0
  334. package/src/eval/js/shared/prelude.txt +246 -0
  335. package/src/eval/js/shared/rewrite-imports.ts +532 -0
  336. package/src/eval/js/shared/runtime.ts +352 -0
  337. package/src/eval/js/shared/types.ts +18 -0
  338. package/src/eval/js/tool-bridge.ts +162 -0
  339. package/src/eval/js/worker-core.ts +132 -0
  340. package/src/eval/js/worker-entry.ts +30 -0
  341. package/src/eval/js/worker-protocol.ts +47 -0
  342. package/src/eval/py/__tests__/prelude.test.ts +19 -0
  343. package/src/eval/py/display.ts +71 -0
  344. package/src/eval/py/executor.ts +742 -0
  345. package/src/eval/py/index.ts +68 -0
  346. package/src/eval/py/kernel.ts +748 -0
  347. package/src/eval/py/prelude.py +658 -0
  348. package/src/eval/py/prelude.ts +3 -0
  349. package/src/eval/py/runner.py +1133 -0
  350. package/src/eval/py/runtime.ts +276 -0
  351. package/src/eval/py/spawn-options.ts +126 -0
  352. package/src/eval/py/tool-bridge.ts +182 -0
  353. package/src/eval/session-id.ts +8 -0
  354. package/src/eval/types.ts +48 -0
  355. package/src/exa/index.ts +2 -0
  356. package/src/exa/mcp-client.ts +370 -0
  357. package/src/exa/types.ts +69 -0
  358. package/src/exec/bash-executor.ts +419 -0
  359. package/src/exec/exec.ts +53 -0
  360. package/src/exec/non-interactive-env.ts +48 -0
  361. package/src/export/custom-share.ts +65 -0
  362. package/src/export/html/index.ts +164 -0
  363. package/src/export/html/template.css +1051 -0
  364. package/src/export/html/template.generated.ts +2 -0
  365. package/src/export/html/template.html +46 -0
  366. package/src/export/html/template.js +2271 -0
  367. package/src/export/html/template.macro.ts +25 -0
  368. package/src/export/html/vendor/highlight.min.js +1213 -0
  369. package/src/export/html/vendor/marked.min.js +6 -0
  370. package/src/export/ttsr.ts +583 -0
  371. package/src/extensibility/custom-commands/bundled/ci-green/index.ts +54 -0
  372. package/src/extensibility/custom-commands/bundled/review/index.ts +489 -0
  373. package/src/extensibility/custom-commands/index.ts +2 -0
  374. package/src/extensibility/custom-commands/loader.ts +238 -0
  375. package/src/extensibility/custom-commands/types.ts +113 -0
  376. package/src/extensibility/custom-tools/index.ts +7 -0
  377. package/src/extensibility/custom-tools/loader.ts +269 -0
  378. package/src/extensibility/custom-tools/types.ts +270 -0
  379. package/src/extensibility/custom-tools/wrapper.ts +47 -0
  380. package/src/extensibility/extensions/compact-handler.ts +40 -0
  381. package/src/extensibility/extensions/get-commands-handler.ts +78 -0
  382. package/src/extensibility/extensions/index.ts +16 -0
  383. package/src/extensibility/extensions/loader.ts +572 -0
  384. package/src/extensibility/extensions/runner.ts +922 -0
  385. package/src/extensibility/extensions/types.ts +1322 -0
  386. package/src/extensibility/extensions/wrapper.ts +223 -0
  387. package/src/extensibility/hooks/index.ts +5 -0
  388. package/src/extensibility/hooks/loader.ts +257 -0
  389. package/src/extensibility/hooks/runner.ts +425 -0
  390. package/src/extensibility/hooks/tool-wrapper.ts +107 -0
  391. package/src/extensibility/hooks/types.ts +606 -0
  392. package/src/extensibility/legacy-pi-ai-shim.ts +24 -0
  393. package/src/extensibility/legacy-pi-coding-agent-shim.ts +15 -0
  394. package/src/extensibility/plugins/doctor.ts +65 -0
  395. package/src/extensibility/plugins/git-url.ts +367 -0
  396. package/src/extensibility/plugins/index.ts +9 -0
  397. package/src/extensibility/plugins/installer.ts +192 -0
  398. package/src/extensibility/plugins/legacy-pi-compat.ts +682 -0
  399. package/src/extensibility/plugins/loader.ts +313 -0
  400. package/src/extensibility/plugins/manager.ts +827 -0
  401. package/src/extensibility/plugins/marketplace/cache.ts +136 -0
  402. package/src/extensibility/plugins/marketplace/fetcher.ts +317 -0
  403. package/src/extensibility/plugins/marketplace/index.ts +6 -0
  404. package/src/extensibility/plugins/marketplace/manager.ts +770 -0
  405. package/src/extensibility/plugins/marketplace/registry.ts +196 -0
  406. package/src/extensibility/plugins/marketplace/source-resolver.ts +147 -0
  407. package/src/extensibility/plugins/marketplace/types.ts +191 -0
  408. package/src/extensibility/plugins/marketplace-auto-update.ts +49 -0
  409. package/src/extensibility/plugins/parser.ts +105 -0
  410. package/src/extensibility/plugins/types.ts +194 -0
  411. package/src/extensibility/shared-events.ts +343 -0
  412. package/src/extensibility/skills.ts +312 -0
  413. package/src/extensibility/slash-commands.ts +227 -0
  414. package/src/extensibility/tool-proxy.ts +25 -0
  415. package/src/extensibility/typebox.ts +418 -0
  416. package/src/extensibility/utils.ts +44 -0
  417. package/src/goals/index.ts +3 -0
  418. package/src/goals/runtime.ts +528 -0
  419. package/src/goals/state.ts +37 -0
  420. package/src/goals/tools/goal-tool.ts +251 -0
  421. package/src/hindsight/backend.ts +354 -0
  422. package/src/hindsight/bank.ts +156 -0
  423. package/src/hindsight/client.ts +598 -0
  424. package/src/hindsight/config.ts +175 -0
  425. package/src/hindsight/content.ts +210 -0
  426. package/src/hindsight/index.ts +8 -0
  427. package/src/hindsight/mental-models.ts +429 -0
  428. package/src/hindsight/seeds.json +32 -0
  429. package/src/hindsight/state.ts +488 -0
  430. package/src/hindsight/transcript.ts +71 -0
  431. package/src/index.ts +59 -0
  432. package/src/internal-urls/agent-protocol.ts +146 -0
  433. package/src/internal-urls/artifact-protocol.ts +107 -0
  434. package/src/internal-urls/docs-index.generated.ts +106 -0
  435. package/src/internal-urls/history-protocol.ts +113 -0
  436. package/src/internal-urls/index.ts +25 -0
  437. package/src/internal-urls/issue-pr-protocol.ts +584 -0
  438. package/src/internal-urls/json-query.ts +126 -0
  439. package/src/internal-urls/local-protocol.ts +287 -0
  440. package/src/internal-urls/mcp-protocol.ts +151 -0
  441. package/src/internal-urls/memory-protocol.ts +169 -0
  442. package/src/internal-urls/omp-protocol.ts +93 -0
  443. package/src/internal-urls/parse.ts +72 -0
  444. package/src/internal-urls/registry-helpers.ts +25 -0
  445. package/src/internal-urls/router.ts +105 -0
  446. package/src/internal-urls/rule-protocol.ts +45 -0
  447. package/src/internal-urls/skill-protocol.ts +96 -0
  448. package/src/internal-urls/types.ts +152 -0
  449. package/src/internal-urls/vault-protocol.ts +936 -0
  450. package/src/irc/bus.ts +292 -0
  451. package/src/lib/xai-http.ts +124 -0
  452. package/src/lsp/client.ts +1193 -0
  453. package/src/lsp/clients/biome-client.ts +264 -0
  454. package/src/lsp/clients/index.ts +50 -0
  455. package/src/lsp/clients/lsp-linter-client.ts +93 -0
  456. package/src/lsp/clients/swiftlint-client.ts +120 -0
  457. package/src/lsp/config.ts +502 -0
  458. package/src/lsp/defaults.json +493 -0
  459. package/src/lsp/diagnostics-ledger.ts +51 -0
  460. package/src/lsp/edits.ts +267 -0
  461. package/src/lsp/index.ts +2477 -0
  462. package/src/lsp/lspmux.ts +233 -0
  463. package/src/lsp/render.ts +694 -0
  464. package/src/lsp/startup-events.ts +13 -0
  465. package/src/lsp/types.ts +455 -0
  466. package/src/lsp/utils.ts +718 -0
  467. package/src/main.ts +1325 -0
  468. package/src/mcp/client.ts +484 -0
  469. package/src/mcp/config-writer.ts +225 -0
  470. package/src/mcp/config.ts +365 -0
  471. package/src/mcp/index.ts +29 -0
  472. package/src/mcp/json-rpc.ts +122 -0
  473. package/src/mcp/loader.ts +124 -0
  474. package/src/mcp/manager.ts +1275 -0
  475. package/src/mcp/oauth-discovery.ts +442 -0
  476. package/src/mcp/oauth-flow.ts +442 -0
  477. package/src/mcp/render.ts +132 -0
  478. package/src/mcp/smithery-auth.ts +104 -0
  479. package/src/mcp/smithery-connect.ts +145 -0
  480. package/src/mcp/smithery-registry.ts +477 -0
  481. package/src/mcp/timeout.ts +59 -0
  482. package/src/mcp/tool-bridge.ts +426 -0
  483. package/src/mcp/tool-cache.ts +117 -0
  484. package/src/mcp/transports/http.ts +519 -0
  485. package/src/mcp/transports/index.ts +6 -0
  486. package/src/mcp/transports/stdio.ts +528 -0
  487. package/src/mcp/types.ts +423 -0
  488. package/src/memories/index.ts +1150 -0
  489. package/src/memories/storage.ts +577 -0
  490. package/src/memory-backend/index.ts +18 -0
  491. package/src/memory-backend/local-backend.ts +39 -0
  492. package/src/memory-backend/off-backend.ts +25 -0
  493. package/src/memory-backend/resolve.ts +25 -0
  494. package/src/memory-backend/runtime.ts +66 -0
  495. package/src/memory-backend/types.ts +166 -0
  496. package/src/mnemopi/backend.ts +547 -0
  497. package/src/mnemopi/config.ts +160 -0
  498. package/src/mnemopi/index.ts +3 -0
  499. package/src/mnemopi/state.ts +584 -0
  500. package/src/modes/acp/acp-agent.ts +2407 -0
  501. package/src/modes/acp/acp-client-bridge.ts +154 -0
  502. package/src/modes/acp/acp-event-mapper.ts +929 -0
  503. package/src/modes/acp/acp-mode.ts +23 -0
  504. package/src/modes/acp/index.ts +2 -0
  505. package/src/modes/acp/terminal-auth.ts +37 -0
  506. package/src/modes/components/agent-dashboard.ts +1206 -0
  507. package/src/modes/components/agent-hub.ts +1071 -0
  508. package/src/modes/components/assistant-message.ts +307 -0
  509. package/src/modes/components/bash-execution.ts +220 -0
  510. package/src/modes/components/bordered-loader.ts +41 -0
  511. package/src/modes/components/branch-summary-message.ts +45 -0
  512. package/src/modes/components/btw-panel.ts +104 -0
  513. package/src/modes/components/chat-block.ts +111 -0
  514. package/src/modes/components/compaction-summary-message.ts +87 -0
  515. package/src/modes/components/copy-selector.ts +206 -0
  516. package/src/modes/components/countdown-timer.ts +75 -0
  517. package/src/modes/components/custom-editor.ts +398 -0
  518. package/src/modes/components/custom-message.ts +63 -0
  519. package/src/modes/components/diff.ts +277 -0
  520. package/src/modes/components/dynamic-border.ts +34 -0
  521. package/src/modes/components/error-banner.ts +33 -0
  522. package/src/modes/components/eval-execution.ts +158 -0
  523. package/src/modes/components/execution-shared.ts +101 -0
  524. package/src/modes/components/extensions/extension-dashboard.ts +399 -0
  525. package/src/modes/components/extensions/extension-list.ts +502 -0
  526. package/src/modes/components/extensions/index.ts +9 -0
  527. package/src/modes/components/extensions/inspector-panel.ts +317 -0
  528. package/src/modes/components/extensions/state-manager.ts +627 -0
  529. package/src/modes/components/extensions/types.ts +186 -0
  530. package/src/modes/components/footer.ts +274 -0
  531. package/src/modes/components/history-search.ts +280 -0
  532. package/src/modes/components/hook-editor.ts +167 -0
  533. package/src/modes/components/hook-input.ts +87 -0
  534. package/src/modes/components/hook-message.ts +66 -0
  535. package/src/modes/components/hook-selector.ts +660 -0
  536. package/src/modes/components/index.ts +38 -0
  537. package/src/modes/components/keybinding-hints.ts +65 -0
  538. package/src/modes/components/late-diagnostics-message.ts +60 -0
  539. package/src/modes/components/login-dialog.ts +164 -0
  540. package/src/modes/components/mcp-add-wizard.ts +1340 -0
  541. package/src/modes/components/message-frame.ts +88 -0
  542. package/src/modes/components/model-selector.ts +1271 -0
  543. package/src/modes/components/oauth-selector.ts +368 -0
  544. package/src/modes/components/omfg-panel.ts +141 -0
  545. package/src/modes/components/overlay-box.ts +108 -0
  546. package/src/modes/components/plan-review-overlay.ts +820 -0
  547. package/src/modes/components/plan-toc.ts +138 -0
  548. package/src/modes/components/plugin-selector.ts +95 -0
  549. package/src/modes/components/plugin-settings.ts +722 -0
  550. package/src/modes/components/queue-mode-selector.ts +56 -0
  551. package/src/modes/components/read-tool-group.ts +670 -0
  552. package/src/modes/components/segment-track.ts +52 -0
  553. package/src/modes/components/session-selector.ts +625 -0
  554. package/src/modes/components/settings-defs.ts +189 -0
  555. package/src/modes/components/settings-selector.ts +651 -0
  556. package/src/modes/components/show-images-selector.ts +45 -0
  557. package/src/modes/components/skill-message.ts +89 -0
  558. package/src/modes/components/status-line/component.ts +869 -0
  559. package/src/modes/components/status-line/context-thresholds.ts +79 -0
  560. package/src/modes/components/status-line/git-utils.ts +42 -0
  561. package/src/modes/components/status-line/index.ts +5 -0
  562. package/src/modes/components/status-line/presets.ts +106 -0
  563. package/src/modes/components/status-line/segments.ts +584 -0
  564. package/src/modes/components/status-line/separators.ts +55 -0
  565. package/src/modes/components/status-line/token-rate.ts +66 -0
  566. package/src/modes/components/status-line/types.ts +108 -0
  567. package/src/modes/components/theme-selector.ts +63 -0
  568. package/src/modes/components/thinking-selector.ts +52 -0
  569. package/src/modes/components/tiny-title-download-progress.ts +90 -0
  570. package/src/modes/components/tips.txt +19 -0
  571. package/src/modes/components/todo-reminder.ts +38 -0
  572. package/src/modes/components/tool-execution.ts +1024 -0
  573. package/src/modes/components/transcript-container.ts +608 -0
  574. package/src/modes/components/tree-selector.ts +978 -0
  575. package/src/modes/components/ttsr-notification.ts +122 -0
  576. package/src/modes/components/user-message-selector.ts +227 -0
  577. package/src/modes/components/user-message.ts +66 -0
  578. package/src/modes/components/visual-truncate.ts +63 -0
  579. package/src/modes/components/welcome.ts +493 -0
  580. package/src/modes/controllers/btw-controller.ts +105 -0
  581. package/src/modes/controllers/command-controller-shared.ts +109 -0
  582. package/src/modes/controllers/command-controller.ts +1566 -0
  583. package/src/modes/controllers/event-controller.ts +1054 -0
  584. package/src/modes/controllers/extension-ui-controller.ts +886 -0
  585. package/src/modes/controllers/input-controller.ts +1073 -0
  586. package/src/modes/controllers/mcp-command-controller.ts +2017 -0
  587. package/src/modes/controllers/omfg-controller.ts +283 -0
  588. package/src/modes/controllers/omfg-rule.ts +647 -0
  589. package/src/modes/controllers/selector-controller.ts +1108 -0
  590. package/src/modes/controllers/ssh-command-controller.ts +384 -0
  591. package/src/modes/controllers/streaming-reveal.ts +279 -0
  592. package/src/modes/controllers/tan-command-controller.ts +173 -0
  593. package/src/modes/controllers/todo-command-controller.ts +485 -0
  594. package/src/modes/data/emojis.json +1 -0
  595. package/src/modes/emoji-autocomplete.ts +285 -0
  596. package/src/modes/gradient-highlight.ts +87 -0
  597. package/src/modes/image-references.ts +117 -0
  598. package/src/modes/index.ts +17 -0
  599. package/src/modes/interactive-mode.ts +3370 -0
  600. package/src/modes/internal-url-autocomplete.ts +143 -0
  601. package/src/modes/loop-limit.ts +140 -0
  602. package/src/modes/magic-keywords.ts +20 -0
  603. package/src/modes/markdown-prose.ts +247 -0
  604. package/src/modes/oauth-manual-input.ts +69 -0
  605. package/src/modes/orchestrate.ts +42 -0
  606. package/src/modes/print-mode.ts +126 -0
  607. package/src/modes/prompt-action-autocomplete.ts +260 -0
  608. package/src/modes/rpc/host-tools.ts +186 -0
  609. package/src/modes/rpc/host-uris.ts +235 -0
  610. package/src/modes/rpc/rpc-client.ts +963 -0
  611. package/src/modes/rpc/rpc-mode.ts +947 -0
  612. package/src/modes/rpc/rpc-subagents.ts +265 -0
  613. package/src/modes/rpc/rpc-types.ts +458 -0
  614. package/src/modes/runtime-init.ts +116 -0
  615. package/src/modes/session-observer-registry.ts +146 -0
  616. package/src/modes/setup-version.ts +11 -0
  617. package/src/modes/setup-wizard/index.ts +99 -0
  618. package/src/modes/setup-wizard/lazy.ts +16 -0
  619. package/src/modes/setup-wizard/scenes/glyph.ts +96 -0
  620. package/src/modes/setup-wizard/scenes/outro.ts +35 -0
  621. package/src/modes/setup-wizard/scenes/providers.ts +69 -0
  622. package/src/modes/setup-wizard/scenes/sign-in.ts +205 -0
  623. package/src/modes/setup-wizard/scenes/splash.ts +201 -0
  624. package/src/modes/setup-wizard/scenes/theme.ts +299 -0
  625. package/src/modes/setup-wizard/scenes/types.ts +48 -0
  626. package/src/modes/setup-wizard/scenes/web-search.ts +129 -0
  627. package/src/modes/setup-wizard/wizard-overlay.ts +275 -0
  628. package/src/modes/shared.ts +47 -0
  629. package/src/modes/theme/dark.json +95 -0
  630. package/src/modes/theme/defaults/alabaster.json +93 -0
  631. package/src/modes/theme/defaults/amethyst.json +96 -0
  632. package/src/modes/theme/defaults/anthracite.json +93 -0
  633. package/src/modes/theme/defaults/basalt.json +91 -0
  634. package/src/modes/theme/defaults/birch.json +95 -0
  635. package/src/modes/theme/defaults/dark-abyss.json +91 -0
  636. package/src/modes/theme/defaults/dark-arctic.json +104 -0
  637. package/src/modes/theme/defaults/dark-aurora.json +95 -0
  638. package/src/modes/theme/defaults/dark-catppuccin.json +107 -0
  639. package/src/modes/theme/defaults/dark-cavern.json +91 -0
  640. package/src/modes/theme/defaults/dark-copper.json +95 -0
  641. package/src/modes/theme/defaults/dark-cosmos.json +90 -0
  642. package/src/modes/theme/defaults/dark-cyberpunk.json +102 -0
  643. package/src/modes/theme/defaults/dark-dracula.json +98 -0
  644. package/src/modes/theme/defaults/dark-eclipse.json +91 -0
  645. package/src/modes/theme/defaults/dark-ember.json +95 -0
  646. package/src/modes/theme/defaults/dark-equinox.json +90 -0
  647. package/src/modes/theme/defaults/dark-forest.json +96 -0
  648. package/src/modes/theme/defaults/dark-github.json +105 -0
  649. package/src/modes/theme/defaults/dark-gruvbox.json +112 -0
  650. package/src/modes/theme/defaults/dark-lavender.json +95 -0
  651. package/src/modes/theme/defaults/dark-lunar.json +89 -0
  652. package/src/modes/theme/defaults/dark-midnight.json +95 -0
  653. package/src/modes/theme/defaults/dark-monochrome.json +94 -0
  654. package/src/modes/theme/defaults/dark-monokai.json +98 -0
  655. package/src/modes/theme/defaults/dark-nebula.json +90 -0
  656. package/src/modes/theme/defaults/dark-nord.json +97 -0
  657. package/src/modes/theme/defaults/dark-ocean.json +101 -0
  658. package/src/modes/theme/defaults/dark-one.json +100 -0
  659. package/src/modes/theme/defaults/dark-poimandres.json +142 -0
  660. package/src/modes/theme/defaults/dark-rainforest.json +91 -0
  661. package/src/modes/theme/defaults/dark-reef.json +91 -0
  662. package/src/modes/theme/defaults/dark-retro.json +92 -0
  663. package/src/modes/theme/defaults/dark-rose-pine.json +96 -0
  664. package/src/modes/theme/defaults/dark-sakura.json +95 -0
  665. package/src/modes/theme/defaults/dark-slate.json +95 -0
  666. package/src/modes/theme/defaults/dark-solarized.json +97 -0
  667. package/src/modes/theme/defaults/dark-solstice.json +90 -0
  668. package/src/modes/theme/defaults/dark-starfall.json +91 -0
  669. package/src/modes/theme/defaults/dark-sunset.json +99 -0
  670. package/src/modes/theme/defaults/dark-swamp.json +90 -0
  671. package/src/modes/theme/defaults/dark-synthwave.json +103 -0
  672. package/src/modes/theme/defaults/dark-taiga.json +91 -0
  673. package/src/modes/theme/defaults/dark-terminal.json +95 -0
  674. package/src/modes/theme/defaults/dark-tokyo-night.json +101 -0
  675. package/src/modes/theme/defaults/dark-tundra.json +91 -0
  676. package/src/modes/theme/defaults/dark-twilight.json +91 -0
  677. package/src/modes/theme/defaults/dark-volcanic.json +91 -0
  678. package/src/modes/theme/defaults/graphite.json +92 -0
  679. package/src/modes/theme/defaults/index.ts +199 -0
  680. package/src/modes/theme/defaults/light-arctic.json +107 -0
  681. package/src/modes/theme/defaults/light-aurora-day.json +91 -0
  682. package/src/modes/theme/defaults/light-canyon.json +91 -0
  683. package/src/modes/theme/defaults/light-catppuccin.json +106 -0
  684. package/src/modes/theme/defaults/light-cirrus.json +90 -0
  685. package/src/modes/theme/defaults/light-coral.json +95 -0
  686. package/src/modes/theme/defaults/light-cyberpunk.json +96 -0
  687. package/src/modes/theme/defaults/light-dawn.json +90 -0
  688. package/src/modes/theme/defaults/light-dunes.json +91 -0
  689. package/src/modes/theme/defaults/light-eucalyptus.json +95 -0
  690. package/src/modes/theme/defaults/light-forest.json +100 -0
  691. package/src/modes/theme/defaults/light-frost.json +95 -0
  692. package/src/modes/theme/defaults/light-github.json +115 -0
  693. package/src/modes/theme/defaults/light-glacier.json +91 -0
  694. package/src/modes/theme/defaults/light-gruvbox.json +108 -0
  695. package/src/modes/theme/defaults/light-haze.json +90 -0
  696. package/src/modes/theme/defaults/light-honeycomb.json +95 -0
  697. package/src/modes/theme/defaults/light-lagoon.json +91 -0
  698. package/src/modes/theme/defaults/light-lavender.json +95 -0
  699. package/src/modes/theme/defaults/light-meadow.json +91 -0
  700. package/src/modes/theme/defaults/light-mint.json +95 -0
  701. package/src/modes/theme/defaults/light-monochrome.json +101 -0
  702. package/src/modes/theme/defaults/light-ocean.json +99 -0
  703. package/src/modes/theme/defaults/light-one.json +99 -0
  704. package/src/modes/theme/defaults/light-opal.json +91 -0
  705. package/src/modes/theme/defaults/light-orchard.json +91 -0
  706. package/src/modes/theme/defaults/light-paper.json +95 -0
  707. package/src/modes/theme/defaults/light-poimandres.json +142 -0
  708. package/src/modes/theme/defaults/light-prism.json +90 -0
  709. package/src/modes/theme/defaults/light-retro.json +98 -0
  710. package/src/modes/theme/defaults/light-sand.json +95 -0
  711. package/src/modes/theme/defaults/light-savanna.json +91 -0
  712. package/src/modes/theme/defaults/light-solarized.json +102 -0
  713. package/src/modes/theme/defaults/light-soleil.json +90 -0
  714. package/src/modes/theme/defaults/light-sunset.json +99 -0
  715. package/src/modes/theme/defaults/light-synthwave.json +98 -0
  716. package/src/modes/theme/defaults/light-tokyo-night.json +111 -0
  717. package/src/modes/theme/defaults/light-wetland.json +91 -0
  718. package/src/modes/theme/defaults/light-zenith.json +89 -0
  719. package/src/modes/theme/defaults/limestone.json +94 -0
  720. package/src/modes/theme/defaults/mahogany.json +97 -0
  721. package/src/modes/theme/defaults/marble.json +93 -0
  722. package/src/modes/theme/defaults/obsidian.json +91 -0
  723. package/src/modes/theme/defaults/onyx.json +91 -0
  724. package/src/modes/theme/defaults/pearl.json +93 -0
  725. package/src/modes/theme/defaults/porcelain.json +91 -0
  726. package/src/modes/theme/defaults/quartz.json +96 -0
  727. package/src/modes/theme/defaults/sandstone.json +95 -0
  728. package/src/modes/theme/defaults/titanium.json +90 -0
  729. package/src/modes/theme/light.json +93 -0
  730. package/src/modes/theme/mermaid-cache.ts +29 -0
  731. package/src/modes/theme/shimmer.ts +235 -0
  732. package/src/modes/theme/theme-schema.json +459 -0
  733. package/src/modes/theme/theme.ts +2676 -0
  734. package/src/modes/turn-budget.ts +31 -0
  735. package/src/modes/types.ts +359 -0
  736. package/src/modes/ultrathink.ts +41 -0
  737. package/src/modes/utils/context-usage.ts +339 -0
  738. package/src/modes/utils/copy-targets.ts +360 -0
  739. package/src/modes/utils/hotkeys-markdown.ts +61 -0
  740. package/src/modes/utils/keybinding-matchers.ts +51 -0
  741. package/src/modes/utils/tools-markdown.ts +27 -0
  742. package/src/modes/utils/ui-helpers.ts +801 -0
  743. package/src/modes/workflow.ts +42 -0
  744. package/src/plan-mode/approved-plan.ts +186 -0
  745. package/src/plan-mode/plan-handoff.ts +37 -0
  746. package/src/plan-mode/plan-protection.ts +31 -0
  747. package/src/plan-mode/state.ts +6 -0
  748. package/src/priority.json +41 -0
  749. package/src/prompts/agents/designer.md +66 -0
  750. package/src/prompts/agents/explore.md +58 -0
  751. package/src/prompts/agents/frontmatter.md +11 -0
  752. package/src/prompts/agents/init.md +33 -0
  753. package/src/prompts/agents/librarian.md +119 -0
  754. package/src/prompts/agents/oracle.md +55 -0
  755. package/src/prompts/agents/plan.md +48 -0
  756. package/src/prompts/agents/reviewer.md +140 -0
  757. package/src/prompts/agents/task.md +16 -0
  758. package/src/prompts/ci-green-request.md +36 -0
  759. package/src/prompts/dry-balance-bench.md +8 -0
  760. package/src/prompts/goals/goal-budget-limit.md +16 -0
  761. package/src/prompts/goals/goal-continuation.md +28 -0
  762. package/src/prompts/goals/goal-mode-active.md +23 -0
  763. package/src/prompts/memories/consolidation.md +30 -0
  764. package/src/prompts/memories/read-path.md +11 -0
  765. package/src/prompts/memories/stage_one_input.md +6 -0
  766. package/src/prompts/memories/stage_one_system.md +21 -0
  767. package/src/prompts/review-custom-request.md +22 -0
  768. package/src/prompts/review-headless-request.md +16 -0
  769. package/src/prompts/review-request.md +69 -0
  770. package/src/prompts/steering/user-interjection.md +10 -0
  771. package/src/prompts/system/agent-creation-architect.md +50 -0
  772. package/src/prompts/system/agent-creation-user.md +6 -0
  773. package/src/prompts/system/auto-continue.md +1 -0
  774. package/src/prompts/system/auto-thinking-difficulty-local.md +14 -0
  775. package/src/prompts/system/auto-thinking-difficulty.md +12 -0
  776. package/src/prompts/system/background-tan-dispatch.md +8 -0
  777. package/src/prompts/system/btw-user.md +8 -0
  778. package/src/prompts/system/commit-message-system.md +14 -0
  779. package/src/prompts/system/custom-system-prompt.md +64 -0
  780. package/src/prompts/system/eager-todo.md +13 -0
  781. package/src/prompts/system/empty-stop-retry.md +6 -0
  782. package/src/prompts/system/irc-incoming.md +7 -0
  783. package/src/prompts/system/manual-continue.md +7 -0
  784. package/src/prompts/system/memory-consolidation-system.md +8 -0
  785. package/src/prompts/system/memory-extraction-system.md +26 -0
  786. package/src/prompts/system/omfg-user.md +50 -0
  787. package/src/prompts/system/orchestrate-notice.md +40 -0
  788. package/src/prompts/system/plan-mode-active.md +109 -0
  789. package/src/prompts/system/plan-mode-approved.md +25 -0
  790. package/src/prompts/system/plan-mode-compact-instructions.md +16 -0
  791. package/src/prompts/system/plan-mode-reference.md +11 -0
  792. package/src/prompts/system/plan-mode-subagent.md +33 -0
  793. package/src/prompts/system/plan-mode-tool-decision-reminder.md +9 -0
  794. package/src/prompts/system/project-prompt.md +52 -0
  795. package/src/prompts/system/subagent-system-prompt.md +64 -0
  796. package/src/prompts/system/subagent-user-prompt.md +3 -0
  797. package/src/prompts/system/subagent-yield-reminder.md +12 -0
  798. package/src/prompts/system/system-prompt.md +258 -0
  799. package/src/prompts/system/tiny-title-system.md +8 -0
  800. package/src/prompts/system/title-system.md +16 -0
  801. package/src/prompts/system/ttsr-interrupt.md +7 -0
  802. package/src/prompts/system/ttsr-tool-reminder.md +5 -0
  803. package/src/prompts/system/ultrathink-notice.md +3 -0
  804. package/src/prompts/system/web-search.md +25 -0
  805. package/src/prompts/system/workflow-notice.md +70 -0
  806. package/src/prompts/tools/apply-patch.md +65 -0
  807. package/src/prompts/tools/ask.md +30 -0
  808. package/src/prompts/tools/ast-edit.md +39 -0
  809. package/src/prompts/tools/ast-grep.md +42 -0
  810. package/src/prompts/tools/async-result.md +8 -0
  811. package/src/prompts/tools/bash.md +46 -0
  812. package/src/prompts/tools/browser.md +73 -0
  813. package/src/prompts/tools/checkpoint.md +16 -0
  814. package/src/prompts/tools/debug.md +34 -0
  815. package/src/prompts/tools/eval.md +92 -0
  816. package/src/prompts/tools/find.md +36 -0
  817. package/src/prompts/tools/github.md +21 -0
  818. package/src/prompts/tools/goal.md +18 -0
  819. package/src/prompts/tools/image-gen.md +7 -0
  820. package/src/prompts/tools/inspect-image-system.md +20 -0
  821. package/src/prompts/tools/inspect-image.md +32 -0
  822. package/src/prompts/tools/irc.md +59 -0
  823. package/src/prompts/tools/job.md +19 -0
  824. package/src/prompts/tools/lsp-late-diagnostic.md +8 -0
  825. package/src/prompts/tools/lsp.md +42 -0
  826. package/src/prompts/tools/memory-edit.md +8 -0
  827. package/src/prompts/tools/patch.md +70 -0
  828. package/src/prompts/tools/read.md +84 -0
  829. package/src/prompts/tools/recall.md +5 -0
  830. package/src/prompts/tools/reflect.md +5 -0
  831. package/src/prompts/tools/render-mermaid.md +9 -0
  832. package/src/prompts/tools/replace.md +30 -0
  833. package/src/prompts/tools/resolve.md +9 -0
  834. package/src/prompts/tools/retain.md +6 -0
  835. package/src/prompts/tools/rewind.md +13 -0
  836. package/src/prompts/tools/search-tool-bm25.md +32 -0
  837. package/src/prompts/tools/search.md +24 -0
  838. package/src/prompts/tools/ssh.md +31 -0
  839. package/src/prompts/tools/task-summary.md +17 -0
  840. package/src/prompts/tools/task.md +88 -0
  841. package/src/prompts/tools/todo.md +62 -0
  842. package/src/prompts/tools/web-search.md +10 -0
  843. package/src/prompts/tools/write.md +14 -0
  844. package/src/registry/agent-lifecycle.ts +218 -0
  845. package/src/registry/agent-registry.ts +151 -0
  846. package/src/sdk.ts +2558 -0
  847. package/src/secrets/index.ts +123 -0
  848. package/src/secrets/obfuscator.ts +298 -0
  849. package/src/secrets/regex.ts +21 -0
  850. package/src/session/agent-session.ts +10121 -0
  851. package/src/session/agent-storage.ts +455 -0
  852. package/src/session/artifacts.ts +135 -0
  853. package/src/session/auth-broker-config.ts +131 -0
  854. package/src/session/auth-storage.ts +29 -0
  855. package/src/session/blob-store.ts +255 -0
  856. package/src/session/client-bridge.ts +85 -0
  857. package/src/session/history-storage.ts +348 -0
  858. package/src/session/indexed-session-storage.ts +430 -0
  859. package/src/session/messages.ts +541 -0
  860. package/src/session/redis-session-storage.ts +170 -0
  861. package/src/session/session-dump-format.ts +209 -0
  862. package/src/session/session-history-format.ts +246 -0
  863. package/src/session/session-manager.ts +3676 -0
  864. package/src/session/session-storage.ts +529 -0
  865. package/src/session/shake-types.ts +43 -0
  866. package/src/session/sql-session-storage.ts +314 -0
  867. package/src/session/streaming-output.ts +1330 -0
  868. package/src/session/tool-choice-queue.ts +213 -0
  869. package/src/session/yield-queue.ts +173 -0
  870. package/src/slash-commands/acp-builtins.ts +70 -0
  871. package/src/slash-commands/builtin-registry.ts +1798 -0
  872. package/src/slash-commands/helpers/context-report.ts +39 -0
  873. package/src/slash-commands/helpers/format.ts +46 -0
  874. package/src/slash-commands/helpers/marketplace-manager.ts +25 -0
  875. package/src/slash-commands/helpers/mcp.ts +532 -0
  876. package/src/slash-commands/helpers/parse.ts +85 -0
  877. package/src/slash-commands/helpers/ssh.ts +195 -0
  878. package/src/slash-commands/helpers/stats-dashboard.ts +85 -0
  879. package/src/slash-commands/helpers/todo.ts +279 -0
  880. package/src/slash-commands/helpers/usage-report.ts +95 -0
  881. package/src/slash-commands/marketplace-install-parser.ts +99 -0
  882. package/src/slash-commands/types.ts +135 -0
  883. package/src/ssh/config-writer.ts +183 -0
  884. package/src/ssh/connection-manager.ts +509 -0
  885. package/src/ssh/ssh-executor.ts +189 -0
  886. package/src/ssh/sshfs-mount.ts +140 -0
  887. package/src/ssh/utils.ts +8 -0
  888. package/src/stt/downloader.ts +71 -0
  889. package/src/stt/index.ts +3 -0
  890. package/src/stt/recorder.ts +351 -0
  891. package/src/stt/setup.ts +52 -0
  892. package/src/stt/stt-controller.ts +160 -0
  893. package/src/stt/transcribe.py +70 -0
  894. package/src/stt/transcriber.ts +91 -0
  895. package/src/stubs/natives/index.ts +814 -0
  896. package/src/stubs/natives/package.json +7 -0
  897. package/src/stubs/tui/index.ts +282 -0
  898. package/src/stubs/tui/package.json +7 -0
  899. package/src/system-prompt.ts +611 -0
  900. package/src/task/agents.ts +167 -0
  901. package/src/task/commands.ts +132 -0
  902. package/src/task/discovery.ts +122 -0
  903. package/src/task/executor.ts +2133 -0
  904. package/src/task/index.ts +1419 -0
  905. package/src/task/name-generator.ts +1577 -0
  906. package/src/task/omp-command.ts +26 -0
  907. package/src/task/output-manager.ts +88 -0
  908. package/src/task/parallel.ts +116 -0
  909. package/src/task/render.ts +1381 -0
  910. package/src/task/repair-args.ts +129 -0
  911. package/src/task/subprocess-tool-registry.ts +88 -0
  912. package/src/task/types.ts +336 -0
  913. package/src/task/worktree.ts +514 -0
  914. package/src/telemetry-export.ts +144 -0
  915. package/src/thinking.ts +167 -0
  916. package/src/tiny/compiled-runtime.ts +179 -0
  917. package/src/tiny/device.ts +111 -0
  918. package/src/tiny/dtype.ts +101 -0
  919. package/src/tiny/models.ts +242 -0
  920. package/src/tiny/text.ts +165 -0
  921. package/src/tiny/title-client.ts +543 -0
  922. package/src/tiny/title-protocol.ts +56 -0
  923. package/src/tiny/worker.ts +568 -0
  924. package/src/tool-discovery/mode.ts +24 -0
  925. package/src/tool-discovery/tool-index.ts +256 -0
  926. package/src/tools/approval.ts +189 -0
  927. package/src/tools/archive-reader.ts +721 -0
  928. package/src/tools/ask.ts +928 -0
  929. package/src/tools/ast-edit.ts +642 -0
  930. package/src/tools/ast-grep.ts +452 -0
  931. package/src/tools/auto-generated-guard.ts +322 -0
  932. package/src/tools/bash-command-fixup.ts +37 -0
  933. package/src/tools/bash-interactive.ts +408 -0
  934. package/src/tools/bash-interceptor.ts +67 -0
  935. package/src/tools/bash-pty-selection.ts +14 -0
  936. package/src/tools/bash-skill-urls.ts +248 -0
  937. package/src/tools/bash.ts +1386 -0
  938. package/src/tools/browser/attach.ts +175 -0
  939. package/src/tools/browser/launch.ts +660 -0
  940. package/src/tools/browser/readable.ts +112 -0
  941. package/src/tools/browser/registry.ts +197 -0
  942. package/src/tools/browser/render.ts +216 -0
  943. package/src/tools/browser/tab-protocol.ts +105 -0
  944. package/src/tools/browser/tab-supervisor.ts +628 -0
  945. package/src/tools/browser/tab-worker-entry.ts +21 -0
  946. package/src/tools/browser/tab-worker.ts +1226 -0
  947. package/src/tools/browser.ts +343 -0
  948. package/src/tools/checkpoint.ts +136 -0
  949. package/src/tools/conflict-detect.ts +718 -0
  950. package/src/tools/context.ts +39 -0
  951. package/src/tools/debug.ts +1067 -0
  952. package/src/tools/eval-backends.ts +27 -0
  953. package/src/tools/eval-render.ts +752 -0
  954. package/src/tools/eval.ts +577 -0
  955. package/src/tools/fetch.ts +1926 -0
  956. package/src/tools/file-recorder.ts +35 -0
  957. package/src/tools/find.ts +609 -0
  958. package/src/tools/fs-cache-invalidation.ts +28 -0
  959. package/src/tools/gh-cache-invalidation.ts +255 -0
  960. package/src/tools/gh-format.ts +12 -0
  961. package/src/tools/gh-renderer.ts +481 -0
  962. package/src/tools/gh.ts +3720 -0
  963. package/src/tools/github-cache.ts +637 -0
  964. package/src/tools/grouped-file-output.ts +210 -0
  965. package/src/tools/image-gen.ts +1517 -0
  966. package/src/tools/index.ts +599 -0
  967. package/src/tools/inspect-image-renderer.ts +132 -0
  968. package/src/tools/inspect-image.ts +174 -0
  969. package/src/tools/irc.ts +723 -0
  970. package/src/tools/job.ts +557 -0
  971. package/src/tools/json-tree.ts +243 -0
  972. package/src/tools/jtd-to-json-schema.ts +219 -0
  973. package/src/tools/jtd-to-typescript.ts +136 -0
  974. package/src/tools/jtd-utils.ts +102 -0
  975. package/src/tools/list-limit.ts +40 -0
  976. package/src/tools/match-line-format.ts +20 -0
  977. package/src/tools/memory-edit.ts +59 -0
  978. package/src/tools/memory-recall.ts +100 -0
  979. package/src/tools/memory-reflect.ts +88 -0
  980. package/src/tools/memory-render.ts +202 -0
  981. package/src/tools/memory-retain.ts +91 -0
  982. package/src/tools/output-meta.ts +754 -0
  983. package/src/tools/output-schema-validator.ts +132 -0
  984. package/src/tools/path-utils.ts +1054 -0
  985. package/src/tools/plan-mode-guard.ts +108 -0
  986. package/src/tools/puppeteer/00_stealth_tampering.txt +63 -0
  987. package/src/tools/puppeteer/01_stealth_activity.txt +20 -0
  988. package/src/tools/puppeteer/02_stealth_hairline.txt +11 -0
  989. package/src/tools/puppeteer/03_stealth_botd.txt +384 -0
  990. package/src/tools/puppeteer/04_stealth_iframe.txt +81 -0
  991. package/src/tools/puppeteer/05_stealth_webgl.txt +75 -0
  992. package/src/tools/puppeteer/06_stealth_screen.txt +72 -0
  993. package/src/tools/puppeteer/07_stealth_fonts.txt +97 -0
  994. package/src/tools/puppeteer/08_stealth_audio.txt +51 -0
  995. package/src/tools/puppeteer/09_stealth_locale.txt +46 -0
  996. package/src/tools/puppeteer/10_stealth_plugins.txt +206 -0
  997. package/src/tools/puppeteer/11_stealth_hardware.txt +8 -0
  998. package/src/tools/puppeteer/12_stealth_codecs.txt +40 -0
  999. package/src/tools/puppeteer/13_stealth_worker.txt +74 -0
  1000. package/src/tools/read.ts +2929 -0
  1001. package/src/tools/render-mermaid.ts +69 -0
  1002. package/src/tools/render-utils.ts +838 -0
  1003. package/src/tools/renderers.ts +77 -0
  1004. package/src/tools/report-tool-issue.ts +534 -0
  1005. package/src/tools/resolve.ts +276 -0
  1006. package/src/tools/review.ts +253 -0
  1007. package/src/tools/search-tool-bm25.ts +351 -0
  1008. package/src/tools/search.ts +1580 -0
  1009. package/src/tools/sqlite-reader.ts +828 -0
  1010. package/src/tools/ssh.ts +349 -0
  1011. package/src/tools/todo.ts +982 -0
  1012. package/src/tools/tool-errors.ts +62 -0
  1013. package/src/tools/tool-result.ts +94 -0
  1014. package/src/tools/tool-timeouts.ts +30 -0
  1015. package/src/tools/tts.ts +133 -0
  1016. package/src/tools/write.ts +1217 -0
  1017. package/src/tools/yield.ts +269 -0
  1018. package/src/tui/code-cell.ts +216 -0
  1019. package/src/tui/file-list.ts +55 -0
  1020. package/src/tui/hyperlink.ts +175 -0
  1021. package/src/tui/index.ts +12 -0
  1022. package/src/tui/output-block.ts +240 -0
  1023. package/src/tui/status-line.ts +54 -0
  1024. package/src/tui/tree-list.ts +84 -0
  1025. package/src/tui/types.ts +15 -0
  1026. package/src/tui/utils.ts +103 -0
  1027. package/src/utils/block-context.ts +312 -0
  1028. package/src/utils/changelog.ts +132 -0
  1029. package/src/utils/clipboard.ts +193 -0
  1030. package/src/utils/command-args.ts +76 -0
  1031. package/src/utils/commit-message-generator.ts +151 -0
  1032. package/src/utils/edit-mode.ts +41 -0
  1033. package/src/utils/enhanced-paste.ts +230 -0
  1034. package/src/utils/event-bus.ts +33 -0
  1035. package/src/utils/external-editor.ts +65 -0
  1036. package/src/utils/file-display-mode.ts +45 -0
  1037. package/src/utils/file-mentions.ts +281 -0
  1038. package/src/utils/git.ts +1833 -0
  1039. package/src/utils/image-loading.ts +132 -0
  1040. package/src/utils/image-resize.ts +309 -0
  1041. package/src/utils/jj.ts +248 -0
  1042. package/src/utils/lang-from-path.ts +239 -0
  1043. package/src/utils/markit.ts +89 -0
  1044. package/src/utils/open.ts +55 -0
  1045. package/src/utils/session-color.ts +68 -0
  1046. package/src/utils/shell-snapshot.ts +187 -0
  1047. package/src/utils/sixel.ts +69 -0
  1048. package/src/utils/title-generator.ts +373 -0
  1049. package/src/utils/tool-choice.ts +33 -0
  1050. package/src/utils/tools-manager.ts +363 -0
  1051. package/src/web/kagi.ts +305 -0
  1052. package/src/web/parallel.ts +353 -0
  1053. package/src/web/scrapers/artifacthub.ts +207 -0
  1054. package/src/web/scrapers/arxiv.ts +83 -0
  1055. package/src/web/scrapers/aur.ts +162 -0
  1056. package/src/web/scrapers/biorxiv.ts +133 -0
  1057. package/src/web/scrapers/bluesky.ts +262 -0
  1058. package/src/web/scrapers/brew.ts +172 -0
  1059. package/src/web/scrapers/cheatsh.ts +68 -0
  1060. package/src/web/scrapers/chocolatey.ts +196 -0
  1061. package/src/web/scrapers/choosealicense.ts +95 -0
  1062. package/src/web/scrapers/cisa-kev.ts +87 -0
  1063. package/src/web/scrapers/clojars.ts +154 -0
  1064. package/src/web/scrapers/coingecko.ts +177 -0
  1065. package/src/web/scrapers/crates-io.ts +97 -0
  1066. package/src/web/scrapers/crossref.ts +136 -0
  1067. package/src/web/scrapers/devto.ts +147 -0
  1068. package/src/web/scrapers/discogs.ts +306 -0
  1069. package/src/web/scrapers/discourse.ts +197 -0
  1070. package/src/web/scrapers/dockerhub.ts +138 -0
  1071. package/src/web/scrapers/docs-rs.ts +653 -0
  1072. package/src/web/scrapers/fdroid.ts +134 -0
  1073. package/src/web/scrapers/firefox-addons.ts +191 -0
  1074. package/src/web/scrapers/flathub.ts +223 -0
  1075. package/src/web/scrapers/github-gist.ts +58 -0
  1076. package/src/web/scrapers/github.ts +704 -0
  1077. package/src/web/scrapers/gitlab.ts +401 -0
  1078. package/src/web/scrapers/go-pkg.ts +266 -0
  1079. package/src/web/scrapers/hackage.ts +140 -0
  1080. package/src/web/scrapers/hackernews.ts +189 -0
  1081. package/src/web/scrapers/hex.ts +105 -0
  1082. package/src/web/scrapers/huggingface.ts +321 -0
  1083. package/src/web/scrapers/iacr.ts +89 -0
  1084. package/src/web/scrapers/index.ts +252 -0
  1085. package/src/web/scrapers/jetbrains-marketplace.ts +159 -0
  1086. package/src/web/scrapers/lemmy.ts +203 -0
  1087. package/src/web/scrapers/lobsters.ts +175 -0
  1088. package/src/web/scrapers/mastodon.ts +292 -0
  1089. package/src/web/scrapers/maven.ts +138 -0
  1090. package/src/web/scrapers/mdn.ts +173 -0
  1091. package/src/web/scrapers/metacpan.ts +222 -0
  1092. package/src/web/scrapers/musicbrainz.ts +250 -0
  1093. package/src/web/scrapers/npm.ts +98 -0
  1094. package/src/web/scrapers/nuget.ts +183 -0
  1095. package/src/web/scrapers/nvd.ts +222 -0
  1096. package/src/web/scrapers/ollama.ts +239 -0
  1097. package/src/web/scrapers/open-vsx.ts +106 -0
  1098. package/src/web/scrapers/opencorporates.ts +292 -0
  1099. package/src/web/scrapers/openlibrary.ts +336 -0
  1100. package/src/web/scrapers/orcid.ts +286 -0
  1101. package/src/web/scrapers/osv.ts +176 -0
  1102. package/src/web/scrapers/packagist.ts +160 -0
  1103. package/src/web/scrapers/pub-dev.ts +143 -0
  1104. package/src/web/scrapers/pubmed.ts +211 -0
  1105. package/src/web/scrapers/pypi.ts +112 -0
  1106. package/src/web/scrapers/rawg.ts +110 -0
  1107. package/src/web/scrapers/readthedocs.ts +120 -0
  1108. package/src/web/scrapers/reddit.ts +95 -0
  1109. package/src/web/scrapers/repology.ts +251 -0
  1110. package/src/web/scrapers/rfc.ts +201 -0
  1111. package/src/web/scrapers/rubygems.ts +103 -0
  1112. package/src/web/scrapers/searchcode.ts +189 -0
  1113. package/src/web/scrapers/sec-edgar.ts +261 -0
  1114. package/src/web/scrapers/semantic-scholar.ts +171 -0
  1115. package/src/web/scrapers/snapcraft.ts +187 -0
  1116. package/src/web/scrapers/sourcegraph.ts +336 -0
  1117. package/src/web/scrapers/spdx.ts +108 -0
  1118. package/src/web/scrapers/spotify.ts +198 -0
  1119. package/src/web/scrapers/stackoverflow.ts +120 -0
  1120. package/src/web/scrapers/terraform.ts +277 -0
  1121. package/src/web/scrapers/tldr.ts +47 -0
  1122. package/src/web/scrapers/twitter.ts +94 -0
  1123. package/src/web/scrapers/types.ts +397 -0
  1124. package/src/web/scrapers/utils.ts +109 -0
  1125. package/src/web/scrapers/vimeo.ts +133 -0
  1126. package/src/web/scrapers/vscode-marketplace.ts +187 -0
  1127. package/src/web/scrapers/w3c.ts +156 -0
  1128. package/src/web/scrapers/wikidata.ts +344 -0
  1129. package/src/web/scrapers/wikipedia.ts +84 -0
  1130. package/src/web/scrapers/youtube.ts +325 -0
  1131. package/src/web/search/index.ts +292 -0
  1132. package/src/web/search/provider.ts +157 -0
  1133. package/src/web/search/providers/anthropic.ts +318 -0
  1134. package/src/web/search/providers/base.ts +89 -0
  1135. package/src/web/search/providers/brave.ts +152 -0
  1136. package/src/web/search/providers/codex.ts +591 -0
  1137. package/src/web/search/providers/exa.ts +400 -0
  1138. package/src/web/search/providers/gemini.ts +460 -0
  1139. package/src/web/search/providers/jina.ts +111 -0
  1140. package/src/web/search/providers/kagi.ts +86 -0
  1141. package/src/web/search/providers/kimi.ts +196 -0
  1142. package/src/web/search/providers/parallel.ts +225 -0
  1143. package/src/web/search/providers/perplexity.ts +730 -0
  1144. package/src/web/search/providers/searxng.ts +313 -0
  1145. package/src/web/search/providers/synthetic.ts +114 -0
  1146. package/src/web/search/providers/tavily.ts +176 -0
  1147. package/src/web/search/providers/utils.ts +128 -0
  1148. package/src/web/search/providers/zai.ts +333 -0
  1149. package/src/web/search/render.ts +262 -0
  1150. package/src/web/search/types.ts +482 -0
  1151. package/src/web/search/utils.ts +17 -0
  1152. package/src/workspace-tree.ts +286 -0
@@ -0,0 +1,1833 @@
1
+ import * as fs from "node:fs";
2
+ import * as os from "node:os";
3
+ import * as path from "node:path";
4
+ import { $which, hasFsCode, isEnoent, Snowflake } from "@oh-my-pi/pi-utils";
5
+ import {
6
+ parseDiffHunks as parseCommitDiffHunks,
7
+ parseFileDiffs,
8
+ parseFileHunks,
9
+ parseNumstat,
10
+ } from "../commit/git/diff";
11
+ import type { FileDiff, FileHunks, NumstatEntry } from "../commit/types";
12
+ import { ToolAbortError, ToolError, throwIfAborted } from "../tools/tool-errors";
13
+
14
+ // ════════════════════════════════════════════════════════════════════════════
15
+ // Types
16
+ // ════════════════════════════════════════════════════════════════════════════
17
+
18
+ export interface GitCommandResult {
19
+ exitCode: number;
20
+ stdout: string;
21
+ stderr: string;
22
+ }
23
+
24
+ export interface GitRepository {
25
+ commonDir: string;
26
+ gitDir: string;
27
+ gitEntryPath: string;
28
+ headPath: string;
29
+ repoRoot: string;
30
+ isReftable?: boolean;
31
+ }
32
+
33
+ export interface GitStatusSummary {
34
+ staged: number;
35
+ unstaged: number;
36
+ untracked: number;
37
+ }
38
+
39
+ export type HunkSelection = {
40
+ path: string;
41
+ hunks: { type: "all" } | { type: "indices"; indices: number[] } | { type: "lines"; start: number; end: number };
42
+ };
43
+
44
+ export interface StageHunksOptions {
45
+ readonly diffCached?: boolean;
46
+ readonly rawDiff?: string;
47
+ readonly signal?: AbortSignal;
48
+ }
49
+ export interface HunkSelectionValidationError {
50
+ readonly path: string;
51
+ readonly message: string;
52
+ }
53
+
54
+ export interface DiffOptions {
55
+ readonly allowFailure?: boolean;
56
+ readonly base?: string;
57
+ readonly binary?: boolean;
58
+ readonly cached?: boolean;
59
+ readonly env?: Record<string, string | undefined>;
60
+ readonly files?: readonly string[];
61
+ readonly head?: string;
62
+ readonly nameOnly?: boolean;
63
+ readonly noIndex?: { left: string; right: string };
64
+ readonly numstat?: boolean;
65
+ readonly signal?: AbortSignal;
66
+ readonly stat?: boolean;
67
+ }
68
+
69
+ export interface StatusOptions {
70
+ readonly pathspecs?: readonly string[];
71
+ readonly porcelainV1?: boolean;
72
+ readonly signal?: AbortSignal;
73
+ readonly untrackedFiles?: "all" | "no" | "normal";
74
+ readonly z?: boolean;
75
+ }
76
+
77
+ export interface CommitOptions {
78
+ readonly allowEmpty?: boolean;
79
+ readonly files?: readonly string[];
80
+ readonly signal?: AbortSignal;
81
+ }
82
+
83
+ export interface PushOptions {
84
+ readonly forceWithLease?: boolean;
85
+ readonly refspec?: string;
86
+ readonly remote?: string;
87
+ readonly signal?: AbortSignal;
88
+ }
89
+
90
+ export interface PatchOptions {
91
+ readonly cached?: boolean;
92
+ readonly check?: boolean;
93
+ readonly env?: Record<string, string | undefined>;
94
+ readonly signal?: AbortSignal;
95
+ }
96
+
97
+ export interface RestoreOptions {
98
+ readonly files?: readonly string[];
99
+ readonly signal?: AbortSignal;
100
+ readonly source?: string;
101
+ readonly staged?: boolean;
102
+ readonly worktree?: boolean;
103
+ }
104
+
105
+ export interface CloneOptions {
106
+ readonly ref?: string;
107
+ readonly sha?: string;
108
+ readonly signal?: AbortSignal;
109
+ }
110
+
111
+ interface GitHeadBase extends GitRepository {
112
+ headContent: string;
113
+ }
114
+
115
+ export interface GitRefHead extends GitHeadBase {
116
+ branchName: string | null;
117
+ commit: string | null;
118
+ kind: "ref";
119
+ ref: string;
120
+ }
121
+
122
+ export interface GitDetachedHead extends GitHeadBase {
123
+ commit: string | null;
124
+ kind: "detached";
125
+ }
126
+
127
+ export type GitHeadState = GitRefHead | GitDetachedHead;
128
+
129
+ export interface GitWorktreeEntry {
130
+ branch?: string;
131
+ detached: boolean;
132
+ head?: string;
133
+ path: string;
134
+ }
135
+
136
+ // ════════════════════════════════════════════════════════════════════════════
137
+ // Error
138
+ // ════════════════════════════════════════════════════════════════════════════
139
+
140
+ export class GitCommandError extends Error {
141
+ readonly args: readonly string[];
142
+ readonly result: GitCommandResult;
143
+
144
+ constructor(args: readonly string[], result: GitCommandResult) {
145
+ super(formatCommandFailure(args, result));
146
+ this.name = "GitCommandError";
147
+ this.args = [...args];
148
+ this.result = result;
149
+ }
150
+ }
151
+
152
+ // ════════════════════════════════════════════════════════════════════════════
153
+ // Internal: Core execution
154
+ // ════════════════════════════════════════════════════════════════════════════
155
+
156
+ const NO_OPTIONAL_LOCKS = "--no-optional-locks";
157
+ const HEAD_REF_PREFIX = "ref:";
158
+ const LOCAL_BRANCH_PREFIX = "refs/heads/";
159
+ const DEFAULT_BRANCH_REFS = ["refs/remotes/origin/HEAD", "refs/remotes/upstream/HEAD"] as const;
160
+ const SHORT_LIVED_GIT_CONFIG: readonly (readonly [key: string, value: string])[] = [
161
+ ["core.fsmonitor", "false"],
162
+ ["core.untrackedCache", "false"],
163
+ ];
164
+ const REMOTE_ALREADY_EXISTS = /remote .* already exists/i;
165
+
166
+ interface CommandOptions {
167
+ readonly env?: Record<string, string | undefined>;
168
+ readonly readOnly?: boolean;
169
+ readonly signal?: AbortSignal;
170
+ readonly stdin?: string | Uint8Array | ArrayBuffer | SharedArrayBuffer;
171
+ }
172
+
173
+ function normalizeStdin(input: CommandOptions["stdin"]): "ignore" | Uint8Array {
174
+ if (input === undefined) return "ignore";
175
+ if (typeof input === "string") return new TextEncoder().encode(input);
176
+ if (input instanceof Uint8Array) return input;
177
+ return new Uint8Array(input);
178
+ }
179
+
180
+ function ensureAvailable(): void {
181
+ if (!$which("git")) {
182
+ throw new Error("git is not installed.");
183
+ }
184
+ }
185
+
186
+ function formatCommandFailure(
187
+ args: readonly string[],
188
+ result: Pick<GitCommandResult, "exitCode" | "stdout" | "stderr">,
189
+ ): string {
190
+ const stderr = result.stderr.trim();
191
+ if (stderr) return stderr;
192
+ const stdout = result.stdout.trim();
193
+ if (stdout) return stdout;
194
+ return `git ${args.join(" ")} failed with exit code ${result.exitCode}`;
195
+ }
196
+
197
+ async function git(cwd: string, args: readonly string[], options: CommandOptions = {}): Promise<GitCommandResult> {
198
+ const commandArgs = withShortLivedGitConfig(options.readOnly ? withNoOptionalLocks(args) : [...args]);
199
+ const child = Bun.spawn(["git", ...commandArgs], {
200
+ cwd,
201
+ env: options.env ? { ...process.env, GIT_OPTIONAL_LOCKS: "0", ...options.env } : undefined,
202
+ signal: options.signal,
203
+ stdin: normalizeStdin(options.stdin),
204
+ stdout: "pipe",
205
+ stderr: "pipe",
206
+ windowsHide: true,
207
+ });
208
+
209
+ if (!child.stdout || !child.stderr) {
210
+ throw new Error("Failed to capture git command output.");
211
+ }
212
+
213
+ const [stdout, stderr, exitCode] = await Promise.all([
214
+ new Response(child.stdout).text(),
215
+ new Response(child.stderr).text(),
216
+ child.exited,
217
+ ]);
218
+
219
+ return { exitCode: exitCode ?? 0, stdout, stderr };
220
+ }
221
+
222
+ function withNoOptionalLocks(args: readonly string[]): string[] {
223
+ if (args.includes(NO_OPTIONAL_LOCKS)) return [...args];
224
+ return [NO_OPTIONAL_LOCKS, ...args];
225
+ }
226
+
227
+ function withShortLivedGitConfig(args: readonly string[]): string[] {
228
+ const prefix: string[] = [];
229
+ for (const [key, value] of SHORT_LIVED_GIT_CONFIG) {
230
+ if (hasGitConfig(args, key, value)) continue;
231
+ prefix.push("-c", `${key}=${value}`);
232
+ }
233
+ return [...prefix, ...args];
234
+ }
235
+
236
+ function hasGitConfig(args: readonly string[], key: string, value: string): boolean {
237
+ const expected = `${key}=${value}`;
238
+ for (let index = 0; index < args.length - 1; index += 1) {
239
+ if (args[index] === "-c" && args[index + 1] === expected) {
240
+ return true;
241
+ }
242
+ }
243
+ return false;
244
+ }
245
+
246
+ async function runChecked(
247
+ cwd: string,
248
+ args: readonly string[],
249
+ options: CommandOptions = {},
250
+ ): Promise<GitCommandResult> {
251
+ ensureAvailable();
252
+ const result = await git(cwd, args, options);
253
+ if (result.exitCode !== 0) {
254
+ throw new GitCommandError(args, result);
255
+ }
256
+ return result;
257
+ }
258
+
259
+ async function runEffect(cwd: string, args: readonly string[], options: CommandOptions = {}): Promise<void> {
260
+ await runChecked(cwd, args, options);
261
+ }
262
+
263
+ async function runText(cwd: string, args: readonly string[], options: CommandOptions = {}): Promise<string> {
264
+ return (await runChecked(cwd, args, options)).stdout;
265
+ }
266
+
267
+ async function tryText(
268
+ cwd: string,
269
+ args: readonly string[],
270
+ options: CommandOptions = {},
271
+ ): Promise<string | undefined> {
272
+ ensureAvailable();
273
+ const result = await git(cwd, args, options);
274
+ if (result.exitCode !== 0) return undefined;
275
+ return result.stdout;
276
+ }
277
+
278
+ // ════════════════════════════════════════════════════════════════════════════
279
+ // Internal: per-repo write serialization
280
+ // ════════════════════════════════════════════════════════════════════════════
281
+
282
+ // Git uses lock files (`.git/config.lock`, commit-graph chain locks,
283
+ // `packed-refs.lock`, …) for many of its mutating operations. Each is created
284
+ // O_EXCL with no waiter, so concurrent in-process git invocations against the
285
+ // same repository fail immediately rather than block. Worktrees share the
286
+ // primary repo's `.git` directory, so racing across worktrees has the same
287
+ // failure mode. We give callers a single per-repo serialization point keyed by
288
+ // the primary repo root: any block that mutates repo state should hold this
289
+ // lock so unrelated callers cannot collide on git's internal locks.
290
+ const repoWriteChain = new Map<string, Promise<unknown>>();
291
+
292
+ /**
293
+ * Serialize an async block that mutates a git repository against other
294
+ * in-process callers operating on the same repository. The lock is keyed by
295
+ * the primary repo root so worktrees of the same repo share a single queue.
296
+ * Failures in one block do not poison the queue for the next caller.
297
+ *
298
+ * Not reentrant: do NOT nest acquisitions for the same repo. Helpers in this
299
+ * module never auto-acquire — callers wrap the critical section themselves.
300
+ */
301
+ export async function withRepoLock<T>(cwd: string, fn: () => Promise<T>, signal?: AbortSignal): Promise<T> {
302
+ const key = (await repo.primaryRoot(cwd, signal)) ?? cwd;
303
+ const prior = repoWriteChain.get(key);
304
+ const run = (async () => {
305
+ if (prior) {
306
+ try {
307
+ await prior;
308
+ } catch {
309
+ // A prior caller failing must not block us from running.
310
+ }
311
+ }
312
+ throwIfAborted(signal);
313
+ return fn();
314
+ })();
315
+ repoWriteChain.set(key, run);
316
+ try {
317
+ return await run;
318
+ } finally {
319
+ if (repoWriteChain.get(key) === run) repoWriteChain.delete(key);
320
+ }
321
+ }
322
+
323
+ function splitLines(text: string): string[] {
324
+ return text
325
+ .split("\n")
326
+ .map(line => line.trim())
327
+ .filter(Boolean);
328
+ }
329
+
330
+ function trimScalar(text: string | undefined): string | undefined {
331
+ const trimmed = text?.trim();
332
+ return trimmed || undefined;
333
+ }
334
+
335
+ // ════════════════════════════════════════════════════════════════════════════
336
+ // Internal: Argument builders
337
+ // ════════════════════════════════════════════════════════════════════════════
338
+
339
+ function buildDiffArgs(options: DiffOptions): string[] {
340
+ const args = ["diff"];
341
+ if (options.binary) args.push("--binary");
342
+ if (options.cached) args.push("--cached");
343
+ if (options.nameOnly) args.push("--name-only");
344
+ if (options.stat) args.push("--stat");
345
+ if (options.numstat) args.push("--numstat");
346
+ if (options.noIndex) {
347
+ args.push("--no-index", options.noIndex.left, options.noIndex.right);
348
+ return args;
349
+ }
350
+ if (options.base) {
351
+ args.push(options.base);
352
+ if (options.head) args.push(options.head);
353
+ }
354
+ if (options.files?.length) args.push("--", ...options.files);
355
+ return args;
356
+ }
357
+
358
+ function buildApplyArgs(patchPath: string, options: PatchOptions): string[] {
359
+ const args = ["apply"];
360
+ if (options.check) args.push("--check");
361
+ if (options.cached) args.push("--cached");
362
+ args.push("--binary", patchPath);
363
+ return args;
364
+ }
365
+
366
+ async function writeTempPatch(content: string): Promise<string> {
367
+ const tempPath = path.join(os.tmpdir(), `omp-git-patch-${Snowflake.next()}.patch`);
368
+ await Bun.write(tempPath, content);
369
+ return tempPath;
370
+ }
371
+
372
+ // ════════════════════════════════════════════════════════════════════════════
373
+ // Internal: Repository resolution
374
+ // ════════════════════════════════════════════════════════════════════════════
375
+
376
+ type EntryType = "directory" | "file";
377
+
378
+ function shouldRetry(err: unknown, n: number) {
379
+ if (isEnoent(err) || hasFsCode(err, "ENFILE") || hasFsCode(err, "EMFILE")) return false;
380
+ if (hasFsCode(err, "EINTR")) return n < EINTR_MAX_RETRIES;
381
+ if (n > EINTR_MAX_RETRIES) throw err;
382
+ throw err;
383
+ }
384
+
385
+ /**
386
+ * Bounded retry for synchronous I/O against `EINTR`. POSIX permits short syscalls
387
+ * to be interrupted by signals; when that happens libc traditionally retries.
388
+ * Node's sync wrappers surface the raw `EINTR` so we replicate the retry locally.
389
+ * Any other error (and persistent EINTR after `EINTR_MAX_RETRIES`) is rethrown
390
+ * for the caller's normal "optional metadata" classifier to handle.
391
+ */
392
+ const EINTR_MAX_RETRIES = 3;
393
+ function retryOnEintrSync<T>(op: () => T): T | null {
394
+ for (let attempt = 0; attempt <= EINTR_MAX_RETRIES; attempt += 1) {
395
+ try {
396
+ return op();
397
+ } catch (err) {
398
+ if (shouldRetry(err, attempt)) continue;
399
+ return null;
400
+ }
401
+ }
402
+ throw new Error("retryOnEintrSync: exhausted without resolution");
403
+ }
404
+ async function retryOnEintr<T>(op: () => Promise<T>): Promise<T | null> {
405
+ for (let attempt = 0; attempt <= EINTR_MAX_RETRIES; attempt += 1) {
406
+ try {
407
+ return await op();
408
+ } catch (err) {
409
+ if (shouldRetry(err, attempt)) continue;
410
+ return null;
411
+ }
412
+ }
413
+ throw new Error("retryOnEintr: exhausted without resolution");
414
+ }
415
+
416
+ function getEntryTypeSync(gitEntryPath: string): EntryType | null {
417
+ return retryOnEintrSync(() => {
418
+ const stat = fs.statSync(gitEntryPath);
419
+ if (stat.isDirectory()) return "directory";
420
+ if (stat.isFile()) return "file";
421
+ return null;
422
+ });
423
+ }
424
+
425
+ async function getEntryType(gitEntryPath: string): Promise<EntryType | null> {
426
+ return retryOnEintr(async () => {
427
+ const stat = await fs.promises.stat(gitEntryPath);
428
+ if (stat.isDirectory()) return "directory";
429
+ if (stat.isFile()) return "file";
430
+ return null;
431
+ });
432
+ }
433
+
434
+ function readOptionalTextSync(filePath: string): string | null {
435
+ return retryOnEintrSync(() => fs.readFileSync(filePath, "utf8"));
436
+ }
437
+
438
+ async function readOptionalText(filePath: string): Promise<string | null> {
439
+ return retryOnEintr(async () => await Bun.file(filePath).text());
440
+ }
441
+
442
+ function parseGitDirPointer(content: string): string | null {
443
+ const match = /^gitdir:\s*(.+)\s*$/iu.exec(content.trim());
444
+ return match?.[1] ?? null;
445
+ }
446
+
447
+ function resolveGitDirSync(gitEntryPath: string, entryType: EntryType): string | null {
448
+ if (entryType === "directory") return gitEntryPath;
449
+ const content = readOptionalTextSync(gitEntryPath);
450
+ if (content === null) return null;
451
+ const parsed = parseGitDirPointer(content);
452
+ if (!parsed) return null;
453
+ const gitDir = path.resolve(path.dirname(gitEntryPath), parsed);
454
+ return getEntryTypeSync(gitDir) === "directory" ? gitDir : null;
455
+ }
456
+
457
+ async function resolveGitDir(gitEntryPath: string, entryType: EntryType): Promise<string | null> {
458
+ if (entryType === "directory") return gitEntryPath;
459
+ const content = await readOptionalText(gitEntryPath);
460
+ if (content === null) return null;
461
+ const parsed = parseGitDirPointer(content);
462
+ if (!parsed) return null;
463
+ const gitDir = path.resolve(path.dirname(gitEntryPath), parsed);
464
+ return (await getEntryType(gitDir)) === "directory" ? gitDir : null;
465
+ }
466
+
467
+ function resolveCommonDirSync(gitDir: string): string {
468
+ const content = readOptionalTextSync(path.join(gitDir, "commondir"));
469
+ const relative = content?.trim();
470
+ if (!relative) return gitDir;
471
+ return path.resolve(gitDir, relative);
472
+ }
473
+
474
+ async function resolveCommonDir(gitDir: string): Promise<string> {
475
+ const content = await readOptionalText(path.join(gitDir, "commondir"));
476
+ const relative = content?.trim();
477
+ if (!relative) return gitDir;
478
+ return path.resolve(gitDir, relative);
479
+ }
480
+ function isLinkedWorktree(repository: GitRepository): boolean {
481
+ return (
482
+ repository.gitDir !== repository.commonDir &&
483
+ getEntryTypeSync(path.join(repository.gitDir, "commondir")) === "file"
484
+ );
485
+ }
486
+
487
+ async function isLinkedWorktreeAsync(repository: GitRepository): Promise<boolean> {
488
+ return (
489
+ repository.gitDir !== repository.commonDir &&
490
+ (await getEntryType(path.join(repository.gitDir, "commondir"))) === "file"
491
+ );
492
+ }
493
+
494
+ function primaryRootFromRepositorySync(repository: GitRepository): string {
495
+ if (path.basename(repository.commonDir) === ".git") return path.dirname(repository.commonDir);
496
+ if (isLinkedWorktree(repository)) return repository.commonDir;
497
+ return repository.repoRoot;
498
+ }
499
+
500
+ async function primaryRootFromRepository(repository: GitRepository): Promise<string> {
501
+ if (path.basename(repository.commonDir) === ".git") return path.dirname(repository.commonDir);
502
+ if (await isLinkedWorktreeAsync(repository)) return repository.commonDir;
503
+ return repository.repoRoot;
504
+ }
505
+
506
+ function resolveRepoFromEntrySync(repoRoot: string, gitEntryPath: string, entryType: EntryType): GitRepository | null {
507
+ const gitDir = resolveGitDirSync(gitEntryPath, entryType);
508
+ if (!gitDir) return null;
509
+ return {
510
+ commonDir: resolveCommonDirSync(gitDir),
511
+ gitDir,
512
+ gitEntryPath,
513
+ headPath: path.join(gitDir, "HEAD"),
514
+ repoRoot,
515
+ };
516
+ }
517
+
518
+ async function resolveRepoFromEntry(
519
+ repoRoot: string,
520
+ gitEntryPath: string,
521
+ entryType: EntryType,
522
+ ): Promise<GitRepository | null> {
523
+ const gitDir = await resolveGitDir(gitEntryPath, entryType);
524
+ if (!gitDir) return null;
525
+ return {
526
+ commonDir: await resolveCommonDir(gitDir),
527
+ gitDir,
528
+ gitEntryPath,
529
+ headPath: path.join(gitDir, "HEAD"),
530
+ repoRoot,
531
+ };
532
+ }
533
+
534
+ function resolveRepositorySync(startDir: string): GitRepository | null {
535
+ let current = path.resolve(startDir);
536
+ while (true) {
537
+ const gitEntryPath = path.join(current, ".git");
538
+ const entryType = getEntryTypeSync(gitEntryPath);
539
+ if (entryType) {
540
+ const repository = resolveRepoFromEntrySync(current, gitEntryPath, entryType);
541
+ if (repository) return repository;
542
+ }
543
+ const parent = path.dirname(current);
544
+ if (parent === current) return null;
545
+ current = parent;
546
+ }
547
+ }
548
+
549
+ async function resolveRepository(startDir: string): Promise<GitRepository | null> {
550
+ let current = path.resolve(startDir);
551
+ while (true) {
552
+ const gitEntryPath = path.join(current, ".git");
553
+ const entryType = await getEntryType(gitEntryPath);
554
+ if (entryType) {
555
+ const repository = await resolveRepoFromEntry(current, gitEntryPath, entryType);
556
+ if (repository) return repository;
557
+ }
558
+ const parent = path.dirname(current);
559
+ if (parent === current) return null;
560
+ current = parent;
561
+ }
562
+ }
563
+
564
+ // ════════════════════════════════════════════════════════════════════════════
565
+ // Internal: Ref resolution
566
+ // ════════════════════════════════════════════════════════════════════════════
567
+
568
+ function getRefLookupDirs(repository: GitRepository): string[] {
569
+ if (repository.gitDir === repository.commonDir) return [repository.gitDir];
570
+ return [repository.gitDir, repository.commonDir];
571
+ }
572
+
573
+ function normalizeRefValue(content: string | null): string | null {
574
+ const trimmed = content?.trim() ?? "";
575
+ return trimmed || null;
576
+ }
577
+
578
+ function parsePackedRefs(content: string | null, targetRef: string): string | null {
579
+ if (!content) return null;
580
+ for (const line of content.split("\n")) {
581
+ const trimmed = line.trim();
582
+ if (!trimmed || trimmed.startsWith("#") || trimmed.startsWith("^")) continue;
583
+ const [sha, refName] = trimmed.split(" ", 2);
584
+ if (refName === targetRef && sha) return sha;
585
+ }
586
+ return null;
587
+ }
588
+
589
+ function stripGitConfigComments(line: string): string {
590
+ let clean = "";
591
+ let inQuotes = false;
592
+ for (let i = 0; i < line.length; i++) {
593
+ const char = line[i];
594
+ if (char === '"') {
595
+ inQuotes = !inQuotes;
596
+ clean += char;
597
+ } else if (!inQuotes && (char === ";" || char === "#")) {
598
+ break;
599
+ } else {
600
+ clean += char;
601
+ }
602
+ }
603
+ return clean.trim();
604
+ }
605
+
606
+ function parseGitConfigHasReftable(content: string): boolean {
607
+ let inExtensions = false;
608
+ for (const line of content.split("\n")) {
609
+ const trimmed = stripGitConfigComments(line);
610
+ if (trimmed.startsWith("[") && trimmed.endsWith("]")) {
611
+ const section = trimmed.slice(1, -1).trim().toLowerCase();
612
+ inExtensions = section === "extensions";
613
+ } else if (inExtensions) {
614
+ const eqIndex = trimmed.indexOf("=");
615
+ if (eqIndex !== -1) {
616
+ const key = trimmed.slice(0, eqIndex).trim().toLowerCase();
617
+ let value = trimmed.slice(eqIndex + 1).trim();
618
+ if (key === "refstorage") {
619
+ if (value.startsWith('"') && value.endsWith('"')) {
620
+ value = value.slice(1, -1).trim();
621
+ }
622
+ const lowerValue = value.toLowerCase();
623
+ if (lowerValue === "reftable" || lowerValue.startsWith("reftable:")) {
624
+ return true;
625
+ }
626
+ }
627
+ }
628
+ }
629
+ }
630
+ return false;
631
+ }
632
+
633
+ function isReftableRepoSync(repository: GitRepository): boolean {
634
+ if (repository.isReftable !== undefined) return repository.isReftable;
635
+ const configPath = path.join(repository.commonDir, "config");
636
+ const content = readOptionalTextSync(configPath);
637
+ repository.isReftable = content ? parseGitConfigHasReftable(content) : false;
638
+ return repository.isReftable;
639
+ }
640
+
641
+ async function isReftableRepo(repository: GitRepository): Promise<boolean> {
642
+ if (repository.isReftable !== undefined) return repository.isReftable;
643
+ const configPath = path.join(repository.commonDir, "config");
644
+ const content = await readOptionalText(configPath);
645
+ repository.isReftable = content ? parseGitConfigHasReftable(content) : false;
646
+ return repository.isReftable;
647
+ }
648
+
649
+ async function resolveHeadStateReftable(repository: GitRepository, signal?: AbortSignal): Promise<GitHeadState | null> {
650
+ throwIfAborted(signal);
651
+ const symResult = await git(repository.repoRoot, ["symbolic-ref", "HEAD"], { readOnly: true, signal }).catch(err => {
652
+ if (signal?.aborted || (err instanceof Error && (err.name === "AbortError" || err.name === "ToolAbortError"))) {
653
+ throw err;
654
+ }
655
+ return null;
656
+ });
657
+ throwIfAborted(signal);
658
+ const revResult = await git(repository.repoRoot, ["rev-parse", "--verify", "HEAD"], {
659
+ readOnly: true,
660
+ signal,
661
+ }).catch(err => {
662
+ if (signal?.aborted || (err instanceof Error && (err.name === "AbortError" || err.name === "ToolAbortError"))) {
663
+ throw err;
664
+ }
665
+ return null;
666
+ });
667
+ const commit = revResult && revResult.exitCode === 0 ? revResult.stdout.trim() || null : null;
668
+
669
+ if (symResult && symResult.exitCode === 0) {
670
+ const ref = symResult.stdout.trim();
671
+ const branchName = ref.startsWith(LOCAL_BRANCH_PREFIX) ? ref.slice(LOCAL_BRANCH_PREFIX.length) : null;
672
+ return {
673
+ ...repository,
674
+ kind: "ref",
675
+ ref,
676
+ branchName,
677
+ commit,
678
+ headContent: `${HEAD_REF_PREFIX} ${ref}`,
679
+ };
680
+ }
681
+
682
+ return {
683
+ ...repository,
684
+ kind: "detached",
685
+ commit,
686
+ headContent: commit || "",
687
+ };
688
+ }
689
+
690
+ function resolveHeadStateReftableSync(repository: GitRepository): GitHeadState | null {
691
+ ensureAvailable();
692
+ const symArgs = withShortLivedGitConfig(withNoOptionalLocks(["symbolic-ref", "HEAD"]));
693
+ const symResult = Bun.spawnSync(["git", ...symArgs], {
694
+ cwd: repository.repoRoot,
695
+ stdout: "pipe",
696
+ stderr: "pipe",
697
+ windowsHide: true,
698
+ });
699
+
700
+ const revArgs = withShortLivedGitConfig(withNoOptionalLocks(["rev-parse", "--verify", "HEAD"]));
701
+ const revResult = Bun.spawnSync(["git", ...revArgs], {
702
+ cwd: repository.repoRoot,
703
+ stdout: "pipe",
704
+ stderr: "pipe",
705
+ windowsHide: true,
706
+ });
707
+ const commit = revResult.exitCode === 0 ? new TextDecoder().decode(revResult.stdout).trim() || null : null;
708
+
709
+ if (symResult.exitCode === 0) {
710
+ const ref = new TextDecoder().decode(symResult.stdout).trim();
711
+ const branchName = ref.startsWith(LOCAL_BRANCH_PREFIX) ? ref.slice(LOCAL_BRANCH_PREFIX.length) : null;
712
+ return {
713
+ ...repository,
714
+ kind: "ref",
715
+ ref,
716
+ branchName,
717
+ commit,
718
+ headContent: `${HEAD_REF_PREFIX} ${ref}`,
719
+ };
720
+ }
721
+
722
+ return {
723
+ ...repository,
724
+ kind: "detached",
725
+ commit,
726
+ headContent: commit || "",
727
+ };
728
+ }
729
+
730
+ function readRefSync(repository: GitRepository, targetRef: string): string | null {
731
+ if (isReftableRepoSync(repository)) {
732
+ ensureAvailable();
733
+ const symArgs = withShortLivedGitConfig(withNoOptionalLocks(["symbolic-ref", targetRef]));
734
+ const symResult = Bun.spawnSync(["git", ...symArgs], {
735
+ cwd: repository.repoRoot,
736
+ stdout: "pipe",
737
+ stderr: "pipe",
738
+ windowsHide: true,
739
+ });
740
+ if (symResult.exitCode === 0) {
741
+ const stdoutText = new TextDecoder().decode(symResult.stdout).trim();
742
+ return `${HEAD_REF_PREFIX} ${stdoutText}`;
743
+ }
744
+ const revArgs = withShortLivedGitConfig(withNoOptionalLocks(["rev-parse", "--verify", targetRef]));
745
+ const revResult = Bun.spawnSync(["git", ...revArgs], {
746
+ cwd: repository.repoRoot,
747
+ stdout: "pipe",
748
+ stderr: "pipe",
749
+ windowsHide: true,
750
+ });
751
+ if (revResult.exitCode === 0) {
752
+ return new TextDecoder().decode(revResult.stdout).trim() || null;
753
+ }
754
+ return null;
755
+ }
756
+
757
+ for (const dir of getRefLookupDirs(repository)) {
758
+ const value = normalizeRefValue(readOptionalTextSync(path.join(dir, targetRef)));
759
+ if (value) return value;
760
+ }
761
+ for (const dir of getRefLookupDirs(repository)) {
762
+ const value = parsePackedRefs(readOptionalTextSync(path.join(dir, "packed-refs")), targetRef);
763
+ if (value) return value;
764
+ }
765
+ return null;
766
+ }
767
+
768
+ async function readRef(repository: GitRepository, targetRef: string, signal?: AbortSignal): Promise<string | null> {
769
+ if (await isReftableRepo(repository)) {
770
+ throwIfAborted(signal);
771
+ const symResult = await git(repository.repoRoot, ["symbolic-ref", targetRef], { readOnly: true, signal }).catch(
772
+ err => {
773
+ if (
774
+ signal?.aborted ||
775
+ (err instanceof Error && (err.name === "AbortError" || err.name === "ToolAbortError"))
776
+ ) {
777
+ throw err;
778
+ }
779
+ return null;
780
+ },
781
+ );
782
+ if (symResult && symResult.exitCode === 0) {
783
+ return `${HEAD_REF_PREFIX} ${symResult.stdout.trim()}`;
784
+ }
785
+ throwIfAborted(signal);
786
+ const revResult = await git(repository.repoRoot, ["rev-parse", "--verify", targetRef], {
787
+ readOnly: true,
788
+ signal,
789
+ }).catch(err => {
790
+ if (
791
+ signal?.aborted ||
792
+ (err instanceof Error && (err.name === "AbortError" || err.name === "ToolAbortError"))
793
+ ) {
794
+ throw err;
795
+ }
796
+ return null;
797
+ });
798
+ if (revResult && revResult.exitCode === 0) {
799
+ return revResult.stdout.trim() || null;
800
+ }
801
+ return null;
802
+ }
803
+
804
+ for (const dir of getRefLookupDirs(repository)) {
805
+ const value = normalizeRefValue(await readOptionalText(path.join(dir, targetRef)));
806
+ if (value) return value;
807
+ }
808
+ for (const dir of getRefLookupDirs(repository)) {
809
+ const value = parsePackedRefs(await readOptionalText(path.join(dir, "packed-refs")), targetRef);
810
+ if (value) return value;
811
+ }
812
+ return null;
813
+ }
814
+
815
+ // ════════════════════════════════════════════════════════════════════════════
816
+ // Internal: Head state parsing
817
+ // ════════════════════════════════════════════════════════════════════════════
818
+
819
+ function parseHeadStateSync(repository: GitRepository, headContent: string): GitHeadState {
820
+ const trimmed = headContent.trim();
821
+ if (!trimmed?.startsWith(HEAD_REF_PREFIX)) {
822
+ return { ...repository, commit: trimmed || null, headContent, kind: "detached" };
823
+ }
824
+ const refValue = trimmed.slice(HEAD_REF_PREFIX.length).trim();
825
+ const branchName = refValue.startsWith(LOCAL_BRANCH_PREFIX) ? refValue.slice(LOCAL_BRANCH_PREFIX.length) : null;
826
+ return {
827
+ ...repository,
828
+ branchName,
829
+ commit: readRefSync(repository, refValue),
830
+ headContent,
831
+ kind: "ref",
832
+ ref: refValue,
833
+ };
834
+ }
835
+
836
+ async function parseHeadState(repository: GitRepository, headContent: string): Promise<GitHeadState> {
837
+ const trimmed = headContent.trim();
838
+ if (!trimmed?.startsWith(HEAD_REF_PREFIX)) {
839
+ return { ...repository, commit: trimmed || null, headContent, kind: "detached" };
840
+ }
841
+ const refValue = trimmed.slice(HEAD_REF_PREFIX.length).trim();
842
+ const branchName = refValue.startsWith(LOCAL_BRANCH_PREFIX) ? refValue.slice(LOCAL_BRANCH_PREFIX.length) : null;
843
+ return {
844
+ ...repository,
845
+ branchName,
846
+ commit: await readRef(repository, refValue),
847
+ headContent,
848
+ kind: "ref",
849
+ ref: refValue,
850
+ };
851
+ }
852
+
853
+ function parseDefaultBranchRef(refPath: string, target: string | null): string | null {
854
+ if (!target?.startsWith(HEAD_REF_PREFIX)) return null;
855
+ const resolvedRef = target.slice(HEAD_REF_PREFIX.length).trim();
856
+ const remotePrefix = refPath.slice(0, -"HEAD".length);
857
+ if (!resolvedRef.startsWith(remotePrefix)) return null;
858
+ return resolvedRef.slice(remotePrefix.length) || null;
859
+ }
860
+
861
+ function stripRemotePrefix(refValue: string): string | null {
862
+ const slash = refValue.indexOf("/");
863
+ if (slash < 0) return refValue || null;
864
+ return refValue.slice(slash + 1) || null;
865
+ }
866
+
867
+ function parseWorktreeList(text: string): GitWorktreeEntry[] {
868
+ const trimmed = text.trim();
869
+ if (!trimmed) return [];
870
+ return trimmed
871
+ .split(/\n\s*\n/)
872
+ .map(block => block.trim())
873
+ .filter(Boolean)
874
+ .map(block => {
875
+ const entry: GitWorktreeEntry = { detached: false, path: "" };
876
+ for (const line of block.split("\n")) {
877
+ if (line.startsWith("worktree ")) entry.path = line.slice("worktree ".length);
878
+ else if (line.startsWith("HEAD ")) entry.head = line.slice("HEAD ".length);
879
+ else if (line.startsWith("branch ")) entry.branch = line.slice("branch ".length);
880
+ else if (line === "detached") entry.detached = true;
881
+ }
882
+ return entry;
883
+ });
884
+ }
885
+
886
+ // ════════════════════════════════════════════════════════════════════════════
887
+ // Internal: Hunk selection
888
+ // ════════════════════════════════════════════════════════════════════════════
889
+
890
+ function extractFileHeader(diffText: string): string {
891
+ const lines = diffText.split("\n");
892
+ const headerLines: string[] = [];
893
+ for (const line of lines) {
894
+ if (line.startsWith("@@")) break;
895
+ headerLines.push(line);
896
+ }
897
+ return headerLines.join("\n");
898
+ }
899
+
900
+ function selectHunks(file: FileHunks, selector: HunkSelection["hunks"]): FileHunks["hunks"] {
901
+ if (selector.type === "indices") {
902
+ const wanted = new Set(selector.indices.map(v => Math.max(1, Math.floor(v))));
903
+ return file.hunks.filter(hunk => wanted.has(hunk.index + 1));
904
+ }
905
+ if (selector.type === "lines") {
906
+ const start = Math.floor(selector.start);
907
+ const end = Math.floor(selector.end);
908
+ return file.hunks.filter(hunk => hunk.newStart <= end && hunk.newStart + hunk.newLines - 1 >= start);
909
+ }
910
+ return file.hunks;
911
+ }
912
+
913
+ export function createHunkSelectionValidator(
914
+ rawDiff: string,
915
+ ): (selections: readonly HunkSelection[]) => HunkSelectionValidationError[] {
916
+ const fileDiffMap = new Map(parseFileDiffs(rawDiff).map(entry => [entry.filename, entry]));
917
+ return selections => validateHunkSelectionsFromMap(fileDiffMap, selections);
918
+ }
919
+
920
+ function validateHunkSelectionsFromMap(
921
+ fileDiffMap: ReadonlyMap<string, FileDiff>,
922
+ selections: readonly HunkSelection[],
923
+ ): HunkSelectionValidationError[] {
924
+ const errors: HunkSelectionValidationError[] = [];
925
+
926
+ for (const selection of selections) {
927
+ const fileDiff = fileDiffMap.get(selection.path);
928
+ if (!fileDiff) continue;
929
+ if (selection.hunks.type === "all") continue;
930
+ if (fileDiff.isBinary) {
931
+ errors.push({ path: selection.path, message: `Cannot select hunks for binary file ${selection.path}` });
932
+ continue;
933
+ }
934
+ const selected = selectHunks(parseFileHunks(fileDiff), selection.hunks);
935
+ if (selected.length === 0) {
936
+ errors.push({ path: selection.path, message: `No hunks selected for ${selection.path}` });
937
+ }
938
+ }
939
+
940
+ return errors;
941
+ }
942
+
943
+ export function validateHunkSelections(
944
+ rawDiff: string,
945
+ selections: readonly HunkSelection[],
946
+ ): HunkSelectionValidationError[] {
947
+ return createHunkSelectionValidator(rawDiff)(selections);
948
+ }
949
+
950
+ function parseStatusPorcelain(text: string): GitStatusSummary {
951
+ let staged = 0;
952
+ let unstaged = 0;
953
+ let untracked = 0;
954
+ for (const line of text.split("\n")) {
955
+ if (!line) continue;
956
+ const x = line[0];
957
+ const y = line[1];
958
+ if (x === "?" && y === "?") {
959
+ untracked += 1;
960
+ continue;
961
+ }
962
+ if (x && x !== " " && x !== "?") staged += 1;
963
+ if (y && y !== " ") unstaged += 1;
964
+ }
965
+ return { staged, unstaged, untracked };
966
+ }
967
+
968
+ // ════════════════════════════════════════════════════════════════════════════
969
+ // API: diff
970
+ // ════════════════════════════════════════════════════════════════════════════
971
+
972
+ /** Run `git diff` with the given options. Returns raw diff text. */
973
+ export const diff = Object.assign(
974
+ async function diff(cwd: string, options: DiffOptions = {}): Promise<string> {
975
+ const args = buildDiffArgs(options);
976
+ if (options.allowFailure) {
977
+ return (await git(cwd, args, { env: options.env, readOnly: true, signal: options.signal })).stdout;
978
+ }
979
+ return runText(cwd, args, { env: options.env, readOnly: true, signal: options.signal });
980
+ },
981
+ {
982
+ /** List changed file paths. */
983
+ async changedFiles(
984
+ cwd: string,
985
+ options: Pick<DiffOptions, "cached" | "files" | "signal"> = {},
986
+ ): Promise<string[]> {
987
+ return splitLines(await diff(cwd, { ...options, nameOnly: true }));
988
+ },
989
+ /** Parsed per-file add/remove counts. */
990
+ async numstat(cwd: string, options: Pick<DiffOptions, "cached" | "signal"> = {}): Promise<NumstatEntry[]> {
991
+ return parseNumstat(await diff(cwd, { ...options, numstat: true }));
992
+ },
993
+ /** Parsed diff hunks for the given files. */
994
+ async hunks(
995
+ cwd: string,
996
+ files: readonly string[],
997
+ options: { cached?: boolean; signal?: AbortSignal } = {},
998
+ ): Promise<FileHunks[]> {
999
+ return parseCommitDiffHunks(
1000
+ await diff(cwd, { cached: options.cached ?? true, files, signal: options.signal }),
1001
+ );
1002
+ },
1003
+ /** Check whether a diff exists (uses `--quiet` for efficiency). */
1004
+ async has(cwd: string, options: Pick<DiffOptions, "cached" | "files" | "signal"> = {}): Promise<boolean> {
1005
+ const args = ["diff"];
1006
+ if (options.cached) args.push("--cached");
1007
+ args.push("--quiet");
1008
+ if (options.files?.length) args.push("--", ...options.files);
1009
+ const result = await git(cwd, args, { readOnly: true, signal: options.signal });
1010
+ if (result.exitCode === 0) return false;
1011
+ if (result.exitCode === 1) return true;
1012
+ throw new GitCommandError(args, result);
1013
+ },
1014
+ /** Diff between two tree-ish objects (`git diff-tree`). */
1015
+ async tree(
1016
+ cwd: string,
1017
+ base: string,
1018
+ headRef: string,
1019
+ options: { binary?: boolean; signal?: AbortSignal; allowFailure?: boolean } = {},
1020
+ ): Promise<string> {
1021
+ const args = ["diff-tree", "-r", "-p"];
1022
+ if (options.binary) args.push("--binary");
1023
+ args.push(base, headRef);
1024
+ if (options.allowFailure) {
1025
+ return (await git(cwd, args, { readOnly: true, signal: options.signal })).stdout;
1026
+ }
1027
+ return runText(cwd, args, { readOnly: true, signal: options.signal });
1028
+ },
1029
+ /** Parse raw diff text into per-file diffs. */
1030
+ parseFiles(text: string): FileDiff[] {
1031
+ return parseFileDiffs(text);
1032
+ },
1033
+ /** Parse raw diff text into per-file hunks. */
1034
+ parseHunks(text: string): FileHunks[] {
1035
+ return parseCommitDiffHunks(text);
1036
+ },
1037
+ },
1038
+ );
1039
+
1040
+ // ════════════════════════════════════════════════════════════════════════════
1041
+ // API: status
1042
+ // ════════════════════════════════════════════════════════════════════════════
1043
+
1044
+ /** Run `git status --porcelain`. Returns raw status text. */
1045
+ export const status = Object.assign(
1046
+ async function status(cwd: string, options: StatusOptions = {}): Promise<string> {
1047
+ const args = ["status"];
1048
+ args.push(options.porcelainV1 ? "--porcelain=v1" : "--porcelain");
1049
+ if (options.z) args.push("-z");
1050
+ if (options.untrackedFiles) args.push(`--untracked-files=${options.untrackedFiles}`);
1051
+ if (options.pathspecs?.length) args.push("--", ...options.pathspecs);
1052
+ return runText(cwd, args, { readOnly: true, signal: options.signal });
1053
+ },
1054
+ {
1055
+ /** Parsed status counts (staged, unstaged, untracked). */
1056
+ async summary(cwd: string, signal?: AbortSignal): Promise<GitStatusSummary | null> {
1057
+ const result = await git(cwd, ["status", "--porcelain"], { readOnly: true, signal });
1058
+ if (result.exitCode !== 0) return null;
1059
+ return parseStatusPorcelain(result.stdout);
1060
+ },
1061
+ /** Parse porcelain status text into counts. */
1062
+ parse: parseStatusPorcelain,
1063
+ },
1064
+ );
1065
+
1066
+ // ════════════════════════════════════════════════════════════════════════════
1067
+ // API: stage
1068
+ // ════════════════════════════════════════════════════════════════════════════
1069
+
1070
+ export const stage = {
1071
+ /** Stage files. Empty array stages all (`git add -A`). */
1072
+ async files(cwd: string, files: readonly string[] = [], signal?: AbortSignal): Promise<void> {
1073
+ const args = files.length === 0 ? ["add", "-A"] : ["add", "--", ...files];
1074
+ await runEffect(cwd, args, { signal });
1075
+ },
1076
+
1077
+ /** Selectively stage hunks from the provided diff or the current working tree diff. */
1078
+ async hunks(cwd: string, selections: HunkSelection[], options: StageHunksOptions = {}): Promise<void> {
1079
+ if (selections.length === 0) return;
1080
+ const rawDiff = options.rawDiff ?? (await diff(cwd, { cached: options.diffCached, signal: options.signal }));
1081
+ const fileDiffs = parseFileDiffs(rawDiff);
1082
+ const fileDiffMap = new Map(fileDiffs.map(entry => [entry.filename, entry]));
1083
+ const patchParts: string[] = [];
1084
+
1085
+ for (const selection of selections) {
1086
+ const fileDiff = fileDiffMap.get(selection.path);
1087
+ if (!fileDiff) throw new Error(`No diff found for ${selection.path}`);
1088
+ if (fileDiff.isBinary) {
1089
+ if (selection.hunks.type !== "all")
1090
+ throw new Error(`Cannot select hunks for binary file ${selection.path}`);
1091
+ patchParts.push(fileDiff.content);
1092
+ continue;
1093
+ }
1094
+ if (selection.hunks.type === "all") {
1095
+ patchParts.push(fileDiff.content);
1096
+ continue;
1097
+ }
1098
+ const fileHunks = parseFileHunks(fileDiff);
1099
+ const selected = selectHunks(fileHunks, selection.hunks);
1100
+ if (selected.length === 0) throw new Error(`No hunks selected for ${selection.path}`);
1101
+ const header = extractFileHeader(fileDiff.content);
1102
+ patchParts.push([header, ...selected.map(h => h.content)].join("\n"));
1103
+ }
1104
+
1105
+ const patchText = patch.join(patchParts);
1106
+ if (!patchText.trim()) return;
1107
+ await patch.applyText(cwd, patchText, { cached: true, signal: options.signal });
1108
+ },
1109
+
1110
+ /** Unstage files. Empty array unstages all (`git reset`). */
1111
+ async reset(cwd: string, files: readonly string[] = [], signal?: AbortSignal): Promise<void> {
1112
+ const args = files.length === 0 ? ["reset"] : ["reset", "--", ...files];
1113
+ await runEffect(cwd, args, { signal });
1114
+ },
1115
+ };
1116
+
1117
+ // ════════════════════════════════════════════════════════════════════════════
1118
+ // API: commit, push, checkout
1119
+ // ════════════════════════════════════════════════════════════════════════════
1120
+
1121
+ /** Create a commit with the given message (passed via stdin). */
1122
+ export async function commit(cwd: string, message: string, options: CommitOptions = {}): Promise<GitCommandResult> {
1123
+ const args = ["commit", "-F", "-"];
1124
+ if (options.allowEmpty) args.push("--allow-empty");
1125
+ if (options.files?.length) args.push("--", ...options.files);
1126
+ return runChecked(cwd, args, { signal: options.signal, stdin: message });
1127
+ }
1128
+
1129
+ /** Push the current branch. */
1130
+ export async function push(cwd: string, options: PushOptions = {}): Promise<void> {
1131
+ const args = ["push"];
1132
+ if (options.forceWithLease) args.push("--force-with-lease");
1133
+ if (options.remote) args.push(options.remote);
1134
+ if (options.refspec) args.push(options.refspec);
1135
+ await runEffect(cwd, args, { signal: options.signal });
1136
+ }
1137
+
1138
+ /** Checkout a ref. */
1139
+ export async function checkout(cwd: string, ref: string, signal?: AbortSignal): Promise<void> {
1140
+ await runEffect(cwd, ["checkout", ref], { signal });
1141
+ }
1142
+
1143
+ /** Fetch a specific refspec from a remote. */
1144
+ export async function fetch(
1145
+ cwd: string,
1146
+ remote: string,
1147
+ source: string,
1148
+ target: string,
1149
+ signal?: AbortSignal,
1150
+ ): Promise<void> {
1151
+ await runEffect(cwd, ["fetch", remote, `+${source}:${target}`], { signal });
1152
+ }
1153
+
1154
+ /** Read a tree-ish into the index. */
1155
+ export async function readTree(
1156
+ cwd: string,
1157
+ treeish: string,
1158
+ options: Pick<CommandOptions, "env" | "signal"> = {},
1159
+ ): Promise<void> {
1160
+ await runEffect(cwd, ["read-tree", treeish], options);
1161
+ }
1162
+
1163
+ /** Write the current index as a tree and return its object id. */
1164
+ export async function writeTree(cwd: string, options: Pick<CommandOptions, "env" | "signal"> = {}): Promise<string> {
1165
+ return (await runText(cwd, ["write-tree"], options)).trim();
1166
+ }
1167
+
1168
+ // ════════════════════════════════════════════════════════════════════════════
1169
+ // API: show
1170
+ // ════════════════════════════════════════════════════════════════════════════
1171
+
1172
+ /** Run `git show` on a revision. */
1173
+ export const show = Object.assign(
1174
+ async function show(
1175
+ cwd: string,
1176
+ revision: string,
1177
+ options: { format?: string; signal?: AbortSignal } = {},
1178
+ ): Promise<string> {
1179
+ return runText(cwd, ["show", `--format=${options.format ?? ""}`, revision], {
1180
+ readOnly: true,
1181
+ signal: options.signal,
1182
+ });
1183
+ },
1184
+ {
1185
+ /** Get the path prefix of the current directory relative to the repo root. */
1186
+ async prefix(cwd: string, signal?: AbortSignal): Promise<string> {
1187
+ return (await runText(cwd, ["rev-parse", "--show-prefix"], { readOnly: true, signal })).trim();
1188
+ },
1189
+ },
1190
+ );
1191
+
1192
+ // ════════════════════════════════════════════════════════════════════════════
1193
+ // API: log
1194
+ // ════════════════════════════════════════════════════════════════════════════
1195
+
1196
+ export const log = {
1197
+ /** Recent commit subjects (one-line each). */
1198
+ async subjects(cwd: string, count: number, signal?: AbortSignal): Promise<string[]> {
1199
+ return splitLines(await runText(cwd, ["log", `-n${count}`, "--pretty=format:%s"], { readOnly: true, signal }));
1200
+ },
1201
+ /** Recent commits as `<short-sha> <subject>` onelines. */
1202
+ async onelines(cwd: string, count: number, signal?: AbortSignal): Promise<string[]> {
1203
+ return splitLines(
1204
+ await runText(cwd, ["log", `-${count}`, "--oneline", "--no-decorate"], { readOnly: true, signal }),
1205
+ );
1206
+ },
1207
+ };
1208
+
1209
+ // ════════════════════════════════════════════════════════════════════════════
1210
+ // API: branch
1211
+ // ════════════════════════════════════════════════════════════════════════════
1212
+
1213
+ export const branch = {
1214
+ /** Current branch name, or null if detached/unavailable. */
1215
+ async current(cwd: string, signal?: AbortSignal): Promise<string | null> {
1216
+ const headState = await resolveHead(cwd);
1217
+ if (headState?.kind === "ref") return headState.branchName ?? headState.ref;
1218
+ const result = await git(cwd, ["symbolic-ref", "--short", "HEAD"], { readOnly: true, signal });
1219
+ if (result.exitCode !== 0) return null;
1220
+ return result.stdout.trim() || null;
1221
+ },
1222
+
1223
+ /** Default branch name (from remote HEAD refs). */
1224
+ async default(cwd: string, signal?: AbortSignal): Promise<string | null> {
1225
+ const repository = await resolveRepository(cwd);
1226
+ if (repository) {
1227
+ for (const refPath of DEFAULT_BRANCH_REFS) {
1228
+ const target = await readRef(repository, refPath, signal);
1229
+ const branchName = parseDefaultBranchRef(refPath, target);
1230
+ if (branchName) return branchName;
1231
+ }
1232
+ }
1233
+ for (const remoteRef of ["origin/HEAD", "upstream/HEAD"]) {
1234
+ const result = await git(cwd, ["rev-parse", "--abbrev-ref", remoteRef], { readOnly: true, signal });
1235
+ if (result.exitCode !== 0) continue;
1236
+ const branchName = stripRemotePrefix(result.stdout.trim());
1237
+ if (branchName) return branchName;
1238
+ }
1239
+ return null;
1240
+ },
1241
+
1242
+ /** Create a new branch at the given start point. */
1243
+ async create(cwd: string, name: string, startPoint = "HEAD", signal?: AbortSignal): Promise<void> {
1244
+ await runEffect(cwd, ["branch", name, startPoint], { signal });
1245
+ },
1246
+
1247
+ /** Force-move a branch to a new start point. */
1248
+ async force(cwd: string, name: string, startPoint: string, signal?: AbortSignal): Promise<void> {
1249
+ await runEffect(cwd, ["branch", "--force", name, startPoint], { signal });
1250
+ },
1251
+
1252
+ /** Delete a branch. Throws on failure. */
1253
+ async delete(cwd: string, name: string, options: { force?: boolean; signal?: AbortSignal } = {}): Promise<void> {
1254
+ await runEffect(cwd, ["branch", options.force === false ? "-d" : "-D", name], { signal: options.signal });
1255
+ },
1256
+
1257
+ /** Delete a branch. Returns false on failure instead of throwing. */
1258
+ async tryDelete(
1259
+ cwd: string,
1260
+ name: string,
1261
+ options: { force?: boolean; signal?: AbortSignal } = {},
1262
+ ): Promise<boolean> {
1263
+ const result = await git(cwd, ["branch", options.force === false ? "-d" : "-D", name], {
1264
+ signal: options.signal,
1265
+ });
1266
+ return result.exitCode === 0;
1267
+ },
1268
+
1269
+ /** Create and checkout a new branch. */
1270
+ async checkoutNew(cwd: string, name: string, signal?: AbortSignal): Promise<void> {
1271
+ await runEffect(cwd, ["checkout", "-b", name], { signal });
1272
+ },
1273
+
1274
+ /** List branches. Pass `{ all: true }` to include remotes. */
1275
+ async list(cwd: string, options: { all?: boolean; signal?: AbortSignal } = {}): Promise<string[]> {
1276
+ const args = ["branch"];
1277
+ if (options.all) args.push("-a");
1278
+ args.push("--format=%(refname:short)");
1279
+ return splitLines(await runText(cwd, args, { readOnly: true, signal: options.signal }));
1280
+ },
1281
+ };
1282
+
1283
+ // ════════════════════════════════════════════════════════════════════════════
1284
+ // API: remote
1285
+ // ════════════════════════════════════════════════════════════════════════════
1286
+
1287
+ export const remote = {
1288
+ /** List remote names. */
1289
+ async list(cwd: string, signal?: AbortSignal): Promise<string[]> {
1290
+ return splitLines(await runText(cwd, ["remote"], { readOnly: true, signal }));
1291
+ },
1292
+
1293
+ /** Get the URL for a remote. */
1294
+ async url(cwd: string, name: string, signal?: AbortSignal): Promise<string | undefined> {
1295
+ return trimScalar(await tryText(cwd, ["remote", "get-url", name], { readOnly: true, signal }));
1296
+ },
1297
+
1298
+ /**
1299
+ * Add a remote pointing at `url`. Idempotent: if a remote named `name`
1300
+ * already exists with the same URL (e.g. an in-process race or a leftover
1301
+ * remote from a previous run), this is treated as success. Throws when the
1302
+ * remote exists with a different URL — that's a real conflict the caller
1303
+ * needs to resolve, not paper over.
1304
+ */
1305
+ async add(cwd: string, name: string, url: string, signal?: AbortSignal): Promise<void> {
1306
+ const result = await git(cwd, ["remote", "add", name, url], { signal });
1307
+ if (result.exitCode === 0) return;
1308
+ if (REMOTE_ALREADY_EXISTS.test(result.stderr)) {
1309
+ const existing = await remote.url(cwd, name, signal);
1310
+ if (existing === url) return;
1311
+ throw new ToolError(`remote ${name} already exists with URL ${existing ?? "(unset)"}, expected ${url}`);
1312
+ }
1313
+ throw new GitCommandError(["remote", "add", name, url], result);
1314
+ },
1315
+ };
1316
+
1317
+ // ════════════════════════════════════════════════════════════════════════════
1318
+ // API: ref
1319
+ // ════════════════════════════════════════════════════════════════════════════
1320
+
1321
+ export const ref = {
1322
+ /** Check if a ref exists. */
1323
+ async exists(cwd: string, refName: string, signal?: AbortSignal): Promise<boolean> {
1324
+ if (refName === "HEAD") return (await head.sha(cwd, signal)) !== null;
1325
+ const repository = await resolveRepository(cwd);
1326
+ if (repository && refName.startsWith("refs/")) return (await readRef(repository, refName, signal)) !== null;
1327
+ const result = await git(cwd, ["show-ref", "--verify", "--quiet", refName], { readOnly: true, signal });
1328
+ return result.exitCode === 0;
1329
+ },
1330
+
1331
+ /** Resolve a ref to its commit SHA. */
1332
+ async resolve(cwd: string, refName: string, signal?: AbortSignal): Promise<string | null> {
1333
+ if (refName === "HEAD") return head.sha(cwd, signal);
1334
+ const repository = await resolveRepository(cwd);
1335
+ if (repository && refName.startsWith("refs/")) return readRef(repository, refName, signal);
1336
+ const result = await git(cwd, ["rev-parse", refName], { readOnly: true, signal });
1337
+ if (result.exitCode !== 0) return null;
1338
+ return result.stdout.trim() || null;
1339
+ },
1340
+
1341
+ /** Tags pointing at a ref. */
1342
+ async tags(cwd: string, refName = "HEAD", signal?: AbortSignal): Promise<string[]> {
1343
+ return splitLines(
1344
+ await runText(
1345
+ cwd,
1346
+ [
1347
+ "for-each-ref",
1348
+ "--points-at",
1349
+ refName,
1350
+ "--sort=-version:refname",
1351
+ "--format=%(refname:strip=2)",
1352
+ "refs/tags",
1353
+ ],
1354
+ { readOnly: true, signal },
1355
+ ),
1356
+ );
1357
+ },
1358
+ };
1359
+
1360
+ // ════════════════════════════════════════════════════════════════════════════
1361
+ // API: config
1362
+ // ════════════════════════════════════════════════════════════════════════════
1363
+
1364
+ export const config = {
1365
+ async get(cwd: string, key: string, signal?: AbortSignal): Promise<string | undefined> {
1366
+ return trimScalar(await tryText(cwd, ["config", "--get", key], { readOnly: true, signal }));
1367
+ },
1368
+
1369
+ async set(cwd: string, key: string, value: string, signal?: AbortSignal): Promise<void> {
1370
+ await runEffect(cwd, ["config", key, value], { signal });
1371
+ },
1372
+
1373
+ async getBranch(cwd: string, branchName: string, key: string, signal?: AbortSignal): Promise<string | undefined> {
1374
+ return config.get(cwd, `branch.${branchName}.${key}`, signal);
1375
+ },
1376
+
1377
+ async setBranch(cwd: string, branchName: string, key: string, value: string, signal?: AbortSignal): Promise<void> {
1378
+ return config.set(cwd, `branch.${branchName}.${key}`, value, signal);
1379
+ },
1380
+ };
1381
+
1382
+ // ════════════════════════════════════════════════════════════════════════════
1383
+ // API: worktree
1384
+ // ════════════════════════════════════════════════════════════════════════════
1385
+
1386
+ export const worktree = {
1387
+ async add(
1388
+ cwd: string,
1389
+ worktreePath: string,
1390
+ refName: string,
1391
+ options: { detach?: boolean; signal?: AbortSignal } = {},
1392
+ ): Promise<void> {
1393
+ const args = ["worktree", "add"];
1394
+ if (options.detach) args.push("--detach");
1395
+ args.push(worktreePath, refName);
1396
+ await runEffect(cwd, args, { signal: options.signal });
1397
+ },
1398
+
1399
+ async remove(
1400
+ cwd: string,
1401
+ worktreePath: string,
1402
+ options: { force?: boolean; signal?: AbortSignal } = {},
1403
+ ): Promise<void> {
1404
+ const args = ["worktree", "remove"];
1405
+ if (options.force ?? true) args.push("-f");
1406
+ args.push(worktreePath);
1407
+ await runEffect(cwd, args, { signal: options.signal });
1408
+ },
1409
+
1410
+ async tryRemove(
1411
+ cwd: string,
1412
+ worktreePath: string,
1413
+ options: { force?: boolean; signal?: AbortSignal } = {},
1414
+ ): Promise<boolean> {
1415
+ const args = ["worktree", "remove"];
1416
+ if (options.force ?? true) args.push("-f");
1417
+ args.push(worktreePath);
1418
+ const result = await git(cwd, args, { signal: options.signal });
1419
+ return result.exitCode === 0;
1420
+ },
1421
+
1422
+ async list(cwd: string, signal?: AbortSignal): Promise<GitWorktreeEntry[]> {
1423
+ return parseWorktreeList(await runText(cwd, ["worktree", "list", "--porcelain"], { readOnly: true, signal }));
1424
+ },
1425
+
1426
+ async prune(cwd: string, signal?: AbortSignal): Promise<void> {
1427
+ await runEffect(cwd, ["worktree", "prune"], { signal });
1428
+ },
1429
+ };
1430
+
1431
+ // ════════════════════════════════════════════════════════════════════════════
1432
+ // API: patch
1433
+ // ════════════════════════════════════════════════════════════════════════════
1434
+
1435
+ export const patch = {
1436
+ /** Apply a patch file. */
1437
+ async apply(cwd: string, patchPath: string, options: PatchOptions = {}): Promise<void> {
1438
+ await runEffect(cwd, buildApplyArgs(patchPath, options), { env: options.env, signal: options.signal });
1439
+ },
1440
+
1441
+ /** Apply a patch from a string (writes to a temp file). */
1442
+ async applyText(cwd: string, patchText: string, options: PatchOptions = {}): Promise<void> {
1443
+ if (!patchText.trim()) return;
1444
+ const tempPath = await writeTempPatch(patchText);
1445
+ try {
1446
+ await patch.apply(cwd, tempPath, options);
1447
+ } finally {
1448
+ await fs.promises.rm(tempPath, { force: true });
1449
+ }
1450
+ },
1451
+
1452
+ /** Check if a patch file can be applied cleanly. */
1453
+ async canApply(cwd: string, patchPath: string, options: Omit<PatchOptions, "check"> = {}): Promise<boolean> {
1454
+ const result = await git(cwd, buildApplyArgs(patchPath, { ...options, check: true }), {
1455
+ env: options.env,
1456
+ readOnly: true,
1457
+ signal: options.signal,
1458
+ });
1459
+ return result.exitCode === 0;
1460
+ },
1461
+
1462
+ /** Check if a patch string can be applied cleanly. */
1463
+ async canApplyText(cwd: string, patchText: string, options: Omit<PatchOptions, "check"> = {}): Promise<boolean> {
1464
+ if (!patchText.trim()) return true;
1465
+ const tempPath = await writeTempPatch(patchText);
1466
+ try {
1467
+ return await patch.canApply(cwd, tempPath, options);
1468
+ } finally {
1469
+ await fs.promises.rm(tempPath, { force: true });
1470
+ }
1471
+ },
1472
+
1473
+ /** Join patch parts into a single patch string. */
1474
+ join(parts: string[]): string {
1475
+ return `${parts
1476
+ .map(part => (part.endsWith("\n") ? part : `${part}\n`))
1477
+ .join("\n")
1478
+ .replace(/\n+$/, "")}\n`;
1479
+ },
1480
+ };
1481
+
1482
+ // ════════════════════════════════════════════════════════════════════════════
1483
+ // API: cherryPick
1484
+ // ════════════════════════════════════════════════════════════════════════════
1485
+
1486
+ export const cherryPick = Object.assign(
1487
+ async function cherryPick(cwd: string, revision: string, signal?: AbortSignal): Promise<void> {
1488
+ await runEffect(cwd, ["cherry-pick", revision], { signal });
1489
+ },
1490
+ {
1491
+ async abort(cwd: string, signal?: AbortSignal): Promise<void> {
1492
+ await runEffect(cwd, ["cherry-pick", "--abort"], { signal });
1493
+ },
1494
+ },
1495
+ );
1496
+
1497
+ // ════════════════════════════════════════════════════════════════════════════
1498
+ // API: stash
1499
+ // ════════════════════════════════════════════════════════════════════════════
1500
+
1501
+ export const stash = {
1502
+ /** Stash working tree + index changes. Returns true when git created a new stash entry. */
1503
+ async push(cwd: string, message?: string): Promise<boolean> {
1504
+ ensureAvailable();
1505
+ const previousStash = await ref.resolve(cwd, "refs/stash");
1506
+ const args = ["stash", "push", "--include-untracked"];
1507
+ if (message) args.push("-m", message);
1508
+ await runEffect(cwd, args);
1509
+ const nextStash = await ref.resolve(cwd, "refs/stash");
1510
+ return nextStash !== null && nextStash !== previousStash;
1511
+ },
1512
+ /** Pop the most recent stash entry, optionally restoring its staged state. */
1513
+ async pop(cwd: string, options?: { index?: boolean }): Promise<void> {
1514
+ const args = ["stash", "pop"];
1515
+ if (options?.index) args.push("--index");
1516
+ await runEffect(cwd, args);
1517
+ },
1518
+ };
1519
+
1520
+ // ════════════════════════════════════════════════════════════════════════════
1521
+ // API: clone, restore, clean
1522
+ // ════════════════════════════════════════════════════════════════════════════
1523
+
1524
+ export async function clone(url: string, targetDir: string, options: CloneOptions = {}): Promise<void> {
1525
+ ensureAvailable();
1526
+ const absoluteTarget = path.resolve(targetDir);
1527
+ await fs.promises.mkdir(path.dirname(absoluteTarget), { recursive: true });
1528
+
1529
+ // `git clone --depth 1 --single-branch` only fetches the tip of the target
1530
+ // branch, so any subsequent `git checkout <sha>` for a non-tip commit fails
1531
+ // with "reference is not a tree". When the caller pinned a specific SHA we
1532
+ // fall back to a full clone so the object is guaranteed to be present.
1533
+ const shallow = !options.sha;
1534
+ const args = ["clone"];
1535
+ if (shallow) args.push("--depth", "1");
1536
+ if (options.ref) args.push("--branch", options.ref, "--single-branch");
1537
+ else if (shallow) args.push("--single-branch");
1538
+ args.push(url, absoluteTarget);
1539
+
1540
+ try {
1541
+ await runEffect(path.dirname(absoluteTarget), args, { signal: options.signal });
1542
+ if (options.sha) {
1543
+ try {
1544
+ await checkout(absoluteTarget, options.sha, options.signal);
1545
+ } catch {
1546
+ await fs.promises.rm(absoluteTarget, { force: true, recursive: true });
1547
+ throw new Error(`Failed to checkout SHA ${options.sha} in cloned repository ${url}`);
1548
+ }
1549
+ }
1550
+ } catch (err) {
1551
+ await fs.promises.rm(absoluteTarget, { force: true, recursive: true });
1552
+ throw err;
1553
+ }
1554
+ }
1555
+
1556
+ export async function restore(cwd: string, options: RestoreOptions = {}): Promise<void> {
1557
+ const args = ["restore"];
1558
+ if (options.source) args.push(`--source=${options.source}`);
1559
+ if (options.staged) args.push("--staged");
1560
+ if (options.worktree) args.push("--worktree");
1561
+ if (options.files?.length) args.push("--", ...options.files);
1562
+ await runEffect(cwd, args, { signal: options.signal });
1563
+ }
1564
+
1565
+ /**
1566
+ * Run `git reset` with options. Default is a soft reset (no flag); pass `hard: true` for a destructive reset.
1567
+ *
1568
+ * NOTE: stage.reset() handles the per-file unstaging case. This helper exists for tree-wide resets.
1569
+ */
1570
+ export async function reset(
1571
+ cwd: string,
1572
+ options: { hard?: boolean; mixed?: boolean; soft?: boolean; target?: string; signal?: AbortSignal } = {},
1573
+ ): Promise<void> {
1574
+ const args = ["reset"];
1575
+ if (options.hard) args.push("--hard");
1576
+ else if (options.mixed) args.push("--mixed");
1577
+ else if (options.soft) args.push("--soft");
1578
+ if (options.target) args.push(options.target);
1579
+ await runEffect(cwd, args, { signal: options.signal });
1580
+ }
1581
+
1582
+ export async function clean(
1583
+ cwd: string,
1584
+ options: { ignoredOnly?: boolean; paths?: readonly string[]; signal?: AbortSignal } = {},
1585
+ ): Promise<void> {
1586
+ const args = ["clean", options.ignoredOnly ? "-fdX" : "-fd"];
1587
+ if (options.paths?.length) args.push("--", ...options.paths);
1588
+ await runEffect(cwd, args, { signal: options.signal });
1589
+ }
1590
+
1591
+ // ════════════════════════════════════════════════════════════════════════════
1592
+ // API: ls
1593
+ // ════════════════════════════════════════════════════════════════════════════
1594
+
1595
+ export const ls = {
1596
+ /** List files tracked or untracked by git. */
1597
+ async files(
1598
+ cwd: string,
1599
+ options: { others?: boolean; excludeStandard?: boolean; signal?: AbortSignal } = {},
1600
+ ): Promise<string[]> {
1601
+ const args = ["ls-files"];
1602
+ if (options.others) args.push("--others");
1603
+ if (options.excludeStandard) args.push("--exclude-standard");
1604
+ return splitLines(await runText(cwd, args, { readOnly: true, signal: options.signal }));
1605
+ },
1606
+
1607
+ /** List untracked files (excludes ignored). */
1608
+ async untracked(cwd: string, signal?: AbortSignal): Promise<string[]> {
1609
+ return ls.files(cwd, { others: true, excludeStandard: true, signal });
1610
+ },
1611
+
1612
+ /** List submodule paths (recursive). */
1613
+ async submodules(cwd: string, signal?: AbortSignal): Promise<string[]> {
1614
+ const output = await git(cwd, ["submodule", "--quiet", "foreach", "--recursive", "echo $sm_path"], {
1615
+ readOnly: true,
1616
+ signal,
1617
+ });
1618
+ return splitLines(output.stdout);
1619
+ },
1620
+ };
1621
+
1622
+ // ════════════════════════════════════════════════════════════════════════════
1623
+ // API: head
1624
+ // ════════════════════════════════════════════════════════════════════════════
1625
+
1626
+ export const head = {
1627
+ /** Full HEAD state (branch, commit, repo info). */
1628
+ async resolve(cwd: string, signal?: AbortSignal): Promise<GitHeadState | null> {
1629
+ const repository = await resolveRepository(cwd);
1630
+ if (!repository) return null;
1631
+ if (await isReftableRepo(repository)) {
1632
+ return resolveHeadStateReftable(repository, signal);
1633
+ }
1634
+ const content = await readOptionalText(repository.headPath);
1635
+ if (content === null) return null;
1636
+ return parseHeadState(repository, content);
1637
+ },
1638
+
1639
+ /** Full HEAD state (synchronous). */
1640
+ resolveSync(cwd: string): GitHeadState | null {
1641
+ const repository = resolveRepositorySync(cwd);
1642
+ if (!repository) return null;
1643
+ if (isReftableRepoSync(repository)) {
1644
+ return resolveHeadStateReftableSync(repository);
1645
+ }
1646
+ const content = readOptionalTextSync(repository.headPath);
1647
+ if (content === null) return null;
1648
+ return parseHeadStateSync(repository, content);
1649
+ },
1650
+
1651
+ /** Current HEAD commit SHA. */
1652
+ async sha(cwd: string, signal?: AbortSignal): Promise<string | null> {
1653
+ const headState = await head.resolve(cwd, signal);
1654
+ if (headState?.commit) return headState.commit;
1655
+ const result = await git(cwd, ["rev-parse", "HEAD"], { readOnly: true, signal });
1656
+ if (result.exitCode !== 0) return null;
1657
+ return result.stdout.trim() || null;
1658
+ },
1659
+
1660
+ /** Abbreviated HEAD commit SHA. */
1661
+ async short(cwd: string, length = 7, signal?: AbortSignal): Promise<string | null> {
1662
+ const result = await git(cwd, ["rev-parse", `--short=${length}`, "HEAD"], { readOnly: true, signal });
1663
+ if (result.exitCode !== 0) return null;
1664
+ return result.stdout.trim() || null;
1665
+ },
1666
+ };
1667
+
1668
+ // ════════════════════════════════════════════════════════════════════════════
1669
+ // API: repo
1670
+ // ════════════════════════════════════════════════════════════════════════════
1671
+
1672
+ export const repo = {
1673
+ /** Resolve the repository root (may be a worktree root). */
1674
+ async root(cwd: string, signal?: AbortSignal): Promise<string | null> {
1675
+ const repository = await resolveRepository(cwd);
1676
+ if (repository) return repository.repoRoot;
1677
+ const result = await git(cwd, ["rev-parse", "--show-toplevel"], { readOnly: true, signal });
1678
+ if (result.exitCode !== 0) return null;
1679
+ return result.stdout.trim() || null;
1680
+ },
1681
+
1682
+ /** Resolve the primary checkout root, or the shared common dir for bare-repo worktrees. */
1683
+ async primaryRoot(cwd: string, signal?: AbortSignal): Promise<string | null> {
1684
+ const repository = await resolveRepository(cwd);
1685
+ if (repository) return primaryRootFromRepository(repository);
1686
+ const repoRoot = await repo.root(cwd, signal);
1687
+ if (!repoRoot) return null;
1688
+ const commonDir = await runText(repoRoot, ["rev-parse", "--path-format=absolute", "--git-common-dir"], {
1689
+ readOnly: true,
1690
+ signal,
1691
+ });
1692
+ if (path.basename(commonDir.trim()) === ".git") return path.dirname(commonDir.trim());
1693
+ return repoRoot;
1694
+ },
1695
+
1696
+ /**
1697
+ * Sync sibling of {@link primaryRoot}. Resolves only via on-disk `.git`/
1698
+ * `commondir` walking — no subprocess fallback — so it stays usable from
1699
+ * paths where async I/O is impractical (e.g. `computeBankScope`). Returns
1700
+ * `null` when `cwd` is outside a repository. Bare-repo worktrees resolve to
1701
+ * the shared common dir (`foo.git`) because they have no primary checkout.
1702
+ */
1703
+ primaryRootSync(cwd: string): string | null {
1704
+ const repository = resolveRepositorySync(cwd);
1705
+ if (!repository) return null;
1706
+ return primaryRootFromRepositorySync(repository);
1707
+ },
1708
+
1709
+ /** Full GitRepository metadata (sync). */
1710
+ resolveSync(cwd: string): GitRepository | null {
1711
+ return resolveRepositorySync(cwd);
1712
+ },
1713
+
1714
+ /** Full GitRepository metadata. */
1715
+ resolve(cwd: string): Promise<GitRepository | null> {
1716
+ return resolveRepository(cwd);
1717
+ },
1718
+
1719
+ /** Check if the repository uses the reftable reference storage format (sync). */
1720
+ isReftableSync(repository: GitRepository): boolean {
1721
+ return isReftableRepoSync(repository);
1722
+ },
1723
+
1724
+ /** Check if the repository uses the reftable reference storage format. */
1725
+ isReftable(repository: GitRepository): Promise<boolean> {
1726
+ return isReftableRepo(repository);
1727
+ },
1728
+ };
1729
+
1730
+ // Helper used during head resolution — defined here to reference `head` namespace.
1731
+ async function resolveHead(cwd: string, signal?: AbortSignal): Promise<GitHeadState | null> {
1732
+ return head.resolve(cwd, signal);
1733
+ }
1734
+
1735
+ // ════════════════════════════════════════════════════════════════════════════
1736
+ // API: github (GitHub CLI)
1737
+ // ════════════════════════════════════════════════════════════════════════════
1738
+
1739
+ export interface GhCommandResult {
1740
+ exitCode: number;
1741
+ stdout: string;
1742
+ stderr: string;
1743
+ }
1744
+
1745
+ export interface GhCommandOptions {
1746
+ repoProvided?: boolean;
1747
+ trimOutput?: boolean;
1748
+ }
1749
+
1750
+ function formatGhFailure(args: readonly string[], stdout: string, stderr: string, options?: GhCommandOptions): string {
1751
+ const message = (stderr || stdout).trim();
1752
+ if (message.includes("gh auth login") || message.includes("not logged into any GitHub hosts")) {
1753
+ return "GitHub CLI is not authenticated. Run `gh auth login`.";
1754
+ }
1755
+ if (
1756
+ !options?.repoProvided &&
1757
+ (message.includes("not a git repository") ||
1758
+ message.includes("no git remotes found") ||
1759
+ message.includes("unable to determine current repository"))
1760
+ ) {
1761
+ return "GitHub repository context is unavailable. Pass `repo` explicitly or run the tool inside a GitHub checkout.";
1762
+ }
1763
+ if (message.length > 0) return message;
1764
+ return `GitHub CLI command failed: gh ${args.join(" ")}`;
1765
+ }
1766
+
1767
+ export const github = {
1768
+ /** Check if `gh` CLI is installed. */
1769
+ available(): boolean {
1770
+ return Boolean($which("gh"));
1771
+ },
1772
+
1773
+ /** Run a raw `gh` CLI command. Does not throw on non-zero exit. */
1774
+ async run(cwd: string, args: string[], signal?: AbortSignal, options?: GhCommandOptions): Promise<GhCommandResult> {
1775
+ throwIfAborted(signal);
1776
+ if (!$which("gh")) {
1777
+ throw new ToolError("GitHub CLI (gh) is not installed. Install it from https://cli.github.com/.");
1778
+ }
1779
+ try {
1780
+ const child = Bun.spawn(["gh", ...args], {
1781
+ cwd,
1782
+ stdin: "ignore",
1783
+ stdout: "pipe",
1784
+ stderr: "pipe",
1785
+ windowsHide: true,
1786
+ signal,
1787
+ });
1788
+ if (!child.stdout || !child.stderr) {
1789
+ throw new ToolError("Failed to capture GitHub CLI output.");
1790
+ }
1791
+ const [stdout, stderr, exitCode] = await Promise.all([
1792
+ new Response(child.stdout).text(),
1793
+ new Response(child.stderr).text(),
1794
+ child.exited,
1795
+ ]);
1796
+ throwIfAborted(signal);
1797
+ const trim = options?.trimOutput !== false;
1798
+ return {
1799
+ exitCode: exitCode ?? 0,
1800
+ stdout: trim ? stdout.trim() : stdout,
1801
+ stderr: trim ? stderr.trim() : stderr,
1802
+ };
1803
+ } catch (error) {
1804
+ if (signal?.aborted) throw new ToolAbortError();
1805
+ throw error;
1806
+ }
1807
+ },
1808
+
1809
+ /** Run `gh` and parse stdout as JSON. Throws on non-zero exit or invalid JSON. */
1810
+ async json<T>(cwd: string, args: string[], signal?: AbortSignal, options?: GhCommandOptions): Promise<T> {
1811
+ const result = await github.run(cwd, args, signal, options);
1812
+ if (result.exitCode !== 0) {
1813
+ throw new ToolError(formatGhFailure(args, result.stdout, result.stderr, options));
1814
+ }
1815
+ if (!result.stdout) {
1816
+ throw new ToolError("GitHub CLI returned empty output.");
1817
+ }
1818
+ try {
1819
+ return JSON.parse(result.stdout) as T;
1820
+ } catch {
1821
+ throw new ToolError("GitHub CLI returned invalid JSON output.");
1822
+ }
1823
+ },
1824
+
1825
+ /** Run `gh` and return stdout as text. Throws on non-zero exit. */
1826
+ async text(cwd: string, args: string[], signal?: AbortSignal, options?: GhCommandOptions): Promise<string> {
1827
+ const result = await github.run(cwd, args, signal, options);
1828
+ if (result.exitCode !== 0) {
1829
+ throw new ToolError(formatGhFailure(args, result.stdout, result.stderr, options));
1830
+ }
1831
+ return result.stdout;
1832
+ },
1833
+ };