onlycode 1.18.0 → 1.20.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 (418) hide show
  1. package/bin/{opencode.cjs → onlycode.cjs} +8 -42
  2. package/package.json +12 -147
  3. package/src/account/account.ts +0 -463
  4. package/src/account/repo.ts +0 -173
  5. package/src/account/schema.ts +0 -99
  6. package/src/account/url.ts +0 -8
  7. package/src/acp/agent.ts +0 -95
  8. package/src/acp/config-option.ts +0 -203
  9. package/src/acp/content.ts +0 -250
  10. package/src/acp/directory.ts +0 -210
  11. package/src/acp/error.ts +0 -90
  12. package/src/acp/event.ts +0 -342
  13. package/src/acp/permission.ts +0 -124
  14. package/src/acp/profile.ts +0 -42
  15. package/src/acp/service.ts +0 -1048
  16. package/src/acp/session.ts +0 -231
  17. package/src/acp/tool.ts +0 -367
  18. package/src/acp/usage.ts +0 -232
  19. package/src/agent/agent.ts +0 -459
  20. package/src/agent/generate.txt +0 -75
  21. package/src/agent/prompt/compaction.txt +0 -9
  22. package/src/agent/prompt/explore.txt +0 -18
  23. package/src/agent/prompt/summary.txt +0 -11
  24. package/src/agent/prompt/title.txt +0 -44
  25. package/src/agent/subagent-permissions.ts +0 -27
  26. package/src/audio.d.ts +0 -14
  27. package/src/auth/index.ts +0 -99
  28. package/src/background/job.ts +0 -39
  29. package/src/bus/global.ts +0 -22
  30. package/src/cli/bootstrap.ts +0 -11
  31. package/src/cli/cmd/account.ts +0 -264
  32. package/src/cli/cmd/acp.ts +0 -73
  33. package/src/cli/cmd/agent.ts +0 -259
  34. package/src/cli/cmd/attach.ts +0 -97
  35. package/src/cli/cmd/cmd.ts +0 -7
  36. package/src/cli/cmd/db.ts +0 -62
  37. package/src/cli/cmd/debug/agent.handler.ts +0 -193
  38. package/src/cli/cmd/debug/agent.ts +0 -27
  39. package/src/cli/cmd/debug/config.ts +0 -14
  40. package/src/cli/cmd/debug/file.ts +0 -73
  41. package/src/cli/cmd/debug/index.ts +0 -87
  42. package/src/cli/cmd/debug/lsp.ts +0 -50
  43. package/src/cli/cmd/debug/ripgrep.ts +0 -79
  44. package/src/cli/cmd/debug/scrap.ts +0 -15
  45. package/src/cli/cmd/debug/skill.ts +0 -15
  46. package/src/cli/cmd/debug/snapshot.ts +0 -50
  47. package/src/cli/cmd/debug/startup.ts +0 -11
  48. package/src/cli/cmd/debug/v2.ts +0 -44
  49. package/src/cli/cmd/export.ts +0 -292
  50. package/src/cli/cmd/generate.ts +0 -54
  51. package/src/cli/cmd/github.handler.ts +0 -1593
  52. package/src/cli/cmd/github.shared.ts +0 -30
  53. package/src/cli/cmd/github.ts +0 -42
  54. package/src/cli/cmd/import.ts +0 -224
  55. package/src/cli/cmd/mcp.ts +0 -849
  56. package/src/cli/cmd/models.ts +0 -66
  57. package/src/cli/cmd/plug.ts +0 -230
  58. package/src/cli/cmd/pr.ts +0 -115
  59. package/src/cli/cmd/prompt-display.ts +0 -1
  60. package/src/cli/cmd/providers.ts +0 -534
  61. package/src/cli/cmd/run/demo.ts +0 -1274
  62. package/src/cli/cmd/run/entry.body.ts +0 -205
  63. package/src/cli/cmd/run/footer.command.tsx +0 -1064
  64. package/src/cli/cmd/run/footer.menu.tsx +0 -351
  65. package/src/cli/cmd/run/footer.permission.tsx +0 -472
  66. package/src/cli/cmd/run/footer.prompt.tsx +0 -1306
  67. package/src/cli/cmd/run/footer.question.tsx +0 -573
  68. package/src/cli/cmd/run/footer.subagent.tsx +0 -173
  69. package/src/cli/cmd/run/footer.ts +0 -1129
  70. package/src/cli/cmd/run/footer.view.tsx +0 -943
  71. package/src/cli/cmd/run/footer.width.ts +0 -27
  72. package/src/cli/cmd/run/permission.shared.ts +0 -256
  73. package/src/cli/cmd/run/prompt.editor.ts +0 -157
  74. package/src/cli/cmd/run/prompt.shared.ts +0 -153
  75. package/src/cli/cmd/run/question.shared.ts +0 -340
  76. package/src/cli/cmd/run/runtime.boot.ts +0 -202
  77. package/src/cli/cmd/run/runtime.lifecycle.ts +0 -406
  78. package/src/cli/cmd/run/runtime.queue.ts +0 -349
  79. package/src/cli/cmd/run/runtime.shared.ts +0 -17
  80. package/src/cli/cmd/run/runtime.stdin.ts +0 -37
  81. package/src/cli/cmd/run/runtime.ts +0 -814
  82. package/src/cli/cmd/run/scrollback.shared.ts +0 -92
  83. package/src/cli/cmd/run/scrollback.surface.ts +0 -431
  84. package/src/cli/cmd/run/scrollback.writer.tsx +0 -352
  85. package/src/cli/cmd/run/session-data.ts +0 -1113
  86. package/src/cli/cmd/run/session-replay.ts +0 -374
  87. package/src/cli/cmd/run/session.shared.ts +0 -196
  88. package/src/cli/cmd/run/splash.ts +0 -275
  89. package/src/cli/cmd/run/stream.transport.ts +0 -1462
  90. package/src/cli/cmd/run/stream.ts +0 -175
  91. package/src/cli/cmd/run/subagent-data.ts +0 -876
  92. package/src/cli/cmd/run/theme.ts +0 -690
  93. package/src/cli/cmd/run/tool.ts +0 -1489
  94. package/src/cli/cmd/run/trace.ts +0 -94
  95. package/src/cli/cmd/run/turn-summary.ts +0 -47
  96. package/src/cli/cmd/run/types.ts +0 -350
  97. package/src/cli/cmd/run/variant.shared.ts +0 -215
  98. package/src/cli/cmd/run.ts +0 -894
  99. package/src/cli/cmd/serve.ts +0 -24
  100. package/src/cli/cmd/session.ts +0 -147
  101. package/src/cli/cmd/stats.ts +0 -393
  102. package/src/cli/cmd/tui.ts +0 -224
  103. package/src/cli/cmd/uninstall.ts +0 -353
  104. package/src/cli/cmd/upgrade.ts +0 -74
  105. package/src/cli/cmd/web.ts +0 -84
  106. package/src/cli/effect/prompt.ts +0 -37
  107. package/src/cli/effect-cmd.ts +0 -96
  108. package/src/cli/error.ts +0 -130
  109. package/src/cli/heap.ts +0 -45
  110. package/src/cli/logo-pixel.ts +0 -35
  111. package/src/cli/logo.ts +0 -1
  112. package/src/cli/network.ts +0 -64
  113. package/src/cli/tui/layer.ts +0 -7
  114. package/src/cli/tui/validate-session.ts +0 -29
  115. package/src/cli/tui/worker.ts +0 -71
  116. package/src/cli/ui.ts +0 -148
  117. package/src/cli/upgrade.ts +0 -53
  118. package/src/command/index.ts +0 -184
  119. package/src/command/template/initialize.txt +0 -66
  120. package/src/command/template/review.txt +0 -101
  121. package/src/config/agent.ts +0 -59
  122. package/src/config/command.ts +0 -39
  123. package/src/config/config.ts +0 -686
  124. package/src/config/entry-name.ts +0 -19
  125. package/src/config/managed.ts +0 -69
  126. package/src/config/markdown.ts +0 -36
  127. package/src/config/parse.ts +0 -79
  128. package/src/config/paths.ts +0 -45
  129. package/src/config/plugin.ts +0 -79
  130. package/src/config/tui-cwd.ts +0 -5
  131. package/src/config/tui-host-attention.ts +0 -21
  132. package/src/config/tui-migrate.ts +0 -132
  133. package/src/config/tui.ts +0 -274
  134. package/src/config/variable.ts +0 -91
  135. package/src/control-plane/adapters/index.ts +0 -41
  136. package/src/control-plane/adapters/worktree.ts +0 -96
  137. package/src/control-plane/dev/README.md +0 -19
  138. package/src/control-plane/dev/debug-workspace-plugin.ts +0 -73
  139. package/src/control-plane/types.ts +0 -59
  140. package/src/control-plane/util.ts +0 -39
  141. package/src/control-plane/workspace-adapter-runtime.ts +0 -51
  142. package/src/control-plane/workspace-context.ts +0 -26
  143. package/src/control-plane/workspace.ts +0 -989
  144. package/src/effect/app-runtime.ts +0 -132
  145. package/src/effect/bootstrap-runtime.ts +0 -23
  146. package/src/effect/bridge.ts +0 -84
  147. package/src/effect/config-service.ts +0 -67
  148. package/src/effect/instance-ref.ts +0 -11
  149. package/src/effect/instance-registry.ts +0 -12
  150. package/src/effect/instance-state.ts +0 -69
  151. package/src/effect/promise.ts +0 -17
  152. package/src/effect/run-service.ts +0 -47
  153. package/src/effect/runner.ts +0 -217
  154. package/src/effect/runtime-flags.ts +0 -79
  155. package/src/env/index.ts +0 -43
  156. package/src/event-v2-bridge.ts +0 -79
  157. package/src/format/formatter.ts +0 -404
  158. package/src/format/index.ts +0 -205
  159. package/src/git/index.ts +0 -350
  160. package/src/id/id.ts +0 -80
  161. package/src/ide/index.ts +0 -61
  162. package/src/image/image.ts +0 -174
  163. package/src/index.ts +0 -142
  164. package/src/installation/index.ts +0 -350
  165. package/src/lsp/client.ts +0 -650
  166. package/src/lsp/diagnostic.ts +0 -29
  167. package/src/lsp/language.ts +0 -121
  168. package/src/lsp/launch.ts +0 -21
  169. package/src/lsp/lsp.ts +0 -511
  170. package/src/lsp/server.ts +0 -1983
  171. package/src/markdown.d.ts +0 -4
  172. package/src/mcp/auth.ts +0 -174
  173. package/src/mcp/catalog.ts +0 -153
  174. package/src/mcp/index.ts +0 -953
  175. package/src/mcp/oauth-callback.ts +0 -233
  176. package/src/mcp/oauth-provider.ts +0 -206
  177. package/src/node.ts +0 -4
  178. package/src/patch/index.ts +0 -686
  179. package/src/permission/arity.ts +0 -163
  180. package/src/permission/evaluate.ts +0 -1
  181. package/src/permission/index.ts +0 -230
  182. package/src/plugin/azure.ts +0 -26
  183. package/src/plugin/cloudflare.ts +0 -76
  184. package/src/plugin/digitalocean.ts +0 -383
  185. package/src/plugin/github-copilot/copilot.ts +0 -414
  186. package/src/plugin/github-copilot/models.ts +0 -246
  187. package/src/plugin/index.ts +0 -316
  188. package/src/plugin/install.ts +0 -439
  189. package/src/plugin/loader.ts +0 -237
  190. package/src/plugin/meta.ts +0 -188
  191. package/src/plugin/openai/README.md +0 -31
  192. package/src/plugin/openai/codex.ts +0 -641
  193. package/src/plugin/openai/ws-pool.ts +0 -270
  194. package/src/plugin/openai/ws.ts +0 -381
  195. package/src/plugin/pty-environment.ts +0 -24
  196. package/src/plugin/shared.ts +0 -323
  197. package/src/plugin/snowflake-cortex.ts +0 -529
  198. package/src/plugin/tui/internal.ts +0 -10
  199. package/src/plugin/tui/runtime.ts +0 -1130
  200. package/src/plugin/xai.ts +0 -716
  201. package/src/project/bootstrap-service.ts +0 -9
  202. package/src/project/bootstrap.ts +0 -76
  203. package/src/project/instance-context.ts +0 -24
  204. package/src/project/instance-layer.ts +0 -11
  205. package/src/project/instance-runtime.ts +0 -16
  206. package/src/project/instance-store.ts +0 -209
  207. package/src/project/project.ts +0 -519
  208. package/src/project/vcs.ts +0 -431
  209. package/src/provider/auth.ts +0 -233
  210. package/src/provider/error.ts +0 -188
  211. package/src/provider/model-status.ts +0 -8
  212. package/src/provider/provider.ts +0 -1975
  213. package/src/provider/transform.ts +0 -1543
  214. package/src/question/index.ts +0 -229
  215. package/src/question/schema.ts +0 -10
  216. package/src/server/auth.ts +0 -48
  217. package/src/server/event.ts +0 -13
  218. package/src/server/global-lifecycle.ts +0 -28
  219. package/src/server/init-projectors.ts +0 -3
  220. package/src/server/mdns.ts +0 -47
  221. package/src/server/projectors.ts +0 -1
  222. package/src/server/proxy-util.ts +0 -48
  223. package/src/server/routes/instance/httpapi/AGENTS.md +0 -39
  224. package/src/server/routes/instance/httpapi/api.ts +0 -78
  225. package/src/server/routes/instance/httpapi/errors.ts +0 -193
  226. package/src/server/routes/instance/httpapi/groups/config.ts +0 -65
  227. package/src/server/routes/instance/httpapi/groups/control-plane.ts +0 -35
  228. package/src/server/routes/instance/httpapi/groups/control.ts +0 -76
  229. package/src/server/routes/instance/httpapi/groups/event.ts +0 -29
  230. package/src/server/routes/instance/httpapi/groups/experimental.ts +0 -275
  231. package/src/server/routes/instance/httpapi/groups/file.ts +0 -185
  232. package/src/server/routes/instance/httpapi/groups/global.ts +0 -138
  233. package/src/server/routes/instance/httpapi/groups/instance.ts +0 -206
  234. package/src/server/routes/instance/httpapi/groups/mcp.ts +0 -156
  235. package/src/server/routes/instance/httpapi/groups/metadata.ts +0 -18
  236. package/src/server/routes/instance/httpapi/groups/permission.ts +0 -61
  237. package/src/server/routes/instance/httpapi/groups/project-copy.ts +0 -32
  238. package/src/server/routes/instance/httpapi/groups/project.ts +0 -93
  239. package/src/server/routes/instance/httpapi/groups/provider.ts +0 -101
  240. package/src/server/routes/instance/httpapi/groups/pty.ts +0 -172
  241. package/src/server/routes/instance/httpapi/groups/query.ts +0 -12
  242. package/src/server/routes/instance/httpapi/groups/question.ts +0 -74
  243. package/src/server/routes/instance/httpapi/groups/session.ts +0 -462
  244. package/src/server/routes/instance/httpapi/groups/sync.ts +0 -113
  245. package/src/server/routes/instance/httpapi/groups/tui.ts +0 -208
  246. package/src/server/routes/instance/httpapi/groups/workspace.ts +0 -141
  247. package/src/server/routes/instance/httpapi/handlers/config.ts +0 -34
  248. package/src/server/routes/instance/httpapi/handlers/control-plane.ts +0 -37
  249. package/src/server/routes/instance/httpapi/handlers/control.ts +0 -43
  250. package/src/server/routes/instance/httpapi/handlers/event.ts +0 -99
  251. package/src/server/routes/instance/httpapi/handlers/experimental.ts +0 -192
  252. package/src/server/routes/instance/httpapi/handlers/file.ts +0 -139
  253. package/src/server/routes/instance/httpapi/handlers/global.ts +0 -156
  254. package/src/server/routes/instance/httpapi/handlers/instance.ts +0 -110
  255. package/src/server/routes/instance/httpapi/handlers/mcp.ts +0 -111
  256. package/src/server/routes/instance/httpapi/handlers/permission.ts +0 -41
  257. package/src/server/routes/instance/httpapi/handlers/project-copy.ts +0 -83
  258. package/src/server/routes/instance/httpapi/handlers/project.ts +0 -63
  259. package/src/server/routes/instance/httpapi/handlers/provider.ts +0 -113
  260. package/src/server/routes/instance/httpapi/handlers/pty.ts +0 -273
  261. package/src/server/routes/instance/httpapi/handlers/question.ts +0 -54
  262. package/src/server/routes/instance/httpapi/handlers/session-errors.ts +0 -21
  263. package/src/server/routes/instance/httpapi/handlers/session.ts +0 -440
  264. package/src/server/routes/instance/httpapi/handlers/sync.ts +0 -89
  265. package/src/server/routes/instance/httpapi/handlers/tui.ts +0 -131
  266. package/src/server/routes/instance/httpapi/handlers/workspace.ts +0 -102
  267. package/src/server/routes/instance/httpapi/lifecycle.ts +0 -54
  268. package/src/server/routes/instance/httpapi/middleware/authorization.ts +0 -150
  269. package/src/server/routes/instance/httpapi/middleware/compression.ts +0 -64
  270. package/src/server/routes/instance/httpapi/middleware/cors-vary.ts +0 -29
  271. package/src/server/routes/instance/httpapi/middleware/error.ts +0 -43
  272. package/src/server/routes/instance/httpapi/middleware/fence.ts +0 -25
  273. package/src/server/routes/instance/httpapi/middleware/instance-context.ts +0 -43
  274. package/src/server/routes/instance/httpapi/middleware/proxy.ts +0 -108
  275. package/src/server/routes/instance/httpapi/middleware/schema-error.ts +0 -41
  276. package/src/server/routes/instance/httpapi/middleware/workspace-routing.ts +0 -250
  277. package/src/server/routes/instance/httpapi/public.ts +0 -535
  278. package/src/server/routes/instance/httpapi/server.ts +0 -298
  279. package/src/server/routes/instance/httpapi/websocket-tracker.ts +0 -57
  280. package/src/server/server.ts +0 -225
  281. package/src/server/shared/fence.ts +0 -60
  282. package/src/server/shared/pty-ticket.ts +0 -15
  283. package/src/server/shared/public-ui.ts +0 -12
  284. package/src/server/shared/tui-control.ts +0 -28
  285. package/src/server/shared/ui.ts +0 -108
  286. package/src/server/shared/workspace-routing.ts +0 -38
  287. package/src/server/tui-event.ts +0 -53
  288. package/src/session/compaction.ts +0 -620
  289. package/src/session/instruction.ts +0 -241
  290. package/src/session/llm/AGENTS.md +0 -90
  291. package/src/session/llm/ai-sdk.ts +0 -288
  292. package/src/session/llm/native-request.ts +0 -196
  293. package/src/session/llm/native-runtime.ts +0 -195
  294. package/src/session/llm/request.ts +0 -216
  295. package/src/session/llm.ts +0 -415
  296. package/src/session/message-error.ts +0 -14
  297. package/src/session/message-v2.ts +0 -747
  298. package/src/session/message.ts +0 -148
  299. package/src/session/overflow.ts +0 -34
  300. package/src/session/processor.ts +0 -1084
  301. package/src/session/prompt/anthropic.txt +0 -109
  302. package/src/session/prompt/beast.txt +0 -151
  303. package/src/session/prompt/build-switch.txt +0 -9
  304. package/src/session/prompt/codex.txt +0 -83
  305. package/src/session/prompt/copilot-gpt-5.txt +0 -147
  306. package/src/session/prompt/default.txt +0 -99
  307. package/src/session/prompt/gemini.txt +0 -159
  308. package/src/session/prompt/gpt.txt +0 -111
  309. package/src/session/prompt/kimi.txt +0 -99
  310. package/src/session/prompt/plan-mode.txt +0 -74
  311. package/src/session/prompt/plan-reminder-anthropic.txt +0 -71
  312. package/src/session/prompt/plan.txt +0 -30
  313. package/src/session/prompt/trinity.txt +0 -101
  314. package/src/session/prompt.ts +0 -1707
  315. package/src/session/reminders.ts +0 -92
  316. package/src/session/retry.ts +0 -201
  317. package/src/session/revert.ts +0 -160
  318. package/src/session/run-state.ts +0 -156
  319. package/src/session/schema.ts +0 -26
  320. package/src/session/session.ts +0 -1119
  321. package/src/session/status.ts +0 -97
  322. package/src/session/summary.ts +0 -165
  323. package/src/session/system.ts +0 -117
  324. package/src/session/todo.ts +0 -90
  325. package/src/session/tools.ts +0 -207
  326. package/src/share/session.ts +0 -61
  327. package/src/share/share-next.ts +0 -385
  328. package/src/skill/discovery.ts +0 -109
  329. package/src/skill/index.ts +0 -366
  330. package/src/snapshot/index.ts +0 -808
  331. package/src/sql.d.ts +0 -4
  332. package/src/storage/schema.ts +0 -5
  333. package/src/storage/storage.ts +0 -329
  334. package/src/sync/README.md +0 -179
  335. package/src/sync/schema.ts +0 -11
  336. package/src/temporary.ts +0 -31
  337. package/src/tool/apply_patch.ts +0 -315
  338. package/src/tool/apply_patch.txt +0 -33
  339. package/src/tool/apply_patch.zh.txt +0 -33
  340. package/src/tool/description.ts +0 -100
  341. package/src/tool/edit.ts +0 -739
  342. package/src/tool/edit.txt +0 -10
  343. package/src/tool/edit.zh.txt +0 -10
  344. package/src/tool/external-directory.ts +0 -49
  345. package/src/tool/glob.ts +0 -78
  346. package/src/tool/glob.txt +0 -6
  347. package/src/tool/glob.zh.txt +0 -6
  348. package/src/tool/grep.ts +0 -114
  349. package/src/tool/grep.txt +0 -8
  350. package/src/tool/grep.zh.txt +0 -8
  351. package/src/tool/invalid.ts +0 -21
  352. package/src/tool/json-schema.ts +0 -164
  353. package/src/tool/lsp.ts +0 -115
  354. package/src/tool/lsp.txt +0 -24
  355. package/src/tool/lsp.zh.txt +0 -24
  356. package/src/tool/mcp-websearch.ts +0 -96
  357. package/src/tool/plan-enter.txt +0 -14
  358. package/src/tool/plan-enter.zh.txt +0 -14
  359. package/src/tool/plan-exit.txt +0 -13
  360. package/src/tool/plan-exit.zh.txt +0 -13
  361. package/src/tool/plan.ts +0 -81
  362. package/src/tool/question.ts +0 -46
  363. package/src/tool/question.txt +0 -10
  364. package/src/tool/question.zh.txt +0 -10
  365. package/src/tool/read.ts +0 -388
  366. package/src/tool/read.txt +0 -14
  367. package/src/tool/read.zh.txt +0 -14
  368. package/src/tool/registry.ts +0 -440
  369. package/src/tool/schema.ts +0 -14
  370. package/src/tool/shell/id.ts +0 -19
  371. package/src/tool/shell/prompt.ts +0 -307
  372. package/src/tool/shell/shell.txt +0 -21
  373. package/src/tool/shell.ts +0 -657
  374. package/src/tool/skill.ts +0 -73
  375. package/src/tool/skill.txt +0 -5
  376. package/src/tool/skill.zh.txt +0 -5
  377. package/src/tool/task.ts +0 -348
  378. package/src/tool/task.txt +0 -19
  379. package/src/tool/task.zh.txt +0 -19
  380. package/src/tool/todo.ts +0 -59
  381. package/src/tool/todowrite.txt +0 -44
  382. package/src/tool/todowrite.zh.txt +0 -44
  383. package/src/tool/tool.ts +0 -183
  384. package/src/tool/truncate.ts +0 -158
  385. package/src/tool/truncation-dir.ts +0 -4
  386. package/src/tool/webfetch.ts +0 -194
  387. package/src/tool/webfetch.txt +0 -13
  388. package/src/tool/webfetch.zh.txt +0 -13
  389. package/src/tool/websearch.ts +0 -145
  390. package/src/tool/websearch.txt +0 -14
  391. package/src/tool/websearch.zh.txt +0 -14
  392. package/src/tool/write.ts +0 -106
  393. package/src/tool/write.txt +0 -8
  394. package/src/tool/write.zh.txt +0 -8
  395. package/src/util/archive.ts +0 -17
  396. package/src/util/bom.ts +0 -27
  397. package/src/util/data-url.ts +0 -9
  398. package/src/util/defer.ts +0 -10
  399. package/src/util/effect-http-client.ts +0 -11
  400. package/src/util/error.ts +0 -1
  401. package/src/util/filesystem.ts +0 -251
  402. package/src/util/html.ts +0 -8
  403. package/src/util/iife.ts +0 -3
  404. package/src/util/lazy.ts +0 -20
  405. package/src/util/local-context.ts +0 -25
  406. package/src/util/locale.ts +0 -2
  407. package/src/util/media.ts +0 -26
  408. package/src/util/process.ts +0 -177
  409. package/src/util/proxy-env.ts +0 -72
  410. package/src/util/queue.ts +0 -32
  411. package/src/util/record.ts +0 -1
  412. package/src/util/repository.ts +0 -232
  413. package/src/util/rpc.ts +0 -66
  414. package/src/util/signal.ts +0 -12
  415. package/src/util/timeout.ts +0 -13
  416. package/src/util/token.ts +0 -1
  417. package/src/util/wildcard.ts +0 -59
  418. package/src/worktree/index.ts +0 -654
