bonecode 1.0.0 → 1.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (840) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +64 -50
  3. package/bin/bonecode +47 -42
  4. package/bone/output/agent/.dockerignore +7 -7
  5. package/bone/output/agent/.env.example +36 -36
  6. package/bone/output/agent/.github/workflows/ci.yaml +58 -58
  7. package/bone/output/agent/AgentDomain.bone.map +349 -349
  8. package/bone/output/agent/AgentDomain.postman_collection.json +957 -957
  9. package/bone/output/agent/Dockerfile +22 -22
  10. package/bone/output/agent/README.md +47 -47
  11. package/bone/output/agent/admin/index.html +739 -739
  12. package/bone/output/agent/docker-compose.yaml +22 -22
  13. package/bone/output/agent/k8s/deployment.yaml +75 -75
  14. package/bone/output/agent/migrations/agent.sql +36 -36
  15. package/bone/output/agent/migrations/agent_instance.sql +36 -36
  16. package/bone/output/agent/migrations/audit_log.sql +18 -18
  17. package/bone/output/agent/migrations/build_step.sql +34 -34
  18. package/bone/output/agent/migrations/event_outbox.sql +31 -31
  19. package/bone/output/agent/migrations/plan.sql +30 -30
  20. package/bone/output/agent/migrations/task.sql +30 -30
  21. package/bone/output/agent/migrations/tool_call.sql +33 -33
  22. package/bone/output/agent/openapi.yaml +1116 -1116
  23. package/bone/output/agent/package.json +35 -35
  24. package/bone/output/agent/schema.graphql +233 -233
  25. package/bone/output/agent/sdk/client.ts +231 -231
  26. package/bone/output/agent/src/algorithms.ts +2 -2
  27. package/bone/output/agent/src/audit.ts +44 -44
  28. package/bone/output/agent/src/auth.ts +57 -57
  29. package/bone/output/agent/src/cron.ts +12 -12
  30. package/bone/output/agent/src/db.ts +31 -31
  31. package/bone/output/agent/src/debug.ts +66 -66
  32. package/bone/output/agent/src/events.ts +243 -243
  33. package/bone/output/agent/src/extensions.ts +54 -54
  34. package/bone/output/agent/src/failure_rules.ts +322 -322
  35. package/bone/output/agent/src/flows.ts +168 -168
  36. package/bone/output/agent/src/health.ts +43 -43
  37. package/bone/output/agent/src/index.ts +99 -99
  38. package/bone/output/agent/src/logger.ts +69 -66
  39. package/bone/output/agent/src/metrics.ts +75 -75
  40. package/bone/output/agent/src/migrate.ts +351 -351
  41. package/bone/output/agent/src/migration_diff.ts +108 -108
  42. package/bone/output/agent/src/notify.ts +125 -125
  43. package/bone/output/agent/src/routes/plan.ts +91 -91
  44. package/bone/output/agent/src/routes/task.ts +105 -105
  45. package/bone/output/agent/src/routes/tool_call.ts +166 -166
  46. package/bone/output/agent/src/schemas.ts +384 -384
  47. package/bone/output/agent/src/state_machines/agent_instance.ts +24 -24
  48. package/bone/output/agent/src/state_machines/build_step.ts +22 -22
  49. package/bone/output/agent/src/state_machines/plan.ts +22 -22
  50. package/bone/output/agent/src/state_machines/task.ts +22 -22
  51. package/bone/output/agent/src/state_machines/tool_call.ts +22 -22
  52. package/bone/output/agent/src/tests.ts +361 -361
  53. package/bone/output/agent/src/websocket.ts +200 -200
  54. package/bone/output/agent/tsconfig.json +24 -24
  55. package/bone/output/rag/.dockerignore +7 -7
  56. package/bone/output/rag/.env.example +36 -36
  57. package/bone/output/rag/.github/workflows/ci.yaml +58 -58
  58. package/bone/output/rag/Dockerfile +22 -22
  59. package/bone/output/rag/RAGDomain.bone.map +286 -286
  60. package/bone/output/rag/RAGDomain.postman_collection.json +922 -922
  61. package/bone/output/rag/README.md +47 -47
  62. package/bone/output/rag/admin/index.html +817 -817
  63. package/bone/output/rag/docker-compose.yaml +22 -22
  64. package/bone/output/rag/k8s/deployment.yaml +75 -75
  65. package/bone/output/rag/migrations/audit_log.sql +18 -18
  66. package/bone/output/rag/migrations/code_chunk.sql +34 -34
  67. package/bone/output/rag/migrations/code_file.sql +33 -33
  68. package/bone/output/rag/migrations/event_outbox.sql +31 -31
  69. package/bone/output/rag/migrations/indexing_job.sql +33 -33
  70. package/bone/output/rag/migrations/knowledge_base.sql +35 -35
  71. package/bone/output/rag/migrations/memory_entry.sql +34 -34
  72. package/bone/output/rag/openapi.yaml +1097 -1097
  73. package/bone/output/rag/package.json +35 -35
  74. package/bone/output/rag/schema.graphql +245 -245
  75. package/bone/output/rag/sdk/client.ts +234 -234
  76. package/bone/output/rag/src/algorithms.ts +2 -2
  77. package/bone/output/rag/src/audit.ts +37 -37
  78. package/bone/output/rag/src/auth.ts +57 -57
  79. package/bone/output/rag/src/cron.ts +12 -12
  80. package/bone/output/rag/src/db.ts +31 -31
  81. package/bone/output/rag/src/debug.ts +66 -66
  82. package/bone/output/rag/src/events.ts +243 -243
  83. package/bone/output/rag/src/extensions.ts +350 -350
  84. package/bone/output/rag/src/failure_rules.ts +314 -314
  85. package/bone/output/rag/src/flows.ts +239 -239
  86. package/bone/output/rag/src/health.ts +43 -43
  87. package/bone/output/rag/src/index.ts +94 -94
  88. package/bone/output/rag/src/logger.ts +69 -66
  89. package/bone/output/rag/src/metrics.ts +75 -75
  90. package/bone/output/rag/src/migrate.ts +363 -363
  91. package/bone/output/rag/src/migration_diff.ts +108 -108
  92. package/bone/output/rag/src/notify.ts +99 -99
  93. package/bone/output/rag/src/routes/code_chunk.ts +75 -75
  94. package/bone/output/rag/src/routes/code_file.ts +101 -101
  95. package/bone/output/rag/src/routes/indexing_job.ts +87 -87
  96. package/bone/output/rag/src/routes/knowledge_base.ts +230 -230
  97. package/bone/output/rag/src/routes/memory_entry.ts +87 -87
  98. package/bone/output/rag/src/schemas.ts +394 -394
  99. package/bone/output/rag/src/state_machines/code_file.ts +23 -23
  100. package/bone/output/rag/src/state_machines/indexing_job.ts +22 -22
  101. package/bone/output/rag/src/state_machines/knowledge_base.ts +23 -23
  102. package/bone/output/rag/src/state_machines/memory_entry.ts +20 -20
  103. package/bone/output/rag/src/tests.ts +339 -339
  104. package/bone/output/rag/tsconfig.json +24 -24
  105. package/bone/output/session/.dockerignore +7 -7
  106. package/bone/output/session/.env.example +36 -36
  107. package/bone/output/session/.github/workflows/ci.yaml +58 -58
  108. package/bone/output/session/Dockerfile +22 -22
  109. package/bone/output/session/README.md +47 -47
  110. package/bone/output/session/SessionDomain.bone.map +349 -349
  111. package/bone/output/session/SessionDomain.postman_collection.json +957 -957
  112. package/bone/output/session/admin/index.html +666 -666
  113. package/bone/output/session/docker-compose.yaml +22 -22
  114. package/bone/output/session/k8s/deployment.yaml +75 -75
  115. package/bone/output/session/migrations/audit_log.sql +18 -18
  116. package/bone/output/session/migrations/event_outbox.sql +31 -31
  117. package/bone/output/session/migrations/message.sql +31 -31
  118. package/bone/output/session/migrations/part.sql +28 -28
  119. package/bone/output/session/migrations/permission.sql +28 -28
  120. package/bone/output/session/migrations/project.sql +28 -28
  121. package/bone/output/session/migrations/session.sql +38 -38
  122. package/bone/output/session/openapi.yaml +1101 -1101
  123. package/bone/output/session/package.json +35 -35
  124. package/bone/output/session/schema.graphql +222 -222
  125. package/bone/output/session/sdk/client.ts +225 -225
  126. package/bone/output/session/src/algorithms.ts +2 -2
  127. package/bone/output/session/src/audit.ts +44 -44
  128. package/bone/output/session/src/auth.ts +57 -57
  129. package/bone/output/session/src/cron.ts +12 -12
  130. package/bone/output/session/src/db.ts +31 -31
  131. package/bone/output/session/src/debug.ts +66 -66
  132. package/bone/output/session/src/events.ts +270 -270
  133. package/bone/output/session/src/extensions.ts +215 -215
  134. package/bone/output/session/src/failure_rules.ts +283 -283
  135. package/bone/output/session/src/flows.ts +168 -168
  136. package/bone/output/session/src/health.ts +43 -43
  137. package/bone/output/session/src/index.ts +99 -99
  138. package/bone/output/session/src/logger.ts +67 -66
  139. package/bone/output/session/src/metrics.ts +75 -75
  140. package/bone/output/session/src/migrate.ts +331 -331
  141. package/bone/output/session/src/migration_diff.ts +108 -108
  142. package/bone/output/session/src/notify.ts +112 -112
  143. package/bone/output/session/src/routes/message.ts +93 -93
  144. package/bone/output/session/src/routes/part.ts +79 -79
  145. package/bone/output/session/src/routes/permission.ts +79 -79
  146. package/bone/output/session/src/routes/project.ts +79 -79
  147. package/bone/output/session/src/routes/session.ts +294 -294
  148. package/bone/output/session/src/schemas.ts +357 -357
  149. package/bone/output/session/src/state_machines/session.ts +23 -23
  150. package/bone/output/session/src/tests.ts +325 -325
  151. package/bone/output/session/src/websocket.ts +223 -200
  152. package/bone/output/session/tsconfig.json +24 -24
  153. package/bone/output/workspace/.dockerignore +7 -7
  154. package/bone/output/workspace/.env.example +36 -36
  155. package/bone/output/workspace/.github/workflows/ci.yaml +58 -58
  156. package/bone/output/workspace/Dockerfile +22 -22
  157. package/bone/output/workspace/README.md +45 -45
  158. package/bone/output/workspace/WorkspaceDomain.bone.map +188 -188
  159. package/bone/output/workspace/WorkspaceDomain.postman_collection.json +620 -620
  160. package/bone/output/workspace/admin/index.html +484 -484
  161. package/bone/output/workspace/docker-compose.yaml +22 -22
  162. package/bone/output/workspace/k8s/deployment.yaml +75 -75
  163. package/bone/output/workspace/migrations/audit_log.sql +18 -18
  164. package/bone/output/workspace/migrations/codebase.sql +34 -34
  165. package/bone/output/workspace/migrations/event_outbox.sql +31 -31
  166. package/bone/output/workspace/migrations/snapshot.sql +32 -32
  167. package/bone/output/workspace/migrations/workspace.sql +33 -33
  168. package/bone/output/workspace/openapi.yaml +721 -721
  169. package/bone/output/workspace/package.json +35 -35
  170. package/bone/output/workspace/schema.graphql +153 -153
  171. package/bone/output/workspace/sdk/client.ts +155 -155
  172. package/bone/output/workspace/src/algorithms.ts +2 -2
  173. package/bone/output/workspace/src/audit.ts +37 -37
  174. package/bone/output/workspace/src/auth.ts +57 -57
  175. package/bone/output/workspace/src/cron.ts +12 -12
  176. package/bone/output/workspace/src/db.ts +31 -31
  177. package/bone/output/workspace/src/debug.ts +66 -66
  178. package/bone/output/workspace/src/events.ts +243 -243
  179. package/bone/output/workspace/src/extensions.ts +44 -44
  180. package/bone/output/workspace/src/failure_rules.ts +152 -152
  181. package/bone/output/workspace/src/health.ts +43 -43
  182. package/bone/output/workspace/src/index.ts +88 -88
  183. package/bone/output/workspace/src/logger.ts +69 -66
  184. package/bone/output/workspace/src/metrics.ts +75 -75
  185. package/bone/output/workspace/src/migrate.ts +219 -219
  186. package/bone/output/workspace/src/migration_diff.ts +108 -108
  187. package/bone/output/workspace/src/notify.ts +73 -73
  188. package/bone/output/workspace/src/routes/codebase.ts +87 -87
  189. package/bone/output/workspace/src/routes/snapshot.ts +127 -127
  190. package/bone/output/workspace/src/routes/workspace.ts +190 -190
  191. package/bone/output/workspace/src/schemas.ts +231 -231
  192. package/bone/output/workspace/src/state_machines/codebase.ts +21 -21
  193. package/bone/output/workspace/src/state_machines/snapshot.ts +20 -20
  194. package/bone/output/workspace/src/state_machines/workspace.ts +21 -21
  195. package/bone/output/workspace/src/tests.ts +248 -248
  196. package/bone/output/workspace/tsconfig.json +24 -24
  197. package/compat/opencode_adapter.ts +282 -34
  198. package/dist/bone/output/agent/src/algorithms.d.ts +1 -0
  199. package/dist/bone/output/agent/src/algorithms.js +3 -0
  200. package/dist/bone/output/agent/src/algorithms.js.map +1 -0
  201. package/dist/bone/output/agent/src/audit.d.ts +3 -0
  202. package/dist/bone/output/agent/src/audit.js +40 -0
  203. package/dist/bone/output/agent/src/audit.js.map +1 -0
  204. package/dist/bone/output/agent/src/auth.d.ts +8 -0
  205. package/dist/bone/output/agent/src/auth.js +56 -0
  206. package/dist/bone/output/agent/src/auth.js.map +1 -0
  207. package/dist/bone/output/agent/src/db.d.ts +6 -0
  208. package/dist/bone/output/agent/src/db.js +63 -0
  209. package/dist/bone/output/agent/src/db.js.map +1 -0
  210. package/dist/bone/output/agent/src/events.d.ts +25 -0
  211. package/dist/bone/output/agent/src/events.js +184 -0
  212. package/dist/bone/output/agent/src/events.js.map +1 -0
  213. package/dist/bone/output/agent/src/logger.d.ts +28 -0
  214. package/dist/bone/output/agent/src/logger.js +45 -0
  215. package/dist/bone/output/agent/src/logger.js.map +1 -0
  216. package/dist/bone/output/agent/src/metrics.d.ts +5 -0
  217. package/dist/bone/output/agent/src/metrics.js +60 -0
  218. package/dist/bone/output/agent/src/metrics.js.map +1 -0
  219. package/dist/bone/output/agent/src/routes/agent_instance.d.ts +1 -0
  220. package/dist/bone/output/agent/src/routes/agent_instance.js +253 -0
  221. package/dist/bone/output/agent/src/routes/agent_instance.js.map +1 -0
  222. package/dist/bone/output/agent/src/routes/build_step.d.ts +1 -0
  223. package/dist/bone/output/agent/src/routes/build_step.js +133 -0
  224. package/dist/bone/output/agent/src/routes/build_step.js.map +1 -0
  225. package/dist/bone/output/agent/src/routes/plan.d.ts +1 -0
  226. package/dist/bone/output/agent/src/routes/plan.js +119 -0
  227. package/dist/bone/output/agent/src/routes/plan.js.map +1 -0
  228. package/dist/bone/output/agent/src/routes/task.d.ts +1 -0
  229. package/dist/bone/output/agent/src/routes/task.js +133 -0
  230. package/dist/bone/output/agent/src/routes/task.js.map +1 -0
  231. package/dist/bone/output/agent/src/routes/tool_call.d.ts +1 -0
  232. package/dist/bone/output/agent/src/routes/tool_call.js +190 -0
  233. package/dist/bone/output/agent/src/routes/tool_call.js.map +1 -0
  234. package/dist/bone/output/agent/src/state_machines/agent_instance.d.ts +9 -0
  235. package/dist/bone/output/agent/src/state_machines/agent_instance.js +22 -0
  236. package/dist/bone/output/agent/src/state_machines/agent_instance.js.map +1 -0
  237. package/dist/bone/output/agent/src/state_machines/build_step.d.ts +9 -0
  238. package/dist/bone/output/agent/src/state_machines/build_step.js +20 -0
  239. package/dist/bone/output/agent/src/state_machines/build_step.js.map +1 -0
  240. package/dist/bone/output/agent/src/state_machines/plan.d.ts +9 -0
  241. package/dist/bone/output/agent/src/state_machines/plan.js +20 -0
  242. package/dist/bone/output/agent/src/state_machines/plan.js.map +1 -0
  243. package/dist/bone/output/agent/src/state_machines/task.d.ts +9 -0
  244. package/dist/bone/output/agent/src/state_machines/task.js +20 -0
  245. package/dist/bone/output/agent/src/state_machines/task.js.map +1 -0
  246. package/dist/bone/output/agent/src/state_machines/tool_call.d.ts +9 -0
  247. package/dist/bone/output/agent/src/state_machines/tool_call.js +20 -0
  248. package/dist/bone/output/agent/src/state_machines/tool_call.js.map +1 -0
  249. package/dist/bone/output/rag/src/algorithms.d.ts +1 -0
  250. package/dist/bone/output/rag/src/algorithms.js +3 -0
  251. package/dist/bone/output/rag/src/algorithms.js.map +1 -0
  252. package/dist/bone/output/rag/src/auth.d.ts +8 -0
  253. package/dist/bone/output/rag/src/auth.js +56 -0
  254. package/dist/bone/output/rag/src/auth.js.map +1 -0
  255. package/dist/bone/output/rag/src/db.d.ts +6 -0
  256. package/dist/bone/output/rag/src/db.js +63 -0
  257. package/dist/bone/output/rag/src/db.js.map +1 -0
  258. package/dist/bone/output/rag/src/events.d.ts +25 -0
  259. package/dist/bone/output/rag/src/events.js +184 -0
  260. package/dist/bone/output/rag/src/events.js.map +1 -0
  261. package/dist/bone/output/rag/src/extensions.d.ts +83 -0
  262. package/dist/bone/output/rag/src/extensions.js +329 -0
  263. package/dist/bone/output/rag/src/extensions.js.map +1 -0
  264. package/dist/bone/output/rag/src/flows.d.ts +24 -0
  265. package/dist/bone/output/rag/src/flows.js +236 -0
  266. package/dist/bone/output/rag/src/flows.js.map +1 -0
  267. package/dist/bone/output/rag/src/logger.d.ts +28 -0
  268. package/dist/bone/output/rag/src/logger.js +45 -0
  269. package/dist/bone/output/rag/src/logger.js.map +1 -0
  270. package/dist/bone/output/rag/src/metrics.d.ts +5 -0
  271. package/dist/bone/output/rag/src/metrics.js +60 -0
  272. package/dist/bone/output/rag/src/metrics.js.map +1 -0
  273. package/dist/bone/output/rag/src/routes/code_chunk.d.ts +1 -0
  274. package/dist/bone/output/rag/src/routes/code_chunk.js +100 -0
  275. package/dist/bone/output/rag/src/routes/code_chunk.js.map +1 -0
  276. package/dist/bone/output/rag/src/routes/code_file.d.ts +1 -0
  277. package/dist/bone/output/rag/src/routes/code_file.js +127 -0
  278. package/dist/bone/output/rag/src/routes/code_file.js.map +1 -0
  279. package/dist/bone/output/rag/src/routes/indexing_job.d.ts +1 -0
  280. package/dist/bone/output/rag/src/routes/indexing_job.js +113 -0
  281. package/dist/bone/output/rag/src/routes/indexing_job.js.map +1 -0
  282. package/dist/bone/output/rag/src/routes/knowledge_base.d.ts +1 -0
  283. package/dist/bone/output/rag/src/routes/knowledge_base.js +242 -0
  284. package/dist/bone/output/rag/src/routes/knowledge_base.js.map +1 -0
  285. package/dist/bone/output/rag/src/routes/memory_entry.d.ts +1 -0
  286. package/dist/bone/output/rag/src/routes/memory_entry.js +113 -0
  287. package/dist/bone/output/rag/src/routes/memory_entry.js.map +1 -0
  288. package/dist/bone/output/rag/src/state_machines/code_file.d.ts +9 -0
  289. package/dist/bone/output/rag/src/state_machines/code_file.js +21 -0
  290. package/dist/bone/output/rag/src/state_machines/code_file.js.map +1 -0
  291. package/dist/bone/output/rag/src/state_machines/indexing_job.d.ts +9 -0
  292. package/dist/bone/output/rag/src/state_machines/indexing_job.js +20 -0
  293. package/dist/bone/output/rag/src/state_machines/indexing_job.js.map +1 -0
  294. package/dist/bone/output/rag/src/state_machines/knowledge_base.d.ts +9 -0
  295. package/dist/bone/output/rag/src/state_machines/knowledge_base.js +21 -0
  296. package/dist/bone/output/rag/src/state_machines/knowledge_base.js.map +1 -0
  297. package/dist/bone/output/rag/src/state_machines/memory_entry.d.ts +9 -0
  298. package/dist/bone/output/rag/src/state_machines/memory_entry.js +18 -0
  299. package/dist/bone/output/rag/src/state_machines/memory_entry.js.map +1 -0
  300. package/dist/bone/output/session/src/algorithms.d.ts +1 -0
  301. package/dist/bone/output/session/src/algorithms.js +3 -0
  302. package/dist/bone/output/session/src/algorithms.js.map +1 -0
  303. package/dist/bone/output/session/src/audit.d.ts +3 -0
  304. package/dist/bone/output/session/src/audit.js +40 -0
  305. package/dist/bone/output/session/src/audit.js.map +1 -0
  306. package/dist/bone/output/session/src/auth.d.ts +8 -0
  307. package/dist/bone/output/session/src/auth.js +56 -0
  308. package/dist/bone/output/session/src/auth.js.map +1 -0
  309. package/dist/bone/output/session/src/db.d.ts +6 -0
  310. package/dist/bone/output/session/src/db.js +63 -0
  311. package/dist/bone/output/session/src/db.js.map +1 -0
  312. package/dist/bone/output/session/src/events.d.ts +26 -0
  313. package/dist/bone/output/session/src/events.js +212 -0
  314. package/dist/bone/output/session/src/events.js.map +1 -0
  315. package/dist/bone/output/session/src/extensions.d.ts +41 -0
  316. package/dist/bone/output/session/src/extensions.js +217 -0
  317. package/dist/bone/output/session/src/extensions.js.map +1 -0
  318. package/dist/bone/output/session/src/logger.d.ts +28 -0
  319. package/dist/bone/output/session/src/logger.js +44 -0
  320. package/dist/bone/output/session/src/logger.js.map +1 -0
  321. package/dist/bone/output/session/src/metrics.d.ts +5 -0
  322. package/dist/bone/output/session/src/metrics.js +60 -0
  323. package/dist/bone/output/session/src/metrics.js.map +1 -0
  324. package/dist/bone/output/session/src/routes/message.d.ts +1 -0
  325. package/dist/bone/output/session/src/routes/message.js +120 -0
  326. package/dist/bone/output/session/src/routes/message.js.map +1 -0
  327. package/dist/bone/output/session/src/routes/part.d.ts +1 -0
  328. package/dist/bone/output/session/src/routes/part.js +106 -0
  329. package/dist/bone/output/session/src/routes/part.js.map +1 -0
  330. package/dist/bone/output/session/src/routes/permission.d.ts +1 -0
  331. package/dist/bone/output/session/src/routes/permission.js +106 -0
  332. package/dist/bone/output/session/src/routes/permission.js.map +1 -0
  333. package/dist/bone/output/session/src/routes/project.d.ts +1 -0
  334. package/dist/bone/output/session/src/routes/project.js +106 -0
  335. package/dist/bone/output/session/src/routes/project.js.map +1 -0
  336. package/dist/bone/output/session/src/routes/session.d.ts +1 -0
  337. package/dist/bone/output/session/src/routes/session.js +308 -0
  338. package/dist/bone/output/session/src/routes/session.js.map +1 -0
  339. package/dist/bone/output/session/src/state_machines/session.d.ts +9 -0
  340. package/dist/bone/output/session/src/state_machines/session.js +21 -0
  341. package/dist/bone/output/session/src/state_machines/session.js.map +1 -0
  342. package/dist/bone/output/session/src/websocket.d.ts +15 -0
  343. package/dist/bone/output/session/src/websocket.js +215 -0
  344. package/dist/bone/output/session/src/websocket.js.map +1 -0
  345. package/dist/bone/output/workspace/src/algorithms.d.ts +1 -0
  346. package/dist/bone/output/workspace/src/algorithms.js +3 -0
  347. package/dist/bone/output/workspace/src/algorithms.js.map +1 -0
  348. package/dist/bone/output/workspace/src/auth.d.ts +8 -0
  349. package/dist/bone/output/workspace/src/auth.js +56 -0
  350. package/dist/bone/output/workspace/src/auth.js.map +1 -0
  351. package/dist/bone/output/workspace/src/db.d.ts +6 -0
  352. package/dist/bone/output/workspace/src/db.js +63 -0
  353. package/dist/bone/output/workspace/src/db.js.map +1 -0
  354. package/dist/bone/output/workspace/src/events.d.ts +25 -0
  355. package/dist/bone/output/workspace/src/events.js +184 -0
  356. package/dist/bone/output/workspace/src/events.js.map +1 -0
  357. package/dist/bone/output/workspace/src/logger.d.ts +28 -0
  358. package/dist/bone/output/workspace/src/logger.js +45 -0
  359. package/dist/bone/output/workspace/src/logger.js.map +1 -0
  360. package/dist/bone/output/workspace/src/metrics.d.ts +5 -0
  361. package/dist/bone/output/workspace/src/metrics.js +60 -0
  362. package/dist/bone/output/workspace/src/metrics.js.map +1 -0
  363. package/dist/bone/output/workspace/src/routes/codebase.d.ts +1 -0
  364. package/dist/bone/output/workspace/src/routes/codebase.js +113 -0
  365. package/dist/bone/output/workspace/src/routes/codebase.js.map +1 -0
  366. package/dist/bone/output/workspace/src/routes/snapshot.d.ts +1 -0
  367. package/dist/bone/output/workspace/src/routes/snapshot.js +151 -0
  368. package/dist/bone/output/workspace/src/routes/snapshot.js.map +1 -0
  369. package/dist/bone/output/workspace/src/routes/workspace.d.ts +1 -0
  370. package/dist/bone/output/workspace/src/routes/workspace.js +209 -0
  371. package/dist/bone/output/workspace/src/routes/workspace.js.map +1 -0
  372. package/dist/bone/output/workspace/src/state_machines/codebase.d.ts +9 -0
  373. package/dist/bone/output/workspace/src/state_machines/codebase.js +19 -0
  374. package/dist/bone/output/workspace/src/state_machines/codebase.js.map +1 -0
  375. package/dist/bone/output/workspace/src/state_machines/snapshot.d.ts +9 -0
  376. package/dist/bone/output/workspace/src/state_machines/snapshot.js +18 -0
  377. package/dist/bone/output/workspace/src/state_machines/snapshot.js.map +1 -0
  378. package/dist/bone/output/workspace/src/state_machines/workspace.d.ts +9 -0
  379. package/dist/bone/output/workspace/src/state_machines/workspace.js +19 -0
  380. package/dist/bone/output/workspace/src/state_machines/workspace.js.map +1 -0
  381. package/dist/compat/opencode_adapter.d.ts +25 -0
  382. package/dist/compat/opencode_adapter.js +599 -0
  383. package/dist/compat/opencode_adapter.js.map +1 -0
  384. package/dist/extensions/chunker.d.ts +24 -0
  385. package/dist/extensions/chunker.js +360 -0
  386. package/dist/extensions/chunker.js.map +1 -0
  387. package/dist/extensions/embedding_provider.d.ts +18 -0
  388. package/dist/extensions/embedding_provider.js +150 -0
  389. package/dist/extensions/embedding_provider.js.map +1 -0
  390. package/dist/extensions/llm_provider.d.ts +33 -0
  391. package/dist/extensions/llm_provider.js +338 -0
  392. package/dist/extensions/llm_provider.js.map +1 -0
  393. package/dist/extensions/mcp_bridge.d.ts +44 -0
  394. package/dist/extensions/mcp_bridge.js +151 -0
  395. package/dist/extensions/mcp_bridge.js.map +1 -0
  396. package/dist/extensions/rag_search.d.ts +38 -0
  397. package/dist/extensions/rag_search.js +242 -0
  398. package/dist/extensions/rag_search.js.map +1 -0
  399. package/dist/extensions/snapshot.d.ts +14 -0
  400. package/dist/extensions/snapshot.js +158 -0
  401. package/dist/extensions/snapshot.js.map +1 -0
  402. package/dist/extensions/tool_executor.d.ts +28 -0
  403. package/dist/extensions/tool_executor.js +268 -0
  404. package/dist/extensions/tool_executor.js.map +1 -0
  405. package/dist/src/cli.d.ts +15 -0
  406. package/dist/src/cli.js +687 -0
  407. package/dist/src/cli.js.map +1 -0
  408. package/dist/src/config.d.ts +44 -0
  409. package/dist/src/config.js +165 -0
  410. package/dist/src/config.js.map +1 -0
  411. package/dist/src/context_builder.d.ts +51 -0
  412. package/dist/src/context_builder.js +558 -0
  413. package/dist/src/context_builder.js.map +1 -0
  414. package/dist/src/db_adapter.d.ts +24 -0
  415. package/dist/src/db_adapter.js +341 -0
  416. package/dist/src/db_adapter.js.map +1 -0
  417. package/dist/src/engine/session/compaction_logic.d.ts +11 -0
  418. package/dist/src/engine/session/compaction_logic.js +113 -0
  419. package/dist/src/engine/session/compaction_logic.js.map +1 -0
  420. package/dist/src/engine/session/instruction_loader.d.ts +5 -0
  421. package/dist/src/engine/session/instruction_loader.js +78 -0
  422. package/dist/src/engine/session/instruction_loader.js.map +1 -0
  423. package/dist/src/engine/session/overflow_check.d.ts +14 -0
  424. package/dist/src/engine/session/overflow_check.js +45 -0
  425. package/dist/src/engine/session/overflow_check.js.map +1 -0
  426. package/dist/src/engine/session/prompt.d.ts +45 -0
  427. package/dist/src/engine/session/prompt.js +584 -0
  428. package/dist/src/engine/session/prompt.js.map +1 -0
  429. package/dist/src/engine/session/provider_transform.d.ts +59 -0
  430. package/dist/src/engine/session/provider_transform.js +193 -0
  431. package/dist/src/engine/session/provider_transform.js.map +1 -0
  432. package/dist/src/engine/session/retry_logic.d.ts +12 -0
  433. package/dist/src/engine/session/retry_logic.js +72 -0
  434. package/dist/src/engine/session/retry_logic.js.map +1 -0
  435. package/dist/src/engine/session/system_prompt.d.ts +9 -0
  436. package/dist/src/engine/session/system_prompt.js +96 -0
  437. package/dist/src/engine/session/system_prompt.js.map +1 -0
  438. package/dist/src/engine/session/tool_registry.d.ts +5 -0
  439. package/dist/src/engine/session/tool_registry.js +117 -0
  440. package/dist/src/engine/session/tool_registry.js.map +1 -0
  441. package/dist/src/export.d.ts +13 -0
  442. package/dist/src/export.js +103 -0
  443. package/dist/src/export.js.map +1 -0
  444. package/dist/src/mdns.d.ts +7 -0
  445. package/dist/src/mdns.js +60 -0
  446. package/dist/src/mdns.js.map +1 -0
  447. package/dist/src/rag_worker.d.ts +38 -0
  448. package/dist/src/rag_worker.js +435 -0
  449. package/dist/src/rag_worker.js.map +1 -0
  450. package/dist/src/server.d.ts +11 -0
  451. package/dist/src/server.js +214 -0
  452. package/dist/src/server.js.map +1 -0
  453. package/dist/src/stats.d.ts +45 -0
  454. package/dist/src/stats.js +233 -0
  455. package/dist/src/stats.js.map +1 -0
  456. package/dist/src/tui.d.ts +29 -0
  457. package/dist/src/tui.js +1053 -0
  458. package/dist/src/tui.js.map +1 -0
  459. package/package.json +21 -5
  460. package/src/cli.ts +314 -113
  461. package/src/db_adapter.ts +354 -0
  462. package/src/engine/account/account.sql.ts +39 -39
  463. package/src/engine/account/account.ts +456 -456
  464. package/src/engine/account/repo.ts +166 -166
  465. package/src/engine/account/schema.ts +99 -99
  466. package/src/engine/account/url.ts +8 -8
  467. package/src/engine/acp/README.md +174 -174
  468. package/src/engine/acp/agent.ts +1968 -1968
  469. package/src/engine/acp/runtime.ts +22 -22
  470. package/src/engine/acp/session.ts +122 -122
  471. package/src/engine/acp/types.ts +24 -24
  472. package/src/engine/agent/agent.ts +463 -463
  473. package/src/engine/agent/generate.txt +75 -75
  474. package/src/engine/agent/prompt/compaction.txt +9 -9
  475. package/src/engine/agent/prompt/explore.txt +18 -18
  476. package/src/engine/agent/prompt/scout.txt +36 -36
  477. package/src/engine/agent/prompt/summary.txt +11 -11
  478. package/src/engine/agent/prompt/title.txt +44 -44
  479. package/src/engine/agent/subagent-permissions.ts +34 -34
  480. package/src/engine/auth/index.ts +96 -96
  481. package/src/engine/background/background/job.ts +200 -200
  482. package/src/engine/background/job.ts +200 -200
  483. package/src/engine/bus/bus-event.ts +45 -45
  484. package/src/engine/bus/global.ts +22 -22
  485. package/src/engine/bus/index.ts +203 -203
  486. package/src/engine/command/command/index.ts +181 -181
  487. package/src/engine/command/command/template/initialize.txt +66 -66
  488. package/src/engine/command/command/template/review.txt +101 -101
  489. package/src/engine/command/index.ts +181 -181
  490. package/src/engine/command/template/initialize.txt +66 -66
  491. package/src/engine/command/template/review.txt +101 -101
  492. package/src/engine/config/agent.ts +172 -172
  493. package/src/engine/config/attachment.ts +25 -25
  494. package/src/engine/config/command.ts +62 -62
  495. package/src/engine/config/config.ts +833 -833
  496. package/src/engine/config/console-state.ts +14 -14
  497. package/src/engine/config/entry-name.ts +16 -16
  498. package/src/engine/config/error.ts +23 -23
  499. package/src/engine/config/formatter.ts +13 -13
  500. package/src/engine/config/layout.ts +6 -6
  501. package/src/engine/config/lsp.ts +43 -43
  502. package/src/engine/config/managed.ts +71 -71
  503. package/src/engine/config/markdown.ts +96 -96
  504. package/src/engine/config/mcp.ts +56 -56
  505. package/src/engine/config/model-id.ts +5 -5
  506. package/src/engine/config/parse.ts +79 -79
  507. package/src/engine/config/paths.ts +45 -45
  508. package/src/engine/config/permission.ts +58 -58
  509. package/src/engine/config/plugin.ts +84 -84
  510. package/src/engine/config/provider.ts +111 -111
  511. package/src/engine/config/reference.ts +23 -23
  512. package/src/engine/config/server.ts +19 -19
  513. package/src/engine/config/skills.ts +14 -14
  514. package/src/engine/config/variable.ts +90 -90
  515. package/src/engine/control-plane/adapters/index.ts +41 -41
  516. package/src/engine/control-plane/adapters/worktree.ts +96 -96
  517. package/src/engine/control-plane/dev/README.md +19 -19
  518. package/src/engine/control-plane/dev/debug-workspace-plugin.ts +73 -73
  519. package/src/engine/control-plane/schema.ts +14 -14
  520. package/src/engine/control-plane/types.ts +59 -59
  521. package/src/engine/control-plane/util.ts +39 -39
  522. package/src/engine/control-plane/workspace-adapter-runtime.ts +51 -51
  523. package/src/engine/control-plane/workspace-context.ts +26 -26
  524. package/src/engine/control-plane/workspace.sql.ts +20 -20
  525. package/src/engine/control-plane/workspace.ts +1072 -1072
  526. package/src/engine/data-migration.ts +161 -161
  527. package/src/engine/effect/app-runtime.ts +143 -143
  528. package/src/engine/effect/bootstrap-runtime.ts +29 -29
  529. package/src/engine/effect/bridge.ts +84 -84
  530. package/src/engine/effect/config-service.ts +67 -67
  531. package/src/engine/effect/instance-ref.ts +11 -11
  532. package/src/engine/effect/instance-registry.ts +12 -12
  533. package/src/engine/effect/instance-state.ts +72 -72
  534. package/src/engine/effect/promise.ts +17 -17
  535. package/src/engine/effect/run-service.ts +47 -47
  536. package/src/engine/effect/runner.ts +217 -217
  537. package/src/engine/effect/runtime-flags.ts +74 -74
  538. package/src/engine/effect/service-use.ts +38 -38
  539. package/src/engine/env/index.ts +37 -37
  540. package/src/engine/event-v2-bridge.ts +89 -89
  541. package/src/engine/file/file/ignore.ts +81 -81
  542. package/src/engine/file/file/index.ts +651 -651
  543. package/src/engine/file/file/protected.ts +59 -59
  544. package/src/engine/file/file/ripgrep.ts +481 -481
  545. package/src/engine/file/file/watcher.ts +167 -167
  546. package/src/engine/file/ignore.ts +81 -81
  547. package/src/engine/file/index.ts +651 -651
  548. package/src/engine/file/protected.ts +59 -59
  549. package/src/engine/file/ripgrep.ts +481 -481
  550. package/src/engine/file/watcher.ts +167 -167
  551. package/src/engine/format/format/formatter.ts +404 -404
  552. package/src/engine/format/format/index.ts +209 -209
  553. package/src/engine/format/formatter.ts +404 -404
  554. package/src/engine/format/index.ts +209 -209
  555. package/src/engine/git/git/index.ts +347 -347
  556. package/src/engine/git/index.ts +347 -347
  557. package/src/engine/id/id.ts +80 -80
  558. package/src/engine/ide/index.ts +70 -70
  559. package/src/engine/image/image/image.ts +176 -176
  560. package/src/engine/image/image.ts +176 -176
  561. package/src/engine/index.ts +251 -251
  562. package/src/engine/installation/index.ts +327 -327
  563. package/src/engine/lsp/client.ts +707 -707
  564. package/src/engine/lsp/diagnostic.ts +29 -29
  565. package/src/engine/lsp/language.ts +121 -121
  566. package/src/engine/lsp/launch.ts +21 -21
  567. package/src/engine/lsp/lsp/client.ts +707 -707
  568. package/src/engine/lsp/lsp/diagnostic.ts +29 -29
  569. package/src/engine/lsp/lsp/language.ts +121 -121
  570. package/src/engine/lsp/lsp/launch.ts +21 -21
  571. package/src/engine/lsp/lsp/lsp.ts +507 -507
  572. package/src/engine/lsp/lsp/server.ts +2064 -2064
  573. package/src/engine/lsp/lsp.ts +507 -507
  574. package/src/engine/lsp/server.ts +2064 -2064
  575. package/src/engine/mcp/auth.ts +146 -146
  576. package/src/engine/mcp/index.ts +958 -958
  577. package/src/engine/mcp/mcp/auth.ts +146 -146
  578. package/src/engine/mcp/mcp/index.ts +958 -958
  579. package/src/engine/mcp/mcp/oauth-callback.ts +232 -232
  580. package/src/engine/mcp/mcp/oauth-provider.ts +214 -214
  581. package/src/engine/mcp/oauth-callback.ts +232 -232
  582. package/src/engine/mcp/oauth-provider.ts +214 -214
  583. package/src/engine/node.ts +6 -6
  584. package/src/engine/patch/index.ts +689 -689
  585. package/src/engine/patch/patch/index.ts +689 -689
  586. package/src/engine/permission/arity.ts +163 -163
  587. package/src/engine/permission/evaluate.ts +15 -15
  588. package/src/engine/permission/index.ts +306 -306
  589. package/src/engine/permission/permission/arity.ts +163 -163
  590. package/src/engine/permission/permission/evaluate.ts +15 -15
  591. package/src/engine/permission/permission/index.ts +306 -306
  592. package/src/engine/permission/permission/schema.ts +13 -13
  593. package/src/engine/permission/schema.ts +13 -13
  594. package/src/engine/plugin/azure.ts +26 -26
  595. package/src/engine/plugin/cloudflare.ts +76 -76
  596. package/src/engine/plugin/codex.ts +622 -622
  597. package/src/engine/plugin/digitalocean.ts +411 -411
  598. package/src/engine/plugin/github-copilot/copilot.ts +394 -394
  599. package/src/engine/plugin/github-copilot/models.ts +196 -196
  600. package/src/engine/plugin/index.ts +295 -295
  601. package/src/engine/plugin/install.ts +439 -439
  602. package/src/engine/plugin/loader.ts +216 -216
  603. package/src/engine/plugin/meta.ts +188 -188
  604. package/src/engine/plugin/shared.ts +323 -323
  605. package/src/engine/project/bootstrap-service.ts +9 -9
  606. package/src/engine/project/bootstrap.ts +75 -75
  607. package/src/engine/project/instance-context.ts +24 -24
  608. package/src/engine/project/instance-layer.ts +11 -11
  609. package/src/engine/project/instance-runtime.ts +16 -16
  610. package/src/engine/project/instance-store.ts +193 -193
  611. package/src/engine/project/project.sql.ts +17 -17
  612. package/src/engine/project/project.ts +537 -537
  613. package/src/engine/project/schema.ts +13 -13
  614. package/src/engine/project/vcs.ts +405 -405
  615. package/src/engine/provider/auth.ts +225 -225
  616. package/src/engine/provider/error.ts +204 -204
  617. package/src/engine/provider/model-status.ts +8 -8
  618. package/src/engine/provider/provider.ts +1843 -1843
  619. package/src/engine/provider/schema.ts +30 -30
  620. package/src/engine/provider/transform.ts +1376 -1376
  621. package/src/engine/pty/index.ts +365 -365
  622. package/src/engine/pty/input.ts +24 -24
  623. package/src/engine/pty/pty/index.ts +365 -365
  624. package/src/engine/pty/pty/input.ts +24 -24
  625. package/src/engine/pty/pty/pty.bun.ts +26 -26
  626. package/src/engine/pty/pty/pty.node.ts +27 -27
  627. package/src/engine/pty/pty/pty.ts +25 -25
  628. package/src/engine/pty/pty/schema.ts +14 -14
  629. package/src/engine/pty/pty/ticket.ts +68 -68
  630. package/src/engine/pty/pty.bun.ts +26 -26
  631. package/src/engine/pty/pty.node.ts +27 -27
  632. package/src/engine/pty/pty.ts +25 -25
  633. package/src/engine/pty/schema.ts +14 -14
  634. package/src/engine/pty/ticket.ts +68 -68
  635. package/src/engine/question/index.ts +213 -213
  636. package/src/engine/question/question/index.ts +213 -213
  637. package/src/engine/question/question/schema.ts +10 -10
  638. package/src/engine/question/schema.ts +10 -10
  639. package/src/engine/reference/reference/reference.ts +241 -241
  640. package/src/engine/reference/reference/repository-cache.ts +147 -147
  641. package/src/engine/reference/reference.ts +241 -241
  642. package/src/engine/reference/repository-cache.ts +147 -147
  643. package/src/engine/session/compaction.ts +651 -651
  644. package/src/engine/session/instruction.ts +238 -238
  645. package/src/engine/session/llm.ts +459 -459
  646. package/src/engine/session/message-error.ts +14 -14
  647. package/src/engine/session/message-v2.ts +1202 -1202
  648. package/src/engine/session/message.ts +146 -146
  649. package/src/engine/session/overflow.ts +32 -32
  650. package/src/engine/session/processor.ts +823 -823
  651. package/src/engine/session/prompt/anthropic.txt +105 -105
  652. package/src/engine/session/prompt/beast.txt +147 -147
  653. package/src/engine/session/prompt/build-switch.txt +5 -5
  654. package/src/engine/session/prompt/codex.txt +79 -79
  655. package/src/engine/session/prompt/copilot-gpt-5.txt +143 -143
  656. package/src/engine/session/prompt/default.txt +105 -105
  657. package/src/engine/session/prompt/gemini.txt +155 -155
  658. package/src/engine/session/prompt/gpt.txt +107 -107
  659. package/src/engine/session/prompt/kimi.txt +95 -95
  660. package/src/engine/session/prompt/max-steps.txt +15 -15
  661. package/src/engine/session/prompt/plan-reminder-anthropic.txt +67 -67
  662. package/src/engine/session/prompt/plan.txt +26 -26
  663. package/src/engine/session/prompt/trinity.txt +97 -97
  664. package/src/engine/session/prompt.ts +66 -9
  665. package/src/engine/session/retry.ts +200 -200
  666. package/src/engine/session/revert.ts +162 -162
  667. package/src/engine/session/run-state.ts +153 -153
  668. package/src/engine/session/schema.ts +26 -26
  669. package/src/engine/session/session.sql.ts +137 -137
  670. package/src/engine/session/session.ts +1011 -1011
  671. package/src/engine/session/status.ts +94 -94
  672. package/src/engine/session/summary.ts +164 -164
  673. package/src/engine/session/system.ts +84 -84
  674. package/src/engine/session/todo.ts +81 -81
  675. package/src/engine/share/session.ts +61 -61
  676. package/src/engine/share/share-next.ts +376 -376
  677. package/src/engine/share/share.sql.ts +13 -13
  678. package/src/engine/shell/shell/shell.ts +215 -215
  679. package/src/engine/shell/shell.ts +215 -215
  680. package/src/engine/skill/discovery.ts +116 -116
  681. package/src/engine/skill/index.ts +336 -336
  682. package/src/engine/skill/prompt/customize-opencode.md +377 -377
  683. package/src/engine/skill/skill/discovery.ts +116 -116
  684. package/src/engine/skill/skill/index.ts +336 -336
  685. package/src/engine/skill/skill/prompt/customize-opencode.md +377 -377
  686. package/src/engine/snapshot/index.ts +762 -762
  687. package/src/engine/snapshot/snapshot/index.ts +762 -762
  688. package/src/engine/sync/README.md +179 -179
  689. package/src/engine/sync/event.sql.ts +17 -17
  690. package/src/engine/sync/index.ts +410 -410
  691. package/src/engine/sync/schema.ts +11 -11
  692. package/src/engine/temporary.ts +33 -33
  693. package/src/engine/tool/apply_patch.ts +313 -313
  694. package/src/engine/tool/apply_patch.txt +33 -33
  695. package/src/engine/tool/edit.ts +711 -711
  696. package/src/engine/tool/edit.txt +10 -10
  697. package/src/engine/tool/external-directory.ts +49 -49
  698. package/src/engine/tool/glob.ts +103 -103
  699. package/src/engine/tool/glob.txt +6 -6
  700. package/src/engine/tool/grep.ts +156 -156
  701. package/src/engine/tool/grep.txt +8 -8
  702. package/src/engine/tool/invalid.ts +21 -21
  703. package/src/engine/tool/json-schema.ts +164 -164
  704. package/src/engine/tool/lsp.ts +113 -113
  705. package/src/engine/tool/lsp.txt +24 -24
  706. package/src/engine/tool/mcp-websearch.ts +96 -96
  707. package/src/engine/tool/plan-enter.txt +14 -14
  708. package/src/engine/tool/plan-exit.txt +13 -13
  709. package/src/engine/tool/plan.ts +78 -78
  710. package/src/engine/tool/question.ts +44 -44
  711. package/src/engine/tool/question.txt +10 -10
  712. package/src/engine/tool/read.ts +337 -337
  713. package/src/engine/tool/read.txt +14 -14
  714. package/src/engine/tool/registry.ts +472 -472
  715. package/src/engine/tool/repo_clone.ts +80 -80
  716. package/src/engine/tool/repo_clone.txt +5 -5
  717. package/src/engine/tool/repo_overview.ts +279 -279
  718. package/src/engine/tool/repo_overview.txt +4 -4
  719. package/src/engine/tool/schema.ts +14 -14
  720. package/src/engine/tool/shell/id.ts +19 -19
  721. package/src/engine/tool/shell/prompt.ts +295 -295
  722. package/src/engine/tool/shell/shell.txt +77 -77
  723. package/src/engine/tool/shell.ts +647 -647
  724. package/src/engine/tool/skill.ts +75 -75
  725. package/src/engine/tool/skill.txt +5 -5
  726. package/src/engine/tool/task.ts +337 -337
  727. package/src/engine/tool/task.txt +58 -58
  728. package/src/engine/tool/task_status.ts +179 -179
  729. package/src/engine/tool/task_status.txt +13 -13
  730. package/src/engine/tool/todo.ts +57 -57
  731. package/src/engine/tool/todowrite.txt +167 -167
  732. package/src/engine/tool/tool/apply_patch.ts +313 -313
  733. package/src/engine/tool/tool/apply_patch.txt +33 -33
  734. package/src/engine/tool/tool/edit.ts +711 -711
  735. package/src/engine/tool/tool/edit.txt +10 -10
  736. package/src/engine/tool/tool/external-directory.ts +49 -49
  737. package/src/engine/tool/tool/glob.ts +103 -103
  738. package/src/engine/tool/tool/glob.txt +6 -6
  739. package/src/engine/tool/tool/grep.ts +156 -156
  740. package/src/engine/tool/tool/grep.txt +8 -8
  741. package/src/engine/tool/tool/invalid.ts +21 -21
  742. package/src/engine/tool/tool/json-schema.ts +164 -164
  743. package/src/engine/tool/tool/lsp.ts +113 -113
  744. package/src/engine/tool/tool/lsp.txt +24 -24
  745. package/src/engine/tool/tool/mcp-websearch.ts +96 -96
  746. package/src/engine/tool/tool/plan-enter.txt +14 -14
  747. package/src/engine/tool/tool/plan-exit.txt +13 -13
  748. package/src/engine/tool/tool/plan.ts +78 -78
  749. package/src/engine/tool/tool/question.ts +44 -44
  750. package/src/engine/tool/tool/question.txt +10 -10
  751. package/src/engine/tool/tool/read.ts +337 -337
  752. package/src/engine/tool/tool/read.txt +14 -14
  753. package/src/engine/tool/tool/registry.ts +472 -472
  754. package/src/engine/tool/tool/repo_clone.ts +80 -80
  755. package/src/engine/tool/tool/repo_clone.txt +5 -5
  756. package/src/engine/tool/tool/repo_overview.ts +279 -279
  757. package/src/engine/tool/tool/repo_overview.txt +4 -4
  758. package/src/engine/tool/tool/schema.ts +14 -14
  759. package/src/engine/tool/tool/shell/id.ts +19 -19
  760. package/src/engine/tool/tool/shell/prompt.ts +295 -295
  761. package/src/engine/tool/tool/shell/shell.txt +77 -77
  762. package/src/engine/tool/tool/shell.ts +647 -647
  763. package/src/engine/tool/tool/skill.ts +75 -75
  764. package/src/engine/tool/tool/skill.txt +5 -5
  765. package/src/engine/tool/tool/task.ts +337 -337
  766. package/src/engine/tool/tool/task.txt +58 -58
  767. package/src/engine/tool/tool/task_status.ts +179 -179
  768. package/src/engine/tool/tool/task_status.txt +13 -13
  769. package/src/engine/tool/tool/todo.ts +57 -57
  770. package/src/engine/tool/tool/todowrite.txt +167 -167
  771. package/src/engine/tool/tool/tool.ts +164 -164
  772. package/src/engine/tool/tool/truncate.ts +160 -160
  773. package/src/engine/tool/tool/truncation-dir.ts +4 -4
  774. package/src/engine/tool/tool/webfetch.ts +192 -192
  775. package/src/engine/tool/tool/webfetch.txt +13 -13
  776. package/src/engine/tool/tool/websearch.ts +143 -143
  777. package/src/engine/tool/tool/websearch.txt +14 -14
  778. package/src/engine/tool/tool/write.ts +104 -104
  779. package/src/engine/tool/tool/write.txt +8 -8
  780. package/src/engine/tool/tool.ts +164 -164
  781. package/src/engine/tool/truncate.ts +160 -160
  782. package/src/engine/tool/truncation-dir.ts +4 -4
  783. package/src/engine/tool/webfetch.ts +192 -192
  784. package/src/engine/tool/webfetch.txt +13 -13
  785. package/src/engine/tool/websearch.ts +143 -143
  786. package/src/engine/tool/websearch.txt +14 -14
  787. package/src/engine/tool/write.ts +104 -104
  788. package/src/engine/tool/write.txt +8 -8
  789. package/src/engine/util/archive.ts +17 -17
  790. package/src/engine/util/bom.ts +31 -31
  791. package/src/engine/util/data-url.ts +9 -9
  792. package/src/engine/util/defer.ts +10 -10
  793. package/src/engine/util/effect-http-client.ts +11 -11
  794. package/src/engine/util/error.ts +88 -88
  795. package/src/engine/util/filesystem.ts +252 -252
  796. package/src/engine/util/format.ts +20 -20
  797. package/src/engine/util/iife.ts +3 -3
  798. package/src/engine/util/lazy.ts +20 -20
  799. package/src/engine/util/local-context.ts +25 -25
  800. package/src/engine/util/locale.ts +86 -86
  801. package/src/engine/util/media.ts +26 -26
  802. package/src/engine/util/process.ts +176 -176
  803. package/src/engine/util/queue.ts +32 -32
  804. package/src/engine/util/record.ts +3 -3
  805. package/src/engine/util/repository.ts +158 -158
  806. package/src/engine/util/rpc.ts +66 -66
  807. package/src/engine/util/signal.ts +12 -12
  808. package/src/engine/util/timeout.ts +13 -13
  809. package/src/engine/util/token.ts +7 -7
  810. package/src/engine/util/util/archive.ts +17 -17
  811. package/src/engine/util/util/bom.ts +31 -31
  812. package/src/engine/util/util/data-url.ts +9 -9
  813. package/src/engine/util/util/defer.ts +10 -10
  814. package/src/engine/util/util/effect-http-client.ts +11 -11
  815. package/src/engine/util/util/error.ts +88 -88
  816. package/src/engine/util/util/filesystem.ts +252 -252
  817. package/src/engine/util/util/format.ts +20 -20
  818. package/src/engine/util/util/iife.ts +3 -3
  819. package/src/engine/util/util/lazy.ts +20 -20
  820. package/src/engine/util/util/local-context.ts +25 -25
  821. package/src/engine/util/util/locale.ts +86 -86
  822. package/src/engine/util/util/media.ts +26 -26
  823. package/src/engine/util/util/process.ts +176 -176
  824. package/src/engine/util/util/queue.ts +32 -32
  825. package/src/engine/util/util/record.ts +3 -3
  826. package/src/engine/util/util/repository.ts +158 -158
  827. package/src/engine/util/util/rpc.ts +66 -66
  828. package/src/engine/util/util/signal.ts +12 -12
  829. package/src/engine/util/util/timeout.ts +13 -13
  830. package/src/engine/util/util/token.ts +7 -7
  831. package/src/engine/util/util/which.ts +14 -14
  832. package/src/engine/util/util/wildcard.ts +59 -59
  833. package/src/engine/util/which.ts +14 -14
  834. package/src/engine/util/wildcard.ts +59 -59
  835. package/src/engine/worktree/index.ts +621 -621
  836. package/src/export.ts +122 -0
  837. package/src/mdns.ts +53 -0
  838. package/src/server.ts +151 -156
  839. package/src/stats.ts +290 -0
  840. package/src/tui.ts +964 -480
