@epoch-ai/cli 2.2.4 → 2.2.5

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 (711) hide show
  1. package/LICENSE +21 -0
  2. package/package.json +18 -179
  3. package/.artifacts/unit/junit.xml +0 -2823
  4. package/.project-map/backups/20260530_223453/.project-map.json +0 -90101
  5. package/.project-map/backups/20260530_223507/.project-map.json +0 -90101
  6. package/.project-map/backups/20260530_223512/.project-map.json +0 -90101
  7. package/.project-map/backups/20260530_223512/map.toon +0 -666
  8. package/.project-map/backups/20260530_223516/.project-map.json +0 -90101
  9. package/.project-map/backups/20260530_223516/map.toon +0 -666
  10. package/.project-map/backups/20260530_223520/.project-map.json +0 -90101
  11. package/.project-map/backups/20260530_223520/map.toon +0 -666
  12. package/AGENTS.md +0 -47
  13. package/BUN_SHELL_MIGRATION_PLAN.md +0 -136
  14. package/Dockerfile +0 -18
  15. package/README.md +0 -15
  16. package/bunfig.toml +0 -7
  17. package/drizzle.config.ts +0 -10
  18. package/git +0 -0
  19. package/migration/20260127222353_familiar_lady_ursula/migration.sql +0 -90
  20. package/migration/20260127222353_familiar_lady_ursula/snapshot.json +0 -796
  21. package/migration/20260211171708_add_project_commands/migration.sql +0 -1
  22. package/migration/20260211171708_add_project_commands/snapshot.json +0 -806
  23. package/migration/20260213144116_wakeful_the_professor/migration.sql +0 -11
  24. package/migration/20260213144116_wakeful_the_professor/snapshot.json +0 -897
  25. package/migration/20260225215848_workspace/migration.sql +0 -7
  26. package/migration/20260225215848_workspace/snapshot.json +0 -959
  27. package/migration/20260227213759_add_session_workspace_id/migration.sql +0 -2
  28. package/migration/20260227213759_add_session_workspace_id/snapshot.json +0 -983
  29. package/migration/20260228203230_blue_harpoon/migration.sql +0 -17
  30. package/migration/20260228203230_blue_harpoon/snapshot.json +0 -1102
  31. package/migration/20260303231226_add_workspace_fields/migration.sql +0 -5
  32. package/migration/20260303231226_add_workspace_fields/snapshot.json +0 -1013
  33. package/migration/20260309230000_move_org_to_state/migration.sql +0 -3
  34. package/migration/20260309230000_move_org_to_state/snapshot.json +0 -1156
  35. package/migration/20260312043431_session_message_cursor/migration.sql +0 -4
  36. package/migration/20260312043431_session_message_cursor/snapshot.json +0 -1168
  37. package/migration/20260323234822_events/migration.sql +0 -13
  38. package/migration/20260323234822_events/snapshot.json +0 -1271
  39. package/migration/20260418092949_add_yolo_to_session/migration.sql +0 -2
  40. package/migration/20260418092949_add_yolo_to_session/snapshot.json +0 -1199
  41. package/migration/20260419120000_add_intervention_to_session/migration.sql +0 -2
  42. package/parsers-config.ts +0 -290
  43. package/script/build-node.ts +0 -71
  44. package/script/build.ts +0 -255
  45. package/script/check-migrations.ts +0 -16
  46. package/script/fix-node-pty.ts +0 -28
  47. package/script/publish.ts +0 -184
  48. package/script/schema.ts +0 -63
  49. package/script/seed-e2e.ts +0 -60
  50. package/script/upgrade-opentui.ts +0 -64
  51. package/specs/effect-migration.md +0 -310
  52. package/specs/tui-plugins.md +0 -436
  53. package/specs/v2.md +0 -14
  54. package/src/account/account.sql.ts +0 -39
  55. package/src/account/index.ts +0 -465
  56. package/src/account/repo.ts +0 -163
  57. package/src/account/schema.ts +0 -91
  58. package/src/acp/README.md +0 -174
  59. package/src/acp/agent.ts +0 -1847
  60. package/src/acp/session.ts +0 -116
  61. package/src/acp/types.ts +0 -24
  62. package/src/agent/agent.ts +0 -445
  63. package/src/agent/generate.txt +0 -75
  64. package/src/agent/prompt/compaction.txt +0 -15
  65. package/src/agent/prompt/explore.txt +0 -9
  66. package/src/agent/prompt/summary.txt +0 -11
  67. package/src/agent/prompt/title.txt +0 -44
  68. package/src/auth/index.ts +0 -110
  69. package/src/bus/bus-event.ts +0 -40
  70. package/src/bus/global.ts +0 -10
  71. package/src/bus/index.ts +0 -232
  72. package/src/cli/bootstrap.ts +0 -17
  73. package/src/cli/cmd/account.ts +0 -257
  74. package/src/cli/cmd/acp.ts +0 -70
  75. package/src/cli/cmd/agent.ts +0 -245
  76. package/src/cli/cmd/cmd.ts +0 -7
  77. package/src/cli/cmd/db.ts +0 -119
  78. package/src/cli/cmd/debug/agent.ts +0 -167
  79. package/src/cli/cmd/debug/config.ts +0 -16
  80. package/src/cli/cmd/debug/file.ts +0 -97
  81. package/src/cli/cmd/debug/index.ts +0 -48
  82. package/src/cli/cmd/debug/lsp.ts +0 -53
  83. package/src/cli/cmd/debug/ripgrep.ts +0 -87
  84. package/src/cli/cmd/debug/scrap.ts +0 -16
  85. package/src/cli/cmd/debug/skill.ts +0 -16
  86. package/src/cli/cmd/debug/snapshot.ts +0 -52
  87. package/src/cli/cmd/export.ts +0 -89
  88. package/src/cli/cmd/generate.ts +0 -38
  89. package/src/cli/cmd/github.ts +0 -1639
  90. package/src/cli/cmd/import.ts +0 -169
  91. package/src/cli/cmd/mcp.ts +0 -754
  92. package/src/cli/cmd/models.ts +0 -78
  93. package/src/cli/cmd/plug.ts +0 -233
  94. package/src/cli/cmd/pr.ts +0 -127
  95. package/src/cli/cmd/providers.ts +0 -478
  96. package/src/cli/cmd/run.ts +0 -681
  97. package/src/cli/cmd/serve.ts +0 -24
  98. package/src/cli/cmd/session.ts +0 -159
  99. package/src/cli/cmd/stats.ts +0 -410
  100. package/src/cli/cmd/tui/app.tsx +0 -945
  101. package/src/cli/cmd/tui/attach.ts +0 -88
  102. package/src/cli/cmd/tui/component/border.tsx +0 -21
  103. package/src/cli/cmd/tui/component/dialog-agent.tsx +0 -31
  104. package/src/cli/cmd/tui/component/dialog-command.tsx +0 -171
  105. package/src/cli/cmd/tui/component/dialog-console-org.tsx +0 -103
  106. package/src/cli/cmd/tui/component/dialog-mcp.tsx +0 -86
  107. package/src/cli/cmd/tui/component/dialog-model.tsx +0 -190
  108. package/src/cli/cmd/tui/component/dialog-provider.tsx +0 -364
  109. package/src/cli/cmd/tui/component/dialog-session-list.tsx +0 -108
  110. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +0 -31
  111. package/src/cli/cmd/tui/component/dialog-skill.tsx +0 -36
  112. package/src/cli/cmd/tui/component/dialog-stash.tsx +0 -87
  113. package/src/cli/cmd/tui/component/dialog-status.tsx +0 -168
  114. package/src/cli/cmd/tui/component/dialog-tag.tsx +0 -44
  115. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +0 -50
  116. package/src/cli/cmd/tui/component/dialog-variant.tsx +0 -39
  117. package/src/cli/cmd/tui/component/dialog-workspace-list.tsx +0 -320
  118. package/src/cli/cmd/tui/component/error-component.tsx +0 -92
  119. package/src/cli/cmd/tui/component/logo.tsx +0 -85
  120. package/src/cli/cmd/tui/component/plugin-route-missing.tsx +0 -14
  121. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +0 -672
  122. package/src/cli/cmd/tui/component/prompt/frecency.tsx +0 -90
  123. package/src/cli/cmd/tui/component/prompt/history.tsx +0 -109
  124. package/src/cli/cmd/tui/component/prompt/index.tsx +0 -1336
  125. package/src/cli/cmd/tui/component/prompt/part.ts +0 -16
  126. package/src/cli/cmd/tui/component/prompt/stash.tsx +0 -101
  127. package/src/cli/cmd/tui/component/spinner.tsx +0 -24
  128. package/src/cli/cmd/tui/component/startup-loading.tsx +0 -63
  129. package/src/cli/cmd/tui/component/textarea-keybindings.ts +0 -73
  130. package/src/cli/cmd/tui/component/todo-item.tsx +0 -32
  131. package/src/cli/cmd/tui/component/workspace/dialog-session-list.tsx +0 -151
  132. package/src/cli/cmd/tui/context/args.tsx +0 -15
  133. package/src/cli/cmd/tui/context/directory.ts +0 -13
  134. package/src/cli/cmd/tui/context/exit.tsx +0 -60
  135. package/src/cli/cmd/tui/context/helper.tsx +0 -25
  136. package/src/cli/cmd/tui/context/keybind.tsx +0 -105
  137. package/src/cli/cmd/tui/context/kv.tsx +0 -52
  138. package/src/cli/cmd/tui/context/local.tsx +0 -456
  139. package/src/cli/cmd/tui/context/plugin-keybinds.ts +0 -41
  140. package/src/cli/cmd/tui/context/prompt.tsx +0 -18
  141. package/src/cli/cmd/tui/context/route.tsx +0 -52
  142. package/src/cli/cmd/tui/context/sdk.tsx +0 -115
  143. package/src/cli/cmd/tui/context/sync.tsx +0 -516
  144. package/src/cli/cmd/tui/context/theme/aura.json +0 -69
  145. package/src/cli/cmd/tui/context/theme/ayu.json +0 -80
  146. package/src/cli/cmd/tui/context/theme/carbonfox.json +0 -248
  147. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +0 -233
  148. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +0 -233
  149. package/src/cli/cmd/tui/context/theme/catppuccin.json +0 -112
  150. package/src/cli/cmd/tui/context/theme/cobalt2.json +0 -228
  151. package/src/cli/cmd/tui/context/theme/cursor.json +0 -249
  152. package/src/cli/cmd/tui/context/theme/dracula.json +0 -219
  153. package/src/cli/cmd/tui/context/theme/epochcli.json +0 -245
  154. package/src/cli/cmd/tui/context/theme/everforest.json +0 -241
  155. package/src/cli/cmd/tui/context/theme/flexoki.json +0 -237
  156. package/src/cli/cmd/tui/context/theme/github.json +0 -233
  157. package/src/cli/cmd/tui/context/theme/gruvbox.json +0 -242
  158. package/src/cli/cmd/tui/context/theme/kanagawa.json +0 -77
  159. package/src/cli/cmd/tui/context/theme/lucent-orng.json +0 -237
  160. package/src/cli/cmd/tui/context/theme/material.json +0 -235
  161. package/src/cli/cmd/tui/context/theme/matrix.json +0 -77
  162. package/src/cli/cmd/tui/context/theme/mercury.json +0 -252
  163. package/src/cli/cmd/tui/context/theme/monokai.json +0 -221
  164. package/src/cli/cmd/tui/context/theme/nightowl.json +0 -221
  165. package/src/cli/cmd/tui/context/theme/nord.json +0 -223
  166. package/src/cli/cmd/tui/context/theme/one-dark.json +0 -84
  167. package/src/cli/cmd/tui/context/theme/orng.json +0 -249
  168. package/src/cli/cmd/tui/context/theme/osaka-jade.json +0 -93
  169. package/src/cli/cmd/tui/context/theme/palenight.json +0 -222
  170. package/src/cli/cmd/tui/context/theme/rosepine.json +0 -234
  171. package/src/cli/cmd/tui/context/theme/solarized.json +0 -223
  172. package/src/cli/cmd/tui/context/theme/synthwave84.json +0 -226
  173. package/src/cli/cmd/tui/context/theme/tokyonight.json +0 -243
  174. package/src/cli/cmd/tui/context/theme/vercel.json +0 -245
  175. package/src/cli/cmd/tui/context/theme/vesper.json +0 -218
  176. package/src/cli/cmd/tui/context/theme/zenburn.json +0 -223
  177. package/src/cli/cmd/tui/context/theme.tsx +0 -1236
  178. package/src/cli/cmd/tui/context/tui-config.tsx +0 -9
  179. package/src/cli/cmd/tui/event.ts +0 -48
  180. package/src/cli/cmd/tui/feature-plugins/home/footer.tsx +0 -93
  181. package/src/cli/cmd/tui/feature-plugins/home/tips-view.tsx +0 -145
  182. package/src/cli/cmd/tui/feature-plugins/home/tips.tsx +0 -50
  183. package/src/cli/cmd/tui/feature-plugins/sidebar/context.tsx +0 -63
  184. package/src/cli/cmd/tui/feature-plugins/sidebar/files.tsx +0 -62
  185. package/src/cli/cmd/tui/feature-plugins/sidebar/footer.tsx +0 -93
  186. package/src/cli/cmd/tui/feature-plugins/sidebar/lsp.tsx +0 -66
  187. package/src/cli/cmd/tui/feature-plugins/sidebar/mcp.tsx +0 -96
  188. package/src/cli/cmd/tui/feature-plugins/sidebar/todo.tsx +0 -48
  189. package/src/cli/cmd/tui/feature-plugins/system/plugins.tsx +0 -270
  190. package/src/cli/cmd/tui/plugin/api.tsx +0 -397
  191. package/src/cli/cmd/tui/plugin/index.ts +0 -3
  192. package/src/cli/cmd/tui/plugin/internal.ts +0 -27
  193. package/src/cli/cmd/tui/plugin/runtime.ts +0 -1031
  194. package/src/cli/cmd/tui/plugin/slots.tsx +0 -60
  195. package/src/cli/cmd/tui/routes/home.tsx +0 -84
  196. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +0 -65
  197. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +0 -110
  198. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +0 -26
  199. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +0 -47
  200. package/src/cli/cmd/tui/routes/session/footer.tsx +0 -91
  201. package/src/cli/cmd/tui/routes/session/index.tsx +0 -2161
  202. package/src/cli/cmd/tui/routes/session/permission.tsx +0 -691
  203. package/src/cli/cmd/tui/routes/session/question.tsx +0 -468
  204. package/src/cli/cmd/tui/routes/session/sidebar.tsx +0 -70
  205. package/src/cli/cmd/tui/routes/session/subagent-footer.tsx +0 -131
  206. package/src/cli/cmd/tui/thread.ts +0 -241
  207. package/src/cli/cmd/tui/ui/dialog-alert.tsx +0 -59
  208. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +0 -89
  209. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +0 -211
  210. package/src/cli/cmd/tui/ui/dialog-help.tsx +0 -40
  211. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +0 -115
  212. package/src/cli/cmd/tui/ui/dialog-select.tsx +0 -417
  213. package/src/cli/cmd/tui/ui/dialog.tsx +0 -192
  214. package/src/cli/cmd/tui/ui/link.tsx +0 -28
  215. package/src/cli/cmd/tui/ui/spinner.ts +0 -368
  216. package/src/cli/cmd/tui/ui/toast.tsx +0 -100
  217. package/src/cli/cmd/tui/util/clipboard.ts +0 -192
  218. package/src/cli/cmd/tui/util/editor.ts +0 -37
  219. package/src/cli/cmd/tui/util/model.ts +0 -23
  220. package/src/cli/cmd/tui/util/provider-origin.ts +0 -20
  221. package/src/cli/cmd/tui/util/scroll.ts +0 -23
  222. package/src/cli/cmd/tui/util/selection.ts +0 -25
  223. package/src/cli/cmd/tui/util/signal.ts +0 -7
  224. package/src/cli/cmd/tui/util/terminal.ts +0 -114
  225. package/src/cli/cmd/tui/util/transcript.ts +0 -112
  226. package/src/cli/cmd/tui/win32.ts +0 -129
  227. package/src/cli/cmd/tui/worker.ts +0 -195
  228. package/src/cli/cmd/uninstall.ts +0 -353
  229. package/src/cli/cmd/upgrade.ts +0 -73
  230. package/src/cli/cmd/web.ts +0 -81
  231. package/src/cli/effect/prompt.ts +0 -25
  232. package/src/cli/error.ts +0 -46
  233. package/src/cli/heap.ts +0 -59
  234. package/src/cli/logo.ts +0 -6
  235. package/src/cli/network.ts +0 -60
  236. package/src/cli/ui.ts +0 -133
  237. package/src/cli/upgrade.ts +0 -31
  238. package/src/command/index.ts +0 -197
  239. package/src/command/template/initialize.txt +0 -66
  240. package/src/command/template/review.txt +0 -101
  241. package/src/config/config.ts +0 -1610
  242. package/src/config/console-state.ts +0 -15
  243. package/src/config/markdown.ts +0 -99
  244. package/src/config/paths.ts +0 -167
  245. package/src/config/tui-migrate.ts +0 -155
  246. package/src/config/tui-schema.ts +0 -37
  247. package/src/config/tui.ts +0 -179
  248. package/src/config/validator.ts +0 -52
  249. package/src/control-plane/adaptors/index.ts +0 -20
  250. package/src/control-plane/adaptors/worktree.ts +0 -42
  251. package/src/control-plane/schema.ts +0 -17
  252. package/src/control-plane/sse.ts +0 -66
  253. package/src/control-plane/types.ts +0 -32
  254. package/src/control-plane/workspace.sql.ts +0 -17
  255. package/src/control-plane/workspace.ts +0 -168
  256. package/src/effect/cross-spawn-spawner.ts +0 -502
  257. package/src/effect/instance-ref.ts +0 -6
  258. package/src/effect/instance-registry.ts +0 -12
  259. package/src/effect/instance-state.ts +0 -82
  260. package/src/effect/run-service.ts +0 -33
  261. package/src/effect/runner.ts +0 -216
  262. package/src/env/index.ts +0 -28
  263. package/src/file/ignore.ts +0 -82
  264. package/src/file/index.ts +0 -686
  265. package/src/file/protected.ts +0 -59
  266. package/src/file/ripgrep.ts +0 -376
  267. package/src/file/time.ts +0 -133
  268. package/src/file/watcher.ts +0 -172
  269. package/src/filesystem/index.ts +0 -236
  270. package/src/flag/flag.ts +0 -157
  271. package/src/format/formatter.ts +0 -413
  272. package/src/format/index.ts +0 -203
  273. package/src/git/index.ts +0 -303
  274. package/src/global/index.ts +0 -54
  275. package/src/id/id.ts +0 -85
  276. package/src/ide/index.ts +0 -74
  277. package/src/index.ts +0 -253
  278. package/src/installation/index.ts +0 -355
  279. package/src/installation/meta.ts +0 -7
  280. package/src/lsp/client.ts +0 -256
  281. package/src/lsp/index.ts +0 -558
  282. package/src/lsp/language.ts +0 -120
  283. package/src/lsp/launch.ts +0 -21
  284. package/src/lsp/server.ts +0 -1968
  285. package/src/mcp/auth.ts +0 -173
  286. package/src/mcp/index.ts +0 -1250
  287. package/src/mcp/oauth-callback.ts +0 -216
  288. package/src/mcp/oauth-provider.ts +0 -185
  289. package/src/mcp/schema-loader.ts +0 -82
  290. package/src/node.ts +0 -1
  291. package/src/npm/index.ts +0 -188
  292. package/src/patch/index.ts +0 -680
  293. package/src/permission/arity.ts +0 -163
  294. package/src/permission/evaluate.ts +0 -15
  295. package/src/permission/index.ts +0 -323
  296. package/src/permission/schema.ts +0 -17
  297. package/src/plugin/cloudflare.ts +0 -67
  298. package/src/plugin/codex.ts +0 -608
  299. package/src/plugin/github-copilot/copilot.ts +0 -361
  300. package/src/plugin/github-copilot/models.ts +0 -144
  301. package/src/plugin/index.ts +0 -288
  302. package/src/plugin/install.ts +0 -439
  303. package/src/plugin/loader.ts +0 -174
  304. package/src/plugin/meta.ts +0 -188
  305. package/src/plugin/shared.ts +0 -323
  306. package/src/project/bootstrap.ts +0 -29
  307. package/src/project/instance.ts +0 -175
  308. package/src/project/project.sql.ts +0 -16
  309. package/src/project/project.ts +0 -519
  310. package/src/project/schema.ts +0 -16
  311. package/src/project/state.ts +0 -70
  312. package/src/project/vcs.ts +0 -240
  313. package/src/provider/auth.ts +0 -253
  314. package/src/provider/error.ts +0 -297
  315. package/src/provider/models.ts +0 -162
  316. package/src/provider/provider.ts +0 -1776
  317. package/src/provider/schema.ts +0 -38
  318. package/src/provider/sdk/copilot/README.md +0 -5
  319. package/src/provider/sdk/copilot/chat/convert-to-openai-compatible-chat-messages.ts +0 -170
  320. package/src/provider/sdk/copilot/chat/get-response-metadata.ts +0 -15
  321. package/src/provider/sdk/copilot/chat/map-openai-compatible-finish-reason.ts +0 -19
  322. package/src/provider/sdk/copilot/chat/openai-compatible-api-types.ts +0 -64
  323. package/src/provider/sdk/copilot/chat/openai-compatible-chat-language-model.ts +0 -814
  324. package/src/provider/sdk/copilot/chat/openai-compatible-chat-options.ts +0 -28
  325. package/src/provider/sdk/copilot/chat/openai-compatible-metadata-extractor.ts +0 -44
  326. package/src/provider/sdk/copilot/chat/openai-compatible-prepare-tools.ts +0 -83
  327. package/src/provider/sdk/copilot/copilot-provider.ts +0 -100
  328. package/src/provider/sdk/copilot/index.ts +0 -2
  329. package/src/provider/sdk/copilot/openai-compatible-error.ts +0 -27
  330. package/src/provider/sdk/copilot/responses/convert-to-openai-responses-input.ts +0 -335
  331. package/src/provider/sdk/copilot/responses/map-openai-responses-finish-reason.ts +0 -22
  332. package/src/provider/sdk/copilot/responses/openai-config.ts +0 -18
  333. package/src/provider/sdk/copilot/responses/openai-error.ts +0 -22
  334. package/src/provider/sdk/copilot/responses/openai-responses-api-types.ts +0 -214
  335. package/src/provider/sdk/copilot/responses/openai-responses-language-model.ts +0 -1769
  336. package/src/provider/sdk/copilot/responses/openai-responses-prepare-tools.ts +0 -173
  337. package/src/provider/sdk/copilot/responses/openai-responses-settings.ts +0 -1
  338. package/src/provider/sdk/copilot/responses/tool/code-interpreter.ts +0 -87
  339. package/src/provider/sdk/copilot/responses/tool/file-search.ts +0 -127
  340. package/src/provider/sdk/copilot/responses/tool/image-generation.ts +0 -114
  341. package/src/provider/sdk/copilot/responses/tool/local-shell.ts +0 -64
  342. package/src/provider/sdk/copilot/responses/tool/web-search-preview.ts +0 -103
  343. package/src/provider/sdk/copilot/responses/tool/web-search.ts +0 -102
  344. package/src/provider/transform.ts +0 -1124
  345. package/src/pty/index.ts +0 -397
  346. package/src/pty/pty.bun.ts +0 -26
  347. package/src/pty/pty.node.ts +0 -27
  348. package/src/pty/pty.ts +0 -25
  349. package/src/pty/schema.ts +0 -17
  350. package/src/question/index.ts +0 -224
  351. package/src/question/schema.ts +0 -17
  352. package/src/server/error.ts +0 -36
  353. package/src/server/event.ts +0 -7
  354. package/src/server/instance.ts +0 -315
  355. package/src/server/mdns.ts +0 -60
  356. package/src/server/middleware.ts +0 -33
  357. package/src/server/projectors.ts +0 -28
  358. package/src/server/proxy.ts +0 -130
  359. package/src/server/router.ts +0 -105
  360. package/src/server/routes/config.ts +0 -92
  361. package/src/server/routes/event.ts +0 -83
  362. package/src/server/routes/experimental.ts +0 -374
  363. package/src/server/routes/file.ts +0 -197
  364. package/src/server/routes/global.ts +0 -312
  365. package/src/server/routes/mcp.ts +0 -225
  366. package/src/server/routes/permission.ts +0 -69
  367. package/src/server/routes/project.ts +0 -118
  368. package/src/server/routes/provider.ts +0 -171
  369. package/src/server/routes/pty.ts +0 -210
  370. package/src/server/routes/question.ts +0 -99
  371. package/src/server/routes/session.ts +0 -984
  372. package/src/server/routes/tui.ts +0 -378
  373. package/src/server/routes/workspace.ts +0 -94
  374. package/src/server/server.ts +0 -353
  375. package/src/session/compaction.ts +0 -86
  376. package/src/session/index.ts +0 -904
  377. package/src/session/instruction.ts +0 -261
  378. package/src/session/llm/monitor.ts +0 -87
  379. package/src/session/llm.ts +0 -1676
  380. package/src/session/message-v2.ts +0 -1082
  381. package/src/session/message.ts +0 -191
  382. package/src/session/overflow.ts +0 -35
  383. package/src/session/processor.ts +0 -635
  384. package/src/session/projectors.ts +0 -136
  385. package/src/session/prompt/build-switch.txt +0 -5
  386. package/src/session/prompt/builder.ts +0 -135
  387. package/src/session/prompt/default.txt +0 -11
  388. package/src/session/prompt/engine.ts +0 -1072
  389. package/src/session/prompt/gemma4.txt +0 -1
  390. package/src/session/prompt/max-steps.txt +0 -16
  391. package/src/session/prompt/orchestrator.ts +0 -426
  392. package/src/session/prompt/plan.txt +0 -28
  393. package/src/session/prompt/qwen.txt +0 -19
  394. package/src/session/prompt/resolver.ts +0 -670
  395. package/src/session/prompt/router.ts +0 -197
  396. package/src/session/prompt/state.ts +0 -96
  397. package/src/session/prompt/types.ts +0 -115
  398. package/src/session/prompt/utils.ts +0 -15
  399. package/src/session/prompt.ts +0 -362
  400. package/src/session/retry.ts +0 -106
  401. package/src/session/revert.ts +0 -176
  402. package/src/session/sanitizer.ts +0 -125
  403. package/src/session/schema.ts +0 -38
  404. package/src/session/session.sql.ts +0 -106
  405. package/src/session/status.ts +0 -102
  406. package/src/session/summary.ts +0 -183
  407. package/src/session/system.ts +0 -79
  408. package/src/session/todo.ts +0 -166
  409. package/src/session/worker.ts +0 -382
  410. package/src/shell/shell.ts +0 -110
  411. package/src/skill/discovery.ts +0 -116
  412. package/src/skill/index.ts +0 -287
  413. package/src/snapshot/index.ts +0 -726
  414. package/src/sql.d.ts +0 -4
  415. package/src/storage/db.bun.ts +0 -8
  416. package/src/storage/db.node.ts +0 -8
  417. package/src/storage/db.ts +0 -174
  418. package/src/storage/json-migration.ts +0 -387
  419. package/src/storage/schema.sql.ts +0 -10
  420. package/src/storage/schema.ts +0 -4
  421. package/src/storage/storage.ts +0 -353
  422. package/src/sync/README.md +0 -179
  423. package/src/sync/event.sql.ts +0 -16
  424. package/src/sync/index.ts +0 -263
  425. package/src/sync/schema.ts +0 -14
  426. package/src/tool/apply_patch.ts +0 -281
  427. package/src/tool/apply_patch.txt +0 -1
  428. package/src/tool/arbitration.txt +0 -5
  429. package/src/tool/bash.ts +0 -494
  430. package/src/tool/bash.txt +0 -2
  431. package/src/tool/batch.ts +0 -183
  432. package/src/tool/batch.txt +0 -1
  433. package/src/tool/codesearch.ts +0 -132
  434. package/src/tool/codesearch.txt +0 -1
  435. package/src/tool/edit.ts +0 -734
  436. package/src/tool/edit.txt +0 -1
  437. package/src/tool/external-directory.ts +0 -46
  438. package/src/tool/glob.ts +0 -73
  439. package/src/tool/glob.txt +0 -2
  440. package/src/tool/grep.ts +0 -156
  441. package/src/tool/grep.txt +0 -2
  442. package/src/tool/invalid.ts +0 -20
  443. package/src/tool/ls.ts +0 -121
  444. package/src/tool/ls.txt +0 -1
  445. package/src/tool/lsp.ts +0 -97
  446. package/src/tool/lsp.txt +0 -1
  447. package/src/tool/multiedit.ts +0 -46
  448. package/src/tool/multiedit.txt +0 -1
  449. package/src/tool/plan-enter.txt +0 -14
  450. package/src/tool/plan-exit.txt +0 -13
  451. package/src/tool/plan.ts +0 -131
  452. package/src/tool/question.ts +0 -46
  453. package/src/tool/question.txt +0 -10
  454. package/src/tool/read.ts +0 -332
  455. package/src/tool/read.txt +0 -1
  456. package/src/tool/registry.ts +0 -288
  457. package/src/tool/revert.ts +0 -37
  458. package/src/tool/schema.ts +0 -17
  459. package/src/tool/skill.ts +0 -105
  460. package/src/tool/task.ts +0 -150
  461. package/src/tool/task.txt +0 -3
  462. package/src/tool/task_complete.ts +0 -21
  463. package/src/tool/tool.ts +0 -112
  464. package/src/tool/truncate.ts +0 -144
  465. package/src/tool/truncation-dir.ts +0 -4
  466. package/src/tool/webfetch.ts +0 -206
  467. package/src/tool/webfetch.txt +0 -1
  468. package/src/tool/websearch.ts +0 -150
  469. package/src/tool/websearch.txt +0 -1
  470. package/src/tool/write.ts +0 -101
  471. package/src/tool/write.txt +0 -1
  472. package/src/util/abort.ts +0 -35
  473. package/src/util/ai-sdk.ts +0 -59
  474. package/src/util/archive.ts +0 -17
  475. package/src/util/color.ts +0 -19
  476. package/src/util/context.ts +0 -25
  477. package/src/util/data-url.ts +0 -9
  478. package/src/util/defer.ts +0 -12
  479. package/src/util/effect-http-client.ts +0 -11
  480. package/src/util/effect-zod.ts +0 -98
  481. package/src/util/error.ts +0 -77
  482. package/src/util/filesystem.ts +0 -245
  483. package/src/util/flock.ts +0 -333
  484. package/src/util/fn.ts +0 -21
  485. package/src/util/format.ts +0 -20
  486. package/src/util/glob.ts +0 -34
  487. package/src/util/hash.ts +0 -7
  488. package/src/util/iife.ts +0 -3
  489. package/src/util/keybind.ts +0 -103
  490. package/src/util/lazy.ts +0 -23
  491. package/src/util/locale.ts +0 -81
  492. package/src/util/lock.ts +0 -98
  493. package/src/util/log-parser.ts +0 -114
  494. package/src/util/log.ts +0 -250
  495. package/src/util/network.ts +0 -23
  496. package/src/util/process.ts +0 -176
  497. package/src/util/queue.ts +0 -32
  498. package/src/util/record.ts +0 -3
  499. package/src/util/rpc.ts +0 -66
  500. package/src/util/schema.ts +0 -53
  501. package/src/util/scrap.ts +0 -10
  502. package/src/util/session-analyzer.ts +0 -331
  503. package/src/util/session-telemetry.ts +0 -91
  504. package/src/util/signal.ts +0 -12
  505. package/src/util/timeout.ts +0 -14
  506. package/src/util/token.ts +0 -7
  507. package/src/util/tokenizer.ts +0 -50
  508. package/src/util/toon.ts +0 -45
  509. package/src/util/update-schema.ts +0 -13
  510. package/src/util/which.ts +0 -14
  511. package/src/util/wildcard.ts +0 -59
  512. package/src/worktree/index.ts +0 -612
  513. package/sst-env.d.ts +0 -10
  514. package/test/AGENTS.md +0 -81
  515. package/test/account/repo.test.ts +0 -326
  516. package/test/account/service.test.ts +0 -393
  517. package/test/acp/agent-interface.test.ts +0 -51
  518. package/test/acp/event-subscription.test.ts +0 -685
  519. package/test/agent/agent.test.ts +0 -716
  520. package/test/auth/auth.test.ts +0 -58
  521. package/test/bus/bus-effect.test.ts +0 -164
  522. package/test/bus/bus-integration.test.ts +0 -87
  523. package/test/bus/bus.test.ts +0 -219
  524. package/test/cli/account.test.ts +0 -26
  525. package/test/cli/cmd/tui/prompt-part.test.ts +0 -47
  526. package/test/cli/github-action.test.ts +0 -198
  527. package/test/cli/github-remote.test.ts +0 -80
  528. package/test/cli/plugin-auth-picker.test.ts +0 -120
  529. package/test/cli/tui/keybind-plugin.test.ts +0 -90
  530. package/test/cli/tui/plugin-add.test.ts +0 -107
  531. package/test/cli/tui/plugin-install.test.ts +0 -89
  532. package/test/cli/tui/plugin-lifecycle.test.ts +0 -225
  533. package/test/cli/tui/plugin-loader-entrypoint.test.ts +0 -492
  534. package/test/cli/tui/plugin-loader-pure.test.ts +0 -72
  535. package/test/cli/tui/plugin-loader.test.ts +0 -752
  536. package/test/cli/tui/plugin-toggle.test.ts +0 -159
  537. package/test/cli/tui/slot-replace.test.tsx +0 -47
  538. package/test/cli/tui/theme-store.test.ts +0 -51
  539. package/test/cli/tui/thread.test.ts +0 -128
  540. package/test/cli/tui/transcript.test.ts +0 -426
  541. package/test/config/agent-color.test.ts +0 -71
  542. package/test/config/config.test.ts +0 -2337
  543. package/test/config/fixtures/empty-frontmatter.md +0 -4
  544. package/test/config/fixtures/frontmatter.md +0 -28
  545. package/test/config/fixtures/markdown-header.md +0 -11
  546. package/test/config/fixtures/no-frontmatter.md +0 -1
  547. package/test/config/fixtures/weird-model-id.md +0 -13
  548. package/test/config/markdown.test.ts +0 -228
  549. package/test/config/tui.test.ts +0 -800
  550. package/test/control-plane/sse.test.ts +0 -56
  551. package/test/effect/cross-spawn-spawner.test.ts +0 -412
  552. package/test/effect/instance-state.test.ts +0 -482
  553. package/test/effect/run-service.test.ts +0 -46
  554. package/test/effect/runner.test.ts +0 -523
  555. package/test/fake/provider.ts +0 -82
  556. package/test/file/fsmonitor.test.ts +0 -62
  557. package/test/file/ignore.test.ts +0 -10
  558. package/test/file/index.test.ts +0 -946
  559. package/test/file/path-traversal.test.ts +0 -198
  560. package/test/file/ripgrep.test.ts +0 -54
  561. package/test/file/time.test.ts +0 -445
  562. package/test/file/watcher.test.ts +0 -247
  563. package/test/filesystem/filesystem.test.ts +0 -319
  564. package/test/fixture/db.ts +0 -11
  565. package/test/fixture/fixture.test.ts +0 -26
  566. package/test/fixture/fixture.ts +0 -172
  567. package/test/fixture/flock-worker.ts +0 -72
  568. package/test/fixture/lsp/fake-lsp-server.js +0 -77
  569. package/test/fixture/plug-worker.ts +0 -93
  570. package/test/fixture/plugin-meta-worker.ts +0 -26
  571. package/test/fixture/skills/agents-sdk/SKILL.md +0 -152
  572. package/test/fixture/skills/agents-sdk/references/callable.md +0 -92
  573. package/test/fixture/skills/cloudflare/SKILL.md +0 -211
  574. package/test/fixture/skills/index.json +0 -6
  575. package/test/fixture/tui-plugin.ts +0 -328
  576. package/test/fixture/tui-runtime.ts +0 -27
  577. package/test/format/format.test.ts +0 -171
  578. package/test/git/git.test.ts +0 -128
  579. package/test/ide/ide.test.ts +0 -82
  580. package/test/installation/installation.test.ts +0 -152
  581. package/test/keybind.test.ts +0 -421
  582. package/test/lib/effect.ts +0 -53
  583. package/test/lib/filesystem.ts +0 -10
  584. package/test/lib/llm-server.ts +0 -794
  585. package/test/lsp/client.test.ts +0 -95
  586. package/test/lsp/index.test.ts +0 -133
  587. package/test/lsp/launch.test.ts +0 -22
  588. package/test/lsp/lifecycle.test.ts +0 -147
  589. package/test/mcp/headers.test.ts +0 -153
  590. package/test/mcp/lifecycle.test.ts +0 -750
  591. package/test/mcp/oauth-auto-connect.test.ts +0 -199
  592. package/test/mcp/oauth-browser.test.ts +0 -249
  593. package/test/mcp/sc-approve-validator.test.ts +0 -431
  594. package/test/memory/abort-leak.test.ts +0 -137
  595. package/test/npm.test.ts +0 -18
  596. package/test/patch/patch.test.ts +0 -348
  597. package/test/permission/arity.test.ts +0 -33
  598. package/test/permission/next.test.ts +0 -1123
  599. package/test/permission-task.test.ts +0 -323
  600. package/test/plugin/auth-override.test.ts +0 -74
  601. package/test/plugin/codex.test.ts +0 -123
  602. package/test/plugin/github-copilot-models.test.ts +0 -117
  603. package/test/plugin/install-concurrency.test.ts +0 -140
  604. package/test/plugin/install.test.ts +0 -570
  605. package/test/plugin/loader-shared.test.ts +0 -1136
  606. package/test/plugin/meta.test.ts +0 -137
  607. package/test/plugin/shared.test.ts +0 -88
  608. package/test/plugin/trigger.test.ts +0 -111
  609. package/test/preload.ts +0 -90
  610. package/test/project/migrate-global.test.ts +0 -140
  611. package/test/project/project.test.ts +0 -459
  612. package/test/project/state.test.ts +0 -115
  613. package/test/project/vcs.test.ts +0 -228
  614. package/test/project/worktree-remove.test.ts +0 -96
  615. package/test/project/worktree.test.ts +0 -173
  616. package/test/provider/amazon-bedrock.test.ts +0 -447
  617. package/test/provider/copilot/convert-to-copilot-messages.test.ts +0 -523
  618. package/test/provider/copilot/copilot-chat-model.test.ts +0 -592
  619. package/test/provider/error.test.ts +0 -49
  620. package/test/provider/gitlab-duo.test.ts +0 -412
  621. package/test/provider/provider.test.ts +0 -2494
  622. package/test/provider/transform.test.ts +0 -2944
  623. package/test/pty/pty-output-isolation.test.ts +0 -141
  624. package/test/pty/pty-session.test.ts +0 -92
  625. package/test/pty/pty-shell.test.ts +0 -59
  626. package/test/question/question.test.ts +0 -453
  627. package/test/server/global-session-list.test.ts +0 -89
  628. package/test/server/project-init-git.test.ts +0 -121
  629. package/test/server/session-actions.test.ts +0 -83
  630. package/test/server/session-list.test.ts +0 -98
  631. package/test/server/session-messages.test.ts +0 -159
  632. package/test/server/session-select.test.ts +0 -84
  633. package/test/session/compaction.test.ts +0 -683
  634. package/test/session/continuity-handover.test.ts +0 -620
  635. package/test/session/deterministic-handover.test.ts +0 -328
  636. package/test/session/doom-protection.test.ts +0 -247
  637. package/test/session/hard-reset.test.ts +0 -179
  638. package/test/session/instruction.test.ts +0 -286
  639. package/test/session/llm/monitor.test.ts +0 -53
  640. package/test/session/llm-sanitizer.test.ts +0 -90
  641. package/test/session/llm-zones-e2e.test.ts +0 -61
  642. package/test/session/llm.test.ts +0 -1308
  643. package/test/session/mcpx-normalization.test.ts +0 -86
  644. package/test/session/mcpx-syntax-recovery.test.ts +0 -28
  645. package/test/session/message-v2.test.ts +0 -957
  646. package/test/session/messages-pagination.test.ts +0 -885
  647. package/test/session/processor-effect.test.ts +0 -805
  648. package/test/session/prompt/builder.test.ts +0 -71
  649. package/test/session/prompt/engine-loop.test.ts +0 -80
  650. package/test/session/prompt/orchestrator.test.ts +0 -108
  651. package/test/session/prompt/resolver.test.ts +0 -211
  652. package/test/session/prompt/router.test.ts +0 -84
  653. package/test/session/prompt/state.test.ts +0 -57
  654. package/test/session/prompt-effect.test.ts +0 -1241
  655. package/test/session/prompt.test.ts +0 -522
  656. package/test/session/refactor-system-zones.test.ts +0 -241
  657. package/test/session/retry.test.ts +0 -232
  658. package/test/session/revert-compact.test.ts +0 -621
  659. package/test/session/sanitizer.test.ts +0 -61
  660. package/test/session/session.test.ts +0 -142
  661. package/test/session/snapshot-tool-race.test.ts +0 -242
  662. package/test/session/structured-output-integration.test.ts +0 -233
  663. package/test/session/structured-output.test.ts +0 -391
  664. package/test/session/system.test.ts +0 -59
  665. package/test/session/telemetry.test.ts +0 -35
  666. package/test/shell/shell.test.ts +0 -73
  667. package/test/skill/discovery.test.ts +0 -116
  668. package/test/skill/skill.test.ts +0 -392
  669. package/test/snapshot/snapshot.test.ts +0 -1404
  670. package/test/storage/db.test.ts +0 -14
  671. package/test/storage/json-migration.test.ts +0 -791
  672. package/test/storage/storage.test.ts +0 -295
  673. package/test/sync/index.test.ts +0 -191
  674. package/test/tool/__snapshots__/tool.test.ts.snap +0 -9
  675. package/test/tool/apply_patch.test.ts +0 -567
  676. package/test/tool/bash.test.ts +0 -1099
  677. package/test/tool/edit.test.ts +0 -681
  678. package/test/tool/external-directory.test.ts +0 -198
  679. package/test/tool/fixtures/large-image.png +0 -0
  680. package/test/tool/fixtures/models-api.json +0 -65179
  681. package/test/tool/grep.test.ts +0 -111
  682. package/test/tool/question.test.ts +0 -126
  683. package/test/tool/read.test.ts +0 -468
  684. package/test/tool/registry.test.ts +0 -126
  685. package/test/tool/skill.test.ts +0 -167
  686. package/test/tool/task.test.ts +0 -49
  687. package/test/tool/tool-define.test.ts +0 -101
  688. package/test/tool/truncation.test.ts +0 -161
  689. package/test/tool/webfetch.test.ts +0 -101
  690. package/test/tool/write.test.ts +0 -354
  691. package/test/util/data-url.test.ts +0 -14
  692. package/test/util/effect-zod.test.ts +0 -61
  693. package/test/util/error.test.ts +0 -38
  694. package/test/util/filesystem.test.ts +0 -656
  695. package/test/util/flock.test.ts +0 -383
  696. package/test/util/format.test.ts +0 -59
  697. package/test/util/glob.test.ts +0 -164
  698. package/test/util/iife.test.ts +0 -36
  699. package/test/util/lazy.test.ts +0 -50
  700. package/test/util/lock.test.ts +0 -72
  701. package/test/util/log-parser.test.ts +0 -61
  702. package/test/util/module.test.ts +0 -59
  703. package/test/util/process.test.ts +0 -128
  704. package/test/util/telemetry-integration.test.ts +0 -104
  705. package/test/util/timeout.test.ts +0 -21
  706. package/test/util/which.test.ts +0 -100
  707. package/test/util/wildcard.test.ts +0 -90
  708. package/test-regex.js +0 -50
  709. package/tsconfig.json +0 -23
  710. /package/bin/{epochcli → epochcli.cjs} +0 -0
  711. /package/{script/postinstall.mjs → postinstall.mjs} +0 -0