@@ -1,686 +0,0 @@
1
- import { LayerNode } from "@opencode-ai/core/effect/layer-node"
2
- import { httpClient } from "@opencode-ai/core/effect/layer-node-platform"
3
- import { serviceUse } from "@opencode-ai/core/effect/service-use"
4
- import path from "path"
5
- import { pathToFileURL } from "url"
6
- import os from "os"
7
- import { mergeDeep } from "remeda"
8
- import { Global } from "@opencode-ai/core/global"
9
- import fsNode from "fs/promises"
10
- import { Flag } from "@opencode-ai/core/flag/flag"
11
- import { Auth } from "../auth"
12
- import { Env } from "../env"
13
- import { applyEdits, modify } from "jsonc-parser"
14
- import { InstallationLocal, InstallationVersion } from "@opencode-ai/core/installation/version"
15
- import { existsSync } from "fs"
16
- import { Account } from "@/account/account"
17
- import { isRecord } from "@/util/record"
18
- import type { ConsoleState } from "@opencode-ai/core/v1/config/console-state"
19
- import { FSUtil } from "@opencode-ai/core/fs-util"
20
- import { InstanceState } from "@/effect/instance-state"
21
- import { Context, Duration, Effect, Exit, Fiber, Layer, Option, Schema } from "effect"
22
- import { FetchHttpClient, HttpClient, HttpClientRequest } from "effect/unstable/http"
23
- import { EffectFlock } from "@opencode-ai/core/util/effect-flock"
24
- import { containsPath, type InstanceContext } from "../project/instance-context"
25
- import { ConfigV1 } from "@opencode-ai/core/v1/config/config"
26
- import { RemoteAuthError } from "@opencode-ai/core/v1/config/error"
27
- import { ConfigPermissionV1 } from "@opencode-ai/core/v1/config/permission"
28
- import { ConfigPluginV1 } from "@opencode-ai/core/v1/config/plugin"
29
- import { ConfigAgent } from "./agent"
30
- import { ConfigCommand } from "./command"
31
- import { ConfigManaged } from "./managed"
32
- import { ConfigParse } from "./parse"
33
- import { ConfigPaths } from "./paths"
34
- import { ConfigPlugin } from "./plugin"
35
- import { ConfigVariable } from "./variable"
36
- import { Npm } from "@opencode-ai/core/npm"
37
- import { withTransientReadRetry } from "@/util/effect-http-client"
38
-
39
- // Custom merge function that concatenates array fields instead of replacing them
40
- // Keep remeda's deep conditional merge type out of hot config-loading paths; TS profiling showed it dominates here.
41
- function mergeConfig(target: Info, source: Info): Info {
42
- return mergeDeep(target, source) as Info
43
- }
44
-
45
- function mergeConfigConcatArrays(target: Info, source: Info): Info {
46
- const merged = mergeConfig(target, source)
47
- if (target.instructions && source.instructions) {
48
- merged.instructions = Array.from(new Set([...target.instructions, ...source.instructions]))
49
- }
50
- return merged
51
- }
52
-
53
- function normalizeLoadedConfig(data: unknown) {
54
- if (!isRecord(data)) return data
55
- const copy = { ...data }
56
- const hadLegacy = "theme" in copy || "keybinds" in copy || "tui" in copy
57
- if (!hadLegacy) return copy
58
- delete copy.theme
59
- delete copy.keybinds
60
- delete copy.tui
61
- return copy
62
- }
63
-
64
- async function substituteWellKnownRemoteConfig(input: {
65
- value: unknown
66
- dir: string
67
- source: string
68
- env: Record<string, string>
69
- }) {
70
- if (!isRecord(input.value) || typeof input.value.url !== "string") return undefined
71
-
72
- const url = await ConfigVariable.substitute({
73
- text: input.value.url,
74
- type: "virtual",
75
- dir: input.dir,
76
- source: input.source,
77
- env: input.env,
78
- })
79
- const headers = isRecord(input.value.headers)
80
- ? Object.fromEntries(
81
- await Promise.all(
82
- Object.entries(input.value.headers)
83
- .filter((entry): entry is [string, string] => typeof entry[1] === "string")
84
- .map(async ([key, value]) => [
85
- key,
86
- await ConfigVariable.substitute({
87
- text: value,
88
- type: "virtual",
89
- dir: input.dir,
90
- source: input.source,
91
- env: input.env,
92
- }),
93
- ]),
94
- ),
95
- )
96
- : undefined
97
-
98
- return { url, headers }
99
- }
100
-
101
- async function resolveLoadedPlugins<T extends { plugin?: ConfigPluginV1.Spec[] }>(config: T, filepath: string) {
102
- if (!config.plugin) return config
103
- for (let i = 0; i < config.plugin.length; i++) {
104
- // Normalize path-like plugin specs while we still know which config file declared them.
105
- // This prevents `./plugin.ts` from being reinterpreted relative to some later merge location.
106
- config.plugin[i] = await ConfigPlugin.resolvePluginSpec(config.plugin[i], filepath)
107
- }
108
- return config
109
- }
110
-
111
- type Info = ConfigV1.Info & {
112
- // plugin_origins is derived state, not a persisted config field. It keeps each winning plugin spec together
113
- // with the file and scope it came from so later runtime code can make location-sensitive decisions.
114
- plugin_origins?: ConfigPlugin.Origin[]
115
- }
116
-
117
- type State = {
118
- config: Info
119
- directories: string[]
120
- deps: Fiber.Fiber<void>[]
121
- consoleState: ConsoleState
122
- }
123
-
124
- export interface Interface {
125
- readonly get: () => Effect.Effect<Info>
126
- readonly getGlobal: () => Effect.Effect<Info>
127
- readonly getConsoleState: () => Effect.Effect<ConsoleState>
128
- readonly update: (config: Info) => Effect.Effect<void>
129
- readonly updateGlobal: (config: Info) => Effect.Effect<{ info: Info; changed: boolean }>
130
- readonly invalidate: () => Effect.Effect<void>
131
- readonly directories: () => Effect.Effect<string[]>
132
- readonly waitForDependencies: () => Effect.Effect<void>
133
- }
134
-
135
- export class Service extends Context.Service<Service, Interface>()("@opencode/Config") {}
136
-
137
- export const use = serviceUse(Service)
138
-
139
- function globalConfigFile() {
140
- const candidates = ["opencode.jsonc", "opencode.json", "config.json"].map((file) =>
141
- path.join(Global.Path.config, file),
142
- )
143
- for (const file of candidates) {
144
- if (existsSync(file)) return file
145
- }
146
- return candidates[0]
147
- }
148
-
149
- function patchJsonc(input: string, patch: unknown, path: string[] = []): string {
150
- if (!isRecord(patch)) {
151
- const edits = modify(input, path, patch, {
152
- formattingOptions: {
153
- insertSpaces: true,
154
- tabSize: 2,
155
- },
156
- })
157
- return applyEdits(input, edits)
158
- }
159
-
160
- return Object.entries(patch).reduce((result, [key, value]) => patchJsonc(result, value, [...path, key]), input)
161
- }
162
-
163
- function writable(info: Info) {
164
- const { plugin_origins: _plugin_origins, ...next } = info
165
- return next
166
- }
167
-
168
- function writableGlobal(info: Info) {
169
- const next = writable(info)
170
- // When a user changes config from a value back to default in the Desktop app, we don't want to leave a blank `"shell": "",` key
171
- if ("shell" in next && next.shell === "") return { ...next, shell: undefined }
172
- return next
173
- }
174
-
175
- export const layer = Layer.effect(
176
- Service,
177
- Effect.gen(function* () {
178
- const fs = yield* FSUtil.Service
179
- const authSvc = yield* Auth.Service
180
- const accountSvc = yield* Account.Service
181
- const env = yield* Env.Service
182
- const npmSvc = yield* Npm.Service
183
- const http = yield* HttpClient.HttpClient
184
-
185
- const readConfigFile = (filepath: string) => fs.readFileStringSafe(filepath).pipe(Effect.orDie)
186
-
187
- const fetchRemoteJson = Effect.fnUntraced(function* <S extends Schema.Top>(
188
- url: string,
189
- headers: Record<string, string> | undefined,
190
- schema: S,
191
- loginOrigin: string,
192
- ) {
193
- const response = yield* HttpClient.filterStatusOk(withTransientReadRetry(http))
194
- .execute(
195
- HttpClientRequest.get(url).pipe(HttpClientRequest.acceptJson, HttpClientRequest.setHeaders(headers ?? {})),
196
- )
197
- .pipe(
198
- Effect.catch((error) => Effect.die(new Error(`failed to fetch remote config from ${url}: ${String(error)}`))),
199
- )
200
- const body = yield* response.text.pipe(
201
- Effect.catch((error) => Effect.die(new Error(`failed to read remote config from ${url}: ${String(error)}`))),
202
- )
203
- // An auth proxy can answer with an HTML login page at HTTP 200 (passes filterStatusOk); treat it as a re-auth error, not a decode failure.
204
- const contentType = (response.headers["content-type"] ?? "").toLowerCase()
205
- if (contentType.includes("html") || /^\s*<!doctype|^\s*<html/i.test(body)) {
206
- return yield* Effect.die(new RemoteAuthError({ url: loginOrigin, remote: url }))
207
- }
208
- return yield* Schema.decodeEffect(Schema.fromJsonString(schema))(body).pipe(
209
- Effect.catch((error) => Effect.die(new Error(`failed to decode remote config from ${url}: ${String(error)}`))),
210
- )
211
- })
212
-
213
- const loadConfig = Effect.fnUntraced(function* (
214
- text: string,
215
- options: { path: string } | { dir: string; source: string },
216
- env?: Record<string, string>,
217
- ) {
218
- const source = "path" in options ? options.path : options.source
219
- const expanded = yield* Effect.promise(() =>
220
- ConfigVariable.substitute(
221
- "path" in options
222
- ? { text, type: "path", path: options.path, env }
223
- : { text, type: "virtual", ...options, env },
224
- ),
225
- )
226
- const parsed = ConfigParse.jsonc(expanded, source)
227
- const data = ConfigParse.schema(ConfigV1.Info, normalizeLoadedConfig(parsed), source)
228
- if (!("path" in options)) return data
229
-
230
- yield* Effect.promise(() => resolveLoadedPlugins(data, options.path))
231
- if (!data.$schema) {
232
- data.$schema = "https://ai.huamp.com/config.json"
233
- const updated = text.replace(/^\s*\{/, '{\n "$schema": "https://ai.huamp.com/config.json",')
234
- yield* fs.writeFileString(options.path, updated).pipe(Effect.catch(() => Effect.void))
235
- }
236
- return data
237
- })
238
-
239
- const loadFile = Effect.fnUntraced(function* (filepath: string, env?: Record<string, string>) {
240
- yield* Effect.logInfo("loading", { path: filepath })
241
- const text = yield* readConfigFile(filepath)
242
- if (!text) return {} as Info
243
- return yield* loadConfig(text, { path: filepath }, env)
244
- })
245
-
246
- const loadGlobal = Effect.fnUntraced(function* (env?: Record<string, string>) {
247
- let result: Info = {}
248
- // Seed the default global config with the schema for editor completion, but avoid writing when the user
249
- // explicitly routes config through env-provided paths or content.
250
- if (!Flag.OPENCODE_CONFIG && !Flag.OPENCODE_CONFIG_DIR && !Flag.OPENCODE_CONFIG_CONTENT) {
251
- const file = globalConfigFile()
252
- if (!existsSync(file)) {
253
- yield* fs
254
- .writeWithDirs(file, JSON.stringify({ $schema: "https://ai.huamp.com/config.json" }, null, 2))
255
- .pipe(Effect.catch(() => Effect.void))
256
- }
257
- }
258
- result = mergeConfig(result, yield* loadFile(path.join(Global.Path.config, "config.json"), env))
259
- result = mergeConfig(result, yield* loadFile(path.join(Global.Path.config, "opencode.json"), env))
260
- result = mergeConfig(result, yield* loadFile(path.join(Global.Path.config, "opencode.jsonc"), env))
261
-
262
- const legacy = path.join(Global.Path.config, "config")
263
- if (existsSync(legacy)) {
264
- yield* Effect.promise(() =>
265
- import(pathToFileURL(legacy).href, { with: { type: "toml" } })
266
- .then(async (mod) => {
267
- const { provider, model, ...rest } = mod.default
268
- if (provider && model) result.model = `${provider}/${model}`
269
- result["$schema"] = "https://ai.huamp.com/config.json"
270
- result = mergeConfig(result, rest)
271
- await fsNode.writeFile(path.join(Global.Path.config, "config.json"), JSON.stringify(result, null, 2))
272
- await fsNode.unlink(legacy)
273
- })
274
- .catch(() => {}),
275
- )
276
- }
277
-
278
- return result
279
- })
280
-
281
- const [cachedGlobal, invalidateGlobal] = yield* Effect.cachedInvalidateWithTTL(
282
- loadGlobal().pipe(
283
- Effect.tapError((error) =>
284
- Effect.logError("failed to load global config, using defaults", { error: String(error) }),
285
- ),
286
- Effect.orElseSucceed((): Info => ({})),
287
- ),
288
- Duration.infinity,
289
- )
290
-
291
- const getGlobal = Effect.fn("Config.getGlobal")(function* () {
292
- return yield* cachedGlobal
293
- })
294
-
295
- const ensureGitignore = Effect.fn("Config.ensureGitignore")(function* (dir: string) {
296
- const gitignore = path.join(dir, ".gitignore")
297
- const hasIgnore = yield* fs.existsSafe(gitignore)
298
- if (!hasIgnore) {
299
- yield* fs
300
- .writeFileString(
301
- gitignore,
302
- ["node_modules", "package.json", "package-lock.json", "bun.lock", ".gitignore"].join("\n"),
303
- )
304
- .pipe(
305
- Effect.catchIf(
306
- (e) => e.reason._tag === "PermissionDenied",
307
- () => Effect.void,
308
- ),
309
- )
310
- }
311
- })
312
-
313
- const loadInstanceState = Effect.fn("Config.loadInstanceState")(
314
- function* (ctx: InstanceContext) {
315
- const auth = yield* authSvc.all().pipe(Effect.orDie)
316
-
317
- let result: Info = {}
318
- const authEnv: Record<string, string> = {}
319
- const consoleManagedProviders = new Set<string>()
320
- let activeOrgName: string | undefined
321
-
322
- const pluginScopeForSource = Effect.fnUntraced(function* (source: string) {
323
- if (source.startsWith("http://") || source.startsWith("https://")) return "global"
324
- if (source === "OPENCODE_CONFIG_CONTENT") return "local"
325
- if (containsPath(source, ctx)) return "local"
326
- return "global"
327
- })
328
-
329
- const mergePluginOrigins = Effect.fnUntraced(function* (
330
- source: string,
331
- // mergePluginOrigins receives raw Specs from one config source, before provenance for this merge step
332
- // is attached.
333
- list: ConfigPluginV1.Spec[] | undefined,
334
- // Scope can be inferred from the source path, but some callers already know whether the config should
335
- // behave as global or local and can pass that explicitly.
336
- kind?: ConfigPlugin.Scope,
337
- ) {
338
- if (!list?.length) return
339
- const hit = kind ?? (yield* pluginScopeForSource(source))
340
- // Merge newly seen plugin origins with previously collected ones, then dedupe by plugin identity while
341
- // keeping the winning source/scope metadata for downstream installs, writes, and diagnostics.
342
- const plugins = ConfigPlugin.deduplicatePluginOrigins([
343
- ...(result.plugin_origins ?? []),
344
- ...list.map((spec) => ({ spec, source, scope: hit })),
345
- ])
346
- result.plugin = plugins.map((item) => item.spec)
347
- result.plugin_origins = plugins
348
- })
349
-
350
- const merge = (source: string, next: Info, kind?: ConfigPlugin.Scope) => {
351
- result = mergeConfigConcatArrays(result, next)
352
- return mergePluginOrigins(source, next.plugin, kind)
353
- }
354
-
355
- for (const [key, value] of Object.entries(auth)) {
356
- if (value.type === "wellknown") {
357
- const url = key.replace(/\/+$/, "")
358
- authEnv[value.key] = value.token
359
- const wellknownURL = `${url}/.well-known/opencode`
360
- yield* Effect.logDebug("fetching remote config", { url: wellknownURL })
361
- const wellknown = yield* fetchRemoteJson(wellknownURL, undefined, ConfigV1.WellKnown, url)
362
- const remote = yield* Effect.promise(() =>
363
- substituteWellKnownRemoteConfig({
364
- value: wellknown.remote_config,
365
- dir: url,
366
- source: wellknownURL,
367
- env: authEnv,
368
- }),
369
- )
370
- const fetchedConfig = remote
371
- ? yield* Effect.gen(function* () {
372
- yield* Effect.logDebug("fetching remote config", { url: remote.url })
373
- const data = yield* fetchRemoteJson(remote.url, remote.headers, Schema.Json, url)
374
- if (isRecord(data) && isRecord(data.config)) return data.config
375
- if (isRecord(data)) return data
376
- return yield* Effect.die(
377
- new Error(`failed to decode remote config from ${remote.url}: expected object`),
378
- )
379
- })
380
- : {}
381
- const remoteConfig = mergeConfig(isRecord(wellknown.config) ? wellknown.config : {}, fetchedConfig)
382
- if (!remoteConfig.$schema) remoteConfig.$schema = "https://ai.huamp.com/config.json"
383
- const source = wellknownURL
384
- const next = yield* loadConfig(
385
- JSON.stringify(remoteConfig),
386
- {
387
- dir: path.dirname(source),
388
- source,
389
- },
390
- authEnv,
391
- )
392
- yield* merge(source, next, "global")
393
- yield* Effect.logDebug("loaded remote config from well-known", { url })
394
- }
395
- }
396
-
397
- const global = Object.keys(authEnv).length ? yield* loadGlobal(authEnv) : yield* getGlobal()
398
- yield* merge(Global.Path.config, global, "global")
399
-
400
- if (Flag.OPENCODE_CONFIG) {
401
- yield* merge(Flag.OPENCODE_CONFIG, yield* loadFile(Flag.OPENCODE_CONFIG, authEnv))
402
- yield* Effect.logDebug("loaded custom config", { path: Flag.OPENCODE_CONFIG })
403
- }
404
-
405
- if (!Flag.OPENCODE_DISABLE_PROJECT_CONFIG) {
406
- for (const file of yield* ConfigPaths.files("opencode", ctx.directory, ctx.worktree).pipe(Effect.orDie)) {
407
- yield* merge(file, yield* loadFile(file, authEnv), "local")
408
- }
409
- }
410
-
411
- result.agent = result.agent || {}
412
- result.mode = result.mode || {}
413
- result.plugin = result.plugin || []
414
-
415
- const directories = yield* ConfigPaths.directories(ctx.directory, ctx.worktree)
416
-
417
- if (Flag.OPENCODE_CONFIG_DIR) {
418
- yield* Effect.logDebug("loading config from OPENCODE_CONFIG_DIR", { path: Flag.OPENCODE_CONFIG_DIR })
419
- }
420
-
421
- const deps: Fiber.Fiber<void>[] = []
422
-
423
- for (const dir of directories) {
424
- if (dir.endsWith(".opencode") || dir === Flag.OPENCODE_CONFIG_DIR) {
425
- for (const file of ["opencode.json", "opencode.jsonc"]) {
426
- const source = path.join(dir, file)
427
- yield* Effect.logDebug(`loading config from ${source}`)
428
- yield* merge(source, yield* loadFile(source, authEnv))
429
- result.agent ??= {}
430
- result.mode ??= {}
431
- result.plugin ??= []
432
- }
433
- }
434
-
435
- yield* ensureGitignore(dir).pipe(Effect.orDie)
436
-
437
- const dep = yield* npmSvc
438
- .install(dir, {
439
- add: [
440
- {
441
- name: "@opencode-ai/plugin",
442
- version: InstallationLocal ? undefined : InstallationVersion,
443
- },
444
- ],
445
- })
446
- .pipe(
447
- Effect.exit,
448
- Effect.tap((exit) =>
449
- Exit.isFailure(exit)
450
- ? Effect.logWarning("background dependency install failed", { dir, error: String(exit.cause) })
451
- : Effect.void,
452
- ),
453
- Effect.asVoid,
454
- Effect.forkDetach,
455
- )
456
- deps.push(dep)
457
-
458
- result.command = mergeDeep(result.command ?? {}, yield* Effect.promise(() => ConfigCommand.load(dir)))
459
- result.agent = mergeDeep(result.agent ?? {}, yield* Effect.promise(() => ConfigAgent.load(dir)))
460
- result.agent = mergeDeep(result.agent ?? {}, yield* Effect.promise(() => ConfigAgent.loadMode(dir)))
461
- // Auto-discovered plugins under `.opencode/plugin(s)` are already local files, so ConfigPlugin.load
462
- // returns normalized Specs and we only need to attach origin metadata here.
463
- const list = yield* Effect.promise(() => ConfigPlugin.load(dir))
464
- yield* mergePluginOrigins(dir, list)
465
- }
466
-
467
- if (process.env.OPENCODE_CONFIG_CONTENT) {
468
- const source = "OPENCODE_CONFIG_CONTENT"
469
- const next = yield* loadConfig(process.env.OPENCODE_CONFIG_CONTENT, {
470
- dir: ctx.directory,
471
- source,
472
- })
473
- yield* merge(source, next, "local")
474
- yield* Effect.logDebug("loaded custom config from OPENCODE_CONFIG_CONTENT")
475
- }
476
-
477
- const activeAccount = Option.getOrUndefined(
478
- yield* accountSvc.active().pipe(Effect.catch(() => Effect.succeed(Option.none()))),
479
- )
480
- if (activeAccount?.active_org_id) {
481
- const accountID = activeAccount.id
482
- const orgID = activeAccount.active_org_id
483
- const url = activeAccount.url
484
- yield* Effect.gen(function* () {
485
- const [configOpt, tokenOpt] = yield* Effect.all(
486
- [accountSvc.config(accountID, orgID), accountSvc.token(accountID)],
487
- { concurrency: 2 },
488
- )
489
- if (Option.isSome(tokenOpt)) {
490
- process.env["OPENCODE_CONSOLE_TOKEN"] = tokenOpt.value
491
- yield* env.set("OPENCODE_CONSOLE_TOKEN", tokenOpt.value)
492
- }
493
-
494
- if (Option.isSome(configOpt)) {
495
- const source = `${url}/api/config`
496
- const next = yield* loadConfig(JSON.stringify(configOpt.value), {
497
- dir: path.dirname(source),
498
- source,
499
- })
500
- for (const providerID of Object.keys(next.provider ?? {})) {
501
- consoleManagedProviders.add(providerID)
502
- }
503
- yield* merge(source, next, "global")
504
- }
505
- }).pipe(
506
- Effect.withSpan("Config.loadActiveOrgConfig"),
507
- Effect.catch((err) =>
508
- Effect.logDebug("failed to fetch remote account config", {
509
- error: err instanceof Error ? err.message : String(err),
510
- }),
511
- ),
512
- )
513
- }
514
-
515
- const managedDir = ConfigManaged.managedConfigDir()
516
- if (existsSync(managedDir)) {
517
- for (const file of ["opencode.json", "opencode.jsonc"]) {
518
- const source = path.join(managedDir, file)
519
- yield* merge(source, yield* loadFile(source), "global")
520
- }
521
- }
522
-
523
- // macOS managed preferences (.mobileconfig deployed via MDM) override everything
524
- const managed = yield* Effect.promise(() => ConfigManaged.readManagedPreferences())
525
- if (managed) {
526
- result = mergeConfigConcatArrays(
527
- result,
528
- yield* loadConfig(managed.text, {
529
- dir: path.dirname(managed.source),
530
- source: managed.source,
531
- }),
532
- )
533
- }
534
-
535
- for (const [name, mode] of Object.entries(result.mode ?? {})) {
536
- result.agent = mergeDeep(result.agent ?? {}, {
537
- [name]: {
538
- ...mode,
539
- mode: "primary" as const,
540
- },
541
- })
542
- }
543
-
544
- if (Flag.OPENCODE_PERMISSION) {
545
- try {
546
- result.permission = mergeDeep(result.permission ?? {}, JSON.parse(Flag.OPENCODE_PERMISSION))
547
- } catch (err) {
548
- yield* Effect.logWarning("OPENCODE_PERMISSION contains invalid JSON, skipping", { err })
549
- }
550
- }
551
-
552
- if (result.tools) {
553
- const perms: Record<string, ConfigPermissionV1.Action> = {}
554
- for (const [tool, enabled] of Object.entries(result.tools)) {
555
- const action: ConfigPermissionV1.Action = enabled ? "allow" : "deny"
556
- if (tool === "write" || tool === "edit" || tool === "patch") {
557
- perms.edit = action
558
- continue
559
- }
560
- perms[tool] = action
561
- }
562
- result.permission = mergeDeep(perms, result.permission ?? {})
563
- }
564
-
565
- if (!result.username) {
566
- try {
567
- result.username = os.userInfo().username || "user"
568
- } catch (err) {
569
- yield* Effect.logWarning("failed to read system username, using fallback", { err })
570
- result.username = "user"
571
- }
572
- }
573
-
574
- if (result.autoshare === true && !result.share) {
575
- result.share = "auto"
576
- }
577
-
578
- if (Flag.OPENCODE_DISABLE_AUTOCOMPACT) {
579
- result.compaction = { ...result.compaction, auto: false }
580
- }
581
- if (Flag.OPENCODE_DISABLE_PRUNE) {
582
- result.compaction = { ...result.compaction, prune: false }
583
- }
584
-
585
- return {
586
- config: result,
587
- directories,
588
- deps,
589
- consoleState: {
590
- consoleManagedProviders: Array.from(consoleManagedProviders),
591
- activeOrgName,
592
- switchableOrgCount: 0,
593
- },
594
- }
595
- },
596
- Effect.provideService(FSUtil.Service, fs),
597
- )
598
-
599
- const state = yield* InstanceState.make<State>(
600
- Effect.fn("Config.state")(function* (ctx) {
601
- return yield* loadInstanceState(ctx).pipe(Effect.orDie)
602
- }),
603
- )
604
-
605
- const get = Effect.fn("Config.get")(function* () {
606
- return yield* InstanceState.use(state, (s) => s.config)
607
- })
608
-
609
- const directories = Effect.fn("Config.directories")(function* () {
610
- return yield* InstanceState.use(state, (s) => s.directories)
611
- })
612
-
613
- const getConsoleState = Effect.fn("Config.getConsoleState")(function* () {
614
- return yield* InstanceState.use(state, (s) => s.consoleState)
615
- })
616
-
617
- const waitForDependencies = Effect.fn("Config.waitForDependencies")(function* () {
618
- yield* InstanceState.useEffect(state, (s) =>
619
- Effect.forEach(s.deps, Fiber.join, { concurrency: "unbounded" }).pipe(Effect.asVoid),
620
- )
621
- })
622
-
623
- const update = Effect.fn("Config.update")(function* (config: Info) {
624
- const dir = yield* InstanceState.directory
625
- const file = path.join(dir, "config.json")
626
- const existing = yield* loadFile(file)
627
- yield* fs
628
- .writeFileString(file, JSON.stringify(mergeDeep(writable(existing), writable(config)), null, 2))
629
- .pipe(Effect.orDie)
630
- })
631
-
632
- const invalidate = Effect.fn("Config.invalidate")(function* () {
633
- yield* invalidateGlobal
634
- })
635
-
636
- const updateGlobal = Effect.fn("Config.updateGlobal")(function* (config: Info) {
637
- const file = globalConfigFile()
638
- const before = (yield* readConfigFile(file)) ?? "{}"
639
- const patch = writableGlobal(config)
640
-
641
- let next: Info
642
- let changed: boolean
643
- if (!file.endsWith(".jsonc")) {
644
- const existing = ConfigParse.schema(ConfigV1.Info, ConfigParse.jsonc(before, file), file)
645
- const merged = mergeDeep(writable(existing), patch)
646
- const serialized = JSON.stringify(merged, null, 2)
647
- changed = serialized !== before
648
- if (changed) yield* fs.writeFileString(file, serialized).pipe(Effect.orDie)
649
- next = merged
650
- } else {
651
- const updated = patchJsonc(before, patch)
652
- next = ConfigParse.schema(ConfigV1.Info, ConfigParse.jsonc(updated, file), file)
653
- changed = updated !== before
654
- if (changed) yield* fs.writeFileString(file, updated).pipe(Effect.orDie)
655
- }
656
-
657
- if (changed) yield* invalidate()
658
- return { info: next, changed }
659
- })
660
-
661
- return Service.of({
662
- get,
663
- getGlobal,
664
- getConsoleState,
665
- update,
666
- updateGlobal,
667
- invalidate,
668
- directories,
669
- waitForDependencies,
670
- })
671
- }),
672
- )
673
-
674
- export const defaultLayer = layer.pipe(
675
- Layer.provide(EffectFlock.defaultLayer),
676
- Layer.provide(FSUtil.defaultLayer),
677
- Layer.provide(Env.defaultLayer),
678
- Layer.provide(Auth.defaultLayer),
679
- Layer.provide(Account.defaultLayer),
680
- Layer.provide(Npm.defaultLayer),
681
- Layer.provide(FetchHttpClient.layer),
682
- )
683
-
684
- export const node = LayerNode.make(layer, [FSUtil.node, Auth.node, Account.node, Env.node, Npm.node, httpClient])
685
-
686
- export * as Config from "./config"