onlycode 1.18.0 → 1.19.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 (417) hide show
  1. package/package.json +11 -146
  2. package/src/account/account.ts +0 -463
  3. package/src/account/repo.ts +0 -173
  4. package/src/account/schema.ts +0 -99
  5. package/src/account/url.ts +0 -8
  6. package/src/acp/agent.ts +0 -95
  7. package/src/acp/config-option.ts +0 -203
  8. package/src/acp/content.ts +0 -250
  9. package/src/acp/directory.ts +0 -210
  10. package/src/acp/error.ts +0 -90
  11. package/src/acp/event.ts +0 -342
  12. package/src/acp/permission.ts +0 -124
  13. package/src/acp/profile.ts +0 -42
  14. package/src/acp/service.ts +0 -1048
  15. package/src/acp/session.ts +0 -231
  16. package/src/acp/tool.ts +0 -367
  17. package/src/acp/usage.ts +0 -232
  18. package/src/agent/agent.ts +0 -459
  19. package/src/agent/generate.txt +0 -75
  20. package/src/agent/prompt/compaction.txt +0 -9
  21. package/src/agent/prompt/explore.txt +0 -18
  22. package/src/agent/prompt/summary.txt +0 -11
  23. package/src/agent/prompt/title.txt +0 -44
  24. package/src/agent/subagent-permissions.ts +0 -27
  25. package/src/audio.d.ts +0 -14
  26. package/src/auth/index.ts +0 -99
  27. package/src/background/job.ts +0 -39
  28. package/src/bus/global.ts +0 -22
  29. package/src/cli/bootstrap.ts +0 -11
  30. package/src/cli/cmd/account.ts +0 -264
  31. package/src/cli/cmd/acp.ts +0 -73
  32. package/src/cli/cmd/agent.ts +0 -259
  33. package/src/cli/cmd/attach.ts +0 -97
  34. package/src/cli/cmd/cmd.ts +0 -7
  35. package/src/cli/cmd/db.ts +0 -62
  36. package/src/cli/cmd/debug/agent.handler.ts +0 -193
  37. package/src/cli/cmd/debug/agent.ts +0 -27
  38. package/src/cli/cmd/debug/config.ts +0 -14
  39. package/src/cli/cmd/debug/file.ts +0 -73
  40. package/src/cli/cmd/debug/index.ts +0 -87
  41. package/src/cli/cmd/debug/lsp.ts +0 -50
  42. package/src/cli/cmd/debug/ripgrep.ts +0 -79
  43. package/src/cli/cmd/debug/scrap.ts +0 -15
  44. package/src/cli/cmd/debug/skill.ts +0 -15
  45. package/src/cli/cmd/debug/snapshot.ts +0 -50
  46. package/src/cli/cmd/debug/startup.ts +0 -11
  47. package/src/cli/cmd/debug/v2.ts +0 -44
  48. package/src/cli/cmd/export.ts +0 -292
  49. package/src/cli/cmd/generate.ts +0 -54
  50. package/src/cli/cmd/github.handler.ts +0 -1593
  51. package/src/cli/cmd/github.shared.ts +0 -30
  52. package/src/cli/cmd/github.ts +0 -42
  53. package/src/cli/cmd/import.ts +0 -224
  54. package/src/cli/cmd/mcp.ts +0 -849
  55. package/src/cli/cmd/models.ts +0 -66
  56. package/src/cli/cmd/plug.ts +0 -230
  57. package/src/cli/cmd/pr.ts +0 -115
  58. package/src/cli/cmd/prompt-display.ts +0 -1
  59. package/src/cli/cmd/providers.ts +0 -534
  60. package/src/cli/cmd/run/demo.ts +0 -1274
  61. package/src/cli/cmd/run/entry.body.ts +0 -205
  62. package/src/cli/cmd/run/footer.command.tsx +0 -1064
  63. package/src/cli/cmd/run/footer.menu.tsx +0 -351
  64. package/src/cli/cmd/run/footer.permission.tsx +0 -472
  65. package/src/cli/cmd/run/footer.prompt.tsx +0 -1306
  66. package/src/cli/cmd/run/footer.question.tsx +0 -573
  67. package/src/cli/cmd/run/footer.subagent.tsx +0 -173
  68. package/src/cli/cmd/run/footer.ts +0 -1129
  69. package/src/cli/cmd/run/footer.view.tsx +0 -943
  70. package/src/cli/cmd/run/footer.width.ts +0 -27
  71. package/src/cli/cmd/run/permission.shared.ts +0 -256
  72. package/src/cli/cmd/run/prompt.editor.ts +0 -157
  73. package/src/cli/cmd/run/prompt.shared.ts +0 -153
  74. package/src/cli/cmd/run/question.shared.ts +0 -340
  75. package/src/cli/cmd/run/runtime.boot.ts +0 -202
  76. package/src/cli/cmd/run/runtime.lifecycle.ts +0 -406
  77. package/src/cli/cmd/run/runtime.queue.ts +0 -349
  78. package/src/cli/cmd/run/runtime.shared.ts +0 -17
  79. package/src/cli/cmd/run/runtime.stdin.ts +0 -37
  80. package/src/cli/cmd/run/runtime.ts +0 -814
  81. package/src/cli/cmd/run/scrollback.shared.ts +0 -92
  82. package/src/cli/cmd/run/scrollback.surface.ts +0 -431
  83. package/src/cli/cmd/run/scrollback.writer.tsx +0 -352
  84. package/src/cli/cmd/run/session-data.ts +0 -1113
  85. package/src/cli/cmd/run/session-replay.ts +0 -374
  86. package/src/cli/cmd/run/session.shared.ts +0 -196
  87. package/src/cli/cmd/run/splash.ts +0 -275
  88. package/src/cli/cmd/run/stream.transport.ts +0 -1462
  89. package/src/cli/cmd/run/stream.ts +0 -175
  90. package/src/cli/cmd/run/subagent-data.ts +0 -876
  91. package/src/cli/cmd/run/theme.ts +0 -690
  92. package/src/cli/cmd/run/tool.ts +0 -1489
  93. package/src/cli/cmd/run/trace.ts +0 -94
  94. package/src/cli/cmd/run/turn-summary.ts +0 -47
  95. package/src/cli/cmd/run/types.ts +0 -350
  96. package/src/cli/cmd/run/variant.shared.ts +0 -215
  97. package/src/cli/cmd/run.ts +0 -894
  98. package/src/cli/cmd/serve.ts +0 -24
  99. package/src/cli/cmd/session.ts +0 -147
  100. package/src/cli/cmd/stats.ts +0 -393
  101. package/src/cli/cmd/tui.ts +0 -224
  102. package/src/cli/cmd/uninstall.ts +0 -353
  103. package/src/cli/cmd/upgrade.ts +0 -74
  104. package/src/cli/cmd/web.ts +0 -84
  105. package/src/cli/effect/prompt.ts +0 -37
  106. package/src/cli/effect-cmd.ts +0 -96
  107. package/src/cli/error.ts +0 -130
  108. package/src/cli/heap.ts +0 -45
  109. package/src/cli/logo-pixel.ts +0 -35
  110. package/src/cli/logo.ts +0 -1
  111. package/src/cli/network.ts +0 -64
  112. package/src/cli/tui/layer.ts +0 -7
  113. package/src/cli/tui/validate-session.ts +0 -29
  114. package/src/cli/tui/worker.ts +0 -71
  115. package/src/cli/ui.ts +0 -148
  116. package/src/cli/upgrade.ts +0 -53
  117. package/src/command/index.ts +0 -184
  118. package/src/command/template/initialize.txt +0 -66
  119. package/src/command/template/review.txt +0 -101
  120. package/src/config/agent.ts +0 -59
  121. package/src/config/command.ts +0 -39
  122. package/src/config/config.ts +0 -686
  123. package/src/config/entry-name.ts +0 -19
  124. package/src/config/managed.ts +0 -69
  125. package/src/config/markdown.ts +0 -36
  126. package/src/config/parse.ts +0 -79
  127. package/src/config/paths.ts +0 -45
  128. package/src/config/plugin.ts +0 -79
  129. package/src/config/tui-cwd.ts +0 -5
  130. package/src/config/tui-host-attention.ts +0 -21
  131. package/src/config/tui-migrate.ts +0 -132
  132. package/src/config/tui.ts +0 -274
  133. package/src/config/variable.ts +0 -91
  134. package/src/control-plane/adapters/index.ts +0 -41
  135. package/src/control-plane/adapters/worktree.ts +0 -96
  136. package/src/control-plane/dev/README.md +0 -19
  137. package/src/control-plane/dev/debug-workspace-plugin.ts +0 -73
  138. package/src/control-plane/types.ts +0 -59
  139. package/src/control-plane/util.ts +0 -39
  140. package/src/control-plane/workspace-adapter-runtime.ts +0 -51
  141. package/src/control-plane/workspace-context.ts +0 -26
  142. package/src/control-plane/workspace.ts +0 -989
  143. package/src/effect/app-runtime.ts +0 -132
  144. package/src/effect/bootstrap-runtime.ts +0 -23
  145. package/src/effect/bridge.ts +0 -84
  146. package/src/effect/config-service.ts +0 -67
  147. package/src/effect/instance-ref.ts +0 -11
  148. package/src/effect/instance-registry.ts +0 -12
  149. package/src/effect/instance-state.ts +0 -69
  150. package/src/effect/promise.ts +0 -17
  151. package/src/effect/run-service.ts +0 -47
  152. package/src/effect/runner.ts +0 -217
  153. package/src/effect/runtime-flags.ts +0 -79
  154. package/src/env/index.ts +0 -43
  155. package/src/event-v2-bridge.ts +0 -79
  156. package/src/format/formatter.ts +0 -404
  157. package/src/format/index.ts +0 -205
  158. package/src/git/index.ts +0 -350
  159. package/src/id/id.ts +0 -80
  160. package/src/ide/index.ts +0 -61
  161. package/src/image/image.ts +0 -174
  162. package/src/index.ts +0 -142
  163. package/src/installation/index.ts +0 -350
  164. package/src/lsp/client.ts +0 -650
  165. package/src/lsp/diagnostic.ts +0 -29
  166. package/src/lsp/language.ts +0 -121
  167. package/src/lsp/launch.ts +0 -21
  168. package/src/lsp/lsp.ts +0 -511
  169. package/src/lsp/server.ts +0 -1983
  170. package/src/markdown.d.ts +0 -4
  171. package/src/mcp/auth.ts +0 -174
  172. package/src/mcp/catalog.ts +0 -153
  173. package/src/mcp/index.ts +0 -953
  174. package/src/mcp/oauth-callback.ts +0 -233
  175. package/src/mcp/oauth-provider.ts +0 -206
  176. package/src/node.ts +0 -4
  177. package/src/patch/index.ts +0 -686
  178. package/src/permission/arity.ts +0 -163
  179. package/src/permission/evaluate.ts +0 -1
  180. package/src/permission/index.ts +0 -230
  181. package/src/plugin/azure.ts +0 -26
  182. package/src/plugin/cloudflare.ts +0 -76
  183. package/src/plugin/digitalocean.ts +0 -383
  184. package/src/plugin/github-copilot/copilot.ts +0 -414
  185. package/src/plugin/github-copilot/models.ts +0 -246
  186. package/src/plugin/index.ts +0 -316
  187. package/src/plugin/install.ts +0 -439
  188. package/src/plugin/loader.ts +0 -237
  189. package/src/plugin/meta.ts +0 -188
  190. package/src/plugin/openai/README.md +0 -31
  191. package/src/plugin/openai/codex.ts +0 -641
  192. package/src/plugin/openai/ws-pool.ts +0 -270
  193. package/src/plugin/openai/ws.ts +0 -381
  194. package/src/plugin/pty-environment.ts +0 -24
  195. package/src/plugin/shared.ts +0 -323
  196. package/src/plugin/snowflake-cortex.ts +0 -529
  197. package/src/plugin/tui/internal.ts +0 -10
  198. package/src/plugin/tui/runtime.ts +0 -1130
  199. package/src/plugin/xai.ts +0 -716
  200. package/src/project/bootstrap-service.ts +0 -9
  201. package/src/project/bootstrap.ts +0 -76
  202. package/src/project/instance-context.ts +0 -24
  203. package/src/project/instance-layer.ts +0 -11
  204. package/src/project/instance-runtime.ts +0 -16
  205. package/src/project/instance-store.ts +0 -209
  206. package/src/project/project.ts +0 -519
  207. package/src/project/vcs.ts +0 -431
  208. package/src/provider/auth.ts +0 -233
  209. package/src/provider/error.ts +0 -188
  210. package/src/provider/model-status.ts +0 -8
  211. package/src/provider/provider.ts +0 -1975
  212. package/src/provider/transform.ts +0 -1543
  213. package/src/question/index.ts +0 -229
  214. package/src/question/schema.ts +0 -10
  215. package/src/server/auth.ts +0 -48
  216. package/src/server/event.ts +0 -13
  217. package/src/server/global-lifecycle.ts +0 -28
  218. package/src/server/init-projectors.ts +0 -3
  219. package/src/server/mdns.ts +0 -47
  220. package/src/server/projectors.ts +0 -1
  221. package/src/server/proxy-util.ts +0 -48
  222. package/src/server/routes/instance/httpapi/AGENTS.md +0 -39
  223. package/src/server/routes/instance/httpapi/api.ts +0 -78
  224. package/src/server/routes/instance/httpapi/errors.ts +0 -193
  225. package/src/server/routes/instance/httpapi/groups/config.ts +0 -65
  226. package/src/server/routes/instance/httpapi/groups/control-plane.ts +0 -35
  227. package/src/server/routes/instance/httpapi/groups/control.ts +0 -76
  228. package/src/server/routes/instance/httpapi/groups/event.ts +0 -29
  229. package/src/server/routes/instance/httpapi/groups/experimental.ts +0 -275
  230. package/src/server/routes/instance/httpapi/groups/file.ts +0 -185
  231. package/src/server/routes/instance/httpapi/groups/global.ts +0 -138
  232. package/src/server/routes/instance/httpapi/groups/instance.ts +0 -206
  233. package/src/server/routes/instance/httpapi/groups/mcp.ts +0 -156
  234. package/src/server/routes/instance/httpapi/groups/metadata.ts +0 -18
  235. package/src/server/routes/instance/httpapi/groups/permission.ts +0 -61
  236. package/src/server/routes/instance/httpapi/groups/project-copy.ts +0 -32
  237. package/src/server/routes/instance/httpapi/groups/project.ts +0 -93
  238. package/src/server/routes/instance/httpapi/groups/provider.ts +0 -101
  239. package/src/server/routes/instance/httpapi/groups/pty.ts +0 -172
  240. package/src/server/routes/instance/httpapi/groups/query.ts +0 -12
  241. package/src/server/routes/instance/httpapi/groups/question.ts +0 -74
  242. package/src/server/routes/instance/httpapi/groups/session.ts +0 -462
  243. package/src/server/routes/instance/httpapi/groups/sync.ts +0 -113
  244. package/src/server/routes/instance/httpapi/groups/tui.ts +0 -208
  245. package/src/server/routes/instance/httpapi/groups/workspace.ts +0 -141
  246. package/src/server/routes/instance/httpapi/handlers/config.ts +0 -34
  247. package/src/server/routes/instance/httpapi/handlers/control-plane.ts +0 -37
  248. package/src/server/routes/instance/httpapi/handlers/control.ts +0 -43
  249. package/src/server/routes/instance/httpapi/handlers/event.ts +0 -99
  250. package/src/server/routes/instance/httpapi/handlers/experimental.ts +0 -192
  251. package/src/server/routes/instance/httpapi/handlers/file.ts +0 -139
  252. package/src/server/routes/instance/httpapi/handlers/global.ts +0 -156
  253. package/src/server/routes/instance/httpapi/handlers/instance.ts +0 -110
  254. package/src/server/routes/instance/httpapi/handlers/mcp.ts +0 -111
  255. package/src/server/routes/instance/httpapi/handlers/permission.ts +0 -41
  256. package/src/server/routes/instance/httpapi/handlers/project-copy.ts +0 -83
  257. package/src/server/routes/instance/httpapi/handlers/project.ts +0 -63
  258. package/src/server/routes/instance/httpapi/handlers/provider.ts +0 -113
  259. package/src/server/routes/instance/httpapi/handlers/pty.ts +0 -273
  260. package/src/server/routes/instance/httpapi/handlers/question.ts +0 -54
  261. package/src/server/routes/instance/httpapi/handlers/session-errors.ts +0 -21
  262. package/src/server/routes/instance/httpapi/handlers/session.ts +0 -440
  263. package/src/server/routes/instance/httpapi/handlers/sync.ts +0 -89
  264. package/src/server/routes/instance/httpapi/handlers/tui.ts +0 -131
  265. package/src/server/routes/instance/httpapi/handlers/workspace.ts +0 -102
  266. package/src/server/routes/instance/httpapi/lifecycle.ts +0 -54
  267. package/src/server/routes/instance/httpapi/middleware/authorization.ts +0 -150
  268. package/src/server/routes/instance/httpapi/middleware/compression.ts +0 -64
  269. package/src/server/routes/instance/httpapi/middleware/cors-vary.ts +0 -29
  270. package/src/server/routes/instance/httpapi/middleware/error.ts +0 -43
  271. package/src/server/routes/instance/httpapi/middleware/fence.ts +0 -25
  272. package/src/server/routes/instance/httpapi/middleware/instance-context.ts +0 -43
  273. package/src/server/routes/instance/httpapi/middleware/proxy.ts +0 -108
  274. package/src/server/routes/instance/httpapi/middleware/schema-error.ts +0 -41
  275. package/src/server/routes/instance/httpapi/middleware/workspace-routing.ts +0 -250
  276. package/src/server/routes/instance/httpapi/public.ts +0 -535
  277. package/src/server/routes/instance/httpapi/server.ts +0 -298
  278. package/src/server/routes/instance/httpapi/websocket-tracker.ts +0 -57
  279. package/src/server/server.ts +0 -225
  280. package/src/server/shared/fence.ts +0 -60
  281. package/src/server/shared/pty-ticket.ts +0 -15
  282. package/src/server/shared/public-ui.ts +0 -12
  283. package/src/server/shared/tui-control.ts +0 -28
  284. package/src/server/shared/ui.ts +0 -108
  285. package/src/server/shared/workspace-routing.ts +0 -38
  286. package/src/server/tui-event.ts +0 -53
  287. package/src/session/compaction.ts +0 -620
  288. package/src/session/instruction.ts +0 -241
  289. package/src/session/llm/AGENTS.md +0 -90
  290. package/src/session/llm/ai-sdk.ts +0 -288
  291. package/src/session/llm/native-request.ts +0 -196
  292. package/src/session/llm/native-runtime.ts +0 -195
  293. package/src/session/llm/request.ts +0 -216
  294. package/src/session/llm.ts +0 -415
  295. package/src/session/message-error.ts +0 -14
  296. package/src/session/message-v2.ts +0 -747
  297. package/src/session/message.ts +0 -148
  298. package/src/session/overflow.ts +0 -34
  299. package/src/session/processor.ts +0 -1084
  300. package/src/session/prompt/anthropic.txt +0 -109
  301. package/src/session/prompt/beast.txt +0 -151
  302. package/src/session/prompt/build-switch.txt +0 -9
  303. package/src/session/prompt/codex.txt +0 -83
  304. package/src/session/prompt/copilot-gpt-5.txt +0 -147
  305. package/src/session/prompt/default.txt +0 -99
  306. package/src/session/prompt/gemini.txt +0 -159
  307. package/src/session/prompt/gpt.txt +0 -111
  308. package/src/session/prompt/kimi.txt +0 -99
  309. package/src/session/prompt/plan-mode.txt +0 -74
  310. package/src/session/prompt/plan-reminder-anthropic.txt +0 -71
  311. package/src/session/prompt/plan.txt +0 -30
  312. package/src/session/prompt/trinity.txt +0 -101
  313. package/src/session/prompt.ts +0 -1707
  314. package/src/session/reminders.ts +0 -92
  315. package/src/session/retry.ts +0 -201
  316. package/src/session/revert.ts +0 -160
  317. package/src/session/run-state.ts +0 -156
  318. package/src/session/schema.ts +0 -26
  319. package/src/session/session.ts +0 -1119
  320. package/src/session/status.ts +0 -97
  321. package/src/session/summary.ts +0 -165
  322. package/src/session/system.ts +0 -117
  323. package/src/session/todo.ts +0 -90
  324. package/src/session/tools.ts +0 -207
  325. package/src/share/session.ts +0 -61
  326. package/src/share/share-next.ts +0 -385
  327. package/src/skill/discovery.ts +0 -109
  328. package/src/skill/index.ts +0 -366
  329. package/src/snapshot/index.ts +0 -808
  330. package/src/sql.d.ts +0 -4
  331. package/src/storage/schema.ts +0 -5
  332. package/src/storage/storage.ts +0 -329
  333. package/src/sync/README.md +0 -179
  334. package/src/sync/schema.ts +0 -11
  335. package/src/temporary.ts +0 -31
  336. package/src/tool/apply_patch.ts +0 -315
  337. package/src/tool/apply_patch.txt +0 -33
  338. package/src/tool/apply_patch.zh.txt +0 -33
  339. package/src/tool/description.ts +0 -100
  340. package/src/tool/edit.ts +0 -739
  341. package/src/tool/edit.txt +0 -10
  342. package/src/tool/edit.zh.txt +0 -10
  343. package/src/tool/external-directory.ts +0 -49
  344. package/src/tool/glob.ts +0 -78
  345. package/src/tool/glob.txt +0 -6
  346. package/src/tool/glob.zh.txt +0 -6
  347. package/src/tool/grep.ts +0 -114
  348. package/src/tool/grep.txt +0 -8
  349. package/src/tool/grep.zh.txt +0 -8
  350. package/src/tool/invalid.ts +0 -21
  351. package/src/tool/json-schema.ts +0 -164
  352. package/src/tool/lsp.ts +0 -115
  353. package/src/tool/lsp.txt +0 -24
  354. package/src/tool/lsp.zh.txt +0 -24
  355. package/src/tool/mcp-websearch.ts +0 -96
  356. package/src/tool/plan-enter.txt +0 -14
  357. package/src/tool/plan-enter.zh.txt +0 -14
  358. package/src/tool/plan-exit.txt +0 -13
  359. package/src/tool/plan-exit.zh.txt +0 -13
  360. package/src/tool/plan.ts +0 -81
  361. package/src/tool/question.ts +0 -46
  362. package/src/tool/question.txt +0 -10
  363. package/src/tool/question.zh.txt +0 -10
  364. package/src/tool/read.ts +0 -388
  365. package/src/tool/read.txt +0 -14
  366. package/src/tool/read.zh.txt +0 -14
  367. package/src/tool/registry.ts +0 -440
  368. package/src/tool/schema.ts +0 -14
  369. package/src/tool/shell/id.ts +0 -19
  370. package/src/tool/shell/prompt.ts +0 -307
  371. package/src/tool/shell/shell.txt +0 -21
  372. package/src/tool/shell.ts +0 -657
  373. package/src/tool/skill.ts +0 -73
  374. package/src/tool/skill.txt +0 -5
  375. package/src/tool/skill.zh.txt +0 -5
  376. package/src/tool/task.ts +0 -348
  377. package/src/tool/task.txt +0 -19
  378. package/src/tool/task.zh.txt +0 -19
  379. package/src/tool/todo.ts +0 -59
  380. package/src/tool/todowrite.txt +0 -44
  381. package/src/tool/todowrite.zh.txt +0 -44
  382. package/src/tool/tool.ts +0 -183
  383. package/src/tool/truncate.ts +0 -158
  384. package/src/tool/truncation-dir.ts +0 -4
  385. package/src/tool/webfetch.ts +0 -194
  386. package/src/tool/webfetch.txt +0 -13
  387. package/src/tool/webfetch.zh.txt +0 -13
  388. package/src/tool/websearch.ts +0 -145
  389. package/src/tool/websearch.txt +0 -14
  390. package/src/tool/websearch.zh.txt +0 -14
  391. package/src/tool/write.ts +0 -106
  392. package/src/tool/write.txt +0 -8
  393. package/src/tool/write.zh.txt +0 -8
  394. package/src/util/archive.ts +0 -17
  395. package/src/util/bom.ts +0 -27
  396. package/src/util/data-url.ts +0 -9
  397. package/src/util/defer.ts +0 -10
  398. package/src/util/effect-http-client.ts +0 -11
  399. package/src/util/error.ts +0 -1
  400. package/src/util/filesystem.ts +0 -251
  401. package/src/util/html.ts +0 -8
  402. package/src/util/iife.ts +0 -3
  403. package/src/util/lazy.ts +0 -20
  404. package/src/util/local-context.ts +0 -25
  405. package/src/util/locale.ts +0 -2
  406. package/src/util/media.ts +0 -26
  407. package/src/util/process.ts +0 -177
  408. package/src/util/proxy-env.ts +0 -72
  409. package/src/util/queue.ts +0 -32
  410. package/src/util/record.ts +0 -1
  411. package/src/util/repository.ts +0 -232
  412. package/src/util/rpc.ts +0 -66
  413. package/src/util/signal.ts +0 -12
  414. package/src/util/timeout.ts +0 -13
  415. package/src/util/token.ts +0 -1
  416. package/src/util/wildcard.ts +0 -59
  417. package/src/worktree/index.ts +0 -654
