@vikasitai/vikasit-code 2.0.7 → 2.0.8

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 +27 -165
  3. package/AGENTS.md +0 -69
  4. package/BUN_SHELL_MIGRATION_PLAN.md +0 -136
  5. package/Dockerfile +0 -18
  6. package/README.md +0 -15
  7. package/bunfig.toml +0 -7
  8. package/drizzle.config.ts +0 -10
  9. package/git +0 -0
  10. package/migration/20260127222353_familiar_lady_ursula/migration.sql +0 -90
  11. package/migration/20260127222353_familiar_lady_ursula/snapshot.json +0 -796
  12. package/migration/20260211171708_add_project_commands/migration.sql +0 -1
  13. package/migration/20260211171708_add_project_commands/snapshot.json +0 -806
  14. package/migration/20260213144116_wakeful_the_professor/migration.sql +0 -11
  15. package/migration/20260213144116_wakeful_the_professor/snapshot.json +0 -897
  16. package/migration/20260225215848_workspace/migration.sql +0 -7
  17. package/migration/20260225215848_workspace/snapshot.json +0 -959
  18. package/migration/20260227213759_add_session_workspace_id/migration.sql +0 -2
  19. package/migration/20260227213759_add_session_workspace_id/snapshot.json +0 -983
  20. package/migration/20260228203230_blue_harpoon/migration.sql +0 -17
  21. package/migration/20260228203230_blue_harpoon/snapshot.json +0 -1102
  22. package/migration/20260303231226_add_workspace_fields/migration.sql +0 -5
  23. package/migration/20260303231226_add_workspace_fields/snapshot.json +0 -1013
  24. package/migration/20260309230000_move_org_to_state/migration.sql +0 -3
  25. package/migration/20260309230000_move_org_to_state/snapshot.json +0 -1156
  26. package/migration/20260312043431_session_message_cursor/migration.sql +0 -4
  27. package/migration/20260312043431_session_message_cursor/snapshot.json +0 -1168
  28. package/migration/20260323234822_events/migration.sql +0 -13
  29. package/migration/20260323234822_events/snapshot.json +0 -1271
  30. package/parsers-config.ts +0 -290
  31. package/script/build-all.sh +0 -40
  32. package/script/build-node.ts +0 -54
  33. package/script/build.ts +0 -281
  34. package/script/check-migrations.ts +0 -16
  35. package/script/fetch-models.ts +0 -101
  36. package/script/publish.ts +0 -184
  37. package/script/schema.ts +0 -63
  38. package/script/seed-e2e.ts +0 -75
  39. package/script/upgrade-opentui.ts +0 -64
  40. package/specs/claude-code-integration.md +0 -563
  41. package/specs/effect-migration.md +0 -291
  42. package/specs/tui-plugins.md +0 -410
  43. package/src/account/account.sql.ts +0 -39
  44. package/src/account/index.ts +0 -397
  45. package/src/account/repo.ts +0 -163
  46. package/src/account/schema.ts +0 -91
  47. package/src/acp/README.md +0 -174
  48. package/src/acp/agent.ts +0 -1743
  49. package/src/acp/session.ts +0 -116
  50. package/src/acp/types.ts +0 -24
  51. package/src/agent/agent-display.ts +0 -82
  52. package/src/agent/agent.ts +0 -475
  53. package/src/agent/extract-names.cjs +0 -42
  54. package/src/agent/generate.txt +0 -75
  55. package/src/agent/mailbox.ts +0 -124
  56. package/src/agent/orchestrator.ts +0 -170
  57. package/src/agent/prompt/code-reviewer.txt +0 -11
  58. package/src/agent/prompt/compaction.txt +0 -14
  59. package/src/agent/prompt/explore.txt +0 -18
  60. package/src/agent/prompt/general.txt +0 -17
  61. package/src/agent/prompt/summary.txt +0 -11
  62. package/src/agent/prompt/title.txt +0 -44
  63. package/src/agent/prompt/verification.txt +0 -11
  64. package/src/auth/index.ts +0 -109
  65. package/src/bun/index.ts +0 -129
  66. package/src/bun/registry.ts +0 -50
  67. package/src/bus/bus-event.ts +0 -40
  68. package/src/bus/global.ts +0 -10
  69. package/src/bus/index.ts +0 -184
  70. package/src/cli/bootstrap.ts +0 -17
  71. package/src/cli/cmd/account.ts +0 -257
  72. package/src/cli/cmd/acp.ts +0 -70
  73. package/src/cli/cmd/agent.ts +0 -245
  74. package/src/cli/cmd/auto-updater.ts +0 -71
  75. package/src/cli/cmd/cmd.ts +0 -7
  76. package/src/cli/cmd/db.ts +0 -119
  77. package/src/cli/cmd/debug/agent.ts +0 -167
  78. package/src/cli/cmd/debug/config.ts +0 -16
  79. package/src/cli/cmd/debug/file.ts +0 -97
  80. package/src/cli/cmd/debug/index.ts +0 -48
  81. package/src/cli/cmd/debug/lsp.ts +0 -53
  82. package/src/cli/cmd/debug/ripgrep.ts +0 -87
  83. package/src/cli/cmd/debug/scrap.ts +0 -16
  84. package/src/cli/cmd/debug/skill.ts +0 -16
  85. package/src/cli/cmd/debug/snapshot.ts +0 -52
  86. package/src/cli/cmd/export.ts +0 -89
  87. package/src/cli/cmd/generate.ts +0 -38
  88. package/src/cli/cmd/github.ts +0 -1646
  89. package/src/cli/cmd/import.ts +0 -207
  90. package/src/cli/cmd/mcp.ts +0 -754
  91. package/src/cli/cmd/models.ts +0 -78
  92. package/src/cli/cmd/plug.ts +0 -231
  93. package/src/cli/cmd/pr.ts +0 -127
  94. package/src/cli/cmd/providers.ts +0 -484
  95. package/src/cli/cmd/run.ts +0 -676
  96. package/src/cli/cmd/search.ts +0 -137
  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 -405
  100. package/src/cli/cmd/tui/app.tsx +0 -996
  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-login.tsx +0 -156
  106. package/src/cli/cmd/tui/component/dialog-mcp.tsx +0 -86
  107. package/src/cli/cmd/tui/component/dialog-model.tsx +0 -179
  108. package/src/cli/cmd/tui/component/dialog-provider.tsx +0 -329
  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 -91
  119. package/src/cli/cmd/tui/component/login-gate.tsx +0 -125
  120. package/src/cli/cmd/tui/component/logo.tsx +0 -85
  121. package/src/cli/cmd/tui/component/plugin-route-missing.tsx +0 -14
  122. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +0 -667
  123. package/src/cli/cmd/tui/component/prompt/frecency.tsx +0 -90
  124. package/src/cli/cmd/tui/component/prompt/history.tsx +0 -108
  125. package/src/cli/cmd/tui/component/prompt/index.tsx +0 -1224
  126. package/src/cli/cmd/tui/component/prompt/part.ts +0 -16
  127. package/src/cli/cmd/tui/component/prompt/stash.tsx +0 -101
  128. package/src/cli/cmd/tui/component/spinner.tsx +0 -24
  129. package/src/cli/cmd/tui/component/startup-loading.tsx +0 -63
  130. package/src/cli/cmd/tui/component/status-bar.ts +0 -54
  131. package/src/cli/cmd/tui/component/textarea-keybindings.ts +0 -73
  132. package/src/cli/cmd/tui/component/todo-item.tsx +0 -32
  133. package/src/cli/cmd/tui/component/workspace/dialog-session-list.tsx +0 -151
  134. package/src/cli/cmd/tui/context/args.tsx +0 -15
  135. package/src/cli/cmd/tui/context/directory.ts +0 -13
  136. package/src/cli/cmd/tui/context/exit.tsx +0 -60
  137. package/src/cli/cmd/tui/context/helper.tsx +0 -25
  138. package/src/cli/cmd/tui/context/keybind.tsx +0 -105
  139. package/src/cli/cmd/tui/context/kv.tsx +0 -52
  140. package/src/cli/cmd/tui/context/local.tsx +0 -412
  141. package/src/cli/cmd/tui/context/plugin-keybinds.ts +0 -41
  142. package/src/cli/cmd/tui/context/prompt.tsx +0 -18
  143. package/src/cli/cmd/tui/context/route.tsx +0 -52
  144. package/src/cli/cmd/tui/context/sdk.tsx +0 -128
  145. package/src/cli/cmd/tui/context/sync.tsx +0 -504
  146. package/src/cli/cmd/tui/context/theme/aura.json +0 -69
  147. package/src/cli/cmd/tui/context/theme/ayu.json +0 -80
  148. package/src/cli/cmd/tui/context/theme/carbonfox.json +0 -248
  149. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +0 -233
  150. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +0 -233
  151. package/src/cli/cmd/tui/context/theme/catppuccin.json +0 -112
  152. package/src/cli/cmd/tui/context/theme/cobalt2.json +0 -228
  153. package/src/cli/cmd/tui/context/theme/cursor.json +0 -249
  154. package/src/cli/cmd/tui/context/theme/dracula.json +0 -219
  155. package/src/cli/cmd/tui/context/theme/everforest.json +0 -241
  156. package/src/cli/cmd/tui/context/theme/flexoki.json +0 -237
  157. package/src/cli/cmd/tui/context/theme/github.json +0 -233
  158. package/src/cli/cmd/tui/context/theme/gruvbox.json +0 -242
  159. package/src/cli/cmd/tui/context/theme/kanagawa.json +0 -77
  160. package/src/cli/cmd/tui/context/theme/lucent-orng.json +0 -237
  161. package/src/cli/cmd/tui/context/theme/material.json +0 -235
  162. package/src/cli/cmd/tui/context/theme/matrix.json +0 -77
  163. package/src/cli/cmd/tui/context/theme/mercury.json +0 -252
  164. package/src/cli/cmd/tui/context/theme/monokai.json +0 -221
  165. package/src/cli/cmd/tui/context/theme/nightowl.json +0 -221
  166. package/src/cli/cmd/tui/context/theme/nord.json +0 -223
  167. package/src/cli/cmd/tui/context/theme/one-dark.json +0 -84
  168. package/src/cli/cmd/tui/context/theme/orng.json +0 -249
  169. package/src/cli/cmd/tui/context/theme/osaka-jade.json +0 -93
  170. package/src/cli/cmd/tui/context/theme/palenight.json +0 -222
  171. package/src/cli/cmd/tui/context/theme/rosepine.json +0 -234
  172. package/src/cli/cmd/tui/context/theme/solarized.json +0 -223
  173. package/src/cli/cmd/tui/context/theme/synthwave84.json +0 -226
  174. package/src/cli/cmd/tui/context/theme/tokyonight.json +0 -243
  175. package/src/cli/cmd/tui/context/theme/vercel.json +0 -245
  176. package/src/cli/cmd/tui/context/theme/vesper.json +0 -218
  177. package/src/cli/cmd/tui/context/theme/vikasit.json +0 -245
  178. package/src/cli/cmd/tui/context/theme/zenburn.json +0 -223
  179. package/src/cli/cmd/tui/context/theme.tsx +0 -1236
  180. package/src/cli/cmd/tui/context/tui-config.tsx +0 -9
  181. package/src/cli/cmd/tui/event.ts +0 -52
  182. package/src/cli/cmd/tui/feature-plugins/home/footer.tsx +0 -93
  183. package/src/cli/cmd/tui/feature-plugins/home/tips-view.tsx +0 -152
  184. package/src/cli/cmd/tui/feature-plugins/home/tips.tsx +0 -50
  185. package/src/cli/cmd/tui/feature-plugins/sidebar/context.tsx +0 -63
  186. package/src/cli/cmd/tui/feature-plugins/sidebar/files.tsx +0 -62
  187. package/src/cli/cmd/tui/feature-plugins/sidebar/footer.tsx +0 -99
  188. package/src/cli/cmd/tui/feature-plugins/sidebar/lsp.tsx +0 -66
  189. package/src/cli/cmd/tui/feature-plugins/sidebar/mcp.tsx +0 -96
  190. package/src/cli/cmd/tui/feature-plugins/sidebar/todo.tsx +0 -48
  191. package/src/cli/cmd/tui/feature-plugins/system/invite.tsx +0 -148
  192. package/src/cli/cmd/tui/feature-plugins/system/plugins.tsx +0 -270
  193. package/src/cli/cmd/tui/plugin/api.tsx +0 -420
  194. package/src/cli/cmd/tui/plugin/index.ts +0 -3
  195. package/src/cli/cmd/tui/plugin/internal.ts +0 -29
  196. package/src/cli/cmd/tui/plugin/runtime.ts +0 -998
  197. package/src/cli/cmd/tui/plugin/slots.tsx +0 -61
  198. package/src/cli/cmd/tui/routes/home.tsx +0 -84
  199. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +0 -65
  200. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +0 -110
  201. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +0 -26
  202. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +0 -47
  203. package/src/cli/cmd/tui/routes/session/footer.tsx +0 -91
  204. package/src/cli/cmd/tui/routes/session/index.tsx +0 -2259
  205. package/src/cli/cmd/tui/routes/session/permission.tsx +0 -685
  206. package/src/cli/cmd/tui/routes/session/question.tsx +0 -467
  207. package/src/cli/cmd/tui/routes/session/sidebar.tsx +0 -68
  208. package/src/cli/cmd/tui/routes/session/subagent-footer.tsx +0 -124
  209. package/src/cli/cmd/tui/thread.ts +0 -232
  210. package/src/cli/cmd/tui/ui/dialog-alert.tsx +0 -59
  211. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +0 -89
  212. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +0 -208
  213. package/src/cli/cmd/tui/ui/dialog-help.tsx +0 -40
  214. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +0 -106
  215. package/src/cli/cmd/tui/ui/dialog-select.tsx +0 -402
  216. package/src/cli/cmd/tui/ui/dialog.tsx +0 -192
  217. package/src/cli/cmd/tui/ui/link.tsx +0 -28
  218. package/src/cli/cmd/tui/ui/spinner.ts +0 -368
  219. package/src/cli/cmd/tui/ui/toast.tsx +0 -100
  220. package/src/cli/cmd/tui/util/clipboard.ts +0 -192
  221. package/src/cli/cmd/tui/util/editor.ts +0 -37
  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 -98
  226. package/src/cli/cmd/tui/win32.ts +0 -129
  227. package/src/cli/cmd/tui/worker.ts +0 -172
  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/logo.ts +0 -20
  234. package/src/cli/network.ts +0 -60
  235. package/src/cli/ui.ts +0 -116
  236. package/src/cli/upgrade.ts +0 -31
  237. package/src/command/index.ts +0 -251
  238. package/src/command/template/branch.txt +0 -1
  239. package/src/command/template/commit.txt +0 -7
  240. package/src/command/template/diff.txt +0 -1
  241. package/src/command/template/initialize.txt +0 -10
  242. package/src/command/template/memory.txt +0 -1
  243. package/src/command/template/review.txt +0 -101
  244. package/src/command/template/summary.txt +0 -1
  245. package/src/config/config.ts +0 -1619
  246. package/src/config/markdown.ts +0 -99
  247. package/src/config/migrate-tui-config.ts +0 -155
  248. package/src/config/paths.ts +0 -174
  249. package/src/config/tui-schema.ts +0 -36
  250. package/src/config/tui.ts +0 -222
  251. package/src/context/contextOptimizer.ts +0 -277
  252. package/src/control-plane/adaptors/index.ts +0 -20
  253. package/src/control-plane/adaptors/worktree.ts +0 -38
  254. package/src/control-plane/schema.ts +0 -17
  255. package/src/control-plane/sse.ts +0 -66
  256. package/src/control-plane/types.ts +0 -21
  257. package/src/control-plane/workspace.sql.ts +0 -17
  258. package/src/control-plane/workspace.ts +0 -154
  259. package/src/cost/index.ts +0 -173
  260. package/src/cost/schema.ts +0 -43
  261. package/src/cron/index.ts +0 -200
  262. package/src/effect/cross-spawn-spawner.ts +0 -479
  263. package/src/effect/instance-registry.ts +0 -12
  264. package/src/effect/instance-state.ts +0 -47
  265. package/src/effect/run-service.ts +0 -19
  266. package/src/effect/runner.ts +0 -216
  267. package/src/env/index.ts +0 -28
  268. package/src/file/ignore.ts +0 -82
  269. package/src/file/index.ts +0 -698
  270. package/src/file/protected.ts +0 -59
  271. package/src/file/ripgrep.ts +0 -376
  272. package/src/file/time.ts +0 -128
  273. package/src/file/watcher.ts +0 -171
  274. package/src/filesystem/index.ts +0 -226
  275. package/src/flag/flag.ts +0 -175
  276. package/src/format/formatter.ts +0 -396
  277. package/src/format/index.ts +0 -199
  278. package/src/global/index.ts +0 -54
  279. package/src/hook/index.ts +0 -132
  280. package/src/hook/schema.ts +0 -31
  281. package/src/id/id.ts +0 -85
  282. package/src/ide/index.ts +0 -74
  283. package/src/index.ts +0 -249
  284. package/src/installation/index.ts +0 -363
  285. package/src/lsp/client.ts +0 -252
  286. package/src/lsp/index.ts +0 -558
  287. package/src/lsp/language.ts +0 -120
  288. package/src/lsp/launch.ts +0 -21
  289. package/src/lsp/server.ts +0 -2093
  290. package/src/mcp/auth.ts +0 -181
  291. package/src/mcp/index.ts +0 -926
  292. package/src/mcp/oauth-callback.ts +0 -215
  293. package/src/mcp/oauth-provider.ts +0 -185
  294. package/src/memory/memory.ts +0 -140
  295. package/src/node.ts +0 -1
  296. package/src/patch/index.ts +0 -680
  297. package/src/permission/arity.ts +0 -163
  298. package/src/permission/auto-classify.ts +0 -114
  299. package/src/permission/evaluate.ts +0 -15
  300. package/src/permission/index.ts +0 -322
  301. package/src/permission/schema.ts +0 -17
  302. package/src/plugin/codex.ts +0 -596
  303. package/src/plugin/copilot.ts +0 -343
  304. package/src/plugin/index.ts +0 -322
  305. package/src/plugin/install.ts +0 -417
  306. package/src/plugin/loader.ts +0 -137
  307. package/src/plugin/meta.ts +0 -188
  308. package/src/plugin/shared.ts +0 -272
  309. package/src/project/bootstrap.ts +0 -31
  310. package/src/project/instance.ts +0 -167
  311. package/src/project/project.sql.ts +0 -16
  312. package/src/project/project.ts +0 -519
  313. package/src/project/schema.ts +0 -16
  314. package/src/project/state.ts +0 -70
  315. package/src/project/vcs.ts +0 -124
  316. package/src/provider/auth.ts +0 -252
  317. package/src/provider/error.ts +0 -197
  318. package/src/provider/github-models.ts +0 -53
  319. package/src/provider/models.ts +0 -139
  320. package/src/provider/provider.ts +0 -1650
  321. package/src/provider/schema.ts +0 -38
  322. package/src/provider/sdk/copilot/README.md +0 -5
  323. package/src/provider/sdk/copilot/chat/convert-to-openai-compatible-chat-messages.ts +0 -170
  324. package/src/provider/sdk/copilot/chat/get-response-metadata.ts +0 -15
  325. package/src/provider/sdk/copilot/chat/map-openai-compatible-finish-reason.ts +0 -19
  326. package/src/provider/sdk/copilot/chat/openai-compatible-api-types.ts +0 -64
  327. package/src/provider/sdk/copilot/chat/openai-compatible-chat-language-model.ts +0 -815
  328. package/src/provider/sdk/copilot/chat/openai-compatible-chat-options.ts +0 -28
  329. package/src/provider/sdk/copilot/chat/openai-compatible-metadata-extractor.ts +0 -44
  330. package/src/provider/sdk/copilot/chat/openai-compatible-prepare-tools.ts +0 -83
  331. package/src/provider/sdk/copilot/copilot-provider.ts +0 -100
  332. package/src/provider/sdk/copilot/index.ts +0 -2
  333. package/src/provider/sdk/copilot/openai-compatible-error.ts +0 -27
  334. package/src/provider/sdk/copilot/responses/convert-to-openai-responses-input.ts +0 -335
  335. package/src/provider/sdk/copilot/responses/map-openai-responses-finish-reason.ts +0 -22
  336. package/src/provider/sdk/copilot/responses/openai-config.ts +0 -18
  337. package/src/provider/sdk/copilot/responses/openai-error.ts +0 -22
  338. package/src/provider/sdk/copilot/responses/openai-responses-api-types.ts +0 -214
  339. package/src/provider/sdk/copilot/responses/openai-responses-language-model.ts +0 -1769
  340. package/src/provider/sdk/copilot/responses/openai-responses-prepare-tools.ts +0 -173
  341. package/src/provider/sdk/copilot/responses/openai-responses-settings.ts +0 -1
  342. package/src/provider/sdk/copilot/responses/tool/code-interpreter.ts +0 -87
  343. package/src/provider/sdk/copilot/responses/tool/file-search.ts +0 -127
  344. package/src/provider/sdk/copilot/responses/tool/image-generation.ts +0 -114
  345. package/src/provider/sdk/copilot/responses/tool/local-shell.ts +0 -64
  346. package/src/provider/sdk/copilot/responses/tool/web-search-preview.ts +0 -103
  347. package/src/provider/sdk/copilot/responses/tool/web-search.ts +0 -102
  348. package/src/provider/transform.ts +0 -1045
  349. package/src/pty/index.ts +0 -397
  350. package/src/pty/schema.ts +0 -17
  351. package/src/question/index.ts +0 -221
  352. package/src/question/schema.ts +0 -17
  353. package/src/server/error.ts +0 -36
  354. package/src/server/event.ts +0 -7
  355. package/src/server/instance.ts +0 -285
  356. package/src/server/mdns.ts +0 -60
  357. package/src/server/middleware.ts +0 -33
  358. package/src/server/projectors.ts +0 -28
  359. package/src/server/router.ts +0 -99
  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 -271
  363. package/src/server/routes/file.ts +0 -197
  364. package/src/server/routes/global.ts +0 -312
  365. package/src/server/routes/instance/httpapi/app.ts +0 -21
  366. package/src/server/routes/mcp.ts +0 -225
  367. package/src/server/routes/permission.ts +0 -69
  368. package/src/server/routes/project.ts +0 -118
  369. package/src/server/routes/provider.ts +0 -171
  370. package/src/server/routes/pty.ts +0 -211
  371. package/src/server/routes/question.ts +0 -99
  372. package/src/server/routes/session.ts +0 -1031
  373. package/src/server/routes/tui.ts +0 -379
  374. package/src/server/routes/workspace.ts +0 -94
  375. package/src/server/server.ts +0 -314
  376. package/src/session/compaction.ts +0 -432
  377. package/src/session/index.ts +0 -885
  378. package/src/session/instruction.ts +0 -192
  379. package/src/session/llm.ts +0 -365
  380. package/src/session/memory.ts +0 -126
  381. package/src/session/message-v2.ts +0 -1030
  382. package/src/session/message.ts +0 -191
  383. package/src/session/overflow.ts +0 -22
  384. package/src/session/processor.ts +0 -697
  385. package/src/session/projectors.ts +0 -135
  386. package/src/session/prompt/anthropic.txt +0 -125
  387. package/src/session/prompt/beast.txt +0 -161
  388. package/src/session/prompt/build-switch.txt +0 -5
  389. package/src/session/prompt/codex.txt +0 -90
  390. package/src/session/prompt/copilot-gpt-5.txt +0 -143
  391. package/src/session/prompt/default.txt +0 -116
  392. package/src/session/prompt/gemini.txt +0 -173
  393. package/src/session/prompt/gpt.txt +0 -120
  394. package/src/session/prompt/kimi.txt +0 -131
  395. package/src/session/prompt/max-steps.txt +0 -16
  396. package/src/session/prompt/plan-reminder-anthropic.txt +0 -67
  397. package/src/session/prompt/plan.txt +0 -26
  398. package/src/session/prompt/trinity.txt +0 -109
  399. package/src/session/prompt.ts +0 -1898
  400. package/src/session/rate-limit.ts +0 -176
  401. package/src/session/retry.ts +0 -107
  402. package/src/session/revert.ts +0 -135
  403. package/src/session/schema.ts +0 -38
  404. package/src/session/session.sql.ts +0 -103
  405. package/src/session/status.ts +0 -102
  406. package/src/session/summary.ts +0 -169
  407. package/src/session/system.ts +0 -76
  408. package/src/session/todo.ts +0 -57
  409. package/src/share/share-next.ts +0 -287
  410. package/src/share/share.sql.ts +0 -13
  411. package/src/shell/shell.ts +0 -110
  412. package/src/skill/discovery.ts +0 -116
  413. package/src/skill/index.ts +0 -277
  414. package/src/snapshot/index.ts +0 -489
  415. package/src/sql.d.ts +0 -4
  416. package/src/storage/db.bun.ts +0 -8
  417. package/src/storage/db.node.ts +0 -8
  418. package/src/storage/db.ts +0 -177
  419. package/src/storage/json-migration.ts +0 -425
  420. package/src/storage/schema.sql.ts +0 -10
  421. package/src/storage/schema.ts +0 -5
  422. package/src/storage/storage.ts +0 -353
  423. package/src/sync/README.md +0 -179
  424. package/src/sync/event.sql.ts +0 -16
  425. package/src/sync/index.ts +0 -263
  426. package/src/sync/schema.ts +0 -14
  427. package/src/task/index.ts +0 -89
  428. package/src/task/schema.ts +0 -39
  429. package/src/tool/apply_patch.ts +0 -281
  430. package/src/tool/apply_patch.txt +0 -33
  431. package/src/tool/bash-security.ts +0 -214
  432. package/src/tool/bash.ts +0 -526
  433. package/src/tool/bash.txt +0 -129
  434. package/src/tool/batch.ts +0 -183
  435. package/src/tool/batch.txt +0 -24
  436. package/src/tool/codesearch.ts +0 -132
  437. package/src/tool/codesearch.txt +0 -12
  438. package/src/tool/cron-create.ts +0 -38
  439. package/src/tool/cron-create.txt +0 -10
  440. package/src/tool/cron-delete.ts +0 -29
  441. package/src/tool/cron-delete.txt +0 -1
  442. package/src/tool/cron-list.ts +0 -37
  443. package/src/tool/cron-list.txt +0 -1
  444. package/src/tool/edit.ts +0 -667
  445. package/src/tool/edit.txt +0 -10
  446. package/src/tool/external-directory.ts +0 -37
  447. package/src/tool/glob.ts +0 -78
  448. package/src/tool/glob.txt +0 -6
  449. package/src/tool/grep.ts +0 -156
  450. package/src/tool/grep.txt +0 -8
  451. package/src/tool/invalid.ts +0 -17
  452. package/src/tool/ls.ts +0 -121
  453. package/src/tool/ls.txt +0 -1
  454. package/src/tool/lsp.ts +0 -97
  455. package/src/tool/lsp.txt +0 -19
  456. package/src/tool/multiedit.ts +0 -46
  457. package/src/tool/multiedit.txt +0 -41
  458. package/src/tool/notebook-edit.ts +0 -136
  459. package/src/tool/notebook-edit.txt +0 -10
  460. package/src/tool/plan-enter.txt +0 -14
  461. package/src/tool/plan-exit.txt +0 -13
  462. package/src/tool/plan.ts +0 -130
  463. package/src/tool/question.ts +0 -33
  464. package/src/tool/question.txt +0 -10
  465. package/src/tool/read.ts +0 -296
  466. package/src/tool/read.txt +0 -14
  467. package/src/tool/registry.ts +0 -333
  468. package/src/tool/schema.ts +0 -17
  469. package/src/tool/sendmessage.ts +0 -57
  470. package/src/tool/sendmessage.txt +0 -6
  471. package/src/tool/skill.ts +0 -105
  472. package/src/tool/task-create.ts +0 -25
  473. package/src/tool/task-create.txt +0 -1
  474. package/src/tool/task-get.ts +0 -27
  475. package/src/tool/task-get.txt +0 -1
  476. package/src/tool/task-list.ts +0 -45
  477. package/src/tool/task-list.txt +0 -1
  478. package/src/tool/task-output.ts +0 -28
  479. package/src/tool/task-output.txt +0 -1
  480. package/src/tool/task-stop.ts +0 -27
  481. package/src/tool/task-stop.txt +0 -1
  482. package/src/tool/task-update.ts +0 -38
  483. package/src/tool/task-update.txt +0 -1
  484. package/src/tool/task.ts +0 -246
  485. package/src/tool/task.txt +0 -65
  486. package/src/tool/todo.ts +0 -31
  487. package/src/tool/todowrite.txt +0 -167
  488. package/src/tool/tool.ts +0 -98
  489. package/src/tool/toolsearch.ts +0 -117
  490. package/src/tool/toolsearch.txt +0 -10
  491. package/src/tool/truncate.ts +0 -144
  492. package/src/tool/truncation-dir.ts +0 -4
  493. package/src/tool/webfetch.ts +0 -206
  494. package/src/tool/webfetch.txt +0 -13
  495. package/src/tool/websearch.ts +0 -150
  496. package/src/tool/websearch.txt +0 -14
  497. package/src/tool/worktree-enter.ts +0 -38
  498. package/src/tool/worktree-enter.txt +0 -8
  499. package/src/tool/worktree-exit.ts +0 -43
  500. package/src/tool/worktree-exit.txt +0 -3
  501. package/src/tool/write.ts +0 -84
  502. package/src/tool/write.txt +0 -8
  503. package/src/util/abort.ts +0 -35
  504. package/src/util/archive.ts +0 -17
  505. package/src/util/color.ts +0 -19
  506. package/src/util/context.ts +0 -25
  507. package/src/util/data-url.ts +0 -9
  508. package/src/util/defer.ts +0 -12
  509. package/src/util/effect-http-client.ts +0 -11
  510. package/src/util/effect-zod.ts +0 -98
  511. package/src/util/error.ts +0 -77
  512. package/src/util/filesystem.ts +0 -220
  513. package/src/util/flock.ts +0 -333
  514. package/src/util/fn.ts +0 -21
  515. package/src/util/format.ts +0 -20
  516. package/src/util/git.ts +0 -35
  517. package/src/util/glob.ts +0 -34
  518. package/src/util/hash.ts +0 -7
  519. package/src/util/iife.ts +0 -3
  520. package/src/util/keybind.ts +0 -103
  521. package/src/util/lazy.ts +0 -23
  522. package/src/util/locale.ts +0 -81
  523. package/src/util/lock.ts +0 -98
  524. package/src/util/log.ts +0 -182
  525. package/src/util/network.ts +0 -9
  526. package/src/util/process.ts +0 -172
  527. package/src/util/queue.ts +0 -32
  528. package/src/util/record.ts +0 -3
  529. package/src/util/rpc.ts +0 -66
  530. package/src/util/schema.ts +0 -53
  531. package/src/util/scrap.ts +0 -10
  532. package/src/util/signal.ts +0 -12
  533. package/src/util/timeout.ts +0 -14
  534. package/src/util/token.ts +0 -7
  535. package/src/util/update-schema.ts +0 -13
  536. package/src/util/which.ts +0 -14
  537. package/src/util/wildcard.ts +0 -59
  538. package/src/vim/agent-namer.ts +0 -86
  539. package/src/vim/types.ts +0 -29
  540. package/src/worktree/index.ts +0 -638
  541. package/sst-env.d.ts +0 -10
  542. package/test/AGENTS.md +0 -81
  543. package/test/account/repo.test.ts +0 -326
  544. package/test/account/service.test.ts +0 -282
  545. package/test/acp/agent-interface.test.ts +0 -51
  546. package/test/acp/event-subscription.test.ts +0 -685
  547. package/test/agent/agent.test.ts +0 -717
  548. package/test/auth/auth.test.ts +0 -58
  549. package/test/bun.test.ts +0 -137
  550. package/test/bus/bus-effect.test.ts +0 -164
  551. package/test/bus/bus-integration.test.ts +0 -87
  552. package/test/bus/bus.test.ts +0 -219
  553. package/test/cli/account.test.ts +0 -26
  554. package/test/cli/cmd/tui/prompt-part.test.ts +0 -47
  555. package/test/cli/github-action.test.ts +0 -198
  556. package/test/cli/github-remote.test.ts +0 -80
  557. package/test/cli/import.test.ts +0 -54
  558. package/test/cli/plugin-auth-picker.test.ts +0 -120
  559. package/test/cli/tui/keybind-plugin.test.ts +0 -90
  560. package/test/cli/tui/plugin-add.test.ts +0 -61
  561. package/test/cli/tui/plugin-install.test.ts +0 -89
  562. package/test/cli/tui/plugin-lifecycle.test.ts +0 -225
  563. package/test/cli/tui/plugin-loader-entrypoint.test.ts +0 -492
  564. package/test/cli/tui/plugin-loader-pure.test.ts +0 -72
  565. package/test/cli/tui/plugin-loader.test.ts +0 -752
  566. package/test/cli/tui/plugin-toggle.test.ts +0 -159
  567. package/test/cli/tui/theme-store.test.ts +0 -51
  568. package/test/cli/tui/thread.test.ts +0 -128
  569. package/test/cli/tui/transcript.test.ts +0 -322
  570. package/test/config/agent-color.test.ts +0 -71
  571. package/test/config/config.test.ts +0 -2218
  572. package/test/config/fixtures/empty-frontmatter.md +0 -4
  573. package/test/config/fixtures/frontmatter.md +0 -28
  574. package/test/config/fixtures/markdown-header.md +0 -11
  575. package/test/config/fixtures/no-frontmatter.md +0 -1
  576. package/test/config/fixtures/weird-model-id.md +0 -13
  577. package/test/config/markdown.test.ts +0 -228
  578. package/test/config/tui.test.ts +0 -673
  579. package/test/control-plane/sse.test.ts +0 -56
  580. package/test/effect/cross-spawn-spawner.test.ts +0 -402
  581. package/test/effect/instance-state.test.ts +0 -384
  582. package/test/effect/run-service.test.ts +0 -46
  583. package/test/effect/runner.test.ts +0 -523
  584. package/test/file/fsmonitor.test.ts +0 -62
  585. package/test/file/ignore.test.ts +0 -10
  586. package/test/file/index.test.ts +0 -946
  587. package/test/file/path-traversal.test.ts +0 -198
  588. package/test/file/ripgrep.test.ts +0 -54
  589. package/test/file/time.test.ts +0 -354
  590. package/test/file/watcher.test.ts +0 -247
  591. package/test/filesystem/filesystem.test.ts +0 -319
  592. package/test/fixture/db.ts +0 -11
  593. package/test/fixture/fixture.test.ts +0 -26
  594. package/test/fixture/fixture.ts +0 -141
  595. package/test/fixture/flock-worker.ts +0 -72
  596. package/test/fixture/lsp/fake-lsp-server.js +0 -77
  597. package/test/fixture/plug-worker.ts +0 -93
  598. package/test/fixture/plugin-meta-worker.ts +0 -26
  599. package/test/fixture/skills/agents-sdk/SKILL.md +0 -152
  600. package/test/fixture/skills/agents-sdk/references/callable.md +0 -92
  601. package/test/fixture/skills/cloudflare/SKILL.md +0 -211
  602. package/test/fixture/skills/index.json +0 -6
  603. package/test/fixture/tui-plugin.ts +0 -335
  604. package/test/fixture/tui-runtime.ts +0 -27
  605. package/test/format/format.test.ts +0 -179
  606. package/test/ide/ide.test.ts +0 -82
  607. package/test/installation/installation.test.ts +0 -151
  608. package/test/keybind.test.ts +0 -421
  609. package/test/lib/effect.ts +0 -37
  610. package/test/lib/filesystem.ts +0 -10
  611. package/test/lsp/client.test.ts +0 -95
  612. package/test/lsp/index.test.ts +0 -55
  613. package/test/lsp/launch.test.ts +0 -22
  614. package/test/lsp/lifecycle.test.ts +0 -147
  615. package/test/mcp/headers.test.ts +0 -153
  616. package/test/mcp/lifecycle.test.ts +0 -750
  617. package/test/mcp/oauth-auto-connect.test.ts +0 -199
  618. package/test/mcp/oauth-browser.test.ts +0 -249
  619. package/test/memory/abort-leak.test.ts +0 -137
  620. package/test/patch/patch.test.ts +0 -348
  621. package/test/permission/arity.test.ts +0 -33
  622. package/test/permission/next.test.ts +0 -1148
  623. package/test/permission-task.test.ts +0 -323
  624. package/test/plugin/auth-override.test.ts +0 -74
  625. package/test/plugin/codex.test.ts +0 -123
  626. package/test/plugin/install-concurrency.test.ts +0 -140
  627. package/test/plugin/install.test.ts +0 -531
  628. package/test/plugin/loader-shared.test.ts +0 -836
  629. package/test/plugin/meta.test.ts +0 -137
  630. package/test/plugin/trigger.test.ts +0 -111
  631. package/test/preload.ts +0 -90
  632. package/test/project/migrate-global.test.ts +0 -140
  633. package/test/project/project.test.ts +0 -459
  634. package/test/project/state.test.ts +0 -115
  635. package/test/project/vcs.test.ts +0 -116
  636. package/test/project/worktree-remove.test.ts +0 -96
  637. package/test/project/worktree.test.ts +0 -173
  638. package/test/provider/amazon-bedrock.test.ts +0 -447
  639. package/test/provider/copilot/convert-to-copilot-messages.test.ts +0 -523
  640. package/test/provider/copilot/copilot-chat-model.test.ts +0 -592
  641. package/test/provider/gitlab-duo.test.ts +0 -412
  642. package/test/provider/provider.test.ts +0 -2284
  643. package/test/provider/transform.test.ts +0 -2758
  644. package/test/pty/pty-output-isolation.test.ts +0 -141
  645. package/test/pty/pty-session.test.ts +0 -92
  646. package/test/pty/pty-shell.test.ts +0 -59
  647. package/test/question/question.test.ts +0 -453
  648. package/test/server/global-session-list.test.ts +0 -89
  649. package/test/server/project-init-git.test.ts +0 -121
  650. package/test/server/session-actions.test.ts +0 -83
  651. package/test/server/session-list.test.ts +0 -98
  652. package/test/server/session-messages.test.ts +0 -159
  653. package/test/server/session-select.test.ts +0 -84
  654. package/test/session/compaction.test.ts +0 -1202
  655. package/test/session/instruction.test.ts +0 -170
  656. package/test/session/llm.test.ts +0 -1098
  657. package/test/session/message-v2.test.ts +0 -957
  658. package/test/session/messages-pagination.test.ts +0 -115
  659. package/test/session/processor-effect.test.ts +0 -872
  660. package/test/session/prompt-concurrency.test.ts +0 -247
  661. package/test/session/prompt-effect.test.ts +0 -1206
  662. package/test/session/prompt.test.ts +0 -518
  663. package/test/session/retry.test.ts +0 -232
  664. package/test/session/revert-compact.test.ts +0 -286
  665. package/test/session/session.test.ts +0 -142
  666. package/test/session/structured-output-integration.test.ts +0 -233
  667. package/test/session/structured-output.test.ts +0 -391
  668. package/test/session/system.test.ts +0 -59
  669. package/test/share/share-next.test.ts +0 -76
  670. package/test/shell/shell.test.ts +0 -73
  671. package/test/skill/discovery.test.ts +0 -116
  672. package/test/skill/skill.test.ts +0 -392
  673. package/test/snapshot/snapshot.test.ts +0 -1235
  674. package/test/storage/db.test.ts +0 -14
  675. package/test/storage/json-migration.test.ts +0 -849
  676. package/test/storage/storage.test.ts +0 -295
  677. package/test/sync/index.test.ts +0 -191
  678. package/test/tool/__snapshots__/tool.test.ts.snap +0 -9
  679. package/test/tool/apply_patch.test.ts +0 -567
  680. package/test/tool/bash.test.ts +0 -984
  681. package/test/tool/edit.test.ts +0 -681
  682. package/test/tool/external-directory.test.ts +0 -198
  683. package/test/tool/fixtures/large-image.png +0 -0
  684. package/test/tool/fixtures/models-api.json +0 -65179
  685. package/test/tool/grep.test.ts +0 -111
  686. package/test/tool/question.test.ts +0 -108
  687. package/test/tool/read.test.ts +0 -546
  688. package/test/tool/registry.test.ts +0 -126
  689. package/test/tool/skill.test.ts +0 -167
  690. package/test/tool/task.test.ts +0 -49
  691. package/test/tool/truncation.test.ts +0 -161
  692. package/test/tool/webfetch.test.ts +0 -101
  693. package/test/tool/write.test.ts +0 -353
  694. package/test/util/data-url.test.ts +0 -14
  695. package/test/util/effect-zod.test.ts +0 -61
  696. package/test/util/error.test.ts +0 -38
  697. package/test/util/filesystem.test.ts +0 -567
  698. package/test/util/flock.test.ts +0 -383
  699. package/test/util/format.test.ts +0 -59
  700. package/test/util/glob.test.ts +0 -164
  701. package/test/util/iife.test.ts +0 -36
  702. package/test/util/lazy.test.ts +0 -50
  703. package/test/util/lock.test.ts +0 -72
  704. package/test/util/module.test.ts +0 -59
  705. package/test/util/process.test.ts +0 -128
  706. package/test/util/timeout.test.ts +0 -21
  707. package/test/util/which.test.ts +0 -100
  708. package/test/util/wildcard.test.ts +0 -90
  709. package/tsconfig.json +0 -23
  710. package/vikasitai-vikasit-code-2.0.5.tgz +0 -0
  711. /package/{script/postinstall.mjs → postinstall.mjs} +0 -0
