bonecode 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (605) hide show
  1. package/ARCHITECTURE.md +183 -0
  2. package/README.md +71 -0
  3. package/bin/bonecode +62 -0
  4. package/bone/migrations/rag_vectors.sql +258 -0
  5. package/bone/output/agent/.dockerignore +7 -0
  6. package/bone/output/agent/.env.example +36 -0
  7. package/bone/output/agent/.github/workflows/ci.yaml +58 -0
  8. package/bone/output/agent/AgentDomain.bone.map +350 -0
  9. package/bone/output/agent/AgentDomain.postman_collection.json +958 -0
  10. package/bone/output/agent/Dockerfile +22 -0
  11. package/bone/output/agent/README.md +47 -0
  12. package/bone/output/agent/admin/index.html +740 -0
  13. package/bone/output/agent/docker-compose.yaml +22 -0
  14. package/bone/output/agent/k8s/deployment.yaml +75 -0
  15. package/bone/output/agent/migrations/agent.sql +36 -0
  16. package/bone/output/agent/migrations/agent_instance.sql +36 -0
  17. package/bone/output/agent/migrations/audit_log.sql +18 -0
  18. package/bone/output/agent/migrations/build_step.sql +34 -0
  19. package/bone/output/agent/migrations/event_outbox.sql +31 -0
  20. package/bone/output/agent/migrations/plan.sql +30 -0
  21. package/bone/output/agent/migrations/task.sql +30 -0
  22. package/bone/output/agent/migrations/tool_call.sql +33 -0
  23. package/bone/output/agent/openapi.yaml +1116 -0
  24. package/bone/output/agent/package.json +36 -0
  25. package/bone/output/agent/schema.graphql +233 -0
  26. package/bone/output/agent/sdk/client.ts +231 -0
  27. package/bone/output/agent/src/algorithms.ts +2 -0
  28. package/bone/output/agent/src/audit.ts +44 -0
  29. package/bone/output/agent/src/auth.ts +57 -0
  30. package/bone/output/agent/src/cron.ts +12 -0
  31. package/bone/output/agent/src/db.ts +32 -0
  32. package/bone/output/agent/src/debug.ts +66 -0
  33. package/bone/output/agent/src/events.ts +243 -0
  34. package/bone/output/agent/src/extensions.ts +54 -0
  35. package/bone/output/agent/src/failure_rules.ts +323 -0
  36. package/bone/output/agent/src/flows.ts +168 -0
  37. package/bone/output/agent/src/health.ts +43 -0
  38. package/bone/output/agent/src/index.ts +100 -0
  39. package/bone/output/agent/src/logger.ts +66 -0
  40. package/bone/output/agent/src/metrics.ts +75 -0
  41. package/bone/output/agent/src/migrate.ts +352 -0
  42. package/bone/output/agent/src/migration_diff.ts +108 -0
  43. package/bone/output/agent/src/notify.ts +125 -0
  44. package/bone/output/agent/src/routes/agent_instance.ts +234 -0
  45. package/bone/output/agent/src/routes/build_step.ts +105 -0
  46. package/bone/output/agent/src/routes/plan.ts +91 -0
  47. package/bone/output/agent/src/routes/task.ts +105 -0
  48. package/bone/output/agent/src/routes/tool_call.ts +166 -0
  49. package/bone/output/agent/src/schemas.ts +384 -0
  50. package/bone/output/agent/src/state_machines/agent_instance.ts +24 -0
  51. package/bone/output/agent/src/state_machines/build_step.ts +22 -0
  52. package/bone/output/agent/src/state_machines/plan.ts +22 -0
  53. package/bone/output/agent/src/state_machines/task.ts +22 -0
  54. package/bone/output/agent/src/state_machines/tool_call.ts +22 -0
  55. package/bone/output/agent/src/tests.ts +362 -0
  56. package/bone/output/agent/src/websocket.ts +201 -0
  57. package/bone/output/agent/tsconfig.json +25 -0
  58. package/bone/output/rag/.dockerignore +7 -0
  59. package/bone/output/rag/.env.example +36 -0
  60. package/bone/output/rag/.github/workflows/ci.yaml +58 -0
  61. package/bone/output/rag/Dockerfile +22 -0
  62. package/bone/output/rag/RAGDomain.bone.map +287 -0
  63. package/bone/output/rag/RAGDomain.postman_collection.json +923 -0
  64. package/bone/output/rag/README.md +47 -0
  65. package/bone/output/rag/admin/index.html +818 -0
  66. package/bone/output/rag/docker-compose.yaml +22 -0
  67. package/bone/output/rag/k8s/deployment.yaml +75 -0
  68. package/bone/output/rag/migrations/audit_log.sql +18 -0
  69. package/bone/output/rag/migrations/code_chunk.sql +34 -0
  70. package/bone/output/rag/migrations/code_file.sql +33 -0
  71. package/bone/output/rag/migrations/event_outbox.sql +31 -0
  72. package/bone/output/rag/migrations/indexing_job.sql +33 -0
  73. package/bone/output/rag/migrations/knowledge_base.sql +35 -0
  74. package/bone/output/rag/migrations/memory_entry.sql +34 -0
  75. package/bone/output/rag/openapi.yaml +1097 -0
  76. package/bone/output/rag/package.json +36 -0
  77. package/bone/output/rag/schema.graphql +245 -0
  78. package/bone/output/rag/sdk/client.ts +234 -0
  79. package/bone/output/rag/src/algorithms.ts +2 -0
  80. package/bone/output/rag/src/audit.ts +37 -0
  81. package/bone/output/rag/src/auth.ts +57 -0
  82. package/bone/output/rag/src/cron.ts +12 -0
  83. package/bone/output/rag/src/db.ts +32 -0
  84. package/bone/output/rag/src/debug.ts +66 -0
  85. package/bone/output/rag/src/events.ts +243 -0
  86. package/bone/output/rag/src/extensions.ts +350 -0
  87. package/bone/output/rag/src/failure_rules.ts +315 -0
  88. package/bone/output/rag/src/flows.ts +239 -0
  89. package/bone/output/rag/src/health.ts +43 -0
  90. package/bone/output/rag/src/index.ts +95 -0
  91. package/bone/output/rag/src/logger.ts +66 -0
  92. package/bone/output/rag/src/metrics.ts +75 -0
  93. package/bone/output/rag/src/migrate.ts +364 -0
  94. package/bone/output/rag/src/migration_diff.ts +108 -0
  95. package/bone/output/rag/src/notify.ts +99 -0
  96. package/bone/output/rag/src/routes/code_chunk.ts +75 -0
  97. package/bone/output/rag/src/routes/code_file.ts +101 -0
  98. package/bone/output/rag/src/routes/indexing_job.ts +87 -0
  99. package/bone/output/rag/src/routes/knowledge_base.ts +230 -0
  100. package/bone/output/rag/src/routes/memory_entry.ts +87 -0
  101. package/bone/output/rag/src/schemas.ts +394 -0
  102. package/bone/output/rag/src/state_machines/code_file.ts +23 -0
  103. package/bone/output/rag/src/state_machines/indexing_job.ts +22 -0
  104. package/bone/output/rag/src/state_machines/knowledge_base.ts +23 -0
  105. package/bone/output/rag/src/state_machines/memory_entry.ts +20 -0
  106. package/bone/output/rag/src/tests.ts +340 -0
  107. package/bone/output/rag/tsconfig.json +25 -0
  108. package/bone/output/session/.dockerignore +7 -0
  109. package/bone/output/session/.env.example +36 -0
  110. package/bone/output/session/.github/workflows/ci.yaml +58 -0
  111. package/bone/output/session/Dockerfile +22 -0
  112. package/bone/output/session/README.md +47 -0
  113. package/bone/output/session/SessionDomain.bone.map +350 -0
  114. package/bone/output/session/SessionDomain.postman_collection.json +958 -0
  115. package/bone/output/session/admin/index.html +667 -0
  116. package/bone/output/session/docker-compose.yaml +22 -0
  117. package/bone/output/session/k8s/deployment.yaml +75 -0
  118. package/bone/output/session/migrations/audit_log.sql +18 -0
  119. package/bone/output/session/migrations/event_outbox.sql +31 -0
  120. package/bone/output/session/migrations/message.sql +31 -0
  121. package/bone/output/session/migrations/part.sql +28 -0
  122. package/bone/output/session/migrations/permission.sql +28 -0
  123. package/bone/output/session/migrations/project.sql +28 -0
  124. package/bone/output/session/migrations/session.sql +38 -0
  125. package/bone/output/session/openapi.yaml +1101 -0
  126. package/bone/output/session/package.json +36 -0
  127. package/bone/output/session/schema.graphql +222 -0
  128. package/bone/output/session/sdk/client.ts +225 -0
  129. package/bone/output/session/src/algorithms.ts +2 -0
  130. package/bone/output/session/src/audit.ts +44 -0
  131. package/bone/output/session/src/auth.ts +57 -0
  132. package/bone/output/session/src/cron.ts +12 -0
  133. package/bone/output/session/src/db.ts +32 -0
  134. package/bone/output/session/src/debug.ts +66 -0
  135. package/bone/output/session/src/events.ts +270 -0
  136. package/bone/output/session/src/extensions.ts +215 -0
  137. package/bone/output/session/src/failure_rules.ts +284 -0
  138. package/bone/output/session/src/flows.ts +168 -0
  139. package/bone/output/session/src/health.ts +43 -0
  140. package/bone/output/session/src/index.ts +100 -0
  141. package/bone/output/session/src/logger.ts +66 -0
  142. package/bone/output/session/src/metrics.ts +75 -0
  143. package/bone/output/session/src/migrate.ts +332 -0
  144. package/bone/output/session/src/migration_diff.ts +108 -0
  145. package/bone/output/session/src/notify.ts +112 -0
  146. package/bone/output/session/src/routes/message.ts +93 -0
  147. package/bone/output/session/src/routes/part.ts +79 -0
  148. package/bone/output/session/src/routes/permission.ts +79 -0
  149. package/bone/output/session/src/routes/project.ts +79 -0
  150. package/bone/output/session/src/routes/session.ts +294 -0
  151. package/bone/output/session/src/schemas.ts +357 -0
  152. package/bone/output/session/src/state_machines/session.ts +23 -0
  153. package/bone/output/session/src/tests.ts +326 -0
  154. package/bone/output/session/src/websocket.ts +201 -0
  155. package/bone/output/session/tsconfig.json +25 -0
  156. package/bone/output/workspace/.dockerignore +7 -0
  157. package/bone/output/workspace/.env.example +36 -0
  158. package/bone/output/workspace/.github/workflows/ci.yaml +58 -0
  159. package/bone/output/workspace/Dockerfile +22 -0
  160. package/bone/output/workspace/README.md +45 -0
  161. package/bone/output/workspace/WorkspaceDomain.bone.map +189 -0
  162. package/bone/output/workspace/WorkspaceDomain.postman_collection.json +621 -0
  163. package/bone/output/workspace/admin/index.html +485 -0
  164. package/bone/output/workspace/docker-compose.yaml +22 -0
  165. package/bone/output/workspace/k8s/deployment.yaml +75 -0
  166. package/bone/output/workspace/migrations/audit_log.sql +18 -0
  167. package/bone/output/workspace/migrations/codebase.sql +34 -0
  168. package/bone/output/workspace/migrations/event_outbox.sql +31 -0
  169. package/bone/output/workspace/migrations/snapshot.sql +32 -0
  170. package/bone/output/workspace/migrations/workspace.sql +33 -0
  171. package/bone/output/workspace/openapi.yaml +721 -0
  172. package/bone/output/workspace/package.json +36 -0
  173. package/bone/output/workspace/schema.graphql +153 -0
  174. package/bone/output/workspace/sdk/client.ts +155 -0
  175. package/bone/output/workspace/src/algorithms.ts +2 -0
  176. package/bone/output/workspace/src/audit.ts +37 -0
  177. package/bone/output/workspace/src/auth.ts +57 -0
  178. package/bone/output/workspace/src/cron.ts +12 -0
  179. package/bone/output/workspace/src/db.ts +32 -0
  180. package/bone/output/workspace/src/debug.ts +66 -0
  181. package/bone/output/workspace/src/events.ts +243 -0
  182. package/bone/output/workspace/src/extensions.ts +44 -0
  183. package/bone/output/workspace/src/failure_rules.ts +153 -0
  184. package/bone/output/workspace/src/health.ts +43 -0
  185. package/bone/output/workspace/src/index.ts +89 -0
  186. package/bone/output/workspace/src/logger.ts +66 -0
  187. package/bone/output/workspace/src/metrics.ts +75 -0
  188. package/bone/output/workspace/src/migrate.ts +220 -0
  189. package/bone/output/workspace/src/migration_diff.ts +108 -0
  190. package/bone/output/workspace/src/notify.ts +73 -0
  191. package/bone/output/workspace/src/routes/codebase.ts +87 -0
  192. package/bone/output/workspace/src/routes/snapshot.ts +127 -0
  193. package/bone/output/workspace/src/routes/workspace.ts +190 -0
  194. package/bone/output/workspace/src/schemas.ts +231 -0
  195. package/bone/output/workspace/src/state_machines/codebase.ts +21 -0
  196. package/bone/output/workspace/src/state_machines/snapshot.ts +20 -0
  197. package/bone/output/workspace/src/state_machines/workspace.ts +21 -0
  198. package/bone/output/workspace/src/tests.ts +249 -0
  199. package/bone/output/workspace/tsconfig.json +25 -0
  200. package/compat/opencode_adapter.ts +410 -0
  201. package/package.json +69 -0
  202. package/scripts/check_benchmark_session.js +34 -0
  203. package/scripts/check_finish_event.js +24 -0
  204. package/scripts/check_parts.js +15 -0
  205. package/scripts/compile.js +79 -0
  206. package/scripts/copy_opencode.ps1 +53 -0
  207. package/scripts/create_functions.sql +129 -0
  208. package/scripts/migrate.js +85 -0
  209. package/scripts/migrate_from_opencode.ts +218 -0
  210. package/scripts/test_agent_loop.js +101 -0
  211. package/scripts/test_api.ps1 +116 -0
  212. package/scripts/test_context_builder.js +136 -0
  213. package/scripts/test_context_builder.ts +97 -0
  214. package/scripts/test_rag.js +189 -0
  215. package/scripts/test_stream_events.js +36 -0
  216. package/scripts/test_websocket_and_saga.js +216 -0
  217. package/src/cli.ts +475 -0
  218. package/src/config.ts +162 -0
  219. package/src/context_builder.ts +598 -0
  220. package/src/engine/account/account.sql.ts +39 -0
  221. package/src/engine/account/account.ts +456 -0
  222. package/src/engine/account/repo.ts +166 -0
  223. package/src/engine/account/schema.ts +99 -0
  224. package/src/engine/account/url.ts +8 -0
  225. package/src/engine/acp/README.md +174 -0
  226. package/src/engine/acp/agent.ts +1968 -0
  227. package/src/engine/acp/runtime.ts +22 -0
  228. package/src/engine/acp/session.ts +122 -0
  229. package/src/engine/acp/types.ts +24 -0
  230. package/src/engine/agent/agent.ts +463 -0
  231. package/src/engine/agent/generate.txt +75 -0
  232. package/src/engine/agent/prompt/compaction.txt +9 -0
  233. package/src/engine/agent/prompt/explore.txt +18 -0
  234. package/src/engine/agent/prompt/scout.txt +36 -0
  235. package/src/engine/agent/prompt/summary.txt +11 -0
  236. package/src/engine/agent/prompt/title.txt +44 -0
  237. package/src/engine/agent/subagent-permissions.ts +34 -0
  238. package/src/engine/auth/index.ts +96 -0
  239. package/src/engine/background/background/job.ts +200 -0
  240. package/src/engine/background/job.ts +200 -0
  241. package/src/engine/bus/bus-event.ts +45 -0
  242. package/src/engine/bus/global.ts +22 -0
  243. package/src/engine/bus/index.ts +203 -0
  244. package/src/engine/command/command/index.ts +181 -0
  245. package/src/engine/command/command/template/initialize.txt +66 -0
  246. package/src/engine/command/command/template/review.txt +101 -0
  247. package/src/engine/command/index.ts +181 -0
  248. package/src/engine/command/template/initialize.txt +66 -0
  249. package/src/engine/command/template/review.txt +101 -0
  250. package/src/engine/config/agent.ts +172 -0
  251. package/src/engine/config/attachment.ts +25 -0
  252. package/src/engine/config/command.ts +62 -0
  253. package/src/engine/config/config.ts +833 -0
  254. package/src/engine/config/console-state.ts +14 -0
  255. package/src/engine/config/entry-name.ts +16 -0
  256. package/src/engine/config/error.ts +23 -0
  257. package/src/engine/config/formatter.ts +13 -0
  258. package/src/engine/config/layout.ts +6 -0
  259. package/src/engine/config/lsp.ts +43 -0
  260. package/src/engine/config/managed.ts +71 -0
  261. package/src/engine/config/markdown.ts +96 -0
  262. package/src/engine/config/mcp.ts +56 -0
  263. package/src/engine/config/model-id.ts +5 -0
  264. package/src/engine/config/parse.ts +79 -0
  265. package/src/engine/config/paths.ts +45 -0
  266. package/src/engine/config/permission.ts +58 -0
  267. package/src/engine/config/plugin.ts +84 -0
  268. package/src/engine/config/provider.ts +111 -0
  269. package/src/engine/config/reference.ts +23 -0
  270. package/src/engine/config/server.ts +19 -0
  271. package/src/engine/config/skills.ts +14 -0
  272. package/src/engine/config/variable.ts +90 -0
  273. package/src/engine/control-plane/adapters/index.ts +41 -0
  274. package/src/engine/control-plane/adapters/worktree.ts +96 -0
  275. package/src/engine/control-plane/dev/README.md +19 -0
  276. package/src/engine/control-plane/dev/debug-workspace-plugin.ts +73 -0
  277. package/src/engine/control-plane/schema.ts +14 -0
  278. package/src/engine/control-plane/types.ts +59 -0
  279. package/src/engine/control-plane/util.ts +39 -0
  280. package/src/engine/control-plane/workspace-adapter-runtime.ts +51 -0
  281. package/src/engine/control-plane/workspace-context.ts +26 -0
  282. package/src/engine/control-plane/workspace.sql.ts +20 -0
  283. package/src/engine/control-plane/workspace.ts +1072 -0
  284. package/src/engine/data-migration.ts +161 -0
  285. package/src/engine/effect/app-runtime.ts +143 -0
  286. package/src/engine/effect/bootstrap-runtime.ts +29 -0
  287. package/src/engine/effect/bridge.ts +84 -0
  288. package/src/engine/effect/config-service.ts +67 -0
  289. package/src/engine/effect/instance-ref.ts +11 -0
  290. package/src/engine/effect/instance-registry.ts +12 -0
  291. package/src/engine/effect/instance-state.ts +72 -0
  292. package/src/engine/effect/promise.ts +17 -0
  293. package/src/engine/effect/run-service.ts +47 -0
  294. package/src/engine/effect/runner.ts +217 -0
  295. package/src/engine/effect/runtime-flags.ts +74 -0
  296. package/src/engine/effect/service-use.ts +38 -0
  297. package/src/engine/env/index.ts +37 -0
  298. package/src/engine/event-v2-bridge.ts +89 -0
  299. package/src/engine/file/file/ignore.ts +81 -0
  300. package/src/engine/file/file/index.ts +651 -0
  301. package/src/engine/file/file/protected.ts +59 -0
  302. package/src/engine/file/file/ripgrep.ts +481 -0
  303. package/src/engine/file/file/watcher.ts +167 -0
  304. package/src/engine/file/ignore.ts +81 -0
  305. package/src/engine/file/index.ts +651 -0
  306. package/src/engine/file/protected.ts +59 -0
  307. package/src/engine/file/ripgrep.ts +481 -0
  308. package/src/engine/file/watcher.ts +167 -0
  309. package/src/engine/format/format/formatter.ts +404 -0
  310. package/src/engine/format/format/index.ts +209 -0
  311. package/src/engine/format/formatter.ts +404 -0
  312. package/src/engine/format/index.ts +209 -0
  313. package/src/engine/git/git/index.ts +347 -0
  314. package/src/engine/git/index.ts +347 -0
  315. package/src/engine/id/id.ts +80 -0
  316. package/src/engine/ide/index.ts +70 -0
  317. package/src/engine/image/image/image.ts +176 -0
  318. package/src/engine/image/image.ts +176 -0
  319. package/src/engine/index.ts +251 -0
  320. package/src/engine/installation/index.ts +327 -0
  321. package/src/engine/lsp/client.ts +707 -0
  322. package/src/engine/lsp/diagnostic.ts +29 -0
  323. package/src/engine/lsp/language.ts +121 -0
  324. package/src/engine/lsp/launch.ts +21 -0
  325. package/src/engine/lsp/lsp/client.ts +707 -0
  326. package/src/engine/lsp/lsp/diagnostic.ts +29 -0
  327. package/src/engine/lsp/lsp/language.ts +121 -0
  328. package/src/engine/lsp/lsp/launch.ts +21 -0
  329. package/src/engine/lsp/lsp/lsp.ts +507 -0
  330. package/src/engine/lsp/lsp/server.ts +2064 -0
  331. package/src/engine/lsp/lsp.ts +507 -0
  332. package/src/engine/lsp/server.ts +2064 -0
  333. package/src/engine/mcp/auth.ts +146 -0
  334. package/src/engine/mcp/index.ts +958 -0
  335. package/src/engine/mcp/mcp/auth.ts +146 -0
  336. package/src/engine/mcp/mcp/index.ts +958 -0
  337. package/src/engine/mcp/mcp/oauth-callback.ts +232 -0
  338. package/src/engine/mcp/mcp/oauth-provider.ts +214 -0
  339. package/src/engine/mcp/oauth-callback.ts +232 -0
  340. package/src/engine/mcp/oauth-provider.ts +214 -0
  341. package/src/engine/node.ts +6 -0
  342. package/src/engine/patch/index.ts +689 -0
  343. package/src/engine/patch/patch/index.ts +689 -0
  344. package/src/engine/permission/arity.ts +163 -0
  345. package/src/engine/permission/evaluate.ts +15 -0
  346. package/src/engine/permission/index.ts +306 -0
  347. package/src/engine/permission/permission/arity.ts +163 -0
  348. package/src/engine/permission/permission/evaluate.ts +15 -0
  349. package/src/engine/permission/permission/index.ts +306 -0
  350. package/src/engine/permission/permission/schema.ts +13 -0
  351. package/src/engine/permission/schema.ts +13 -0
  352. package/src/engine/plugin/azure.ts +26 -0
  353. package/src/engine/plugin/cloudflare.ts +76 -0
  354. package/src/engine/plugin/codex.ts +622 -0
  355. package/src/engine/plugin/digitalocean.ts +411 -0
  356. package/src/engine/plugin/github-copilot/copilot.ts +394 -0
  357. package/src/engine/plugin/github-copilot/models.ts +196 -0
  358. package/src/engine/plugin/index.ts +295 -0
  359. package/src/engine/plugin/install.ts +439 -0
  360. package/src/engine/plugin/loader.ts +216 -0
  361. package/src/engine/plugin/meta.ts +188 -0
  362. package/src/engine/plugin/shared.ts +323 -0
  363. package/src/engine/project/bootstrap-service.ts +9 -0
  364. package/src/engine/project/bootstrap.ts +75 -0
  365. package/src/engine/project/instance-context.ts +24 -0
  366. package/src/engine/project/instance-layer.ts +11 -0
  367. package/src/engine/project/instance-runtime.ts +16 -0
  368. package/src/engine/project/instance-store.ts +193 -0
  369. package/src/engine/project/project.sql.ts +17 -0
  370. package/src/engine/project/project.ts +537 -0
  371. package/src/engine/project/schema.ts +13 -0
  372. package/src/engine/project/vcs.ts +405 -0
  373. package/src/engine/provider/auth.ts +225 -0
  374. package/src/engine/provider/error.ts +204 -0
  375. package/src/engine/provider/model-status.ts +8 -0
  376. package/src/engine/provider/provider.ts +1843 -0
  377. package/src/engine/provider/schema.ts +30 -0
  378. package/src/engine/provider/sdk/copilot/AGENTS.md +1 -0
  379. package/src/engine/provider/transform.ts +1376 -0
  380. package/src/engine/pty/index.ts +365 -0
  381. package/src/engine/pty/input.ts +24 -0
  382. package/src/engine/pty/pty/index.ts +365 -0
  383. package/src/engine/pty/pty/input.ts +24 -0
  384. package/src/engine/pty/pty/pty.bun.ts +26 -0
  385. package/src/engine/pty/pty/pty.node.ts +27 -0
  386. package/src/engine/pty/pty/pty.ts +25 -0
  387. package/src/engine/pty/pty/schema.ts +14 -0
  388. package/src/engine/pty/pty/ticket.ts +68 -0
  389. package/src/engine/pty/pty.bun.ts +26 -0
  390. package/src/engine/pty/pty.node.ts +27 -0
  391. package/src/engine/pty/pty.ts +25 -0
  392. package/src/engine/pty/schema.ts +14 -0
  393. package/src/engine/pty/ticket.ts +68 -0
  394. package/src/engine/question/index.ts +213 -0
  395. package/src/engine/question/question/index.ts +213 -0
  396. package/src/engine/question/question/schema.ts +10 -0
  397. package/src/engine/question/schema.ts +10 -0
  398. package/src/engine/reference/reference/reference.ts +241 -0
  399. package/src/engine/reference/reference/repository-cache.ts +147 -0
  400. package/src/engine/reference/reference.ts +241 -0
  401. package/src/engine/reference/repository-cache.ts +147 -0
  402. package/src/engine/session/compaction.ts +651 -0
  403. package/src/engine/session/compaction_logic.ts +120 -0
  404. package/src/engine/session/instruction.ts +238 -0
  405. package/src/engine/session/instruction_loader.ts +54 -0
  406. package/src/engine/session/llm.ts +459 -0
  407. package/src/engine/session/message-error.ts +14 -0
  408. package/src/engine/session/message-v2.ts +1202 -0
  409. package/src/engine/session/message.ts +146 -0
  410. package/src/engine/session/overflow.ts +32 -0
  411. package/src/engine/session/overflow_check.ts +46 -0
  412. package/src/engine/session/processor.ts +823 -0
  413. package/src/engine/session/prompt/anthropic.txt +105 -0
  414. package/src/engine/session/prompt/beast.txt +147 -0
  415. package/src/engine/session/prompt/build-switch.txt +5 -0
  416. package/src/engine/session/prompt/codex.txt +79 -0
  417. package/src/engine/session/prompt/copilot-gpt-5.txt +143 -0
  418. package/src/engine/session/prompt/default.txt +105 -0
  419. package/src/engine/session/prompt/gemini.txt +155 -0
  420. package/src/engine/session/prompt/gpt.txt +107 -0
  421. package/src/engine/session/prompt/kimi.txt +95 -0
  422. package/src/engine/session/prompt/max-steps.txt +16 -0
  423. package/src/engine/session/prompt/plan-reminder-anthropic.txt +67 -0
  424. package/src/engine/session/prompt/plan.txt +26 -0
  425. package/src/engine/session/prompt/trinity.txt +97 -0
  426. package/src/engine/session/prompt.ts +671 -0
  427. package/src/engine/session/provider_transform.ts +187 -0
  428. package/src/engine/session/retry.ts +200 -0
  429. package/src/engine/session/retry_logic.ts +65 -0
  430. package/src/engine/session/revert.ts +162 -0
  431. package/src/engine/session/run-state.ts +153 -0
  432. package/src/engine/session/schema.ts +26 -0
  433. package/src/engine/session/session.sql.ts +137 -0
  434. package/src/engine/session/session.ts +1011 -0
  435. package/src/engine/session/status.ts +94 -0
  436. package/src/engine/session/summary.ts +164 -0
  437. package/src/engine/session/system.ts +84 -0
  438. package/src/engine/session/system_prompt.ts +65 -0
  439. package/src/engine/session/todo.ts +81 -0
  440. package/src/engine/session/tool_registry.ts +162 -0
  441. package/src/engine/share/session.ts +61 -0
  442. package/src/engine/share/share-next.ts +376 -0
  443. package/src/engine/share/share.sql.ts +13 -0
  444. package/src/engine/shell/shell/shell.ts +215 -0
  445. package/src/engine/shell/shell.ts +215 -0
  446. package/src/engine/skill/discovery.ts +116 -0
  447. package/src/engine/skill/index.ts +336 -0
  448. package/src/engine/skill/prompt/customize-opencode.md +377 -0
  449. package/src/engine/skill/skill/discovery.ts +116 -0
  450. package/src/engine/skill/skill/index.ts +336 -0
  451. package/src/engine/skill/skill/prompt/customize-opencode.md +377 -0
  452. package/src/engine/snapshot/index.ts +762 -0
  453. package/src/engine/snapshot/snapshot/index.ts +762 -0
  454. package/src/engine/sync/README.md +179 -0
  455. package/src/engine/sync/event.sql.ts +17 -0
  456. package/src/engine/sync/index.ts +410 -0
  457. package/src/engine/sync/schema.ts +11 -0
  458. package/src/engine/temporary.ts +33 -0
  459. package/src/engine/tool/apply_patch.ts +313 -0
  460. package/src/engine/tool/apply_patch.txt +33 -0
  461. package/src/engine/tool/edit.ts +711 -0
  462. package/src/engine/tool/edit.txt +10 -0
  463. package/src/engine/tool/external-directory.ts +49 -0
  464. package/src/engine/tool/glob.ts +103 -0
  465. package/src/engine/tool/glob.txt +6 -0
  466. package/src/engine/tool/grep.ts +156 -0
  467. package/src/engine/tool/grep.txt +8 -0
  468. package/src/engine/tool/invalid.ts +21 -0
  469. package/src/engine/tool/json-schema.ts +164 -0
  470. package/src/engine/tool/lsp.ts +113 -0
  471. package/src/engine/tool/lsp.txt +24 -0
  472. package/src/engine/tool/mcp-websearch.ts +96 -0
  473. package/src/engine/tool/plan-enter.txt +14 -0
  474. package/src/engine/tool/plan-exit.txt +13 -0
  475. package/src/engine/tool/plan.ts +78 -0
  476. package/src/engine/tool/question.ts +44 -0
  477. package/src/engine/tool/question.txt +10 -0
  478. package/src/engine/tool/read.ts +337 -0
  479. package/src/engine/tool/read.txt +14 -0
  480. package/src/engine/tool/registry.ts +472 -0
  481. package/src/engine/tool/repo_clone.ts +80 -0
  482. package/src/engine/tool/repo_clone.txt +5 -0
  483. package/src/engine/tool/repo_overview.ts +279 -0
  484. package/src/engine/tool/repo_overview.txt +4 -0
  485. package/src/engine/tool/schema.ts +14 -0
  486. package/src/engine/tool/shell/id.ts +19 -0
  487. package/src/engine/tool/shell/prompt.ts +295 -0
  488. package/src/engine/tool/shell/shell.txt +77 -0
  489. package/src/engine/tool/shell.ts +647 -0
  490. package/src/engine/tool/skill.ts +75 -0
  491. package/src/engine/tool/skill.txt +5 -0
  492. package/src/engine/tool/task.ts +337 -0
  493. package/src/engine/tool/task.txt +58 -0
  494. package/src/engine/tool/task_status.ts +179 -0
  495. package/src/engine/tool/task_status.txt +13 -0
  496. package/src/engine/tool/todo.ts +57 -0
  497. package/src/engine/tool/todowrite.txt +167 -0
  498. package/src/engine/tool/tool/apply_patch.ts +313 -0
  499. package/src/engine/tool/tool/apply_patch.txt +33 -0
  500. package/src/engine/tool/tool/edit.ts +711 -0
  501. package/src/engine/tool/tool/edit.txt +10 -0
  502. package/src/engine/tool/tool/external-directory.ts +49 -0
  503. package/src/engine/tool/tool/glob.ts +103 -0
  504. package/src/engine/tool/tool/glob.txt +6 -0
  505. package/src/engine/tool/tool/grep.ts +156 -0
  506. package/src/engine/tool/tool/grep.txt +8 -0
  507. package/src/engine/tool/tool/invalid.ts +21 -0
  508. package/src/engine/tool/tool/json-schema.ts +164 -0
  509. package/src/engine/tool/tool/lsp.ts +113 -0
  510. package/src/engine/tool/tool/lsp.txt +24 -0
  511. package/src/engine/tool/tool/mcp-websearch.ts +96 -0
  512. package/src/engine/tool/tool/plan-enter.txt +14 -0
  513. package/src/engine/tool/tool/plan-exit.txt +13 -0
  514. package/src/engine/tool/tool/plan.ts +78 -0
  515. package/src/engine/tool/tool/question.ts +44 -0
  516. package/src/engine/tool/tool/question.txt +10 -0
  517. package/src/engine/tool/tool/read.ts +337 -0
  518. package/src/engine/tool/tool/read.txt +14 -0
  519. package/src/engine/tool/tool/registry.ts +472 -0
  520. package/src/engine/tool/tool/repo_clone.ts +80 -0
  521. package/src/engine/tool/tool/repo_clone.txt +5 -0
  522. package/src/engine/tool/tool/repo_overview.ts +279 -0
  523. package/src/engine/tool/tool/repo_overview.txt +4 -0
  524. package/src/engine/tool/tool/schema.ts +14 -0
  525. package/src/engine/tool/tool/shell/id.ts +19 -0
  526. package/src/engine/tool/tool/shell/prompt.ts +295 -0
  527. package/src/engine/tool/tool/shell/shell.txt +77 -0
  528. package/src/engine/tool/tool/shell.ts +647 -0
  529. package/src/engine/tool/tool/skill.ts +75 -0
  530. package/src/engine/tool/tool/skill.txt +5 -0
  531. package/src/engine/tool/tool/task.ts +337 -0
  532. package/src/engine/tool/tool/task.txt +58 -0
  533. package/src/engine/tool/tool/task_status.ts +179 -0
  534. package/src/engine/tool/tool/task_status.txt +13 -0
  535. package/src/engine/tool/tool/todo.ts +57 -0
  536. package/src/engine/tool/tool/todowrite.txt +167 -0
  537. package/src/engine/tool/tool/tool.ts +164 -0
  538. package/src/engine/tool/tool/truncate.ts +160 -0
  539. package/src/engine/tool/tool/truncation-dir.ts +4 -0
  540. package/src/engine/tool/tool/webfetch.ts +192 -0
  541. package/src/engine/tool/tool/webfetch.txt +13 -0
  542. package/src/engine/tool/tool/websearch.ts +143 -0
  543. package/src/engine/tool/tool/websearch.txt +14 -0
  544. package/src/engine/tool/tool/write.ts +104 -0
  545. package/src/engine/tool/tool/write.txt +8 -0
  546. package/src/engine/tool/tool.ts +164 -0
  547. package/src/engine/tool/truncate.ts +160 -0
  548. package/src/engine/tool/truncation-dir.ts +4 -0
  549. package/src/engine/tool/webfetch.ts +192 -0
  550. package/src/engine/tool/webfetch.txt +13 -0
  551. package/src/engine/tool/websearch.ts +143 -0
  552. package/src/engine/tool/websearch.txt +14 -0
  553. package/src/engine/tool/write.ts +104 -0
  554. package/src/engine/tool/write.txt +8 -0
  555. package/src/engine/util/archive.ts +17 -0
  556. package/src/engine/util/bom.ts +31 -0
  557. package/src/engine/util/data-url.ts +9 -0
  558. package/src/engine/util/defer.ts +10 -0
  559. package/src/engine/util/effect-http-client.ts +11 -0
  560. package/src/engine/util/error.ts +88 -0
  561. package/src/engine/util/filesystem.ts +252 -0
  562. package/src/engine/util/format.ts +20 -0
  563. package/src/engine/util/iife.ts +3 -0
  564. package/src/engine/util/lazy.ts +20 -0
  565. package/src/engine/util/local-context.ts +25 -0
  566. package/src/engine/util/locale.ts +86 -0
  567. package/src/engine/util/media.ts +26 -0
  568. package/src/engine/util/process.ts +176 -0
  569. package/src/engine/util/queue.ts +32 -0
  570. package/src/engine/util/record.ts +3 -0
  571. package/src/engine/util/repository.ts +158 -0
  572. package/src/engine/util/rpc.ts +66 -0
  573. package/src/engine/util/signal.ts +12 -0
  574. package/src/engine/util/timeout.ts +13 -0
  575. package/src/engine/util/token.ts +7 -0
  576. package/src/engine/util/util/archive.ts +17 -0
  577. package/src/engine/util/util/bom.ts +31 -0
  578. package/src/engine/util/util/data-url.ts +9 -0
  579. package/src/engine/util/util/defer.ts +10 -0
  580. package/src/engine/util/util/effect-http-client.ts +11 -0
  581. package/src/engine/util/util/error.ts +88 -0
  582. package/src/engine/util/util/filesystem.ts +252 -0
  583. package/src/engine/util/util/format.ts +20 -0
  584. package/src/engine/util/util/iife.ts +3 -0
  585. package/src/engine/util/util/lazy.ts +20 -0
  586. package/src/engine/util/util/local-context.ts +25 -0
  587. package/src/engine/util/util/locale.ts +86 -0
  588. package/src/engine/util/util/media.ts +26 -0
  589. package/src/engine/util/util/process.ts +176 -0
  590. package/src/engine/util/util/queue.ts +32 -0
  591. package/src/engine/util/util/record.ts +3 -0
  592. package/src/engine/util/util/repository.ts +158 -0
  593. package/src/engine/util/util/rpc.ts +66 -0
  594. package/src/engine/util/util/signal.ts +12 -0
  595. package/src/engine/util/util/timeout.ts +13 -0
  596. package/src/engine/util/util/token.ts +7 -0
  597. package/src/engine/util/util/which.ts +14 -0
  598. package/src/engine/util/util/wildcard.ts +59 -0
  599. package/src/engine/util/which.ts +14 -0
  600. package/src/engine/util/wildcard.ts +59 -0
  601. package/src/engine/worktree/index.ts +621 -0
  602. package/src/rag_worker.ts +519 -0
  603. package/src/server.ts +201 -0
  604. package/src/tui.ts +637 -0
  605. package/tsconfig.json +24 -0
