bonecode 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (605) hide show
  1. package/ARCHITECTURE.md +183 -0
  2. package/README.md +71 -0
  3. package/bin/bonecode +62 -0
  4. package/bone/migrations/rag_vectors.sql +258 -0
  5. package/bone/output/agent/.dockerignore +7 -0
  6. package/bone/output/agent/.env.example +36 -0
  7. package/bone/output/agent/.github/workflows/ci.yaml +58 -0
  8. package/bone/output/agent/AgentDomain.bone.map +350 -0
  9. package/bone/output/agent/AgentDomain.postman_collection.json +958 -0
  10. package/bone/output/agent/Dockerfile +22 -0
  11. package/bone/output/agent/README.md +47 -0
  12. package/bone/output/agent/admin/index.html +740 -0
  13. package/bone/output/agent/docker-compose.yaml +22 -0
  14. package/bone/output/agent/k8s/deployment.yaml +75 -0
  15. package/bone/output/agent/migrations/agent.sql +36 -0
  16. package/bone/output/agent/migrations/agent_instance.sql +36 -0
  17. package/bone/output/agent/migrations/audit_log.sql +18 -0
  18. package/bone/output/agent/migrations/build_step.sql +34 -0
  19. package/bone/output/agent/migrations/event_outbox.sql +31 -0
  20. package/bone/output/agent/migrations/plan.sql +30 -0
  21. package/bone/output/agent/migrations/task.sql +30 -0
  22. package/bone/output/agent/migrations/tool_call.sql +33 -0
  23. package/bone/output/agent/openapi.yaml +1116 -0
  24. package/bone/output/agent/package.json +36 -0
  25. package/bone/output/agent/schema.graphql +233 -0
  26. package/bone/output/agent/sdk/client.ts +231 -0
  27. package/bone/output/agent/src/algorithms.ts +2 -0
  28. package/bone/output/agent/src/audit.ts +44 -0
  29. package/bone/output/agent/src/auth.ts +57 -0
  30. package/bone/output/agent/src/cron.ts +12 -0
  31. package/bone/output/agent/src/db.ts +32 -0
  32. package/bone/output/agent/src/debug.ts +66 -0
  33. package/bone/output/agent/src/events.ts +243 -0
  34. package/bone/output/agent/src/extensions.ts +54 -0
  35. package/bone/output/agent/src/failure_rules.ts +323 -0
  36. package/bone/output/agent/src/flows.ts +168 -0
  37. package/bone/output/agent/src/health.ts +43 -0
  38. package/bone/output/agent/src/index.ts +100 -0
  39. package/bone/output/agent/src/logger.ts +66 -0
  40. package/bone/output/agent/src/metrics.ts +75 -0
  41. package/bone/output/agent/src/migrate.ts +352 -0
  42. package/bone/output/agent/src/migration_diff.ts +108 -0
  43. package/bone/output/agent/src/notify.ts +125 -0
  44. package/bone/output/agent/src/routes/agent_instance.ts +234 -0
  45. package/bone/output/agent/src/routes/build_step.ts +105 -0
  46. package/bone/output/agent/src/routes/plan.ts +91 -0
  47. package/bone/output/agent/src/routes/task.ts +105 -0
  48. package/bone/output/agent/src/routes/tool_call.ts +166 -0
  49. package/bone/output/agent/src/schemas.ts +384 -0
  50. package/bone/output/agent/src/state_machines/agent_instance.ts +24 -0
  51. package/bone/output/agent/src/state_machines/build_step.ts +22 -0
  52. package/bone/output/agent/src/state_machines/plan.ts +22 -0
  53. package/bone/output/agent/src/state_machines/task.ts +22 -0
  54. package/bone/output/agent/src/state_machines/tool_call.ts +22 -0
  55. package/bone/output/agent/src/tests.ts +362 -0
  56. package/bone/output/agent/src/websocket.ts +201 -0
  57. package/bone/output/agent/tsconfig.json +25 -0
  58. package/bone/output/rag/.dockerignore +7 -0
  59. package/bone/output/rag/.env.example +36 -0
  60. package/bone/output/rag/.github/workflows/ci.yaml +58 -0
  61. package/bone/output/rag/Dockerfile +22 -0
  62. package/bone/output/rag/RAGDomain.bone.map +287 -0
  63. package/bone/output/rag/RAGDomain.postman_collection.json +923 -0
  64. package/bone/output/rag/README.md +47 -0
  65. package/bone/output/rag/admin/index.html +818 -0
  66. package/bone/output/rag/docker-compose.yaml +22 -0
  67. package/bone/output/rag/k8s/deployment.yaml +75 -0
  68. package/bone/output/rag/migrations/audit_log.sql +18 -0
  69. package/bone/output/rag/migrations/code_chunk.sql +34 -0
  70. package/bone/output/rag/migrations/code_file.sql +33 -0
  71. package/bone/output/rag/migrations/event_outbox.sql +31 -0
  72. package/bone/output/rag/migrations/indexing_job.sql +33 -0
  73. package/bone/output/rag/migrations/knowledge_base.sql +35 -0
  74. package/bone/output/rag/migrations/memory_entry.sql +34 -0
  75. package/bone/output/rag/openapi.yaml +1097 -0
  76. package/bone/output/rag/package.json +36 -0
  77. package/bone/output/rag/schema.graphql +245 -0
  78. package/bone/output/rag/sdk/client.ts +234 -0
  79. package/bone/output/rag/src/algorithms.ts +2 -0
  80. package/bone/output/rag/src/audit.ts +37 -0
  81. package/bone/output/rag/src/auth.ts +57 -0
  82. package/bone/output/rag/src/cron.ts +12 -0
  83. package/bone/output/rag/src/db.ts +32 -0
  84. package/bone/output/rag/src/debug.ts +66 -0
  85. package/bone/output/rag/src/events.ts +243 -0
  86. package/bone/output/rag/src/extensions.ts +350 -0
  87. package/bone/output/rag/src/failure_rules.ts +315 -0
  88. package/bone/output/rag/src/flows.ts +239 -0
  89. package/bone/output/rag/src/health.ts +43 -0
  90. package/bone/output/rag/src/index.ts +95 -0
  91. package/bone/output/rag/src/logger.ts +66 -0
  92. package/bone/output/rag/src/metrics.ts +75 -0
  93. package/bone/output/rag/src/migrate.ts +364 -0
  94. package/bone/output/rag/src/migration_diff.ts +108 -0
  95. package/bone/output/rag/src/notify.ts +99 -0
  96. package/bone/output/rag/src/routes/code_chunk.ts +75 -0
  97. package/bone/output/rag/src/routes/code_file.ts +101 -0
  98. package/bone/output/rag/src/routes/indexing_job.ts +87 -0
  99. package/bone/output/rag/src/routes/knowledge_base.ts +230 -0
  100. package/bone/output/rag/src/routes/memory_entry.ts +87 -0
  101. package/bone/output/rag/src/schemas.ts +394 -0
  102. package/bone/output/rag/src/state_machines/code_file.ts +23 -0
  103. package/bone/output/rag/src/state_machines/indexing_job.ts +22 -0
  104. package/bone/output/rag/src/state_machines/knowledge_base.ts +23 -0
  105. package/bone/output/rag/src/state_machines/memory_entry.ts +20 -0
  106. package/bone/output/rag/src/tests.ts +340 -0
  107. package/bone/output/rag/tsconfig.json +25 -0
  108. package/bone/output/session/.dockerignore +7 -0
  109. package/bone/output/session/.env.example +36 -0
  110. package/bone/output/session/.github/workflows/ci.yaml +58 -0
  111. package/bone/output/session/Dockerfile +22 -0
  112. package/bone/output/session/README.md +47 -0
  113. package/bone/output/session/SessionDomain.bone.map +350 -0
  114. package/bone/output/session/SessionDomain.postman_collection.json +958 -0
  115. package/bone/output/session/admin/index.html +667 -0
  116. package/bone/output/session/docker-compose.yaml +22 -0
  117. package/bone/output/session/k8s/deployment.yaml +75 -0
  118. package/bone/output/session/migrations/audit_log.sql +18 -0
  119. package/bone/output/session/migrations/event_outbox.sql +31 -0
  120. package/bone/output/session/migrations/message.sql +31 -0
  121. package/bone/output/session/migrations/part.sql +28 -0
  122. package/bone/output/session/migrations/permission.sql +28 -0
  123. package/bone/output/session/migrations/project.sql +28 -0
  124. package/bone/output/session/migrations/session.sql +38 -0
  125. package/bone/output/session/openapi.yaml +1101 -0
  126. package/bone/output/session/package.json +36 -0
  127. package/bone/output/session/schema.graphql +222 -0
  128. package/bone/output/session/sdk/client.ts +225 -0
  129. package/bone/output/session/src/algorithms.ts +2 -0
  130. package/bone/output/session/src/audit.ts +44 -0
  131. package/bone/output/session/src/auth.ts +57 -0
  132. package/bone/output/session/src/cron.ts +12 -0
  133. package/bone/output/session/src/db.ts +32 -0
  134. package/bone/output/session/src/debug.ts +66 -0
  135. package/bone/output/session/src/events.ts +270 -0
  136. package/bone/output/session/src/extensions.ts +215 -0
  137. package/bone/output/session/src/failure_rules.ts +284 -0
  138. package/bone/output/session/src/flows.ts +168 -0
  139. package/bone/output/session/src/health.ts +43 -0
  140. package/bone/output/session/src/index.ts +100 -0
  141. package/bone/output/session/src/logger.ts +66 -0
  142. package/bone/output/session/src/metrics.ts +75 -0
  143. package/bone/output/session/src/migrate.ts +332 -0
  144. package/bone/output/session/src/migration_diff.ts +108 -0
  145. package/bone/output/session/src/notify.ts +112 -0
  146. package/bone/output/session/src/routes/message.ts +93 -0
  147. package/bone/output/session/src/routes/part.ts +79 -0
  148. package/bone/output/session/src/routes/permission.ts +79 -0
  149. package/bone/output/session/src/routes/project.ts +79 -0
  150. package/bone/output/session/src/routes/session.ts +294 -0
  151. package/bone/output/session/src/schemas.ts +357 -0
  152. package/bone/output/session/src/state_machines/session.ts +23 -0
  153. package/bone/output/session/src/tests.ts +326 -0
  154. package/bone/output/session/src/websocket.ts +201 -0
  155. package/bone/output/session/tsconfig.json +25 -0
  156. package/bone/output/workspace/.dockerignore +7 -0
  157. package/bone/output/workspace/.env.example +36 -0
  158. package/bone/output/workspace/.github/workflows/ci.yaml +58 -0
  159. package/bone/output/workspace/Dockerfile +22 -0
  160. package/bone/output/workspace/README.md +45 -0
  161. package/bone/output/workspace/WorkspaceDomain.bone.map +189 -0
  162. package/bone/output/workspace/WorkspaceDomain.postman_collection.json +621 -0
  163. package/bone/output/workspace/admin/index.html +485 -0
  164. package/bone/output/workspace/docker-compose.yaml +22 -0
  165. package/bone/output/workspace/k8s/deployment.yaml +75 -0
  166. package/bone/output/workspace/migrations/audit_log.sql +18 -0
  167. package/bone/output/workspace/migrations/codebase.sql +34 -0
  168. package/bone/output/workspace/migrations/event_outbox.sql +31 -0
  169. package/bone/output/workspace/migrations/snapshot.sql +32 -0
  170. package/bone/output/workspace/migrations/workspace.sql +33 -0
  171. package/bone/output/workspace/openapi.yaml +721 -0
  172. package/bone/output/workspace/package.json +36 -0
  173. package/bone/output/workspace/schema.graphql +153 -0
  174. package/bone/output/workspace/sdk/client.ts +155 -0
  175. package/bone/output/workspace/src/algorithms.ts +2 -0
  176. package/bone/output/workspace/src/audit.ts +37 -0
  177. package/bone/output/workspace/src/auth.ts +57 -0
  178. package/bone/output/workspace/src/cron.ts +12 -0
  179. package/bone/output/workspace/src/db.ts +32 -0
  180. package/bone/output/workspace/src/debug.ts +66 -0
  181. package/bone/output/workspace/src/events.ts +243 -0
  182. package/bone/output/workspace/src/extensions.ts +44 -0
  183. package/bone/output/workspace/src/failure_rules.ts +153 -0
  184. package/bone/output/workspace/src/health.ts +43 -0
  185. package/bone/output/workspace/src/index.ts +89 -0
  186. package/bone/output/workspace/src/logger.ts +66 -0
  187. package/bone/output/workspace/src/metrics.ts +75 -0
  188. package/bone/output/workspace/src/migrate.ts +220 -0
  189. package/bone/output/workspace/src/migration_diff.ts +108 -0
  190. package/bone/output/workspace/src/notify.ts +73 -0
  191. package/bone/output/workspace/src/routes/codebase.ts +87 -0
  192. package/bone/output/workspace/src/routes/snapshot.ts +127 -0
  193. package/bone/output/workspace/src/routes/workspace.ts +190 -0
  194. package/bone/output/workspace/src/schemas.ts +231 -0
  195. package/bone/output/workspace/src/state_machines/codebase.ts +21 -0
  196. package/bone/output/workspace/src/state_machines/snapshot.ts +20 -0
  197. package/bone/output/workspace/src/state_machines/workspace.ts +21 -0
  198. package/bone/output/workspace/src/tests.ts +249 -0
  199. package/bone/output/workspace/tsconfig.json +25 -0
  200. package/compat/opencode_adapter.ts +410 -0
  201. package/package.json +69 -0
  202. package/scripts/check_benchmark_session.js +34 -0
  203. package/scripts/check_finish_event.js +24 -0
  204. package/scripts/check_parts.js +15 -0
  205. package/scripts/compile.js +79 -0
  206. package/scripts/copy_opencode.ps1 +53 -0
  207. package/scripts/create_functions.sql +129 -0
  208. package/scripts/migrate.js +85 -0
  209. package/scripts/migrate_from_opencode.ts +218 -0
  210. package/scripts/test_agent_loop.js +101 -0
  211. package/scripts/test_api.ps1 +116 -0
  212. package/scripts/test_context_builder.js +136 -0
  213. package/scripts/test_context_builder.ts +97 -0
  214. package/scripts/test_rag.js +189 -0
  215. package/scripts/test_stream_events.js +36 -0
  216. package/scripts/test_websocket_and_saga.js +216 -0
  217. package/src/cli.ts +475 -0
  218. package/src/config.ts +162 -0
  219. package/src/context_builder.ts +598 -0
  220. package/src/engine/account/account.sql.ts +39 -0
  221. package/src/engine/account/account.ts +456 -0
  222. package/src/engine/account/repo.ts +166 -0
  223. package/src/engine/account/schema.ts +99 -0
  224. package/src/engine/account/url.ts +8 -0
  225. package/src/engine/acp/README.md +174 -0
  226. package/src/engine/acp/agent.ts +1968 -0
  227. package/src/engine/acp/runtime.ts +22 -0
  228. package/src/engine/acp/session.ts +122 -0
  229. package/src/engine/acp/types.ts +24 -0
  230. package/src/engine/agent/agent.ts +463 -0
  231. package/src/engine/agent/generate.txt +75 -0
  232. package/src/engine/agent/prompt/compaction.txt +9 -0
  233. package/src/engine/agent/prompt/explore.txt +18 -0
  234. package/src/engine/agent/prompt/scout.txt +36 -0
  235. package/src/engine/agent/prompt/summary.txt +11 -0
  236. package/src/engine/agent/prompt/title.txt +44 -0
  237. package/src/engine/agent/subagent-permissions.ts +34 -0
  238. package/src/engine/auth/index.ts +96 -0
  239. package/src/engine/background/background/job.ts +200 -0
  240. package/src/engine/background/job.ts +200 -0
  241. package/src/engine/bus/bus-event.ts +45 -0
  242. package/src/engine/bus/global.ts +22 -0
  243. package/src/engine/bus/index.ts +203 -0
  244. package/src/engine/command/command/index.ts +181 -0
  245. package/src/engine/command/command/template/initialize.txt +66 -0
  246. package/src/engine/command/command/template/review.txt +101 -0
  247. package/src/engine/command/index.ts +181 -0
  248. package/src/engine/command/template/initialize.txt +66 -0
  249. package/src/engine/command/template/review.txt +101 -0
  250. package/src/engine/config/agent.ts +172 -0
  251. package/src/engine/config/attachment.ts +25 -0
  252. package/src/engine/config/command.ts +62 -0
  253. package/src/engine/config/config.ts +833 -0
  254. package/src/engine/config/console-state.ts +14 -0
  255. package/src/engine/config/entry-name.ts +16 -0
  256. package/src/engine/config/error.ts +23 -0
  257. package/src/engine/config/formatter.ts +13 -0
  258. package/src/engine/config/layout.ts +6 -0
  259. package/src/engine/config/lsp.ts +43 -0
  260. package/src/engine/config/managed.ts +71 -0
  261. package/src/engine/config/markdown.ts +96 -0
  262. package/src/engine/config/mcp.ts +56 -0
  263. package/src/engine/config/model-id.ts +5 -0
  264. package/src/engine/config/parse.ts +79 -0
  265. package/src/engine/config/paths.ts +45 -0
  266. package/src/engine/config/permission.ts +58 -0
  267. package/src/engine/config/plugin.ts +84 -0
  268. package/src/engine/config/provider.ts +111 -0
  269. package/src/engine/config/reference.ts +23 -0
  270. package/src/engine/config/server.ts +19 -0
  271. package/src/engine/config/skills.ts +14 -0
  272. package/src/engine/config/variable.ts +90 -0
  273. package/src/engine/control-plane/adapters/index.ts +41 -0
  274. package/src/engine/control-plane/adapters/worktree.ts +96 -0
  275. package/src/engine/control-plane/dev/README.md +19 -0
  276. package/src/engine/control-plane/dev/debug-workspace-plugin.ts +73 -0
  277. package/src/engine/control-plane/schema.ts +14 -0
  278. package/src/engine/control-plane/types.ts +59 -0
  279. package/src/engine/control-plane/util.ts +39 -0
  280. package/src/engine/control-plane/workspace-adapter-runtime.ts +51 -0
  281. package/src/engine/control-plane/workspace-context.ts +26 -0
  282. package/src/engine/control-plane/workspace.sql.ts +20 -0
  283. package/src/engine/control-plane/workspace.ts +1072 -0
  284. package/src/engine/data-migration.ts +161 -0
  285. package/src/engine/effect/app-runtime.ts +143 -0
  286. package/src/engine/effect/bootstrap-runtime.ts +29 -0
  287. package/src/engine/effect/bridge.ts +84 -0
  288. package/src/engine/effect/config-service.ts +67 -0
  289. package/src/engine/effect/instance-ref.ts +11 -0
  290. package/src/engine/effect/instance-registry.ts +12 -0
  291. package/src/engine/effect/instance-state.ts +72 -0
  292. package/src/engine/effect/promise.ts +17 -0
  293. package/src/engine/effect/run-service.ts +47 -0
  294. package/src/engine/effect/runner.ts +217 -0
  295. package/src/engine/effect/runtime-flags.ts +74 -0
  296. package/src/engine/effect/service-use.ts +38 -0
  297. package/src/engine/env/index.ts +37 -0
  298. package/src/engine/event-v2-bridge.ts +89 -0
  299. package/src/engine/file/file/ignore.ts +81 -0
  300. package/src/engine/file/file/index.ts +651 -0
  301. package/src/engine/file/file/protected.ts +59 -0
  302. package/src/engine/file/file/ripgrep.ts +481 -0
  303. package/src/engine/file/file/watcher.ts +167 -0
  304. package/src/engine/file/ignore.ts +81 -0
  305. package/src/engine/file/index.ts +651 -0
  306. package/src/engine/file/protected.ts +59 -0
  307. package/src/engine/file/ripgrep.ts +481 -0
  308. package/src/engine/file/watcher.ts +167 -0
  309. package/src/engine/format/format/formatter.ts +404 -0
  310. package/src/engine/format/format/index.ts +209 -0
  311. package/src/engine/format/formatter.ts +404 -0
  312. package/src/engine/format/index.ts +209 -0
  313. package/src/engine/git/git/index.ts +347 -0
  314. package/src/engine/git/index.ts +347 -0
  315. package/src/engine/id/id.ts +80 -0
  316. package/src/engine/ide/index.ts +70 -0
  317. package/src/engine/image/image/image.ts +176 -0
  318. package/src/engine/image/image.ts +176 -0
  319. package/src/engine/index.ts +251 -0
  320. package/src/engine/installation/index.ts +327 -0
  321. package/src/engine/lsp/client.ts +707 -0
  322. package/src/engine/lsp/diagnostic.ts +29 -0
  323. package/src/engine/lsp/language.ts +121 -0
  324. package/src/engine/lsp/launch.ts +21 -0
  325. package/src/engine/lsp/lsp/client.ts +707 -0
  326. package/src/engine/lsp/lsp/diagnostic.ts +29 -0
  327. package/src/engine/lsp/lsp/language.ts +121 -0
  328. package/src/engine/lsp/lsp/launch.ts +21 -0
  329. package/src/engine/lsp/lsp/lsp.ts +507 -0
  330. package/src/engine/lsp/lsp/server.ts +2064 -0
  331. package/src/engine/lsp/lsp.ts +507 -0
  332. package/src/engine/lsp/server.ts +2064 -0
  333. package/src/engine/mcp/auth.ts +146 -0
  334. package/src/engine/mcp/index.ts +958 -0
  335. package/src/engine/mcp/mcp/auth.ts +146 -0
  336. package/src/engine/mcp/mcp/index.ts +958 -0
  337. package/src/engine/mcp/mcp/oauth-callback.ts +232 -0
  338. package/src/engine/mcp/mcp/oauth-provider.ts +214 -0
  339. package/src/engine/mcp/oauth-callback.ts +232 -0
  340. package/src/engine/mcp/oauth-provider.ts +214 -0
  341. package/src/engine/node.ts +6 -0
  342. package/src/engine/patch/index.ts +689 -0
  343. package/src/engine/patch/patch/index.ts +689 -0
  344. package/src/engine/permission/arity.ts +163 -0
  345. package/src/engine/permission/evaluate.ts +15 -0
  346. package/src/engine/permission/index.ts +306 -0
  347. package/src/engine/permission/permission/arity.ts +163 -0
  348. package/src/engine/permission/permission/evaluate.ts +15 -0
  349. package/src/engine/permission/permission/index.ts +306 -0
  350. package/src/engine/permission/permission/schema.ts +13 -0
  351. package/src/engine/permission/schema.ts +13 -0
  352. package/src/engine/plugin/azure.ts +26 -0
  353. package/src/engine/plugin/cloudflare.ts +76 -0
  354. package/src/engine/plugin/codex.ts +622 -0
  355. package/src/engine/plugin/digitalocean.ts +411 -0
  356. package/src/engine/plugin/github-copilot/copilot.ts +394 -0
  357. package/src/engine/plugin/github-copilot/models.ts +196 -0
  358. package/src/engine/plugin/index.ts +295 -0
  359. package/src/engine/plugin/install.ts +439 -0
  360. package/src/engine/plugin/loader.ts +216 -0
  361. package/src/engine/plugin/meta.ts +188 -0
  362. package/src/engine/plugin/shared.ts +323 -0
  363. package/src/engine/project/bootstrap-service.ts +9 -0
  364. package/src/engine/project/bootstrap.ts +75 -0
  365. package/src/engine/project/instance-context.ts +24 -0
  366. package/src/engine/project/instance-layer.ts +11 -0
  367. package/src/engine/project/instance-runtime.ts +16 -0
  368. package/src/engine/project/instance-store.ts +193 -0
  369. package/src/engine/project/project.sql.ts +17 -0
  370. package/src/engine/project/project.ts +537 -0
  371. package/src/engine/project/schema.ts +13 -0
  372. package/src/engine/project/vcs.ts +405 -0
  373. package/src/engine/provider/auth.ts +225 -0
  374. package/src/engine/provider/error.ts +204 -0
  375. package/src/engine/provider/model-status.ts +8 -0
  376. package/src/engine/provider/provider.ts +1843 -0
  377. package/src/engine/provider/schema.ts +30 -0
  378. package/src/engine/provider/sdk/copilot/AGENTS.md +1 -0
  379. package/src/engine/provider/transform.ts +1376 -0
  380. package/src/engine/pty/index.ts +365 -0
  381. package/src/engine/pty/input.ts +24 -0
  382. package/src/engine/pty/pty/index.ts +365 -0
  383. package/src/engine/pty/pty/input.ts +24 -0
  384. package/src/engine/pty/pty/pty.bun.ts +26 -0
  385. package/src/engine/pty/pty/pty.node.ts +27 -0
  386. package/src/engine/pty/pty/pty.ts +25 -0
  387. package/src/engine/pty/pty/schema.ts +14 -0
  388. package/src/engine/pty/pty/ticket.ts +68 -0
  389. package/src/engine/pty/pty.bun.ts +26 -0
  390. package/src/engine/pty/pty.node.ts +27 -0
  391. package/src/engine/pty/pty.ts +25 -0
  392. package/src/engine/pty/schema.ts +14 -0
  393. package/src/engine/pty/ticket.ts +68 -0
  394. package/src/engine/question/index.ts +213 -0
  395. package/src/engine/question/question/index.ts +213 -0
  396. package/src/engine/question/question/schema.ts +10 -0
  397. package/src/engine/question/schema.ts +10 -0
  398. package/src/engine/reference/reference/reference.ts +241 -0
  399. package/src/engine/reference/reference/repository-cache.ts +147 -0
  400. package/src/engine/reference/reference.ts +241 -0
  401. package/src/engine/reference/repository-cache.ts +147 -0
  402. package/src/engine/session/compaction.ts +651 -0
  403. package/src/engine/session/compaction_logic.ts +120 -0
  404. package/src/engine/session/instruction.ts +238 -0
  405. package/src/engine/session/instruction_loader.ts +54 -0
  406. package/src/engine/session/llm.ts +459 -0
  407. package/src/engine/session/message-error.ts +14 -0
  408. package/src/engine/session/message-v2.ts +1202 -0
  409. package/src/engine/session/message.ts +146 -0
  410. package/src/engine/session/overflow.ts +32 -0
  411. package/src/engine/session/overflow_check.ts +46 -0
  412. package/src/engine/session/processor.ts +823 -0
  413. package/src/engine/session/prompt/anthropic.txt +105 -0
  414. package/src/engine/session/prompt/beast.txt +147 -0
  415. package/src/engine/session/prompt/build-switch.txt +5 -0
  416. package/src/engine/session/prompt/codex.txt +79 -0
  417. package/src/engine/session/prompt/copilot-gpt-5.txt +143 -0
  418. package/src/engine/session/prompt/default.txt +105 -0
  419. package/src/engine/session/prompt/gemini.txt +155 -0
  420. package/src/engine/session/prompt/gpt.txt +107 -0
  421. package/src/engine/session/prompt/kimi.txt +95 -0
  422. package/src/engine/session/prompt/max-steps.txt +16 -0
  423. package/src/engine/session/prompt/plan-reminder-anthropic.txt +67 -0
  424. package/src/engine/session/prompt/plan.txt +26 -0
  425. package/src/engine/session/prompt/trinity.txt +97 -0
  426. package/src/engine/session/prompt.ts +671 -0
  427. package/src/engine/session/provider_transform.ts +187 -0
  428. package/src/engine/session/retry.ts +200 -0
  429. package/src/engine/session/retry_logic.ts +65 -0
  430. package/src/engine/session/revert.ts +162 -0
  431. package/src/engine/session/run-state.ts +153 -0
  432. package/src/engine/session/schema.ts +26 -0
  433. package/src/engine/session/session.sql.ts +137 -0
  434. package/src/engine/session/session.ts +1011 -0
  435. package/src/engine/session/status.ts +94 -0
  436. package/src/engine/session/summary.ts +164 -0
  437. package/src/engine/session/system.ts +84 -0
  438. package/src/engine/session/system_prompt.ts +65 -0
  439. package/src/engine/session/todo.ts +81 -0
  440. package/src/engine/session/tool_registry.ts +162 -0
  441. package/src/engine/share/session.ts +61 -0
  442. package/src/engine/share/share-next.ts +376 -0
  443. package/src/engine/share/share.sql.ts +13 -0
  444. package/src/engine/shell/shell/shell.ts +215 -0
  445. package/src/engine/shell/shell.ts +215 -0
  446. package/src/engine/skill/discovery.ts +116 -0
  447. package/src/engine/skill/index.ts +336 -0
  448. package/src/engine/skill/prompt/customize-opencode.md +377 -0
  449. package/src/engine/skill/skill/discovery.ts +116 -0
  450. package/src/engine/skill/skill/index.ts +336 -0
  451. package/src/engine/skill/skill/prompt/customize-opencode.md +377 -0
  452. package/src/engine/snapshot/index.ts +762 -0
  453. package/src/engine/snapshot/snapshot/index.ts +762 -0
  454. package/src/engine/sync/README.md +179 -0
  455. package/src/engine/sync/event.sql.ts +17 -0
  456. package/src/engine/sync/index.ts +410 -0
  457. package/src/engine/sync/schema.ts +11 -0
  458. package/src/engine/temporary.ts +33 -0
  459. package/src/engine/tool/apply_patch.ts +313 -0
  460. package/src/engine/tool/apply_patch.txt +33 -0
  461. package/src/engine/tool/edit.ts +711 -0
  462. package/src/engine/tool/edit.txt +10 -0
  463. package/src/engine/tool/external-directory.ts +49 -0
  464. package/src/engine/tool/glob.ts +103 -0
  465. package/src/engine/tool/glob.txt +6 -0
  466. package/src/engine/tool/grep.ts +156 -0
  467. package/src/engine/tool/grep.txt +8 -0
  468. package/src/engine/tool/invalid.ts +21 -0
  469. package/src/engine/tool/json-schema.ts +164 -0
  470. package/src/engine/tool/lsp.ts +113 -0
  471. package/src/engine/tool/lsp.txt +24 -0
  472. package/src/engine/tool/mcp-websearch.ts +96 -0
  473. package/src/engine/tool/plan-enter.txt +14 -0
  474. package/src/engine/tool/plan-exit.txt +13 -0
  475. package/src/engine/tool/plan.ts +78 -0
  476. package/src/engine/tool/question.ts +44 -0
  477. package/src/engine/tool/question.txt +10 -0
  478. package/src/engine/tool/read.ts +337 -0
  479. package/src/engine/tool/read.txt +14 -0
  480. package/src/engine/tool/registry.ts +472 -0
  481. package/src/engine/tool/repo_clone.ts +80 -0
  482. package/src/engine/tool/repo_clone.txt +5 -0
  483. package/src/engine/tool/repo_overview.ts +279 -0
  484. package/src/engine/tool/repo_overview.txt +4 -0
  485. package/src/engine/tool/schema.ts +14 -0
  486. package/src/engine/tool/shell/id.ts +19 -0
  487. package/src/engine/tool/shell/prompt.ts +295 -0
  488. package/src/engine/tool/shell/shell.txt +77 -0
  489. package/src/engine/tool/shell.ts +647 -0
  490. package/src/engine/tool/skill.ts +75 -0
  491. package/src/engine/tool/skill.txt +5 -0
  492. package/src/engine/tool/task.ts +337 -0
  493. package/src/engine/tool/task.txt +58 -0
  494. package/src/engine/tool/task_status.ts +179 -0
  495. package/src/engine/tool/task_status.txt +13 -0
  496. package/src/engine/tool/todo.ts +57 -0
  497. package/src/engine/tool/todowrite.txt +167 -0
  498. package/src/engine/tool/tool/apply_patch.ts +313 -0
  499. package/src/engine/tool/tool/apply_patch.txt +33 -0
  500. package/src/engine/tool/tool/edit.ts +711 -0
  501. package/src/engine/tool/tool/edit.txt +10 -0
  502. package/src/engine/tool/tool/external-directory.ts +49 -0
  503. package/src/engine/tool/tool/glob.ts +103 -0
  504. package/src/engine/tool/tool/glob.txt +6 -0
  505. package/src/engine/tool/tool/grep.ts +156 -0
  506. package/src/engine/tool/tool/grep.txt +8 -0
  507. package/src/engine/tool/tool/invalid.ts +21 -0
  508. package/src/engine/tool/tool/json-schema.ts +164 -0
  509. package/src/engine/tool/tool/lsp.ts +113 -0
  510. package/src/engine/tool/tool/lsp.txt +24 -0
  511. package/src/engine/tool/tool/mcp-websearch.ts +96 -0
  512. package/src/engine/tool/tool/plan-enter.txt +14 -0
  513. package/src/engine/tool/tool/plan-exit.txt +13 -0
  514. package/src/engine/tool/tool/plan.ts +78 -0
  515. package/src/engine/tool/tool/question.ts +44 -0
  516. package/src/engine/tool/tool/question.txt +10 -0
  517. package/src/engine/tool/tool/read.ts +337 -0
  518. package/src/engine/tool/tool/read.txt +14 -0
  519. package/src/engine/tool/tool/registry.ts +472 -0
  520. package/src/engine/tool/tool/repo_clone.ts +80 -0
  521. package/src/engine/tool/tool/repo_clone.txt +5 -0
  522. package/src/engine/tool/tool/repo_overview.ts +279 -0
  523. package/src/engine/tool/tool/repo_overview.txt +4 -0
  524. package/src/engine/tool/tool/schema.ts +14 -0
  525. package/src/engine/tool/tool/shell/id.ts +19 -0
  526. package/src/engine/tool/tool/shell/prompt.ts +295 -0
  527. package/src/engine/tool/tool/shell/shell.txt +77 -0
  528. package/src/engine/tool/tool/shell.ts +647 -0
  529. package/src/engine/tool/tool/skill.ts +75 -0
  530. package/src/engine/tool/tool/skill.txt +5 -0
  531. package/src/engine/tool/tool/task.ts +337 -0
  532. package/src/engine/tool/tool/task.txt +58 -0
  533. package/src/engine/tool/tool/task_status.ts +179 -0
  534. package/src/engine/tool/tool/task_status.txt +13 -0
  535. package/src/engine/tool/tool/todo.ts +57 -0
  536. package/src/engine/tool/tool/todowrite.txt +167 -0
  537. package/src/engine/tool/tool/tool.ts +164 -0
  538. package/src/engine/tool/tool/truncate.ts +160 -0
  539. package/src/engine/tool/tool/truncation-dir.ts +4 -0
  540. package/src/engine/tool/tool/webfetch.ts +192 -0
  541. package/src/engine/tool/tool/webfetch.txt +13 -0
  542. package/src/engine/tool/tool/websearch.ts +143 -0
  543. package/src/engine/tool/tool/websearch.txt +14 -0
  544. package/src/engine/tool/tool/write.ts +104 -0
  545. package/src/engine/tool/tool/write.txt +8 -0
  546. package/src/engine/tool/tool.ts +164 -0
  547. package/src/engine/tool/truncate.ts +160 -0
  548. package/src/engine/tool/truncation-dir.ts +4 -0
  549. package/src/engine/tool/webfetch.ts +192 -0
  550. package/src/engine/tool/webfetch.txt +13 -0
  551. package/src/engine/tool/websearch.ts +143 -0
  552. package/src/engine/tool/websearch.txt +14 -0
  553. package/src/engine/tool/write.ts +104 -0
  554. package/src/engine/tool/write.txt +8 -0
  555. package/src/engine/util/archive.ts +17 -0
  556. package/src/engine/util/bom.ts +31 -0
  557. package/src/engine/util/data-url.ts +9 -0
  558. package/src/engine/util/defer.ts +10 -0
  559. package/src/engine/util/effect-http-client.ts +11 -0
  560. package/src/engine/util/error.ts +88 -0
  561. package/src/engine/util/filesystem.ts +252 -0
  562. package/src/engine/util/format.ts +20 -0
  563. package/src/engine/util/iife.ts +3 -0
  564. package/src/engine/util/lazy.ts +20 -0
  565. package/src/engine/util/local-context.ts +25 -0
  566. package/src/engine/util/locale.ts +86 -0
  567. package/src/engine/util/media.ts +26 -0
  568. package/src/engine/util/process.ts +176 -0
  569. package/src/engine/util/queue.ts +32 -0
  570. package/src/engine/util/record.ts +3 -0
  571. package/src/engine/util/repository.ts +158 -0
  572. package/src/engine/util/rpc.ts +66 -0
  573. package/src/engine/util/signal.ts +12 -0
  574. package/src/engine/util/timeout.ts +13 -0
  575. package/src/engine/util/token.ts +7 -0
  576. package/src/engine/util/util/archive.ts +17 -0
  577. package/src/engine/util/util/bom.ts +31 -0
  578. package/src/engine/util/util/data-url.ts +9 -0
  579. package/src/engine/util/util/defer.ts +10 -0
  580. package/src/engine/util/util/effect-http-client.ts +11 -0
  581. package/src/engine/util/util/error.ts +88 -0
  582. package/src/engine/util/util/filesystem.ts +252 -0
  583. package/src/engine/util/util/format.ts +20 -0
  584. package/src/engine/util/util/iife.ts +3 -0
  585. package/src/engine/util/util/lazy.ts +20 -0
  586. package/src/engine/util/util/local-context.ts +25 -0
  587. package/src/engine/util/util/locale.ts +86 -0
  588. package/src/engine/util/util/media.ts +26 -0
  589. package/src/engine/util/util/process.ts +176 -0
  590. package/src/engine/util/util/queue.ts +32 -0
  591. package/src/engine/util/util/record.ts +3 -0
  592. package/src/engine/util/util/repository.ts +158 -0
  593. package/src/engine/util/util/rpc.ts +66 -0
  594. package/src/engine/util/util/signal.ts +12 -0
  595. package/src/engine/util/util/timeout.ts +13 -0
  596. package/src/engine/util/util/token.ts +7 -0
  597. package/src/engine/util/util/which.ts +14 -0
  598. package/src/engine/util/util/wildcard.ts +59 -0
  599. package/src/engine/util/which.ts +14 -0
  600. package/src/engine/util/wildcard.ts +59 -0
  601. package/src/engine/worktree/index.ts +621 -0
  602. package/src/rag_worker.ts +519 -0
  603. package/src/server.ts +201 -0
  604. package/src/tui.ts +637 -0
  605. package/tsconfig.json +24 -0
