@getpaseo/server 0.1.16 → 0.1.18

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