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,188 @@
1
+ import path from "path"
2
+ import { fileURLToPath } from "url"
3
+
4
+ import { Flag } from "@opencode-ai/core/flag/flag"
5
+ import { Global } from "@opencode-ai/core/global"
6
+ import { Filesystem } from "@/util/filesystem"
7
+ import { Flock } from "@opencode-ai/core/util/flock"
8
+
9
+ import { parsePluginSpecifier, pluginSource } from "./shared"
10
+
11
+ type Source = "file" | "npm"
12
+
13
+ export type Theme = {
14
+ src: string
15
+ dest: string
16
+ mtime?: number
17
+ size?: number
18
+ }
19
+
20
+ export type Entry = {
21
+ id: string
22
+ source: Source
23
+ spec: string
24
+ target: string
25
+ requested?: string
26
+ version?: string
27
+ modified?: number
28
+ first_time: number
29
+ last_time: number
30
+ time_changed: number
31
+ load_count: number
32
+ fingerprint: string
33
+ themes?: Record<string, Theme>
34
+ }
35
+
36
+ export type State = "first" | "updated" | "same"
37
+
38
+ export type Touch = {
39
+ spec: string
40
+ target: string
41
+ id: string
42
+ }
43
+
44
+ type Store = Record<string, Entry>
45
+ type Core = Omit<Entry, "first_time" | "last_time" | "time_changed" | "load_count" | "fingerprint" | "themes">
46
+ type Row = Touch & { core: Core }
47
+
48
+ function storePath() {
49
+ return Flag.OPENCODE_PLUGIN_META_FILE ?? path.join(Global.Path.state, "plugin-meta.json")
50
+ }
51
+
52
+ function lock(file: string) {
53
+ return `plugin-meta:${file}`
54
+ }
55
+
56
+ function fileTarget(spec: string, target: string) {
57
+ if (spec.startsWith("file://")) return fileURLToPath(spec)
58
+ if (target.startsWith("file://")) return fileURLToPath(target)
59
+ return
60
+ }
61
+
62
+ async function modifiedAt(file: string) {
63
+ const stat = await Filesystem.statAsync(file)
64
+ if (!stat) return
65
+ const mtime = stat.mtimeMs
66
+ return Math.floor(typeof mtime === "bigint" ? Number(mtime) : mtime)
67
+ }
68
+
69
+ function resolvedTarget(target: string) {
70
+ if (target.startsWith("file://")) return fileURLToPath(target)
71
+ return target
72
+ }
73
+
74
+ async function npmVersion(target: string) {
75
+ const resolved = resolvedTarget(target)
76
+ const stat = await Filesystem.statAsync(resolved)
77
+ const dir = stat?.isDirectory() ? resolved : path.dirname(resolved)
78
+ return Filesystem.readJson<{ version?: string }>(path.join(dir, "package.json"))
79
+ .then((item) => item.version)
80
+ .catch(() => undefined)
81
+ }
82
+
83
+ async function entryCore(item: Touch): Promise<Core> {
84
+ const spec = item.spec
85
+ const target = item.target
86
+ const source = pluginSource(spec)
87
+ if (source === "file") {
88
+ const file = fileTarget(spec, target)
89
+ return {
90
+ id: item.id,
91
+ source,
92
+ spec,
93
+ target,
94
+ modified: file ? await modifiedAt(file) : undefined,
95
+ }
96
+ }
97
+
98
+ return {
99
+ id: item.id,
100
+ source,
101
+ spec,
102
+ target,
103
+ requested: parsePluginSpecifier(spec).version,
104
+ version: await npmVersion(target),
105
+ }
106
+ }
107
+
108
+ function fingerprint(value: Core) {
109
+ if (value.source === "file") return [value.target, value.modified ?? ""].join("|")
110
+ return [value.target, value.requested ?? "", value.version ?? ""].join("|")
111
+ }
112
+
113
+ async function read(file: string): Promise<Store> {
114
+ return Filesystem.readJson<Store>(file).catch(() => ({}) as Store)
115
+ }
116
+
117
+ async function row(item: Touch): Promise<Row> {
118
+ return {
119
+ ...item,
120
+ core: await entryCore(item),
121
+ }
122
+ }
123
+
124
+ function next(prev: Entry | undefined, core: Core, now: number): { state: State; entry: Entry } {
125
+ const entry: Entry = {
126
+ ...core,
127
+ first_time: prev?.first_time ?? now,
128
+ last_time: now,
129
+ time_changed: prev?.time_changed ?? now,
130
+ load_count: (prev?.load_count ?? 0) + 1,
131
+ fingerprint: fingerprint(core),
132
+ themes: prev?.themes,
133
+ }
134
+ const state: State = !prev ? "first" : prev.fingerprint === entry.fingerprint ? "same" : "updated"
135
+ if (state === "updated") entry.time_changed = now
136
+ return {
137
+ state,
138
+ entry,
139
+ }
140
+ }
141
+
142
+ export async function touchMany(items: Touch[]): Promise<Array<{ state: State; entry: Entry }>> {
143
+ if (!items.length) return []
144
+ const file = storePath()
145
+ const rows = await Promise.all(items.map((item) => row(item)))
146
+
147
+ return Flock.withLock(lock(file), async () => {
148
+ const store = await read(file)
149
+ const now = Date.now()
150
+ const out: Array<{ state: State; entry: Entry }> = []
151
+ for (const item of rows) {
152
+ const hit = next(store[item.id], item.core, now)
153
+ store[item.id] = hit.entry
154
+ out.push(hit)
155
+ }
156
+ await Filesystem.writeJson(file, store)
157
+ return out
158
+ })
159
+ }
160
+
161
+ export async function touch(spec: string, target: string, id: string): Promise<{ state: State; entry: Entry }> {
162
+ return touchMany([{ spec, target, id }]).then((item) => {
163
+ const hit = item[0]
164
+ if (hit) return hit
165
+ throw new Error("Failed to touch plugin metadata.")
166
+ })
167
+ }
168
+
169
+ export async function setTheme(id: string, name: string, theme: Theme): Promise<void> {
170
+ const file = storePath()
171
+ await Flock.withLock(lock(file), async () => {
172
+ const store = await read(file)
173
+ const entry = store[id]
174
+ if (!entry) return
175
+ entry.themes = {
176
+ ...entry.themes,
177
+ [name]: theme,
178
+ }
179
+ await Filesystem.writeJson(file, store)
180
+ })
181
+ }
182
+
183
+ export async function list(): Promise<Store> {
184
+ const file = storePath()
185
+ return Flock.withLock(lock(file), async () => read(file))
186
+ }
187
+
188
+ export * as PluginMeta from "./meta"
@@ -0,0 +1,323 @@
1
+ import path from "path"
2
+ import { fileURLToPath, pathToFileURL } from "url"
3
+ import npa from "npm-package-arg"
4
+ import semver from "semver"
5
+ import { Filesystem } from "@/util/filesystem"
6
+ import { isRecord } from "@/util/record"
7
+ import { Npm } from "@opencode-ai/core/npm"
8
+
9
+ // Old npm package names for plugins that are now built-in
10
+ export const DEPRECATED_PLUGIN_PACKAGES = ["opencode-openai-codex-auth", "opencode-copilot-auth"]
11
+
12
+ export function isDeprecatedPlugin(spec: string) {
13
+ return DEPRECATED_PLUGIN_PACKAGES.some((pkg) => spec.includes(pkg))
14
+ }
15
+
16
+ function parse(spec: string) {
17
+ try {
18
+ return npa(spec)
19
+ } catch {}
20
+ }
21
+
22
+ export function parsePluginSpecifier(spec: string) {
23
+ const hit = parse(spec)
24
+ if (hit?.type === "alias" && !hit.name) {
25
+ const sub = (hit as npa.AliasResult).subSpec
26
+ if (sub?.name) {
27
+ const version = !sub.rawSpec || sub.rawSpec === "*" ? "latest" : sub.rawSpec
28
+ return { pkg: sub.name, version }
29
+ }
30
+ }
31
+ if (!hit?.name) return { pkg: spec, version: "" }
32
+ if (hit.raw === hit.name) return { pkg: hit.name, version: "latest" }
33
+ return { pkg: hit.name, version: hit.rawSpec }
34
+ }
35
+
36
+ export type PluginSource = "file" | "npm"
37
+ export type PluginKind = "server" | "tui"
38
+ type PluginMode = "strict" | "detect"
39
+
40
+ export type PluginPackage = {
41
+ dir: string
42
+ pkg: string
43
+ json: Record<string, unknown>
44
+ }
45
+
46
+ export type PluginEntry = {
47
+ spec: string
48
+ source: PluginSource
49
+ target: string
50
+ pkg?: PluginPackage
51
+ entry?: string
52
+ }
53
+
54
+ const INDEX_FILES = ["index.ts", "index.tsx", "index.js", "index.mjs", "index.cjs"]
55
+
56
+ export function pluginSource(spec: string): PluginSource {
57
+ if (isPathPluginSpec(spec)) return "file"
58
+ return "npm"
59
+ }
60
+
61
+ function resolveExportPath(raw: string, dir: string) {
62
+ if (raw.startsWith("file://")) return fileURLToPath(raw)
63
+ if (path.isAbsolute(raw)) return raw
64
+ return path.resolve(dir, raw)
65
+ }
66
+
67
+ function isAbsolutePath(raw: string) {
68
+ return path.isAbsolute(raw) || /^[A-Za-z]:[\\/]/.test(raw)
69
+ }
70
+
71
+ function extractExportValue(value: unknown): string | undefined {
72
+ if (typeof value === "string") return value
73
+ if (!isRecord(value)) return undefined
74
+ for (const key of ["import", "default"]) {
75
+ const nested = value[key]
76
+ if (typeof nested === "string") return nested
77
+ }
78
+ return undefined
79
+ }
80
+
81
+ function packageMain(pkg: PluginPackage) {
82
+ const value = pkg.json.main
83
+ if (typeof value !== "string") return
84
+ const next = value.trim()
85
+ if (!next) return
86
+ return next
87
+ }
88
+
89
+ function resolvePackageFile(spec: string, raw: string, kind: string, pkg: PluginPackage) {
90
+ const resolved = resolveExportPath(raw, pkg.dir)
91
+ const root = Filesystem.resolve(pkg.dir)
92
+ const next = Filesystem.resolve(resolved)
93
+ if (!Filesystem.contains(root, next)) {
94
+ throw new Error(`Plugin ${spec} resolved ${kind} entry outside plugin directory`)
95
+ }
96
+ return next
97
+ }
98
+
99
+ function resolvePackagePath(spec: string, raw: string, kind: PluginKind, pkg: PluginPackage) {
100
+ return pathToFileURL(resolvePackageFile(spec, raw, kind, pkg)).href
101
+ }
102
+
103
+ function resolvePackageEntrypoint(spec: string, kind: PluginKind, pkg: PluginPackage) {
104
+ const exports = pkg.json.exports
105
+ if (isRecord(exports)) {
106
+ const raw = extractExportValue(exports[`./${kind}`])
107
+ if (raw) return resolvePackagePath(spec, raw, kind, pkg)
108
+ }
109
+
110
+ if (kind !== "server") return
111
+ const main = packageMain(pkg)
112
+ if (!main) return
113
+ return resolvePackagePath(spec, main, kind, pkg)
114
+ }
115
+
116
+ function targetPath(target: string) {
117
+ if (target.startsWith("file://")) return fileURLToPath(target)
118
+ if (path.isAbsolute(target)) return target
119
+ }
120
+
121
+ async function resolveDirectoryIndex(dir: string) {
122
+ for (const name of INDEX_FILES) {
123
+ const file = path.join(dir, name)
124
+ if (await Filesystem.exists(file)) return file
125
+ }
126
+ }
127
+
128
+ async function resolveTargetDirectory(target: string) {
129
+ const file = targetPath(target)
130
+ if (!file) return
131
+ const stat = await Filesystem.statAsync(file)
132
+ if (!stat?.isDirectory()) return
133
+ return file
134
+ }
135
+
136
+ async function resolvePluginEntrypoint(spec: string, target: string, kind: PluginKind, pkg?: PluginPackage) {
137
+ const source = pluginSource(spec)
138
+ const hit =
139
+ pkg ?? (source === "npm" ? await readPluginPackage(target) : await readPluginPackage(target).catch(() => undefined))
140
+ if (!hit) return target
141
+
142
+ const entry = resolvePackageEntrypoint(spec, kind, hit)
143
+ if (entry) return entry
144
+
145
+ const dir = await resolveTargetDirectory(target)
146
+
147
+ if (kind === "tui") {
148
+ if (source === "file" && dir) {
149
+ const index = await resolveDirectoryIndex(dir)
150
+ if (index) return pathToFileURL(index).href
151
+ }
152
+
153
+ if (source === "npm") return
154
+ if (dir) return
155
+
156
+ return target
157
+ }
158
+
159
+ if (dir && isRecord(hit.json.exports)) {
160
+ if (source === "file") {
161
+ const index = await resolveDirectoryIndex(dir)
162
+ if (index) return pathToFileURL(index).href
163
+ }
164
+
165
+ return
166
+ }
167
+
168
+ return target
169
+ }
170
+
171
+ export function isPathPluginSpec(spec: string) {
172
+ return spec.startsWith("file://") || spec.startsWith(".") || isAbsolutePath(spec)
173
+ }
174
+
175
+ export async function resolvePathPluginTarget(spec: string) {
176
+ const raw = spec.startsWith("file://") ? fileURLToPath(spec) : spec
177
+ const file = path.isAbsolute(raw) || /^[A-Za-z]:[\\/]/.test(raw) ? raw : path.resolve(raw)
178
+ const stat = await Filesystem.statAsync(file)
179
+ if (!stat?.isDirectory()) {
180
+ if (spec.startsWith("file://")) return spec
181
+ return pathToFileURL(file).href
182
+ }
183
+
184
+ if (await Filesystem.exists(path.join(file, "package.json"))) {
185
+ return pathToFileURL(file).href
186
+ }
187
+
188
+ const index = await resolveDirectoryIndex(file)
189
+ if (index) return pathToFileURL(index).href
190
+
191
+ throw new Error(`Plugin directory ${file} is missing package.json or index file`)
192
+ }
193
+
194
+ export async function checkPluginCompatibility(target: string, opencodeVersion: string, pkg?: PluginPackage) {
195
+ if (!semver.valid(opencodeVersion) || semver.major(opencodeVersion) === 0) return
196
+ const hit = pkg ?? (await readPluginPackage(target).catch(() => undefined))
197
+ if (!hit) return
198
+ const engines = hit.json.engines
199
+ if (!isRecord(engines)) return
200
+ const range = engines.opencode
201
+ if (typeof range !== "string") return
202
+ if (!semver.satisfies(opencodeVersion, range)) {
203
+ throw new Error(`Plugin requires opencode ${range} but running ${opencodeVersion}`)
204
+ }
205
+ }
206
+
207
+ export async function resolvePluginTarget(spec: string) {
208
+ if (isPathPluginSpec(spec)) return resolvePathPluginTarget(spec)
209
+ const hit = parse(spec)
210
+ const pkg = hit?.name && hit.raw === hit.name ? `${hit.name}@latest` : spec
211
+ const result = await Npm.add(pkg)
212
+ return result.directory
213
+ }
214
+
215
+ export async function readPluginPackage(target: string): Promise<PluginPackage> {
216
+ const file = target.startsWith("file://") ? fileURLToPath(target) : target
217
+ const stat = await Filesystem.statAsync(file)
218
+ const dir = stat?.isDirectory() ? file : path.dirname(file)
219
+ const pkg = path.join(dir, "package.json")
220
+ const json = await Filesystem.readJson<Record<string, unknown>>(pkg)
221
+ return { dir, pkg, json }
222
+ }
223
+
224
+ export async function createPluginEntry(spec: string, target: string, kind: PluginKind): Promise<PluginEntry> {
225
+ const source = pluginSource(spec)
226
+ const pkg =
227
+ source === "npm" ? await readPluginPackage(target) : await readPluginPackage(target).catch(() => undefined)
228
+ const entry = await resolvePluginEntrypoint(spec, target, kind, pkg)
229
+ return {
230
+ spec,
231
+ source,
232
+ target,
233
+ pkg,
234
+ entry,
235
+ }
236
+ }
237
+
238
+ export function readPackageThemes(spec: string, pkg: PluginPackage) {
239
+ const field = pkg.json["oc-themes"]
240
+ if (field === undefined) return []
241
+ if (!Array.isArray(field)) {
242
+ throw new TypeError(`Plugin ${spec} has invalid oc-themes field`)
243
+ }
244
+
245
+ const list = field.map((item) => {
246
+ if (typeof item !== "string") {
247
+ throw new TypeError(`Plugin ${spec} has invalid oc-themes entry`)
248
+ }
249
+
250
+ const raw = item.trim()
251
+ if (!raw) {
252
+ throw new TypeError(`Plugin ${spec} has empty oc-themes entry`)
253
+ }
254
+ if (raw.startsWith("file://") || isAbsolutePath(raw)) {
255
+ throw new TypeError(`Plugin ${spec} oc-themes entry must be relative: ${item}`)
256
+ }
257
+
258
+ return resolvePackageFile(spec, raw, "oc-themes", pkg)
259
+ })
260
+
261
+ return Array.from(new Set(list))
262
+ }
263
+
264
+ export function readPluginId(id: unknown, spec: string) {
265
+ if (id === undefined) return
266
+ if (typeof id !== "string") throw new TypeError(`Plugin ${spec} has invalid id type ${typeof id}`)
267
+ const value = id.trim()
268
+ if (!value) throw new TypeError(`Plugin ${spec} has an empty id`)
269
+ return value
270
+ }
271
+
272
+ export function readV1Plugin(
273
+ mod: Record<string, unknown>,
274
+ spec: string,
275
+ kind: PluginKind,
276
+ mode: PluginMode = "strict",
277
+ ) {
278
+ const value = mod.default
279
+ if (!isRecord(value)) {
280
+ if (mode === "detect") return
281
+ throw new TypeError(`Plugin ${spec} must default export an object with ${kind}()`)
282
+ }
283
+ if (mode === "detect" && !("id" in value) && !("server" in value) && !("tui" in value)) return
284
+
285
+ const server = "server" in value ? value.server : undefined
286
+ const tui = "tui" in value ? value.tui : undefined
287
+ if (server !== undefined && typeof server !== "function") {
288
+ throw new TypeError(`Plugin ${spec} has invalid server export`)
289
+ }
290
+ if (tui !== undefined && typeof tui !== "function") {
291
+ throw new TypeError(`Plugin ${spec} has invalid tui export`)
292
+ }
293
+ if (server !== undefined && tui !== undefined) {
294
+ throw new TypeError(`Plugin ${spec} must default export either server() or tui(), not both`)
295
+ }
296
+ if (kind === "server" && server === undefined) {
297
+ throw new TypeError(`Plugin ${spec} must default export an object with server()`)
298
+ }
299
+ if (kind === "tui" && tui === undefined) {
300
+ throw new TypeError(`Plugin ${spec} must default export an object with tui()`)
301
+ }
302
+
303
+ return value
304
+ }
305
+
306
+ export async function resolvePluginId(
307
+ source: PluginSource,
308
+ spec: string,
309
+ target: string,
310
+ id: string | undefined,
311
+ pkg?: PluginPackage,
312
+ ) {
313
+ if (source === "file") {
314
+ if (id) return id
315
+ throw new TypeError(`Path plugin ${spec} must export id`)
316
+ }
317
+ if (id) return id
318
+ const hit = pkg ?? (await readPluginPackage(target))
319
+ if (typeof hit.json.name !== "string" || !hit.json.name.trim()) {
320
+ throw new TypeError(`Plugin package ${hit.pkg} is missing name`)
321
+ }
322
+ return hit.json.name.trim()
323
+ }
@@ -0,0 +1,9 @@
1
+ import { Context, Effect } from "effect"
2
+
3
+ export interface Interface {
4
+ readonly run: Effect.Effect<void>
5
+ }
6
+
7
+ export class Service extends Context.Service<Service, Interface>()("@opencode/InstanceBootstrap") {}
8
+
9
+ export * as InstanceBootstrap from "./bootstrap-service"
@@ -0,0 +1,75 @@
1
+ import { Plugin } from "../plugin"
2
+ import { Format } from "../format"
3
+ import { LSP } from "@/lsp/lsp"
4
+ import { File } from "../file"
5
+ import { Snapshot } from "../snapshot"
6
+ import * as Project from "./project"
7
+ import * as Vcs from "./vcs"
8
+ import { Bus } from "../bus"
9
+ import { InstanceState } from "@/effect/instance-state"
10
+ import { FileWatcher } from "@/file/watcher"
11
+ import { ShareNext } from "@/share/share-next"
12
+ import { Effect, Layer } from "effect"
13
+ import { Config } from "@/config/config"
14
+ import { Service } from "./bootstrap-service"
15
+ import { Reference } from "@/reference/reference"
16
+
17
+ export { Service } from "./bootstrap-service"
18
+ export type { Interface } from "./bootstrap-service"
19
+
20
+ export const layer = Layer.effect(
21
+ Service,
22
+ Effect.gen(function* () {
23
+ // Yield each bootstrap dep at layer init so `run` itself has R = never.
24
+ // InstanceStore imports only the lightweight tag from bootstrap-service.ts,
25
+ // so it can depend on bootstrap without importing this implementation graph.
26
+ const config = yield* Config.Service
27
+ const file = yield* File.Service
28
+ const fileWatcher = yield* FileWatcher.Service
29
+ const format = yield* Format.Service
30
+ const lsp = yield* LSP.Service
31
+ const plugin = yield* Plugin.Service
32
+ const project = yield* Project.Service
33
+ const reference = yield* Reference.Service
34
+ const shareNext = yield* ShareNext.Service
35
+ const snapshot = yield* Snapshot.Service
36
+ const vcs = yield* Vcs.Service
37
+
38
+ const run = Effect.gen(function* () {
39
+ const ctx = yield* InstanceState.context
40
+ yield* Effect.logInfo("bootstrapping").pipe(Effect.annotateLogs("directory", ctx.directory))
41
+ // everything depends on config so eager load it for nice traces
42
+ yield* config.get()
43
+ // Plugin can mutate config so it has to be initialized before anything else.
44
+ yield* plugin.init()
45
+ // Each service self-manages its own slow work via Effect.forkScoped against
46
+ // its per-instance state scope. We just await materialization here.
47
+ yield* Effect.forEach(
48
+ [reference, lsp, shareNext, format, file, fileWatcher, vcs, snapshot, project],
49
+ (s) => s.init().pipe(Effect.catchCause((cause) => Effect.logWarning("init failed", { cause }))),
50
+ { concurrency: "unbounded", discard: true },
51
+ ).pipe(Effect.withSpan("InstanceBootstrap.init"))
52
+ }).pipe(Effect.withSpan("InstanceBootstrap"))
53
+
54
+ return Service.of({ run })
55
+ }),
56
+ )
57
+
58
+ export const defaultLayer: Layer.Layer<Service> = layer.pipe(
59
+ Layer.provide([
60
+ Bus.layer,
61
+ Config.defaultLayer,
62
+ File.defaultLayer,
63
+ FileWatcher.defaultLayer,
64
+ Format.defaultLayer,
65
+ LSP.defaultLayer,
66
+ Plugin.defaultLayer,
67
+ Project.defaultLayer,
68
+ Reference.defaultLayer,
69
+ ShareNext.defaultLayer,
70
+ Snapshot.defaultLayer,
71
+ Vcs.defaultLayer,
72
+ ]),
73
+ )
74
+
75
+ export * as InstanceBootstrap from "./bootstrap"
@@ -0,0 +1,24 @@
1
+ import { LocalContext } from "@/util/local-context"
2
+ import { AppFileSystem } from "@opencode-ai/core/filesystem"
3
+ import type * as Project from "./project"
4
+
5
+ export interface InstanceContext {
6
+ directory: string
7
+ worktree: string
8
+ project: Project.Info
9
+ }
10
+
11
+ export const context = LocalContext.create<InstanceContext>("instance")
12
+
13
+ /**
14
+ * Check if a path is within the project boundary.
15
+ * Returns true if path is inside ctx.directory OR ctx.worktree.
16
+ * Paths within the worktree but outside the working directory should not trigger external_directory permission.
17
+ */
18
+ export function containsPath(filepath: string, ctx: InstanceContext): boolean {
19
+ if (AppFileSystem.contains(ctx.directory, filepath)) return true
20
+ // Non-git projects set worktree to "/" which would match ANY absolute path.
21
+ // Skip worktree check in this case to preserve external_directory permissions.
22
+ if (ctx.worktree === "/") return false
23
+ return AppFileSystem.contains(ctx.worktree, filepath)
24
+ }
@@ -0,0 +1,11 @@
1
+ import { Effect, Layer } from "effect"
2
+ import { InstanceStore } from "./instance-store"
3
+
4
+ export const layer = Layer.unwrap(
5
+ Effect.promise(async () => {
6
+ const { InstanceBootstrap } = await import("./bootstrap")
7
+ return InstanceStore.defaultLayer.pipe(Layer.provide(InstanceBootstrap.defaultLayer))
8
+ }),
9
+ )
10
+
11
+ export * as InstanceLayer from "./instance-layer"
@@ -0,0 +1,16 @@
1
+ import { AppRuntime } from "@/effect/app-runtime"
2
+ import { type InstanceContext } from "./instance-context"
3
+ import { InstanceStore, type LoadInput } from "./instance-store"
4
+
5
+ // Bridge for Promise/ALS callers that cannot yet yield InstanceStore.Service.
6
+ // Delete this module once those callers are migrated to Effect boundaries that
7
+ // provide InstanceStore directly.
8
+
9
+ export const load = (input: LoadInput) => AppRuntime.runPromise(InstanceStore.Service.use((store) => store.load(input)))
10
+ export const disposeInstance = (ctx: InstanceContext) =>
11
+ AppRuntime.runPromise(InstanceStore.Service.use((store) => store.dispose(ctx)))
12
+ export const disposeAllInstances = () => AppRuntime.runPromise(InstanceStore.Service.use((store) => store.disposeAll()))
13
+ export const reloadInstance = (input: LoadInput) =>
14
+ AppRuntime.runPromise(InstanceStore.Service.use((store) => store.reload(input)))
15
+
16
+ export * as InstanceRuntime from "./instance-runtime"