@@ -0,0 +1,762 @@
1
+ import { Cause, Duration, Effect, Layer, Schedule, Schema, Semaphore, Context } from "effect"
2
+ import { ChildProcess, ChildProcessSpawner } from "effect/unstable/process"
3
+ import { formatPatch, structuredPatch } from "diff"
4
+ import path from "path"
5
+ import { AppProcess } from "@opencode-ai/core/process"
6
+ import { InstanceState } from "@/effect/instance-state"
7
+ import { AppFileSystem } from "@opencode-ai/core/filesystem"
8
+ import { Hash } from "@opencode-ai/core/util/hash"
9
+ import { Config } from "@/config/config"
10
+ import { Global } from "@opencode-ai/core/global"
11
+ import * as Log from "@opencode-ai/core/util/log"
12
+
13
+ export const Patch = Schema.Struct({
14
+ hash: Schema.String,
15
+ files: Schema.mutable(Schema.Array(Schema.String)),
16
+ })
17
+ export type Patch = typeof Patch.Type
18
+
19
+ export const FileDiff = Schema.Struct({
20
+ // Optional because legacy/imported `summary_diffs` on disk may omit
21
+ // file details and patch text. Required Schema rejected the whole
22
+ // session response and broke session loading on Desktop.
23
+ file: Schema.optional(Schema.String),
24
+ patch: Schema.optional(Schema.String),
25
+ additions: Schema.Finite,
26
+ deletions: Schema.Finite,
27
+ status: Schema.optional(Schema.Literals(["added", "deleted", "modified"])),
28
+ }).annotate({ identifier: "SnapshotFileDiff" })
29
+ export type FileDiff = typeof FileDiff.Type
30
+
31
+ const log = Log.create({ service: "snapshot" })
32
+ const prune = "7.days"
33
+ const limit = 2 * 1024 * 1024
34
+ const core = ["-c", "core.longpaths=true", "-c", "core.symlinks=true"]
35
+ const cfg = ["-c", "core.autocrlf=false", ...core]
36
+ const quote = [...cfg, "-c", "core.quotepath=false"]
37
+ interface GitResult {
38
+ readonly code: ChildProcessSpawner.ExitCode
39
+ readonly text: string
40
+ readonly stderr: string
41
+ }
42
+
43
+ type State = Omit<Interface, "init">
44
+
45
+ export interface Interface {
46
+ readonly init: () => Effect.Effect<void>
47
+ readonly cleanup: () => Effect.Effect<void>
48
+ readonly track: () => Effect.Effect<string | undefined>
49
+ readonly patch: (hash: string) => Effect.Effect<Patch>
50
+ readonly restore: (snapshot: string) => Effect.Effect<void>
51
+ readonly revert: (patches: Patch[]) => Effect.Effect<void>
52
+ readonly diff: (hash: string) => Effect.Effect<string>
53
+ readonly diffFull: (from: string, to: string) => Effect.Effect<FileDiff[]>
54
+ }
55
+
56
+ export class Service extends Context.Service<Service, Interface>()("@opencode/Snapshot") {}
57
+
58
+ export const layer: Layer.Layer<Service, never, AppFileSystem.Service | AppProcess.Service | Config.Service> =
59
+ Layer.effect(
60
+ Service,
61
+ Effect.gen(function* () {
62
+ const fs = yield* AppFileSystem.Service
63
+ const appProcess = yield* AppProcess.Service
64
+ const config = yield* Config.Service
65
+ const locks = new Map<string, Semaphore.Semaphore>()
66
+
67
+ const lock = (key: string) => {
68
+ const hit = locks.get(key)
69
+ if (hit) return hit
70
+
71
+ const next = Semaphore.makeUnsafe(1)
72
+ locks.set(key, next)
73
+ return next
74
+ }
75
+
76
+ const state = yield* InstanceState.make<State>(
77
+ Effect.fn("Snapshot.state")(function* (ctx) {
78
+ const state = {
79
+ directory: ctx.directory,
80
+ worktree: ctx.worktree,
81
+ gitdir: path.join(Global.Path.data, "snapshot", ctx.project.id, Hash.fast(ctx.worktree)),
82
+ vcs: ctx.project.vcs,
83
+ }
84
+
85
+ const args = (cmd: string[]) => ["--git-dir", state.gitdir, "--work-tree", state.worktree, ...cmd]
86
+
87
+ const feed = (list: string[]) => list.join("\0") + "\0"
88
+
89
+ const git = Effect.fnUntraced(
90
+ function* (cmd: string[], opts?: { cwd?: string; env?: Record<string, string>; stdin?: string }) {
91
+ const result = yield* appProcess.run(
92
+ ChildProcess.make("git", cmd, { cwd: opts?.cwd, env: opts?.env, extendEnv: true }),
93
+ { stdin: opts?.stdin },
94
+ )
95
+ return {
96
+ code: ChildProcessSpawner.ExitCode(result.exitCode),
97
+ text: result.stdout.toString("utf8"),
98
+ stderr: result.stderr.toString("utf8"),
99
+ } satisfies GitResult
100
+ },
101
+ Effect.catch((err) =>
102
+ Effect.succeed({
103
+ code: ChildProcessSpawner.ExitCode(1),
104
+ text: "",
105
+ stderr: err instanceof Error ? err.message : String(err),
106
+ }),
107
+ ),
108
+ )
109
+
110
+ const ignore = Effect.fnUntraced(function* (files: string[]) {
111
+ if (!files.length) return new Set<string>()
112
+ const check = yield* git(
113
+ [
114
+ ...quote,
115
+ "--git-dir",
116
+ path.join(state.worktree, ".git"),
117
+ "--work-tree",
118
+ state.worktree,
119
+ "check-ignore",
120
+ "--no-index",
121
+ "--stdin",
122
+ "-z",
123
+ ],
124
+ {
125
+ cwd: state.directory,
126
+ stdin: feed(files),
127
+ },
128
+ )
129
+ if (check.code !== 0 && check.code !== 1) return new Set<string>()
130
+ return new Set(check.text.split("\0").filter(Boolean))
131
+ })
132
+
133
+ const drop = Effect.fnUntraced(function* (files: string[]) {
134
+ if (!files.length) return
135
+ yield* git(
136
+ [
137
+ ...cfg,
138
+ ...args(["rm", "--cached", "-f", "--ignore-unmatch", "--pathspec-from-file=-", "--pathspec-file-nul"]),
139
+ ],
140
+ {
141
+ cwd: state.directory,
142
+ stdin: feed(files),
143
+ },
144
+ )
145
+ })
146
+
147
+ const stage = Effect.fnUntraced(function* (files: string[]) {
148
+ if (!files.length) return
149
+ const result = yield* git(
150
+ [...cfg, ...args(["add", "--all", "--sparse", "--pathspec-from-file=-", "--pathspec-file-nul"])],
151
+ {
152
+ cwd: state.directory,
153
+ stdin: feed(files),
154
+ },
155
+ )
156
+ if (result.code === 0) return
157
+ log.warn("failed to add snapshot files", {
158
+ exitCode: result.code,
159
+ stderr: result.stderr,
160
+ })
161
+ })
162
+
163
+ const exists = (file: string) => fs.exists(file).pipe(Effect.orDie)
164
+ const read = (file: string) => fs.readFileString(file).pipe(Effect.catch(() => Effect.succeed("")))
165
+ const remove = (file: string) => fs.remove(file).pipe(Effect.catch(() => Effect.void))
166
+ const locked = <A, E, R>(fx: Effect.Effect<A, E, R>) => lock(state.gitdir).withPermits(1)(fx)
167
+
168
+ const enabled = Effect.fnUntraced(function* () {
169
+ if (state.vcs !== "git") return false
170
+ return (yield* config.get()).snapshot !== false
171
+ })
172
+
173
+ const excludes = Effect.fnUntraced(function* () {
174
+ const result = yield* git(["rev-parse", "--path-format=absolute", "--git-path", "info/exclude"], {
175
+ cwd: state.worktree,
176
+ })
177
+ const file = result.text.trim()
178
+ if (!file) return
179
+ if (!(yield* exists(file))) return
180
+ return file
181
+ })
182
+
183
+ const sync = Effect.fnUntraced(function* (list: string[] = []) {
184
+ const file = yield* excludes()
185
+ const target = path.join(state.gitdir, "info", "exclude")
186
+ const text = [
187
+ file ? (yield* read(file)).trimEnd() : "",
188
+ ...list.map((item) => `/${item.replaceAll("\\", "/")}`),
189
+ ]
190
+ .filter(Boolean)
191
+ .join("\n")
192
+ yield* fs.ensureDir(path.join(state.gitdir, "info")).pipe(Effect.orDie)
193
+ yield* fs.writeFileString(target, text ? `${text}\n` : "").pipe(Effect.orDie)
194
+ })
195
+
196
+ const add = Effect.fnUntraced(function* () {
197
+ yield* sync()
198
+ const [diff, other] = yield* Effect.all(
199
+ [
200
+ git([...quote, ...args(["diff-files", "--name-only", "-z", "--", "."])], {
201
+ cwd: state.directory,
202
+ }),
203
+ git([...quote, ...args(["ls-files", "--others", "--exclude-standard", "-z", "--", "."])], {
204
+ cwd: state.directory,
205
+ }),
206
+ ],
207
+ { concurrency: 2 },
208
+ )
209
+ if (diff.code !== 0 || other.code !== 0) {
210
+ log.warn("failed to list snapshot files", {
211
+ diffCode: diff.code,
212
+ diffStderr: diff.stderr,
213
+ otherCode: other.code,
214
+ otherStderr: other.stderr,
215
+ })
216
+ return
217
+ }
218
+
219
+ const tracked = diff.text.split("\0").filter(Boolean)
220
+ const untracked = other.text.split("\0").filter(Boolean)
221
+ const all = Array.from(new Set([...tracked, ...untracked]))
222
+ if (!all.length) return
223
+
224
+ // Resolve source-repo ignore rules against the exact candidate set.
225
+ // --no-index keeps this pattern-based even when a path is already tracked.
226
+ const ignored = yield* ignore(all)
227
+
228
+ // Remove newly-ignored files from snapshot index to prevent re-adding
229
+ if (ignored.size > 0) {
230
+ const ignoredFiles = Array.from(ignored)
231
+ log.info("removing gitignored files from snapshot", { count: ignoredFiles.length })
232
+ yield* drop(ignoredFiles)
233
+ }
234
+
235
+ const allow = all.filter((item) => !ignored.has(item))
236
+ if (!allow.length) return
237
+
238
+ const large = new Set(
239
+ (yield* Effect.all(
240
+ allow.map((item) =>
241
+ fs
242
+ .stat(path.join(state.directory, item))
243
+ .pipe(Effect.catch(() => Effect.void))
244
+ .pipe(
245
+ Effect.map((stat) => {
246
+ if (!stat || stat.type !== "File") return
247
+ const size = typeof stat.size === "bigint" ? Number(stat.size) : stat.size
248
+ return size > limit ? item : undefined
249
+ }),
250
+ ),
251
+ ),
252
+ { concurrency: 8 },
253
+ )).filter((item): item is string => Boolean(item)),
254
+ )
255
+ const block = new Set(untracked.filter((item) => large.has(item)))
256
+ yield* sync(Array.from(block))
257
+ // Stage only the allowed candidate paths so snapshot updates stay scoped.
258
+ yield* stage(allow.filter((item) => !block.has(item)))
259
+ })
260
+
261
+ const cleanup = Effect.fnUntraced(function* () {
262
+ return yield* locked(
263
+ Effect.gen(function* () {
264
+ if (!(yield* enabled())) return
265
+ if (!(yield* exists(state.gitdir))) return
266
+ const result = yield* git(args(["gc", `--prune=${prune}`]), { cwd: state.directory })
267
+ if (result.code !== 0) {
268
+ log.warn("cleanup failed", {
269
+ exitCode: result.code,
270
+ stderr: result.stderr,
271
+ })
272
+ return
273
+ }
274
+ log.info("cleanup", { prune })
275
+ }),
276
+ )
277
+ })
278
+
279
+ const track = Effect.fnUntraced(function* () {
280
+ return yield* locked(
281
+ Effect.gen(function* () {
282
+ if (!(yield* enabled())) return
283
+ const existed = yield* exists(state.gitdir)
284
+ yield* fs.ensureDir(state.gitdir).pipe(Effect.orDie)
285
+ if (!existed) {
286
+ yield* git(["init"], {
287
+ env: { GIT_DIR: state.gitdir, GIT_WORK_TREE: state.worktree },
288
+ })
289
+ yield* git(["--git-dir", state.gitdir, "config", "core.autocrlf", "false"])
290
+ yield* git(["--git-dir", state.gitdir, "config", "core.longpaths", "true"])
291
+ yield* git(["--git-dir", state.gitdir, "config", "core.symlinks", "true"])
292
+ yield* git(["--git-dir", state.gitdir, "config", "core.fsmonitor", "false"])
293
+ log.info("initialized")
294
+ }
295
+ yield* add()
296
+ const result = yield* git(args(["write-tree"]), { cwd: state.directory })
297
+ const hash = result.text.trim()
298
+ log.info("tracking", { hash, cwd: state.directory, git: state.gitdir })
299
+ return hash
300
+ }),
301
+ )
302
+ })
303
+
304
+ const patch = Effect.fnUntraced(function* (hash: string) {
305
+ return yield* locked(
306
+ Effect.gen(function* () {
307
+ yield* add()
308
+ const result = yield* git(
309
+ [...quote, ...args(["diff", "--cached", "--no-ext-diff", "--name-only", hash, "--", "."])],
310
+ {
311
+ cwd: state.directory,
312
+ },
313
+ )
314
+ if (result.code !== 0) {
315
+ log.warn("failed to get diff", { hash, exitCode: result.code })
316
+ return { hash, files: [] }
317
+ }
318
+ const files = result.text
319
+ .trim()
320
+ .split("\n")
321
+ .map((x) => x.trim())
322
+ .filter(Boolean)
323
+
324
+ // Hide ignored-file removals from the user-facing patch output.
325
+ const ignored = yield* ignore(files)
326
+
327
+ return {
328
+ hash,
329
+ files: files
330
+ .filter((item) => !ignored.has(item))
331
+ .map((x) => path.join(state.worktree, x).replaceAll("\\", "/")),
332
+ }
333
+ }),
334
+ )
335
+ })
336
+
337
+ const restore = Effect.fnUntraced(function* (snapshot: string) {
338
+ return yield* locked(
339
+ Effect.gen(function* () {
340
+ log.info("restore", { commit: snapshot })
341
+ const result = yield* git([...core, ...args(["read-tree", snapshot])], { cwd: state.worktree })
342
+ if (result.code === 0) {
343
+ const checkout = yield* git([...core, ...args(["checkout-index", "-a", "-f"])], {
344
+ cwd: state.worktree,
345
+ })
346
+ if (checkout.code === 0) return
347
+ log.error("failed to restore snapshot", {
348
+ snapshot,
349
+ exitCode: checkout.code,
350
+ stderr: checkout.stderr,
351
+ })
352
+ return
353
+ }
354
+ log.error("failed to restore snapshot", {
355
+ snapshot,
356
+ exitCode: result.code,
357
+ stderr: result.stderr,
358
+ })
359
+ }),
360
+ )
361
+ })
362
+
363
+ const revert = Effect.fnUntraced(function* (patches: Patch[]) {
364
+ return yield* locked(
365
+ Effect.gen(function* () {
366
+ const ops: { hash: string; file: string; rel: string }[] = []
367
+ const seen = new Set<string>()
368
+ for (const item of patches) {
369
+ for (const file of item.files) {
370
+ if (seen.has(file)) continue
371
+ seen.add(file)
372
+ ops.push({
373
+ hash: item.hash,
374
+ file,
375
+ rel: path.relative(state.worktree, file).replaceAll("\\", "/"),
376
+ })
377
+ }
378
+ }
379
+
380
+ const single = Effect.fnUntraced(function* (op: (typeof ops)[number]) {
381
+ log.info("reverting", { file: op.file, hash: op.hash })
382
+ const result = yield* git([...core, ...args(["checkout", op.hash, "--", op.file])], {
383
+ cwd: state.worktree,
384
+ })
385
+ if (result.code === 0) return
386
+ const tree = yield* git([...core, ...args(["ls-tree", op.hash, "--", op.rel])], {
387
+ cwd: state.worktree,
388
+ })
389
+ if (tree.code === 0 && tree.text.trim()) {
390
+ log.info("file existed in snapshot but checkout failed, keeping", { file: op.file, hash: op.hash })
391
+ return
392
+ }
393
+ log.info("file did not exist in snapshot, deleting", { file: op.file, hash: op.hash })
394
+ yield* remove(op.file)
395
+ })
396
+
397
+ const clash = (a: string, b: string) => a === b || a.startsWith(`${b}/`) || b.startsWith(`${a}/`)
398
+
399
+ for (let i = 0; i < ops.length; ) {
400
+ const first = ops[i]!
401
+ const run = [first]
402
+ let j = i + 1
403
+ // Only batch adjacent files when their paths cannot affect each other.
404
+ while (j < ops.length && run.length < 100) {
405
+ const next = ops[j]!
406
+ if (next.hash !== first.hash) break
407
+ if (run.some((item) => clash(item.rel, next.rel))) break
408
+ run.push(next)
409
+ j += 1
410
+ }
411
+
412
+ if (run.length === 1) {
413
+ yield* single(first)
414
+ i = j
415
+ continue
416
+ }
417
+
418
+ const tree = yield* git(
419
+ [...core, ...args(["ls-tree", "--name-only", first.hash, "--", ...run.map((item) => item.rel)])],
420
+ {
421
+ cwd: state.worktree,
422
+ },
423
+ )
424
+
425
+ if (tree.code !== 0) {
426
+ log.info("batched ls-tree failed, falling back to single-file revert", {
427
+ hash: first.hash,
428
+ files: run.length,
429
+ })
430
+ for (const op of run) {
431
+ yield* single(op)
432
+ }
433
+ i = j
434
+ continue
435
+ }
436
+
437
+ const have = new Set(
438
+ tree.text
439
+ .trim()
440
+ .split("\n")
441
+ .map((item) => item.trim())
442
+ .filter(Boolean),
443
+ )
444
+ const list = run.filter((item) => have.has(item.rel))
445
+ if (list.length) {
446
+ log.info("reverting", { hash: first.hash, files: list.length })
447
+ const result = yield* git(
448
+ [...core, ...args(["checkout", first.hash, "--", ...list.map((item) => item.file)])],
449
+ {
450
+ cwd: state.worktree,
451
+ },
452
+ )
453
+ if (result.code !== 0) {
454
+ log.info("batched checkout failed, falling back to single-file revert", {
455
+ hash: first.hash,
456
+ files: list.length,
457
+ })
458
+ for (const op of run) {
459
+ yield* single(op)
460
+ }
461
+ i = j
462
+ continue
463
+ }
464
+ }
465
+
466
+ for (const op of run) {
467
+ if (have.has(op.rel)) continue
468
+ log.info("file did not exist in snapshot, deleting", { file: op.file, hash: op.hash })
469
+ yield* remove(op.file)
470
+ }
471
+
472
+ i = j
473
+ }
474
+ }),
475
+ )
476
+ })
477
+
478
+ const diff = Effect.fnUntraced(function* (hash: string) {
479
+ return yield* locked(
480
+ Effect.gen(function* () {
481
+ yield* add()
482
+ const result = yield* git([...quote, ...args(["diff", "--cached", "--no-ext-diff", hash, "--", "."])], {
483
+ cwd: state.worktree,
484
+ })
485
+ if (result.code !== 0) {
486
+ log.warn("failed to get diff", {
487
+ hash,
488
+ exitCode: result.code,
489
+ stderr: result.stderr,
490
+ })
491
+ return ""
492
+ }
493
+ return result.text.trim()
494
+ }),
495
+ )
496
+ })
497
+
498
+ const diffFull = Effect.fnUntraced(function* (from: string, to: string) {
499
+ return yield* locked(
500
+ Effect.gen(function* () {
501
+ type Row = {
502
+ file: string
503
+ status: "added" | "deleted" | "modified"
504
+ binary: boolean
505
+ additions: number
506
+ deletions: number
507
+ }
508
+
509
+ type Ref = {
510
+ file: string
511
+ side: "before" | "after"
512
+ ref: string
513
+ }
514
+
515
+ const show = Effect.fnUntraced(function* (row: Row) {
516
+ if (row.binary) return ["", ""]
517
+ if (row.status === "added") {
518
+ return [
519
+ "",
520
+ yield* git([...cfg, ...args(["show", `${to}:${row.file}`])]).pipe(
521
+ Effect.map((item) => item.text),
522
+ ),
523
+ ]
524
+ }
525
+ if (row.status === "deleted") {
526
+ return [
527
+ yield* git([...cfg, ...args(["show", `${from}:${row.file}`])]).pipe(
528
+ Effect.map((item) => item.text),
529
+ ),
530
+ "",
531
+ ]
532
+ }
533
+ return yield* Effect.all(
534
+ [
535
+ git([...cfg, ...args(["show", `${from}:${row.file}`])]).pipe(Effect.map((item) => item.text)),
536
+ git([...cfg, ...args(["show", `${to}:${row.file}`])]).pipe(Effect.map((item) => item.text)),
537
+ ],
538
+ { concurrency: 2 },
539
+ )
540
+ })
541
+
542
+ const load = Effect.fnUntraced(
543
+ function* (rows: Row[]) {
544
+ const refs = rows.flatMap((row) => {
545
+ if (row.binary) return []
546
+ if (row.status === "added")
547
+ return [{ file: row.file, side: "after", ref: `${to}:${row.file}` } satisfies Ref]
548
+ if (row.status === "deleted") {
549
+ return [{ file: row.file, side: "before", ref: `${from}:${row.file}` } satisfies Ref]
550
+ }
551
+ return [
552
+ { file: row.file, side: "before", ref: `${from}:${row.file}` } satisfies Ref,
553
+ { file: row.file, side: "after", ref: `${to}:${row.file}` } satisfies Ref,
554
+ ]
555
+ })
556
+ if (!refs.length) return new Map<string, { before: string; after: string }>()
557
+
558
+ const batch = yield* appProcess.run(
559
+ ChildProcess.make("git", [...cfg, ...args(["cat-file", "--batch"])], {
560
+ cwd: state.directory,
561
+ extendEnv: true,
562
+ }),
563
+ { stdin: refs.map((item) => item.ref).join("\n") + "\n" },
564
+ )
565
+ if (batch.exitCode !== 0) {
566
+ log.info("git cat-file --batch failed during snapshot diff, falling back to per-file git show", {
567
+ stderr: batch.stderr.toString("utf8"),
568
+ refs: refs.length,
569
+ })
570
+ return
571
+ }
572
+ const out = batch.stdout
573
+
574
+ const fail = (msg: string, extra?: Record<string, string>) => {
575
+ log.info(msg, { ...extra, refs: refs.length })
576
+ return undefined
577
+ }
578
+
579
+ const map = new Map<string, { before: string; after: string }>()
580
+ const dec = new TextDecoder()
581
+ let i = 0
582
+ for (const ref of refs) {
583
+ let end = i
584
+ while (end < out.length && out[end] !== 10) end += 1
585
+ if (end >= out.length) {
586
+ return fail(
587
+ "git cat-file --batch returned a truncated header during snapshot diff, falling back to per-file git show",
588
+ )
589
+ }
590
+
591
+ const head = dec.decode(out.slice(i, end))
592
+ i = end + 1
593
+ const hit = map.get(ref.file) ?? { before: "", after: "" }
594
+ if (head.endsWith(" missing")) {
595
+ map.set(ref.file, hit)
596
+ continue
597
+ }
598
+
599
+ const match = head.match(/^[0-9a-f]+ blob (\d+)$/)
600
+ if (!match) {
601
+ return fail(
602
+ "git cat-file --batch returned an unexpected header during snapshot diff, falling back to per-file git show",
603
+ { head },
604
+ )
605
+ }
606
+
607
+ const size = Number(match[1])
608
+ if (!Number.isInteger(size) || size < 0 || i + size >= out.length || out[i + size] !== 10) {
609
+ return fail(
610
+ "git cat-file --batch returned truncated content during snapshot diff, falling back to per-file git show",
611
+ { head },
612
+ )
613
+ }
614
+
615
+ const text = dec.decode(out.slice(i, i + size))
616
+ if (ref.side === "before") hit.before = text
617
+ if (ref.side === "after") hit.after = text
618
+ map.set(ref.file, hit)
619
+ i += size + 1
620
+ }
621
+
622
+ if (i !== out.length) {
623
+ return fail(
624
+ "git cat-file --batch returned trailing data during snapshot diff, falling back to per-file git show",
625
+ )
626
+ }
627
+
628
+ return map
629
+ },
630
+ Effect.scoped,
631
+ Effect.catch(() =>
632
+ Effect.succeed<Map<string, { before: string; after: string }> | undefined>(undefined),
633
+ ),
634
+ )
635
+
636
+ const result: FileDiff[] = []
637
+ const status = new Map<string, "added" | "deleted" | "modified">()
638
+
639
+ const statuses = yield* git(
640
+ [...quote, ...args(["diff", "--no-ext-diff", "--name-status", "--no-renames", from, to, "--", "."])],
641
+ { cwd: state.directory },
642
+ )
643
+
644
+ for (const line of statuses.text.trim().split("\n")) {
645
+ if (!line) continue
646
+ const [code, file] = line.split("\t")
647
+ if (!code || !file) continue
648
+ status.set(file, code.startsWith("A") ? "added" : code.startsWith("D") ? "deleted" : "modified")
649
+ }
650
+
651
+ const numstat = yield* git(
652
+ [...quote, ...args(["diff", "--no-ext-diff", "--no-renames", "--numstat", from, to, "--", "."])],
653
+ {
654
+ cwd: state.directory,
655
+ },
656
+ )
657
+
658
+ const rows = numstat.text
659
+ .trim()
660
+ .split("\n")
661
+ .filter(Boolean)
662
+ .flatMap((line) => {
663
+ const [adds, dels, file] = line.split("\t")
664
+ if (!file) return []
665
+ const binary = adds === "-" && dels === "-"
666
+ const additions = binary ? 0 : parseInt(adds)
667
+ const deletions = binary ? 0 : parseInt(dels)
668
+ return [
669
+ {
670
+ file,
671
+ status: status.get(file) ?? "modified",
672
+ binary,
673
+ additions: Number.isFinite(additions) ? additions : 0,
674
+ deletions: Number.isFinite(deletions) ? deletions : 0,
675
+ } satisfies Row,
676
+ ]
677
+ })
678
+
679
+ // Hide ignored-file removals from the user-facing diff output.
680
+ const ignored = yield* ignore(rows.map((r) => r.file))
681
+ if (ignored.size > 0) {
682
+ const filtered = rows.filter((r) => !ignored.has(r.file))
683
+ rows.length = 0
684
+ rows.push(...filtered)
685
+ }
686
+
687
+ const step = 100
688
+ const patch = (file: string, before: string, after: string) =>
689
+ formatPatch(structuredPatch(file, file, before, after, "", "", { context: Number.MAX_SAFE_INTEGER }))
690
+
691
+ for (let i = 0; i < rows.length; i += step) {
692
+ const run = rows.slice(i, i + step)
693
+ const text = yield* load(run)
694
+
695
+ for (const row of run) {
696
+ const hit = text?.get(row.file) ?? { before: "", after: "" }
697
+ const [before, after] = row.binary ? ["", ""] : text ? [hit.before, hit.after] : yield* show(row)
698
+ result.push({
699
+ file: row.file,
700
+ patch: row.binary ? "" : patch(row.file, before, after),
701
+ additions: row.additions,
702
+ deletions: row.deletions,
703
+ status: row.status,
704
+ })
705
+ }
706
+ }
707
+
708
+ return result
709
+ }),
710
+ )
711
+ })
712
+
713
+ yield* cleanup().pipe(
714
+ Effect.catchCause((cause) => {
715
+ log.error("cleanup loop failed", { cause: Cause.pretty(cause) })
716
+ return Effect.void
717
+ }),
718
+ Effect.repeat(Schedule.spaced(Duration.hours(1))),
719
+ Effect.delay(Duration.minutes(1)),
720
+ Effect.forkScoped,
721
+ )
722
+
723
+ return { cleanup, track, patch, restore, revert, diff, diffFull }
724
+ }),
725
+ )
726
+
727
+ return Service.of({
728
+ init: Effect.fn("Snapshot.init")(function* () {
729
+ yield* InstanceState.get(state)
730
+ }),
731
+ cleanup: Effect.fn("Snapshot.cleanup")(function* () {
732
+ return yield* InstanceState.useEffect(state, (s) => s.cleanup())
733
+ }),
734
+ track: Effect.fn("Snapshot.track")(function* () {
735
+ return yield* InstanceState.useEffect(state, (s) => s.track())
736
+ }),
737
+ patch: Effect.fn("Snapshot.patch")(function* (hash: string) {
738
+ return yield* InstanceState.useEffect(state, (s) => s.patch(hash))
739
+ }),
740
+ restore: Effect.fn("Snapshot.restore")(function* (snapshot: string) {
741
+ return yield* InstanceState.useEffect(state, (s) => s.restore(snapshot))
742
+ }),
743
+ revert: Effect.fn("Snapshot.revert")(function* (patches: Patch[]) {
744
+ return yield* InstanceState.useEffect(state, (s) => s.revert(patches))
745
+ }),
746
+ diff: Effect.fn("Snapshot.diff")(function* (hash: string) {
747
+ return yield* InstanceState.useEffect(state, (s) => s.diff(hash))
748
+ }),
749
+ diffFull: Effect.fn("Snapshot.diffFull")(function* (from: string, to: string) {
750
+ return yield* InstanceState.useEffect(state, (s) => s.diffFull(from, to))
751
+ }),
752
+ })
753
+ }),
754
+ )
755
+
756
+ export const defaultLayer = layer.pipe(
757
+ Layer.provide(AppProcess.defaultLayer),
758
+ Layer.provide(AppFileSystem.defaultLayer),
759
+ Layer.provide(Config.defaultLayer),
760
+ )
761
+
762
+ export * as Snapshot from "."