@sleepy-ai/cli 0.1.6 → 0.1.7

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 (799) hide show
  1. package/{src/skill/compose/LICENSE-superpowers → LICENSE} +3 -6
  2. package/README.md +139 -6
  3. package/package.json +35 -199
  4. package/postinstall.mjs +102 -0
  5. package/.sleepycode-test-fixtures-50158/sleepycode-test-eqsxv84ythu/.watcher-b67v64k2mnv +0 -1
  6. package/Dockerfile +0 -18
  7. package/drizzle.config.ts +0 -10
  8. package/git +0 -0
  9. package/migration/20260127222353_familiar_lady_ursula/migration.sql +0 -90
  10. package/migration/20260127222353_familiar_lady_ursula/snapshot.json +0 -796
  11. package/migration/20260211171708_add_project_commands/migration.sql +0 -1
  12. package/migration/20260211171708_add_project_commands/snapshot.json +0 -806
  13. package/migration/20260213144116_wakeful_the_professor/migration.sql +0 -11
  14. package/migration/20260213144116_wakeful_the_professor/snapshot.json +0 -897
  15. package/migration/20260225215848_workspace/migration.sql +0 -7
  16. package/migration/20260225215848_workspace/snapshot.json +0 -959
  17. package/migration/20260227213759_add_session_workspace_id/migration.sql +0 -2
  18. package/migration/20260227213759_add_session_workspace_id/snapshot.json +0 -983
  19. package/migration/20260228203230_blue_harpoon/migration.sql +0 -17
  20. package/migration/20260228203230_blue_harpoon/snapshot.json +0 -1102
  21. package/migration/20260303231226_add_workspace_fields/migration.sql +0 -5
  22. package/migration/20260303231226_add_workspace_fields/snapshot.json +0 -1013
  23. package/migration/20260309230000_move_org_to_state/migration.sql +0 -3
  24. package/migration/20260309230000_move_org_to_state/snapshot.json +0 -1156
  25. package/migration/20260312043431_session_message_cursor/migration.sql +0 -4
  26. package/migration/20260312043431_session_message_cursor/snapshot.json +0 -1168
  27. package/migration/20260323234822_events/migration.sql +0 -13
  28. package/migration/20260323234822_events/snapshot.json +0 -1271
  29. package/migration/20260410174513_workspace-name/migration.sql +0 -16
  30. package/migration/20260410174513_workspace-name/snapshot.json +0 -1271
  31. package/migration/20260413175956_chief_energizer/migration.sql +0 -13
  32. package/migration/20260413175956_chief_energizer/snapshot.json +0 -1399
  33. package/migration/20260422160000_context_inheritance/migration.sql +0 -3
  34. package/migration/20260422170000_task_registry/migration.sql +0 -18
  35. package/migration/20260423145421_remove_session_entry/migration.sql +0 -4
  36. package/migration/20260515000000_actor_rename/migration.sql +0 -7
  37. package/migration/20260515010000_memory_fts/migration.sql +0 -33
  38. package/migration/20260515020000_user_task/migration.sql +0 -29
  39. package/migration/20260519000000_last_checkpoint_message_id/migration.sql +0 -1
  40. package/migration/20260521000000_message_agent_id/migration.sql +0 -2
  41. package/migration/20260521000100_actor_registry_v6/migration.sql +0 -25
  42. package/migration/20260521010000_memory_fts_v6/migration.sql +0 -33
  43. package/migration/20260521020000_memory_fts_triggers/migration.sql +0 -17
  44. package/migration/20260526000000_agent_id_main/migration.sql +0 -14
  45. package/migration/20260527000000_actor_lifecycle/migration.sql +0 -8
  46. package/migration/20260527000100_inbox/migration.sql +0 -12
  47. package/migration/20260529000000_task_todo_redesign/migration.sql +0 -16
  48. package/migration/20260603000000_task_in_progress_owner/migration.sql +0 -1
  49. package/migration/20260603000000_workflow_run/migration.sql +0 -17
  50. package/migration/20260604000000_workflow_script_sha/migration.sql +0 -1
  51. package/migration/20260608000000_claude_import/migration.sql +0 -7
  52. package/migration/20260608010000_claude_import_message_ids/migration.sql +0 -1
  53. package/migration/20260609000000_history_fts/migration.sql +0 -29
  54. package/migration/20260609230000_workflow_agent_timeout/migration.sql +0 -1
  55. package/migration/20260612000000_external_import/migration.sql +0 -16
  56. package/parsers-config.ts +0 -290
  57. package/src/account/account.sql.ts +0 -39
  58. package/src/account/account.ts +0 -456
  59. package/src/account/repo.ts +0 -166
  60. package/src/account/schema.ts +0 -99
  61. package/src/account/url.ts +0 -8
  62. package/src/acp/README.md +0 -174
  63. package/src/acp/agent.ts +0 -1787
  64. package/src/acp/session.ts +0 -116
  65. package/src/acp/types.ts +0 -24
  66. package/src/actor/actor.sql.ts +0 -38
  67. package/src/actor/events.ts +0 -67
  68. package/src/actor/index.ts +0 -2
  69. package/src/actor/registry.ts +0 -412
  70. package/src/actor/return-header.ts +0 -24
  71. package/src/actor/schema.ts +0 -47
  72. package/src/actor/spawn-ref.ts +0 -16
  73. package/src/actor/spawn.ts +0 -756
  74. package/src/actor/turn.ts +0 -49
  75. package/src/actor/waiter.ts +0 -166
  76. package/src/agent/agent.ts +0 -574
  77. package/src/agent/config.ts +0 -5
  78. package/src/agent/generate.txt +0 -75
  79. package/src/agent/prompt/checkpoint-writer.txt +0 -167
  80. package/src/agent/prompt/compaction.txt +0 -9
  81. package/src/agent/prompt/distill.txt +0 -199
  82. package/src/agent/prompt/dream.txt +0 -155
  83. package/src/agent/prompt/explore.txt +0 -18
  84. package/src/agent/prompt/summary.txt +0 -11
  85. package/src/agent/prompt/title.txt +0 -44
  86. package/src/audio.d.ts +0 -9
  87. package/src/auth/index.ts +0 -97
  88. package/src/bus/bus-event.ts +0 -33
  89. package/src/bus/global.ts +0 -12
  90. package/src/bus/index.ts +0 -193
  91. package/src/cli/bootstrap.ts +0 -33
  92. package/src/cli/cmd/account.ts +0 -258
  93. package/src/cli/cmd/acp.ts +0 -70
  94. package/src/cli/cmd/agent.ts +0 -248
  95. package/src/cli/cmd/cmd.ts +0 -7
  96. package/src/cli/cmd/db.ts +0 -120
  97. package/src/cli/cmd/debug/agent.ts +0 -192
  98. package/src/cli/cmd/debug/config.ts +0 -17
  99. package/src/cli/cmd/debug/file.ts +0 -100
  100. package/src/cli/cmd/debug/index.ts +0 -48
  101. package/src/cli/cmd/debug/lsp.ts +0 -61
  102. package/src/cli/cmd/debug/ripgrep.ts +0 -105
  103. package/src/cli/cmd/debug/scrap.ts +0 -16
  104. package/src/cli/cmd/debug/skill.ts +0 -23
  105. package/src/cli/cmd/debug/snapshot.ts +0 -53
  106. package/src/cli/cmd/doctor.ts +0 -116
  107. package/src/cli/cmd/export.ts +0 -306
  108. package/src/cli/cmd/generate.ts +0 -50
  109. package/src/cli/cmd/github.ts +0 -1647
  110. package/src/cli/cmd/import.ts +0 -208
  111. package/src/cli/cmd/init.ts +0 -97
  112. package/src/cli/cmd/login.ts +0 -402
  113. package/src/cli/cmd/mcp.ts +0 -812
  114. package/src/cli/cmd/models.ts +0 -88
  115. package/src/cli/cmd/plug.ts +0 -233
  116. package/src/cli/cmd/pr.ts +0 -138
  117. package/src/cli/cmd/providers.ts +0 -713
  118. package/src/cli/cmd/run-completion.ts +0 -77
  119. package/src/cli/cmd/run.ts +0 -694
  120. package/src/cli/cmd/serve.ts +0 -30
  121. package/src/cli/cmd/session.ts +0 -181
  122. package/src/cli/cmd/stats.ts +0 -413
  123. package/src/cli/cmd/tui/app.tsx +0 -1181
  124. package/src/cli/cmd/tui/asset/TEN_VAD_LICENSE +0 -12
  125. package/src/cli/cmd/tui/asset/charge.wav +0 -0
  126. package/src/cli/cmd/tui/asset/pulse-a.wav +0 -0
  127. package/src/cli/cmd/tui/asset/pulse-b.wav +0 -0
  128. package/src/cli/cmd/tui/asset/pulse-c.wav +0 -0
  129. package/src/cli/cmd/tui/asset/ten_vad.wasm +0 -0
  130. package/src/cli/cmd/tui/asset/ten_vad_loader.js +0 -30
  131. package/src/cli/cmd/tui/attach.ts +0 -84
  132. package/src/cli/cmd/tui/component/background-image.tsx +0 -150
  133. package/src/cli/cmd/tui/component/banner-session-expired.tsx +0 -48
  134. package/src/cli/cmd/tui/component/bg-pulse.tsx +0 -130
  135. package/src/cli/cmd/tui/component/border.tsx +0 -21
  136. package/src/cli/cmd/tui/component/dialog-agent.tsx +0 -31
  137. package/src/cli/cmd/tui/component/dialog-agreement.tsx +0 -111
  138. package/src/cli/cmd/tui/component/dialog-command.tsx +0 -219
  139. package/src/cli/cmd/tui/component/dialog-console-org.tsx +0 -103
  140. package/src/cli/cmd/tui/component/dialog-go-upsell.tsx +0 -157
  141. package/src/cli/cmd/tui/component/dialog-image-list.tsx +0 -111
  142. package/src/cli/cmd/tui/component/dialog-logo-design.tsx +0 -37
  143. package/src/cli/cmd/tui/component/dialog-mcp.tsx +0 -86
  144. package/src/cli/cmd/tui/component/dialog-model.tsx +0 -299
  145. package/src/cli/cmd/tui/component/dialog-provider.tsx +0 -449
  146. package/src/cli/cmd/tui/component/dialog-session-delete-failed.tsx +0 -101
  147. package/src/cli/cmd/tui/component/dialog-session-list.tsx +0 -269
  148. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +0 -31
  149. package/src/cli/cmd/tui/component/dialog-skill.tsx +0 -42
  150. package/src/cli/cmd/tui/component/dialog-sleepy-login.tsx +0 -288
  151. package/src/cli/cmd/tui/component/dialog-stash.tsx +0 -87
  152. package/src/cli/cmd/tui/component/dialog-status.tsx +0 -170
  153. package/src/cli/cmd/tui/component/dialog-tag.tsx +0 -44
  154. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +0 -50
  155. package/src/cli/cmd/tui/component/dialog-token-plan.tsx +0 -77
  156. package/src/cli/cmd/tui/component/dialog-variant.tsx +0 -39
  157. package/src/cli/cmd/tui/component/dialog-workflows.tsx +0 -51
  158. package/src/cli/cmd/tui/component/dialog-workspace-create.tsx +0 -289
  159. package/src/cli/cmd/tui/component/dialog-workspace-unavailable.tsx +0 -81
  160. package/src/cli/cmd/tui/component/dialog-worktree.tsx +0 -90
  161. package/src/cli/cmd/tui/component/error-component.tsx +0 -92
  162. package/src/cli/cmd/tui/component/logo.tsx +0 -961
  163. package/src/cli/cmd/tui/component/plugin-route-missing.tsx +0 -14
  164. package/src/cli/cmd/tui/component/prompt/autocomplete-detect.ts +0 -34
  165. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +0 -668
  166. package/src/cli/cmd/tui/component/prompt/cwd.ts +0 -0
  167. package/src/cli/cmd/tui/component/prompt/frecency.tsx +0 -90
  168. package/src/cli/cmd/tui/component/prompt/history.tsx +0 -108
  169. package/src/cli/cmd/tui/component/prompt/index.tsx +0 -1869
  170. package/src/cli/cmd/tui/component/prompt/offset.ts +0 -45
  171. package/src/cli/cmd/tui/component/prompt/part.ts +0 -36
  172. package/src/cli/cmd/tui/component/prompt/stash.tsx +0 -101
  173. package/src/cli/cmd/tui/component/spinner.tsx +0 -24
  174. package/src/cli/cmd/tui/component/starry-background.tsx +0 -305
  175. package/src/cli/cmd/tui/component/startup-loading.tsx +0 -67
  176. package/src/cli/cmd/tui/component/task-item.tsx +0 -63
  177. package/src/cli/cmd/tui/component/textarea-keybindings.ts +0 -73
  178. package/src/cli/cmd/tui/component/todo-item.tsx +0 -32
  179. package/src/cli/cmd/tui/component/workflow-tree.tsx +0 -177
  180. package/src/cli/cmd/tui/config/cwd.ts +0 -5
  181. package/src/cli/cmd/tui/config/tui-migrate.ts +0 -151
  182. package/src/cli/cmd/tui/config/tui-schema.ts +0 -38
  183. package/src/cli/cmd/tui/config/tui.ts +0 -219
  184. package/src/cli/cmd/tui/context/args.tsx +0 -16
  185. package/src/cli/cmd/tui/context/directory.ts +0 -15
  186. package/src/cli/cmd/tui/context/event.ts +0 -45
  187. package/src/cli/cmd/tui/context/exit.tsx +0 -65
  188. package/src/cli/cmd/tui/context/helper.tsx +0 -25
  189. package/src/cli/cmd/tui/context/keybind.tsx +0 -105
  190. package/src/cli/cmd/tui/context/kv.tsx +0 -86
  191. package/src/cli/cmd/tui/context/language.tsx +0 -91
  192. package/src/cli/cmd/tui/context/local.tsx +0 -469
  193. package/src/cli/cmd/tui/context/plugin-keybinds.ts +0 -41
  194. package/src/cli/cmd/tui/context/project.tsx +0 -109
  195. package/src/cli/cmd/tui/context/prompt.tsx +0 -18
  196. package/src/cli/cmd/tui/context/route.tsx +0 -68
  197. package/src/cli/cmd/tui/context/sdk.tsx +0 -150
  198. package/src/cli/cmd/tui/context/sync.tsx +0 -884
  199. package/src/cli/cmd/tui/context/theme/aura.json +0 -69
  200. package/src/cli/cmd/tui/context/theme/ayu.json +0 -80
  201. package/src/cli/cmd/tui/context/theme/carbonfox.json +0 -248
  202. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +0 -230
  203. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +0 -230
  204. package/src/cli/cmd/tui/context/theme/catppuccin.json +0 -112
  205. package/src/cli/cmd/tui/context/theme/cobalt2.json +0 -225
  206. package/src/cli/cmd/tui/context/theme/cursor.json +0 -249
  207. package/src/cli/cmd/tui/context/theme/dracula.json +0 -219
  208. package/src/cli/cmd/tui/context/theme/everforest.json +0 -241
  209. package/src/cli/cmd/tui/context/theme/flexoki.json +0 -237
  210. package/src/cli/cmd/tui/context/theme/github.json +0 -233
  211. package/src/cli/cmd/tui/context/theme/gruvbox.json +0 -242
  212. package/src/cli/cmd/tui/context/theme/kanagawa.json +0 -77
  213. package/src/cli/cmd/tui/context/theme/lucent-orng.json +0 -234
  214. package/src/cli/cmd/tui/context/theme/material.json +0 -235
  215. package/src/cli/cmd/tui/context/theme/matrix.json +0 -77
  216. package/src/cli/cmd/tui/context/theme/mercury.json +0 -252
  217. package/src/cli/cmd/tui/context/theme/monokai.json +0 -221
  218. package/src/cli/cmd/tui/context/theme/nightowl.json +0 -221
  219. package/src/cli/cmd/tui/context/theme/nord.json +0 -223
  220. package/src/cli/cmd/tui/context/theme/one-dark.json +0 -84
  221. package/src/cli/cmd/tui/context/theme/orng.json +0 -249
  222. package/src/cli/cmd/tui/context/theme/osaka-jade.json +0 -93
  223. package/src/cli/cmd/tui/context/theme/palenight.json +0 -222
  224. package/src/cli/cmd/tui/context/theme/rosepine.json +0 -234
  225. package/src/cli/cmd/tui/context/theme/sleepycode.json +0 -245
  226. package/src/cli/cmd/tui/context/theme/solarized.json +0 -223
  227. package/src/cli/cmd/tui/context/theme/synthwave84.json +0 -226
  228. package/src/cli/cmd/tui/context/theme/tokyonight.json +0 -243
  229. package/src/cli/cmd/tui/context/theme/vercel.json +0 -245
  230. package/src/cli/cmd/tui/context/theme/vesper.json +0 -218
  231. package/src/cli/cmd/tui/context/theme/zenburn.json +0 -223
  232. package/src/cli/cmd/tui/context/theme.tsx +0 -1298
  233. package/src/cli/cmd/tui/context/thinking.ts +0 -48
  234. package/src/cli/cmd/tui/context/tui-config.tsx +0 -9
  235. package/src/cli/cmd/tui/event.ts +0 -62
  236. package/src/cli/cmd/tui/feature-plugins/home/footer.tsx +0 -93
  237. package/src/cli/cmd/tui/feature-plugins/home/tips-view.tsx +0 -193
  238. package/src/cli/cmd/tui/feature-plugins/home/tips.tsx +0 -54
  239. package/src/cli/cmd/tui/feature-plugins/sidebar/context.tsx +0 -267
  240. package/src/cli/cmd/tui/feature-plugins/sidebar/cwd.tsx +0 -45
  241. package/src/cli/cmd/tui/feature-plugins/sidebar/files.tsx +0 -62
  242. package/src/cli/cmd/tui/feature-plugins/sidebar/footer.tsx +0 -93
  243. package/src/cli/cmd/tui/feature-plugins/sidebar/goal.tsx +0 -84
  244. package/src/cli/cmd/tui/feature-plugins/sidebar/instructions.tsx +0 -54
  245. package/src/cli/cmd/tui/feature-plugins/sidebar/lsp.tsx +0 -66
  246. package/src/cli/cmd/tui/feature-plugins/sidebar/mascot.tsx +0 -69
  247. package/src/cli/cmd/tui/feature-plugins/sidebar/mcp.tsx +0 -98
  248. package/src/cli/cmd/tui/feature-plugins/sidebar/task.tsx +0 -95
  249. package/src/cli/cmd/tui/feature-plugins/sidebar/todo.tsx +0 -51
  250. package/src/cli/cmd/tui/feature-plugins/sidebar/tps.ts +0 -31
  251. package/src/cli/cmd/tui/feature-plugins/system/plugins.tsx +0 -274
  252. package/src/cli/cmd/tui/i18n/en.ts +0 -465
  253. package/src/cli/cmd/tui/i18n/es.ts +0 -508
  254. package/src/cli/cmd/tui/i18n/fr.ts +0 -515
  255. package/src/cli/cmd/tui/i18n/ja.ts +0 -463
  256. package/src/cli/cmd/tui/i18n/locales.ts +0 -82
  257. package/src/cli/cmd/tui/i18n/ru.ts +0 -527
  258. package/src/cli/cmd/tui/i18n/slash-command.ts +0 -11
  259. package/src/cli/cmd/tui/i18n/zh.ts +0 -454
  260. package/src/cli/cmd/tui/i18n/zht.ts +0 -430
  261. package/src/cli/cmd/tui/layer.ts +0 -6
  262. package/src/cli/cmd/tui/plugin/api.tsx +0 -402
  263. package/src/cli/cmd/tui/plugin/index.ts +0 -3
  264. package/src/cli/cmd/tui/plugin/internal.ts +0 -37
  265. package/src/cli/cmd/tui/plugin/runtime.ts +0 -1057
  266. package/src/cli/cmd/tui/plugin/slots.tsx +0 -60
  267. package/src/cli/cmd/tui/routes/home.tsx +0 -165
  268. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +0 -76
  269. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +0 -116
  270. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +0 -47
  271. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +0 -47
  272. package/src/cli/cmd/tui/routes/session/footer.tsx +0 -91
  273. package/src/cli/cmd/tui/routes/session/index.tsx +0 -3073
  274. package/src/cli/cmd/tui/routes/session/permission.tsx +0 -697
  275. package/src/cli/cmd/tui/routes/session/question.tsx +0 -488
  276. package/src/cli/cmd/tui/routes/session/sidebar.tsx +0 -97
  277. package/src/cli/cmd/tui/routes/session/subagent-footer.tsx +0 -143
  278. package/src/cli/cmd/tui/thread.ts +0 -324
  279. package/src/cli/cmd/tui/ui/dialog-alert.tsx +0 -61
  280. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +0 -95
  281. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +0 -223
  282. package/src/cli/cmd/tui/ui/dialog-help.tsx +0 -42
  283. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +0 -123
  284. package/src/cli/cmd/tui/ui/dialog-select.tsx +0 -467
  285. package/src/cli/cmd/tui/ui/dialog.tsx +0 -207
  286. package/src/cli/cmd/tui/ui/link.tsx +0 -28
  287. package/src/cli/cmd/tui/ui/spinner.ts +0 -378
  288. package/src/cli/cmd/tui/ui/toast.tsx +0 -102
  289. package/src/cli/cmd/tui/util/clipboard.ts +0 -203
  290. package/src/cli/cmd/tui/util/editor.ts +0 -36
  291. package/src/cli/cmd/tui/util/image-protocol.ts +0 -35
  292. package/src/cli/cmd/tui/util/index.ts +0 -6
  293. package/src/cli/cmd/tui/util/model.ts +0 -23
  294. package/src/cli/cmd/tui/util/pinyin.ts +0 -20
  295. package/src/cli/cmd/tui/util/provider-origin.ts +0 -7
  296. package/src/cli/cmd/tui/util/revert-diff.ts +0 -18
  297. package/src/cli/cmd/tui/util/scroll.ts +0 -23
  298. package/src/cli/cmd/tui/util/selection.ts +0 -23
  299. package/src/cli/cmd/tui/util/signal.ts +0 -41
  300. package/src/cli/cmd/tui/util/sound.ts +0 -154
  301. package/src/cli/cmd/tui/util/system-locale.ts +0 -209
  302. package/src/cli/cmd/tui/util/terminal.ts +0 -110
  303. package/src/cli/cmd/tui/util/transcript.ts +0 -112
  304. package/src/cli/cmd/tui/util/vad.ts +0 -229
  305. package/src/cli/cmd/tui/util/voice.ts +0 -450
  306. package/src/cli/cmd/tui/win32.ts +0 -130
  307. package/src/cli/cmd/tui/worker.ts +0 -104
  308. package/src/cli/cmd/uninstall.ts +0 -351
  309. package/src/cli/cmd/upgrade.ts +0 -79
  310. package/src/cli/cmd/web.ts +0 -96
  311. package/src/cli/effect/prompt.ts +0 -25
  312. package/src/cli/error.ts +0 -82
  313. package/src/cli/heap.ts +0 -59
  314. package/src/cli/i18n.ts +0 -15
  315. package/src/cli/logo.ts +0 -53
  316. package/src/cli/network.ts +0 -68
  317. package/src/cli/ui.ts +0 -133
  318. package/src/cli/upgrade.ts +0 -41
  319. package/src/command/index.ts +0 -276
  320. package/src/command/template/initialize.txt +0 -66
  321. package/src/command/template/review.txt +0 -101
  322. package/src/config/agent.ts +0 -197
  323. package/src/config/command.ts +0 -69
  324. package/src/config/compose.ts +0 -26
  325. package/src/config/config.ts +0 -1047
  326. package/src/config/console-state.ts +0 -16
  327. package/src/config/entry-name.ts +0 -16
  328. package/src/config/error.ts +0 -21
  329. package/src/config/formatter.ts +0 -17
  330. package/src/config/history.ts +0 -21
  331. package/src/config/index.ts +0 -17
  332. package/src/config/keybinds.ts +0 -127
  333. package/src/config/layout.ts +0 -10
  334. package/src/config/lsp.ts +0 -45
  335. package/src/config/managed.ts +0 -70
  336. package/src/config/markdown.ts +0 -97
  337. package/src/config/mcp.ts +0 -172
  338. package/src/config/model-id.ts +0 -14
  339. package/src/config/parse.ts +0 -44
  340. package/src/config/paths.ts +0 -85
  341. package/src/config/permission.ts +0 -76
  342. package/src/config/plugin.ts +0 -88
  343. package/src/config/provider.ts +0 -118
  344. package/src/config/server.ts +0 -20
  345. package/src/config/skills.ts +0 -16
  346. package/src/config/variable.ts +0 -90
  347. package/src/control-plane/adaptors/index.ts +0 -52
  348. package/src/control-plane/adaptors/worktree.ts +0 -47
  349. package/src/control-plane/dev/debug-workspace-plugin.ts +0 -73
  350. package/src/control-plane/schema.ts +0 -19
  351. package/src/control-plane/sse.ts +0 -66
  352. package/src/control-plane/types.ts +0 -34
  353. package/src/control-plane/util.ts +0 -37
  354. package/src/control-plane/workspace-context.ts +0 -26
  355. package/src/control-plane/workspace.sql.ts +0 -17
  356. package/src/control-plane/workspace.ts +0 -615
  357. package/src/effect/app-runtime.ts +0 -146
  358. package/src/effect/bootstrap-runtime.ts +0 -33
  359. package/src/effect/bridge.ts +0 -48
  360. package/src/effect/cross-spawn-spawner.ts +0 -514
  361. package/src/effect/index.ts +0 -5
  362. package/src/effect/instance-ref.ts +0 -11
  363. package/src/effect/instance-registry.ts +0 -12
  364. package/src/effect/instance-state.ts +0 -81
  365. package/src/effect/logger.ts +0 -73
  366. package/src/effect/memo-map.ts +0 -3
  367. package/src/effect/observability.ts +0 -107
  368. package/src/effect/run-service.ts +0 -52
  369. package/src/effect/runner.ts +0 -210
  370. package/src/effect/runtime.ts +0 -19
  371. package/src/env/index.ts +0 -37
  372. package/src/file/ignore.ts +0 -81
  373. package/src/file/index.ts +0 -664
  374. package/src/file/protected.ts +0 -59
  375. package/src/file/ripgrep.ts +0 -485
  376. package/src/file/watcher.ts +0 -163
  377. package/src/flag/flag.ts +0 -185
  378. package/src/format/formatter.ts +0 -403
  379. package/src/format/index.ts +0 -203
  380. package/src/git/index.ts +0 -260
  381. package/src/global/index.ts +0 -54
  382. package/src/history/backfill.ts +0 -162
  383. package/src/history/extract.ts +0 -67
  384. package/src/history/fts-query.ts +0 -15
  385. package/src/history/fts.sql.ts +0 -20
  386. package/src/history/index.ts +0 -10
  387. package/src/history/resolve.ts +0 -65
  388. package/src/history/service.ts +0 -258
  389. package/src/history/writer.ts +0 -112
  390. package/src/id/id.ts +0 -87
  391. package/src/ide/index.ts +0 -73
  392. package/src/inbox/inbox-ref.ts +0 -38
  393. package/src/inbox/inbox.sql.ts +0 -26
  394. package/src/inbox/inbox.ts +0 -223
  395. package/src/inbox/index.ts +0 -3
  396. package/src/inbox/render.ts +0 -40
  397. package/src/index.ts +0 -268
  398. package/src/installation/index.ts +0 -362
  399. package/src/installation/version.ts +0 -8
  400. package/src/lsp/client.ts +0 -249
  401. package/src/lsp/diagnostic.ts +0 -29
  402. package/src/lsp/index.ts +0 -3
  403. package/src/lsp/language.ts +0 -120
  404. package/src/lsp/launch.ts +0 -21
  405. package/src/lsp/lsp.ts +0 -519
  406. package/src/lsp/server.ts +0 -1956
  407. package/src/mcp/auth.ts +0 -144
  408. package/src/mcp/index.ts +0 -939
  409. package/src/mcp/oauth-callback.ts +0 -236
  410. package/src/mcp/oauth-provider.ts +0 -214
  411. package/src/memory/fts-query.ts +0 -37
  412. package/src/memory/fts.sql.ts +0 -19
  413. package/src/memory/index.ts +0 -1
  414. package/src/memory/paths.ts +0 -116
  415. package/src/memory/reconcile.ts +0 -144
  416. package/src/memory/service.ts +0 -144
  417. package/src/metrics/client.ts +0 -40
  418. package/src/metrics/event.ts +0 -43
  419. package/src/metrics/index.ts +0 -5
  420. package/src/metrics/installation.ts +0 -18
  421. package/src/metrics/subscriber.ts +0 -58
  422. package/src/metrics/util.ts +0 -9
  423. package/src/node.ts +0 -6
  424. package/src/npm/config.ts +0 -0
  425. package/src/npm/index.ts +0 -293
  426. package/src/npmcli-config.d.ts +0 -43
  427. package/src/patch/index.ts +0 -680
  428. package/src/permission/arity.ts +0 -163
  429. package/src/permission/evaluate.ts +0 -15
  430. package/src/permission/index.ts +0 -382
  431. package/src/permission/schema.ts +0 -17
  432. package/src/plugin/checkpoint-splitover.ts +0 -60
  433. package/src/plugin/cloudflare.ts +0 -76
  434. package/src/plugin/codex.ts +0 -611
  435. package/src/plugin/github-copilot/copilot.ts +0 -368
  436. package/src/plugin/github-copilot/models.ts +0 -153
  437. package/src/plugin/index.ts +0 -637
  438. package/src/plugin/install.ts +0 -439
  439. package/src/plugin/loader.ts +0 -216
  440. package/src/plugin/matcher.ts +0 -33
  441. package/src/plugin/meta.ts +0 -188
  442. package/src/plugin/shared.ts +0 -323
  443. package/src/plugin/sleepy.ts +0 -244
  444. package/src/plugin/subagent-progress-checker.ts +0 -147
  445. package/src/project/bootstrap.ts +0 -59
  446. package/src/project/index.ts +0 -2
  447. package/src/project/instance.ts +0 -215
  448. package/src/project/project-id.ts +0 -48
  449. package/src/project/project.sql.ts +0 -16
  450. package/src/project/project.ts +0 -522
  451. package/src/project/schema.ts +0 -15
  452. package/src/project/vcs.ts +0 -227
  453. package/src/project/workspace-trust.ts +0 -67
  454. package/src/provider/auth.ts +0 -234
  455. package/src/provider/error.ts +0 -216
  456. package/src/provider/index.ts +0 -5
  457. package/src/provider/models-snapshot.d.ts +0 -2
  458. package/src/provider/models-snapshot.js +0 -3
  459. package/src/provider/models.ts +0 -180
  460. package/src/provider/provider.ts +0 -2098
  461. package/src/provider/schema.ts +0 -36
  462. package/src/provider/sdk/copilot/README.md +0 -5
  463. package/src/provider/sdk/copilot/chat/convert-to-openai-compatible-chat-messages.ts +0 -170
  464. package/src/provider/sdk/copilot/chat/get-response-metadata.ts +0 -15
  465. package/src/provider/sdk/copilot/chat/map-openai-compatible-finish-reason.ts +0 -19
  466. package/src/provider/sdk/copilot/chat/openai-compatible-api-types.ts +0 -64
  467. package/src/provider/sdk/copilot/chat/openai-compatible-chat-language-model.ts +0 -815
  468. package/src/provider/sdk/copilot/chat/openai-compatible-chat-options.ts +0 -28
  469. package/src/provider/sdk/copilot/chat/openai-compatible-metadata-extractor.ts +0 -44
  470. package/src/provider/sdk/copilot/chat/openai-compatible-prepare-tools.ts +0 -83
  471. package/src/provider/sdk/copilot/copilot-provider.ts +0 -100
  472. package/src/provider/sdk/copilot/index.ts +0 -2
  473. package/src/provider/sdk/copilot/openai-compatible-error.ts +0 -27
  474. package/src/provider/sdk/copilot/responses/convert-to-openai-responses-input.ts +0 -335
  475. package/src/provider/sdk/copilot/responses/map-openai-responses-finish-reason.ts +0 -22
  476. package/src/provider/sdk/copilot/responses/openai-config.ts +0 -18
  477. package/src/provider/sdk/copilot/responses/openai-error.ts +0 -22
  478. package/src/provider/sdk/copilot/responses/openai-responses-api-types.ts +0 -214
  479. package/src/provider/sdk/copilot/responses/openai-responses-language-model.ts +0 -1770
  480. package/src/provider/sdk/copilot/responses/openai-responses-prepare-tools.ts +0 -173
  481. package/src/provider/sdk/copilot/responses/openai-responses-settings.ts +0 -1
  482. package/src/provider/sdk/copilot/responses/tool/code-interpreter.ts +0 -87
  483. package/src/provider/sdk/copilot/responses/tool/file-search.ts +0 -127
  484. package/src/provider/sdk/copilot/responses/tool/image-generation.ts +0 -114
  485. package/src/provider/sdk/copilot/responses/tool/local-shell.ts +0 -64
  486. package/src/provider/sdk/copilot/responses/tool/web-search-preview.ts +0 -103
  487. package/src/provider/sdk/copilot/responses/tool/web-search.ts +0 -102
  488. package/src/provider/session-check.ts +0 -163
  489. package/src/provider/transform.ts +0 -1376
  490. package/src/pty/index.ts +0 -364
  491. package/src/pty/pty.bun.ts +0 -26
  492. package/src/pty/pty.node.ts +0 -27
  493. package/src/pty/pty.ts +0 -25
  494. package/src/pty/schema.ts +0 -17
  495. package/src/question/index.ts +0 -252
  496. package/src/question/schema.ts +0 -17
  497. package/src/server/adapter.bun.ts +0 -40
  498. package/src/server/adapter.node.ts +0 -66
  499. package/src/server/adapter.ts +0 -21
  500. package/src/server/auth.ts +0 -16
  501. package/src/server/error.ts +0 -53
  502. package/src/server/event.ts +0 -7
  503. package/src/server/fence.ts +0 -81
  504. package/src/server/mdns.ts +0 -60
  505. package/src/server/middleware.ts +0 -94
  506. package/src/server/projectors.ts +0 -28
  507. package/src/server/proxy.ts +0 -171
  508. package/src/server/pty-ticket.ts +0 -42
  509. package/src/server/rate-limit.ts +0 -38
  510. package/src/server/routes/control/index.ts +0 -160
  511. package/src/server/routes/control/workspace.ts +0 -203
  512. package/src/server/routes/global.ts +0 -367
  513. package/src/server/routes/instance/bash-interactive.ts +0 -82
  514. package/src/server/routes/instance/config.ts +0 -89
  515. package/src/server/routes/instance/event.ts +0 -108
  516. package/src/server/routes/instance/experimental.ts +0 -408
  517. package/src/server/routes/instance/file.ts +0 -190
  518. package/src/server/routes/instance/httpapi/config.ts +0 -51
  519. package/src/server/routes/instance/httpapi/permission.ts +0 -72
  520. package/src/server/routes/instance/httpapi/project.ts +0 -62
  521. package/src/server/routes/instance/httpapi/provider.ts +0 -142
  522. package/src/server/routes/instance/httpapi/question.ts +0 -121
  523. package/src/server/routes/instance/httpapi/server.ts +0 -153
  524. package/src/server/routes/instance/index.ts +0 -301
  525. package/src/server/routes/instance/mcp.ts +0 -260
  526. package/src/server/routes/instance/middleware.ts +0 -44
  527. package/src/server/routes/instance/permission.ts +0 -73
  528. package/src/server/routes/instance/project.ts +0 -122
  529. package/src/server/routes/instance/provider.ts +0 -158
  530. package/src/server/routes/instance/pty.ts +0 -302
  531. package/src/server/routes/instance/question.ts +0 -162
  532. package/src/server/routes/instance/session.ts +0 -1328
  533. package/src/server/routes/instance/sync.ts +0 -143
  534. package/src/server/routes/instance/trace.ts +0 -59
  535. package/src/server/routes/instance/tui.ts +0 -384
  536. package/src/server/routes/instance/workflows.ts +0 -142
  537. package/src/server/routes/ui.ts +0 -37
  538. package/src/server/server.ts +0 -146
  539. package/src/server/workspace.ts +0 -122
  540. package/src/session/auto-dream.ts +0 -123
  541. package/src/session/boundary.ts +0 -77
  542. package/src/session/budgeted-read.ts +0 -118
  543. package/src/session/checkpoint-align.ts +0 -29
  544. package/src/session/checkpoint-context.ts +0 -36
  545. package/src/session/checkpoint-paths.ts +0 -86
  546. package/src/session/checkpoint-progress-reconcile.ts +0 -111
  547. package/src/session/checkpoint-retry.ts +0 -192
  548. package/src/session/checkpoint-templates.ts +0 -114
  549. package/src/session/checkpoint-validator.ts +0 -259
  550. package/src/session/checkpoint.ts +0 -1560
  551. package/src/session/classify.ts +0 -117
  552. package/src/session/claude-import.ts +0 -381
  553. package/src/session/codex-import.ts +0 -416
  554. package/src/session/compaction.ts +0 -545
  555. package/src/session/external-import.sql.ts +0 -18
  556. package/src/session/external-import.ts +0 -136
  557. package/src/session/goal.ts +0 -232
  558. package/src/session/index.ts +0 -1
  559. package/src/session/instruction.ts +0 -276
  560. package/src/session/last-message-info.ts +0 -32
  561. package/src/session/llm-request-prefix.ts +0 -82
  562. package/src/session/llm.ts +0 -737
  563. package/src/session/max-mode.ts +0 -410
  564. package/src/session/message-v2.ts +0 -1138
  565. package/src/session/message.ts +0 -191
  566. package/src/session/opencode-import.ts +0 -281
  567. package/src/session/overflow.ts +0 -53
  568. package/src/session/prefix-capture-ref.ts +0 -48
  569. package/src/session/processor.ts +0 -983
  570. package/src/session/projectors.ts +0 -137
  571. package/src/session/prompt/anthropic.txt +0 -154
  572. package/src/session/prompt/beast.txt +0 -155
  573. package/src/session/prompt/build-switch.txt +0 -5
  574. package/src/session/prompt/codex.txt +0 -79
  575. package/src/session/prompt/compose.txt +0 -119
  576. package/src/session/prompt/copilot-gpt-5.txt +0 -143
  577. package/src/session/prompt/deepseek.txt +0 -131
  578. package/src/session/prompt/default.old.txt +0 -151
  579. package/src/session/prompt/default.txt +0 -172
  580. package/src/session/prompt/gemini.txt +0 -155
  581. package/src/session/prompt/glm.txt +0 -51
  582. package/src/session/prompt/gpt.txt +0 -107
  583. package/src/session/prompt/kimi.txt +0 -95
  584. package/src/session/prompt/max-steps.txt +0 -16
  585. package/src/session/prompt/minimax.txt +0 -140
  586. package/src/session/prompt/text-loop-recovery.ts +0 -40
  587. package/src/session/prompt/text-ngram-detection.ts +0 -87
  588. package/src/session/prompt/trinity.txt +0 -97
  589. package/src/session/prompt.ts +0 -3878
  590. package/src/session/prune.ts +0 -481
  591. package/src/session/retry.ts +0 -178
  592. package/src/session/revert.ts +0 -161
  593. package/src/session/run-state.ts +0 -135
  594. package/src/session/schema.ts +0 -36
  595. package/src/session/session.sql.ts +0 -110
  596. package/src/session/session.ts +0 -908
  597. package/src/session/status.ts +0 -89
  598. package/src/session/summary.ts +0 -163
  599. package/src/session/system.ts +0 -96
  600. package/src/session/todo.ts +0 -77
  601. package/src/session/trajectory.ts +0 -98
  602. package/src/share/index.ts +0 -2
  603. package/src/share/session.ts +0 -57
  604. package/src/share/share-next.ts +0 -381
  605. package/src/share/share.sql.ts +0 -13
  606. package/src/shell/shell.ts +0 -124
  607. package/src/skill/builtin/.bundle/self-extend/SKILL.md +0 -131
  608. package/src/skill/builtin/.bundle/self-extend/reference/hook-api.md +0 -242
  609. package/src/skill/builtin/.bundle/self-extend/reference/skill-api.md +0 -114
  610. package/src/skill/builtin/.bundle/self-extend/reference/tool-api.md +0 -115
  611. package/src/skill/builtin/.bundle/self-extend/reference/tui-api.md +0 -258
  612. package/src/skill/builtin/bundle.macro.ts +0 -30
  613. package/src/skill/builtin/extract.ts +0 -41
  614. package/src/skill/compose/.bundle/ask/SKILL.md +0 -58
  615. package/src/skill/compose/.bundle/brainstorm/SKILL.md +0 -200
  616. package/src/skill/compose/.bundle/brainstorm/scripts/frame-template.html +0 -214
  617. package/src/skill/compose/.bundle/brainstorm/scripts/helper.js +0 -88
  618. package/src/skill/compose/.bundle/brainstorm/scripts/server.cjs +0 -354
  619. package/src/skill/compose/.bundle/brainstorm/scripts/start-server.sh +0 -148
  620. package/src/skill/compose/.bundle/brainstorm/scripts/stop-server.sh +0 -56
  621. package/src/skill/compose/.bundle/brainstorm/spec-document-reviewer-prompt.md +0 -50
  622. package/src/skill/compose/.bundle/brainstorm/visual-companion.md +0 -258
  623. package/src/skill/compose/.bundle/debug/CREATION-LOG.md +0 -119
  624. package/src/skill/compose/.bundle/debug/SKILL.md +0 -297
  625. package/src/skill/compose/.bundle/debug/condition-based-waiting-example.ts +0 -158
  626. package/src/skill/compose/.bundle/debug/condition-based-waiting.md +0 -106
  627. package/src/skill/compose/.bundle/debug/defense-in-depth.md +0 -122
  628. package/src/skill/compose/.bundle/debug/find-polluter.sh +0 -63
  629. package/src/skill/compose/.bundle/debug/root-cause-tracing.md +0 -144
  630. package/src/skill/compose/.bundle/debug/test-academic.md +0 -14
  631. package/src/skill/compose/.bundle/debug/test-pressure-1.md +0 -58
  632. package/src/skill/compose/.bundle/debug/test-pressure-2.md +0 -68
  633. package/src/skill/compose/.bundle/debug/test-pressure-3.md +0 -69
  634. package/src/skill/compose/.bundle/execute/SKILL.md +0 -71
  635. package/src/skill/compose/.bundle/feedback/SKILL.md +0 -214
  636. package/src/skill/compose/.bundle/merge/SKILL.md +0 -252
  637. package/src/skill/compose/.bundle/new-skill/SKILL.md +0 -211
  638. package/src/skill/compose/.bundle/parallel/SKILL.md +0 -168
  639. package/src/skill/compose/.bundle/plan/SKILL.md +0 -183
  640. package/src/skill/compose/.bundle/plan/plan-document-reviewer-prompt.md +0 -50
  641. package/src/skill/compose/.bundle/report/SKILL.md +0 -180
  642. package/src/skill/compose/.bundle/review/SKILL.md +0 -104
  643. package/src/skill/compose/.bundle/review/code-reviewer.md +0 -178
  644. package/src/skill/compose/.bundle/subagent/SKILL.md +0 -325
  645. package/src/skill/compose/.bundle/subagent/code-quality-reviewer-prompt.md +0 -26
  646. package/src/skill/compose/.bundle/subagent/implementer-prompt.md +0 -128
  647. package/src/skill/compose/.bundle/subagent/spec-reviewer-prompt.md +0 -118
  648. package/src/skill/compose/.bundle/tdd/SKILL.md +0 -360
  649. package/src/skill/compose/.bundle/tdd/testing-anti-patterns.md +0 -299
  650. package/src/skill/compose/.bundle/verify/SKILL.md +0 -140
  651. package/src/skill/compose/.bundle/worktree/SKILL.md +0 -234
  652. package/src/skill/compose/LICENSE-karpathy +0 -28
  653. package/src/skill/compose/bundle.macro.ts +0 -30
  654. package/src/skill/compose/extract.ts +0 -85
  655. package/src/skill/discovery.ts +0 -114
  656. package/src/skill/index.ts +0 -328
  657. package/src/snapshot/index.ts +0 -777
  658. package/src/sql.d.ts +0 -4
  659. package/src/storage/db.bun.ts +0 -8
  660. package/src/storage/db.node.ts +0 -8
  661. package/src/storage/db.ts +0 -172
  662. package/src/storage/index.ts +0 -26
  663. package/src/storage/json-migration.ts +0 -426
  664. package/src/storage/read-sqlite.bun.ts +0 -11
  665. package/src/storage/read-sqlite.node.ts +0 -13
  666. package/src/storage/read-sqlite.ts +0 -10
  667. package/src/storage/schema.sql.ts +0 -10
  668. package/src/storage/schema.ts +0 -7
  669. package/src/storage/storage.ts +0 -331
  670. package/src/sync/README.md +0 -179
  671. package/src/sync/event.sql.ts +0 -16
  672. package/src/sync/index.ts +0 -278
  673. package/src/sync/schema.ts +0 -14
  674. package/src/task/events.ts +0 -28
  675. package/src/task/gate-state.ts +0 -54
  676. package/src/task/gate.ts +0 -116
  677. package/src/task/index.ts +0 -1
  678. package/src/task/registry.ts +0 -394
  679. package/src/task/schema.ts +0 -43
  680. package/src/task/task.sql.ts +0 -50
  681. package/src/team/events.ts +0 -22
  682. package/src/team/index.ts +0 -113
  683. package/src/team/schema.ts +0 -31
  684. package/src/temporary.ts +0 -33
  685. package/src/tool/actor.shell.txt +0 -72
  686. package/src/tool/actor.ts +0 -804
  687. package/src/tool/actor.txt +0 -103
  688. package/src/tool/apply_patch.ts +0 -308
  689. package/src/tool/apply_patch.txt +0 -33
  690. package/src/tool/bash-interactive.ts +0 -183
  691. package/src/tool/bash.ts +0 -704
  692. package/src/tool/bash.txt +0 -123
  693. package/src/tool/change-directory.ts +0 -91
  694. package/src/tool/codesearch.ts +0 -63
  695. package/src/tool/codesearch.txt +0 -12
  696. package/src/tool/edit.ts +0 -693
  697. package/src/tool/edit.txt +0 -10
  698. package/src/tool/external-directory.ts +0 -132
  699. package/src/tool/glob.ts +0 -100
  700. package/src/tool/glob.txt +0 -6
  701. package/src/tool/grep.ts +0 -145
  702. package/src/tool/grep.txt +0 -8
  703. package/src/tool/history.ts +0 -146
  704. package/src/tool/history.txt +0 -17
  705. package/src/tool/index.ts +0 -4
  706. package/src/tool/invalid.ts +0 -20
  707. package/src/tool/invocation-style.ts +0 -17
  708. package/src/tool/lsp.ts +0 -91
  709. package/src/tool/lsp.txt +0 -19
  710. package/src/tool/mcp-exa.ts +0 -78
  711. package/src/tool/memory-path-guard.ts +0 -162
  712. package/src/tool/memory.ts +0 -81
  713. package/src/tool/memory.txt +0 -69
  714. package/src/tool/multiedit.ts +0 -54
  715. package/src/tool/multiedit.txt +0 -41
  716. package/src/tool/notebook-edit.ts +0 -225
  717. package/src/tool/notebook-edit.txt +0 -10
  718. package/src/tool/plan-enter.txt +0 -16
  719. package/src/tool/plan-exit.txt +0 -14
  720. package/src/tool/plan.ts +0 -179
  721. package/src/tool/question.ts +0 -67
  722. package/src/tool/question.txt +0 -10
  723. package/src/tool/read-state.ts +0 -44
  724. package/src/tool/read.ts +0 -327
  725. package/src/tool/read.txt +0 -14
  726. package/src/tool/recoverable.ts +0 -35
  727. package/src/tool/registry.ts +0 -429
  728. package/src/tool/schema.ts +0 -17
  729. package/src/tool/session-cwd.ts +0 -35
  730. package/src/tool/shell-tokenize.ts +0 -374
  731. package/src/tool/shell-wrap.ts +0 -235
  732. package/src/tool/skill.ts +0 -76
  733. package/src/tool/skill.txt +0 -5
  734. package/src/tool/task.shell.txt +0 -57
  735. package/src/tool/task.ts +0 -456
  736. package/src/tool/task.txt +0 -56
  737. package/src/tool/tool.ts +0 -166
  738. package/src/tool/truncate.ts +0 -201
  739. package/src/tool/truncation-dir.ts +0 -4
  740. package/src/tool/webfetch.ts +0 -208
  741. package/src/tool/webfetch.txt +0 -13
  742. package/src/tool/websearch/index.ts +0 -104
  743. package/src/tool/websearch/sleepy.ts +0 -118
  744. package/src/tool/websearch/websearch.txt +0 -14
  745. package/src/tool/workflow.ts +0 -357
  746. package/src/tool/workflow.txt +0 -25
  747. package/src/tool/write.ts +0 -88
  748. package/src/tool/write.txt +0 -10
  749. package/src/util/abort.ts +0 -35
  750. package/src/util/archive.ts +0 -15
  751. package/src/util/color.ts +0 -17
  752. package/src/util/data-url.ts +0 -9
  753. package/src/util/defer.ts +0 -10
  754. package/src/util/effect-http-client.ts +0 -11
  755. package/src/util/effect-zod.ts +0 -367
  756. package/src/util/env-info.ts +0 -62
  757. package/src/util/error.ts +0 -78
  758. package/src/util/filesystem.ts +0 -243
  759. package/src/util/fn.ts +0 -21
  760. package/src/util/format.ts +0 -20
  761. package/src/util/iife.ts +0 -3
  762. package/src/util/index.ts +0 -14
  763. package/src/util/keybind.ts +0 -101
  764. package/src/util/lazy.ts +0 -18
  765. package/src/util/local-context.ts +0 -23
  766. package/src/util/locale.ts +0 -79
  767. package/src/util/lock.ts +0 -96
  768. package/src/util/log.ts +0 -234
  769. package/src/util/media.ts +0 -26
  770. package/src/util/network.ts +0 -9
  771. package/src/util/process.ts +0 -174
  772. package/src/util/provider-priority.ts +0 -48
  773. package/src/util/queue.ts +0 -60
  774. package/src/util/record.ts +0 -3
  775. package/src/util/rpc.ts +0 -64
  776. package/src/util/schema.ts +0 -53
  777. package/src/util/scrap.ts +0 -10
  778. package/src/util/signal.ts +0 -12
  779. package/src/util/sleepy-process.ts +0 -24
  780. package/src/util/ssrf.ts +0 -116
  781. package/src/util/timeout.ts +0 -14
  782. package/src/util/token.ts +0 -5
  783. package/src/util/tool-compat.ts +0 -144
  784. package/src/util/update-schema.ts +0 -13
  785. package/src/util/which.ts +0 -14
  786. package/src/util/wildcard.ts +0 -57
  787. package/src/workflow/builtin/compose.js +0 -749
  788. package/src/workflow/builtin/deep-research.js +0 -398
  789. package/src/workflow/builtin.ts +0 -59
  790. package/src/workflow/events.ts +0 -72
  791. package/src/workflow/meta.ts +0 -335
  792. package/src/workflow/persistence.ts +0 -312
  793. package/src/workflow/resolve.ts +0 -45
  794. package/src/workflow/runtime-ref.ts +0 -18
  795. package/src/workflow/runtime.ts +0 -1447
  796. package/src/workflow/sandbox.ts +0 -286
  797. package/src/workflow/workflow.sql.ts +0 -31
  798. package/src/workflow/workspace.ts +0 -69
  799. package/src/worktree/index.ts +0 -629
