onlycode 1.18.0 → 1.19.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.
- package/package.json +11 -146
- package/src/account/account.ts +0 -463
- package/src/account/repo.ts +0 -173
- package/src/account/schema.ts +0 -99
- package/src/account/url.ts +0 -8
- package/src/acp/agent.ts +0 -95
- package/src/acp/config-option.ts +0 -203
- package/src/acp/content.ts +0 -250
- package/src/acp/directory.ts +0 -210
- package/src/acp/error.ts +0 -90
- package/src/acp/event.ts +0 -342
- package/src/acp/permission.ts +0 -124
- package/src/acp/profile.ts +0 -42
- package/src/acp/service.ts +0 -1048
- package/src/acp/session.ts +0 -231
- package/src/acp/tool.ts +0 -367
- package/src/acp/usage.ts +0 -232
- package/src/agent/agent.ts +0 -459
- package/src/agent/generate.txt +0 -75
- package/src/agent/prompt/compaction.txt +0 -9
- package/src/agent/prompt/explore.txt +0 -18
- package/src/agent/prompt/summary.txt +0 -11
- package/src/agent/prompt/title.txt +0 -44
- package/src/agent/subagent-permissions.ts +0 -27
- package/src/audio.d.ts +0 -14
- package/src/auth/index.ts +0 -99
- package/src/background/job.ts +0 -39
- package/src/bus/global.ts +0 -22
- package/src/cli/bootstrap.ts +0 -11
- package/src/cli/cmd/account.ts +0 -264
- package/src/cli/cmd/acp.ts +0 -73
- package/src/cli/cmd/agent.ts +0 -259
- package/src/cli/cmd/attach.ts +0 -97
- package/src/cli/cmd/cmd.ts +0 -7
- package/src/cli/cmd/db.ts +0 -62
- package/src/cli/cmd/debug/agent.handler.ts +0 -193
- package/src/cli/cmd/debug/agent.ts +0 -27
- package/src/cli/cmd/debug/config.ts +0 -14
- package/src/cli/cmd/debug/file.ts +0 -73
- package/src/cli/cmd/debug/index.ts +0 -87
- package/src/cli/cmd/debug/lsp.ts +0 -50
- package/src/cli/cmd/debug/ripgrep.ts +0 -79
- package/src/cli/cmd/debug/scrap.ts +0 -15
- package/src/cli/cmd/debug/skill.ts +0 -15
- package/src/cli/cmd/debug/snapshot.ts +0 -50
- package/src/cli/cmd/debug/startup.ts +0 -11
- package/src/cli/cmd/debug/v2.ts +0 -44
- package/src/cli/cmd/export.ts +0 -292
- package/src/cli/cmd/generate.ts +0 -54
- package/src/cli/cmd/github.handler.ts +0 -1593
- package/src/cli/cmd/github.shared.ts +0 -30
- package/src/cli/cmd/github.ts +0 -42
- package/src/cli/cmd/import.ts +0 -224
- package/src/cli/cmd/mcp.ts +0 -849
- package/src/cli/cmd/models.ts +0 -66
- package/src/cli/cmd/plug.ts +0 -230
- package/src/cli/cmd/pr.ts +0 -115
- package/src/cli/cmd/prompt-display.ts +0 -1
- package/src/cli/cmd/providers.ts +0 -534
- package/src/cli/cmd/run/demo.ts +0 -1274
- package/src/cli/cmd/run/entry.body.ts +0 -205
- package/src/cli/cmd/run/footer.command.tsx +0 -1064
- package/src/cli/cmd/run/footer.menu.tsx +0 -351
- package/src/cli/cmd/run/footer.permission.tsx +0 -472
- package/src/cli/cmd/run/footer.prompt.tsx +0 -1306
- package/src/cli/cmd/run/footer.question.tsx +0 -573
- package/src/cli/cmd/run/footer.subagent.tsx +0 -173
- package/src/cli/cmd/run/footer.ts +0 -1129
- package/src/cli/cmd/run/footer.view.tsx +0 -943
- package/src/cli/cmd/run/footer.width.ts +0 -27
- package/src/cli/cmd/run/permission.shared.ts +0 -256
- package/src/cli/cmd/run/prompt.editor.ts +0 -157
- package/src/cli/cmd/run/prompt.shared.ts +0 -153
- package/src/cli/cmd/run/question.shared.ts +0 -340
- package/src/cli/cmd/run/runtime.boot.ts +0 -202
- package/src/cli/cmd/run/runtime.lifecycle.ts +0 -406
- package/src/cli/cmd/run/runtime.queue.ts +0 -349
- package/src/cli/cmd/run/runtime.shared.ts +0 -17
- package/src/cli/cmd/run/runtime.stdin.ts +0 -37
- package/src/cli/cmd/run/runtime.ts +0 -814
- package/src/cli/cmd/run/scrollback.shared.ts +0 -92
- package/src/cli/cmd/run/scrollback.surface.ts +0 -431
- package/src/cli/cmd/run/scrollback.writer.tsx +0 -352
- package/src/cli/cmd/run/session-data.ts +0 -1113
- package/src/cli/cmd/run/session-replay.ts +0 -374
- package/src/cli/cmd/run/session.shared.ts +0 -196
- package/src/cli/cmd/run/splash.ts +0 -275
- package/src/cli/cmd/run/stream.transport.ts +0 -1462
- package/src/cli/cmd/run/stream.ts +0 -175
- package/src/cli/cmd/run/subagent-data.ts +0 -876
- package/src/cli/cmd/run/theme.ts +0 -690
- package/src/cli/cmd/run/tool.ts +0 -1489
- package/src/cli/cmd/run/trace.ts +0 -94
- package/src/cli/cmd/run/turn-summary.ts +0 -47
- package/src/cli/cmd/run/types.ts +0 -350
- package/src/cli/cmd/run/variant.shared.ts +0 -215
- package/src/cli/cmd/run.ts +0 -894
- package/src/cli/cmd/serve.ts +0 -24
- package/src/cli/cmd/session.ts +0 -147
- package/src/cli/cmd/stats.ts +0 -393
- package/src/cli/cmd/tui.ts +0 -224
- package/src/cli/cmd/uninstall.ts +0 -353
- package/src/cli/cmd/upgrade.ts +0 -74
- package/src/cli/cmd/web.ts +0 -84
- package/src/cli/effect/prompt.ts +0 -37
- package/src/cli/effect-cmd.ts +0 -96
- package/src/cli/error.ts +0 -130
- package/src/cli/heap.ts +0 -45
- package/src/cli/logo-pixel.ts +0 -35
- package/src/cli/logo.ts +0 -1
- package/src/cli/network.ts +0 -64
- package/src/cli/tui/layer.ts +0 -7
- package/src/cli/tui/validate-session.ts +0 -29
- package/src/cli/tui/worker.ts +0 -71
- package/src/cli/ui.ts +0 -148
- package/src/cli/upgrade.ts +0 -53
- package/src/command/index.ts +0 -184
- package/src/command/template/initialize.txt +0 -66
- package/src/command/template/review.txt +0 -101
- package/src/config/agent.ts +0 -59
- package/src/config/command.ts +0 -39
- package/src/config/config.ts +0 -686
- package/src/config/entry-name.ts +0 -19
- package/src/config/managed.ts +0 -69
- package/src/config/markdown.ts +0 -36
- package/src/config/parse.ts +0 -79
- package/src/config/paths.ts +0 -45
- package/src/config/plugin.ts +0 -79
- package/src/config/tui-cwd.ts +0 -5
- package/src/config/tui-host-attention.ts +0 -21
- package/src/config/tui-migrate.ts +0 -132
- package/src/config/tui.ts +0 -274
- package/src/config/variable.ts +0 -91
- package/src/control-plane/adapters/index.ts +0 -41
- package/src/control-plane/adapters/worktree.ts +0 -96
- package/src/control-plane/dev/README.md +0 -19
- package/src/control-plane/dev/debug-workspace-plugin.ts +0 -73
- package/src/control-plane/types.ts +0 -59
- package/src/control-plane/util.ts +0 -39
- package/src/control-plane/workspace-adapter-runtime.ts +0 -51
- package/src/control-plane/workspace-context.ts +0 -26
- package/src/control-plane/workspace.ts +0 -989
- package/src/effect/app-runtime.ts +0 -132
- package/src/effect/bootstrap-runtime.ts +0 -23
- package/src/effect/bridge.ts +0 -84
- package/src/effect/config-service.ts +0 -67
- package/src/effect/instance-ref.ts +0 -11
- package/src/effect/instance-registry.ts +0 -12
- package/src/effect/instance-state.ts +0 -69
- package/src/effect/promise.ts +0 -17
- package/src/effect/run-service.ts +0 -47
- package/src/effect/runner.ts +0 -217
- package/src/effect/runtime-flags.ts +0 -79
- package/src/env/index.ts +0 -43
- package/src/event-v2-bridge.ts +0 -79
- package/src/format/formatter.ts +0 -404
- package/src/format/index.ts +0 -205
- package/src/git/index.ts +0 -350
- package/src/id/id.ts +0 -80
- package/src/ide/index.ts +0 -61
- package/src/image/image.ts +0 -174
- package/src/index.ts +0 -142
- package/src/installation/index.ts +0 -350
- package/src/lsp/client.ts +0 -650
- package/src/lsp/diagnostic.ts +0 -29
- package/src/lsp/language.ts +0 -121
- package/src/lsp/launch.ts +0 -21
- package/src/lsp/lsp.ts +0 -511
- package/src/lsp/server.ts +0 -1983
- package/src/markdown.d.ts +0 -4
- package/src/mcp/auth.ts +0 -174
- package/src/mcp/catalog.ts +0 -153
- package/src/mcp/index.ts +0 -953
- package/src/mcp/oauth-callback.ts +0 -233
- package/src/mcp/oauth-provider.ts +0 -206
- package/src/node.ts +0 -4
- package/src/patch/index.ts +0 -686
- package/src/permission/arity.ts +0 -163
- package/src/permission/evaluate.ts +0 -1
- package/src/permission/index.ts +0 -230
- package/src/plugin/azure.ts +0 -26
- package/src/plugin/cloudflare.ts +0 -76
- package/src/plugin/digitalocean.ts +0 -383
- package/src/plugin/github-copilot/copilot.ts +0 -414
- package/src/plugin/github-copilot/models.ts +0 -246
- package/src/plugin/index.ts +0 -316
- package/src/plugin/install.ts +0 -439
- package/src/plugin/loader.ts +0 -237
- package/src/plugin/meta.ts +0 -188
- package/src/plugin/openai/README.md +0 -31
- package/src/plugin/openai/codex.ts +0 -641
- package/src/plugin/openai/ws-pool.ts +0 -270
- package/src/plugin/openai/ws.ts +0 -381
- package/src/plugin/pty-environment.ts +0 -24
- package/src/plugin/shared.ts +0 -323
- package/src/plugin/snowflake-cortex.ts +0 -529
- package/src/plugin/tui/internal.ts +0 -10
- package/src/plugin/tui/runtime.ts +0 -1130
- package/src/plugin/xai.ts +0 -716
- package/src/project/bootstrap-service.ts +0 -9
- package/src/project/bootstrap.ts +0 -76
- package/src/project/instance-context.ts +0 -24
- package/src/project/instance-layer.ts +0 -11
- package/src/project/instance-runtime.ts +0 -16
- package/src/project/instance-store.ts +0 -209
- package/src/project/project.ts +0 -519
- package/src/project/vcs.ts +0 -431
- package/src/provider/auth.ts +0 -233
- package/src/provider/error.ts +0 -188
- package/src/provider/model-status.ts +0 -8
- package/src/provider/provider.ts +0 -1975
- package/src/provider/transform.ts +0 -1543
- package/src/question/index.ts +0 -229
- package/src/question/schema.ts +0 -10
- package/src/server/auth.ts +0 -48
- package/src/server/event.ts +0 -13
- package/src/server/global-lifecycle.ts +0 -28
- package/src/server/init-projectors.ts +0 -3
- package/src/server/mdns.ts +0 -47
- package/src/server/projectors.ts +0 -1
- package/src/server/proxy-util.ts +0 -48
- package/src/server/routes/instance/httpapi/AGENTS.md +0 -39
- package/src/server/routes/instance/httpapi/api.ts +0 -78
- package/src/server/routes/instance/httpapi/errors.ts +0 -193
- package/src/server/routes/instance/httpapi/groups/config.ts +0 -65
- package/src/server/routes/instance/httpapi/groups/control-plane.ts +0 -35
- package/src/server/routes/instance/httpapi/groups/control.ts +0 -76
- package/src/server/routes/instance/httpapi/groups/event.ts +0 -29
- package/src/server/routes/instance/httpapi/groups/experimental.ts +0 -275
- package/src/server/routes/instance/httpapi/groups/file.ts +0 -185
- package/src/server/routes/instance/httpapi/groups/global.ts +0 -138
- package/src/server/routes/instance/httpapi/groups/instance.ts +0 -206
- package/src/server/routes/instance/httpapi/groups/mcp.ts +0 -156
- package/src/server/routes/instance/httpapi/groups/metadata.ts +0 -18
- package/src/server/routes/instance/httpapi/groups/permission.ts +0 -61
- package/src/server/routes/instance/httpapi/groups/project-copy.ts +0 -32
- package/src/server/routes/instance/httpapi/groups/project.ts +0 -93
- package/src/server/routes/instance/httpapi/groups/provider.ts +0 -101
- package/src/server/routes/instance/httpapi/groups/pty.ts +0 -172
- package/src/server/routes/instance/httpapi/groups/query.ts +0 -12
- package/src/server/routes/instance/httpapi/groups/question.ts +0 -74
- package/src/server/routes/instance/httpapi/groups/session.ts +0 -462
- package/src/server/routes/instance/httpapi/groups/sync.ts +0 -113
- package/src/server/routes/instance/httpapi/groups/tui.ts +0 -208
- package/src/server/routes/instance/httpapi/groups/workspace.ts +0 -141
- package/src/server/routes/instance/httpapi/handlers/config.ts +0 -34
- package/src/server/routes/instance/httpapi/handlers/control-plane.ts +0 -37
- package/src/server/routes/instance/httpapi/handlers/control.ts +0 -43
- package/src/server/routes/instance/httpapi/handlers/event.ts +0 -99
- package/src/server/routes/instance/httpapi/handlers/experimental.ts +0 -192
- package/src/server/routes/instance/httpapi/handlers/file.ts +0 -139
- package/src/server/routes/instance/httpapi/handlers/global.ts +0 -156
- package/src/server/routes/instance/httpapi/handlers/instance.ts +0 -110
- package/src/server/routes/instance/httpapi/handlers/mcp.ts +0 -111
- package/src/server/routes/instance/httpapi/handlers/permission.ts +0 -41
- package/src/server/routes/instance/httpapi/handlers/project-copy.ts +0 -83
- package/src/server/routes/instance/httpapi/handlers/project.ts +0 -63
- package/src/server/routes/instance/httpapi/handlers/provider.ts +0 -113
- package/src/server/routes/instance/httpapi/handlers/pty.ts +0 -273
- package/src/server/routes/instance/httpapi/handlers/question.ts +0 -54
- package/src/server/routes/instance/httpapi/handlers/session-errors.ts +0 -21
- package/src/server/routes/instance/httpapi/handlers/session.ts +0 -440
- package/src/server/routes/instance/httpapi/handlers/sync.ts +0 -89
- package/src/server/routes/instance/httpapi/handlers/tui.ts +0 -131
- package/src/server/routes/instance/httpapi/handlers/workspace.ts +0 -102
- package/src/server/routes/instance/httpapi/lifecycle.ts +0 -54
- package/src/server/routes/instance/httpapi/middleware/authorization.ts +0 -150
- package/src/server/routes/instance/httpapi/middleware/compression.ts +0 -64
- package/src/server/routes/instance/httpapi/middleware/cors-vary.ts +0 -29
- package/src/server/routes/instance/httpapi/middleware/error.ts +0 -43
- package/src/server/routes/instance/httpapi/middleware/fence.ts +0 -25
- package/src/server/routes/instance/httpapi/middleware/instance-context.ts +0 -43
- package/src/server/routes/instance/httpapi/middleware/proxy.ts +0 -108
- package/src/server/routes/instance/httpapi/middleware/schema-error.ts +0 -41
- package/src/server/routes/instance/httpapi/middleware/workspace-routing.ts +0 -250
- package/src/server/routes/instance/httpapi/public.ts +0 -535
- package/src/server/routes/instance/httpapi/server.ts +0 -298
- package/src/server/routes/instance/httpapi/websocket-tracker.ts +0 -57
- package/src/server/server.ts +0 -225
- package/src/server/shared/fence.ts +0 -60
- package/src/server/shared/pty-ticket.ts +0 -15
- package/src/server/shared/public-ui.ts +0 -12
- package/src/server/shared/tui-control.ts +0 -28
- package/src/server/shared/ui.ts +0 -108
- package/src/server/shared/workspace-routing.ts +0 -38
- package/src/server/tui-event.ts +0 -53
- package/src/session/compaction.ts +0 -620
- package/src/session/instruction.ts +0 -241
- package/src/session/llm/AGENTS.md +0 -90
- package/src/session/llm/ai-sdk.ts +0 -288
- package/src/session/llm/native-request.ts +0 -196
- package/src/session/llm/native-runtime.ts +0 -195
- package/src/session/llm/request.ts +0 -216
- package/src/session/llm.ts +0 -415
- package/src/session/message-error.ts +0 -14
- package/src/session/message-v2.ts +0 -747
- package/src/session/message.ts +0 -148
- package/src/session/overflow.ts +0 -34
- package/src/session/processor.ts +0 -1084
- package/src/session/prompt/anthropic.txt +0 -109
- package/src/session/prompt/beast.txt +0 -151
- package/src/session/prompt/build-switch.txt +0 -9
- package/src/session/prompt/codex.txt +0 -83
- package/src/session/prompt/copilot-gpt-5.txt +0 -147
- package/src/session/prompt/default.txt +0 -99
- package/src/session/prompt/gemini.txt +0 -159
- package/src/session/prompt/gpt.txt +0 -111
- package/src/session/prompt/kimi.txt +0 -99
- package/src/session/prompt/plan-mode.txt +0 -74
- package/src/session/prompt/plan-reminder-anthropic.txt +0 -71
- package/src/session/prompt/plan.txt +0 -30
- package/src/session/prompt/trinity.txt +0 -101
- package/src/session/prompt.ts +0 -1707
- package/src/session/reminders.ts +0 -92
- package/src/session/retry.ts +0 -201
- package/src/session/revert.ts +0 -160
- package/src/session/run-state.ts +0 -156
- package/src/session/schema.ts +0 -26
- package/src/session/session.ts +0 -1119
- package/src/session/status.ts +0 -97
- package/src/session/summary.ts +0 -165
- package/src/session/system.ts +0 -117
- package/src/session/todo.ts +0 -90
- package/src/session/tools.ts +0 -207
- package/src/share/session.ts +0 -61
- package/src/share/share-next.ts +0 -385
- package/src/skill/discovery.ts +0 -109
- package/src/skill/index.ts +0 -366
- package/src/snapshot/index.ts +0 -808
- package/src/sql.d.ts +0 -4
- package/src/storage/schema.ts +0 -5
- package/src/storage/storage.ts +0 -329
- package/src/sync/README.md +0 -179
- package/src/sync/schema.ts +0 -11
- package/src/temporary.ts +0 -31
- package/src/tool/apply_patch.ts +0 -315
- package/src/tool/apply_patch.txt +0 -33
- package/src/tool/apply_patch.zh.txt +0 -33
- package/src/tool/description.ts +0 -100
- package/src/tool/edit.ts +0 -739
- package/src/tool/edit.txt +0 -10
- package/src/tool/edit.zh.txt +0 -10
- package/src/tool/external-directory.ts +0 -49
- package/src/tool/glob.ts +0 -78
- package/src/tool/glob.txt +0 -6
- package/src/tool/glob.zh.txt +0 -6
- package/src/tool/grep.ts +0 -114
- package/src/tool/grep.txt +0 -8
- package/src/tool/grep.zh.txt +0 -8
- package/src/tool/invalid.ts +0 -21
- package/src/tool/json-schema.ts +0 -164
- package/src/tool/lsp.ts +0 -115
- package/src/tool/lsp.txt +0 -24
- package/src/tool/lsp.zh.txt +0 -24
- package/src/tool/mcp-websearch.ts +0 -96
- package/src/tool/plan-enter.txt +0 -14
- package/src/tool/plan-enter.zh.txt +0 -14
- package/src/tool/plan-exit.txt +0 -13
- package/src/tool/plan-exit.zh.txt +0 -13
- package/src/tool/plan.ts +0 -81
- package/src/tool/question.ts +0 -46
- package/src/tool/question.txt +0 -10
- package/src/tool/question.zh.txt +0 -10
- package/src/tool/read.ts +0 -388
- package/src/tool/read.txt +0 -14
- package/src/tool/read.zh.txt +0 -14
- package/src/tool/registry.ts +0 -440
- package/src/tool/schema.ts +0 -14
- package/src/tool/shell/id.ts +0 -19
- package/src/tool/shell/prompt.ts +0 -307
- package/src/tool/shell/shell.txt +0 -21
- package/src/tool/shell.ts +0 -657
- package/src/tool/skill.ts +0 -73
- package/src/tool/skill.txt +0 -5
- package/src/tool/skill.zh.txt +0 -5
- package/src/tool/task.ts +0 -348
- package/src/tool/task.txt +0 -19
- package/src/tool/task.zh.txt +0 -19
- package/src/tool/todo.ts +0 -59
- package/src/tool/todowrite.txt +0 -44
- package/src/tool/todowrite.zh.txt +0 -44
- package/src/tool/tool.ts +0 -183
- package/src/tool/truncate.ts +0 -158
- package/src/tool/truncation-dir.ts +0 -4
- package/src/tool/webfetch.ts +0 -194
- package/src/tool/webfetch.txt +0 -13
- package/src/tool/webfetch.zh.txt +0 -13
- package/src/tool/websearch.ts +0 -145
- package/src/tool/websearch.txt +0 -14
- package/src/tool/websearch.zh.txt +0 -14
- package/src/tool/write.ts +0 -106
- package/src/tool/write.txt +0 -8
- package/src/tool/write.zh.txt +0 -8
- package/src/util/archive.ts +0 -17
- package/src/util/bom.ts +0 -27
- package/src/util/data-url.ts +0 -9
- package/src/util/defer.ts +0 -10
- package/src/util/effect-http-client.ts +0 -11
- package/src/util/error.ts +0 -1
- package/src/util/filesystem.ts +0 -251
- package/src/util/html.ts +0 -8
- package/src/util/iife.ts +0 -3
- package/src/util/lazy.ts +0 -20
- package/src/util/local-context.ts +0 -25
- package/src/util/locale.ts +0 -2
- package/src/util/media.ts +0 -26
- package/src/util/process.ts +0 -177
- package/src/util/proxy-env.ts +0 -72
- package/src/util/queue.ts +0 -32
- package/src/util/record.ts +0 -1
- package/src/util/repository.ts +0 -232
- package/src/util/rpc.ts +0 -66
- package/src/util/signal.ts +0 -12
- package/src/util/timeout.ts +0 -13
- package/src/util/token.ts +0 -1
- package/src/util/wildcard.ts +0 -59
- package/src/worktree/index.ts +0 -654
package/src/cli/cmd/mcp.ts
DELETED
|
@@ -1,849 +0,0 @@
|
|
|
1
|
-
import { cmd } from "./cmd"
|
|
2
|
-
import { ConfigV1 } from "@opencode-ai/core/v1/config/config"
|
|
3
|
-
import { effectCmd } from "../effect-cmd"
|
|
4
|
-
import { Cause } from "effect"
|
|
5
|
-
import { Client } from "@modelcontextprotocol/sdk/client/index.js"
|
|
6
|
-
import { StreamableHTTPClientTransport } from "@modelcontextprotocol/sdk/client/streamableHttp.js"
|
|
7
|
-
import { UnauthorizedError } from "@modelcontextprotocol/sdk/client/auth.js"
|
|
8
|
-
import { LATEST_PROTOCOL_VERSION } from "@modelcontextprotocol/sdk/types.js"
|
|
9
|
-
import * as prompts from "@clack/prompts"
|
|
10
|
-
import { UI } from "../ui"
|
|
11
|
-
import { MCP } from "../../mcp"
|
|
12
|
-
import { McpAuth } from "../../mcp/auth"
|
|
13
|
-
import { McpOAuthProvider } from "../../mcp/oauth-provider"
|
|
14
|
-
import { Config } from "@/config/config"
|
|
15
|
-
import { ConfigMCPV1 } from "@opencode-ai/core/v1/config/mcp"
|
|
16
|
-
import { InstanceRef } from "@/effect/instance-ref"
|
|
17
|
-
import { InstallationVersion } from "@opencode-ai/core/installation/version"
|
|
18
|
-
import path from "path"
|
|
19
|
-
import { Global } from "@opencode-ai/core/global"
|
|
20
|
-
import { modify, applyEdits } from "jsonc-parser"
|
|
21
|
-
import { Filesystem } from "@/util/filesystem"
|
|
22
|
-
import { EventV2Bridge } from "@/event-v2-bridge"
|
|
23
|
-
import { EventV2 } from "@opencode-ai/core/event"
|
|
24
|
-
import { Effect } from "effect"
|
|
25
|
-
|
|
26
|
-
function getAuthStatusIcon(status: MCP.AuthStatus): string {
|
|
27
|
-
switch (status) {
|
|
28
|
-
case "authenticated":
|
|
29
|
-
return "✓"
|
|
30
|
-
case "expired":
|
|
31
|
-
return "⚠"
|
|
32
|
-
case "not_authenticated":
|
|
33
|
-
return "✗"
|
|
34
|
-
}
|
|
35
|
-
}
|
|
36
|
-
|
|
37
|
-
function getAuthStatusText(status: MCP.AuthStatus): string {
|
|
38
|
-
switch (status) {
|
|
39
|
-
case "authenticated":
|
|
40
|
-
return "authenticated"
|
|
41
|
-
case "expired":
|
|
42
|
-
return "expired"
|
|
43
|
-
case "not_authenticated":
|
|
44
|
-
return "not authenticated"
|
|
45
|
-
}
|
|
46
|
-
}
|
|
47
|
-
|
|
48
|
-
type McpEntry = NonNullable<ConfigV1.Info["mcp"]>[string]
|
|
49
|
-
|
|
50
|
-
type McpConfigured = ConfigMCPV1.Info
|
|
51
|
-
function isMcpConfigured(config: McpEntry): config is McpConfigured {
|
|
52
|
-
return typeof config === "object" && config !== null && "type" in config
|
|
53
|
-
}
|
|
54
|
-
|
|
55
|
-
type McpRemote = Extract<McpConfigured, { type: "remote" }>
|
|
56
|
-
function isMcpRemote(config: McpEntry): config is McpRemote {
|
|
57
|
-
return isMcpConfigured(config) && config.type === "remote"
|
|
58
|
-
}
|
|
59
|
-
|
|
60
|
-
function configuredServers(config: ConfigV1.Info) {
|
|
61
|
-
return Object.entries(config.mcp ?? {}).filter((entry): entry is [string, McpConfigured] => isMcpConfigured(entry[1]))
|
|
62
|
-
}
|
|
63
|
-
|
|
64
|
-
function oauthServers(config: ConfigV1.Info) {
|
|
65
|
-
return configuredServers(config).filter(
|
|
66
|
-
(entry): entry is [string, McpRemote] => isMcpRemote(entry[1]) && entry[1].oauth !== false,
|
|
67
|
-
)
|
|
68
|
-
}
|
|
69
|
-
|
|
70
|
-
function listState() {
|
|
71
|
-
return Effect.gen(function* () {
|
|
72
|
-
const cfg = yield* Config.Service
|
|
73
|
-
const mcp = yield* MCP.Service
|
|
74
|
-
const config = yield* cfg.get()
|
|
75
|
-
const statuses = yield* mcp.status()
|
|
76
|
-
const stored = yield* Effect.all(
|
|
77
|
-
Object.fromEntries(configuredServers(config).map(([name]) => [name, mcp.hasStoredTokens(name)])),
|
|
78
|
-
{ concurrency: "unbounded" },
|
|
79
|
-
)
|
|
80
|
-
return { config, statuses, stored }
|
|
81
|
-
})
|
|
82
|
-
}
|
|
83
|
-
|
|
84
|
-
function authState() {
|
|
85
|
-
return Effect.gen(function* () {
|
|
86
|
-
const cfg = yield* Config.Service
|
|
87
|
-
const mcp = yield* MCP.Service
|
|
88
|
-
const config = yield* cfg.get()
|
|
89
|
-
const auth = yield* Effect.all(
|
|
90
|
-
Object.fromEntries(oauthServers(config).map(([name]) => [name, mcp.getAuthStatus(name)])),
|
|
91
|
-
{ concurrency: "unbounded" },
|
|
92
|
-
)
|
|
93
|
-
return { config, auth }
|
|
94
|
-
})
|
|
95
|
-
}
|
|
96
|
-
|
|
97
|
-
export const McpCommand = cmd({
|
|
98
|
-
command: "mcp",
|
|
99
|
-
describe: "manage MCP (Model Context Protocol) servers",
|
|
100
|
-
builder: (yargs) =>
|
|
101
|
-
yargs
|
|
102
|
-
.command(McpAddCommand)
|
|
103
|
-
.command(McpListCommand)
|
|
104
|
-
.command(McpAuthCommand)
|
|
105
|
-
.command(McpLogoutCommand)
|
|
106
|
-
.command(McpDebugCommand)
|
|
107
|
-
.demandCommand(),
|
|
108
|
-
async handler() {},
|
|
109
|
-
})
|
|
110
|
-
|
|
111
|
-
export const McpListCommand = effectCmd({
|
|
112
|
-
command: "list",
|
|
113
|
-
aliases: ["ls"],
|
|
114
|
-
describe: "list MCP servers and their status",
|
|
115
|
-
handler: Effect.fn("Cli.mcp.list")(function* () {
|
|
116
|
-
UI.empty()
|
|
117
|
-
prompts.intro("MCP Servers")
|
|
118
|
-
|
|
119
|
-
const { config, statuses, stored } = yield* listState()
|
|
120
|
-
const servers = configuredServers(config)
|
|
121
|
-
|
|
122
|
-
if (servers.length === 0) {
|
|
123
|
-
prompts.log.warn("No MCP servers configured")
|
|
124
|
-
prompts.outro("Add servers with: opencode mcp add")
|
|
125
|
-
return
|
|
126
|
-
}
|
|
127
|
-
|
|
128
|
-
for (const [name, serverConfig] of servers) {
|
|
129
|
-
const status = statuses[name]
|
|
130
|
-
const hasOAuth = isMcpRemote(serverConfig) && !!serverConfig.oauth
|
|
131
|
-
const hasStoredTokens = stored[name]
|
|
132
|
-
|
|
133
|
-
let statusIcon: string
|
|
134
|
-
let statusText: string
|
|
135
|
-
let hint = ""
|
|
136
|
-
|
|
137
|
-
if (!status) {
|
|
138
|
-
statusIcon = "○"
|
|
139
|
-
statusText = "not initialized"
|
|
140
|
-
} else if (status.status === "connected") {
|
|
141
|
-
statusIcon = "✓"
|
|
142
|
-
statusText = "connected"
|
|
143
|
-
if (hasOAuth && hasStoredTokens) {
|
|
144
|
-
hint = " (OAuth)"
|
|
145
|
-
}
|
|
146
|
-
} else if (status.status === "disabled") {
|
|
147
|
-
statusIcon = "○"
|
|
148
|
-
statusText = "disabled"
|
|
149
|
-
} else if (status.status === "needs_auth") {
|
|
150
|
-
statusIcon = "⚠"
|
|
151
|
-
statusText = "needs authentication"
|
|
152
|
-
} else if (status.status === "needs_client_registration") {
|
|
153
|
-
statusIcon = "✗"
|
|
154
|
-
statusText = "needs client registration"
|
|
155
|
-
hint = "\n " + status.error
|
|
156
|
-
} else {
|
|
157
|
-
statusIcon = "✗"
|
|
158
|
-
statusText = "failed"
|
|
159
|
-
hint = "\n " + status.error
|
|
160
|
-
}
|
|
161
|
-
|
|
162
|
-
const typeHint = serverConfig.type === "remote" ? serverConfig.url : serverConfig.command.join(" ")
|
|
163
|
-
prompts.log.info(
|
|
164
|
-
`${statusIcon} ${name} ${UI.Style.TEXT_DIM}${statusText}${hint}\n ${UI.Style.TEXT_DIM}${typeHint}`,
|
|
165
|
-
)
|
|
166
|
-
}
|
|
167
|
-
|
|
168
|
-
prompts.outro(`${servers.length} server(s)`)
|
|
169
|
-
}),
|
|
170
|
-
})
|
|
171
|
-
|
|
172
|
-
export const McpAuthCommand = effectCmd({
|
|
173
|
-
command: "auth [name]",
|
|
174
|
-
describe: "authenticate with an OAuth-enabled MCP server",
|
|
175
|
-
builder: (yargs) =>
|
|
176
|
-
yargs
|
|
177
|
-
.positional("name", {
|
|
178
|
-
describe: "name of the MCP server",
|
|
179
|
-
type: "string",
|
|
180
|
-
})
|
|
181
|
-
.command(McpAuthListCommand),
|
|
182
|
-
handler: Effect.fn("Cli.mcp.auth")(function* (args) {
|
|
183
|
-
UI.empty()
|
|
184
|
-
prompts.intro("MCP OAuth Authentication")
|
|
185
|
-
|
|
186
|
-
const { config, auth } = yield* authState()
|
|
187
|
-
const mcpServers = config.mcp ?? {}
|
|
188
|
-
const servers = oauthServers(config)
|
|
189
|
-
|
|
190
|
-
if (servers.length === 0) {
|
|
191
|
-
prompts.log.warn("No OAuth-capable MCP servers configured")
|
|
192
|
-
prompts.log.info("Remote MCP servers support OAuth by default. Add a remote server in opencode.json:")
|
|
193
|
-
prompts.log.info(`
|
|
194
|
-
"mcp": {
|
|
195
|
-
"my-server": {
|
|
196
|
-
"type": "remote",
|
|
197
|
-
"url": "https://example.com/mcp"
|
|
198
|
-
}
|
|
199
|
-
}`)
|
|
200
|
-
prompts.outro("Done")
|
|
201
|
-
return
|
|
202
|
-
}
|
|
203
|
-
|
|
204
|
-
let serverName = args.name
|
|
205
|
-
if (!serverName) {
|
|
206
|
-
// Build options with auth status
|
|
207
|
-
const options = servers.map(([name, cfg]) => {
|
|
208
|
-
const authStatus = auth[name]
|
|
209
|
-
const icon = getAuthStatusIcon(authStatus)
|
|
210
|
-
const statusText = getAuthStatusText(authStatus)
|
|
211
|
-
const url = cfg.url
|
|
212
|
-
return {
|
|
213
|
-
label: `${icon} ${name} (${statusText})`,
|
|
214
|
-
value: name,
|
|
215
|
-
hint: url,
|
|
216
|
-
}
|
|
217
|
-
})
|
|
218
|
-
|
|
219
|
-
const selected = yield* Effect.promise(() =>
|
|
220
|
-
prompts.select({
|
|
221
|
-
message: "Select MCP server to authenticate",
|
|
222
|
-
options,
|
|
223
|
-
}),
|
|
224
|
-
)
|
|
225
|
-
if (prompts.isCancel(selected)) throw new UI.CancelledError()
|
|
226
|
-
serverName = selected
|
|
227
|
-
}
|
|
228
|
-
|
|
229
|
-
const serverConfig = mcpServers[serverName]
|
|
230
|
-
if (!serverConfig) {
|
|
231
|
-
prompts.log.error(`MCP server not found: ${serverName}`)
|
|
232
|
-
prompts.outro("Done")
|
|
233
|
-
return
|
|
234
|
-
}
|
|
235
|
-
|
|
236
|
-
if (!isMcpRemote(serverConfig) || serverConfig.oauth === false) {
|
|
237
|
-
prompts.log.error(`MCP server ${serverName} is not an OAuth-capable remote server`)
|
|
238
|
-
prompts.outro("Done")
|
|
239
|
-
return
|
|
240
|
-
}
|
|
241
|
-
|
|
242
|
-
// Check if already authenticated
|
|
243
|
-
const authStatus = auth[serverName] ?? (yield* MCP.Service.use((mcp) => mcp.getAuthStatus(serverName)))
|
|
244
|
-
if (authStatus === "authenticated") {
|
|
245
|
-
const confirm = yield* Effect.promise(() =>
|
|
246
|
-
prompts.confirm({
|
|
247
|
-
message: `${serverName} already has valid credentials. Re-authenticate?`,
|
|
248
|
-
}),
|
|
249
|
-
)
|
|
250
|
-
if (prompts.isCancel(confirm) || !confirm) {
|
|
251
|
-
prompts.outro("Cancelled")
|
|
252
|
-
return
|
|
253
|
-
}
|
|
254
|
-
} else if (authStatus === "expired") {
|
|
255
|
-
prompts.log.warn(`${serverName} has expired credentials. Re-authenticating...`)
|
|
256
|
-
}
|
|
257
|
-
|
|
258
|
-
const spinner = prompts.spinner()
|
|
259
|
-
spinner.start("Starting OAuth flow...")
|
|
260
|
-
|
|
261
|
-
// Subscribe to browser open failure events to show URL for manual opening
|
|
262
|
-
const events = yield* EventV2Bridge.Service
|
|
263
|
-
const unsubscribe = yield* events.listen((event) => {
|
|
264
|
-
if (event.type !== MCP.BrowserOpenFailed.type) return Effect.void
|
|
265
|
-
const data = event.data as EventV2.Data<typeof MCP.BrowserOpenFailed>
|
|
266
|
-
if (data.mcpName === serverName) {
|
|
267
|
-
spinner.stop("Could not open browser automatically")
|
|
268
|
-
prompts.log.warn("Please open this URL in your browser to authenticate:")
|
|
269
|
-
prompts.log.info(data.url)
|
|
270
|
-
spinner.start("Waiting for authorization...")
|
|
271
|
-
}
|
|
272
|
-
return Effect.void
|
|
273
|
-
})
|
|
274
|
-
|
|
275
|
-
yield* MCP.Service.use((mcp) => mcp.authenticate(serverName)).pipe(
|
|
276
|
-
Effect.tap((status) =>
|
|
277
|
-
Effect.sync(() => {
|
|
278
|
-
if (status.status === "connected") {
|
|
279
|
-
spinner.stop("Authentication successful!")
|
|
280
|
-
} else if (status.status === "needs_client_registration") {
|
|
281
|
-
spinner.stop("Authentication failed", 1)
|
|
282
|
-
prompts.log.error(status.error)
|
|
283
|
-
prompts.log.info("Add clientId to your MCP server config:")
|
|
284
|
-
prompts.log.info(`
|
|
285
|
-
"mcp": {
|
|
286
|
-
"${serverName}": {
|
|
287
|
-
"type": "remote",
|
|
288
|
-
"url": "${serverConfig.url}",
|
|
289
|
-
"oauth": {
|
|
290
|
-
"clientId": "your-client-id",
|
|
291
|
-
"clientSecret": "your-client-secret"
|
|
292
|
-
}
|
|
293
|
-
}
|
|
294
|
-
}`)
|
|
295
|
-
} else if (status.status === "failed") {
|
|
296
|
-
spinner.stop("Authentication failed", 1)
|
|
297
|
-
prompts.log.error(status.error)
|
|
298
|
-
} else {
|
|
299
|
-
spinner.stop("Unexpected status: " + status.status, 1)
|
|
300
|
-
}
|
|
301
|
-
}),
|
|
302
|
-
),
|
|
303
|
-
Effect.catchCause((cause) =>
|
|
304
|
-
Effect.sync(() => {
|
|
305
|
-
spinner.stop("Authentication failed", 1)
|
|
306
|
-
const error = Cause.squash(cause)
|
|
307
|
-
prompts.log.error(error instanceof Error ? error.message : String(error))
|
|
308
|
-
}),
|
|
309
|
-
),
|
|
310
|
-
Effect.ensuring(unsubscribe),
|
|
311
|
-
)
|
|
312
|
-
|
|
313
|
-
prompts.outro("Done")
|
|
314
|
-
}),
|
|
315
|
-
})
|
|
316
|
-
|
|
317
|
-
export const McpAuthListCommand = effectCmd({
|
|
318
|
-
command: "list",
|
|
319
|
-
aliases: ["ls"],
|
|
320
|
-
describe: "list OAuth-capable MCP servers and their auth status",
|
|
321
|
-
handler: Effect.fn("Cli.mcp.auth.list")(function* () {
|
|
322
|
-
UI.empty()
|
|
323
|
-
prompts.intro("MCP OAuth Status")
|
|
324
|
-
|
|
325
|
-
const { config, auth } = yield* authState()
|
|
326
|
-
const servers = oauthServers(config)
|
|
327
|
-
|
|
328
|
-
if (servers.length === 0) {
|
|
329
|
-
prompts.log.warn("No OAuth-capable MCP servers configured")
|
|
330
|
-
prompts.outro("Done")
|
|
331
|
-
return
|
|
332
|
-
}
|
|
333
|
-
|
|
334
|
-
for (const [name, serverConfig] of servers) {
|
|
335
|
-
const authStatus = auth[name]
|
|
336
|
-
const icon = getAuthStatusIcon(authStatus)
|
|
337
|
-
const statusText = getAuthStatusText(authStatus)
|
|
338
|
-
const url = serverConfig.url
|
|
339
|
-
|
|
340
|
-
prompts.log.info(`${icon} ${name} ${UI.Style.TEXT_DIM}${statusText}\n ${UI.Style.TEXT_DIM}${url}`)
|
|
341
|
-
}
|
|
342
|
-
|
|
343
|
-
prompts.outro(`${servers.length} OAuth-capable server(s)`)
|
|
344
|
-
}),
|
|
345
|
-
})
|
|
346
|
-
|
|
347
|
-
export const McpLogoutCommand = effectCmd({
|
|
348
|
-
command: "logout [name]",
|
|
349
|
-
describe: "remove OAuth credentials for an MCP server",
|
|
350
|
-
builder: (yargs) =>
|
|
351
|
-
yargs.positional("name", {
|
|
352
|
-
describe: "name of the MCP server",
|
|
353
|
-
type: "string",
|
|
354
|
-
}),
|
|
355
|
-
handler: Effect.fn("Cli.mcp.logout")(function* (args) {
|
|
356
|
-
UI.empty()
|
|
357
|
-
prompts.intro("MCP OAuth Logout")
|
|
358
|
-
|
|
359
|
-
const credentials = yield* McpAuth.Service.use((auth) => auth.all())
|
|
360
|
-
const serverNames = Object.keys(credentials)
|
|
361
|
-
|
|
362
|
-
if (serverNames.length === 0) {
|
|
363
|
-
prompts.log.warn("No MCP OAuth credentials stored")
|
|
364
|
-
prompts.outro("Done")
|
|
365
|
-
return
|
|
366
|
-
}
|
|
367
|
-
|
|
368
|
-
let serverName = args.name
|
|
369
|
-
if (!serverName) {
|
|
370
|
-
const selected = yield* Effect.promise(() =>
|
|
371
|
-
prompts.select({
|
|
372
|
-
message: "Select MCP server to logout",
|
|
373
|
-
options: serverNames.map((name) => {
|
|
374
|
-
const entry = credentials[name]
|
|
375
|
-
const hasTokens = !!entry.tokens
|
|
376
|
-
const hasClient = !!entry.clientInfo
|
|
377
|
-
let hint = ""
|
|
378
|
-
if (hasTokens && hasClient) hint = "tokens + client"
|
|
379
|
-
else if (hasTokens) hint = "tokens"
|
|
380
|
-
else if (hasClient) hint = "client registration"
|
|
381
|
-
return {
|
|
382
|
-
label: name,
|
|
383
|
-
value: name,
|
|
384
|
-
hint,
|
|
385
|
-
}
|
|
386
|
-
}),
|
|
387
|
-
}),
|
|
388
|
-
)
|
|
389
|
-
if (prompts.isCancel(selected)) throw new UI.CancelledError()
|
|
390
|
-
serverName = selected
|
|
391
|
-
}
|
|
392
|
-
|
|
393
|
-
if (!credentials[serverName]) {
|
|
394
|
-
prompts.log.error(`No credentials found for: ${serverName}`)
|
|
395
|
-
prompts.outro("Done")
|
|
396
|
-
return
|
|
397
|
-
}
|
|
398
|
-
|
|
399
|
-
yield* MCP.Service.use((mcp) => mcp.removeAuth(serverName))
|
|
400
|
-
prompts.log.success(`Removed OAuth credentials for ${serverName}`)
|
|
401
|
-
prompts.outro("Done")
|
|
402
|
-
}),
|
|
403
|
-
})
|
|
404
|
-
|
|
405
|
-
async function resolveConfigPath(baseDir: string, global = false) {
|
|
406
|
-
// Check for existing config files (prefer .jsonc over .json, check .opencode/ subdirectory too)
|
|
407
|
-
const candidates = [path.join(baseDir, "opencode.json"), path.join(baseDir, "opencode.jsonc")]
|
|
408
|
-
|
|
409
|
-
if (!global) {
|
|
410
|
-
candidates.push(path.join(baseDir, ".opencode", "opencode.json"), path.join(baseDir, ".opencode", "opencode.jsonc"))
|
|
411
|
-
}
|
|
412
|
-
|
|
413
|
-
for (const candidate of candidates) {
|
|
414
|
-
if (await Filesystem.exists(candidate)) {
|
|
415
|
-
return candidate
|
|
416
|
-
}
|
|
417
|
-
}
|
|
418
|
-
|
|
419
|
-
// Default to opencode.json if none exist
|
|
420
|
-
return candidates[0]
|
|
421
|
-
}
|
|
422
|
-
|
|
423
|
-
async function addMcpToConfig(name: string, mcpConfig: ConfigMCPV1.Info, configPath: string) {
|
|
424
|
-
let text = "{}"
|
|
425
|
-
if (await Filesystem.exists(configPath)) {
|
|
426
|
-
text = await Filesystem.readText(configPath)
|
|
427
|
-
}
|
|
428
|
-
|
|
429
|
-
// Use jsonc-parser to modify while preserving comments
|
|
430
|
-
const edits = modify(text, ["mcp", name], mcpConfig, {
|
|
431
|
-
formattingOptions: { tabSize: 2, insertSpaces: true },
|
|
432
|
-
})
|
|
433
|
-
const result = applyEdits(text, edits)
|
|
434
|
-
|
|
435
|
-
await Filesystem.write(configPath, result)
|
|
436
|
-
|
|
437
|
-
return configPath
|
|
438
|
-
}
|
|
439
|
-
|
|
440
|
-
export const McpAddCommand = effectCmd({
|
|
441
|
-
command: "add [name]",
|
|
442
|
-
describe: "add an MCP server",
|
|
443
|
-
builder: (yargs) =>
|
|
444
|
-
yargs
|
|
445
|
-
.positional("name", {
|
|
446
|
-
describe: "name of the MCP server",
|
|
447
|
-
type: "string",
|
|
448
|
-
})
|
|
449
|
-
.option("url", {
|
|
450
|
-
describe: "URL for a remote MCP server",
|
|
451
|
-
type: "string",
|
|
452
|
-
})
|
|
453
|
-
.option("env", {
|
|
454
|
-
describe: "environment variable for a local MCP server (KEY=VALUE)",
|
|
455
|
-
type: "string",
|
|
456
|
-
array: true,
|
|
457
|
-
})
|
|
458
|
-
.option("header", {
|
|
459
|
-
describe: "HTTP header for a remote MCP server (KEY=VALUE)",
|
|
460
|
-
type: "string",
|
|
461
|
-
array: true,
|
|
462
|
-
}),
|
|
463
|
-
handler: Effect.fn("Cli.mcp.add")(function* (args) {
|
|
464
|
-
const maybeCtx = yield* InstanceRef
|
|
465
|
-
if (!maybeCtx) return yield* Effect.die("InstanceRef not provided")
|
|
466
|
-
const ctx = maybeCtx
|
|
467
|
-
yield* Effect.promise(async () => {
|
|
468
|
-
const command = args["--"] ?? []
|
|
469
|
-
if (!args.name && (args.url || args.env?.length || args.header?.length || command.length)) {
|
|
470
|
-
throw new Error("A server name is required for non-interactive MCP configuration")
|
|
471
|
-
}
|
|
472
|
-
if (args.name) {
|
|
473
|
-
if (!!args.url === !!command.length) {
|
|
474
|
-
throw new Error("Provide either --url <url> or a command after --")
|
|
475
|
-
}
|
|
476
|
-
if (args.url && !URL.canParse(args.url)) {
|
|
477
|
-
throw new Error(`Invalid URL: ${args.url}`)
|
|
478
|
-
}
|
|
479
|
-
if (args.url && args.env?.length) {
|
|
480
|
-
throw new Error("--env is only valid for local MCP servers")
|
|
481
|
-
}
|
|
482
|
-
if (command.length && args.header?.length) {
|
|
483
|
-
throw new Error("--header is only valid for remote MCP servers")
|
|
484
|
-
}
|
|
485
|
-
|
|
486
|
-
const entries = (values: string[], kind: string) =>
|
|
487
|
-
Object.fromEntries(
|
|
488
|
-
values.map((entry) => {
|
|
489
|
-
const index = entry.indexOf("=")
|
|
490
|
-
if (index < 1) throw new Error(`Invalid ${kind}: ${entry}. Expected KEY=VALUE`)
|
|
491
|
-
return [entry.slice(0, index), entry.slice(index + 1)]
|
|
492
|
-
}),
|
|
493
|
-
)
|
|
494
|
-
const environment = entries(args.env ?? [], "environment variable")
|
|
495
|
-
const headers = entries(args.header ?? [], "HTTP header")
|
|
496
|
-
const mcpConfig: ConfigMCPV1.Info = args.url
|
|
497
|
-
? {
|
|
498
|
-
type: "remote",
|
|
499
|
-
url: args.url,
|
|
500
|
-
...(Object.keys(headers).length ? { headers } : {}),
|
|
501
|
-
}
|
|
502
|
-
: {
|
|
503
|
-
type: "local",
|
|
504
|
-
command,
|
|
505
|
-
...(Object.keys(environment).length ? { environment } : {}),
|
|
506
|
-
}
|
|
507
|
-
|
|
508
|
-
const configPath = await resolveConfigPath(Global.Path.config, true)
|
|
509
|
-
await addMcpToConfig(args.name, mcpConfig, configPath)
|
|
510
|
-
prompts.log.success(`MCP server "${args.name}" added to ${configPath}`)
|
|
511
|
-
return
|
|
512
|
-
}
|
|
513
|
-
|
|
514
|
-
UI.empty()
|
|
515
|
-
prompts.intro("Add MCP server")
|
|
516
|
-
|
|
517
|
-
const project = ctx.project
|
|
518
|
-
|
|
519
|
-
// Resolve config paths eagerly for hints
|
|
520
|
-
const [projectConfigPath, globalConfigPath] = await Promise.all([
|
|
521
|
-
resolveConfigPath(ctx.worktree),
|
|
522
|
-
resolveConfigPath(Global.Path.config, true),
|
|
523
|
-
])
|
|
524
|
-
|
|
525
|
-
// Determine scope
|
|
526
|
-
let configPath = globalConfigPath
|
|
527
|
-
if (project.vcs === "git") {
|
|
528
|
-
const scopeResult = await prompts.select({
|
|
529
|
-
message: "Location",
|
|
530
|
-
options: [
|
|
531
|
-
{
|
|
532
|
-
label: "Current project",
|
|
533
|
-
value: projectConfigPath,
|
|
534
|
-
hint: projectConfigPath,
|
|
535
|
-
},
|
|
536
|
-
{
|
|
537
|
-
label: "Global",
|
|
538
|
-
value: globalConfigPath,
|
|
539
|
-
hint: globalConfigPath,
|
|
540
|
-
},
|
|
541
|
-
],
|
|
542
|
-
})
|
|
543
|
-
if (prompts.isCancel(scopeResult)) throw new UI.CancelledError()
|
|
544
|
-
configPath = scopeResult
|
|
545
|
-
}
|
|
546
|
-
|
|
547
|
-
const name = await prompts.text({
|
|
548
|
-
message: "Enter MCP server name",
|
|
549
|
-
validate: (x) => (x && x.length > 0 ? undefined : "Required"),
|
|
550
|
-
})
|
|
551
|
-
if (prompts.isCancel(name)) throw new UI.CancelledError()
|
|
552
|
-
|
|
553
|
-
const type = await prompts.select({
|
|
554
|
-
message: "Select MCP server type",
|
|
555
|
-
options: [
|
|
556
|
-
{
|
|
557
|
-
label: "Local",
|
|
558
|
-
value: "local",
|
|
559
|
-
hint: "Run a local command",
|
|
560
|
-
},
|
|
561
|
-
{
|
|
562
|
-
label: "Remote",
|
|
563
|
-
value: "remote",
|
|
564
|
-
hint: "Connect to a remote URL",
|
|
565
|
-
},
|
|
566
|
-
],
|
|
567
|
-
})
|
|
568
|
-
if (prompts.isCancel(type)) throw new UI.CancelledError()
|
|
569
|
-
|
|
570
|
-
if (type === "local") {
|
|
571
|
-
const command = await prompts.text({
|
|
572
|
-
message: "Enter command to run",
|
|
573
|
-
placeholder: "e.g., opencode x @modelcontextprotocol/server-filesystem",
|
|
574
|
-
validate: (x) => (x && x.length > 0 ? undefined : "Required"),
|
|
575
|
-
})
|
|
576
|
-
if (prompts.isCancel(command)) throw new UI.CancelledError()
|
|
577
|
-
|
|
578
|
-
const mcpConfig: ConfigMCPV1.Info = {
|
|
579
|
-
type: "local",
|
|
580
|
-
command: command.split(" "),
|
|
581
|
-
}
|
|
582
|
-
|
|
583
|
-
await addMcpToConfig(name, mcpConfig, configPath)
|
|
584
|
-
prompts.log.success(`MCP server "${name}" added to ${configPath}`)
|
|
585
|
-
prompts.outro("MCP server added successfully")
|
|
586
|
-
return
|
|
587
|
-
}
|
|
588
|
-
|
|
589
|
-
if (type === "remote") {
|
|
590
|
-
const url = await prompts.text({
|
|
591
|
-
message: "Enter MCP server URL",
|
|
592
|
-
placeholder: "e.g., https://example.com/mcp",
|
|
593
|
-
validate: (x) => {
|
|
594
|
-
if (!x) return "Required"
|
|
595
|
-
if (x.length === 0) return "Required"
|
|
596
|
-
const isValid = URL.canParse(x)
|
|
597
|
-
return isValid ? undefined : "Invalid URL"
|
|
598
|
-
},
|
|
599
|
-
})
|
|
600
|
-
if (prompts.isCancel(url)) throw new UI.CancelledError()
|
|
601
|
-
|
|
602
|
-
const useOAuth = await prompts.confirm({
|
|
603
|
-
message: "Does this server require OAuth authentication?",
|
|
604
|
-
initialValue: false,
|
|
605
|
-
})
|
|
606
|
-
if (prompts.isCancel(useOAuth)) throw new UI.CancelledError()
|
|
607
|
-
|
|
608
|
-
let mcpConfig: ConfigMCPV1.Info
|
|
609
|
-
|
|
610
|
-
if (useOAuth) {
|
|
611
|
-
const hasClientId = await prompts.confirm({
|
|
612
|
-
message: "Do you have a pre-registered client ID?",
|
|
613
|
-
initialValue: false,
|
|
614
|
-
})
|
|
615
|
-
if (prompts.isCancel(hasClientId)) throw new UI.CancelledError()
|
|
616
|
-
|
|
617
|
-
if (hasClientId) {
|
|
618
|
-
const clientId = await prompts.text({
|
|
619
|
-
message: "Enter client ID",
|
|
620
|
-
validate: (x) => (x && x.length > 0 ? undefined : "Required"),
|
|
621
|
-
})
|
|
622
|
-
if (prompts.isCancel(clientId)) throw new UI.CancelledError()
|
|
623
|
-
|
|
624
|
-
const hasSecret = await prompts.confirm({
|
|
625
|
-
message: "Do you have a client secret?",
|
|
626
|
-
initialValue: false,
|
|
627
|
-
})
|
|
628
|
-
if (prompts.isCancel(hasSecret)) throw new UI.CancelledError()
|
|
629
|
-
|
|
630
|
-
let clientSecret: string | undefined
|
|
631
|
-
if (hasSecret) {
|
|
632
|
-
const secret = await prompts.password({
|
|
633
|
-
message: "Enter client secret",
|
|
634
|
-
})
|
|
635
|
-
if (prompts.isCancel(secret)) throw new UI.CancelledError()
|
|
636
|
-
clientSecret = secret
|
|
637
|
-
}
|
|
638
|
-
|
|
639
|
-
mcpConfig = {
|
|
640
|
-
type: "remote",
|
|
641
|
-
url,
|
|
642
|
-
oauth: {
|
|
643
|
-
clientId,
|
|
644
|
-
...(clientSecret && { clientSecret }),
|
|
645
|
-
},
|
|
646
|
-
}
|
|
647
|
-
} else {
|
|
648
|
-
mcpConfig = {
|
|
649
|
-
type: "remote",
|
|
650
|
-
url,
|
|
651
|
-
oauth: {},
|
|
652
|
-
}
|
|
653
|
-
}
|
|
654
|
-
} else {
|
|
655
|
-
mcpConfig = {
|
|
656
|
-
type: "remote",
|
|
657
|
-
url,
|
|
658
|
-
}
|
|
659
|
-
}
|
|
660
|
-
|
|
661
|
-
await addMcpToConfig(name, mcpConfig, configPath)
|
|
662
|
-
prompts.log.success(`MCP server "${name}" added to ${configPath}`)
|
|
663
|
-
}
|
|
664
|
-
|
|
665
|
-
prompts.outro("MCP server added successfully")
|
|
666
|
-
})
|
|
667
|
-
}),
|
|
668
|
-
})
|
|
669
|
-
|
|
670
|
-
export const McpDebugCommand = effectCmd({
|
|
671
|
-
command: "debug <name>",
|
|
672
|
-
describe: "debug OAuth connection for an MCP server",
|
|
673
|
-
builder: (yargs) =>
|
|
674
|
-
yargs.positional("name", {
|
|
675
|
-
describe: "name of the MCP server",
|
|
676
|
-
type: "string",
|
|
677
|
-
demandOption: true,
|
|
678
|
-
}),
|
|
679
|
-
handler: Effect.fn("Cli.mcp.debug")(function* (args) {
|
|
680
|
-
const config = yield* Config.Service.use((cfg) => cfg.get())
|
|
681
|
-
const mcp = yield* MCP.Service
|
|
682
|
-
const auth = yield* McpAuth.Service
|
|
683
|
-
yield* Effect.promise(async () => {
|
|
684
|
-
UI.empty()
|
|
685
|
-
prompts.intro("MCP OAuth Debug")
|
|
686
|
-
|
|
687
|
-
const mcpServers = config.mcp ?? {}
|
|
688
|
-
const serverName = args.name
|
|
689
|
-
|
|
690
|
-
const serverConfig = mcpServers[serverName]
|
|
691
|
-
if (!serverConfig) {
|
|
692
|
-
prompts.log.error(`MCP server not found: ${serverName}`)
|
|
693
|
-
prompts.outro("Done")
|
|
694
|
-
return
|
|
695
|
-
}
|
|
696
|
-
|
|
697
|
-
if (!isMcpRemote(serverConfig)) {
|
|
698
|
-
prompts.log.error(`MCP server ${serverName} is not a remote server`)
|
|
699
|
-
prompts.outro("Done")
|
|
700
|
-
return
|
|
701
|
-
}
|
|
702
|
-
|
|
703
|
-
if (serverConfig.oauth === false) {
|
|
704
|
-
prompts.log.warn(`MCP server ${serverName} has OAuth explicitly disabled`)
|
|
705
|
-
prompts.outro("Done")
|
|
706
|
-
return
|
|
707
|
-
}
|
|
708
|
-
|
|
709
|
-
prompts.log.info(`Server: ${serverName}`)
|
|
710
|
-
prompts.log.info(`URL: ${serverConfig.url}`)
|
|
711
|
-
|
|
712
|
-
// Check stored auth status — services already in hand, run inline.
|
|
713
|
-
const { authStatus, entry } = await Effect.runPromise(
|
|
714
|
-
Effect.all({
|
|
715
|
-
authStatus: mcp.getAuthStatus(serverName),
|
|
716
|
-
entry: auth.get(serverName),
|
|
717
|
-
}),
|
|
718
|
-
)
|
|
719
|
-
prompts.log.info(`Auth status: ${getAuthStatusIcon(authStatus)} ${getAuthStatusText(authStatus)}`)
|
|
720
|
-
|
|
721
|
-
if (entry?.tokens) {
|
|
722
|
-
prompts.log.info(` Access token: ${entry.tokens.accessToken.substring(0, 20)}...`)
|
|
723
|
-
if (entry.tokens.expiresAt) {
|
|
724
|
-
const expiresDate = new Date(entry.tokens.expiresAt * 1000)
|
|
725
|
-
const isExpired = entry.tokens.expiresAt < Date.now() / 1000
|
|
726
|
-
prompts.log.info(` Expires: ${expiresDate.toISOString()} ${isExpired ? "(EXPIRED)" : ""}`)
|
|
727
|
-
}
|
|
728
|
-
if (entry.tokens.refreshToken) {
|
|
729
|
-
prompts.log.info(` Refresh token: present`)
|
|
730
|
-
}
|
|
731
|
-
}
|
|
732
|
-
if (entry?.clientInfo) {
|
|
733
|
-
prompts.log.info(` Client ID: ${entry.clientInfo.clientId}`)
|
|
734
|
-
if (entry.clientInfo.clientSecretExpiresAt) {
|
|
735
|
-
const expiresDate = new Date(entry.clientInfo.clientSecretExpiresAt * 1000)
|
|
736
|
-
prompts.log.info(` Client secret expires: ${expiresDate.toISOString()}`)
|
|
737
|
-
}
|
|
738
|
-
}
|
|
739
|
-
|
|
740
|
-
const spinner = prompts.spinner()
|
|
741
|
-
spinner.start("Testing connection...")
|
|
742
|
-
|
|
743
|
-
// Test basic HTTP connectivity first
|
|
744
|
-
try {
|
|
745
|
-
const response = await fetch(serverConfig.url, {
|
|
746
|
-
method: "POST",
|
|
747
|
-
headers: {
|
|
748
|
-
...serverConfig.headers,
|
|
749
|
-
"Content-Type": "application/json",
|
|
750
|
-
Accept: "application/json, text/event-stream",
|
|
751
|
-
},
|
|
752
|
-
body: JSON.stringify({
|
|
753
|
-
jsonrpc: "2.0",
|
|
754
|
-
method: "initialize",
|
|
755
|
-
params: {
|
|
756
|
-
protocolVersion: LATEST_PROTOCOL_VERSION,
|
|
757
|
-
capabilities: {},
|
|
758
|
-
clientInfo: { name: "opencode-debug", version: InstallationVersion },
|
|
759
|
-
},
|
|
760
|
-
id: 1,
|
|
761
|
-
}),
|
|
762
|
-
})
|
|
763
|
-
|
|
764
|
-
spinner.stop(`HTTP response: ${response.status} ${response.statusText}`)
|
|
765
|
-
|
|
766
|
-
// Check for WWW-Authenticate header
|
|
767
|
-
const wwwAuth = response.headers.get("www-authenticate")
|
|
768
|
-
if (wwwAuth) {
|
|
769
|
-
prompts.log.info(`WWW-Authenticate: ${wwwAuth}`)
|
|
770
|
-
}
|
|
771
|
-
|
|
772
|
-
if (response.status === 401) {
|
|
773
|
-
prompts.log.warn("Server returned 401 Unauthorized")
|
|
774
|
-
|
|
775
|
-
// Try to discover OAuth metadata
|
|
776
|
-
const oauthConfig = typeof serverConfig.oauth === "object" ? serverConfig.oauth : undefined
|
|
777
|
-
const authProvider = new McpOAuthProvider(
|
|
778
|
-
serverName,
|
|
779
|
-
serverConfig.url,
|
|
780
|
-
{
|
|
781
|
-
clientId: oauthConfig?.clientId,
|
|
782
|
-
clientSecret: oauthConfig?.clientSecret,
|
|
783
|
-
scope: oauthConfig?.scope,
|
|
784
|
-
redirectUri: oauthConfig?.redirectUri,
|
|
785
|
-
},
|
|
786
|
-
{
|
|
787
|
-
onRedirect: async () => {},
|
|
788
|
-
},
|
|
789
|
-
auth,
|
|
790
|
-
)
|
|
791
|
-
|
|
792
|
-
prompts.log.info("Testing OAuth flow (without completing authorization)...")
|
|
793
|
-
|
|
794
|
-
// Try creating transport with auth provider to trigger discovery
|
|
795
|
-
const transport = new StreamableHTTPClientTransport(new URL(serverConfig.url), {
|
|
796
|
-
authProvider,
|
|
797
|
-
requestInit: serverConfig.headers ? { headers: serverConfig.headers } : undefined,
|
|
798
|
-
})
|
|
799
|
-
|
|
800
|
-
try {
|
|
801
|
-
const client = new Client({
|
|
802
|
-
name: "opencode-debug",
|
|
803
|
-
version: InstallationVersion,
|
|
804
|
-
})
|
|
805
|
-
await client.connect(transport)
|
|
806
|
-
prompts.log.success("Connection successful (already authenticated)")
|
|
807
|
-
await client.close()
|
|
808
|
-
} catch (error) {
|
|
809
|
-
if (error instanceof UnauthorizedError) {
|
|
810
|
-
prompts.log.info(`OAuth flow triggered: ${error.message}`)
|
|
811
|
-
|
|
812
|
-
// Check if dynamic registration would be attempted
|
|
813
|
-
const clientInfo = await authProvider.clientInformation()
|
|
814
|
-
if (clientInfo) {
|
|
815
|
-
prompts.log.info(`Client ID available: ${clientInfo.client_id}`)
|
|
816
|
-
} else {
|
|
817
|
-
prompts.log.info("No client ID - dynamic registration will be attempted")
|
|
818
|
-
}
|
|
819
|
-
} else {
|
|
820
|
-
prompts.log.error(`Connection error: ${error instanceof Error ? error.message : String(error)}`)
|
|
821
|
-
}
|
|
822
|
-
}
|
|
823
|
-
} else if (response.status >= 200 && response.status < 300) {
|
|
824
|
-
prompts.log.success("Server responded successfully (no auth required or already authenticated)")
|
|
825
|
-
const body = await response.text()
|
|
826
|
-
try {
|
|
827
|
-
const json = JSON.parse(body)
|
|
828
|
-
if (json.result?.serverInfo) {
|
|
829
|
-
prompts.log.info(`Server info: ${JSON.stringify(json.result.serverInfo)}`)
|
|
830
|
-
}
|
|
831
|
-
} catch {
|
|
832
|
-
// Not JSON, ignore
|
|
833
|
-
}
|
|
834
|
-
} else {
|
|
835
|
-
prompts.log.warn(`Unexpected status: ${response.status}`)
|
|
836
|
-
const body = await response.text().catch(() => "")
|
|
837
|
-
if (body) {
|
|
838
|
-
prompts.log.info(`Response body: ${body.substring(0, 500)}`)
|
|
839
|
-
}
|
|
840
|
-
}
|
|
841
|
-
} catch (error) {
|
|
842
|
-
spinner.stop("Connection failed", 1)
|
|
843
|
-
prompts.log.error(`Error: ${error instanceof Error ? error.message : String(error)}`)
|
|
844
|
-
}
|
|
845
|
-
|
|
846
|
-
prompts.outro("Debug complete")
|
|
847
|
-
})
|
|
848
|
-
}),
|
|
849
|
-
})
|