@getpaseo/server 0.1.37 → 0.1.38

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 (295) hide show
  1. package/dist/scripts/dev-runner.js +1 -1
  2. package/dist/scripts/dev-runner.js.map +1 -1
  3. package/dist/scripts/{daemon-runner.js → supervisor-entrypoint.js} +7 -9
  4. package/dist/scripts/supervisor-entrypoint.js.map +1 -0
  5. package/dist/scripts/supervisor.js +17 -1
  6. package/dist/scripts/supervisor.js.map +1 -1
  7. package/dist/server/server/bootstrap.d.ts +0 -4
  8. package/dist/server/server/bootstrap.d.ts.map +1 -1
  9. package/dist/server/server/bootstrap.js +8 -22
  10. package/dist/server/server/bootstrap.js.map +1 -1
  11. package/dist/server/server/index.js +0 -4
  12. package/dist/server/server/index.js.map +1 -1
  13. package/dist/server/server/pid-lock.d.ts +7 -2
  14. package/dist/server/server/pid-lock.d.ts.map +1 -1
  15. package/dist/server/server/pid-lock.js +21 -0
  16. package/dist/server/server/pid-lock.js.map +1 -1
  17. package/dist/server/server/session.d.ts.map +1 -1
  18. package/dist/server/server/session.js +2 -2
  19. package/dist/server/server/session.js.map +1 -1
  20. package/dist/src/server/pid-lock.js +21 -0
  21. package/dist/src/server/pid-lock.js.map +1 -1
  22. package/package.json +3 -3
  23. package/dist/scripts/daemon-runner.js.map +0 -1
  24. package/dist/src/server/agent/activity-curator.js +0 -243
  25. package/dist/src/server/agent/activity-curator.js.map +0 -1
  26. package/dist/src/server/agent/agent-manager.js +0 -1855
  27. package/dist/src/server/agent/agent-manager.js.map +0 -1
  28. package/dist/src/server/agent/agent-metadata-generator.js +0 -161
  29. package/dist/src/server/agent/agent-metadata-generator.js.map +0 -1
  30. package/dist/src/server/agent/agent-projections.js +0 -254
  31. package/dist/src/server/agent/agent-projections.js.map +0 -1
  32. package/dist/src/server/agent/agent-response-loop.js +0 -304
  33. package/dist/src/server/agent/agent-response-loop.js.map +0 -1
  34. package/dist/src/server/agent/agent-sdk-types.js +0 -12
  35. package/dist/src/server/agent/agent-sdk-types.js.map +0 -1
  36. package/dist/src/server/agent/agent-storage.js +0 -302
  37. package/dist/src/server/agent/agent-storage.js.map +0 -1
  38. package/dist/src/server/agent/agent-title-limits.js +0 -3
  39. package/dist/src/server/agent/agent-title-limits.js.map +0 -1
  40. package/dist/src/server/agent/audio-utils.js +0 -19
  41. package/dist/src/server/agent/audio-utils.js.map +0 -1
  42. package/dist/src/server/agent/dictation-debug.js +0 -50
  43. package/dist/src/server/agent/dictation-debug.js.map +0 -1
  44. package/dist/src/server/agent/mcp-server.js +0 -754
  45. package/dist/src/server/agent/mcp-server.js.map +0 -1
  46. package/dist/src/server/agent/orchestrator-instructions.js +0 -51
  47. package/dist/src/server/agent/orchestrator-instructions.js.map +0 -1
  48. package/dist/src/server/agent/pcm16-resampler.js +0 -63
  49. package/dist/src/server/agent/pcm16-resampler.js.map +0 -1
  50. package/dist/src/server/agent/provider-launch-config.js +0 -213
  51. package/dist/src/server/agent/provider-launch-config.js.map +0 -1
  52. package/dist/src/server/agent/provider-manifest.js +0 -127
  53. package/dist/src/server/agent/provider-manifest.js.map +0 -1
  54. package/dist/src/server/agent/provider-registry.js +0 -45
  55. package/dist/src/server/agent/provider-registry.js.map +0 -1
  56. package/dist/src/server/agent/providers/claude/partial-json.js +0 -306
  57. package/dist/src/server/agent/providers/claude/partial-json.js.map +0 -1
  58. package/dist/src/server/agent/providers/claude/sdk-model-resolver.js +0 -104
  59. package/dist/src/server/agent/providers/claude/sdk-model-resolver.js.map +0 -1
  60. package/dist/src/server/agent/providers/claude/sidechain-tracker.js +0 -230
  61. package/dist/src/server/agent/providers/claude/sidechain-tracker.js.map +0 -1
  62. package/dist/src/server/agent/providers/claude/task-notification-tool-call.js +0 -267
  63. package/dist/src/server/agent/providers/claude/task-notification-tool-call.js.map +0 -1
  64. package/dist/src/server/agent/providers/claude/tool-call-detail-parser.js +0 -121
  65. package/dist/src/server/agent/providers/claude/tool-call-detail-parser.js.map +0 -1
  66. package/dist/src/server/agent/providers/claude/tool-call-mapper.js +0 -252
  67. package/dist/src/server/agent/providers/claude/tool-call-mapper.js.map +0 -1
  68. package/dist/src/server/agent/providers/claude-agent.js +0 -3166
  69. package/dist/src/server/agent/providers/claude-agent.js.map +0 -1
  70. package/dist/src/server/agent/providers/codex/tool-call-detail-parser.js +0 -104
  71. package/dist/src/server/agent/providers/codex/tool-call-detail-parser.js.map +0 -1
  72. package/dist/src/server/agent/providers/codex/tool-call-mapper.js +0 -758
  73. package/dist/src/server/agent/providers/codex/tool-call-mapper.js.map +0 -1
  74. package/dist/src/server/agent/providers/codex-app-server-agent.js +0 -2949
  75. package/dist/src/server/agent/providers/codex-app-server-agent.js.map +0 -1
  76. package/dist/src/server/agent/providers/codex-rollout-timeline.js +0 -544
  77. package/dist/src/server/agent/providers/codex-rollout-timeline.js.map +0 -1
  78. package/dist/src/server/agent/providers/opencode/tool-call-detail-parser.js +0 -39
  79. package/dist/src/server/agent/providers/opencode/tool-call-detail-parser.js.map +0 -1
  80. package/dist/src/server/agent/providers/opencode/tool-call-mapper.js +0 -144
  81. package/dist/src/server/agent/providers/opencode/tool-call-mapper.js.map +0 -1
  82. package/dist/src/server/agent/providers/opencode-agent.js +0 -1193
  83. package/dist/src/server/agent/providers/opencode-agent.js.map +0 -1
  84. package/dist/src/server/agent/providers/tool-call-detail-primitives.js +0 -686
  85. package/dist/src/server/agent/providers/tool-call-detail-primitives.js.map +0 -1
  86. package/dist/src/server/agent/providers/tool-call-mapper-utils.js +0 -115
  87. package/dist/src/server/agent/providers/tool-call-mapper-utils.js.map +0 -1
  88. package/dist/src/server/agent/recordings-debug.js +0 -19
  89. package/dist/src/server/agent/recordings-debug.js.map +0 -1
  90. package/dist/src/server/agent/stt-debug.js +0 -33
  91. package/dist/src/server/agent/stt-debug.js.map +0 -1
  92. package/dist/src/server/agent/stt-manager.js +0 -232
  93. package/dist/src/server/agent/stt-manager.js.map +0 -1
  94. package/dist/src/server/agent/timeline-append.js +0 -27
  95. package/dist/src/server/agent/timeline-append.js.map +0 -1
  96. package/dist/src/server/agent/timeline-projection.js +0 -215
  97. package/dist/src/server/agent/timeline-projection.js.map +0 -1
  98. package/dist/src/server/agent/tool-name-normalization.js +0 -45
  99. package/dist/src/server/agent/tool-name-normalization.js.map +0 -1
  100. package/dist/src/server/agent/tts-debug.js +0 -24
  101. package/dist/src/server/agent/tts-debug.js.map +0 -1
  102. package/dist/src/server/agent/tts-manager.js +0 -374
  103. package/dist/src/server/agent/tts-manager.js.map +0 -1
  104. package/dist/src/server/agent/wait-for-agent-tracker.js +0 -53
  105. package/dist/src/server/agent/wait-for-agent-tracker.js.map +0 -1
  106. package/dist/src/server/agent-attention-policy.js +0 -40
  107. package/dist/src/server/agent-attention-policy.js.map +0 -1
  108. package/dist/src/server/allowed-hosts.js +0 -94
  109. package/dist/src/server/allowed-hosts.js.map +0 -1
  110. package/dist/src/server/bootstrap.js +0 -620
  111. package/dist/src/server/bootstrap.js.map +0 -1
  112. package/dist/src/server/chat/chat-mentions.js +0 -71
  113. package/dist/src/server/chat/chat-mentions.js.map +0 -1
  114. package/dist/src/server/chat/chat-rpc-schemas.js +0 -103
  115. package/dist/src/server/chat/chat-rpc-schemas.js.map +0 -1
  116. package/dist/src/server/chat/chat-service.js +0 -330
  117. package/dist/src/server/chat/chat-service.js.map +0 -1
  118. package/dist/src/server/chat/chat-types.js +0 -22
  119. package/dist/src/server/chat/chat-types.js.map +0 -1
  120. package/dist/src/server/checkout-diff-manager.js +0 -272
  121. package/dist/src/server/checkout-diff-manager.js.map +0 -1
  122. package/dist/src/server/checkout-git-utils.js +0 -37
  123. package/dist/src/server/checkout-git-utils.js.map +0 -1
  124. package/dist/src/server/client-message-id.js +0 -12
  125. package/dist/src/server/client-message-id.js.map +0 -1
  126. package/dist/src/server/config.js +0 -73
  127. package/dist/src/server/config.js.map +0 -1
  128. package/dist/src/server/connection-offer.js +0 -59
  129. package/dist/src/server/connection-offer.js.map +0 -1
  130. package/dist/src/server/daemon-keypair.js +0 -40
  131. package/dist/src/server/daemon-keypair.js.map +0 -1
  132. package/dist/src/server/daemon-version.js +0 -22
  133. package/dist/src/server/daemon-version.js.map +0 -1
  134. package/dist/src/server/dictation/dictation-stream-manager.js +0 -571
  135. package/dist/src/server/dictation/dictation-stream-manager.js.map +0 -1
  136. package/dist/src/server/file-download/token-store.js +0 -40
  137. package/dist/src/server/file-download/token-store.js.map +0 -1
  138. package/dist/src/server/file-explorer/service.js +0 -180
  139. package/dist/src/server/file-explorer/service.js.map +0 -1
  140. package/dist/src/server/json-utils.js +0 -45
  141. package/dist/src/server/json-utils.js.map +0 -1
  142. package/dist/src/server/loop/rpc-schemas.js +0 -159
  143. package/dist/src/server/loop/rpc-schemas.js.map +0 -1
  144. package/dist/src/server/loop-service.js +0 -741
  145. package/dist/src/server/loop-service.js.map +0 -1
  146. package/dist/src/server/messages.js +0 -29
  147. package/dist/src/server/messages.js.map +0 -1
  148. package/dist/src/server/package-version.js +0 -46
  149. package/dist/src/server/package-version.js.map +0 -1
  150. package/dist/src/server/pairing-offer.js +0 -45
  151. package/dist/src/server/pairing-offer.js.map +0 -1
  152. package/dist/src/server/pairing-qr.js +0 -45
  153. package/dist/src/server/pairing-qr.js.map +0 -1
  154. package/dist/src/server/path-utils.js +0 -20
  155. package/dist/src/server/path-utils.js.map +0 -1
  156. package/dist/src/server/persisted-config.js +0 -265
  157. package/dist/src/server/persisted-config.js.map +0 -1
  158. package/dist/src/server/persistence-hooks.js +0 -60
  159. package/dist/src/server/persistence-hooks.js.map +0 -1
  160. package/dist/src/server/push/push-service.js +0 -68
  161. package/dist/src/server/push/push-service.js.map +0 -1
  162. package/dist/src/server/push/token-store.js +0 -70
  163. package/dist/src/server/push/token-store.js.map +0 -1
  164. package/dist/src/server/relay-transport.js +0 -461
  165. package/dist/src/server/relay-transport.js.map +0 -1
  166. package/dist/src/server/schedule/cron.js +0 -103
  167. package/dist/src/server/schedule/cron.js.map +0 -1
  168. package/dist/src/server/schedule/rpc-schemas.js +0 -112
  169. package/dist/src/server/schedule/rpc-schemas.js.map +0 -1
  170. package/dist/src/server/schedule/service.js +0 -397
  171. package/dist/src/server/schedule/service.js.map +0 -1
  172. package/dist/src/server/schedule/store.js +0 -56
  173. package/dist/src/server/schedule/store.js.map +0 -1
  174. package/dist/src/server/schedule/types.js +0 -73
  175. package/dist/src/server/schedule/types.js.map +0 -1
  176. package/dist/src/server/server-id.js +0 -63
  177. package/dist/src/server/server-id.js.map +0 -1
  178. package/dist/src/server/session.js +0 -6381
  179. package/dist/src/server/session.js.map +0 -1
  180. package/dist/src/server/speech/audio.js +0 -101
  181. package/dist/src/server/speech/audio.js.map +0 -1
  182. package/dist/src/server/speech/provider-resolver.js +0 -7
  183. package/dist/src/server/speech/provider-resolver.js.map +0 -1
  184. package/dist/src/server/speech/providers/local/config.js +0 -74
  185. package/dist/src/server/speech/providers/local/config.js.map +0 -1
  186. package/dist/src/server/speech/providers/local/models.js +0 -17
  187. package/dist/src/server/speech/providers/local/models.js.map +0 -1
  188. package/dist/src/server/speech/providers/local/pocket/pocket-tts-onnx.js +0 -436
  189. package/dist/src/server/speech/providers/local/pocket/pocket-tts-onnx.js.map +0 -1
  190. package/dist/src/server/speech/providers/local/runtime.js +0 -238
  191. package/dist/src/server/speech/providers/local/runtime.js.map +0 -1
  192. package/dist/src/server/speech/providers/local/sherpa/model-catalog.js +0 -166
  193. package/dist/src/server/speech/providers/local/sherpa/model-catalog.js.map +0 -1
  194. package/dist/src/server/speech/providers/local/sherpa/model-downloader.js +0 -165
  195. package/dist/src/server/speech/providers/local/sherpa/model-downloader.js.map +0 -1
  196. package/dist/src/server/speech/providers/local/sherpa/sherpa-offline-recognizer.js +0 -73
  197. package/dist/src/server/speech/providers/local/sherpa/sherpa-offline-recognizer.js.map +0 -1
  198. package/dist/src/server/speech/providers/local/sherpa/sherpa-online-recognizer.js +0 -84
  199. package/dist/src/server/speech/providers/local/sherpa/sherpa-online-recognizer.js.map +0 -1
  200. package/dist/src/server/speech/providers/local/sherpa/sherpa-onnx-loader.js +0 -11
  201. package/dist/src/server/speech/providers/local/sherpa/sherpa-onnx-loader.js.map +0 -1
  202. package/dist/src/server/speech/providers/local/sherpa/sherpa-onnx-node-loader.js +0 -102
  203. package/dist/src/server/speech/providers/local/sherpa/sherpa-onnx-node-loader.js.map +0 -1
  204. package/dist/src/server/speech/providers/local/sherpa/sherpa-parakeet-realtime-session.js +0 -135
  205. package/dist/src/server/speech/providers/local/sherpa/sherpa-parakeet-realtime-session.js.map +0 -1
  206. package/dist/src/server/speech/providers/local/sherpa/sherpa-parakeet-stt.js +0 -130
  207. package/dist/src/server/speech/providers/local/sherpa/sherpa-parakeet-stt.js.map +0 -1
  208. package/dist/src/server/speech/providers/local/sherpa/sherpa-realtime-session.js +0 -110
  209. package/dist/src/server/speech/providers/local/sherpa/sherpa-realtime-session.js.map +0 -1
  210. package/dist/src/server/speech/providers/local/sherpa/sherpa-stt.js +0 -138
  211. package/dist/src/server/speech/providers/local/sherpa/sherpa-stt.js.map +0 -1
  212. package/dist/src/server/speech/providers/local/sherpa/sherpa-tts.js +0 -98
  213. package/dist/src/server/speech/providers/local/sherpa/sherpa-tts.js.map +0 -1
  214. package/dist/src/server/speech/providers/local/sherpa/silero-vad-provider.js +0 -23
  215. package/dist/src/server/speech/providers/local/sherpa/silero-vad-provider.js.map +0 -1
  216. package/dist/src/server/speech/providers/local/sherpa/silero-vad-session.js +0 -107
  217. package/dist/src/server/speech/providers/local/sherpa/silero-vad-session.js.map +0 -1
  218. package/dist/src/server/speech/providers/openai/config.js +0 -80
  219. package/dist/src/server/speech/providers/openai/config.js.map +0 -1
  220. package/dist/src/server/speech/providers/openai/realtime-transcription-session.js +0 -168
  221. package/dist/src/server/speech/providers/openai/realtime-transcription-session.js.map +0 -1
  222. package/dist/src/server/speech/providers/openai/runtime.js +0 -112
  223. package/dist/src/server/speech/providers/openai/runtime.js.map +0 -1
  224. package/dist/src/server/speech/providers/openai/stt.js +0 -206
  225. package/dist/src/server/speech/providers/openai/stt.js.map +0 -1
  226. package/dist/src/server/speech/providers/openai/tts.js +0 -46
  227. package/dist/src/server/speech/providers/openai/tts.js.map +0 -1
  228. package/dist/src/server/speech/speech-config-resolver.js +0 -102
  229. package/dist/src/server/speech/speech-config-resolver.js.map +0 -1
  230. package/dist/src/server/speech/speech-provider.js +0 -2
  231. package/dist/src/server/speech/speech-provider.js.map +0 -1
  232. package/dist/src/server/speech/speech-runtime.js +0 -530
  233. package/dist/src/server/speech/speech-runtime.js.map +0 -1
  234. package/dist/src/server/speech/speech-types.js +0 -8
  235. package/dist/src/server/speech/speech-types.js.map +0 -1
  236. package/dist/src/server/speech/turn-detection-provider.js +0 -2
  237. package/dist/src/server/speech/turn-detection-provider.js.map +0 -1
  238. package/dist/src/server/utils/diff-highlighter.js +0 -257
  239. package/dist/src/server/utils/diff-highlighter.js.map +0 -1
  240. package/dist/src/server/voice/fixed-duration-pcm-ring-buffer.js +0 -35
  241. package/dist/src/server/voice/fixed-duration-pcm-ring-buffer.js.map +0 -1
  242. package/dist/src/server/voice/voice-turn-controller.js +0 -159
  243. package/dist/src/server/voice/voice-turn-controller.js.map +0 -1
  244. package/dist/src/server/voice-config.js +0 -51
  245. package/dist/src/server/voice-config.js.map +0 -1
  246. package/dist/src/server/voice-mcp-bridge-command.js +0 -31
  247. package/dist/src/server/voice-mcp-bridge-command.js.map +0 -1
  248. package/dist/src/server/voice-mcp-bridge.js +0 -109
  249. package/dist/src/server/voice-mcp-bridge.js.map +0 -1
  250. package/dist/src/server/voice-permission-policy.js +0 -13
  251. package/dist/src/server/voice-permission-policy.js.map +0 -1
  252. package/dist/src/server/voice-types.js +0 -2
  253. package/dist/src/server/voice-types.js.map +0 -1
  254. package/dist/src/server/websocket-server.js +0 -1048
  255. package/dist/src/server/websocket-server.js.map +0 -1
  256. package/dist/src/server/workspace-registry-bootstrap.js +0 -98
  257. package/dist/src/server/workspace-registry-bootstrap.js.map +0 -1
  258. package/dist/src/server/workspace-registry-model.js +0 -175
  259. package/dist/src/server/workspace-registry-model.js.map +0 -1
  260. package/dist/src/server/workspace-registry.js +0 -151
  261. package/dist/src/server/workspace-registry.js.map +0 -1
  262. package/dist/src/server/worktree-bootstrap.js +0 -508
  263. package/dist/src/server/worktree-bootstrap.js.map +0 -1
  264. package/dist/src/shared/agent-attention-notification.js +0 -130
  265. package/dist/src/shared/agent-attention-notification.js.map +0 -1
  266. package/dist/src/shared/agent-lifecycle.js +0 -8
  267. package/dist/src/shared/agent-lifecycle.js.map +0 -1
  268. package/dist/src/shared/connection-offer.js +0 -17
  269. package/dist/src/shared/connection-offer.js.map +0 -1
  270. package/dist/src/shared/daemon-endpoints.js +0 -122
  271. package/dist/src/shared/daemon-endpoints.js.map +0 -1
  272. package/dist/src/shared/messages.js +0 -2107
  273. package/dist/src/shared/messages.js.map +0 -1
  274. package/dist/src/shared/path-utils.js +0 -16
  275. package/dist/src/shared/path-utils.js.map +0 -1
  276. package/dist/src/shared/terminal-stream-protocol.js +0 -99
  277. package/dist/src/shared/terminal-stream-protocol.js.map +0 -1
  278. package/dist/src/shared/tool-call-display.js +0 -122
  279. package/dist/src/shared/tool-call-display.js.map +0 -1
  280. package/dist/src/terminal/terminal-manager.js +0 -136
  281. package/dist/src/terminal/terminal-manager.js.map +0 -1
  282. package/dist/src/terminal/terminal.js +0 -333
  283. package/dist/src/terminal/terminal.js.map +0 -1
  284. package/dist/src/utils/checkout-git.js +0 -1518
  285. package/dist/src/utils/checkout-git.js.map +0 -1
  286. package/dist/src/utils/directory-suggestions.js +0 -671
  287. package/dist/src/utils/directory-suggestions.js.map +0 -1
  288. package/dist/src/utils/path.js +0 -15
  289. package/dist/src/utils/path.js.map +0 -1
  290. package/dist/src/utils/project-icon.js +0 -389
  291. package/dist/src/utils/project-icon.js.map +0 -1
  292. package/dist/src/utils/worktree-metadata.js +0 -116
  293. package/dist/src/utils/worktree-metadata.js.map +0 -1
  294. package/dist/src/utils/worktree.js +0 -744
  295. package/dist/src/utils/worktree.js.map +0 -1
