@camaradesuk/git-worktree-tools 1.10.0 → 1.12.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (431) hide show
  1. package/README.md +4 -4
  2. package/dist/lib/config.d.ts +2 -1
  3. package/dist/lib/config.d.ts.map +1 -1
  4. package/dist/lib/config.js +11 -0
  5. package/dist/lib/config.js.map +1 -1
  6. package/package.json +12 -3
  7. package/schemas/worktreerc.schema.json +1 -1
  8. package/dist/api/list.test.d.ts +0 -5
  9. package/dist/api/list.test.d.ts.map +0 -1
  10. package/dist/api/list.test.js +0 -390
  11. package/dist/api/list.test.js.map +0 -1
  12. package/dist/cli/cleanpr.test.d.ts +0 -2
  13. package/dist/cli/cleanpr.test.d.ts.map +0 -1
  14. package/dist/cli/cleanpr.test.js +0 -954
  15. package/dist/cli/cleanpr.test.js.map +0 -1
  16. package/dist/cli/lswt.test.d.ts +0 -2
  17. package/dist/cli/lswt.test.d.ts.map +0 -1
  18. package/dist/cli/lswt.test.js +0 -376
  19. package/dist/cli/lswt.test.js.map +0 -1
  20. package/dist/cli/newpr.test.d.ts +0 -2
  21. package/dist/cli/newpr.test.d.ts.map +0 -1
  22. package/dist/cli/newpr.test.js +0 -1182
  23. package/dist/cli/newpr.test.js.map +0 -1
  24. package/dist/cli/prs.test.d.ts +0 -8
  25. package/dist/cli/prs.test.d.ts.map +0 -1
  26. package/dist/cli/prs.test.js +0 -463
  27. package/dist/cli/prs.test.js.map +0 -1
  28. package/dist/cli/wt/clean.test.d.ts +0 -8
  29. package/dist/cli/wt/clean.test.d.ts.map +0 -1
  30. package/dist/cli/wt/clean.test.js +0 -624
  31. package/dist/cli/wt/clean.test.js.map +0 -1
  32. package/dist/cli/wt/completion.test.d.ts +0 -5
  33. package/dist/cli/wt/completion.test.d.ts.map +0 -1
  34. package/dist/cli/wt/completion.test.js +0 -275
  35. package/dist/cli/wt/completion.test.js.map +0 -1
  36. package/dist/cli/wt/config.test.d.ts +0 -7
  37. package/dist/cli/wt/config.test.d.ts.map +0 -1
  38. package/dist/cli/wt/config.test.js +0 -440
  39. package/dist/cli/wt/config.test.js.map +0 -1
  40. package/dist/cli/wt/entry.test.d.ts +0 -8
  41. package/dist/cli/wt/entry.test.d.ts.map +0 -1
  42. package/dist/cli/wt/entry.test.js +0 -201
  43. package/dist/cli/wt/entry.test.js.map +0 -1
  44. package/dist/cli/wt/init.test.d.ts +0 -5
  45. package/dist/cli/wt/init.test.d.ts.map +0 -1
  46. package/dist/cli/wt/init.test.js +0 -165
  47. package/dist/cli/wt/init.test.js.map +0 -1
  48. package/dist/cli/wt/init.unit.test.d.ts +0 -5
  49. package/dist/cli/wt/init.unit.test.d.ts.map +0 -1
  50. package/dist/cli/wt/init.unit.test.js +0 -432
  51. package/dist/cli/wt/init.unit.test.js.map +0 -1
  52. package/dist/cli/wt/interactive-menu.test.d.ts +0 -12
  53. package/dist/cli/wt/interactive-menu.test.d.ts.map +0 -1
  54. package/dist/cli/wt/interactive-menu.test.js +0 -796
  55. package/dist/cli/wt/interactive-menu.test.js.map +0 -1
  56. package/dist/cli/wt/list.test.d.ts +0 -10
  57. package/dist/cli/wt/list.test.d.ts.map +0 -1
  58. package/dist/cli/wt/list.test.js +0 -157
  59. package/dist/cli/wt/list.test.js.map +0 -1
  60. package/dist/cli/wt/prs.test.d.ts +0 -5
  61. package/dist/cli/wt/prs.test.d.ts.map +0 -1
  62. package/dist/cli/wt/prs.test.js +0 -410
  63. package/dist/cli/wt/prs.test.js.map +0 -1
  64. package/dist/cli/wt/run-command.test.d.ts +0 -5
  65. package/dist/cli/wt/run-command.test.d.ts.map +0 -1
  66. package/dist/cli/wt/run-command.test.js +0 -88
  67. package/dist/cli/wt/run-command.test.js.map +0 -1
  68. package/dist/cli/wt/state.test.d.ts +0 -9
  69. package/dist/cli/wt/state.test.d.ts.map +0 -1
  70. package/dist/cli/wt/state.test.js +0 -127
  71. package/dist/cli/wt/state.test.js.map +0 -1
  72. package/dist/cli/wt/wt.test.d.ts +0 -8
  73. package/dist/cli/wt/wt.test.d.ts.map +0 -1
  74. package/dist/cli/wt/wt.test.js +0 -739
  75. package/dist/cli/wt/wt.test.js.map +0 -1
  76. package/dist/cli/wt.unit.test.d.ts +0 -7
  77. package/dist/cli/wt.unit.test.d.ts.map +0 -1
  78. package/dist/cli/wt.unit.test.js +0 -160
  79. package/dist/cli/wt.unit.test.js.map +0 -1
  80. package/dist/cli/wtconfig.test.d.ts +0 -5
  81. package/dist/cli/wtconfig.test.d.ts.map +0 -1
  82. package/dist/cli/wtconfig.test.js +0 -1289
  83. package/dist/cli/wtconfig.test.js.map +0 -1
  84. package/dist/cli/wtlink.test.d.ts +0 -2
  85. package/dist/cli/wtlink.test.d.ts.map +0 -1
  86. package/dist/cli/wtlink.test.js +0 -249
  87. package/dist/cli/wtlink.test.js.map +0 -1
  88. package/dist/cli/wtstate.test.d.ts +0 -5
  89. package/dist/cli/wtstate.test.d.ts.map +0 -1
  90. package/dist/cli/wtstate.test.js +0 -193
  91. package/dist/cli/wtstate.test.js.map +0 -1
  92. package/dist/e2e/cleanpr/cleanpr.e2e.test.d.ts +0 -2
  93. package/dist/e2e/cleanpr/cleanpr.e2e.test.d.ts.map +0 -1
  94. package/dist/e2e/cleanpr/cleanpr.e2e.test.js +0 -326
  95. package/dist/e2e/cleanpr/cleanpr.e2e.test.js.map +0 -1
  96. package/dist/e2e/cli.e2e.test.d.ts +0 -2
  97. package/dist/e2e/cli.e2e.test.d.ts.map +0 -1
  98. package/dist/e2e/cli.e2e.test.js +0 -417
  99. package/dist/e2e/cli.e2e.test.js.map +0 -1
  100. package/dist/e2e/lswt/lswt.e2e.test.d.ts +0 -2
  101. package/dist/e2e/lswt/lswt.e2e.test.d.ts.map +0 -1
  102. package/dist/e2e/lswt/lswt.e2e.test.js +0 -361
  103. package/dist/e2e/lswt/lswt.e2e.test.js.map +0 -1
  104. package/dist/e2e/newpr/newpr.e2e.test.d.ts +0 -2
  105. package/dist/e2e/newpr/newpr.e2e.test.d.ts.map +0 -1
  106. package/dist/e2e/newpr/newpr.e2e.test.js +0 -286
  107. package/dist/e2e/newpr/newpr.e2e.test.js.map +0 -1
  108. package/dist/e2e/newpr/scenarios.e2e.test.d.ts +0 -2
  109. package/dist/e2e/newpr/scenarios.e2e.test.d.ts.map +0 -1
  110. package/dist/e2e/newpr/scenarios.e2e.test.js +0 -426
  111. package/dist/e2e/newpr/scenarios.e2e.test.js.map +0 -1
  112. package/dist/e2e/newpr-full-flow.e2e.test.d.ts +0 -2
  113. package/dist/e2e/newpr-full-flow.e2e.test.d.ts.map +0 -1
  114. package/dist/e2e/newpr-full-flow.e2e.test.js +0 -280
  115. package/dist/e2e/newpr-full-flow.e2e.test.js.map +0 -1
  116. package/dist/e2e/prs/prs.e2e.test.d.ts +0 -7
  117. package/dist/e2e/prs/prs.e2e.test.d.ts.map +0 -1
  118. package/dist/e2e/prs/prs.e2e.test.js +0 -606
  119. package/dist/e2e/prs/prs.e2e.test.js.map +0 -1
  120. package/dist/e2e/workflows/pr-lifecycle.e2e.test.d.ts +0 -2
  121. package/dist/e2e/workflows/pr-lifecycle.e2e.test.d.ts.map +0 -1
  122. package/dist/e2e/workflows/pr-lifecycle.e2e.test.js +0 -298
  123. package/dist/e2e/workflows/pr-lifecycle.e2e.test.js.map +0 -1
  124. package/dist/e2e/wt/interactive-menu.e2e.test.d.ts +0 -8
  125. package/dist/e2e/wt/interactive-menu.e2e.test.d.ts.map +0 -1
  126. package/dist/e2e/wt/interactive-menu.e2e.test.js +0 -583
  127. package/dist/e2e/wt/interactive-menu.e2e.test.js.map +0 -1
  128. package/dist/e2e/wt/wt.e2e.test.d.ts +0 -9
  129. package/dist/e2e/wt/wt.e2e.test.d.ts.map +0 -1
  130. package/dist/e2e/wt/wt.e2e.test.js +0 -597
  131. package/dist/e2e/wt/wt.e2e.test.js.map +0 -1
  132. package/dist/e2e/wtlink/wtlink.e2e.test.d.ts +0 -2
  133. package/dist/e2e/wtlink/wtlink.e2e.test.d.ts.map +0 -1
  134. package/dist/e2e/wtlink/wtlink.e2e.test.js +0 -416
  135. package/dist/e2e/wtlink/wtlink.e2e.test.js.map +0 -1
  136. package/dist/integration/git.integration.test.d.ts +0 -2
  137. package/dist/integration/git.integration.test.d.ts.map +0 -1
  138. package/dist/integration/git.integration.test.js +0 -336
  139. package/dist/integration/git.integration.test.js.map +0 -1
  140. package/dist/integration/lswt-remote-pr.integration.test.d.ts +0 -2
  141. package/dist/integration/lswt-remote-pr.integration.test.d.ts.map +0 -1
  142. package/dist/integration/lswt-remote-pr.integration.test.js +0 -222
  143. package/dist/integration/lswt-remote-pr.integration.test.js.map +0 -1
  144. package/dist/integration/newpr-branchfrom-head.integration.test.d.ts +0 -2
  145. package/dist/integration/newpr-branchfrom-head.integration.test.d.ts.map +0 -1
  146. package/dist/integration/newpr-branchfrom-head.integration.test.js +0 -498
  147. package/dist/integration/newpr-branchfrom-head.integration.test.js.map +0 -1
  148. package/dist/integration/newpr.integration.test.d.ts +0 -2
  149. package/dist/integration/newpr.integration.test.d.ts.map +0 -1
  150. package/dist/integration/newpr.integration.test.js +0 -460
  151. package/dist/integration/newpr.integration.test.js.map +0 -1
  152. package/dist/integration/prs.integration.test.d.ts +0 -8
  153. package/dist/integration/prs.integration.test.d.ts.map +0 -1
  154. package/dist/integration/prs.integration.test.js +0 -478
  155. package/dist/integration/prs.integration.test.js.map +0 -1
  156. package/dist/lib/ai/base-provider.test.d.ts +0 -7
  157. package/dist/lib/ai/base-provider.test.d.ts.map +0 -1
  158. package/dist/lib/ai/base-provider.test.js +0 -319
  159. package/dist/lib/ai/base-provider.test.js.map +0 -1
  160. package/dist/lib/ai/cli-provider.test.d.ts +0 -5
  161. package/dist/lib/ai/cli-provider.test.d.ts.map +0 -1
  162. package/dist/lib/ai/cli-provider.test.js +0 -460
  163. package/dist/lib/ai/cli-provider.test.js.map +0 -1
  164. package/dist/lib/ai/fallback-provider.test.d.ts +0 -7
  165. package/dist/lib/ai/fallback-provider.test.d.ts.map +0 -1
  166. package/dist/lib/ai/fallback-provider.test.js +0 -165
  167. package/dist/lib/ai/fallback-provider.test.js.map +0 -1
  168. package/dist/lib/ai/generation-service.test.d.ts +0 -7
  169. package/dist/lib/ai/generation-service.test.d.ts.map +0 -1
  170. package/dist/lib/ai/generation-service.test.js +0 -213
  171. package/dist/lib/ai/generation-service.test.js.map +0 -1
  172. package/dist/lib/ai/provider-manager.test.d.ts +0 -5
  173. package/dist/lib/ai/provider-manager.test.d.ts.map +0 -1
  174. package/dist/lib/ai/provider-manager.test.js +0 -312
  175. package/dist/lib/ai/provider-manager.test.js.map +0 -1
  176. package/dist/lib/ai/repo-docs.test.d.ts +0 -5
  177. package/dist/lib/ai/repo-docs.test.d.ts.map +0 -1
  178. package/dist/lib/ai/repo-docs.test.js +0 -357
  179. package/dist/lib/ai/repo-docs.test.js.map +0 -1
  180. package/dist/lib/cleanpr/args.test.d.ts +0 -2
  181. package/dist/lib/cleanpr/args.test.d.ts.map +0 -1
  182. package/dist/lib/cleanpr/args.test.js +0 -269
  183. package/dist/lib/cleanpr/args.test.js.map +0 -1
  184. package/dist/lib/cleanpr/cleanup.test.d.ts +0 -2
  185. package/dist/lib/cleanpr/cleanup.test.d.ts.map +0 -1
  186. package/dist/lib/cleanpr/cleanup.test.js +0 -296
  187. package/dist/lib/cleanpr/cleanup.test.js.map +0 -1
  188. package/dist/lib/cleanpr/worktree-info.test.d.ts +0 -2
  189. package/dist/lib/cleanpr/worktree-info.test.d.ts.map +0 -1
  190. package/dist/lib/cleanpr/worktree-info.test.js +0 -228
  191. package/dist/lib/cleanpr/worktree-info.test.js.map +0 -1
  192. package/dist/lib/colors.test.d.ts +0 -2
  193. package/dist/lib/colors.test.d.ts.map +0 -1
  194. package/dist/lib/colors.test.js +0 -142
  195. package/dist/lib/colors.test.js.map +0 -1
  196. package/dist/lib/config-editor.test.d.ts +0 -11
  197. package/dist/lib/config-editor.test.d.ts.map +0 -1
  198. package/dist/lib/config-editor.test.js +0 -526
  199. package/dist/lib/config-editor.test.js.map +0 -1
  200. package/dist/lib/config-migration/detector.test.d.ts +0 -5
  201. package/dist/lib/config-migration/detector.test.d.ts.map +0 -1
  202. package/dist/lib/config-migration/detector.test.js +0 -201
  203. package/dist/lib/config-migration/detector.test.js.map +0 -1
  204. package/dist/lib/config-migration/reporter.test.d.ts +0 -5
  205. package/dist/lib/config-migration/reporter.test.d.ts.map +0 -1
  206. package/dist/lib/config-migration/reporter.test.js +0 -305
  207. package/dist/lib/config-migration/reporter.test.js.map +0 -1
  208. package/dist/lib/config-migration/runner.test.d.ts +0 -5
  209. package/dist/lib/config-migration/runner.test.d.ts.map +0 -1
  210. package/dist/lib/config-migration/runner.test.js +0 -235
  211. package/dist/lib/config-migration/runner.test.js.map +0 -1
  212. package/dist/lib/config-validation.test.d.ts +0 -5
  213. package/dist/lib/config-validation.test.d.ts.map +0 -1
  214. package/dist/lib/config-validation.test.js +0 -423
  215. package/dist/lib/config-validation.test.js.map +0 -1
  216. package/dist/lib/config.test.d.ts +0 -2
  217. package/dist/lib/config.test.d.ts.map +0 -1
  218. package/dist/lib/config.test.js +0 -554
  219. package/dist/lib/config.test.js.map +0 -1
  220. package/dist/lib/constants.test.d.ts +0 -5
  221. package/dist/lib/constants.test.d.ts.map +0 -1
  222. package/dist/lib/constants.test.js +0 -180
  223. package/dist/lib/constants.test.js.map +0 -1
  224. package/dist/lib/deprecation.test.d.ts +0 -2
  225. package/dist/lib/deprecation.test.d.ts.map +0 -1
  226. package/dist/lib/deprecation.test.js +0 -71
  227. package/dist/lib/deprecation.test.js.map +0 -1
  228. package/dist/lib/errors.test.d.ts +0 -2
  229. package/dist/lib/errors.test.d.ts.map +0 -1
  230. package/dist/lib/errors.test.js +0 -117
  231. package/dist/lib/errors.test.js.map +0 -1
  232. package/dist/lib/git.test.d.ts +0 -2
  233. package/dist/lib/git.test.d.ts.map +0 -1
  234. package/dist/lib/git.test.js +0 -608
  235. package/dist/lib/git.test.js.map +0 -1
  236. package/dist/lib/github.test.d.ts +0 -2
  237. package/dist/lib/github.test.d.ts.map +0 -1
  238. package/dist/lib/github.test.js +0 -441
  239. package/dist/lib/github.test.js.map +0 -1
  240. package/dist/lib/global-check.test.d.ts +0 -5
  241. package/dist/lib/global-check.test.d.ts.map +0 -1
  242. package/dist/lib/global-check.test.js +0 -150
  243. package/dist/lib/global-check.test.js.map +0 -1
  244. package/dist/lib/global-config.test.d.ts +0 -5
  245. package/dist/lib/global-config.test.d.ts.map +0 -1
  246. package/dist/lib/global-config.test.js +0 -282
  247. package/dist/lib/global-config.test.js.map +0 -1
  248. package/dist/lib/hooks/confirmation.test.d.ts +0 -7
  249. package/dist/lib/hooks/confirmation.test.d.ts.map +0 -1
  250. package/dist/lib/hooks/confirmation.test.js +0 -300
  251. package/dist/lib/hooks/confirmation.test.js.map +0 -1
  252. package/dist/lib/hooks/executor.test.d.ts +0 -5
  253. package/dist/lib/hooks/executor.test.d.ts.map +0 -1
  254. package/dist/lib/hooks/executor.test.js +0 -648
  255. package/dist/lib/hooks/executor.test.js.map +0 -1
  256. package/dist/lib/hooks/templates.test.d.ts +0 -5
  257. package/dist/lib/hooks/templates.test.d.ts.map +0 -1
  258. package/dist/lib/hooks/templates.test.js +0 -163
  259. package/dist/lib/hooks/templates.test.js.map +0 -1
  260. package/dist/lib/hooks/types.test.d.ts +0 -5
  261. package/dist/lib/hooks/types.test.d.ts.map +0 -1
  262. package/dist/lib/hooks/types.test.js +0 -132
  263. package/dist/lib/hooks/types.test.js.map +0 -1
  264. package/dist/lib/json-output.test.d.ts +0 -5
  265. package/dist/lib/json-output.test.d.ts.map +0 -1
  266. package/dist/lib/json-output.test.js +0 -261
  267. package/dist/lib/json-output.test.js.map +0 -1
  268. package/dist/lib/logger.test.d.ts +0 -14
  269. package/dist/lib/logger.test.d.ts.map +0 -1
  270. package/dist/lib/logger.test.js +0 -692
  271. package/dist/lib/logger.test.js.map +0 -1
  272. package/dist/lib/lswt/action-executors.test.d.ts +0 -2
  273. package/dist/lib/lswt/action-executors.test.d.ts.map +0 -1
  274. package/dist/lib/lswt/action-executors.test.js +0 -1127
  275. package/dist/lib/lswt/action-executors.test.js.map +0 -1
  276. package/dist/lib/lswt/actions.test.d.ts +0 -2
  277. package/dist/lib/lswt/actions.test.d.ts.map +0 -1
  278. package/dist/lib/lswt/actions.test.js +0 -497
  279. package/dist/lib/lswt/actions.test.js.map +0 -1
  280. package/dist/lib/lswt/args.test.d.ts +0 -2
  281. package/dist/lib/lswt/args.test.d.ts.map +0 -1
  282. package/dist/lib/lswt/args.test.js +0 -195
  283. package/dist/lib/lswt/args.test.js.map +0 -1
  284. package/dist/lib/lswt/environment.test.d.ts +0 -2
  285. package/dist/lib/lswt/environment.test.d.ts.map +0 -1
  286. package/dist/lib/lswt/environment.test.js +0 -544
  287. package/dist/lib/lswt/environment.test.js.map +0 -1
  288. package/dist/lib/lswt/formatters.test.d.ts +0 -2
  289. package/dist/lib/lswt/formatters.test.d.ts.map +0 -1
  290. package/dist/lib/lswt/formatters.test.js +0 -323
  291. package/dist/lib/lswt/formatters.test.js.map +0 -1
  292. package/dist/lib/lswt/fuzzy-search.test.d.ts +0 -5
  293. package/dist/lib/lswt/fuzzy-search.test.d.ts.map +0 -1
  294. package/dist/lib/lswt/fuzzy-search.test.js +0 -207
  295. package/dist/lib/lswt/fuzzy-search.test.js.map +0 -1
  296. package/dist/lib/lswt/interactive.test.d.ts +0 -2
  297. package/dist/lib/lswt/interactive.test.d.ts.map +0 -1
  298. package/dist/lib/lswt/interactive.test.js +0 -771
  299. package/dist/lib/lswt/interactive.test.js.map +0 -1
  300. package/dist/lib/lswt/table.test.d.ts +0 -5
  301. package/dist/lib/lswt/table.test.d.ts.map +0 -1
  302. package/dist/lib/lswt/table.test.js +0 -262
  303. package/dist/lib/lswt/table.test.js.map +0 -1
  304. package/dist/lib/lswt/worktree-info.test.d.ts +0 -2
  305. package/dist/lib/lswt/worktree-info.test.d.ts.map +0 -1
  306. package/dist/lib/lswt/worktree-info.test.js +0 -484
  307. package/dist/lib/lswt/worktree-info.test.js.map +0 -1
  308. package/dist/lib/newpr/action-deps.test.d.ts +0 -5
  309. package/dist/lib/newpr/action-deps.test.d.ts.map +0 -1
  310. package/dist/lib/newpr/action-deps.test.js +0 -111
  311. package/dist/lib/newpr/action-deps.test.js.map +0 -1
  312. package/dist/lib/newpr/actions.test.d.ts +0 -2
  313. package/dist/lib/newpr/actions.test.d.ts.map +0 -1
  314. package/dist/lib/newpr/actions.test.js +0 -254
  315. package/dist/lib/newpr/actions.test.js.map +0 -1
  316. package/dist/lib/newpr/args.test.d.ts +0 -2
  317. package/dist/lib/newpr/args.test.d.ts.map +0 -1
  318. package/dist/lib/newpr/args.test.js +0 -479
  319. package/dist/lib/newpr/args.test.js.map +0 -1
  320. package/dist/lib/newpr/hook-runner.test.d.ts +0 -7
  321. package/dist/lib/newpr/hook-runner.test.d.ts.map +0 -1
  322. package/dist/lib/newpr/hook-runner.test.js +0 -422
  323. package/dist/lib/newpr/hook-runner.test.js.map +0 -1
  324. package/dist/lib/newpr/plan-generator.test.d.ts +0 -7
  325. package/dist/lib/newpr/plan-generator.test.d.ts.map +0 -1
  326. package/dist/lib/newpr/plan-generator.test.js +0 -387
  327. package/dist/lib/newpr/plan-generator.test.js.map +0 -1
  328. package/dist/lib/newpr/scenario-handler.test.d.ts +0 -2
  329. package/dist/lib/newpr/scenario-handler.test.d.ts.map +0 -1
  330. package/dist/lib/newpr/scenario-handler.test.js +0 -256
  331. package/dist/lib/newpr/scenario-handler.test.js.map +0 -1
  332. package/dist/lib/prompts.test.d.ts +0 -2
  333. package/dist/lib/prompts.test.d.ts.map +0 -1
  334. package/dist/lib/prompts.test.js +0 -807
  335. package/dist/lib/prompts.test.js.map +0 -1
  336. package/dist/lib/prs/actions.test.d.ts +0 -5
  337. package/dist/lib/prs/actions.test.d.ts.map +0 -1
  338. package/dist/lib/prs/actions.test.js +0 -356
  339. package/dist/lib/prs/actions.test.js.map +0 -1
  340. package/dist/lib/prs/command.test.d.ts +0 -11
  341. package/dist/lib/prs/command.test.d.ts.map +0 -1
  342. package/dist/lib/prs/command.test.js +0 -409
  343. package/dist/lib/prs/command.test.js.map +0 -1
  344. package/dist/lib/prs/data.test.d.ts +0 -5
  345. package/dist/lib/prs/data.test.d.ts.map +0 -1
  346. package/dist/lib/prs/data.test.js +0 -417
  347. package/dist/lib/prs/data.test.js.map +0 -1
  348. package/dist/lib/prs/details.test.d.ts +0 -5
  349. package/dist/lib/prs/details.test.d.ts.map +0 -1
  350. package/dist/lib/prs/details.test.js +0 -325
  351. package/dist/lib/prs/details.test.js.map +0 -1
  352. package/dist/lib/prs/filters.test.d.ts +0 -5
  353. package/dist/lib/prs/filters.test.d.ts.map +0 -1
  354. package/dist/lib/prs/filters.test.js +0 -312
  355. package/dist/lib/prs/filters.test.js.map +0 -1
  356. package/dist/lib/prs/formatters.test.d.ts +0 -2
  357. package/dist/lib/prs/formatters.test.d.ts.map +0 -1
  358. package/dist/lib/prs/formatters.test.js +0 -387
  359. package/dist/lib/prs/formatters.test.js.map +0 -1
  360. package/dist/lib/prs/interactive.test.d.ts +0 -5
  361. package/dist/lib/prs/interactive.test.d.ts.map +0 -1
  362. package/dist/lib/prs/interactive.test.js +0 -517
  363. package/dist/lib/prs/interactive.test.js.map +0 -1
  364. package/dist/lib/schema.test.d.ts +0 -10
  365. package/dist/lib/schema.test.d.ts.map +0 -1
  366. package/dist/lib/schema.test.js +0 -309
  367. package/dist/lib/schema.test.js.map +0 -1
  368. package/dist/lib/state-detection.test.d.ts +0 -2
  369. package/dist/lib/state-detection.test.d.ts.map +0 -1
  370. package/dist/lib/state-detection.test.js +0 -451
  371. package/dist/lib/state-detection.test.js.map +0 -1
  372. package/dist/lib/ui/error.test.d.ts +0 -2
  373. package/dist/lib/ui/error.test.d.ts.map +0 -1
  374. package/dist/lib/ui/error.test.js +0 -143
  375. package/dist/lib/ui/error.test.js.map +0 -1
  376. package/dist/lib/ui/output.test.d.ts +0 -2
  377. package/dist/lib/ui/output.test.d.ts.map +0 -1
  378. package/dist/lib/ui/output.test.js +0 -59
  379. package/dist/lib/ui/output.test.js.map +0 -1
  380. package/dist/lib/ui/status.test.d.ts +0 -2
  381. package/dist/lib/ui/status.test.d.ts.map +0 -1
  382. package/dist/lib/ui/status.test.js +0 -158
  383. package/dist/lib/ui/status.test.js.map +0 -1
  384. package/dist/lib/ui/table.test.d.ts +0 -2
  385. package/dist/lib/ui/table.test.d.ts.map +0 -1
  386. package/dist/lib/ui/table.test.js +0 -115
  387. package/dist/lib/ui/table.test.js.map +0 -1
  388. package/dist/lib/ui/theme.test.d.ts +0 -2
  389. package/dist/lib/ui/theme.test.d.ts.map +0 -1
  390. package/dist/lib/ui/theme.test.js +0 -76
  391. package/dist/lib/ui/theme.test.js.map +0 -1
  392. package/dist/lib/wtconfig/config-manager.test.d.ts +0 -5
  393. package/dist/lib/wtconfig/config-manager.test.d.ts.map +0 -1
  394. package/dist/lib/wtconfig/config-manager.test.js +0 -501
  395. package/dist/lib/wtconfig/config-manager.test.js.map +0 -1
  396. package/dist/lib/wtconfig/environment.test.d.ts +0 -5
  397. package/dist/lib/wtconfig/environment.test.d.ts.map +0 -1
  398. package/dist/lib/wtconfig/environment.test.js +0 -285
  399. package/dist/lib/wtconfig/environment.test.js.map +0 -1
  400. package/dist/lib/wtlink/config-manifest.test.d.ts +0 -2
  401. package/dist/lib/wtlink/config-manifest.test.d.ts.map +0 -1
  402. package/dist/lib/wtlink/config-manifest.test.js +0 -486
  403. package/dist/lib/wtlink/config-manifest.test.js.map +0 -1
  404. package/dist/lib/wtlink/link-configs.test.d.ts +0 -2
  405. package/dist/lib/wtlink/link-configs.test.d.ts.map +0 -1
  406. package/dist/lib/wtlink/link-configs.test.js +0 -612
  407. package/dist/lib/wtlink/link-configs.test.js.map +0 -1
  408. package/dist/lib/wtlink/main-menu.test.d.ts +0 -5
  409. package/dist/lib/wtlink/main-menu.test.d.ts.map +0 -1
  410. package/dist/lib/wtlink/main-menu.test.js +0 -126
  411. package/dist/lib/wtlink/main-menu.test.js.map +0 -1
  412. package/dist/lib/wtlink/manage-manifest.test.d.ts +0 -2
  413. package/dist/lib/wtlink/manage-manifest.test.d.ts.map +0 -1
  414. package/dist/lib/wtlink/manage-manifest.test.js +0 -714
  415. package/dist/lib/wtlink/manage-manifest.test.js.map +0 -1
  416. package/dist/lib/wtlink/validate-manifest.test.d.ts +0 -2
  417. package/dist/lib/wtlink/validate-manifest.test.d.ts.map +0 -1
  418. package/dist/lib/wtlink/validate-manifest.test.js +0 -220
  419. package/dist/lib/wtlink/validate-manifest.test.js.map +0 -1
  420. package/dist/lib/wtstate/analyze.test.d.ts +0 -5
  421. package/dist/lib/wtstate/analyze.test.d.ts.map +0 -1
  422. package/dist/lib/wtstate/analyze.test.js +0 -282
  423. package/dist/lib/wtstate/analyze.test.js.map +0 -1
  424. package/dist/lib/wtstate/args.test.d.ts +0 -5
  425. package/dist/lib/wtstate/args.test.d.ts.map +0 -1
  426. package/dist/lib/wtstate/args.test.js +0 -120
  427. package/dist/lib/wtstate/args.test.js.map +0 -1
  428. package/dist/mcp/server.test.d.ts +0 -9
  429. package/dist/mcp/server.test.d.ts.map +0 -1
  430. package/dist/mcp/server.test.js +0 -550
  431. package/dist/mcp/server.test.js.map +0 -1