@@ -1,1119 +0,0 @@
1
- import { LayerNode } from "@opencode-ai/core/effect/layer-node"
2
- import { PermissionV1 } from "@opencode-ai/core/v1/permission"
3
- import { Slug } from "@opencode-ai/core/util/slug"
4
- import { SessionV1 } from "@opencode-ai/core/v1/session"
5
- import { serviceUse } from "@opencode-ai/core/effect/service-use"
6
- import path from "path"
7
- import { BackgroundJob } from "@/background/job"
8
- import { Decimal } from "decimal.js"
9
- import type { ProviderMetadata, Usage } from "@opencode-ai/llm"
10
- import { InstallationVersion } from "@opencode-ai/core/installation/version"
11
- import { Database } from "@opencode-ai/core/database/database"
12
- import { makeRuntime } from "@opencode-ai/core/effect/runtime"
13
- import { EventV2Bridge } from "@/event-v2-bridge"
14
- import { EventV2 } from "@opencode-ai/core/event"
15
- import { SessionV2 } from "@opencode-ai/core/session"
16
- import { SessionExecution } from "@opencode-ai/core/session/execution"
17
-
18
- import { NotFoundError } from "@/storage/storage"
19
- import { eq } from "drizzle-orm"
20
- import { and } from "drizzle-orm"
21
- import { gte } from "drizzle-orm"
22
- import { isNull } from "drizzle-orm"
23
- import { desc } from "drizzle-orm"
24
- import { like } from "drizzle-orm"
25
- import { sql } from "drizzle-orm"
26
- import { inArray } from "drizzle-orm"
27
- import { lt } from "drizzle-orm"
28
- import { or } from "drizzle-orm"
29
- import type { SQL } from "drizzle-orm"
30
- import { PartTable, SessionTable } from "@opencode-ai/core/session/sql"
31
- import { ProjectTable } from "@opencode-ai/core/project/sql"
32
- import { MessageV2 } from "./message-v2"
33
- import type { InstanceContext } from "../project/instance-context"
34
- import { InstanceState } from "@/effect/instance-state"
35
- import { Snapshot } from "@/snapshot"
36
- import { ProjectV2 } from "@opencode-ai/core/project"
37
- import { WorkspaceV2 } from "@opencode-ai/core/workspace"
38
- import { SessionID, MessageID, PartID } from "./schema"
39
-
40
- import type { Provider } from "@/provider/provider"
41
- import { Permission } from "@/permission"
42
- import { Global } from "@opencode-ai/core/global"
43
- import { Effect, Layer, Option, Context, Schema, Types } from "effect"
44
- import { NonNegativeInt, optionalOmitUndefined } from "@opencode-ai/core/schema"
45
- import { RuntimeFlags } from "@/effect/runtime-flags"
46
- import { ProviderV2 } from "@opencode-ai/core/provider"
47
- import { ModelV2 } from "@opencode-ai/core/model"
48
-
49
- const runtime = makeRuntime(Database.Service, Database.defaultLayer)
50
-
51
- const parentTitlePrefix = "New session - "
52
- const childTitlePrefix = "Child session - "
53
-
54
- export function isDefaultTitle(title: string) {
55
- return new RegExp(
56
- `^(${parentTitlePrefix}|${childTitlePrefix})\\d{4}-\\d{2}-\\d{2}T\\d{2}:\\d{2}:\\d{2}\\.\\d{3}Z$`,
57
- ).test(title)
58
- }
59
-
60
- type SessionRow = typeof SessionTable.$inferSelect
61
-
62
- export function fromRow(row: SessionRow): Info {
63
- const summary =
64
- row.summary_additions !== null || row.summary_deletions !== null || row.summary_files !== null
65
- ? {
66
- additions: row.summary_additions ?? 0,
67
- deletions: row.summary_deletions ?? 0,
68
- files: row.summary_files ?? 0,
69
- diffs: row.summary_diffs ?? undefined,
70
- }
71
- : undefined
72
- const share = row.share_url ? { url: row.share_url } : undefined
73
- const revert = row.revert ?? undefined
74
- return {
75
- id: row.id,
76
- slug: row.slug,
77
- projectID: row.project_id,
78
- workspaceID: row.workspace_id ?? undefined,
79
- directory: row.directory,
80
- path: row.path ?? undefined,
81
- parentID: row.parent_id ?? undefined,
82
- title: row.title,
83
- agent: row.agent ?? undefined,
84
- model: row.model
85
- ? {
86
- id: ModelV2.ID.make(row.model.id),
87
- providerID: ProviderV2.ID.make(row.model.providerID),
88
- variant: row.model.variant,
89
- }
90
- : undefined,
91
- version: row.version,
92
- summary,
93
- cost: row.cost,
94
- tokens: {
95
- input: row.tokens_input,
96
- output: row.tokens_output,
97
- reasoning: row.tokens_reasoning,
98
- cache: {
99
- read: row.tokens_cache_read,
100
- write: row.tokens_cache_write,
101
- },
102
- },
103
- share,
104
- metadata: row.metadata ?? undefined,
105
- revert,
106
- permission: row.permission ? [...row.permission] : undefined,
107
- time: {
108
- created: row.time_created,
109
- updated: row.time_updated,
110
- compacting: row.time_compacting ?? undefined,
111
- archived: row.time_archived ?? undefined,
112
- },
113
- }
114
- }
115
-
116
- export function toRow(info: Info) {
117
- return {
118
- id: info.id,
119
- project_id: info.projectID,
120
- workspace_id: info.workspaceID,
121
- parent_id: info.parentID,
122
- slug: info.slug,
123
- directory: info.directory,
124
- path: info.path,
125
- title: info.title,
126
- agent: info.agent,
127
- model: info.model,
128
- version: info.version,
129
- share_url: info.share?.url,
130
- summary_additions: info.summary?.additions,
131
- summary_deletions: info.summary?.deletions,
132
- summary_files: info.summary?.files,
133
- summary_diffs: info.summary?.diffs,
134
- metadata: info.metadata,
135
- cost: info.cost ?? 0,
136
- tokens_input: (info.tokens ?? EmptyTokens).input,
137
- tokens_output: (info.tokens ?? EmptyTokens).output,
138
- tokens_reasoning: (info.tokens ?? EmptyTokens).reasoning,
139
- tokens_cache_read: (info.tokens ?? EmptyTokens).cache.read,
140
- tokens_cache_write: (info.tokens ?? EmptyTokens).cache.write,
141
- revert: info.revert ?? null,
142
- permission: info.permission,
143
- time_created: info.time.created,
144
- time_updated: info.time.updated,
145
- time_compacting: info.time.compacting,
146
- time_archived: info.time.archived,
147
- }
148
- }
149
-
150
- function getForkedTitle(title: string): string {
151
- const match = title.match(/^(.+) \(fork #(\d+)\)$/)
152
- if (match) {
153
- const base = match[1]
154
- const num = parseInt(match[2], 10)
155
- return `${base} (fork #${num + 1})`
156
- }
157
- return `${title} (fork #1)`
158
- }
159
-
160
- function sessionPath(worktree: string, cwd: string) {
161
- return path.relative(path.resolve(worktree), cwd).replaceAll("\\", "/")
162
- }
163
-
164
- const Summary = Schema.Struct({
165
- additions: Schema.Finite,
166
- deletions: Schema.Finite,
167
- files: Schema.Finite,
168
- diffs: optionalOmitUndefined(Schema.Array(Snapshot.FileDiff)),
169
- })
170
-
171
- const Tokens = Schema.Struct({
172
- input: Schema.Finite,
173
- output: Schema.Finite,
174
- reasoning: Schema.Finite,
175
- cache: Schema.Struct({
176
- read: Schema.Finite,
177
- write: Schema.Finite,
178
- }),
179
- })
180
-
181
- const EmptyTokens = { input: 0, output: 0, reasoning: 0, cache: { read: 0, write: 0 } }
182
-
183
- const Share = Schema.Struct({
184
- url: Schema.String,
185
- })
186
-
187
- // Legacy HTTP accepted negative values here. Keep archive timestamps permissive
188
- // while excluding non-finite values that cannot round-trip through JSON.
189
- export const ArchivedTimestamp = Schema.Finite
190
-
191
- const Time = Schema.Struct({
192
- created: NonNegativeInt,
193
- updated: NonNegativeInt,
194
- compacting: optionalOmitUndefined(NonNegativeInt),
195
- archived: optionalOmitUndefined(ArchivedTimestamp),
196
- })
197
-
198
- const Revert = Schema.Struct({
199
- messageID: MessageID,
200
- partID: optionalOmitUndefined(PartID),
201
- snapshot: optionalOmitUndefined(Schema.String),
202
- diff: optionalOmitUndefined(Schema.String),
203
- })
204
-
205
- const Model = Schema.Struct({
206
- id: ModelV2.ID,
207
- providerID: ProviderV2.ID,
208
- variant: optionalOmitUndefined(Schema.String),
209
- })
210
-
211
- export const Metadata = Schema.Record(Schema.String, Schema.Any)
212
-
213
- export const Info = Schema.Struct({
214
- id: SessionID,
215
- slug: Schema.String,
216
- projectID: ProjectV2.ID,
217
- workspaceID: optionalOmitUndefined(WorkspaceV2.ID),
218
- directory: Schema.String,
219
- path: optionalOmitUndefined(Schema.String),
220
- parentID: optionalOmitUndefined(SessionID),
221
- summary: optionalOmitUndefined(Summary),
222
- cost: optionalOmitUndefined(Schema.Finite),
223
- tokens: optionalOmitUndefined(Tokens),
224
- share: optionalOmitUndefined(Share),
225
- title: Schema.String,
226
- agent: optionalOmitUndefined(Schema.String),
227
- model: optionalOmitUndefined(Model),
228
- version: Schema.String,
229
- metadata: optionalOmitUndefined(Metadata),
230
- time: Time,
231
- permission: optionalOmitUndefined(PermissionV1.Ruleset),
232
- revert: optionalOmitUndefined(Revert),
233
- }).annotate({ identifier: "Session" })
234
- export type Info = Types.DeepMutable<Schema.Schema.Type<typeof Info>>
235
-
236
- export const ProjectInfo = Schema.Struct({
237
- id: ProjectV2.ID,
238
- name: optionalOmitUndefined(Schema.String),
239
- worktree: Schema.String,
240
- }).annotate({ identifier: "ProjectSummary" })
241
- export type ProjectInfo = Types.DeepMutable<Schema.Schema.Type<typeof ProjectInfo>>
242
-
243
- export const GlobalInfo = Schema.Struct({
244
- ...Info.fields,
245
- project: Schema.NullOr(ProjectInfo),
246
- }).annotate({ identifier: "GlobalSession" })
247
- export type GlobalInfo = Types.DeepMutable<Schema.Schema.Type<typeof GlobalInfo>>
248
-
249
- export const CreateInput = Schema.optional(
250
- Schema.Struct({
251
- parentID: Schema.optional(SessionID),
252
- title: Schema.optional(Schema.String),
253
- agent: Schema.optional(Schema.String),
254
- model: Schema.optional(Model),
255
- metadata: Schema.optional(Metadata),
256
- permission: Schema.optional(PermissionV1.Ruleset),
257
- workspaceID: Schema.optional(WorkspaceV2.ID),
258
- }),
259
- )
260
- export type CreateInput = Types.DeepMutable<Schema.Schema.Type<typeof CreateInput>>
261
-
262
- export const ForkInput = Schema.Struct({
263
- sessionID: SessionID,
264
- messageID: Schema.optional(MessageID),
265
- })
266
- export const GetInput = SessionID
267
- export const ChildrenInput = SessionID
268
- export const RemoveInput = SessionID
269
- export const SetTitleInput = Schema.Struct({ sessionID: SessionID, title: Schema.String })
270
- export const SetArchivedInput = Schema.Struct({
271
- sessionID: SessionID,
272
- time: Schema.optional(ArchivedTimestamp),
273
- })
274
- export const SetMetadataInput = Schema.Struct({
275
- sessionID: SessionID,
276
- metadata: Metadata,
277
- })
278
- export const SetPermissionInput = Schema.Struct({
279
- sessionID: SessionID,
280
- permission: PermissionV1.Ruleset,
281
- })
282
- export const SetRevertInput = Schema.Struct({
283
- sessionID: SessionID,
284
- revert: Schema.optional(Revert),
285
- summary: Schema.optional(Summary),
286
- })
287
- export const MessagesInput = Schema.Struct({
288
- sessionID: SessionID,
289
- limit: Schema.optional(NonNegativeInt),
290
- })
291
- export type ListInput = {
292
- directory?: string
293
- scope?: "project"
294
- path?: string
295
- workspaceID?: WorkspaceV2.ID
296
- roots?: boolean
297
- start?: number
298
- search?: string
299
- limit?: number
300
- }
301
-
302
- export type GlobalListInput = {
303
- directory?: string
304
- roots?: boolean
305
- start?: number
306
- cursor?: number
307
- search?: string
308
- limit?: number
309
- archived?: boolean
310
- }
311
-
312
- const CreatedEventSchema = Schema.Struct({
313
- sessionID: SessionID,
314
- info: Info,
315
- })
316
-
317
- const UpdatedShare = Schema.Struct({
318
- url: Schema.optional(Schema.NullOr(Schema.String)),
319
- })
320
-
321
- const UpdatedTime = Schema.Struct({
322
- created: Schema.optional(Schema.NullOr(NonNegativeInt)),
323
- updated: Schema.optional(Schema.NullOr(NonNegativeInt)),
324
- compacting: Schema.optional(Schema.NullOr(NonNegativeInt)),
325
- archived: Schema.optional(Schema.NullOr(ArchivedTimestamp)),
326
- })
327
-
328
- const UpdatedInfo = Schema.Struct({
329
- id: Schema.optional(Schema.NullOr(SessionID)),
330
- slug: Schema.optional(Schema.NullOr(Schema.String)),
331
- projectID: Schema.optional(Schema.NullOr(ProjectV2.ID)),
332
- workspaceID: Schema.optional(Schema.NullOr(WorkspaceV2.ID)),
333
- directory: Schema.optional(Schema.NullOr(Schema.String)),
334
- path: Schema.optional(Schema.NullOr(Schema.String)),
335
- parentID: Schema.optional(Schema.NullOr(SessionID)),
336
- summary: Schema.optional(Schema.NullOr(Summary)),
337
- cost: Schema.optional(Schema.Finite),
338
- tokens: Schema.optional(Tokens),
339
- share: Schema.optional(UpdatedShare),
340
- title: Schema.optional(Schema.NullOr(Schema.String)),
341
- agent: Schema.optional(Schema.NullOr(Schema.String)),
342
- model: Schema.optional(Schema.NullOr(Model)),
343
- version: Schema.optional(Schema.NullOr(Schema.String)),
344
- metadata: Schema.optional(Schema.NullOr(Metadata)),
345
- time: Schema.optional(UpdatedTime),
346
- permission: Schema.optional(Schema.NullOr(PermissionV1.Ruleset)),
347
- revert: Schema.optional(Schema.NullOr(Revert)),
348
- })
349
-
350
- const UpdatedEventSchema = Schema.Struct({
351
- sessionID: SessionID,
352
- info: UpdatedInfo,
353
- })
354
-
355
- export const Event = {
356
- Created: SessionV1.Event.Created,
357
- Updated: SessionV1.Event.Updated,
358
- Deleted: SessionV1.Event.Deleted,
359
- Diff: EventV2.define({
360
- type: "session.diff",
361
- schema: {
362
- sessionID: SessionID,
363
- diff: Schema.Array(Snapshot.FileDiff),
364
- },
365
- }),
366
- Error: EventV2.define({
367
- type: "session.error",
368
- schema: {
369
- sessionID: Schema.optional(SessionID),
370
- // Reuses SessionV1.Assistant.fields.error (already Schema.optional) so
371
- // the derived schema keeps the same discriminated-union shape on the event stream.
372
- error: SessionV1.Assistant.fields.error,
373
- },
374
- }),
375
- }
376
-
377
- export function plan(input: { slug: string; time: { created: number } }, instance: InstanceContext) {
378
- const base = instance.project.vcs
379
- ? path.join(instance.worktree, ".opencode", "plans")
380
- : path.join(Global.Path.data, "plans")
381
- return path.join(base, [input.time.created, input.slug].join("-") + ".md")
382
- }
383
-
384
- export const getUsage = (input: { model: Provider.Model; usage: Usage; metadata?: ProviderMetadata }) => {
385
- const safe = (value: number) => {
386
- if (!Number.isFinite(value)) return 0
387
- return Math.max(0, value)
388
- }
389
- const inputTokens = safe(input.usage.inputTokens ?? 0)
390
- const outputTokens = safe(input.usage.outputTokens ?? 0)
391
- const reasoningTokens = safe(input.usage.reasoningTokens ?? 0)
392
-
393
- const cacheReadInputTokens = safe(input.usage.cacheReadInputTokens ?? 0)
394
- const cacheWriteInputTokens = safe(
395
- Number(
396
- input.usage.cacheWriteInputTokens ??
397
- input.metadata?.["anthropic"]?.["cacheCreationInputTokens"] ??
398
- // google-vertex-anthropic returns metadata under "vertex" key
399
- // (AnthropicMessagesLanguageModel custom provider key from 'vertex.anthropic.messages')
400
- input.metadata?.["vertex"]?.["cacheCreationInputTokens"] ??
401
- // @ts-expect-error
402
- input.metadata?.["bedrock"]?.["usage"]?.["cacheWriteInputTokens"] ??
403
- // @ts-expect-error
404
- input.metadata?.["venice"]?.["usage"]?.["cacheCreationInputTokens"] ??
405
- 0,
406
- ),
407
- )
408
-
409
- // AI SDK v6 normalized inputTokens to include cached tokens across all providers
410
- // (including Anthropic/Bedrock which previously excluded them). Always subtract cache
411
- // tokens to get the non-cached input count for separate cost calculation.
412
- const adjustedInputTokens = safe(inputTokens - cacheReadInputTokens - cacheWriteInputTokens)
413
-
414
- const total = input.usage.totalTokens
415
-
416
- const tokens = {
417
- total,
418
- input: adjustedInputTokens,
419
- output: safe(outputTokens - reasoningTokens),
420
- reasoning: reasoningTokens,
421
- cache: {
422
- write: cacheWriteInputTokens,
423
- read: cacheReadInputTokens,
424
- },
425
- }
426
-
427
- const contextTokens = inputTokens
428
- const costInfo =
429
- input.model.cost?.tiers
430
- ?.filter((item) => item.tier.type === "context" && contextTokens > item.tier.size)
431
- .sort((a, b) => b.tier.size - a.tier.size)[0] ??
432
- (input.model.cost?.experimentalOver200K && contextTokens > 200_000
433
- ? input.model.cost.experimentalOver200K
434
- : input.model.cost)
435
- const totalNanoAiu = input.metadata?.["copilot"]?.["totalNanoAiu"]
436
- return {
437
- cost:
438
- typeof totalNanoAiu === "number" && Number.isFinite(totalNanoAiu) && totalNanoAiu >= 0
439
- ? new Decimal(totalNanoAiu).div(100_000_000_000).toNumber()
440
- : safe(
441
- new Decimal(0)
442
- .add(new Decimal(tokens.input).mul(costInfo?.input ?? 0).div(1_000_000))
443
- .add(new Decimal(tokens.output).mul(costInfo?.output ?? 0).div(1_000_000))
444
- .add(new Decimal(tokens.cache.read).mul(costInfo?.cache?.read ?? 0).div(1_000_000))
445
- .add(new Decimal(tokens.cache.write).mul(costInfo?.cache?.write ?? 0).div(1_000_000))
446
- // TODO: update models.dev to have better pricing model, for now:
447
- // charge reasoning tokens at the same rate as output tokens
448
- .add(new Decimal(tokens.reasoning).mul(costInfo?.output ?? 0).div(1_000_000))
449
- .toNumber(),
450
- ),
451
- tokens,
452
- }
453
- }
454
-
455
- export class BusyError extends Schema.TaggedErrorClass<BusyError>()("SessionBusyError", {
456
- sessionID: SessionID,
457
- }) {}
458
-
459
- export type NotFound = NotFoundError
460
-
461
- export interface Interface {
462
- readonly list: (input?: ListInput) => Effect.Effect<Info[]>
463
- readonly listGlobal: (input?: GlobalListInput) => Effect.Effect<GlobalInfo[]>
464
- readonly create: (input?: {
465
- parentID?: SessionID
466
- title?: string
467
- agent?: string
468
- model?: Schema.Schema.Type<typeof Model>
469
- metadata?: typeof Metadata.Type
470
- permission?: PermissionV1.Ruleset
471
- workspaceID?: WorkspaceV2.ID
472
- }) => Effect.Effect<Info>
473
- readonly fork: (input: { sessionID: SessionID; messageID?: MessageID }) => Effect.Effect<Info, NotFound>
474
- readonly touch: (sessionID: SessionID) => Effect.Effect<void>
475
- readonly get: (id: SessionID) => Effect.Effect<Info, NotFound>
476
- readonly setTitle: (input: { sessionID: SessionID; title: string }) => Effect.Effect<void>
477
- readonly setArchived: (input: { sessionID: SessionID; time?: number }) => Effect.Effect<void>
478
- readonly setMetadata: (input: typeof SetMetadataInput.Type) => Effect.Effect<void>
479
- readonly setPermission: (input: { sessionID: SessionID; permission: PermissionV1.Ruleset }) => Effect.Effect<void>
480
- readonly setRevert: (input: {
481
- sessionID: SessionID
482
- revert: Info["revert"]
483
- summary: Info["summary"]
484
- }) => Effect.Effect<void>
485
- readonly clearRevert: (sessionID: SessionID) => Effect.Effect<void>
486
- readonly setSummary: (input: { sessionID: SessionID; summary: Info["summary"] }) => Effect.Effect<void>
487
- readonly setShare: (input: { sessionID: SessionID; share: Info["share"] }) => Effect.Effect<void>
488
- readonly setWorkspace: (input: { sessionID: SessionID; workspaceID: Info["workspaceID"] }) => Effect.Effect<void>
489
- readonly diff: (sessionID: SessionID) => Effect.Effect<Snapshot.FileDiff[]>
490
- readonly messages: (input: { sessionID: SessionID; limit?: number }) => Effect.Effect<SessionV1.WithParts[], NotFound>
491
- readonly children: (parentID: SessionID) => Effect.Effect<Info[]>
492
- readonly remove: (sessionID: SessionID) => Effect.Effect<void, NotFound>
493
- readonly updateMessage: <T extends SessionV1.Info>(msg: T) => Effect.Effect<T>
494
- readonly removeMessage: (input: { sessionID: SessionID; messageID: MessageID }) => Effect.Effect<MessageID>
495
- readonly removePart: (input: { sessionID: SessionID; messageID: MessageID; partID: PartID }) => Effect.Effect<PartID>
496
- readonly getPart: (input: {
497
- sessionID: SessionID
498
- messageID: MessageID
499
- partID: PartID
500
- }) => Effect.Effect<SessionV1.Part | undefined>
501
- readonly updatePart: <T extends SessionV1.Part>(part: T) => Effect.Effect<T>
502
- readonly updatePartDelta: (input: {
503
- sessionID: SessionID
504
- messageID: MessageID
505
- partID: PartID
506
- field: string
507
- delta: string
508
- }) => Effect.Effect<void>
509
- /** Finds the first message matching the predicate, searching newest-first. */
510
- readonly findMessage: (
511
- sessionID: SessionID,
512
- predicate: (msg: SessionV1.WithParts) => boolean,
513
- ) => Effect.Effect<Option.Option<SessionV1.WithParts>, NotFound>
514
- }
515
-
516
- export class Service extends Context.Service<Service, Interface>()("@opencode/Session") {}
517
-
518
- export const use = serviceUse(Service)
519
-
520
- export type Patch = Omit<Partial<Info>, "time" | "share" | "summary" | "revert" | "permission"> & {
521
- time?: Partial<Info["time"]>
522
- share?: Partial<NonNullable<Info["share"]>> | null
523
- summary?: Info["summary"] | null
524
- revert?: Info["revert"] | null
525
- permission?: Info["permission"] | null
526
- }
527
-
528
- export const layer: Layer.Layer<
529
- Service,
530
- never,
531
- BackgroundJob.Service | RuntimeFlags.Service | Database.Service | EventV2Bridge.Service
532
- > = Layer.effect(
533
- Service,
534
- Effect.gen(function* () {
535
- const { db } = yield* Database.Service
536
- const database = yield* Database.Service
537
- const background = yield* BackgroundJob.Service
538
- const events = yield* EventV2Bridge.Service
539
- const flags = yield* RuntimeFlags.Service
540
-
541
- const createNext = Effect.fn("Session.createNext")(function* (input: {
542
- id?: SessionID
543
- title?: string
544
- agent?: string
545
- model?: Schema.Schema.Type<typeof Model>
546
- parentID?: SessionID
547
- workspaceID?: WorkspaceV2.ID
548
- directory: string
549
- path?: string
550
- metadata?: typeof Metadata.Type
551
- permission?: PermissionV1.Ruleset
552
- }) {
553
- const ctx = yield* InstanceState.context
554
- const result: Info = {
555
- id: SessionID.descending(input.id),
556
- slug: Slug.create(),
557
- version: InstallationVersion,
558
- projectID: ctx.project.id,
559
- directory: input.directory,
560
- path: input.path,
561
- workspaceID: input.workspaceID,
562
- parentID: input.parentID,
563
- title: input.title ?? (input.parentID ? childTitlePrefix : parentTitlePrefix) + new Date().toISOString(),
564
- agent: input.agent,
565
- model: input.model,
566
- metadata: input.metadata,
567
- permission: input.permission ? [...input.permission] : undefined,
568
- cost: 0,
569
- tokens: EmptyTokens,
570
- time: {
571
- created: Date.now(),
572
- updated: Date.now(),
573
- },
574
- }
575
- yield* Effect.logInfo("created", result)
576
-
577
- yield* events.publish(SessionV1.Event.Created, { sessionID: result.id, info: result })
578
-
579
- return result
580
- })
581
-
582
- const get = Effect.fn("Session.get")(function* (id: SessionID) {
583
- const row = yield* db.select().from(SessionTable).where(eq(SessionTable.id, id)).get().pipe(Effect.orDie)
584
- if (!row) return yield* Effect.fail(new NotFoundError({ message: `Session not found: ${id}` }))
585
- return fromRow(row)
586
- })
587
-
588
- const list = Effect.fn("Session.list")(function* (input?: ListInput) {
589
- const ctx = yield* InstanceState.context
590
- return yield* listByProject(db, {
591
- projectID: ctx.project.id,
592
- experimentalWorkspaces: flags.experimentalWorkspaces,
593
- ...input,
594
- })
595
- })
596
-
597
- const listGlobal = Effect.fn("Session.listGlobal")(function* (input?: GlobalListInput) {
598
- const conditions: SQL[] = []
599
- if (input?.directory) conditions.push(eq(SessionTable.directory, input.directory))
600
- if (input?.roots) conditions.push(isNull(SessionTable.parent_id))
601
- if (input?.start) conditions.push(gte(SessionTable.time_updated, input.start))
602
- if (input?.cursor) conditions.push(lt(SessionTable.time_updated, input.cursor))
603
- if (input?.search) conditions.push(like(SessionTable.title, `%${input.search}%`))
604
- if (!input?.archived) conditions.push(isNull(SessionTable.time_archived))
605
-
606
- const query =
607
- conditions.length > 0
608
- ? db
609
- .select()
610
- .from(SessionTable)
611
- .where(and(...conditions))
612
- : db.select().from(SessionTable)
613
- const rows = yield* query
614
- .orderBy(desc(SessionTable.time_updated), desc(SessionTable.id))
615
- .limit(input?.limit ?? 100)
616
- .all()
617
- .pipe(Effect.orDie)
618
- const ids = [...new Set(rows.map((row) => row.project_id))]
619
- const projects = new Map<string, ProjectInfo>()
620
- if (ids.length > 0) {
621
- const items = yield* db
622
- .select({ id: ProjectTable.id, name: ProjectTable.name, worktree: ProjectTable.worktree })
623
- .from(ProjectTable)
624
- .where(inArray(ProjectTable.id, ids))
625
- .all()
626
- .pipe(Effect.orDie)
627
- for (const item of items) {
628
- projects.set(item.id, {
629
- id: item.id,
630
- name: item.name ?? undefined,
631
- worktree: item.worktree,
632
- })
633
- }
634
- }
635
- return rows.map((row) => ({ ...fromRow(row), project: projects.get(row.project_id) ?? null }))
636
- })
637
-
638
- const children = Effect.fn("Session.children")(function* (parentID: SessionID) {
639
- const rows = yield* db
640
- .select()
641
- .from(SessionTable)
642
- .where(and(eq(SessionTable.parent_id, parentID)))
643
- .all()
644
- .pipe(Effect.orDie)
645
- return rows.map(fromRow)
646
- })
647
-
648
- const remove: Interface["remove"] = Effect.fnUntraced(function* (sessionID: SessionID) {
649
- const session = yield* get(sessionID)
650
- try {
651
- // `remove` needs to work in all cases, such as broken sessions that
652
- // run cleanup without instance state.
653
- const hasInstance = yield* InstanceState.directory.pipe(
654
- Effect.as(true),
655
- Effect.catchCause(() => Effect.succeed(false)),
656
- )
657
-
658
- if (hasInstance) yield* cancelBackgroundJobs(background, sessionID)
659
- const kids = yield* children(sessionID)
660
- for (const child of kids) {
661
- yield* remove(child.id)
662
- }
663
-
664
- yield* events.publish(SessionV1.Event.Deleted, { sessionID, info: session })
665
- yield* events.remove(sessionID)
666
- } catch (error) {
667
- yield* Effect.logError("failed to remove session", { sessionID, error })
668
- }
669
- })
670
-
671
- const updateMessage = <T extends SessionV1.Info>(msg: T): Effect.Effect<T> =>
672
- Effect.gen(function* () {
673
- yield* events.publish(SessionV1.Event.MessageUpdated, { sessionID: msg.sessionID, info: msg })
674
- return msg
675
- }).pipe(Effect.withSpan("Session.updateMessage"))
676
-
677
- const updatePart = <T extends SessionV1.Part>(part: T): Effect.Effect<T> =>
678
- Effect.gen(function* () {
679
- yield* events.publish(SessionV1.Event.PartUpdated, {
680
- sessionID: part.sessionID,
681
- part: structuredClone(part),
682
- time: Date.now(),
683
- })
684
- return part
685
- }).pipe(Effect.withSpan("Session.updatePart"))
686
-
687
- const getPart: Interface["getPart"] = Effect.fn("Session.getPart")(function* (input) {
688
- const row = yield* db
689
- .select()
690
- .from(PartTable)
691
- .where(
692
- and(
693
- eq(PartTable.session_id, input.sessionID),
694
- eq(PartTable.message_id, input.messageID),
695
- eq(PartTable.id, input.partID),
696
- ),
697
- )
698
- .get()
699
- .pipe(Effect.orDie)
700
- if (!row) return
701
- return {
702
- ...row.data,
703
- id: row.id,
704
- sessionID: row.session_id,
705
- messageID: row.message_id,
706
- } as SessionV1.Part
707
- })
708
-
709
- const create = Effect.fn("Session.create")(function* (input?: {
710
- parentID?: SessionID
711
- title?: string
712
- agent?: string
713
- model?: Schema.Schema.Type<typeof Model>
714
- metadata?: typeof Metadata.Type
715
- permission?: PermissionV1.Ruleset
716
- workspaceID?: WorkspaceV2.ID
717
- }) {
718
- const ctx = yield* InstanceState.context
719
- const workspace = yield* InstanceState.workspaceID
720
- return yield* createNext({
721
- parentID: input?.parentID,
722
- directory: ctx.directory,
723
- path: sessionPath(ctx.worktree, ctx.directory),
724
- title: input?.title,
725
- agent: input?.agent,
726
- model: input?.model,
727
- metadata: input?.metadata,
728
- permission: input?.permission,
729
- workspaceID: input?.workspaceID ?? workspace,
730
- })
731
- })
732
-
733
- const fork = Effect.fn("Session.fork")(function* (input: { sessionID: SessionID; messageID?: MessageID }) {
734
- const ctx = yield* InstanceState.context
735
- const original = yield* get(input.sessionID)
736
- const title = getForkedTitle(original.title)
737
- const session = yield* createNext({
738
- directory: ctx.directory,
739
- path: sessionPath(ctx.worktree, ctx.directory),
740
- workspaceID: original.workspaceID,
741
- title,
742
- metadata: structuredClone(original.metadata),
743
- })
744
- const msgs = yield* messages({ sessionID: input.sessionID })
745
- const idMap = new Map<string, MessageID>()
746
-
747
- for (const msg of msgs) {
748
- if (input.messageID && msg.info.id >= input.messageID) break
749
- const newID = MessageID.ascending()
750
- idMap.set(msg.info.id, newID)
751
-
752
- const parentID = msg.info.role === "assistant" && msg.info.parentID ? idMap.get(msg.info.parentID) : undefined
753
- const cloned = yield* updateMessage({
754
- ...msg.info,
755
- sessionID: session.id,
756
- id: newID,
757
- ...(parentID && { parentID }),
758
- })
759
-
760
- for (const part of msg.parts) {
761
- const p: SessionV1.Part = {
762
- ...part,
763
- id: PartID.ascending(),
764
- messageID: cloned.id,
765
- sessionID: session.id,
766
- }
767
- if (p.type === "compaction" && p.tail_start_id) {
768
- p.tail_start_id = idMap.get(p.tail_start_id)
769
- }
770
- yield* updatePart(p)
771
- }
772
- }
773
- return session
774
- })
775
-
776
- const patch = (sessionID: SessionID, info: Patch) =>
777
- Effect.gen(function* () {
778
- const current = yield* get(sessionID)
779
- const next = {
780
- ...current,
781
- ...info,
782
- time: info.time ? { ...current.time, ...info.time } : current.time,
783
- share: info.share === null ? undefined : info.share ? { ...current.share, ...info.share } : current.share,
784
- summary: info.summary === null ? undefined : (info.summary ?? current.summary),
785
- revert: info.revert === null ? undefined : (info.revert ?? current.revert),
786
- permission: info.permission === null ? undefined : (info.permission ?? current.permission),
787
- } as Info
788
- yield* events.publish(SessionV1.Event.Updated, { sessionID, info: next })
789
- })
790
-
791
- const touch = Effect.fn("Session.touch")(function* (sessionID: SessionID) {
792
- yield* patch(sessionID, { time: { updated: Date.now() } }).pipe(Effect.orDie)
793
- })
794
-
795
- const setTitle = Effect.fn("Session.setTitle")(function* (input: { sessionID: SessionID; title: string }) {
796
- yield* patch(input.sessionID, { title: input.title }).pipe(Effect.orDie)
797
- })
798
-
799
- const setArchived = Effect.fn("Session.setArchived")(function* (input: { sessionID: SessionID; time?: number }) {
800
- yield* patch(input.sessionID, { time: { archived: input.time } }).pipe(Effect.orDie)
801
- })
802
-
803
- const setMetadata = Effect.fn("Session.setMetadata")(function* (input: typeof SetMetadataInput.Type) {
804
- yield* patch(input.sessionID, { metadata: input.metadata, time: { updated: Date.now() } }).pipe(Effect.orDie)
805
- })
806
-
807
- const setPermission = Effect.fn("Session.setPermission")(function* (input: {
808
- sessionID: SessionID
809
- permission: PermissionV1.Ruleset
810
- }) {
811
- yield* patch(input.sessionID, { permission: [...input.permission], time: { updated: Date.now() } }).pipe(
812
- Effect.orDie,
813
- )
814
- })
815
-
816
- const setRevert = Effect.fn("Session.setRevert")(function* (input: {
817
- sessionID: SessionID
818
- revert: Info["revert"]
819
- summary: Info["summary"]
820
- }) {
821
- yield* patch(input.sessionID, {
822
- summary: input.summary,
823
- time: { updated: Date.now() },
824
- revert: input.revert,
825
- }).pipe(Effect.orDie)
826
- })
827
-
828
- const clearRevert = Effect.fn("Session.clearRevert")(function* (sessionID: SessionID) {
829
- yield* patch(sessionID, { time: { updated: Date.now() }, revert: null }).pipe(Effect.orDie)
830
- })
831
-
832
- const setSummary = Effect.fn("Session.setSummary")(function* (input: {
833
- sessionID: SessionID
834
- summary: Info["summary"]
835
- }) {
836
- yield* patch(input.sessionID, { time: { updated: Date.now() }, summary: input.summary }).pipe(Effect.orDie)
837
- })
838
-
839
- const setShare = Effect.fn("Session.setShare")(function* (input: { sessionID: SessionID; share: Info["share"] }) {
840
- yield* patch(input.sessionID, { share: input.share ?? null, time: { updated: Date.now() } }).pipe(Effect.orDie)
841
- })
842
-
843
- const setWorkspace = Effect.fn("Session.setWorkspace")(function* (input: {
844
- sessionID: SessionID
845
- workspaceID: Info["workspaceID"]
846
- }) {
847
- yield* patch(input.sessionID, { workspaceID: input.workspaceID, time: { updated: Date.now() } }).pipe(
848
- Effect.orDie,
849
- )
850
- })
851
-
852
- const diff = Effect.fn("Session.diff")(function* (sessionID: SessionID) {
853
- void sessionID
854
- return [] as Snapshot.FileDiff[]
855
- })
856
-
857
- const messages: Interface["messages"] = Effect.fn("Session.messages")(function* (input) {
858
- if (input.limit) {
859
- return (yield* MessageV2.page({ sessionID: input.sessionID, limit: input.limit }).pipe(
860
- Effect.provideService(Database.Service, database),
861
- )).items
862
- }
863
-
864
- const size = 50
865
- const result = [] as SessionV1.WithParts[]
866
- let before: string | undefined
867
- while (true) {
868
- const page = yield* MessageV2.page({ sessionID: input.sessionID, limit: size, before }).pipe(
869
- Effect.provideService(Database.Service, database),
870
- )
871
- if (page.items.length === 0) break
872
- for (let i = page.items.length - 1; i >= 0; i--) {
873
- const item = page.items[i]
874
- if (item) result.push(item)
875
- }
876
- if (!page.more || !page.cursor) break
877
- before = page.cursor
878
- }
879
- return result.reverse()
880
- })
881
-
882
- const removeMessage = Effect.fn("Session.removeMessage")(function* (input: {
883
- sessionID: SessionID
884
- messageID: MessageID
885
- }) {
886
- yield* events.publish(SessionV1.Event.MessageRemoved, {
887
- sessionID: input.sessionID,
888
- messageID: input.messageID,
889
- })
890
- return input.messageID
891
- })
892
-
893
- const removePart = Effect.fn("Session.removePart")(function* (input: {
894
- sessionID: SessionID
895
- messageID: MessageID
896
- partID: PartID
897
- }) {
898
- yield* events.publish(SessionV1.Event.PartRemoved, {
899
- sessionID: input.sessionID,
900
- messageID: input.messageID,
901
- partID: input.partID,
902
- })
903
- return input.partID
904
- })
905
-
906
- const updatePartDelta = Effect.fnUntraced(function* (input: {
907
- sessionID: SessionID
908
- messageID: MessageID
909
- partID: PartID
910
- field: string
911
- delta: string
912
- }) {
913
- yield* events.publish(MessageV2.Event.PartDelta, input)
914
- })
915
-
916
- /** Finds the first message matching the predicate, searching newest-first. */
917
- const findMessage: Interface["findMessage"] = Effect.fn("Session.findMessage")(function* (sessionID, predicate) {
918
- const size = 50
919
- let before: string | undefined
920
- while (true) {
921
- const page = yield* MessageV2.page({ sessionID, limit: size, before }).pipe(
922
- Effect.provideService(Database.Service, database),
923
- )
924
- if (page.items.length === 0) break
925
- for (let i = page.items.length - 1; i >= 0; i--) {
926
- const item = page.items[i]
927
- if (item && predicate(item)) return Option.some(item)
928
- }
929
- if (!page.more || !page.cursor) break
930
- before = page.cursor
931
- }
932
- return Option.none<SessionV1.WithParts>()
933
- })
934
-
935
- return Service.of({
936
- list,
937
- listGlobal,
938
- create,
939
- fork,
940
- touch,
941
- get,
942
- setTitle,
943
- setArchived,
944
- setMetadata,
945
- setPermission,
946
- setRevert,
947
- clearRevert,
948
- setSummary,
949
- setShare,
950
- setWorkspace,
951
- diff,
952
- messages,
953
- children,
954
- remove,
955
- updateMessage,
956
- removeMessage,
957
- removePart,
958
- updatePart,
959
- getPart,
960
- updatePartDelta,
961
- findMessage,
962
- })
963
- }),
964
- )
965
-
966
- export const defaultLayer = layer.pipe(
967
- Layer.provide(BackgroundJob.defaultLayer),
968
- Layer.provide(Database.defaultLayer),
969
- Layer.provide(EventV2Bridge.defaultLayer),
970
- Layer.provide(SessionExecution.noopLayer),
971
- Layer.provide(SessionV2.defaultLayer),
972
- Layer.provide(RuntimeFlags.defaultLayer),
973
- )
974
-
975
- const cancelBackgroundJobs = Effect.fn("Session.cancelBackgroundJobs")(function* (
976
- background: BackgroundJob.Interface,
977
- sessionID: SessionID,
978
- ) {
979
- const jobs = yield* background.list()
980
- yield* Effect.forEach(
981
- jobs.filter((job) => {
982
- if (job.status !== "running") return false
983
- if (job.id === sessionID) return true
984
- if (job.metadata?.sessionId === sessionID) return true
985
- return job.metadata?.parentSessionId === sessionID
986
- }),
987
- (job) => background.cancel(job.id),
988
- { concurrency: "unbounded", discard: true },
989
- )
990
- })
991
-
992
- function listByProject(
993
- db: Database.Interface["db"],
994
- input: ListInput & {
995
- projectID: ProjectV2.ID
996
- experimentalWorkspaces: boolean
997
- },
998
- ) {
999
- const conditions = [eq(SessionTable.project_id, input.projectID)]
1000
-
1001
- if (input.workspaceID) {
1002
- conditions.push(eq(SessionTable.workspace_id, input.workspaceID))
1003
- }
1004
- if (input.path !== undefined) {
1005
- if (input.path) {
1006
- const conds = [
1007
- eq(SessionTable.path, input.path),
1008
- like(SessionTable.path, sql.param(`${input.path}/%`, SessionTable.path)),
1009
- ]
1010
-
1011
- conditions.push(
1012
- input.directory
1013
- ? or(...conds, and(isNull(SessionTable.path), eq(SessionTable.directory, input.directory))!)!
1014
- : or(...conds)!,
1015
- )
1016
- }
1017
- } else if (input.scope !== "project") {
1018
- if (input.directory) {
1019
- conditions.push(eq(SessionTable.directory, input.directory))
1020
- }
1021
- }
1022
- if (input.roots) {
1023
- conditions.push(isNull(SessionTable.parent_id))
1024
- }
1025
- if (input.start) {
1026
- conditions.push(gte(SessionTable.time_updated, input.start))
1027
- }
1028
- if (input.search) {
1029
- conditions.push(like(SessionTable.title, `%${input.search}%`))
1030
- }
1031
-
1032
- const limit = input.limit ?? 100
1033
-
1034
- return db
1035
- .select()
1036
- .from(SessionTable)
1037
- .where(and(...conditions))
1038
- .orderBy(desc(SessionTable.time_updated))
1039
- .limit(limit)
1040
- .all()
1041
- .pipe(
1042
- Effect.orDie,
1043
- Effect.map((rows) => rows.map(fromRow)),
1044
- )
1045
- }
1046
-
1047
- export function* listGlobal(input?: {
1048
- directory?: string
1049
- roots?: boolean
1050
- start?: number
1051
- cursor?: number
1052
- search?: string
1053
- limit?: number
1054
- archived?: boolean
1055
- }) {
1056
- const conditions: SQL[] = []
1057
-
1058
- if (input?.directory) {
1059
- conditions.push(eq(SessionTable.directory, input.directory))
1060
- }
1061
- if (input?.roots) {
1062
- conditions.push(isNull(SessionTable.parent_id))
1063
- }
1064
- if (input?.start) {
1065
- conditions.push(gte(SessionTable.time_updated, input.start))
1066
- }
1067
- if (input?.cursor) {
1068
- conditions.push(lt(SessionTable.time_updated, input.cursor))
1069
- }
1070
- if (input?.search) {
1071
- conditions.push(like(SessionTable.title, `%${input.search}%`))
1072
- }
1073
- if (!input?.archived) {
1074
- conditions.push(isNull(SessionTable.time_archived))
1075
- }
1076
-
1077
- const limit = input?.limit ?? 100
1078
-
1079
- const rows = runtime.runSync(({ db }) => {
1080
- const query =
1081
- conditions.length > 0
1082
- ? db
1083
- .select()
1084
- .from(SessionTable)
1085
- .where(and(...conditions))
1086
- : db.select().from(SessionTable)
1087
- return query.orderBy(desc(SessionTable.time_updated), desc(SessionTable.id)).limit(limit).all().pipe(Effect.orDie)
1088
- })
1089
-
1090
- const ids = [...new Set(rows.map((row) => row.project_id))]
1091
- const projects = new Map<string, ProjectInfo>()
1092
-
1093
- if (ids.length > 0) {
1094
- const items = runtime.runSync(({ db }) =>
1095
- db
1096
- .select({ id: ProjectTable.id, name: ProjectTable.name, worktree: ProjectTable.worktree })
1097
- .from(ProjectTable)
1098
- .where(inArray(ProjectTable.id, ids))
1099
- .all()
1100
- .pipe(Effect.orDie),
1101
- )
1102
- for (const item of items) {
1103
- projects.set(item.id, {
1104
- id: item.id,
1105
- name: item.name ?? undefined,
1106
- worktree: item.worktree,
1107
- })
1108
- }
1109
- }
1110
-
1111
- for (const row of rows) {
1112
- const project = projects.get(row.project_id) ?? null
1113
- yield { ...fromRow(row), project }
1114
- }
1115
- }
1116
-
1117
- export const node = LayerNode.make(layer, [BackgroundJob.node, RuntimeFlags.node, Database.node, EventV2Bridge.node])
1118
-
1119
- export * as Session from "./session"