@@ -1,744 +0,0 @@
1
- import { exec, spawn } from "child_process";
2
- import { promisify } from "util";
3
- import { existsSync, mkdirSync, readFileSync, realpathSync, rmSync, statSync } from "fs";
4
- import { join, basename, dirname, resolve, sep } from "path";
5
- import net from "node:net";
6
- import { createHash } from "node:crypto";
7
- import { createNameId } from "mnemonic-id";
8
- import { normalizeBaseRefName, readPaseoWorktreeMetadata, readPaseoWorktreeRuntimePort, writePaseoWorktreeMetadata, writePaseoWorktreeRuntimeMetadata, } from "./worktree-metadata.js";
9
- import { resolvePaseoHome } from "../server/paseo-home.js";
10
- const execAsync = promisify(exec);
11
- const READ_ONLY_GIT_ENV = {
12
- ...process.env,
13
- GIT_OPTIONAL_LOCKS: "0",
14
- };
15
- export class WorktreeSetupError extends Error {
16
- constructor(message, results) {
17
- super(message);
18
- this.name = "WorktreeSetupError";
19
- this.results = results;
20
- }
21
- }
22
- export class WorktreeDestroyError extends Error {
23
- constructor(message, results) {
24
- super(message);
25
- this.name = "WorktreeDestroyError";
26
- this.results = results;
27
- }
28
- }
29
- function readPaseoConfig(repoRoot) {
30
- const paseoConfigPath = join(repoRoot, "paseo.json");
31
- if (!existsSync(paseoConfigPath)) {
32
- return null;
33
- }
34
- try {
35
- return JSON.parse(readFileSync(paseoConfigPath, "utf8"));
36
- }
37
- catch {
38
- throw new Error(`Failed to parse paseo.json`);
39
- }
40
- }
41
- export function getWorktreeSetupCommands(repoRoot) {
42
- const config = readPaseoConfig(repoRoot);
43
- const setupCommands = config?.worktree?.setup;
44
- if (!setupCommands || setupCommands.length === 0) {
45
- return [];
46
- }
47
- return setupCommands.filter((cmd) => typeof cmd === "string" && cmd.trim().length > 0);
48
- }
49
- export function getWorktreeDestroyCommands(repoRoot) {
50
- const config = readPaseoConfig(repoRoot);
51
- const destroyCommands = config?.worktree?.destroy;
52
- if (!destroyCommands || destroyCommands.length === 0) {
53
- return [];
54
- }
55
- return destroyCommands.filter((cmd) => typeof cmd === "string" && cmd.trim().length > 0);
56
- }
57
- export function getWorktreeTerminalSpecs(repoRoot) {
58
- const config = readPaseoConfig(repoRoot);
59
- const terminals = config?.worktree?.terminals;
60
- if (!Array.isArray(terminals) || terminals.length === 0) {
61
- return [];
62
- }
63
- const specs = [];
64
- for (const terminal of terminals) {
65
- if (!terminal || typeof terminal !== "object") {
66
- continue;
67
- }
68
- const rawCommand = terminal.command;
69
- if (typeof rawCommand !== "string") {
70
- continue;
71
- }
72
- const command = rawCommand.trim();
73
- if (!command) {
74
- continue;
75
- }
76
- const rawName = terminal.name;
77
- const name = typeof rawName === "string" && rawName.trim().length > 0 ? rawName.trim() : undefined;
78
- specs.push({
79
- ...(name ? { name } : {}),
80
- command,
81
- });
82
- }
83
- return specs;
84
- }
85
- async function execSetupCommand(command, options) {
86
- const startedAt = Date.now();
87
- try {
88
- const { stdout, stderr } = await execAsync(command, {
89
- cwd: options.cwd,
90
- env: options.env,
91
- shell: "/bin/bash",
92
- });
93
- return {
94
- command,
95
- cwd: options.cwd,
96
- stdout: stdout ?? "",
97
- stderr: stderr ?? "",
98
- exitCode: 0,
99
- durationMs: Date.now() - startedAt,
100
- };
101
- }
102
- catch (error) {
103
- return {
104
- command,
105
- cwd: options.cwd,
106
- stdout: error?.stdout ?? "",
107
- stderr: error?.stderr ?? (error instanceof Error ? error.message : String(error)),
108
- exitCode: typeof error?.code === "number" ? error.code : null,
109
- durationMs: Date.now() - startedAt,
110
- };
111
- }
112
- }
113
- async function execSetupCommandStreamed(options) {
114
- return new Promise((resolve) => {
115
- const startedAt = Date.now();
116
- const stdoutChunks = [];
117
- const stderrChunks = [];
118
- let settled = false;
119
- const finish = (exitCode) => {
120
- if (settled) {
121
- return;
122
- }
123
- settled = true;
124
- const result = {
125
- command: options.command,
126
- cwd: options.cwd,
127
- stdout: stdoutChunks.join(""),
128
- stderr: stderrChunks.join(""),
129
- exitCode,
130
- durationMs: Date.now() - startedAt,
131
- };
132
- options.onEvent?.({
133
- type: "command_completed",
134
- index: options.index,
135
- total: options.total,
136
- command: options.command,
137
- cwd: options.cwd,
138
- exitCode: result.exitCode,
139
- durationMs: result.durationMs,
140
- stdout: result.stdout,
141
- stderr: result.stderr,
142
- });
143
- resolve(result);
144
- };
145
- options.onEvent?.({
146
- type: "command_started",
147
- index: options.index,
148
- total: options.total,
149
- command: options.command,
150
- cwd: options.cwd,
151
- });
152
- const child = spawn("/bin/bash", ["-lc", options.command], {
153
- cwd: options.cwd,
154
- env: options.env,
155
- stdio: ["ignore", "pipe", "pipe"],
156
- });
157
- child.stdout?.on("data", (chunk) => {
158
- const text = chunk.toString();
159
- stdoutChunks.push(text);
160
- options.onEvent?.({
161
- type: "output",
162
- index: options.index,
163
- total: options.total,
164
- command: options.command,
165
- cwd: options.cwd,
166
- stream: "stdout",
167
- chunk: text,
168
- });
169
- });
170
- child.stderr?.on("data", (chunk) => {
171
- const text = chunk.toString();
172
- stderrChunks.push(text);
173
- options.onEvent?.({
174
- type: "output",
175
- index: options.index,
176
- total: options.total,
177
- command: options.command,
178
- cwd: options.cwd,
179
- stream: "stderr",
180
- chunk: text,
181
- });
182
- });
183
- child.on("error", (error) => {
184
- stderrChunks.push(error instanceof Error ? error.message : String(error));
185
- finish(null);
186
- });
187
- child.on("close", (code) => {
188
- finish(typeof code === "number" ? code : null);
189
- });
190
- });
191
- }
192
- async function getAvailablePort() {
193
- return new Promise((resolve, reject) => {
194
- const server = net.createServer();
195
- server.once("error", reject);
196
- server.listen(0, () => {
197
- const address = server.address();
198
- if (!address || typeof address === "string") {
199
- server.close(() => reject(new Error("Failed to acquire available port")));
200
- return;
201
- }
202
- server.close((error) => {
203
- if (error) {
204
- reject(error);
205
- return;
206
- }
207
- resolve(address.port);
208
- });
209
- });
210
- });
211
- }
212
- async function assertPortAvailable(port) {
213
- await new Promise((resolve, reject) => {
214
- const server = net.createServer();
215
- server.once("error", (error) => {
216
- const message = error?.code === "EADDRINUSE"
217
- ? `Persisted worktree port ${port} is already in use`
218
- : error instanceof Error
219
- ? error.message
220
- : String(error);
221
- reject(new Error(message));
222
- });
223
- server.listen(port, () => {
224
- server.close((error) => {
225
- if (error) {
226
- reject(error);
227
- return;
228
- }
229
- resolve();
230
- });
231
- });
232
- });
233
- }
234
- async function inferRepoRootPathFromWorktreePath(worktreePath) {
235
- try {
236
- const commonDir = await getGitCommonDir(worktreePath);
237
- const normalizedCommonDir = normalizePathForOwnership(commonDir);
238
- // Normal repo/worktree: common dir is <repoRoot>/.git
239
- if (basename(normalizedCommonDir) === ".git") {
240
- return dirname(normalizedCommonDir);
241
- }
242
- // Bare repo: common dir is the repo dir itself
243
- return normalizedCommonDir;
244
- }
245
- catch {
246
- // Fallback: best-effort resolve toplevel (will be the worktree root in typical cases)
247
- try {
248
- const { stdout } = await execAsync("git rev-parse --path-format=absolute --show-toplevel", {
249
- cwd: worktreePath,
250
- env: READ_ONLY_GIT_ENV,
251
- });
252
- const topLevel = stdout.trim();
253
- if (topLevel) {
254
- return normalizePathForOwnership(topLevel);
255
- }
256
- }
257
- catch {
258
- // ignore
259
- }
260
- return normalizePathForOwnership(worktreePath);
261
- }
262
- }
263
- export async function runWorktreeSetupCommands(options) {
264
- // Read paseo.json from the worktree (it will have the same content as the source repo)
265
- const setupCommands = getWorktreeSetupCommands(options.worktreePath);
266
- if (setupCommands.length === 0) {
267
- return [];
268
- }
269
- const runtimeEnv = options.runtimeEnv ??
270
- (await resolveWorktreeRuntimeEnv({
271
- worktreePath: options.worktreePath,
272
- branchName: options.branchName,
273
- ...(options.repoRootPath ? { repoRootPath: options.repoRootPath } : {}),
274
- }));
275
- const setupEnv = {
276
- ...process.env,
277
- ...runtimeEnv,
278
- };
279
- const results = [];
280
- for (const [index, cmd] of setupCommands.entries()) {
281
- const result = options.onEvent
282
- ? await execSetupCommandStreamed({
283
- command: cmd,
284
- cwd: options.worktreePath,
285
- env: setupEnv,
286
- index: index + 1,
287
- total: setupCommands.length,
288
- onEvent: options.onEvent,
289
- })
290
- : await execSetupCommand(cmd, {
291
- cwd: options.worktreePath,
292
- env: setupEnv,
293
- });
294
- results.push(result);
295
- if (result.exitCode !== 0) {
296
- if (options.cleanupOnFailure) {
297
- try {
298
- await execAsync(`git worktree remove "${options.worktreePath}" --force`, {
299
- cwd: options.worktreePath,
300
- });
301
- }
302
- catch {
303
- rmSync(options.worktreePath, { recursive: true, force: true });
304
- }
305
- }
306
- throw new WorktreeSetupError(`Worktree setup command failed: ${cmd}\n${result.stderr}`.trim(), results);
307
- }
308
- }
309
- return results;
310
- }
311
- async function resolveBranchNameForWorktreePath(worktreePath) {
312
- try {
313
- const { stdout } = await execAsync("git branch --show-current", {
314
- cwd: worktreePath,
315
- env: READ_ONLY_GIT_ENV,
316
- });
317
- const branchName = stdout.trim();
318
- if (branchName.length > 0) {
319
- return branchName;
320
- }
321
- }
322
- catch {
323
- // ignore
324
- }
325
- return basename(worktreePath);
326
- }
327
- export async function resolveWorktreeRuntimeEnv(options) {
328
- const repoRootPath = options.repoRootPath ?? (await inferRepoRootPathFromWorktreePath(options.worktreePath));
329
- const branchName = options.branchName ?? (await resolveBranchNameForWorktreePath(options.worktreePath));
330
- let worktreePort = readPaseoWorktreeRuntimePort(options.worktreePath);
331
- if (worktreePort === null) {
332
- worktreePort = await getAvailablePort();
333
- const metadata = readPaseoWorktreeMetadata(options.worktreePath);
334
- if (metadata) {
335
- writePaseoWorktreeRuntimeMetadata(options.worktreePath, { worktreePort });
336
- }
337
- }
338
- else {
339
- await assertPortAvailable(worktreePort);
340
- }
341
- return {
342
- // Source checkout path is the original git repo root (shared across worktrees), not the
343
- // worktree itself. This allows setup scripts to copy local files (e.g. .env) from the
344
- // source checkout.
345
- PASEO_SOURCE_CHECKOUT_PATH: repoRootPath,
346
- // Backward-compatible alias.
347
- PASEO_ROOT_PATH: repoRootPath,
348
- PASEO_WORKTREE_PATH: options.worktreePath,
349
- PASEO_BRANCH_NAME: branchName,
350
- PASEO_WORKTREE_PORT: String(worktreePort),
351
- };
352
- }
353
- export async function runWorktreeDestroyCommands(options) {
354
- // Read paseo.json from the worktree (it will have the same content as the source repo)
355
- const destroyCommands = getWorktreeDestroyCommands(options.worktreePath);
356
- if (destroyCommands.length === 0) {
357
- return [];
358
- }
359
- const repoRootPath = options.repoRootPath ?? (await inferRepoRootPathFromWorktreePath(options.worktreePath));
360
- const branchName = options.branchName ?? (await resolveBranchNameForWorktreePath(options.worktreePath));
361
- const destroyEnv = {
362
- ...process.env,
363
- // Source checkout path is the original git repo root (shared across worktrees), not the
364
- // worktree itself. This allows destroy scripts to clean resources using paths from the
365
- // source checkout.
366
- PASEO_SOURCE_CHECKOUT_PATH: repoRootPath,
367
- // Backward-compatible alias.
368
- PASEO_ROOT_PATH: repoRootPath,
369
- PASEO_WORKTREE_PATH: options.worktreePath,
370
- PASEO_BRANCH_NAME: branchName,
371
- };
372
- const results = [];
373
- for (const cmd of destroyCommands) {
374
- const result = await execSetupCommand(cmd, {
375
- cwd: options.worktreePath,
376
- env: destroyEnv,
377
- });
378
- results.push(result);
379
- if (result.exitCode !== 0) {
380
- throw new WorktreeDestroyError(`Worktree destroy command failed: ${cmd}\n${result.stderr}`.trim(), results);
381
- }
382
- }
383
- return results;
384
- }
385
- /**
386
- * Get the git common directory (shared across worktrees) for a given cwd.
387
- * This is where refs, objects, etc. are stored.
388
- */
389
- export async function getGitCommonDir(cwd) {
390
- const { stdout } = await execAsync("git rev-parse --path-format=absolute --git-common-dir", {
391
- cwd,
392
- env: READ_ONLY_GIT_ENV,
393
- });
394
- const commonDir = stdout.trim();
395
- if (!commonDir) {
396
- throw new Error("Not in a git repository");
397
- }
398
- return commonDir;
399
- }
400
- /**
401
- * Validate that a string is a valid git branch name slug
402
- * Must be lowercase, alphanumeric, hyphens only
403
- */
404
- export function validateBranchSlug(slug) {
405
- if (!slug || slug.length === 0) {
406
- return { valid: false, error: "Branch name cannot be empty" };
407
- }
408
- if (slug.length > 100) {
409
- return { valid: false, error: "Branch name too long (max 100 characters)" };
410
- }
411
- // Check for valid characters: lowercase letters, numbers, hyphens, forward slashes
412
- const validPattern = /^[a-z0-9-/]+$/;
413
- if (!validPattern.test(slug)) {
414
- return {
415
- valid: false,
416
- error: "Branch name must contain only lowercase letters, numbers, hyphens, and forward slashes",
417
- };
418
- }
419
- // Cannot start or end with hyphen
420
- if (slug.startsWith("-") || slug.endsWith("-")) {
421
- return {
422
- valid: false,
423
- error: "Branch name cannot start or end with a hyphen",
424
- };
425
- }
426
- // Cannot have consecutive hyphens
427
- if (slug.includes("--")) {
428
- return { valid: false, error: "Branch name cannot have consecutive hyphens" };
429
- }
430
- return { valid: true };
431
- }
432
- const MAX_SLUG_LENGTH = 50;
433
- /**
434
- * Convert string to kebab-case for branch names
435
- */
436
- export function slugify(input) {
437
- const slug = input
438
- .toLowerCase()
439
- .replace(/[^a-z0-9]+/g, "-")
440
- .replace(/^-+|-+$/g, "");
441
- if (slug.length <= MAX_SLUG_LENGTH) {
442
- return slug;
443
- }
444
- // Truncate at word boundary (hyphen) if possible
445
- const truncated = slug.slice(0, MAX_SLUG_LENGTH);
446
- const lastHyphen = truncated.lastIndexOf("-");
447
- if (lastHyphen > MAX_SLUG_LENGTH / 2) {
448
- return truncated.slice(0, lastHyphen);
449
- }
450
- return truncated.replace(/-+$/, "");
451
- }
452
- function generateWorktreeSlug() {
453
- return createNameId();
454
- }
455
- const WORKTREE_PROJECT_HASH_LENGTH = 8;
456
- function deriveShortAlphanumericHash(value) {
457
- const digest = createHash("sha256").update(value).digest();
458
- let hashValue = 0n;
459
- for (let index = 0; index < 8; index += 1) {
460
- hashValue = (hashValue << 8n) | BigInt(digest[index] ?? 0);
461
- }
462
- return hashValue.toString(36).padStart(13, "0").slice(0, WORKTREE_PROJECT_HASH_LENGTH);
463
- }
464
- export async function deriveWorktreeProjectHash(cwd) {
465
- try {
466
- const commonDir = await getGitCommonDir(cwd);
467
- const normalizedCommonDir = normalizePathForOwnership(commonDir);
468
- const repoRoot = basename(normalizedCommonDir) === ".git" ? dirname(normalizedCommonDir) : normalizedCommonDir;
469
- return deriveShortAlphanumericHash(repoRoot);
470
- }
471
- catch {
472
- return deriveShortAlphanumericHash(normalizePathForOwnership(cwd));
473
- }
474
- }
475
- export async function getPaseoWorktreesRoot(cwd, paseoHome) {
476
- const home = paseoHome ? resolve(paseoHome) : resolvePaseoHome();
477
- const projectHash = await deriveWorktreeProjectHash(cwd);
478
- return join(home, "worktrees", projectHash);
479
- }
480
- export async function computeWorktreePath(cwd, slug, paseoHome) {
481
- const worktreesRoot = await getPaseoWorktreesRoot(cwd, paseoHome);
482
- return join(worktreesRoot, slug);
483
- }
484
- function normalizePathForOwnership(input) {
485
- try {
486
- return realpathSync(input);
487
- }
488
- catch {
489
- return resolve(input);
490
- }
491
- }
492
- function resolveRepoRootFromGitCommonDir(commonDir) {
493
- const normalizedCommonDir = normalizePathForOwnership(commonDir);
494
- return basename(normalizedCommonDir) === ".git"
495
- ? dirname(normalizedCommonDir)
496
- : normalizedCommonDir;
497
- }
498
- export async function isPaseoOwnedWorktreeCwd(cwd, options) {
499
- let gitCommonDir;
500
- try {
501
- gitCommonDir = await getGitCommonDir(cwd);
502
- }
503
- catch {
504
- return {
505
- allowed: false,
506
- worktreePath: normalizePathForOwnership(cwd),
507
- };
508
- }
509
- const repoRoot = resolveRepoRootFromGitCommonDir(gitCommonDir);
510
- const worktreesRoot = await getPaseoWorktreesRoot(cwd, options?.paseoHome);
511
- const resolvedRoot = normalizePathForOwnership(worktreesRoot) + sep;
512
- const resolvedCwd = normalizePathForOwnership(cwd);
513
- if (!resolvedCwd.startsWith(resolvedRoot)) {
514
- return {
515
- allowed: false,
516
- repoRoot,
517
- worktreeRoot: worktreesRoot,
518
- worktreePath: resolvedCwd,
519
- };
520
- }
521
- const worktrees = await listPaseoWorktrees({ cwd, paseoHome: options?.paseoHome });
522
- const allowed = worktrees.some((entry) => {
523
- const worktreePath = resolve(entry.path);
524
- return resolvedCwd === worktreePath || resolvedCwd.startsWith(worktreePath + sep);
525
- });
526
- return {
527
- allowed,
528
- repoRoot,
529
- worktreeRoot: worktreesRoot,
530
- worktreePath: resolvedCwd,
531
- };
532
- }
533
- function parseWorktreeList(output) {
534
- const entries = [];
535
- let current = null;
536
- for (const line of output.split("\n")) {
537
- if (line.startsWith("worktree ")) {
538
- if (current?.path) {
539
- entries.push(current);
540
- }
541
- current = { path: line.slice("worktree ".length).trim() };
542
- continue;
543
- }
544
- if (!current) {
545
- continue;
546
- }
547
- if (line.startsWith("branch ")) {
548
- const ref = line.slice("branch ".length).trim();
549
- current.branchName = ref.startsWith("refs/heads/") ? ref.slice("refs/heads/".length) : ref;
550
- }
551
- else if (line.startsWith("HEAD ")) {
552
- current.head = line.slice("HEAD ".length).trim();
553
- }
554
- else if (line.trim().length === 0) {
555
- if (current.path) {
556
- entries.push(current);
557
- }
558
- current = null;
559
- }
560
- }
561
- if (current?.path) {
562
- entries.push(current);
563
- }
564
- return entries;
565
- }
566
- function resolveWorktreeCreatedAtIso(worktreePath) {
567
- try {
568
- const stats = statSync(worktreePath);
569
- const birthtimeMs = stats.birthtimeMs;
570
- const createdAtMs = Number.isFinite(birthtimeMs) && birthtimeMs > 0 ? birthtimeMs : stats.ctimeMs;
571
- return new Date(createdAtMs).toISOString();
572
- }
573
- catch {
574
- return new Date(0).toISOString();
575
- }
576
- }
577
- export async function listPaseoWorktrees({ cwd, paseoHome, }) {
578
- const worktreesRoot = await getPaseoWorktreesRoot(cwd, paseoHome);
579
- const { stdout } = await execAsync("git worktree list --porcelain", {
580
- cwd,
581
- env: READ_ONLY_GIT_ENV,
582
- });
583
- const rootPrefix = normalizePathForOwnership(worktreesRoot) + sep;
584
- return parseWorktreeList(stdout)
585
- .map((entry) => ({ ...entry, path: normalizePathForOwnership(entry.path) }))
586
- .filter((entry) => entry.path.startsWith(rootPrefix))
587
- .map((entry) => ({
588
- ...entry,
589
- createdAt: resolveWorktreeCreatedAtIso(entry.path),
590
- }));
591
- }
592
- export async function resolvePaseoWorktreeRootForCwd(cwd, options) {
593
- let gitCommonDir;
594
- try {
595
- gitCommonDir = await getGitCommonDir(cwd);
596
- }
597
- catch {
598
- return null;
599
- }
600
- const worktreesRoot = await getPaseoWorktreesRoot(cwd, options?.paseoHome);
601
- const resolvedRoot = normalizePathForOwnership(worktreesRoot) + sep;
602
- let worktreeRoot = null;
603
- try {
604
- const { stdout } = await execAsync("git rev-parse --path-format=absolute --show-toplevel", {
605
- cwd,
606
- env: READ_ONLY_GIT_ENV,
607
- });
608
- const trimmed = stdout.trim();
609
- worktreeRoot = trimmed.length > 0 ? trimmed : null;
610
- }
611
- catch {
612
- worktreeRoot = null;
613
- }
614
- if (!worktreeRoot) {
615
- return null;
616
- }
617
- const resolvedWorktreeRoot = normalizePathForOwnership(worktreeRoot);
618
- if (!resolvedWorktreeRoot.startsWith(resolvedRoot)) {
619
- return null;
620
- }
621
- const knownWorktrees = await listPaseoWorktrees({
622
- cwd,
623
- paseoHome: options?.paseoHome,
624
- });
625
- const match = knownWorktrees.find((entry) => entry.path === resolvedWorktreeRoot);
626
- if (!match) {
627
- return null;
628
- }
629
- return {
630
- repoRoot: gitCommonDir,
631
- worktreeRoot: worktreesRoot,
632
- worktreePath: match.path,
633
- };
634
- }
635
- export async function deletePaseoWorktree({ cwd, worktreePath, worktreeSlug, paseoHome, }) {
636
- if (!worktreePath && !worktreeSlug) {
637
- throw new Error("worktreePath or worktreeSlug is required");
638
- }
639
- const worktreesRoot = await getPaseoWorktreesRoot(cwd, paseoHome);
640
- const resolvedRoot = normalizePathForOwnership(worktreesRoot) + sep;
641
- const requestedPath = worktreePath ?? join(worktreesRoot, worktreeSlug);
642
- const resolvedRequested = normalizePathForOwnership(requestedPath);
643
- const resolvedWorktree = (await resolvePaseoWorktreeRootForCwd(requestedPath, { paseoHome }))?.worktreePath ??
644
- resolvedRequested;
645
- if (!resolvedWorktree.startsWith(resolvedRoot)) {
646
- throw new Error("Refusing to delete non-Paseo worktree");
647
- }
648
- await runWorktreeDestroyCommands({
649
- worktreePath: resolvedWorktree,
650
- });
651
- await execAsync(`git worktree remove "${resolvedWorktree}" --force`, {
652
- cwd,
653
- });
654
- if (existsSync(resolvedWorktree)) {
655
- rmSync(resolvedWorktree, { recursive: true, force: true });
656
- }
657
- }
658
- /**
659
- * Create a git worktree with proper naming conventions
660
- */
661
- export async function createWorktree({ branchName, cwd, baseBranch, worktreeSlug, runSetup = true, paseoHome, }) {
662
- // Validate branch name
663
- const validation = validateBranchSlug(branchName);
664
- if (!validation.valid) {
665
- throw new Error(`Invalid branch name: ${validation.error}`);
666
- }
667
- const normalizedBaseBranch = baseBranch ? normalizeBaseRefName(baseBranch) : "";
668
- if (!normalizedBaseBranch) {
669
- throw new Error("Base branch is required when creating a Paseo worktree");
670
- }
671
- if (normalizedBaseBranch === "HEAD") {
672
- throw new Error("Base branch cannot be HEAD when creating a Paseo worktree");
673
- }
674
- // Resolve the base branch - try local first, then remote
675
- let resolvedBaseBranch = normalizedBaseBranch;
676
- try {
677
- await execAsync(`git rev-parse --verify ${normalizedBaseBranch}`, { cwd });
678
- }
679
- catch {
680
- // Local branch doesn't exist, try remote (origin/{branch})
681
- try {
682
- await execAsync(`git rev-parse --verify origin/${normalizedBaseBranch}`, { cwd });
683
- resolvedBaseBranch = `origin/${normalizedBaseBranch}`;
684
- }
685
- catch {
686
- throw new Error(`Base branch not found: ${normalizedBaseBranch}`);
687
- }
688
- }
689
- let worktreePath;
690
- const desiredSlug = worktreeSlug || generateWorktreeSlug();
691
- worktreePath = join(await getPaseoWorktreesRoot(cwd, paseoHome), desiredSlug);
692
- mkdirSync(dirname(worktreePath), { recursive: true });
693
- // Check if branch already exists
694
- let branchExists = false;
695
- try {
696
- await execAsync(`git show-ref --verify --quiet refs/heads/${branchName}`, { cwd });
697
- branchExists = true;
698
- }
699
- catch {
700
- branchExists = false;
701
- }
702
- // Always create a new branch for the worktree
703
- // If branchName already exists, use it as base and create worktree-slug as branch name
704
- // If branchName doesn't exist, create it from baseBranch (resolved to remote if needed)
705
- const base = branchExists ? branchName : resolvedBaseBranch;
706
- const candidateBranch = branchExists ? desiredSlug : branchName;
707
- // Find unique branch name if collision
708
- let newBranchName = candidateBranch;
709
- let suffix = 1;
710
- while (true) {
711
- try {
712
- await execAsync(`git show-ref --verify --quiet refs/heads/${newBranchName}`, { cwd });
713
- // Branch exists, try with suffix
714
- newBranchName = `${candidateBranch}-${suffix}`;
715
- suffix++;
716
- }
717
- catch {
718
- break;
719
- }
720
- }
721
- // Also handle worktree path collision
722
- let finalWorktreePath = worktreePath;
723
- let pathSuffix = 1;
724
- while (existsSync(finalWorktreePath)) {
725
- finalWorktreePath = `${worktreePath}-${pathSuffix}`;
726
- pathSuffix++;
727
- }
728
- const command = `git worktree add "${finalWorktreePath}" -b "${newBranchName}" "${base}"`;
729
- await execAsync(command, { cwd });
730
- worktreePath = normalizePathForOwnership(finalWorktreePath);
731
- writePaseoWorktreeMetadata(worktreePath, { baseRefName: normalizedBaseBranch });
732
- if (runSetup) {
733
- await runWorktreeSetupCommands({
734
- worktreePath,
735
- branchName: newBranchName,
736
- cleanupOnFailure: true,
737
- });
738
- }
739
- return {
740
- branchName: newBranchName,
741
- worktreePath,
742
- };
743
- }
744
- //# sourceMappingURL=worktree.js.map