@@ -1,954 +0,0 @@
1
- import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
2
- // Mock modules before importing the CLI
3
- vi.mock('../lib/git.js', () => ({
4
- getRepoRoot: vi.fn(),
5
- removeWorktree: vi.fn(),
6
- pruneWorktrees: vi.fn(),
7
- }));
8
- vi.mock('../lib/github.js', () => ({
9
- isGhInstalled: vi.fn(),
10
- }));
11
- vi.mock('../lib/prompts.js', () => ({
12
- promptChoice: vi.fn(),
13
- promptConfirm: vi.fn(),
14
- withSpinner: vi.fn(async (message, fn) => fn()),
15
- }));
16
- vi.mock('../lib/config.js', () => ({
17
- loadConfig: vi.fn(),
18
- }));
19
- vi.mock('../lib/cleanpr/index.js', () => ({
20
- parseArgs: vi.fn(),
21
- getHelpText: vi.fn(),
22
- gatherPrWorktreeInfo: vi.fn(),
23
- createDefaultDeps: vi.fn(),
24
- groupWorktreesByState: vi.fn(),
25
- getCleanableWorktrees: vi.fn(),
26
- findWorktreeByPrNumber: vi.fn(),
27
- cleanWorktree: vi.fn(),
28
- summarizeResults: vi.fn(),
29
- }));
30
- // Import after mocking
31
- import * as git from '../lib/git.js';
32
- import * as github from '../lib/github.js';
33
- import * as prompts from '../lib/prompts.js';
34
- import { loadConfig } from '../lib/config.js';
35
- import * as cleanpr from '../lib/cleanpr/index.js';
36
- describe('cli/cleanpr', () => {
37
- let mockConsoleLog;
38
- let mockConsoleError;
39
- let mockProcessExit;
40
- let originalArgv;
41
- const defaultConfig = {
42
- configVersion: 1,
43
- baseBranch: 'main',
44
- worktreePattern: '{repo}.pr{number}',
45
- worktreeParent: '..',
46
- draftPr: false,
47
- sharedRepos: [],
48
- branchPrefix: 'feature',
49
- previewLabel: 'preview',
50
- syncPatterns: [],
51
- preferredEditor: 'auto',
52
- ai: { provider: 'none' },
53
- hooks: {},
54
- hookDefaults: { timeout: 30000, maxTimeout: 60000 },
55
- plugins: [],
56
- generators: {},
57
- integrations: {},
58
- logging: { level: 'info', timestamps: true },
59
- global: { warnNotGlobal: true },
60
- wtlink: { enabled: [], disabled: [] },
61
- linkConfigFiles: false,
62
- };
63
- const makeWorktreeInfo = (overrides = {}) => ({
64
- path: '/repo.pr123',
65
- branch: 'feature-1',
66
- commit: 'abc123',
67
- prNumber: 123,
68
- prState: 'MERGED',
69
- hasChanges: false,
70
- ...overrides,
71
- });
72
- beforeEach(() => {
73
- vi.resetAllMocks();
74
- // Reset withSpinner mock (resetAllMocks clears the implementation)
75
- vi.mocked(prompts.withSpinner).mockImplementation(async (message, fn) => {
76
- return await fn();
77
- });
78
- mockConsoleLog = vi.spyOn(console, 'log').mockImplementation(() => { });
79
- mockConsoleError = vi.spyOn(console, 'error').mockImplementation(() => { });
80
- // @ts-expect-error - process.exit mock type is complex
81
- mockProcessExit = vi.spyOn(process, 'exit').mockImplementation((() => { }));
82
- originalArgv = process.argv;
83
- });
84
- afterEach(() => {
85
- mockConsoleLog.mockRestore();
86
- mockConsoleError.mockRestore();
87
- mockProcessExit.mockRestore();
88
- process.argv = originalArgv;
89
- vi.resetModules();
90
- });
91
- async function runCli(args = []) {
92
- process.argv = ['node', 'cleanpr', ...args];
93
- await import('./cleanpr.js');
94
- // Allow time for all async operations to complete
95
- await new Promise((resolve) => setTimeout(resolve, 100));
96
- }
97
- describe('help option', () => {
98
- it('prints help and exits 0 on --help', async () => {
99
- vi.mocked(cleanpr.parseArgs).mockReturnValue({ kind: 'help' });
100
- vi.mocked(cleanpr.getHelpText).mockReturnValue('Usage: cleanpr [options]');
101
- await runCli(['--help']);
102
- expect(cleanpr.getHelpText).toHaveBeenCalled();
103
- expect(mockConsoleLog).toHaveBeenCalledWith('Usage: cleanpr [options]');
104
- expect(mockProcessExit).toHaveBeenCalledWith(0);
105
- });
106
- });
107
- describe('error handling', () => {
108
- it('prints error and exits 1 on parse error', async () => {
109
- vi.mocked(cleanpr.parseArgs).mockReturnValue({
110
- kind: 'error',
111
- message: 'Invalid option: --invalid',
112
- });
113
- await runCli(['--invalid']);
114
- expect(mockConsoleError).toHaveBeenCalled();
115
- expect(mockProcessExit).toHaveBeenCalledWith(1);
116
- });
117
- it('exits 1 when gh not installed', async () => {
118
- vi.mocked(cleanpr.parseArgs).mockReturnValue({
119
- kind: 'success',
120
- prNumber: null,
121
- options: {
122
- all: false,
123
- force: false,
124
- deleteRemote: false,
125
- interactive: true,
126
- json: false,
127
- dryRun: false,
128
- },
129
- });
130
- vi.mocked(github.isGhInstalled).mockReturnValue(false);
131
- await runCli([]);
132
- expect(mockConsoleError).toHaveBeenCalledWith(expect.stringContaining('GitHub CLI'));
133
- expect(mockProcessExit).toHaveBeenCalledWith(1);
134
- });
135
- it('exits 1 when not in git repo', async () => {
136
- vi.mocked(cleanpr.parseArgs).mockReturnValue({
137
- kind: 'success',
138
- prNumber: null,
139
- options: {
140
- all: false,
141
- force: false,
142
- deleteRemote: false,
143
- interactive: true,
144
- json: false,
145
- dryRun: false,
146
- },
147
- });
148
- vi.mocked(github.isGhInstalled).mockReturnValue(true);
149
- vi.mocked(git.getRepoRoot).mockReturnValue('');
150
- await runCli([]);
151
- expect(mockConsoleError).toHaveBeenCalled();
152
- expect(mockProcessExit).toHaveBeenCalledWith(1);
153
- });
154
- });
155
- describe('interactive mode', () => {
156
- it('shows info message when no PR worktrees found', async () => {
157
- vi.mocked(cleanpr.parseArgs).mockReturnValue({
158
- kind: 'success',
159
- prNumber: null,
160
- options: {
161
- all: false,
162
- force: false,
163
- deleteRemote: false,
164
- interactive: true,
165
- json: false,
166
- dryRun: false,
167
- },
168
- });
169
- vi.mocked(github.isGhInstalled).mockReturnValue(true);
170
- vi.mocked(git.getRepoRoot).mockReturnValue('/repo');
171
- vi.mocked(loadConfig).mockReturnValue(defaultConfig);
172
- vi.mocked(cleanpr.createDefaultDeps).mockReturnValue({});
173
- vi.mocked(cleanpr.gatherPrWorktreeInfo).mockResolvedValue([]);
174
- vi.mocked(cleanpr.groupWorktreesByState).mockReturnValue({
175
- merged: [],
176
- closed: [],
177
- open: [],
178
- unknown: [],
179
- });
180
- vi.mocked(cleanpr.getCleanableWorktrees).mockReturnValue([]);
181
- await runCli([]);
182
- expect(mockConsoleLog).toHaveBeenCalledWith(expect.stringContaining('No PR worktrees'));
183
- });
184
- it('displays grouped worktrees in interactive mode', async () => {
185
- const mockWorktrees = [
186
- makeWorktreeInfo({ prNumber: 1, prState: 'MERGED' }),
187
- makeWorktreeInfo({ prNumber: 2, prState: 'CLOSED' }),
188
- makeWorktreeInfo({ prNumber: 3, prState: 'OPEN' }),
189
- ];
190
- vi.mocked(cleanpr.parseArgs).mockReturnValue({
191
- kind: 'success',
192
- prNumber: null,
193
- options: {
194
- all: false,
195
- force: false,
196
- deleteRemote: false,
197
- interactive: true,
198
- json: false,
199
- dryRun: false,
200
- },
201
- });
202
- vi.mocked(github.isGhInstalled).mockReturnValue(true);
203
- vi.mocked(git.getRepoRoot).mockReturnValue('/repo');
204
- vi.mocked(loadConfig).mockReturnValue(defaultConfig);
205
- vi.mocked(cleanpr.createDefaultDeps).mockReturnValue({});
206
- vi.mocked(cleanpr.gatherPrWorktreeInfo).mockResolvedValue(mockWorktrees);
207
- vi.mocked(cleanpr.groupWorktreesByState).mockReturnValue({
208
- merged: [mockWorktrees[0]],
209
- closed: [mockWorktrees[1]],
210
- open: [mockWorktrees[2]],
211
- unknown: [],
212
- });
213
- vi.mocked(cleanpr.getCleanableWorktrees).mockReturnValue([]);
214
- await runCli([]);
215
- expect(cleanpr.groupWorktreesByState).toHaveBeenCalledWith(mockWorktrees);
216
- expect(mockConsoleLog).toHaveBeenCalledWith(expect.stringContaining('PR Worktrees'));
217
- });
218
- it('shows no cleanable message when only open PRs exist', async () => {
219
- const mockWorktrees = [makeWorktreeInfo({ prNumber: 1, prState: 'OPEN' })];
220
- vi.mocked(cleanpr.parseArgs).mockReturnValue({
221
- kind: 'success',
222
- prNumber: null,
223
- options: {
224
- all: false,
225
- force: false,
226
- deleteRemote: false,
227
- interactive: true,
228
- json: false,
229
- dryRun: false,
230
- },
231
- });
232
- vi.mocked(github.isGhInstalled).mockReturnValue(true);
233
- vi.mocked(git.getRepoRoot).mockReturnValue('/repo');
234
- vi.mocked(loadConfig).mockReturnValue(defaultConfig);
235
- vi.mocked(cleanpr.createDefaultDeps).mockReturnValue({});
236
- vi.mocked(cleanpr.gatherPrWorktreeInfo).mockResolvedValue(mockWorktrees);
237
- vi.mocked(cleanpr.groupWorktreesByState).mockReturnValue({
238
- merged: [],
239
- closed: [],
240
- open: mockWorktrees,
241
- unknown: [],
242
- });
243
- vi.mocked(cleanpr.getCleanableWorktrees).mockReturnValue([]);
244
- await runCli([]);
245
- expect(mockConsoleLog).toHaveBeenCalledWith(expect.stringContaining('No merged or closed PRs'));
246
- });
247
- it('handles user cancel in interactive mode', async () => {
248
- const mockWorktrees = [makeWorktreeInfo({ prNumber: 1, prState: 'MERGED' })];
249
- const { promptChoice } = await import('../lib/prompts.js');
250
- vi.mocked(cleanpr.parseArgs).mockReturnValue({
251
- kind: 'success',
252
- prNumber: null,
253
- options: {
254
- all: false,
255
- force: false,
256
- deleteRemote: false,
257
- interactive: true,
258
- json: false,
259
- dryRun: false,
260
- },
261
- });
262
- vi.mocked(github.isGhInstalled).mockReturnValue(true);
263
- vi.mocked(git.getRepoRoot).mockReturnValue('/repo');
264
- vi.mocked(loadConfig).mockReturnValue(defaultConfig);
265
- vi.mocked(cleanpr.createDefaultDeps).mockReturnValue({});
266
- vi.mocked(cleanpr.gatherPrWorktreeInfo).mockResolvedValue(mockWorktrees);
267
- vi.mocked(cleanpr.groupWorktreesByState).mockReturnValue({
268
- merged: mockWorktrees,
269
- closed: [],
270
- open: [],
271
- unknown: [],
272
- });
273
- vi.mocked(cleanpr.getCleanableWorktrees).mockReturnValue(mockWorktrees);
274
- vi.mocked(promptChoice).mockResolvedValue('cancel');
275
- await runCli([]);
276
- expect(mockConsoleLog).toHaveBeenCalledWith(expect.stringContaining('Cancelled'));
277
- });
278
- it('handles all option in interactive mode', async () => {
279
- const mockWorktrees = [makeWorktreeInfo({ prNumber: 1, prState: 'MERGED' })];
280
- const { promptChoice, promptConfirm } = await import('../lib/prompts.js');
281
- vi.mocked(cleanpr.parseArgs).mockReturnValue({
282
- kind: 'success',
283
- prNumber: null,
284
- options: {
285
- all: false,
286
- force: false,
287
- deleteRemote: false,
288
- interactive: true,
289
- json: false,
290
- dryRun: false,
291
- },
292
- });
293
- vi.mocked(github.isGhInstalled).mockReturnValue(true);
294
- vi.mocked(git.getRepoRoot).mockReturnValue('/repo');
295
- vi.mocked(loadConfig).mockReturnValue(defaultConfig);
296
- vi.mocked(cleanpr.createDefaultDeps).mockReturnValue({});
297
- vi.mocked(cleanpr.gatherPrWorktreeInfo).mockResolvedValue(mockWorktrees);
298
- vi.mocked(cleanpr.groupWorktreesByState).mockReturnValue({
299
- merged: mockWorktrees,
300
- closed: [],
301
- open: [],
302
- unknown: [],
303
- });
304
- vi.mocked(cleanpr.getCleanableWorktrees).mockReturnValue(mockWorktrees);
305
- vi.mocked(promptChoice).mockResolvedValue('all');
306
- vi.mocked(promptConfirm).mockResolvedValue(false); // Don't delete remote
307
- vi.mocked(cleanpr.cleanWorktree).mockReturnValue({
308
- success: true,
309
- message: 'Cleaned PR #1',
310
- prNumber: 1,
311
- localBranchDeleted: true,
312
- remoteBranchDeleted: false,
313
- });
314
- vi.mocked(cleanpr.summarizeResults).mockReturnValue({ cleaned: 1, total: 1, failed: 0 });
315
- await runCli([]);
316
- expect(cleanpr.cleanWorktree).toHaveBeenCalled();
317
- expect(cleanpr.summarizeResults).toHaveBeenCalled();
318
- });
319
- it('handles merged option in interactive mode', async () => {
320
- const mockWorktrees = [
321
- makeWorktreeInfo({ prNumber: 1, prState: 'MERGED' }),
322
- makeWorktreeInfo({ prNumber: 2, prState: 'CLOSED' }),
323
- ];
324
- const { promptChoice, promptConfirm } = await import('../lib/prompts.js');
325
- vi.mocked(cleanpr.parseArgs).mockReturnValue({
326
- kind: 'success',
327
- prNumber: null,
328
- options: {
329
- all: false,
330
- force: false,
331
- deleteRemote: false,
332
- interactive: true,
333
- json: false,
334
- dryRun: false,
335
- },
336
- });
337
- vi.mocked(github.isGhInstalled).mockReturnValue(true);
338
- vi.mocked(git.getRepoRoot).mockReturnValue('/repo');
339
- vi.mocked(loadConfig).mockReturnValue(defaultConfig);
340
- vi.mocked(cleanpr.createDefaultDeps).mockReturnValue({});
341
- vi.mocked(cleanpr.gatherPrWorktreeInfo).mockResolvedValue(mockWorktrees);
342
- vi.mocked(cleanpr.groupWorktreesByState).mockReturnValue({
343
- merged: [mockWorktrees[0]],
344
- closed: [mockWorktrees[1]],
345
- open: [],
346
- unknown: [],
347
- });
348
- vi.mocked(cleanpr.getCleanableWorktrees).mockReturnValue(mockWorktrees);
349
- vi.mocked(promptChoice).mockResolvedValue('merged');
350
- vi.mocked(promptConfirm).mockResolvedValue(false);
351
- vi.mocked(cleanpr.cleanWorktree).mockReturnValue({
352
- success: true,
353
- message: 'Cleaned PR #1',
354
- prNumber: 1,
355
- localBranchDeleted: true,
356
- remoteBranchDeleted: false,
357
- });
358
- vi.mocked(cleanpr.summarizeResults).mockReturnValue({ cleaned: 1, total: 1, failed: 0 });
359
- await runCli([]);
360
- // Should only clean merged (PR #1), not closed (PR #2)
361
- expect(cleanpr.cleanWorktree).toHaveBeenCalledTimes(1);
362
- });
363
- it('handles select option in interactive mode', async () => {
364
- const mockWorktrees = [makeWorktreeInfo({ prNumber: 1, prState: 'MERGED' })];
365
- const { promptChoice, promptConfirm } = await import('../lib/prompts.js');
366
- vi.mocked(cleanpr.parseArgs).mockReturnValue({
367
- kind: 'success',
368
- prNumber: null,
369
- options: {
370
- all: false,
371
- force: false,
372
- deleteRemote: false,
373
- interactive: true,
374
- json: false,
375
- dryRun: false,
376
- },
377
- });
378
- vi.mocked(github.isGhInstalled).mockReturnValue(true);
379
- vi.mocked(git.getRepoRoot).mockReturnValue('/repo');
380
- vi.mocked(loadConfig).mockReturnValue(defaultConfig);
381
- vi.mocked(cleanpr.createDefaultDeps).mockReturnValue({});
382
- vi.mocked(cleanpr.gatherPrWorktreeInfo).mockResolvedValue(mockWorktrees);
383
- vi.mocked(cleanpr.groupWorktreesByState).mockReturnValue({
384
- merged: mockWorktrees,
385
- closed: [],
386
- open: [],
387
- unknown: [],
388
- });
389
- vi.mocked(cleanpr.getCleanableWorktrees).mockReturnValue(mockWorktrees);
390
- // First prompt: select action, Second: individual confirm, Third: remote deletion
391
- vi.mocked(promptChoice).mockResolvedValue('select');
392
- vi.mocked(promptConfirm)
393
- .mockResolvedValueOnce(true) // Yes, clean this PR
394
- .mockResolvedValueOnce(false); // No, don't delete remote
395
- vi.mocked(cleanpr.cleanWorktree).mockReturnValue({
396
- success: true,
397
- message: 'Cleaned PR #1',
398
- prNumber: 1,
399
- localBranchDeleted: true,
400
- remoteBranchDeleted: false,
401
- });
402
- vi.mocked(cleanpr.summarizeResults).mockReturnValue({ cleaned: 1, total: 1, failed: 0 });
403
- await runCli([]);
404
- expect(cleanpr.cleanWorktree).toHaveBeenCalled();
405
- });
406
- it('skips PR when user declines in select mode', async () => {
407
- const mockWorktrees = [makeWorktreeInfo({ prNumber: 1, prState: 'MERGED' })];
408
- const { promptChoice, promptConfirm } = await import('../lib/prompts.js');
409
- vi.mocked(cleanpr.parseArgs).mockReturnValue({
410
- kind: 'success',
411
- prNumber: null,
412
- options: {
413
- all: false,
414
- force: false,
415
- deleteRemote: false,
416
- interactive: true,
417
- json: false,
418
- dryRun: false,
419
- },
420
- });
421
- vi.mocked(github.isGhInstalled).mockReturnValue(true);
422
- vi.mocked(git.getRepoRoot).mockReturnValue('/repo');
423
- vi.mocked(loadConfig).mockReturnValue(defaultConfig);
424
- vi.mocked(cleanpr.createDefaultDeps).mockReturnValue({});
425
- vi.mocked(cleanpr.gatherPrWorktreeInfo).mockResolvedValue(mockWorktrees);
426
- vi.mocked(cleanpr.groupWorktreesByState).mockReturnValue({
427
- merged: mockWorktrees,
428
- closed: [],
429
- open: [],
430
- unknown: [],
431
- });
432
- vi.mocked(cleanpr.getCleanableWorktrees).mockReturnValue(mockWorktrees);
433
- vi.mocked(promptChoice).mockResolvedValue('select');
434
- vi.mocked(promptConfirm).mockResolvedValueOnce(false); // No, don't clean this PR
435
- await runCli([]);
436
- expect(mockConsoleLog).toHaveBeenCalledWith(expect.stringContaining('Nothing to clean'));
437
- });
438
- });
439
- describe('--all flag', () => {
440
- it('cleans all merged/closed worktrees in batch mode', async () => {
441
- const mockWorktrees = [makeWorktreeInfo()];
442
- vi.mocked(cleanpr.parseArgs).mockReturnValue({
443
- kind: 'success',
444
- prNumber: null,
445
- options: {
446
- all: true,
447
- force: false,
448
- deleteRemote: false,
449
- interactive: false,
450
- json: false,
451
- dryRun: false,
452
- },
453
- });
454
- vi.mocked(github.isGhInstalled).mockReturnValue(true);
455
- vi.mocked(git.getRepoRoot).mockReturnValue('/repo');
456
- vi.mocked(loadConfig).mockReturnValue(defaultConfig);
457
- vi.mocked(cleanpr.createDefaultDeps).mockReturnValue({});
458
- vi.mocked(cleanpr.gatherPrWorktreeInfo).mockResolvedValue(mockWorktrees);
459
- vi.mocked(cleanpr.getCleanableWorktrees).mockReturnValue(mockWorktrees);
460
- vi.mocked(cleanpr.cleanWorktree).mockReturnValue({
461
- success: true,
462
- message: 'Cleaned PR #123',
463
- prNumber: 123,
464
- localBranchDeleted: true,
465
- remoteBranchDeleted: false,
466
- });
467
- vi.mocked(cleanpr.summarizeResults).mockReturnValue({ cleaned: 1, total: 1, failed: 0 });
468
- await runCli(['--all']);
469
- // Verify wiring: cleanWorktree receives correct worktree info and options
470
- expect(cleanpr.cleanWorktree).toHaveBeenCalledWith(expect.objectContaining({ prNumber: 123 }), // worktree info
471
- expect.objectContaining({ force: false, deleteRemote: false }), // options
472
- expect.any(Object) // deps
473
- );
474
- expect(cleanpr.summarizeResults).toHaveBeenCalled();
475
- });
476
- it('shows info when no cleanable worktrees in --all mode', async () => {
477
- vi.mocked(cleanpr.parseArgs).mockReturnValue({
478
- kind: 'success',
479
- prNumber: null,
480
- options: {
481
- all: true,
482
- force: false,
483
- deleteRemote: false,
484
- interactive: false,
485
- json: false,
486
- dryRun: false,
487
- },
488
- });
489
- vi.mocked(github.isGhInstalled).mockReturnValue(true);
490
- vi.mocked(git.getRepoRoot).mockReturnValue('/repo');
491
- vi.mocked(loadConfig).mockReturnValue(defaultConfig);
492
- vi.mocked(cleanpr.createDefaultDeps).mockReturnValue({});
493
- vi.mocked(cleanpr.gatherPrWorktreeInfo).mockResolvedValue([]);
494
- vi.mocked(cleanpr.getCleanableWorktrees).mockReturnValue([]);
495
- await runCli(['--all']);
496
- expect(mockConsoleLog).toHaveBeenCalledWith(expect.stringContaining('No merged or closed'));
497
- });
498
- });
499
- describe('specific PR mode', () => {
500
- it('cleans specific PR worktree by number', async () => {
501
- const mockWorktree = makeWorktreeInfo({ prNumber: 42 });
502
- vi.mocked(cleanpr.parseArgs).mockReturnValue({
503
- kind: 'success',
504
- prNumber: 42,
505
- options: {
506
- all: false,
507
- force: false,
508
- deleteRemote: false,
509
- interactive: true,
510
- json: false,
511
- dryRun: false,
512
- },
513
- });
514
- vi.mocked(github.isGhInstalled).mockReturnValue(true);
515
- vi.mocked(git.getRepoRoot).mockReturnValue('/repo');
516
- vi.mocked(loadConfig).mockReturnValue(defaultConfig);
517
- vi.mocked(cleanpr.createDefaultDeps).mockReturnValue({});
518
- vi.mocked(cleanpr.gatherPrWorktreeInfo).mockResolvedValue([mockWorktree]);
519
- vi.mocked(cleanpr.findWorktreeByPrNumber).mockReturnValue(mockWorktree);
520
- vi.mocked(cleanpr.cleanWorktree).mockReturnValue({
521
- success: true,
522
- message: 'Cleaned PR #42',
523
- prNumber: 42,
524
- localBranchDeleted: true,
525
- remoteBranchDeleted: false,
526
- });
527
- await runCli(['42']);
528
- // Verify wiring: findWorktreeByPrNumber receives worktrees and PR number
529
- expect(cleanpr.findWorktreeByPrNumber).toHaveBeenCalledWith([mockWorktree], 42);
530
- // Verify wiring: cleanWorktree receives correct worktree info and options
531
- expect(cleanpr.cleanWorktree).toHaveBeenCalledWith(expect.objectContaining({ prNumber: 42 }), // worktree info
532
- expect.objectContaining({ force: false, deleteRemote: false }), // options
533
- expect.any(Object) // deps
534
- );
535
- });
536
- it('exits 1 when specific PR not found', async () => {
537
- vi.mocked(cleanpr.parseArgs).mockReturnValue({
538
- kind: 'success',
539
- prNumber: 999,
540
- options: {
541
- all: false,
542
- force: false,
543
- deleteRemote: false,
544
- interactive: true,
545
- json: false,
546
- dryRun: false,
547
- },
548
- });
549
- vi.mocked(github.isGhInstalled).mockReturnValue(true);
550
- vi.mocked(git.getRepoRoot).mockReturnValue('/repo');
551
- vi.mocked(loadConfig).mockReturnValue(defaultConfig);
552
- vi.mocked(cleanpr.createDefaultDeps).mockReturnValue({});
553
- vi.mocked(cleanpr.gatherPrWorktreeInfo).mockResolvedValue([]);
554
- vi.mocked(cleanpr.findWorktreeByPrNumber).mockReturnValue(undefined);
555
- await runCli(['999']);
556
- expect(mockConsoleError).toHaveBeenCalledWith(expect.stringContaining('No worktree found'));
557
- expect(mockProcessExit).toHaveBeenCalledWith(1);
558
- });
559
- });
560
- describe('JSON output mode', () => {
561
- it('outputs JSON error when gh not installed with --json', async () => {
562
- vi.mocked(cleanpr.parseArgs).mockReturnValue({
563
- kind: 'success',
564
- prNumber: null,
565
- options: {
566
- all: true,
567
- force: false,
568
- deleteRemote: false,
569
- interactive: false,
570
- json: true,
571
- dryRun: false,
572
- },
573
- });
574
- vi.mocked(github.isGhInstalled).mockReturnValue(false);
575
- await runCli(['--json', '--all']);
576
- expect(mockConsoleLog).toHaveBeenCalledWith(expect.stringContaining('"success": false'));
577
- expect(mockConsoleLog).toHaveBeenCalledWith(expect.stringContaining('GH_NOT_INSTALLED'));
578
- expect(mockProcessExit).toHaveBeenCalledWith(1);
579
- });
580
- it('outputs JSON error when not in git repo with --json', async () => {
581
- vi.mocked(cleanpr.parseArgs).mockReturnValue({
582
- kind: 'success',
583
- prNumber: null,
584
- options: {
585
- all: true,
586
- force: false,
587
- deleteRemote: false,
588
- interactive: false,
589
- json: true,
590
- dryRun: false,
591
- },
592
- });
593
- vi.mocked(github.isGhInstalled).mockReturnValue(true);
594
- vi.mocked(git.getRepoRoot).mockReturnValue('');
595
- await runCli(['--json', '--all']);
596
- expect(mockConsoleLog).toHaveBeenCalledWith(expect.stringContaining('"success": false'));
597
- expect(mockConsoleLog).toHaveBeenCalledWith(expect.stringContaining('NOT_GIT_REPO'));
598
- expect(mockProcessExit).toHaveBeenCalledWith(1);
599
- });
600
- it('outputs JSON error on parse error with --json', async () => {
601
- vi.mocked(cleanpr.parseArgs).mockReturnValue({
602
- kind: 'error',
603
- message: 'Invalid option: --invalid',
604
- });
605
- await runCli(['--json', '--invalid']);
606
- expect(mockConsoleLog).toHaveBeenCalledWith(expect.stringContaining('"success": false'));
607
- expect(mockConsoleLog).toHaveBeenCalledWith(expect.stringContaining('INVALID_ARGUMENT'));
608
- expect(mockProcessExit).toHaveBeenCalledWith(1);
609
- });
610
- it('outputs JSON success when cleaning all with --json', async () => {
611
- const mockWorktrees = [makeWorktreeInfo()];
612
- vi.mocked(cleanpr.parseArgs).mockReturnValue({
613
- kind: 'success',
614
- prNumber: null,
615
- options: {
616
- all: true,
617
- force: false,
618
- deleteRemote: false,
619
- interactive: false,
620
- json: true,
621
- dryRun: false,
622
- },
623
- });
624
- vi.mocked(github.isGhInstalled).mockReturnValue(true);
625
- vi.mocked(git.getRepoRoot).mockReturnValue('/repo');
626
- vi.mocked(loadConfig).mockReturnValue(defaultConfig);
627
- vi.mocked(cleanpr.createDefaultDeps).mockReturnValue({});
628
- vi.mocked(cleanpr.gatherPrWorktreeInfo).mockResolvedValue(mockWorktrees);
629
- vi.mocked(cleanpr.getCleanableWorktrees).mockReturnValue(mockWorktrees);
630
- vi.mocked(cleanpr.cleanWorktree).mockReturnValue({
631
- success: true,
632
- message: 'Cleaned PR #123',
633
- prNumber: 123,
634
- localBranchDeleted: true,
635
- remoteBranchDeleted: false,
636
- });
637
- await runCli(['--json', '--all']);
638
- expect(mockConsoleLog).toHaveBeenCalledWith(expect.stringContaining('"success": true'));
639
- expect(mockConsoleLog).toHaveBeenCalledWith(expect.stringContaining('"totalCleaned": 1'));
640
- });
641
- it('outputs JSON empty result when no cleanable worktrees with --json --all', async () => {
642
- vi.mocked(cleanpr.parseArgs).mockReturnValue({
643
- kind: 'success',
644
- prNumber: null,
645
- options: {
646
- all: true,
647
- force: false,
648
- deleteRemote: false,
649
- interactive: false,
650
- json: true,
651
- dryRun: false,
652
- },
653
- });
654
- vi.mocked(github.isGhInstalled).mockReturnValue(true);
655
- vi.mocked(git.getRepoRoot).mockReturnValue('/repo');
656
- vi.mocked(loadConfig).mockReturnValue(defaultConfig);
657
- vi.mocked(cleanpr.createDefaultDeps).mockReturnValue({});
658
- vi.mocked(cleanpr.gatherPrWorktreeInfo).mockResolvedValue([]);
659
- vi.mocked(cleanpr.getCleanableWorktrees).mockReturnValue([]);
660
- await runCli(['--json', '--all']);
661
- expect(mockConsoleLog).toHaveBeenCalledWith(expect.stringContaining('"success": true'));
662
- expect(mockConsoleLog).toHaveBeenCalledWith(expect.stringContaining('"totalCleaned": 0'));
663
- });
664
- it('outputs JSON error when specific PR not found with --json', async () => {
665
- vi.mocked(cleanpr.parseArgs).mockReturnValue({
666
- kind: 'success',
667
- prNumber: 999,
668
- options: {
669
- all: false,
670
- force: false,
671
- deleteRemote: false,
672
- interactive: true,
673
- json: true,
674
- dryRun: false,
675
- },
676
- });
677
- vi.mocked(github.isGhInstalled).mockReturnValue(true);
678
- vi.mocked(git.getRepoRoot).mockReturnValue('/repo');
679
- vi.mocked(loadConfig).mockReturnValue(defaultConfig);
680
- vi.mocked(cleanpr.createDefaultDeps).mockReturnValue({});
681
- vi.mocked(cleanpr.gatherPrWorktreeInfo).mockResolvedValue([]);
682
- vi.mocked(cleanpr.findWorktreeByPrNumber).mockReturnValue(undefined);
683
- await runCli(['--json', '999']);
684
- expect(mockConsoleLog).toHaveBeenCalledWith(expect.stringContaining('"success": false'));
685
- expect(mockConsoleLog).toHaveBeenCalledWith(expect.stringContaining('PR_NOT_FOUND'));
686
- expect(mockProcessExit).toHaveBeenCalledWith(1);
687
- });
688
- it('outputs JSON success when cleaning specific PR with --json', async () => {
689
- const mockWorktree = makeWorktreeInfo({ prNumber: 42 });
690
- vi.mocked(cleanpr.parseArgs).mockReturnValue({
691
- kind: 'success',
692
- prNumber: 42,
693
- options: {
694
- all: false,
695
- force: false,
696
- deleteRemote: false,
697
- interactive: false,
698
- json: true,
699
- dryRun: false,
700
- },
701
- });
702
- vi.mocked(github.isGhInstalled).mockReturnValue(true);
703
- vi.mocked(git.getRepoRoot).mockReturnValue('/repo');
704
- vi.mocked(loadConfig).mockReturnValue(defaultConfig);
705
- vi.mocked(cleanpr.createDefaultDeps).mockReturnValue({});
706
- vi.mocked(cleanpr.gatherPrWorktreeInfo).mockResolvedValue([mockWorktree]);
707
- vi.mocked(cleanpr.findWorktreeByPrNumber).mockReturnValue(mockWorktree);
708
- vi.mocked(cleanpr.cleanWorktree).mockReturnValue({
709
- success: true,
710
- message: 'Cleaned PR #42',
711
- prNumber: 42,
712
- localBranchDeleted: true,
713
- remoteBranchDeleted: true,
714
- });
715
- await runCli(['--json', '42']);
716
- expect(mockConsoleLog).toHaveBeenCalledWith(expect.stringContaining('"success": true'));
717
- expect(mockConsoleLog).toHaveBeenCalledWith(expect.stringContaining('"totalCleaned": 1'));
718
- });
719
- it('outputs JSON error when interactive mode requested with --json', async () => {
720
- vi.mocked(cleanpr.parseArgs).mockReturnValue({
721
- kind: 'success',
722
- prNumber: null,
723
- options: {
724
- all: false,
725
- force: false,
726
- deleteRemote: false,
727
- interactive: true,
728
- json: true,
729
- dryRun: false,
730
- },
731
- });
732
- vi.mocked(github.isGhInstalled).mockReturnValue(true);
733
- vi.mocked(git.getRepoRoot).mockReturnValue('/repo');
734
- vi.mocked(loadConfig).mockReturnValue(defaultConfig);
735
- vi.mocked(cleanpr.createDefaultDeps).mockReturnValue({});
736
- vi.mocked(cleanpr.gatherPrWorktreeInfo).mockResolvedValue([makeWorktreeInfo()]);
737
- await runCli(['--json']);
738
- expect(mockConsoleLog).toHaveBeenCalledWith(expect.stringContaining('"success": false'));
739
- expect(mockConsoleLog).toHaveBeenCalledWith(expect.stringContaining('INVALID_ARGUMENT'));
740
- expect(mockConsoleLog).toHaveBeenCalledWith(expect.stringContaining('Interactive mode not supported'));
741
- expect(mockProcessExit).toHaveBeenCalledWith(1);
742
- });
743
- });
744
- describe('dry-run mode', () => {
745
- it('outputs dry-run results with --dry-run --all', async () => {
746
- const mockWorktrees = [makeWorktreeInfo()];
747
- vi.mocked(cleanpr.parseArgs).mockReturnValue({
748
- kind: 'success',
749
- prNumber: null,
750
- options: {
751
- all: true,
752
- force: false,
753
- deleteRemote: false,
754
- interactive: false,
755
- json: false,
756
- dryRun: true,
757
- },
758
- });
759
- vi.mocked(github.isGhInstalled).mockReturnValue(true);
760
- vi.mocked(git.getRepoRoot).mockReturnValue('/repo');
761
- vi.mocked(loadConfig).mockReturnValue(defaultConfig);
762
- vi.mocked(cleanpr.createDefaultDeps).mockReturnValue({});
763
- vi.mocked(cleanpr.gatherPrWorktreeInfo).mockResolvedValue(mockWorktrees);
764
- vi.mocked(cleanpr.getCleanableWorktrees).mockReturnValue(mockWorktrees);
765
- vi.mocked(cleanpr.cleanWorktree).mockReturnValue({
766
- success: true,
767
- message: 'Would clean PR #123',
768
- prNumber: 123,
769
- localBranchDeleted: true,
770
- remoteBranchDeleted: false,
771
- });
772
- vi.mocked(cleanpr.summarizeResults).mockReturnValue({ cleaned: 1, total: 1, failed: 0 });
773
- await runCli(['--dry-run', '--all']);
774
- expect(mockConsoleLog).toHaveBeenCalledWith(expect.stringContaining('Would clean'));
775
- });
776
- it('outputs JSON dry-run results with --json --dry-run --all', async () => {
777
- const mockWorktrees = [makeWorktreeInfo()];
778
- vi.mocked(cleanpr.parseArgs).mockReturnValue({
779
- kind: 'success',
780
- prNumber: null,
781
- options: {
782
- all: true,
783
- force: false,
784
- deleteRemote: false,
785
- interactive: false,
786
- json: true,
787
- dryRun: true,
788
- },
789
- });
790
- vi.mocked(github.isGhInstalled).mockReturnValue(true);
791
- vi.mocked(git.getRepoRoot).mockReturnValue('/repo');
792
- vi.mocked(loadConfig).mockReturnValue(defaultConfig);
793
- vi.mocked(cleanpr.createDefaultDeps).mockReturnValue({});
794
- vi.mocked(cleanpr.gatherPrWorktreeInfo).mockResolvedValue(mockWorktrees);
795
- vi.mocked(cleanpr.getCleanableWorktrees).mockReturnValue(mockWorktrees);
796
- vi.mocked(cleanpr.cleanWorktree).mockReturnValue({
797
- success: true,
798
- message: 'Would clean PR #123',
799
- prNumber: 123,
800
- localBranchDeleted: true,
801
- remoteBranchDeleted: false,
802
- });
803
- await runCli(['--json', '--dry-run', '--all']);
804
- expect(mockConsoleLog).toHaveBeenCalledWith(expect.stringContaining('"wouldClean"'));
805
- expect(mockConsoleLog).toHaveBeenCalledWith(expect.stringContaining('"totalWouldClean": 1'));
806
- });
807
- it('outputs dry-run results for specific PR with --dry-run', async () => {
808
- const mockWorktree = makeWorktreeInfo({ prNumber: 42 });
809
- vi.mocked(cleanpr.parseArgs).mockReturnValue({
810
- kind: 'success',
811
- prNumber: 42,
812
- options: {
813
- all: false,
814
- force: false,
815
- deleteRemote: false,
816
- interactive: false,
817
- json: false,
818
- dryRun: true,
819
- },
820
- });
821
- vi.mocked(github.isGhInstalled).mockReturnValue(true);
822
- vi.mocked(git.getRepoRoot).mockReturnValue('/repo');
823
- vi.mocked(loadConfig).mockReturnValue(defaultConfig);
824
- vi.mocked(cleanpr.createDefaultDeps).mockReturnValue({});
825
- vi.mocked(cleanpr.gatherPrWorktreeInfo).mockResolvedValue([mockWorktree]);
826
- vi.mocked(cleanpr.findWorktreeByPrNumber).mockReturnValue(mockWorktree);
827
- vi.mocked(cleanpr.cleanWorktree).mockReturnValue({
828
- success: true,
829
- message: 'Would clean PR #42',
830
- prNumber: 42,
831
- localBranchDeleted: true,
832
- remoteBranchDeleted: false,
833
- });
834
- await runCli(['--dry-run', '42']);
835
- expect(cleanpr.cleanWorktree).toHaveBeenCalledWith(expect.objectContaining({ prNumber: 42 }), expect.objectContaining({ dryRun: true }), expect.any(Object));
836
- });
837
- });
838
- describe('failed cleanup handling', () => {
839
- it('handles failed cleanup result for specific PR', async () => {
840
- const mockWorktree = makeWorktreeInfo({ prNumber: 42 });
841
- vi.mocked(cleanpr.parseArgs).mockReturnValue({
842
- kind: 'success',
843
- prNumber: 42,
844
- options: {
845
- all: false,
846
- force: false,
847
- deleteRemote: false,
848
- interactive: false,
849
- json: false,
850
- dryRun: false,
851
- },
852
- });
853
- vi.mocked(github.isGhInstalled).mockReturnValue(true);
854
- vi.mocked(git.getRepoRoot).mockReturnValue('/repo');
855
- vi.mocked(loadConfig).mockReturnValue(defaultConfig);
856
- vi.mocked(cleanpr.createDefaultDeps).mockReturnValue({});
857
- vi.mocked(cleanpr.gatherPrWorktreeInfo).mockResolvedValue([mockWorktree]);
858
- vi.mocked(cleanpr.findWorktreeByPrNumber).mockReturnValue(mockWorktree);
859
- vi.mocked(cleanpr.cleanWorktree).mockReturnValue({
860
- success: false,
861
- message: 'Failed to clean PR #42: worktree has uncommitted changes',
862
- prNumber: 42,
863
- localBranchDeleted: false,
864
- remoteBranchDeleted: false,
865
- });
866
- await runCli(['42']);
867
- expect(mockProcessExit).toHaveBeenCalledWith(1);
868
- });
869
- it('handles failed cleanup in --all mode with warnings', async () => {
870
- const mockWorktrees = [
871
- makeWorktreeInfo({ prNumber: 1 }),
872
- makeWorktreeInfo({ prNumber: 2, hasChanges: true }),
873
- ];
874
- vi.mocked(cleanpr.parseArgs).mockReturnValue({
875
- kind: 'success',
876
- prNumber: null,
877
- options: {
878
- all: true,
879
- force: false,
880
- deleteRemote: false,
881
- interactive: false,
882
- json: false,
883
- dryRun: false,
884
- },
885
- });
886
- vi.mocked(github.isGhInstalled).mockReturnValue(true);
887
- vi.mocked(git.getRepoRoot).mockReturnValue('/repo');
888
- vi.mocked(loadConfig).mockReturnValue(defaultConfig);
889
- vi.mocked(cleanpr.createDefaultDeps).mockReturnValue({});
890
- vi.mocked(cleanpr.gatherPrWorktreeInfo).mockResolvedValue(mockWorktrees);
891
- vi.mocked(cleanpr.getCleanableWorktrees).mockReturnValue(mockWorktrees);
892
- vi.mocked(cleanpr.cleanWorktree)
893
- .mockReturnValueOnce({
894
- success: true,
895
- message: 'Cleaned PR #1',
896
- prNumber: 1,
897
- localBranchDeleted: true,
898
- remoteBranchDeleted: false,
899
- })
900
- .mockReturnValueOnce({
901
- success: false,
902
- message: 'Skipped PR #2: has uncommitted changes',
903
- prNumber: 2,
904
- localBranchDeleted: false,
905
- remoteBranchDeleted: false,
906
- });
907
- vi.mocked(cleanpr.summarizeResults).mockReturnValue({ cleaned: 1, total: 2, failed: 1 });
908
- await runCli(['--all']);
909
- expect(cleanpr.summarizeResults).toHaveBeenCalled();
910
- });
911
- it('includes skipped items in JSON output', async () => {
912
- const mockWorktrees = [
913
- makeWorktreeInfo({ prNumber: 1 }),
914
- makeWorktreeInfo({ prNumber: 2, hasChanges: true }),
915
- ];
916
- vi.mocked(cleanpr.parseArgs).mockReturnValue({
917
- kind: 'success',
918
- prNumber: null,
919
- options: {
920
- all: true,
921
- force: false,
922
- deleteRemote: false,
923
- interactive: false,
924
- json: true,
925
- dryRun: false,
926
- },
927
- });
928
- vi.mocked(github.isGhInstalled).mockReturnValue(true);
929
- vi.mocked(git.getRepoRoot).mockReturnValue('/repo');
930
- vi.mocked(loadConfig).mockReturnValue(defaultConfig);
931
- vi.mocked(cleanpr.createDefaultDeps).mockReturnValue({});
932
- vi.mocked(cleanpr.gatherPrWorktreeInfo).mockResolvedValue(mockWorktrees);
933
- vi.mocked(cleanpr.getCleanableWorktrees).mockReturnValue(mockWorktrees);
934
- vi.mocked(cleanpr.cleanWorktree)
935
- .mockReturnValueOnce({
936
- success: true,
937
- message: 'Cleaned PR #1',
938
- prNumber: 1,
939
- localBranchDeleted: true,
940
- remoteBranchDeleted: false,
941
- })
942
- .mockReturnValueOnce({
943
- success: false,
944
- message: 'Skipped PR #2: has uncommitted changes',
945
- prNumber: 2,
946
- localBranchDeleted: false,
947
- remoteBranchDeleted: false,
948
- });
949
- await runCli(['--json', '--all']);
950
- expect(mockConsoleLog).toHaveBeenCalledWith(expect.stringContaining('"totalSkipped": 1'));
951
- });
952
- });
953
- });
954
- //# sourceMappingURL=cleanpr.test.js.map