@@ -0,0 +1,111 @@
1
+ import { Schema } from "effect"
2
+ import { PositiveInt } from "@opencode-ai/core/schema"
3
+ import { ModelStatus } from "@/provider/model-status"
4
+
5
+ export const Model = Schema.Struct({
6
+ id: Schema.optional(Schema.String),
7
+ name: Schema.optional(Schema.String),
8
+ family: Schema.optional(Schema.String),
9
+ release_date: Schema.optional(Schema.String),
10
+ attachment: Schema.optional(Schema.Boolean),
11
+ reasoning: Schema.optional(Schema.Boolean),
12
+ temperature: Schema.optional(Schema.Boolean),
13
+ tool_call: Schema.optional(Schema.Boolean),
14
+ interleaved: Schema.optional(
15
+ Schema.Union([
16
+ Schema.Literal(true),
17
+ Schema.Struct({
18
+ field: Schema.Literals(["reasoning_content", "reasoning_details"]),
19
+ }),
20
+ ]),
21
+ ),
22
+ cost: Schema.optional(
23
+ Schema.Struct({
24
+ input: Schema.Finite,
25
+ output: Schema.Finite,
26
+ cache_read: Schema.optional(Schema.Finite),
27
+ cache_write: Schema.optional(Schema.Finite),
28
+ context_over_200k: Schema.optional(
29
+ Schema.Struct({
30
+ input: Schema.Finite,
31
+ output: Schema.Finite,
32
+ cache_read: Schema.optional(Schema.Finite),
33
+ cache_write: Schema.optional(Schema.Finite),
34
+ }),
35
+ ),
36
+ }),
37
+ ),
38
+ limit: Schema.optional(
39
+ Schema.Struct({
40
+ context: Schema.Finite,
41
+ input: Schema.optional(Schema.Finite),
42
+ output: Schema.Finite,
43
+ }),
44
+ ),
45
+ modalities: Schema.optional(
46
+ Schema.Struct({
47
+ input: Schema.mutable(Schema.Array(Schema.Literals(["text", "audio", "image", "video", "pdf"]))),
48
+ output: Schema.mutable(Schema.Array(Schema.Literals(["text", "audio", "image", "video", "pdf"]))),
49
+ }),
50
+ ),
51
+ experimental: Schema.optional(Schema.Boolean),
52
+ status: Schema.optional(ModelStatus),
53
+ provider: Schema.optional(
54
+ Schema.Struct({ npm: Schema.optional(Schema.String), api: Schema.optional(Schema.String) }),
55
+ ),
56
+ options: Schema.optional(Schema.Record(Schema.String, Schema.Any)),
57
+ headers: Schema.optional(Schema.Record(Schema.String, Schema.String)),
58
+ variants: Schema.optional(
59
+ Schema.Record(
60
+ Schema.String,
61
+ Schema.StructWithRest(
62
+ Schema.Struct({
63
+ disabled: Schema.optional(Schema.Boolean).annotate({ description: "Disable this variant for the model" }),
64
+ }),
65
+ [Schema.Record(Schema.String, Schema.Any)],
66
+ ),
67
+ ).annotate({ description: "Variant-specific configuration" }),
68
+ ),
69
+ })
70
+
71
+ export const Info = Schema.Struct({
72
+ api: Schema.optional(Schema.String),
73
+ name: Schema.optional(Schema.String),
74
+ env: Schema.optional(Schema.mutable(Schema.Array(Schema.String))),
75
+ id: Schema.optional(Schema.String),
76
+ npm: Schema.optional(Schema.String),
77
+ whitelist: Schema.optional(Schema.mutable(Schema.Array(Schema.String))),
78
+ blacklist: Schema.optional(Schema.mutable(Schema.Array(Schema.String))),
79
+ options: Schema.optional(
80
+ Schema.StructWithRest(
81
+ Schema.Struct({
82
+ apiKey: Schema.optional(Schema.String),
83
+ baseURL: Schema.optional(Schema.String),
84
+ enterpriseUrl: Schema.optional(Schema.String).annotate({
85
+ description: "GitHub Enterprise URL for copilot authentication",
86
+ }),
87
+ setCacheKey: Schema.optional(Schema.Boolean).annotate({
88
+ description: "Enable promptCacheKey for this provider (default false)",
89
+ }),
90
+ timeout: Schema.optional(
91
+ Schema.Union([PositiveInt, Schema.Literal(false)]).annotate({
92
+ description:
93
+ "Timeout in milliseconds for requests to this provider. Default is 300000 (5 minutes). Set to false to disable timeout.",
94
+ }),
95
+ ).annotate({
96
+ description:
97
+ "Timeout in milliseconds for requests to this provider. Default is 300000 (5 minutes). Set to false to disable timeout.",
98
+ }),
99
+ chunkTimeout: Schema.optional(PositiveInt).annotate({
100
+ description:
101
+ "Timeout in milliseconds between streamed SSE chunks for this provider. If no chunk arrives within this window, the request is aborted.",
102
+ }),
103
+ }),
104
+ [Schema.Record(Schema.String, Schema.Any)],
105
+ ),
106
+ ),
107
+ models: Schema.optional(Schema.Record(Schema.String, Model)),
108
+ }).annotate({ identifier: "ProviderConfig" })
109
+ export type Info = Schema.Schema.Type<typeof Info>
110
+
111
+ export * as ConfigProvider from "./provider"
@@ -0,0 +1,23 @@
1
+ export * as ConfigReference from "./reference"
2
+
3
+ import { Schema } from "effect"
4
+
5
+ const Git = Schema.Struct({
6
+ repository: Schema.String.annotate({
7
+ description: "Git repository URL, host/path reference, or GitHub owner/repo shorthand",
8
+ }),
9
+ branch: Schema.optional(Schema.String).annotate({
10
+ description: "Branch or ref Scout should clone and inspect",
11
+ }),
12
+ })
13
+
14
+ const Local = Schema.Struct({
15
+ path: Schema.String.annotate({
16
+ description: "Absolute path, ~/ path, or workspace-relative path to a local reference directory",
17
+ }),
18
+ })
19
+
20
+ export const Entry = Schema.Union([Schema.String, Git, Local]).annotate({ identifier: "ReferenceConfigEntry" })
21
+
22
+ export const Info = Schema.Record(Schema.String, Entry).annotate({ identifier: "ReferenceConfig" })
23
+ export type Info = Schema.Schema.Type<typeof Info>
@@ -0,0 +1,19 @@
1
+ import { Schema } from "effect"
2
+ import { PositiveInt } from "@opencode-ai/core/schema"
3
+
4
+ export const Server = Schema.Struct({
5
+ port: Schema.optional(PositiveInt).annotate({
6
+ description: "Port to listen on",
7
+ }),
8
+ hostname: Schema.optional(Schema.String).annotate({ description: "Hostname to listen on" }),
9
+ mdns: Schema.optional(Schema.Boolean).annotate({ description: "Enable mDNS service discovery" }),
10
+ mdnsDomain: Schema.optional(Schema.String).annotate({
11
+ description: "Custom domain name for mDNS service (default: opencode.local)",
12
+ }),
13
+ cors: Schema.optional(Schema.mutable(Schema.Array(Schema.String))).annotate({
14
+ description: "Additional domains to allow for CORS",
15
+ }),
16
+ }).annotate({ identifier: "ServerConfig" })
17
+ export type Server = Schema.Schema.Type<typeof Server>
18
+
19
+ export * as ConfigServer from "./server"
@@ -0,0 +1,14 @@
1
+ import { Schema } from "effect"
2
+
3
+ export const Info = Schema.Struct({
4
+ paths: Schema.optional(Schema.Array(Schema.String)).annotate({
5
+ description: "Additional paths to skill folders",
6
+ }),
7
+ urls: Schema.optional(Schema.Array(Schema.String)).annotate({
8
+ description: "URLs to fetch skills from (e.g., https://example.com/.well-known/skills/)",
9
+ }),
10
+ })
11
+
12
+ export type Info = Schema.Schema.Type<typeof Info>
13
+
14
+ export * as ConfigSkills from "./skills"
@@ -0,0 +1,90 @@
1
+ export * as ConfigVariable from "./variable"
2
+
3
+ import path from "path"
4
+ import os from "os"
5
+ import { Filesystem } from "@/util/filesystem"
6
+ import { InvalidError } from "./error"
7
+
8
+ type ParseSource =
9
+ | {
10
+ type: "path"
11
+ path: string
12
+ }
13
+ | {
14
+ type: "virtual"
15
+ source: string
16
+ dir: string
17
+ }
18
+
19
+ type SubstituteInput = ParseSource & {
20
+ text: string
21
+ missing?: "error" | "empty"
22
+ }
23
+
24
+ function source(input: ParseSource) {
25
+ return input.type === "path" ? input.path : input.source
26
+ }
27
+
28
+ function dir(input: ParseSource) {
29
+ return input.type === "path" ? path.dirname(input.path) : input.dir
30
+ }
31
+
32
+ /** Apply {env:VAR} and {file:path} substitutions to config text. */
33
+ export async function substitute(input: SubstituteInput) {
34
+ const missing = input.missing ?? "error"
35
+ let text = input.text.replace(/\{env:([^}]+)\}/g, (_, varName) => {
36
+ return process.env[varName] || ""
37
+ })
38
+
39
+ const fileMatches = Array.from(text.matchAll(/\{file:[^}]+\}/g))
40
+ if (!fileMatches.length) return text
41
+
42
+ const configDir = dir(input)
43
+ const configSource = source(input)
44
+ let out = ""
45
+ let cursor = 0
46
+
47
+ for (const match of fileMatches) {
48
+ const token = match[0]
49
+ const index = match.index!
50
+ out += text.slice(cursor, index)
51
+
52
+ const lineStart = text.lastIndexOf("\n", index - 1) + 1
53
+ const prefix = text.slice(lineStart, index).trimStart()
54
+ if (prefix.startsWith("//")) {
55
+ out += token
56
+ cursor = index + token.length
57
+ continue
58
+ }
59
+
60
+ let filePath = token.replace(/^\{file:/, "").replace(/\}$/, "")
61
+ if (filePath.startsWith("~/")) {
62
+ filePath = path.join(os.homedir(), filePath.slice(2))
63
+ }
64
+
65
+ const resolvedPath = path.isAbsolute(filePath) ? filePath : path.resolve(configDir, filePath)
66
+ const fileContent = (
67
+ await Filesystem.readText(resolvedPath).catch((error: NodeJS.ErrnoException) => {
68
+ if (missing === "empty") return ""
69
+
70
+ const errMsg = `bad file reference: "${token}"`
71
+ if (error.code === "ENOENT") {
72
+ throw new InvalidError(
73
+ {
74
+ path: configSource,
75
+ message: errMsg + ` ${resolvedPath} does not exist`,
76
+ },
77
+ { cause: error },
78
+ )
79
+ }
80
+ throw new InvalidError({ path: configSource, message: errMsg }, { cause: error })
81
+ })
82
+ ).trim()
83
+
84
+ out += JSON.stringify(fileContent).slice(1, -1)
85
+ cursor = index + token.length
86
+ }
87
+
88
+ out += text.slice(cursor)
89
+ return out
90
+ }
@@ -0,0 +1,41 @@
1
+ import type { ProjectID } from "@/project/schema"
2
+ import type { WorkspaceAdapter, WorkspaceAdapterEntry } from "../types"
3
+ import { WorktreeAdapter } from "./worktree"
4
+
5
+ const BUILTIN: Record<string, WorkspaceAdapter> = {
6
+ worktree: WorktreeAdapter,
7
+ }
8
+
9
+ const state = new Map<ProjectID, Map<string, WorkspaceAdapter>>()
10
+
11
+ export function getAdapter(projectID: ProjectID, type: string): WorkspaceAdapter {
12
+ const custom = state.get(projectID)?.get(type)
13
+ if (custom) return custom
14
+
15
+ const builtin = BUILTIN[type]
16
+ if (builtin) return builtin
17
+
18
+ throw new Error(`Unknown workspace adapter: ${type}`)
19
+ }
20
+
21
+ export function listAdapters(projectID: ProjectID): WorkspaceAdapterEntry[] {
22
+ return registeredAdapters(projectID).map(([type, adapter]) => ({
23
+ type,
24
+ name: adapter.name,
25
+ description: adapter.description,
26
+ }))
27
+ }
28
+
29
+ export function registeredAdapters(projectID: ProjectID): [string, WorkspaceAdapter][] {
30
+ const adapters = new Map(Object.entries(BUILTIN))
31
+ for (const [type, adapter] of state.get(projectID)?.entries() ?? []) adapters.set(type, adapter)
32
+ return [...adapters.entries()]
33
+ }
34
+
35
+ // Plugins can be loaded per-project so we need to scope them. If you
36
+ // want to install a global one pass `ProjectID.global`
37
+ export function registerAdapter(projectID: ProjectID, type: string, adapter: WorkspaceAdapter) {
38
+ const adapters = state.get(projectID) ?? new Map<string, WorkspaceAdapter>()
39
+ adapters.set(type, adapter)
40
+ state.set(projectID, adapters)
41
+ }
@@ -0,0 +1,96 @@
1
+ import { Effect, Schema } from "effect"
2
+ import { InstanceRef, WorkspaceRef } from "@/effect/instance-ref"
3
+ import { type WorkspaceAdapter, type WorkspaceAdapterContext, WorkspaceInfo } from "../types"
4
+
5
+ const WorktreeConfig = Schema.Struct({
6
+ name: WorkspaceInfo.fields.name,
7
+ branch: Schema.optional(Schema.NullOr(Schema.String)),
8
+ directory: Schema.String,
9
+ })
10
+ const decodeWorktreeConfig = Schema.decodeUnknownSync(WorktreeConfig)
11
+
12
+ async function loadWorktree() {
13
+ const [{ AppRuntime }, { Worktree }] = await Promise.all([import("@/effect/app-runtime"), import("@/worktree")])
14
+ return { AppRuntime, Worktree }
15
+ }
16
+
17
+ function requireInstance(context: WorkspaceAdapterContext | undefined) {
18
+ if (!context?.instance) throw new Error("Worktree adapter requires an instance context")
19
+ return context.instance
20
+ }
21
+
22
+ const provideContext = <A, E, R>(effect: Effect.Effect<A, E, R>, context: WorkspaceAdapterContext | undefined) =>
23
+ effect.pipe(
24
+ Effect.provideService(InstanceRef, requireInstance(context)),
25
+ Effect.provideService(WorkspaceRef, context?.workspaceID),
26
+ )
27
+
28
+ export const WorktreeAdapter: WorkspaceAdapter = {
29
+ name: "Worktree",
30
+ description: "Create a git worktree",
31
+ async configure(info, context) {
32
+ const { AppRuntime, Worktree } = await loadWorktree()
33
+ const next = await AppRuntime.runPromise(
34
+ provideContext(
35
+ Worktree.Service.use((svc) => svc.makeWorktreeInfo({ detached: true })),
36
+ context,
37
+ ),
38
+ )
39
+ return {
40
+ ...info,
41
+ name: next.name,
42
+ directory: next.directory,
43
+ }
44
+ },
45
+ async create(info, _env, _from, context) {
46
+ const { AppRuntime, Worktree } = await loadWorktree()
47
+ const config = decodeWorktreeConfig(info)
48
+ await AppRuntime.runPromise(
49
+ provideContext(
50
+ Worktree.Service.use((svc) =>
51
+ svc.createFromInfo({
52
+ name: config.name,
53
+ directory: config.directory,
54
+ ...(config.branch ? { branch: config.branch } : {}),
55
+ }),
56
+ ),
57
+ context,
58
+ ),
59
+ )
60
+ },
61
+ async list(context) {
62
+ const { AppRuntime, Worktree } = await loadWorktree()
63
+ const ctx = requireInstance(context)
64
+ return (
65
+ await AppRuntime.runPromise(
66
+ provideContext(
67
+ Worktree.Service.use((svc) => svc.list()),
68
+ context,
69
+ ),
70
+ )
71
+ ).map((info) => ({
72
+ type: "worktree",
73
+ name: info.name,
74
+ branch: info.branch,
75
+ directory: info.directory,
76
+ projectID: ctx.project.id,
77
+ }))
78
+ },
79
+ async remove(info, context) {
80
+ const { AppRuntime, Worktree } = await loadWorktree()
81
+ const config = decodeWorktreeConfig(info)
82
+ await AppRuntime.runPromise(
83
+ provideContext(
84
+ Worktree.Service.use((svc) => svc.remove({ directory: config.directory })),
85
+ context,
86
+ ),
87
+ )
88
+ },
89
+ target(info) {
90
+ const config = decodeWorktreeConfig(info)
91
+ return {
92
+ type: "local",
93
+ directory: config.directory,
94
+ }
95
+ },
96
+ }
@@ -0,0 +1,19 @@
1
+ This is a plugin to simulate a remote environment locally. Add this to `.opencode/opencode.jsonc`:
2
+
3
+ ```json
4
+ "plugin": ["../packages/opencode/src/control-plane/dev/debug-workspace-plugin.ts"],
5
+ ```
6
+
7
+ In a separate terminal, run a separate OpenCode server. This will act like a remote server and the local instance will proxy all requests to it:
8
+
9
+ ```
10
+ ./packages/opencode/script/run-workspace-server
11
+ ```
12
+
13
+ With the plugin install, you can now run OpenCode and create a `debug` workspace type. This will create a "remote" workspace which talks to the second workspace server started above.
14
+
15
+ How this works:
16
+
17
+ - The workspace server needs to know the workspace id and port to run. It waits for this information to be written to a file and starts the server when the data is written.
18
+ - The debug plugin writes this information in the `create` call to the workspace. So create a `debug` workspace will always kick off a new external server.
19
+ - The server script watches for file changes, so whenver you create a new `debug` workspace it will restart with the new information. This means that there is only ever one working `debug` workspace at a time; when you create a new one all previous sessions will show that it can't connect because previous debug workspaces do not exist.
@@ -0,0 +1,73 @@
1
+ import type { Plugin } from "@opencode-ai/plugin"
2
+ import { rename, writeFile } from "node:fs/promises"
3
+ import { randomInt } from "node:crypto"
4
+ import { setTimeout as sleep } from "node:timers/promises"
5
+
6
+ const DEV_DATA_FILE = "/tmp/opencode-workspace-dev-data.json"
7
+ const DEV_DATA_TEMP_FILE = `${DEV_DATA_FILE}.tmp`
8
+
9
+ async function waitForHealth(port: number) {
10
+ const url = `http://127.0.0.1:${port}/global/health`
11
+ const started = Date.now()
12
+
13
+ while (Date.now() - started < 30_000) {
14
+ try {
15
+ const response = await fetch(url)
16
+ if (response.ok) {
17
+ return
18
+ }
19
+ } catch {}
20
+
21
+ await sleep(250)
22
+ }
23
+
24
+ throw new Error(`Timed out waiting for debug server health check at ${url}`)
25
+ }
26
+
27
+ let PORT: number | undefined
28
+
29
+ async function writeDebugData(port: number, id: string, env: Record<string, string | undefined>) {
30
+ await writeFile(
31
+ DEV_DATA_TEMP_FILE,
32
+ JSON.stringify(
33
+ {
34
+ port,
35
+ id,
36
+ env,
37
+ },
38
+ null,
39
+ 2,
40
+ ),
41
+ )
42
+
43
+ await rename(DEV_DATA_TEMP_FILE, DEV_DATA_FILE)
44
+ }
45
+
46
+ export const DebugWorkspacePlugin: Plugin = async ({ experimental_workspace }) => {
47
+ experimental_workspace.register("debug", {
48
+ name: "Debug",
49
+ description: "Create a debugging server",
50
+ configure(config) {
51
+ return config
52
+ },
53
+ async create(config, env) {
54
+ const port = randomInt(5000, 9001)
55
+ PORT = port
56
+
57
+ await writeDebugData(port, config.id, env)
58
+
59
+ await waitForHealth(port)
60
+ },
61
+ async remove(_config) {},
62
+ target(_config) {
63
+ return {
64
+ type: "remote",
65
+ url: `http://localhost:${PORT!}/`,
66
+ }
67
+ },
68
+ })
69
+
70
+ return {}
71
+ }
72
+
73
+ export default DebugWorkspacePlugin
@@ -0,0 +1,14 @@
1
+ import { Schema } from "effect"
2
+
3
+ import { Identifier } from "@/id/id"
4
+ import { withStatics } from "@opencode-ai/core/schema"
5
+
6
+ const workspaceIdSchema = Schema.String.check(Schema.isStartsWith("wrk")).pipe(Schema.brand("WorkspaceID"))
7
+
8
+ export type WorkspaceID = typeof workspaceIdSchema.Type
9
+
10
+ export const WorkspaceID = workspaceIdSchema.pipe(
11
+ withStatics((schema: typeof workspaceIdSchema) => ({
12
+ ascending: (id?: string) => schema.make(Identifier.ascending("workspace", id)),
13
+ })),
14
+ )
@@ -0,0 +1,59 @@
1
+ import { Schema, Struct } from "effect"
2
+ import { ProjectID } from "@/project/schema"
3
+ import type { InstanceContext } from "@/project/instance-context"
4
+ import { WorkspaceID } from "./schema"
5
+ import type { DeepMutable } from "@opencode-ai/core/schema"
6
+
7
+ export const WorkspaceInfo = Schema.Struct({
8
+ id: WorkspaceID,
9
+ type: Schema.String,
10
+ name: Schema.String,
11
+ branch: Schema.optional(Schema.NullOr(Schema.String)),
12
+ directory: Schema.optional(Schema.NullOr(Schema.String)),
13
+ extra: Schema.optional(Schema.NullOr(Schema.Unknown)),
14
+ projectID: ProjectID,
15
+ }).annotate({ identifier: "Workspace" })
16
+ export type WorkspaceInfo = DeepMutable<Schema.Schema.Type<typeof WorkspaceInfo>>
17
+
18
+ export const WorkspaceListedInfo = Schema.Struct(Struct.omit(WorkspaceInfo.fields, ["id"])).annotate({
19
+ identifier: "WorkspaceListedInfo",
20
+ })
21
+ export type WorkspaceListedInfo = DeepMutable<Schema.Schema.Type<typeof WorkspaceListedInfo>>
22
+
23
+ export const WorkspaceAdapterEntry = Schema.Struct({
24
+ type: Schema.String,
25
+ name: Schema.String,
26
+ description: Schema.String,
27
+ })
28
+ export type WorkspaceAdapterEntry = Schema.Schema.Type<typeof WorkspaceAdapterEntry>
29
+
30
+ export type Target =
31
+ | {
32
+ type: "local"
33
+ directory: string
34
+ }
35
+ | {
36
+ type: "remote"
37
+ url: string | URL
38
+ headers?: HeadersInit
39
+ }
40
+
41
+ export type WorkspaceAdapterContext = {
42
+ readonly instance?: InstanceContext
43
+ readonly workspaceID?: WorkspaceID
44
+ }
45
+
46
+ export type WorkspaceAdapter = {
47
+ name: string
48
+ description: string
49
+ configure(info: WorkspaceInfo, context?: WorkspaceAdapterContext): WorkspaceInfo | Promise<WorkspaceInfo>
50
+ create(
51
+ info: WorkspaceInfo,
52
+ env: Record<string, string | undefined>,
53
+ from?: WorkspaceInfo,
54
+ context?: WorkspaceAdapterContext,
55
+ ): Promise<void>
56
+ list?(context?: WorkspaceAdapterContext): WorkspaceListedInfo[] | Promise<WorkspaceListedInfo[]>
57
+ remove(info: WorkspaceInfo, context?: WorkspaceAdapterContext): Promise<void>
58
+ target(info: WorkspaceInfo, context?: WorkspaceAdapterContext): Target | Promise<Target>
59
+ }
@@ -0,0 +1,39 @@
1
+ import { GlobalBus, type GlobalEvent } from "@/bus/global"
2
+ import { Effect } from "effect"
3
+
4
+ export function waitEvent(input: { timeout: number; signal?: AbortSignal; fn: (event: GlobalEvent) => boolean }) {
5
+ if (input.signal?.aborted) return Effect.fail(input.signal.reason ?? new Error("Request aborted"))
6
+
7
+ return Effect.callback<void, unknown>((resume) => {
8
+ const abort = () => {
9
+ cleanup()
10
+ resume(Effect.fail(input.signal?.reason ?? new Error("Request aborted")))
11
+ }
12
+
13
+ const handler = (event: GlobalEvent) => {
14
+ try {
15
+ if (!input.fn(event)) return
16
+ cleanup()
17
+ resume(Effect.void)
18
+ } catch (error) {
19
+ cleanup()
20
+ resume(Effect.fail(error))
21
+ }
22
+ }
23
+
24
+ const cleanup = () => {
25
+ clearTimeout(timeout)
26
+ GlobalBus.off("event", handler)
27
+ input.signal?.removeEventListener("abort", abort)
28
+ }
29
+
30
+ const timeout = setTimeout(() => {
31
+ cleanup()
32
+ resume(Effect.fail(new Error("Timed out waiting for global event")))
33
+ }, input.timeout)
34
+
35
+ GlobalBus.on("event", handler)
36
+ input.signal?.addEventListener("abort", abort, { once: true })
37
+ return Effect.sync(cleanup)
38
+ })
39
+ }
@@ -0,0 +1,51 @@
1
+ import { Effect } from "effect"
2
+ import { EffectBridge } from "@/effect/bridge"
3
+ import { InstanceRef, WorkspaceRef } from "@/effect/instance-ref"
4
+ import { getAdapter } from "./adapters"
5
+ import type { WorkspaceAdapter, WorkspaceInfo } from "./types"
6
+
7
+ const context = Effect.gen(function* () {
8
+ return {
9
+ instance: yield* InstanceRef,
10
+ workspaceID: yield* WorkspaceRef,
11
+ }
12
+ })
13
+
14
+ export const target = (info: WorkspaceInfo) =>
15
+ Effect.gen(function* () {
16
+ const adapter = getAdapter(info.projectID, info.type)
17
+ const ctx = yield* context
18
+ return yield* EffectBridge.fromPromise(() => adapter.target(info, ctx))
19
+ })
20
+
21
+ export const configure = (adapter: WorkspaceAdapter, info: WorkspaceInfo) =>
22
+ Effect.gen(function* () {
23
+ const ctx = yield* context
24
+ return yield* EffectBridge.fromPromise(() => adapter.configure(info, ctx))
25
+ })
26
+
27
+ export const create = (
28
+ adapter: WorkspaceAdapter,
29
+ info: WorkspaceInfo,
30
+ env: Record<string, string | undefined>,
31
+ from?: WorkspaceInfo,
32
+ ) =>
33
+ Effect.gen(function* () {
34
+ const ctx = yield* context
35
+ return yield* EffectBridge.fromPromise(() => adapter.create(info, env, from, ctx))
36
+ })
37
+
38
+ export const list = (adapter: WorkspaceAdapter) =>
39
+ Effect.gen(function* () {
40
+ const ctx = yield* context
41
+ return yield* EffectBridge.fromPromise(() => Promise.resolve(adapter.list?.(ctx) ?? []))
42
+ })
43
+
44
+ export const remove = (info: WorkspaceInfo) =>
45
+ Effect.gen(function* () {
46
+ const adapter = getAdapter(info.projectID, info.type)
47
+ const ctx = yield* context
48
+ return yield* EffectBridge.fromPromise(() => adapter.remove(info, ctx))
49
+ })
50
+
51
+ export * as WorkspaceAdapterRuntime from "./workspace-adapter-runtime"
@@ -0,0 +1,26 @@
1
+ import { LocalContext } from "@/util/local-context"
2
+ import type { WorkspaceID } from "../control-plane/schema"
3
+
4
+ export interface WorkspaceContext {
5
+ workspaceID: WorkspaceID | undefined
6
+ }
7
+
8
+ const context = LocalContext.create<WorkspaceContext>("instance")
9
+
10
+ export const WorkspaceContext = {
11
+ async provide<R>(input: { workspaceID?: WorkspaceID; fn: () => R }): Promise<R> {
12
+ return context.provide({ workspaceID: input.workspaceID }, () => input.fn())
13
+ },
14
+
15
+ restore<R>(workspaceID: WorkspaceID, fn: () => R): R {
16
+ return context.provide({ workspaceID }, fn)
17
+ },
18
+
19
+ get workspaceID() {
20
+ try {
21
+ return context.use().workspaceID
22
+ } catch {
23
+ return undefined
24
+ }
25
+ },
26
+ }