@@ -1,1404 +0,0 @@
1
- import { afterEach, test, expect } from "bun:test"
2
- import { $ } from "bun"
3
- import fs from "fs/promises"
4
- import path from "path"
5
- import { Snapshot } from "../../src/snapshot"
6
- import { Instance } from "../../src/project/instance"
7
- import { Filesystem } from "../../src/util/filesystem"
8
- import { tmpdir } from "../fixture/fixture"
9
-
10
- // Git always outputs /-separated paths internally. Snapshot.patch() joins them
11
- // with path.join (which produces \ on Windows) then normalizes back to /.
12
- // This helper does the same for expected values so assertions match cross-platform.
13
- const fwd = (...parts: string[]) => path.join(...parts).replaceAll("\\", "/")
14
-
15
- afterEach(async () => {
16
- await Instance.disposeAll()
17
- })
18
-
19
- async function bootstrap() {
20
- return tmpdir({
21
- git: true,
22
- init: async (dir) => {
23
- const unique = Math.random().toString(36).slice(2)
24
- const aContent = `A${unique}`
25
- const bContent = `B${unique}`
26
- await Filesystem.write(`${dir}/a.txt`, aContent)
27
- await Filesystem.write(`${dir}/b.txt`, bContent)
28
- await $`git add .`.cwd(dir).quiet()
29
- await $`git commit --no-gpg-sign -m init`.cwd(dir).quiet()
30
- return {
31
- aContent,
32
- bContent,
33
- }
34
- },
35
- })
36
- }
37
-
38
- test("tracks deleted files correctly", async () => {
39
- await using tmp = await bootstrap()
40
- await Instance.provide({
41
- directory: tmp.path,
42
- fn: async () => {
43
- const before = await Snapshot.track()
44
- expect(before).toBeTruthy()
45
-
46
- await $`rm ${tmp.path}/a.txt`.quiet()
47
-
48
- expect((await Snapshot.patch(before!)).files).toContain(fwd(tmp.path, "a.txt"))
49
- },
50
- })
51
- })
52
-
53
- test("revert should remove new files", async () => {
54
- await using tmp = await bootstrap()
55
- await Instance.provide({
56
- directory: tmp.path,
57
- fn: async () => {
58
- const before = await Snapshot.track()
59
- expect(before).toBeTruthy()
60
-
61
- await Filesystem.write(`${tmp.path}/new.txt`, "NEW")
62
-
63
- await Snapshot.revert([await Snapshot.patch(before!)])
64
-
65
- expect(
66
- await fs
67
- .access(`${tmp.path}/new.txt`)
68
- .then(() => true)
69
- .catch(() => false),
70
- ).toBe(false)
71
- },
72
- })
73
- })
74
-
75
- test("revert in subdirectory", async () => {
76
- await using tmp = await bootstrap()
77
- await Instance.provide({
78
- directory: tmp.path,
79
- fn: async () => {
80
- const before = await Snapshot.track()
81
- expect(before).toBeTruthy()
82
-
83
- await $`mkdir -p ${tmp.path}/sub`.quiet()
84
- await Filesystem.write(`${tmp.path}/sub/file.txt`, "SUB")
85
-
86
- await Snapshot.revert([await Snapshot.patch(before!)])
87
-
88
- expect(
89
- await fs
90
- .access(`${tmp.path}/sub/file.txt`)
91
- .then(() => true)
92
- .catch(() => false),
93
- ).toBe(false)
94
- // Note: revert currently only removes files, not directories
95
- // The empty subdirectory will remain
96
- },
97
- })
98
- })
99
-
100
- test("multiple file operations", async () => {
101
- await using tmp = await bootstrap()
102
- await Instance.provide({
103
- directory: tmp.path,
104
- fn: async () => {
105
- const before = await Snapshot.track()
106
- expect(before).toBeTruthy()
107
-
108
- await $`rm ${tmp.path}/a.txt`.quiet()
109
- await Filesystem.write(`${tmp.path}/c.txt`, "C")
110
- await $`mkdir -p ${tmp.path}/dir`.quiet()
111
- await Filesystem.write(`${tmp.path}/dir/d.txt`, "D")
112
- await Filesystem.write(`${tmp.path}/b.txt`, "MODIFIED")
113
-
114
- await Snapshot.revert([await Snapshot.patch(before!)])
115
-
116
- expect(await fs.readFile(`${tmp.path}/a.txt`, "utf-8")).toBe(tmp.extra.aContent)
117
- expect(
118
- await fs
119
- .access(`${tmp.path}/c.txt`)
120
- .then(() => true)
121
- .catch(() => false),
122
- ).toBe(false)
123
- // Note: revert currently only removes files, not directories
124
- // The empty directory will remain
125
- expect(await fs.readFile(`${tmp.path}/b.txt`, "utf-8")).toBe(tmp.extra.bContent)
126
- },
127
- })
128
- })
129
-
130
- test("empty directory handling", async () => {
131
- await using tmp = await bootstrap()
132
- await Instance.provide({
133
- directory: tmp.path,
134
- fn: async () => {
135
- const before = await Snapshot.track()
136
- expect(before).toBeTruthy()
137
-
138
- await $`mkdir ${tmp.path}/empty`.quiet()
139
-
140
- expect((await Snapshot.patch(before!)).files.length).toBe(0)
141
- },
142
- })
143
- })
144
-
145
- test("binary file handling", async () => {
146
- await using tmp = await bootstrap()
147
- await Instance.provide({
148
- directory: tmp.path,
149
- fn: async () => {
150
- const before = await Snapshot.track()
151
- expect(before).toBeTruthy()
152
-
153
- await Filesystem.write(`${tmp.path}/image.png`, new Uint8Array([0x89, 0x50, 0x4e, 0x47]))
154
-
155
- const patch = await Snapshot.patch(before!)
156
- expect(patch.files).toContain(fwd(tmp.path, "image.png"))
157
-
158
- await Snapshot.revert([patch])
159
- expect(
160
- await fs
161
- .access(`${tmp.path}/image.png`)
162
- .then(() => true)
163
- .catch(() => false),
164
- ).toBe(false)
165
- },
166
- })
167
- })
168
-
169
- test("symlink handling", async () => {
170
- await using tmp = await bootstrap()
171
- await Instance.provide({
172
- directory: tmp.path,
173
- fn: async () => {
174
- const before = await Snapshot.track()
175
- expect(before).toBeTruthy()
176
-
177
- await fs.symlink(`${tmp.path}/a.txt`, `${tmp.path}/link.txt`, "file")
178
-
179
- expect((await Snapshot.patch(before!)).files).toContain(fwd(tmp.path, "link.txt"))
180
- },
181
- })
182
- })
183
-
184
- test("file under size limit handling", async () => {
185
- await using tmp = await bootstrap()
186
- await Instance.provide({
187
- directory: tmp.path,
188
- fn: async () => {
189
- const before = await Snapshot.track()
190
- expect(before).toBeTruthy()
191
-
192
- await Filesystem.write(`${tmp.path}/large.txt`, "x".repeat(1024 * 1024))
193
-
194
- expect((await Snapshot.patch(before!)).files).toContain(fwd(tmp.path, "large.txt"))
195
- },
196
- })
197
- })
198
-
199
- test("large added files are skipped", async () => {
200
- await using tmp = await bootstrap()
201
- await Instance.provide({
202
- directory: tmp.path,
203
- fn: async () => {
204
- const before = await Snapshot.track()
205
- expect(before).toBeTruthy()
206
-
207
- await Filesystem.write(`${tmp.path}/huge.txt`, new Uint8Array(2 * 1024 * 1024 + 1))
208
-
209
- expect((await Snapshot.patch(before!)).files).toEqual([])
210
- expect(await Snapshot.diff(before!)).toBe("")
211
- expect(await Snapshot.track()).toBe(before)
212
- },
213
- })
214
- })
215
-
216
- test("nested directory revert", async () => {
217
- await using tmp = await bootstrap()
218
- await Instance.provide({
219
- directory: tmp.path,
220
- fn: async () => {
221
- const before = await Snapshot.track()
222
- expect(before).toBeTruthy()
223
-
224
- await $`mkdir -p ${tmp.path}/level1/level2/level3`.quiet()
225
- await Filesystem.write(`${tmp.path}/level1/level2/level3/deep.txt`, "DEEP")
226
-
227
- await Snapshot.revert([await Snapshot.patch(before!)])
228
-
229
- expect(
230
- await fs
231
- .access(`${tmp.path}/level1/level2/level3/deep.txt`)
232
- .then(() => true)
233
- .catch(() => false),
234
- ).toBe(false)
235
- },
236
- })
237
- })
238
-
239
- test("special characters in filenames", async () => {
240
- await using tmp = await bootstrap()
241
- await Instance.provide({
242
- directory: tmp.path,
243
- fn: async () => {
244
- const before = await Snapshot.track()
245
- expect(before).toBeTruthy()
246
-
247
- await Filesystem.write(`${tmp.path}/file with spaces.txt`, "SPACES")
248
- await Filesystem.write(`${tmp.path}/file-with-dashes.txt`, "DASHES")
249
- await Filesystem.write(`${tmp.path}/file_with_underscores.txt`, "UNDERSCORES")
250
-
251
- const files = (await Snapshot.patch(before!)).files
252
- expect(files).toContain(fwd(tmp.path, "file with spaces.txt"))
253
- expect(files).toContain(fwd(tmp.path, "file-with-dashes.txt"))
254
- expect(files).toContain(fwd(tmp.path, "file_with_underscores.txt"))
255
- },
256
- })
257
- })
258
-
259
- test("revert with empty patches", async () => {
260
- await using tmp = await bootstrap()
261
- await Instance.provide({
262
- directory: tmp.path,
263
- fn: async () => {
264
- // Should not crash with empty patches
265
- expect(Snapshot.revert([])).resolves.toBeUndefined()
266
-
267
- // Should not crash with patches that have empty file lists
268
- expect(Snapshot.revert([{ hash: "dummy", files: [] }])).resolves.toBeUndefined()
269
- },
270
- })
271
- })
272
-
273
- test("patch with invalid hash", async () => {
274
- await using tmp = await bootstrap()
275
- await Instance.provide({
276
- directory: tmp.path,
277
- fn: async () => {
278
- const before = await Snapshot.track()
279
- expect(before).toBeTruthy()
280
-
281
- // Create a change
282
- await Filesystem.write(`${tmp.path}/test.txt`, "TEST")
283
-
284
- // Try to patch with invalid hash - should handle gracefully
285
- const patch = await Snapshot.patch("invalid-hash-12345")
286
- expect(patch.files).toEqual([])
287
- expect(patch.hash).toBe("invalid-hash-12345")
288
- },
289
- })
290
- })
291
-
292
- test("revert non-existent file", async () => {
293
- await using tmp = await bootstrap()
294
- await Instance.provide({
295
- directory: tmp.path,
296
- fn: async () => {
297
- const before = await Snapshot.track()
298
- expect(before).toBeTruthy()
299
-
300
- // Try to revert a file that doesn't exist in the snapshot
301
- // This should not crash
302
- expect(
303
- Snapshot.revert([
304
- {
305
- hash: before!,
306
- files: [`${tmp.path}/nonexistent.txt`],
307
- },
308
- ]),
309
- ).resolves.toBeUndefined()
310
- },
311
- })
312
- })
313
-
314
- test("unicode filenames", async () => {
315
- await using tmp = await bootstrap()
316
- await Instance.provide({
317
- directory: tmp.path,
318
- fn: async () => {
319
- const before = await Snapshot.track()
320
- expect(before).toBeTruthy()
321
-
322
- const unicodeFiles = [
323
- { path: fwd(tmp.path, "文件.txt"), content: "chinese content" },
324
- { path: fwd(tmp.path, "🚀rocket.txt"), content: "emoji content" },
325
- { path: fwd(tmp.path, "café.txt"), content: "accented content" },
326
- { path: fwd(tmp.path, "файл.txt"), content: "cyrillic content" },
327
- ]
328
-
329
- for (const file of unicodeFiles) {
330
- await Filesystem.write(file.path, file.content)
331
- }
332
-
333
- const patch = await Snapshot.patch(before!)
334
- expect(patch.files.length).toBe(4)
335
-
336
- for (const file of unicodeFiles) {
337
- expect(patch.files).toContain(file.path)
338
- }
339
-
340
- await Snapshot.revert([patch])
341
-
342
- for (const file of unicodeFiles) {
343
- expect(
344
- await fs
345
- .access(file.path)
346
- .then(() => true)
347
- .catch(() => false),
348
- ).toBe(false)
349
- }
350
- },
351
- })
352
- })
353
-
354
- test.skip("unicode filenames modification and restore", async () => {
355
- await using tmp = await bootstrap()
356
- await Instance.provide({
357
- directory: tmp.path,
358
- fn: async () => {
359
- const chineseFile = fwd(tmp.path, "文件.txt")
360
- const cyrillicFile = fwd(tmp.path, "файл.txt")
361
-
362
- await Filesystem.write(chineseFile, "original chinese")
363
- await Filesystem.write(cyrillicFile, "original cyrillic")
364
-
365
- const before = await Snapshot.track()
366
- expect(before).toBeTruthy()
367
-
368
- await Filesystem.write(chineseFile, "modified chinese")
369
- await Filesystem.write(cyrillicFile, "modified cyrillic")
370
-
371
- const patch = await Snapshot.patch(before!)
372
- expect(patch.files).toContain(chineseFile)
373
- expect(patch.files).toContain(cyrillicFile)
374
-
375
- await Snapshot.revert([patch])
376
-
377
- expect(await fs.readFile(chineseFile, "utf-8")).toBe("original chinese")
378
- expect(await fs.readFile(cyrillicFile, "utf-8")).toBe("original cyrillic")
379
- },
380
- })
381
- })
382
-
383
- test("unicode filenames in subdirectories", async () => {
384
- await using tmp = await bootstrap()
385
- await Instance.provide({
386
- directory: tmp.path,
387
- fn: async () => {
388
- const before = await Snapshot.track()
389
- expect(before).toBeTruthy()
390
-
391
- await $`mkdir -p "${tmp.path}/目录/подкаталог"`.quiet()
392
- const deepFile = fwd(tmp.path, "目录", "подкаталог", "文件.txt")
393
- await Filesystem.write(deepFile, "deep unicode content")
394
-
395
- const patch = await Snapshot.patch(before!)
396
- expect(patch.files).toContain(deepFile)
397
-
398
- await Snapshot.revert([patch])
399
- expect(
400
- await fs
401
- .access(deepFile)
402
- .then(() => true)
403
- .catch(() => false),
404
- ).toBe(false)
405
- },
406
- })
407
- })
408
-
409
- test("very long filenames", async () => {
410
- await using tmp = await bootstrap()
411
- await Instance.provide({
412
- directory: tmp.path,
413
- fn: async () => {
414
- const before = await Snapshot.track()
415
- expect(before).toBeTruthy()
416
-
417
- const longName = "a".repeat(200) + ".txt"
418
- const longFile = fwd(tmp.path, longName)
419
-
420
- await Filesystem.write(longFile, "long filename content")
421
-
422
- const patch = await Snapshot.patch(before!)
423
- expect(patch.files).toContain(longFile)
424
-
425
- await Snapshot.revert([patch])
426
- expect(
427
- await fs
428
- .access(longFile)
429
- .then(() => true)
430
- .catch(() => false),
431
- ).toBe(false)
432
- },
433
- })
434
- })
435
-
436
- test("hidden files", async () => {
437
- await using tmp = await bootstrap()
438
- await Instance.provide({
439
- directory: tmp.path,
440
- fn: async () => {
441
- const before = await Snapshot.track()
442
- expect(before).toBeTruthy()
443
-
444
- await Filesystem.write(`${tmp.path}/.hidden`, "hidden content")
445
- await Filesystem.write(`${tmp.path}/.gitignore`, "*.log")
446
- await Filesystem.write(`${tmp.path}/.config`, "config content")
447
-
448
- const patch = await Snapshot.patch(before!)
449
- expect(patch.files).toContain(fwd(tmp.path, ".hidden"))
450
- expect(patch.files).toContain(fwd(tmp.path, ".gitignore"))
451
- expect(patch.files).toContain(fwd(tmp.path, ".config"))
452
- },
453
- })
454
- })
455
-
456
- test("nested symlinks", async () => {
457
- await using tmp = await bootstrap()
458
- await Instance.provide({
459
- directory: tmp.path,
460
- fn: async () => {
461
- const before = await Snapshot.track()
462
- expect(before).toBeTruthy()
463
-
464
- await $`mkdir -p ${tmp.path}/sub/dir`.quiet()
465
- await Filesystem.write(`${tmp.path}/sub/dir/target.txt`, "target content")
466
- await fs.symlink(`${tmp.path}/sub/dir/target.txt`, `${tmp.path}/sub/dir/link.txt`, "file")
467
- await fs.symlink(`${tmp.path}/sub`, `${tmp.path}/sub-link`, "dir")
468
-
469
- const patch = await Snapshot.patch(before!)
470
- expect(patch.files).toContain(fwd(tmp.path, "sub", "dir", "link.txt"))
471
- expect(patch.files).toContain(fwd(tmp.path, "sub-link"))
472
- },
473
- })
474
- })
475
-
476
- test("file permissions and ownership changes", async () => {
477
- await using tmp = await bootstrap()
478
- await Instance.provide({
479
- directory: tmp.path,
480
- fn: async () => {
481
- const before = await Snapshot.track()
482
- expect(before).toBeTruthy()
483
-
484
- // Change permissions multiple times
485
- await $`chmod 600 ${tmp.path}/a.txt`.quiet()
486
- await $`chmod 755 ${tmp.path}/a.txt`.quiet()
487
- await $`chmod 644 ${tmp.path}/a.txt`.quiet()
488
-
489
- const patch = await Snapshot.patch(before!)
490
- // Note: git doesn't track permission changes on existing files by default
491
- // Only tracks executable bit when files are first added
492
- expect(patch.files.length).toBe(0)
493
- },
494
- })
495
- })
496
-
497
- test("circular symlinks", async () => {
498
- await using tmp = await bootstrap()
499
- await Instance.provide({
500
- directory: tmp.path,
501
- fn: async () => {
502
- const before = await Snapshot.track()
503
- expect(before).toBeTruthy()
504
-
505
- // Create circular symlink
506
- await fs.symlink(`${tmp.path}/circular`, `${tmp.path}/circular`, "dir").catch(() => {})
507
-
508
- const patch = await Snapshot.patch(before!)
509
- expect(patch.files.length).toBeGreaterThanOrEqual(0) // Should not crash
510
- },
511
- })
512
- })
513
-
514
- test("gitignore changes", async () => {
515
- await using tmp = await bootstrap()
516
- await Instance.provide({
517
- directory: tmp.path,
518
- fn: async () => {
519
- const before = await Snapshot.track()
520
- expect(before).toBeTruthy()
521
-
522
- await Filesystem.write(`${tmp.path}/.gitignore`, "*.ignored")
523
- await Filesystem.write(`${tmp.path}/test.ignored`, "ignored content")
524
- await Filesystem.write(`${tmp.path}/normal.txt`, "normal content")
525
-
526
- const patch = await Snapshot.patch(before!)
527
-
528
- // Should track gitignore itself
529
- expect(patch.files).toContain(fwd(tmp.path, ".gitignore"))
530
- // Should track normal files
531
- expect(patch.files).toContain(fwd(tmp.path, "normal.txt"))
532
- // Should not track ignored files (git won't see them)
533
- expect(patch.files).not.toContain(fwd(tmp.path, "test.ignored"))
534
- },
535
- })
536
- })
537
-
538
- test("git info exclude changes", async () => {
539
- await using tmp = await bootstrap()
540
- await Instance.provide({
541
- directory: tmp.path,
542
- fn: async () => {
543
- const before = await Snapshot.track()
544
- expect(before).toBeTruthy()
545
-
546
- const file = `${tmp.path}/.git/info/exclude`
547
- const text = await Bun.file(file).text()
548
- await Bun.write(file, `${text.trimEnd()}\nignored.txt\n`)
549
- await Bun.write(`${tmp.path}/ignored.txt`, "ignored content")
550
- await Bun.write(`${tmp.path}/normal.txt`, "normal content")
551
-
552
- const patch = await Snapshot.patch(before!)
553
- expect(patch.files).toContain(fwd(tmp.path, "normal.txt"))
554
- expect(patch.files).not.toContain(fwd(tmp.path, "ignored.txt"))
555
-
556
- const after = await Snapshot.track()
557
- const diffs = await Snapshot.diffFull(before!, after!)
558
- expect(diffs.some((x) => x.file === "normal.txt")).toBe(true)
559
- expect(diffs.some((x) => x.file === "ignored.txt")).toBe(false)
560
- },
561
- })
562
- })
563
-
564
- test("git info exclude keeps global excludes", async () => {
565
- await using tmp = await bootstrap()
566
- await Instance.provide({
567
- directory: tmp.path,
568
- fn: async () => {
569
- const global = `${tmp.path}/global.ignore`
570
- const config = `${tmp.path}/global.gitconfig`
571
- await Bun.write(global, "global.tmp\n")
572
- await Bun.write(config, `[core]\n\texcludesFile = ${global.replaceAll("\\", "/")}\n`)
573
-
574
- const prev = process.env.GIT_CONFIG_GLOBAL
575
- process.env.GIT_CONFIG_GLOBAL = config
576
- try {
577
- const before = await Snapshot.track()
578
- expect(before).toBeTruthy()
579
-
580
- const file = `${tmp.path}/.git/info/exclude`
581
- const text = await Bun.file(file).text()
582
- await Bun.write(file, `${text.trimEnd()}\ninfo.tmp\n`)
583
-
584
- await Bun.write(`${tmp.path}/global.tmp`, "global content")
585
- await Bun.write(`${tmp.path}/info.tmp`, "info content")
586
- await Bun.write(`${tmp.path}/normal.txt`, "normal content")
587
-
588
- const patch = await Snapshot.patch(before!)
589
- expect(patch.files).toContain(fwd(tmp.path, "normal.txt"))
590
- expect(patch.files).not.toContain(fwd(tmp.path, "global.tmp"))
591
- expect(patch.files).not.toContain(fwd(tmp.path, "info.tmp"))
592
- } finally {
593
- if (prev) process.env.GIT_CONFIG_GLOBAL = prev
594
- else delete process.env.GIT_CONFIG_GLOBAL
595
- }
596
- },
597
- })
598
- })
599
-
600
- test("concurrent file operations during patch", async () => {
601
- await using tmp = await bootstrap()
602
- await Instance.provide({
603
- directory: tmp.path,
604
- fn: async () => {
605
- const before = await Snapshot.track()
606
- expect(before).toBeTruthy()
607
-
608
- // Start creating files
609
- const createPromise = (async () => {
610
- for (let i = 0; i < 10; i++) {
611
- await Filesystem.write(`${tmp.path}/concurrent${i}.txt`, `concurrent${i}`)
612
- // Small delay to simulate concurrent operations
613
- await new Promise((resolve) => setTimeout(resolve, 1))
614
- }
615
- })()
616
-
617
- // Get patch while files are being created
618
- const patchPromise = Snapshot.patch(before!)
619
-
620
- await createPromise
621
- const patch = await patchPromise
622
-
623
- // Should capture some or all of the concurrent files
624
- expect(patch.files.length).toBeGreaterThanOrEqual(0)
625
- },
626
- })
627
- })
628
-
629
- test("snapshot state isolation between projects", async () => {
630
- // Test that different projects don't interfere with each other
631
- await using tmp1 = await bootstrap()
632
- await using tmp2 = await bootstrap()
633
-
634
- await Instance.provide({
635
- directory: tmp1.path,
636
- fn: async () => {
637
- const before1 = await Snapshot.track()
638
- await Filesystem.write(`${tmp1.path}/project1.txt`, "project1 content")
639
- const patch1 = await Snapshot.patch(before1!)
640
- expect(patch1.files).toContain(fwd(tmp1.path, "project1.txt"))
641
- },
642
- })
643
-
644
- await Instance.provide({
645
- directory: tmp2.path,
646
- fn: async () => {
647
- const before2 = await Snapshot.track()
648
- await Filesystem.write(`${tmp2.path}/project2.txt`, "project2 content")
649
- const patch2 = await Snapshot.patch(before2!)
650
- expect(patch2.files).toContain(fwd(tmp2.path, "project2.txt"))
651
-
652
- // Ensure project1 files don't appear in project2
653
- expect(patch2.files).not.toContain(fwd(tmp1?.path ?? "", "project1.txt"))
654
- },
655
- })
656
- })
657
-
658
- test("patch detects changes in secondary worktree", async () => {
659
- await using tmp = await bootstrap()
660
- const worktreePath = `${tmp.path}-worktree`
661
- await $`git worktree add ${worktreePath} HEAD`.cwd(tmp.path).quiet()
662
-
663
- try {
664
- await Instance.provide({
665
- directory: tmp.path,
666
- fn: async () => {
667
- expect(await Snapshot.track()).toBeTruthy()
668
- },
669
- })
670
-
671
- await Instance.provide({
672
- directory: worktreePath,
673
- fn: async () => {
674
- const before = await Snapshot.track()
675
- expect(before).toBeTruthy()
676
-
677
- const worktreeFile = fwd(worktreePath, "worktree.txt")
678
- await Filesystem.write(worktreeFile, "worktree content")
679
-
680
- const patch = await Snapshot.patch(before!)
681
- expect(patch.files).toContain(worktreeFile)
682
- },
683
- })
684
- } finally {
685
- await $`git worktree remove --force ${worktreePath}`.cwd(tmp.path).quiet().nothrow()
686
- await $`rm -rf ${worktreePath}`.quiet()
687
- }
688
- })
689
-
690
- test("revert only removes files in invoking worktree", async () => {
691
- await using tmp = await bootstrap()
692
- const worktreePath = `${tmp.path}-worktree`
693
- await $`git worktree add ${worktreePath} HEAD`.cwd(tmp.path).quiet()
694
-
695
- try {
696
- await Instance.provide({
697
- directory: tmp.path,
698
- fn: async () => {
699
- expect(await Snapshot.track()).toBeTruthy()
700
- },
701
- })
702
- const primaryFile = `${tmp.path}/worktree.txt`
703
- await Filesystem.write(primaryFile, "primary content")
704
-
705
- await Instance.provide({
706
- directory: worktreePath,
707
- fn: async () => {
708
- const before = await Snapshot.track()
709
- expect(before).toBeTruthy()
710
-
711
- const worktreeFile = fwd(worktreePath, "worktree.txt")
712
- await Filesystem.write(worktreeFile, "worktree content")
713
-
714
- const patch = await Snapshot.patch(before!)
715
- await Snapshot.revert([patch])
716
-
717
- expect(
718
- await fs
719
- .access(worktreeFile)
720
- .then(() => true)
721
- .catch(() => false),
722
- ).toBe(false)
723
- },
724
- })
725
-
726
- expect(await fs.readFile(primaryFile, "utf-8")).toBe("primary content")
727
- } finally {
728
- await $`git worktree remove --force ${worktreePath}`.cwd(tmp.path).quiet().nothrow()
729
- await $`rm -rf ${worktreePath}`.quiet()
730
- await $`rm -f ${tmp.path}/worktree.txt`.quiet()
731
- }
732
- })
733
-
734
- test("diff reports worktree-only/shared edits and ignores primary-only", async () => {
735
- await using tmp = await bootstrap()
736
- const worktreePath = `${tmp.path}-worktree`
737
- await $`git worktree add ${worktreePath} HEAD`.cwd(tmp.path).quiet()
738
-
739
- try {
740
- await Instance.provide({
741
- directory: tmp.path,
742
- fn: async () => {
743
- expect(await Snapshot.track()).toBeTruthy()
744
- },
745
- })
746
-
747
- await Instance.provide({
748
- directory: worktreePath,
749
- fn: async () => {
750
- const before = await Snapshot.track()
751
- expect(before).toBeTruthy()
752
-
753
- await Filesystem.write(`${worktreePath}/worktree-only.txt`, "worktree diff content")
754
- await Filesystem.write(`${worktreePath}/shared.txt`, "worktree edit")
755
- await Filesystem.write(`${tmp.path}/shared.txt`, "primary edit")
756
- await Filesystem.write(`${tmp.path}/primary-only.txt`, "primary change")
757
-
758
- const diff = await Snapshot.diff(before!)
759
- expect(diff).toContain("worktree-only.txt")
760
- expect(diff).toContain("shared.txt")
761
- expect(diff).not.toContain("primary-only.txt")
762
- },
763
- })
764
- } finally {
765
- await $`git worktree remove --force ${worktreePath}`.cwd(tmp.path).quiet().nothrow()
766
- await $`rm -rf ${worktreePath}`.quiet()
767
- await $`rm -f ${tmp.path}/shared.txt`.quiet()
768
- await $`rm -f ${tmp.path}/primary-only.txt`.quiet()
769
- }
770
- })
771
-
772
- test("track with no changes returns same hash", async () => {
773
- await using tmp = await bootstrap()
774
- await Instance.provide({
775
- directory: tmp.path,
776
- fn: async () => {
777
- const hash1 = await Snapshot.track()
778
- expect(hash1).toBeTruthy()
779
-
780
- // Track again with no changes
781
- const hash2 = await Snapshot.track()
782
- expect(hash2).toBe(hash1!)
783
-
784
- // Track again
785
- const hash3 = await Snapshot.track()
786
- expect(hash3).toBe(hash1!)
787
- },
788
- })
789
- })
790
-
791
- test("diff function with various changes", async () => {
792
- await using tmp = await bootstrap()
793
- await Instance.provide({
794
- directory: tmp.path,
795
- fn: async () => {
796
- const before = await Snapshot.track()
797
- expect(before).toBeTruthy()
798
-
799
- // Make various changes
800
- await $`rm ${tmp.path}/a.txt`.quiet()
801
- await Filesystem.write(`${tmp.path}/new.txt`, "new content")
802
- await Filesystem.write(`${tmp.path}/b.txt`, "modified content")
803
-
804
- const diff = await Snapshot.diff(before!)
805
- expect(diff).toContain("a.txt")
806
- expect(diff).toContain("b.txt")
807
- expect(diff).toContain("new.txt")
808
- },
809
- })
810
- })
811
-
812
- test("restore function", async () => {
813
- await using tmp = await bootstrap()
814
- await Instance.provide({
815
- directory: tmp.path,
816
- fn: async () => {
817
- const before = await Snapshot.track()
818
- expect(before).toBeTruthy()
819
-
820
- // Make changes
821
- await $`rm ${tmp.path}/a.txt`.quiet()
822
- await Filesystem.write(`${tmp.path}/new.txt`, "new content")
823
- await Filesystem.write(`${tmp.path}/b.txt`, "modified")
824
-
825
- // Restore to original state
826
- await Snapshot.restore(before!)
827
-
828
- expect(
829
- await fs
830
- .access(`${tmp.path}/a.txt`)
831
- .then(() => true)
832
- .catch(() => false),
833
- ).toBe(true)
834
- expect(await fs.readFile(`${tmp.path}/a.txt`, "utf-8")).toBe(tmp.extra.aContent)
835
- expect(
836
- await fs
837
- .access(`${tmp.path}/new.txt`)
838
- .then(() => true)
839
- .catch(() => false),
840
- ).toBe(true) // New files should remain
841
- expect(await fs.readFile(`${tmp.path}/b.txt`, "utf-8")).toBe(tmp.extra.bContent)
842
- },
843
- })
844
- })
845
-
846
- test("revert should not delete files that existed but were deleted in snapshot", async () => {
847
- await using tmp = await bootstrap()
848
- await Instance.provide({
849
- directory: tmp.path,
850
- fn: async () => {
851
- const snapshot1 = await Snapshot.track()
852
- expect(snapshot1).toBeTruthy()
853
-
854
- await $`rm ${tmp.path}/a.txt`.quiet()
855
-
856
- const snapshot2 = await Snapshot.track()
857
- expect(snapshot2).toBeTruthy()
858
-
859
- await Filesystem.write(`${tmp.path}/a.txt`, "recreated content")
860
-
861
- const patch = await Snapshot.patch(snapshot2!)
862
- expect(patch.files).toContain(fwd(tmp.path, "a.txt"))
863
-
864
- await Snapshot.revert([patch])
865
-
866
- expect(
867
- await fs
868
- .access(`${tmp.path}/a.txt`)
869
- .then(() => true)
870
- .catch(() => false),
871
- ).toBe(false)
872
- },
873
- })
874
- })
875
-
876
- test("revert preserves file that existed in snapshot when deleted then recreated", async () => {
877
- await using tmp = await bootstrap()
878
- await Instance.provide({
879
- directory: tmp.path,
880
- fn: async () => {
881
- await Filesystem.write(`${tmp.path}/existing.txt`, "original content")
882
-
883
- const snapshot = await Snapshot.track()
884
- expect(snapshot).toBeTruthy()
885
-
886
- await $`rm ${tmp.path}/existing.txt`.quiet()
887
- await Filesystem.write(`${tmp.path}/existing.txt`, "recreated")
888
- await Filesystem.write(`${tmp.path}/newfile.txt`, "new")
889
-
890
- const patch = await Snapshot.patch(snapshot!)
891
- expect(patch.files).toContain(fwd(tmp.path, "existing.txt"))
892
- expect(patch.files).toContain(fwd(tmp.path, "newfile.txt"))
893
-
894
- await Snapshot.revert([patch])
895
-
896
- expect(
897
- await fs
898
- .access(`${tmp.path}/newfile.txt`)
899
- .then(() => true)
900
- .catch(() => false),
901
- ).toBe(false)
902
- expect(
903
- await fs
904
- .access(`${tmp.path}/existing.txt`)
905
- .then(() => true)
906
- .catch(() => false),
907
- ).toBe(true)
908
- expect(await fs.readFile(`${tmp.path}/existing.txt`, "utf-8")).toBe("original content")
909
- },
910
- })
911
- })
912
-
913
- test("diffFull sets status based on git change type", async () => {
914
- await using tmp = await bootstrap()
915
- await Instance.provide({
916
- directory: tmp.path,
917
- fn: async () => {
918
- await Filesystem.write(`${tmp.path}/grow.txt`, "one\n")
919
- await Filesystem.write(`${tmp.path}/trim.txt`, "line1\nline2\n")
920
- await Filesystem.write(`${tmp.path}/delete.txt`, "gone")
921
-
922
- const before = await Snapshot.track()
923
- expect(before).toBeTruthy()
924
-
925
- await Filesystem.write(`${tmp.path}/grow.txt`, "one\ntwo\n")
926
- await Filesystem.write(`${tmp.path}/trim.txt`, "line1\n")
927
- await $`rm ${tmp.path}/delete.txt`.quiet()
928
- await Filesystem.write(`${tmp.path}/added.txt`, "new")
929
-
930
- const after = await Snapshot.track()
931
- expect(after).toBeTruthy()
932
-
933
- const diffs = await Snapshot.diffFull(before!, after!)
934
- expect(diffs.length).toBe(4)
935
-
936
- const added = diffs.find((d) => d.file === "added.txt")
937
- expect(added).toBeDefined()
938
- expect(added!.status).toBe("added")
939
-
940
- const deleted = diffs.find((d) => d.file === "delete.txt")
941
- expect(deleted).toBeDefined()
942
- expect(deleted!.status).toBe("deleted")
943
-
944
- const grow = diffs.find((d) => d.file === "grow.txt")
945
- expect(grow).toBeDefined()
946
- expect(grow!.status).toBe("modified")
947
- expect(grow!.additions).toBeGreaterThan(0)
948
- expect(grow!.deletions).toBe(0)
949
-
950
- const trim = diffs.find((d) => d.file === "trim.txt")
951
- expect(trim).toBeDefined()
952
- expect(trim!.status).toBe("modified")
953
- expect(trim!.additions).toBe(0)
954
- expect(trim!.deletions).toBeGreaterThan(0)
955
- },
956
- })
957
- })
958
-
959
- test("diffFull with new file additions", async () => {
960
- await using tmp = await bootstrap()
961
- await Instance.provide({
962
- directory: tmp.path,
963
- fn: async () => {
964
- const before = await Snapshot.track()
965
- expect(before).toBeTruthy()
966
-
967
- await Filesystem.write(`${tmp.path}/new.txt`, "new content")
968
-
969
- const after = await Snapshot.track()
970
- expect(after).toBeTruthy()
971
-
972
- const diffs = await Snapshot.diffFull(before!, after!)
973
- expect(diffs.length).toBe(1)
974
-
975
- const newFileDiff = diffs[0]
976
- expect(newFileDiff.file).toBe("new.txt")
977
- expect(newFileDiff.before).toBe("")
978
- expect(newFileDiff.after).toBe("new content")
979
- expect(newFileDiff.additions).toBe(1)
980
- expect(newFileDiff.deletions).toBe(0)
981
- },
982
- })
983
- })
984
-
985
- test("diffFull with a large interleaved mixed diff", async () => {
986
- await using tmp = await bootstrap()
987
- await Instance.provide({
988
- directory: tmp.path,
989
- fn: async () => {
990
- const ids = Array.from({ length: 60 }, (_, i) => i.toString().padStart(3, "0"))
991
- const mod = ids.map((id) => fwd(tmp.path, "mix", `${id}-mod.txt`))
992
- const del = ids.map((id) => fwd(tmp.path, "mix", `${id}-del.txt`))
993
- const add = ids.map((id) => fwd(tmp.path, "mix", `${id}-add.txt`))
994
- const bin = ids.map((id) => fwd(tmp.path, "mix", `${id}-bin.bin`))
995
-
996
- await $`mkdir -p ${tmp.path}/mix`.quiet()
997
- await Promise.all([
998
- ...mod.map((file, i) => Filesystem.write(file, `before-${ids[i]}-é\n🙂\nline`)),
999
- ...del.map((file, i) => Filesystem.write(file, `gone-${ids[i]}\n你好`)),
1000
- ...bin.map((file, i) => Filesystem.write(file, new Uint8Array([0, i, 255, i % 251]))),
1001
- ])
1002
-
1003
- const before = await Snapshot.track()
1004
- expect(before).toBeTruthy()
1005
-
1006
- await Promise.all([
1007
- ...mod.map((file, i) => Filesystem.write(file, `after-${ids[i]}-é\n🚀\nline`)),
1008
- ...add.map((file, i) => Filesystem.write(file, `new-${ids[i]}\nこんにちは`)),
1009
- ...bin.map((file, i) => Filesystem.write(file, new Uint8Array([9, i, 8, i % 251]))),
1010
- ...del.map((file) => fs.rm(file)),
1011
- ])
1012
-
1013
- const after = await Snapshot.track()
1014
- expect(after).toBeTruthy()
1015
-
1016
- const diffs = await Snapshot.diffFull(before!, after!)
1017
- expect(diffs).toHaveLength(ids.length * 4)
1018
-
1019
- const map = new Map(diffs.map((item) => [item.file, item]))
1020
- for (let i = 0; i < ids.length; i++) {
1021
- const m = map.get(fwd("mix", `${ids[i]}-mod.txt`))
1022
- expect(m).toBeDefined()
1023
- expect(m!.before).toBe(`before-${ids[i]}-é\n🙂\nline`)
1024
- expect(m!.after).toBe(`after-${ids[i]}-é\n🚀\nline`)
1025
- expect(m!.status).toBe("modified")
1026
-
1027
- const d = map.get(fwd("mix", `${ids[i]}-del.txt`))
1028
- expect(d).toBeDefined()
1029
- expect(d!.before).toBe(`gone-${ids[i]}\n你好`)
1030
- expect(d!.after).toBe("")
1031
- expect(d!.status).toBe("deleted")
1032
-
1033
- const a = map.get(fwd("mix", `${ids[i]}-add.txt`))
1034
- expect(a).toBeDefined()
1035
- expect(a!.before).toBe("")
1036
- expect(a!.after).toBe(`new-${ids[i]}\nこんにちは`)
1037
- expect(a!.status).toBe("added")
1038
-
1039
- const b = map.get(fwd("mix", `${ids[i]}-bin.bin`))
1040
- expect(b).toBeDefined()
1041
- expect(b!.before).toBe("")
1042
- expect(b!.after).toBe("")
1043
- expect(b!.additions).toBe(0)
1044
- expect(b!.deletions).toBe(0)
1045
- expect(b!.status).toBe("modified")
1046
- }
1047
- },
1048
- })
1049
- })
1050
-
1051
- test("diffFull preserves git diff order across batch boundaries", async () => {
1052
- await using tmp = await bootstrap()
1053
- await Instance.provide({
1054
- directory: tmp.path,
1055
- fn: async () => {
1056
- const ids = Array.from({ length: 140 }, (_, i) => i.toString().padStart(3, "0"))
1057
-
1058
- await $`mkdir -p ${tmp.path}/order`.quiet()
1059
- await Promise.all(ids.map((id) => Filesystem.write(`${tmp.path}/order/${id}.txt`, `before-${id}`)))
1060
-
1061
- const before = await Snapshot.track()
1062
- expect(before).toBeTruthy()
1063
-
1064
- await Promise.all(ids.map((id) => Filesystem.write(`${tmp.path}/order/${id}.txt`, `after-${id}`)))
1065
-
1066
- const after = await Snapshot.track()
1067
- expect(after).toBeTruthy()
1068
-
1069
- const expected = ids.map((id) => `order/${id}.txt`)
1070
-
1071
- const diffs = await Snapshot.diffFull(before!, after!)
1072
- expect(diffs.map((item) => item.file)).toEqual(expected)
1073
- },
1074
- })
1075
- })
1076
-
1077
- test("diffFull with file modifications", async () => {
1078
- await using tmp = await bootstrap()
1079
- await Instance.provide({
1080
- directory: tmp.path,
1081
- fn: async () => {
1082
- const before = await Snapshot.track()
1083
- expect(before).toBeTruthy()
1084
-
1085
- await Filesystem.write(`${tmp.path}/b.txt`, "modified content")
1086
-
1087
- const after = await Snapshot.track()
1088
- expect(after).toBeTruthy()
1089
-
1090
- const diffs = await Snapshot.diffFull(before!, after!)
1091
- expect(diffs.length).toBe(1)
1092
-
1093
- const modifiedFileDiff = diffs[0]
1094
- expect(modifiedFileDiff.file).toBe("b.txt")
1095
- expect(modifiedFileDiff.before).toBe(tmp.extra.bContent)
1096
- expect(modifiedFileDiff.after).toBe("modified content")
1097
- expect(modifiedFileDiff.additions).toBeGreaterThan(0)
1098
- expect(modifiedFileDiff.deletions).toBeGreaterThan(0)
1099
- },
1100
- })
1101
- })
1102
-
1103
- test("diffFull with file deletions", async () => {
1104
- await using tmp = await bootstrap()
1105
- await Instance.provide({
1106
- directory: tmp.path,
1107
- fn: async () => {
1108
- const before = await Snapshot.track()
1109
- expect(before).toBeTruthy()
1110
-
1111
- await $`rm ${tmp.path}/a.txt`.quiet()
1112
-
1113
- const after = await Snapshot.track()
1114
- expect(after).toBeTruthy()
1115
-
1116
- const diffs = await Snapshot.diffFull(before!, after!)
1117
- expect(diffs.length).toBe(1)
1118
-
1119
- const removedFileDiff = diffs[0]
1120
- expect(removedFileDiff.file).toBe("a.txt")
1121
- expect(removedFileDiff.before).toBe(tmp.extra.aContent)
1122
- expect(removedFileDiff.after).toBe("")
1123
- expect(removedFileDiff.additions).toBe(0)
1124
- expect(removedFileDiff.deletions).toBe(1)
1125
- },
1126
- })
1127
- })
1128
-
1129
- test("diffFull with multiple line additions", async () => {
1130
- await using tmp = await bootstrap()
1131
- await Instance.provide({
1132
- directory: tmp.path,
1133
- fn: async () => {
1134
- const before = await Snapshot.track()
1135
- expect(before).toBeTruthy()
1136
-
1137
- await Filesystem.write(`${tmp.path}/multi.txt`, "line1\nline2\nline3")
1138
-
1139
- const after = await Snapshot.track()
1140
- expect(after).toBeTruthy()
1141
-
1142
- const diffs = await Snapshot.diffFull(before!, after!)
1143
- expect(diffs.length).toBe(1)
1144
-
1145
- const multiDiff = diffs[0]
1146
- expect(multiDiff.file).toBe("multi.txt")
1147
- expect(multiDiff.before).toBe("")
1148
- expect(multiDiff.after).toBe("line1\nline2\nline3")
1149
- expect(multiDiff.additions).toBe(3)
1150
- expect(multiDiff.deletions).toBe(0)
1151
- },
1152
- })
1153
- })
1154
-
1155
- test("diffFull with addition and deletion", async () => {
1156
- await using tmp = await bootstrap()
1157
- await Instance.provide({
1158
- directory: tmp.path,
1159
- fn: async () => {
1160
- const before = await Snapshot.track()
1161
- expect(before).toBeTruthy()
1162
-
1163
- await Filesystem.write(`${tmp.path}/added.txt`, "added content")
1164
- await $`rm ${tmp.path}/a.txt`.quiet()
1165
-
1166
- const after = await Snapshot.track()
1167
- expect(after).toBeTruthy()
1168
-
1169
- const diffs = await Snapshot.diffFull(before!, after!)
1170
- expect(diffs.length).toBe(2)
1171
-
1172
- const addedFileDiff = diffs.find((d) => d.file === "added.txt")
1173
- expect(addedFileDiff).toBeDefined()
1174
- expect(addedFileDiff!.before).toBe("")
1175
- expect(addedFileDiff!.after).toBe("added content")
1176
- expect(addedFileDiff!.additions).toBe(1)
1177
- expect(addedFileDiff!.deletions).toBe(0)
1178
-
1179
- const removedFileDiff = diffs.find((d) => d.file === "a.txt")
1180
- expect(removedFileDiff).toBeDefined()
1181
- expect(removedFileDiff!.before).toBe(tmp.extra.aContent)
1182
- expect(removedFileDiff!.after).toBe("")
1183
- expect(removedFileDiff!.additions).toBe(0)
1184
- expect(removedFileDiff!.deletions).toBe(1)
1185
- },
1186
- })
1187
- })
1188
-
1189
- test("diffFull with multiple additions and deletions", async () => {
1190
- await using tmp = await bootstrap()
1191
- await Instance.provide({
1192
- directory: tmp.path,
1193
- fn: async () => {
1194
- const before = await Snapshot.track()
1195
- expect(before).toBeTruthy()
1196
-
1197
- await Filesystem.write(`${tmp.path}/multi1.txt`, "line1\nline2\nline3")
1198
- await Filesystem.write(`${tmp.path}/multi2.txt`, "single line")
1199
- await $`rm ${tmp.path}/a.txt`.quiet()
1200
- await $`rm ${tmp.path}/b.txt`.quiet()
1201
-
1202
- const after = await Snapshot.track()
1203
- expect(after).toBeTruthy()
1204
-
1205
- const diffs = await Snapshot.diffFull(before!, after!)
1206
- expect(diffs.length).toBe(4)
1207
-
1208
- const multi1Diff = diffs.find((d) => d.file === "multi1.txt")
1209
- expect(multi1Diff).toBeDefined()
1210
- expect(multi1Diff!.additions).toBe(3)
1211
- expect(multi1Diff!.deletions).toBe(0)
1212
-
1213
- const multi2Diff = diffs.find((d) => d.file === "multi2.txt")
1214
- expect(multi2Diff).toBeDefined()
1215
- expect(multi2Diff!.additions).toBe(1)
1216
- expect(multi2Diff!.deletions).toBe(0)
1217
-
1218
- const removedADiff = diffs.find((d) => d.file === "a.txt")
1219
- expect(removedADiff).toBeDefined()
1220
- expect(removedADiff!.additions).toBe(0)
1221
- expect(removedADiff!.deletions).toBe(1)
1222
-
1223
- const removedBDiff = diffs.find((d) => d.file === "b.txt")
1224
- expect(removedBDiff).toBeDefined()
1225
- expect(removedBDiff!.additions).toBe(0)
1226
- expect(removedBDiff!.deletions).toBe(1)
1227
- },
1228
- })
1229
- })
1230
-
1231
- test("diffFull with no changes", async () => {
1232
- await using tmp = await bootstrap()
1233
- await Instance.provide({
1234
- directory: tmp.path,
1235
- fn: async () => {
1236
- const before = await Snapshot.track()
1237
- expect(before).toBeTruthy()
1238
-
1239
- const after = await Snapshot.track()
1240
- expect(after).toBeTruthy()
1241
-
1242
- const diffs = await Snapshot.diffFull(before!, after!)
1243
- expect(diffs.length).toBe(0)
1244
- },
1245
- })
1246
- })
1247
-
1248
- test("diffFull with binary file changes", async () => {
1249
- await using tmp = await bootstrap()
1250
- await Instance.provide({
1251
- directory: tmp.path,
1252
- fn: async () => {
1253
- const before = await Snapshot.track()
1254
- expect(before).toBeTruthy()
1255
-
1256
- await Filesystem.write(`${tmp.path}/binary.bin`, new Uint8Array([0x00, 0x01, 0x02, 0x03]))
1257
-
1258
- const after = await Snapshot.track()
1259
- expect(after).toBeTruthy()
1260
-
1261
- const diffs = await Snapshot.diffFull(before!, after!)
1262
- expect(diffs.length).toBe(1)
1263
-
1264
- const binaryDiff = diffs[0]
1265
- expect(binaryDiff.file).toBe("binary.bin")
1266
- expect(binaryDiff.before).toBe("")
1267
- },
1268
- })
1269
- })
1270
-
1271
- test("diffFull with whitespace changes", async () => {
1272
- await using tmp = await bootstrap()
1273
- await Instance.provide({
1274
- directory: tmp.path,
1275
- fn: async () => {
1276
- await Filesystem.write(`${tmp.path}/whitespace.txt`, "line1\nline2")
1277
- const before = await Snapshot.track()
1278
- expect(before).toBeTruthy()
1279
-
1280
- await Filesystem.write(`${tmp.path}/whitespace.txt`, "line1\n\nline2\n")
1281
-
1282
- const after = await Snapshot.track()
1283
- expect(after).toBeTruthy()
1284
-
1285
- const diffs = await Snapshot.diffFull(before!, after!)
1286
- expect(diffs.length).toBe(1)
1287
-
1288
- const whitespaceDiff = diffs[0]
1289
- expect(whitespaceDiff.file).toBe("whitespace.txt")
1290
- expect(whitespaceDiff.additions).toBeGreaterThan(0)
1291
- },
1292
- })
1293
- })
1294
-
1295
- test("revert with overlapping files across patches uses first patch hash", async () => {
1296
- await using tmp = await bootstrap()
1297
- await Instance.provide({
1298
- directory: tmp.path,
1299
- fn: async () => {
1300
- // Write initial content and snapshot
1301
- await Filesystem.write(`${tmp.path}/shared.txt`, "v1")
1302
- const snap1 = await Snapshot.track()
1303
- expect(snap1).toBeTruthy()
1304
-
1305
- // Modify and snapshot again
1306
- await Filesystem.write(`${tmp.path}/shared.txt`, "v2")
1307
- const snap2 = await Snapshot.track()
1308
- expect(snap2).toBeTruthy()
1309
-
1310
- // Modify once more so both patches include shared.txt
1311
- await Filesystem.write(`${tmp.path}/shared.txt`, "v3")
1312
-
1313
- const patch1 = await Snapshot.patch(snap1!)
1314
- const patch2 = await Snapshot.patch(snap2!)
1315
-
1316
- // Both patches should include shared.txt
1317
- expect(patch1.files).toContain(fwd(tmp.path, "shared.txt"))
1318
- expect(patch2.files).toContain(fwd(tmp.path, "shared.txt"))
1319
-
1320
- // Revert with patch1 first — should use snap1's hash (restoring "v1")
1321
- await Snapshot.revert([patch1, patch2])
1322
-
1323
- const content = await fs.readFile(`${tmp.path}/shared.txt`, "utf-8")
1324
- expect(content).toBe("v1")
1325
- },
1326
- })
1327
- })
1328
-
1329
- test("revert preserves patch order when the same hash appears again", async () => {
1330
- await using tmp = await bootstrap()
1331
- await Instance.provide({
1332
- directory: tmp.path,
1333
- fn: async () => {
1334
- await $`mkdir -p ${tmp.path}/foo`.quiet()
1335
- await Filesystem.write(`${tmp.path}/foo/bar`, "v1")
1336
- await Filesystem.write(`${tmp.path}/a.txt`, "v1")
1337
-
1338
- const snap1 = await Snapshot.track()
1339
- expect(snap1).toBeTruthy()
1340
-
1341
- await $`rm -rf ${tmp.path}/foo`.quiet()
1342
- await Filesystem.write(`${tmp.path}/foo`, "v2")
1343
- await Filesystem.write(`${tmp.path}/a.txt`, "v2")
1344
-
1345
- const snap2 = await Snapshot.track()
1346
- expect(snap2).toBeTruthy()
1347
-
1348
- await $`rm -rf ${tmp.path}/foo`.quiet()
1349
- await Filesystem.write(`${tmp.path}/a.txt`, "v3")
1350
-
1351
- await Snapshot.revert([
1352
- { hash: snap1!, files: [fwd(tmp.path, "a.txt")] },
1353
- { hash: snap2!, files: [fwd(tmp.path, "foo")] },
1354
- { hash: snap1!, files: [fwd(tmp.path, "foo", "bar")] },
1355
- ])
1356
-
1357
- expect(await fs.readFile(`${tmp.path}/a.txt`, "utf-8")).toBe("v1")
1358
- expect((await fs.stat(`${tmp.path}/foo`)).isDirectory()).toBe(true)
1359
- expect(await fs.readFile(`${tmp.path}/foo/bar`, "utf-8")).toBe("v1")
1360
- },
1361
- })
1362
- })
1363
-
1364
- test("revert handles large mixed batches across chunk boundaries", async () => {
1365
- await using tmp = await bootstrap()
1366
- await Instance.provide({
1367
- directory: tmp.path,
1368
- fn: async () => {
1369
- const base = Array.from({ length: 140 }, (_, i) => fwd(tmp.path, "batch", `${i}.txt`))
1370
- const fresh = Array.from({ length: 140 }, (_, i) => fwd(tmp.path, "fresh", `${i}.txt`))
1371
-
1372
- await $`mkdir -p ${tmp.path}/batch ${tmp.path}/fresh`.quiet()
1373
- await Promise.all(base.map((file, i) => Filesystem.write(file, `base-${i}`)))
1374
-
1375
- const snap = await Snapshot.track()
1376
- expect(snap).toBeTruthy()
1377
-
1378
- await Promise.all(base.map((file, i) => Filesystem.write(file, `next-${i}`)))
1379
- await Promise.all(fresh.map((file, i) => Filesystem.write(file, `fresh-${i}`)))
1380
-
1381
- const patch = await Snapshot.patch(snap!)
1382
- expect(patch.files.length).toBe(base.length + fresh.length)
1383
-
1384
- await Snapshot.revert([patch])
1385
-
1386
- await Promise.all(
1387
- base.map(async (file, i) => {
1388
- expect(await fs.readFile(file, "utf-8")).toBe(`base-${i}`)
1389
- }),
1390
- )
1391
-
1392
- await Promise.all(
1393
- fresh.map(async (file) => {
1394
- expect(
1395
- await fs
1396
- .access(file)
1397
- .then(() => true)
1398
- .catch(() => false),
1399
- ).toBe(false)
1400
- }),
1401
- )
1402
- },
1403
- })
1404
- })