@@ -1,1235 +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 file modifications", async () => {
986
- await using tmp = await bootstrap()
987
- await Instance.provide({
988
- directory: tmp.path,
989
- fn: async () => {
990
- const before = await Snapshot.track()
991
- expect(before).toBeTruthy()
992
-
993
- await Filesystem.write(`${tmp.path}/b.txt`, "modified content")
994
-
995
- const after = await Snapshot.track()
996
- expect(after).toBeTruthy()
997
-
998
- const diffs = await Snapshot.diffFull(before!, after!)
999
- expect(diffs.length).toBe(1)
1000
-
1001
- const modifiedFileDiff = diffs[0]
1002
- expect(modifiedFileDiff.file).toBe("b.txt")
1003
- expect(modifiedFileDiff.before).toBe(tmp.extra.bContent)
1004
- expect(modifiedFileDiff.after).toBe("modified content")
1005
- expect(modifiedFileDiff.additions).toBeGreaterThan(0)
1006
- expect(modifiedFileDiff.deletions).toBeGreaterThan(0)
1007
- },
1008
- })
1009
- })
1010
-
1011
- test("diffFull with file deletions", async () => {
1012
- await using tmp = await bootstrap()
1013
- await Instance.provide({
1014
- directory: tmp.path,
1015
- fn: async () => {
1016
- const before = await Snapshot.track()
1017
- expect(before).toBeTruthy()
1018
-
1019
- await $`rm ${tmp.path}/a.txt`.quiet()
1020
-
1021
- const after = await Snapshot.track()
1022
- expect(after).toBeTruthy()
1023
-
1024
- const diffs = await Snapshot.diffFull(before!, after!)
1025
- expect(diffs.length).toBe(1)
1026
-
1027
- const removedFileDiff = diffs[0]
1028
- expect(removedFileDiff.file).toBe("a.txt")
1029
- expect(removedFileDiff.before).toBe(tmp.extra.aContent)
1030
- expect(removedFileDiff.after).toBe("")
1031
- expect(removedFileDiff.additions).toBe(0)
1032
- expect(removedFileDiff.deletions).toBe(1)
1033
- },
1034
- })
1035
- })
1036
-
1037
- test("diffFull with multiple line additions", async () => {
1038
- await using tmp = await bootstrap()
1039
- await Instance.provide({
1040
- directory: tmp.path,
1041
- fn: async () => {
1042
- const before = await Snapshot.track()
1043
- expect(before).toBeTruthy()
1044
-
1045
- await Filesystem.write(`${tmp.path}/multi.txt`, "line1\nline2\nline3")
1046
-
1047
- const after = await Snapshot.track()
1048
- expect(after).toBeTruthy()
1049
-
1050
- const diffs = await Snapshot.diffFull(before!, after!)
1051
- expect(diffs.length).toBe(1)
1052
-
1053
- const multiDiff = diffs[0]
1054
- expect(multiDiff.file).toBe("multi.txt")
1055
- expect(multiDiff.before).toBe("")
1056
- expect(multiDiff.after).toBe("line1\nline2\nline3")
1057
- expect(multiDiff.additions).toBe(3)
1058
- expect(multiDiff.deletions).toBe(0)
1059
- },
1060
- })
1061
- })
1062
-
1063
- test("diffFull with addition and deletion", async () => {
1064
- await using tmp = await bootstrap()
1065
- await Instance.provide({
1066
- directory: tmp.path,
1067
- fn: async () => {
1068
- const before = await Snapshot.track()
1069
- expect(before).toBeTruthy()
1070
-
1071
- await Filesystem.write(`${tmp.path}/added.txt`, "added content")
1072
- await $`rm ${tmp.path}/a.txt`.quiet()
1073
-
1074
- const after = await Snapshot.track()
1075
- expect(after).toBeTruthy()
1076
-
1077
- const diffs = await Snapshot.diffFull(before!, after!)
1078
- expect(diffs.length).toBe(2)
1079
-
1080
- const addedFileDiff = diffs.find((d) => d.file === "added.txt")
1081
- expect(addedFileDiff).toBeDefined()
1082
- expect(addedFileDiff!.before).toBe("")
1083
- expect(addedFileDiff!.after).toBe("added content")
1084
- expect(addedFileDiff!.additions).toBe(1)
1085
- expect(addedFileDiff!.deletions).toBe(0)
1086
-
1087
- const removedFileDiff = diffs.find((d) => d.file === "a.txt")
1088
- expect(removedFileDiff).toBeDefined()
1089
- expect(removedFileDiff!.before).toBe(tmp.extra.aContent)
1090
- expect(removedFileDiff!.after).toBe("")
1091
- expect(removedFileDiff!.additions).toBe(0)
1092
- expect(removedFileDiff!.deletions).toBe(1)
1093
- },
1094
- })
1095
- })
1096
-
1097
- test("diffFull with multiple additions and deletions", async () => {
1098
- await using tmp = await bootstrap()
1099
- await Instance.provide({
1100
- directory: tmp.path,
1101
- fn: async () => {
1102
- const before = await Snapshot.track()
1103
- expect(before).toBeTruthy()
1104
-
1105
- await Filesystem.write(`${tmp.path}/multi1.txt`, "line1\nline2\nline3")
1106
- await Filesystem.write(`${tmp.path}/multi2.txt`, "single line")
1107
- await $`rm ${tmp.path}/a.txt`.quiet()
1108
- await $`rm ${tmp.path}/b.txt`.quiet()
1109
-
1110
- const after = await Snapshot.track()
1111
- expect(after).toBeTruthy()
1112
-
1113
- const diffs = await Snapshot.diffFull(before!, after!)
1114
- expect(diffs.length).toBe(4)
1115
-
1116
- const multi1Diff = diffs.find((d) => d.file === "multi1.txt")
1117
- expect(multi1Diff).toBeDefined()
1118
- expect(multi1Diff!.additions).toBe(3)
1119
- expect(multi1Diff!.deletions).toBe(0)
1120
-
1121
- const multi2Diff = diffs.find((d) => d.file === "multi2.txt")
1122
- expect(multi2Diff).toBeDefined()
1123
- expect(multi2Diff!.additions).toBe(1)
1124
- expect(multi2Diff!.deletions).toBe(0)
1125
-
1126
- const removedADiff = diffs.find((d) => d.file === "a.txt")
1127
- expect(removedADiff).toBeDefined()
1128
- expect(removedADiff!.additions).toBe(0)
1129
- expect(removedADiff!.deletions).toBe(1)
1130
-
1131
- const removedBDiff = diffs.find((d) => d.file === "b.txt")
1132
- expect(removedBDiff).toBeDefined()
1133
- expect(removedBDiff!.additions).toBe(0)
1134
- expect(removedBDiff!.deletions).toBe(1)
1135
- },
1136
- })
1137
- })
1138
-
1139
- test("diffFull with no changes", async () => {
1140
- await using tmp = await bootstrap()
1141
- await Instance.provide({
1142
- directory: tmp.path,
1143
- fn: async () => {
1144
- const before = await Snapshot.track()
1145
- expect(before).toBeTruthy()
1146
-
1147
- const after = await Snapshot.track()
1148
- expect(after).toBeTruthy()
1149
-
1150
- const diffs = await Snapshot.diffFull(before!, after!)
1151
- expect(diffs.length).toBe(0)
1152
- },
1153
- })
1154
- })
1155
-
1156
- test("diffFull with binary file changes", async () => {
1157
- await using tmp = await bootstrap()
1158
- await Instance.provide({
1159
- directory: tmp.path,
1160
- fn: async () => {
1161
- const before = await Snapshot.track()
1162
- expect(before).toBeTruthy()
1163
-
1164
- await Filesystem.write(`${tmp.path}/binary.bin`, new Uint8Array([0x00, 0x01, 0x02, 0x03]))
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(1)
1171
-
1172
- const binaryDiff = diffs[0]
1173
- expect(binaryDiff.file).toBe("binary.bin")
1174
- expect(binaryDiff.before).toBe("")
1175
- },
1176
- })
1177
- })
1178
-
1179
- test("diffFull with whitespace changes", async () => {
1180
- await using tmp = await bootstrap()
1181
- await Instance.provide({
1182
- directory: tmp.path,
1183
- fn: async () => {
1184
- await Filesystem.write(`${tmp.path}/whitespace.txt`, "line1\nline2")
1185
- const before = await Snapshot.track()
1186
- expect(before).toBeTruthy()
1187
-
1188
- await Filesystem.write(`${tmp.path}/whitespace.txt`, "line1\n\nline2\n")
1189
-
1190
- const after = await Snapshot.track()
1191
- expect(after).toBeTruthy()
1192
-
1193
- const diffs = await Snapshot.diffFull(before!, after!)
1194
- expect(diffs.length).toBe(1)
1195
-
1196
- const whitespaceDiff = diffs[0]
1197
- expect(whitespaceDiff.file).toBe("whitespace.txt")
1198
- expect(whitespaceDiff.additions).toBeGreaterThan(0)
1199
- },
1200
- })
1201
- })
1202
-
1203
- test("revert with overlapping files across patches uses first patch hash", async () => {
1204
- await using tmp = await bootstrap()
1205
- await Instance.provide({
1206
- directory: tmp.path,
1207
- fn: async () => {
1208
- // Write initial content and snapshot
1209
- await Filesystem.write(`${tmp.path}/shared.txt`, "v1")
1210
- const snap1 = await Snapshot.track()
1211
- expect(snap1).toBeTruthy()
1212
-
1213
- // Modify and snapshot again
1214
- await Filesystem.write(`${tmp.path}/shared.txt`, "v2")
1215
- const snap2 = await Snapshot.track()
1216
- expect(snap2).toBeTruthy()
1217
-
1218
- // Modify once more so both patches include shared.txt
1219
- await Filesystem.write(`${tmp.path}/shared.txt`, "v3")
1220
-
1221
- const patch1 = await Snapshot.patch(snap1!)
1222
- const patch2 = await Snapshot.patch(snap2!)
1223
-
1224
- // Both patches should include shared.txt
1225
- expect(patch1.files).toContain(fwd(tmp.path, "shared.txt"))
1226
- expect(patch2.files).toContain(fwd(tmp.path, "shared.txt"))
1227
-
1228
- // Revert with patch1 first — should use snap1's hash (restoring "v1")
1229
- await Snapshot.revert([patch1, patch2])
1230
-
1231
- const content = await fs.readFile(`${tmp.path}/shared.txt`, "utf-8")
1232
- expect(content).toBe("v1")
1233
- },
1234
- })
1235
- })