@@ -1,1560 +0,0 @@
1
- import fs from "fs/promises"
2
- import path from "path"
3
- import { Global } from "@/global"
4
- import { Bus } from "@/bus"
5
- import { Config } from "@/config"
6
- import { Memory } from "@/memory"
7
- import { MemoryFtsTable } from "@/memory/fts.sql"
8
- import { TaskRegistry } from "@/task/registry"
9
- import { ActorRegistry } from "@/actor/registry"
10
- import type { AgentOutcome, ForkContext } from "@/actor/spawn"
11
- import { spawnRef } from "@/actor/spawn-ref"
12
- import { prefixCaptureRef } from "./prefix-capture-ref"
13
- import { Database, and, eq, or } from "@/storage"
14
- import { Instance } from "@/project/instance"
15
- import { ProjectID } from "@/project/schema"
16
- import { SessionTable } from "./session.sql"
17
- import * as Session from "./session"
18
- import { MessageV2 } from "./message-v2"
19
- import { SessionID, MessageID, PartID } from "./schema"
20
- import { Log, Token } from "../util"
21
- import { Effect, Layer, Deferred, Context, Scope } from "effect"
22
- import { makeRuntime } from "@/effect/run-service"
23
- import type { ActorPromptOps } from "@/tool/actor"
24
- import type { ProviderID, ModelID } from "../provider/schema"
25
- import PROMPT_CHECKPOINT_WRITER from "@/agent/prompt/checkpoint-writer.txt"
26
- import { WriterCachePerf } from "@/actor/events"
27
- import {
28
- metaDir,
29
- checkpointPath,
30
- memoryPath,
31
- notesPath,
32
- globalMemoryPath,
33
- migrateProjectMemory,
34
- } from "./checkpoint-paths"
35
- import { readBudgeted, readBudgetedSectionAware } from "./budgeted-read"
36
- import type { LastMessageInfo } from "./last-message-info"
37
- import { CHECKPOINT_TEMPLATE, MEMORY_TEMPLATE, NOTES_TEMPLATE, CHECKPOINT_SECTION_BUDGETS } from "./checkpoint-templates"
38
- import { adjustBoundaryForApiInvariants } from "./boundary"
39
- import { alignToNonToolResultUser } from "./checkpoint-align"
40
- import { loadPriorDiscoveredTitles } from "./checkpoint-retry"
41
- import * as CheckpointContext from "./checkpoint-context"
42
- import { buildProgressDiff } from "./checkpoint-progress-reconcile"
43
-
44
- const log = Log.create({ service: "session.checkpoint" })
45
-
46
- function truncate(s: string, max: number): string {
47
- return s.length <= max ? s : s.slice(0, max - 60) + "\n... (truncated, full body at file)"
48
- }
49
-
50
- /**
51
- * Truncate verbatim user input that exceeds per-message cap. Keeps head (~60%)
52
- * + tail (~30%) with an elision marker pointing at messageID for full recall
53
- * via the history tool's operation=around. ~4 chars/token approximation matches
54
- * Token.estimate.
55
- */
56
- function truncateVerbatimUserMsg(text: string, capTokens: number, messageID: string): string {
57
- if (Token.estimate(text) <= capTokens) return text
58
- // slice() cuts on UTF-16 code units, which can split a surrogate pair at the
59
- // boundary; trim a dangling high surrogate off the head and a leading low
60
- // surrogate off the tail so emoji / non-BMP chars don't render as garbage.
61
- const head = text.slice(0, Math.floor(capTokens * 0.6) * 4).replace(/[\uD800-\uDBFF]$/, "")
62
- const tail = text.slice(-Math.floor(capTokens * 0.3) * 4).replace(/^[\uDC00-\uDFFF]/, "")
63
- const elidedTokens = Token.estimate(text) - Token.estimate(head) - Token.estimate(tail)
64
- return [
65
- head,
66
- `[…elided ${elidedTokens} tokens; messageID=${messageID}; use the history tool with operation=around to fetch full content]`,
67
- tail,
68
- ].join("\n")
69
- }
70
-
71
- /**
72
- * Concatenate text-typed parts of a user message into a single string. Skips
73
- * tool/file/image/etc. parts and synthetic text (e.g. rebuild-boundary content
74
- * injected by insertRebuildBoundary) — only true user prose contributes.
75
- */
76
- function userMsgText(parts: Array<{ type: string; text?: string; synthetic?: boolean }>): string {
77
- return parts
78
- .filter((p) => p.type === "text" && !p.synthetic && typeof p.text === "string" && p.text.length > 0)
79
- .map((p) => p.text!)
80
- .join("\n")
81
- }
82
-
83
- function autonomousLoopReminder(): string {
84
- return [
85
- "<system-reminder>",
86
- "You are mid-loop in an autonomous task. Continue your work loop:",
87
- "respond to the tool results below and proceed to the next iteration.",
88
- "</system-reminder>",
89
- ].join("\n")
90
- }
91
-
92
- function stopReminder(focusTaskID: string | undefined): string {
93
- const taskHint = focusTaskID
94
- ? `Consult this session's tasks/${focusTaskID}/progress.md head section.`
95
- : "Consult the most recently active task's progress.md head section."
96
- return [
97
- "<system-reminder>",
98
- "The previous assistant turn ended with a stop. Before stopping again,",
99
- taskHint,
100
- "Compare the Task spec to the latest Progress entries. If the task is",
101
- "incomplete, proceed to the next concrete step. Only stop when the spec",
102
- "is genuinely satisfied or you need user input you cannot infer.",
103
- "</system-reminder>",
104
- ].join("\n")
105
- }
106
-
107
- function toolResultContinueReminder(): string {
108
- return [
109
- "<system-reminder>",
110
- "Tool results above are real history from the autonomous loop. Process",
111
- "them and continue to the next iteration. Do not pause to summarize.",
112
- "</system-reminder>",
113
- ].join("\n")
114
- }
115
-
116
- async function ensureCheckpointTemplate(checkpointFile: string): Promise<void> {
117
- if (!(await Bun.file(checkpointFile).exists())) {
118
- await fs.mkdir(path.dirname(checkpointFile), { recursive: true })
119
- await Bun.write(checkpointFile, CHECKPOINT_TEMPLATE)
120
- }
121
- }
122
-
123
- async function ensureMemoryTemplate(memoryFile: string): Promise<void> {
124
- if (!(await Bun.file(memoryFile).exists())) {
125
- await fs.mkdir(path.dirname(memoryFile), { recursive: true })
126
- await Bun.write(memoryFile, MEMORY_TEMPLATE)
127
- }
128
- }
129
-
130
- async function ensureNotesTemplate(notesFile: string): Promise<void> {
131
- if (!(await Bun.file(notesFile).exists())) {
132
- await fs.mkdir(path.dirname(notesFile), { recursive: true })
133
- await Bun.write(notesFile, NOTES_TEMPLATE)
134
- }
135
- }
136
-
137
- // Tail preservation budget (token-budgeted boundary).
138
- // Session-memory compact: minimum guarantees the LLM has enough
139
- // recent-context anchor (avoids the agent-Read-loop failure mode from
140
- // v4 → v5 spec rationale); maximum is a SOFT ceiling on backward
141
- // expansion — i.e. when the natural tail is below the floor we expand
142
- // backward UP TO maxTokens, but if the natural tail already exceeds
143
- // maxTokens we leave it alone. Single-message-granularity cap would
144
- // break tool_use/result pairing.
145
- //
146
- // 20K is the empirical sweet spot — observed compact output is ~20K,
147
- // not the 40K nominal default. The 40K appears in source as fallback,
148
- // but the upstream config likely tunes it lower in production.
149
- const TAIL_MIN_TOKENS = 10_000
150
- const TAIL_MAX_TOKENS = 20_000
151
- const TAIL_MIN_TEXT_BLOCK_MESSAGES = 5
152
-
153
- // Rebuild-time microcompact (see
154
- // docs/superpowers/specs/2026-06-03-rebuild-tail-microcompact-design.md).
155
- //
156
- // After computing the boundary, msgs strictly newer than the boundary
157
- // survive into the rebuild context. Their tool_use parts are kept (so the
158
- // LLM still sees what action was taken), but for tools in this whitelist
159
- // the tool_result content is replaced with a placeholder. Result is either
160
- // large-and-regeneratable (read/bash/grep/glob/webfetch/websearch) or
161
- // essentially a "done" confirmation (edit/write/multiedit). Tools NOT here
162
- // carry state the LLM references later (actor/task/question/skill/memory).
163
- const COMPACTABLE_TOOL_NAMES = new Set<string>([
164
- "read",
165
- "bash",
166
- "grep",
167
- "glob",
168
- "webfetch",
169
- "websearch",
170
- "edit",
171
- "write",
172
- "multiedit",
173
- "apply_patch",
174
- "codesearch",
175
- ])
176
-
177
- function estimateMessageTokens(m: { parts: Array<{ type: string; [k: string]: unknown }> }): number {
178
- // Same estimator used elsewhere in checkpoint.ts (Token.estimate over JSON).
179
- // Sum across all parts of the message.
180
- let sum = 0
181
- for (const p of m.parts) {
182
- // JSON.stringify throws on circular structures. Parser-produced parts are
183
- // plain POJOs, but a plugin-injected part could contain a cycle. Fall back
184
- // to a conservative NON-ZERO estimate so a bad part is never counted as
185
- // "free" — a 0 here would let the tail-boundary algorithm swallow the part
186
- // for nothing and skew the budget. The constant overstates a typical part,
187
- // which is the safe direction (boundary walks back, never forward).
188
- try {
189
- sum += Token.estimate(JSON.stringify(p))
190
- } catch {
191
- sum += 1000
192
- }
193
- }
194
- return sum
195
- }
196
-
197
- function hasTextBlocks(m: { parts: Array<{ type: string }> }): boolean {
198
- return m.parts.some((p) => p.type === "text" || p.type === "reasoning")
199
- }
200
-
201
- /**
202
- * Token-budgeted, role-aware boundary choice for the preserved tail.
203
- *
204
- * Returns the ID of the FIRST message to preserve (boundary message ID;
205
- * everything strictly before this ID is summarized into checkpoint.md and
206
- * discarded from the rebuild context).
207
- *
208
- * Algorithm (token-budgeted boundary):
209
- *
210
- * 1. Start at the last finished assistant index minus 1, take it+successors
211
- * as the candidate tail (preserves spec 2 starting point so reasonable
212
- * tails are unchanged).
213
- * 2. If tail tokens already >= TAIL_MAX_TOKENS: leave boundary as-is and
214
- * return. Do NOT pull boundary forward — message-granularity truncation
215
- * would split tool_use/tool_result pairs (downstream
216
- * adjustBoundaryForApiInvariants would just walk back, net no-op + risk
217
- * of thinking-block breaks). The cap is a SOFT ceiling on backward
218
- * expansion, not a hard upper bound on tail size. If a single
219
- * assistant turn legitimately produces 60K of tool_result, the tail
220
- * will be 60K and that's fine.
221
- * 3. Else if tail tokens < TAIL_MIN_TOKENS or text-block messages < min:
222
- * walk backward (earlier) one message at a time until both minimums
223
- * met OR TAIL_MAX_TOKENS hit OR no more messages.
224
- *
225
- * The downstream adjustBoundaryForApiInvariants call (in
226
- * tryStartCheckpointWriter) handles tool_use/tool_result pairing and
227
- * thinking-block atomicity — this function does NOT need to.
228
- *
229
- * Edge cases:
230
- * - msgs.length === 0: return "" (matches old behavior).
231
- * - No finished assistant: return msgs[0].info.id (degenerate; caller should
232
- * not be invoking trim here, but stay safe).
233
- * - lastAsstIdx === 0: return msgs[0].info.id (degenerate tail).
234
- */
235
- export function computeBoundary(
236
- msgs: ReadonlyArray<{ info: { id: string; role: "user" | "assistant"; finish?: string }; parts: Array<{ type: string; [k: string]: unknown }> }>,
237
- ): string {
238
- if (msgs.length === 0) return ""
239
- const lastAsstIdx = msgs.findLastIndex(
240
- (m) => m.info.role === "assistant" && m.info.finish !== undefined,
241
- )
242
- if (lastAsstIdx <= 0) return msgs[lastAsstIdx >= 0 ? lastAsstIdx : 0].info.id
243
-
244
- // Token estimate per message (computed once).
245
- const tokens = msgs.map((m) => estimateMessageTokens(m))
246
-
247
- // Spec 2 starting point: lastAsstIdx - 1.
248
- let startIdx = lastAsstIdx - 1
249
- let tailSum = 0
250
- let textBlockCount = 0
251
- for (let i = startIdx; i < msgs.length; i++) {
252
- tailSum += tokens[i]
253
- if (hasTextBlocks(msgs[i])) textBlockCount += 1
254
- }
255
-
256
- // Natural tail already >= cap: leave it alone (soft ceiling; do NOT pull
257
- // boundary forward — see jsdoc rationale).
258
- if (tailSum >= TAIL_MAX_TOKENS) {
259
- return msgs[startIdx].info.id
260
- }
261
-
262
- // Tail too small — pull boundary earlier (include more history)
263
- // until both floors met, capped at TAIL_MAX_TOKENS.
264
- while (
265
- startIdx > 0 &&
266
- tailSum < TAIL_MAX_TOKENS &&
267
- (tailSum < TAIL_MIN_TOKENS || textBlockCount < TAIL_MIN_TEXT_BLOCK_MESSAGES)
268
- ) {
269
- startIdx -= 1
270
- tailSum += tokens[startIdx]
271
- if (hasTextBlocks(msgs[startIdx])) textBlockCount += 1
272
- }
273
- return msgs[startIdx].info.id
274
- }
275
-
276
- function renderSectionBudgets(budgets: Record<string, number>): string {
277
- const entries = Object.entries(budgets)
278
- if (entries.length === 0) {
279
- throw new Error("CHECKPOINT_SECTION_BUDGETS is empty — F43 substitution would produce an empty prompt block")
280
- }
281
- const cols = 3
282
- const lines: string[] = ["Section budgets (~tokens):"]
283
- for (let i = 0; i < entries.length; i += cols) {
284
- const row = entries
285
- .slice(i, i + cols)
286
- .map(([k, v]) => `${k}: ${v}`)
287
- .join(" ")
288
- lines.push(` ${row}`)
289
- }
290
- return lines.join("\n")
291
- }
292
-
293
- /**
294
- * Composes the full writer prompt for the checkpoint subagent.
295
- *
296
- * The body wraps PROMPT_CHECKPOINT_WRITER with an ABSOLUTE-PATHS preamble
297
- * that pins CHECKPOINT_PATH/MEMORY_PATH/TASK_MEM_DIR to the current session's
298
- * dirs — without this, the model frequently invents legacy `/data/checkpoints/`
299
- * style paths from training-data lookalikes.
300
- */
301
- function composeWriterPrompt(input: {
302
- checkpointFile: string
303
- memoryFile: string
304
- taskMemDir: string
305
- notesFile: string
306
- rangeDesc: string
307
- progressDiff: string // Spec ② Chain 2: empty string when nothing to reconcile
308
- }): string {
309
- return [
310
- "<system-reminder>",
311
- "You are now operating in checkpoint-writer mode. Ignore the general coding-assistant framing in the system prompt above. The read, write, edit, glob, grep, and task tools are available; do not invoke others.",
312
- "",
313
- "========================================================================",
314
- "ABSOLUTE PATHS — USE THESE VERBATIM. NEVER COMPUTE, INFER, OR MODIFY.",
315
- "========================================================================",
316
- "",
317
- `CHECKPOINT_PATH = ${input.checkpointFile}`,
318
- `MEMORY_PATH = ${input.memoryFile}`,
319
- `TASK_MEM_DIR = ${input.taskMemDir}`,
320
- `NOTES_PATH = ${input.notesFile}`,
321
- "",
322
- "When using the Write tool, the first arg MUST be one of these literal",
323
- "absolute paths (or for task narrative, TASK_MEM_DIR + '/' + task_id +",
324
- "'/progress.md' or '/notes.md'). Do NOT abbreviate. Do NOT change",
325
- "parent directories. Do NOT insert paths from memory of similar projects.",
326
- "If you find yourself typing '/data/checkpoints/' as a parent, STOP — that",
327
- "is the legacy v2 layout and is wrong. The current parent for the",
328
- "checkpoint file is the directory portion of CHECKPOINT_PATH above.",
329
- "========================================================================",
330
- "",
331
- input.progressDiff,
332
- "",
333
- PROMPT_CHECKPOINT_WRITER.replace("{{SECTION_BUDGETS}}", renderSectionBudgets(CHECKPOINT_SECTION_BUDGETS)),
334
- "</system-reminder>",
335
- "",
336
- `Write the next checkpoint for this session.`,
337
- "",
338
- input.rangeDesc,
339
- "",
340
- "Use the `task` tool for ALL task state ops (create / start / progress / done / abandon / approve / rename / block / unblock / batch_create). Use the Write tool for the checkpoint, memory, and task narrative files at the CHECKPOINT_PATH / MEMORY_PATH / TASK_MEM_DIR locations declared above. After all writes and tool calls, stop immediately.",
341
- ].join("\n")
342
- }
343
-
344
- function aggregateWriterCacheMetrics(
345
- sessions: Session.Interface,
346
- sessionID: SessionID,
347
- actorID: string,
348
- ) {
349
- return Effect.gen(function* () {
350
- const msgs = yield* sessions.messages({ sessionID, agentID: "*" })
351
- let totalInput = 0
352
- let cacheRead = 0
353
- let cacheWrite = 0
354
- let assistantCount = 0
355
- for (const m of msgs) {
356
- if (m.info.role !== "assistant") continue
357
- if (m.info.agentID !== actorID) continue
358
- // Count every assistant LLM call in the slice, even ones without
359
- // billing-token data (errors / mid-stream interrupts). Token sums
360
- // skip the no-data rows; the call count includes them so downstream
361
- // consumers can distinguish "low cache hit" from "few calls".
362
- assistantCount += 1
363
- const t = m.info.tokens
364
- if (!t) continue
365
- totalInput += (t.input ?? 0) + (t.cache?.read ?? 0) + (t.cache?.write ?? 0)
366
- cacheRead += t.cache?.read ?? 0
367
- cacheWrite += t.cache?.write ?? 0
368
- }
369
- const billable = totalInput - cacheRead - cacheWrite
370
- const denom = cacheRead + Math.max(billable, 0)
371
- const hitRate = denom > 0 ? cacheRead / denom : 0
372
- return {
373
- total_input_tokens: totalInput,
374
- cache_read_tokens: cacheRead,
375
- cache_write_tokens: cacheWrite,
376
- cache_hit_rate: hitRate,
377
- num_llm_calls: assistantCount,
378
- }
379
- })
380
- }
381
-
382
- // ---------------------------------------------------------------------------
383
- // Service interface
384
- // ---------------------------------------------------------------------------
385
-
386
- export type TryStartCheckpointWriterInput = {
387
- sessionID: SessionID
388
- model: { providerID: string; modelID: string }
389
- promptOps: ActorPromptOps
390
- }
391
-
392
- /**
393
- * Outcome of a tryStartCheckpointWriter call:
394
- * - "started": no writer was running for this session, a fresh one was forked.
395
- * - "queued": a writer is already running. The new request is held in the
396
- * 1-slot pending queue and will fire once the current writer
397
- * settles. If a pending request already exists it's evicted —
398
- * newest wins because its range is a strict superset of the
399
- * older pending range, so the older one would just duplicate
400
- * work. (F40)
401
- * - "skipped": the request was rejected outright — empty session, system-
402
- * spawned subagent, or Actor service unavailable. No writer
403
- * will fire for this request now or later.
404
- */
405
- export type TryStartCheckpointWriterResult = "started" | "queued" | "skipped"
406
-
407
- export interface Interface {
408
- readonly tryStartCheckpointWriter: (
409
- input: TryStartCheckpointWriterInput,
410
- ) => Effect.Effect<TryStartCheckpointWriterResult>
411
-
412
- readonly waitForWriter: (sessionID: SessionID) => Effect.Effect<WriterOutcome | "no-writer">
413
-
414
- /**
415
- * Await all in-flight writers across sessions up to `timeoutMs`. Used by
416
- * the CLI shutdown path so headless `sleepy run` invocations don't exit
417
- * while a forked checkpoint writer is still waiting on its LLM round-trip.
418
- * Returns the count of writers that completed vs. still pending when the
419
- * timeout fired.
420
- */
421
- readonly drainWriters: (input?: { timeoutMs?: number }) => Effect.Effect<{
422
- drained: number
423
- timedOut: number
424
- }>
425
-
426
- readonly hasCheckpoint: (sessionID: SessionID) => Effect.Effect<boolean>
427
-
428
- /**
429
- * Returns true when the session has any memory artifacts:
430
- * either a populated `<data>/memory/sessions/<sid>/` directory, or
431
- * any tasks recorded in the task registry. Used by the per-user-message
432
- * recall reminder so it fires whenever there is anything to recall —
433
- * not only when classic v2 checkpoints exist.
434
- */
435
- readonly hasMemoryOrTasks: (sessionID: SessionID) => Effect.Effect<boolean>
436
-
437
- /** Returns the content of the latest checkpoint file, or undefined if none exists. */
438
- readonly loadLatest: (sessionID: SessionID) => Effect.Effect<string | undefined>
439
-
440
- /** Returns the content of the last N checkpoint files, ordered oldest to newest. */
441
- readonly loadCheckpoints: (sessionID: SessionID, count: number) => Effect.Effect<string[]>
442
-
443
- /** Returns a human-readable index overview for injection into rebuild context. */
444
- readonly renderIndex: (sessionID: SessionID) => Effect.Effect<string>
445
-
446
- /**
447
- * Returns the rebuild-time context that should be injected after trim.
448
- * Format:
449
- * <system-reminder>Verify-before-act note...</system-reminder>
450
- * ## Accumulated learnings (chronological)
451
- * ### From checkpoint #1 (<topic>)
452
- * <Learning body>
453
- * ...
454
- * ## Current snapshot (as of checkpoint #N)
455
- * <Snapshot body>
456
- *
457
- * Stale Snapshots from older checkpoints are intentionally dropped. Returns
458
- * an empty string if no checkpoints exist. When checkpoints exist but all
459
- * Learning sections are empty, emits "(no prior learnings)" placeholder;
460
- * when the latest checkpoint has no Snapshot section, emits
461
- * "(latest checkpoint has no Snapshot section)" placeholder — the full
462
- * structure is always produced so the verify-before-act reminder is
463
- * consistently visible.
464
- */
465
- readonly renderRebuildContext: (
466
- sessionID: SessionID,
467
- opts?: { lastMessageInfo?: LastMessageInfo; agentID?: string },
468
- ) => Effect.Effect<string>
469
-
470
- readonly lastBoundary: (sessionID: SessionID) => Effect.Effect<MessageID | undefined>
471
-
472
- readonly isWriterRunning: (sessionID: SessionID) => Effect.Effect<boolean>
473
-
474
- /**
475
- * Insert a synthetic checkpoint-boundary user message (boundary marker +
476
- * index overview + rebuild context + active-actors text) just after the
477
- * given boundary. Inserts nothing and returns false when rebuild context is
478
- * empty. Never deletes DB messages.
479
- */
480
- readonly insertRebuildBoundary: (input: {
481
- sessionID: SessionID
482
- boundary: MessageID
483
- lastMessageInfo?: LastMessageInfo
484
- agentID?: string
485
- agent: string
486
- model: { providerID: string; modelID: string }
487
- boundaryCreatedAt?: number
488
- }) => Effect.Effect<boolean>
489
- }
490
-
491
- export class Service extends Context.Service<Service, Interface>()("@opencode/SessionCheckpoint") {}
492
-
493
- // ---------------------------------------------------------------------------
494
- // Writer state per session
495
- // ---------------------------------------------------------------------------
496
-
497
- export type WriterOutcome = "success" | "failure"
498
-
499
- interface WriterState {
500
- // Holds the AgentOutcome Deferred returned by Actor.spawn so callers can
501
- // await writer settlement (waitForWriter / drainWriters). The public
502
- // WriterOutcome translation happens in waitForWriter.
503
- writing: Deferred.Deferred<AgentOutcome>
504
- // F40: 1-slot pending queue. When set, holds the input for a writer that
505
- // should fire as soon as `writing` settles. Newer requests evict older
506
- // pending values — the newest range is always a strict superset of the
507
- // older one, so older pending checkpoints would only duplicate work.
508
- pending?: TryStartCheckpointWriterInput
509
- }
510
-
511
- // ---------------------------------------------------------------------------
512
- // Layer implementation
513
- // ---------------------------------------------------------------------------
514
-
515
- export const layer: Layer.Layer<
516
- Service,
517
- never,
518
- Session.Service | Bus.Service | Config.Service | Memory.Service | TaskRegistry.Service | ActorRegistry.Service
519
- > = Layer.effect(
520
- Service,
521
- Effect.gen(function* () {
522
- const session = yield* Session.Service
523
- const config = yield* Config.Service
524
- const memory = yield* Memory.Service
525
- const taskRegistry = yield* TaskRegistry.Service
526
- const actorRegistry = yield* ActorRegistry.Service
527
- const bus = yield* Bus.Service
528
- const scope = yield* Scope.Scope
529
-
530
- // Plain Map in the layer closure — same approach as compaction.ts
531
- const writers = new Map<SessionID, WriterState>()
532
-
533
- const tryStartCheckpointWriter: (
534
- input: TryStartCheckpointWriterInput,
535
- ) => Effect.Effect<TryStartCheckpointWriterResult> = Effect.fn("SessionCheckpoint.tryStartCheckpointWriter")(function* (
536
- input: TryStartCheckpointWriterInput,
537
- ) {
538
- // F40: writer1 still running. Evict any prior pending and queue this
539
- // request — newest wins because its range is a strict superset of the
540
- // older pending range, so older pending checkpoints would only
541
- // duplicate the work.
542
- const existing = writers.get(input.sessionID)
543
- if (existing) {
544
- if (existing.pending) {
545
- log.info("writer pending evicted (newer range arrived)", { sessionID: input.sessionID })
546
- } else {
547
- log.info("writer already running, queueing", { sessionID: input.sessionID })
548
- }
549
- existing.pending = input
550
- return "queued" as const
551
- }
552
-
553
- // Defensive: skip if called for a system-spawned session. With Task 27's
554
- // writer-as-subagent migration this becomes mostly impossible, but the
555
- // guard stays so future paths that fold a system-spawn actor into the
556
- // main loop don't accidentally re-enter the writer.
557
- if (yield* actorRegistry.isSystemSpawned(input.sessionID, "main")) {
558
- log.info("tryStartCheckpointWriter skipping system-spawned session")
559
- return "skipped" as const
560
- }
561
-
562
- // Mirror parent runLoop's view (prompt.ts:2036-2040) so the writer's
563
- // ForkContext is byte-equal at the watermark moment. Reading the
564
- // unfiltered session stream would let computeBoundary land on a
565
- // subagent/prior-writer assistant turn and misalign the prefix cache.
566
- const sessionInfo = yield* session.get(input.sessionID)
567
- const msgs = yield* MessageV2.filterCompactedEffect(input.sessionID, {
568
- contextFrom: sessionInfo.contextFrom,
569
- contextWatermark: sessionInfo.contextWatermark,
570
- agentID: "main",
571
- })
572
- if (msgs.length === 0) {
573
- log.info("no messages, skipping checkpoint", { sessionID: input.sessionID })
574
- return "skipped" as const
575
- }
576
-
577
- // Compute boundary for last_checkpoint_message_id bookkeeping. Layer 6
578
- // (Task 16): role-aware adjustment to ensure tool_use/tool_result pairs
579
- // and same-message.id thinking blocks aren't split. OpenCode's ToolPart
580
- // carries both use (input) and result (output) on the SAME message, so
581
- // we project each ToolPart to both a tool_use and a tool_result block —
582
- // pairing is intrinsically satisfied today and the algorithm acts as a
583
- // no-op. Wiring is in place so future tool_result extraction
584
- // (separate user message) will walk the boundary correctly
585
- // without further changes here.
586
- const candidateID = computeBoundary(msgs)
587
- const candidateIdx = msgs.findIndex((m) => m.info.id === candidateID)
588
- const adjustedIdx = adjustBoundaryForApiInvariants(
589
- msgs.map((m) => ({
590
- role: m.info.role,
591
- id: m.info.id,
592
- content: m.parts.flatMap((p) =>
593
- p.type === "tool"
594
- ? [
595
- { type: "tool_use", id: p.callID },
596
- { type: "tool_result", tool_use_id: p.callID },
597
- ]
598
- : [],
599
- ),
600
- })),
601
- Math.max(candidateIdx, 0),
602
- )
603
- const endMessageID = msgs[adjustedIdx]?.info.id ?? candidateID
604
-
605
- // v5 paths: single checkpoint.md per session, single memory.md per
606
- // project (carries across sessions in the same repo), task narrative
607
- // under <sid>/tasks/<id>/. Resolve projectID once HERE — Instance.current
608
- // is ALS-bound and lost once the writer subagent fiber detaches.
609
- const projectID =
610
- (yield* Effect.try({
611
- try: () => Instance.current?.project?.id as ProjectID | undefined,
612
- catch: () => undefined,
613
- }).pipe(Effect.orElseSucceed(() => undefined))) ?? ProjectID.global
614
- const sessMemDir = metaDir(input.sessionID)
615
- const projectMemDir = path.join(Global.Path.data, "memory", "projects", projectID)
616
- const checkpointFile = checkpointPath(input.sessionID)
617
- const memoryFile = memoryPath(projectID)
618
- const taskMemDir = path.join(sessMemDir, "tasks")
619
- const notesFile = notesPath(input.sessionID)
620
-
621
- // Ensure dirs exist before writer fires
622
- yield* Effect.promise(() => fs.mkdir(sessMemDir, { recursive: true }))
623
- yield* Effect.promise(() => fs.mkdir(taskMemDir, { recursive: true }))
624
- yield* Effect.promise(() => fs.mkdir(projectMemDir, { recursive: true }))
625
-
626
- // Migrate legacy lowercase memory.md → MEMORY.md before templating/reading.
627
- yield* Effect.promise(() => migrateProjectMemory(projectID))
628
-
629
- // Bootstrap checkpoint.md, memory.md, and notes.md from templates if missing.
630
- // Self-contained helpers also mkdir parent so they're safe in isolation.
631
- yield* Effect.promise(() => ensureCheckpointTemplate(checkpointFile))
632
- yield* Effect.promise(() => ensureMemoryTemplate(memoryFile))
633
- yield* Effect.promise(() => ensureNotesTemplate(notesFile))
634
-
635
- // v5: single-file checkpoint, check if prior content exists
636
- const checkpointExists = yield* Effect.promise(() => Bun.file(checkpointFile).exists())
637
- const memoryExists = yield* Effect.promise(() => Bun.file(memoryFile).exists())
638
- const rangeDesc = checkpointExists
639
- ? [
640
- `Previous checkpoint: ${checkpointFile}`,
641
- memoryExists ? `Previous memory: ${memoryFile}` : "",
642
- "Read BOTH the prior checkpoint (to dedupe Discovered/Dead-end titles AND to carry forward Live Resources, Execution-context frames, and Session-metadata fields that are still alive) AND the prior memory (project memory) before writing yours.",
643
- ]
644
- .filter((s) => s.length > 0)
645
- .join("\n")
646
- : "This is the first checkpoint of this session. No prior checkpoint exists; MEMORY.md and the task narrative directory likely don't exist yet either."
647
-
648
- const progressDiff = yield* Effect.promise(() => buildProgressDiff(input.sessionID))
649
- const promptText = composeWriterPrompt({ checkpointFile, memoryFile, taskMemDir, notesFile, rangeDesc, progressDiff })
650
-
651
- // v6: spawn writer as subagent — shared sessionID, automatic
652
- // ActorRegistry registration, automatic tool whitelist enforcement
653
- // via permission system. Replaces the legacy session.create + manual
654
- // forkDetach + WriterState tracking that lived here pre-Task-27.
655
- //
656
- // Resolved via spawnRef rather than `yield* Actor.Service` to break the
657
- // (Actor → SessionPrompt → SessionCheckpoint → Actor) layer cycle.
658
- const actor = spawnRef.current
659
- if (!actor) {
660
- log.warn("tryStartCheckpointWriter skipping — Actor service unavailable", { sessionID: input.sessionID })
661
- return "skipped" as const
662
- }
663
-
664
- // Axis B: branch forkContext shape on config.checkpoint.fork.
665
- // - true → preserve existing prefix-cache parent-fork behavior
666
- // (parent agent's system + tools, full slice up to watermark).
667
- // - false → cold-start: writer's own system + tools, delta slice since
668
- // last_checkpoint_message_id (aligned past tool_use/tool_result).
669
- // See spec 2026-06-09-checkpoint-writer-child-session-and-no-fork-fallback-design.md §3.
670
- //
671
- // Default-behavior change at this PR: previously the writer always forked
672
- // the parent's full prefix (effectively fork: true). The default is now
673
- // false (no-fork delta-only). Users on cache-breakpoint providers
674
- // (Anthropic) who want to retain the prefix-cache benefit must set
675
- // `checkpoint.fork: true` in their config. See the spec at
676
- // docs/superpowers/specs/2026-06-09-checkpoint-writer-child-session-and-no-fork-fallback-design.md §4.5.
677
- const cfg = yield* config.get()
678
- const forkMode = cfg.checkpoint?.fork ?? false
679
-
680
- const parentRow = yield* Effect.sync(() =>
681
- Database.use((d) =>
682
- d.select({ last: SessionTable.last_checkpoint_message_id })
683
- .from(SessionTable)
684
- .where(eq(SessionTable.id, input.sessionID))
685
- .get(),
686
- ),
687
- ).pipe(Effect.catch(() => Effect.succeed(undefined as { last: MessageID | null } | undefined)))
688
- const lastCheckpointMessageID = parentRow?.last ?? undefined
689
-
690
- // Hoisted watermark + delta computation: must run BEFORE session.create
691
- // so an empty-delta fork:false call short-circuits to "skipped" without
692
- // creating a child session or invoking actor.spawn. Pre-fix, the
693
- // empty-delta path fell through to spawn → runLoop's
694
- // `isForkAgent && !forkCtx → break` → settle watcher resolved success →
695
- // parent's last_checkpoint_message_id advanced silently (stale checkpoint).
696
- const watermarkIdx = msgs.findIndex((m) => m.info.id === endMessageID)
697
- if (watermarkIdx < 0) {
698
- log.warn("tryStartCheckpointWriter: watermark message not found, skipping", {
699
- sessionID: input.sessionID,
700
- endMessageID,
701
- })
702
- return "skipped" as const
703
- }
704
-
705
- // For fork:false only: precompute the aligned delta and bail if empty.
706
- // fork:true uses msgs.slice(0, watermarkIdx + 1) which is never empty
707
- // given msgs.length > 0 and watermarkIdx >= 0.
708
- const lastIdx = lastCheckpointMessageID
709
- ? msgs.findIndex((m) => m.info.id === lastCheckpointMessageID)
710
- : -1
711
- const rawDeltaStart = lastIdx >= 0 ? lastIdx + 1 : 0
712
- const alignedStart = alignToNonToolResultUser(
713
- msgs.map((m) => ({ info: { role: m.info.role }, parts: m.parts })),
714
- rawDeltaStart,
715
- )
716
- const delta = forkMode ? [] : msgs.slice(alignedStart, watermarkIdx + 1)
717
- if (!forkMode && delta.length === 0) {
718
- // Empty delta under fork:false signals either (a) a degenerate
719
- // session, or (b) a bug elsewhere advanced last_checkpoint_message_id
720
- // past the watermark. Either way, spawning a writer would be a
721
- // silent no-op that would still advance the watermark on settle —
722
- // skip visibly so it's observable in logs.
723
- log.warn("tryStartCheckpointWriter: empty delta under fork:false, skipping", {
724
- sessionID: input.sessionID,
725
- endMessageID,
726
- lastCheckpointMessageID,
727
- })
728
- return "skipped" as const
729
- }
730
-
731
- // Capture parent's view at the watermark for prefix-cache alignment.
732
- // See docs/superpowers/specs/2026-05-26-fork-agent-prefix-cache-design.md
733
- //
734
- // prefixCaptureRef is populated by SessionPrompt.layer to break the
735
- // (ToolRegistry → SessionCheckpoint → ToolRegistry) layer cycle.
736
- const buildPrefix = prefixCaptureRef.current
737
- if (!buildPrefix) {
738
- log.warn("tryStartCheckpointWriter: prefixCaptureRef not set, spawning without forkContext", {
739
- sessionID: input.sessionID,
740
- })
741
- }
742
- const forkCtx: ForkContext | undefined = yield* (buildPrefix
743
- ? Effect.gen(function* () {
744
- const watermarkMsg = msgs[watermarkIdx]
745
- const parentAgentName = (watermarkMsg.info as { agent?: string }).agent
746
- // NOTE: parentAgentName guard is scoped to the forkMode:true branch only —
747
- // fork:false is agent-name-independent (always passes "checkpoint-writer"
748
- // to buildPrefix), so a missing parent agent field must not gate it.
749
-
750
- if (forkMode) {
751
- if (!parentAgentName) {
752
- log.warn(
753
- "tryStartCheckpointWriter: watermark has no agent, fork:true requires parent agent — falling back to no forkContext",
754
- { sessionID: input.sessionID, endMessageID },
755
- )
756
- return undefined as ForkContext | undefined
757
- }
758
- // fork:true — preserve existing prefix-cache parent-fork behavior.
759
- // Build system + tools + inheritedMessages snapshot via capture ref
760
- // using the parent agent's identity and the full slice up to watermark.
761
- // The closure inside SessionPrompt.layer resolves Agent.Info and Provider.Model.
762
- const msgsAtWatermark = msgs.slice(0, watermarkIdx + 1)
763
- const prefix = yield* buildPrefix({
764
- sessionID: input.sessionID,
765
- agentName: parentAgentName,
766
- providerID: input.model.providerID,
767
- modelID: input.model.modelID,
768
- msgs: msgsAtWatermark,
769
- })
770
- return {
771
- system: prefix.system,
772
- tools: prefix.tools,
773
- inheritedMessages: prefix.inheritedMessages,
774
- parentPermission: prefix.parentPermission,
775
- watermarkMsgID: endMessageID as MessageID,
776
- model: {
777
- providerID: input.model.providerID as ProviderID,
778
- modelID: input.model.modelID as ModelID,
779
- },
780
- } satisfies ForkContext
781
- }
782
-
783
- // fork:false — cold-start: use the writer's own system + tools and
784
- // a delta slice since the last checkpoint. capture() resolves the
785
- // checkpoint-writer agent's own definition when agentName is
786
- // "checkpoint-writer", so a single call returns:
787
- // - system + tools = writer's (because agentName === "checkpoint-writer"),
788
- // - inheritedMessages = the delta we pass in, converted to ModelMessage[],
789
- // - parentPermission = writer's own permission (used for tool-availability filter).
790
- // Earlier draft considered two buildPrefix calls (one with msgs:[] for
791
- // system+tools, one with msgs:delta for messages); rejected because
792
- // buildLLMRequestPrefix `Effect.die`s if msgs has no user message.
793
- // Delta is precomputed (and the empty-delta case is short-circuited)
794
- // above, before session.create.
795
- const writerPrefix = yield* buildPrefix({
796
- sessionID: input.sessionID,
797
- agentName: "checkpoint-writer",
798
- providerID: input.model.providerID,
799
- modelID: input.model.modelID,
800
- msgs: delta,
801
- })
802
-
803
- return {
804
- system: writerPrefix.system,
805
- tools: writerPrefix.tools,
806
- inheritedMessages: writerPrefix.inheritedMessages,
807
- parentPermission: writerPrefix.parentPermission,
808
- watermarkMsgID: endMessageID as MessageID,
809
- model: {
810
- providerID: input.model.providerID as ProviderID,
811
- modelID: input.model.modelID as ModelID,
812
- },
813
- } satisfies ForkContext
814
- })
815
- : Effect.succeed(undefined as ForkContext | undefined))
816
-
817
- // Axis A: writer always runs in a fresh child session. This isolates the
818
- // writer's messages and actor registration from the parent so:
819
- // - parent's message table sees zero new rows,
820
- // - parent's `sync.data.actor[parent]` does not include the writer,
821
- // - Ctrl+X subagent cycle / SubagentFooter / DialogSubagent / etc. are
822
- // all naturally clean (they all key on sessionID).
823
- // The writer's checkpoint.md / memory.md / progress paths are absolute and
824
- // computed from input.sessionID (parent) above, so file writes still target
825
- // the parent's artifacts. Settle watcher below also targets parent.
826
- // See spec 2026-06-09-checkpoint-writer-child-session-and-no-fork-fallback-design.md §2.
827
- const writerChildSession = yield* session.create({
828
- parentID: input.sessionID,
829
- title: `checkpoint-writer: ${rangeDesc}`,
830
- })
831
-
832
- // Estimate delta tokens for observability. forkCtx.inheritedMessages is
833
- // ModelMessage[]; an exact count requires the tokenizer, but a rough
834
- // length heuristic is sufficient for the log line.
835
- const deltaApproxBytes = JSON.stringify(forkCtx?.inheritedMessages ?? []).length
836
- log.info("tryStartCheckpointWriter spawning", {
837
- sessionID: input.sessionID,
838
- childSessionID: writerChildSession.id,
839
- mode: forkMode ? "fork" : "no-fork",
840
- deltaApproxBytes,
841
- rangeDesc,
842
- })
843
-
844
- const result = yield* actor.spawn({
845
- mode: "subagent",
846
- sessionID: writerChildSession.id,
847
- // Axis A: writer runs under child session, but its checkpoint.md /
848
- // memory.md / progress paths AND CheckpointContext entries are keyed
849
- // on the PARENT. The splitover plugin reads these via actor.preStop
850
- // and must see parentSessionID to re-derive the right paths — without
851
- // it, checkpointPath(child) returns an empty file and the plugin
852
- // emits a false topic-missing reflection that loops the writer up to
853
- // MAX_PRE_REACT.
854
- parentSessionID: input.sessionID,
855
- agentType: "checkpoint-writer",
856
- description: `checkpoint writer for session ${input.sessionID} covering ${rangeDesc}`,
857
- task: promptText,
858
- context: "full",
859
- tools: ["read", "write", "edit", "apply_patch", "glob", "grep", "task"],
860
- model: {
861
- providerID: input.model.providerID as ProviderID,
862
- modelID: input.model.modelID as ModelID,
863
- },
864
- background: true,
865
- forkContext: forkCtx,
866
- })
867
-
868
- const actorID = result.actorID
869
-
870
- // Capture priorTitles (from checkpoint.md as it stood at the watermark)
871
- // and register the per-actor context entry BEFORE the writer's first
872
- // turn so the splitover plugin's preStop hook can read it. The set
873
- // runs in microseconds; the writer's first LLM round-trip takes
874
- // seconds — no race in practice. See spec §6.1.
875
- const priorTitles = yield* Effect.promise(() => loadPriorDiscoveredTitles(input.sessionID))
876
- CheckpointContext.set(input.sessionID, actorID, {
877
- priorTitles,
878
- expectedRevisions: [],
879
- })
880
-
881
- writers.set(input.sessionID, { writing: result.outcome })
882
-
883
- // Bookkeeping: the parent's last_checkpoint_message_id advances when the
884
- // writer settles. Fork into the layer's scope so the watcher survives
885
- // tryStartCheckpointWriter returning (background: true semantics) but is still tied
886
- // to the layer's lifetime — no orphan fiber on shutdown.
887
- yield* Effect.gen(function* () {
888
- const outcome = yield* Deferred.await(result.outcome)
889
- yield* Effect.sync(() =>
890
- Database.use((d) =>
891
- d.update(SessionTable)
892
- .set({ last_checkpoint_message_id: endMessageID as MessageID })
893
- .where(eq(SessionTable.id, input.sessionID))
894
- .run(),
895
- ),
896
- )
897
-
898
- // F40: capture pending before deleting the slot so a queued writer
899
- // (held while writer1 was running) can fire as a fresh writer.
900
- const pending = writers.get(input.sessionID)?.pending
901
- writers.delete(input.sessionID)
902
-
903
- // F44: aggregate writer slice tokens and emit cache-perf metric so
904
- // prefix-cache reuse is empirically observable. Degrades to zeros if
905
- // aggregation fails (e.g. session messages unavailable post-shutdown).
906
- const stats = yield* aggregateWriterCacheMetrics(session, input.sessionID, result.actorID).pipe(
907
- Effect.catch(() =>
908
- Effect.succeed({
909
- total_input_tokens: 0,
910
- cache_read_tokens: 0,
911
- cache_write_tokens: 0,
912
- cache_hit_rate: 0,
913
- num_llm_calls: 0,
914
- }),
915
- ),
916
- )
917
- yield* bus
918
- .publish(WriterCachePerf, {
919
- sessionID: input.sessionID,
920
- writerActorID: result.actorID,
921
- status: outcome.status === "success" ? ("completed" as const) : ("failed" as const),
922
- ...stats,
923
- })
924
- .pipe(Effect.ignore)
925
-
926
- // F40: drain pending. If a queued request exists, fire a fresh writer
927
- // for it. Errors are swallowed — the queued writer's failure should
928
- // not interrupt the original writer's settlement watcher.
929
- if (pending) {
930
- log.info("draining pending writer", { sessionID: input.sessionID })
931
- yield* tryStartCheckpointWriter(pending).pipe(Effect.ignore)
932
- }
933
- }).pipe(
934
- Effect.ensuring(
935
- Effect.sync(() => CheckpointContext.remove(input.sessionID, actorID)),
936
- ),
937
- Effect.forkIn(scope),
938
- )
939
-
940
- return "started" as const
941
- })
942
-
943
- const waitForWriter = Effect.fn("SessionCheckpoint.waitForWriter")(function* (sessionID: SessionID) {
944
- const state = writers.get(sessionID)
945
- if (!state) return "no-writer" as const
946
-
947
- // v2 writers manage 3 file types and frequently take 60-180s; pad to
948
- // 5min so a long-but-honest writer is not mistaken for a failure by
949
- // the prune retry watcher. AgentOutcome → WriterOutcome translation:
950
- // success → "success", failure / cancelled → "failure".
951
- const outcome = yield* Deferred.await(state.writing).pipe(
952
- Effect.timeout(300_000),
953
- Effect.catch(() => Effect.succeed<AgentOutcome>({ status: "failure", error: "timeout" })),
954
- )
955
- return outcome.status === "success" ? ("success" as const) : ("failure" as const)
956
- })
957
-
958
- const drainWriters = Effect.fn("SessionCheckpoint.drainWriters")(function* (input?: { timeoutMs?: number }) {
959
- const timeoutMs = input?.timeoutMs ?? 120_000
960
- const pending = [...writers.values()]
961
- if (pending.length === 0) return { drained: 0, timedOut: 0 }
962
- log.info("draining checkpoint writers before shutdown", {
963
- count: pending.length,
964
- timeoutMs,
965
- })
966
-
967
- // Deferred.await ignores fiber interruption during shutdown because
968
- // it resolves via Deferred.succeed in the detached writer. We only
969
- // need a collective upper bound so a stuck writer doesn't block exit.
970
- yield* Effect.all(
971
- pending.map((state) => Deferred.await(state.writing)),
972
- { concurrency: "unbounded" },
973
- ).pipe(
974
- Effect.timeout(timeoutMs),
975
- Effect.catch(() => Effect.succeed(undefined)),
976
- )
977
-
978
- // Writers delete themselves from the map on success/failure, so anything
979
- // still present after the timeout is a writer that didn't settle in time.
980
- const timedOut = writers.size
981
- const drained = pending.length - timedOut
982
- if (timedOut > 0) log.warn("drain timed out, writers still pending", { drained, timedOut })
983
- else log.info("drain complete", { drained })
984
- return { drained, timedOut }
985
- })
986
-
987
- const hasCheckpoint = Effect.fn("SessionCheckpoint.hasCheckpoint")(function* (sessionID: SessionID) {
988
- return yield* Effect.promise(() => Bun.file(checkpointPath(sessionID)).exists())
989
- })
990
-
991
- const hasMemoryOrTasks = Effect.fn("SessionCheckpoint.hasMemoryOrTasks")(function* (sessionID: SessionID) {
992
- const memoryRoot = yield* memory.root()
993
- const sessMemDir = path.join(memoryRoot, "sessions", sessionID)
994
- const memEntries = yield* Effect.promise(() =>
995
- fs.readdir(sessMemDir).catch(() => [] as string[]),
996
- )
997
- if (memEntries.length > 0) return true
998
- const tasks = yield* taskRegistry.list({ session_id: sessionID, include_terminal: true })
999
- return tasks.length > 0
1000
- })
1001
-
1002
- const loadLatest = Effect.fn("SessionCheckpoint.loadLatest")(function* (sessionID: SessionID) {
1003
- const content = yield* Effect.promise(() =>
1004
- Bun.file(checkpointPath(sessionID)).text().catch(() => ""),
1005
- )
1006
- return content || undefined
1007
- })
1008
-
1009
- const loadCheckpoints = Effect.fn("SessionCheckpoint.loadCheckpoints")(function* (
1010
- sessionID: SessionID,
1011
- _count: number,
1012
- ) {
1013
- const content = yield* Effect.promise(() =>
1014
- Bun.file(checkpointPath(sessionID)).text().catch(() => ""),
1015
- )
1016
- return content ? [content] : []
1017
- })
1018
-
1019
- const renderIndex = Effect.fn("SessionCheckpoint.renderIndex")(function* (sessionID: SessionID) {
1020
- const snapFile = checkpointPath(sessionID)
1021
- const exists = yield* Effect.promise(() => Bun.file(snapFile).exists())
1022
- if (!exists) return "No checkpoints yet for this session."
1023
-
1024
- const content = yield* Effect.promise(() => Bun.file(snapFile).text().catch(() => ""))
1025
- const topicMatch = content.match(/^Topic:\s*(.+)$/m)
1026
- const topic = topicMatch ? topicMatch[1].trim() : "(unknown)"
1027
-
1028
- const dir = metaDir(sessionID)
1029
- const lines: string[] = []
1030
- lines.push("## Checkpoint")
1031
- lines.push("")
1032
- lines.push(`Directory: ${dir}/`)
1033
- lines.push("")
1034
- lines.push(`Current checkpoint (${topic}): checkpoint.md [shown below]`)
1035
- lines.push("")
1036
- lines.push(`Use read("${snapFile}") to access the full checkpoint.`)
1037
-
1038
- return lines.join("\n")
1039
- })
1040
-
1041
- const renderRebuildContext = Effect.fn("SessionCheckpoint.renderRebuildContext")(function* (
1042
- sessionID: SessionID,
1043
- opts?: { lastMessageInfo?: LastMessageInfo; agentID?: string },
1044
- ) {
1045
- // renderRebuildContext is for the user-facing main agent's context rebuild.
1046
- // Subagent-mode actors (system-spawned writers, model-spawned subagents)
1047
- // share the parent's session but don't have their own checkpoint state to
1048
- // render — return empty so the rebuild path is a no-op for them.
1049
- // Note: agentID === "main" must pass through. After F49+F50 the main
1050
- // agent's lastUser.agentID is "main" (DB row→info reconstruction in
1051
- // message-v2.ts populates info.agentID from agent_id column), and the
1052
- // runLoop calls this with that value. Treating "main" as subagent here
1053
- // would skip rebuild → fall through to F39 compaction → context loss.
1054
- if (opts?.agentID && opts.agentID !== "main") return ""
1055
-
1056
- const inFlight = writers.get(sessionID)
1057
- if (inFlight) {
1058
- log.info("rebuild waiting for in-flight writer", { sessionID })
1059
- yield* Effect.race(
1060
- Deferred.await(inFlight.writing).pipe(Effect.as("done" as const)),
1061
- Effect.sleep("60 seconds").pipe(
1062
- Effect.tap(() => Effect.sync(() => log.warn("writer wait timeout — using on-disk checkpoint", { sessionID }))),
1063
- Effect.as("timeout" as const),
1064
- ),
1065
- ).pipe(Effect.catch(() => Effect.succeed("error" as const)))
1066
- }
1067
-
1068
- const cfg = yield* config.get()
1069
- const caps = cfg.checkpoint?.push_caps ?? {}
1070
- const memoryRoot = yield* memory.root()
1071
-
1072
- const sessMemDir = path.join(memoryRoot, "sessions", sessionID)
1073
-
1074
- // Resolve current project ID once. Used by Section 7 (project memory
1075
- // read) and Section 8 (FTS scope filter). ALS-bound — must be resolved
1076
- // before any deferred work.
1077
- const currentProjectID = yield* Effect.try({
1078
- try: () => Instance.current?.project?.id as ProjectID | undefined,
1079
- catch: () => undefined as ProjectID | undefined,
1080
- }).pipe(Effect.catch(() => Effect.succeed<ProjectID | undefined>(undefined)))
1081
- const projectID = currentProjectID ?? ProjectID.global
1082
-
1083
- // Section data: tasks (SQL), session checkpoint (file), project memory (file).
1084
- const tasks = yield* taskRegistry.list({ session_id: sessionID, include_terminal: true })
1085
-
1086
- const checkpointResult = yield* Effect.promise(() =>
1087
- readBudgetedSectionAware(checkpointPath(sessionID), caps.checkpoint ?? 11_000),
1088
- )
1089
- const checkpointText = checkpointResult?.text ?? ""
1090
-
1091
- yield* Effect.promise(() => migrateProjectMemory(projectID))
1092
- const memoryResult = yield* Effect.promise(() =>
1093
- readBudgetedSectionAware(memoryPath(projectID), caps.memory ?? 10_000),
1094
- )
1095
- const memoryText = memoryResult?.text ?? ""
1096
-
1097
- const notesResult = yield* Effect.promise(() =>
1098
- readBudgeted(notesPath(sessionID), caps.notes ?? 6000),
1099
- )
1100
- const notesText = notesResult?.text ?? ""
1101
-
1102
- const globalResult = yield* Effect.promise(() =>
1103
- readBudgetedSectionAware(globalMemoryPath(), caps.global ?? 6000),
1104
- )
1105
- const globalText = globalResult?.text ?? ""
1106
-
1107
- const actors = yield* actorRegistry.listActive()
1108
-
1109
- // Pull recent user messages (verbatim, FIFO-bounded). Done before the
1110
- // early-bail check so a session whose only signal is "user typed N
1111
- // prompts" still emits the section.
1112
- const recentUserCap = caps.recent_user ?? 16_000
1113
- const recentUserPerMsg = caps.recent_user_per_msg ?? 2_000
1114
- const recentUserEntries: string[] = []
1115
- if (recentUserCap > 0) {
1116
- // Fixed page ceiling sized comfortably above the token budget: at the
1117
- // 2K per-msg cap, 200 msgs ≈ 400K tokens, far past any recent_user cap,
1118
- // so the token loop below — not this limit — is what bounds the section.
1119
- // The only way 200 msgs could underflow the budget is hundreds of sub-
1120
- // 80-token prompts ("ok"/"continue"), which carry no anchors worth
1121
- // paging deeper for. Keeping a constant avoids a magic tokens/msg
1122
- // heuristic and a larger fetch+hydrate on every checkpoint.
1123
- // Exclude rebuild/compaction boundary messages: insertRebuildBoundary
1124
- // writes a role:"user" row carrying a checkpoint part + synthetic text
1125
- // holding the *previous* rebuild context. Re-ingesting it would fold
1126
- // each prior rebuild back in recursively (fractal bloat). userMsgText
1127
- // also drops synthetic text parts as a second guard.
1128
- const userMsgs = MessageV2.page({ sessionID, agentID: "main", limit: 200 }).items.filter(
1129
- (m) =>
1130
- m.info.role === "user" &&
1131
- !m.parts.some((p) => p.type === "tool" || p.type === "checkpoint" || p.type === "compaction"),
1132
- )
1133
- // Iterate most-recent backward so FIFO drops oldest when total cap hits.
1134
- let remaining = recentUserCap
1135
- for (let i = userMsgs.length - 1; i >= 0; i--) {
1136
- const rawText = userMsgText(userMsgs[i].parts)
1137
- if (!rawText.trim()) continue
1138
- const entry = truncateVerbatimUserMsg(rawText, recentUserPerMsg, userMsgs[i].info.id)
1139
- const cost = Token.estimate(entry)
1140
- if (remaining - cost < 0) break
1141
- recentUserEntries.unshift(entry)
1142
- remaining -= cost
1143
- }
1144
- }
1145
-
1146
- // Bail early if absolutely nothing to push: no tasks, no memory content, no live actors,
1147
- // no user messages.
1148
- if (
1149
- tasks.length === 0 &&
1150
- !checkpointText.trim() &&
1151
- !memoryText.trim() &&
1152
- !globalText.trim() &&
1153
- actors.length === 0 &&
1154
- recentUserEntries.length === 0
1155
- ) {
1156
- return ""
1157
- }
1158
-
1159
- const lines: string[] = []
1160
-
1161
- // F17: Explicit "already loaded" header. Anchors the active recall
1162
- // protocol's "look for this header" instruction in buildMemoryInstructions.
1163
- lines.push(
1164
- "The following blocks are auto-loaded from your session memory. They are already in your context — do not Read them as whole files. Use Grep for specific facts instead.",
1165
- )
1166
- lines.push("")
1167
-
1168
- // Section 3: tasks ledger (hierarchical with subtasks).
1169
- lines.push("## Tasks ledger")
1170
- if (tasks.length === 0) {
1171
- lines.push("(none)")
1172
- } else {
1173
- const topLevel = tasks.filter((t) => !t.parent_task_id)
1174
- const byParent = new Map<string, typeof tasks>()
1175
- for (const t of tasks) {
1176
- if (!t.parent_task_id) continue
1177
- const bucket = byParent.get(t.parent_task_id) ?? []
1178
- bucket.push(t)
1179
- byParent.set(t.parent_task_id, bucket)
1180
- }
1181
- const statusIcon = (s: string) =>
1182
- ({ open: "🔵", in_progress: "🔄", blocked: "🟡", done: "✅", abandoned: "❌" })[s] ?? s
1183
- const ledgerLines: string[] = []
1184
- for (const t of topLevel) {
1185
- ledgerLines.push(`- ${t.id} ${t.status} — ${t.summary}`)
1186
- const subs = byParent.get(t.id) ?? []
1187
- if (subs.length === 0) continue
1188
- const sublist = subs
1189
- .map((s) => `${statusIcon(s.status)}${s.id}`)
1190
- .join(" / ")
1191
- ledgerLines.push(` Subtasks: ${sublist}`)
1192
- }
1193
- lines.push(truncate(ledgerLines.join("\n"), caps.tasks_ledger ?? 2000))
1194
- }
1195
- lines.push("")
1196
-
1197
- // Section 5: session checkpoint (full body, capped).
1198
- if (checkpointText.trim()) {
1199
- lines.push("## Session checkpoint")
1200
- lines.push(checkpointText.trim())
1201
- lines.push("")
1202
- }
1203
-
1204
- // Section 6: active actors ledger (one line per running actor).
1205
- if (actors.length > 0) {
1206
- lines.push("## Active actors")
1207
- let actorBudget = caps.actor_ledger ?? 500
1208
- for (const a of actors) {
1209
- const line = `- ${a.actorID} — ${a.status}, "${a.description}" (agent=${a.agent})`
1210
- const cost = Token.estimate(line)
1211
- if (actorBudget - cost < 0) break
1212
- lines.push(line)
1213
- actorBudget -= cost
1214
- }
1215
- lines.push("")
1216
- }
1217
-
1218
- // Section 6.5: recent user input (verbatim, FIFO, budget-bounded).
1219
- // Preserves original user prose from the live DB — writer summaries
1220
- // paraphrase user commands, losing anchors like exact flags or pasted
1221
- // content. (entries computed earlier so the early-bail guard sees them.)
1222
- if (recentUserEntries.length > 0) {
1223
- lines.push("## Recent user input (verbatim)")
1224
- lines.push(...recentUserEntries)
1225
- lines.push("")
1226
- }
1227
-
1228
- // Section 7: project memory (full body, capped).
1229
- if (memoryText.trim()) {
1230
- lines.push("## Project memory")
1231
- lines.push(memoryText.trim())
1232
- lines.push("")
1233
- }
1234
-
1235
- // Section 7.4: global memory (full body, capped). User-level cross-project
1236
- // preferences. Placed after project memory (more actionable) and before
1237
- // session notes (more volatile).
1238
- if (globalText.trim()) {
1239
- lines.push("## Global memory")
1240
- lines.push(globalText.trim())
1241
- lines.push("")
1242
- }
1243
-
1244
- // F14 Section 7.5: session notes (full body, capped). Skip if empty.
1245
- if (notesText.trim()) {
1246
- lines.push("## Session notes")
1247
- lines.push(notesText.trim())
1248
- lines.push("")
1249
- }
1250
-
1251
- // Section 8: memory keys index (paths only, omit already-pushed).
1252
- // SQL-scoped to the current session/project + global so other
1253
- // sessions' files are not leaked. Falls back to skipping the projects
1254
- // scope when the current project is the global/non-git fallback.
1255
- // Reconcile first so files written off-tool (e.g. by the checkpoint
1256
- // writer subagent) are visible in the FTS index here.
1257
- yield* memory.reconcile().pipe(Effect.ignore)
1258
- const pushedPaths = new Set(
1259
- [
1260
- memoryPath(projectID),
1261
- checkpointPath(sessionID),
1262
- globalMemoryPath(),
1263
- ].filter((p) => p.length > 0),
1264
- )
1265
-
1266
- const scopeFilter =
1267
- currentProjectID && currentProjectID !== ProjectID.global
1268
- ? or(
1269
- eq(MemoryFtsTable.scope, "global"),
1270
- and(eq(MemoryFtsTable.scope, "sessions"), eq(MemoryFtsTable.scope_id, sessionID as string)),
1271
- and(eq(MemoryFtsTable.scope, "projects"), eq(MemoryFtsTable.scope_id, currentProjectID)),
1272
- )
1273
- : or(
1274
- eq(MemoryFtsTable.scope, "global"),
1275
- and(eq(MemoryFtsTable.scope, "sessions"), eq(MemoryFtsTable.scope_id, sessionID as string)),
1276
- )
1277
- const scopedPaths = yield* Effect.sync(() =>
1278
- Database.use((db) =>
1279
- db.select({ path: MemoryFtsTable.path }).from(MemoryFtsTable).where(scopeFilter).all(),
1280
- ),
1281
- )
1282
- const keyEntries = scopedPaths
1283
- .map((r) => r.path)
1284
- .filter((p) => !pushedPaths.has(p) && !p.includes(`${path.sep}checkpoint${path.sep}learning-`))
1285
- .map((p) => p.replace(memoryRoot + path.sep, ""))
1286
- if (keyEntries.length > 0) {
1287
- lines.push("## Memory keys index")
1288
- let kBudget = caps.memory_titles ?? 500
1289
- for (const entry of keyEntries) {
1290
- const cost = Token.estimate(entry)
1291
- if (kBudget - cost < 0) break
1292
- lines.push(`- ${entry}`)
1293
- kBudget -= cost
1294
- }
1295
- lines.push("")
1296
- }
1297
-
1298
- // Section 10: explicit seam framing for LLM continuity post-rebuild.
1299
- // Compaction-summary pattern: tells the model
1300
- // that preserved messages below are real history, not pseudo-content,
1301
- // so it resumes mid-loop instead of asking "what would you like me
1302
- // to do".
1303
- lines.push("")
1304
- lines.push(
1305
- "This session is being continued from a previous conversation that hit a checkpoint. The session checkpoint and project memory above cover the earlier portion of the conversation.",
1306
- )
1307
- lines.push("")
1308
- lines.push(
1309
- "Recent messages are preserved verbatim below — the assistant turn (and any tool results) you'll see is real history, not pseudo-content. Continue your task by responding to the most recent state.",
1310
- )
1311
- lines.push("")
1312
- lines.push(
1313
- "Resume directly. Do not acknowledge this memory dump, do not recap, do not preface with \"I'll continue\" or similar. Pick up the last task as if the break never happened.",
1314
- )
1315
-
1316
- // Section 11: tail-aware system reminder. Picks the appropriate nudge
1317
- // based on how the preserved tail ends: tool-calls → continue loop,
1318
- // stop → check task spec before stopping again, tool → process results,
1319
- // user → no addendum needed.
1320
- const info = opts?.lastMessageInfo
1321
- if (info) {
1322
- const reminder = (() => {
1323
- switch (info.role) {
1324
- case "assistant":
1325
- if (info.finish === "tool-calls") return autonomousLoopReminder()
1326
- return stopReminder(undefined)
1327
- case "tool":
1328
- return toolResultContinueReminder()
1329
- case "user":
1330
- return ""
1331
- }
1332
- })()
1333
- if (reminder) {
1334
- lines.push("")
1335
- lines.push(reminder)
1336
- }
1337
- }
1338
-
1339
- return lines.join("\n")
1340
- })
1341
-
1342
- const lastBoundary = Effect.fn("SessionCheckpoint.lastBoundary")(function* (sessionID: SessionID) {
1343
- const row = yield* Effect.sync(() =>
1344
- Database.use((db) =>
1345
- db.select({ last_checkpoint_message_id: SessionTable.last_checkpoint_message_id })
1346
- .from(SessionTable)
1347
- .where(eq(SessionTable.id, sessionID))
1348
- .get(),
1349
- ),
1350
- )
1351
- return row?.last_checkpoint_message_id as MessageID | undefined
1352
- })
1353
-
1354
- const isWriterRunning = Effect.fn("SessionCheckpoint.isWriterRunning")(function* (sessionID: SessionID) {
1355
- return writers.has(sessionID)
1356
- })
1357
-
1358
- const insertRebuildBoundary = Effect.fn("SessionCheckpoint.insertRebuildBoundary")(function* (input: {
1359
- sessionID: SessionID
1360
- boundary: MessageID
1361
- lastMessageInfo?: LastMessageInfo
1362
- agentID?: string
1363
- agent: string
1364
- model: { providerID: string; modelID: string }
1365
- boundaryCreatedAt?: number
1366
- }) {
1367
- const rebuildContext = yield* renderRebuildContext(input.sessionID, {
1368
- lastMessageInfo: input.lastMessageInfo,
1369
- agentID: input.agentID,
1370
- }).pipe(Effect.catch(() => Effect.succeed("")))
1371
- if (!rebuildContext) return false
1372
-
1373
- const indexText = yield* renderIndex(input.sessionID).pipe(Effect.catch(() => Effect.succeed("")))
1374
-
1375
- const syntheticTime = (input.boundaryCreatedAt ?? Date.now()) + 1
1376
- const msg = yield* session.updateMessage({
1377
- id: MessageID.ascending(),
1378
- role: "user" as const,
1379
- model: { providerID: input.model.providerID as ProviderID, modelID: input.model.modelID as ModelID },
1380
- sessionID: input.sessionID,
1381
- agent: input.agent,
1382
- time: { created: syntheticTime },
1383
- })
1384
-
1385
- yield* session.updatePart({
1386
- id: PartID.ascending(),
1387
- messageID: msg.id,
1388
- sessionID: input.sessionID,
1389
- type: "checkpoint",
1390
- checkpointDir: "",
1391
- checkpointNumber: 0,
1392
- coveredUpTo: input.boundary,
1393
- })
1394
-
1395
- if (indexText) {
1396
- yield* session.updatePart({
1397
- id: PartID.ascending(),
1398
- messageID: msg.id,
1399
- sessionID: input.sessionID,
1400
- type: "text",
1401
- synthetic: true,
1402
- text: indexText,
1403
- })
1404
- }
1405
-
1406
- yield* session.updatePart({
1407
- id: PartID.ascending(),
1408
- messageID: msg.id,
1409
- sessionID: input.sessionID,
1410
- type: "text",
1411
- synthetic: true,
1412
- text: rebuildContext,
1413
- })
1414
-
1415
- const actorsText = yield* actorRegistry
1416
- .renderForAgent(input.sessionID)
1417
- .pipe(Effect.catch(() => Effect.succeed("")))
1418
- if (actorsText) {
1419
- yield* session.updatePart({
1420
- id: PartID.ascending(),
1421
- messageID: msg.id,
1422
- sessionID: input.sessionID,
1423
- type: "text",
1424
- synthetic: true,
1425
- text: actorsText,
1426
- })
1427
- }
1428
-
1429
- // Microcompact: messages strictly newer than the boundary will survive
1430
- // into the rebuild context. Clear tool_result content for compactable
1431
- // tools so the first uncached request after rebuild is smaller. tool_use
1432
- // is preserved — LLM still sees what action was taken; result body
1433
- // becomes "[Old tool result content cleared]" via the converter at
1434
- // message-v2.ts (ToolStateCompleted → output).
1435
- // See docs/superpowers/specs/2026-06-03-rebuild-tail-microcompact-design.md.
1436
- //
1437
- // boundaryTime resolution (fail-closed):
1438
- // 1. Prefer explicit input.boundaryCreatedAt (production callers
1439
- // compute it but may pass undefined if the boundary message is no
1440
- // longer in their filterCompactedEffect slice).
1441
- // 2. Else look up input.boundary in allMsgs (full DB, includes
1442
- // pre-marker history).
1443
- // 3. Else SKIP — the previous fallback of 0 would clear EVERY
1444
- // completed compactable tool result in the entire session,
1445
- // corrupting future checkpoint writer input. Log a warning.
1446
- const allMsgs = yield* session.messages({ sessionID: input.sessionID, agentID: "*" })
1447
- const boundaryTime =
1448
- input.boundaryCreatedAt ??
1449
- allMsgs.find((m) => m.info.id === input.boundary)?.info.time.created
1450
- if (boundaryTime === undefined) {
1451
- log.warn("microcompact skipped: no boundary timestamp available", {
1452
- sessionID: input.sessionID,
1453
- boundary: input.boundary,
1454
- })
1455
- return true
1456
- }
1457
- let cleared = 0
1458
- for (const m of allMsgs) {
1459
- if (m.info.id === msg.id) continue
1460
- if (m.info.time.created <= boundaryTime) continue
1461
- for (const part of m.parts) {
1462
- if (part.type !== "tool") continue
1463
- if (!COMPACTABLE_TOOL_NAMES.has(part.tool)) continue
1464
- if (part.state.status !== "completed") continue
1465
- if (part.state.time.compacted) continue
1466
- part.state.time.compacted = Date.now()
1467
- yield* session.updatePart(part)
1468
- cleared += 1
1469
- }
1470
- }
1471
- if (cleared > 0) {
1472
- log.info("rebuild microcompact", { sessionID: input.sessionID, cleared })
1473
- }
1474
-
1475
- return true
1476
- })
1477
-
1478
- return Service.of({
1479
- tryStartCheckpointWriter,
1480
- waitForWriter,
1481
- drainWriters,
1482
- hasCheckpoint,
1483
- hasMemoryOrTasks,
1484
- loadLatest,
1485
- loadCheckpoints,
1486
- renderIndex,
1487
- renderRebuildContext,
1488
- lastBoundary,
1489
- isWriterRunning,
1490
- insertRebuildBoundary,
1491
- })
1492
- }),
1493
- )
1494
-
1495
- // ---------------------------------------------------------------------------
1496
- // Default layer
1497
- // ---------------------------------------------------------------------------
1498
-
1499
- // `defaultLayer` no longer requires `Actor.Service`: SessionCheckpoint reaches
1500
- // the Actor implementation through the late-bound `spawnRef` (see
1501
- // `actor/spawn-ref.ts`). This deliberately breaks the otherwise-unresolvable
1502
- // layer cycle Actor → SessionPrompt → SessionCheckpoint → Actor. The AppLayer
1503
- // constructs `Actor.defaultLayer` separately; its initialiser populates
1504
- // `spawnRef`, which `tryStartCheckpointWriter` reads at call time.
1505
- export const defaultLayer = Layer.suspend(() =>
1506
- layer.pipe(
1507
- Layer.provide(Session.defaultLayer),
1508
- Layer.provide(Bus.layer),
1509
- Layer.provide(Config.defaultLayer),
1510
- Layer.provide(Memory.defaultLayer),
1511
- Layer.provide(TaskRegistry.defaultLayer),
1512
- Layer.provide(ActorRegistry.defaultLayer),
1513
- ),
1514
- )
1515
-
1516
- // ---------------------------------------------------------------------------
1517
- // Convenience wrappers
1518
- // ---------------------------------------------------------------------------
1519
-
1520
- const { runPromise } = makeRuntime(Service, defaultLayer)
1521
-
1522
- export async function hasCheckpoint(input: { sessionID: SessionID }) {
1523
- return runPromise((svc) => svc.hasCheckpoint(input.sessionID))
1524
- }
1525
-
1526
- export async function loadLatest(input: { sessionID: SessionID }) {
1527
- return runPromise((svc) => svc.loadLatest(input.sessionID))
1528
- }
1529
-
1530
- export async function loadCheckpoints(input: { sessionID: SessionID; count: number }) {
1531
- return runPromise((svc) => svc.loadCheckpoints(input.sessionID, input.count))
1532
- }
1533
-
1534
- export async function renderIndex(input: { sessionID: SessionID }) {
1535
- return runPromise((svc) => svc.renderIndex(input.sessionID))
1536
- }
1537
-
1538
- export async function renderRebuildContext(input: {
1539
- sessionID: SessionID
1540
- lastMessageInfo?: LastMessageInfo
1541
- agentID?: string
1542
- }) {
1543
- return runPromise((svc) =>
1544
- svc.renderRebuildContext(input.sessionID, { lastMessageInfo: input.lastMessageInfo, agentID: input.agentID }),
1545
- )
1546
- }
1547
-
1548
- export async function lastBoundary(input: { sessionID: SessionID }) {
1549
- return runPromise((svc) => svc.lastBoundary(input.sessionID))
1550
- }
1551
-
1552
- export async function isWriterRunning(input: { sessionID: SessionID }) {
1553
- return runPromise((svc) => svc.isWriterRunning(input.sessionID))
1554
- }
1555
-
1556
- export * as SessionCheckpoint from "./checkpoint"
1557
-
1558
- // Test-only re-export so test code can call composeWriterPrompt without
1559
- // triggering the full SessionCheckpoint Service stack.
1560
- export { composeWriterPrompt as composeWriterPromptForTest }