@@ -1,1202 +1,1202 @@
1
- import { BusEvent } from "@/bus/bus-event"
2
- import { SessionID, MessageID, PartID } from "./schema"
3
- import { NamedError } from "@opencode-ai/core/util/error"
4
- import { APICallError, convertToModelMessages, LoadAPIKeyError, type ModelMessage, type UIMessage } from "ai"
5
- import { LSP } from "@/lsp/lsp"
6
- import { Snapshot } from "@/snapshot"
7
- import { SyncEvent } from "../sync"
8
- import { Database } from "@/storage/db"
9
- import { NotFoundError } from "@/storage/storage"
10
- import { and } from "drizzle-orm"
11
- import { desc } from "drizzle-orm"
12
- import { eq } from "drizzle-orm"
13
- import { inArray } from "drizzle-orm"
14
- import { lt } from "drizzle-orm"
15
- import { or } from "drizzle-orm"
16
- import { MessageTable, PartTable, SessionTable } from "./session.sql"
17
- import * as ProviderError from "@/provider/error"
18
- import { iife } from "@/util/iife"
19
- import { errorMessage } from "@/util/error"
20
- import { isMedia } from "@/util/media"
21
- import type { SystemError } from "bun"
22
- import type { Provider } from "@/provider/provider"
23
- import { ModelID, ProviderID } from "@/provider/schema"
24
- import { Effect, Schema, Types } from "effect"
25
- import { NonNegativeInt } from "@opencode-ai/core/schema"
26
- import * as EffectLogger from "@opencode-ai/core/effect/logger"
27
- import { MessageError } from "./message-error"
28
- import { AuthError, OutputLengthError } from "./message-error"
29
- export { AuthError, OutputLengthError } from "./message-error"
30
-
31
- /** Error shape thrown by Bun's fetch() when gzip/br decompression fails mid-stream */
32
- interface FetchDecompressionError extends Error {
33
- code: "ZlibError"
34
- errno: number
35
- path: string
36
- }
37
-
38
- export const SYNTHETIC_ATTACHMENT_PROMPT = "Attached media from tool result:"
39
- export { isMedia }
40
-
41
- export const AbortedError = NamedError.create("MessageAbortedError", { message: Schema.String })
42
- export const StructuredOutputError = NamedError.create("StructuredOutputError", {
43
- message: Schema.String,
44
- retries: NonNegativeInt,
45
- })
46
- export const APIError = NamedError.create("APIError", {
47
- message: Schema.String,
48
- statusCode: Schema.optional(NonNegativeInt),
49
- isRetryable: Schema.Boolean,
50
- responseHeaders: Schema.optional(Schema.Record(Schema.String, Schema.String)),
51
- responseBody: Schema.optional(Schema.String),
52
- metadata: Schema.optional(Schema.Record(Schema.String, Schema.String)),
53
- })
54
- export type APIError = Schema.Schema.Type<typeof APIError.Schema>
55
- export const ContextOverflowError = NamedError.create("ContextOverflowError", {
56
- message: Schema.String,
57
- responseBody: Schema.optional(Schema.String),
58
- })
59
-
60
- export class OutputFormatText extends Schema.Class<OutputFormatText>("OutputFormatText")({
61
- type: Schema.Literal("text"),
62
- }) {}
63
-
64
- export class OutputFormatJsonSchema extends Schema.Class<OutputFormatJsonSchema>("OutputFormatJsonSchema")({
65
- type: Schema.Literal("json_schema"),
66
- schema: Schema.Record(Schema.String, Schema.Any).annotate({ identifier: "JSONSchema" }),
67
- retryCount: NonNegativeInt.pipe(Schema.optional, Schema.withDecodingDefault(Effect.succeed(2))),
68
- }) {}
69
-
70
- export const Format = Schema.Union([OutputFormatText, OutputFormatJsonSchema]).annotate({
71
- discriminator: "type",
72
- identifier: "OutputFormat",
73
- })
74
- export type OutputFormat = Schema.Schema.Type<typeof Format>
75
-
76
- const partBase = {
77
- id: PartID,
78
- sessionID: SessionID,
79
- messageID: MessageID,
80
- }
81
-
82
- export const SnapshotPart = Schema.Struct({
83
- ...partBase,
84
- type: Schema.Literal("snapshot"),
85
- snapshot: Schema.String,
86
- }).annotate({ identifier: "SnapshotPart" })
87
- export type SnapshotPart = Types.DeepMutable<Schema.Schema.Type<typeof SnapshotPart>>
88
-
89
- export const PatchPart = Schema.Struct({
90
- ...partBase,
91
- type: Schema.Literal("patch"),
92
- hash: Schema.String,
93
- files: Schema.Array(Schema.String),
94
- }).annotate({ identifier: "PatchPart" })
95
- export type PatchPart = Types.DeepMutable<Schema.Schema.Type<typeof PatchPart>>
96
-
97
- export const TextPart = Schema.Struct({
98
- ...partBase,
99
- type: Schema.Literal("text"),
100
- text: Schema.String,
101
- synthetic: Schema.optional(Schema.Boolean),
102
- ignored: Schema.optional(Schema.Boolean),
103
- time: Schema.optional(
104
- Schema.Struct({
105
- start: NonNegativeInt,
106
- end: Schema.optional(NonNegativeInt),
107
- }),
108
- ),
109
- metadata: Schema.optional(Schema.Record(Schema.String, Schema.Any)),
110
- }).annotate({ identifier: "TextPart" })
111
- export type TextPart = Types.DeepMutable<Schema.Schema.Type<typeof TextPart>>
112
-
113
- export const ReasoningPart = Schema.Struct({
114
- ...partBase,
115
- type: Schema.Literal("reasoning"),
116
- text: Schema.String,
117
- metadata: Schema.optional(Schema.Record(Schema.String, Schema.Any)),
118
- time: Schema.Struct({
119
- start: NonNegativeInt,
120
- end: Schema.optional(NonNegativeInt),
121
- }),
122
- }).annotate({ identifier: "ReasoningPart" })
123
- export type ReasoningPart = Types.DeepMutable<Schema.Schema.Type<typeof ReasoningPart>>
124
-
125
- const filePartSourceBase = {
126
- text: Schema.Struct({
127
- value: Schema.String,
128
- start: Schema.Finite,
129
- end: Schema.Finite,
130
- }).annotate({ identifier: "FilePartSourceText" }),
131
- }
132
-
133
- export const FileSource = Schema.Struct({
134
- ...filePartSourceBase,
135
- type: Schema.Literal("file"),
136
- path: Schema.String,
137
- }).annotate({ identifier: "FileSource" })
138
-
139
- export const SymbolSource = Schema.Struct({
140
- ...filePartSourceBase,
141
- type: Schema.Literal("symbol"),
142
- path: Schema.String,
143
- range: LSP.Range,
144
- name: Schema.String,
145
- kind: NonNegativeInt,
146
- }).annotate({ identifier: "SymbolSource" })
147
-
148
- export const ResourceSource = Schema.Struct({
149
- ...filePartSourceBase,
150
- type: Schema.Literal("resource"),
151
- clientName: Schema.String,
152
- uri: Schema.String,
153
- }).annotate({ identifier: "ResourceSource" })
154
-
155
- export const FilePartSource = Schema.Union([FileSource, SymbolSource, ResourceSource]).annotate({
156
- discriminator: "type",
157
- identifier: "FilePartSource",
158
- })
159
-
160
- export const FilePart = Schema.Struct({
161
- ...partBase,
162
- type: Schema.Literal("file"),
163
- mime: Schema.String,
164
- filename: Schema.optional(Schema.String),
165
- url: Schema.String,
166
- source: Schema.optional(FilePartSource),
167
- }).annotate({ identifier: "FilePart" })
168
- export type FilePart = Types.DeepMutable<Schema.Schema.Type<typeof FilePart>>
169
-
170
- export const AgentPart = Schema.Struct({
171
- ...partBase,
172
- type: Schema.Literal("agent"),
173
- name: Schema.String,
174
- source: Schema.optional(
175
- Schema.Struct({
176
- value: Schema.String,
177
- start: NonNegativeInt,
178
- end: NonNegativeInt,
179
- }),
180
- ),
181
- }).annotate({ identifier: "AgentPart" })
182
- export type AgentPart = Types.DeepMutable<Schema.Schema.Type<typeof AgentPart>>
183
-
184
- export const CompactionPart = Schema.Struct({
185
- ...partBase,
186
- type: Schema.Literal("compaction"),
187
- auto: Schema.Boolean,
188
- overflow: Schema.optional(Schema.Boolean),
189
- tail_start_id: Schema.optional(MessageID),
190
- }).annotate({ identifier: "CompactionPart" })
191
- export type CompactionPart = Types.DeepMutable<Schema.Schema.Type<typeof CompactionPart>>
192
-
193
- export const SubtaskPart = Schema.Struct({
194
- ...partBase,
195
- type: Schema.Literal("subtask"),
196
- prompt: Schema.String,
197
- description: Schema.String,
198
- agent: Schema.String,
199
- model: Schema.optional(
200
- Schema.Struct({
201
- providerID: ProviderID,
202
- modelID: ModelID,
203
- }),
204
- ),
205
- command: Schema.optional(Schema.String),
206
- }).annotate({ identifier: "SubtaskPart" })
207
- export type SubtaskPart = Types.DeepMutable<Schema.Schema.Type<typeof SubtaskPart>>
208
-
209
- export const RetryPart = Schema.Struct({
210
- ...partBase,
211
- type: Schema.Literal("retry"),
212
- attempt: NonNegativeInt,
213
- error: APIError.EffectSchema,
214
- time: Schema.Struct({
215
- created: NonNegativeInt,
216
- }),
217
- }).annotate({ identifier: "RetryPart" })
218
- export type RetryPart = Omit<Types.DeepMutable<Schema.Schema.Type<typeof RetryPart>>, "error"> & {
219
- error: APIError
220
- }
221
-
222
- export const StepStartPart = Schema.Struct({
223
- ...partBase,
224
- type: Schema.Literal("step-start"),
225
- snapshot: Schema.optional(Schema.String),
226
- }).annotate({ identifier: "StepStartPart" })
227
- export type StepStartPart = Types.DeepMutable<Schema.Schema.Type<typeof StepStartPart>>
228
-
229
- export const StepFinishPart = Schema.Struct({
230
- ...partBase,
231
- type: Schema.Literal("step-finish"),
232
- reason: Schema.String,
233
- snapshot: Schema.optional(Schema.String),
234
- cost: Schema.Finite,
235
- tokens: Schema.Struct({
236
- total: Schema.optional(Schema.Finite),
237
- input: Schema.Finite,
238
- output: Schema.Finite,
239
- reasoning: Schema.Finite,
240
- cache: Schema.Struct({
241
- read: Schema.Finite,
242
- write: Schema.Finite,
243
- }),
244
- }),
245
- }).annotate({ identifier: "StepFinishPart" })
246
- export type StepFinishPart = Types.DeepMutable<Schema.Schema.Type<typeof StepFinishPart>>
247
-
248
- export const ToolStatePending = Schema.Struct({
249
- status: Schema.Literal("pending"),
250
- input: Schema.Record(Schema.String, Schema.Any),
251
- raw: Schema.String,
252
- }).annotate({ identifier: "ToolStatePending" })
253
- export type ToolStatePending = Types.DeepMutable<Schema.Schema.Type<typeof ToolStatePending>>
254
-
255
- export const ToolStateRunning = Schema.Struct({
256
- status: Schema.Literal("running"),
257
- input: Schema.Record(Schema.String, Schema.Any),
258
- title: Schema.optional(Schema.String),
259
- metadata: Schema.optional(Schema.Record(Schema.String, Schema.Any)),
260
- time: Schema.Struct({
261
- start: NonNegativeInt,
262
- }),
263
- }).annotate({ identifier: "ToolStateRunning" })
264
- export type ToolStateRunning = Types.DeepMutable<Schema.Schema.Type<typeof ToolStateRunning>>
265
-
266
- export const ToolStateCompleted = Schema.Struct({
267
- status: Schema.Literal("completed"),
268
- input: Schema.Record(Schema.String, Schema.Any),
269
- output: Schema.String,
270
- title: Schema.String,
271
- metadata: Schema.Record(Schema.String, Schema.Any),
272
- time: Schema.Struct({
273
- start: NonNegativeInt,
274
- end: NonNegativeInt,
275
- compacted: Schema.optional(NonNegativeInt),
276
- }),
277
- attachments: Schema.optional(Schema.Array(FilePart)),
278
- }).annotate({ identifier: "ToolStateCompleted" })
279
- export type ToolStateCompleted = Types.DeepMutable<Schema.Schema.Type<typeof ToolStateCompleted>>
280
-
281
- function truncateToolOutput(text: string, maxChars?: number) {
282
- if (!maxChars || text.length <= maxChars) return text
283
- const omitted = text.length - maxChars
284
- return `${text.slice(0, maxChars)}\n[Tool output truncated for compaction: omitted ${omitted} chars]`
285
- }
286
-
287
- export const ToolStateError = Schema.Struct({
288
- status: Schema.Literal("error"),
289
- input: Schema.Record(Schema.String, Schema.Any),
290
- error: Schema.String,
291
- metadata: Schema.optional(Schema.Record(Schema.String, Schema.Any)),
292
- time: Schema.Struct({
293
- start: NonNegativeInt,
294
- end: NonNegativeInt,
295
- }),
296
- }).annotate({ identifier: "ToolStateError" })
297
- export type ToolStateError = Types.DeepMutable<Schema.Schema.Type<typeof ToolStateError>>
298
-
299
- export const ToolState = Schema.Union([
300
- ToolStatePending,
301
- ToolStateRunning,
302
- ToolStateCompleted,
303
- ToolStateError,
304
- ]).annotate({
305
- discriminator: "status",
306
- identifier: "ToolState",
307
- })
308
- export type ToolState = ToolStatePending | ToolStateRunning | ToolStateCompleted | ToolStateError
309
-
310
- export const ToolPart = Schema.Struct({
311
- ...partBase,
312
- type: Schema.Literal("tool"),
313
- callID: Schema.String,
314
- tool: Schema.String,
315
- state: ToolState,
316
- metadata: Schema.optional(Schema.Record(Schema.String, Schema.Any)),
317
- }).annotate({ identifier: "ToolPart" })
318
- export type ToolPart = Omit<Types.DeepMutable<Schema.Schema.Type<typeof ToolPart>>, "state"> & {
319
- state: ToolState
320
- }
321
-
322
- const messageBase = {
323
- id: MessageID,
324
- sessionID: SessionID,
325
- }
326
-
327
- export const User = Schema.Struct({
328
- ...messageBase,
329
- role: Schema.Literal("user"),
330
- time: Schema.Struct({
331
- created: NonNegativeInt,
332
- }),
333
- format: Schema.optional(Format),
334
- summary: Schema.optional(
335
- Schema.Struct({
336
- title: Schema.optional(Schema.String),
337
- body: Schema.optional(Schema.String),
338
- diffs: Schema.Array(Snapshot.FileDiff),
339
- }),
340
- ),
341
- agent: Schema.String,
342
- model: Schema.Struct({
343
- providerID: ProviderID,
344
- modelID: ModelID,
345
- variant: Schema.optional(Schema.String),
346
- }),
347
- system: Schema.optional(Schema.String),
348
- tools: Schema.optional(Schema.Record(Schema.String, Schema.Boolean)),
349
- }).annotate({ identifier: "UserMessage" })
350
- export type User = Types.DeepMutable<Schema.Schema.Type<typeof User>>
351
-
352
- export const Part = Schema.Union([
353
- TextPart,
354
- SubtaskPart,
355
- ReasoningPart,
356
- FilePart,
357
- ToolPart,
358
- StepStartPart,
359
- StepFinishPart,
360
- SnapshotPart,
361
- PatchPart,
362
- AgentPart,
363
- RetryPart,
364
- CompactionPart,
365
- ]).annotate({ discriminator: "type", identifier: "Part" })
366
- export type Part =
367
- | TextPart
368
- | SubtaskPart
369
- | ReasoningPart
370
- | FilePart
371
- | ToolPart
372
- | StepStartPart
373
- | StepFinishPart
374
- | SnapshotPart
375
- | PatchPart
376
- | AgentPart
377
- | RetryPart
378
- | CompactionPart
379
-
380
- const AssistantErrorSchema = Schema.Union([
381
- ...MessageError.Shared,
382
- AbortedError.EffectSchema,
383
- StructuredOutputError.EffectSchema,
384
- ContextOverflowError.EffectSchema,
385
- APIError.EffectSchema,
386
- ]).annotate({ discriminator: "name" })
387
- type AssistantError = Schema.Schema.Type<typeof AssistantErrorSchema>
388
-
389
- // ── Prompt input schemas ─────────────────────────────────────────────────────
390
- //
391
- // Consumers of `SessionPrompt.PromptInput.parts` send part drafts without the
392
- // ambient IDs (`messageID`, `sessionID`) that live on stored parts, and may
393
- // omit `id` to let the server allocate one. These Schema-Struct variants
394
- // carry that shape so prompt decoding can accept drafts without stored IDs.
395
-
396
- export const TextPartInput = Schema.Struct({
397
- id: Schema.optional(PartID),
398
- type: Schema.Literal("text"),
399
- text: Schema.String,
400
- synthetic: Schema.optional(Schema.Boolean),
401
- ignored: Schema.optional(Schema.Boolean),
402
- time: Schema.optional(
403
- Schema.Struct({
404
- start: NonNegativeInt,
405
- end: Schema.optional(NonNegativeInt),
406
- }),
407
- ),
408
- metadata: Schema.optional(Schema.Record(Schema.String, Schema.Any)),
409
- }).annotate({ identifier: "TextPartInput" })
410
- export type TextPartInput = Types.DeepMutable<Schema.Schema.Type<typeof TextPartInput>>
411
-
412
- export const FilePartInput = Schema.Struct({
413
- id: Schema.optional(PartID),
414
- type: Schema.Literal("file"),
415
- mime: Schema.String,
416
- filename: Schema.optional(Schema.String),
417
- url: Schema.String,
418
- source: Schema.optional(FilePartSource),
419
- }).annotate({ identifier: "FilePartInput" })
420
- export type FilePartInput = Types.DeepMutable<Schema.Schema.Type<typeof FilePartInput>>
421
-
422
- export const AgentPartInput = Schema.Struct({
423
- id: Schema.optional(PartID),
424
- type: Schema.Literal("agent"),
425
- name: Schema.String,
426
- source: Schema.optional(
427
- Schema.Struct({
428
- value: Schema.String,
429
- start: NonNegativeInt,
430
- end: NonNegativeInt,
431
- }),
432
- ),
433
- }).annotate({ identifier: "AgentPartInput" })
434
- export type AgentPartInput = Types.DeepMutable<Schema.Schema.Type<typeof AgentPartInput>>
435
-
436
- export const SubtaskPartInput = Schema.Struct({
437
- id: Schema.optional(PartID),
438
- type: Schema.Literal("subtask"),
439
- prompt: Schema.String,
440
- description: Schema.String,
441
- agent: Schema.String,
442
- model: Schema.optional(
443
- Schema.Struct({
444
- providerID: ProviderID,
445
- modelID: ModelID,
446
- }),
447
- ),
448
- command: Schema.optional(Schema.String),
449
- }).annotate({ identifier: "SubtaskPartInput" })
450
- export type SubtaskPartInput = Types.DeepMutable<Schema.Schema.Type<typeof SubtaskPartInput>>
451
-
452
- export const Assistant = Schema.Struct({
453
- ...messageBase,
454
- role: Schema.Literal("assistant"),
455
- time: Schema.Struct({
456
- created: NonNegativeInt,
457
- completed: Schema.optional(NonNegativeInt),
458
- }),
459
- error: Schema.optional(AssistantErrorSchema),
460
- parentID: MessageID,
461
- modelID: ModelID,
462
- providerID: ProviderID,
463
- /**
464
- * @deprecated
465
- */
466
- mode: Schema.String,
467
- agent: Schema.String,
468
- path: Schema.Struct({
469
- cwd: Schema.String,
470
- root: Schema.String,
471
- }),
472
- summary: Schema.optional(Schema.Boolean),
473
- cost: Schema.Finite,
474
- tokens: Schema.Struct({
475
- total: Schema.optional(Schema.Finite),
476
- input: Schema.Finite,
477
- output: Schema.Finite,
478
- reasoning: Schema.Finite,
479
- cache: Schema.Struct({
480
- read: Schema.Finite,
481
- write: Schema.Finite,
482
- }),
483
- }),
484
- structured: Schema.optional(Schema.Any),
485
- variant: Schema.optional(Schema.String),
486
- finish: Schema.optional(Schema.String),
487
- }).annotate({ identifier: "AssistantMessage" })
488
- export type Assistant = Omit<Types.DeepMutable<Schema.Schema.Type<typeof Assistant>>, "error"> & {
489
- error?: AssistantError
490
- }
491
-
492
- export const Info = Schema.Union([User, Assistant]).annotate({ discriminator: "role", identifier: "Message" })
493
- export type Info = User | Assistant
494
-
495
- const UpdatedEventSchema = Schema.Struct({
496
- sessionID: SessionID,
497
- info: Info,
498
- })
499
-
500
- const RemovedEventSchema = Schema.Struct({
501
- sessionID: SessionID,
502
- messageID: MessageID,
503
- })
504
-
505
- const PartUpdatedEventSchema = Schema.Struct({
506
- sessionID: SessionID,
507
- part: Part,
508
- time: NonNegativeInt,
509
- })
510
-
511
- const PartRemovedEventSchema = Schema.Struct({
512
- sessionID: SessionID,
513
- messageID: MessageID,
514
- partID: PartID,
515
- })
516
-
517
- export const Event = {
518
- Updated: SyncEvent.define({
519
- type: "message.updated",
520
- version: 1,
521
- aggregate: "sessionID",
522
- schema: UpdatedEventSchema,
523
- }),
524
- Removed: SyncEvent.define({
525
- type: "message.removed",
526
- version: 1,
527
- aggregate: "sessionID",
528
- schema: RemovedEventSchema,
529
- }),
530
- PartUpdated: SyncEvent.define({
531
- type: "message.part.updated",
532
- version: 1,
533
- aggregate: "sessionID",
534
- schema: PartUpdatedEventSchema,
535
- }),
536
- PartDelta: BusEvent.define(
537
- "message.part.delta",
538
- Schema.Struct({
539
- sessionID: SessionID,
540
- messageID: MessageID,
541
- partID: PartID,
542
- field: Schema.String,
543
- delta: Schema.String,
544
- }),
545
- ),
546
- PartRemoved: SyncEvent.define({
547
- type: "message.part.removed",
548
- version: 1,
549
- aggregate: "sessionID",
550
- schema: PartRemovedEventSchema,
551
- }),
552
- }
553
-
554
- export const WithParts = Schema.Struct({
555
- info: Info,
556
- parts: Schema.Array(Part),
557
- })
558
- export type WithParts = {
559
- info: Info
560
- parts: Part[]
561
- }
562
-
563
- const Cursor = Schema.Struct({
564
- id: MessageID,
565
- time: Schema.Finite.check(Schema.isGreaterThanOrEqualTo(0)),
566
- })
567
- type Cursor = typeof Cursor.Type
568
-
569
- const decodeCursor = Schema.decodeUnknownSync(Cursor)
570
-
571
- export const cursor = {
572
- encode(input: Cursor) {
573
- return Buffer.from(JSON.stringify(input)).toString("base64url")
574
- },
575
- decode(input: string) {
576
- return decodeCursor(JSON.parse(Buffer.from(input, "base64url").toString("utf8")))
577
- },
578
- }
579
-
580
- const info = (row: typeof MessageTable.$inferSelect) =>
581
- ({
582
- ...row.data,
583
- id: row.id,
584
- sessionID: row.session_id,
585
- }) as Info
586
-
587
- const part = (row: typeof PartTable.$inferSelect) =>
588
- ({
589
- ...row.data,
590
- id: row.id,
591
- sessionID: row.session_id,
592
- messageID: row.message_id,
593
- }) as Part
594
-
595
- const older = (row: Cursor) =>
596
- or(lt(MessageTable.time_created, row.time), and(eq(MessageTable.time_created, row.time), lt(MessageTable.id, row.id)))
597
-
598
- function hydrate(rows: (typeof MessageTable.$inferSelect)[]) {
599
- const ids = rows.map((row) => row.id)
600
- const partByMessage = new Map<string, Part[]>()
601
- if (ids.length > 0) {
602
- const partRows = Database.use((db) =>
603
- db
604
- .select()
605
- .from(PartTable)
606
- .where(inArray(PartTable.message_id, ids))
607
- .orderBy(PartTable.message_id, PartTable.id)
608
- .all(),
609
- )
610
- for (const row of partRows) {
611
- const next = part(row)
612
- const list = partByMessage.get(row.message_id)
613
- if (list) list.push(next)
614
- else partByMessage.set(row.message_id, [next])
615
- }
616
- }
617
-
618
- return rows.map((row) => ({
619
- info: info(row),
620
- parts: partByMessage.get(row.id) ?? [],
621
- }))
622
- }
623
-
624
- function providerMeta(metadata: Record<string, any> | undefined) {
625
- if (!metadata) return undefined
626
- const { providerExecuted: _, ...rest } = metadata
627
- return Object.keys(rest).length > 0 ? rest : undefined
628
- }
629
-
630
- export const toModelMessagesEffect = Effect.fnUntraced(function* (
631
- input: WithParts[],
632
- model: Provider.Model,
633
- options?: { stripMedia?: boolean; toolOutputMaxChars?: number },
634
- ) {
635
- const result: UIMessage[] = []
636
- const toolNames = new Set<string>()
637
- // Track media from tool results that need to be injected as user messages
638
- // for providers that don't support that media type in tool results.
639
- //
640
- // OpenAI-compatible APIs only support string content in tool results, so we need
641
- // to extract media and inject as user messages. Some SDKs only support a subset
642
- // of media in tool results; e.g. Bedrock supports images but not PDFs there.
643
- //
644
- // Only apply this workaround if the model actually supports that media input -
645
- // otherwise unsupportedParts() will turn it into a user-visible error.
646
- const supportsMediaInToolResult = (attachment: { mime: string }) => {
647
- if (model.api.npm === "@ai-sdk/anthropic") return true
648
- if (model.api.npm === "@ai-sdk/openai") return true
649
- if (model.api.npm === "@ai-sdk/amazon-bedrock") return attachment.mime.startsWith("image/")
650
- if (model.api.npm === "@ai-sdk/google-vertex/anthropic") return true
651
- if (model.api.npm === "@ai-sdk/google") {
652
- const id = model.api.id.toLowerCase()
653
- return id.includes("gemini-3") && !id.includes("gemini-2")
654
- }
655
- return false
656
- }
657
-
658
- const toModelOutput = (options: { toolCallId: string; input: unknown; output: unknown }) => {
659
- const output = options.output
660
- if (typeof output === "string") {
661
- return { type: "text", value: output }
662
- }
663
-
664
- if (typeof output === "object") {
665
- const outputObject = output as {
666
- text: string
667
- attachments?: Array<{ mime: string; url: string }>
668
- }
669
- const attachments = (outputObject.attachments ?? []).filter((attachment) => {
670
- return attachment.url.startsWith("data:") && attachment.url.includes(",")
671
- })
672
-
673
- return {
674
- type: "content",
675
- value: [
676
- ...(outputObject.text ? [{ type: "text", text: outputObject.text }] : []),
677
- ...attachments.map((attachment) => ({
678
- type: "media",
679
- mediaType: attachment.mime,
680
- data: iife(() => {
681
- const commaIndex = attachment.url.indexOf(",")
682
- return commaIndex === -1 ? attachment.url : attachment.url.slice(commaIndex + 1)
683
- }),
684
- })),
685
- ],
686
- }
687
- }
688
-
689
- return { type: "json", value: output as never }
690
- }
691
-
692
- for (const msg of input) {
693
- if (msg.parts.length === 0) continue
694
-
695
- if (msg.info.role === "user") {
696
- const userMessage: UIMessage = {
697
- id: msg.info.id,
698
- role: "user",
699
- parts: [],
700
- }
701
- for (const part of msg.parts) {
702
- // User message parts should never be empty
703
- if (part.type === "text" && !part.ignored && part.text !== "")
704
- userMessage.parts.push({
705
- type: "text",
706
- text: part.text,
707
- })
708
- // text/plain and directory files are converted into text parts, ignore them
709
- if (part.type === "file" && part.mime !== "text/plain" && part.mime !== "application/x-directory") {
710
- if (options?.stripMedia && isMedia(part.mime)) {
711
- userMessage.parts.push({
712
- type: "text",
713
- text: `[Attached ${part.mime}: ${part.filename ?? "file"}]`,
714
- })
715
- } else {
716
- userMessage.parts.push({
717
- type: "file",
718
- url: part.url,
719
- mediaType: part.mime,
720
- filename: part.filename,
721
- })
722
- }
723
- }
724
-
725
- if (part.type === "compaction") {
726
- userMessage.parts.push({
727
- type: "text",
728
- text: "What did we do so far?",
729
- })
730
- }
731
- if (part.type === "subtask") {
732
- userMessage.parts.push({
733
- type: "text",
734
- text: "The following tool was executed by the user",
735
- })
736
- }
737
- }
738
- if (userMessage.parts.length > 0) result.push(userMessage)
739
- }
740
-
741
- if (msg.info.role === "assistant") {
742
- const differentModel = `${model.providerID}/${model.id}` !== `${msg.info.providerID}/${msg.info.modelID}`
743
- const media: Array<{ mime: string; url: string; filename?: string }> = []
744
-
745
- if (
746
- msg.info.error &&
747
- !(
748
- AbortedError.isInstance(msg.info.error) &&
749
- msg.parts.some((part) => part.type !== "step-start" && part.type !== "reasoning")
750
- )
751
- ) {
752
- continue
753
- }
754
- const assistantMessage: UIMessage = {
755
- id: msg.info.id,
756
- role: "assistant",
757
- parts: [],
758
- }
759
- // Anthropic adaptive thinking can persist assistant turns like:
760
- // step-start, reasoning(signature), text(""), step-start,
761
- // reasoning(signature). The empty text part is a structural separator,
762
- // but it does not carry the signature metadata itself. Dropping it shifts
763
- // signed thinking positions after step-start splitting/provider regrouping;
764
- // keeping it as "" is filtered by the AI SDK and rejected by Anthropic.
765
- // It is unclear whether this shape originates in our stream processing,
766
- // a proxy, or a lower-level library, but preserving a non-empty separator
767
- // here is the only safe replay point we have.
768
- // Use a single space so the separator survives replay without changing
769
- // the neighboring signed reasoning blocks.
770
- const hasSignedReasoning = msg.parts.some((part) => {
771
- if (part.type !== "reasoning") return false
772
- return part.metadata?.anthropic?.signature != null
773
- })
774
- for (const part of msg.parts) {
775
- if (part.type === "text") {
776
- const text = part.text === "" && hasSignedReasoning ? " " : part.text
777
- assistantMessage.parts.push({
778
- type: "text",
779
- text,
780
- ...(differentModel ? {} : { providerMetadata: part.metadata }),
781
- })
782
- }
783
- if (part.type === "step-start")
784
- assistantMessage.parts.push({
785
- type: "step-start",
786
- })
787
- if (part.type === "tool") {
788
- toolNames.add(part.tool)
789
- if (part.state.status === "completed") {
790
- const outputText = part.state.time.compacted
791
- ? "[Old tool result content cleared]"
792
- : truncateToolOutput(part.state.output, options?.toolOutputMaxChars)
793
- const attachments = part.state.time.compacted || options?.stripMedia ? [] : (part.state.attachments ?? [])
794
-
795
- // For providers that don't support media in tool results, extract media files
796
- // (images, PDFs) to be sent as a separate user message
797
- const mediaAttachments = attachments.filter((a) => isMedia(a.mime))
798
- const extractedMedia = mediaAttachments.filter((a) => !supportsMediaInToolResult(a))
799
- if (extractedMedia.length > 0) {
800
- media.push(...extractedMedia)
801
- }
802
- const finalAttachments = attachments.filter((a) => !isMedia(a.mime) || supportsMediaInToolResult(a))
803
-
804
- const output =
805
- finalAttachments.length > 0
806
- ? {
807
- text: outputText,
808
- attachments: finalAttachments,
809
- }
810
- : outputText
811
-
812
- assistantMessage.parts.push({
813
- type: ("tool-" + part.tool) as `tool-${string}`,
814
- state: "output-available",
815
- toolCallId: part.callID,
816
- input: part.state.input,
817
- output,
818
- ...(part.metadata?.providerExecuted ? { providerExecuted: true } : {}),
819
- ...(differentModel ? {} : { callProviderMetadata: providerMeta(part.metadata) }),
820
- })
821
- }
822
- if (part.state.status === "error") {
823
- const output = part.state.metadata?.interrupted === true ? part.state.metadata.output : undefined
824
- if (typeof output === "string") {
825
- assistantMessage.parts.push({
826
- type: ("tool-" + part.tool) as `tool-${string}`,
827
- state: "output-available",
828
- toolCallId: part.callID,
829
- input: part.state.input,
830
- output,
831
- ...(part.metadata?.providerExecuted ? { providerExecuted: true } : {}),
832
- ...(differentModel ? {} : { callProviderMetadata: providerMeta(part.metadata) }),
833
- })
834
- } else {
835
- assistantMessage.parts.push({
836
- type: ("tool-" + part.tool) as `tool-${string}`,
837
- state: "output-error",
838
- toolCallId: part.callID,
839
- input: part.state.input,
840
- errorText: part.state.error,
841
- ...(part.metadata?.providerExecuted ? { providerExecuted: true } : {}),
842
- ...(differentModel ? {} : { callProviderMetadata: providerMeta(part.metadata) }),
843
- })
844
- }
845
- }
846
- // Handle pending/running tool calls to prevent dangling tool_use blocks
847
- // Anthropic/Claude APIs require every tool_use to have a corresponding tool_result
848
- if (part.state.status === "pending" || part.state.status === "running")
849
- assistantMessage.parts.push({
850
- type: ("tool-" + part.tool) as `tool-${string}`,
851
- state: "output-error",
852
- toolCallId: part.callID,
853
- input: part.state.input,
854
- errorText: "[Tool execution was interrupted]",
855
- ...(part.metadata?.providerExecuted ? { providerExecuted: true } : {}),
856
- ...(differentModel ? {} : { callProviderMetadata: providerMeta(part.metadata) }),
857
- })
858
- }
859
- if (part.type === "reasoning") {
860
- if (differentModel) {
861
- if (part.text.trim().length > 0)
862
- assistantMessage.parts.push({
863
- type: "text",
864
- text: part.text,
865
- })
866
- continue
867
- }
868
- assistantMessage.parts.push({
869
- type: "reasoning",
870
- text: part.text,
871
- providerMetadata: part.metadata,
872
- })
873
- }
874
- }
875
- if (assistantMessage.parts.length > 0) {
876
- result.push(assistantMessage)
877
- // Inject pending media as a user message for providers that don't support
878
- // media (images, PDFs) in tool results
879
- if (media.length > 0) {
880
- result.push({
881
- id: MessageID.ascending(),
882
- role: "user",
883
- parts: [
884
- {
885
- type: "text" as const,
886
- text: SYNTHETIC_ATTACHMENT_PROMPT,
887
- },
888
- ...media.map((attachment) => ({
889
- type: "file" as const,
890
- url: attachment.url,
891
- mediaType: attachment.mime,
892
- filename: attachment.filename,
893
- })),
894
- ],
895
- })
896
- }
897
- }
898
- }
899
- }
900
-
901
- const tools = Object.fromEntries(Array.from(toolNames).map((toolName) => [toolName, { toModelOutput }]))
902
-
903
- return yield* Effect.promise(() =>
904
- convertToModelMessages(
905
- result.filter((msg) => msg.parts.some((part) => part.type !== "step-start")),
906
- {
907
- //@ts-expect-error (convertToModelMessages expects a ToolSet but only actually needs tools[name]?.toModelOutput)
908
- tools,
909
- },
910
- ),
911
- )
912
- })
913
-
914
- export function toModelMessages(
915
- input: WithParts[],
916
- model: Provider.Model,
917
- options?: { stripMedia?: boolean; toolOutputMaxChars?: number },
918
- ): Promise<ModelMessage[]> {
919
- return Effect.runPromise(toModelMessagesEffect(input, model, options).pipe(Effect.provide(EffectLogger.layer)))
920
- }
921
-
922
- export const page = Effect.fn("MessageV2.page")(function* (input: {
923
- sessionID: SessionID
924
- limit: number
925
- before?: string
926
- }) {
927
- const before = input.before ? cursor.decode(input.before) : undefined
928
- const where = before
929
- ? and(eq(MessageTable.session_id, input.sessionID), older(before))
930
- : eq(MessageTable.session_id, input.sessionID)
931
- const rows = Database.use((db) =>
932
- db
933
- .select()
934
- .from(MessageTable)
935
- .where(where)
936
- .orderBy(desc(MessageTable.time_created), desc(MessageTable.id))
937
- .limit(input.limit + 1)
938
- .all(),
939
- )
940
- if (rows.length === 0) {
941
- const row = Database.use((db) =>
942
- db.select({ id: SessionTable.id }).from(SessionTable).where(eq(SessionTable.id, input.sessionID)).get(),
943
- )
944
- if (!row) return yield* new NotFoundError({ message: `Session not found: ${input.sessionID}` })
945
- return {
946
- items: [] as WithParts[],
947
- more: false,
948
- }
949
- }
950
-
951
- const more = rows.length > input.limit
952
- const slice = more ? rows.slice(0, input.limit) : rows
953
- const items = hydrate(slice)
954
- items.reverse()
955
- const tail = slice.at(-1)
956
- return {
957
- items,
958
- more,
959
- cursor: more && tail ? cursor.encode({ id: tail.id, time: tail.time_created }) : undefined,
960
- }
961
- })
962
-
963
- export function* stream(sessionID: SessionID) {
964
- const size = 50
965
- let before: string | undefined
966
- while (true) {
967
- const next = Effect.runSync(
968
- page({ sessionID, limit: size, before }).pipe(
969
- Effect.catchIf(NotFoundError.isInstance, () =>
970
- Effect.succeed({ items: [] as WithParts[], more: false, cursor: undefined }),
971
- ),
972
- ),
973
- )
974
- if (next.items.length === 0) break
975
- for (let i = next.items.length - 1; i >= 0; i--) {
976
- yield next.items[i]
977
- }
978
- if (!next.more || !next.cursor) break
979
- before = next.cursor
980
- }
981
- }
982
-
983
- export function parts(message_id: MessageID) {
984
- const rows = Database.use((db) =>
985
- db.select().from(PartTable).where(eq(PartTable.message_id, message_id)).orderBy(PartTable.id).all(),
986
- )
987
- return rows.map(
988
- (row) =>
989
- ({
990
- ...row.data,
991
- id: row.id,
992
- sessionID: row.session_id,
993
- messageID: row.message_id,
994
- }) as Part,
995
- )
996
- }
997
-
998
- export const get = Effect.fn("MessageV2.get")(function* (input: { sessionID: SessionID; messageID: MessageID }) {
999
- const row = Database.use((db) =>
1000
- db
1001
- .select()
1002
- .from(MessageTable)
1003
- .where(and(eq(MessageTable.id, input.messageID), eq(MessageTable.session_id, input.sessionID)))
1004
- .get(),
1005
- )
1006
- if (!row) return yield* new NotFoundError({ message: `Message not found: ${input.messageID}` })
1007
- return {
1008
- info: info(row),
1009
- parts: parts(input.messageID),
1010
- }
1011
- })
1012
-
1013
- export function filterCompacted(msgs: Iterable<WithParts>) {
1014
- const result = [] as WithParts[]
1015
- const completed = new Set<string>()
1016
- let retain: MessageID | undefined
1017
- for (const msg of msgs) {
1018
- result.push(msg)
1019
- if (retain) {
1020
- if (msg.info.id === retain) break
1021
- continue
1022
- }
1023
- if (msg.info.role === "user" && completed.has(msg.info.id)) {
1024
- const part = msg.parts.find((item): item is CompactionPart => item.type === "compaction")
1025
- if (!part) continue
1026
- if (!part.tail_start_id) break
1027
- retain = part.tail_start_id
1028
- if (msg.info.id === retain) break
1029
- continue
1030
- }
1031
- if (msg.info.role === "user" && completed.has(msg.info.id) && msg.parts.some((part) => part.type === "compaction"))
1032
- break
1033
- if (msg.info.role === "assistant" && msg.info.summary && msg.info.finish && !msg.info.error)
1034
- completed.add(msg.info.parentID)
1035
- }
1036
- result.reverse()
1037
- const compactionIndex = result.findLastIndex(
1038
- (msg) =>
1039
- msg.info.role === "user" &&
1040
- msg.parts.some((item): item is CompactionPart => item.type === "compaction" && item.tail_start_id !== undefined),
1041
- )
1042
- const compaction = result[compactionIndex]
1043
- const part = compaction?.parts.find(
1044
- (item): item is CompactionPart => item.type === "compaction" && item.tail_start_id !== undefined,
1045
- )
1046
- const summaryIndex = compaction
1047
- ? result.findIndex(
1048
- (msg, index) =>
1049
- index > compactionIndex &&
1050
- msg.info.role === "assistant" &&
1051
- msg.info.summary &&
1052
- msg.info.parentID === compaction.info.id,
1053
- )
1054
- : -1
1055
- const tailIndex = part?.tail_start_id ? result.findIndex((msg) => msg.info.id === part.tail_start_id) : -1
1056
- if (tailIndex >= 0 && tailIndex < compactionIndex && summaryIndex > compactionIndex) {
1057
- return [
1058
- ...result.slice(compactionIndex, summaryIndex + 1),
1059
- ...result.slice(tailIndex, compactionIndex),
1060
- ...result.slice(summaryIndex + 1),
1061
- ]
1062
- }
1063
- return result
1064
- }
1065
-
1066
- export const filterCompactedEffect = Effect.fnUntraced(function* (sessionID: SessionID) {
1067
- return filterCompacted(stream(sessionID))
1068
- })
1069
-
1070
- // filterCompacted reorders messages for model consumption
1071
- // ([compaction-user, summary, ...retained tail..., continue-user]), so array
1072
- // position is not chronological. Derive each binding by max id (MessageID
1073
- // is monotonic via MessageID.ascending) so a pre-compaction overflowing tail
1074
- // assistant doesn't get mistaken for the most recent turn. tasks are
1075
- // compaction/subtask parts attached to user messages newer than the latest
1076
- // finished assistant — i.e. unprocessed work.
1077
- export function latest(msgs: WithParts[]) {
1078
- let user: User | undefined
1079
- let assistant: Assistant | undefined
1080
- let finished: Assistant | undefined
1081
- for (const msg of msgs) {
1082
- const info = msg.info
1083
- if (info.role === "user" && (!user || info.id > user.id)) user = info
1084
- if (info.role === "assistant" && (!assistant || info.id > assistant.id)) assistant = info
1085
- if (info.role === "assistant" && info.finish && (!finished || info.id > finished.id)) finished = info
1086
- }
1087
- const tasks = msgs.flatMap((m) =>
1088
- finished && m.info.id <= finished.id
1089
- ? []
1090
- : m.parts.filter((p): p is CompactionPart | SubtaskPart => p.type === "compaction" || p.type === "subtask"),
1091
- )
1092
- return { user, assistant, finished, tasks }
1093
- }
1094
-
1095
- export function fromError(
1096
- e: unknown,
1097
- ctx: { providerID: ProviderID; aborted?: boolean },
1098
- ): NonNullable<Assistant["error"]> {
1099
- switch (true) {
1100
- case e instanceof DOMException && e.name === "AbortError":
1101
- return new AbortedError(
1102
- { message: e.message },
1103
- {
1104
- cause: e,
1105
- },
1106
- ).toObject()
1107
- case OutputLengthError.isInstance(e):
1108
- return e
1109
- case LoadAPIKeyError.isInstance(e):
1110
- return new AuthError(
1111
- {
1112
- providerID: ctx.providerID,
1113
- message: e.message,
1114
- },
1115
- { cause: e },
1116
- ).toObject()
1117
- case (e as SystemError)?.code === "ECONNRESET":
1118
- return new APIError(
1119
- {
1120
- message: "Connection reset by server",
1121
- isRetryable: true,
1122
- metadata: {
1123
- code: (e as SystemError).code ?? "",
1124
- syscall: (e as SystemError).syscall ?? "",
1125
- message: (e as SystemError).message ?? "",
1126
- },
1127
- },
1128
- { cause: e },
1129
- ).toObject()
1130
- case e instanceof Error && (e as FetchDecompressionError).code === "ZlibError":
1131
- if (ctx.aborted) {
1132
- return new AbortedError({ message: e.message }, { cause: e }).toObject()
1133
- }
1134
- return new APIError(
1135
- {
1136
- message: "Response decompression failed",
1137
- isRetryable: true,
1138
- metadata: {
1139
- code: (e as FetchDecompressionError).code,
1140
- message: e.message,
1141
- },
1142
- },
1143
- { cause: e },
1144
- ).toObject()
1145
- case APICallError.isInstance(e):
1146
- const parsed = ProviderError.parseAPICallError({
1147
- providerID: ctx.providerID,
1148
- error: e,
1149
- })
1150
- if (parsed.type === "context_overflow") {
1151
- return new ContextOverflowError(
1152
- {
1153
- message: parsed.message,
1154
- responseBody: parsed.responseBody,
1155
- },
1156
- { cause: e },
1157
- ).toObject()
1158
- }
1159
-
1160
- return new APIError(
1161
- {
1162
- message: parsed.message,
1163
- statusCode: parsed.statusCode,
1164
- isRetryable: parsed.isRetryable,
1165
- responseHeaders: parsed.responseHeaders,
1166
- responseBody: parsed.responseBody,
1167
- metadata: parsed.metadata,
1168
- },
1169
- { cause: e },
1170
- ).toObject()
1171
- case e instanceof Error:
1172
- return new NamedError.Unknown({ message: errorMessage(e) }, { cause: e }).toObject()
1173
- default:
1174
- try {
1175
- const parsed = ProviderError.parseStreamError(e)
1176
- if (parsed) {
1177
- if (parsed.type === "context_overflow") {
1178
- return new ContextOverflowError(
1179
- {
1180
- message: parsed.message,
1181
- responseBody: parsed.responseBody,
1182
- },
1183
- { cause: e },
1184
- ).toObject()
1185
- }
1186
- return new APIError(
1187
- {
1188
- message: parsed.message,
1189
- isRetryable: parsed.isRetryable,
1190
- responseBody: parsed.responseBody,
1191
- },
1192
- {
1193
- cause: e,
1194
- },
1195
- ).toObject()
1196
- }
1197
- } catch {}
1198
- return new NamedError.Unknown({ message: JSON.stringify(e) }, { cause: e }).toObject()
1199
- }
1200
- }
1201
-
1202
- export * as MessageV2 from "./message-v2"
1
+ import { BusEvent } from "@/bus/bus-event"
2
+ import { SessionID, MessageID, PartID } from "./schema"
3
+ import { NamedError } from "@opencode-ai/core/util/error"
4
+ import { APICallError, convertToModelMessages, LoadAPIKeyError, type ModelMessage, type UIMessage } from "ai"
5
+ import { LSP } from "@/lsp/lsp"
6
+ import { Snapshot } from "@/snapshot"
7
+ import { SyncEvent } from "../sync"
8
+ import { Database } from "@/storage/db"
9
+ import { NotFoundError } from "@/storage/storage"
10
+ import { and } from "drizzle-orm"
11
+ import { desc } from "drizzle-orm"
12
+ import { eq } from "drizzle-orm"
13
+ import { inArray } from "drizzle-orm"
14
+ import { lt } from "drizzle-orm"
15
+ import { or } from "drizzle-orm"
16
+ import { MessageTable, PartTable, SessionTable } from "./session.sql"
17
+ import * as ProviderError from "@/provider/error"
18
+ import { iife } from "@/util/iife"
19
+ import { errorMessage } from "@/util/error"
20
+ import { isMedia } from "@/util/media"
21
+ import type { SystemError } from "bun"
22
+ import type { Provider } from "@/provider/provider"
23
+ import { ModelID, ProviderID } from "@/provider/schema"
24
+ import { Effect, Schema, Types } from "effect"
25
+ import { NonNegativeInt } from "@opencode-ai/core/schema"
26
+ import * as EffectLogger from "@opencode-ai/core/effect/logger"
27
+ import { MessageError } from "./message-error"
28
+ import { AuthError, OutputLengthError } from "./message-error"
29
+ export { AuthError, OutputLengthError } from "./message-error"
30
+
31
+ /** Error shape thrown by Bun's fetch() when gzip/br decompression fails mid-stream */
32
+ interface FetchDecompressionError extends Error {
33
+ code: "ZlibError"
34
+ errno: number
35
+ path: string
36
+ }
37
+
38
+ export const SYNTHETIC_ATTACHMENT_PROMPT = "Attached media from tool result:"
39
+ export { isMedia }
40
+
41
+ export const AbortedError = NamedError.create("MessageAbortedError", { message: Schema.String })
42
+ export const StructuredOutputError = NamedError.create("StructuredOutputError", {
43
+ message: Schema.String,
44
+ retries: NonNegativeInt,
45
+ })
46
+ export const APIError = NamedError.create("APIError", {
47
+ message: Schema.String,
48
+ statusCode: Schema.optional(NonNegativeInt),
49
+ isRetryable: Schema.Boolean,
50
+ responseHeaders: Schema.optional(Schema.Record(Schema.String, Schema.String)),
51
+ responseBody: Schema.optional(Schema.String),
52
+ metadata: Schema.optional(Schema.Record(Schema.String, Schema.String)),
53
+ })
54
+ export type APIError = Schema.Schema.Type<typeof APIError.Schema>
55
+ export const ContextOverflowError = NamedError.create("ContextOverflowError", {
56
+ message: Schema.String,
57
+ responseBody: Schema.optional(Schema.String),
58
+ })
59
+
60
+ export class OutputFormatText extends Schema.Class<OutputFormatText>("OutputFormatText")({
61
+ type: Schema.Literal("text"),
62
+ }) {}
63
+
64
+ export class OutputFormatJsonSchema extends Schema.Class<OutputFormatJsonSchema>("OutputFormatJsonSchema")({
65
+ type: Schema.Literal("json_schema"),
66
+ schema: Schema.Record(Schema.String, Schema.Any).annotate({ identifier: "JSONSchema" }),
67
+ retryCount: NonNegativeInt.pipe(Schema.optional, Schema.withDecodingDefault(Effect.succeed(2))),
68
+ }) {}
69
+
70
+ export const Format = Schema.Union([OutputFormatText, OutputFormatJsonSchema]).annotate({
71
+ discriminator: "type",
72
+ identifier: "OutputFormat",
73
+ })
74
+ export type OutputFormat = Schema.Schema.Type<typeof Format>
75
+
76
+ const partBase = {
77
+ id: PartID,
78
+ sessionID: SessionID,
79
+ messageID: MessageID,
80
+ }
81
+
82
+ export const SnapshotPart = Schema.Struct({
83
+ ...partBase,
84
+ type: Schema.Literal("snapshot"),
85
+ snapshot: Schema.String,
86
+ }).annotate({ identifier: "SnapshotPart" })
87
+ export type SnapshotPart = Types.DeepMutable<Schema.Schema.Type<typeof SnapshotPart>>
88
+
89
+ export const PatchPart = Schema.Struct({
90
+ ...partBase,
91
+ type: Schema.Literal("patch"),
92
+ hash: Schema.String,
93
+ files: Schema.Array(Schema.String),
94
+ }).annotate({ identifier: "PatchPart" })
95
+ export type PatchPart = Types.DeepMutable<Schema.Schema.Type<typeof PatchPart>>
96
+
97
+ export const TextPart = Schema.Struct({
98
+ ...partBase,
99
+ type: Schema.Literal("text"),
100
+ text: Schema.String,
101
+ synthetic: Schema.optional(Schema.Boolean),
102
+ ignored: Schema.optional(Schema.Boolean),
103
+ time: Schema.optional(
104
+ Schema.Struct({
105
+ start: NonNegativeInt,
106
+ end: Schema.optional(NonNegativeInt),
107
+ }),
108
+ ),
109
+ metadata: Schema.optional(Schema.Record(Schema.String, Schema.Any)),
110
+ }).annotate({ identifier: "TextPart" })
111
+ export type TextPart = Types.DeepMutable<Schema.Schema.Type<typeof TextPart>>
112
+
113
+ export const ReasoningPart = Schema.Struct({
114
+ ...partBase,
115
+ type: Schema.Literal("reasoning"),
116
+ text: Schema.String,
117
+ metadata: Schema.optional(Schema.Record(Schema.String, Schema.Any)),
118
+ time: Schema.Struct({
119
+ start: NonNegativeInt,
120
+ end: Schema.optional(NonNegativeInt),
121
+ }),
122
+ }).annotate({ identifier: "ReasoningPart" })
123
+ export type ReasoningPart = Types.DeepMutable<Schema.Schema.Type<typeof ReasoningPart>>
124
+
125
+ const filePartSourceBase = {
126
+ text: Schema.Struct({
127
+ value: Schema.String,
128
+ start: Schema.Finite,
129
+ end: Schema.Finite,
130
+ }).annotate({ identifier: "FilePartSourceText" }),
131
+ }
132
+
133
+ export const FileSource = Schema.Struct({
134
+ ...filePartSourceBase,
135
+ type: Schema.Literal("file"),
136
+ path: Schema.String,
137
+ }).annotate({ identifier: "FileSource" })
138
+
139
+ export const SymbolSource = Schema.Struct({
140
+ ...filePartSourceBase,
141
+ type: Schema.Literal("symbol"),
142
+ path: Schema.String,
143
+ range: LSP.Range,
144
+ name: Schema.String,
145
+ kind: NonNegativeInt,
146
+ }).annotate({ identifier: "SymbolSource" })
147
+
148
+ export const ResourceSource = Schema.Struct({
149
+ ...filePartSourceBase,
150
+ type: Schema.Literal("resource"),
151
+ clientName: Schema.String,
152
+ uri: Schema.String,
153
+ }).annotate({ identifier: "ResourceSource" })
154
+
155
+ export const FilePartSource = Schema.Union([FileSource, SymbolSource, ResourceSource]).annotate({
156
+ discriminator: "type",
157
+ identifier: "FilePartSource",
158
+ })
159
+
160
+ export const FilePart = Schema.Struct({
161
+ ...partBase,
162
+ type: Schema.Literal("file"),
163
+ mime: Schema.String,
164
+ filename: Schema.optional(Schema.String),
165
+ url: Schema.String,
166
+ source: Schema.optional(FilePartSource),
167
+ }).annotate({ identifier: "FilePart" })
168
+ export type FilePart = Types.DeepMutable<Schema.Schema.Type<typeof FilePart>>
169
+
170
+ export const AgentPart = Schema.Struct({
171
+ ...partBase,
172
+ type: Schema.Literal("agent"),
173
+ name: Schema.String,
174
+ source: Schema.optional(
175
+ Schema.Struct({
176
+ value: Schema.String,
177
+ start: NonNegativeInt,
178
+ end: NonNegativeInt,
179
+ }),
180
+ ),
181
+ }).annotate({ identifier: "AgentPart" })
182
+ export type AgentPart = Types.DeepMutable<Schema.Schema.Type<typeof AgentPart>>
183
+
184
+ export const CompactionPart = Schema.Struct({
185
+ ...partBase,
186
+ type: Schema.Literal("compaction"),
187
+ auto: Schema.Boolean,
188
+ overflow: Schema.optional(Schema.Boolean),
189
+ tail_start_id: Schema.optional(MessageID),
190
+ }).annotate({ identifier: "CompactionPart" })
191
+ export type CompactionPart = Types.DeepMutable<Schema.Schema.Type<typeof CompactionPart>>
192
+
193
+ export const SubtaskPart = Schema.Struct({
194
+ ...partBase,
195
+ type: Schema.Literal("subtask"),
196
+ prompt: Schema.String,
197
+ description: Schema.String,
198
+ agent: Schema.String,
199
+ model: Schema.optional(
200
+ Schema.Struct({
201
+ providerID: ProviderID,
202
+ modelID: ModelID,
203
+ }),
204
+ ),
205
+ command: Schema.optional(Schema.String),
206
+ }).annotate({ identifier: "SubtaskPart" })
207
+ export type SubtaskPart = Types.DeepMutable<Schema.Schema.Type<typeof SubtaskPart>>
208
+
209
+ export const RetryPart = Schema.Struct({
210
+ ...partBase,
211
+ type: Schema.Literal("retry"),
212
+ attempt: NonNegativeInt,
213
+ error: APIError.EffectSchema,
214
+ time: Schema.Struct({
215
+ created: NonNegativeInt,
216
+ }),
217
+ }).annotate({ identifier: "RetryPart" })
218
+ export type RetryPart = Omit<Types.DeepMutable<Schema.Schema.Type<typeof RetryPart>>, "error"> & {
219
+ error: APIError
220
+ }
221
+
222
+ export const StepStartPart = Schema.Struct({
223
+ ...partBase,
224
+ type: Schema.Literal("step-start"),
225
+ snapshot: Schema.optional(Schema.String),
226
+ }).annotate({ identifier: "StepStartPart" })
227
+ export type StepStartPart = Types.DeepMutable<Schema.Schema.Type<typeof StepStartPart>>
228
+
229
+ export const StepFinishPart = Schema.Struct({
230
+ ...partBase,
231
+ type: Schema.Literal("step-finish"),
232
+ reason: Schema.String,
233
+ snapshot: Schema.optional(Schema.String),
234
+ cost: Schema.Finite,
235
+ tokens: Schema.Struct({
236
+ total: Schema.optional(Schema.Finite),
237
+ input: Schema.Finite,
238
+ output: Schema.Finite,
239
+ reasoning: Schema.Finite,
240
+ cache: Schema.Struct({
241
+ read: Schema.Finite,
242
+ write: Schema.Finite,
243
+ }),
244
+ }),
245
+ }).annotate({ identifier: "StepFinishPart" })
246
+ export type StepFinishPart = Types.DeepMutable<Schema.Schema.Type<typeof StepFinishPart>>
247
+
248
+ export const ToolStatePending = Schema.Struct({
249
+ status: Schema.Literal("pending"),
250
+ input: Schema.Record(Schema.String, Schema.Any),
251
+ raw: Schema.String,
252
+ }).annotate({ identifier: "ToolStatePending" })
253
+ export type ToolStatePending = Types.DeepMutable<Schema.Schema.Type<typeof ToolStatePending>>
254
+
255
+ export const ToolStateRunning = Schema.Struct({
256
+ status: Schema.Literal("running"),
257
+ input: Schema.Record(Schema.String, Schema.Any),
258
+ title: Schema.optional(Schema.String),
259
+ metadata: Schema.optional(Schema.Record(Schema.String, Schema.Any)),
260
+ time: Schema.Struct({
261
+ start: NonNegativeInt,
262
+ }),
263
+ }).annotate({ identifier: "ToolStateRunning" })
264
+ export type ToolStateRunning = Types.DeepMutable<Schema.Schema.Type<typeof ToolStateRunning>>
265
+
266
+ export const ToolStateCompleted = Schema.Struct({
267
+ status: Schema.Literal("completed"),
268
+ input: Schema.Record(Schema.String, Schema.Any),
269
+ output: Schema.String,
270
+ title: Schema.String,
271
+ metadata: Schema.Record(Schema.String, Schema.Any),
272
+ time: Schema.Struct({
273
+ start: NonNegativeInt,
274
+ end: NonNegativeInt,
275
+ compacted: Schema.optional(NonNegativeInt),
276
+ }),
277
+ attachments: Schema.optional(Schema.Array(FilePart)),
278
+ }).annotate({ identifier: "ToolStateCompleted" })
279
+ export type ToolStateCompleted = Types.DeepMutable<Schema.Schema.Type<typeof ToolStateCompleted>>
280
+
281
+ function truncateToolOutput(text: string, maxChars?: number) {
282
+ if (!maxChars || text.length <= maxChars) return text
283
+ const omitted = text.length - maxChars
284
+ return `${text.slice(0, maxChars)}\n[Tool output truncated for compaction: omitted ${omitted} chars]`
285
+ }
286
+
287
+ export const ToolStateError = Schema.Struct({
288
+ status: Schema.Literal("error"),
289
+ input: Schema.Record(Schema.String, Schema.Any),
290
+ error: Schema.String,
291
+ metadata: Schema.optional(Schema.Record(Schema.String, Schema.Any)),
292
+ time: Schema.Struct({
293
+ start: NonNegativeInt,
294
+ end: NonNegativeInt,
295
+ }),
296
+ }).annotate({ identifier: "ToolStateError" })
297
+ export type ToolStateError = Types.DeepMutable<Schema.Schema.Type<typeof ToolStateError>>
298
+
299
+ export const ToolState = Schema.Union([
300
+ ToolStatePending,
301
+ ToolStateRunning,
302
+ ToolStateCompleted,
303
+ ToolStateError,
304
+ ]).annotate({
305
+ discriminator: "status",
306
+ identifier: "ToolState",
307
+ })
308
+ export type ToolState = ToolStatePending | ToolStateRunning | ToolStateCompleted | ToolStateError
309
+
310
+ export const ToolPart = Schema.Struct({
311
+ ...partBase,
312
+ type: Schema.Literal("tool"),
313
+ callID: Schema.String,
314
+ tool: Schema.String,
315
+ state: ToolState,
316
+ metadata: Schema.optional(Schema.Record(Schema.String, Schema.Any)),
317
+ }).annotate({ identifier: "ToolPart" })
318
+ export type ToolPart = Omit<Types.DeepMutable<Schema.Schema.Type<typeof ToolPart>>, "state"> & {
319
+ state: ToolState
320
+ }
321
+
322
+ const messageBase = {
323
+ id: MessageID,
324
+ sessionID: SessionID,
325
+ }
326
+
327
+ export const User = Schema.Struct({
328
+ ...messageBase,
329
+ role: Schema.Literal("user"),
330
+ time: Schema.Struct({
331
+ created: NonNegativeInt,
332
+ }),
333
+ format: Schema.optional(Format),
334
+ summary: Schema.optional(
335
+ Schema.Struct({
336
+ title: Schema.optional(Schema.String),
337
+ body: Schema.optional(Schema.String),
338
+ diffs: Schema.Array(Snapshot.FileDiff),
339
+ }),
340
+ ),
341
+ agent: Schema.String,
342
+ model: Schema.Struct({
343
+ providerID: ProviderID,
344
+ modelID: ModelID,
345
+ variant: Schema.optional(Schema.String),
346
+ }),
347
+ system: Schema.optional(Schema.String),
348
+ tools: Schema.optional(Schema.Record(Schema.String, Schema.Boolean)),
349
+ }).annotate({ identifier: "UserMessage" })
350
+ export type User = Types.DeepMutable<Schema.Schema.Type<typeof User>>
351
+
352
+ export const Part = Schema.Union([
353
+ TextPart,
354
+ SubtaskPart,
355
+ ReasoningPart,
356
+ FilePart,
357
+ ToolPart,
358
+ StepStartPart,
359
+ StepFinishPart,
360
+ SnapshotPart,
361
+ PatchPart,
362
+ AgentPart,
363
+ RetryPart,
364
+ CompactionPart,
365
+ ]).annotate({ discriminator: "type", identifier: "Part" })
366
+ export type Part =
367
+ | TextPart
368
+ | SubtaskPart
369
+ | ReasoningPart
370
+ | FilePart
371
+ | ToolPart
372
+ | StepStartPart
373
+ | StepFinishPart
374
+ | SnapshotPart
375
+ | PatchPart
376
+ | AgentPart
377
+ | RetryPart
378
+ | CompactionPart
379
+
380
+ const AssistantErrorSchema = Schema.Union([
381
+ ...MessageError.Shared,
382
+ AbortedError.EffectSchema,
383
+ StructuredOutputError.EffectSchema,
384
+ ContextOverflowError.EffectSchema,
385
+ APIError.EffectSchema,
386
+ ]).annotate({ discriminator: "name" })
387
+ type AssistantError = Schema.Schema.Type<typeof AssistantErrorSchema>
388
+
389
+ // ── Prompt input schemas ─────────────────────────────────────────────────────
390
+ //
391
+ // Consumers of `SessionPrompt.PromptInput.parts` send part drafts without the
392
+ // ambient IDs (`messageID`, `sessionID`) that live on stored parts, and may
393
+ // omit `id` to let the server allocate one. These Schema-Struct variants
394
+ // carry that shape so prompt decoding can accept drafts without stored IDs.
395
+
396
+ export const TextPartInput = Schema.Struct({
397
+ id: Schema.optional(PartID),
398
+ type: Schema.Literal("text"),
399
+ text: Schema.String,
400
+ synthetic: Schema.optional(Schema.Boolean),
401
+ ignored: Schema.optional(Schema.Boolean),
402
+ time: Schema.optional(
403
+ Schema.Struct({
404
+ start: NonNegativeInt,
405
+ end: Schema.optional(NonNegativeInt),
406
+ }),
407
+ ),
408
+ metadata: Schema.optional(Schema.Record(Schema.String, Schema.Any)),
409
+ }).annotate({ identifier: "TextPartInput" })
410
+ export type TextPartInput = Types.DeepMutable<Schema.Schema.Type<typeof TextPartInput>>
411
+
412
+ export const FilePartInput = Schema.Struct({
413
+ id: Schema.optional(PartID),
414
+ type: Schema.Literal("file"),
415
+ mime: Schema.String,
416
+ filename: Schema.optional(Schema.String),
417
+ url: Schema.String,
418
+ source: Schema.optional(FilePartSource),
419
+ }).annotate({ identifier: "FilePartInput" })
420
+ export type FilePartInput = Types.DeepMutable<Schema.Schema.Type<typeof FilePartInput>>
421
+
422
+ export const AgentPartInput = Schema.Struct({
423
+ id: Schema.optional(PartID),
424
+ type: Schema.Literal("agent"),
425
+ name: Schema.String,
426
+ source: Schema.optional(
427
+ Schema.Struct({
428
+ value: Schema.String,
429
+ start: NonNegativeInt,
430
+ end: NonNegativeInt,
431
+ }),
432
+ ),
433
+ }).annotate({ identifier: "AgentPartInput" })
434
+ export type AgentPartInput = Types.DeepMutable<Schema.Schema.Type<typeof AgentPartInput>>
435
+
436
+ export const SubtaskPartInput = Schema.Struct({
437
+ id: Schema.optional(PartID),
438
+ type: Schema.Literal("subtask"),
439
+ prompt: Schema.String,
440
+ description: Schema.String,
441
+ agent: Schema.String,
442
+ model: Schema.optional(
443
+ Schema.Struct({
444
+ providerID: ProviderID,
445
+ modelID: ModelID,
446
+ }),
447
+ ),
448
+ command: Schema.optional(Schema.String),
449
+ }).annotate({ identifier: "SubtaskPartInput" })
450
+ export type SubtaskPartInput = Types.DeepMutable<Schema.Schema.Type<typeof SubtaskPartInput>>
451
+
452
+ export const Assistant = Schema.Struct({
453
+ ...messageBase,
454
+ role: Schema.Literal("assistant"),
455
+ time: Schema.Struct({
456
+ created: NonNegativeInt,
457
+ completed: Schema.optional(NonNegativeInt),
458
+ }),
459
+ error: Schema.optional(AssistantErrorSchema),
460
+ parentID: MessageID,
461
+ modelID: ModelID,
462
+ providerID: ProviderID,
463
+ /**
464
+ * @deprecated
465
+ */
466
+ mode: Schema.String,
467
+ agent: Schema.String,
468
+ path: Schema.Struct({
469
+ cwd: Schema.String,
470
+ root: Schema.String,
471
+ }),
472
+ summary: Schema.optional(Schema.Boolean),
473
+ cost: Schema.Finite,
474
+ tokens: Schema.Struct({
475
+ total: Schema.optional(Schema.Finite),
476
+ input: Schema.Finite,
477
+ output: Schema.Finite,
478
+ reasoning: Schema.Finite,
479
+ cache: Schema.Struct({
480
+ read: Schema.Finite,
481
+ write: Schema.Finite,
482
+ }),
483
+ }),
484
+ structured: Schema.optional(Schema.Any),
485
+ variant: Schema.optional(Schema.String),
486
+ finish: Schema.optional(Schema.String),
487
+ }).annotate({ identifier: "AssistantMessage" })
488
+ export type Assistant = Omit<Types.DeepMutable<Schema.Schema.Type<typeof Assistant>>, "error"> & {
489
+ error?: AssistantError
490
+ }
491
+
492
+ export const Info = Schema.Union([User, Assistant]).annotate({ discriminator: "role", identifier: "Message" })
493
+ export type Info = User | Assistant
494
+
495
+ const UpdatedEventSchema = Schema.Struct({
496
+ sessionID: SessionID,
497
+ info: Info,
498
+ })
499
+
500
+ const RemovedEventSchema = Schema.Struct({
501
+ sessionID: SessionID,
502
+ messageID: MessageID,
503
+ })
504
+
505
+ const PartUpdatedEventSchema = Schema.Struct({
506
+ sessionID: SessionID,
507
+ part: Part,
508
+ time: NonNegativeInt,
509
+ })
510
+
511
+ const PartRemovedEventSchema = Schema.Struct({
512
+ sessionID: SessionID,
513
+ messageID: MessageID,
514
+ partID: PartID,
515
+ })
516
+
517
+ export const Event = {
518
+ Updated: SyncEvent.define({
519
+ type: "message.updated",
520
+ version: 1,
521
+ aggregate: "sessionID",
522
+ schema: UpdatedEventSchema,
523
+ }),
524
+ Removed: SyncEvent.define({
525
+ type: "message.removed",
526
+ version: 1,
527
+ aggregate: "sessionID",
528
+ schema: RemovedEventSchema,
529
+ }),
530
+ PartUpdated: SyncEvent.define({
531
+ type: "message.part.updated",
532
+ version: 1,
533
+ aggregate: "sessionID",
534
+ schema: PartUpdatedEventSchema,
535
+ }),
536
+ PartDelta: BusEvent.define(
537
+ "message.part.delta",
538
+ Schema.Struct({
539
+ sessionID: SessionID,
540
+ messageID: MessageID,
541
+ partID: PartID,
542
+ field: Schema.String,
543
+ delta: Schema.String,
544
+ }),
545
+ ),
546
+ PartRemoved: SyncEvent.define({
547
+ type: "message.part.removed",
548
+ version: 1,
549
+ aggregate: "sessionID",
550
+ schema: PartRemovedEventSchema,
551
+ }),
552
+ }
553
+
554
+ export const WithParts = Schema.Struct({
555
+ info: Info,
556
+ parts: Schema.Array(Part),
557
+ })
558
+ export type WithParts = {
559
+ info: Info
560
+ parts: Part[]
561
+ }
562
+
563
+ const Cursor = Schema.Struct({
564
+ id: MessageID,
565
+ time: Schema.Finite.check(Schema.isGreaterThanOrEqualTo(0)),
566
+ })
567
+ type Cursor = typeof Cursor.Type
568
+
569
+ const decodeCursor = Schema.decodeUnknownSync(Cursor)
570
+
571
+ export const cursor = {
572
+ encode(input: Cursor) {
573
+ return Buffer.from(JSON.stringify(input)).toString("base64url")
574
+ },
575
+ decode(input: string) {
576
+ return decodeCursor(JSON.parse(Buffer.from(input, "base64url").toString("utf8")))
577
+ },
578
+ }
579
+
580
+ const info = (row: typeof MessageTable.$inferSelect) =>
581
+ ({
582
+ ...row.data,
583
+ id: row.id,
584
+ sessionID: row.session_id,
585
+ }) as Info
586
+
587
+ const part = (row: typeof PartTable.$inferSelect) =>
588
+ ({
589
+ ...row.data,
590
+ id: row.id,
591
+ sessionID: row.session_id,
592
+ messageID: row.message_id,
593
+ }) as Part
594
+
595
+ const older = (row: Cursor) =>
596
+ or(lt(MessageTable.time_created, row.time), and(eq(MessageTable.time_created, row.time), lt(MessageTable.id, row.id)))
597
+
598
+ function hydrate(rows: (typeof MessageTable.$inferSelect)[]) {
599
+ const ids = rows.map((row) => row.id)
600
+ const partByMessage = new Map<string, Part[]>()
601
+ if (ids.length > 0) {
602
+ const partRows = Database.use((db) =>
603
+ db
604
+ .select()
605
+ .from(PartTable)
606
+ .where(inArray(PartTable.message_id, ids))
607
+ .orderBy(PartTable.message_id, PartTable.id)
608
+ .all(),
609
+ )
610
+ for (const row of partRows) {
611
+ const next = part(row)
612
+ const list = partByMessage.get(row.message_id)
613
+ if (list) list.push(next)
614
+ else partByMessage.set(row.message_id, [next])
615
+ }
616
+ }
617
+
618
+ return rows.map((row) => ({
619
+ info: info(row),
620
+ parts: partByMessage.get(row.id) ?? [],
621
+ }))
622
+ }
623
+
624
+ function providerMeta(metadata: Record<string, any> | undefined) {
625
+ if (!metadata) return undefined
626
+ const { providerExecuted: _, ...rest } = metadata
627
+ return Object.keys(rest).length > 0 ? rest : undefined
628
+ }
629
+
630
+ export const toModelMessagesEffect = Effect.fnUntraced(function* (
631
+ input: WithParts[],
632
+ model: Provider.Model,
633
+ options?: { stripMedia?: boolean; toolOutputMaxChars?: number },
634
+ ) {
635
+ const result: UIMessage[] = []
636
+ const toolNames = new Set<string>()
637
+ // Track media from tool results that need to be injected as user messages
638
+ // for providers that don't support that media type in tool results.
639
+ //
640
+ // OpenAI-compatible APIs only support string content in tool results, so we need
641
+ // to extract media and inject as user messages. Some SDKs only support a subset
642
+ // of media in tool results; e.g. Bedrock supports images but not PDFs there.
643
+ //
644
+ // Only apply this workaround if the model actually supports that media input -
645
+ // otherwise unsupportedParts() will turn it into a user-visible error.
646
+ const supportsMediaInToolResult = (attachment: { mime: string }) => {
647
+ if (model.api.npm === "@ai-sdk/anthropic") return true
648
+ if (model.api.npm === "@ai-sdk/openai") return true
649
+ if (model.api.npm === "@ai-sdk/amazon-bedrock") return attachment.mime.startsWith("image/")
650
+ if (model.api.npm === "@ai-sdk/google-vertex/anthropic") return true
651
+ if (model.api.npm === "@ai-sdk/google") {
652
+ const id = model.api.id.toLowerCase()
653
+ return id.includes("gemini-3") && !id.includes("gemini-2")
654
+ }
655
+ return false
656
+ }
657
+
658
+ const toModelOutput = (options: { toolCallId: string; input: unknown; output: unknown }) => {
659
+ const output = options.output
660
+ if (typeof output === "string") {
661
+ return { type: "text", value: output }
662
+ }
663
+
664
+ if (typeof output === "object") {
665
+ const outputObject = output as {
666
+ text: string
667
+ attachments?: Array<{ mime: string; url: string }>
668
+ }
669
+ const attachments = (outputObject.attachments ?? []).filter((attachment) => {
670
+ return attachment.url.startsWith("data:") && attachment.url.includes(",")
671
+ })
672
+
673
+ return {
674
+ type: "content",
675
+ value: [
676
+ ...(outputObject.text ? [{ type: "text", text: outputObject.text }] : []),
677
+ ...attachments.map((attachment) => ({
678
+ type: "media",
679
+ mediaType: attachment.mime,
680
+ data: iife(() => {
681
+ const commaIndex = attachment.url.indexOf(",")
682
+ return commaIndex === -1 ? attachment.url : attachment.url.slice(commaIndex + 1)
683
+ }),
684
+ })),
685
+ ],
686
+ }
687
+ }
688
+
689
+ return { type: "json", value: output as never }
690
+ }
691
+
692
+ for (const msg of input) {
693
+ if (msg.parts.length === 0) continue
694
+
695
+ if (msg.info.role === "user") {
696
+ const userMessage: UIMessage = {
697
+ id: msg.info.id,
698
+ role: "user",
699
+ parts: [],
700
+ }
701
+ for (const part of msg.parts) {
702
+ // User message parts should never be empty
703
+ if (part.type === "text" && !part.ignored && part.text !== "")
704
+ userMessage.parts.push({
705
+ type: "text",
706
+ text: part.text,
707
+ })
708
+ // text/plain and directory files are converted into text parts, ignore them
709
+ if (part.type === "file" && part.mime !== "text/plain" && part.mime !== "application/x-directory") {
710
+ if (options?.stripMedia && isMedia(part.mime)) {
711
+ userMessage.parts.push({
712
+ type: "text",
713
+ text: `[Attached ${part.mime}: ${part.filename ?? "file"}]`,
714
+ })
715
+ } else {
716
+ userMessage.parts.push({
717
+ type: "file",
718
+ url: part.url,
719
+ mediaType: part.mime,
720
+ filename: part.filename,
721
+ })
722
+ }
723
+ }
724
+
725
+ if (part.type === "compaction") {
726
+ userMessage.parts.push({
727
+ type: "text",
728
+ text: "What did we do so far?",
729
+ })
730
+ }
731
+ if (part.type === "subtask") {
732
+ userMessage.parts.push({
733
+ type: "text",
734
+ text: "The following tool was executed by the user",
735
+ })
736
+ }
737
+ }
738
+ if (userMessage.parts.length > 0) result.push(userMessage)
739
+ }
740
+
741
+ if (msg.info.role === "assistant") {
742
+ const differentModel = `${model.providerID}/${model.id}` !== `${msg.info.providerID}/${msg.info.modelID}`
743
+ const media: Array<{ mime: string; url: string; filename?: string }> = []
744
+
745
+ if (
746
+ msg.info.error &&
747
+ !(
748
+ AbortedError.isInstance(msg.info.error) &&
749
+ msg.parts.some((part) => part.type !== "step-start" && part.type !== "reasoning")
750
+ )
751
+ ) {
752
+ continue
753
+ }
754
+ const assistantMessage: UIMessage = {
755
+ id: msg.info.id,
756
+ role: "assistant",
757
+ parts: [],
758
+ }
759
+ // Anthropic adaptive thinking can persist assistant turns like:
760
+ // step-start, reasoning(signature), text(""), step-start,
761
+ // reasoning(signature). The empty text part is a structural separator,
762
+ // but it does not carry the signature metadata itself. Dropping it shifts
763
+ // signed thinking positions after step-start splitting/provider regrouping;
764
+ // keeping it as "" is filtered by the AI SDK and rejected by Anthropic.
765
+ // It is unclear whether this shape originates in our stream processing,
766
+ // a proxy, or a lower-level library, but preserving a non-empty separator
767
+ // here is the only safe replay point we have.
768
+ // Use a single space so the separator survives replay without changing
769
+ // the neighboring signed reasoning blocks.
770
+ const hasSignedReasoning = msg.parts.some((part) => {
771
+ if (part.type !== "reasoning") return false
772
+ return part.metadata?.anthropic?.signature != null
773
+ })
774
+ for (const part of msg.parts) {
775
+ if (part.type === "text") {
776
+ const text = part.text === "" && hasSignedReasoning ? " " : part.text
777
+ assistantMessage.parts.push({
778
+ type: "text",
779
+ text,
780
+ ...(differentModel ? {} : { providerMetadata: part.metadata }),
781
+ })
782
+ }
783
+ if (part.type === "step-start")
784
+ assistantMessage.parts.push({
785
+ type: "step-start",
786
+ })
787
+ if (part.type === "tool") {
788
+ toolNames.add(part.tool)
789
+ if (part.state.status === "completed") {
790
+ const outputText = part.state.time.compacted
791
+ ? "[Old tool result content cleared]"
792
+ : truncateToolOutput(part.state.output, options?.toolOutputMaxChars)
793
+ const attachments = part.state.time.compacted || options?.stripMedia ? [] : (part.state.attachments ?? [])
794
+
795
+ // For providers that don't support media in tool results, extract media files
796
+ // (images, PDFs) to be sent as a separate user message
797
+ const mediaAttachments = attachments.filter((a) => isMedia(a.mime))
798
+ const extractedMedia = mediaAttachments.filter((a) => !supportsMediaInToolResult(a))
799
+ if (extractedMedia.length > 0) {
800
+ media.push(...extractedMedia)
801
+ }
802
+ const finalAttachments = attachments.filter((a) => !isMedia(a.mime) || supportsMediaInToolResult(a))
803
+
804
+ const output =
805
+ finalAttachments.length > 0
806
+ ? {
807
+ text: outputText,
808
+ attachments: finalAttachments,
809
+ }
810
+ : outputText
811
+
812
+ assistantMessage.parts.push({
813
+ type: ("tool-" + part.tool) as `tool-${string}`,
814
+ state: "output-available",
815
+ toolCallId: part.callID,
816
+ input: part.state.input,
817
+ output,
818
+ ...(part.metadata?.providerExecuted ? { providerExecuted: true } : {}),
819
+ ...(differentModel ? {} : { callProviderMetadata: providerMeta(part.metadata) }),
820
+ })
821
+ }
822
+ if (part.state.status === "error") {
823
+ const output = part.state.metadata?.interrupted === true ? part.state.metadata.output : undefined
824
+ if (typeof output === "string") {
825
+ assistantMessage.parts.push({
826
+ type: ("tool-" + part.tool) as `tool-${string}`,
827
+ state: "output-available",
828
+ toolCallId: part.callID,
829
+ input: part.state.input,
830
+ output,
831
+ ...(part.metadata?.providerExecuted ? { providerExecuted: true } : {}),
832
+ ...(differentModel ? {} : { callProviderMetadata: providerMeta(part.metadata) }),
833
+ })
834
+ } else {
835
+ assistantMessage.parts.push({
836
+ type: ("tool-" + part.tool) as `tool-${string}`,
837
+ state: "output-error",
838
+ toolCallId: part.callID,
839
+ input: part.state.input,
840
+ errorText: part.state.error,
841
+ ...(part.metadata?.providerExecuted ? { providerExecuted: true } : {}),
842
+ ...(differentModel ? {} : { callProviderMetadata: providerMeta(part.metadata) }),
843
+ })
844
+ }
845
+ }
846
+ // Handle pending/running tool calls to prevent dangling tool_use blocks
847
+ // Anthropic/Claude APIs require every tool_use to have a corresponding tool_result
848
+ if (part.state.status === "pending" || part.state.status === "running")
849
+ assistantMessage.parts.push({
850
+ type: ("tool-" + part.tool) as `tool-${string}`,
851
+ state: "output-error",
852
+ toolCallId: part.callID,
853
+ input: part.state.input,
854
+ errorText: "[Tool execution was interrupted]",
855
+ ...(part.metadata?.providerExecuted ? { providerExecuted: true } : {}),
856
+ ...(differentModel ? {} : { callProviderMetadata: providerMeta(part.metadata) }),
857
+ })
858
+ }
859
+ if (part.type === "reasoning") {
860
+ if (differentModel) {
861
+ if (part.text.trim().length > 0)
862
+ assistantMessage.parts.push({
863
+ type: "text",
864
+ text: part.text,
865
+ })
866
+ continue
867
+ }
868
+ assistantMessage.parts.push({
869
+ type: "reasoning",
870
+ text: part.text,
871
+ providerMetadata: part.metadata,
872
+ })
873
+ }
874
+ }
875
+ if (assistantMessage.parts.length > 0) {
876
+ result.push(assistantMessage)
877
+ // Inject pending media as a user message for providers that don't support
878
+ // media (images, PDFs) in tool results
879
+ if (media.length > 0) {
880
+ result.push({
881
+ id: MessageID.ascending(),
882
+ role: "user",
883
+ parts: [
884
+ {
885
+ type: "text" as const,
886
+ text: SYNTHETIC_ATTACHMENT_PROMPT,
887
+ },
888
+ ...media.map((attachment) => ({
889
+ type: "file" as const,
890
+ url: attachment.url,
891
+ mediaType: attachment.mime,
892
+ filename: attachment.filename,
893
+ })),
894
+ ],
895
+ })
896
+ }
897
+ }
898
+ }
899
+ }
900
+
901
+ const tools = Object.fromEntries(Array.from(toolNames).map((toolName) => [toolName, { toModelOutput }]))
902
+
903
+ return yield* Effect.promise(() =>
904
+ convertToModelMessages(
905
+ result.filter((msg) => msg.parts.some((part) => part.type !== "step-start")),
906
+ {
907
+ //@ts-expect-error (convertToModelMessages expects a ToolSet but only actually needs tools[name]?.toModelOutput)
908
+ tools,
909
+ },
910
+ ),
911
+ )
912
+ })
913
+
914
+ export function toModelMessages(
915
+ input: WithParts[],
916
+ model: Provider.Model,
917
+ options?: { stripMedia?: boolean; toolOutputMaxChars?: number },
918
+ ): Promise<ModelMessage[]> {
919
+ return Effect.runPromise(toModelMessagesEffect(input, model, options).pipe(Effect.provide(EffectLogger.layer)))
920
+ }
921
+
922
+ export const page = Effect.fn("MessageV2.page")(function* (input: {
923
+ sessionID: SessionID
924
+ limit: number
925
+ before?: string
926
+ }) {
927
+ const before = input.before ? cursor.decode(input.before) : undefined
928
+ const where = before
929
+ ? and(eq(MessageTable.session_id, input.sessionID), older(before))
930
+ : eq(MessageTable.session_id, input.sessionID)
931
+ const rows = Database.use((db) =>
932
+ db
933
+ .select()
934
+ .from(MessageTable)
935
+ .where(where)
936
+ .orderBy(desc(MessageTable.time_created), desc(MessageTable.id))
937
+ .limit(input.limit + 1)
938
+ .all(),
939
+ )
940
+ if (rows.length === 0) {
941
+ const row = Database.use((db) =>
942
+ db.select({ id: SessionTable.id }).from(SessionTable).where(eq(SessionTable.id, input.sessionID)).get(),
943
+ )
944
+ if (!row) return yield* new NotFoundError({ message: `Session not found: ${input.sessionID}` })
945
+ return {
946
+ items: [] as WithParts[],
947
+ more: false,
948
+ }
949
+ }
950
+
951
+ const more = rows.length > input.limit
952
+ const slice = more ? rows.slice(0, input.limit) : rows
953
+ const items = hydrate(slice)
954
+ items.reverse()
955
+ const tail = slice.at(-1)
956
+ return {
957
+ items,
958
+ more,
959
+ cursor: more && tail ? cursor.encode({ id: tail.id, time: tail.time_created }) : undefined,
960
+ }
961
+ })
962
+
963
+ export function* stream(sessionID: SessionID) {
964
+ const size = 50
965
+ let before: string | undefined
966
+ while (true) {
967
+ const next = Effect.runSync(
968
+ page({ sessionID, limit: size, before }).pipe(
969
+ Effect.catchIf(NotFoundError.isInstance, () =>
970
+ Effect.succeed({ items: [] as WithParts[], more: false, cursor: undefined }),
971
+ ),
972
+ ),
973
+ )
974
+ if (next.items.length === 0) break
975
+ for (let i = next.items.length - 1; i >= 0; i--) {
976
+ yield next.items[i]
977
+ }
978
+ if (!next.more || !next.cursor) break
979
+ before = next.cursor
980
+ }
981
+ }
982
+
983
+ export function parts(message_id: MessageID) {
984
+ const rows = Database.use((db) =>
985
+ db.select().from(PartTable).where(eq(PartTable.message_id, message_id)).orderBy(PartTable.id).all(),
986
+ )
987
+ return rows.map(
988
+ (row) =>
989
+ ({
990
+ ...row.data,
991
+ id: row.id,
992
+ sessionID: row.session_id,
993
+ messageID: row.message_id,
994
+ }) as Part,
995
+ )
996
+ }
997
+
998
+ export const get = Effect.fn("MessageV2.get")(function* (input: { sessionID: SessionID; messageID: MessageID }) {
999
+ const row = Database.use((db) =>
1000
+ db
1001
+ .select()
1002
+ .from(MessageTable)
1003
+ .where(and(eq(MessageTable.id, input.messageID), eq(MessageTable.session_id, input.sessionID)))
1004
+ .get(),
1005
+ )
1006
+ if (!row) return yield* new NotFoundError({ message: `Message not found: ${input.messageID}` })
1007
+ return {
1008
+ info: info(row),
1009
+ parts: parts(input.messageID),
1010
+ }
1011
+ })
1012
+
1013
+ export function filterCompacted(msgs: Iterable<WithParts>) {
1014
+ const result = [] as WithParts[]
1015
+ const completed = new Set<string>()
1016
+ let retain: MessageID | undefined
1017
+ for (const msg of msgs) {
1018
+ result.push(msg)
1019
+ if (retain) {
1020
+ if (msg.info.id === retain) break
1021
+ continue
1022
+ }
1023
+ if (msg.info.role === "user" && completed.has(msg.info.id)) {
1024
+ const part = msg.parts.find((item): item is CompactionPart => item.type === "compaction")
1025
+ if (!part) continue
1026
+ if (!part.tail_start_id) break
1027
+ retain = part.tail_start_id
1028
+ if (msg.info.id === retain) break
1029
+ continue
1030
+ }
1031
+ if (msg.info.role === "user" && completed.has(msg.info.id) && msg.parts.some((part) => part.type === "compaction"))
1032
+ break
1033
+ if (msg.info.role === "assistant" && msg.info.summary && msg.info.finish && !msg.info.error)
1034
+ completed.add(msg.info.parentID)
1035
+ }
1036
+ result.reverse()
1037
+ const compactionIndex = result.findLastIndex(
1038
+ (msg) =>
1039
+ msg.info.role === "user" &&
1040
+ msg.parts.some((item): item is CompactionPart => item.type === "compaction" && item.tail_start_id !== undefined),
1041
+ )
1042
+ const compaction = result[compactionIndex]
1043
+ const part = compaction?.parts.find(
1044
+ (item): item is CompactionPart => item.type === "compaction" && item.tail_start_id !== undefined,
1045
+ )
1046
+ const summaryIndex = compaction
1047
+ ? result.findIndex(
1048
+ (msg, index) =>
1049
+ index > compactionIndex &&
1050
+ msg.info.role === "assistant" &&
1051
+ msg.info.summary &&
1052
+ msg.info.parentID === compaction.info.id,
1053
+ )
1054
+ : -1
1055
+ const tailIndex = part?.tail_start_id ? result.findIndex((msg) => msg.info.id === part.tail_start_id) : -1
1056
+ if (tailIndex >= 0 && tailIndex < compactionIndex && summaryIndex > compactionIndex) {
1057
+ return [
1058
+ ...result.slice(compactionIndex, summaryIndex + 1),
1059
+ ...result.slice(tailIndex, compactionIndex),
1060
+ ...result.slice(summaryIndex + 1),
1061
+ ]
1062
+ }
1063
+ return result
1064
+ }
1065
+
1066
+ export const filterCompactedEffect = Effect.fnUntraced(function* (sessionID: SessionID) {
1067
+ return filterCompacted(stream(sessionID))
1068
+ })
1069
+
1070
+ // filterCompacted reorders messages for model consumption
1071
+ // ([compaction-user, summary, ...retained tail..., continue-user]), so array
1072
+ // position is not chronological. Derive each binding by max id (MessageID
1073
+ // is monotonic via MessageID.ascending) so a pre-compaction overflowing tail
1074
+ // assistant doesn't get mistaken for the most recent turn. tasks are
1075
+ // compaction/subtask parts attached to user messages newer than the latest
1076
+ // finished assistant — i.e. unprocessed work.
1077
+ export function latest(msgs: WithParts[]) {
1078
+ let user: User | undefined
1079
+ let assistant: Assistant | undefined
1080
+ let finished: Assistant | undefined
1081
+ for (const msg of msgs) {
1082
+ const info = msg.info
1083
+ if (info.role === "user" && (!user || info.id > user.id)) user = info
1084
+ if (info.role === "assistant" && (!assistant || info.id > assistant.id)) assistant = info
1085
+ if (info.role === "assistant" && info.finish && (!finished || info.id > finished.id)) finished = info
1086
+ }
1087
+ const tasks = msgs.flatMap((m) =>
1088
+ finished && m.info.id <= finished.id
1089
+ ? []
1090
+ : m.parts.filter((p): p is CompactionPart | SubtaskPart => p.type === "compaction" || p.type === "subtask"),
1091
+ )
1092
+ return { user, assistant, finished, tasks }
1093
+ }
1094
+
1095
+ export function fromError(
1096
+ e: unknown,
1097
+ ctx: { providerID: ProviderID; aborted?: boolean },
1098
+ ): NonNullable<Assistant["error"]> {
1099
+ switch (true) {
1100
+ case e instanceof DOMException && e.name === "AbortError":
1101
+ return new AbortedError(
1102
+ { message: e.message },
1103
+ {
1104
+ cause: e,
1105
+ },
1106
+ ).toObject()
1107
+ case OutputLengthError.isInstance(e):
1108
+ return e
1109
+ case LoadAPIKeyError.isInstance(e):
1110
+ return new AuthError(
1111
+ {
1112
+ providerID: ctx.providerID,
1113
+ message: e.message,
1114
+ },
1115
+ { cause: e },
1116
+ ).toObject()
1117
+ case (e as SystemError)?.code === "ECONNRESET":
1118
+ return new APIError(
1119
+ {
1120
+ message: "Connection reset by server",
1121
+ isRetryable: true,
1122
+ metadata: {
1123
+ code: (e as SystemError).code ?? "",
1124
+ syscall: (e as SystemError).syscall ?? "",
1125
+ message: (e as SystemError).message ?? "",
1126
+ },
1127
+ },
1128
+ { cause: e },
1129
+ ).toObject()
1130
+ case e instanceof Error && (e as FetchDecompressionError).code === "ZlibError":
1131
+ if (ctx.aborted) {
1132
+ return new AbortedError({ message: e.message }, { cause: e }).toObject()
1133
+ }
1134
+ return new APIError(
1135
+ {
1136
+ message: "Response decompression failed",
1137
+ isRetryable: true,
1138
+ metadata: {
1139
+ code: (e as FetchDecompressionError).code,
1140
+ message: e.message,
1141
+ },
1142
+ },
1143
+ { cause: e },
1144
+ ).toObject()
1145
+ case APICallError.isInstance(e):
1146
+ const parsed = ProviderError.parseAPICallError({
1147
+ providerID: ctx.providerID,
1148
+ error: e,
1149
+ })
1150
+ if (parsed.type === "context_overflow") {
1151
+ return new ContextOverflowError(
1152
+ {
1153
+ message: parsed.message,
1154
+ responseBody: parsed.responseBody,
1155
+ },
1156
+ { cause: e },
1157
+ ).toObject()
1158
+ }
1159
+
1160
+ return new APIError(
1161
+ {
1162
+ message: parsed.message,
1163
+ statusCode: parsed.statusCode,
1164
+ isRetryable: parsed.isRetryable,
1165
+ responseHeaders: parsed.responseHeaders,
1166
+ responseBody: parsed.responseBody,
1167
+ metadata: parsed.metadata,
1168
+ },
1169
+ { cause: e },
1170
+ ).toObject()
1171
+ case e instanceof Error:
1172
+ return new NamedError.Unknown({ message: errorMessage(e) }, { cause: e }).toObject()
1173
+ default:
1174
+ try {
1175
+ const parsed = ProviderError.parseStreamError(e)
1176
+ if (parsed) {
1177
+ if (parsed.type === "context_overflow") {
1178
+ return new ContextOverflowError(
1179
+ {
1180
+ message: parsed.message,
1181
+ responseBody: parsed.responseBody,
1182
+ },
1183
+ { cause: e },
1184
+ ).toObject()
1185
+ }
1186
+ return new APIError(
1187
+ {
1188
+ message: parsed.message,
1189
+ isRetryable: parsed.isRetryable,
1190
+ responseBody: parsed.responseBody,
1191
+ },
1192
+ {
1193
+ cause: e,
1194
+ },
1195
+ ).toObject()
1196
+ }
1197
+ } catch {}
1198
+ return new NamedError.Unknown({ message: JSON.stringify(e) }, { cause: e }).toObject()
1199
+ }
1200
+ }
1201
+
1202
+ export * as MessageV2 from "./message-v2"