opencode-v2 1.1.53
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/AGENTS.md +27 -0
- package/Dockerfile +18 -0
- package/README.md +15 -0
- package/bin/opencode +84 -0
- package/bunfig.toml +5 -0
- package/package.json +126 -0
- package/parsers-config.ts +253 -0
- package/script/build.ts +193 -0
- package/script/postinstall.mjs +125 -0
- package/script/publish.ts +181 -0
- package/script/schema.ts +47 -0
- package/script/seed-e2e.ts +50 -0
- package/src/acp/README.md +164 -0
- package/src/acp/agent.ts +1676 -0
- package/src/acp/session.ts +117 -0
- package/src/acp/types.ts +23 -0
- package/src/agent/agent.ts +414 -0
- package/src/agent/generate.txt +75 -0
- package/src/agent/prompt/compaction.txt +12 -0
- package/src/agent/prompt/explore.txt +18 -0
- package/src/agent/prompt/summary.txt +11 -0
- package/src/agent/prompt/title.txt +44 -0
- package/src/auth/index.ts +70 -0
- package/src/bun/index.ts +137 -0
- package/src/bun/registry.ts +48 -0
- package/src/bus/bus-event.ts +43 -0
- package/src/bus/global.ts +10 -0
- package/src/bus/index.ts +105 -0
- package/src/cli/bootstrap.ts +17 -0
- package/src/cli/cmd/acp.ts +70 -0
- package/src/cli/cmd/agent.ts +257 -0
- package/src/cli/cmd/auth.ts +400 -0
- package/src/cli/cmd/cmd.ts +7 -0
- package/src/cli/cmd/debug/agent.ts +167 -0
- package/src/cli/cmd/debug/config.ts +16 -0
- package/src/cli/cmd/debug/file.ts +97 -0
- package/src/cli/cmd/debug/index.ts +48 -0
- package/src/cli/cmd/debug/lsp.ts +52 -0
- package/src/cli/cmd/debug/ripgrep.ts +87 -0
- package/src/cli/cmd/debug/scrap.ts +16 -0
- package/src/cli/cmd/debug/skill.ts +16 -0
- package/src/cli/cmd/debug/snapshot.ts +52 -0
- package/src/cli/cmd/export.ts +88 -0
- package/src/cli/cmd/generate.ts +38 -0
- package/src/cli/cmd/github.ts +1540 -0
- package/src/cli/cmd/import.ts +147 -0
- package/src/cli/cmd/mcp.ts +755 -0
- package/src/cli/cmd/models.ts +77 -0
- package/src/cli/cmd/pr.ts +112 -0
- package/src/cli/cmd/run.ts +617 -0
- package/src/cli/cmd/serve.ts +20 -0
- package/src/cli/cmd/session.ts +135 -0
- package/src/cli/cmd/stats.ts +426 -0
- package/src/cli/cmd/tui/app.tsx +801 -0
- package/src/cli/cmd/tui/attach.ts +52 -0
- package/src/cli/cmd/tui/component/border.tsx +21 -0
- package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
- package/src/cli/cmd/tui/component/dialog-command.tsx +148 -0
- package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
- package/src/cli/cmd/tui/component/dialog-model.tsx +234 -0
- package/src/cli/cmd/tui/component/dialog-provider.tsx +266 -0
- package/src/cli/cmd/tui/component/dialog-session-list.tsx +108 -0
- package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
- package/src/cli/cmd/tui/component/dialog-skill.tsx +36 -0
- package/src/cli/cmd/tui/component/dialog-stash.tsx +87 -0
- package/src/cli/cmd/tui/component/dialog-status.tsx +177 -0
- package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
- package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
- package/src/cli/cmd/tui/component/logo.tsx +85 -0
- package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +666 -0
- package/src/cli/cmd/tui/component/prompt/frecency.tsx +89 -0
- package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
- package/src/cli/cmd/tui/component/prompt/index.tsx +1132 -0
- package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
- package/src/cli/cmd/tui/component/spinner.tsx +24 -0
- package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
- package/src/cli/cmd/tui/component/tips.tsx +153 -0
- package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
- package/src/cli/cmd/tui/context/args.tsx +15 -0
- package/src/cli/cmd/tui/context/directory.ts +13 -0
- package/src/cli/cmd/tui/context/exit.tsx +52 -0
- package/src/cli/cmd/tui/context/helper.tsx +25 -0
- package/src/cli/cmd/tui/context/keybind.tsx +100 -0
- package/src/cli/cmd/tui/context/kv.tsx +52 -0
- package/src/cli/cmd/tui/context/local.tsx +409 -0
- package/src/cli/cmd/tui/context/prompt.tsx +18 -0
- package/src/cli/cmd/tui/context/route.tsx +46 -0
- package/src/cli/cmd/tui/context/sdk.tsx +101 -0
- package/src/cli/cmd/tui/context/sync.tsx +470 -0
- package/src/cli/cmd/tui/context/theme/aura.json +69 -0
- package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
- package/src/cli/cmd/tui/context/theme/carbonfox.json +248 -0
- package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
- package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
- package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
- package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
- package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
- package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
- package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
- package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
- package/src/cli/cmd/tui/context/theme/github.json +233 -0
- package/src/cli/cmd/tui/context/theme/gruvbox.json +242 -0
- package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
- package/src/cli/cmd/tui/context/theme/lucent-orng.json +237 -0
- package/src/cli/cmd/tui/context/theme/material.json +235 -0
- package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
- package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
- package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
- package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
- package/src/cli/cmd/tui/context/theme/nord.json +223 -0
- package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
- package/src/cli/cmd/tui/context/theme/orng.json +249 -0
- package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
- package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
- package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
- package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
- package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
- package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
- package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
- package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
- package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
- package/src/cli/cmd/tui/context/theme.tsx +1152 -0
- package/src/cli/cmd/tui/event.ts +48 -0
- package/src/cli/cmd/tui/routes/home.tsx +140 -0
- package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
- package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
- package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
- package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
- package/src/cli/cmd/tui/routes/session/footer.tsx +91 -0
- package/src/cli/cmd/tui/routes/session/header.tsx +142 -0
- package/src/cli/cmd/tui/routes/session/index.tsx +2126 -0
- package/src/cli/cmd/tui/routes/session/permission.tsx +508 -0
- package/src/cli/cmd/tui/routes/session/question.tsx +466 -0
- package/src/cli/cmd/tui/routes/session/sidebar.tsx +313 -0
- package/src/cli/cmd/tui/thread.ts +175 -0
- package/src/cli/cmd/tui/ui/dialog-alert.tsx +68 -0
- package/src/cli/cmd/tui/ui/dialog-confirm.tsx +93 -0
- package/src/cli/cmd/tui/ui/dialog-export-options.tsx +215 -0
- package/src/cli/cmd/tui/ui/dialog-help.tsx +49 -0
- package/src/cli/cmd/tui/ui/dialog-prompt.tsx +88 -0
- package/src/cli/cmd/tui/ui/dialog-select.tsx +399 -0
- package/src/cli/cmd/tui/ui/dialog.tsx +167 -0
- package/src/cli/cmd/tui/ui/link.tsx +28 -0
- package/src/cli/cmd/tui/ui/spinner.ts +368 -0
- package/src/cli/cmd/tui/ui/toast.tsx +100 -0
- package/src/cli/cmd/tui/util/clipboard.ts +159 -0
- package/src/cli/cmd/tui/util/editor.ts +32 -0
- package/src/cli/cmd/tui/util/signal.ts +7 -0
- package/src/cli/cmd/tui/util/terminal.ts +114 -0
- package/src/cli/cmd/tui/util/transcript.ts +98 -0
- package/src/cli/cmd/tui/worker.ts +152 -0
- package/src/cli/cmd/uninstall.ts +357 -0
- package/src/cli/cmd/upgrade.ts +73 -0
- package/src/cli/cmd/web.ts +81 -0
- package/src/cli/error.ts +57 -0
- package/src/cli/logo.ts +6 -0
- package/src/cli/network.ts +60 -0
- package/src/cli/ui.ts +113 -0
- package/src/cli/upgrade.ts +25 -0
- package/src/command/index.ts +150 -0
- package/src/command/template/initialize.txt +10 -0
- package/src/command/template/review.txt +99 -0
- package/src/config/config.ts +1477 -0
- package/src/config/markdown.ts +98 -0
- package/src/env/index.ts +28 -0
- package/src/file/ignore.ts +83 -0
- package/src/file/index.ts +583 -0
- package/src/file/ripgrep.ts +375 -0
- package/src/file/time.ts +69 -0
- package/src/file/watcher.ts +127 -0
- package/src/flag/flag.ts +97 -0
- package/src/format/formatter.ts +366 -0
- package/src/format/index.ts +137 -0
- package/src/global/index.ts +55 -0
- package/src/id/id.ts +83 -0
- package/src/ide/index.ts +76 -0
- package/src/index.ts +159 -0
- package/src/installation/index.ts +246 -0
- package/src/lsp/client.ts +252 -0
- package/src/lsp/index.ts +485 -0
- package/src/lsp/language.ts +119 -0
- package/src/lsp/server.ts +2046 -0
- package/src/mcp/auth.ts +132 -0
- package/src/mcp/index.ts +934 -0
- package/src/mcp/oauth-callback.ts +200 -0
- package/src/mcp/oauth-provider.ts +154 -0
- package/src/patch/index.ts +680 -0
- package/src/permission/arity.ts +163 -0
- package/src/permission/index.ts +210 -0
- package/src/permission/next.ts +280 -0
- package/src/plugin/codex.ts +624 -0
- package/src/plugin/copilot.ts +327 -0
- package/src/plugin/index.ts +138 -0
- package/src/project/bootstrap.ts +35 -0
- package/src/project/instance.ts +114 -0
- package/src/project/project.ts +371 -0
- package/src/project/state.ts +70 -0
- package/src/project/vcs.ts +76 -0
- package/src/provider/auth.ts +147 -0
- package/src/provider/models.ts +133 -0
- package/src/provider/provider.ts +1262 -0
- package/src/provider/sdk/copilot/README.md +5 -0
- package/src/provider/sdk/copilot/chat/convert-to-openai-compatible-chat-messages.ts +164 -0
- package/src/provider/sdk/copilot/chat/get-response-metadata.ts +15 -0
- package/src/provider/sdk/copilot/chat/map-openai-compatible-finish-reason.ts +17 -0
- package/src/provider/sdk/copilot/chat/openai-compatible-api-types.ts +64 -0
- package/src/provider/sdk/copilot/chat/openai-compatible-chat-language-model.ts +780 -0
- package/src/provider/sdk/copilot/chat/openai-compatible-chat-options.ts +28 -0
- package/src/provider/sdk/copilot/chat/openai-compatible-metadata-extractor.ts +44 -0
- package/src/provider/sdk/copilot/chat/openai-compatible-prepare-tools.ts +87 -0
- package/src/provider/sdk/copilot/copilot-provider.ts +100 -0
- package/src/provider/sdk/copilot/index.ts +2 -0
- package/src/provider/sdk/copilot/openai-compatible-error.ts +27 -0
- package/src/provider/sdk/copilot/responses/convert-to-openai-responses-input.ts +303 -0
- package/src/provider/sdk/copilot/responses/map-openai-responses-finish-reason.ts +22 -0
- package/src/provider/sdk/copilot/responses/openai-config.ts +18 -0
- package/src/provider/sdk/copilot/responses/openai-error.ts +22 -0
- package/src/provider/sdk/copilot/responses/openai-responses-api-types.ts +207 -0
- package/src/provider/sdk/copilot/responses/openai-responses-language-model.ts +1732 -0
- package/src/provider/sdk/copilot/responses/openai-responses-prepare-tools.ts +177 -0
- package/src/provider/sdk/copilot/responses/openai-responses-settings.ts +1 -0
- package/src/provider/sdk/copilot/responses/tool/code-interpreter.ts +88 -0
- package/src/provider/sdk/copilot/responses/tool/file-search.ts +128 -0
- package/src/provider/sdk/copilot/responses/tool/image-generation.ts +115 -0
- package/src/provider/sdk/copilot/responses/tool/local-shell.ts +65 -0
- package/src/provider/sdk/copilot/responses/tool/web-search-preview.ts +104 -0
- package/src/provider/sdk/copilot/responses/tool/web-search.ts +103 -0
- package/src/provider/transform.ts +828 -0
- package/src/pty/index.ts +250 -0
- package/src/question/index.ts +171 -0
- package/src/scheduler/index.ts +61 -0
- package/src/server/error.ts +36 -0
- package/src/server/event.ts +7 -0
- package/src/server/mdns.ts +60 -0
- package/src/server/routes/config.ts +92 -0
- package/src/server/routes/experimental.ts +208 -0
- package/src/server/routes/file.ts +197 -0
- package/src/server/routes/global.ts +183 -0
- package/src/server/routes/mcp.ts +225 -0
- package/src/server/routes/permission.ts +68 -0
- package/src/server/routes/project.ts +82 -0
- package/src/server/routes/provider.ts +165 -0
- package/src/server/routes/pty.ts +169 -0
- package/src/server/routes/question.ts +98 -0
- package/src/server/routes/session.ts +939 -0
- package/src/server/routes/tui.ts +379 -0
- package/src/server/server.ts +613 -0
- package/src/session/compaction.ts +226 -0
- package/src/session/index.ts +524 -0
- package/src/session/instruction.ts +197 -0
- package/src/session/llm.ts +289 -0
- package/src/session/message-v2.ts +802 -0
- package/src/session/message.ts +189 -0
- package/src/session/processor.ts +407 -0
- package/src/session/prompt/agent.txt +43 -0
- package/src/session/prompt/anthropic-20250930.txt +166 -0
- package/src/session/prompt/anthropic.txt +105 -0
- package/src/session/prompt/beast.txt +147 -0
- package/src/session/prompt/build-switch.txt +5 -0
- package/src/session/prompt/codex_header.txt +79 -0
- package/src/session/prompt/copilot-gpt-5.txt +143 -0
- package/src/session/prompt/gemini.txt +155 -0
- package/src/session/prompt/max-steps.txt +16 -0
- package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
- package/src/session/prompt/plan.txt +26 -0
- package/src/session/prompt/qwen.txt +109 -0
- package/src/session/prompt/research.txt +81 -0
- package/src/session/prompt/trinity.txt +97 -0
- package/src/session/prompt.ts +1952 -0
- package/src/session/retry.ts +97 -0
- package/src/session/revert.ts +121 -0
- package/src/session/status.ts +76 -0
- package/src/session/summary.ts +217 -0
- package/src/session/system.ts +54 -0
- package/src/session/todo.ts +37 -0
- package/src/share/share-next.ts +200 -0
- package/src/share/share.ts +92 -0
- package/src/shell/shell.ts +67 -0
- package/src/skill/discovery.ts +97 -0
- package/src/skill/index.ts +1 -0
- package/src/skill/skill.ts +188 -0
- package/src/snapshot/index.ts +255 -0
- package/src/storage/storage.ts +227 -0
- package/src/tool/agent-enter.txt +1 -0
- package/src/tool/agent-exit.txt +1 -0
- package/src/tool/agent.ts +237 -0
- package/src/tool/apply_patch.ts +281 -0
- package/src/tool/apply_patch.txt +33 -0
- package/src/tool/bash.ts +269 -0
- package/src/tool/bash.txt +115 -0
- package/src/tool/batch.ts +175 -0
- package/src/tool/batch.txt +24 -0
- package/src/tool/chat-enter.txt +15 -0
- package/src/tool/chat-exit.txt +7 -0
- package/src/tool/chat.ts +217 -0
- package/src/tool/codesearch.ts +132 -0
- package/src/tool/codesearch.txt +12 -0
- package/src/tool/edit.ts +655 -0
- package/src/tool/edit.txt +10 -0
- package/src/tool/external-directory.ts +32 -0
- package/src/tool/glob.ts +78 -0
- package/src/tool/glob.txt +6 -0
- package/src/tool/grep.ts +147 -0
- package/src/tool/grep.txt +8 -0
- package/src/tool/invalid.ts +17 -0
- package/src/tool/ls.ts +121 -0
- package/src/tool/ls.txt +1 -0
- package/src/tool/lsp.ts +96 -0
- package/src/tool/lsp.txt +19 -0
- package/src/tool/multiedit.ts +46 -0
- package/src/tool/multiedit.txt +41 -0
- package/src/tool/plan-enter.txt +14 -0
- package/src/tool/plan-exit.txt +13 -0
- package/src/tool/plan.ts +130 -0
- package/src/tool/question.ts +33 -0
- package/src/tool/question.txt +10 -0
- package/src/tool/read.ts +211 -0
- package/src/tool/read.txt +12 -0
- package/src/tool/registry.ts +167 -0
- package/src/tool/research-enter.txt +1 -0
- package/src/tool/research-exit.txt +1 -0
- package/src/tool/research.ts +134 -0
- package/src/tool/skill.ts +123 -0
- package/src/tool/task.ts +165 -0
- package/src/tool/task.txt +60 -0
- package/src/tool/todo.ts +53 -0
- package/src/tool/todoread.txt +14 -0
- package/src/tool/todowrite.txt +167 -0
- package/src/tool/tool.ts +89 -0
- package/src/tool/truncation.ts +106 -0
- package/src/tool/webfetch.ts +186 -0
- package/src/tool/webfetch.txt +13 -0
- package/src/tool/websearch.ts +150 -0
- package/src/tool/websearch.txt +14 -0
- package/src/tool/write.ts +85 -0
- package/src/tool/write.txt +8 -0
- package/src/util/abort.ts +35 -0
- package/src/util/archive.ts +16 -0
- package/src/util/color.ts +19 -0
- package/src/util/context.ts +25 -0
- package/src/util/defer.ts +12 -0
- package/src/util/eventloop.ts +20 -0
- package/src/util/filesystem.ts +93 -0
- package/src/util/fn.ts +11 -0
- package/src/util/format.ts +20 -0
- package/src/util/iife.ts +3 -0
- package/src/util/keybind.ts +103 -0
- package/src/util/lazy.ts +18 -0
- package/src/util/locale.ts +81 -0
- package/src/util/lock.ts +98 -0
- package/src/util/log.ts +180 -0
- package/src/util/proxied.ts +3 -0
- package/src/util/queue.ts +32 -0
- package/src/util/rpc.ts +66 -0
- package/src/util/scrap.ts +10 -0
- package/src/util/signal.ts +12 -0
- package/src/util/timeout.ts +14 -0
- package/src/util/token.ts +7 -0
- package/src/util/wildcard.ts +56 -0
- package/src/worktree/index.ts +574 -0
- package/sst-env.d.ts +9 -0
- package/test/acp/agent-interface.test.ts +51 -0
- package/test/acp/event-subscription.test.ts +436 -0
- package/test/agent/agent.test.ts +675 -0
- package/test/bun.test.ts +53 -0
- package/test/cli/github-action.test.ts +161 -0
- package/test/cli/github-remote.test.ts +80 -0
- package/test/cli/import.test.ts +38 -0
- package/test/cli/tui/transcript.test.ts +322 -0
- package/test/config/agent-color.test.ts +71 -0
- package/test/config/config.test.ts +1802 -0
- package/test/config/fixtures/empty-frontmatter.md +4 -0
- package/test/config/fixtures/frontmatter.md +28 -0
- package/test/config/fixtures/markdown-header.md +11 -0
- package/test/config/fixtures/no-frontmatter.md +1 -0
- package/test/config/fixtures/weird-model-id.md +13 -0
- package/test/config/markdown.test.ts +228 -0
- package/test/file/ignore.test.ts +10 -0
- package/test/file/path-traversal.test.ts +198 -0
- package/test/file/ripgrep.test.ts +39 -0
- package/test/fixture/fixture.ts +45 -0
- package/test/fixture/lsp/fake-lsp-server.js +77 -0
- package/test/ide/ide.test.ts +82 -0
- package/test/keybind.test.ts +421 -0
- package/test/lsp/client.test.ts +95 -0
- package/test/mcp/headers.test.ts +153 -0
- package/test/mcp/oauth-browser.test.ts +249 -0
- package/test/memory/abort-leak.test.ts +136 -0
- package/test/patch/patch.test.ts +348 -0
- package/test/permission/arity.test.ts +33 -0
- package/test/permission/next.test.ts +690 -0
- package/test/permission-task.test.ts +319 -0
- package/test/plugin/auth-override.test.ts +44 -0
- package/test/plugin/codex.test.ts +123 -0
- package/test/preload.ts +63 -0
- package/test/project/project.test.ts +120 -0
- package/test/provider/amazon-bedrock.test.ts +445 -0
- package/test/provider/copilot/convert-to-copilot-messages.test.ts +523 -0
- package/test/provider/copilot/copilot-chat-model.test.ts +592 -0
- package/test/provider/gitlab-duo.test.ts +262 -0
- package/test/provider/provider.test.ts +2129 -0
- package/test/provider/transform.test.ts +2022 -0
- package/test/question/question.test.ts +300 -0
- package/test/scheduler.test.ts +73 -0
- package/test/server/session-list.test.ts +39 -0
- package/test/server/session-select.test.ts +78 -0
- package/test/session/compaction.test.ts +293 -0
- package/test/session/instruction.test.ts +170 -0
- package/test/session/llm.test.ts +691 -0
- package/test/session/message-v2.test.ts +786 -0
- package/test/session/prompt-missing-file.test.ts +53 -0
- package/test/session/prompt-special-chars.test.ts +56 -0
- package/test/session/prompt-variant.test.ts +60 -0
- package/test/session/retry.test.ts +179 -0
- package/test/session/revert-compact.test.ts +285 -0
- package/test/session/session.test.ts +71 -0
- package/test/skill/discovery.test.ts +60 -0
- package/test/skill/skill.test.ts +388 -0
- package/test/snapshot/snapshot.test.ts +1040 -0
- package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
- package/test/tool/apply_patch.test.ts +559 -0
- package/test/tool/bash.test.ts +399 -0
- package/test/tool/external-directory.test.ts +127 -0
- package/test/tool/fixtures/large-image.png +0 -0
- package/test/tool/fixtures/models-api.json +38413 -0
- package/test/tool/grep.test.ts +110 -0
- package/test/tool/question.test.ts +107 -0
- package/test/tool/read.test.ts +358 -0
- package/test/tool/registry.test.ts +122 -0
- package/test/tool/skill.test.ts +112 -0
- package/test/tool/truncation.test.ts +159 -0
- package/test/util/filesystem.test.ts +39 -0
- package/test/util/format.test.ts +59 -0
- package/test/util/iife.test.ts +36 -0
- package/test/util/lazy.test.ts +50 -0
- package/test/util/lock.test.ts +72 -0
- package/test/util/timeout.test.ts +21 -0
- package/test/util/wildcard.test.ts +75 -0
- package/tsconfig.json +16 -0
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
You are a title generator. You output ONLY a thread title. Nothing else.
|
|
2
|
+
|
|
3
|
+
<task>
|
|
4
|
+
Generate a brief title that would help the user find this conversation later.
|
|
5
|
+
|
|
6
|
+
Follow all rules in <rules>
|
|
7
|
+
Use the <examples> so you know what a good title looks like.
|
|
8
|
+
Your output must be:
|
|
9
|
+
- A single line
|
|
10
|
+
- ≤50 characters
|
|
11
|
+
- No explanations
|
|
12
|
+
</task>
|
|
13
|
+
|
|
14
|
+
<rules>
|
|
15
|
+
- you MUST use the same language as the user message you are summarizing
|
|
16
|
+
- Title must be grammatically correct and read naturally - no word salad
|
|
17
|
+
- Never include tool names in the title (e.g. "read tool", "bash tool", "edit tool")
|
|
18
|
+
- Focus on the main topic or question the user needs to retrieve
|
|
19
|
+
- Vary your phrasing - avoid repetitive patterns like always starting with "Analyzing"
|
|
20
|
+
- When a file is mentioned, focus on WHAT the user wants to do WITH the file, not just that they shared it
|
|
21
|
+
- Keep exact: technical terms, numbers, filenames, HTTP codes
|
|
22
|
+
- Remove: the, this, my, a, an
|
|
23
|
+
- Never assume tech stack
|
|
24
|
+
- Never use tools
|
|
25
|
+
- NEVER respond to questions, just generate a title for the conversation
|
|
26
|
+
- The title should NEVER include "summarizing" or "generating" when generating a title
|
|
27
|
+
- DO NOT SAY YOU CANNOT GENERATE A TITLE OR COMPLAIN ABOUT THE INPUT
|
|
28
|
+
- Always output something meaningful, even if the input is minimal.
|
|
29
|
+
- If the user message is short or conversational (e.g. "hello", "lol", "what's up", "hey"):
|
|
30
|
+
→ create a title that reflects the user's tone or intent (such as Greeting, Quick check-in, Light chat, Intro message, etc.)
|
|
31
|
+
</rules>
|
|
32
|
+
|
|
33
|
+
<examples>
|
|
34
|
+
"debug 500 errors in production" → Debugging production 500 errors
|
|
35
|
+
"refactor user service" → Refactoring user service
|
|
36
|
+
"why is app.js failing" → app.js failure investigation
|
|
37
|
+
"implement rate limiting" → Rate limiting implementation
|
|
38
|
+
"how do I connect postgres to my API" → Postgres API connection
|
|
39
|
+
"best practices for React hooks" → React hooks best practices
|
|
40
|
+
"@src/auth.ts can you add refresh token support" → Auth refresh token support
|
|
41
|
+
"@utils/parser.ts this is broken" → Parser bug fix
|
|
42
|
+
"look at @config.json" → Config review
|
|
43
|
+
"@App.tsx add dark mode toggle" → Dark mode toggle in App
|
|
44
|
+
</examples>
|
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
import path from "path"
|
|
2
|
+
import { Global } from "../global"
|
|
3
|
+
import z from "zod"
|
|
4
|
+
|
|
5
|
+
export const OAUTH_DUMMY_KEY = "opencode-oauth-dummy-key"
|
|
6
|
+
|
|
7
|
+
export namespace Auth {
|
|
8
|
+
export const Oauth = z
|
|
9
|
+
.object({
|
|
10
|
+
type: z.literal("oauth"),
|
|
11
|
+
refresh: z.string(),
|
|
12
|
+
access: z.string(),
|
|
13
|
+
expires: z.number(),
|
|
14
|
+
accountId: z.string().optional(),
|
|
15
|
+
enterpriseUrl: z.string().optional(),
|
|
16
|
+
})
|
|
17
|
+
.meta({ ref: "OAuth" })
|
|
18
|
+
|
|
19
|
+
export const Api = z
|
|
20
|
+
.object({
|
|
21
|
+
type: z.literal("api"),
|
|
22
|
+
key: z.string(),
|
|
23
|
+
})
|
|
24
|
+
.meta({ ref: "ApiAuth" })
|
|
25
|
+
|
|
26
|
+
export const WellKnown = z
|
|
27
|
+
.object({
|
|
28
|
+
type: z.literal("wellknown"),
|
|
29
|
+
key: z.string(),
|
|
30
|
+
token: z.string(),
|
|
31
|
+
})
|
|
32
|
+
.meta({ ref: "WellKnownAuth" })
|
|
33
|
+
|
|
34
|
+
export const Info = z.discriminatedUnion("type", [Oauth, Api, WellKnown]).meta({ ref: "Auth" })
|
|
35
|
+
export type Info = z.infer<typeof Info>
|
|
36
|
+
|
|
37
|
+
const filepath = path.join(Global.Path.data, "auth.json")
|
|
38
|
+
|
|
39
|
+
export async function get(providerID: string) {
|
|
40
|
+
const auth = await all()
|
|
41
|
+
return auth[providerID]
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
export async function all(): Promise<Record<string, Info>> {
|
|
45
|
+
const file = Bun.file(filepath)
|
|
46
|
+
const data = await file.json().catch(() => ({}) as Record<string, unknown>)
|
|
47
|
+
return Object.entries(data).reduce(
|
|
48
|
+
(acc, [key, value]) => {
|
|
49
|
+
const parsed = Info.safeParse(value)
|
|
50
|
+
if (!parsed.success) return acc
|
|
51
|
+
acc[key] = parsed.data
|
|
52
|
+
return acc
|
|
53
|
+
},
|
|
54
|
+
{} as Record<string, Info>,
|
|
55
|
+
)
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
export async function set(key: string, info: Info) {
|
|
59
|
+
const file = Bun.file(filepath)
|
|
60
|
+
const data = await all()
|
|
61
|
+
await Bun.write(file, JSON.stringify({ ...data, [key]: info }, null, 2), { mode: 0o600 })
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
export async function remove(key: string) {
|
|
65
|
+
const file = Bun.file(filepath)
|
|
66
|
+
const data = await all()
|
|
67
|
+
delete data[key]
|
|
68
|
+
await Bun.write(file, JSON.stringify(data, null, 2), { mode: 0o600 })
|
|
69
|
+
}
|
|
70
|
+
}
|
package/src/bun/index.ts
ADDED
|
@@ -0,0 +1,137 @@
|
|
|
1
|
+
import z from "zod"
|
|
2
|
+
import { Global } from "../global"
|
|
3
|
+
import { Log } from "../util/log"
|
|
4
|
+
import path from "path"
|
|
5
|
+
import { Filesystem } from "../util/filesystem"
|
|
6
|
+
import { NamedError } from "@opencode-ai/util/error"
|
|
7
|
+
import { readableStreamToText } from "bun"
|
|
8
|
+
import { Lock } from "../util/lock"
|
|
9
|
+
import { PackageRegistry } from "./registry"
|
|
10
|
+
import { proxied } from "@/util/proxied"
|
|
11
|
+
|
|
12
|
+
export namespace BunProc {
|
|
13
|
+
const log = Log.create({ service: "bun" })
|
|
14
|
+
|
|
15
|
+
export async function run(cmd: string[], options?: Bun.SpawnOptions.OptionsObject<any, any, any>) {
|
|
16
|
+
log.info("running", {
|
|
17
|
+
cmd: [which(), ...cmd],
|
|
18
|
+
...options,
|
|
19
|
+
})
|
|
20
|
+
const result = Bun.spawn([which(), ...cmd], {
|
|
21
|
+
...options,
|
|
22
|
+
stdout: "pipe",
|
|
23
|
+
stderr: "pipe",
|
|
24
|
+
env: {
|
|
25
|
+
...process.env,
|
|
26
|
+
...options?.env,
|
|
27
|
+
BUN_BE_BUN: "1",
|
|
28
|
+
},
|
|
29
|
+
})
|
|
30
|
+
const code = await result.exited
|
|
31
|
+
const stdout = result.stdout
|
|
32
|
+
? typeof result.stdout === "number"
|
|
33
|
+
? result.stdout
|
|
34
|
+
: await readableStreamToText(result.stdout)
|
|
35
|
+
: undefined
|
|
36
|
+
const stderr = result.stderr
|
|
37
|
+
? typeof result.stderr === "number"
|
|
38
|
+
? result.stderr
|
|
39
|
+
: await readableStreamToText(result.stderr)
|
|
40
|
+
: undefined
|
|
41
|
+
log.info("done", {
|
|
42
|
+
code,
|
|
43
|
+
stdout,
|
|
44
|
+
stderr,
|
|
45
|
+
})
|
|
46
|
+
if (code !== 0) {
|
|
47
|
+
throw new Error(`Command failed with exit code ${result.exitCode}`)
|
|
48
|
+
}
|
|
49
|
+
return result
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
export function which() {
|
|
53
|
+
return process.execPath
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
export const InstallFailedError = NamedError.create(
|
|
57
|
+
"BunInstallFailedError",
|
|
58
|
+
z.object({
|
|
59
|
+
pkg: z.string(),
|
|
60
|
+
version: z.string(),
|
|
61
|
+
}),
|
|
62
|
+
)
|
|
63
|
+
|
|
64
|
+
export async function install(pkg: string, version = "latest") {
|
|
65
|
+
// Use lock to ensure only one install at a time
|
|
66
|
+
using _ = await Lock.write("bun-install")
|
|
67
|
+
|
|
68
|
+
const mod = path.join(Global.Path.cache, "node_modules", pkg)
|
|
69
|
+
const pkgjson = Bun.file(path.join(Global.Path.cache, "package.json"))
|
|
70
|
+
const parsed = await pkgjson.json().catch(async () => {
|
|
71
|
+
const result = { dependencies: {} }
|
|
72
|
+
await Bun.write(pkgjson.name!, JSON.stringify(result, null, 2))
|
|
73
|
+
return result
|
|
74
|
+
})
|
|
75
|
+
const dependencies = parsed.dependencies ?? {}
|
|
76
|
+
if (!parsed.dependencies) parsed.dependencies = dependencies
|
|
77
|
+
const modExists = await Filesystem.exists(mod)
|
|
78
|
+
const cachedVersion = dependencies[pkg]
|
|
79
|
+
|
|
80
|
+
if (!modExists || !cachedVersion) {
|
|
81
|
+
// continue to install
|
|
82
|
+
} else if (version !== "latest" && cachedVersion === version) {
|
|
83
|
+
return mod
|
|
84
|
+
} else if (version === "latest") {
|
|
85
|
+
const isOutdated = await PackageRegistry.isOutdated(pkg, cachedVersion, Global.Path.cache)
|
|
86
|
+
if (!isOutdated) return mod
|
|
87
|
+
log.info("Cached version is outdated, proceeding with install", { pkg, cachedVersion })
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
// Build command arguments
|
|
91
|
+
const args = [
|
|
92
|
+
"add",
|
|
93
|
+
"--force",
|
|
94
|
+
"--exact",
|
|
95
|
+
// TODO: get rid of this case (see: https://github.com/oven-sh/bun/issues/19936)
|
|
96
|
+
...(proxied() ? ["--no-cache"] : []),
|
|
97
|
+
"--cwd",
|
|
98
|
+
Global.Path.cache,
|
|
99
|
+
pkg + "@" + version,
|
|
100
|
+
]
|
|
101
|
+
|
|
102
|
+
// Let Bun handle registry resolution:
|
|
103
|
+
// - If .npmrc files exist, Bun will use them automatically
|
|
104
|
+
// - If no .npmrc files exist, Bun will default to https://registry.npmjs.org
|
|
105
|
+
// - No need to pass --registry flag
|
|
106
|
+
log.info("installing package using Bun's default registry resolution", {
|
|
107
|
+
pkg,
|
|
108
|
+
version,
|
|
109
|
+
})
|
|
110
|
+
|
|
111
|
+
await BunProc.run(args, {
|
|
112
|
+
cwd: Global.Path.cache,
|
|
113
|
+
}).catch((e) => {
|
|
114
|
+
throw new InstallFailedError(
|
|
115
|
+
{ pkg, version },
|
|
116
|
+
{
|
|
117
|
+
cause: e,
|
|
118
|
+
},
|
|
119
|
+
)
|
|
120
|
+
})
|
|
121
|
+
|
|
122
|
+
// Resolve actual version from installed package when using "latest"
|
|
123
|
+
// This ensures subsequent starts use the cached version until explicitly updated
|
|
124
|
+
let resolvedVersion = version
|
|
125
|
+
if (version === "latest") {
|
|
126
|
+
const installedPkgJson = Bun.file(path.join(mod, "package.json"))
|
|
127
|
+
const installedPkg = await installedPkgJson.json().catch(() => null)
|
|
128
|
+
if (installedPkg?.version) {
|
|
129
|
+
resolvedVersion = installedPkg.version
|
|
130
|
+
}
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
parsed.dependencies[pkg] = resolvedVersion
|
|
134
|
+
await Bun.write(pkgjson.name!, JSON.stringify(parsed, null, 2))
|
|
135
|
+
return mod
|
|
136
|
+
}
|
|
137
|
+
}
|
|
@@ -0,0 +1,48 @@
|
|
|
1
|
+
import { readableStreamToText, semver } from "bun"
|
|
2
|
+
import { Log } from "../util/log"
|
|
3
|
+
|
|
4
|
+
export namespace PackageRegistry {
|
|
5
|
+
const log = Log.create({ service: "bun" })
|
|
6
|
+
|
|
7
|
+
function which() {
|
|
8
|
+
return process.execPath
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
export async function info(pkg: string, field: string, cwd?: string): Promise<string | null> {
|
|
12
|
+
const result = Bun.spawn([which(), "info", pkg, field], {
|
|
13
|
+
cwd,
|
|
14
|
+
stdout: "pipe",
|
|
15
|
+
stderr: "pipe",
|
|
16
|
+
env: {
|
|
17
|
+
...process.env,
|
|
18
|
+
BUN_BE_BUN: "1",
|
|
19
|
+
},
|
|
20
|
+
})
|
|
21
|
+
|
|
22
|
+
const code = await result.exited
|
|
23
|
+
const stdout = result.stdout ? await readableStreamToText(result.stdout) : ""
|
|
24
|
+
const stderr = result.stderr ? await readableStreamToText(result.stderr) : ""
|
|
25
|
+
|
|
26
|
+
if (code !== 0) {
|
|
27
|
+
log.warn("bun info failed", { pkg, field, code, stderr })
|
|
28
|
+
return null
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
const value = stdout.trim()
|
|
32
|
+
if (!value) return null
|
|
33
|
+
return value
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
export async function isOutdated(pkg: string, cachedVersion: string, cwd?: string): Promise<boolean> {
|
|
37
|
+
const latestVersion = await info(pkg, "version", cwd)
|
|
38
|
+
if (!latestVersion) {
|
|
39
|
+
log.warn("Failed to resolve latest version, using cached", { pkg, cachedVersion })
|
|
40
|
+
return false
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
const isRange = /[\s^~*xX<>|=]/.test(cachedVersion)
|
|
44
|
+
if (isRange) return !semver.satisfies(latestVersion, cachedVersion)
|
|
45
|
+
|
|
46
|
+
return semver.order(cachedVersion, latestVersion) === -1
|
|
47
|
+
}
|
|
48
|
+
}
|
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
import z from "zod"
|
|
2
|
+
import type { ZodType } from "zod"
|
|
3
|
+
import { Log } from "../util/log"
|
|
4
|
+
|
|
5
|
+
export namespace BusEvent {
|
|
6
|
+
const log = Log.create({ service: "event" })
|
|
7
|
+
|
|
8
|
+
export type Definition = ReturnType<typeof define>
|
|
9
|
+
|
|
10
|
+
const registry = new Map<string, Definition>()
|
|
11
|
+
|
|
12
|
+
export function define<Type extends string, Properties extends ZodType>(type: Type, properties: Properties) {
|
|
13
|
+
const result = {
|
|
14
|
+
type,
|
|
15
|
+
properties,
|
|
16
|
+
}
|
|
17
|
+
registry.set(type, result)
|
|
18
|
+
return result
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
export function payloads() {
|
|
22
|
+
return z
|
|
23
|
+
.discriminatedUnion(
|
|
24
|
+
"type",
|
|
25
|
+
registry
|
|
26
|
+
.entries()
|
|
27
|
+
.map(([type, def]) => {
|
|
28
|
+
return z
|
|
29
|
+
.object({
|
|
30
|
+
type: z.literal(type),
|
|
31
|
+
properties: def.properties,
|
|
32
|
+
})
|
|
33
|
+
.meta({
|
|
34
|
+
ref: "Event" + "." + def.type,
|
|
35
|
+
})
|
|
36
|
+
})
|
|
37
|
+
.toArray() as any,
|
|
38
|
+
)
|
|
39
|
+
.meta({
|
|
40
|
+
ref: "Event",
|
|
41
|
+
})
|
|
42
|
+
}
|
|
43
|
+
}
|
package/src/bus/index.ts
ADDED
|
@@ -0,0 +1,105 @@
|
|
|
1
|
+
import z from "zod"
|
|
2
|
+
import { Log } from "../util/log"
|
|
3
|
+
import { Instance } from "../project/instance"
|
|
4
|
+
import { BusEvent } from "./bus-event"
|
|
5
|
+
import { GlobalBus } from "./global"
|
|
6
|
+
|
|
7
|
+
export namespace Bus {
|
|
8
|
+
const log = Log.create({ service: "bus" })
|
|
9
|
+
type Subscription = (event: any) => void
|
|
10
|
+
|
|
11
|
+
export const InstanceDisposed = BusEvent.define(
|
|
12
|
+
"server.instance.disposed",
|
|
13
|
+
z.object({
|
|
14
|
+
directory: z.string(),
|
|
15
|
+
}),
|
|
16
|
+
)
|
|
17
|
+
|
|
18
|
+
const state = Instance.state(
|
|
19
|
+
() => {
|
|
20
|
+
const subscriptions = new Map<any, Subscription[]>()
|
|
21
|
+
|
|
22
|
+
return {
|
|
23
|
+
subscriptions,
|
|
24
|
+
}
|
|
25
|
+
},
|
|
26
|
+
async (entry) => {
|
|
27
|
+
const wildcard = entry.subscriptions.get("*")
|
|
28
|
+
if (!wildcard) return
|
|
29
|
+
const event = {
|
|
30
|
+
type: InstanceDisposed.type,
|
|
31
|
+
properties: {
|
|
32
|
+
directory: Instance.directory,
|
|
33
|
+
},
|
|
34
|
+
}
|
|
35
|
+
for (const sub of [...wildcard]) {
|
|
36
|
+
sub(event)
|
|
37
|
+
}
|
|
38
|
+
},
|
|
39
|
+
)
|
|
40
|
+
|
|
41
|
+
export async function publish<Definition extends BusEvent.Definition>(
|
|
42
|
+
def: Definition,
|
|
43
|
+
properties: z.output<Definition["properties"]>,
|
|
44
|
+
) {
|
|
45
|
+
const payload = {
|
|
46
|
+
type: def.type,
|
|
47
|
+
properties,
|
|
48
|
+
}
|
|
49
|
+
log.info("publishing", {
|
|
50
|
+
type: def.type,
|
|
51
|
+
})
|
|
52
|
+
const pending = []
|
|
53
|
+
for (const key of [def.type, "*"]) {
|
|
54
|
+
const match = state().subscriptions.get(key)
|
|
55
|
+
for (const sub of match ?? []) {
|
|
56
|
+
pending.push(sub(payload))
|
|
57
|
+
}
|
|
58
|
+
}
|
|
59
|
+
GlobalBus.emit("event", {
|
|
60
|
+
directory: Instance.directory,
|
|
61
|
+
payload,
|
|
62
|
+
})
|
|
63
|
+
return Promise.all(pending)
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
export function subscribe<Definition extends BusEvent.Definition>(
|
|
67
|
+
def: Definition,
|
|
68
|
+
callback: (event: { type: Definition["type"]; properties: z.infer<Definition["properties"]> }) => void,
|
|
69
|
+
) {
|
|
70
|
+
return raw(def.type, callback)
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
export function once<Definition extends BusEvent.Definition>(
|
|
74
|
+
def: Definition,
|
|
75
|
+
callback: (event: {
|
|
76
|
+
type: Definition["type"]
|
|
77
|
+
properties: z.infer<Definition["properties"]>
|
|
78
|
+
}) => "done" | undefined,
|
|
79
|
+
) {
|
|
80
|
+
const unsub = subscribe(def, (event) => {
|
|
81
|
+
if (callback(event)) unsub()
|
|
82
|
+
})
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
export function subscribeAll(callback: (event: any) => void) {
|
|
86
|
+
return raw("*", callback)
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
function raw(type: string, callback: (event: any) => void) {
|
|
90
|
+
log.info("subscribing", { type })
|
|
91
|
+
const subscriptions = state().subscriptions
|
|
92
|
+
let match = subscriptions.get(type) ?? []
|
|
93
|
+
match.push(callback)
|
|
94
|
+
subscriptions.set(type, match)
|
|
95
|
+
|
|
96
|
+
return () => {
|
|
97
|
+
log.info("unsubscribing", { type })
|
|
98
|
+
const match = subscriptions.get(type)
|
|
99
|
+
if (!match) return
|
|
100
|
+
const index = match.indexOf(callback)
|
|
101
|
+
if (index === -1) return
|
|
102
|
+
match.splice(index, 1)
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
}
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
import { InstanceBootstrap } from "../project/bootstrap"
|
|
2
|
+
import { Instance } from "../project/instance"
|
|
3
|
+
|
|
4
|
+
export async function bootstrap<T>(directory: string, cb: () => Promise<T>) {
|
|
5
|
+
return Instance.provide({
|
|
6
|
+
directory,
|
|
7
|
+
init: InstanceBootstrap,
|
|
8
|
+
fn: async () => {
|
|
9
|
+
try {
|
|
10
|
+
const result = await cb()
|
|
11
|
+
return result
|
|
12
|
+
} finally {
|
|
13
|
+
await Instance.dispose()
|
|
14
|
+
}
|
|
15
|
+
},
|
|
16
|
+
})
|
|
17
|
+
}
|
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
import { Log } from "@/util/log"
|
|
2
|
+
import { bootstrap } from "../bootstrap"
|
|
3
|
+
import { cmd } from "./cmd"
|
|
4
|
+
import { AgentSideConnection, ndJsonStream } from "@agentclientprotocol/sdk"
|
|
5
|
+
import { ACP } from "@/acp/agent"
|
|
6
|
+
import { Server } from "@/server/server"
|
|
7
|
+
import { createOpencodeClient } from "@opencode-ai/sdk/v2"
|
|
8
|
+
import { withNetworkOptions, resolveNetworkOptions } from "../network"
|
|
9
|
+
|
|
10
|
+
const log = Log.create({ service: "acp-command" })
|
|
11
|
+
|
|
12
|
+
export const AcpCommand = cmd({
|
|
13
|
+
command: "acp",
|
|
14
|
+
describe: "start ACP (Agent Client Protocol) server",
|
|
15
|
+
builder: (yargs) => {
|
|
16
|
+
return withNetworkOptions(yargs).option("cwd", {
|
|
17
|
+
describe: "working directory",
|
|
18
|
+
type: "string",
|
|
19
|
+
default: process.cwd(),
|
|
20
|
+
})
|
|
21
|
+
},
|
|
22
|
+
handler: async (args) => {
|
|
23
|
+
process.env.OPENCODE_CLIENT = "acp"
|
|
24
|
+
await bootstrap(process.cwd(), async () => {
|
|
25
|
+
const opts = await resolveNetworkOptions(args)
|
|
26
|
+
const server = Server.listen(opts)
|
|
27
|
+
|
|
28
|
+
const sdk = createOpencodeClient({
|
|
29
|
+
baseUrl: `http://${server.hostname}:${server.port}`,
|
|
30
|
+
})
|
|
31
|
+
|
|
32
|
+
const input = new WritableStream<Uint8Array>({
|
|
33
|
+
write(chunk) {
|
|
34
|
+
return new Promise<void>((resolve, reject) => {
|
|
35
|
+
process.stdout.write(chunk, (err) => {
|
|
36
|
+
if (err) {
|
|
37
|
+
reject(err)
|
|
38
|
+
} else {
|
|
39
|
+
resolve()
|
|
40
|
+
}
|
|
41
|
+
})
|
|
42
|
+
})
|
|
43
|
+
},
|
|
44
|
+
})
|
|
45
|
+
const output = new ReadableStream<Uint8Array>({
|
|
46
|
+
start(controller) {
|
|
47
|
+
process.stdin.on("data", (chunk: Buffer) => {
|
|
48
|
+
controller.enqueue(new Uint8Array(chunk))
|
|
49
|
+
})
|
|
50
|
+
process.stdin.on("end", () => controller.close())
|
|
51
|
+
process.stdin.on("error", (err) => controller.error(err))
|
|
52
|
+
},
|
|
53
|
+
})
|
|
54
|
+
|
|
55
|
+
const stream = ndJsonStream(input, output)
|
|
56
|
+
const agent = await ACP.init({ sdk })
|
|
57
|
+
|
|
58
|
+
new AgentSideConnection((conn) => {
|
|
59
|
+
return agent.create(conn, { sdk })
|
|
60
|
+
}, stream)
|
|
61
|
+
|
|
62
|
+
log.info("setup connection")
|
|
63
|
+
process.stdin.resume()
|
|
64
|
+
await new Promise((resolve, reject) => {
|
|
65
|
+
process.stdin.on("end", resolve)
|
|
66
|
+
process.stdin.on("error", reject)
|
|
67
|
+
})
|
|
68
|
+
})
|
|
69
|
+
},
|
|
70
|
+
})
|