Repository: opencode-ai/opencode Branch: main Commit: 73ee493265ac Files: 162 Total size: 1.3 MB Directory structure: gitextract_5ee_h7l1/ ├── .github/ │ └── workflows/ │ ├── build.yml │ └── release.yml ├── .gitignore ├── .goreleaser.yml ├── .opencode.json ├── LICENSE ├── README.md ├── cmd/ │ ├── root.go │ └── schema/ │ ├── README.md │ └── main.go ├── go.mod ├── go.sum ├── install ├── internal/ │ ├── app/ │ │ ├── app.go │ │ └── lsp.go │ ├── completions/ │ │ └── files-folders.go │ ├── config/ │ │ ├── config.go │ │ └── init.go │ ├── db/ │ │ ├── connect.go │ │ ├── db.go │ │ ├── embed.go │ │ ├── files.sql.go │ │ ├── messages.sql.go │ │ ├── migrations/ │ │ │ ├── 20250424200609_initial.sql │ │ │ └── 20250515105448_add_summary_message_id.sql │ │ ├── models.go │ │ ├── querier.go │ │ ├── sessions.sql.go │ │ └── sql/ │ │ ├── files.sql │ │ ├── messages.sql │ │ └── sessions.sql │ ├── diff/ │ │ ├── diff.go │ │ └── patch.go │ ├── fileutil/ │ │ └── fileutil.go │ ├── format/ │ │ ├── format.go │ │ └── spinner.go │ ├── history/ │ │ └── file.go │ ├── llm/ │ │ ├── agent/ │ │ │ ├── agent-tool.go │ │ │ ├── agent.go │ │ │ ├── mcp-tools.go │ │ │ └── tools.go │ │ ├── models/ │ │ │ ├── anthropic.go │ │ │ ├── azure.go │ │ │ ├── copilot.go │ │ │ ├── gemini.go │ │ │ ├── groq.go │ │ │ ├── local.go │ │ │ ├── models.go │ │ │ ├── openai.go │ │ │ ├── openrouter.go │ │ │ ├── vertexai.go │ │ │ └── xai.go │ │ ├── prompt/ │ │ │ ├── coder.go │ │ │ ├── prompt.go │ │ │ ├── prompt_test.go │ │ │ ├── summarizer.go │ │ │ ├── task.go │ │ │ └── title.go │ │ ├── provider/ │ │ │ ├── anthropic.go │ │ │ ├── azure.go │ │ │ ├── bedrock.go │ │ │ ├── copilot.go │ │ │ ├── gemini.go │ │ │ ├── openai.go │ │ │ ├── provider.go │ │ │ └── vertexai.go │ │ └── tools/ │ │ ├── bash.go │ │ ├── diagnostics.go │ │ ├── edit.go │ │ ├── fetch.go │ │ ├── file.go │ │ ├── glob.go │ │ ├── grep.go │ │ ├── ls.go │ │ ├── ls_test.go │ │ ├── patch.go │ │ ├── shell/ │ │ │ └── shell.go │ │ ├── sourcegraph.go │ │ ├── tools.go │ │ ├── view.go │ │ └── write.go │ ├── logging/ │ │ ├── logger.go │ │ ├── message.go │ │ └── writer.go │ ├── lsp/ │ │ ├── client.go │ │ ├── handlers.go │ │ ├── language.go │ │ ├── methods.go │ │ ├── protocol/ │ │ │ ├── LICENSE │ │ │ ├── interface.go │ │ │ ├── pattern_interfaces.go │ │ │ ├── tables.go │ │ │ ├── tsdocument-changes.go │ │ │ ├── tsjson.go │ │ │ ├── tsprotocol.go │ │ │ └── uri.go │ │ ├── protocol.go │ │ ├── transport.go │ │ ├── util/ │ │ │ └── edit.go │ │ └── watcher/ │ │ └── watcher.go │ ├── message/ │ │ ├── attachment.go │ │ ├── content.go │ │ └── message.go │ ├── permission/ │ │ └── permission.go │ ├── pubsub/ │ │ ├── broker.go │ │ └── events.go │ ├── session/ │ │ └── session.go │ ├── tui/ │ │ ├── components/ │ │ │ ├── chat/ │ │ │ │ ├── chat.go │ │ │ │ ├── editor.go │ │ │ │ ├── list.go │ │ │ │ ├── message.go │ │ │ │ └── sidebar.go │ │ │ ├── core/ │ │ │ │ └── status.go │ │ │ ├── dialog/ │ │ │ │ ├── arguments.go │ │ │ │ ├── commands.go │ │ │ │ ├── complete.go │ │ │ │ ├── custom_commands.go │ │ │ │ ├── custom_commands_test.go │ │ │ │ ├── filepicker.go │ │ │ │ ├── help.go │ │ │ │ ├── init.go │ │ │ │ ├── models.go │ │ │ │ ├── permission.go │ │ │ │ ├── quit.go │ │ │ │ ├── session.go │ │ │ │ └── theme.go │ │ │ ├── logs/ │ │ │ │ ├── details.go │ │ │ │ └── table.go │ │ │ └── util/ │ │ │ └── simple-list.go │ │ ├── image/ │ │ │ └── images.go │ │ ├── layout/ │ │ │ ├── container.go │ │ │ ├── layout.go │ │ │ ├── overlay.go │ │ │ └── split.go │ │ ├── page/ │ │ │ ├── chat.go │ │ │ ├── logs.go │ │ │ └── page.go │ │ ├── styles/ │ │ │ ├── background.go │ │ │ ├── icons.go │ │ │ ├── markdown.go │ │ │ └── styles.go │ │ ├── theme/ │ │ │ ├── catppuccin.go │ │ │ ├── dracula.go │ │ │ ├── flexoki.go │ │ │ ├── gruvbox.go │ │ │ ├── manager.go │ │ │ ├── monokai.go │ │ │ ├── onedark.go │ │ │ ├── opencode.go │ │ │ ├── theme.go │ │ │ ├── theme_test.go │ │ │ ├── tokyonight.go │ │ │ └── tron.go │ │ ├── tui.go │ │ └── util/ │ │ └── util.go │ └── version/ │ └── version.go ├── main.go ├── opencode-schema.json ├── scripts/ │ ├── check_hidden_chars.sh │ ├── release │ └── snapshot └── sqlc.yaml ================================================ FILE CONTENTS ================================================ ================================================ FILE: .github/workflows/build.yml ================================================ name: build on: workflow_dispatch: push: branches: - main concurrency: ${{ github.workflow }}-${{ github.ref }} permissions: contents: write packages: write jobs: build: runs-on: ubuntu-latest steps: - uses: actions/checkout@v3 with: fetch-depth: 0 - run: git fetch --force --tags - uses: actions/setup-go@v5 with: go-version: ">=1.23.2" cache: true cache-dependency-path: go.sum - run: go mod download - uses: goreleaser/goreleaser-action@v6 with: distribution: goreleaser version: latest args: build --snapshot --clean ================================================ FILE: .github/workflows/release.yml ================================================ name: release on: workflow_dispatch: push: tags: - "*" concurrency: ${{ github.workflow }}-${{ github.ref }} permissions: contents: write packages: write jobs: goreleaser: runs-on: ubuntu-latest steps: - uses: actions/checkout@v3 with: fetch-depth: 0 - run: git fetch --force --tags - uses: actions/setup-go@v5 with: go-version: ">=1.23.2" cache: true cache-dependency-path: go.sum - run: go mod download - uses: goreleaser/goreleaser-action@v6 with: distribution: goreleaser version: latest args: release --clean env: GITHUB_TOKEN: ${{ secrets.HOMEBREW_GITHUB_TOKEN }} AUR_KEY: ${{ secrets.AUR_KEY }} ================================================ FILE: .gitignore ================================================ # Binaries for programs and plugins *.exe *.exe~ *.dll *.so *.dylib # Test binary, built with `go test -c` *.test # Output of the go coverage tool, specifically when used with LiteIDE *.out # Dependency directories (remove the comment below to include it) # vendor/ # Go workspace file go.work # IDE specific files .idea/ .vscode/ *.swp *.swo # OS specific files .DS_Store .DS_Store? ._* .Spotlight-V100 .Trashes ehthumbs.db Thumbs.db *.log # Binary output directory /bin/ /dist/ # Local environment variables .env .env.local .opencode/ opencode opencode.md ================================================ FILE: .goreleaser.yml ================================================ version: 2 project_name: opencode before: hooks: builds: - env: - CGO_ENABLED=0 goos: - linux - darwin goarch: - amd64 - arm64 ldflags: - -s -w -X github.com/opencode-ai/opencode/internal/version.Version={{.Version}} main: ./main.go archives: - format: tar.gz name_template: >- opencode- {{- if eq .Os "darwin" }}mac- {{- else if eq .Os "windows" }}windows- {{- else if eq .Os "linux" }}linux-{{end}} {{- if eq .Arch "amd64" }}x86_64 {{- else if eq .Arch "#86" }}i386 {{- else }}{{ .Arch }}{{ end }} {{- if .Arm }}v{{ .Arm }}{{ end }} format_overrides: - goos: windows format: zip checksum: name_template: "checksums.txt" snapshot: name_template: "0.0.0-{{ .Timestamp }}" aurs: - name: opencode-ai homepage: "https://github.com/opencode-ai/opencode" description: "terminal based agent that can build anything" maintainers: - "kujtimiihoxha " license: "MIT" private_key: "{{ .Env.AUR_KEY }}" git_url: "ssh://aur@aur.archlinux.org/opencode-ai-bin.git" provides: - opencode conflicts: - opencode package: |- install -Dm755 ./opencode "${pkgdir}/usr/bin/opencode" brews: - repository: owner: opencode-ai name: homebrew-tap nfpms: - maintainer: kujtimiihoxha description: terminal based agent that can build anything formats: - deb - rpm file_name_template: >- {{ .ProjectName }}- {{- if eq .Os "darwin" }}mac {{- else }}{{ .Os }}{{ end }}-{{ .Arch }} changelog: sort: asc filters: exclude: - "^docs:" - "^doc:" - "^test:" - "^ci:" - "^ignore:" - "^example:" - "^wip:" ================================================ FILE: .opencode.json ================================================ { "$schema": "./opencode-schema.json", "lsp": { "gopls": { "command": "gopls" } } } ================================================ FILE: LICENSE ================================================ MIT License Copyright (c) 2025 Kujtim Hoxha Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. ================================================ FILE: README.md ================================================ # Archived: Project has Moved This repository is no longer maintained and has been archived for provenance. The project has continued under the name [Crush][crush], developed by the original author and the Charm team. Please follow [Crush][crush] for ongoing development. [crush]: https://github.com/charmbracelet/crush # ⌬ OpenCode

> **⚠️ Early Development Notice:** This project is in early development and is not yet ready for production use. Features may change, break, or be incomplete. Use at your own risk. A powerful terminal-based AI assistant for developers, providing intelligent coding assistance directly in your terminal. ## Overview OpenCode is a Go-based CLI application that brings AI assistance to your terminal. It provides a TUI (Terminal User Interface) for interacting with various AI models to help with coding tasks, debugging, and more.

For a quick video overview, check out OpenCode + Gemini 2.5 Pro: BYE Claude Code! I'm SWITCHING To the FASTEST AI Coder!

## Features - **Interactive TUI**: Built with [Bubble Tea](https://github.com/charmbracelet/bubbletea) for a smooth terminal experience - **Multiple AI Providers**: Support for OpenAI, Anthropic Claude, Google Gemini, AWS Bedrock, Groq, Azure OpenAI, and OpenRouter - **Session Management**: Save and manage multiple conversation sessions - **Tool Integration**: AI can execute commands, search files, and modify code - **Vim-like Editor**: Integrated editor with text input capabilities - **Persistent Storage**: SQLite database for storing conversations and sessions - **LSP Integration**: Language Server Protocol support for code intelligence - **File Change Tracking**: Track and visualize file changes during sessions - **External Editor Support**: Open your preferred editor for composing messages - **Named Arguments for Custom Commands**: Create powerful custom commands with multiple named placeholders ## Installation ### Using the Install Script ```bash # Install the latest version curl -fsSL https://raw.githubusercontent.com/opencode-ai/opencode/refs/heads/main/install | bash # Install a specific version curl -fsSL https://raw.githubusercontent.com/opencode-ai/opencode/refs/heads/main/install | VERSION=0.1.0 bash ``` ### Using Homebrew (macOS and Linux) ```bash brew install opencode-ai/tap/opencode ``` ### Using AUR (Arch Linux) ```bash # Using yay yay -S opencode-ai-bin # Using paru paru -S opencode-ai-bin ``` ### Using Go ```bash go install github.com/opencode-ai/opencode@latest ``` ## Configuration OpenCode looks for configuration in the following locations: - `$HOME/.opencode.json` - `$XDG_CONFIG_HOME/opencode/.opencode.json` - `./.opencode.json` (local directory) ### Auto Compact Feature OpenCode includes an auto compact feature that automatically summarizes your conversation when it approaches the model's context window limit. When enabled (default setting), this feature: - Monitors token usage during your conversation - Automatically triggers summarization when usage reaches 95% of the model's context window - Creates a new session with the summary, allowing you to continue your work without losing context - Helps prevent "out of context" errors that can occur with long conversations You can enable or disable this feature in your configuration file: ```json { "autoCompact": true // default is true } ``` ### Environment Variables You can configure OpenCode using environment variables: | Environment Variable | Purpose | | -------------------------- | -------------------------------------------------------------------------------- | | `ANTHROPIC_API_KEY` | For Claude models | | `OPENAI_API_KEY` | For OpenAI models | | `GEMINI_API_KEY` | For Google Gemini models | | `GITHUB_TOKEN` | For Github Copilot models (see [Using Github Copilot](#using-github-copilot)) | | `VERTEXAI_PROJECT` | For Google Cloud VertexAI (Gemini) | | `VERTEXAI_LOCATION` | For Google Cloud VertexAI (Gemini) | | `GROQ_API_KEY` | For Groq models | | `AWS_ACCESS_KEY_ID` | For AWS Bedrock (Claude) | | `AWS_SECRET_ACCESS_KEY` | For AWS Bedrock (Claude) | | `AWS_REGION` | For AWS Bedrock (Claude) | | `AZURE_OPENAI_ENDPOINT` | For Azure OpenAI models | | `AZURE_OPENAI_API_KEY` | For Azure OpenAI models (optional when using Entra ID) | | `AZURE_OPENAI_API_VERSION` | For Azure OpenAI models | | `LOCAL_ENDPOINT` | For self-hosted models | | `SHELL` | Default shell to use (if not specified in config) | ### Shell Configuration OpenCode allows you to configure the shell used by the bash tool. By default, it uses the shell specified in the `SHELL` environment variable, or falls back to `/bin/bash` if not set. You can override this in your configuration file: ```json { "shell": { "path": "/bin/zsh", "args": ["-l"] } } ``` This is useful if you want to use a different shell than your default system shell, or if you need to pass specific arguments to the shell. ### Configuration File Structure ```json { "data": { "directory": ".opencode" }, "providers": { "openai": { "apiKey": "your-api-key", "disabled": false }, "anthropic": { "apiKey": "your-api-key", "disabled": false }, "copilot": { "disabled": false }, "groq": { "apiKey": "your-api-key", "disabled": false }, "openrouter": { "apiKey": "your-api-key", "disabled": false } }, "agents": { "coder": { "model": "claude-3.7-sonnet", "maxTokens": 5000 }, "task": { "model": "claude-3.7-sonnet", "maxTokens": 5000 }, "title": { "model": "claude-3.7-sonnet", "maxTokens": 80 } }, "shell": { "path": "/bin/bash", "args": ["-l"] }, "mcpServers": { "example": { "type": "stdio", "command": "path/to/mcp-server", "env": [], "args": [] } }, "lsp": { "go": { "disabled": false, "command": "gopls" } }, "debug": false, "debugLSP": false, "autoCompact": true } ``` ## Supported AI Models OpenCode supports a variety of AI models from different providers: ### OpenAI - GPT-4.1 family (gpt-4.1, gpt-4.1-mini, gpt-4.1-nano) - GPT-4.5 Preview - GPT-4o family (gpt-4o, gpt-4o-mini) - O1 family (o1, o1-pro, o1-mini) - O3 family (o3, o3-mini) - O4 Mini ### Anthropic - Claude 4 Sonnet - Claude 4 Opus - Claude 3.5 Sonnet - Claude 3.5 Haiku - Claude 3.7 Sonnet - Claude 3 Haiku - Claude 3 Opus ### GitHub Copilot - GPT-3.5 Turbo - GPT-4 - GPT-4o - GPT-4o Mini - GPT-4.1 - Claude 3.5 Sonnet - Claude 3.7 Sonnet - Claude 3.7 Sonnet Thinking - Claude Sonnet 4 - O1 - O3 Mini - O4 Mini - Gemini 2.0 Flash - Gemini 2.5 Pro ### Google - Gemini 2.5 - Gemini 2.5 Flash - Gemini 2.0 Flash - Gemini 2.0 Flash Lite ### AWS Bedrock - Claude 3.7 Sonnet ### Groq - Llama 4 Maverick (17b-128e-instruct) - Llama 4 Scout (17b-16e-instruct) - QWEN QWQ-32b - Deepseek R1 distill Llama 70b - Llama 3.3 70b Versatile ### Azure OpenAI - GPT-4.1 family (gpt-4.1, gpt-4.1-mini, gpt-4.1-nano) - GPT-4.5 Preview - GPT-4o family (gpt-4o, gpt-4o-mini) - O1 family (o1, o1-mini) - O3 family (o3, o3-mini) - O4 Mini ### Google Cloud VertexAI - Gemini 2.5 - Gemini 2.5 Flash ## Usage ```bash # Start OpenCode opencode # Start with debug logging opencode -d # Start with a specific working directory opencode -c /path/to/project ``` ## Non-interactive Prompt Mode You can run OpenCode in non-interactive mode by passing a prompt directly as a command-line argument. This is useful for scripting, automation, or when you want a quick answer without launching the full TUI. ```bash # Run a single prompt and print the AI's response to the terminal opencode -p "Explain the use of context in Go" # Get response in JSON format opencode -p "Explain the use of context in Go" -f json # Run without showing the spinner (useful for scripts) opencode -p "Explain the use of context in Go" -q ``` In this mode, OpenCode will process your prompt, print the result to standard output, and then exit. All permissions are auto-approved for the session. By default, a spinner animation is displayed while the model is processing your query. You can disable this spinner with the `-q` or `--quiet` flag, which is particularly useful when running OpenCode from scripts or automated workflows. ### Output Formats OpenCode supports the following output formats in non-interactive mode: | Format | Description | | ------ | ------------------------------- | | `text` | Plain text output (default) | | `json` | Output wrapped in a JSON object | The output format is implemented as a strongly-typed `OutputFormat` in the codebase, ensuring type safety and validation when processing outputs. ## Command-line Flags | Flag | Short | Description | | ----------------- | ----- | --------------------------------------------------- | | `--help` | `-h` | Display help information | | `--debug` | `-d` | Enable debug mode | | `--cwd` | `-c` | Set current working directory | | `--prompt` | `-p` | Run a single prompt in non-interactive mode | | `--output-format` | `-f` | Output format for non-interactive mode (text, json) | | `--quiet` | `-q` | Hide spinner in non-interactive mode | ## Keyboard Shortcuts ### Global Shortcuts | Shortcut | Action | | -------- | ------------------------------------------------------- | | `Ctrl+C` | Quit application | | `Ctrl+?` | Toggle help dialog | | `?` | Toggle help dialog (when not in editing mode) | | `Ctrl+L` | View logs | | `Ctrl+A` | Switch session | | `Ctrl+K` | Command dialog | | `Ctrl+O` | Toggle model selection dialog | | `Esc` | Close current overlay/dialog or return to previous mode | ### Chat Page Shortcuts | Shortcut | Action | | -------- | --------------------------------------- | | `Ctrl+N` | Create new session | | `Ctrl+X` | Cancel current operation/generation | | `i` | Focus editor (when not in writing mode) | | `Esc` | Exit writing mode and focus messages | ### Editor Shortcuts | Shortcut | Action | | ------------------- | ----------------------------------------- | | `Ctrl+S` | Send message (when editor is focused) | | `Enter` or `Ctrl+S` | Send message (when editor is not focused) | | `Ctrl+E` | Open external editor | | `Esc` | Blur editor and focus messages | ### Session Dialog Shortcuts | Shortcut | Action | | ---------- | ---------------- | | `↑` or `k` | Previous session | | `↓` or `j` | Next session | | `Enter` | Select session | | `Esc` | Close dialog | ### Model Dialog Shortcuts | Shortcut | Action | | ---------- | ----------------- | | `↑` or `k` | Move up | | `↓` or `j` | Move down | | `←` or `h` | Previous provider | | `→` or `l` | Next provider | | `Esc` | Close dialog | ### Permission Dialog Shortcuts | Shortcut | Action | | ----------------------- | ---------------------------- | | `←` or `left` | Switch options left | | `→` or `right` or `tab` | Switch options right | | `Enter` or `space` | Confirm selection | | `a` | Allow permission | | `A` | Allow permission for session | | `d` | Deny permission | ### Logs Page Shortcuts | Shortcut | Action | | ------------------ | ------------------- | | `Backspace` or `q` | Return to chat page | ## AI Assistant Tools OpenCode's AI assistant has access to various tools to help with coding tasks: ### File and Code Tools | Tool | Description | Parameters | | ------------- | --------------------------- | ---------------------------------------------------------------------------------------- | | `glob` | Find files by pattern | `pattern` (required), `path` (optional) | | `grep` | Search file contents | `pattern` (required), `path` (optional), `include` (optional), `literal_text` (optional) | | `ls` | List directory contents | `path` (optional), `ignore` (optional array of patterns) | | `view` | View file contents | `file_path` (required), `offset` (optional), `limit` (optional) | | `write` | Write to files | `file_path` (required), `content` (required) | | `edit` | Edit files | Various parameters for file editing | | `patch` | Apply patches to files | `file_path` (required), `diff` (required) | | `diagnostics` | Get diagnostics information | `file_path` (optional) | ### Other Tools | Tool | Description | Parameters | | ------------- | -------------------------------------- | ----------------------------------------------------------------------------------------- | | `bash` | Execute shell commands | `command` (required), `timeout` (optional) | | `fetch` | Fetch data from URLs | `url` (required), `format` (required), `timeout` (optional) | | `sourcegraph` | Search code across public repositories | `query` (required), `count` (optional), `context_window` (optional), `timeout` (optional) | | `agent` | Run sub-tasks with the AI agent | `prompt` (required) | ## Architecture OpenCode is built with a modular architecture: - **cmd**: Command-line interface using Cobra - **internal/app**: Core application services - **internal/config**: Configuration management - **internal/db**: Database operations and migrations - **internal/llm**: LLM providers and tools integration - **internal/tui**: Terminal UI components and layouts - **internal/logging**: Logging infrastructure - **internal/message**: Message handling - **internal/session**: Session management - **internal/lsp**: Language Server Protocol integration ## Custom Commands OpenCode supports custom commands that can be created by users to quickly send predefined prompts to the AI assistant. ### Creating Custom Commands Custom commands are predefined prompts stored as Markdown files in one of three locations: 1. **User Commands** (prefixed with `user:`): ``` $XDG_CONFIG_HOME/opencode/commands/ ``` (typically `~/.config/opencode/commands/` on Linux/macOS) or ``` $HOME/.opencode/commands/ ``` 2. **Project Commands** (prefixed with `project:`): ``` /.opencode/commands/ ``` Each `.md` file in these directories becomes a custom command. The file name (without extension) becomes the command ID. For example, creating a file at `~/.config/opencode/commands/prime-context.md` with content: ```markdown RUN git ls-files READ README.md ``` This creates a command called `user:prime-context`. ### Command Arguments OpenCode supports named arguments in custom commands using placeholders in the format `$NAME` (where NAME consists of uppercase letters, numbers, and underscores, and must start with a letter). For example: ```markdown # Fetch Context for Issue $ISSUE_NUMBER RUN gh issue view $ISSUE_NUMBER --json title,body,comments RUN git grep --author="$AUTHOR_NAME" -n . RUN grep -R "$SEARCH_PATTERN" $DIRECTORY ``` When you run a command with arguments, OpenCode will prompt you to enter values for each unique placeholder. Named arguments provide several benefits: - Clear identification of what each argument represents - Ability to use the same argument multiple times - Better organization for commands with multiple inputs ### Organizing Commands You can organize commands in subdirectories: ``` ~/.config/opencode/commands/git/commit.md ``` This creates a command with ID `user:git:commit`. ### Using Custom Commands 1. Press `Ctrl+K` to open the command dialog 2. Select your custom command (prefixed with either `user:` or `project:`) 3. Press Enter to execute the command The content of the command file will be sent as a message to the AI assistant. ### Built-in Commands OpenCode includes several built-in commands: | Command | Description | | ------------------ | --------------------------------------------------------------------------------------------------- | | Initialize Project | Creates or updates the OpenCode.md memory file with project-specific information | | Compact Session | Manually triggers the summarization of the current session, creating a new session with the summary | ## MCP (Model Context Protocol) OpenCode implements the Model Context Protocol (MCP) to extend its capabilities through external tools. MCP provides a standardized way for the AI assistant to interact with external services and tools. ### MCP Features - **External Tool Integration**: Connect to external tools and services via a standardized protocol - **Tool Discovery**: Automatically discover available tools from MCP servers - **Multiple Connection Types**: - **Stdio**: Communicate with tools via standard input/output - **SSE**: Communicate with tools via Server-Sent Events - **Security**: Permission system for controlling access to MCP tools ### Configuring MCP Servers MCP servers are defined in the configuration file under the `mcpServers` section: ```json { "mcpServers": { "example": { "type": "stdio", "command": "path/to/mcp-server", "env": [], "args": [] }, "web-example": { "type": "sse", "url": "https://example.com/mcp", "headers": { "Authorization": "Bearer token" } } } } ``` ### MCP Tool Usage Once configured, MCP tools are automatically available to the AI assistant alongside built-in tools. They follow the same permission model as other tools, requiring user approval before execution. ## LSP (Language Server Protocol) OpenCode integrates with Language Server Protocol to provide code intelligence features across multiple programming languages. ### LSP Features - **Multi-language Support**: Connect to language servers for different programming languages - **Diagnostics**: Receive error checking and linting information - **File Watching**: Automatically notify language servers of file changes ### Configuring LSP Language servers are configured in the configuration file under the `lsp` section: ```json { "lsp": { "go": { "disabled": false, "command": "gopls" }, "typescript": { "disabled": false, "command": "typescript-language-server", "args": ["--stdio"] } } } ``` ### LSP Integration with AI The AI assistant can access LSP features through the `diagnostics` tool, allowing it to: - Check for errors in your code - Suggest fixes based on diagnostics While the LSP client implementation supports the full LSP protocol (including completions, hover, definition, etc.), currently only diagnostics are exposed to the AI assistant. ## Using Github Copilot _Copilot support is currently experimental._ ### Requirements - [Copilot chat in the IDE](https://github.com/settings/copilot) enabled in GitHub settings - One of: - VSCode Github Copilot chat extension - Github `gh` CLI - Neovim Github Copilot plugin (`copilot.vim` or `copilot.lua`) - Github token with copilot permissions If using one of the above plugins or cli tools, make sure you use the authenticate the tool with your github account. This should create a github token at one of the following locations: - ~/.config/github-copilot/[hosts,apps].json - $XDG_CONFIG_HOME/github-copilot/[hosts,apps].json If using an explicit github token, you may either set the $GITHUB_TOKEN environment variable or add it to the opencode.json config file at `providers.copilot.apiKey`. ## Using a self-hosted model provider OpenCode can also load and use models from a self-hosted (OpenAI-like) provider. This is useful for developers who want to experiment with custom models. ### Configuring a self-hosted provider You can use a self-hosted model by setting the `LOCAL_ENDPOINT` environment variable. This will cause OpenCode to load and use the models from the specified endpoint. ```bash LOCAL_ENDPOINT=http://localhost:1235/v1 ``` ### Configuring a self-hosted model You can also configure a self-hosted model in the configuration file under the `agents` section: ```json { "agents": { "coder": { "model": "local.granite-3.3-2b-instruct@q8_0", "reasoningEffort": "high" } } } ``` ## Development ### Prerequisites - Go 1.24.0 or higher ### Building from Source ```bash # Clone the repository git clone https://github.com/opencode-ai/opencode.git cd opencode # Build go build -o opencode # Run ./opencode ``` ## Acknowledgments OpenCode gratefully acknowledges the contributions and support from these key individuals: - [@isaacphi](https://github.com/isaacphi) - For the [mcp-language-server](https://github.com/isaacphi/mcp-language-server) project which provided the foundation for our LSP client implementation - [@adamdottv](https://github.com/adamdottv) - For the design direction and UI/UX architecture Special thanks to the broader open source community whose tools and libraries have made this project possible. ## License OpenCode is licensed under the MIT License. See the [LICENSE](LICENSE) file for details. ## Contributing Contributions are welcome! Here's how you can contribute: 1. Fork the repository 2. Create a feature branch (`git checkout -b feature/amazing-feature`) 3. Commit your changes (`git commit -m 'Add some amazing feature'`) 4. Push to the branch (`git push origin feature/amazing-feature`) 5. Open a Pull Request Please make sure to update tests as appropriate and follow the existing code style. ================================================ FILE: cmd/root.go ================================================ package cmd import ( "context" "fmt" "os" "sync" "time" tea "github.com/charmbracelet/bubbletea" zone "github.com/lrstanley/bubblezone" "github.com/opencode-ai/opencode/internal/app" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/db" "github.com/opencode-ai/opencode/internal/format" "github.com/opencode-ai/opencode/internal/llm/agent" "github.com/opencode-ai/opencode/internal/logging" "github.com/opencode-ai/opencode/internal/pubsub" "github.com/opencode-ai/opencode/internal/tui" "github.com/opencode-ai/opencode/internal/version" "github.com/spf13/cobra" ) var rootCmd = &cobra.Command{ Use: "opencode", Short: "Terminal-based AI assistant for software development", Long: `OpenCode is a powerful terminal-based AI assistant that helps with software development tasks. It provides an interactive chat interface with AI capabilities, code analysis, and LSP integration to assist developers in writing, debugging, and understanding code directly from the terminal.`, Example: ` # Run in interactive mode opencode # Run with debug logging opencode -d # Run with debug logging in a specific directory opencode -d -c /path/to/project # Print version opencode -v # Run a single non-interactive prompt opencode -p "Explain the use of context in Go" # Run a single non-interactive prompt with JSON output format opencode -p "Explain the use of context in Go" -f json `, RunE: func(cmd *cobra.Command, args []string) error { // If the help flag is set, show the help message if cmd.Flag("help").Changed { cmd.Help() return nil } if cmd.Flag("version").Changed { fmt.Println(version.Version) return nil } // Load the config debug, _ := cmd.Flags().GetBool("debug") cwd, _ := cmd.Flags().GetString("cwd") prompt, _ := cmd.Flags().GetString("prompt") outputFormat, _ := cmd.Flags().GetString("output-format") quiet, _ := cmd.Flags().GetBool("quiet") // Validate format option if !format.IsValid(outputFormat) { return fmt.Errorf("invalid format option: %s\n%s", outputFormat, format.GetHelpText()) } if cwd != "" { err := os.Chdir(cwd) if err != nil { return fmt.Errorf("failed to change directory: %v", err) } } if cwd == "" { c, err := os.Getwd() if err != nil { return fmt.Errorf("failed to get current working directory: %v", err) } cwd = c } _, err := config.Load(cwd, debug) if err != nil { return err } // Connect DB, this will also run migrations conn, err := db.Connect() if err != nil { return err } // Create main context for the application ctx, cancel := context.WithCancel(context.Background()) defer cancel() app, err := app.New(ctx, conn) if err != nil { logging.Error("Failed to create app: %v", err) return err } // Defer shutdown here so it runs for both interactive and non-interactive modes defer app.Shutdown() // Initialize MCP tools early for both modes initMCPTools(ctx, app) // Non-interactive mode if prompt != "" { // Run non-interactive flow using the App method return app.RunNonInteractive(ctx, prompt, outputFormat, quiet) } // Interactive mode // Set up the TUI zone.NewGlobal() program := tea.NewProgram( tui.New(app), tea.WithAltScreen(), ) // Setup the subscriptions, this will send services events to the TUI ch, cancelSubs := setupSubscriptions(app, ctx) // Create a context for the TUI message handler tuiCtx, tuiCancel := context.WithCancel(ctx) var tuiWg sync.WaitGroup tuiWg.Add(1) // Set up message handling for the TUI go func() { defer tuiWg.Done() defer logging.RecoverPanic("TUI-message-handler", func() { attemptTUIRecovery(program) }) for { select { case <-tuiCtx.Done(): logging.Info("TUI message handler shutting down") return case msg, ok := <-ch: if !ok { logging.Info("TUI message channel closed") return } program.Send(msg) } } }() // Cleanup function for when the program exits cleanup := func() { // Shutdown the app app.Shutdown() // Cancel subscriptions first cancelSubs() // Then cancel TUI message handler tuiCancel() // Wait for TUI message handler to finish tuiWg.Wait() logging.Info("All goroutines cleaned up") } // Run the TUI result, err := program.Run() cleanup() if err != nil { logging.Error("TUI error: %v", err) return fmt.Errorf("TUI error: %v", err) } logging.Info("TUI exited with result: %v", result) return nil }, } // attemptTUIRecovery tries to recover the TUI after a panic func attemptTUIRecovery(program *tea.Program) { logging.Info("Attempting to recover TUI after panic") // We could try to restart the TUI or gracefully exit // For now, we'll just quit the program to avoid further issues program.Quit() } func initMCPTools(ctx context.Context, app *app.App) { go func() { defer logging.RecoverPanic("MCP-goroutine", nil) // Create a context with timeout for the initial MCP tools fetch ctxWithTimeout, cancel := context.WithTimeout(ctx, 30*time.Second) defer cancel() // Set this up once with proper error handling agent.GetMcpTools(ctxWithTimeout, app.Permissions) logging.Info("MCP message handling goroutine exiting") }() } func setupSubscriber[T any]( ctx context.Context, wg *sync.WaitGroup, name string, subscriber func(context.Context) <-chan pubsub.Event[T], outputCh chan<- tea.Msg, ) { wg.Add(1) go func() { defer wg.Done() defer logging.RecoverPanic(fmt.Sprintf("subscription-%s", name), nil) subCh := subscriber(ctx) for { select { case event, ok := <-subCh: if !ok { logging.Info("subscription channel closed", "name", name) return } var msg tea.Msg = event select { case outputCh <- msg: case <-time.After(2 * time.Second): logging.Warn("message dropped due to slow consumer", "name", name) case <-ctx.Done(): logging.Info("subscription cancelled", "name", name) return } case <-ctx.Done(): logging.Info("subscription cancelled", "name", name) return } } }() } func setupSubscriptions(app *app.App, parentCtx context.Context) (chan tea.Msg, func()) { ch := make(chan tea.Msg, 100) wg := sync.WaitGroup{} ctx, cancel := context.WithCancel(parentCtx) // Inherit from parent context setupSubscriber(ctx, &wg, "logging", logging.Subscribe, ch) setupSubscriber(ctx, &wg, "sessions", app.Sessions.Subscribe, ch) setupSubscriber(ctx, &wg, "messages", app.Messages.Subscribe, ch) setupSubscriber(ctx, &wg, "permissions", app.Permissions.Subscribe, ch) setupSubscriber(ctx, &wg, "coderAgent", app.CoderAgent.Subscribe, ch) cleanupFunc := func() { logging.Info("Cancelling all subscriptions") cancel() // Signal all goroutines to stop waitCh := make(chan struct{}) go func() { defer logging.RecoverPanic("subscription-cleanup", nil) wg.Wait() close(waitCh) }() select { case <-waitCh: logging.Info("All subscription goroutines completed successfully") close(ch) // Only close after all writers are confirmed done case <-time.After(5 * time.Second): logging.Warn("Timed out waiting for some subscription goroutines to complete") close(ch) } } return ch, cleanupFunc } func Execute() { err := rootCmd.Execute() if err != nil { os.Exit(1) } } func init() { rootCmd.Flags().BoolP("help", "h", false, "Help") rootCmd.Flags().BoolP("version", "v", false, "Version") rootCmd.Flags().BoolP("debug", "d", false, "Debug") rootCmd.Flags().StringP("cwd", "c", "", "Current working directory") rootCmd.Flags().StringP("prompt", "p", "", "Prompt to run in non-interactive mode") // Add format flag with validation logic rootCmd.Flags().StringP("output-format", "f", format.Text.String(), "Output format for non-interactive mode (text, json)") // Add quiet flag to hide spinner in non-interactive mode rootCmd.Flags().BoolP("quiet", "q", false, "Hide spinner in non-interactive mode") // Register custom validation for the format flag rootCmd.RegisterFlagCompletionFunc("output-format", func(cmd *cobra.Command, args []string, toComplete string) ([]string, cobra.ShellCompDirective) { return format.SupportedFormats, cobra.ShellCompDirectiveNoFileComp }) } ================================================ FILE: cmd/schema/README.md ================================================ # OpenCode Configuration Schema Generator This tool generates a JSON Schema for the OpenCode configuration file. The schema can be used to validate configuration files and provide autocompletion in editors that support JSON Schema. ## Usage ```bash go run cmd/schema/main.go > opencode-schema.json ``` This will generate a JSON Schema file that can be used to validate configuration files. ## Schema Features The generated schema includes: - All configuration options with descriptions - Default values where applicable - Validation for enum values (e.g., model IDs, provider types) - Required fields - Type checking ## Using the Schema You can use the generated schema in several ways: 1. **Editor Integration**: Many editors (VS Code, JetBrains IDEs, etc.) support JSON Schema for validation and autocompletion. You can configure your editor to use the generated schema for `.opencode.json` files. 2. **Validation Tools**: You can use tools like [jsonschema](https://github.com/Julian/jsonschema) to validate your configuration files against the schema. 3. **Documentation**: The schema serves as documentation for the configuration options. ## Example Configuration Here's an example configuration that conforms to the schema: ```json { "data": { "directory": ".opencode" }, "debug": false, "providers": { "anthropic": { "apiKey": "your-api-key" } }, "agents": { "coder": { "model": "claude-3.7-sonnet", "maxTokens": 5000, "reasoningEffort": "medium" }, "task": { "model": "claude-3.7-sonnet", "maxTokens": 5000 }, "title": { "model": "claude-3.7-sonnet", "maxTokens": 80 } } } ``` ================================================ FILE: cmd/schema/main.go ================================================ package main import ( "encoding/json" "fmt" "os" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/llm/models" ) // JSONSchemaType represents a JSON Schema type type JSONSchemaType struct { Type string `json:"type,omitempty"` Description string `json:"description,omitempty"` Properties map[string]any `json:"properties,omitempty"` Required []string `json:"required,omitempty"` AdditionalProperties any `json:"additionalProperties,omitempty"` Enum []any `json:"enum,omitempty"` Items map[string]any `json:"items,omitempty"` OneOf []map[string]any `json:"oneOf,omitempty"` AnyOf []map[string]any `json:"anyOf,omitempty"` Default any `json:"default,omitempty"` } func main() { schema := generateSchema() // Pretty print the schema encoder := json.NewEncoder(os.Stdout) encoder.SetIndent("", " ") if err := encoder.Encode(schema); err != nil { fmt.Fprintf(os.Stderr, "Error encoding schema: %v\n", err) os.Exit(1) } } func generateSchema() map[string]any { schema := map[string]any{ "$schema": "http://json-schema.org/draft-07/schema#", "title": "OpenCode Configuration", "description": "Configuration schema for the OpenCode application", "type": "object", "properties": map[string]any{}, } // Add Data configuration schema["properties"].(map[string]any)["data"] = map[string]any{ "type": "object", "description": "Storage configuration", "properties": map[string]any{ "directory": map[string]any{ "type": "string", "description": "Directory where application data is stored", "default": ".opencode", }, }, "required": []string{"directory"}, } // Add working directory schema["properties"].(map[string]any)["wd"] = map[string]any{ "type": "string", "description": "Working directory for the application", } // Add debug flags schema["properties"].(map[string]any)["debug"] = map[string]any{ "type": "boolean", "description": "Enable debug mode", "default": false, } schema["properties"].(map[string]any)["debugLSP"] = map[string]any{ "type": "boolean", "description": "Enable LSP debug mode", "default": false, } schema["properties"].(map[string]any)["contextPaths"] = map[string]any{ "type": "array", "description": "Context paths for the application", "items": map[string]any{ "type": "string", }, "default": []string{ ".github/copilot-instructions.md", ".cursorrules", ".cursor/rules/", "CLAUDE.md", "CLAUDE.local.md", "opencode.md", "opencode.local.md", "OpenCode.md", "OpenCode.local.md", "OPENCODE.md", "OPENCODE.local.md", }, } schema["properties"].(map[string]any)["tui"] = map[string]any{ "type": "object", "description": "Terminal User Interface configuration", "properties": map[string]any{ "theme": map[string]any{ "type": "string", "description": "TUI theme name", "default": "opencode", "enum": []string{ "opencode", "catppuccin", "dracula", "flexoki", "gruvbox", "monokai", "onedark", "tokyonight", "tron", }, }, }, } // Add MCP servers schema["properties"].(map[string]any)["mcpServers"] = map[string]any{ "type": "object", "description": "Model Control Protocol server configurations", "additionalProperties": map[string]any{ "type": "object", "description": "MCP server configuration", "properties": map[string]any{ "command": map[string]any{ "type": "string", "description": "Command to execute for the MCP server", }, "env": map[string]any{ "type": "array", "description": "Environment variables for the MCP server", "items": map[string]any{ "type": "string", }, }, "args": map[string]any{ "type": "array", "description": "Command arguments for the MCP server", "items": map[string]any{ "type": "string", }, }, "type": map[string]any{ "type": "string", "description": "Type of MCP server", "enum": []string{"stdio", "sse"}, "default": "stdio", }, "url": map[string]any{ "type": "string", "description": "URL for SSE type MCP servers", }, "headers": map[string]any{ "type": "object", "description": "HTTP headers for SSE type MCP servers", "additionalProperties": map[string]any{ "type": "string", }, }, }, "required": []string{"command"}, }, } // Add providers providerSchema := map[string]any{ "type": "object", "description": "LLM provider configurations", "additionalProperties": map[string]any{ "type": "object", "description": "Provider configuration", "properties": map[string]any{ "apiKey": map[string]any{ "type": "string", "description": "API key for the provider", }, "disabled": map[string]any{ "type": "boolean", "description": "Whether the provider is disabled", "default": false, }, }, }, } // Add known providers knownProviders := []string{ string(models.ProviderAnthropic), string(models.ProviderOpenAI), string(models.ProviderGemini), string(models.ProviderGROQ), string(models.ProviderOpenRouter), string(models.ProviderBedrock), string(models.ProviderAzure), string(models.ProviderVertexAI), } providerSchema["additionalProperties"].(map[string]any)["properties"].(map[string]any)["provider"] = map[string]any{ "type": "string", "description": "Provider type", "enum": knownProviders, } schema["properties"].(map[string]any)["providers"] = providerSchema // Add agents agentSchema := map[string]any{ "type": "object", "description": "Agent configurations", "additionalProperties": map[string]any{ "type": "object", "description": "Agent configuration", "properties": map[string]any{ "model": map[string]any{ "type": "string", "description": "Model ID for the agent", }, "maxTokens": map[string]any{ "type": "integer", "description": "Maximum tokens for the agent", "minimum": 1, }, "reasoningEffort": map[string]any{ "type": "string", "description": "Reasoning effort for models that support it (OpenAI, Anthropic)", "enum": []string{"low", "medium", "high"}, }, }, "required": []string{"model"}, }, } // Add model enum modelEnum := []string{} for modelID := range models.SupportedModels { modelEnum = append(modelEnum, string(modelID)) } agentSchema["additionalProperties"].(map[string]any)["properties"].(map[string]any)["model"].(map[string]any)["enum"] = modelEnum // Add specific agent properties agentProperties := map[string]any{} knownAgents := []string{ string(config.AgentCoder), string(config.AgentTask), string(config.AgentTitle), } for _, agentName := range knownAgents { agentProperties[agentName] = map[string]any{ "$ref": "#/definitions/agent", } } // Create a combined schema that allows both specific agents and additional ones combinedAgentSchema := map[string]any{ "type": "object", "description": "Agent configurations", "properties": agentProperties, "additionalProperties": agentSchema["additionalProperties"], } schema["properties"].(map[string]any)["agents"] = combinedAgentSchema schema["definitions"] = map[string]any{ "agent": agentSchema["additionalProperties"], } // Add LSP configuration schema["properties"].(map[string]any)["lsp"] = map[string]any{ "type": "object", "description": "Language Server Protocol configurations", "additionalProperties": map[string]any{ "type": "object", "description": "LSP configuration for a language", "properties": map[string]any{ "disabled": map[string]any{ "type": "boolean", "description": "Whether the LSP is disabled", "default": false, }, "command": map[string]any{ "type": "string", "description": "Command to execute for the LSP server", }, "args": map[string]any{ "type": "array", "description": "Command arguments for the LSP server", "items": map[string]any{ "type": "string", }, }, "options": map[string]any{ "type": "object", "description": "Additional options for the LSP server", }, }, "required": []string{"command"}, }, } return schema } ================================================ FILE: go.mod ================================================ module github.com/opencode-ai/opencode go 1.24.0 require ( github.com/Azure/azure-sdk-for-go/sdk/azidentity v1.7.0 github.com/JohannesKaufmann/html-to-markdown v1.6.0 github.com/PuerkitoBio/goquery v1.9.2 github.com/alecthomas/chroma/v2 v2.15.0 github.com/anthropics/anthropic-sdk-go v1.4.0 github.com/aymanbagabas/go-udiff v0.2.0 github.com/bmatcuk/doublestar/v4 v4.8.1 github.com/catppuccin/go v0.3.0 github.com/charmbracelet/bubbles v0.21.0 github.com/charmbracelet/bubbletea v1.3.5 github.com/charmbracelet/glamour v0.9.1 github.com/charmbracelet/lipgloss v1.1.0 github.com/charmbracelet/x/ansi v0.8.0 github.com/fsnotify/fsnotify v1.8.0 github.com/go-logfmt/logfmt v0.6.0 github.com/google/uuid v1.6.0 github.com/lrstanley/bubblezone v0.0.0-20250315020633-c249a3fe1231 github.com/mark3labs/mcp-go v0.17.0 github.com/muesli/ansi v0.0.0-20230316100256-276c6243b2f6 github.com/muesli/reflow v0.3.0 github.com/muesli/termenv v0.16.0 github.com/ncruces/go-sqlite3 v0.25.0 github.com/openai/openai-go v0.1.0-beta.2 github.com/pressly/goose/v3 v3.24.2 github.com/sergi/go-diff v1.3.2-0.20230802210424-5b0b94c5c0d3 github.com/spf13/cobra v1.9.1 github.com/spf13/viper v1.20.0 github.com/stretchr/testify v1.10.0 ) require ( cloud.google.com/go v0.116.0 // indirect cloud.google.com/go/auth v0.13.0 // indirect cloud.google.com/go/compute/metadata v0.6.0 // indirect github.com/Azure/azure-sdk-for-go/sdk/azcore v1.17.0 // indirect github.com/Azure/azure-sdk-for-go/sdk/internal v1.10.0 // indirect github.com/AzureAD/microsoft-authentication-library-for-go v1.2.2 // indirect github.com/andybalholm/cascadia v1.3.2 // indirect github.com/atotto/clipboard v0.1.4 // indirect github.com/aws/aws-sdk-go-v2 v1.30.3 // indirect github.com/aws/aws-sdk-go-v2/aws/protocol/eventstream v1.6.3 // indirect github.com/aws/aws-sdk-go-v2/config v1.27.27 // indirect github.com/aws/aws-sdk-go-v2/credentials v1.17.27 // indirect github.com/aws/aws-sdk-go-v2/feature/ec2/imds v1.16.11 // indirect github.com/aws/aws-sdk-go-v2/internal/configsources v1.3.15 // indirect github.com/aws/aws-sdk-go-v2/internal/endpoints/v2 v2.6.15 // indirect github.com/aws/aws-sdk-go-v2/internal/ini v1.8.0 // indirect github.com/aws/aws-sdk-go-v2/service/internal/accept-encoding v1.11.3 // indirect github.com/aws/aws-sdk-go-v2/service/internal/presigned-url v1.11.17 // indirect github.com/aws/aws-sdk-go-v2/service/sso v1.22.4 // indirect github.com/aws/aws-sdk-go-v2/service/ssooidc v1.26.4 // indirect github.com/aws/aws-sdk-go-v2/service/sts v1.30.3 // indirect github.com/aws/smithy-go v1.20.3 // indirect github.com/aymanbagabas/go-osc52/v2 v2.0.1 // indirect github.com/aymerick/douceur v0.2.0 // indirect github.com/charmbracelet/colorprofile v0.2.3-0.20250311203215-f60798e515dc // indirect github.com/charmbracelet/x/cellbuf v0.0.13-0.20250311204145-2c3ea96c31dd // indirect github.com/charmbracelet/x/term v0.2.1 // indirect github.com/davecgh/go-spew v1.1.1 // indirect github.com/disintegration/imaging v1.6.2 github.com/dlclark/regexp2 v1.11.4 // indirect github.com/erikgeiser/coninput v0.0.0-20211004153227-1c3628e74d0f // indirect github.com/felixge/httpsnoop v1.0.4 // indirect github.com/go-logr/logr v1.4.2 // indirect github.com/go-logr/stdr v1.2.2 // indirect github.com/go-viper/mapstructure/v2 v2.2.1 // indirect github.com/golang-jwt/jwt/v5 v5.2.2 // indirect github.com/google/go-cmp v0.7.0 // indirect github.com/google/s2a-go v0.1.8 // indirect github.com/googleapis/enterprise-certificate-proxy v0.3.4 // indirect github.com/googleapis/gax-go/v2 v2.14.1 // indirect github.com/gorilla/css v1.0.1 // indirect github.com/gorilla/websocket v1.5.3 // indirect github.com/inconshreveable/mousetrap v1.1.0 // indirect github.com/kylelemons/godebug v1.1.0 // indirect github.com/lithammer/fuzzysearch v1.1.8 github.com/lucasb-eyer/go-colorful v1.2.0 github.com/mattn/go-isatty v0.0.20 // indirect github.com/mattn/go-localereader v0.0.1 // indirect github.com/mattn/go-runewidth v0.0.16 // indirect github.com/mfridman/interpolate v0.0.2 // indirect github.com/microcosm-cc/bluemonday v1.0.27 // indirect github.com/muesli/cancelreader v0.2.2 // indirect github.com/ncruces/julianday v1.0.0 // indirect github.com/pelletier/go-toml/v2 v2.2.3 // indirect github.com/pkg/browser v0.0.0-20240102092130-5ac0b6a4141c // indirect github.com/pmezard/go-difflib v1.0.0 // indirect github.com/rivo/uniseg v0.4.7 // indirect github.com/rogpeppe/go-internal v1.14.1 // indirect github.com/sagikazarmark/locafero v0.7.0 // indirect github.com/sethvargo/go-retry v0.3.0 // indirect github.com/sourcegraph/conc v0.3.0 // indirect github.com/spf13/afero v1.12.0 // indirect github.com/spf13/cast v1.7.1 // indirect github.com/spf13/pflag v1.0.6 // indirect github.com/subosito/gotenv v1.6.0 // indirect github.com/tetratelabs/wazero v1.9.0 // indirect github.com/tidwall/gjson v1.18.0 // indirect github.com/tidwall/match v1.1.1 // indirect github.com/tidwall/pretty v1.2.1 // indirect github.com/tidwall/sjson v1.2.5 // indirect github.com/xo/terminfo v0.0.0-20220910002029-abceb7e1c41e // indirect github.com/yosida95/uritemplate/v3 v3.0.2 // indirect github.com/yuin/goldmark v1.7.8 // indirect github.com/yuin/goldmark-emoji v1.0.5 // indirect go.opentelemetry.io/auto/sdk v1.1.0 // indirect go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.54.0 // indirect go.opentelemetry.io/otel v1.35.0 // indirect go.opentelemetry.io/otel/metric v1.35.0 // indirect go.opentelemetry.io/otel/trace v1.35.0 // indirect go.uber.org/multierr v1.11.0 // indirect golang.org/x/crypto v0.37.0 // indirect golang.org/x/image v0.26.0 // indirect golang.org/x/net v0.39.0 // indirect golang.org/x/sync v0.13.0 // indirect golang.org/x/sys v0.32.0 // indirect golang.org/x/term v0.31.0 // indirect golang.org/x/text v0.24.0 // indirect google.golang.org/genai v1.3.0 google.golang.org/genproto/googleapis/rpc v0.0.0-20250324211829-b45e905df463 // indirect google.golang.org/grpc v1.71.0 // indirect google.golang.org/protobuf v1.36.6 // indirect gopkg.in/yaml.v3 v3.0.1 // indirect ) ================================================ FILE: go.sum ================================================ cloud.google.com/go v0.116.0 h1:B3fRrSDkLRt5qSHWe40ERJvhvnQwdZiHu0bJOpldweE= cloud.google.com/go v0.116.0/go.mod h1:cEPSRWPzZEswwdr9BxE6ChEn01dWlTaF05LiC2Xs70U= cloud.google.com/go/auth v0.13.0 h1:8Fu8TZy167JkW8Tj3q7dIkr2v4cndv41ouecJx0PAHs= cloud.google.com/go/auth v0.13.0/go.mod h1:COOjD9gwfKNKz+IIduatIhYJQIc0mG3H102r/EMxX6Q= cloud.google.com/go/compute/metadata v0.6.0 h1:A6hENjEsCDtC1k8byVsgwvVcioamEHvZ4j01OwKxG9I= cloud.google.com/go/compute/metadata v0.6.0/go.mod h1:FjyFAW1MW0C203CEOMDTu3Dk1FlqW3Rga40jzHL4hfg= github.com/Azure/azure-sdk-for-go/sdk/azcore v1.17.0 h1:g0EZJwz7xkXQiZAI5xi9f3WWFYBlX1CPTrR+NDToRkQ= github.com/Azure/azure-sdk-for-go/sdk/azcore v1.17.0/go.mod h1:XCW7KnZet0Opnr7HccfUw1PLc4CjHqpcaxW8DHklNkQ= github.com/Azure/azure-sdk-for-go/sdk/azidentity v1.7.0 h1:tfLQ34V6F7tVSwoTf/4lH5sE0o6eCJuNDTmH09nDpbc= github.com/Azure/azure-sdk-for-go/sdk/azidentity v1.7.0/go.mod h1:9kIvujWAA58nmPmWB1m23fyWic1kYZMxD9CxaWn4Qpg= github.com/Azure/azure-sdk-for-go/sdk/internal v1.10.0 h1:ywEEhmNahHBihViHepv3xPBn1663uRv2t2q/ESv9seY= github.com/Azure/azure-sdk-for-go/sdk/internal v1.10.0/go.mod h1:iZDifYGJTIgIIkYRNWPENUnqx6bJ2xnSDFI2tjwZNuY= github.com/AzureAD/microsoft-authentication-library-for-go v1.2.2 h1:XHOnouVk1mxXfQidrMEnLlPk9UMeRtyBTnEFtxkV0kU= github.com/AzureAD/microsoft-authentication-library-for-go v1.2.2/go.mod h1:wP83P5OoQ5p6ip3ScPr0BAq0BvuPAvacpEuSzyouqAI= github.com/JohannesKaufmann/html-to-markdown v1.6.0 h1:04VXMiE50YYfCfLboJCLcgqF5x+rHJnb1ssNmqpLH/k= github.com/JohannesKaufmann/html-to-markdown v1.6.0/go.mod h1:NUI78lGg/a7vpEJTz/0uOcYMaibytE4BUOQS8k78yPQ= github.com/MakeNowJust/heredoc v1.0.0 h1:cXCdzVdstXyiTqTvfqk9SDHpKNjxuom+DOlyEeQ4pzQ= github.com/MakeNowJust/heredoc v1.0.0/go.mod h1:mG5amYoWBHf8vpLOuehzbGGw0EHxpZZ6lCpQ4fNJ8LE= github.com/PuerkitoBio/goquery v1.9.2 h1:4/wZksC3KgkQw7SQgkKotmKljk0M6V8TUvA8Wb4yPeE= github.com/PuerkitoBio/goquery v1.9.2/go.mod h1:GHPCaP0ODyyxqcNoFGYlAprUFH81NuRPd0GX3Zu2Mvk= github.com/alecthomas/assert/v2 v2.11.0 h1:2Q9r3ki8+JYXvGsDyBXwH3LcJ+WK5D0gc5E8vS6K3D0= github.com/alecthomas/assert/v2 v2.11.0/go.mod h1:Bze95FyfUr7x34QZrjL+XP+0qgp/zg8yS+TtBj1WA3k= github.com/alecthomas/chroma/v2 v2.15.0 h1:LxXTQHFoYrstG2nnV9y2X5O94sOBzf0CIUpSTbpxvMc= github.com/alecthomas/chroma/v2 v2.15.0/go.mod h1:gUhVLrPDXPtp/f+L1jo9xepo9gL4eLwRuGAunSZMkio= github.com/alecthomas/repr v0.4.0 h1:GhI2A8MACjfegCPVq9f1FLvIBS+DrQ2KQBFZP1iFzXc= github.com/alecthomas/repr v0.4.0/go.mod h1:Fr0507jx4eOXV7AlPV6AVZLYrLIuIeSOWtW57eE/O/4= github.com/andybalholm/cascadia v1.3.2 h1:3Xi6Dw5lHF15JtdcmAHD3i1+T8plmv7BQ/nsViSLyss= github.com/andybalholm/cascadia v1.3.2/go.mod h1:7gtRlve5FxPPgIgX36uWBX58OdBsSS6lUvCFb+h7KvU= github.com/anthropics/anthropic-sdk-go v1.4.0 h1:fU1jKxYbQdQDiEXCxeW5XZRIOwKevn/PMg8Ay1nnUx0= github.com/anthropics/anthropic-sdk-go v1.4.0/go.mod h1:AapDW22irxK2PSumZiQXYUFvsdQgkwIWlpESweWZI/c= github.com/atotto/clipboard v0.1.4 h1:EH0zSVneZPSuFR11BlR9YppQTVDbh5+16AmcJi4g1z4= github.com/atotto/clipboard v0.1.4/go.mod h1:ZY9tmq7sm5xIbd9bOK4onWV4S6X0u6GY7Vn0Yu86PYI= github.com/aws/aws-sdk-go-v2 v1.30.3 h1:jUeBtG0Ih+ZIFH0F4UkmL9w3cSpaMv9tYYDbzILP8dY= github.com/aws/aws-sdk-go-v2 v1.30.3/go.mod h1:nIQjQVp5sfpQcTc9mPSr1B0PaWK5ByX9MOoDadSN4lc= github.com/aws/aws-sdk-go-v2/aws/protocol/eventstream v1.6.3 h1:tW1/Rkad38LA15X4UQtjXZXNKsCgkshC3EbmcUmghTg= github.com/aws/aws-sdk-go-v2/aws/protocol/eventstream v1.6.3/go.mod h1:UbnqO+zjqk3uIt9yCACHJ9IVNhyhOCnYk8yA19SAWrM= github.com/aws/aws-sdk-go-v2/config v1.27.27 h1:HdqgGt1OAP0HkEDDShEl0oSYa9ZZBSOmKpdpsDMdO90= github.com/aws/aws-sdk-go-v2/config v1.27.27/go.mod h1:MVYamCg76dFNINkZFu4n4RjDixhVr51HLj4ErWzrVwg= github.com/aws/aws-sdk-go-v2/credentials v1.17.27 h1:2raNba6gr2IfA0eqqiP2XiQ0UVOpGPgDSi0I9iAP+UI= github.com/aws/aws-sdk-go-v2/credentials v1.17.27/go.mod h1:gniiwbGahQByxan6YjQUMcW4Aov6bLC3m+evgcoN4r4= github.com/aws/aws-sdk-go-v2/feature/ec2/imds v1.16.11 h1:KreluoV8FZDEtI6Co2xuNk/UqI9iwMrOx/87PBNIKqw= github.com/aws/aws-sdk-go-v2/feature/ec2/imds v1.16.11/go.mod h1:SeSUYBLsMYFoRvHE0Tjvn7kbxaUhl75CJi1sbfhMxkU= github.com/aws/aws-sdk-go-v2/internal/configsources v1.3.15 h1:SoNJ4RlFEQEbtDcCEt+QG56MY4fm4W8rYirAmq+/DdU= github.com/aws/aws-sdk-go-v2/internal/configsources v1.3.15/go.mod h1:U9ke74k1n2bf+RIgoX1SXFed1HLs51OgUSs+Ph0KJP8= github.com/aws/aws-sdk-go-v2/internal/endpoints/v2 v2.6.15 h1:C6WHdGnTDIYETAm5iErQUiVNsclNx9qbJVPIt03B6bI= github.com/aws/aws-sdk-go-v2/internal/endpoints/v2 v2.6.15/go.mod h1:ZQLZqhcu+JhSrA9/NXRm8SkDvsycE+JkV3WGY41e+IM= github.com/aws/aws-sdk-go-v2/internal/ini v1.8.0 h1:hT8rVHwugYE2lEfdFE0QWVo81lF7jMrYJVDWI+f+VxU= github.com/aws/aws-sdk-go-v2/internal/ini v1.8.0/go.mod h1:8tu/lYfQfFe6IGnaOdrpVgEL2IrrDOf6/m9RQum4NkY= github.com/aws/aws-sdk-go-v2/service/internal/accept-encoding v1.11.3 h1:dT3MqvGhSoaIhRseqw2I0yH81l7wiR2vjs57O51EAm8= github.com/aws/aws-sdk-go-v2/service/internal/accept-encoding v1.11.3/go.mod h1:GlAeCkHwugxdHaueRr4nhPuY+WW+gR8UjlcqzPr1SPI= github.com/aws/aws-sdk-go-v2/service/internal/presigned-url v1.11.17 h1:HGErhhrxZlQ044RiM+WdoZxp0p+EGM62y3L6pwA4olE= github.com/aws/aws-sdk-go-v2/service/internal/presigned-url v1.11.17/go.mod h1:RkZEx4l0EHYDJpWppMJ3nD9wZJAa8/0lq9aVC+r2UII= github.com/aws/aws-sdk-go-v2/service/sso v1.22.4 h1:BXx0ZIxvrJdSgSvKTZ+yRBeSqqgPM89VPlulEcl37tM= github.com/aws/aws-sdk-go-v2/service/sso v1.22.4/go.mod h1:ooyCOXjvJEsUw7x+ZDHeISPMhtwI3ZCB7ggFMcFfWLU= github.com/aws/aws-sdk-go-v2/service/ssooidc v1.26.4 h1:yiwVzJW2ZxZTurVbYWA7QOrAaCYQR72t0wrSBfoesUE= github.com/aws/aws-sdk-go-v2/service/ssooidc v1.26.4/go.mod h1:0oxfLkpz3rQ/CHlx5hB7H69YUpFiI1tql6Q6Ne+1bCw= github.com/aws/aws-sdk-go-v2/service/sts v1.30.3 h1:ZsDKRLXGWHk8WdtyYMoGNO7bTudrvuKpDKgMVRlepGE= github.com/aws/aws-sdk-go-v2/service/sts v1.30.3/go.mod h1:zwySh8fpFyXp9yOr/KVzxOl8SRqgf/IDw5aUt9UKFcQ= github.com/aws/smithy-go v1.20.3 h1:ryHwveWzPV5BIof6fyDvor6V3iUL7nTfiTKXHiW05nE= github.com/aws/smithy-go v1.20.3/go.mod h1:krry+ya/rV9RDcV/Q16kpu6ypI4K2czasz0NC3qS14E= github.com/aymanbagabas/go-osc52/v2 v2.0.1 h1:HwpRHbFMcZLEVr42D4p7XBqjyuxQH5SMiErDT4WkJ2k= github.com/aymanbagabas/go-osc52/v2 v2.0.1/go.mod h1:uYgXzlJ7ZpABp8OJ+exZzJJhRNQ2ASbcXHWsFqH8hp8= github.com/aymanbagabas/go-udiff v0.2.0 h1:TK0fH4MteXUDspT88n8CKzvK0X9O2xu9yQjWpi6yML8= github.com/aymanbagabas/go-udiff v0.2.0/go.mod h1:RE4Ex0qsGkTAJoQdQQCA0uG+nAzJO/pI/QwceO5fgrA= github.com/aymerick/douceur v0.2.0 h1:Mv+mAeH1Q+n9Fr+oyamOlAkUNPWPlA8PPGR0QAaYuPk= github.com/aymerick/douceur v0.2.0/go.mod h1:wlT5vV2O3h55X9m7iVYN0TBM0NH/MmbLnd30/FjWUq4= github.com/bmatcuk/doublestar/v4 v4.8.1 h1:54Bopc5c2cAvhLRAzqOGCYHYyhcDHsFF4wWIR5wKP38= github.com/bmatcuk/doublestar/v4 v4.8.1/go.mod h1:xBQ8jztBU6kakFMg+8WGxn0c6z1fTSPVIjEY1Wr7jzc= github.com/catppuccin/go v0.3.0 h1:d+0/YicIq+hSTo5oPuRi5kOpqkVA5tAsU6dNhvRu+aY= github.com/catppuccin/go v0.3.0/go.mod h1:8IHJuMGaUUjQM82qBrGNBv7LFq6JI3NnQCF6MOlZjpc= github.com/charmbracelet/bubbles v0.21.0 h1:9TdC97SdRVg/1aaXNVWfFH3nnLAwOXr8Fn6u6mfQdFs= github.com/charmbracelet/bubbles v0.21.0/go.mod h1:HF+v6QUR4HkEpz62dx7ym2xc71/KBHg+zKwJtMw+qtg= github.com/charmbracelet/bubbletea v1.3.5 h1:JAMNLTbqMOhSwoELIr0qyP4VidFq72/6E9j7HHmRKQc= github.com/charmbracelet/bubbletea v1.3.5/go.mod h1:TkCnmH+aBd4LrXhXcqrKiYwRs7qyQx5rBgH5fVY3v54= github.com/charmbracelet/colorprofile v0.2.3-0.20250311203215-f60798e515dc h1:4pZI35227imm7yK2bGPcfpFEmuY1gc2YSTShr4iJBfs= github.com/charmbracelet/colorprofile v0.2.3-0.20250311203215-f60798e515dc/go.mod h1:X4/0JoqgTIPSFcRA/P6INZzIuyqdFY5rm8tb41s9okk= github.com/charmbracelet/glamour v0.9.1 h1:11dEfiGP8q1BEqvGoIjivuc2rBk+5qEXdPtaQ2WoiCM= github.com/charmbracelet/glamour v0.9.1/go.mod h1:+SHvIS8qnwhgTpVMiXwn7OfGomSqff1cHBCI8jLOetk= github.com/charmbracelet/lipgloss v1.1.0 h1:vYXsiLHVkK7fp74RkV7b2kq9+zDLoEU4MZoFqR/noCY= github.com/charmbracelet/lipgloss v1.1.0/go.mod h1:/6Q8FR2o+kj8rz4Dq0zQc3vYf7X+B0binUUBwA0aL30= github.com/charmbracelet/x/ansi v0.8.0 h1:9GTq3xq9caJW8ZrBTe0LIe2fvfLR/bYXKTx2llXn7xE= github.com/charmbracelet/x/ansi v0.8.0/go.mod h1:wdYl/ONOLHLIVmQaxbIYEC/cRKOQyjTkowiI4blgS9Q= github.com/charmbracelet/x/cellbuf v0.0.13-0.20250311204145-2c3ea96c31dd h1:vy0GVL4jeHEwG5YOXDmi86oYw2yuYUGqz6a8sLwg0X8= github.com/charmbracelet/x/cellbuf v0.0.13-0.20250311204145-2c3ea96c31dd/go.mod h1:xe0nKWGd3eJgtqZRaN9RjMtK7xUYchjzPr7q6kcvCCs= github.com/charmbracelet/x/exp/golden v0.0.0-20241011142426-46044092ad91 h1:payRxjMjKgx2PaCWLZ4p3ro9y97+TVLZNaRZgJwSVDQ= github.com/charmbracelet/x/exp/golden v0.0.0-20241011142426-46044092ad91/go.mod h1:wDlXFlCrmJ8J+swcL/MnGUuYnqgQdW9rhSD61oNMb6U= github.com/charmbracelet/x/term v0.2.1 h1:AQeHeLZ1OqSXhrAWpYUtZyX1T3zVxfpZuEQMIQaGIAQ= github.com/charmbracelet/x/term v0.2.1/go.mod h1:oQ4enTYFV7QN4m0i9mzHrViD7TQKvNEEkHUMCmsxdUg= github.com/cpuguy83/go-md2man/v2 v2.0.6/go.mod h1:oOW0eioCTA6cOiMLiUPZOpcVxMig6NIQQ7OS05n1F4g= github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/disintegration/imaging v1.6.2 h1:w1LecBlG2Lnp8B3jk5zSuNqd7b4DXhcjwek1ei82L+c= github.com/disintegration/imaging v1.6.2/go.mod h1:44/5580QXChDfwIclfc/PCwrr44amcmDAg8hxG0Ewe4= github.com/dlclark/regexp2 v1.11.4 h1:rPYF9/LECdNymJufQKmri9gV604RvvABwgOA8un7yAo= github.com/dlclark/regexp2 v1.11.4/go.mod h1:DHkYz0B9wPfa6wondMfaivmHpzrQ3v9q8cnmRbL6yW8= github.com/dustin/go-humanize v1.0.1 h1:GzkhY7T5VNhEkwH0PVJgjz+fX1rhBrR7pRT3mDkpeCY= github.com/dustin/go-humanize v1.0.1/go.mod h1:Mu1zIs6XwVuF/gI1OepvI0qD18qycQx+mFykh5fBlto= github.com/erikgeiser/coninput v0.0.0-20211004153227-1c3628e74d0f h1:Y/CXytFA4m6baUTXGLOoWe4PQhGxaX0KpnayAqC48p4= github.com/erikgeiser/coninput v0.0.0-20211004153227-1c3628e74d0f/go.mod h1:vw97MGsxSvLiUE2X8qFplwetxpGLQrlU1Q9AUEIzCaM= github.com/felixge/httpsnoop v1.0.4 h1:NFTV2Zj1bL4mc9sqWACXbQFVBBg2W3GPvqp8/ESS2Wg= github.com/felixge/httpsnoop v1.0.4/go.mod h1:m8KPJKqk1gH5J9DgRY2ASl2lWCfGKXixSwevea8zH2U= github.com/frankban/quicktest v1.14.6 h1:7Xjx+VpznH+oBnejlPUj8oUpdxnVs4f8XU8WnHkI4W8= github.com/frankban/quicktest v1.14.6/go.mod h1:4ptaffx2x8+WTWXmUCuVU6aPUX1/Mz7zb5vbUoiM6w0= github.com/fsnotify/fsnotify v1.8.0 h1:dAwr6QBTBZIkG8roQaJjGof0pp0EeF+tNV7YBP3F/8M= github.com/fsnotify/fsnotify v1.8.0/go.mod h1:8jBTzvmWwFyi3Pb8djgCCO5IBqzKJ/Jwo8TRcHyHii0= github.com/go-logfmt/logfmt v0.6.0 h1:wGYYu3uicYdqXVgoYbvnkrPVXkuLM1p1ifugDMEdRi4= github.com/go-logfmt/logfmt v0.6.0/go.mod h1:WYhtIu8zTZfxdn5+rREduYbwxfcBr/Vr6KEVveWlfTs= github.com/go-logr/logr v1.2.2/go.mod h1:jdQByPbusPIv2/zmleS9BjJVeZ6kBagPoEUsqbVz/1A= github.com/go-logr/logr v1.4.2 h1:6pFjapn8bFcIbiKo3XT4j/BhANplGihG6tvd+8rYgrY= github.com/go-logr/logr v1.4.2/go.mod h1:9T104GzyrTigFIr8wt5mBrctHMim0Nb2HLGrmQ40KvY= github.com/go-logr/stdr v1.2.2 h1:hSWxHoqTgW2S2qGc0LTAI563KZ5YKYRhT3MFKZMbjag= github.com/go-logr/stdr v1.2.2/go.mod h1:mMo/vtBO5dYbehREoey6XUKy/eSumjCCveDpRre4VKE= github.com/go-viper/mapstructure/v2 v2.2.1 h1:ZAaOCxANMuZx5RCeg0mBdEZk7DZasvvZIxtHqx8aGss= github.com/go-viper/mapstructure/v2 v2.2.1/go.mod h1:oJDH3BJKyqBA2TXFhDsKDGDTlndYOZ6rGS0BRZIxGhM= github.com/golang-jwt/jwt/v5 v5.2.2 h1:Rl4B7itRWVtYIHFrSNd7vhTiz9UpLdi6gZhZ3wEeDy8= github.com/golang-jwt/jwt/v5 v5.2.2/go.mod h1:pqrtFR0X4osieyHYxtmOUWsAWrfe1Q5UVIyoH402zdk= github.com/golang/protobuf v1.5.4 h1:i7eJL8qZTpSEXOPTxNKhASYpMn+8e5Q6AdndVa1dWek= github.com/golang/protobuf v1.5.4/go.mod h1:lnTiLA8Wa4RWRcIUkrtSVa5nRhsEGBg48fD6rSs7xps= github.com/google/go-cmp v0.7.0 h1:wk8382ETsv4JYUZwIsn6YpYiWiBsYLSJiTsyBybVuN8= github.com/google/go-cmp v0.7.0/go.mod h1:pXiqmnSA92OHEEa9HXL2W4E7lf9JzCmGVUdgjX3N/iU= github.com/google/s2a-go v0.1.8 h1:zZDs9gcbt9ZPLV0ndSyQk6Kacx2g/X+SKYovpnz3SMM= github.com/google/s2a-go v0.1.8/go.mod h1:6iNWHTpQ+nfNRN5E00MSdfDwVesa8hhS32PhPO8deJA= github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0= github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= github.com/googleapis/enterprise-certificate-proxy v0.3.4 h1:XYIDZApgAnrN1c855gTgghdIA6Stxb52D5RnLI1SLyw= github.com/googleapis/enterprise-certificate-proxy v0.3.4/go.mod h1:YKe7cfqYXjKGpGvmSg28/fFvhNzinZQm8DGnaburhGA= github.com/googleapis/gax-go/v2 v2.14.1 h1:hb0FFeiPaQskmvakKu5EbCbpntQn48jyHuvrkurSS/Q= github.com/googleapis/gax-go/v2 v2.14.1/go.mod h1:Hb/NubMaVM88SrNkvl8X/o8XWwDJEPqouaLeN2IUxoA= github.com/gorilla/css v1.0.1 h1:ntNaBIghp6JmvWnxbZKANoLyuXTPZ4cAMlo6RyhlbO8= github.com/gorilla/css v1.0.1/go.mod h1:BvnYkspnSzMmwRK+b8/xgNPLiIuNZr6vbZBTPQ2A3b0= github.com/gorilla/websocket v1.5.3 h1:saDtZ6Pbx/0u+bgYQ3q96pZgCzfhKXGPqt7kZ72aNNg= github.com/gorilla/websocket v1.5.3/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE= github.com/hexops/gotextdiff v1.0.3 h1:gitA9+qJrrTCsiCl7+kh75nPqQt1cx4ZkudSTLoUqJM= github.com/hexops/gotextdiff v1.0.3/go.mod h1:pSWU5MAI3yDq+fZBTazCSJysOMbxWL1BSow5/V2vxeg= github.com/inconshreveable/mousetrap v1.1.0 h1:wN+x4NVGpMsO7ErUn/mUI3vEoE6Jt13X2s0bqwp9tc8= github.com/inconshreveable/mousetrap v1.1.0/go.mod h1:vpF70FUmC8bwa3OWnCshd2FqLfsEA9PFc4w1p2J65bw= github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE= github.com/kr/pretty v0.3.1/go.mod h1:hoEshYVHaxMs3cyo3Yncou5ZscifuDolrwPKZanG3xk= github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= github.com/kylelemons/godebug v1.1.0 h1:RPNrshWIDI6G2gRW9EHilWtl7Z6Sb1BR0xunSBf0SNc= github.com/kylelemons/godebug v1.1.0/go.mod h1:9/0rRGxNHcop5bhtWyNeEfOS8JIWk580+fNqagV/RAw= github.com/lithammer/fuzzysearch v1.1.8 h1:/HIuJnjHuXS8bKaiTMeeDlW2/AyIWk2brx1V8LFgLN4= github.com/lithammer/fuzzysearch v1.1.8/go.mod h1:IdqeyBClc3FFqSzYq/MXESsS4S0FsZ5ajtkr5xPLts4= github.com/lrstanley/bubblezone v0.0.0-20250315020633-c249a3fe1231 h1:9rjt7AfnrXKNSZhp36A3/4QAZAwGGCGD/p8Bse26zms= github.com/lrstanley/bubblezone v0.0.0-20250315020633-c249a3fe1231/go.mod h1:S5etECMx+sZnW0Gm100Ma9J1PgVCTgNyFaqGu2b08b4= github.com/lucasb-eyer/go-colorful v1.2.0 h1:1nnpGOrhyZZuNyfu1QjKiUICQ74+3FNCN69Aj6K7nkY= github.com/lucasb-eyer/go-colorful v1.2.0/go.mod h1:R4dSotOR9KMtayYi1e77YzuveK+i7ruzyGqttikkLy0= github.com/mark3labs/mcp-go v0.17.0 h1:5Ps6T7qXr7De/2QTqs9h6BKeZ/qdeUeGrgM5lPzi930= github.com/mark3labs/mcp-go v0.17.0/go.mod h1:KmJndYv7GIgcPVwEKJjNcbhVQ+hJGJhrCCB/9xITzpE= github.com/mattn/go-isatty v0.0.20 h1:xfD0iDuEKnDkl03q4limB+vH+GxLEtL/jb4xVJSWWEY= github.com/mattn/go-isatty v0.0.20/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y= github.com/mattn/go-localereader v0.0.1 h1:ygSAOl7ZXTx4RdPYinUpg6W99U8jWvWi9Ye2JC/oIi4= github.com/mattn/go-localereader v0.0.1/go.mod h1:8fBrzywKY7BI3czFoHkuzRoWE9C+EiG4R1k4Cjx5p88= github.com/mattn/go-runewidth v0.0.12/go.mod h1:RAqKPSqVFrSLVXbA8x7dzmKdmGzieGRCM46jaSJTDAk= github.com/mattn/go-runewidth v0.0.16 h1:E5ScNMtiwvlvB5paMFdw9p4kSQzbXFikJ5SQO6TULQc= github.com/mattn/go-runewidth v0.0.16/go.mod h1:Jdepj2loyihRzMpdS35Xk/zdY8IAYHsh153qUoGf23w= github.com/mfridman/interpolate v0.0.2 h1:pnuTK7MQIxxFz1Gr+rjSIx9u7qVjf5VOoM/u6BbAxPY= github.com/mfridman/interpolate v0.0.2/go.mod h1:p+7uk6oE07mpE/Ik1b8EckO0O4ZXiGAfshKBWLUM9Xg= github.com/microcosm-cc/bluemonday v1.0.27 h1:MpEUotklkwCSLeH+Qdx1VJgNqLlpY2KXwXFM08ygZfk= github.com/microcosm-cc/bluemonday v1.0.27/go.mod h1:jFi9vgW+H7c3V0lb6nR74Ib/DIB5OBs92Dimizgw2cA= github.com/muesli/ansi v0.0.0-20230316100256-276c6243b2f6 h1:ZK8zHtRHOkbHy6Mmr5D264iyp3TiX5OmNcI5cIARiQI= github.com/muesli/ansi v0.0.0-20230316100256-276c6243b2f6/go.mod h1:CJlz5H+gyd6CUWT45Oy4q24RdLyn7Md9Vj2/ldJBSIo= github.com/muesli/cancelreader v0.2.2 h1:3I4Kt4BQjOR54NavqnDogx/MIoWBFa0StPA8ELUXHmA= github.com/muesli/cancelreader v0.2.2/go.mod h1:3XuTXfFS2VjM+HTLZY9Ak0l6eUKfijIfMUZ4EgX0QYo= github.com/muesli/reflow v0.3.0 h1:IFsN6K9NfGtjeggFP+68I4chLZV2yIKsXJFNZ+eWh6s= github.com/muesli/reflow v0.3.0/go.mod h1:pbwTDkVPibjO2kyvBQRBxTWEEGDGq0FlB1BIKtnHY/8= github.com/muesli/termenv v0.16.0 h1:S5AlUN9dENB57rsbnkPyfdGuWIlkmzJjbFf0Tf5FWUc= github.com/muesli/termenv v0.16.0/go.mod h1:ZRfOIKPFDYQoDFF4Olj7/QJbW60Ol/kL1pU3VfY/Cnk= github.com/ncruces/go-sqlite3 v0.25.0 h1:trugKUs98Zwy9KwRr/EUxZHL92LYt7UqcKqAfpGpK+I= github.com/ncruces/go-sqlite3 v0.25.0/go.mod h1:n6Z7036yFilJx04yV0mi5JWaF66rUmXn1It9Ux8dx68= github.com/ncruces/go-strftime v0.1.9 h1:bY0MQC28UADQmHmaF5dgpLmImcShSi2kHU9XLdhx/f4= github.com/ncruces/go-strftime v0.1.9/go.mod h1:Fwc5htZGVVkseilnfgOVb9mKy6w1naJmn9CehxcKcls= github.com/ncruces/julianday v1.0.0 h1:fH0OKwa7NWvniGQtxdJRxAgkBMolni2BjDHaWTxqt7M= github.com/ncruces/julianday v1.0.0/go.mod h1:Dusn2KvZrrovOMJuOt0TNXL6tB7U2E8kvza5fFc9G7g= github.com/openai/openai-go v0.1.0-beta.2 h1:Ra5nCFkbEl9w+UJwAciC4kqnIBUCcJazhmMA0/YN894= github.com/openai/openai-go v0.1.0-beta.2/go.mod h1:g461MYGXEXBVdV5SaR/5tNzNbSfwTBBefwc+LlDCK0Y= github.com/pelletier/go-toml/v2 v2.2.3 h1:YmeHyLY8mFWbdkNWwpr+qIL2bEqT0o95WSdkNHvL12M= github.com/pelletier/go-toml/v2 v2.2.3/go.mod h1:MfCQTFTvCcUyyvvwm1+G6H/jORL20Xlb6rzQu9GuUkc= github.com/pkg/browser v0.0.0-20240102092130-5ac0b6a4141c h1:+mdjkGKdHQG3305AYmdv1U2eRNDiU2ErMBj1gwrq8eQ= github.com/pkg/browser v0.0.0-20240102092130-5ac0b6a4141c/go.mod h1:7rwL4CYBLnjLxUqIJNnCWiEdr3bn6IUYi15bNlnbCCU= github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= github.com/pressly/goose/v3 v3.24.2 h1:c/ie0Gm8rnIVKvnDQ/scHErv46jrDv9b4I0WRcFJzYU= github.com/pressly/goose/v3 v3.24.2/go.mod h1:kjefwFB0eR4w30Td2Gj2Mznyw94vSP+2jJYkOVNbD1k= github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec h1:W09IVJc94icq4NjY3clb7Lk8O1qJ8BdBEF8z0ibU0rE= github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec/go.mod h1:qqbHyh8v60DhA7CoWK5oRCqLrMHRGoxYCSS9EjAz6Eo= github.com/rivo/uniseg v0.1.0/go.mod h1:J6wj4VEh+S6ZtnVlnTBMWIodfgj8LQOQFoIToxlJtxc= github.com/rivo/uniseg v0.2.0/go.mod h1:J6wj4VEh+S6ZtnVlnTBMWIodfgj8LQOQFoIToxlJtxc= github.com/rivo/uniseg v0.4.7 h1:WUdvkW8uEhrYfLC4ZzdpI2ztxP1I582+49Oc5Mq64VQ= github.com/rivo/uniseg v0.4.7/go.mod h1:FN3SvrM+Zdj16jyLfmOkMNblXMcoc8DfTHruCPUcx88= github.com/rogpeppe/go-internal v1.14.1 h1:UQB4HGPB6osV0SQTLymcB4TgvyWu6ZyliaW0tI/otEQ= github.com/rogpeppe/go-internal v1.14.1/go.mod h1:MaRKkUm5W0goXpeCfT7UZI6fk/L7L7so1lCWt35ZSgc= github.com/russross/blackfriday/v2 v2.1.0/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM= github.com/sagikazarmark/locafero v0.7.0 h1:5MqpDsTGNDhY8sGp0Aowyf0qKsPrhewaLSsFaodPcyo= github.com/sagikazarmark/locafero v0.7.0/go.mod h1:2za3Cg5rMaTMoG/2Ulr9AwtFaIppKXTRYnozin4aB5k= github.com/sebdah/goldie/v2 v2.5.3 h1:9ES/mNN+HNUbNWpVAlrzuZ7jE+Nrczbj8uFRjM7624Y= github.com/sebdah/goldie/v2 v2.5.3/go.mod h1:oZ9fp0+se1eapSRjfYbsV/0Hqhbuu3bJVvKI/NNtssI= github.com/sergi/go-diff v1.0.0/go.mod h1:0CfEIISq7TuYL3j771MWULgwwjU+GofnZX9QAmXWZgo= github.com/sergi/go-diff v1.3.1/go.mod h1:aMJSSKb2lpPvRNec0+w3fl7LP9IOFzdc9Pa4NFbPK1I= github.com/sergi/go-diff v1.3.2-0.20230802210424-5b0b94c5c0d3 h1:n661drycOFuPLCN3Uc8sB6B/s6Z4t2xvBgU1htSHuq8= github.com/sergi/go-diff v1.3.2-0.20230802210424-5b0b94c5c0d3/go.mod h1:A0bzQcvG0E7Rwjx0REVgAGH58e96+X0MeOfepqsbeW4= github.com/sethvargo/go-retry v0.3.0 h1:EEt31A35QhrcRZtrYFDTBg91cqZVnFL2navjDrah2SE= github.com/sethvargo/go-retry v0.3.0/go.mod h1:mNX17F0C/HguQMyMyJxcnU471gOZGxCLyYaFyAZraas= github.com/sourcegraph/conc v0.3.0 h1:OQTbbt6P72L20UqAkXXuLOj79LfEanQ+YQFNpLA9ySo= github.com/sourcegraph/conc v0.3.0/go.mod h1:Sdozi7LEKbFPqYX2/J+iBAM6HpqSLTASQIKqDmF7Mt0= github.com/spf13/afero v1.12.0 h1:UcOPyRBYczmFn6yvphxkn9ZEOY65cpwGKb5mL36mrqs= github.com/spf13/afero v1.12.0/go.mod h1:ZTlWwG4/ahT8W7T0WQ5uYmjI9duaLQGy3Q2OAl4sk/4= github.com/spf13/cast v1.7.1 h1:cuNEagBQEHWN1FnbGEjCXL2szYEXqfJPbP2HNUaca9Y= github.com/spf13/cast v1.7.1/go.mod h1:ancEpBxwJDODSW/UG4rDrAqiKolqNNh2DX3mk86cAdo= github.com/spf13/cobra v1.9.1 h1:CXSaggrXdbHK9CF+8ywj8Amf7PBRmPCOJugH954Nnlo= github.com/spf13/cobra v1.9.1/go.mod h1:nDyEzZ8ogv936Cinf6g1RU9MRY64Ir93oCnqb9wxYW0= github.com/spf13/pflag v1.0.6 h1:jFzHGLGAlb3ruxLB8MhbI6A8+AQX/2eW4qeyNZXNp2o= github.com/spf13/pflag v1.0.6/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg= github.com/spf13/viper v1.20.0 h1:zrxIyR3RQIOsarIrgL8+sAvALXul9jeEPa06Y0Ph6vY= github.com/spf13/viper v1.20.0/go.mod h1:P9Mdzt1zoHIG8m2eZQinpiBjo6kCmZSKBClNNqjJvu4= github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4= github.com/stretchr/testify v1.10.0 h1:Xv5erBjTwe/5IxqUQTdXv5kgmIvbHo3QQyRwhJsOfJA= github.com/stretchr/testify v1.10.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY= github.com/subosito/gotenv v1.6.0 h1:9NlTDc1FTs4qu0DDq7AEtTPNw6SVm7uBMsUCUjABIf8= github.com/subosito/gotenv v1.6.0/go.mod h1:Dk4QP5c2W3ibzajGcXpNraDfq2IrhjMIvMSWPKKo0FU= github.com/tetratelabs/wazero v1.9.0 h1:IcZ56OuxrtaEz8UYNRHBrUa9bYeX9oVY93KspZZBf/I= github.com/tetratelabs/wazero v1.9.0/go.mod h1:TSbcXCfFP0L2FGkRPxHphadXPjo1T6W+CseNNY7EkjM= github.com/tidwall/gjson v1.14.2/go.mod h1:/wbyibRr2FHMks5tjHJ5F8dMZh3AcwJEMf5vlfC0lxk= github.com/tidwall/gjson v1.18.0 h1:FIDeeyB800efLX89e5a8Y0BNH+LOngJyGrIWxG2FKQY= github.com/tidwall/gjson v1.18.0/go.mod h1:/wbyibRr2FHMks5tjHJ5F8dMZh3AcwJEMf5vlfC0lxk= github.com/tidwall/match v1.1.1 h1:+Ho715JplO36QYgwN9PGYNhgZvoUSc9X2c80KVTi+GA= github.com/tidwall/match v1.1.1/go.mod h1:eRSPERbgtNPcGhD8UCthc6PmLEQXEWd3PRB5JTxsfmM= github.com/tidwall/pretty v1.2.0/go.mod h1:ITEVvHYasfjBbM0u2Pg8T2nJnzm8xPwvNhhsoaGGjNU= github.com/tidwall/pretty v1.2.1 h1:qjsOFOWWQl+N3RsoF5/ssm1pHmJJwhjlSbZ51I6wMl4= github.com/tidwall/pretty v1.2.1/go.mod h1:ITEVvHYasfjBbM0u2Pg8T2nJnzm8xPwvNhhsoaGGjNU= github.com/tidwall/sjson v1.2.5 h1:kLy8mja+1c9jlljvWTlSazM7cKDRfJuR/bOJhcY5NcY= github.com/tidwall/sjson v1.2.5/go.mod h1:Fvgq9kS/6ociJEDnK0Fk1cpYF4FIW6ZF7LAe+6jwd28= github.com/xo/terminfo v0.0.0-20220910002029-abceb7e1c41e h1:JVG44RsyaB9T2KIHavMF/ppJZNG9ZpyihvCd0w101no= github.com/xo/terminfo v0.0.0-20220910002029-abceb7e1c41e/go.mod h1:RbqR21r5mrJuqunuUZ/Dhy/avygyECGrLceyNeo4LiM= github.com/yosida95/uritemplate/v3 v3.0.2 h1:Ed3Oyj9yrmi9087+NczuL5BwkIc4wvTb5zIM+UJPGz4= github.com/yosida95/uritemplate/v3 v3.0.2/go.mod h1:ILOh0sOhIJR3+L/8afwt/kE++YT040gmv5BQTMR2HP4= github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY= github.com/yuin/goldmark v1.7.1/go.mod h1:uzxRWxtg69N339t3louHJ7+O03ezfj6PlliRlaOzY1E= github.com/yuin/goldmark v1.7.8 h1:iERMLn0/QJeHFhxSt3p6PeN9mGnvIKSpG9YYorDMnic= github.com/yuin/goldmark v1.7.8/go.mod h1:uzxRWxtg69N339t3louHJ7+O03ezfj6PlliRlaOzY1E= github.com/yuin/goldmark-emoji v1.0.5 h1:EMVWyCGPlXJfUXBXpuMu+ii3TIaxbVBnEX9uaDC4cIk= github.com/yuin/goldmark-emoji v1.0.5/go.mod h1:tTkZEbwu5wkPmgTcitqddVxY9osFZiavD+r4AzQrh1U= go.opentelemetry.io/auto/sdk v1.1.0 h1:cH53jehLUN6UFLY71z+NDOiNJqDdPRaXzTel0sJySYA= go.opentelemetry.io/auto/sdk v1.1.0/go.mod h1:3wSPjt5PWp2RhlCcmmOial7AvC4DQqZb7a7wCow3W8A= go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.54.0 h1:TT4fX+nBOA/+LUkobKGW1ydGcn+G3vRw9+g5HwCphpk= go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.54.0/go.mod h1:L7UH0GbB0p47T4Rri3uHjbpCFYrVrwc1I25QhNPiGK8= go.opentelemetry.io/otel v1.35.0 h1:xKWKPxrxB6OtMCbmMY021CqC45J+3Onta9MqjhnusiQ= go.opentelemetry.io/otel v1.35.0/go.mod h1:UEqy8Zp11hpkUrL73gSlELM0DupHoiq72dR+Zqel/+Y= go.opentelemetry.io/otel/metric v1.35.0 h1:0znxYu2SNyuMSQT4Y9WDWej0VpcsxkuklLa4/siN90M= go.opentelemetry.io/otel/metric v1.35.0/go.mod h1:nKVFgxBZ2fReX6IlyW28MgZojkoAkJGaE8CpgeAU3oE= go.opentelemetry.io/otel/sdk v1.34.0 h1:95zS4k/2GOy069d321O8jWgYsW3MzVV+KuSPKp7Wr1A= go.opentelemetry.io/otel/sdk v1.34.0/go.mod h1:0e/pNiaMAqaykJGKbi+tSjWfNNHMTxoC9qANsCzbyxU= go.opentelemetry.io/otel/sdk/metric v1.34.0 h1:5CeK9ujjbFVL5c1PhLuStg1wxA7vQv7ce1EK0Gyvahk= go.opentelemetry.io/otel/sdk/metric v1.34.0/go.mod h1:jQ/r8Ze28zRKoNRdkjCZxfs6YvBTG1+YIqyFVFYec5w= go.opentelemetry.io/otel/trace v1.35.0 h1:dPpEfJu1sDIqruz7BHFG3c7528f6ddfSWfFDVt/xgMs= go.opentelemetry.io/otel/trace v1.35.0/go.mod h1:WUk7DtFp1Aw2MkvqGdwiXYDZZNvA/1J8o6xRXLrIkyc= go.uber.org/multierr v1.11.0 h1:blXXJkSxSSfBVBlC76pxqeO+LN3aDfLQo+309xJstO0= go.uber.org/multierr v1.11.0/go.mod h1:20+QtiLqy0Nd6FdQB9TLXag12DsQkrbs3htMFfDN80Y= golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= golang.org/x/crypto v0.19.0/go.mod h1:Iy9bg/ha4yyC70EfRS8jz+B6ybOBKMaSxLj6P6oBDfU= golang.org/x/crypto v0.22.0/go.mod h1:vr6Su+7cTlO45qkww3VDJlzDn0ctJvRgYbC2NvXHt+M= golang.org/x/crypto v0.23.0/go.mod h1:CKFgDieR+mRhux2Lsu27y0fO304Db0wZe70UKqHu0v8= golang.org/x/crypto v0.37.0 h1:kJNSjF/Xp7kU0iB2Z+9viTPMW4EqqsrywMXLJOOsXSE= golang.org/x/crypto v0.37.0/go.mod h1:vg+k43peMZ0pUMhYmVAWysMK35e6ioLh3wB8ZCAfbVc= golang.org/x/exp v0.0.0-20250305212735-054e65f0b394 h1:nDVHiLt8aIbd/VzvPWN6kSOPE7+F/fNFDSXLVYkE/Iw= golang.org/x/exp v0.0.0-20250305212735-054e65f0b394/go.mod h1:sIifuuw/Yco/y6yb6+bDNfyeQ/MdPUy/hKEMYQV17cM= golang.org/x/image v0.0.0-20191009234506-e7c1f5e7dbb8/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0= golang.org/x/image v0.26.0 h1:4XjIFEZWQmCZi6Wv8BoxsDhRU3RVnLX04dToTDAEPlY= golang.org/x/image v0.26.0/go.mod h1:lcxbMFAovzpnJxzXS3nyL83K27tmqtKzIJpctK8YO5c= golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4= golang.org/x/mod v0.8.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs= golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= golang.org/x/net v0.0.0-20220722155237-a158d28d115b/go.mod h1:XRhObCWvk6IyKnWLug+ECip1KBveYUHfp+8e9klMJ9c= golang.org/x/net v0.6.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs= golang.org/x/net v0.9.0/go.mod h1:d48xBJpPfHeWQsugry2m+kC02ZBRGRgulfHnEXEuWns= golang.org/x/net v0.10.0/go.mod h1:0qNGK6F8kojg2nk9dLZ2mShWaEBan6FAoqfSigmmuDg= golang.org/x/net v0.21.0/go.mod h1:bIjVDfnllIU7BJ2DNgfnXvpSvtn8VRwhlsaeUTyUS44= golang.org/x/net v0.24.0/go.mod h1:2Q7sJY5mzlzWjKtYUEXSlBWCdyaioyXzRB2RtU8KVE8= golang.org/x/net v0.25.0/go.mod h1:JkAGAh7GEvH74S6FOH42FLoXpXbE/aqXSrIQjXgsiwM= golang.org/x/net v0.39.0 h1:ZCu7HMWDxpXpaiKdhzIfaltL9Lp31x/3fCP11bc6/fY= golang.org/x/net v0.39.0/go.mod h1:X7NRbYVEA+ewNkCNyJ513WmMdQ3BineSwVtN2zD/d+E= golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.13.0 h1:AauUjRAJ9OSnvULf/ARrrVywoJDy0YS2AwQ98I37610= golang.org/x/sync v0.13.0/go.mod h1:1dzgHSNfp02xaA81J2MS99Qcpr2w7fw1gpm99rleRqA= golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210809222454-d867a43fc93e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.1.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.7.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.8.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.17.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= golang.org/x/sys v0.19.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= golang.org/x/sys v0.20.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= golang.org/x/sys v0.32.0 h1:s77OFDvIQeibCmezSnk/q6iAfkdiQaJi4VzroCFrN20= golang.org/x/sys v0.32.0/go.mod h1:BJP2sWEmIv4KK5OTEluFJCKSidICx8ciO85XgH3Ak8k= golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= golang.org/x/term v0.5.0/go.mod h1:jMB1sMXY+tzblOD4FWmEbocvup2/aLOaQEp7JmGp78k= golang.org/x/term v0.7.0/go.mod h1:P32HKFT3hSsZrRxla30E9HqToFYAQPCMs/zFMBUFqPY= golang.org/x/term v0.8.0/go.mod h1:xPskH00ivmX89bAKVGSKKtLOWNx2+17Eiy94tnKShWo= golang.org/x/term v0.17.0/go.mod h1:lLRBjIVuehSbZlaOtGMbcMncT+aqLLLmKrsjNrUguwk= golang.org/x/term v0.19.0/go.mod h1:2CuTdWZ7KHSQwUzKva0cbMg6q2DMI3Mmxp+gKJbskEk= golang.org/x/term v0.20.0/go.mod h1:8UkIAJTvZgivsXaD6/pH6U9ecQzZ45awqEOzuCvwpFY= golang.org/x/term v0.31.0 h1:erwDkOK1Msy6offm1mOgvspSkslFnIGsFnxOKoufg3o= golang.org/x/term v0.31.0/go.mod h1:R4BeIy7D95HzImkxGkTW1UQTtP54tio2RyHz7PwK0aw= golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= golang.org/x/text v0.7.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= golang.org/x/text v0.9.0/go.mod h1:e1OnstbJyHTd6l/uOt8jFFHp6TRDWZR/bV3emEE/zU8= golang.org/x/text v0.14.0/go.mod h1:18ZOQIKpY8NJVqYksKHtTdi31H5itFRjB5/qKTNYzSU= golang.org/x/text v0.15.0/go.mod h1:18ZOQIKpY8NJVqYksKHtTdi31H5itFRjB5/qKTNYzSU= golang.org/x/text v0.24.0 h1:dd5Bzh4yt5KYA8f9CJHCP4FB4D51c2c6JvN37xJJkJ0= golang.org/x/text v0.24.0/go.mod h1:L8rBsPeo2pSS+xqN0d5u2ikmjtmoJbDBT1b7nHvFCdU= golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc= golang.org/x/tools v0.6.0/go.mod h1:Xwgl3UAJ/d3gWutnCtw505GrjyAbvKui8lOU390QaIU= golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= google.golang.org/genai v1.3.0 h1:tXhPJF30skOjnnDY7ZnjK3q7IKy4PuAlEA0fk7uEaEI= google.golang.org/genai v1.3.0/go.mod h1:TyfOKRz/QyCaj6f/ZDt505x+YreXnY40l2I6k8TvgqY= google.golang.org/genproto/googleapis/rpc v0.0.0-20250324211829-b45e905df463 h1:e0AIkUUhxyBKh6ssZNrAMeqhA7RKUj42346d1y02i2g= google.golang.org/genproto/googleapis/rpc v0.0.0-20250324211829-b45e905df463/go.mod h1:qQ0YXyHHx3XkvlzUtpXDkS29lDSafHMZBAZDc03LQ3A= google.golang.org/grpc v1.71.0 h1:kF77BGdPTQ4/JZWMlb9VpJ5pa25aqvVqogsxNHHdeBg= google.golang.org/grpc v1.71.0/go.mod h1:H0GRtasmQOh9LkFoCPDu3ZrwUtD1YGE+b2vYBYd/8Ec= google.golang.org/protobuf v1.36.6 h1:z1NpPI8ku2WgiWnf+t9wTPsn6eP1L7ksHUlkfLvd9xY= google.golang.org/protobuf v1.36.6/go.mod h1:jduwjTPXsFjZGTmRluh+L6NjiWu7pchiJ2/5YcXBHnY= gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk= gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q= gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY= gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ= gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= modernc.org/libc v1.61.13 h1:3LRd6ZO1ezsFiX1y+bHd1ipyEHIJKvuprv0sLTBwLW8= modernc.org/libc v1.61.13/go.mod h1:8F/uJWL/3nNil0Lgt1Dpz+GgkApWh04N3el3hxJcA6E= modernc.org/mathutil v1.7.1 h1:GCZVGXdaN8gTqB1Mf/usp1Y/hSqgI2vAGGP4jZMCxOU= modernc.org/mathutil v1.7.1/go.mod h1:4p5IwJITfppl0G4sUEDtCr4DthTaT47/N3aT6MhfgJg= modernc.org/memory v1.9.1 h1:V/Z1solwAVmMW1yttq3nDdZPJqV1rM05Ccq6KMSZ34g= modernc.org/memory v1.9.1/go.mod h1:/JP4VbVC+K5sU2wZi9bHoq2MAkCnrt2r98UGeSK7Mjw= modernc.org/sqlite v1.36.2 h1:vjcSazuoFve9Wm0IVNHgmJECoOXLZM1KfMXbcX2axHA= modernc.org/sqlite v1.36.2/go.mod h1:ADySlx7K4FdY5MaJcEv86hTJ0PjedAloTUuif0YS3ws= ================================================ FILE: install ================================================ #!/usr/bin/env bash set -euo pipefail APP=opencode RED='\033[0;31m' GREEN='\033[0;32m' YELLOW='\033[1;33m' ORANGE='\033[38;2;255;140;0m' NC='\033[0m' # No Color requested_version=${VERSION:-} os=$(uname -s | tr '[:upper:]' '[:lower:]') if [[ "$os" == "darwin" ]]; then os="mac" fi arch=$(uname -m) if [[ "$arch" == "aarch64" ]]; then arch="arm64" fi filename="$APP-$os-$arch.tar.gz" case "$filename" in *"-linux-"*) [[ "$arch" == "x86_64" || "$arch" == "arm64" || "$arch" == "i386" ]] || exit 1 ;; *"-mac-"*) [[ "$arch" == "x86_64" || "$arch" == "arm64" ]] || exit 1 ;; *) echo "${RED}Unsupported OS/Arch: $os/$arch${NC}" exit 1 ;; esac INSTALL_DIR=$HOME/.opencode/bin mkdir -p "$INSTALL_DIR" if [ -z "$requested_version" ]; then url="https://github.com/opencode-ai/opencode/releases/latest/download/$filename" specific_version=$(curl -s https://api.github.com/repos/opencode-ai/opencode/releases/latest | awk -F'"' '/"tag_name": "/ {gsub(/^v/, "", $4); print $4}') if [[ $? -ne 0 ]]; then echo "${RED}Failed to fetch version information${NC}" exit 1 fi else url="https://github.com/opencode-ai/opencode/releases/download/v${requested_version}/$filename" specific_version=$requested_version fi print_message() { local level=$1 local message=$2 local color="" case $level in info) color="${GREEN}" ;; warning) color="${YELLOW}" ;; error) color="${RED}" ;; esac echo -e "${color}${message}${NC}" } check_version() { if command -v opencode >/dev/null 2>&1; then opencode_path=$(which opencode) ## TODO: check if version is installed # installed_version=$(opencode version) installed_version="0.0.1" installed_version=$(echo $installed_version | awk '{print $2}') if [[ "$installed_version" != "$specific_version" ]]; then print_message info "Installed version: ${YELLOW}$installed_version." else print_message info "Version ${YELLOW}$specific_version${GREEN} already installed" exit 0 fi fi } download_and_install() { print_message info "Downloading ${ORANGE}opencode ${GREEN}version: ${YELLOW}$specific_version ${GREEN}..." mkdir -p opencodetmp && cd opencodetmp curl -# -L $url | tar xz mv opencode $INSTALL_DIR cd .. && rm -rf opencodetmp } check_version download_and_install add_to_path() { local config_file=$1 local command=$2 if [[ -w $config_file ]]; then echo -e "\n# opencode" >> "$config_file" echo "$command" >> "$config_file" print_message info "Successfully added ${ORANGE}opencode ${GREEN}to \$PATH in $config_file" else print_message warning "Manually add the directory to $config_file (or similar):" print_message info " $command" fi } XDG_CONFIG_HOME=${XDG_CONFIG_HOME:-$HOME/.config} current_shell=$(basename "$SHELL") case $current_shell in fish) config_files="$HOME/.config/fish/config.fish" ;; zsh) config_files="$HOME/.zshrc $HOME/.zshenv $XDG_CONFIG_HOME/zsh/.zshrc $XDG_CONFIG_HOME/zsh/.zshenv" ;; bash) config_files="$HOME/.bashrc $HOME/.bash_profile $HOME/.profile $XDG_CONFIG_HOME/bash/.bashrc $XDG_CONFIG_HOME/bash/.bash_profile" ;; ash) config_files="$HOME/.ashrc $HOME/.profile /etc/profile" ;; sh) config_files="$HOME/.ashrc $HOME/.profile /etc/profile" ;; *) # Default case if none of the above matches config_files="$HOME/.bashrc $HOME/.bash_profile $XDG_CONFIG_HOME/bash/.bashrc $XDG_CONFIG_HOME/bash/.bash_profile" ;; esac config_file="" for file in $config_files; do if [[ -f $file ]]; then config_file=$file break fi done if [[ -z $config_file ]]; then print_message error "No config file found for $current_shell. Checked files: ${config_files[@]}" exit 1 fi if [[ ":$PATH:" != *":$INSTALL_DIR:"* ]]; then case $current_shell in fish) add_to_path "$config_file" "fish_add_path $INSTALL_DIR" ;; zsh) add_to_path "$config_file" "export PATH=$INSTALL_DIR:\$PATH" ;; bash) add_to_path "$config_file" "export PATH=$INSTALL_DIR:\$PATH" ;; ash) add_to_path "$config_file" "export PATH=$INSTALL_DIR:\$PATH" ;; sh) add_to_path "$config_file" "export PATH=$INSTALL_DIR:\$PATH" ;; *) print_message warning "Manually add the directory to $config_file (or similar):" print_message info " export PATH=$INSTALL_DIR:\$PATH" ;; esac fi if [ -n "${GITHUB_ACTIONS-}" ] && [ "${GITHUB_ACTIONS}" == "true" ]; then echo "$INSTALL_DIR" >> $GITHUB_PATH print_message info "Added $INSTALL_DIR to \$GITHUB_PATH" fi ================================================ FILE: internal/app/app.go ================================================ package app import ( "context" "database/sql" "errors" "fmt" "maps" "sync" "time" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/db" "github.com/opencode-ai/opencode/internal/format" "github.com/opencode-ai/opencode/internal/history" "github.com/opencode-ai/opencode/internal/llm/agent" "github.com/opencode-ai/opencode/internal/logging" "github.com/opencode-ai/opencode/internal/lsp" "github.com/opencode-ai/opencode/internal/message" "github.com/opencode-ai/opencode/internal/permission" "github.com/opencode-ai/opencode/internal/session" "github.com/opencode-ai/opencode/internal/tui/theme" ) type App struct { Sessions session.Service Messages message.Service History history.Service Permissions permission.Service CoderAgent agent.Service LSPClients map[string]*lsp.Client clientsMutex sync.RWMutex watcherCancelFuncs []context.CancelFunc cancelFuncsMutex sync.Mutex watcherWG sync.WaitGroup } func New(ctx context.Context, conn *sql.DB) (*App, error) { q := db.New(conn) sessions := session.NewService(q) messages := message.NewService(q) files := history.NewService(q, conn) app := &App{ Sessions: sessions, Messages: messages, History: files, Permissions: permission.NewPermissionService(), LSPClients: make(map[string]*lsp.Client), } // Initialize theme based on configuration app.initTheme() // Initialize LSP clients in the background go app.initLSPClients(ctx) var err error app.CoderAgent, err = agent.NewAgent( config.AgentCoder, app.Sessions, app.Messages, agent.CoderAgentTools( app.Permissions, app.Sessions, app.Messages, app.History, app.LSPClients, ), ) if err != nil { logging.Error("Failed to create coder agent", err) return nil, err } return app, nil } // initTheme sets the application theme based on the configuration func (app *App) initTheme() { cfg := config.Get() if cfg == nil || cfg.TUI.Theme == "" { return // Use default theme } // Try to set the theme from config err := theme.SetTheme(cfg.TUI.Theme) if err != nil { logging.Warn("Failed to set theme from config, using default theme", "theme", cfg.TUI.Theme, "error", err) } else { logging.Debug("Set theme from config", "theme", cfg.TUI.Theme) } } // RunNonInteractive handles the execution flow when a prompt is provided via CLI flag. func (a *App) RunNonInteractive(ctx context.Context, prompt string, outputFormat string, quiet bool) error { logging.Info("Running in non-interactive mode") // Start spinner if not in quiet mode var spinner *format.Spinner if !quiet { spinner = format.NewSpinner("Thinking...") spinner.Start() defer spinner.Stop() } const maxPromptLengthForTitle = 100 titlePrefix := "Non-interactive: " var titleSuffix string if len(prompt) > maxPromptLengthForTitle { titleSuffix = prompt[:maxPromptLengthForTitle] + "..." } else { titleSuffix = prompt } title := titlePrefix + titleSuffix sess, err := a.Sessions.Create(ctx, title) if err != nil { return fmt.Errorf("failed to create session for non-interactive mode: %w", err) } logging.Info("Created session for non-interactive run", "session_id", sess.ID) // Automatically approve all permission requests for this non-interactive session a.Permissions.AutoApproveSession(sess.ID) done, err := a.CoderAgent.Run(ctx, sess.ID, prompt) if err != nil { return fmt.Errorf("failed to start agent processing stream: %w", err) } result := <-done if result.Error != nil { if errors.Is(result.Error, context.Canceled) || errors.Is(result.Error, agent.ErrRequestCancelled) { logging.Info("Agent processing cancelled", "session_id", sess.ID) return nil } return fmt.Errorf("agent processing failed: %w", result.Error) } // Stop spinner before printing output if !quiet && spinner != nil { spinner.Stop() } // Get the text content from the response content := "No content available" if result.Message.Content().String() != "" { content = result.Message.Content().String() } fmt.Println(format.FormatOutput(content, outputFormat)) logging.Info("Non-interactive run completed", "session_id", sess.ID) return nil } // Shutdown performs a clean shutdown of the application func (app *App) Shutdown() { // Cancel all watcher goroutines app.cancelFuncsMutex.Lock() for _, cancel := range app.watcherCancelFuncs { cancel() } app.cancelFuncsMutex.Unlock() app.watcherWG.Wait() // Perform additional cleanup for LSP clients app.clientsMutex.RLock() clients := make(map[string]*lsp.Client, len(app.LSPClients)) maps.Copy(clients, app.LSPClients) app.clientsMutex.RUnlock() for name, client := range clients { shutdownCtx, cancel := context.WithTimeout(context.Background(), 5*time.Second) if err := client.Shutdown(shutdownCtx); err != nil { logging.Error("Failed to shutdown LSP client", "name", name, "error", err) } cancel() } } ================================================ FILE: internal/app/lsp.go ================================================ package app import ( "context" "time" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/logging" "github.com/opencode-ai/opencode/internal/lsp" "github.com/opencode-ai/opencode/internal/lsp/watcher" ) func (app *App) initLSPClients(ctx context.Context) { cfg := config.Get() // Initialize LSP clients for name, clientConfig := range cfg.LSP { // Start each client initialization in its own goroutine go app.createAndStartLSPClient(ctx, name, clientConfig.Command, clientConfig.Args...) } logging.Info("LSP clients initialization started in background") } // createAndStartLSPClient creates a new LSP client, initializes it, and starts its workspace watcher func (app *App) createAndStartLSPClient(ctx context.Context, name string, command string, args ...string) { // Create a specific context for initialization with a timeout logging.Info("Creating LSP client", "name", name, "command", command, "args", args) // Create the LSP client lspClient, err := lsp.NewClient(ctx, command, args...) if err != nil { logging.Error("Failed to create LSP client for", name, err) return } // Create a longer timeout for initialization (some servers take time to start) initCtx, cancel := context.WithTimeout(ctx, 30*time.Second) defer cancel() // Initialize with the initialization context _, err = lspClient.InitializeLSPClient(initCtx, config.WorkingDirectory()) if err != nil { logging.Error("Initialize failed", "name", name, "error", err) // Clean up the client to prevent resource leaks lspClient.Close() return } // Wait for the server to be ready if err := lspClient.WaitForServerReady(initCtx); err != nil { logging.Error("Server failed to become ready", "name", name, "error", err) // We'll continue anyway, as some functionality might still work lspClient.SetServerState(lsp.StateError) } else { logging.Info("LSP server is ready", "name", name) lspClient.SetServerState(lsp.StateReady) } logging.Info("LSP client initialized", "name", name) // Create a child context that can be canceled when the app is shutting down watchCtx, cancelFunc := context.WithCancel(ctx) // Create a context with the server name for better identification watchCtx = context.WithValue(watchCtx, "serverName", name) // Create the workspace watcher workspaceWatcher := watcher.NewWorkspaceWatcher(lspClient) // Store the cancel function to be called during cleanup app.cancelFuncsMutex.Lock() app.watcherCancelFuncs = append(app.watcherCancelFuncs, cancelFunc) app.cancelFuncsMutex.Unlock() // Add the watcher to a WaitGroup to track active goroutines app.watcherWG.Add(1) // Add to map with mutex protection before starting goroutine app.clientsMutex.Lock() app.LSPClients[name] = lspClient app.clientsMutex.Unlock() go app.runWorkspaceWatcher(watchCtx, name, workspaceWatcher) } // runWorkspaceWatcher executes the workspace watcher for an LSP client func (app *App) runWorkspaceWatcher(ctx context.Context, name string, workspaceWatcher *watcher.WorkspaceWatcher) { defer app.watcherWG.Done() defer logging.RecoverPanic("LSP-"+name, func() { // Try to restart the client app.restartLSPClient(ctx, name) }) workspaceWatcher.WatchWorkspace(ctx, config.WorkingDirectory()) logging.Info("Workspace watcher stopped", "client", name) } // restartLSPClient attempts to restart a crashed or failed LSP client func (app *App) restartLSPClient(ctx context.Context, name string) { // Get the original configuration cfg := config.Get() clientConfig, exists := cfg.LSP[name] if !exists { logging.Error("Cannot restart client, configuration not found", "client", name) return } // Clean up the old client if it exists app.clientsMutex.Lock() oldClient, exists := app.LSPClients[name] if exists { delete(app.LSPClients, name) // Remove from map before potentially slow shutdown } app.clientsMutex.Unlock() if exists && oldClient != nil { // Try to shut it down gracefully, but don't block on errors shutdownCtx, cancel := context.WithTimeout(context.Background(), 5*time.Second) _ = oldClient.Shutdown(shutdownCtx) cancel() } // Create a new client using the shared function app.createAndStartLSPClient(ctx, name, clientConfig.Command, clientConfig.Args...) logging.Info("Successfully restarted LSP client", "client", name) } ================================================ FILE: internal/completions/files-folders.go ================================================ package completions import ( "bytes" "fmt" "os/exec" "path/filepath" "github.com/lithammer/fuzzysearch/fuzzy" "github.com/opencode-ai/opencode/internal/fileutil" "github.com/opencode-ai/opencode/internal/logging" "github.com/opencode-ai/opencode/internal/tui/components/dialog" ) type filesAndFoldersContextGroup struct { prefix string } func (cg *filesAndFoldersContextGroup) GetId() string { return cg.prefix } func (cg *filesAndFoldersContextGroup) GetEntry() dialog.CompletionItemI { return dialog.NewCompletionItem(dialog.CompletionItem{ Title: "Files & Folders", Value: "files", }) } func processNullTerminatedOutput(outputBytes []byte) []string { if len(outputBytes) > 0 && outputBytes[len(outputBytes)-1] == 0 { outputBytes = outputBytes[:len(outputBytes)-1] } if len(outputBytes) == 0 { return []string{} } split := bytes.Split(outputBytes, []byte{0}) matches := make([]string, 0, len(split)) for _, p := range split { if len(p) == 0 { continue } path := string(p) path = filepath.Join(".", path) if !fileutil.SkipHidden(path) { matches = append(matches, path) } } return matches } func (cg *filesAndFoldersContextGroup) getFiles(query string) ([]string, error) { cmdRg := fileutil.GetRgCmd("") // No glob pattern for this use case cmdFzf := fileutil.GetFzfCmd(query) var matches []string // Case 1: Both rg and fzf available if cmdRg != nil && cmdFzf != nil { rgPipe, err := cmdRg.StdoutPipe() if err != nil { return nil, fmt.Errorf("failed to get rg stdout pipe: %w", err) } defer rgPipe.Close() cmdFzf.Stdin = rgPipe var fzfOut bytes.Buffer var fzfErr bytes.Buffer cmdFzf.Stdout = &fzfOut cmdFzf.Stderr = &fzfErr if err := cmdFzf.Start(); err != nil { return nil, fmt.Errorf("failed to start fzf: %w", err) } errRg := cmdRg.Run() errFzf := cmdFzf.Wait() if errRg != nil { logging.Warn(fmt.Sprintf("rg command failed during pipe: %v", errRg)) } if errFzf != nil { if exitErr, ok := errFzf.(*exec.ExitError); ok && exitErr.ExitCode() == 1 { return []string{}, nil // No matches from fzf } return nil, fmt.Errorf("fzf command failed: %w\nStderr: %s", errFzf, fzfErr.String()) } matches = processNullTerminatedOutput(fzfOut.Bytes()) // Case 2: Only rg available } else if cmdRg != nil { logging.Debug("Using Ripgrep with fuzzy match fallback for file completions") var rgOut bytes.Buffer var rgErr bytes.Buffer cmdRg.Stdout = &rgOut cmdRg.Stderr = &rgErr if err := cmdRg.Run(); err != nil { return nil, fmt.Errorf("rg command failed: %w\nStderr: %s", err, rgErr.String()) } allFiles := processNullTerminatedOutput(rgOut.Bytes()) matches = fuzzy.Find(query, allFiles) // Case 3: Only fzf available } else if cmdFzf != nil { logging.Debug("Using FZF with doublestar fallback for file completions") files, _, err := fileutil.GlobWithDoublestar("**/*", ".", 0) if err != nil { return nil, fmt.Errorf("failed to list files for fzf: %w", err) } allFiles := make([]string, 0, len(files)) for _, file := range files { if !fileutil.SkipHidden(file) { allFiles = append(allFiles, file) } } var fzfIn bytes.Buffer for _, file := range allFiles { fzfIn.WriteString(file) fzfIn.WriteByte(0) } cmdFzf.Stdin = &fzfIn var fzfOut bytes.Buffer var fzfErr bytes.Buffer cmdFzf.Stdout = &fzfOut cmdFzf.Stderr = &fzfErr if err := cmdFzf.Run(); err != nil { if exitErr, ok := err.(*exec.ExitError); ok && exitErr.ExitCode() == 1 { return []string{}, nil } return nil, fmt.Errorf("fzf command failed: %w\nStderr: %s", err, fzfErr.String()) } matches = processNullTerminatedOutput(fzfOut.Bytes()) // Case 4: Fallback to doublestar with fuzzy match } else { logging.Debug("Using doublestar with fuzzy match for file completions") allFiles, _, err := fileutil.GlobWithDoublestar("**/*", ".", 0) if err != nil { return nil, fmt.Errorf("failed to glob files: %w", err) } filteredFiles := make([]string, 0, len(allFiles)) for _, file := range allFiles { if !fileutil.SkipHidden(file) { filteredFiles = append(filteredFiles, file) } } matches = fuzzy.Find(query, filteredFiles) } return matches, nil } func (cg *filesAndFoldersContextGroup) GetChildEntries(query string) ([]dialog.CompletionItemI, error) { matches, err := cg.getFiles(query) if err != nil { return nil, err } items := make([]dialog.CompletionItemI, 0, len(matches)) for _, file := range matches { item := dialog.NewCompletionItem(dialog.CompletionItem{ Title: file, Value: file, }) items = append(items, item) } return items, nil } func NewFileAndFolderContextGroup() dialog.CompletionProvider { return &filesAndFoldersContextGroup{ prefix: "file", } } ================================================ FILE: internal/config/config.go ================================================ // Package config manages application configuration from various sources. package config import ( "encoding/json" "fmt" "log/slog" "os" "path/filepath" "runtime" "strings" "github.com/opencode-ai/opencode/internal/llm/models" "github.com/opencode-ai/opencode/internal/logging" "github.com/spf13/viper" ) // MCPType defines the type of MCP (Model Control Protocol) server. type MCPType string // Supported MCP types const ( MCPStdio MCPType = "stdio" MCPSse MCPType = "sse" ) // MCPServer defines the configuration for a Model Control Protocol server. type MCPServer struct { Command string `json:"command"` Env []string `json:"env"` Args []string `json:"args"` Type MCPType `json:"type"` URL string `json:"url"` Headers map[string]string `json:"headers"` } type AgentName string const ( AgentCoder AgentName = "coder" AgentSummarizer AgentName = "summarizer" AgentTask AgentName = "task" AgentTitle AgentName = "title" ) // Agent defines configuration for different LLM models and their token limits. type Agent struct { Model models.ModelID `json:"model"` MaxTokens int64 `json:"maxTokens"` ReasoningEffort string `json:"reasoningEffort"` // For openai models low,medium,heigh } // Provider defines configuration for an LLM provider. type Provider struct { APIKey string `json:"apiKey"` Disabled bool `json:"disabled"` } // Data defines storage configuration. type Data struct { Directory string `json:"directory,omitempty"` } // LSPConfig defines configuration for Language Server Protocol integration. type LSPConfig struct { Disabled bool `json:"enabled"` Command string `json:"command"` Args []string `json:"args"` Options any `json:"options"` } // TUIConfig defines the configuration for the Terminal User Interface. type TUIConfig struct { Theme string `json:"theme,omitempty"` } // ShellConfig defines the configuration for the shell used by the bash tool. type ShellConfig struct { Path string `json:"path,omitempty"` Args []string `json:"args,omitempty"` } // Config is the main configuration structure for the application. type Config struct { Data Data `json:"data"` WorkingDir string `json:"wd,omitempty"` MCPServers map[string]MCPServer `json:"mcpServers,omitempty"` Providers map[models.ModelProvider]Provider `json:"providers,omitempty"` LSP map[string]LSPConfig `json:"lsp,omitempty"` Agents map[AgentName]Agent `json:"agents,omitempty"` Debug bool `json:"debug,omitempty"` DebugLSP bool `json:"debugLSP,omitempty"` ContextPaths []string `json:"contextPaths,omitempty"` TUI TUIConfig `json:"tui"` Shell ShellConfig `json:"shell,omitempty"` AutoCompact bool `json:"autoCompact,omitempty"` } // Application constants const ( defaultDataDirectory = ".opencode" defaultLogLevel = "info" appName = "opencode" MaxTokensFallbackDefault = 4096 ) var defaultContextPaths = []string{ ".github/copilot-instructions.md", ".cursorrules", ".cursor/rules/", "CLAUDE.md", "CLAUDE.local.md", "opencode.md", "opencode.local.md", "OpenCode.md", "OpenCode.local.md", "OPENCODE.md", "OPENCODE.local.md", } // Global configuration instance var cfg *Config // Load initializes the configuration from environment variables and config files. // If debug is true, debug mode is enabled and log level is set to debug. // It returns an error if configuration loading fails. func Load(workingDir string, debug bool) (*Config, error) { if cfg != nil { return cfg, nil } cfg = &Config{ WorkingDir: workingDir, MCPServers: make(map[string]MCPServer), Providers: make(map[models.ModelProvider]Provider), LSP: make(map[string]LSPConfig), } configureViper() setDefaults(debug) // Read global config if err := readConfig(viper.ReadInConfig()); err != nil { return cfg, err } // Load and merge local config mergeLocalConfig(workingDir) setProviderDefaults() // Apply configuration to the struct if err := viper.Unmarshal(cfg); err != nil { return cfg, fmt.Errorf("failed to unmarshal config: %w", err) } applyDefaultValues() defaultLevel := slog.LevelInfo if cfg.Debug { defaultLevel = slog.LevelDebug } if os.Getenv("OPENCODE_DEV_DEBUG") == "true" { loggingFile := fmt.Sprintf("%s/%s", cfg.Data.Directory, "debug.log") messagesPath := fmt.Sprintf("%s/%s", cfg.Data.Directory, "messages") // if file does not exist create it if _, err := os.Stat(loggingFile); os.IsNotExist(err) { if err := os.MkdirAll(cfg.Data.Directory, 0o755); err != nil { return cfg, fmt.Errorf("failed to create directory: %w", err) } if _, err := os.Create(loggingFile); err != nil { return cfg, fmt.Errorf("failed to create log file: %w", err) } } if _, err := os.Stat(messagesPath); os.IsNotExist(err) { if err := os.MkdirAll(messagesPath, 0o756); err != nil { return cfg, fmt.Errorf("failed to create directory: %w", err) } } logging.MessageDir = messagesPath sloggingFileWriter, err := os.OpenFile(loggingFile, os.O_CREATE|os.O_WRONLY|os.O_APPEND, 0o666) if err != nil { return cfg, fmt.Errorf("failed to open log file: %w", err) } // Configure logger logger := slog.New(slog.NewTextHandler(sloggingFileWriter, &slog.HandlerOptions{ Level: defaultLevel, })) slog.SetDefault(logger) } else { // Configure logger logger := slog.New(slog.NewTextHandler(logging.NewWriter(), &slog.HandlerOptions{ Level: defaultLevel, })) slog.SetDefault(logger) } // Validate configuration if err := Validate(); err != nil { return cfg, fmt.Errorf("config validation failed: %w", err) } if cfg.Agents == nil { cfg.Agents = make(map[AgentName]Agent) } // Override the max tokens for title agent cfg.Agents[AgentTitle] = Agent{ Model: cfg.Agents[AgentTitle].Model, MaxTokens: 80, } return cfg, nil } // configureViper sets up viper's configuration paths and environment variables. func configureViper() { viper.SetConfigName(fmt.Sprintf(".%s", appName)) viper.SetConfigType("json") viper.AddConfigPath("$HOME") viper.AddConfigPath(fmt.Sprintf("$XDG_CONFIG_HOME/%s", appName)) viper.AddConfigPath(fmt.Sprintf("$HOME/.config/%s", appName)) viper.SetEnvPrefix(strings.ToUpper(appName)) viper.AutomaticEnv() } // setDefaults configures default values for configuration options. func setDefaults(debug bool) { viper.SetDefault("data.directory", defaultDataDirectory) viper.SetDefault("contextPaths", defaultContextPaths) viper.SetDefault("tui.theme", "opencode") viper.SetDefault("autoCompact", true) // Set default shell from environment or fallback to /bin/bash shellPath := os.Getenv("SHELL") if shellPath == "" { shellPath = "/bin/bash" } viper.SetDefault("shell.path", shellPath) viper.SetDefault("shell.args", []string{"-l"}) if debug { viper.SetDefault("debug", true) viper.Set("log.level", "debug") } else { viper.SetDefault("debug", false) viper.SetDefault("log.level", defaultLogLevel) } } // setProviderDefaults configures LLM provider defaults based on provider provided by // environment variables and configuration file. func setProviderDefaults() { // Set all API keys we can find in the environment // Note: Viper does not default if the json apiKey is "" if apiKey := os.Getenv("ANTHROPIC_API_KEY"); apiKey != "" { viper.SetDefault("providers.anthropic.apiKey", apiKey) } if apiKey := os.Getenv("OPENAI_API_KEY"); apiKey != "" { viper.SetDefault("providers.openai.apiKey", apiKey) } if apiKey := os.Getenv("GEMINI_API_KEY"); apiKey != "" { viper.SetDefault("providers.gemini.apiKey", apiKey) } if apiKey := os.Getenv("GROQ_API_KEY"); apiKey != "" { viper.SetDefault("providers.groq.apiKey", apiKey) } if apiKey := os.Getenv("OPENROUTER_API_KEY"); apiKey != "" { viper.SetDefault("providers.openrouter.apiKey", apiKey) } if apiKey := os.Getenv("XAI_API_KEY"); apiKey != "" { viper.SetDefault("providers.xai.apiKey", apiKey) } if apiKey := os.Getenv("AZURE_OPENAI_ENDPOINT"); apiKey != "" { // api-key may be empty when using Entra ID credentials – that's okay viper.SetDefault("providers.azure.apiKey", os.Getenv("AZURE_OPENAI_API_KEY")) } if apiKey, err := LoadGitHubToken(); err == nil && apiKey != "" { viper.SetDefault("providers.copilot.apiKey", apiKey) if viper.GetString("providers.copilot.apiKey") == "" { viper.Set("providers.copilot.apiKey", apiKey) } } // Use this order to set the default models // 1. Copilot // 2. Anthropic // 3. OpenAI // 4. Google Gemini // 5. Groq // 6. OpenRouter // 7. AWS Bedrock // 8. Azure // 9. Google Cloud VertexAI // copilot configuration if key := viper.GetString("providers.copilot.apiKey"); strings.TrimSpace(key) != "" { viper.SetDefault("agents.coder.model", models.CopilotGPT4o) viper.SetDefault("agents.summarizer.model", models.CopilotGPT4o) viper.SetDefault("agents.task.model", models.CopilotGPT4o) viper.SetDefault("agents.title.model", models.CopilotGPT4o) return } // Anthropic configuration if key := viper.GetString("providers.anthropic.apiKey"); strings.TrimSpace(key) != "" { viper.SetDefault("agents.coder.model", models.Claude4Sonnet) viper.SetDefault("agents.summarizer.model", models.Claude4Sonnet) viper.SetDefault("agents.task.model", models.Claude4Sonnet) viper.SetDefault("agents.title.model", models.Claude4Sonnet) return } // OpenAI configuration if key := viper.GetString("providers.openai.apiKey"); strings.TrimSpace(key) != "" { viper.SetDefault("agents.coder.model", models.GPT41) viper.SetDefault("agents.summarizer.model", models.GPT41) viper.SetDefault("agents.task.model", models.GPT41Mini) viper.SetDefault("agents.title.model", models.GPT41Mini) return } // Google Gemini configuration if key := viper.GetString("providers.gemini.apiKey"); strings.TrimSpace(key) != "" { viper.SetDefault("agents.coder.model", models.Gemini25) viper.SetDefault("agents.summarizer.model", models.Gemini25) viper.SetDefault("agents.task.model", models.Gemini25Flash) viper.SetDefault("agents.title.model", models.Gemini25Flash) return } // Groq configuration if key := viper.GetString("providers.groq.apiKey"); strings.TrimSpace(key) != "" { viper.SetDefault("agents.coder.model", models.QWENQwq) viper.SetDefault("agents.summarizer.model", models.QWENQwq) viper.SetDefault("agents.task.model", models.QWENQwq) viper.SetDefault("agents.title.model", models.QWENQwq) return } // OpenRouter configuration if key := viper.GetString("providers.openrouter.apiKey"); strings.TrimSpace(key) != "" { viper.SetDefault("agents.coder.model", models.OpenRouterClaude37Sonnet) viper.SetDefault("agents.summarizer.model", models.OpenRouterClaude37Sonnet) viper.SetDefault("agents.task.model", models.OpenRouterClaude37Sonnet) viper.SetDefault("agents.title.model", models.OpenRouterClaude35Haiku) return } // XAI configuration if key := viper.GetString("providers.xai.apiKey"); strings.TrimSpace(key) != "" { viper.SetDefault("agents.coder.model", models.XAIGrok3Beta) viper.SetDefault("agents.summarizer.model", models.XAIGrok3Beta) viper.SetDefault("agents.task.model", models.XAIGrok3Beta) viper.SetDefault("agents.title.model", models.XAiGrok3MiniFastBeta) return } // AWS Bedrock configuration if hasAWSCredentials() { viper.SetDefault("agents.coder.model", models.BedrockClaude37Sonnet) viper.SetDefault("agents.summarizer.model", models.BedrockClaude37Sonnet) viper.SetDefault("agents.task.model", models.BedrockClaude37Sonnet) viper.SetDefault("agents.title.model", models.BedrockClaude37Sonnet) return } // Azure OpenAI configuration if os.Getenv("AZURE_OPENAI_ENDPOINT") != "" { viper.SetDefault("agents.coder.model", models.AzureGPT41) viper.SetDefault("agents.summarizer.model", models.AzureGPT41) viper.SetDefault("agents.task.model", models.AzureGPT41Mini) viper.SetDefault("agents.title.model", models.AzureGPT41Mini) return } // Google Cloud VertexAI configuration if hasVertexAICredentials() { viper.SetDefault("agents.coder.model", models.VertexAIGemini25) viper.SetDefault("agents.summarizer.model", models.VertexAIGemini25) viper.SetDefault("agents.task.model", models.VertexAIGemini25Flash) viper.SetDefault("agents.title.model", models.VertexAIGemini25Flash) return } } // hasAWSCredentials checks if AWS credentials are available in the environment. func hasAWSCredentials() bool { // Check for explicit AWS credentials if os.Getenv("AWS_ACCESS_KEY_ID") != "" && os.Getenv("AWS_SECRET_ACCESS_KEY") != "" { return true } // Check for AWS profile if os.Getenv("AWS_PROFILE") != "" || os.Getenv("AWS_DEFAULT_PROFILE") != "" { return true } // Check for AWS region if os.Getenv("AWS_REGION") != "" || os.Getenv("AWS_DEFAULT_REGION") != "" { return true } // Check if running on EC2 with instance profile if os.Getenv("AWS_CONTAINER_CREDENTIALS_RELATIVE_URI") != "" || os.Getenv("AWS_CONTAINER_CREDENTIALS_FULL_URI") != "" { return true } return false } // hasVertexAICredentials checks if VertexAI credentials are available in the environment. func hasVertexAICredentials() bool { // Check for explicit VertexAI parameters if os.Getenv("VERTEXAI_PROJECT") != "" && os.Getenv("VERTEXAI_LOCATION") != "" { return true } // Check for Google Cloud project and location if os.Getenv("GOOGLE_CLOUD_PROJECT") != "" && (os.Getenv("GOOGLE_CLOUD_REGION") != "" || os.Getenv("GOOGLE_CLOUD_LOCATION") != "") { return true } return false } func hasCopilotCredentials() bool { // Check for explicit Copilot parameters if token, _ := LoadGitHubToken(); token != "" { return true } return false } // readConfig handles the result of reading a configuration file. func readConfig(err error) error { if err == nil { return nil } // It's okay if the config file doesn't exist if _, ok := err.(viper.ConfigFileNotFoundError); ok { return nil } return fmt.Errorf("failed to read config: %w", err) } // mergeLocalConfig loads and merges configuration from the local directory. func mergeLocalConfig(workingDir string) { local := viper.New() local.SetConfigName(fmt.Sprintf(".%s", appName)) local.SetConfigType("json") local.AddConfigPath(workingDir) // Merge local config if it exists if err := local.ReadInConfig(); err == nil { viper.MergeConfigMap(local.AllSettings()) } } // applyDefaultValues sets default values for configuration fields that need processing. func applyDefaultValues() { // Set default MCP type if not specified for k, v := range cfg.MCPServers { if v.Type == "" { v.Type = MCPStdio cfg.MCPServers[k] = v } } } // It validates model IDs and providers, ensuring they are supported. func validateAgent(cfg *Config, name AgentName, agent Agent) error { // Check if model exists // TODO: If a copilot model is specified, but model is not found, // it might be new model. The https://api.githubcopilot.com/models // endpoint should be queried to validate if the model is supported. model, modelExists := models.SupportedModels[agent.Model] if !modelExists { logging.Warn("unsupported model configured, reverting to default", "agent", name, "configured_model", agent.Model) // Set default model based on available providers if setDefaultModelForAgent(name) { logging.Info("set default model for agent", "agent", name, "model", cfg.Agents[name].Model) } else { return fmt.Errorf("no valid provider available for agent %s", name) } return nil } // Check if provider for the model is configured provider := model.Provider providerCfg, providerExists := cfg.Providers[provider] if !providerExists { // Provider not configured, check if we have environment variables apiKey := getProviderAPIKey(provider) if apiKey == "" { logging.Warn("provider not configured for model, reverting to default", "agent", name, "model", agent.Model, "provider", provider) // Set default model based on available providers if setDefaultModelForAgent(name) { logging.Info("set default model for agent", "agent", name, "model", cfg.Agents[name].Model) } else { return fmt.Errorf("no valid provider available for agent %s", name) } } else { // Add provider with API key from environment cfg.Providers[provider] = Provider{ APIKey: apiKey, } logging.Info("added provider from environment", "provider", provider) } } else if providerCfg.Disabled || providerCfg.APIKey == "" { // Provider is disabled or has no API key logging.Warn("provider is disabled or has no API key, reverting to default", "agent", name, "model", agent.Model, "provider", provider) // Set default model based on available providers if setDefaultModelForAgent(name) { logging.Info("set default model for agent", "agent", name, "model", cfg.Agents[name].Model) } else { return fmt.Errorf("no valid provider available for agent %s", name) } } // Validate max tokens if agent.MaxTokens <= 0 { logging.Warn("invalid max tokens, setting to default", "agent", name, "model", agent.Model, "max_tokens", agent.MaxTokens) // Update the agent with default max tokens updatedAgent := cfg.Agents[name] if model.DefaultMaxTokens > 0 { updatedAgent.MaxTokens = model.DefaultMaxTokens } else { updatedAgent.MaxTokens = MaxTokensFallbackDefault } cfg.Agents[name] = updatedAgent } else if model.ContextWindow > 0 && agent.MaxTokens > model.ContextWindow/2 { // Ensure max tokens doesn't exceed half the context window (reasonable limit) logging.Warn("max tokens exceeds half the context window, adjusting", "agent", name, "model", agent.Model, "max_tokens", agent.MaxTokens, "context_window", model.ContextWindow) // Update the agent with adjusted max tokens updatedAgent := cfg.Agents[name] updatedAgent.MaxTokens = model.ContextWindow / 2 cfg.Agents[name] = updatedAgent } // Validate reasoning effort for models that support reasoning if model.CanReason && provider == models.ProviderOpenAI || provider == models.ProviderLocal { if agent.ReasoningEffort == "" { // Set default reasoning effort for models that support it logging.Info("setting default reasoning effort for model that supports reasoning", "agent", name, "model", agent.Model) // Update the agent with default reasoning effort updatedAgent := cfg.Agents[name] updatedAgent.ReasoningEffort = "medium" cfg.Agents[name] = updatedAgent } else { // Check if reasoning effort is valid (low, medium, high) effort := strings.ToLower(agent.ReasoningEffort) if effort != "low" && effort != "medium" && effort != "high" { logging.Warn("invalid reasoning effort, setting to medium", "agent", name, "model", agent.Model, "reasoning_effort", agent.ReasoningEffort) // Update the agent with valid reasoning effort updatedAgent := cfg.Agents[name] updatedAgent.ReasoningEffort = "medium" cfg.Agents[name] = updatedAgent } } } else if !model.CanReason && agent.ReasoningEffort != "" { // Model doesn't support reasoning but reasoning effort is set logging.Warn("model doesn't support reasoning but reasoning effort is set, ignoring", "agent", name, "model", agent.Model, "reasoning_effort", agent.ReasoningEffort) // Update the agent to remove reasoning effort updatedAgent := cfg.Agents[name] updatedAgent.ReasoningEffort = "" cfg.Agents[name] = updatedAgent } return nil } // Validate checks if the configuration is valid and applies defaults where needed. func Validate() error { if cfg == nil { return fmt.Errorf("config not loaded") } // Validate agent models for name, agent := range cfg.Agents { if err := validateAgent(cfg, name, agent); err != nil { return err } } // Validate providers for provider, providerCfg := range cfg.Providers { if providerCfg.APIKey == "" && !providerCfg.Disabled { fmt.Printf("provider has no API key, marking as disabled %s", provider) logging.Warn("provider has no API key, marking as disabled", "provider", provider) providerCfg.Disabled = true cfg.Providers[provider] = providerCfg } } // Validate LSP configurations for language, lspConfig := range cfg.LSP { if lspConfig.Command == "" && !lspConfig.Disabled { logging.Warn("LSP configuration has no command, marking as disabled", "language", language) lspConfig.Disabled = true cfg.LSP[language] = lspConfig } } return nil } // getProviderAPIKey gets the API key for a provider from environment variables func getProviderAPIKey(provider models.ModelProvider) string { switch provider { case models.ProviderAnthropic: return os.Getenv("ANTHROPIC_API_KEY") case models.ProviderOpenAI: return os.Getenv("OPENAI_API_KEY") case models.ProviderGemini: return os.Getenv("GEMINI_API_KEY") case models.ProviderGROQ: return os.Getenv("GROQ_API_KEY") case models.ProviderAzure: return os.Getenv("AZURE_OPENAI_API_KEY") case models.ProviderOpenRouter: return os.Getenv("OPENROUTER_API_KEY") case models.ProviderBedrock: if hasAWSCredentials() { return "aws-credentials-available" } case models.ProviderVertexAI: if hasVertexAICredentials() { return "vertex-ai-credentials-available" } } return "" } // setDefaultModelForAgent sets a default model for an agent based on available providers func setDefaultModelForAgent(agent AgentName) bool { if hasCopilotCredentials() { maxTokens := int64(5000) if agent == AgentTitle { maxTokens = 80 } cfg.Agents[agent] = Agent{ Model: models.CopilotGPT4o, MaxTokens: maxTokens, } return true } // Check providers in order of preference if apiKey := os.Getenv("ANTHROPIC_API_KEY"); apiKey != "" { maxTokens := int64(5000) if agent == AgentTitle { maxTokens = 80 } cfg.Agents[agent] = Agent{ Model: models.Claude37Sonnet, MaxTokens: maxTokens, } return true } if apiKey := os.Getenv("OPENAI_API_KEY"); apiKey != "" { var model models.ModelID maxTokens := int64(5000) reasoningEffort := "" switch agent { case AgentTitle: model = models.GPT41Mini maxTokens = 80 case AgentTask: model = models.GPT41Mini default: model = models.GPT41 } // Check if model supports reasoning if modelInfo, ok := models.SupportedModels[model]; ok && modelInfo.CanReason { reasoningEffort = "medium" } cfg.Agents[agent] = Agent{ Model: model, MaxTokens: maxTokens, ReasoningEffort: reasoningEffort, } return true } if apiKey := os.Getenv("OPENROUTER_API_KEY"); apiKey != "" { var model models.ModelID maxTokens := int64(5000) reasoningEffort := "" switch agent { case AgentTitle: model = models.OpenRouterClaude35Haiku maxTokens = 80 case AgentTask: model = models.OpenRouterClaude37Sonnet default: model = models.OpenRouterClaude37Sonnet } // Check if model supports reasoning if modelInfo, ok := models.SupportedModels[model]; ok && modelInfo.CanReason { reasoningEffort = "medium" } cfg.Agents[agent] = Agent{ Model: model, MaxTokens: maxTokens, ReasoningEffort: reasoningEffort, } return true } if apiKey := os.Getenv("GEMINI_API_KEY"); apiKey != "" { var model models.ModelID maxTokens := int64(5000) if agent == AgentTitle { model = models.Gemini25Flash maxTokens = 80 } else { model = models.Gemini25 } cfg.Agents[agent] = Agent{ Model: model, MaxTokens: maxTokens, } return true } if apiKey := os.Getenv("GROQ_API_KEY"); apiKey != "" { maxTokens := int64(5000) if agent == AgentTitle { maxTokens = 80 } cfg.Agents[agent] = Agent{ Model: models.QWENQwq, MaxTokens: maxTokens, } return true } if hasAWSCredentials() { maxTokens := int64(5000) if agent == AgentTitle { maxTokens = 80 } cfg.Agents[agent] = Agent{ Model: models.BedrockClaude37Sonnet, MaxTokens: maxTokens, ReasoningEffort: "medium", // Claude models support reasoning } return true } if hasVertexAICredentials() { var model models.ModelID maxTokens := int64(5000) if agent == AgentTitle { model = models.VertexAIGemini25Flash maxTokens = 80 } else { model = models.VertexAIGemini25 } cfg.Agents[agent] = Agent{ Model: model, MaxTokens: maxTokens, } return true } return false } func updateCfgFile(updateCfg func(config *Config)) error { if cfg == nil { return fmt.Errorf("config not loaded") } // Get the config file path configFile := viper.ConfigFileUsed() var configData []byte if configFile == "" { homeDir, err := os.UserHomeDir() if err != nil { return fmt.Errorf("failed to get home directory: %w", err) } configFile = filepath.Join(homeDir, fmt.Sprintf(".%s.json", appName)) logging.Info("config file not found, creating new one", "path", configFile) configData = []byte(`{}`) } else { // Read the existing config file data, err := os.ReadFile(configFile) if err != nil { return fmt.Errorf("failed to read config file: %w", err) } configData = data } // Parse the JSON var userCfg *Config if err := json.Unmarshal(configData, &userCfg); err != nil { return fmt.Errorf("failed to parse config file: %w", err) } updateCfg(userCfg) // Write the updated config back to file updatedData, err := json.MarshalIndent(userCfg, "", " ") if err != nil { return fmt.Errorf("failed to marshal config: %w", err) } if err := os.WriteFile(configFile, updatedData, 0o644); err != nil { return fmt.Errorf("failed to write config file: %w", err) } return nil } // Get returns the current configuration. // It's safe to call this function multiple times. func Get() *Config { return cfg } // WorkingDirectory returns the current working directory from the configuration. func WorkingDirectory() string { if cfg == nil { panic("config not loaded") } return cfg.WorkingDir } func UpdateAgentModel(agentName AgentName, modelID models.ModelID) error { if cfg == nil { panic("config not loaded") } existingAgentCfg := cfg.Agents[agentName] model, ok := models.SupportedModels[modelID] if !ok { return fmt.Errorf("model %s not supported", modelID) } maxTokens := existingAgentCfg.MaxTokens if model.DefaultMaxTokens > 0 { maxTokens = model.DefaultMaxTokens } newAgentCfg := Agent{ Model: modelID, MaxTokens: maxTokens, ReasoningEffort: existingAgentCfg.ReasoningEffort, } cfg.Agents[agentName] = newAgentCfg if err := validateAgent(cfg, agentName, newAgentCfg); err != nil { // revert config update on failure cfg.Agents[agentName] = existingAgentCfg return fmt.Errorf("failed to update agent model: %w", err) } return updateCfgFile(func(config *Config) { if config.Agents == nil { config.Agents = make(map[AgentName]Agent) } config.Agents[agentName] = newAgentCfg }) } // UpdateTheme updates the theme in the configuration and writes it to the config file. func UpdateTheme(themeName string) error { if cfg == nil { return fmt.Errorf("config not loaded") } // Update the in-memory config cfg.TUI.Theme = themeName // Update the file config return updateCfgFile(func(config *Config) { config.TUI.Theme = themeName }) } // Tries to load Github token from all possible locations func LoadGitHubToken() (string, error) { // First check environment variable if token := os.Getenv("GITHUB_TOKEN"); token != "" { return token, nil } // Get config directory var configDir string if xdgConfig := os.Getenv("XDG_CONFIG_HOME"); xdgConfig != "" { configDir = xdgConfig } else if runtime.GOOS == "windows" { if localAppData := os.Getenv("LOCALAPPDATA"); localAppData != "" { configDir = localAppData } else { configDir = filepath.Join(os.Getenv("HOME"), "AppData", "Local") } } else { configDir = filepath.Join(os.Getenv("HOME"), ".config") } // Try both hosts.json and apps.json files filePaths := []string{ filepath.Join(configDir, "github-copilot", "hosts.json"), filepath.Join(configDir, "github-copilot", "apps.json"), } for _, filePath := range filePaths { data, err := os.ReadFile(filePath) if err != nil { continue } var config map[string]map[string]interface{} if err := json.Unmarshal(data, &config); err != nil { continue } for key, value := range config { if strings.Contains(key, "github.com") { if oauthToken, ok := value["oauth_token"].(string); ok { return oauthToken, nil } } } } return "", fmt.Errorf("GitHub token not found in standard locations") } ================================================ FILE: internal/config/init.go ================================================ package config import ( "fmt" "os" "path/filepath" ) const ( // InitFlagFilename is the name of the file that indicates whether the project has been initialized InitFlagFilename = "init" ) // ProjectInitFlag represents the initialization status for a project directory type ProjectInitFlag struct { Initialized bool `json:"initialized"` } // ShouldShowInitDialog checks if the initialization dialog should be shown for the current directory func ShouldShowInitDialog() (bool, error) { if cfg == nil { return false, fmt.Errorf("config not loaded") } // Create the flag file path flagFilePath := filepath.Join(cfg.Data.Directory, InitFlagFilename) // Check if the flag file exists _, err := os.Stat(flagFilePath) if err == nil { // File exists, don't show the dialog return false, nil } // If the error is not "file not found", return the error if !os.IsNotExist(err) { return false, fmt.Errorf("failed to check init flag file: %w", err) } // File doesn't exist, show the dialog return true, nil } // MarkProjectInitialized marks the current project as initialized func MarkProjectInitialized() error { if cfg == nil { return fmt.Errorf("config not loaded") } // Create the flag file path flagFilePath := filepath.Join(cfg.Data.Directory, InitFlagFilename) // Create an empty file to mark the project as initialized file, err := os.Create(flagFilePath) if err != nil { return fmt.Errorf("failed to create init flag file: %w", err) } defer file.Close() return nil } ================================================ FILE: internal/db/connect.go ================================================ package db import ( "database/sql" "fmt" "os" "path/filepath" _ "github.com/ncruces/go-sqlite3/driver" _ "github.com/ncruces/go-sqlite3/embed" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/logging" "github.com/pressly/goose/v3" ) func Connect() (*sql.DB, error) { dataDir := config.Get().Data.Directory if dataDir == "" { return nil, fmt.Errorf("data.dir is not set") } if err := os.MkdirAll(dataDir, 0o700); err != nil { return nil, fmt.Errorf("failed to create data directory: %w", err) } dbPath := filepath.Join(dataDir, "opencode.db") // Open the SQLite database db, err := sql.Open("sqlite3", dbPath) if err != nil { return nil, fmt.Errorf("failed to open database: %w", err) } // Verify connection if err = db.Ping(); err != nil { db.Close() return nil, fmt.Errorf("failed to connect to database: %w", err) } // Set pragmas for better performance pragmas := []string{ "PRAGMA foreign_keys = ON;", "PRAGMA journal_mode = WAL;", "PRAGMA page_size = 4096;", "PRAGMA cache_size = -8000;", "PRAGMA synchronous = NORMAL;", } for _, pragma := range pragmas { if _, err = db.Exec(pragma); err != nil { logging.Error("Failed to set pragma", pragma, err) } else { logging.Debug("Set pragma", "pragma", pragma) } } goose.SetBaseFS(FS) if err := goose.SetDialect("sqlite3"); err != nil { logging.Error("Failed to set dialect", "error", err) return nil, fmt.Errorf("failed to set dialect: %w", err) } if err := goose.Up(db, "migrations"); err != nil { logging.Error("Failed to apply migrations", "error", err) return nil, fmt.Errorf("failed to apply migrations: %w", err) } return db, nil } ================================================ FILE: internal/db/db.go ================================================ // Code generated by sqlc. DO NOT EDIT. // versions: // sqlc v1.29.0 package db import ( "context" "database/sql" "fmt" ) type DBTX interface { ExecContext(context.Context, string, ...interface{}) (sql.Result, error) PrepareContext(context.Context, string) (*sql.Stmt, error) QueryContext(context.Context, string, ...interface{}) (*sql.Rows, error) QueryRowContext(context.Context, string, ...interface{}) *sql.Row } func New(db DBTX) *Queries { return &Queries{db: db} } func Prepare(ctx context.Context, db DBTX) (*Queries, error) { q := Queries{db: db} var err error if q.createFileStmt, err = db.PrepareContext(ctx, createFile); err != nil { return nil, fmt.Errorf("error preparing query CreateFile: %w", err) } if q.createMessageStmt, err = db.PrepareContext(ctx, createMessage); err != nil { return nil, fmt.Errorf("error preparing query CreateMessage: %w", err) } if q.createSessionStmt, err = db.PrepareContext(ctx, createSession); err != nil { return nil, fmt.Errorf("error preparing query CreateSession: %w", err) } if q.deleteFileStmt, err = db.PrepareContext(ctx, deleteFile); err != nil { return nil, fmt.Errorf("error preparing query DeleteFile: %w", err) } if q.deleteMessageStmt, err = db.PrepareContext(ctx, deleteMessage); err != nil { return nil, fmt.Errorf("error preparing query DeleteMessage: %w", err) } if q.deleteSessionStmt, err = db.PrepareContext(ctx, deleteSession); err != nil { return nil, fmt.Errorf("error preparing query DeleteSession: %w", err) } if q.deleteSessionFilesStmt, err = db.PrepareContext(ctx, deleteSessionFiles); err != nil { return nil, fmt.Errorf("error preparing query DeleteSessionFiles: %w", err) } if q.deleteSessionMessagesStmt, err = db.PrepareContext(ctx, deleteSessionMessages); err != nil { return nil, fmt.Errorf("error preparing query DeleteSessionMessages: %w", err) } if q.getFileStmt, err = db.PrepareContext(ctx, getFile); err != nil { return nil, fmt.Errorf("error preparing query GetFile: %w", err) } if q.getFileByPathAndSessionStmt, err = db.PrepareContext(ctx, getFileByPathAndSession); err != nil { return nil, fmt.Errorf("error preparing query GetFileByPathAndSession: %w", err) } if q.getMessageStmt, err = db.PrepareContext(ctx, getMessage); err != nil { return nil, fmt.Errorf("error preparing query GetMessage: %w", err) } if q.getSessionByIDStmt, err = db.PrepareContext(ctx, getSessionByID); err != nil { return nil, fmt.Errorf("error preparing query GetSessionByID: %w", err) } if q.listFilesByPathStmt, err = db.PrepareContext(ctx, listFilesByPath); err != nil { return nil, fmt.Errorf("error preparing query ListFilesByPath: %w", err) } if q.listFilesBySessionStmt, err = db.PrepareContext(ctx, listFilesBySession); err != nil { return nil, fmt.Errorf("error preparing query ListFilesBySession: %w", err) } if q.listLatestSessionFilesStmt, err = db.PrepareContext(ctx, listLatestSessionFiles); err != nil { return nil, fmt.Errorf("error preparing query ListLatestSessionFiles: %w", err) } if q.listMessagesBySessionStmt, err = db.PrepareContext(ctx, listMessagesBySession); err != nil { return nil, fmt.Errorf("error preparing query ListMessagesBySession: %w", err) } if q.listNewFilesStmt, err = db.PrepareContext(ctx, listNewFiles); err != nil { return nil, fmt.Errorf("error preparing query ListNewFiles: %w", err) } if q.listSessionsStmt, err = db.PrepareContext(ctx, listSessions); err != nil { return nil, fmt.Errorf("error preparing query ListSessions: %w", err) } if q.updateFileStmt, err = db.PrepareContext(ctx, updateFile); err != nil { return nil, fmt.Errorf("error preparing query UpdateFile: %w", err) } if q.updateMessageStmt, err = db.PrepareContext(ctx, updateMessage); err != nil { return nil, fmt.Errorf("error preparing query UpdateMessage: %w", err) } if q.updateSessionStmt, err = db.PrepareContext(ctx, updateSession); err != nil { return nil, fmt.Errorf("error preparing query UpdateSession: %w", err) } return &q, nil } func (q *Queries) Close() error { var err error if q.createFileStmt != nil { if cerr := q.createFileStmt.Close(); cerr != nil { err = fmt.Errorf("error closing createFileStmt: %w", cerr) } } if q.createMessageStmt != nil { if cerr := q.createMessageStmt.Close(); cerr != nil { err = fmt.Errorf("error closing createMessageStmt: %w", cerr) } } if q.createSessionStmt != nil { if cerr := q.createSessionStmt.Close(); cerr != nil { err = fmt.Errorf("error closing createSessionStmt: %w", cerr) } } if q.deleteFileStmt != nil { if cerr := q.deleteFileStmt.Close(); cerr != nil { err = fmt.Errorf("error closing deleteFileStmt: %w", cerr) } } if q.deleteMessageStmt != nil { if cerr := q.deleteMessageStmt.Close(); cerr != nil { err = fmt.Errorf("error closing deleteMessageStmt: %w", cerr) } } if q.deleteSessionStmt != nil { if cerr := q.deleteSessionStmt.Close(); cerr != nil { err = fmt.Errorf("error closing deleteSessionStmt: %w", cerr) } } if q.deleteSessionFilesStmt != nil { if cerr := q.deleteSessionFilesStmt.Close(); cerr != nil { err = fmt.Errorf("error closing deleteSessionFilesStmt: %w", cerr) } } if q.deleteSessionMessagesStmt != nil { if cerr := q.deleteSessionMessagesStmt.Close(); cerr != nil { err = fmt.Errorf("error closing deleteSessionMessagesStmt: %w", cerr) } } if q.getFileStmt != nil { if cerr := q.getFileStmt.Close(); cerr != nil { err = fmt.Errorf("error closing getFileStmt: %w", cerr) } } if q.getFileByPathAndSessionStmt != nil { if cerr := q.getFileByPathAndSessionStmt.Close(); cerr != nil { err = fmt.Errorf("error closing getFileByPathAndSessionStmt: %w", cerr) } } if q.getMessageStmt != nil { if cerr := q.getMessageStmt.Close(); cerr != nil { err = fmt.Errorf("error closing getMessageStmt: %w", cerr) } } if q.getSessionByIDStmt != nil { if cerr := q.getSessionByIDStmt.Close(); cerr != nil { err = fmt.Errorf("error closing getSessionByIDStmt: %w", cerr) } } if q.listFilesByPathStmt != nil { if cerr := q.listFilesByPathStmt.Close(); cerr != nil { err = fmt.Errorf("error closing listFilesByPathStmt: %w", cerr) } } if q.listFilesBySessionStmt != nil { if cerr := q.listFilesBySessionStmt.Close(); cerr != nil { err = fmt.Errorf("error closing listFilesBySessionStmt: %w", cerr) } } if q.listLatestSessionFilesStmt != nil { if cerr := q.listLatestSessionFilesStmt.Close(); cerr != nil { err = fmt.Errorf("error closing listLatestSessionFilesStmt: %w", cerr) } } if q.listMessagesBySessionStmt != nil { if cerr := q.listMessagesBySessionStmt.Close(); cerr != nil { err = fmt.Errorf("error closing listMessagesBySessionStmt: %w", cerr) } } if q.listNewFilesStmt != nil { if cerr := q.listNewFilesStmt.Close(); cerr != nil { err = fmt.Errorf("error closing listNewFilesStmt: %w", cerr) } } if q.listSessionsStmt != nil { if cerr := q.listSessionsStmt.Close(); cerr != nil { err = fmt.Errorf("error closing listSessionsStmt: %w", cerr) } } if q.updateFileStmt != nil { if cerr := q.updateFileStmt.Close(); cerr != nil { err = fmt.Errorf("error closing updateFileStmt: %w", cerr) } } if q.updateMessageStmt != nil { if cerr := q.updateMessageStmt.Close(); cerr != nil { err = fmt.Errorf("error closing updateMessageStmt: %w", cerr) } } if q.updateSessionStmt != nil { if cerr := q.updateSessionStmt.Close(); cerr != nil { err = fmt.Errorf("error closing updateSessionStmt: %w", cerr) } } return err } func (q *Queries) exec(ctx context.Context, stmt *sql.Stmt, query string, args ...interface{}) (sql.Result, error) { switch { case stmt != nil && q.tx != nil: return q.tx.StmtContext(ctx, stmt).ExecContext(ctx, args...) case stmt != nil: return stmt.ExecContext(ctx, args...) default: return q.db.ExecContext(ctx, query, args...) } } func (q *Queries) query(ctx context.Context, stmt *sql.Stmt, query string, args ...interface{}) (*sql.Rows, error) { switch { case stmt != nil && q.tx != nil: return q.tx.StmtContext(ctx, stmt).QueryContext(ctx, args...) case stmt != nil: return stmt.QueryContext(ctx, args...) default: return q.db.QueryContext(ctx, query, args...) } } func (q *Queries) queryRow(ctx context.Context, stmt *sql.Stmt, query string, args ...interface{}) *sql.Row { switch { case stmt != nil && q.tx != nil: return q.tx.StmtContext(ctx, stmt).QueryRowContext(ctx, args...) case stmt != nil: return stmt.QueryRowContext(ctx, args...) default: return q.db.QueryRowContext(ctx, query, args...) } } type Queries struct { db DBTX tx *sql.Tx createFileStmt *sql.Stmt createMessageStmt *sql.Stmt createSessionStmt *sql.Stmt deleteFileStmt *sql.Stmt deleteMessageStmt *sql.Stmt deleteSessionStmt *sql.Stmt deleteSessionFilesStmt *sql.Stmt deleteSessionMessagesStmt *sql.Stmt getFileStmt *sql.Stmt getFileByPathAndSessionStmt *sql.Stmt getMessageStmt *sql.Stmt getSessionByIDStmt *sql.Stmt listFilesByPathStmt *sql.Stmt listFilesBySessionStmt *sql.Stmt listLatestSessionFilesStmt *sql.Stmt listMessagesBySessionStmt *sql.Stmt listNewFilesStmt *sql.Stmt listSessionsStmt *sql.Stmt updateFileStmt *sql.Stmt updateMessageStmt *sql.Stmt updateSessionStmt *sql.Stmt } func (q *Queries) WithTx(tx *sql.Tx) *Queries { return &Queries{ db: tx, tx: tx, createFileStmt: q.createFileStmt, createMessageStmt: q.createMessageStmt, createSessionStmt: q.createSessionStmt, deleteFileStmt: q.deleteFileStmt, deleteMessageStmt: q.deleteMessageStmt, deleteSessionStmt: q.deleteSessionStmt, deleteSessionFilesStmt: q.deleteSessionFilesStmt, deleteSessionMessagesStmt: q.deleteSessionMessagesStmt, getFileStmt: q.getFileStmt, getFileByPathAndSessionStmt: q.getFileByPathAndSessionStmt, getMessageStmt: q.getMessageStmt, getSessionByIDStmt: q.getSessionByIDStmt, listFilesByPathStmt: q.listFilesByPathStmt, listFilesBySessionStmt: q.listFilesBySessionStmt, listLatestSessionFilesStmt: q.listLatestSessionFilesStmt, listMessagesBySessionStmt: q.listMessagesBySessionStmt, listNewFilesStmt: q.listNewFilesStmt, listSessionsStmt: q.listSessionsStmt, updateFileStmt: q.updateFileStmt, updateMessageStmt: q.updateMessageStmt, updateSessionStmt: q.updateSessionStmt, } } ================================================ FILE: internal/db/embed.go ================================================ package db import "embed" //go:embed migrations/*.sql var FS embed.FS ================================================ FILE: internal/db/files.sql.go ================================================ // Code generated by sqlc. DO NOT EDIT. // versions: // sqlc v1.29.0 // source: files.sql package db import ( "context" ) const createFile = `-- name: CreateFile :one INSERT INTO files ( id, session_id, path, content, version, created_at, updated_at ) VALUES ( ?, ?, ?, ?, ?, strftime('%s', 'now'), strftime('%s', 'now') ) RETURNING id, session_id, path, content, version, created_at, updated_at ` type CreateFileParams struct { ID string `json:"id"` SessionID string `json:"session_id"` Path string `json:"path"` Content string `json:"content"` Version string `json:"version"` } func (q *Queries) CreateFile(ctx context.Context, arg CreateFileParams) (File, error) { row := q.queryRow(ctx, q.createFileStmt, createFile, arg.ID, arg.SessionID, arg.Path, arg.Content, arg.Version, ) var i File err := row.Scan( &i.ID, &i.SessionID, &i.Path, &i.Content, &i.Version, &i.CreatedAt, &i.UpdatedAt, ) return i, err } const deleteFile = `-- name: DeleteFile :exec DELETE FROM files WHERE id = ? ` func (q *Queries) DeleteFile(ctx context.Context, id string) error { _, err := q.exec(ctx, q.deleteFileStmt, deleteFile, id) return err } const deleteSessionFiles = `-- name: DeleteSessionFiles :exec DELETE FROM files WHERE session_id = ? ` func (q *Queries) DeleteSessionFiles(ctx context.Context, sessionID string) error { _, err := q.exec(ctx, q.deleteSessionFilesStmt, deleteSessionFiles, sessionID) return err } const getFile = `-- name: GetFile :one SELECT id, session_id, path, content, version, created_at, updated_at FROM files WHERE id = ? LIMIT 1 ` func (q *Queries) GetFile(ctx context.Context, id string) (File, error) { row := q.queryRow(ctx, q.getFileStmt, getFile, id) var i File err := row.Scan( &i.ID, &i.SessionID, &i.Path, &i.Content, &i.Version, &i.CreatedAt, &i.UpdatedAt, ) return i, err } const getFileByPathAndSession = `-- name: GetFileByPathAndSession :one SELECT id, session_id, path, content, version, created_at, updated_at FROM files WHERE path = ? AND session_id = ? ORDER BY created_at DESC LIMIT 1 ` type GetFileByPathAndSessionParams struct { Path string `json:"path"` SessionID string `json:"session_id"` } func (q *Queries) GetFileByPathAndSession(ctx context.Context, arg GetFileByPathAndSessionParams) (File, error) { row := q.queryRow(ctx, q.getFileByPathAndSessionStmt, getFileByPathAndSession, arg.Path, arg.SessionID) var i File err := row.Scan( &i.ID, &i.SessionID, &i.Path, &i.Content, &i.Version, &i.CreatedAt, &i.UpdatedAt, ) return i, err } const listFilesByPath = `-- name: ListFilesByPath :many SELECT id, session_id, path, content, version, created_at, updated_at FROM files WHERE path = ? ORDER BY created_at DESC ` func (q *Queries) ListFilesByPath(ctx context.Context, path string) ([]File, error) { rows, err := q.query(ctx, q.listFilesByPathStmt, listFilesByPath, path) if err != nil { return nil, err } defer rows.Close() items := []File{} for rows.Next() { var i File if err := rows.Scan( &i.ID, &i.SessionID, &i.Path, &i.Content, &i.Version, &i.CreatedAt, &i.UpdatedAt, ); err != nil { return nil, err } items = append(items, i) } if err := rows.Close(); err != nil { return nil, err } if err := rows.Err(); err != nil { return nil, err } return items, nil } const listFilesBySession = `-- name: ListFilesBySession :many SELECT id, session_id, path, content, version, created_at, updated_at FROM files WHERE session_id = ? ORDER BY created_at ASC ` func (q *Queries) ListFilesBySession(ctx context.Context, sessionID string) ([]File, error) { rows, err := q.query(ctx, q.listFilesBySessionStmt, listFilesBySession, sessionID) if err != nil { return nil, err } defer rows.Close() items := []File{} for rows.Next() { var i File if err := rows.Scan( &i.ID, &i.SessionID, &i.Path, &i.Content, &i.Version, &i.CreatedAt, &i.UpdatedAt, ); err != nil { return nil, err } items = append(items, i) } if err := rows.Close(); err != nil { return nil, err } if err := rows.Err(); err != nil { return nil, err } return items, nil } const listLatestSessionFiles = `-- name: ListLatestSessionFiles :many SELECT f.id, f.session_id, f.path, f.content, f.version, f.created_at, f.updated_at FROM files f INNER JOIN ( SELECT path, MAX(created_at) as max_created_at FROM files GROUP BY path ) latest ON f.path = latest.path AND f.created_at = latest.max_created_at WHERE f.session_id = ? ORDER BY f.path ` func (q *Queries) ListLatestSessionFiles(ctx context.Context, sessionID string) ([]File, error) { rows, err := q.query(ctx, q.listLatestSessionFilesStmt, listLatestSessionFiles, sessionID) if err != nil { return nil, err } defer rows.Close() items := []File{} for rows.Next() { var i File if err := rows.Scan( &i.ID, &i.SessionID, &i.Path, &i.Content, &i.Version, &i.CreatedAt, &i.UpdatedAt, ); err != nil { return nil, err } items = append(items, i) } if err := rows.Close(); err != nil { return nil, err } if err := rows.Err(); err != nil { return nil, err } return items, nil } const listNewFiles = `-- name: ListNewFiles :many SELECT id, session_id, path, content, version, created_at, updated_at FROM files WHERE is_new = 1 ORDER BY created_at DESC ` func (q *Queries) ListNewFiles(ctx context.Context) ([]File, error) { rows, err := q.query(ctx, q.listNewFilesStmt, listNewFiles) if err != nil { return nil, err } defer rows.Close() items := []File{} for rows.Next() { var i File if err := rows.Scan( &i.ID, &i.SessionID, &i.Path, &i.Content, &i.Version, &i.CreatedAt, &i.UpdatedAt, ); err != nil { return nil, err } items = append(items, i) } if err := rows.Close(); err != nil { return nil, err } if err := rows.Err(); err != nil { return nil, err } return items, nil } const updateFile = `-- name: UpdateFile :one UPDATE files SET content = ?, version = ?, updated_at = strftime('%s', 'now') WHERE id = ? RETURNING id, session_id, path, content, version, created_at, updated_at ` type UpdateFileParams struct { Content string `json:"content"` Version string `json:"version"` ID string `json:"id"` } func (q *Queries) UpdateFile(ctx context.Context, arg UpdateFileParams) (File, error) { row := q.queryRow(ctx, q.updateFileStmt, updateFile, arg.Content, arg.Version, arg.ID) var i File err := row.Scan( &i.ID, &i.SessionID, &i.Path, &i.Content, &i.Version, &i.CreatedAt, &i.UpdatedAt, ) return i, err } ================================================ FILE: internal/db/messages.sql.go ================================================ // Code generated by sqlc. DO NOT EDIT. // versions: // sqlc v1.29.0 // source: messages.sql package db import ( "context" "database/sql" ) const createMessage = `-- name: CreateMessage :one INSERT INTO messages ( id, session_id, role, parts, model, created_at, updated_at ) VALUES ( ?, ?, ?, ?, ?, strftime('%s', 'now'), strftime('%s', 'now') ) RETURNING id, session_id, role, parts, model, created_at, updated_at, finished_at ` type CreateMessageParams struct { ID string `json:"id"` SessionID string `json:"session_id"` Role string `json:"role"` Parts string `json:"parts"` Model sql.NullString `json:"model"` } func (q *Queries) CreateMessage(ctx context.Context, arg CreateMessageParams) (Message, error) { row := q.queryRow(ctx, q.createMessageStmt, createMessage, arg.ID, arg.SessionID, arg.Role, arg.Parts, arg.Model, ) var i Message err := row.Scan( &i.ID, &i.SessionID, &i.Role, &i.Parts, &i.Model, &i.CreatedAt, &i.UpdatedAt, &i.FinishedAt, ) return i, err } const deleteMessage = `-- name: DeleteMessage :exec DELETE FROM messages WHERE id = ? ` func (q *Queries) DeleteMessage(ctx context.Context, id string) error { _, err := q.exec(ctx, q.deleteMessageStmt, deleteMessage, id) return err } const deleteSessionMessages = `-- name: DeleteSessionMessages :exec DELETE FROM messages WHERE session_id = ? ` func (q *Queries) DeleteSessionMessages(ctx context.Context, sessionID string) error { _, err := q.exec(ctx, q.deleteSessionMessagesStmt, deleteSessionMessages, sessionID) return err } const getMessage = `-- name: GetMessage :one SELECT id, session_id, role, parts, model, created_at, updated_at, finished_at FROM messages WHERE id = ? LIMIT 1 ` func (q *Queries) GetMessage(ctx context.Context, id string) (Message, error) { row := q.queryRow(ctx, q.getMessageStmt, getMessage, id) var i Message err := row.Scan( &i.ID, &i.SessionID, &i.Role, &i.Parts, &i.Model, &i.CreatedAt, &i.UpdatedAt, &i.FinishedAt, ) return i, err } const listMessagesBySession = `-- name: ListMessagesBySession :many SELECT id, session_id, role, parts, model, created_at, updated_at, finished_at FROM messages WHERE session_id = ? ORDER BY created_at ASC ` func (q *Queries) ListMessagesBySession(ctx context.Context, sessionID string) ([]Message, error) { rows, err := q.query(ctx, q.listMessagesBySessionStmt, listMessagesBySession, sessionID) if err != nil { return nil, err } defer rows.Close() items := []Message{} for rows.Next() { var i Message if err := rows.Scan( &i.ID, &i.SessionID, &i.Role, &i.Parts, &i.Model, &i.CreatedAt, &i.UpdatedAt, &i.FinishedAt, ); err != nil { return nil, err } items = append(items, i) } if err := rows.Close(); err != nil { return nil, err } if err := rows.Err(); err != nil { return nil, err } return items, nil } const updateMessage = `-- name: UpdateMessage :exec UPDATE messages SET parts = ?, finished_at = ?, updated_at = strftime('%s', 'now') WHERE id = ? ` type UpdateMessageParams struct { Parts string `json:"parts"` FinishedAt sql.NullInt64 `json:"finished_at"` ID string `json:"id"` } func (q *Queries) UpdateMessage(ctx context.Context, arg UpdateMessageParams) error { _, err := q.exec(ctx, q.updateMessageStmt, updateMessage, arg.Parts, arg.FinishedAt, arg.ID) return err } ================================================ FILE: internal/db/migrations/20250424200609_initial.sql ================================================ -- +goose Up -- +goose StatementBegin -- Sessions CREATE TABLE IF NOT EXISTS sessions ( id TEXT PRIMARY KEY, parent_session_id TEXT, title TEXT NOT NULL, message_count INTEGER NOT NULL DEFAULT 0 CHECK (message_count >= 0), prompt_tokens INTEGER NOT NULL DEFAULT 0 CHECK (prompt_tokens >= 0), completion_tokens INTEGER NOT NULL DEFAULT 0 CHECK (completion_tokens>= 0), cost REAL NOT NULL DEFAULT 0.0 CHECK (cost >= 0.0), updated_at INTEGER NOT NULL, -- Unix timestamp in milliseconds created_at INTEGER NOT NULL -- Unix timestamp in milliseconds ); CREATE TRIGGER IF NOT EXISTS update_sessions_updated_at AFTER UPDATE ON sessions BEGIN UPDATE sessions SET updated_at = strftime('%s', 'now') WHERE id = new.id; END; -- Files CREATE TABLE IF NOT EXISTS files ( id TEXT PRIMARY KEY, session_id TEXT NOT NULL, path TEXT NOT NULL, content TEXT NOT NULL, version TEXT NOT NULL, created_at INTEGER NOT NULL, -- Unix timestamp in milliseconds updated_at INTEGER NOT NULL, -- Unix timestamp in milliseconds FOREIGN KEY (session_id) REFERENCES sessions (id) ON DELETE CASCADE, UNIQUE(path, session_id, version) ); CREATE INDEX IF NOT EXISTS idx_files_session_id ON files (session_id); CREATE INDEX IF NOT EXISTS idx_files_path ON files (path); CREATE TRIGGER IF NOT EXISTS update_files_updated_at AFTER UPDATE ON files BEGIN UPDATE files SET updated_at = strftime('%s', 'now') WHERE id = new.id; END; -- Messages CREATE TABLE IF NOT EXISTS messages ( id TEXT PRIMARY KEY, session_id TEXT NOT NULL, role TEXT NOT NULL, parts TEXT NOT NULL default '[]', model TEXT, created_at INTEGER NOT NULL, -- Unix timestamp in milliseconds updated_at INTEGER NOT NULL, -- Unix timestamp in milliseconds finished_at INTEGER, -- Unix timestamp in milliseconds FOREIGN KEY (session_id) REFERENCES sessions (id) ON DELETE CASCADE ); CREATE INDEX IF NOT EXISTS idx_messages_session_id ON messages (session_id); CREATE TRIGGER IF NOT EXISTS update_messages_updated_at AFTER UPDATE ON messages BEGIN UPDATE messages SET updated_at = strftime('%s', 'now') WHERE id = new.id; END; CREATE TRIGGER IF NOT EXISTS update_session_message_count_on_insert AFTER INSERT ON messages BEGIN UPDATE sessions SET message_count = message_count + 1 WHERE id = new.session_id; END; CREATE TRIGGER IF NOT EXISTS update_session_message_count_on_delete AFTER DELETE ON messages BEGIN UPDATE sessions SET message_count = message_count - 1 WHERE id = old.session_id; END; -- +goose StatementEnd -- +goose Down -- +goose StatementBegin DROP TRIGGER IF EXISTS update_sessions_updated_at; DROP TRIGGER IF EXISTS update_messages_updated_at; DROP TRIGGER IF EXISTS update_files_updated_at; DROP TRIGGER IF EXISTS update_session_message_count_on_delete; DROP TRIGGER IF EXISTS update_session_message_count_on_insert; DROP TABLE IF EXISTS sessions; DROP TABLE IF EXISTS messages; DROP TABLE IF EXISTS files; -- +goose StatementEnd ================================================ FILE: internal/db/migrations/20250515105448_add_summary_message_id.sql ================================================ -- +goose Up -- +goose StatementBegin ALTER TABLE sessions ADD COLUMN summary_message_id TEXT; -- +goose StatementEnd -- +goose Down -- +goose StatementBegin ALTER TABLE sessions DROP COLUMN summary_message_id; -- +goose StatementEnd ================================================ FILE: internal/db/models.go ================================================ // Code generated by sqlc. DO NOT EDIT. // versions: // sqlc v1.29.0 package db import ( "database/sql" ) type File struct { ID string `json:"id"` SessionID string `json:"session_id"` Path string `json:"path"` Content string `json:"content"` Version string `json:"version"` CreatedAt int64 `json:"created_at"` UpdatedAt int64 `json:"updated_at"` } type Message struct { ID string `json:"id"` SessionID string `json:"session_id"` Role string `json:"role"` Parts string `json:"parts"` Model sql.NullString `json:"model"` CreatedAt int64 `json:"created_at"` UpdatedAt int64 `json:"updated_at"` FinishedAt sql.NullInt64 `json:"finished_at"` } type Session struct { ID string `json:"id"` ParentSessionID sql.NullString `json:"parent_session_id"` Title string `json:"title"` MessageCount int64 `json:"message_count"` PromptTokens int64 `json:"prompt_tokens"` CompletionTokens int64 `json:"completion_tokens"` Cost float64 `json:"cost"` UpdatedAt int64 `json:"updated_at"` CreatedAt int64 `json:"created_at"` SummaryMessageID sql.NullString `json:"summary_message_id"` } ================================================ FILE: internal/db/querier.go ================================================ // Code generated by sqlc. DO NOT EDIT. // versions: // sqlc v1.29.0 package db import ( "context" ) type Querier interface { CreateFile(ctx context.Context, arg CreateFileParams) (File, error) CreateMessage(ctx context.Context, arg CreateMessageParams) (Message, error) CreateSession(ctx context.Context, arg CreateSessionParams) (Session, error) DeleteFile(ctx context.Context, id string) error DeleteMessage(ctx context.Context, id string) error DeleteSession(ctx context.Context, id string) error DeleteSessionFiles(ctx context.Context, sessionID string) error DeleteSessionMessages(ctx context.Context, sessionID string) error GetFile(ctx context.Context, id string) (File, error) GetFileByPathAndSession(ctx context.Context, arg GetFileByPathAndSessionParams) (File, error) GetMessage(ctx context.Context, id string) (Message, error) GetSessionByID(ctx context.Context, id string) (Session, error) ListFilesByPath(ctx context.Context, path string) ([]File, error) ListFilesBySession(ctx context.Context, sessionID string) ([]File, error) ListLatestSessionFiles(ctx context.Context, sessionID string) ([]File, error) ListMessagesBySession(ctx context.Context, sessionID string) ([]Message, error) ListNewFiles(ctx context.Context) ([]File, error) ListSessions(ctx context.Context) ([]Session, error) UpdateFile(ctx context.Context, arg UpdateFileParams) (File, error) UpdateMessage(ctx context.Context, arg UpdateMessageParams) error UpdateSession(ctx context.Context, arg UpdateSessionParams) (Session, error) } var _ Querier = (*Queries)(nil) ================================================ FILE: internal/db/sessions.sql.go ================================================ // Code generated by sqlc. DO NOT EDIT. // versions: // sqlc v1.29.0 // source: sessions.sql package db import ( "context" "database/sql" ) const createSession = `-- name: CreateSession :one INSERT INTO sessions ( id, parent_session_id, title, message_count, prompt_tokens, completion_tokens, cost, summary_message_id, updated_at, created_at ) VALUES ( ?, ?, ?, ?, ?, ?, ?, null, strftime('%s', 'now'), strftime('%s', 'now') ) RETURNING id, parent_session_id, title, message_count, prompt_tokens, completion_tokens, cost, updated_at, created_at, summary_message_id ` type CreateSessionParams struct { ID string `json:"id"` ParentSessionID sql.NullString `json:"parent_session_id"` Title string `json:"title"` MessageCount int64 `json:"message_count"` PromptTokens int64 `json:"prompt_tokens"` CompletionTokens int64 `json:"completion_tokens"` Cost float64 `json:"cost"` } func (q *Queries) CreateSession(ctx context.Context, arg CreateSessionParams) (Session, error) { row := q.queryRow(ctx, q.createSessionStmt, createSession, arg.ID, arg.ParentSessionID, arg.Title, arg.MessageCount, arg.PromptTokens, arg.CompletionTokens, arg.Cost, ) var i Session err := row.Scan( &i.ID, &i.ParentSessionID, &i.Title, &i.MessageCount, &i.PromptTokens, &i.CompletionTokens, &i.Cost, &i.UpdatedAt, &i.CreatedAt, &i.SummaryMessageID, ) return i, err } const deleteSession = `-- name: DeleteSession :exec DELETE FROM sessions WHERE id = ? ` func (q *Queries) DeleteSession(ctx context.Context, id string) error { _, err := q.exec(ctx, q.deleteSessionStmt, deleteSession, id) return err } const getSessionByID = `-- name: GetSessionByID :one SELECT id, parent_session_id, title, message_count, prompt_tokens, completion_tokens, cost, updated_at, created_at, summary_message_id FROM sessions WHERE id = ? LIMIT 1 ` func (q *Queries) GetSessionByID(ctx context.Context, id string) (Session, error) { row := q.queryRow(ctx, q.getSessionByIDStmt, getSessionByID, id) var i Session err := row.Scan( &i.ID, &i.ParentSessionID, &i.Title, &i.MessageCount, &i.PromptTokens, &i.CompletionTokens, &i.Cost, &i.UpdatedAt, &i.CreatedAt, &i.SummaryMessageID, ) return i, err } const listSessions = `-- name: ListSessions :many SELECT id, parent_session_id, title, message_count, prompt_tokens, completion_tokens, cost, updated_at, created_at, summary_message_id FROM sessions WHERE parent_session_id is NULL ORDER BY created_at DESC ` func (q *Queries) ListSessions(ctx context.Context) ([]Session, error) { rows, err := q.query(ctx, q.listSessionsStmt, listSessions) if err != nil { return nil, err } defer rows.Close() items := []Session{} for rows.Next() { var i Session if err := rows.Scan( &i.ID, &i.ParentSessionID, &i.Title, &i.MessageCount, &i.PromptTokens, &i.CompletionTokens, &i.Cost, &i.UpdatedAt, &i.CreatedAt, &i.SummaryMessageID, ); err != nil { return nil, err } items = append(items, i) } if err := rows.Close(); err != nil { return nil, err } if err := rows.Err(); err != nil { return nil, err } return items, nil } const updateSession = `-- name: UpdateSession :one UPDATE sessions SET title = ?, prompt_tokens = ?, completion_tokens = ?, summary_message_id = ?, cost = ? WHERE id = ? RETURNING id, parent_session_id, title, message_count, prompt_tokens, completion_tokens, cost, updated_at, created_at, summary_message_id ` type UpdateSessionParams struct { Title string `json:"title"` PromptTokens int64 `json:"prompt_tokens"` CompletionTokens int64 `json:"completion_tokens"` SummaryMessageID sql.NullString `json:"summary_message_id"` Cost float64 `json:"cost"` ID string `json:"id"` } func (q *Queries) UpdateSession(ctx context.Context, arg UpdateSessionParams) (Session, error) { row := q.queryRow(ctx, q.updateSessionStmt, updateSession, arg.Title, arg.PromptTokens, arg.CompletionTokens, arg.SummaryMessageID, arg.Cost, arg.ID, ) var i Session err := row.Scan( &i.ID, &i.ParentSessionID, &i.Title, &i.MessageCount, &i.PromptTokens, &i.CompletionTokens, &i.Cost, &i.UpdatedAt, &i.CreatedAt, &i.SummaryMessageID, ) return i, err } ================================================ FILE: internal/db/sql/files.sql ================================================ -- name: GetFile :one SELECT * FROM files WHERE id = ? LIMIT 1; -- name: GetFileByPathAndSession :one SELECT * FROM files WHERE path = ? AND session_id = ? ORDER BY created_at DESC LIMIT 1; -- name: ListFilesBySession :many SELECT * FROM files WHERE session_id = ? ORDER BY created_at ASC; -- name: ListFilesByPath :many SELECT * FROM files WHERE path = ? ORDER BY created_at DESC; -- name: CreateFile :one INSERT INTO files ( id, session_id, path, content, version, created_at, updated_at ) VALUES ( ?, ?, ?, ?, ?, strftime('%s', 'now'), strftime('%s', 'now') ) RETURNING *; -- name: UpdateFile :one UPDATE files SET content = ?, version = ?, updated_at = strftime('%s', 'now') WHERE id = ? RETURNING *; -- name: DeleteFile :exec DELETE FROM files WHERE id = ?; -- name: DeleteSessionFiles :exec DELETE FROM files WHERE session_id = ?; -- name: ListLatestSessionFiles :many SELECT f.* FROM files f INNER JOIN ( SELECT path, MAX(created_at) as max_created_at FROM files GROUP BY path ) latest ON f.path = latest.path AND f.created_at = latest.max_created_at WHERE f.session_id = ? ORDER BY f.path; -- name: ListNewFiles :many SELECT * FROM files WHERE is_new = 1 ORDER BY created_at DESC; ================================================ FILE: internal/db/sql/messages.sql ================================================ -- name: GetMessage :one SELECT * FROM messages WHERE id = ? LIMIT 1; -- name: ListMessagesBySession :many SELECT * FROM messages WHERE session_id = ? ORDER BY created_at ASC; -- name: CreateMessage :one INSERT INTO messages ( id, session_id, role, parts, model, created_at, updated_at ) VALUES ( ?, ?, ?, ?, ?, strftime('%s', 'now'), strftime('%s', 'now') ) RETURNING *; -- name: UpdateMessage :exec UPDATE messages SET parts = ?, finished_at = ?, updated_at = strftime('%s', 'now') WHERE id = ?; -- name: DeleteMessage :exec DELETE FROM messages WHERE id = ?; -- name: DeleteSessionMessages :exec DELETE FROM messages WHERE session_id = ?; ================================================ FILE: internal/db/sql/sessions.sql ================================================ -- name: CreateSession :one INSERT INTO sessions ( id, parent_session_id, title, message_count, prompt_tokens, completion_tokens, cost, summary_message_id, updated_at, created_at ) VALUES ( ?, ?, ?, ?, ?, ?, ?, null, strftime('%s', 'now'), strftime('%s', 'now') ) RETURNING *; -- name: GetSessionByID :one SELECT * FROM sessions WHERE id = ? LIMIT 1; -- name: ListSessions :many SELECT * FROM sessions WHERE parent_session_id is NULL ORDER BY created_at DESC; -- name: UpdateSession :one UPDATE sessions SET title = ?, prompt_tokens = ?, completion_tokens = ?, summary_message_id = ?, cost = ? WHERE id = ? RETURNING *; -- name: DeleteSession :exec DELETE FROM sessions WHERE id = ?; ================================================ FILE: internal/diff/diff.go ================================================ package diff import ( "bytes" "fmt" "io" "regexp" "strconv" "strings" "github.com/alecthomas/chroma/v2" "github.com/alecthomas/chroma/v2/formatters" "github.com/alecthomas/chroma/v2/lexers" "github.com/alecthomas/chroma/v2/styles" "github.com/aymanbagabas/go-udiff" "github.com/charmbracelet/lipgloss" "github.com/charmbracelet/x/ansi" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/tui/theme" "github.com/sergi/go-diff/diffmatchpatch" ) // ------------------------------------------------------------------------- // Core Types // ------------------------------------------------------------------------- // LineType represents the kind of line in a diff. type LineType int const ( LineContext LineType = iota // Line exists in both files LineAdded // Line added in the new file LineRemoved // Line removed from the old file ) // Segment represents a portion of a line for intra-line highlighting type Segment struct { Start int End int Type LineType Text string } // DiffLine represents a single line in a diff type DiffLine struct { OldLineNo int // Line number in old file (0 for added lines) NewLineNo int // Line number in new file (0 for removed lines) Kind LineType // Type of line (added, removed, context) Content string // Content of the line Segments []Segment // Segments for intraline highlighting } // Hunk represents a section of changes in a diff type Hunk struct { Header string Lines []DiffLine } // DiffResult contains the parsed result of a diff type DiffResult struct { OldFile string NewFile string Hunks []Hunk } // linePair represents a pair of lines for side-by-side display type linePair struct { left *DiffLine right *DiffLine } // ------------------------------------------------------------------------- // Parse Configuration // ------------------------------------------------------------------------- // ParseConfig configures the behavior of diff parsing type ParseConfig struct { ContextSize int // Number of context lines to include } // ParseOption modifies a ParseConfig type ParseOption func(*ParseConfig) // WithContextSize sets the number of context lines to include func WithContextSize(size int) ParseOption { return func(p *ParseConfig) { if size >= 0 { p.ContextSize = size } } } // ------------------------------------------------------------------------- // Side-by-Side Configuration // ------------------------------------------------------------------------- // SideBySideConfig configures the rendering of side-by-side diffs type SideBySideConfig struct { TotalWidth int } // SideBySideOption modifies a SideBySideConfig type SideBySideOption func(*SideBySideConfig) // NewSideBySideConfig creates a SideBySideConfig with default values func NewSideBySideConfig(opts ...SideBySideOption) SideBySideConfig { config := SideBySideConfig{ TotalWidth: 160, // Default width for side-by-side view } for _, opt := range opts { opt(&config) } return config } // WithTotalWidth sets the total width for side-by-side view func WithTotalWidth(width int) SideBySideOption { return func(s *SideBySideConfig) { if width > 0 { s.TotalWidth = width } } } // ------------------------------------------------------------------------- // Diff Parsing // ------------------------------------------------------------------------- // ParseUnifiedDiff parses a unified diff format string into structured data func ParseUnifiedDiff(diff string) (DiffResult, error) { var result DiffResult var currentHunk *Hunk hunkHeaderRe := regexp.MustCompile(`^@@ -(\d+),?(\d*) \+(\d+),?(\d*) @@`) lines := strings.Split(diff, "\n") var oldLine, newLine int inFileHeader := true for _, line := range lines { // Parse file headers if inFileHeader { if strings.HasPrefix(line, "--- a/") { result.OldFile = strings.TrimPrefix(line, "--- a/") continue } if strings.HasPrefix(line, "+++ b/") { result.NewFile = strings.TrimPrefix(line, "+++ b/") inFileHeader = false continue } } // Parse hunk headers if matches := hunkHeaderRe.FindStringSubmatch(line); matches != nil { if currentHunk != nil { result.Hunks = append(result.Hunks, *currentHunk) } currentHunk = &Hunk{ Header: line, Lines: []DiffLine{}, } oldStart, _ := strconv.Atoi(matches[1]) newStart, _ := strconv.Atoi(matches[3]) oldLine = oldStart newLine = newStart continue } // Ignore "No newline at end of file" markers if strings.HasPrefix(line, "\\ No newline at end of file") { continue } if currentHunk == nil { continue } // Process the line based on its prefix if len(line) > 0 { switch line[0] { case '+': currentHunk.Lines = append(currentHunk.Lines, DiffLine{ OldLineNo: 0, NewLineNo: newLine, Kind: LineAdded, Content: line[1:], }) newLine++ case '-': currentHunk.Lines = append(currentHunk.Lines, DiffLine{ OldLineNo: oldLine, NewLineNo: 0, Kind: LineRemoved, Content: line[1:], }) oldLine++ default: currentHunk.Lines = append(currentHunk.Lines, DiffLine{ OldLineNo: oldLine, NewLineNo: newLine, Kind: LineContext, Content: line, }) oldLine++ newLine++ } } else { // Handle empty lines currentHunk.Lines = append(currentHunk.Lines, DiffLine{ OldLineNo: oldLine, NewLineNo: newLine, Kind: LineContext, Content: "", }) oldLine++ newLine++ } } // Add the last hunk if there is one if currentHunk != nil { result.Hunks = append(result.Hunks, *currentHunk) } return result, nil } // HighlightIntralineChanges updates lines in a hunk to show character-level differences func HighlightIntralineChanges(h *Hunk) { var updated []DiffLine dmp := diffmatchpatch.New() for i := 0; i < len(h.Lines); i++ { // Look for removed line followed by added line if i+1 < len(h.Lines) && h.Lines[i].Kind == LineRemoved && h.Lines[i+1].Kind == LineAdded { oldLine := h.Lines[i] newLine := h.Lines[i+1] // Find character-level differences patches := dmp.DiffMain(oldLine.Content, newLine.Content, false) patches = dmp.DiffCleanupSemantic(patches) patches = dmp.DiffCleanupMerge(patches) patches = dmp.DiffCleanupEfficiency(patches) segments := make([]Segment, 0) removeStart := 0 addStart := 0 for _, patch := range patches { switch patch.Type { case diffmatchpatch.DiffDelete: segments = append(segments, Segment{ Start: removeStart, End: removeStart + len(patch.Text), Type: LineRemoved, Text: patch.Text, }) removeStart += len(patch.Text) case diffmatchpatch.DiffInsert: segments = append(segments, Segment{ Start: addStart, End: addStart + len(patch.Text), Type: LineAdded, Text: patch.Text, }) addStart += len(patch.Text) default: // Context text, no highlighting needed removeStart += len(patch.Text) addStart += len(patch.Text) } } oldLine.Segments = segments newLine.Segments = segments updated = append(updated, oldLine, newLine) i++ // Skip the next line as we've already processed it } else { updated = append(updated, h.Lines[i]) } } h.Lines = updated } // pairLines converts a flat list of diff lines to pairs for side-by-side display func pairLines(lines []DiffLine) []linePair { var pairs []linePair i := 0 for i < len(lines) { switch lines[i].Kind { case LineRemoved: // Check if the next line is an addition, if so pair them if i+1 < len(lines) && lines[i+1].Kind == LineAdded { pairs = append(pairs, linePair{left: &lines[i], right: &lines[i+1]}) i += 2 } else { pairs = append(pairs, linePair{left: &lines[i], right: nil}) i++ } case LineAdded: pairs = append(pairs, linePair{left: nil, right: &lines[i]}) i++ case LineContext: pairs = append(pairs, linePair{left: &lines[i], right: &lines[i]}) i++ } } return pairs } // ------------------------------------------------------------------------- // Syntax Highlighting // ------------------------------------------------------------------------- // SyntaxHighlight applies syntax highlighting to text based on file extension func SyntaxHighlight(w io.Writer, source, fileName, formatter string, bg lipgloss.TerminalColor) error { t := theme.CurrentTheme() // Determine the language lexer to use l := lexers.Match(fileName) if l == nil { l = lexers.Analyse(source) } if l == nil { l = lexers.Fallback } l = chroma.Coalesce(l) // Get the formatter f := formatters.Get(formatter) if f == nil { f = formatters.Fallback } // Dynamic theme based on current theme values syntaxThemeXml := fmt.Sprintf(` `, getColor(t.Background()), // Background getColor(t.Text()), // Text getColor(t.Text()), // Other getColor(t.Error()), // Error getColor(t.SyntaxKeyword()), // Keyword getColor(t.SyntaxKeyword()), // KeywordConstant getColor(t.SyntaxKeyword()), // KeywordDeclaration getColor(t.SyntaxKeyword()), // KeywordNamespace getColor(t.SyntaxKeyword()), // KeywordPseudo getColor(t.SyntaxKeyword()), // KeywordReserved getColor(t.SyntaxType()), // KeywordType getColor(t.Text()), // Name getColor(t.SyntaxVariable()), // NameAttribute getColor(t.SyntaxType()), // NameBuiltin getColor(t.SyntaxVariable()), // NameBuiltinPseudo getColor(t.SyntaxType()), // NameClass getColor(t.SyntaxVariable()), // NameConstant getColor(t.SyntaxFunction()), // NameDecorator getColor(t.SyntaxVariable()), // NameEntity getColor(t.SyntaxType()), // NameException getColor(t.SyntaxFunction()), // NameFunction getColor(t.Text()), // NameLabel getColor(t.SyntaxType()), // NameNamespace getColor(t.SyntaxVariable()), // NameOther getColor(t.SyntaxKeyword()), // NameTag getColor(t.SyntaxVariable()), // NameVariable getColor(t.SyntaxVariable()), // NameVariableClass getColor(t.SyntaxVariable()), // NameVariableGlobal getColor(t.SyntaxVariable()), // NameVariableInstance getColor(t.SyntaxString()), // Literal getColor(t.SyntaxString()), // LiteralDate getColor(t.SyntaxString()), // LiteralString getColor(t.SyntaxString()), // LiteralStringBacktick getColor(t.SyntaxString()), // LiteralStringChar getColor(t.SyntaxString()), // LiteralStringDoc getColor(t.SyntaxString()), // LiteralStringDouble getColor(t.SyntaxString()), // LiteralStringEscape getColor(t.SyntaxString()), // LiteralStringHeredoc getColor(t.SyntaxString()), // LiteralStringInterpol getColor(t.SyntaxString()), // LiteralStringOther getColor(t.SyntaxString()), // LiteralStringRegex getColor(t.SyntaxString()), // LiteralStringSingle getColor(t.SyntaxString()), // LiteralStringSymbol getColor(t.SyntaxNumber()), // LiteralNumber getColor(t.SyntaxNumber()), // LiteralNumberBin getColor(t.SyntaxNumber()), // LiteralNumberFloat getColor(t.SyntaxNumber()), // LiteralNumberHex getColor(t.SyntaxNumber()), // LiteralNumberInteger getColor(t.SyntaxNumber()), // LiteralNumberIntegerLong getColor(t.SyntaxNumber()), // LiteralNumberOct getColor(t.SyntaxOperator()), // Operator getColor(t.SyntaxKeyword()), // OperatorWord getColor(t.SyntaxPunctuation()), // Punctuation getColor(t.SyntaxComment()), // Comment getColor(t.SyntaxComment()), // CommentHashbang getColor(t.SyntaxComment()), // CommentMultiline getColor(t.SyntaxComment()), // CommentSingle getColor(t.SyntaxComment()), // CommentSpecial getColor(t.SyntaxKeyword()), // CommentPreproc getColor(t.Text()), // Generic getColor(t.Error()), // GenericDeleted getColor(t.Text()), // GenericEmph getColor(t.Error()), // GenericError getColor(t.Text()), // GenericHeading getColor(t.Success()), // GenericInserted getColor(t.TextMuted()), // GenericOutput getColor(t.Text()), // GenericPrompt getColor(t.Text()), // GenericStrong getColor(t.Text()), // GenericSubheading getColor(t.Error()), // GenericTraceback getColor(t.Text()), // TextWhitespace ) r := strings.NewReader(syntaxThemeXml) style := chroma.MustNewXMLStyle(r) // Modify the style to use the provided background s, err := style.Builder().Transform( func(t chroma.StyleEntry) chroma.StyleEntry { r, g, b, _ := bg.RGBA() t.Background = chroma.NewColour(uint8(r>>8), uint8(g>>8), uint8(b>>8)) return t }, ).Build() if err != nil { s = styles.Fallback } // Tokenize and format it, err := l.Tokenise(nil, source) if err != nil { return err } return f.Format(w, s, it) } // getColor returns the appropriate hex color string based on terminal background func getColor(adaptiveColor lipgloss.AdaptiveColor) string { if lipgloss.HasDarkBackground() { return adaptiveColor.Dark } return adaptiveColor.Light } // highlightLine applies syntax highlighting to a single line func highlightLine(fileName string, line string, bg lipgloss.TerminalColor) string { var buf bytes.Buffer err := SyntaxHighlight(&buf, line, fileName, "terminal16m", bg) if err != nil { return line } return buf.String() } // createStyles generates the lipgloss styles needed for rendering diffs func createStyles(t theme.Theme) (removedLineStyle, addedLineStyle, contextLineStyle, lineNumberStyle lipgloss.Style) { removedLineStyle = lipgloss.NewStyle().Background(t.DiffRemovedBg()) addedLineStyle = lipgloss.NewStyle().Background(t.DiffAddedBg()) contextLineStyle = lipgloss.NewStyle().Background(t.DiffContextBg()) lineNumberStyle = lipgloss.NewStyle().Foreground(t.DiffLineNumber()) return } // ------------------------------------------------------------------------- // Rendering Functions // ------------------------------------------------------------------------- func lipglossToHex(color lipgloss.Color) string { r, g, b, a := color.RGBA() // Scale uint32 values (0-65535) to uint8 (0-255). r8 := uint8(r >> 8) g8 := uint8(g >> 8) b8 := uint8(b >> 8) a8 := uint8(a >> 8) return fmt.Sprintf("#%02x%02x%02x%02x", r8, g8, b8, a8) } // applyHighlighting applies intra-line highlighting to a piece of text func applyHighlighting(content string, segments []Segment, segmentType LineType, highlightBg lipgloss.AdaptiveColor) string { // Find all ANSI sequences in the content ansiRegex := regexp.MustCompile(`\x1b(?:[@-Z\\-_]|\[[0-9?]*(?:;[0-9?]*)*[@-~])`) ansiMatches := ansiRegex.FindAllStringIndex(content, -1) // Build a mapping of visible character positions to their actual indices visibleIdx := 0 ansiSequences := make(map[int]string) lastAnsiSeq := "\x1b[0m" // Default reset sequence for i := 0; i < len(content); { isAnsi := false for _, match := range ansiMatches { if match[0] == i { ansiSequences[visibleIdx] = content[match[0]:match[1]] lastAnsiSeq = content[match[0]:match[1]] i = match[1] isAnsi = true break } } if isAnsi { continue } // For non-ANSI positions, store the last ANSI sequence if _, exists := ansiSequences[visibleIdx]; !exists { ansiSequences[visibleIdx] = lastAnsiSeq } visibleIdx++ i++ } // Apply highlighting var sb strings.Builder inSelection := false currentPos := 0 // Get the appropriate color based on terminal background bgColor := lipgloss.Color(getColor(highlightBg)) fgColor := lipgloss.Color(getColor(theme.CurrentTheme().Background())) for i := 0; i < len(content); { // Check if we're at an ANSI sequence isAnsi := false for _, match := range ansiMatches { if match[0] == i { sb.WriteString(content[match[0]:match[1]]) // Preserve ANSI sequence i = match[1] isAnsi = true break } } if isAnsi { continue } // Check for segment boundaries for _, seg := range segments { if seg.Type == segmentType { if currentPos == seg.Start { inSelection = true } if currentPos == seg.End { inSelection = false } } } // Get current character char := string(content[i]) if inSelection { // Get the current styling currentStyle := ansiSequences[currentPos] // Apply foreground and background highlight sb.WriteString("\x1b[38;2;") r, g, b, _ := fgColor.RGBA() sb.WriteString(fmt.Sprintf("%d;%d;%dm", r>>8, g>>8, b>>8)) sb.WriteString("\x1b[48;2;") r, g, b, _ = bgColor.RGBA() sb.WriteString(fmt.Sprintf("%d;%d;%dm", r>>8, g>>8, b>>8)) sb.WriteString(char) // Reset foreground and background sb.WriteString("\x1b[39m") // Reapply the original ANSI sequence sb.WriteString(currentStyle) } else { // Not in selection, just copy the character sb.WriteString(char) } currentPos++ i++ } return sb.String() } // renderLeftColumn formats the left side of a side-by-side diff func renderLeftColumn(fileName string, dl *DiffLine, colWidth int) string { t := theme.CurrentTheme() if dl == nil { contextLineStyle := lipgloss.NewStyle().Background(t.DiffContextBg()) return contextLineStyle.Width(colWidth).Render("") } removedLineStyle, _, contextLineStyle, lineNumberStyle := createStyles(t) // Determine line style based on line type var marker string var bgStyle lipgloss.Style switch dl.Kind { case LineRemoved: marker = removedLineStyle.Foreground(t.DiffRemoved()).Render("-") bgStyle = removedLineStyle lineNumberStyle = lineNumberStyle.Foreground(t.DiffRemoved()).Background(t.DiffRemovedLineNumberBg()) case LineAdded: marker = "?" bgStyle = contextLineStyle case LineContext: marker = contextLineStyle.Render(" ") bgStyle = contextLineStyle } // Format line number lineNum := "" if dl.OldLineNo > 0 { lineNum = fmt.Sprintf("%6d", dl.OldLineNo) } // Create the line prefix prefix := lineNumberStyle.Render(lineNum + " " + marker) // Apply syntax highlighting content := highlightLine(fileName, dl.Content, bgStyle.GetBackground()) // Apply intra-line highlighting for removed lines if dl.Kind == LineRemoved && len(dl.Segments) > 0 { content = applyHighlighting(content, dl.Segments, LineRemoved, t.DiffHighlightRemoved()) } // Add a padding space for removed lines if dl.Kind == LineRemoved { content = bgStyle.Render(" ") + content } // Create the final line and truncate if needed lineText := prefix + content return bgStyle.MaxHeight(1).Width(colWidth).Render( ansi.Truncate( lineText, colWidth, lipgloss.NewStyle().Background(bgStyle.GetBackground()).Foreground(t.TextMuted()).Render("..."), ), ) } // renderRightColumn formats the right side of a side-by-side diff func renderRightColumn(fileName string, dl *DiffLine, colWidth int) string { t := theme.CurrentTheme() if dl == nil { contextLineStyle := lipgloss.NewStyle().Background(t.DiffContextBg()) return contextLineStyle.Width(colWidth).Render("") } _, addedLineStyle, contextLineStyle, lineNumberStyle := createStyles(t) // Determine line style based on line type var marker string var bgStyle lipgloss.Style switch dl.Kind { case LineAdded: marker = addedLineStyle.Foreground(t.DiffAdded()).Render("+") bgStyle = addedLineStyle lineNumberStyle = lineNumberStyle.Foreground(t.DiffAdded()).Background(t.DiffAddedLineNumberBg()) case LineRemoved: marker = "?" bgStyle = contextLineStyle case LineContext: marker = contextLineStyle.Render(" ") bgStyle = contextLineStyle } // Format line number lineNum := "" if dl.NewLineNo > 0 { lineNum = fmt.Sprintf("%6d", dl.NewLineNo) } // Create the line prefix prefix := lineNumberStyle.Render(lineNum + " " + marker) // Apply syntax highlighting content := highlightLine(fileName, dl.Content, bgStyle.GetBackground()) // Apply intra-line highlighting for added lines if dl.Kind == LineAdded && len(dl.Segments) > 0 { content = applyHighlighting(content, dl.Segments, LineAdded, t.DiffHighlightAdded()) } // Add a padding space for added lines if dl.Kind == LineAdded { content = bgStyle.Render(" ") + content } // Create the final line and truncate if needed lineText := prefix + content return bgStyle.MaxHeight(1).Width(colWidth).Render( ansi.Truncate( lineText, colWidth, lipgloss.NewStyle().Background(bgStyle.GetBackground()).Foreground(t.TextMuted()).Render("..."), ), ) } // ------------------------------------------------------------------------- // Public API // ------------------------------------------------------------------------- // RenderSideBySideHunk formats a hunk for side-by-side display func RenderSideBySideHunk(fileName string, h Hunk, opts ...SideBySideOption) string { // Apply options to create the configuration config := NewSideBySideConfig(opts...) // Make a copy of the hunk so we don't modify the original hunkCopy := Hunk{Lines: make([]DiffLine, len(h.Lines))} copy(hunkCopy.Lines, h.Lines) // Highlight changes within lines HighlightIntralineChanges(&hunkCopy) // Pair lines for side-by-side display pairs := pairLines(hunkCopy.Lines) // Calculate column width colWidth := config.TotalWidth / 2 leftWidth := colWidth rightWidth := config.TotalWidth - colWidth var sb strings.Builder for _, p := range pairs { leftStr := renderLeftColumn(fileName, p.left, leftWidth) rightStr := renderRightColumn(fileName, p.right, rightWidth) sb.WriteString(leftStr + rightStr + "\n") } return sb.String() } // FormatDiff creates a side-by-side formatted view of a diff func FormatDiff(diffText string, opts ...SideBySideOption) (string, error) { diffResult, err := ParseUnifiedDiff(diffText) if err != nil { return "", err } var sb strings.Builder for _, h := range diffResult.Hunks { sb.WriteString(RenderSideBySideHunk(diffResult.OldFile, h, opts...)) } return sb.String(), nil } // GenerateDiff creates a unified diff from two file contents func GenerateDiff(beforeContent, afterContent, fileName string) (string, int, int) { // remove the cwd prefix and ensure consistent path format // this prevents issues with absolute paths in different environments cwd := config.WorkingDirectory() fileName = strings.TrimPrefix(fileName, cwd) fileName = strings.TrimPrefix(fileName, "/") var ( unified = udiff.Unified("a/"+fileName, "b/"+fileName, beforeContent, afterContent) additions = 0 removals = 0 ) lines := strings.SplitSeq(unified, "\n") for line := range lines { if strings.HasPrefix(line, "+") && !strings.HasPrefix(line, "+++") { additions++ } else if strings.HasPrefix(line, "-") && !strings.HasPrefix(line, "---") { removals++ } } return unified, additions, removals } ================================================ FILE: internal/diff/patch.go ================================================ package diff import ( "errors" "fmt" "os" "path/filepath" "strings" ) type ActionType string const ( ActionAdd ActionType = "add" ActionDelete ActionType = "delete" ActionUpdate ActionType = "update" ) type FileChange struct { Type ActionType OldContent *string NewContent *string MovePath *string } type Commit struct { Changes map[string]FileChange } type Chunk struct { OrigIndex int // line index of the first line in the original file DelLines []string // lines to delete InsLines []string // lines to insert } type PatchAction struct { Type ActionType NewFile *string Chunks []Chunk MovePath *string } type Patch struct { Actions map[string]PatchAction } type DiffError struct { message string } func (e DiffError) Error() string { return e.message } // Helper functions for error handling func NewDiffError(message string) DiffError { return DiffError{message: message} } func fileError(action, reason, path string) DiffError { return NewDiffError(fmt.Sprintf("%s File Error: %s: %s", action, reason, path)) } func contextError(index int, context string, isEOF bool) DiffError { prefix := "Invalid Context" if isEOF { prefix = "Invalid EOF Context" } return NewDiffError(fmt.Sprintf("%s %d:\n%s", prefix, index, context)) } type Parser struct { currentFiles map[string]string lines []string index int patch Patch fuzz int } func NewParser(currentFiles map[string]string, lines []string) *Parser { return &Parser{ currentFiles: currentFiles, lines: lines, index: 0, patch: Patch{Actions: make(map[string]PatchAction, len(currentFiles))}, fuzz: 0, } } func (p *Parser) isDone(prefixes []string) bool { if p.index >= len(p.lines) { return true } for _, prefix := range prefixes { if strings.HasPrefix(p.lines[p.index], prefix) { return true } } return false } func (p *Parser) startsWith(prefix any) bool { var prefixes []string switch v := prefix.(type) { case string: prefixes = []string{v} case []string: prefixes = v } for _, pfx := range prefixes { if strings.HasPrefix(p.lines[p.index], pfx) { return true } } return false } func (p *Parser) readStr(prefix string, returnEverything bool) string { if p.index >= len(p.lines) { return "" // Changed from panic to return empty string for safer operation } if strings.HasPrefix(p.lines[p.index], prefix) { var text string if returnEverything { text = p.lines[p.index] } else { text = p.lines[p.index][len(prefix):] } p.index++ return text } return "" } func (p *Parser) Parse() error { endPatchPrefixes := []string{"*** End Patch"} for !p.isDone(endPatchPrefixes) { path := p.readStr("*** Update File: ", false) if path != "" { if _, exists := p.patch.Actions[path]; exists { return fileError("Update", "Duplicate Path", path) } moveTo := p.readStr("*** Move to: ", false) if _, exists := p.currentFiles[path]; !exists { return fileError("Update", "Missing File", path) } text := p.currentFiles[path] action, err := p.parseUpdateFile(text) if err != nil { return err } if moveTo != "" { action.MovePath = &moveTo } p.patch.Actions[path] = action continue } path = p.readStr("*** Delete File: ", false) if path != "" { if _, exists := p.patch.Actions[path]; exists { return fileError("Delete", "Duplicate Path", path) } if _, exists := p.currentFiles[path]; !exists { return fileError("Delete", "Missing File", path) } p.patch.Actions[path] = PatchAction{Type: ActionDelete, Chunks: []Chunk{}} continue } path = p.readStr("*** Add File: ", false) if path != "" { if _, exists := p.patch.Actions[path]; exists { return fileError("Add", "Duplicate Path", path) } if _, exists := p.currentFiles[path]; exists { return fileError("Add", "File already exists", path) } action, err := p.parseAddFile() if err != nil { return err } p.patch.Actions[path] = action continue } return NewDiffError(fmt.Sprintf("Unknown Line: %s", p.lines[p.index])) } if !p.startsWith("*** End Patch") { return NewDiffError("Missing End Patch") } p.index++ return nil } func (p *Parser) parseUpdateFile(text string) (PatchAction, error) { action := PatchAction{Type: ActionUpdate, Chunks: []Chunk{}} fileLines := strings.Split(text, "\n") index := 0 endPrefixes := []string{ "*** End Patch", "*** Update File:", "*** Delete File:", "*** Add File:", "*** End of File", } for !p.isDone(endPrefixes) { defStr := p.readStr("@@ ", false) sectionStr := "" if defStr == "" && p.index < len(p.lines) && p.lines[p.index] == "@@" { sectionStr = p.lines[p.index] p.index++ } if defStr == "" && sectionStr == "" && index != 0 { return action, NewDiffError(fmt.Sprintf("Invalid Line:\n%s", p.lines[p.index])) } if strings.TrimSpace(defStr) != "" { found := false for i := range fileLines[:index] { if fileLines[i] == defStr { found = true break } } if !found { for i := index; i < len(fileLines); i++ { if fileLines[i] == defStr { index = i + 1 found = true break } } } if !found { for i := range fileLines[:index] { if strings.TrimSpace(fileLines[i]) == strings.TrimSpace(defStr) { found = true break } } } if !found { for i := index; i < len(fileLines); i++ { if strings.TrimSpace(fileLines[i]) == strings.TrimSpace(defStr) { index = i + 1 p.fuzz++ found = true break } } } } nextChunkContext, chunks, endPatchIndex, eof := peekNextSection(p.lines, p.index) newIndex, fuzz := findContext(fileLines, nextChunkContext, index, eof) if newIndex == -1 { ctxText := strings.Join(nextChunkContext, "\n") return action, contextError(index, ctxText, eof) } p.fuzz += fuzz for _, ch := range chunks { ch.OrigIndex += newIndex action.Chunks = append(action.Chunks, ch) } index = newIndex + len(nextChunkContext) p.index = endPatchIndex } return action, nil } func (p *Parser) parseAddFile() (PatchAction, error) { lines := make([]string, 0, 16) // Preallocate space for better performance endPrefixes := []string{ "*** End Patch", "*** Update File:", "*** Delete File:", "*** Add File:", } for !p.isDone(endPrefixes) { s := p.readStr("", true) if !strings.HasPrefix(s, "+") { return PatchAction{}, NewDiffError(fmt.Sprintf("Invalid Add File Line: %s", s)) } lines = append(lines, s[1:]) } newFile := strings.Join(lines, "\n") return PatchAction{ Type: ActionAdd, NewFile: &newFile, Chunks: []Chunk{}, }, nil } // Refactored to use a matcher function for each comparison type func findContextCore(lines []string, context []string, start int) (int, int) { if len(context) == 0 { return start, 0 } // Try exact match if idx, fuzz := tryFindMatch(lines, context, start, func(a, b string) bool { return a == b }); idx >= 0 { return idx, fuzz } // Try trimming right whitespace if idx, fuzz := tryFindMatch(lines, context, start, func(a, b string) bool { return strings.TrimRight(a, " \t") == strings.TrimRight(b, " \t") }); idx >= 0 { return idx, fuzz } // Try trimming all whitespace if idx, fuzz := tryFindMatch(lines, context, start, func(a, b string) bool { return strings.TrimSpace(a) == strings.TrimSpace(b) }); idx >= 0 { return idx, fuzz } return -1, 0 } // Helper function to DRY up the match logic func tryFindMatch(lines []string, context []string, start int, compareFunc func(string, string) bool, ) (int, int) { for i := start; i < len(lines); i++ { if i+len(context) <= len(lines) { match := true for j := range context { if !compareFunc(lines[i+j], context[j]) { match = false break } } if match { // Return fuzz level: 0 for exact, 1 for trimRight, 100 for trimSpace var fuzz int if compareFunc("a ", "a") && !compareFunc("a", "b") { fuzz = 1 } else if compareFunc("a ", "a") { fuzz = 100 } return i, fuzz } } } return -1, 0 } func findContext(lines []string, context []string, start int, eof bool) (int, int) { if eof { newIndex, fuzz := findContextCore(lines, context, len(lines)-len(context)) if newIndex != -1 { return newIndex, fuzz } newIndex, fuzz = findContextCore(lines, context, start) return newIndex, fuzz + 10000 } return findContextCore(lines, context, start) } func peekNextSection(lines []string, initialIndex int) ([]string, []Chunk, int, bool) { index := initialIndex old := make([]string, 0, 32) // Preallocate for better performance delLines := make([]string, 0, 8) insLines := make([]string, 0, 8) chunks := make([]Chunk, 0, 4) mode := "keep" // End conditions for the section endSectionConditions := func(s string) bool { return strings.HasPrefix(s, "@@") || strings.HasPrefix(s, "*** End Patch") || strings.HasPrefix(s, "*** Update File:") || strings.HasPrefix(s, "*** Delete File:") || strings.HasPrefix(s, "*** Add File:") || strings.HasPrefix(s, "*** End of File") || s == "***" || strings.HasPrefix(s, "***") } for index < len(lines) { s := lines[index] if endSectionConditions(s) { break } index++ lastMode := mode line := s if len(line) > 0 { switch line[0] { case '+': mode = "add" case '-': mode = "delete" case ' ': mode = "keep" default: mode = "keep" line = " " + line } } else { mode = "keep" line = " " } line = line[1:] if mode == "keep" && lastMode != mode { if len(insLines) > 0 || len(delLines) > 0 { chunks = append(chunks, Chunk{ OrigIndex: len(old) - len(delLines), DelLines: delLines, InsLines: insLines, }) } delLines = make([]string, 0, 8) insLines = make([]string, 0, 8) } switch mode { case "delete": delLines = append(delLines, line) old = append(old, line) case "add": insLines = append(insLines, line) default: old = append(old, line) } } if len(insLines) > 0 || len(delLines) > 0 { chunks = append(chunks, Chunk{ OrigIndex: len(old) - len(delLines), DelLines: delLines, InsLines: insLines, }) } if index < len(lines) && lines[index] == "*** End of File" { index++ return old, chunks, index, true } return old, chunks, index, false } func TextToPatch(text string, orig map[string]string) (Patch, int, error) { text = strings.TrimSpace(text) lines := strings.Split(text, "\n") if len(lines) < 2 || !strings.HasPrefix(lines[0], "*** Begin Patch") || lines[len(lines)-1] != "*** End Patch" { return Patch{}, 0, NewDiffError("Invalid patch text") } parser := NewParser(orig, lines) parser.index = 1 if err := parser.Parse(); err != nil { return Patch{}, 0, err } return parser.patch, parser.fuzz, nil } func IdentifyFilesNeeded(text string) []string { text = strings.TrimSpace(text) lines := strings.Split(text, "\n") result := make(map[string]bool) for _, line := range lines { if strings.HasPrefix(line, "*** Update File: ") { result[line[len("*** Update File: "):]] = true } if strings.HasPrefix(line, "*** Delete File: ") { result[line[len("*** Delete File: "):]] = true } } files := make([]string, 0, len(result)) for file := range result { files = append(files, file) } return files } func IdentifyFilesAdded(text string) []string { text = strings.TrimSpace(text) lines := strings.Split(text, "\n") result := make(map[string]bool) for _, line := range lines { if strings.HasPrefix(line, "*** Add File: ") { result[line[len("*** Add File: "):]] = true } } files := make([]string, 0, len(result)) for file := range result { files = append(files, file) } return files } func getUpdatedFile(text string, action PatchAction, path string) (string, error) { if action.Type != ActionUpdate { return "", errors.New("expected UPDATE action") } origLines := strings.Split(text, "\n") destLines := make([]string, 0, len(origLines)) // Preallocate with capacity origIndex := 0 for _, chunk := range action.Chunks { if chunk.OrigIndex > len(origLines) { return "", NewDiffError(fmt.Sprintf("%s: chunk.orig_index %d > len(lines) %d", path, chunk.OrigIndex, len(origLines))) } if origIndex > chunk.OrigIndex { return "", NewDiffError(fmt.Sprintf("%s: orig_index %d > chunk.orig_index %d", path, origIndex, chunk.OrigIndex)) } destLines = append(destLines, origLines[origIndex:chunk.OrigIndex]...) delta := chunk.OrigIndex - origIndex origIndex += delta if len(chunk.InsLines) > 0 { destLines = append(destLines, chunk.InsLines...) } origIndex += len(chunk.DelLines) } destLines = append(destLines, origLines[origIndex:]...) return strings.Join(destLines, "\n"), nil } func PatchToCommit(patch Patch, orig map[string]string) (Commit, error) { commit := Commit{Changes: make(map[string]FileChange, len(patch.Actions))} for pathKey, action := range patch.Actions { switch action.Type { case ActionDelete: oldContent := orig[pathKey] commit.Changes[pathKey] = FileChange{ Type: ActionDelete, OldContent: &oldContent, } case ActionAdd: commit.Changes[pathKey] = FileChange{ Type: ActionAdd, NewContent: action.NewFile, } case ActionUpdate: newContent, err := getUpdatedFile(orig[pathKey], action, pathKey) if err != nil { return Commit{}, err } oldContent := orig[pathKey] fileChange := FileChange{ Type: ActionUpdate, OldContent: &oldContent, NewContent: &newContent, } if action.MovePath != nil { fileChange.MovePath = action.MovePath } commit.Changes[pathKey] = fileChange } } return commit, nil } func AssembleChanges(orig map[string]string, updatedFiles map[string]string) Commit { commit := Commit{Changes: make(map[string]FileChange, len(updatedFiles))} for p, newContent := range updatedFiles { oldContent, exists := orig[p] if exists && oldContent == newContent { continue } if exists && newContent != "" { commit.Changes[p] = FileChange{ Type: ActionUpdate, OldContent: &oldContent, NewContent: &newContent, } } else if newContent != "" { commit.Changes[p] = FileChange{ Type: ActionAdd, NewContent: &newContent, } } else if exists { commit.Changes[p] = FileChange{ Type: ActionDelete, OldContent: &oldContent, } } else { return commit // Changed from panic to simply return current commit } } return commit } func LoadFiles(paths []string, openFn func(string) (string, error)) (map[string]string, error) { orig := make(map[string]string, len(paths)) for _, p := range paths { content, err := openFn(p) if err != nil { return nil, fileError("Open", "File not found", p) } orig[p] = content } return orig, nil } func ApplyCommit(commit Commit, writeFn func(string, string) error, removeFn func(string) error) error { for p, change := range commit.Changes { switch change.Type { case ActionDelete: if err := removeFn(p); err != nil { return err } case ActionAdd: if change.NewContent == nil { return NewDiffError(fmt.Sprintf("Add action for %s has nil new_content", p)) } if err := writeFn(p, *change.NewContent); err != nil { return err } case ActionUpdate: if change.NewContent == nil { return NewDiffError(fmt.Sprintf("Update action for %s has nil new_content", p)) } if change.MovePath != nil { if err := writeFn(*change.MovePath, *change.NewContent); err != nil { return err } if err := removeFn(p); err != nil { return err } } else { if err := writeFn(p, *change.NewContent); err != nil { return err } } } } return nil } func ProcessPatch(text string, openFn func(string) (string, error), writeFn func(string, string) error, removeFn func(string) error) (string, error) { if !strings.HasPrefix(text, "*** Begin Patch") { return "", NewDiffError("Patch must start with *** Begin Patch") } paths := IdentifyFilesNeeded(text) orig, err := LoadFiles(paths, openFn) if err != nil { return "", err } patch, fuzz, err := TextToPatch(text, orig) if err != nil { return "", err } if fuzz > 0 { return "", NewDiffError(fmt.Sprintf("Patch contains fuzzy matches (fuzz level: %d)", fuzz)) } commit, err := PatchToCommit(patch, orig) if err != nil { return "", err } if err := ApplyCommit(commit, writeFn, removeFn); err != nil { return "", err } return "Patch applied successfully", nil } func OpenFile(p string) (string, error) { data, err := os.ReadFile(p) if err != nil { return "", err } return string(data), nil } func WriteFile(p string, content string) error { if filepath.IsAbs(p) { return NewDiffError("We do not support absolute paths.") } dir := filepath.Dir(p) if dir != "." { if err := os.MkdirAll(dir, 0o755); err != nil { return err } } return os.WriteFile(p, []byte(content), 0o644) } func RemoveFile(p string) error { return os.Remove(p) } func ValidatePatch(patchText string, files map[string]string) (bool, string, error) { if !strings.HasPrefix(patchText, "*** Begin Patch") { return false, "Patch must start with *** Begin Patch", nil } neededFiles := IdentifyFilesNeeded(patchText) for _, filePath := range neededFiles { if _, exists := files[filePath]; !exists { return false, fmt.Sprintf("File not found: %s", filePath), nil } } patch, fuzz, err := TextToPatch(patchText, files) if err != nil { return false, err.Error(), nil } if fuzz > 0 { return false, fmt.Sprintf("Patch contains fuzzy matches (fuzz level: %d)", fuzz), nil } _, err = PatchToCommit(patch, files) if err != nil { return false, err.Error(), nil } return true, "Patch is valid", nil } ================================================ FILE: internal/fileutil/fileutil.go ================================================ package fileutil import ( "fmt" "io/fs" "os" "os/exec" "path/filepath" "sort" "strings" "time" "github.com/bmatcuk/doublestar/v4" "github.com/opencode-ai/opencode/internal/logging" ) var ( rgPath string fzfPath string ) func init() { var err error rgPath, err = exec.LookPath("rg") if err != nil { logging.Warn("Ripgrep (rg) not found in $PATH. Some features might be limited or slower.") rgPath = "" } fzfPath, err = exec.LookPath("fzf") if err != nil { logging.Warn("FZF not found in $PATH. Some features might be limited or slower.") fzfPath = "" } } func GetRgCmd(globPattern string) *exec.Cmd { if rgPath == "" { return nil } rgArgs := []string{ "--files", "-L", "--null", } if globPattern != "" { if !filepath.IsAbs(globPattern) && !strings.HasPrefix(globPattern, "/") { globPattern = "/" + globPattern } rgArgs = append(rgArgs, "--glob", globPattern) } cmd := exec.Command(rgPath, rgArgs...) cmd.Dir = "." return cmd } func GetFzfCmd(query string) *exec.Cmd { if fzfPath == "" { return nil } fzfArgs := []string{ "--filter", query, "--read0", "--print0", } cmd := exec.Command(fzfPath, fzfArgs...) cmd.Dir = "." return cmd } type FileInfo struct { Path string ModTime time.Time } func SkipHidden(path string) bool { // Check for hidden files (starting with a dot) base := filepath.Base(path) if base != "." && strings.HasPrefix(base, ".") { return true } commonIgnoredDirs := map[string]bool{ ".opencode": true, "node_modules": true, "vendor": true, "dist": true, "build": true, "target": true, ".git": true, ".idea": true, ".vscode": true, "__pycache__": true, "bin": true, "obj": true, "out": true, "coverage": true, "tmp": true, "temp": true, "logs": true, "generated": true, "bower_components": true, "jspm_packages": true, } parts := strings.Split(path, string(os.PathSeparator)) for _, part := range parts { if commonIgnoredDirs[part] { return true } } return false } func GlobWithDoublestar(pattern, searchPath string, limit int) ([]string, bool, error) { fsys := os.DirFS(searchPath) relPattern := strings.TrimPrefix(pattern, "/") var matches []FileInfo err := doublestar.GlobWalk(fsys, relPattern, func(path string, d fs.DirEntry) error { if d.IsDir() { return nil } if SkipHidden(path) { return nil } info, err := d.Info() if err != nil { return nil } absPath := path if !strings.HasPrefix(absPath, searchPath) && searchPath != "." { absPath = filepath.Join(searchPath, absPath) } else if !strings.HasPrefix(absPath, "/") && searchPath == "." { absPath = filepath.Join(searchPath, absPath) // Ensure relative paths are joined correctly } matches = append(matches, FileInfo{Path: absPath, ModTime: info.ModTime()}) if limit > 0 && len(matches) >= limit*2 { return fs.SkipAll } return nil }) if err != nil { return nil, false, fmt.Errorf("glob walk error: %w", err) } sort.Slice(matches, func(i, j int) bool { return matches[i].ModTime.After(matches[j].ModTime) }) truncated := false if limit > 0 && len(matches) > limit { matches = matches[:limit] truncated = true } results := make([]string, len(matches)) for i, m := range matches { results[i] = m.Path } return results, truncated, nil } ================================================ FILE: internal/format/format.go ================================================ package format import ( "encoding/json" "fmt" "strings" ) // OutputFormat represents the output format type for non-interactive mode type OutputFormat string const ( // Text format outputs the AI response as plain text. Text OutputFormat = "text" // JSON format outputs the AI response wrapped in a JSON object. JSON OutputFormat = "json" ) // String returns the string representation of the OutputFormat func (f OutputFormat) String() string { return string(f) } // SupportedFormats is a list of all supported output formats as strings var SupportedFormats = []string{ string(Text), string(JSON), } // Parse converts a string to an OutputFormat func Parse(s string) (OutputFormat, error) { s = strings.ToLower(strings.TrimSpace(s)) switch s { case string(Text): return Text, nil case string(JSON): return JSON, nil default: return "", fmt.Errorf("invalid format: %s", s) } } // IsValid checks if the provided format string is supported func IsValid(s string) bool { _, err := Parse(s) return err == nil } // GetHelpText returns a formatted string describing all supported formats func GetHelpText() string { return fmt.Sprintf(`Supported output formats: - %s: Plain text output (default) - %s: Output wrapped in a JSON object`, Text, JSON) } // FormatOutput formats the AI response according to the specified format func FormatOutput(content string, formatStr string) string { format, err := Parse(formatStr) if err != nil { // Default to text format on error return content } switch format { case JSON: return formatAsJSON(content) case Text: fallthrough default: return content } } // formatAsJSON wraps the content in a simple JSON object func formatAsJSON(content string) string { // Use the JSON package to properly escape the content response := struct { Response string `json:"response"` }{ Response: content, } jsonBytes, err := json.MarshalIndent(response, "", " ") if err != nil { // In case of an error, return a manually formatted JSON jsonEscaped := strings.Replace(content, "\\", "\\\\", -1) jsonEscaped = strings.Replace(jsonEscaped, "\"", "\\\"", -1) jsonEscaped = strings.Replace(jsonEscaped, "\n", "\\n", -1) jsonEscaped = strings.Replace(jsonEscaped, "\r", "\\r", -1) jsonEscaped = strings.Replace(jsonEscaped, "\t", "\\t", -1) return fmt.Sprintf("{\n \"response\": \"%s\"\n}", jsonEscaped) } return string(jsonBytes) } ================================================ FILE: internal/format/spinner.go ================================================ package format import ( "context" "fmt" "os" "github.com/charmbracelet/bubbles/spinner" tea "github.com/charmbracelet/bubbletea" ) // Spinner wraps the bubbles spinner for non-interactive mode type Spinner struct { model spinner.Model done chan struct{} prog *tea.Program ctx context.Context cancel context.CancelFunc } // spinnerModel is the tea.Model for the spinner type spinnerModel struct { spinner spinner.Model message string quitting bool } func (m spinnerModel) Init() tea.Cmd { return m.spinner.Tick } func (m spinnerModel) Update(msg tea.Msg) (tea.Model, tea.Cmd) { switch msg := msg.(type) { case tea.KeyMsg: m.quitting = true return m, tea.Quit case spinner.TickMsg: var cmd tea.Cmd m.spinner, cmd = m.spinner.Update(msg) return m, cmd case quitMsg: m.quitting = true return m, tea.Quit default: return m, nil } } func (m spinnerModel) View() string { if m.quitting { return "" } return fmt.Sprintf("%s %s", m.spinner.View(), m.message) } // quitMsg is sent when we want to quit the spinner type quitMsg struct{} // NewSpinner creates a new spinner with the given message func NewSpinner(message string) *Spinner { s := spinner.New() s.Spinner = spinner.Dot s.Style = s.Style.Foreground(s.Style.GetForeground()) ctx, cancel := context.WithCancel(context.Background()) model := spinnerModel{ spinner: s, message: message, } prog := tea.NewProgram(model, tea.WithOutput(os.Stderr), tea.WithoutCatchPanics()) return &Spinner{ model: s, done: make(chan struct{}), prog: prog, ctx: ctx, cancel: cancel, } } // Start begins the spinner animation func (s *Spinner) Start() { go func() { defer close(s.done) go func() { <-s.ctx.Done() s.prog.Send(quitMsg{}) }() _, err := s.prog.Run() if err != nil { fmt.Fprintf(os.Stderr, "Error running spinner: %v\n", err) } }() } // Stop ends the spinner animation func (s *Spinner) Stop() { s.cancel() <-s.done } ================================================ FILE: internal/history/file.go ================================================ package history import ( "context" "database/sql" "fmt" "strconv" "strings" "time" "github.com/google/uuid" "github.com/opencode-ai/opencode/internal/db" "github.com/opencode-ai/opencode/internal/pubsub" ) const ( InitialVersion = "initial" ) type File struct { ID string SessionID string Path string Content string Version string CreatedAt int64 UpdatedAt int64 } type Service interface { pubsub.Suscriber[File] Create(ctx context.Context, sessionID, path, content string) (File, error) CreateVersion(ctx context.Context, sessionID, path, content string) (File, error) Get(ctx context.Context, id string) (File, error) GetByPathAndSession(ctx context.Context, path, sessionID string) (File, error) ListBySession(ctx context.Context, sessionID string) ([]File, error) ListLatestSessionFiles(ctx context.Context, sessionID string) ([]File, error) Update(ctx context.Context, file File) (File, error) Delete(ctx context.Context, id string) error DeleteSessionFiles(ctx context.Context, sessionID string) error } type service struct { *pubsub.Broker[File] db *sql.DB q *db.Queries } func NewService(q *db.Queries, db *sql.DB) Service { return &service{ Broker: pubsub.NewBroker[File](), q: q, db: db, } } func (s *service) Create(ctx context.Context, sessionID, path, content string) (File, error) { return s.createWithVersion(ctx, sessionID, path, content, InitialVersion) } func (s *service) CreateVersion(ctx context.Context, sessionID, path, content string) (File, error) { // Get the latest version for this path files, err := s.q.ListFilesByPath(ctx, path) if err != nil { return File{}, err } if len(files) == 0 { // No previous versions, create initial return s.Create(ctx, sessionID, path, content) } // Get the latest version latestFile := files[0] // Files are ordered by created_at DESC latestVersion := latestFile.Version // Generate the next version var nextVersion string if latestVersion == InitialVersion { nextVersion = "v1" } else if strings.HasPrefix(latestVersion, "v") { versionNum, err := strconv.Atoi(latestVersion[1:]) if err != nil { // If we can't parse the version, just use a timestamp-based version nextVersion = fmt.Sprintf("v%d", latestFile.CreatedAt) } else { nextVersion = fmt.Sprintf("v%d", versionNum+1) } } else { // If the version format is unexpected, use a timestamp-based version nextVersion = fmt.Sprintf("v%d", latestFile.CreatedAt) } return s.createWithVersion(ctx, sessionID, path, content, nextVersion) } func (s *service) createWithVersion(ctx context.Context, sessionID, path, content, version string) (File, error) { // Maximum number of retries for transaction conflicts const maxRetries = 3 var file File var err error // Retry loop for transaction conflicts for attempt := range maxRetries { // Start a transaction tx, txErr := s.db.Begin() if txErr != nil { return File{}, fmt.Errorf("failed to begin transaction: %w", txErr) } // Create a new queries instance with the transaction qtx := s.q.WithTx(tx) // Try to create the file within the transaction dbFile, txErr := qtx.CreateFile(ctx, db.CreateFileParams{ ID: uuid.New().String(), SessionID: sessionID, Path: path, Content: content, Version: version, }) if txErr != nil { // Rollback the transaction tx.Rollback() // Check if this is a uniqueness constraint violation if strings.Contains(txErr.Error(), "UNIQUE constraint failed") { if attempt < maxRetries-1 { // If we have retries left, generate a new version and try again if strings.HasPrefix(version, "v") { versionNum, parseErr := strconv.Atoi(version[1:]) if parseErr == nil { version = fmt.Sprintf("v%d", versionNum+1) continue } } // If we can't parse the version, use a timestamp-based version version = fmt.Sprintf("v%d", time.Now().Unix()) continue } } return File{}, txErr } // Commit the transaction if txErr = tx.Commit(); txErr != nil { return File{}, fmt.Errorf("failed to commit transaction: %w", txErr) } file = s.fromDBItem(dbFile) s.Publish(pubsub.CreatedEvent, file) return file, nil } return file, err } func (s *service) Get(ctx context.Context, id string) (File, error) { dbFile, err := s.q.GetFile(ctx, id) if err != nil { return File{}, err } return s.fromDBItem(dbFile), nil } func (s *service) GetByPathAndSession(ctx context.Context, path, sessionID string) (File, error) { dbFile, err := s.q.GetFileByPathAndSession(ctx, db.GetFileByPathAndSessionParams{ Path: path, SessionID: sessionID, }) if err != nil { return File{}, err } return s.fromDBItem(dbFile), nil } func (s *service) ListBySession(ctx context.Context, sessionID string) ([]File, error) { dbFiles, err := s.q.ListFilesBySession(ctx, sessionID) if err != nil { return nil, err } files := make([]File, len(dbFiles)) for i, dbFile := range dbFiles { files[i] = s.fromDBItem(dbFile) } return files, nil } func (s *service) ListLatestSessionFiles(ctx context.Context, sessionID string) ([]File, error) { dbFiles, err := s.q.ListLatestSessionFiles(ctx, sessionID) if err != nil { return nil, err } files := make([]File, len(dbFiles)) for i, dbFile := range dbFiles { files[i] = s.fromDBItem(dbFile) } return files, nil } func (s *service) Update(ctx context.Context, file File) (File, error) { dbFile, err := s.q.UpdateFile(ctx, db.UpdateFileParams{ ID: file.ID, Content: file.Content, Version: file.Version, }) if err != nil { return File{}, err } updatedFile := s.fromDBItem(dbFile) s.Publish(pubsub.UpdatedEvent, updatedFile) return updatedFile, nil } func (s *service) Delete(ctx context.Context, id string) error { file, err := s.Get(ctx, id) if err != nil { return err } err = s.q.DeleteFile(ctx, id) if err != nil { return err } s.Publish(pubsub.DeletedEvent, file) return nil } func (s *service) DeleteSessionFiles(ctx context.Context, sessionID string) error { files, err := s.ListBySession(ctx, sessionID) if err != nil { return err } for _, file := range files { err = s.Delete(ctx, file.ID) if err != nil { return err } } return nil } func (s *service) fromDBItem(item db.File) File { return File{ ID: item.ID, SessionID: item.SessionID, Path: item.Path, Content: item.Content, Version: item.Version, CreatedAt: item.CreatedAt, UpdatedAt: item.UpdatedAt, } } ================================================ FILE: internal/llm/agent/agent-tool.go ================================================ package agent import ( "context" "encoding/json" "fmt" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/llm/tools" "github.com/opencode-ai/opencode/internal/lsp" "github.com/opencode-ai/opencode/internal/message" "github.com/opencode-ai/opencode/internal/session" ) type agentTool struct { sessions session.Service messages message.Service lspClients map[string]*lsp.Client } const ( AgentToolName = "agent" ) type AgentParams struct { Prompt string `json:"prompt"` } func (b *agentTool) Info() tools.ToolInfo { return tools.ToolInfo{ Name: AgentToolName, Description: "Launch a new agent that has access to the following tools: GlobTool, GrepTool, LS, View. When you are searching for a keyword or file and are not confident that you will find the right match on the first try, use the Agent tool to perform the search for you. For example:\n\n- If you are searching for a keyword like \"config\" or \"logger\", or for questions like \"which file does X?\", the Agent tool is strongly recommended\n- If you want to read a specific file path, use the View or GlobTool tool instead of the Agent tool, to find the match more quickly\n- If you are searching for a specific class definition like \"class Foo\", use the GlobTool tool instead, to find the match more quickly\n\nUsage notes:\n1. Launch multiple agents concurrently whenever possible, to maximize performance; to do that, use a single message with multiple tool uses\n2. When the agent is done, it will return a single message back to you. The result returned by the agent is not visible to the user. To show the user the result, you should send a text message back to the user with a concise summary of the result.\n3. Each agent invocation is stateless. You will not be able to send additional messages to the agent, nor will the agent be able to communicate with you outside of its final report. Therefore, your prompt should contain a highly detailed task description for the agent to perform autonomously and you should specify exactly what information the agent should return back to you in its final and only message to you.\n4. The agent's outputs should generally be trusted\n5. IMPORTANT: The agent can not use Bash, Replace, Edit, so can not modify files. If you want to use these tools, use them directly instead of going through the agent.", Parameters: map[string]any{ "prompt": map[string]any{ "type": "string", "description": "The task for the agent to perform", }, }, Required: []string{"prompt"}, } } func (b *agentTool) Run(ctx context.Context, call tools.ToolCall) (tools.ToolResponse, error) { var params AgentParams if err := json.Unmarshal([]byte(call.Input), ¶ms); err != nil { return tools.NewTextErrorResponse(fmt.Sprintf("error parsing parameters: %s", err)), nil } if params.Prompt == "" { return tools.NewTextErrorResponse("prompt is required"), nil } sessionID, messageID := tools.GetContextValues(ctx) if sessionID == "" || messageID == "" { return tools.ToolResponse{}, fmt.Errorf("session_id and message_id are required") } agent, err := NewAgent(config.AgentTask, b.sessions, b.messages, TaskAgentTools(b.lspClients)) if err != nil { return tools.ToolResponse{}, fmt.Errorf("error creating agent: %s", err) } session, err := b.sessions.CreateTaskSession(ctx, call.ID, sessionID, "New Agent Session") if err != nil { return tools.ToolResponse{}, fmt.Errorf("error creating session: %s", err) } done, err := agent.Run(ctx, session.ID, params.Prompt) if err != nil { return tools.ToolResponse{}, fmt.Errorf("error generating agent: %s", err) } result := <-done if result.Error != nil { return tools.ToolResponse{}, fmt.Errorf("error generating agent: %s", result.Error) } response := result.Message if response.Role != message.Assistant { return tools.NewTextErrorResponse("no response"), nil } updatedSession, err := b.sessions.Get(ctx, session.ID) if err != nil { return tools.ToolResponse{}, fmt.Errorf("error getting session: %s", err) } parentSession, err := b.sessions.Get(ctx, sessionID) if err != nil { return tools.ToolResponse{}, fmt.Errorf("error getting parent session: %s", err) } parentSession.Cost += updatedSession.Cost _, err = b.sessions.Save(ctx, parentSession) if err != nil { return tools.ToolResponse{}, fmt.Errorf("error saving parent session: %s", err) } return tools.NewTextResponse(response.Content().String()), nil } func NewAgentTool( Sessions session.Service, Messages message.Service, LspClients map[string]*lsp.Client, ) tools.BaseTool { return &agentTool{ sessions: Sessions, messages: Messages, lspClients: LspClients, } } ================================================ FILE: internal/llm/agent/agent.go ================================================ package agent import ( "context" "errors" "fmt" "strings" "sync" "time" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/llm/models" "github.com/opencode-ai/opencode/internal/llm/prompt" "github.com/opencode-ai/opencode/internal/llm/provider" "github.com/opencode-ai/opencode/internal/llm/tools" "github.com/opencode-ai/opencode/internal/logging" "github.com/opencode-ai/opencode/internal/message" "github.com/opencode-ai/opencode/internal/permission" "github.com/opencode-ai/opencode/internal/pubsub" "github.com/opencode-ai/opencode/internal/session" ) // Common errors var ( ErrRequestCancelled = errors.New("request cancelled by user") ErrSessionBusy = errors.New("session is currently processing another request") ) type AgentEventType string const ( AgentEventTypeError AgentEventType = "error" AgentEventTypeResponse AgentEventType = "response" AgentEventTypeSummarize AgentEventType = "summarize" ) type AgentEvent struct { Type AgentEventType Message message.Message Error error // When summarizing SessionID string Progress string Done bool } type Service interface { pubsub.Suscriber[AgentEvent] Model() models.Model Run(ctx context.Context, sessionID string, content string, attachments ...message.Attachment) (<-chan AgentEvent, error) Cancel(sessionID string) IsSessionBusy(sessionID string) bool IsBusy() bool Update(agentName config.AgentName, modelID models.ModelID) (models.Model, error) Summarize(ctx context.Context, sessionID string) error } type agent struct { *pubsub.Broker[AgentEvent] sessions session.Service messages message.Service tools []tools.BaseTool provider provider.Provider titleProvider provider.Provider summarizeProvider provider.Provider activeRequests sync.Map } func NewAgent( agentName config.AgentName, sessions session.Service, messages message.Service, agentTools []tools.BaseTool, ) (Service, error) { agentProvider, err := createAgentProvider(agentName) if err != nil { return nil, err } var titleProvider provider.Provider // Only generate titles for the coder agent if agentName == config.AgentCoder { titleProvider, err = createAgentProvider(config.AgentTitle) if err != nil { return nil, err } } var summarizeProvider provider.Provider if agentName == config.AgentCoder { summarizeProvider, err = createAgentProvider(config.AgentSummarizer) if err != nil { return nil, err } } agent := &agent{ Broker: pubsub.NewBroker[AgentEvent](), provider: agentProvider, messages: messages, sessions: sessions, tools: agentTools, titleProvider: titleProvider, summarizeProvider: summarizeProvider, activeRequests: sync.Map{}, } return agent, nil } func (a *agent) Model() models.Model { return a.provider.Model() } func (a *agent) Cancel(sessionID string) { // Cancel regular requests if cancelFunc, exists := a.activeRequests.LoadAndDelete(sessionID); exists { if cancel, ok := cancelFunc.(context.CancelFunc); ok { logging.InfoPersist(fmt.Sprintf("Request cancellation initiated for session: %s", sessionID)) cancel() } } // Also check for summarize requests if cancelFunc, exists := a.activeRequests.LoadAndDelete(sessionID + "-summarize"); exists { if cancel, ok := cancelFunc.(context.CancelFunc); ok { logging.InfoPersist(fmt.Sprintf("Summarize cancellation initiated for session: %s", sessionID)) cancel() } } } func (a *agent) IsBusy() bool { busy := false a.activeRequests.Range(func(key, value interface{}) bool { if cancelFunc, ok := value.(context.CancelFunc); ok { if cancelFunc != nil { busy = true return false // Stop iterating } } return true // Continue iterating }) return busy } func (a *agent) IsSessionBusy(sessionID string) bool { _, busy := a.activeRequests.Load(sessionID) return busy } func (a *agent) generateTitle(ctx context.Context, sessionID string, content string) error { if content == "" { return nil } if a.titleProvider == nil { return nil } session, err := a.sessions.Get(ctx, sessionID) if err != nil { return err } ctx = context.WithValue(ctx, tools.SessionIDContextKey, sessionID) parts := []message.ContentPart{message.TextContent{Text: content}} response, err := a.titleProvider.SendMessages( ctx, []message.Message{ { Role: message.User, Parts: parts, }, }, make([]tools.BaseTool, 0), ) if err != nil { return err } title := strings.TrimSpace(strings.ReplaceAll(response.Content, "\n", " ")) if title == "" { return nil } session.Title = title _, err = a.sessions.Save(ctx, session) return err } func (a *agent) err(err error) AgentEvent { return AgentEvent{ Type: AgentEventTypeError, Error: err, } } func (a *agent) Run(ctx context.Context, sessionID string, content string, attachments ...message.Attachment) (<-chan AgentEvent, error) { if !a.provider.Model().SupportsAttachments && attachments != nil { attachments = nil } events := make(chan AgentEvent) if a.IsSessionBusy(sessionID) { return nil, ErrSessionBusy } genCtx, cancel := context.WithCancel(ctx) a.activeRequests.Store(sessionID, cancel) go func() { logging.Debug("Request started", "sessionID", sessionID) defer logging.RecoverPanic("agent.Run", func() { events <- a.err(fmt.Errorf("panic while running the agent")) }) var attachmentParts []message.ContentPart for _, attachment := range attachments { attachmentParts = append(attachmentParts, message.BinaryContent{Path: attachment.FilePath, MIMEType: attachment.MimeType, Data: attachment.Content}) } result := a.processGeneration(genCtx, sessionID, content, attachmentParts) if result.Error != nil && !errors.Is(result.Error, ErrRequestCancelled) && !errors.Is(result.Error, context.Canceled) { logging.ErrorPersist(result.Error.Error()) } logging.Debug("Request completed", "sessionID", sessionID) a.activeRequests.Delete(sessionID) cancel() a.Publish(pubsub.CreatedEvent, result) events <- result close(events) }() return events, nil } func (a *agent) processGeneration(ctx context.Context, sessionID, content string, attachmentParts []message.ContentPart) AgentEvent { cfg := config.Get() // List existing messages; if none, start title generation asynchronously. msgs, err := a.messages.List(ctx, sessionID) if err != nil { return a.err(fmt.Errorf("failed to list messages: %w", err)) } if len(msgs) == 0 { go func() { defer logging.RecoverPanic("agent.Run", func() { logging.ErrorPersist("panic while generating title") }) titleErr := a.generateTitle(context.Background(), sessionID, content) if titleErr != nil { logging.ErrorPersist(fmt.Sprintf("failed to generate title: %v", titleErr)) } }() } session, err := a.sessions.Get(ctx, sessionID) if err != nil { return a.err(fmt.Errorf("failed to get session: %w", err)) } if session.SummaryMessageID != "" { summaryMsgInex := -1 for i, msg := range msgs { if msg.ID == session.SummaryMessageID { summaryMsgInex = i break } } if summaryMsgInex != -1 { msgs = msgs[summaryMsgInex:] msgs[0].Role = message.User } } userMsg, err := a.createUserMessage(ctx, sessionID, content, attachmentParts) if err != nil { return a.err(fmt.Errorf("failed to create user message: %w", err)) } // Append the new user message to the conversation history. msgHistory := append(msgs, userMsg) for { // Check for cancellation before each iteration select { case <-ctx.Done(): return a.err(ctx.Err()) default: // Continue processing } agentMessage, toolResults, err := a.streamAndHandleEvents(ctx, sessionID, msgHistory) if err != nil { if errors.Is(err, context.Canceled) { agentMessage.AddFinish(message.FinishReasonCanceled) a.messages.Update(context.Background(), agentMessage) return a.err(ErrRequestCancelled) } return a.err(fmt.Errorf("failed to process events: %w", err)) } if cfg.Debug { seqId := (len(msgHistory) + 1) / 2 toolResultFilepath := logging.WriteToolResultsJson(sessionID, seqId, toolResults) logging.Info("Result", "message", agentMessage.FinishReason(), "toolResults", "{}", "filepath", toolResultFilepath) } else { logging.Info("Result", "message", agentMessage.FinishReason(), "toolResults", toolResults) } if (agentMessage.FinishReason() == message.FinishReasonToolUse) && toolResults != nil { // We are not done, we need to respond with the tool response msgHistory = append(msgHistory, agentMessage, *toolResults) continue } return AgentEvent{ Type: AgentEventTypeResponse, Message: agentMessage, Done: true, } } } func (a *agent) createUserMessage(ctx context.Context, sessionID, content string, attachmentParts []message.ContentPart) (message.Message, error) { parts := []message.ContentPart{message.TextContent{Text: content}} parts = append(parts, attachmentParts...) return a.messages.Create(ctx, sessionID, message.CreateMessageParams{ Role: message.User, Parts: parts, }) } func (a *agent) streamAndHandleEvents(ctx context.Context, sessionID string, msgHistory []message.Message) (message.Message, *message.Message, error) { ctx = context.WithValue(ctx, tools.SessionIDContextKey, sessionID) eventChan := a.provider.StreamResponse(ctx, msgHistory, a.tools) assistantMsg, err := a.messages.Create(ctx, sessionID, message.CreateMessageParams{ Role: message.Assistant, Parts: []message.ContentPart{}, Model: a.provider.Model().ID, }) if err != nil { return assistantMsg, nil, fmt.Errorf("failed to create assistant message: %w", err) } // Add the session and message ID into the context if needed by tools. ctx = context.WithValue(ctx, tools.MessageIDContextKey, assistantMsg.ID) // Process each event in the stream. for event := range eventChan { if processErr := a.processEvent(ctx, sessionID, &assistantMsg, event); processErr != nil { a.finishMessage(ctx, &assistantMsg, message.FinishReasonCanceled) return assistantMsg, nil, processErr } if ctx.Err() != nil { a.finishMessage(context.Background(), &assistantMsg, message.FinishReasonCanceled) return assistantMsg, nil, ctx.Err() } } toolResults := make([]message.ToolResult, len(assistantMsg.ToolCalls())) toolCalls := assistantMsg.ToolCalls() for i, toolCall := range toolCalls { select { case <-ctx.Done(): a.finishMessage(context.Background(), &assistantMsg, message.FinishReasonCanceled) // Make all future tool calls cancelled for j := i; j < len(toolCalls); j++ { toolResults[j] = message.ToolResult{ ToolCallID: toolCalls[j].ID, Content: "Tool execution canceled by user", IsError: true, } } goto out default: // Continue processing var tool tools.BaseTool for _, availableTool := range a.tools { if availableTool.Info().Name == toolCall.Name { tool = availableTool break } // Monkey patch for Copilot Sonnet-4 tool repetition obfuscation // if strings.HasPrefix(toolCall.Name, availableTool.Info().Name) && // strings.HasPrefix(toolCall.Name, availableTool.Info().Name+availableTool.Info().Name) { // tool = availableTool // break // } } // Tool not found if tool == nil { toolResults[i] = message.ToolResult{ ToolCallID: toolCall.ID, Content: fmt.Sprintf("Tool not found: %s", toolCall.Name), IsError: true, } continue } toolResult, toolErr := tool.Run(ctx, tools.ToolCall{ ID: toolCall.ID, Name: toolCall.Name, Input: toolCall.Input, }) if toolErr != nil { if errors.Is(toolErr, permission.ErrorPermissionDenied) { toolResults[i] = message.ToolResult{ ToolCallID: toolCall.ID, Content: "Permission denied", IsError: true, } for j := i + 1; j < len(toolCalls); j++ { toolResults[j] = message.ToolResult{ ToolCallID: toolCalls[j].ID, Content: "Tool execution canceled by user", IsError: true, } } a.finishMessage(ctx, &assistantMsg, message.FinishReasonPermissionDenied) break } } toolResults[i] = message.ToolResult{ ToolCallID: toolCall.ID, Content: toolResult.Content, Metadata: toolResult.Metadata, IsError: toolResult.IsError, } } } out: if len(toolResults) == 0 { return assistantMsg, nil, nil } parts := make([]message.ContentPart, 0) for _, tr := range toolResults { parts = append(parts, tr) } msg, err := a.messages.Create(context.Background(), assistantMsg.SessionID, message.CreateMessageParams{ Role: message.Tool, Parts: parts, }) if err != nil { return assistantMsg, nil, fmt.Errorf("failed to create cancelled tool message: %w", err) } return assistantMsg, &msg, err } func (a *agent) finishMessage(ctx context.Context, msg *message.Message, finishReson message.FinishReason) { msg.AddFinish(finishReson) _ = a.messages.Update(ctx, *msg) } func (a *agent) processEvent(ctx context.Context, sessionID string, assistantMsg *message.Message, event provider.ProviderEvent) error { select { case <-ctx.Done(): return ctx.Err() default: // Continue processing. } switch event.Type { case provider.EventThinkingDelta: assistantMsg.AppendReasoningContent(event.Content) return a.messages.Update(ctx, *assistantMsg) case provider.EventContentDelta: assistantMsg.AppendContent(event.Content) return a.messages.Update(ctx, *assistantMsg) case provider.EventToolUseStart: assistantMsg.AddToolCall(*event.ToolCall) return a.messages.Update(ctx, *assistantMsg) // TODO: see how to handle this // case provider.EventToolUseDelta: // tm := time.Unix(assistantMsg.UpdatedAt, 0) // assistantMsg.AppendToolCallInput(event.ToolCall.ID, event.ToolCall.Input) // if time.Since(tm) > 1000*time.Millisecond { // err := a.messages.Update(ctx, *assistantMsg) // assistantMsg.UpdatedAt = time.Now().Unix() // return err // } case provider.EventToolUseStop: assistantMsg.FinishToolCall(event.ToolCall.ID) return a.messages.Update(ctx, *assistantMsg) case provider.EventError: if errors.Is(event.Error, context.Canceled) { logging.InfoPersist(fmt.Sprintf("Event processing canceled for session: %s", sessionID)) return context.Canceled } logging.ErrorPersist(event.Error.Error()) return event.Error case provider.EventComplete: assistantMsg.SetToolCalls(event.Response.ToolCalls) assistantMsg.AddFinish(event.Response.FinishReason) if err := a.messages.Update(ctx, *assistantMsg); err != nil { return fmt.Errorf("failed to update message: %w", err) } return a.TrackUsage(ctx, sessionID, a.provider.Model(), event.Response.Usage) } return nil } func (a *agent) TrackUsage(ctx context.Context, sessionID string, model models.Model, usage provider.TokenUsage) error { sess, err := a.sessions.Get(ctx, sessionID) if err != nil { return fmt.Errorf("failed to get session: %w", err) } cost := model.CostPer1MInCached/1e6*float64(usage.CacheCreationTokens) + model.CostPer1MOutCached/1e6*float64(usage.CacheReadTokens) + model.CostPer1MIn/1e6*float64(usage.InputTokens) + model.CostPer1MOut/1e6*float64(usage.OutputTokens) sess.Cost += cost sess.CompletionTokens = usage.OutputTokens + usage.CacheReadTokens sess.PromptTokens = usage.InputTokens + usage.CacheCreationTokens _, err = a.sessions.Save(ctx, sess) if err != nil { return fmt.Errorf("failed to save session: %w", err) } return nil } func (a *agent) Update(agentName config.AgentName, modelID models.ModelID) (models.Model, error) { if a.IsBusy() { return models.Model{}, fmt.Errorf("cannot change model while processing requests") } if err := config.UpdateAgentModel(agentName, modelID); err != nil { return models.Model{}, fmt.Errorf("failed to update config: %w", err) } provider, err := createAgentProvider(agentName) if err != nil { return models.Model{}, fmt.Errorf("failed to create provider for model %s: %w", modelID, err) } a.provider = provider return a.provider.Model(), nil } func (a *agent) Summarize(ctx context.Context, sessionID string) error { if a.summarizeProvider == nil { return fmt.Errorf("summarize provider not available") } // Check if session is busy if a.IsSessionBusy(sessionID) { return ErrSessionBusy } // Create a new context with cancellation summarizeCtx, cancel := context.WithCancel(ctx) // Store the cancel function in activeRequests to allow cancellation a.activeRequests.Store(sessionID+"-summarize", cancel) go func() { defer a.activeRequests.Delete(sessionID + "-summarize") defer cancel() event := AgentEvent{ Type: AgentEventTypeSummarize, Progress: "Starting summarization...", } a.Publish(pubsub.CreatedEvent, event) // Get all messages from the session msgs, err := a.messages.List(summarizeCtx, sessionID) if err != nil { event = AgentEvent{ Type: AgentEventTypeError, Error: fmt.Errorf("failed to list messages: %w", err), Done: true, } a.Publish(pubsub.CreatedEvent, event) return } summarizeCtx = context.WithValue(summarizeCtx, tools.SessionIDContextKey, sessionID) if len(msgs) == 0 { event = AgentEvent{ Type: AgentEventTypeError, Error: fmt.Errorf("no messages to summarize"), Done: true, } a.Publish(pubsub.CreatedEvent, event) return } event = AgentEvent{ Type: AgentEventTypeSummarize, Progress: "Analyzing conversation...", } a.Publish(pubsub.CreatedEvent, event) // Add a system message to guide the summarization summarizePrompt := "Provide a detailed but concise summary of our conversation above. Focus on information that would be helpful for continuing the conversation, including what we did, what we're doing, which files we're working on, and what we're going to do next." // Create a new message with the summarize prompt promptMsg := message.Message{ Role: message.User, Parts: []message.ContentPart{message.TextContent{Text: summarizePrompt}}, } // Append the prompt to the messages msgsWithPrompt := append(msgs, promptMsg) event = AgentEvent{ Type: AgentEventTypeSummarize, Progress: "Generating summary...", } a.Publish(pubsub.CreatedEvent, event) // Send the messages to the summarize provider response, err := a.summarizeProvider.SendMessages( summarizeCtx, msgsWithPrompt, make([]tools.BaseTool, 0), ) if err != nil { event = AgentEvent{ Type: AgentEventTypeError, Error: fmt.Errorf("failed to summarize: %w", err), Done: true, } a.Publish(pubsub.CreatedEvent, event) return } summary := strings.TrimSpace(response.Content) if summary == "" { event = AgentEvent{ Type: AgentEventTypeError, Error: fmt.Errorf("empty summary returned"), Done: true, } a.Publish(pubsub.CreatedEvent, event) return } event = AgentEvent{ Type: AgentEventTypeSummarize, Progress: "Creating new session...", } a.Publish(pubsub.CreatedEvent, event) oldSession, err := a.sessions.Get(summarizeCtx, sessionID) if err != nil { event = AgentEvent{ Type: AgentEventTypeError, Error: fmt.Errorf("failed to get session: %w", err), Done: true, } a.Publish(pubsub.CreatedEvent, event) return } // Create a message in the new session with the summary msg, err := a.messages.Create(summarizeCtx, oldSession.ID, message.CreateMessageParams{ Role: message.Assistant, Parts: []message.ContentPart{ message.TextContent{Text: summary}, message.Finish{ Reason: message.FinishReasonEndTurn, Time: time.Now().Unix(), }, }, Model: a.summarizeProvider.Model().ID, }) if err != nil { event = AgentEvent{ Type: AgentEventTypeError, Error: fmt.Errorf("failed to create summary message: %w", err), Done: true, } a.Publish(pubsub.CreatedEvent, event) return } oldSession.SummaryMessageID = msg.ID oldSession.CompletionTokens = response.Usage.OutputTokens oldSession.PromptTokens = 0 model := a.summarizeProvider.Model() usage := response.Usage cost := model.CostPer1MInCached/1e6*float64(usage.CacheCreationTokens) + model.CostPer1MOutCached/1e6*float64(usage.CacheReadTokens) + model.CostPer1MIn/1e6*float64(usage.InputTokens) + model.CostPer1MOut/1e6*float64(usage.OutputTokens) oldSession.Cost += cost _, err = a.sessions.Save(summarizeCtx, oldSession) if err != nil { event = AgentEvent{ Type: AgentEventTypeError, Error: fmt.Errorf("failed to save session: %w", err), Done: true, } a.Publish(pubsub.CreatedEvent, event) } event = AgentEvent{ Type: AgentEventTypeSummarize, SessionID: oldSession.ID, Progress: "Summary complete", Done: true, } a.Publish(pubsub.CreatedEvent, event) // Send final success event with the new session ID }() return nil } func createAgentProvider(agentName config.AgentName) (provider.Provider, error) { cfg := config.Get() agentConfig, ok := cfg.Agents[agentName] if !ok { return nil, fmt.Errorf("agent %s not found", agentName) } model, ok := models.SupportedModels[agentConfig.Model] if !ok { return nil, fmt.Errorf("model %s not supported", agentConfig.Model) } providerCfg, ok := cfg.Providers[model.Provider] if !ok { return nil, fmt.Errorf("provider %s not supported", model.Provider) } if providerCfg.Disabled { return nil, fmt.Errorf("provider %s is not enabled", model.Provider) } maxTokens := model.DefaultMaxTokens if agentConfig.MaxTokens > 0 { maxTokens = agentConfig.MaxTokens } opts := []provider.ProviderClientOption{ provider.WithAPIKey(providerCfg.APIKey), provider.WithModel(model), provider.WithSystemMessage(prompt.GetAgentPrompt(agentName, model.Provider)), provider.WithMaxTokens(maxTokens), } if model.Provider == models.ProviderOpenAI || model.Provider == models.ProviderLocal && model.CanReason { opts = append( opts, provider.WithOpenAIOptions( provider.WithReasoningEffort(agentConfig.ReasoningEffort), ), ) } else if model.Provider == models.ProviderAnthropic && model.CanReason && agentName == config.AgentCoder { opts = append( opts, provider.WithAnthropicOptions( provider.WithAnthropicShouldThinkFn(provider.DefaultShouldThinkFn), ), ) } agentProvider, err := provider.NewProvider( model.Provider, opts..., ) if err != nil { return nil, fmt.Errorf("could not create provider: %v", err) } return agentProvider, nil } ================================================ FILE: internal/llm/agent/mcp-tools.go ================================================ package agent import ( "context" "encoding/json" "fmt" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/llm/tools" "github.com/opencode-ai/opencode/internal/logging" "github.com/opencode-ai/opencode/internal/permission" "github.com/opencode-ai/opencode/internal/version" "github.com/mark3labs/mcp-go/client" "github.com/mark3labs/mcp-go/mcp" ) type mcpTool struct { mcpName string tool mcp.Tool mcpConfig config.MCPServer permissions permission.Service } type MCPClient interface { Initialize( ctx context.Context, request mcp.InitializeRequest, ) (*mcp.InitializeResult, error) ListTools(ctx context.Context, request mcp.ListToolsRequest) (*mcp.ListToolsResult, error) CallTool(ctx context.Context, request mcp.CallToolRequest) (*mcp.CallToolResult, error) Close() error } func (b *mcpTool) Info() tools.ToolInfo { required := b.tool.InputSchema.Required if required == nil { required = make([]string, 0) } return tools.ToolInfo{ Name: fmt.Sprintf("%s_%s", b.mcpName, b.tool.Name), Description: b.tool.Description, Parameters: b.tool.InputSchema.Properties, Required: required, } } func runTool(ctx context.Context, c MCPClient, toolName string, input string) (tools.ToolResponse, error) { defer c.Close() initRequest := mcp.InitializeRequest{} initRequest.Params.ProtocolVersion = mcp.LATEST_PROTOCOL_VERSION initRequest.Params.ClientInfo = mcp.Implementation{ Name: "OpenCode", Version: version.Version, } _, err := c.Initialize(ctx, initRequest) if err != nil { return tools.NewTextErrorResponse(err.Error()), nil } toolRequest := mcp.CallToolRequest{} toolRequest.Params.Name = toolName var args map[string]any if err = json.Unmarshal([]byte(input), &args); err != nil { return tools.NewTextErrorResponse(fmt.Sprintf("error parsing parameters: %s", err)), nil } toolRequest.Params.Arguments = args result, err := c.CallTool(ctx, toolRequest) if err != nil { return tools.NewTextErrorResponse(err.Error()), nil } output := "" for _, v := range result.Content { if v, ok := v.(mcp.TextContent); ok { output = v.Text } else { output = fmt.Sprintf("%v", v) } } return tools.NewTextResponse(output), nil } func (b *mcpTool) Run(ctx context.Context, params tools.ToolCall) (tools.ToolResponse, error) { sessionID, messageID := tools.GetContextValues(ctx) if sessionID == "" || messageID == "" { return tools.ToolResponse{}, fmt.Errorf("session ID and message ID are required for creating a new file") } permissionDescription := fmt.Sprintf("execute %s with the following parameters: %s", b.Info().Name, params.Input) p := b.permissions.Request( permission.CreatePermissionRequest{ SessionID: sessionID, Path: config.WorkingDirectory(), ToolName: b.Info().Name, Action: "execute", Description: permissionDescription, Params: params.Input, }, ) if !p { return tools.NewTextErrorResponse("permission denied"), nil } switch b.mcpConfig.Type { case config.MCPStdio: c, err := client.NewStdioMCPClient( b.mcpConfig.Command, b.mcpConfig.Env, b.mcpConfig.Args..., ) if err != nil { return tools.NewTextErrorResponse(err.Error()), nil } return runTool(ctx, c, b.tool.Name, params.Input) case config.MCPSse: c, err := client.NewSSEMCPClient( b.mcpConfig.URL, client.WithHeaders(b.mcpConfig.Headers), ) if err != nil { return tools.NewTextErrorResponse(err.Error()), nil } return runTool(ctx, c, b.tool.Name, params.Input) } return tools.NewTextErrorResponse("invalid mcp type"), nil } func NewMcpTool(name string, tool mcp.Tool, permissions permission.Service, mcpConfig config.MCPServer) tools.BaseTool { return &mcpTool{ mcpName: name, tool: tool, mcpConfig: mcpConfig, permissions: permissions, } } var mcpTools []tools.BaseTool func getTools(ctx context.Context, name string, m config.MCPServer, permissions permission.Service, c MCPClient) []tools.BaseTool { var stdioTools []tools.BaseTool initRequest := mcp.InitializeRequest{} initRequest.Params.ProtocolVersion = mcp.LATEST_PROTOCOL_VERSION initRequest.Params.ClientInfo = mcp.Implementation{ Name: "OpenCode", Version: version.Version, } _, err := c.Initialize(ctx, initRequest) if err != nil { logging.Error("error initializing mcp client", "error", err) return stdioTools } toolsRequest := mcp.ListToolsRequest{} tools, err := c.ListTools(ctx, toolsRequest) if err != nil { logging.Error("error listing tools", "error", err) return stdioTools } for _, t := range tools.Tools { stdioTools = append(stdioTools, NewMcpTool(name, t, permissions, m)) } defer c.Close() return stdioTools } func GetMcpTools(ctx context.Context, permissions permission.Service) []tools.BaseTool { if len(mcpTools) > 0 { return mcpTools } for name, m := range config.Get().MCPServers { switch m.Type { case config.MCPStdio: c, err := client.NewStdioMCPClient( m.Command, m.Env, m.Args..., ) if err != nil { logging.Error("error creating mcp client", "error", err) continue } mcpTools = append(mcpTools, getTools(ctx, name, m, permissions, c)...) case config.MCPSse: c, err := client.NewSSEMCPClient( m.URL, client.WithHeaders(m.Headers), ) if err != nil { logging.Error("error creating mcp client", "error", err) continue } mcpTools = append(mcpTools, getTools(ctx, name, m, permissions, c)...) } } return mcpTools } ================================================ FILE: internal/llm/agent/tools.go ================================================ package agent import ( "context" "github.com/opencode-ai/opencode/internal/history" "github.com/opencode-ai/opencode/internal/llm/tools" "github.com/opencode-ai/opencode/internal/lsp" "github.com/opencode-ai/opencode/internal/message" "github.com/opencode-ai/opencode/internal/permission" "github.com/opencode-ai/opencode/internal/session" ) func CoderAgentTools( permissions permission.Service, sessions session.Service, messages message.Service, history history.Service, lspClients map[string]*lsp.Client, ) []tools.BaseTool { ctx := context.Background() otherTools := GetMcpTools(ctx, permissions) if len(lspClients) > 0 { otherTools = append(otherTools, tools.NewDiagnosticsTool(lspClients)) } return append( []tools.BaseTool{ tools.NewBashTool(permissions), tools.NewEditTool(lspClients, permissions, history), tools.NewFetchTool(permissions), tools.NewGlobTool(), tools.NewGrepTool(), tools.NewLsTool(), tools.NewSourcegraphTool(), tools.NewViewTool(lspClients), tools.NewPatchTool(lspClients, permissions, history), tools.NewWriteTool(lspClients, permissions, history), NewAgentTool(sessions, messages, lspClients), }, otherTools..., ) } func TaskAgentTools(lspClients map[string]*lsp.Client) []tools.BaseTool { return []tools.BaseTool{ tools.NewGlobTool(), tools.NewGrepTool(), tools.NewLsTool(), tools.NewSourcegraphTool(), tools.NewViewTool(lspClients), } } ================================================ FILE: internal/llm/models/anthropic.go ================================================ package models const ( ProviderAnthropic ModelProvider = "anthropic" // Models Claude35Sonnet ModelID = "claude-3.5-sonnet" Claude3Haiku ModelID = "claude-3-haiku" Claude37Sonnet ModelID = "claude-3.7-sonnet" Claude35Haiku ModelID = "claude-3.5-haiku" Claude3Opus ModelID = "claude-3-opus" Claude4Opus ModelID = "claude-4-opus" Claude4Sonnet ModelID = "claude-4-sonnet" ) // https://docs.anthropic.com/en/docs/about-claude/models/all-models var AnthropicModels = map[ModelID]Model{ Claude35Sonnet: { ID: Claude35Sonnet, Name: "Claude 3.5 Sonnet", Provider: ProviderAnthropic, APIModel: "claude-3-5-sonnet-latest", CostPer1MIn: 3.0, CostPer1MInCached: 3.75, CostPer1MOutCached: 0.30, CostPer1MOut: 15.0, ContextWindow: 200000, DefaultMaxTokens: 5000, SupportsAttachments: true, }, Claude3Haiku: { ID: Claude3Haiku, Name: "Claude 3 Haiku", Provider: ProviderAnthropic, APIModel: "claude-3-haiku-20240307", // doesn't support "-latest" CostPer1MIn: 0.25, CostPer1MInCached: 0.30, CostPer1MOutCached: 0.03, CostPer1MOut: 1.25, ContextWindow: 200000, DefaultMaxTokens: 4096, SupportsAttachments: true, }, Claude37Sonnet: { ID: Claude37Sonnet, Name: "Claude 3.7 Sonnet", Provider: ProviderAnthropic, APIModel: "claude-3-7-sonnet-latest", CostPer1MIn: 3.0, CostPer1MInCached: 3.75, CostPer1MOutCached: 0.30, CostPer1MOut: 15.0, ContextWindow: 200000, DefaultMaxTokens: 50000, CanReason: true, SupportsAttachments: true, }, Claude35Haiku: { ID: Claude35Haiku, Name: "Claude 3.5 Haiku", Provider: ProviderAnthropic, APIModel: "claude-3-5-haiku-latest", CostPer1MIn: 0.80, CostPer1MInCached: 1.0, CostPer1MOutCached: 0.08, CostPer1MOut: 4.0, ContextWindow: 200000, DefaultMaxTokens: 4096, SupportsAttachments: true, }, Claude3Opus: { ID: Claude3Opus, Name: "Claude 3 Opus", Provider: ProviderAnthropic, APIModel: "claude-3-opus-latest", CostPer1MIn: 15.0, CostPer1MInCached: 18.75, CostPer1MOutCached: 1.50, CostPer1MOut: 75.0, ContextWindow: 200000, DefaultMaxTokens: 4096, SupportsAttachments: true, }, Claude4Sonnet: { ID: Claude4Sonnet, Name: "Claude 4 Sonnet", Provider: ProviderAnthropic, APIModel: "claude-sonnet-4-20250514", CostPer1MIn: 3.0, CostPer1MInCached: 3.75, CostPer1MOutCached: 0.30, CostPer1MOut: 15.0, ContextWindow: 200000, DefaultMaxTokens: 50000, CanReason: true, SupportsAttachments: true, }, Claude4Opus: { ID: Claude4Opus, Name: "Claude 4 Opus", Provider: ProviderAnthropic, APIModel: "claude-opus-4-20250514", CostPer1MIn: 15.0, CostPer1MInCached: 18.75, CostPer1MOutCached: 1.50, CostPer1MOut: 75.0, ContextWindow: 200000, DefaultMaxTokens: 4096, SupportsAttachments: true, }, } ================================================ FILE: internal/llm/models/azure.go ================================================ package models const ProviderAzure ModelProvider = "azure" const ( AzureGPT41 ModelID = "azure.gpt-4.1" AzureGPT41Mini ModelID = "azure.gpt-4.1-mini" AzureGPT41Nano ModelID = "azure.gpt-4.1-nano" AzureGPT45Preview ModelID = "azure.gpt-4.5-preview" AzureGPT4o ModelID = "azure.gpt-4o" AzureGPT4oMini ModelID = "azure.gpt-4o-mini" AzureO1 ModelID = "azure.o1" AzureO1Mini ModelID = "azure.o1-mini" AzureO3 ModelID = "azure.o3" AzureO3Mini ModelID = "azure.o3-mini" AzureO4Mini ModelID = "azure.o4-mini" ) var AzureModels = map[ModelID]Model{ AzureGPT41: { ID: AzureGPT41, Name: "Azure OpenAI – GPT 4.1", Provider: ProviderAzure, APIModel: "gpt-4.1", CostPer1MIn: OpenAIModels[GPT41].CostPer1MIn, CostPer1MInCached: OpenAIModels[GPT41].CostPer1MInCached, CostPer1MOut: OpenAIModels[GPT41].CostPer1MOut, CostPer1MOutCached: OpenAIModels[GPT41].CostPer1MOutCached, ContextWindow: OpenAIModels[GPT41].ContextWindow, DefaultMaxTokens: OpenAIModels[GPT41].DefaultMaxTokens, SupportsAttachments: true, }, AzureGPT41Mini: { ID: AzureGPT41Mini, Name: "Azure OpenAI – GPT 4.1 mini", Provider: ProviderAzure, APIModel: "gpt-4.1-mini", CostPer1MIn: OpenAIModels[GPT41Mini].CostPer1MIn, CostPer1MInCached: OpenAIModels[GPT41Mini].CostPer1MInCached, CostPer1MOut: OpenAIModels[GPT41Mini].CostPer1MOut, CostPer1MOutCached: OpenAIModels[GPT41Mini].CostPer1MOutCached, ContextWindow: OpenAIModels[GPT41Mini].ContextWindow, DefaultMaxTokens: OpenAIModels[GPT41Mini].DefaultMaxTokens, SupportsAttachments: true, }, AzureGPT41Nano: { ID: AzureGPT41Nano, Name: "Azure OpenAI – GPT 4.1 nano", Provider: ProviderAzure, APIModel: "gpt-4.1-nano", CostPer1MIn: OpenAIModels[GPT41Nano].CostPer1MIn, CostPer1MInCached: OpenAIModels[GPT41Nano].CostPer1MInCached, CostPer1MOut: OpenAIModels[GPT41Nano].CostPer1MOut, CostPer1MOutCached: OpenAIModels[GPT41Nano].CostPer1MOutCached, ContextWindow: OpenAIModels[GPT41Nano].ContextWindow, DefaultMaxTokens: OpenAIModels[GPT41Nano].DefaultMaxTokens, SupportsAttachments: true, }, AzureGPT45Preview: { ID: AzureGPT45Preview, Name: "Azure OpenAI – GPT 4.5 preview", Provider: ProviderAzure, APIModel: "gpt-4.5-preview", CostPer1MIn: OpenAIModels[GPT45Preview].CostPer1MIn, CostPer1MInCached: OpenAIModels[GPT45Preview].CostPer1MInCached, CostPer1MOut: OpenAIModels[GPT45Preview].CostPer1MOut, CostPer1MOutCached: OpenAIModels[GPT45Preview].CostPer1MOutCached, ContextWindow: OpenAIModels[GPT45Preview].ContextWindow, DefaultMaxTokens: OpenAIModels[GPT45Preview].DefaultMaxTokens, SupportsAttachments: true, }, AzureGPT4o: { ID: AzureGPT4o, Name: "Azure OpenAI – GPT-4o", Provider: ProviderAzure, APIModel: "gpt-4o", CostPer1MIn: OpenAIModels[GPT4o].CostPer1MIn, CostPer1MInCached: OpenAIModels[GPT4o].CostPer1MInCached, CostPer1MOut: OpenAIModels[GPT4o].CostPer1MOut, CostPer1MOutCached: OpenAIModels[GPT4o].CostPer1MOutCached, ContextWindow: OpenAIModels[GPT4o].ContextWindow, DefaultMaxTokens: OpenAIModels[GPT4o].DefaultMaxTokens, SupportsAttachments: true, }, AzureGPT4oMini: { ID: AzureGPT4oMini, Name: "Azure OpenAI – GPT-4o mini", Provider: ProviderAzure, APIModel: "gpt-4o-mini", CostPer1MIn: OpenAIModels[GPT4oMini].CostPer1MIn, CostPer1MInCached: OpenAIModels[GPT4oMini].CostPer1MInCached, CostPer1MOut: OpenAIModels[GPT4oMini].CostPer1MOut, CostPer1MOutCached: OpenAIModels[GPT4oMini].CostPer1MOutCached, ContextWindow: OpenAIModels[GPT4oMini].ContextWindow, DefaultMaxTokens: OpenAIModels[GPT4oMini].DefaultMaxTokens, SupportsAttachments: true, }, AzureO1: { ID: AzureO1, Name: "Azure OpenAI – O1", Provider: ProviderAzure, APIModel: "o1", CostPer1MIn: OpenAIModels[O1].CostPer1MIn, CostPer1MInCached: OpenAIModels[O1].CostPer1MInCached, CostPer1MOut: OpenAIModels[O1].CostPer1MOut, CostPer1MOutCached: OpenAIModels[O1].CostPer1MOutCached, ContextWindow: OpenAIModels[O1].ContextWindow, DefaultMaxTokens: OpenAIModels[O1].DefaultMaxTokens, CanReason: OpenAIModels[O1].CanReason, SupportsAttachments: true, }, AzureO1Mini: { ID: AzureO1Mini, Name: "Azure OpenAI – O1 mini", Provider: ProviderAzure, APIModel: "o1-mini", CostPer1MIn: OpenAIModels[O1Mini].CostPer1MIn, CostPer1MInCached: OpenAIModels[O1Mini].CostPer1MInCached, CostPer1MOut: OpenAIModels[O1Mini].CostPer1MOut, CostPer1MOutCached: OpenAIModels[O1Mini].CostPer1MOutCached, ContextWindow: OpenAIModels[O1Mini].ContextWindow, DefaultMaxTokens: OpenAIModels[O1Mini].DefaultMaxTokens, CanReason: OpenAIModels[O1Mini].CanReason, SupportsAttachments: true, }, AzureO3: { ID: AzureO3, Name: "Azure OpenAI – O3", Provider: ProviderAzure, APIModel: "o3", CostPer1MIn: OpenAIModels[O3].CostPer1MIn, CostPer1MInCached: OpenAIModels[O3].CostPer1MInCached, CostPer1MOut: OpenAIModels[O3].CostPer1MOut, CostPer1MOutCached: OpenAIModels[O3].CostPer1MOutCached, ContextWindow: OpenAIModels[O3].ContextWindow, DefaultMaxTokens: OpenAIModels[O3].DefaultMaxTokens, CanReason: OpenAIModels[O3].CanReason, SupportsAttachments: true, }, AzureO3Mini: { ID: AzureO3Mini, Name: "Azure OpenAI – O3 mini", Provider: ProviderAzure, APIModel: "o3-mini", CostPer1MIn: OpenAIModels[O3Mini].CostPer1MIn, CostPer1MInCached: OpenAIModels[O3Mini].CostPer1MInCached, CostPer1MOut: OpenAIModels[O3Mini].CostPer1MOut, CostPer1MOutCached: OpenAIModels[O3Mini].CostPer1MOutCached, ContextWindow: OpenAIModels[O3Mini].ContextWindow, DefaultMaxTokens: OpenAIModels[O3Mini].DefaultMaxTokens, CanReason: OpenAIModels[O3Mini].CanReason, SupportsAttachments: false, }, AzureO4Mini: { ID: AzureO4Mini, Name: "Azure OpenAI – O4 mini", Provider: ProviderAzure, APIModel: "o4-mini", CostPer1MIn: OpenAIModels[O4Mini].CostPer1MIn, CostPer1MInCached: OpenAIModels[O4Mini].CostPer1MInCached, CostPer1MOut: OpenAIModels[O4Mini].CostPer1MOut, CostPer1MOutCached: OpenAIModels[O4Mini].CostPer1MOutCached, ContextWindow: OpenAIModels[O4Mini].ContextWindow, DefaultMaxTokens: OpenAIModels[O4Mini].DefaultMaxTokens, CanReason: OpenAIModels[O4Mini].CanReason, SupportsAttachments: true, }, } ================================================ FILE: internal/llm/models/copilot.go ================================================ package models const ( ProviderCopilot ModelProvider = "copilot" // GitHub Copilot models CopilotGTP35Turbo ModelID = "copilot.gpt-3.5-turbo" CopilotGPT4o ModelID = "copilot.gpt-4o" CopilotGPT4oMini ModelID = "copilot.gpt-4o-mini" CopilotGPT41 ModelID = "copilot.gpt-4.1" CopilotClaude35 ModelID = "copilot.claude-3.5-sonnet" CopilotClaude37 ModelID = "copilot.claude-3.7-sonnet" CopilotClaude4 ModelID = "copilot.claude-sonnet-4" CopilotO1 ModelID = "copilot.o1" CopilotO3Mini ModelID = "copilot.o3-mini" CopilotO4Mini ModelID = "copilot.o4-mini" CopilotGemini20 ModelID = "copilot.gemini-2.0-flash" CopilotGemini25 ModelID = "copilot.gemini-2.5-pro" CopilotGPT4 ModelID = "copilot.gpt-4" CopilotClaude37Thought ModelID = "copilot.claude-3.7-sonnet-thought" ) var CopilotAnthropicModels = []ModelID{ CopilotClaude35, CopilotClaude37, CopilotClaude37Thought, CopilotClaude4, } // GitHub Copilot models available through GitHub's API var CopilotModels = map[ModelID]Model{ CopilotGTP35Turbo: { ID: CopilotGTP35Turbo, Name: "GitHub Copilot GPT-3.5-turbo", Provider: ProviderCopilot, APIModel: "gpt-3.5-turbo", CostPer1MIn: 0.0, // Included in GitHub Copilot subscription CostPer1MInCached: 0.0, CostPer1MOutCached: 0.0, CostPer1MOut: 0.0, ContextWindow: 16_384, DefaultMaxTokens: 4096, SupportsAttachments: true, }, CopilotGPT4o: { ID: CopilotGPT4o, Name: "GitHub Copilot GPT-4o", Provider: ProviderCopilot, APIModel: "gpt-4o", CostPer1MIn: 0.0, // Included in GitHub Copilot subscription CostPer1MInCached: 0.0, CostPer1MOutCached: 0.0, CostPer1MOut: 0.0, ContextWindow: 128_000, DefaultMaxTokens: 16384, SupportsAttachments: true, }, CopilotGPT4oMini: { ID: CopilotGPT4oMini, Name: "GitHub Copilot GPT-4o Mini", Provider: ProviderCopilot, APIModel: "gpt-4o-mini", CostPer1MIn: 0.0, // Included in GitHub Copilot subscription CostPer1MInCached: 0.0, CostPer1MOutCached: 0.0, CostPer1MOut: 0.0, ContextWindow: 128_000, DefaultMaxTokens: 4096, SupportsAttachments: true, }, CopilotGPT41: { ID: CopilotGPT41, Name: "GitHub Copilot GPT-4.1", Provider: ProviderCopilot, APIModel: "gpt-4.1", CostPer1MIn: 0.0, // Included in GitHub Copilot subscription CostPer1MInCached: 0.0, CostPer1MOutCached: 0.0, CostPer1MOut: 0.0, ContextWindow: 128_000, DefaultMaxTokens: 16384, CanReason: true, SupportsAttachments: true, }, CopilotClaude35: { ID: CopilotClaude35, Name: "GitHub Copilot Claude 3.5 Sonnet", Provider: ProviderCopilot, APIModel: "claude-3.5-sonnet", CostPer1MIn: 0.0, // Included in GitHub Copilot subscription CostPer1MInCached: 0.0, CostPer1MOutCached: 0.0, CostPer1MOut: 0.0, ContextWindow: 90_000, DefaultMaxTokens: 8192, SupportsAttachments: true, }, CopilotClaude37: { ID: CopilotClaude37, Name: "GitHub Copilot Claude 3.7 Sonnet", Provider: ProviderCopilot, APIModel: "claude-3.7-sonnet", CostPer1MIn: 0.0, // Included in GitHub Copilot subscription CostPer1MInCached: 0.0, CostPer1MOutCached: 0.0, CostPer1MOut: 0.0, ContextWindow: 200_000, DefaultMaxTokens: 16384, SupportsAttachments: true, }, CopilotClaude4: { ID: CopilotClaude4, Name: "GitHub Copilot Claude Sonnet 4", Provider: ProviderCopilot, APIModel: "claude-sonnet-4", CostPer1MIn: 0.0, // Included in GitHub Copilot subscription CostPer1MInCached: 0.0, CostPer1MOutCached: 0.0, CostPer1MOut: 0.0, ContextWindow: 128_000, DefaultMaxTokens: 16000, SupportsAttachments: true, }, CopilotO1: { ID: CopilotO1, Name: "GitHub Copilot o1", Provider: ProviderCopilot, APIModel: "o1", CostPer1MIn: 0.0, // Included in GitHub Copilot subscription CostPer1MInCached: 0.0, CostPer1MOutCached: 0.0, CostPer1MOut: 0.0, ContextWindow: 200_000, DefaultMaxTokens: 100_000, CanReason: true, SupportsAttachments: false, }, CopilotO3Mini: { ID: CopilotO3Mini, Name: "GitHub Copilot o3-mini", Provider: ProviderCopilot, APIModel: "o3-mini", CostPer1MIn: 0.0, // Included in GitHub Copilot subscription CostPer1MInCached: 0.0, CostPer1MOutCached: 0.0, CostPer1MOut: 0.0, ContextWindow: 200_000, DefaultMaxTokens: 100_000, CanReason: true, SupportsAttachments: false, }, CopilotO4Mini: { ID: CopilotO4Mini, Name: "GitHub Copilot o4-mini", Provider: ProviderCopilot, APIModel: "o4-mini", CostPer1MIn: 0.0, // Included in GitHub Copilot subscription CostPer1MInCached: 0.0, CostPer1MOutCached: 0.0, CostPer1MOut: 0.0, ContextWindow: 128_000, DefaultMaxTokens: 16_384, CanReason: true, SupportsAttachments: true, }, CopilotGemini20: { ID: CopilotGemini20, Name: "GitHub Copilot Gemini 2.0 Flash", Provider: ProviderCopilot, APIModel: "gemini-2.0-flash-001", CostPer1MIn: 0.0, // Included in GitHub Copilot subscription CostPer1MInCached: 0.0, CostPer1MOutCached: 0.0, CostPer1MOut: 0.0, ContextWindow: 1_000_000, DefaultMaxTokens: 8192, SupportsAttachments: true, }, CopilotGemini25: { ID: CopilotGemini25, Name: "GitHub Copilot Gemini 2.5 Pro", Provider: ProviderCopilot, APIModel: "gemini-2.5-pro", CostPer1MIn: 0.0, // Included in GitHub Copilot subscription CostPer1MInCached: 0.0, CostPer1MOutCached: 0.0, CostPer1MOut: 0.0, ContextWindow: 128_000, DefaultMaxTokens: 64000, SupportsAttachments: true, }, CopilotGPT4: { ID: CopilotGPT4, Name: "GitHub Copilot GPT-4", Provider: ProviderCopilot, APIModel: "gpt-4", CostPer1MIn: 0.0, // Included in GitHub Copilot subscription CostPer1MInCached: 0.0, CostPer1MOutCached: 0.0, CostPer1MOut: 0.0, ContextWindow: 32_768, DefaultMaxTokens: 4096, SupportsAttachments: true, }, CopilotClaude37Thought: { ID: CopilotClaude37Thought, Name: "GitHub Copilot Claude 3.7 Sonnet Thinking", Provider: ProviderCopilot, APIModel: "claude-3.7-sonnet-thought", CostPer1MIn: 0.0, // Included in GitHub Copilot subscription CostPer1MInCached: 0.0, CostPer1MOutCached: 0.0, CostPer1MOut: 0.0, ContextWindow: 200_000, DefaultMaxTokens: 16384, CanReason: true, SupportsAttachments: true, }, } ================================================ FILE: internal/llm/models/gemini.go ================================================ package models const ( ProviderGemini ModelProvider = "gemini" // Models Gemini25Flash ModelID = "gemini-2.5-flash" Gemini25 ModelID = "gemini-2.5" Gemini20Flash ModelID = "gemini-2.0-flash" Gemini20FlashLite ModelID = "gemini-2.0-flash-lite" ) var GeminiModels = map[ModelID]Model{ Gemini25Flash: { ID: Gemini25Flash, Name: "Gemini 2.5 Flash", Provider: ProviderGemini, APIModel: "gemini-2.5-flash-preview-04-17", CostPer1MIn: 0.15, CostPer1MInCached: 0, CostPer1MOutCached: 0, CostPer1MOut: 0.60, ContextWindow: 1000000, DefaultMaxTokens: 50000, SupportsAttachments: true, }, Gemini25: { ID: Gemini25, Name: "Gemini 2.5 Pro", Provider: ProviderGemini, APIModel: "gemini-2.5-pro-preview-05-06", CostPer1MIn: 1.25, CostPer1MInCached: 0, CostPer1MOutCached: 0, CostPer1MOut: 10, ContextWindow: 1000000, DefaultMaxTokens: 50000, SupportsAttachments: true, }, Gemini20Flash: { ID: Gemini20Flash, Name: "Gemini 2.0 Flash", Provider: ProviderGemini, APIModel: "gemini-2.0-flash", CostPer1MIn: 0.10, CostPer1MInCached: 0, CostPer1MOutCached: 0, CostPer1MOut: 0.40, ContextWindow: 1000000, DefaultMaxTokens: 6000, SupportsAttachments: true, }, Gemini20FlashLite: { ID: Gemini20FlashLite, Name: "Gemini 2.0 Flash Lite", Provider: ProviderGemini, APIModel: "gemini-2.0-flash-lite", CostPer1MIn: 0.05, CostPer1MInCached: 0, CostPer1MOutCached: 0, CostPer1MOut: 0.30, ContextWindow: 1000000, DefaultMaxTokens: 6000, SupportsAttachments: true, }, } ================================================ FILE: internal/llm/models/groq.go ================================================ package models const ( ProviderGROQ ModelProvider = "groq" // GROQ QWENQwq ModelID = "qwen-qwq" // GROQ preview models Llama4Scout ModelID = "meta-llama/llama-4-scout-17b-16e-instruct" Llama4Maverick ModelID = "meta-llama/llama-4-maverick-17b-128e-instruct" Llama3_3_70BVersatile ModelID = "llama-3.3-70b-versatile" DeepseekR1DistillLlama70b ModelID = "deepseek-r1-distill-llama-70b" ) var GroqModels = map[ModelID]Model{ // // GROQ QWENQwq: { ID: QWENQwq, Name: "Qwen Qwq", Provider: ProviderGROQ, APIModel: "qwen-qwq-32b", CostPer1MIn: 0.29, CostPer1MInCached: 0.275, CostPer1MOutCached: 0.0, CostPer1MOut: 0.39, ContextWindow: 128_000, DefaultMaxTokens: 50000, // for some reason, the groq api doesn't like the reasoningEffort parameter CanReason: false, SupportsAttachments: false, }, Llama4Scout: { ID: Llama4Scout, Name: "Llama4Scout", Provider: ProviderGROQ, APIModel: "meta-llama/llama-4-scout-17b-16e-instruct", CostPer1MIn: 0.11, CostPer1MInCached: 0, CostPer1MOutCached: 0, CostPer1MOut: 0.34, ContextWindow: 128_000, // 10M when? SupportsAttachments: true, }, Llama4Maverick: { ID: Llama4Maverick, Name: "Llama4Maverick", Provider: ProviderGROQ, APIModel: "meta-llama/llama-4-maverick-17b-128e-instruct", CostPer1MIn: 0.20, CostPer1MInCached: 0, CostPer1MOutCached: 0, CostPer1MOut: 0.20, ContextWindow: 128_000, SupportsAttachments: true, }, Llama3_3_70BVersatile: { ID: Llama3_3_70BVersatile, Name: "Llama3_3_70BVersatile", Provider: ProviderGROQ, APIModel: "llama-3.3-70b-versatile", CostPer1MIn: 0.59, CostPer1MInCached: 0, CostPer1MOutCached: 0, CostPer1MOut: 0.79, ContextWindow: 128_000, SupportsAttachments: false, }, DeepseekR1DistillLlama70b: { ID: DeepseekR1DistillLlama70b, Name: "DeepseekR1DistillLlama70b", Provider: ProviderGROQ, APIModel: "deepseek-r1-distill-llama-70b", CostPer1MIn: 0.75, CostPer1MInCached: 0, CostPer1MOutCached: 0, CostPer1MOut: 0.99, ContextWindow: 128_000, CanReason: true, SupportsAttachments: false, }, } ================================================ FILE: internal/llm/models/local.go ================================================ package models import ( "cmp" "encoding/json" "net/http" "net/url" "os" "regexp" "strings" "unicode" "github.com/opencode-ai/opencode/internal/logging" "github.com/spf13/viper" ) const ( ProviderLocal ModelProvider = "local" localModelsPath = "v1/models" lmStudioBetaModelsPath = "api/v0/models" ) func init() { if endpoint := os.Getenv("LOCAL_ENDPOINT"); endpoint != "" { localEndpoint, err := url.Parse(endpoint) if err != nil { logging.Debug("Failed to parse local endpoint", "error", err, "endpoint", endpoint, ) return } load := func(url *url.URL, path string) []localModel { url.Path = path return listLocalModels(url.String()) } models := load(localEndpoint, lmStudioBetaModelsPath) if len(models) == 0 { models = load(localEndpoint, localModelsPath) } if len(models) == 0 { logging.Debug("No local models found", "endpoint", endpoint, ) return } loadLocalModels(models) viper.SetDefault("providers.local.apiKey", "dummy") ProviderPopularity[ProviderLocal] = 0 } } type localModelList struct { Data []localModel `json:"data"` } type localModel struct { ID string `json:"id"` Object string `json:"object"` Type string `json:"type"` Publisher string `json:"publisher"` Arch string `json:"arch"` CompatibilityType string `json:"compatibility_type"` Quantization string `json:"quantization"` State string `json:"state"` MaxContextLength int64 `json:"max_context_length"` LoadedContextLength int64 `json:"loaded_context_length"` } func listLocalModels(modelsEndpoint string) []localModel { res, err := http.Get(modelsEndpoint) if err != nil { logging.Debug("Failed to list local models", "error", err, "endpoint", modelsEndpoint, ) return []localModel{} } defer res.Body.Close() if res.StatusCode != http.StatusOK { logging.Debug("Failed to list local models", "status", res.StatusCode, "endpoint", modelsEndpoint, ) return []localModel{} } var modelList localModelList if err = json.NewDecoder(res.Body).Decode(&modelList); err != nil { logging.Debug("Failed to list local models", "error", err, "endpoint", modelsEndpoint, ) return []localModel{} } var supportedModels []localModel for _, model := range modelList.Data { if strings.HasSuffix(modelsEndpoint, lmStudioBetaModelsPath) { if model.Object != "model" || model.Type != "llm" { logging.Debug("Skipping unsupported LMStudio model", "endpoint", modelsEndpoint, "id", model.ID, "object", model.Object, "type", model.Type, ) continue } } supportedModels = append(supportedModels, model) } return supportedModels } func loadLocalModels(models []localModel) { for i, m := range models { model := convertLocalModel(m) SupportedModels[model.ID] = model if i == 0 || m.State == "loaded" { viper.SetDefault("agents.coder.model", model.ID) viper.SetDefault("agents.summarizer.model", model.ID) viper.SetDefault("agents.task.model", model.ID) viper.SetDefault("agents.title.model", model.ID) } } } func convertLocalModel(model localModel) Model { return Model{ ID: ModelID("local." + model.ID), Name: friendlyModelName(model.ID), Provider: ProviderLocal, APIModel: model.ID, ContextWindow: cmp.Or(model.LoadedContextLength, 4096), DefaultMaxTokens: cmp.Or(model.LoadedContextLength, 4096), CanReason: true, SupportsAttachments: true, } } var modelInfoRegex = regexp.MustCompile(`(?i)^([a-z0-9]+)(?:[-_]?([rv]?\d[\.\d]*))?(?:[-_]?([a-z]+))?.*`) func friendlyModelName(modelID string) string { mainID := modelID tag := "" if slash := strings.LastIndex(mainID, "/"); slash != -1 { mainID = mainID[slash+1:] } if at := strings.Index(modelID, "@"); at != -1 { mainID = modelID[:at] tag = modelID[at+1:] } match := modelInfoRegex.FindStringSubmatch(mainID) if match == nil { return modelID } capitalize := func(s string) string { if s == "" { return "" } runes := []rune(s) runes[0] = unicode.ToUpper(runes[0]) return string(runes) } family := capitalize(match[1]) version := "" label := "" if len(match) > 2 && match[2] != "" { version = strings.ToUpper(match[2]) } if len(match) > 3 && match[3] != "" { label = capitalize(match[3]) } var parts []string if family != "" { parts = append(parts, family) } if version != "" { parts = append(parts, version) } if label != "" { parts = append(parts, label) } if tag != "" { parts = append(parts, tag) } return strings.Join(parts, " ") } ================================================ FILE: internal/llm/models/models.go ================================================ package models import "maps" type ( ModelID string ModelProvider string ) type Model struct { ID ModelID `json:"id"` Name string `json:"name"` Provider ModelProvider `json:"provider"` APIModel string `json:"api_model"` CostPer1MIn float64 `json:"cost_per_1m_in"` CostPer1MOut float64 `json:"cost_per_1m_out"` CostPer1MInCached float64 `json:"cost_per_1m_in_cached"` CostPer1MOutCached float64 `json:"cost_per_1m_out_cached"` ContextWindow int64 `json:"context_window"` DefaultMaxTokens int64 `json:"default_max_tokens"` CanReason bool `json:"can_reason"` SupportsAttachments bool `json:"supports_attachments"` } // Model IDs const ( // GEMINI // Bedrock BedrockClaude37Sonnet ModelID = "bedrock.claude-3.7-sonnet" ) const ( ProviderBedrock ModelProvider = "bedrock" // ForTests ProviderMock ModelProvider = "__mock" ) // Providers in order of popularity var ProviderPopularity = map[ModelProvider]int{ ProviderCopilot: 1, ProviderAnthropic: 2, ProviderOpenAI: 3, ProviderGemini: 4, ProviderGROQ: 5, ProviderOpenRouter: 6, ProviderBedrock: 7, ProviderAzure: 8, ProviderVertexAI: 9, } var SupportedModels = map[ModelID]Model{ // // // GEMINI // GEMINI25: { // ID: GEMINI25, // Name: "Gemini 2.5 Pro", // Provider: ProviderGemini, // APIModel: "gemini-2.5-pro-exp-03-25", // CostPer1MIn: 0, // CostPer1MInCached: 0, // CostPer1MOutCached: 0, // CostPer1MOut: 0, // }, // // GRMINI20Flash: { // ID: GRMINI20Flash, // Name: "Gemini 2.0 Flash", // Provider: ProviderGemini, // APIModel: "gemini-2.0-flash", // CostPer1MIn: 0.1, // CostPer1MInCached: 0, // CostPer1MOutCached: 0.025, // CostPer1MOut: 0.4, // }, // // // Bedrock BedrockClaude37Sonnet: { ID: BedrockClaude37Sonnet, Name: "Bedrock: Claude 3.7 Sonnet", Provider: ProviderBedrock, APIModel: "anthropic.claude-3-7-sonnet-20250219-v1:0", CostPer1MIn: 3.0, CostPer1MInCached: 3.75, CostPer1MOutCached: 0.30, CostPer1MOut: 15.0, }, } func init() { maps.Copy(SupportedModels, AnthropicModels) maps.Copy(SupportedModels, OpenAIModels) maps.Copy(SupportedModels, GeminiModels) maps.Copy(SupportedModels, GroqModels) maps.Copy(SupportedModels, AzureModels) maps.Copy(SupportedModels, OpenRouterModels) maps.Copy(SupportedModels, XAIModels) maps.Copy(SupportedModels, VertexAIGeminiModels) maps.Copy(SupportedModels, CopilotModels) } ================================================ FILE: internal/llm/models/openai.go ================================================ package models const ( ProviderOpenAI ModelProvider = "openai" GPT41 ModelID = "gpt-4.1" GPT41Mini ModelID = "gpt-4.1-mini" GPT41Nano ModelID = "gpt-4.1-nano" GPT45Preview ModelID = "gpt-4.5-preview" GPT4o ModelID = "gpt-4o" GPT4oMini ModelID = "gpt-4o-mini" O1 ModelID = "o1" O1Pro ModelID = "o1-pro" O1Mini ModelID = "o1-mini" O3 ModelID = "o3" O3Mini ModelID = "o3-mini" O4Mini ModelID = "o4-mini" ) var OpenAIModels = map[ModelID]Model{ GPT41: { ID: GPT41, Name: "GPT 4.1", Provider: ProviderOpenAI, APIModel: "gpt-4.1", CostPer1MIn: 2.00, CostPer1MInCached: 0.50, CostPer1MOutCached: 0.0, CostPer1MOut: 8.00, ContextWindow: 1_047_576, DefaultMaxTokens: 20000, SupportsAttachments: true, }, GPT41Mini: { ID: GPT41Mini, Name: "GPT 4.1 mini", Provider: ProviderOpenAI, APIModel: "gpt-4.1", CostPer1MIn: 0.40, CostPer1MInCached: 0.10, CostPer1MOutCached: 0.0, CostPer1MOut: 1.60, ContextWindow: 200_000, DefaultMaxTokens: 20000, SupportsAttachments: true, }, GPT41Nano: { ID: GPT41Nano, Name: "GPT 4.1 nano", Provider: ProviderOpenAI, APIModel: "gpt-4.1-nano", CostPer1MIn: 0.10, CostPer1MInCached: 0.025, CostPer1MOutCached: 0.0, CostPer1MOut: 0.40, ContextWindow: 1_047_576, DefaultMaxTokens: 20000, SupportsAttachments: true, }, GPT45Preview: { ID: GPT45Preview, Name: "GPT 4.5 preview", Provider: ProviderOpenAI, APIModel: "gpt-4.5-preview", CostPer1MIn: 75.00, CostPer1MInCached: 37.50, CostPer1MOutCached: 0.0, CostPer1MOut: 150.00, ContextWindow: 128_000, DefaultMaxTokens: 15000, SupportsAttachments: true, }, GPT4o: { ID: GPT4o, Name: "GPT 4o", Provider: ProviderOpenAI, APIModel: "gpt-4o", CostPer1MIn: 2.50, CostPer1MInCached: 1.25, CostPer1MOutCached: 0.0, CostPer1MOut: 10.00, ContextWindow: 128_000, DefaultMaxTokens: 4096, SupportsAttachments: true, }, GPT4oMini: { ID: GPT4oMini, Name: "GPT 4o mini", Provider: ProviderOpenAI, APIModel: "gpt-4o-mini", CostPer1MIn: 0.15, CostPer1MInCached: 0.075, CostPer1MOutCached: 0.0, CostPer1MOut: 0.60, ContextWindow: 128_000, SupportsAttachments: true, }, O1: { ID: O1, Name: "O1", Provider: ProviderOpenAI, APIModel: "o1", CostPer1MIn: 15.00, CostPer1MInCached: 7.50, CostPer1MOutCached: 0.0, CostPer1MOut: 60.00, ContextWindow: 200_000, DefaultMaxTokens: 50000, CanReason: true, SupportsAttachments: true, }, O1Pro: { ID: O1Pro, Name: "o1 pro", Provider: ProviderOpenAI, APIModel: "o1-pro", CostPer1MIn: 150.00, CostPer1MInCached: 0.0, CostPer1MOutCached: 0.0, CostPer1MOut: 600.00, ContextWindow: 200_000, DefaultMaxTokens: 50000, CanReason: true, SupportsAttachments: true, }, O1Mini: { ID: O1Mini, Name: "o1 mini", Provider: ProviderOpenAI, APIModel: "o1-mini", CostPer1MIn: 1.10, CostPer1MInCached: 0.55, CostPer1MOutCached: 0.0, CostPer1MOut: 4.40, ContextWindow: 128_000, DefaultMaxTokens: 50000, CanReason: true, SupportsAttachments: true, }, O3: { ID: O3, Name: "o3", Provider: ProviderOpenAI, APIModel: "o3", CostPer1MIn: 10.00, CostPer1MInCached: 2.50, CostPer1MOutCached: 0.0, CostPer1MOut: 40.00, ContextWindow: 200_000, CanReason: true, SupportsAttachments: true, }, O3Mini: { ID: O3Mini, Name: "o3 mini", Provider: ProviderOpenAI, APIModel: "o3-mini", CostPer1MIn: 1.10, CostPer1MInCached: 0.55, CostPer1MOutCached: 0.0, CostPer1MOut: 4.40, ContextWindow: 200_000, DefaultMaxTokens: 50000, CanReason: true, SupportsAttachments: false, }, O4Mini: { ID: O4Mini, Name: "o4 mini", Provider: ProviderOpenAI, APIModel: "o4-mini", CostPer1MIn: 1.10, CostPer1MInCached: 0.275, CostPer1MOutCached: 0.0, CostPer1MOut: 4.40, ContextWindow: 128_000, DefaultMaxTokens: 50000, CanReason: true, SupportsAttachments: true, }, } ================================================ FILE: internal/llm/models/openrouter.go ================================================ package models const ( ProviderOpenRouter ModelProvider = "openrouter" OpenRouterGPT41 ModelID = "openrouter.gpt-4.1" OpenRouterGPT41Mini ModelID = "openrouter.gpt-4.1-mini" OpenRouterGPT41Nano ModelID = "openrouter.gpt-4.1-nano" OpenRouterGPT45Preview ModelID = "openrouter.gpt-4.5-preview" OpenRouterGPT4o ModelID = "openrouter.gpt-4o" OpenRouterGPT4oMini ModelID = "openrouter.gpt-4o-mini" OpenRouterO1 ModelID = "openrouter.o1" OpenRouterO1Pro ModelID = "openrouter.o1-pro" OpenRouterO1Mini ModelID = "openrouter.o1-mini" OpenRouterO3 ModelID = "openrouter.o3" OpenRouterO3Mini ModelID = "openrouter.o3-mini" OpenRouterO4Mini ModelID = "openrouter.o4-mini" OpenRouterGemini25Flash ModelID = "openrouter.gemini-2.5-flash" OpenRouterGemini25 ModelID = "openrouter.gemini-2.5" OpenRouterClaude35Sonnet ModelID = "openrouter.claude-3.5-sonnet" OpenRouterClaude3Haiku ModelID = "openrouter.claude-3-haiku" OpenRouterClaude37Sonnet ModelID = "openrouter.claude-3.7-sonnet" OpenRouterClaude35Haiku ModelID = "openrouter.claude-3.5-haiku" OpenRouterClaude3Opus ModelID = "openrouter.claude-3-opus" OpenRouterDeepSeekR1Free ModelID = "openrouter.deepseek-r1-free" ) var OpenRouterModels = map[ModelID]Model{ OpenRouterGPT41: { ID: OpenRouterGPT41, Name: "OpenRouter – GPT 4.1", Provider: ProviderOpenRouter, APIModel: "openai/gpt-4.1", CostPer1MIn: OpenAIModels[GPT41].CostPer1MIn, CostPer1MInCached: OpenAIModels[GPT41].CostPer1MInCached, CostPer1MOut: OpenAIModels[GPT41].CostPer1MOut, CostPer1MOutCached: OpenAIModels[GPT41].CostPer1MOutCached, ContextWindow: OpenAIModels[GPT41].ContextWindow, DefaultMaxTokens: OpenAIModels[GPT41].DefaultMaxTokens, }, OpenRouterGPT41Mini: { ID: OpenRouterGPT41Mini, Name: "OpenRouter – GPT 4.1 mini", Provider: ProviderOpenRouter, APIModel: "openai/gpt-4.1-mini", CostPer1MIn: OpenAIModels[GPT41Mini].CostPer1MIn, CostPer1MInCached: OpenAIModels[GPT41Mini].CostPer1MInCached, CostPer1MOut: OpenAIModels[GPT41Mini].CostPer1MOut, CostPer1MOutCached: OpenAIModels[GPT41Mini].CostPer1MOutCached, ContextWindow: OpenAIModels[GPT41Mini].ContextWindow, DefaultMaxTokens: OpenAIModels[GPT41Mini].DefaultMaxTokens, }, OpenRouterGPT41Nano: { ID: OpenRouterGPT41Nano, Name: "OpenRouter – GPT 4.1 nano", Provider: ProviderOpenRouter, APIModel: "openai/gpt-4.1-nano", CostPer1MIn: OpenAIModels[GPT41Nano].CostPer1MIn, CostPer1MInCached: OpenAIModels[GPT41Nano].CostPer1MInCached, CostPer1MOut: OpenAIModels[GPT41Nano].CostPer1MOut, CostPer1MOutCached: OpenAIModels[GPT41Nano].CostPer1MOutCached, ContextWindow: OpenAIModels[GPT41Nano].ContextWindow, DefaultMaxTokens: OpenAIModels[GPT41Nano].DefaultMaxTokens, }, OpenRouterGPT45Preview: { ID: OpenRouterGPT45Preview, Name: "OpenRouter – GPT 4.5 preview", Provider: ProviderOpenRouter, APIModel: "openai/gpt-4.5-preview", CostPer1MIn: OpenAIModels[GPT45Preview].CostPer1MIn, CostPer1MInCached: OpenAIModels[GPT45Preview].CostPer1MInCached, CostPer1MOut: OpenAIModels[GPT45Preview].CostPer1MOut, CostPer1MOutCached: OpenAIModels[GPT45Preview].CostPer1MOutCached, ContextWindow: OpenAIModels[GPT45Preview].ContextWindow, DefaultMaxTokens: OpenAIModels[GPT45Preview].DefaultMaxTokens, }, OpenRouterGPT4o: { ID: OpenRouterGPT4o, Name: "OpenRouter – GPT 4o", Provider: ProviderOpenRouter, APIModel: "openai/gpt-4o", CostPer1MIn: OpenAIModels[GPT4o].CostPer1MIn, CostPer1MInCached: OpenAIModels[GPT4o].CostPer1MInCached, CostPer1MOut: OpenAIModels[GPT4o].CostPer1MOut, CostPer1MOutCached: OpenAIModels[GPT4o].CostPer1MOutCached, ContextWindow: OpenAIModels[GPT4o].ContextWindow, DefaultMaxTokens: OpenAIModels[GPT4o].DefaultMaxTokens, }, OpenRouterGPT4oMini: { ID: OpenRouterGPT4oMini, Name: "OpenRouter – GPT 4o mini", Provider: ProviderOpenRouter, APIModel: "openai/gpt-4o-mini", CostPer1MIn: OpenAIModels[GPT4oMini].CostPer1MIn, CostPer1MInCached: OpenAIModels[GPT4oMini].CostPer1MInCached, CostPer1MOut: OpenAIModels[GPT4oMini].CostPer1MOut, CostPer1MOutCached: OpenAIModels[GPT4oMini].CostPer1MOutCached, ContextWindow: OpenAIModels[GPT4oMini].ContextWindow, }, OpenRouterO1: { ID: OpenRouterO1, Name: "OpenRouter – O1", Provider: ProviderOpenRouter, APIModel: "openai/o1", CostPer1MIn: OpenAIModels[O1].CostPer1MIn, CostPer1MInCached: OpenAIModels[O1].CostPer1MInCached, CostPer1MOut: OpenAIModels[O1].CostPer1MOut, CostPer1MOutCached: OpenAIModels[O1].CostPer1MOutCached, ContextWindow: OpenAIModels[O1].ContextWindow, DefaultMaxTokens: OpenAIModels[O1].DefaultMaxTokens, CanReason: OpenAIModels[O1].CanReason, }, OpenRouterO1Pro: { ID: OpenRouterO1Pro, Name: "OpenRouter – o1 pro", Provider: ProviderOpenRouter, APIModel: "openai/o1-pro", CostPer1MIn: OpenAIModels[O1Pro].CostPer1MIn, CostPer1MInCached: OpenAIModels[O1Pro].CostPer1MInCached, CostPer1MOut: OpenAIModels[O1Pro].CostPer1MOut, CostPer1MOutCached: OpenAIModels[O1Pro].CostPer1MOutCached, ContextWindow: OpenAIModels[O1Pro].ContextWindow, DefaultMaxTokens: OpenAIModels[O1Pro].DefaultMaxTokens, CanReason: OpenAIModels[O1Pro].CanReason, }, OpenRouterO1Mini: { ID: OpenRouterO1Mini, Name: "OpenRouter – o1 mini", Provider: ProviderOpenRouter, APIModel: "openai/o1-mini", CostPer1MIn: OpenAIModels[O1Mini].CostPer1MIn, CostPer1MInCached: OpenAIModels[O1Mini].CostPer1MInCached, CostPer1MOut: OpenAIModels[O1Mini].CostPer1MOut, CostPer1MOutCached: OpenAIModels[O1Mini].CostPer1MOutCached, ContextWindow: OpenAIModels[O1Mini].ContextWindow, DefaultMaxTokens: OpenAIModels[O1Mini].DefaultMaxTokens, CanReason: OpenAIModels[O1Mini].CanReason, }, OpenRouterO3: { ID: OpenRouterO3, Name: "OpenRouter – o3", Provider: ProviderOpenRouter, APIModel: "openai/o3", CostPer1MIn: OpenAIModels[O3].CostPer1MIn, CostPer1MInCached: OpenAIModels[O3].CostPer1MInCached, CostPer1MOut: OpenAIModels[O3].CostPer1MOut, CostPer1MOutCached: OpenAIModels[O3].CostPer1MOutCached, ContextWindow: OpenAIModels[O3].ContextWindow, DefaultMaxTokens: OpenAIModels[O3].DefaultMaxTokens, CanReason: OpenAIModels[O3].CanReason, }, OpenRouterO3Mini: { ID: OpenRouterO3Mini, Name: "OpenRouter – o3 mini", Provider: ProviderOpenRouter, APIModel: "openai/o3-mini-high", CostPer1MIn: OpenAIModels[O3Mini].CostPer1MIn, CostPer1MInCached: OpenAIModels[O3Mini].CostPer1MInCached, CostPer1MOut: OpenAIModels[O3Mini].CostPer1MOut, CostPer1MOutCached: OpenAIModels[O3Mini].CostPer1MOutCached, ContextWindow: OpenAIModels[O3Mini].ContextWindow, DefaultMaxTokens: OpenAIModels[O3Mini].DefaultMaxTokens, CanReason: OpenAIModels[O3Mini].CanReason, }, OpenRouterO4Mini: { ID: OpenRouterO4Mini, Name: "OpenRouter – o4 mini", Provider: ProviderOpenRouter, APIModel: "openai/o4-mini-high", CostPer1MIn: OpenAIModels[O4Mini].CostPer1MIn, CostPer1MInCached: OpenAIModels[O4Mini].CostPer1MInCached, CostPer1MOut: OpenAIModels[O4Mini].CostPer1MOut, CostPer1MOutCached: OpenAIModels[O4Mini].CostPer1MOutCached, ContextWindow: OpenAIModels[O4Mini].ContextWindow, DefaultMaxTokens: OpenAIModels[O4Mini].DefaultMaxTokens, CanReason: OpenAIModels[O4Mini].CanReason, }, OpenRouterGemini25Flash: { ID: OpenRouterGemini25Flash, Name: "OpenRouter – Gemini 2.5 Flash", Provider: ProviderOpenRouter, APIModel: "google/gemini-2.5-flash-preview:thinking", CostPer1MIn: GeminiModels[Gemini25Flash].CostPer1MIn, CostPer1MInCached: GeminiModels[Gemini25Flash].CostPer1MInCached, CostPer1MOut: GeminiModels[Gemini25Flash].CostPer1MOut, CostPer1MOutCached: GeminiModels[Gemini25Flash].CostPer1MOutCached, ContextWindow: GeminiModels[Gemini25Flash].ContextWindow, DefaultMaxTokens: GeminiModels[Gemini25Flash].DefaultMaxTokens, }, OpenRouterGemini25: { ID: OpenRouterGemini25, Name: "OpenRouter – Gemini 2.5 Pro", Provider: ProviderOpenRouter, APIModel: "google/gemini-2.5-pro-preview-03-25", CostPer1MIn: GeminiModels[Gemini25].CostPer1MIn, CostPer1MInCached: GeminiModels[Gemini25].CostPer1MInCached, CostPer1MOut: GeminiModels[Gemini25].CostPer1MOut, CostPer1MOutCached: GeminiModels[Gemini25].CostPer1MOutCached, ContextWindow: GeminiModels[Gemini25].ContextWindow, DefaultMaxTokens: GeminiModels[Gemini25].DefaultMaxTokens, }, OpenRouterClaude35Sonnet: { ID: OpenRouterClaude35Sonnet, Name: "OpenRouter – Claude 3.5 Sonnet", Provider: ProviderOpenRouter, APIModel: "anthropic/claude-3.5-sonnet", CostPer1MIn: AnthropicModels[Claude35Sonnet].CostPer1MIn, CostPer1MInCached: AnthropicModels[Claude35Sonnet].CostPer1MInCached, CostPer1MOut: AnthropicModels[Claude35Sonnet].CostPer1MOut, CostPer1MOutCached: AnthropicModels[Claude35Sonnet].CostPer1MOutCached, ContextWindow: AnthropicModels[Claude35Sonnet].ContextWindow, DefaultMaxTokens: AnthropicModels[Claude35Sonnet].DefaultMaxTokens, }, OpenRouterClaude3Haiku: { ID: OpenRouterClaude3Haiku, Name: "OpenRouter – Claude 3 Haiku", Provider: ProviderOpenRouter, APIModel: "anthropic/claude-3-haiku", CostPer1MIn: AnthropicModels[Claude3Haiku].CostPer1MIn, CostPer1MInCached: AnthropicModels[Claude3Haiku].CostPer1MInCached, CostPer1MOut: AnthropicModels[Claude3Haiku].CostPer1MOut, CostPer1MOutCached: AnthropicModels[Claude3Haiku].CostPer1MOutCached, ContextWindow: AnthropicModels[Claude3Haiku].ContextWindow, DefaultMaxTokens: AnthropicModels[Claude3Haiku].DefaultMaxTokens, }, OpenRouterClaude37Sonnet: { ID: OpenRouterClaude37Sonnet, Name: "OpenRouter – Claude 3.7 Sonnet", Provider: ProviderOpenRouter, APIModel: "anthropic/claude-3.7-sonnet", CostPer1MIn: AnthropicModels[Claude37Sonnet].CostPer1MIn, CostPer1MInCached: AnthropicModels[Claude37Sonnet].CostPer1MInCached, CostPer1MOut: AnthropicModels[Claude37Sonnet].CostPer1MOut, CostPer1MOutCached: AnthropicModels[Claude37Sonnet].CostPer1MOutCached, ContextWindow: AnthropicModels[Claude37Sonnet].ContextWindow, DefaultMaxTokens: AnthropicModels[Claude37Sonnet].DefaultMaxTokens, CanReason: AnthropicModels[Claude37Sonnet].CanReason, }, OpenRouterClaude35Haiku: { ID: OpenRouterClaude35Haiku, Name: "OpenRouter – Claude 3.5 Haiku", Provider: ProviderOpenRouter, APIModel: "anthropic/claude-3.5-haiku", CostPer1MIn: AnthropicModels[Claude35Haiku].CostPer1MIn, CostPer1MInCached: AnthropicModels[Claude35Haiku].CostPer1MInCached, CostPer1MOut: AnthropicModels[Claude35Haiku].CostPer1MOut, CostPer1MOutCached: AnthropicModels[Claude35Haiku].CostPer1MOutCached, ContextWindow: AnthropicModels[Claude35Haiku].ContextWindow, DefaultMaxTokens: AnthropicModels[Claude35Haiku].DefaultMaxTokens, }, OpenRouterClaude3Opus: { ID: OpenRouterClaude3Opus, Name: "OpenRouter – Claude 3 Opus", Provider: ProviderOpenRouter, APIModel: "anthropic/claude-3-opus", CostPer1MIn: AnthropicModels[Claude3Opus].CostPer1MIn, CostPer1MInCached: AnthropicModels[Claude3Opus].CostPer1MInCached, CostPer1MOut: AnthropicModels[Claude3Opus].CostPer1MOut, CostPer1MOutCached: AnthropicModels[Claude3Opus].CostPer1MOutCached, ContextWindow: AnthropicModels[Claude3Opus].ContextWindow, DefaultMaxTokens: AnthropicModels[Claude3Opus].DefaultMaxTokens, }, OpenRouterDeepSeekR1Free: { ID: OpenRouterDeepSeekR1Free, Name: "OpenRouter – DeepSeek R1 Free", Provider: ProviderOpenRouter, APIModel: "deepseek/deepseek-r1-0528:free", CostPer1MIn: 0, CostPer1MInCached: 0, CostPer1MOut: 0, CostPer1MOutCached: 0, ContextWindow: 163_840, DefaultMaxTokens: 10000, }, } ================================================ FILE: internal/llm/models/vertexai.go ================================================ package models const ( ProviderVertexAI ModelProvider = "vertexai" // Models VertexAIGemini25Flash ModelID = "vertexai.gemini-2.5-flash" VertexAIGemini25 ModelID = "vertexai.gemini-2.5" ) var VertexAIGeminiModels = map[ModelID]Model{ VertexAIGemini25Flash: { ID: VertexAIGemini25Flash, Name: "VertexAI: Gemini 2.5 Flash", Provider: ProviderVertexAI, APIModel: "gemini-2.5-flash-preview-04-17", CostPer1MIn: GeminiModels[Gemini25Flash].CostPer1MIn, CostPer1MInCached: GeminiModels[Gemini25Flash].CostPer1MInCached, CostPer1MOut: GeminiModels[Gemini25Flash].CostPer1MOut, CostPer1MOutCached: GeminiModels[Gemini25Flash].CostPer1MOutCached, ContextWindow: GeminiModels[Gemini25Flash].ContextWindow, DefaultMaxTokens: GeminiModels[Gemini25Flash].DefaultMaxTokens, SupportsAttachments: true, }, VertexAIGemini25: { ID: VertexAIGemini25, Name: "VertexAI: Gemini 2.5 Pro", Provider: ProviderVertexAI, APIModel: "gemini-2.5-pro-preview-03-25", CostPer1MIn: GeminiModels[Gemini25].CostPer1MIn, CostPer1MInCached: GeminiModels[Gemini25].CostPer1MInCached, CostPer1MOut: GeminiModels[Gemini25].CostPer1MOut, CostPer1MOutCached: GeminiModels[Gemini25].CostPer1MOutCached, ContextWindow: GeminiModels[Gemini25].ContextWindow, DefaultMaxTokens: GeminiModels[Gemini25].DefaultMaxTokens, SupportsAttachments: true, }, } ================================================ FILE: internal/llm/models/xai.go ================================================ package models const ( ProviderXAI ModelProvider = "xai" XAIGrok3Beta ModelID = "grok-3-beta" XAIGrok3MiniBeta ModelID = "grok-3-mini-beta" XAIGrok3FastBeta ModelID = "grok-3-fast-beta" XAiGrok3MiniFastBeta ModelID = "grok-3-mini-fast-beta" ) var XAIModels = map[ModelID]Model{ XAIGrok3Beta: { ID: XAIGrok3Beta, Name: "Grok3 Beta", Provider: ProviderXAI, APIModel: "grok-3-beta", CostPer1MIn: 3.0, CostPer1MInCached: 0, CostPer1MOut: 15, CostPer1MOutCached: 0, ContextWindow: 131_072, DefaultMaxTokens: 20_000, }, XAIGrok3MiniBeta: { ID: XAIGrok3MiniBeta, Name: "Grok3 Mini Beta", Provider: ProviderXAI, APIModel: "grok-3-mini-beta", CostPer1MIn: 0.3, CostPer1MInCached: 0, CostPer1MOut: 0.5, CostPer1MOutCached: 0, ContextWindow: 131_072, DefaultMaxTokens: 20_000, }, XAIGrok3FastBeta: { ID: XAIGrok3FastBeta, Name: "Grok3 Fast Beta", Provider: ProviderXAI, APIModel: "grok-3-fast-beta", CostPer1MIn: 5, CostPer1MInCached: 0, CostPer1MOut: 25, CostPer1MOutCached: 0, ContextWindow: 131_072, DefaultMaxTokens: 20_000, }, XAiGrok3MiniFastBeta: { ID: XAiGrok3MiniFastBeta, Name: "Grok3 Mini Fast Beta", Provider: ProviderXAI, APIModel: "grok-3-mini-fast-beta", CostPer1MIn: 0.6, CostPer1MInCached: 0, CostPer1MOut: 4.0, CostPer1MOutCached: 0, ContextWindow: 131_072, DefaultMaxTokens: 20_000, }, } ================================================ FILE: internal/llm/prompt/coder.go ================================================ package prompt import ( "context" "fmt" "os" "path/filepath" "runtime" "time" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/llm/models" "github.com/opencode-ai/opencode/internal/llm/tools" ) func CoderPrompt(provider models.ModelProvider) string { basePrompt := baseAnthropicCoderPrompt switch provider { case models.ProviderOpenAI: basePrompt = baseOpenAICoderPrompt } envInfo := getEnvironmentInfo() return fmt.Sprintf("%s\n\n%s\n%s", basePrompt, envInfo, lspInformation()) } const baseOpenAICoderPrompt = ` You are operating as and within the OpenCode CLI, a terminal-based agentic coding assistant built by OpenAI. It wraps OpenAI models to enable natural language interaction with a local codebase. You are expected to be precise, safe, and helpful. You can: - Receive user prompts, project context, and files. - Stream responses and emit function calls (e.g., shell commands, code edits). - Apply patches, run commands, and manage user approvals based on policy. - Work inside a sandboxed, git-backed workspace with rollback support. - Log telemetry so sessions can be replayed or inspected later. - More details on your functionality are available at "opencode --help" You are an agent - please keep going until the user's query is completely resolved, before ending your turn and yielding back to the user. Only terminate your turn when you are sure that the problem is solved. If you are not sure about file content or codebase structure pertaining to the user's request, use your tools to read files and gather the relevant information: do NOT guess or make up an answer. Please resolve the user's task by editing and testing the code files in your current code execution session. You are a deployed coding agent. Your session allows for you to modify and run code. The repo(s) are already cloned in your working directory, and you must fully solve the problem for your answer to be considered correct. You MUST adhere to the following criteria when executing the task: - Working on the repo(s) in the current environment is allowed, even if they are proprietary. - Analyzing code for vulnerabilities is allowed. - Showing user code and tool call details is allowed. - User instructions may overwrite the *CODING GUIDELINES* section in this developer message. - If completing the user's task requires writing or modifying files: - Your code and final answer should follow these *CODING GUIDELINES*: - Fix the problem at the root cause rather than applying surface-level patches, when possible. - Avoid unneeded complexity in your solution. - Ignore unrelated bugs or broken tests; it is not your responsibility to fix them. - Update documentation as necessary. - Keep changes consistent with the style of the existing codebase. Changes should be minimal and focused on the task. - Use "git log" and "git blame" to search the history of the codebase if additional context is required; internet access is disabled. - NEVER add copyright or license headers unless specifically requested. - You do not need to "git commit" your changes; this will be done automatically for you. - Once you finish coding, you must - Check "git status" to sanity check your changes; revert any scratch files or changes. - Remove all inline comments you added as much as possible, even if they look normal. Check using "git diff". Inline comments must be generally avoided, unless active maintainers of the repo, after long careful study of the code and the issue, will still misinterpret the code without the comments. - Check if you accidentally add copyright or license headers. If so, remove them. - For smaller tasks, describe in brief bullet points - For more complex tasks, include brief high-level description, use bullet points, and include details that would be relevant to a code reviewer. - If completing the user's task DOES NOT require writing or modifying files (e.g., the user asks a question about the code base): - Respond in a friendly tune as a remote teammate, who is knowledgeable, capable and eager to help with coding. - When your task involves writing or modifying files: - Do NOT tell the user to "save the file" or "copy the code into a file" if you already created or modified the file using "apply_patch". Instead, reference the file as already saved. - Do NOT show the full contents of large files you have already written, unless the user explicitly asks for them. - When doing things with paths, always use use the full path, if the working directory is /abc/xyz and you want to edit the file abc.go in the working dir refer to it as /abc/xyz/abc.go. - If you send a path not including the working dir, the working dir will be prepended to it. - Remember the user does not see the full output of tools ` const baseAnthropicCoderPrompt = `You are OpenCode, an interactive CLI tool that helps users with software engineering tasks. Use the instructions below and the tools available to you to assist the user. IMPORTANT: Before you begin work, think about what the code you're editing is supposed to do based on the filenames directory structure. # Memory If the current working directory contains a file called OpenCode.md, it will be automatically added to your context. This file serves multiple purposes: 1. Storing frequently used bash commands (build, test, lint, etc.) so you can use them without searching each time 2. Recording the user's code style preferences (naming conventions, preferred libraries, etc.) 3. Maintaining useful information about the codebase structure and organization When you spend time searching for commands to typecheck, lint, build, or test, you should ask the user if it's okay to add those commands to OpenCode.md. Similarly, when learning about code style preferences or important codebase information, ask if it's okay to add that to OpenCode.md so you can remember it for next time. # Tone and style You should be concise, direct, and to the point. When you run a non-trivial bash command, you should explain what the command does and why you are running it, to make sure the user understands what you are doing (this is especially important when you are running a command that will make changes to the user's system). Remember that your output will be displayed on a command line interface. Your responses can use Github-flavored markdown for formatting, and will be rendered in a monospace font using the CommonMark specification. Output text to communicate with the user; all text you output outside of tool use is displayed to the user. Only use tools to complete tasks. Never use tools like Bash or code comments as means to communicate with the user during the session. If you cannot or will not help the user with something, please do not say why or what it could lead to, since this comes across as preachy and annoying. Please offer helpful alternatives if possible, and otherwise keep your response to 1-2 sentences. IMPORTANT: You should minimize output tokens as much as possible while maintaining helpfulness, quality, and accuracy. Only address the specific query or task at hand, avoiding tangential information unless absolutely critical for completing the request. If you can answer in 1-3 sentences or a short paragraph, please do. IMPORTANT: You should NOT answer with unnecessary preamble or postamble (such as explaining your code or summarizing your action), unless the user asks you to. IMPORTANT: Keep your responses short, since they will be displayed on a command line interface. You MUST answer concisely with fewer than 4 lines (not including tool use or code generation), unless user asks for detail. Answer the user's question directly, without elaboration, explanation, or details. One word answers are best. Avoid introductions, conclusions, and explanations. You MUST avoid text before/after your response, such as "The answer is .", "Here is the content of the file..." or "Based on the information provided, the answer is..." or "Here is what I will do next...". Here are some examples to demonstrate appropriate verbosity: user: 2 + 2 assistant: 4 user: what is 2+2? assistant: 4 user: is 11 a prime number? assistant: true user: what command should I run to list files in the current directory? assistant: ls user: what command should I run to watch files in the current directory? assistant: [use the ls tool to list the files in the current directory, then read docs/commands in the relevant file to find out how to watch files] npm run dev user: How many golf balls fit inside a jetta? assistant: 150000 user: what files are in the directory src/? assistant: [runs ls and sees foo.c, bar.c, baz.c] user: which file contains the implementation of foo? assistant: src/foo.c user: write tests for new feature assistant: [uses grep and glob search tools to find where similar tests are defined, uses concurrent read file tool use blocks in one tool call to read relevant files at the same time, uses edit/patch file tool to write new tests] # Proactiveness You are allowed to be proactive, but only when the user asks you to do something. You should strive to strike a balance between: 1. Doing the right thing when asked, including taking actions and follow-up actions 2. Not surprising the user with actions you take without asking For example, if the user asks you how to approach something, you should do your best to answer their question first, and not immediately jump into taking actions. 3. Do not add additional code explanation summary unless requested by the user. After working on a file, just stop, rather than providing an explanation of what you did. # Following conventions When making changes to files, first understand the file's code conventions. Mimic code style, use existing libraries and utilities, and follow existing patterns. - NEVER assume that a given library is available, even if it is well known. Whenever you write code that uses a library or framework, first check that this codebase already uses the given library. For example, you might look at neighboring files, or check the package.json (or cargo.toml, and so on depending on the language). - When you create a new component, first look at existing components to see how they're written; then consider framework choice, naming conventions, typing, and other conventions. - When you edit a piece of code, first look at the code's surrounding context (especially its imports) to understand the code's choice of frameworks and libraries. Then consider how to make the given change in a way that is most idiomatic. - Always follow security best practices. Never introduce code that exposes or logs secrets and keys. Never commit secrets or keys to the repository. # Code style - Do not add comments to the code you write, unless the user asks you to, or the code is complex and requires additional context. # Doing tasks The user will primarily request you perform software engineering tasks. This includes solving bugs, adding new functionality, refactoring code, explaining code, and more. For these tasks the following steps are recommended: 1. Use the available search tools to understand the codebase and the user's query. You are encouraged to use the search tools extensively both in parallel and sequentially. 2. Implement the solution using all tools available to you 3. Verify the solution if possible with tests. NEVER assume specific test framework or test script. Check the README or search codebase to determine the testing approach. 4. VERY IMPORTANT: When you have completed a task, you MUST run the lint and typecheck commands (eg. npm run lint, npm run typecheck, ruff, etc.) if they were provided to you to ensure your code is correct. If you are unable to find the correct command, ask the user for the command to run and if they supply it, proactively suggest writing it to opencode.md so that you will know to run it next time. NEVER commit changes unless the user explicitly asks you to. It is VERY IMPORTANT to only commit when explicitly asked, otherwise the user will feel that you are being too proactive. # Tool usage policy - When doing file search, prefer to use the Agent tool in order to reduce context usage. - If you intend to call multiple tools and there are no dependencies between the calls, make all of the independent calls in the same function_calls block. - IMPORTANT: The user does not see the full output of the tool responses, so if you need the output of the tool for the response make sure to summarize it for the user. You MUST answer concisely with fewer than 4 lines of text (not including tool use or code generation), unless user asks for detail.` func getEnvironmentInfo() string { cwd := config.WorkingDirectory() isGit := isGitRepo(cwd) platform := runtime.GOOS date := time.Now().Format("1/2/2006") ls := tools.NewLsTool() r, _ := ls.Run(context.Background(), tools.ToolCall{ Input: `{"path":"."}`, }) return fmt.Sprintf(`Here is useful information about the environment you are running in: Working directory: %s Is directory a git repo: %s Platform: %s Today's date: %s %s `, cwd, boolToYesNo(isGit), platform, date, r.Content) } func isGitRepo(dir string) bool { _, err := os.Stat(filepath.Join(dir, ".git")) return err == nil } func lspInformation() string { cfg := config.Get() hasLSP := false for _, v := range cfg.LSP { if !v.Disabled { hasLSP = true break } } if !hasLSP { return "" } return `# LSP Information Tools that support it will also include useful diagnostics such as linting and typechecking. - These diagnostics will be automatically enabled when you run the tool, and will be displayed in the output at the bottom within the and tags. - Take necessary actions to fix the issues. - You should ignore diagnostics of files that you did not change or are not related or caused by your changes unless the user explicitly asks you to fix them. ` } func boolToYesNo(b bool) string { if b { return "Yes" } return "No" } ================================================ FILE: internal/llm/prompt/prompt.go ================================================ package prompt import ( "fmt" "os" "path/filepath" "strings" "sync" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/llm/models" "github.com/opencode-ai/opencode/internal/logging" ) func GetAgentPrompt(agentName config.AgentName, provider models.ModelProvider) string { basePrompt := "" switch agentName { case config.AgentCoder: basePrompt = CoderPrompt(provider) case config.AgentTitle: basePrompt = TitlePrompt(provider) case config.AgentTask: basePrompt = TaskPrompt(provider) case config.AgentSummarizer: basePrompt = SummarizerPrompt(provider) default: basePrompt = "You are a helpful assistant" } if agentName == config.AgentCoder || agentName == config.AgentTask { // Add context from project-specific instruction files if they exist contextContent := getContextFromPaths() logging.Debug("Context content", "Context", contextContent) if contextContent != "" { return fmt.Sprintf("%s\n\n# Project-Specific Context\n Make sure to follow the instructions in the context below\n%s", basePrompt, contextContent) } } return basePrompt } var ( onceContext sync.Once contextContent string ) func getContextFromPaths() string { onceContext.Do(func() { var ( cfg = config.Get() workDir = cfg.WorkingDir contextPaths = cfg.ContextPaths ) contextContent = processContextPaths(workDir, contextPaths) }) return contextContent } func processContextPaths(workDir string, paths []string) string { var ( wg sync.WaitGroup resultCh = make(chan string) ) // Track processed files to avoid duplicates processedFiles := make(map[string]bool) var processedMutex sync.Mutex for _, path := range paths { wg.Add(1) go func(p string) { defer wg.Done() if strings.HasSuffix(p, "/") { filepath.WalkDir(filepath.Join(workDir, p), func(path string, d os.DirEntry, err error) error { if err != nil { return err } if !d.IsDir() { // Check if we've already processed this file (case-insensitive) processedMutex.Lock() lowerPath := strings.ToLower(path) if !processedFiles[lowerPath] { processedFiles[lowerPath] = true processedMutex.Unlock() if result := processFile(path); result != "" { resultCh <- result } } else { processedMutex.Unlock() } } return nil }) } else { fullPath := filepath.Join(workDir, p) // Check if we've already processed this file (case-insensitive) processedMutex.Lock() lowerPath := strings.ToLower(fullPath) if !processedFiles[lowerPath] { processedFiles[lowerPath] = true processedMutex.Unlock() result := processFile(fullPath) if result != "" { resultCh <- result } } else { processedMutex.Unlock() } } }(path) } go func() { wg.Wait() close(resultCh) }() results := make([]string, 0) for result := range resultCh { results = append(results, result) } return strings.Join(results, "\n") } func processFile(filePath string) string { content, err := os.ReadFile(filePath) if err != nil { return "" } return "# From:" + filePath + "\n" + string(content) } ================================================ FILE: internal/llm/prompt/prompt_test.go ================================================ package prompt import ( "fmt" "os" "path/filepath" "testing" "github.com/opencode-ai/opencode/internal/config" "github.com/stretchr/testify/assert" "github.com/stretchr/testify/require" ) func TestGetContextFromPaths(t *testing.T) { t.Parallel() tmpDir := t.TempDir() _, err := config.Load(tmpDir, false) if err != nil { t.Fatalf("Failed to load config: %v", err) } cfg := config.Get() cfg.WorkingDir = tmpDir cfg.ContextPaths = []string{ "file.txt", "directory/", } testFiles := []string{ "file.txt", "directory/file_a.txt", "directory/file_b.txt", "directory/file_c.txt", } createTestFiles(t, tmpDir, testFiles) context := getContextFromPaths() expectedContext := fmt.Sprintf("# From:%s/file.txt\nfile.txt: test content\n# From:%s/directory/file_a.txt\ndirectory/file_a.txt: test content\n# From:%s/directory/file_b.txt\ndirectory/file_b.txt: test content\n# From:%s/directory/file_c.txt\ndirectory/file_c.txt: test content", tmpDir, tmpDir, tmpDir, tmpDir) assert.Equal(t, expectedContext, context) } func createTestFiles(t *testing.T, tmpDir string, testFiles []string) { t.Helper() for _, path := range testFiles { fullPath := filepath.Join(tmpDir, path) if path[len(path)-1] == '/' { err := os.MkdirAll(fullPath, 0755) require.NoError(t, err) } else { dir := filepath.Dir(fullPath) err := os.MkdirAll(dir, 0755) require.NoError(t, err) err = os.WriteFile(fullPath, []byte(path+": test content"), 0644) require.NoError(t, err) } } } ================================================ FILE: internal/llm/prompt/summarizer.go ================================================ package prompt import "github.com/opencode-ai/opencode/internal/llm/models" func SummarizerPrompt(_ models.ModelProvider) string { return `You are a helpful AI assistant tasked with summarizing conversations. When asked to summarize, provide a detailed but concise summary of the conversation. Focus on information that would be helpful for continuing the conversation, including: - What was done - What is currently being worked on - Which files are being modified - What needs to be done next Your summary should be comprehensive enough to provide context but concise enough to be quickly understood.` } ================================================ FILE: internal/llm/prompt/task.go ================================================ package prompt import ( "fmt" "github.com/opencode-ai/opencode/internal/llm/models" ) func TaskPrompt(_ models.ModelProvider) string { agentPrompt := `You are an agent for OpenCode. Given the user's prompt, you should use the tools available to you to answer the user's question. Notes: 1. IMPORTANT: You should be concise, direct, and to the point, since your responses will be displayed on a command line interface. Answer the user's question directly, without elaboration, explanation, or details. One word answers are best. Avoid introductions, conclusions, and explanations. You MUST avoid text before/after your response, such as "The answer is .", "Here is the content of the file..." or "Based on the information provided, the answer is..." or "Here is what I will do next...". 2. When relevant, share file names and code snippets relevant to the query 3. Any file paths you return in your final response MUST be absolute. DO NOT use relative paths.` return fmt.Sprintf("%s\n%s\n", agentPrompt, getEnvironmentInfo()) } ================================================ FILE: internal/llm/prompt/title.go ================================================ package prompt import "github.com/opencode-ai/opencode/internal/llm/models" func TitlePrompt(_ models.ModelProvider) string { return `you will generate a short title based on the first message a user begins a conversation with - ensure it is not more than 50 characters long - the title should be a summary of the user's message - it should be one line long - do not use quotes or colons - the entire text you return will be used as the title - never return anything that is more than one sentence (one line) long` } ================================================ FILE: internal/llm/provider/anthropic.go ================================================ package provider import ( "context" "encoding/json" "errors" "fmt" "io" "strings" "time" "github.com/anthropics/anthropic-sdk-go" "github.com/anthropics/anthropic-sdk-go/bedrock" "github.com/anthropics/anthropic-sdk-go/option" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/llm/models" toolsPkg "github.com/opencode-ai/opencode/internal/llm/tools" "github.com/opencode-ai/opencode/internal/logging" "github.com/opencode-ai/opencode/internal/message" ) type anthropicOptions struct { useBedrock bool disableCache bool shouldThink func(userMessage string) bool } type AnthropicOption func(*anthropicOptions) type anthropicClient struct { providerOptions providerClientOptions options anthropicOptions client anthropic.Client } type AnthropicClient ProviderClient func newAnthropicClient(opts providerClientOptions) AnthropicClient { anthropicOpts := anthropicOptions{} for _, o := range opts.anthropicOptions { o(&anthropicOpts) } anthropicClientOptions := []option.RequestOption{} if opts.apiKey != "" { anthropicClientOptions = append(anthropicClientOptions, option.WithAPIKey(opts.apiKey)) } if anthropicOpts.useBedrock { anthropicClientOptions = append(anthropicClientOptions, bedrock.WithLoadDefaultConfig(context.Background())) } client := anthropic.NewClient(anthropicClientOptions...) return &anthropicClient{ providerOptions: opts, options: anthropicOpts, client: client, } } func (a *anthropicClient) convertMessages(messages []message.Message) (anthropicMessages []anthropic.MessageParam) { for i, msg := range messages { cache := false if i > len(messages)-3 { cache = true } switch msg.Role { case message.User: content := anthropic.NewTextBlock(msg.Content().String()) if cache && !a.options.disableCache { content.OfText.CacheControl = anthropic.CacheControlEphemeralParam{ Type: "ephemeral", } } var contentBlocks []anthropic.ContentBlockParamUnion contentBlocks = append(contentBlocks, content) for _, binaryContent := range msg.BinaryContent() { base64Image := binaryContent.String(models.ProviderAnthropic) imageBlock := anthropic.NewImageBlockBase64(binaryContent.MIMEType, base64Image) contentBlocks = append(contentBlocks, imageBlock) } anthropicMessages = append(anthropicMessages, anthropic.NewUserMessage(contentBlocks...)) case message.Assistant: blocks := []anthropic.ContentBlockParamUnion{} if msg.Content().String() != "" { content := anthropic.NewTextBlock(msg.Content().String()) if cache && !a.options.disableCache { content.OfText.CacheControl = anthropic.CacheControlEphemeralParam{ Type: "ephemeral", } } blocks = append(blocks, content) } for _, toolCall := range msg.ToolCalls() { var inputMap map[string]any err := json.Unmarshal([]byte(toolCall.Input), &inputMap) if err != nil { continue } blocks = append(blocks, anthropic.NewToolUseBlock(toolCall.ID, inputMap, toolCall.Name)) } if len(blocks) == 0 { logging.Warn("There is a message without content, investigate, this should not happen") continue } anthropicMessages = append(anthropicMessages, anthropic.NewAssistantMessage(blocks...)) case message.Tool: results := make([]anthropic.ContentBlockParamUnion, len(msg.ToolResults())) for i, toolResult := range msg.ToolResults() { results[i] = anthropic.NewToolResultBlock(toolResult.ToolCallID, toolResult.Content, toolResult.IsError) } anthropicMessages = append(anthropicMessages, anthropic.NewUserMessage(results...)) } } return } func (a *anthropicClient) convertTools(tools []toolsPkg.BaseTool) []anthropic.ToolUnionParam { anthropicTools := make([]anthropic.ToolUnionParam, len(tools)) for i, tool := range tools { info := tool.Info() toolParam := anthropic.ToolParam{ Name: info.Name, Description: anthropic.String(info.Description), InputSchema: anthropic.ToolInputSchemaParam{ Properties: info.Parameters, // TODO: figure out how we can tell claude the required fields? }, } if i == len(tools)-1 && !a.options.disableCache { toolParam.CacheControl = anthropic.CacheControlEphemeralParam{ Type: "ephemeral", } } anthropicTools[i] = anthropic.ToolUnionParam{OfTool: &toolParam} } return anthropicTools } func (a *anthropicClient) finishReason(reason string) message.FinishReason { switch reason { case "end_turn": return message.FinishReasonEndTurn case "max_tokens": return message.FinishReasonMaxTokens case "tool_use": return message.FinishReasonToolUse case "stop_sequence": return message.FinishReasonEndTurn default: return message.FinishReasonUnknown } } func (a *anthropicClient) preparedMessages(messages []anthropic.MessageParam, tools []anthropic.ToolUnionParam) anthropic.MessageNewParams { var thinkingParam anthropic.ThinkingConfigParamUnion lastMessage := messages[len(messages)-1] isUser := lastMessage.Role == anthropic.MessageParamRoleUser messageContent := "" temperature := anthropic.Float(0) if isUser { for _, m := range lastMessage.Content { if m.OfText != nil && m.OfText.Text != "" { messageContent = m.OfText.Text } } if messageContent != "" && a.options.shouldThink != nil && a.options.shouldThink(messageContent) { thinkingParam = anthropic.ThinkingConfigParamOfEnabled(int64(float64(a.providerOptions.maxTokens) * 0.8)) temperature = anthropic.Float(1) } } return anthropic.MessageNewParams{ Model: anthropic.Model(a.providerOptions.model.APIModel), MaxTokens: a.providerOptions.maxTokens, Temperature: temperature, Messages: messages, Tools: tools, Thinking: thinkingParam, System: []anthropic.TextBlockParam{ { Text: a.providerOptions.systemMessage, CacheControl: anthropic.CacheControlEphemeralParam{ Type: "ephemeral", }, }, }, } } func (a *anthropicClient) send(ctx context.Context, messages []message.Message, tools []toolsPkg.BaseTool) (resposne *ProviderResponse, err error) { preparedMessages := a.preparedMessages(a.convertMessages(messages), a.convertTools(tools)) cfg := config.Get() if cfg.Debug { jsonData, _ := json.Marshal(preparedMessages) logging.Debug("Prepared messages", "messages", string(jsonData)) } attempts := 0 for { attempts++ anthropicResponse, err := a.client.Messages.New( ctx, preparedMessages, ) // If there is an error we are going to see if we can retry the call if err != nil { logging.Error("Error in Anthropic API call", "error", err) retry, after, retryErr := a.shouldRetry(attempts, err) if retryErr != nil { return nil, retryErr } if retry { logging.WarnPersist(fmt.Sprintf("Retrying due to rate limit... attempt %d of %d", attempts, maxRetries), logging.PersistTimeArg, time.Millisecond*time.Duration(after+100)) select { case <-ctx.Done(): return nil, ctx.Err() case <-time.After(time.Duration(after) * time.Millisecond): continue } } return nil, retryErr } content := "" for _, block := range anthropicResponse.Content { if text, ok := block.AsAny().(anthropic.TextBlock); ok { content += text.Text } } return &ProviderResponse{ Content: content, ToolCalls: a.toolCalls(*anthropicResponse), Usage: a.usage(*anthropicResponse), }, nil } } func (a *anthropicClient) stream(ctx context.Context, messages []message.Message, tools []toolsPkg.BaseTool) <-chan ProviderEvent { preparedMessages := a.preparedMessages(a.convertMessages(messages), a.convertTools(tools)) cfg := config.Get() var sessionId string requestSeqId := (len(messages) + 1) / 2 if cfg.Debug { if sid, ok := ctx.Value(toolsPkg.SessionIDContextKey).(string); ok { sessionId = sid } jsonData, _ := json.Marshal(preparedMessages) if sessionId != "" { filepath := logging.WriteRequestMessageJson(sessionId, requestSeqId, preparedMessages) logging.Debug("Prepared messages", "filepath", filepath) } else { logging.Debug("Prepared messages", "messages", string(jsonData)) } } attempts := 0 eventChan := make(chan ProviderEvent) go func() { for { attempts++ anthropicStream := a.client.Messages.NewStreaming( ctx, preparedMessages, ) accumulatedMessage := anthropic.Message{} currentToolCallID := "" for anthropicStream.Next() { event := anthropicStream.Current() err := accumulatedMessage.Accumulate(event) if err != nil { logging.Warn("Error accumulating message", "error", err) continue } switch event := event.AsAny().(type) { case anthropic.ContentBlockStartEvent: if event.ContentBlock.Type == "text" { eventChan <- ProviderEvent{Type: EventContentStart} } else if event.ContentBlock.Type == "tool_use" { currentToolCallID = event.ContentBlock.ID eventChan <- ProviderEvent{ Type: EventToolUseStart, ToolCall: &message.ToolCall{ ID: event.ContentBlock.ID, Name: event.ContentBlock.Name, Finished: false, }, } } case anthropic.ContentBlockDeltaEvent: if event.Delta.Type == "thinking_delta" && event.Delta.Thinking != "" { eventChan <- ProviderEvent{ Type: EventThinkingDelta, Thinking: event.Delta.Thinking, } } else if event.Delta.Type == "text_delta" && event.Delta.Text != "" { eventChan <- ProviderEvent{ Type: EventContentDelta, Content: event.Delta.Text, } } else if event.Delta.Type == "input_json_delta" { if currentToolCallID != "" { eventChan <- ProviderEvent{ Type: EventToolUseDelta, ToolCall: &message.ToolCall{ ID: currentToolCallID, Finished: false, Input: event.Delta.JSON.PartialJSON.Raw(), }, } } } case anthropic.ContentBlockStopEvent: if currentToolCallID != "" { eventChan <- ProviderEvent{ Type: EventToolUseStop, ToolCall: &message.ToolCall{ ID: currentToolCallID, }, } currentToolCallID = "" } else { eventChan <- ProviderEvent{Type: EventContentStop} } case anthropic.MessageStopEvent: content := "" for _, block := range accumulatedMessage.Content { if text, ok := block.AsAny().(anthropic.TextBlock); ok { content += text.Text } } eventChan <- ProviderEvent{ Type: EventComplete, Response: &ProviderResponse{ Content: content, ToolCalls: a.toolCalls(accumulatedMessage), Usage: a.usage(accumulatedMessage), FinishReason: a.finishReason(string(accumulatedMessage.StopReason)), }, } } } err := anthropicStream.Err() if err == nil || errors.Is(err, io.EOF) { close(eventChan) return } // If there is an error we are going to see if we can retry the call retry, after, retryErr := a.shouldRetry(attempts, err) if retryErr != nil { eventChan <- ProviderEvent{Type: EventError, Error: retryErr} close(eventChan) return } if retry { logging.WarnPersist(fmt.Sprintf("Retrying due to rate limit... attempt %d of %d", attempts, maxRetries), logging.PersistTimeArg, time.Millisecond*time.Duration(after+100)) select { case <-ctx.Done(): // context cancelled if ctx.Err() != nil { eventChan <- ProviderEvent{Type: EventError, Error: ctx.Err()} } close(eventChan) return case <-time.After(time.Duration(after) * time.Millisecond): continue } } if ctx.Err() != nil { eventChan <- ProviderEvent{Type: EventError, Error: ctx.Err()} } close(eventChan) return } }() return eventChan } func (a *anthropicClient) shouldRetry(attempts int, err error) (bool, int64, error) { var apierr *anthropic.Error if !errors.As(err, &apierr) { return false, 0, err } if apierr.StatusCode != 429 && apierr.StatusCode != 529 { return false, 0, err } if attempts > maxRetries { return false, 0, fmt.Errorf("maximum retry attempts reached for rate limit: %d retries", maxRetries) } retryMs := 0 retryAfterValues := apierr.Response.Header.Values("Retry-After") backoffMs := 2000 * (1 << (attempts - 1)) jitterMs := int(float64(backoffMs) * 0.2) retryMs = backoffMs + jitterMs if len(retryAfterValues) > 0 { if _, err := fmt.Sscanf(retryAfterValues[0], "%d", &retryMs); err == nil { retryMs = retryMs * 1000 } } return true, int64(retryMs), nil } func (a *anthropicClient) toolCalls(msg anthropic.Message) []message.ToolCall { var toolCalls []message.ToolCall for _, block := range msg.Content { switch variant := block.AsAny().(type) { case anthropic.ToolUseBlock: toolCall := message.ToolCall{ ID: variant.ID, Name: variant.Name, Input: string(variant.Input), Type: string(variant.Type), Finished: true, } toolCalls = append(toolCalls, toolCall) } } return toolCalls } func (a *anthropicClient) usage(msg anthropic.Message) TokenUsage { return TokenUsage{ InputTokens: msg.Usage.InputTokens, OutputTokens: msg.Usage.OutputTokens, CacheCreationTokens: msg.Usage.CacheCreationInputTokens, CacheReadTokens: msg.Usage.CacheReadInputTokens, } } func WithAnthropicBedrock(useBedrock bool) AnthropicOption { return func(options *anthropicOptions) { options.useBedrock = useBedrock } } func WithAnthropicDisableCache() AnthropicOption { return func(options *anthropicOptions) { options.disableCache = true } } func DefaultShouldThinkFn(s string) bool { return strings.Contains(strings.ToLower(s), "think") } func WithAnthropicShouldThinkFn(fn func(string) bool) AnthropicOption { return func(options *anthropicOptions) { options.shouldThink = fn } } ================================================ FILE: internal/llm/provider/azure.go ================================================ package provider import ( "os" "github.com/Azure/azure-sdk-for-go/sdk/azidentity" "github.com/openai/openai-go" "github.com/openai/openai-go/azure" "github.com/openai/openai-go/option" ) type azureClient struct { *openaiClient } type AzureClient ProviderClient func newAzureClient(opts providerClientOptions) AzureClient { endpoint := os.Getenv("AZURE_OPENAI_ENDPOINT") // ex: https://foo.openai.azure.com apiVersion := os.Getenv("AZURE_OPENAI_API_VERSION") // ex: 2025-04-01-preview if endpoint == "" || apiVersion == "" { return &azureClient{openaiClient: newOpenAIClient(opts).(*openaiClient)} } reqOpts := []option.RequestOption{ azure.WithEndpoint(endpoint, apiVersion), } if opts.apiKey != "" || os.Getenv("AZURE_OPENAI_API_KEY") != "" { key := opts.apiKey if key == "" { key = os.Getenv("AZURE_OPENAI_API_KEY") } reqOpts = append(reqOpts, azure.WithAPIKey(key)) } else if cred, err := azidentity.NewDefaultAzureCredential(nil); err == nil { reqOpts = append(reqOpts, azure.WithTokenCredential(cred)) } base := &openaiClient{ providerOptions: opts, client: openai.NewClient(reqOpts...), } return &azureClient{openaiClient: base} } ================================================ FILE: internal/llm/provider/bedrock.go ================================================ package provider import ( "context" "errors" "fmt" "os" "strings" "github.com/opencode-ai/opencode/internal/llm/tools" "github.com/opencode-ai/opencode/internal/message" ) type bedrockOptions struct { // Bedrock specific options can be added here } type BedrockOption func(*bedrockOptions) type bedrockClient struct { providerOptions providerClientOptions options bedrockOptions childProvider ProviderClient } type BedrockClient ProviderClient func newBedrockClient(opts providerClientOptions) BedrockClient { bedrockOpts := bedrockOptions{} // Apply bedrock specific options if they are added in the future // Get AWS region from environment region := os.Getenv("AWS_REGION") if region == "" { region = os.Getenv("AWS_DEFAULT_REGION") } if region == "" { region = "us-east-1" // default region } if len(region) < 2 { return &bedrockClient{ providerOptions: opts, options: bedrockOpts, childProvider: nil, // Will cause an error when used } } // Prefix the model name with region regionPrefix := region[:2] modelName := opts.model.APIModel opts.model.APIModel = fmt.Sprintf("%s.%s", regionPrefix, modelName) // Determine which provider to use based on the model if strings.Contains(string(opts.model.APIModel), "anthropic") { // Create Anthropic client with Bedrock configuration anthropicOpts := opts anthropicOpts.anthropicOptions = append(anthropicOpts.anthropicOptions, WithAnthropicBedrock(true), WithAnthropicDisableCache(), ) return &bedrockClient{ providerOptions: opts, options: bedrockOpts, childProvider: newAnthropicClient(anthropicOpts), } } // Return client with nil childProvider if model is not supported // This will cause an error when used return &bedrockClient{ providerOptions: opts, options: bedrockOpts, childProvider: nil, } } func (b *bedrockClient) send(ctx context.Context, messages []message.Message, tools []tools.BaseTool) (*ProviderResponse, error) { if b.childProvider == nil { return nil, errors.New("unsupported model for bedrock provider") } return b.childProvider.send(ctx, messages, tools) } func (b *bedrockClient) stream(ctx context.Context, messages []message.Message, tools []tools.BaseTool) <-chan ProviderEvent { eventChan := make(chan ProviderEvent) if b.childProvider == nil { go func() { eventChan <- ProviderEvent{ Type: EventError, Error: errors.New("unsupported model for bedrock provider"), } close(eventChan) }() return eventChan } return b.childProvider.stream(ctx, messages, tools) } ================================================ FILE: internal/llm/provider/copilot.go ================================================ package provider import ( "context" "encoding/json" "errors" "fmt" "io" "net/http" "os" "time" "github.com/openai/openai-go" "github.com/openai/openai-go/option" "github.com/openai/openai-go/shared" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/llm/models" toolsPkg "github.com/opencode-ai/opencode/internal/llm/tools" "github.com/opencode-ai/opencode/internal/logging" "github.com/opencode-ai/opencode/internal/message" ) type copilotOptions struct { reasoningEffort string extraHeaders map[string]string bearerToken string } type CopilotOption func(*copilotOptions) type copilotClient struct { providerOptions providerClientOptions options copilotOptions client openai.Client httpClient *http.Client } type CopilotClient ProviderClient // CopilotTokenResponse represents the response from GitHub's token exchange endpoint type CopilotTokenResponse struct { Token string `json:"token"` ExpiresAt int64 `json:"expires_at"` } func (c *copilotClient) isAnthropicModel() bool { for _, modelId := range models.CopilotAnthropicModels { if c.providerOptions.model.ID == modelId { return true } } return false } // loadGitHubToken loads the GitHub OAuth token from the standard GitHub CLI/Copilot locations // exchangeGitHubToken exchanges a GitHub token for a Copilot bearer token func (c *copilotClient) exchangeGitHubToken(githubToken string) (string, error) { req, err := http.NewRequest("GET", "https://api.github.com/copilot_internal/v2/token", nil) if err != nil { return "", fmt.Errorf("failed to create token exchange request: %w", err) } req.Header.Set("Authorization", "Token "+githubToken) req.Header.Set("User-Agent", "OpenCode/1.0") resp, err := c.httpClient.Do(req) if err != nil { return "", fmt.Errorf("failed to exchange GitHub token: %w", err) } defer resp.Body.Close() if resp.StatusCode != http.StatusOK { body, _ := io.ReadAll(resp.Body) return "", fmt.Errorf("token exchange failed with status %d: %s", resp.StatusCode, string(body)) } var tokenResp CopilotTokenResponse if err := json.NewDecoder(resp.Body).Decode(&tokenResp); err != nil { return "", fmt.Errorf("failed to decode token response: %w", err) } return tokenResp.Token, nil } func newCopilotClient(opts providerClientOptions) CopilotClient { copilotOpts := copilotOptions{ reasoningEffort: "medium", } // Apply copilot-specific options for _, o := range opts.copilotOptions { o(&copilotOpts) } // Create HTTP client for token exchange httpClient := &http.Client{ Timeout: 30 * time.Second, } var bearerToken string // If bearer token is already provided, use it if copilotOpts.bearerToken != "" { bearerToken = copilotOpts.bearerToken } else { // Try to get GitHub token from multiple sources var githubToken string // 1. Environment variable githubToken = os.Getenv("GITHUB_TOKEN") // 2. API key from options if githubToken == "" { githubToken = opts.apiKey } // 3. Standard GitHub CLI/Copilot locations if githubToken == "" { var err error githubToken, err = config.LoadGitHubToken() if err != nil { logging.Debug("Failed to load GitHub token from standard locations", "error", err) } } if githubToken == "" { logging.Error("GitHub token is required for Copilot provider. Set GITHUB_TOKEN environment variable, configure it in opencode.json, or ensure GitHub CLI/Copilot is properly authenticated.") return &copilotClient{ providerOptions: opts, options: copilotOpts, httpClient: httpClient, } } // Create a temporary client for token exchange tempClient := &copilotClient{ providerOptions: opts, options: copilotOpts, httpClient: httpClient, } // Exchange GitHub token for bearer token var err error bearerToken, err = tempClient.exchangeGitHubToken(githubToken) if err != nil { logging.Error("Failed to exchange GitHub token for Copilot bearer token", "error", err) return &copilotClient{ providerOptions: opts, options: copilotOpts, httpClient: httpClient, } } } copilotOpts.bearerToken = bearerToken // GitHub Copilot API base URL baseURL := "https://api.githubcopilot.com" openaiClientOptions := []option.RequestOption{ option.WithBaseURL(baseURL), option.WithAPIKey(bearerToken), // Use bearer token as API key } // Add GitHub Copilot specific headers openaiClientOptions = append(openaiClientOptions, option.WithHeader("Editor-Version", "OpenCode/1.0"), option.WithHeader("Editor-Plugin-Version", "OpenCode/1.0"), option.WithHeader("Copilot-Integration-Id", "vscode-chat"), ) // Add any extra headers if copilotOpts.extraHeaders != nil { for key, value := range copilotOpts.extraHeaders { openaiClientOptions = append(openaiClientOptions, option.WithHeader(key, value)) } } client := openai.NewClient(openaiClientOptions...) // logging.Debug("Copilot client created", "opts", opts, "copilotOpts", copilotOpts, "model", opts.model) return &copilotClient{ providerOptions: opts, options: copilotOpts, client: client, httpClient: httpClient, } } func (c *copilotClient) convertMessages(messages []message.Message) (copilotMessages []openai.ChatCompletionMessageParamUnion) { // Add system message first copilotMessages = append(copilotMessages, openai.SystemMessage(c.providerOptions.systemMessage)) for _, msg := range messages { switch msg.Role { case message.User: var content []openai.ChatCompletionContentPartUnionParam textBlock := openai.ChatCompletionContentPartTextParam{Text: msg.Content().String()} content = append(content, openai.ChatCompletionContentPartUnionParam{OfText: &textBlock}) for _, binaryContent := range msg.BinaryContent() { imageURL := openai.ChatCompletionContentPartImageImageURLParam{URL: binaryContent.String(models.ProviderCopilot)} imageBlock := openai.ChatCompletionContentPartImageParam{ImageURL: imageURL} content = append(content, openai.ChatCompletionContentPartUnionParam{OfImageURL: &imageBlock}) } copilotMessages = append(copilotMessages, openai.UserMessage(content)) case message.Assistant: assistantMsg := openai.ChatCompletionAssistantMessageParam{ Role: "assistant", } if msg.Content().String() != "" { assistantMsg.Content = openai.ChatCompletionAssistantMessageParamContentUnion{ OfString: openai.String(msg.Content().String()), } } if len(msg.ToolCalls()) > 0 { assistantMsg.ToolCalls = make([]openai.ChatCompletionMessageToolCallParam, len(msg.ToolCalls())) for i, call := range msg.ToolCalls() { assistantMsg.ToolCalls[i] = openai.ChatCompletionMessageToolCallParam{ ID: call.ID, Type: "function", Function: openai.ChatCompletionMessageToolCallFunctionParam{ Name: call.Name, Arguments: call.Input, }, } } } copilotMessages = append(copilotMessages, openai.ChatCompletionMessageParamUnion{ OfAssistant: &assistantMsg, }) case message.Tool: for _, result := range msg.ToolResults() { copilotMessages = append(copilotMessages, openai.ToolMessage(result.Content, result.ToolCallID), ) } } } return } func (c *copilotClient) convertTools(tools []toolsPkg.BaseTool) []openai.ChatCompletionToolParam { copilotTools := make([]openai.ChatCompletionToolParam, len(tools)) for i, tool := range tools { info := tool.Info() copilotTools[i] = openai.ChatCompletionToolParam{ Function: openai.FunctionDefinitionParam{ Name: info.Name, Description: openai.String(info.Description), Parameters: openai.FunctionParameters{ "type": "object", "properties": info.Parameters, "required": info.Required, }, }, } } return copilotTools } func (c *copilotClient) finishReason(reason string) message.FinishReason { switch reason { case "stop": return message.FinishReasonEndTurn case "length": return message.FinishReasonMaxTokens case "tool_calls": return message.FinishReasonToolUse default: return message.FinishReasonUnknown } } func (c *copilotClient) preparedParams(messages []openai.ChatCompletionMessageParamUnion, tools []openai.ChatCompletionToolParam) openai.ChatCompletionNewParams { params := openai.ChatCompletionNewParams{ Model: openai.ChatModel(c.providerOptions.model.APIModel), Messages: messages, Tools: tools, } if c.providerOptions.model.CanReason == true { params.MaxCompletionTokens = openai.Int(c.providerOptions.maxTokens) switch c.options.reasoningEffort { case "low": params.ReasoningEffort = shared.ReasoningEffortLow case "medium": params.ReasoningEffort = shared.ReasoningEffortMedium case "high": params.ReasoningEffort = shared.ReasoningEffortHigh default: params.ReasoningEffort = shared.ReasoningEffortMedium } } else { params.MaxTokens = openai.Int(c.providerOptions.maxTokens) } return params } func (c *copilotClient) send(ctx context.Context, messages []message.Message, tools []toolsPkg.BaseTool) (response *ProviderResponse, err error) { params := c.preparedParams(c.convertMessages(messages), c.convertTools(tools)) cfg := config.Get() var sessionId string requestSeqId := (len(messages) + 1) / 2 if cfg.Debug { // jsonData, _ := json.Marshal(params) // logging.Debug("Prepared messages", "messages", string(jsonData)) if sid, ok := ctx.Value(toolsPkg.SessionIDContextKey).(string); ok { sessionId = sid } jsonData, _ := json.Marshal(params) if sessionId != "" { filepath := logging.WriteRequestMessageJson(sessionId, requestSeqId, params) logging.Debug("Prepared messages", "filepath", filepath) } else { logging.Debug("Prepared messages", "messages", string(jsonData)) } } attempts := 0 for { attempts++ copilotResponse, err := c.client.Chat.Completions.New( ctx, params, ) // If there is an error we are going to see if we can retry the call if err != nil { retry, after, retryErr := c.shouldRetry(attempts, err) if retryErr != nil { return nil, retryErr } if retry { logging.WarnPersist(fmt.Sprintf("Retrying due to rate limit... attempt %d of %d", attempts, maxRetries), logging.PersistTimeArg, time.Millisecond*time.Duration(after+100)) select { case <-ctx.Done(): return nil, ctx.Err() case <-time.After(time.Duration(after) * time.Millisecond): continue } } return nil, retryErr } content := "" if copilotResponse.Choices[0].Message.Content != "" { content = copilotResponse.Choices[0].Message.Content } toolCalls := c.toolCalls(*copilotResponse) finishReason := c.finishReason(string(copilotResponse.Choices[0].FinishReason)) if len(toolCalls) > 0 { finishReason = message.FinishReasonToolUse } return &ProviderResponse{ Content: content, ToolCalls: toolCalls, Usage: c.usage(*copilotResponse), FinishReason: finishReason, }, nil } } func (c *copilotClient) stream(ctx context.Context, messages []message.Message, tools []toolsPkg.BaseTool) <-chan ProviderEvent { params := c.preparedParams(c.convertMessages(messages), c.convertTools(tools)) params.StreamOptions = openai.ChatCompletionStreamOptionsParam{ IncludeUsage: openai.Bool(true), } cfg := config.Get() var sessionId string requestSeqId := (len(messages) + 1) / 2 if cfg.Debug { if sid, ok := ctx.Value(toolsPkg.SessionIDContextKey).(string); ok { sessionId = sid } jsonData, _ := json.Marshal(params) if sessionId != "" { filepath := logging.WriteRequestMessageJson(sessionId, requestSeqId, params) logging.Debug("Prepared messages", "filepath", filepath) } else { logging.Debug("Prepared messages", "messages", string(jsonData)) } } attempts := 0 eventChan := make(chan ProviderEvent) go func() { for { attempts++ copilotStream := c.client.Chat.Completions.NewStreaming( ctx, params, ) acc := openai.ChatCompletionAccumulator{} currentContent := "" toolCalls := make([]message.ToolCall, 0) var currentToolCallId string var currentToolCall openai.ChatCompletionMessageToolCall var msgToolCalls []openai.ChatCompletionMessageToolCall for copilotStream.Next() { chunk := copilotStream.Current() acc.AddChunk(chunk) if cfg.Debug { logging.AppendToStreamSessionLogJson(sessionId, requestSeqId, chunk) } for _, choice := range chunk.Choices { if choice.Delta.Content != "" { eventChan <- ProviderEvent{ Type: EventContentDelta, Content: choice.Delta.Content, } currentContent += choice.Delta.Content } } if c.isAnthropicModel() { // Monkeypatch adapter for Sonnet-4 multi-tool use for _, choice := range chunk.Choices { if choice.Delta.ToolCalls != nil && len(choice.Delta.ToolCalls) > 0 { toolCall := choice.Delta.ToolCalls[0] // Detect tool use start if currentToolCallId == "" { if toolCall.ID != "" { currentToolCallId = toolCall.ID currentToolCall = openai.ChatCompletionMessageToolCall{ ID: toolCall.ID, Type: "function", Function: openai.ChatCompletionMessageToolCallFunction{ Name: toolCall.Function.Name, Arguments: toolCall.Function.Arguments, }, } } } else { // Delta tool use if toolCall.ID == "" { currentToolCall.Function.Arguments += toolCall.Function.Arguments } else { // Detect new tool use if toolCall.ID != currentToolCallId { msgToolCalls = append(msgToolCalls, currentToolCall) currentToolCallId = toolCall.ID currentToolCall = openai.ChatCompletionMessageToolCall{ ID: toolCall.ID, Type: "function", Function: openai.ChatCompletionMessageToolCallFunction{ Name: toolCall.Function.Name, Arguments: toolCall.Function.Arguments, }, } } } } } if choice.FinishReason == "tool_calls" { msgToolCalls = append(msgToolCalls, currentToolCall) acc.ChatCompletion.Choices[0].Message.ToolCalls = msgToolCalls } } } } err := copilotStream.Err() if err == nil || errors.Is(err, io.EOF) { if cfg.Debug { respFilepath := logging.WriteChatResponseJson(sessionId, requestSeqId, acc.ChatCompletion) logging.Debug("Chat completion response", "filepath", respFilepath) } // Stream completed successfully finishReason := c.finishReason(string(acc.ChatCompletion.Choices[0].FinishReason)) if len(acc.ChatCompletion.Choices[0].Message.ToolCalls) > 0 { toolCalls = append(toolCalls, c.toolCalls(acc.ChatCompletion)...) } if len(toolCalls) > 0 { finishReason = message.FinishReasonToolUse } eventChan <- ProviderEvent{ Type: EventComplete, Response: &ProviderResponse{ Content: currentContent, ToolCalls: toolCalls, Usage: c.usage(acc.ChatCompletion), FinishReason: finishReason, }, } close(eventChan) return } // If there is an error we are going to see if we can retry the call retry, after, retryErr := c.shouldRetry(attempts, err) if retryErr != nil { eventChan <- ProviderEvent{Type: EventError, Error: retryErr} close(eventChan) return } // shouldRetry is not catching the max retries... // TODO: Figure out why if attempts > maxRetries { logging.Warn("Maximum retry attempts reached for rate limit", "attempts", attempts, "max_retries", maxRetries) retry = false } if retry { logging.WarnPersist(fmt.Sprintf("Retrying due to rate limit... attempt %d of %d (paused for %d ms)", attempts, maxRetries, after), logging.PersistTimeArg, time.Millisecond*time.Duration(after+100)) select { case <-ctx.Done(): // context cancelled if ctx.Err() == nil { eventChan <- ProviderEvent{Type: EventError, Error: ctx.Err()} } close(eventChan) return case <-time.After(time.Duration(after) * time.Millisecond): continue } } eventChan <- ProviderEvent{Type: EventError, Error: retryErr} close(eventChan) return } }() return eventChan } func (c *copilotClient) shouldRetry(attempts int, err error) (bool, int64, error) { var apierr *openai.Error if !errors.As(err, &apierr) { return false, 0, err } // Check for token expiration (401 Unauthorized) if apierr.StatusCode == 401 { // Try to refresh the bearer token var githubToken string // 1. Environment variable githubToken = os.Getenv("GITHUB_TOKEN") // 2. API key from options if githubToken == "" { githubToken = c.providerOptions.apiKey } // 3. Standard GitHub CLI/Copilot locations if githubToken == "" { var err error githubToken, err = config.LoadGitHubToken() if err != nil { logging.Debug("Failed to load GitHub token from standard locations during retry", "error", err) } } if githubToken != "" { newBearerToken, tokenErr := c.exchangeGitHubToken(githubToken) if tokenErr == nil { c.options.bearerToken = newBearerToken // Update the client with the new token // Note: This is a simplified approach. In a production system, // you might want to recreate the entire client with the new token logging.Info("Refreshed Copilot bearer token") return true, 1000, nil // Retry immediately with new token } logging.Error("Failed to refresh Copilot bearer token", "error", tokenErr) } return false, 0, fmt.Errorf("authentication failed: %w", err) } logging.Debug("Copilot API Error", "status", apierr.StatusCode, "headers", apierr.Response.Header, "body", apierr.RawJSON()) if apierr.StatusCode != 429 && apierr.StatusCode != 500 { return false, 0, err } if apierr.StatusCode == 500 { logging.Warn("Copilot API returned 500 error, retrying", "error", err) } if attempts > maxRetries { return false, 0, fmt.Errorf("maximum retry attempts reached for rate limit: %d retries", maxRetries) } retryMs := 0 retryAfterValues := apierr.Response.Header.Values("Retry-After") backoffMs := 2000 * (1 << (attempts - 1)) jitterMs := int(float64(backoffMs) * 0.2) retryMs = backoffMs + jitterMs if len(retryAfterValues) > 0 { if _, err := fmt.Sscanf(retryAfterValues[0], "%d", &retryMs); err == nil { retryMs = retryMs * 1000 } } return true, int64(retryMs), nil } func (c *copilotClient) toolCalls(completion openai.ChatCompletion) []message.ToolCall { var toolCalls []message.ToolCall if len(completion.Choices) > 0 && len(completion.Choices[0].Message.ToolCalls) > 0 { for _, call := range completion.Choices[0].Message.ToolCalls { toolCall := message.ToolCall{ ID: call.ID, Name: call.Function.Name, Input: call.Function.Arguments, Type: "function", Finished: true, } toolCalls = append(toolCalls, toolCall) } } return toolCalls } func (c *copilotClient) usage(completion openai.ChatCompletion) TokenUsage { cachedTokens := completion.Usage.PromptTokensDetails.CachedTokens inputTokens := completion.Usage.PromptTokens - cachedTokens return TokenUsage{ InputTokens: inputTokens, OutputTokens: completion.Usage.CompletionTokens, CacheCreationTokens: 0, // GitHub Copilot doesn't provide this directly CacheReadTokens: cachedTokens, } } func WithCopilotReasoningEffort(effort string) CopilotOption { return func(options *copilotOptions) { defaultReasoningEffort := "medium" switch effort { case "low", "medium", "high": defaultReasoningEffort = effort default: logging.Warn("Invalid reasoning effort, using default: medium") } options.reasoningEffort = defaultReasoningEffort } } func WithCopilotExtraHeaders(headers map[string]string) CopilotOption { return func(options *copilotOptions) { options.extraHeaders = headers } } func WithCopilotBearerToken(bearerToken string) CopilotOption { return func(options *copilotOptions) { options.bearerToken = bearerToken } } ================================================ FILE: internal/llm/provider/gemini.go ================================================ package provider import ( "context" "encoding/json" "errors" "fmt" "io" "strings" "time" "github.com/google/uuid" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/llm/tools" "github.com/opencode-ai/opencode/internal/logging" "github.com/opencode-ai/opencode/internal/message" "google.golang.org/genai" ) type geminiOptions struct { disableCache bool } type GeminiOption func(*geminiOptions) type geminiClient struct { providerOptions providerClientOptions options geminiOptions client *genai.Client } type GeminiClient ProviderClient func newGeminiClient(opts providerClientOptions) GeminiClient { geminiOpts := geminiOptions{} for _, o := range opts.geminiOptions { o(&geminiOpts) } client, err := genai.NewClient(context.Background(), &genai.ClientConfig{APIKey: opts.apiKey, Backend: genai.BackendGeminiAPI}) if err != nil { logging.Error("Failed to create Gemini client", "error", err) return nil } return &geminiClient{ providerOptions: opts, options: geminiOpts, client: client, } } func (g *geminiClient) convertMessages(messages []message.Message) []*genai.Content { var history []*genai.Content for _, msg := range messages { switch msg.Role { case message.User: var parts []*genai.Part parts = append(parts, &genai.Part{Text: msg.Content().String()}) for _, binaryContent := range msg.BinaryContent() { imageFormat := strings.Split(binaryContent.MIMEType, "/") parts = append(parts, &genai.Part{InlineData: &genai.Blob{ MIMEType: imageFormat[1], Data: binaryContent.Data, }}) } history = append(history, &genai.Content{ Parts: parts, Role: "user", }) case message.Assistant: var assistantParts []*genai.Part if msg.Content().String() != "" { assistantParts = append(assistantParts, &genai.Part{Text: msg.Content().String()}) } if len(msg.ToolCalls()) > 0 { for _, call := range msg.ToolCalls() { args, _ := parseJsonToMap(call.Input) assistantParts = append(assistantParts, &genai.Part{ FunctionCall: &genai.FunctionCall{ Name: call.Name, Args: args, }, }) } } if len(assistantParts) > 0 { history = append(history, &genai.Content{ Role: "model", Parts: assistantParts, }) } case message.Tool: for _, result := range msg.ToolResults() { response := map[string]interface{}{"result": result.Content} parsed, err := parseJsonToMap(result.Content) if err == nil { response = parsed } var toolCall message.ToolCall for _, m := range messages { if m.Role == message.Assistant { for _, call := range m.ToolCalls() { if call.ID == result.ToolCallID { toolCall = call break } } } } history = append(history, &genai.Content{ Parts: []*genai.Part{ { FunctionResponse: &genai.FunctionResponse{ Name: toolCall.Name, Response: response, }, }, }, Role: "function", }) } } } return history } func (g *geminiClient) convertTools(tools []tools.BaseTool) []*genai.Tool { geminiTool := &genai.Tool{} geminiTool.FunctionDeclarations = make([]*genai.FunctionDeclaration, 0, len(tools)) for _, tool := range tools { info := tool.Info() declaration := &genai.FunctionDeclaration{ Name: info.Name, Description: info.Description, Parameters: &genai.Schema{ Type: genai.TypeObject, Properties: convertSchemaProperties(info.Parameters), Required: info.Required, }, } geminiTool.FunctionDeclarations = append(geminiTool.FunctionDeclarations, declaration) } return []*genai.Tool{geminiTool} } func (g *geminiClient) finishReason(reason genai.FinishReason) message.FinishReason { switch { case reason == genai.FinishReasonStop: return message.FinishReasonEndTurn case reason == genai.FinishReasonMaxTokens: return message.FinishReasonMaxTokens default: return message.FinishReasonUnknown } } func (g *geminiClient) send(ctx context.Context, messages []message.Message, tools []tools.BaseTool) (*ProviderResponse, error) { // Convert messages geminiMessages := g.convertMessages(messages) cfg := config.Get() if cfg.Debug { jsonData, _ := json.Marshal(geminiMessages) logging.Debug("Prepared messages", "messages", string(jsonData)) } history := geminiMessages[:len(geminiMessages)-1] // All but last message lastMsg := geminiMessages[len(geminiMessages)-1] config := &genai.GenerateContentConfig{ MaxOutputTokens: int32(g.providerOptions.maxTokens), SystemInstruction: &genai.Content{ Parts: []*genai.Part{{Text: g.providerOptions.systemMessage}}, }, } if len(tools) > 0 { config.Tools = g.convertTools(tools) } chat, _ := g.client.Chats.Create(ctx, g.providerOptions.model.APIModel, config, history) attempts := 0 for { attempts++ var toolCalls []message.ToolCall var lastMsgParts []genai.Part for _, part := range lastMsg.Parts { lastMsgParts = append(lastMsgParts, *part) } resp, err := chat.SendMessage(ctx, lastMsgParts...) // If there is an error we are going to see if we can retry the call if err != nil { retry, after, retryErr := g.shouldRetry(attempts, err) if retryErr != nil { return nil, retryErr } if retry { logging.WarnPersist(fmt.Sprintf("Retrying due to rate limit... attempt %d of %d", attempts, maxRetries), logging.PersistTimeArg, time.Millisecond*time.Duration(after+100)) select { case <-ctx.Done(): return nil, ctx.Err() case <-time.After(time.Duration(after) * time.Millisecond): continue } } return nil, retryErr } content := "" if len(resp.Candidates) > 0 && resp.Candidates[0].Content != nil { for _, part := range resp.Candidates[0].Content.Parts { switch { case part.Text != "": content = string(part.Text) case part.FunctionCall != nil: id := "call_" + uuid.New().String() args, _ := json.Marshal(part.FunctionCall.Args) toolCalls = append(toolCalls, message.ToolCall{ ID: id, Name: part.FunctionCall.Name, Input: string(args), Type: "function", Finished: true, }) } } } finishReason := message.FinishReasonEndTurn if len(resp.Candidates) > 0 { finishReason = g.finishReason(resp.Candidates[0].FinishReason) } if len(toolCalls) > 0 { finishReason = message.FinishReasonToolUse } return &ProviderResponse{ Content: content, ToolCalls: toolCalls, Usage: g.usage(resp), FinishReason: finishReason, }, nil } } func (g *geminiClient) stream(ctx context.Context, messages []message.Message, tools []tools.BaseTool) <-chan ProviderEvent { // Convert messages geminiMessages := g.convertMessages(messages) cfg := config.Get() if cfg.Debug { jsonData, _ := json.Marshal(geminiMessages) logging.Debug("Prepared messages", "messages", string(jsonData)) } history := geminiMessages[:len(geminiMessages)-1] // All but last message lastMsg := geminiMessages[len(geminiMessages)-1] config := &genai.GenerateContentConfig{ MaxOutputTokens: int32(g.providerOptions.maxTokens), SystemInstruction: &genai.Content{ Parts: []*genai.Part{{Text: g.providerOptions.systemMessage}}, }, } if len(tools) > 0 { config.Tools = g.convertTools(tools) } chat, _ := g.client.Chats.Create(ctx, g.providerOptions.model.APIModel, config, history) attempts := 0 eventChan := make(chan ProviderEvent) go func() { defer close(eventChan) for { attempts++ currentContent := "" toolCalls := []message.ToolCall{} var finalResp *genai.GenerateContentResponse eventChan <- ProviderEvent{Type: EventContentStart} var lastMsgParts []genai.Part for _, part := range lastMsg.Parts { lastMsgParts = append(lastMsgParts, *part) } for resp, err := range chat.SendMessageStream(ctx, lastMsgParts...) { if err != nil { retry, after, retryErr := g.shouldRetry(attempts, err) if retryErr != nil { eventChan <- ProviderEvent{Type: EventError, Error: retryErr} return } if retry { logging.WarnPersist(fmt.Sprintf("Retrying due to rate limit... attempt %d of %d", attempts, maxRetries), logging.PersistTimeArg, time.Millisecond*time.Duration(after+100)) select { case <-ctx.Done(): if ctx.Err() != nil { eventChan <- ProviderEvent{Type: EventError, Error: ctx.Err()} } return case <-time.After(time.Duration(after) * time.Millisecond): break } } else { eventChan <- ProviderEvent{Type: EventError, Error: err} return } } finalResp = resp if len(resp.Candidates) > 0 && resp.Candidates[0].Content != nil { for _, part := range resp.Candidates[0].Content.Parts { switch { case part.Text != "": delta := string(part.Text) if delta != "" { eventChan <- ProviderEvent{ Type: EventContentDelta, Content: delta, } currentContent += delta } case part.FunctionCall != nil: id := "call_" + uuid.New().String() args, _ := json.Marshal(part.FunctionCall.Args) newCall := message.ToolCall{ ID: id, Name: part.FunctionCall.Name, Input: string(args), Type: "function", Finished: true, } isNew := true for _, existing := range toolCalls { if existing.Name == newCall.Name && existing.Input == newCall.Input { isNew = false break } } if isNew { toolCalls = append(toolCalls, newCall) } } } } } eventChan <- ProviderEvent{Type: EventContentStop} if finalResp != nil { finishReason := message.FinishReasonEndTurn if len(finalResp.Candidates) > 0 { finishReason = g.finishReason(finalResp.Candidates[0].FinishReason) } if len(toolCalls) > 0 { finishReason = message.FinishReasonToolUse } eventChan <- ProviderEvent{ Type: EventComplete, Response: &ProviderResponse{ Content: currentContent, ToolCalls: toolCalls, Usage: g.usage(finalResp), FinishReason: finishReason, }, } return } } }() return eventChan } func (g *geminiClient) shouldRetry(attempts int, err error) (bool, int64, error) { // Check if error is a rate limit error if attempts > maxRetries { return false, 0, fmt.Errorf("maximum retry attempts reached for rate limit: %d retries", maxRetries) } // Gemini doesn't have a standard error type we can check against // So we'll check the error message for rate limit indicators if errors.Is(err, io.EOF) { return false, 0, err } errMsg := err.Error() isRateLimit := false // Check for common rate limit error messages if contains(errMsg, "rate limit", "quota exceeded", "too many requests") { isRateLimit = true } if !isRateLimit { return false, 0, err } // Calculate backoff with jitter backoffMs := 2000 * (1 << (attempts - 1)) jitterMs := int(float64(backoffMs) * 0.2) retryMs := backoffMs + jitterMs return true, int64(retryMs), nil } func (g *geminiClient) toolCalls(resp *genai.GenerateContentResponse) []message.ToolCall { var toolCalls []message.ToolCall if len(resp.Candidates) > 0 && resp.Candidates[0].Content != nil { for _, part := range resp.Candidates[0].Content.Parts { if part.FunctionCall != nil { id := "call_" + uuid.New().String() args, _ := json.Marshal(part.FunctionCall.Args) toolCalls = append(toolCalls, message.ToolCall{ ID: id, Name: part.FunctionCall.Name, Input: string(args), Type: "function", }) } } } return toolCalls } func (g *geminiClient) usage(resp *genai.GenerateContentResponse) TokenUsage { if resp == nil || resp.UsageMetadata == nil { return TokenUsage{} } return TokenUsage{ InputTokens: int64(resp.UsageMetadata.PromptTokenCount), OutputTokens: int64(resp.UsageMetadata.CandidatesTokenCount), CacheCreationTokens: 0, // Not directly provided by Gemini CacheReadTokens: int64(resp.UsageMetadata.CachedContentTokenCount), } } func WithGeminiDisableCache() GeminiOption { return func(options *geminiOptions) { options.disableCache = true } } // Helper functions func parseJsonToMap(jsonStr string) (map[string]interface{}, error) { var result map[string]interface{} err := json.Unmarshal([]byte(jsonStr), &result) return result, err } func convertSchemaProperties(parameters map[string]interface{}) map[string]*genai.Schema { properties := make(map[string]*genai.Schema) for name, param := range parameters { properties[name] = convertToSchema(param) } return properties } func convertToSchema(param interface{}) *genai.Schema { schema := &genai.Schema{Type: genai.TypeString} paramMap, ok := param.(map[string]interface{}) if !ok { return schema } if desc, ok := paramMap["description"].(string); ok { schema.Description = desc } typeVal, hasType := paramMap["type"] if !hasType { return schema } typeStr, ok := typeVal.(string) if !ok { return schema } schema.Type = mapJSONTypeToGenAI(typeStr) switch typeStr { case "array": schema.Items = processArrayItems(paramMap) case "object": if props, ok := paramMap["properties"].(map[string]interface{}); ok { schema.Properties = convertSchemaProperties(props) } } return schema } func processArrayItems(paramMap map[string]interface{}) *genai.Schema { items, ok := paramMap["items"].(map[string]interface{}) if !ok { return nil } return convertToSchema(items) } func mapJSONTypeToGenAI(jsonType string) genai.Type { switch jsonType { case "string": return genai.TypeString case "number": return genai.TypeNumber case "integer": return genai.TypeInteger case "boolean": return genai.TypeBoolean case "array": return genai.TypeArray case "object": return genai.TypeObject default: return genai.TypeString // Default to string for unknown types } } func contains(s string, substrs ...string) bool { for _, substr := range substrs { if strings.Contains(strings.ToLower(s), strings.ToLower(substr)) { return true } } return false } ================================================ FILE: internal/llm/provider/openai.go ================================================ package provider import ( "context" "encoding/json" "errors" "fmt" "io" "time" "github.com/openai/openai-go" "github.com/openai/openai-go/option" "github.com/openai/openai-go/shared" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/llm/models" "github.com/opencode-ai/opencode/internal/llm/tools" "github.com/opencode-ai/opencode/internal/logging" "github.com/opencode-ai/opencode/internal/message" ) type openaiOptions struct { baseURL string disableCache bool reasoningEffort string extraHeaders map[string]string } type OpenAIOption func(*openaiOptions) type openaiClient struct { providerOptions providerClientOptions options openaiOptions client openai.Client } type OpenAIClient ProviderClient func newOpenAIClient(opts providerClientOptions) OpenAIClient { openaiOpts := openaiOptions{ reasoningEffort: "medium", } for _, o := range opts.openaiOptions { o(&openaiOpts) } openaiClientOptions := []option.RequestOption{} if opts.apiKey != "" { openaiClientOptions = append(openaiClientOptions, option.WithAPIKey(opts.apiKey)) } if openaiOpts.baseURL != "" { openaiClientOptions = append(openaiClientOptions, option.WithBaseURL(openaiOpts.baseURL)) } if openaiOpts.extraHeaders != nil { for key, value := range openaiOpts.extraHeaders { openaiClientOptions = append(openaiClientOptions, option.WithHeader(key, value)) } } client := openai.NewClient(openaiClientOptions...) return &openaiClient{ providerOptions: opts, options: openaiOpts, client: client, } } func (o *openaiClient) convertMessages(messages []message.Message) (openaiMessages []openai.ChatCompletionMessageParamUnion) { // Add system message first openaiMessages = append(openaiMessages, openai.SystemMessage(o.providerOptions.systemMessage)) for _, msg := range messages { switch msg.Role { case message.User: var content []openai.ChatCompletionContentPartUnionParam textBlock := openai.ChatCompletionContentPartTextParam{Text: msg.Content().String()} content = append(content, openai.ChatCompletionContentPartUnionParam{OfText: &textBlock}) for _, binaryContent := range msg.BinaryContent() { imageURL := openai.ChatCompletionContentPartImageImageURLParam{URL: binaryContent.String(models.ProviderOpenAI)} imageBlock := openai.ChatCompletionContentPartImageParam{ImageURL: imageURL} content = append(content, openai.ChatCompletionContentPartUnionParam{OfImageURL: &imageBlock}) } openaiMessages = append(openaiMessages, openai.UserMessage(content)) case message.Assistant: assistantMsg := openai.ChatCompletionAssistantMessageParam{ Role: "assistant", } if msg.Content().String() != "" { assistantMsg.Content = openai.ChatCompletionAssistantMessageParamContentUnion{ OfString: openai.String(msg.Content().String()), } } if len(msg.ToolCalls()) > 0 { assistantMsg.ToolCalls = make([]openai.ChatCompletionMessageToolCallParam, len(msg.ToolCalls())) for i, call := range msg.ToolCalls() { assistantMsg.ToolCalls[i] = openai.ChatCompletionMessageToolCallParam{ ID: call.ID, Type: "function", Function: openai.ChatCompletionMessageToolCallFunctionParam{ Name: call.Name, Arguments: call.Input, }, } } } openaiMessages = append(openaiMessages, openai.ChatCompletionMessageParamUnion{ OfAssistant: &assistantMsg, }) case message.Tool: for _, result := range msg.ToolResults() { openaiMessages = append(openaiMessages, openai.ToolMessage(result.Content, result.ToolCallID), ) } } } return } func (o *openaiClient) convertTools(tools []tools.BaseTool) []openai.ChatCompletionToolParam { openaiTools := make([]openai.ChatCompletionToolParam, len(tools)) for i, tool := range tools { info := tool.Info() openaiTools[i] = openai.ChatCompletionToolParam{ Function: openai.FunctionDefinitionParam{ Name: info.Name, Description: openai.String(info.Description), Parameters: openai.FunctionParameters{ "type": "object", "properties": info.Parameters, "required": info.Required, }, }, } } return openaiTools } func (o *openaiClient) finishReason(reason string) message.FinishReason { switch reason { case "stop": return message.FinishReasonEndTurn case "length": return message.FinishReasonMaxTokens case "tool_calls": return message.FinishReasonToolUse default: return message.FinishReasonUnknown } } func (o *openaiClient) preparedParams(messages []openai.ChatCompletionMessageParamUnion, tools []openai.ChatCompletionToolParam) openai.ChatCompletionNewParams { params := openai.ChatCompletionNewParams{ Model: openai.ChatModel(o.providerOptions.model.APIModel), Messages: messages, Tools: tools, } if o.providerOptions.model.CanReason == true { params.MaxCompletionTokens = openai.Int(o.providerOptions.maxTokens) switch o.options.reasoningEffort { case "low": params.ReasoningEffort = shared.ReasoningEffortLow case "medium": params.ReasoningEffort = shared.ReasoningEffortMedium case "high": params.ReasoningEffort = shared.ReasoningEffortHigh default: params.ReasoningEffort = shared.ReasoningEffortMedium } } else { params.MaxTokens = openai.Int(o.providerOptions.maxTokens) } return params } func (o *openaiClient) send(ctx context.Context, messages []message.Message, tools []tools.BaseTool) (response *ProviderResponse, err error) { params := o.preparedParams(o.convertMessages(messages), o.convertTools(tools)) cfg := config.Get() if cfg.Debug { jsonData, _ := json.Marshal(params) logging.Debug("Prepared messages", "messages", string(jsonData)) } attempts := 0 for { attempts++ openaiResponse, err := o.client.Chat.Completions.New( ctx, params, ) // If there is an error we are going to see if we can retry the call if err != nil { retry, after, retryErr := o.shouldRetry(attempts, err) if retryErr != nil { return nil, retryErr } if retry { logging.WarnPersist(fmt.Sprintf("Retrying due to rate limit... attempt %d of %d", attempts, maxRetries), logging.PersistTimeArg, time.Millisecond*time.Duration(after+100)) select { case <-ctx.Done(): return nil, ctx.Err() case <-time.After(time.Duration(after) * time.Millisecond): continue } } return nil, retryErr } content := "" if openaiResponse.Choices[0].Message.Content != "" { content = openaiResponse.Choices[0].Message.Content } toolCalls := o.toolCalls(*openaiResponse) finishReason := o.finishReason(string(openaiResponse.Choices[0].FinishReason)) if len(toolCalls) > 0 { finishReason = message.FinishReasonToolUse } return &ProviderResponse{ Content: content, ToolCalls: toolCalls, Usage: o.usage(*openaiResponse), FinishReason: finishReason, }, nil } } func (o *openaiClient) stream(ctx context.Context, messages []message.Message, tools []tools.BaseTool) <-chan ProviderEvent { params := o.preparedParams(o.convertMessages(messages), o.convertTools(tools)) params.StreamOptions = openai.ChatCompletionStreamOptionsParam{ IncludeUsage: openai.Bool(true), } cfg := config.Get() if cfg.Debug { jsonData, _ := json.Marshal(params) logging.Debug("Prepared messages", "messages", string(jsonData)) } attempts := 0 eventChan := make(chan ProviderEvent) go func() { for { attempts++ openaiStream := o.client.Chat.Completions.NewStreaming( ctx, params, ) acc := openai.ChatCompletionAccumulator{} currentContent := "" toolCalls := make([]message.ToolCall, 0) for openaiStream.Next() { chunk := openaiStream.Current() acc.AddChunk(chunk) for _, choice := range chunk.Choices { if choice.Delta.Content != "" { eventChan <- ProviderEvent{ Type: EventContentDelta, Content: choice.Delta.Content, } currentContent += choice.Delta.Content } } } err := openaiStream.Err() if err == nil || errors.Is(err, io.EOF) { // Stream completed successfully finishReason := o.finishReason(string(acc.ChatCompletion.Choices[0].FinishReason)) if len(acc.ChatCompletion.Choices[0].Message.ToolCalls) > 0 { toolCalls = append(toolCalls, o.toolCalls(acc.ChatCompletion)...) } if len(toolCalls) > 0 { finishReason = message.FinishReasonToolUse } eventChan <- ProviderEvent{ Type: EventComplete, Response: &ProviderResponse{ Content: currentContent, ToolCalls: toolCalls, Usage: o.usage(acc.ChatCompletion), FinishReason: finishReason, }, } close(eventChan) return } // If there is an error we are going to see if we can retry the call retry, after, retryErr := o.shouldRetry(attempts, err) if retryErr != nil { eventChan <- ProviderEvent{Type: EventError, Error: retryErr} close(eventChan) return } if retry { logging.WarnPersist(fmt.Sprintf("Retrying due to rate limit... attempt %d of %d", attempts, maxRetries), logging.PersistTimeArg, time.Millisecond*time.Duration(after+100)) select { case <-ctx.Done(): // context cancelled if ctx.Err() == nil { eventChan <- ProviderEvent{Type: EventError, Error: ctx.Err()} } close(eventChan) return case <-time.After(time.Duration(after) * time.Millisecond): continue } } eventChan <- ProviderEvent{Type: EventError, Error: retryErr} close(eventChan) return } }() return eventChan } func (o *openaiClient) shouldRetry(attempts int, err error) (bool, int64, error) { var apierr *openai.Error if !errors.As(err, &apierr) { return false, 0, err } if apierr.StatusCode != 429 && apierr.StatusCode != 500 { return false, 0, err } if attempts > maxRetries { return false, 0, fmt.Errorf("maximum retry attempts reached for rate limit: %d retries", maxRetries) } retryMs := 0 retryAfterValues := apierr.Response.Header.Values("Retry-After") backoffMs := 2000 * (1 << (attempts - 1)) jitterMs := int(float64(backoffMs) * 0.2) retryMs = backoffMs + jitterMs if len(retryAfterValues) > 0 { if _, err := fmt.Sscanf(retryAfterValues[0], "%d", &retryMs); err == nil { retryMs = retryMs * 1000 } } return true, int64(retryMs), nil } func (o *openaiClient) toolCalls(completion openai.ChatCompletion) []message.ToolCall { var toolCalls []message.ToolCall if len(completion.Choices) > 0 && len(completion.Choices[0].Message.ToolCalls) > 0 { for _, call := range completion.Choices[0].Message.ToolCalls { toolCall := message.ToolCall{ ID: call.ID, Name: call.Function.Name, Input: call.Function.Arguments, Type: "function", Finished: true, } toolCalls = append(toolCalls, toolCall) } } return toolCalls } func (o *openaiClient) usage(completion openai.ChatCompletion) TokenUsage { cachedTokens := completion.Usage.PromptTokensDetails.CachedTokens inputTokens := completion.Usage.PromptTokens - cachedTokens return TokenUsage{ InputTokens: inputTokens, OutputTokens: completion.Usage.CompletionTokens, CacheCreationTokens: 0, // OpenAI doesn't provide this directly CacheReadTokens: cachedTokens, } } func WithOpenAIBaseURL(baseURL string) OpenAIOption { return func(options *openaiOptions) { options.baseURL = baseURL } } func WithOpenAIExtraHeaders(headers map[string]string) OpenAIOption { return func(options *openaiOptions) { options.extraHeaders = headers } } func WithOpenAIDisableCache() OpenAIOption { return func(options *openaiOptions) { options.disableCache = true } } func WithReasoningEffort(effort string) OpenAIOption { return func(options *openaiOptions) { defaultReasoningEffort := "medium" switch effort { case "low", "medium", "high": defaultReasoningEffort = effort default: logging.Warn("Invalid reasoning effort, using default: medium") } options.reasoningEffort = defaultReasoningEffort } } ================================================ FILE: internal/llm/provider/provider.go ================================================ package provider import ( "context" "fmt" "os" "github.com/opencode-ai/opencode/internal/llm/models" "github.com/opencode-ai/opencode/internal/llm/tools" "github.com/opencode-ai/opencode/internal/message" ) type EventType string const maxRetries = 8 const ( EventContentStart EventType = "content_start" EventToolUseStart EventType = "tool_use_start" EventToolUseDelta EventType = "tool_use_delta" EventToolUseStop EventType = "tool_use_stop" EventContentDelta EventType = "content_delta" EventThinkingDelta EventType = "thinking_delta" EventContentStop EventType = "content_stop" EventComplete EventType = "complete" EventError EventType = "error" EventWarning EventType = "warning" ) type TokenUsage struct { InputTokens int64 OutputTokens int64 CacheCreationTokens int64 CacheReadTokens int64 } type ProviderResponse struct { Content string ToolCalls []message.ToolCall Usage TokenUsage FinishReason message.FinishReason } type ProviderEvent struct { Type EventType Content string Thinking string Response *ProviderResponse ToolCall *message.ToolCall Error error } type Provider interface { SendMessages(ctx context.Context, messages []message.Message, tools []tools.BaseTool) (*ProviderResponse, error) StreamResponse(ctx context.Context, messages []message.Message, tools []tools.BaseTool) <-chan ProviderEvent Model() models.Model } type providerClientOptions struct { apiKey string model models.Model maxTokens int64 systemMessage string anthropicOptions []AnthropicOption openaiOptions []OpenAIOption geminiOptions []GeminiOption bedrockOptions []BedrockOption copilotOptions []CopilotOption } type ProviderClientOption func(*providerClientOptions) type ProviderClient interface { send(ctx context.Context, messages []message.Message, tools []tools.BaseTool) (*ProviderResponse, error) stream(ctx context.Context, messages []message.Message, tools []tools.BaseTool) <-chan ProviderEvent } type baseProvider[C ProviderClient] struct { options providerClientOptions client C } func NewProvider(providerName models.ModelProvider, opts ...ProviderClientOption) (Provider, error) { clientOptions := providerClientOptions{} for _, o := range opts { o(&clientOptions) } switch providerName { case models.ProviderCopilot: return &baseProvider[CopilotClient]{ options: clientOptions, client: newCopilotClient(clientOptions), }, nil case models.ProviderAnthropic: return &baseProvider[AnthropicClient]{ options: clientOptions, client: newAnthropicClient(clientOptions), }, nil case models.ProviderOpenAI: return &baseProvider[OpenAIClient]{ options: clientOptions, client: newOpenAIClient(clientOptions), }, nil case models.ProviderGemini: return &baseProvider[GeminiClient]{ options: clientOptions, client: newGeminiClient(clientOptions), }, nil case models.ProviderBedrock: return &baseProvider[BedrockClient]{ options: clientOptions, client: newBedrockClient(clientOptions), }, nil case models.ProviderGROQ: clientOptions.openaiOptions = append(clientOptions.openaiOptions, WithOpenAIBaseURL("https://api.groq.com/openai/v1"), ) return &baseProvider[OpenAIClient]{ options: clientOptions, client: newOpenAIClient(clientOptions), }, nil case models.ProviderAzure: return &baseProvider[AzureClient]{ options: clientOptions, client: newAzureClient(clientOptions), }, nil case models.ProviderVertexAI: return &baseProvider[VertexAIClient]{ options: clientOptions, client: newVertexAIClient(clientOptions), }, nil case models.ProviderOpenRouter: clientOptions.openaiOptions = append(clientOptions.openaiOptions, WithOpenAIBaseURL("https://openrouter.ai/api/v1"), WithOpenAIExtraHeaders(map[string]string{ "HTTP-Referer": "opencode.ai", "X-Title": "OpenCode", }), ) return &baseProvider[OpenAIClient]{ options: clientOptions, client: newOpenAIClient(clientOptions), }, nil case models.ProviderXAI: clientOptions.openaiOptions = append(clientOptions.openaiOptions, WithOpenAIBaseURL("https://api.x.ai/v1"), ) return &baseProvider[OpenAIClient]{ options: clientOptions, client: newOpenAIClient(clientOptions), }, nil case models.ProviderLocal: clientOptions.openaiOptions = append(clientOptions.openaiOptions, WithOpenAIBaseURL(os.Getenv("LOCAL_ENDPOINT")), ) return &baseProvider[OpenAIClient]{ options: clientOptions, client: newOpenAIClient(clientOptions), }, nil case models.ProviderMock: // TODO: implement mock client for test panic("not implemented") } return nil, fmt.Errorf("provider not supported: %s", providerName) } func (p *baseProvider[C]) cleanMessages(messages []message.Message) (cleaned []message.Message) { for _, msg := range messages { // The message has no content if len(msg.Parts) == 0 { continue } cleaned = append(cleaned, msg) } return } func (p *baseProvider[C]) SendMessages(ctx context.Context, messages []message.Message, tools []tools.BaseTool) (*ProviderResponse, error) { messages = p.cleanMessages(messages) return p.client.send(ctx, messages, tools) } func (p *baseProvider[C]) Model() models.Model { return p.options.model } func (p *baseProvider[C]) StreamResponse(ctx context.Context, messages []message.Message, tools []tools.BaseTool) <-chan ProviderEvent { messages = p.cleanMessages(messages) return p.client.stream(ctx, messages, tools) } func WithAPIKey(apiKey string) ProviderClientOption { return func(options *providerClientOptions) { options.apiKey = apiKey } } func WithModel(model models.Model) ProviderClientOption { return func(options *providerClientOptions) { options.model = model } } func WithMaxTokens(maxTokens int64) ProviderClientOption { return func(options *providerClientOptions) { options.maxTokens = maxTokens } } func WithSystemMessage(systemMessage string) ProviderClientOption { return func(options *providerClientOptions) { options.systemMessage = systemMessage } } func WithAnthropicOptions(anthropicOptions ...AnthropicOption) ProviderClientOption { return func(options *providerClientOptions) { options.anthropicOptions = anthropicOptions } } func WithOpenAIOptions(openaiOptions ...OpenAIOption) ProviderClientOption { return func(options *providerClientOptions) { options.openaiOptions = openaiOptions } } func WithGeminiOptions(geminiOptions ...GeminiOption) ProviderClientOption { return func(options *providerClientOptions) { options.geminiOptions = geminiOptions } } func WithBedrockOptions(bedrockOptions ...BedrockOption) ProviderClientOption { return func(options *providerClientOptions) { options.bedrockOptions = bedrockOptions } } func WithCopilotOptions(copilotOptions ...CopilotOption) ProviderClientOption { return func(options *providerClientOptions) { options.copilotOptions = copilotOptions } } ================================================ FILE: internal/llm/provider/vertexai.go ================================================ package provider import ( "context" "os" "github.com/opencode-ai/opencode/internal/logging" "google.golang.org/genai" ) type VertexAIClient ProviderClient func newVertexAIClient(opts providerClientOptions) VertexAIClient { geminiOpts := geminiOptions{} for _, o := range opts.geminiOptions { o(&geminiOpts) } client, err := genai.NewClient(context.Background(), &genai.ClientConfig{ Project: os.Getenv("VERTEXAI_PROJECT"), Location: os.Getenv("VERTEXAI_LOCATION"), Backend: genai.BackendVertexAI, }) if err != nil { logging.Error("Failed to create VertexAI client", "error", err) return nil } return &geminiClient{ providerOptions: opts, options: geminiOpts, client: client, } } ================================================ FILE: internal/llm/tools/bash.go ================================================ package tools import ( "context" "encoding/json" "fmt" "strings" "time" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/llm/tools/shell" "github.com/opencode-ai/opencode/internal/permission" ) type BashParams struct { Command string `json:"command"` Timeout int `json:"timeout"` } type BashPermissionsParams struct { Command string `json:"command"` Timeout int `json:"timeout"` } type BashResponseMetadata struct { StartTime int64 `json:"start_time"` EndTime int64 `json:"end_time"` } type bashTool struct { permissions permission.Service } const ( BashToolName = "bash" DefaultTimeout = 1 * 60 * 1000 // 1 minutes in milliseconds MaxTimeout = 10 * 60 * 1000 // 10 minutes in milliseconds MaxOutputLength = 30000 ) var bannedCommands = []string{ "alias", "curl", "curlie", "wget", "axel", "aria2c", "nc", "telnet", "lynx", "w3m", "links", "httpie", "xh", "http-prompt", "chrome", "firefox", "safari", } var safeReadOnlyCommands = []string{ "ls", "echo", "pwd", "date", "cal", "uptime", "whoami", "id", "groups", "env", "printenv", "set", "unset", "which", "type", "whereis", "whatis", "uname", "hostname", "df", "du", "free", "top", "ps", "kill", "killall", "nice", "nohup", "time", "timeout", "git status", "git log", "git diff", "git show", "git branch", "git tag", "git remote", "git ls-files", "git ls-remote", "git rev-parse", "git config --get", "git config --list", "git describe", "git blame", "git grep", "git shortlog", "go version", "go help", "go list", "go env", "go doc", "go vet", "go fmt", "go mod", "go test", "go build", "go run", "go install", "go clean", } func bashDescription() string { bannedCommandsStr := strings.Join(bannedCommands, ", ") return fmt.Sprintf(`Executes a given bash command in a persistent shell session with optional timeout, ensuring proper handling and security measures. Before executing the command, please follow these steps: 1. Directory Verification: - If the command will create new directories or files, first use the LS tool to verify the parent directory exists and is the correct location - For example, before running "mkdir foo/bar", first use LS to check that "foo" exists and is the intended parent directory 2. Security Check: - For security and to limit the threat of a prompt injection attack, some commands are limited or banned. If you use a disallowed command, you will receive an error message explaining the restriction. Explain the error to the User. - Verify that the command is not one of the banned commands: %s. 3. Command Execution: - After ensuring proper quoting, execute the command. - Capture the output of the command. 4. Output Processing: - If the output exceeds %d characters, output will be truncated before being returned to you. - Prepare the output for display to the user. 5. Return Result: - Provide the processed output of the command. - If any errors occurred during execution, include those in the output. Usage notes: - The command argument is required. - You can specify an optional timeout in milliseconds (up to 600000ms / 10 minutes). If not specified, commands will timeout after 30 minutes. - VERY IMPORTANT: You MUST avoid using search commands like 'find' and 'grep'. Instead use Grep, Glob, or Agent tools to search. You MUST avoid read tools like 'cat', 'head', 'tail', and 'ls', and use FileRead and LS tools to read files. - When issuing multiple commands, use the ';' or '&&' operator to separate them. DO NOT use newlines (newlines are ok in quoted strings). - IMPORTANT: All commands share the same shell session. Shell state (environment variables, virtual environments, current directory, etc.) persist between commands. For example, if you set an environment variable as part of a command, the environment variable will persist for subsequent commands. - Try to maintain your current working directory throughout the session by using absolute paths and avoiding usage of 'cd'. You may use 'cd' if the User explicitly requests it. pytest /foo/bar/tests cd /foo/bar && pytest tests # Committing changes with git When the user asks you to create a new git commit, follow these steps carefully: 1. Start with a single message that contains exactly three tool_use blocks that do the following (it is VERY IMPORTANT that you send these tool_use blocks in a single message, otherwise it will feel slow to the user!): - Run a git status command to see all untracked files. - Run a git diff command to see both staged and unstaged changes that will be committed. - Run a git log command to see recent commit messages, so that you can follow this repository's commit message style. 2. Use the git context at the start of this conversation to determine which files are relevant to your commit. Add relevant untracked files to the staging area. Do not commit files that were already modified at the start of this conversation, if they are not relevant to your commit. 3. Analyze all staged changes (both previously staged and newly added) and draft a commit message. Wrap your analysis process in tags: - List the files that have been changed or added - Summarize the nature of the changes (eg. new feature, enhancement to an existing feature, bug fix, refactoring, test, docs, etc.) - Brainstorm the purpose or motivation behind these changes - Do not use tools to explore code, beyond what is available in the git context - Assess the impact of these changes on the overall project - Check for any sensitive information that shouldn't be committed - Draft a concise (1-2 sentences) commit message that focuses on the "why" rather than the "what" - Ensure your language is clear, concise, and to the point - Ensure the message accurately reflects the changes and their purpose (i.e. "add" means a wholly new feature, "update" means an enhancement to an existing feature, "fix" means a bug fix, etc.) - Ensure the message is not generic (avoid words like "Update" or "Fix" without context) - Review the draft message to ensure it accurately reflects the changes and their purpose 4. Create the commit with a message ending with: 🤖 Generated with opencode Co-Authored-By: opencode - In order to ensure good formatting, ALWAYS pass the commit message via a HEREDOC, a la this example: git commit -m "$(cat <<'EOF' Commit message here. 🤖 Generated with opencode Co-Authored-By: opencode EOF )" 5. If the commit fails due to pre-commit hook changes, retry the commit ONCE to include these automated changes. If it fails again, it usually means a pre-commit hook is preventing the commit. If the commit succeeds but you notice that files were modified by the pre-commit hook, you MUST amend your commit to include them. 6. Finally, run git status to make sure the commit succeeded. Important notes: - When possible, combine the "git add" and "git commit" commands into a single "git commit -am" command, to speed things up - However, be careful not to stage files (e.g. with 'git add .') for commits that aren't part of the change, they may have untracked files they want to keep around, but not commit. - NEVER update the git config - DO NOT push to the remote repository - IMPORTANT: Never use git commands with the -i flag (like git rebase -i or git add -i) since they require interactive input which is not supported. - If there are no changes to commit (i.e., no untracked files and no modifications), do not create an empty commit - Ensure your commit message is meaningful and concise. It should explain the purpose of the changes, not just describe them. - Return an empty response - the user will see the git output directly # Creating pull requests Use the gh command via the Bash tool for ALL GitHub-related tasks including working with issues, pull requests, checks, and releases. If given a Github URL use the gh command to get the information needed. IMPORTANT: When the user asks you to create a pull request, follow these steps carefully: 1. Understand the current state of the branch. Remember to send a single message that contains multiple tool_use blocks (it is VERY IMPORTANT that you do this in a single message, otherwise it will feel slow to the user!): - Run a git status command to see all untracked files. - Run a git diff command to see both staged and unstaged changes that will be committed. - Check if the current branch tracks a remote branch and is up to date with the remote, so you know if you need to push to the remote - Run a git log command and 'git diff main...HEAD' to understand the full commit history for the current branch (from the time it diverged from the 'main' branch.) 2. Create new branch if needed 3. Commit changes if needed 4. Push to remote with -u flag if needed 5. Analyze all changes that will be included in the pull request, making sure to look at all relevant commits (not just the latest commit, but all commits that will be included in the pull request!), and draft a pull request summary. Wrap your analysis process in tags: - List the commits since diverging from the main branch - Summarize the nature of the changes (eg. new feature, enhancement to an existing feature, bug fix, refactoring, test, docs, etc.) - Brainstorm the purpose or motivation behind these changes - Assess the impact of these changes on the overall project - Do not use tools to explore code, beyond what is available in the git context - Check for any sensitive information that shouldn't be committed - Draft a concise (1-2 bullet points) pull request summary that focuses on the "why" rather than the "what" - Ensure the summary accurately reflects all changes since diverging from the main branch - Ensure your language is clear, concise, and to the point - Ensure the summary accurately reflects the changes and their purpose (ie. "add" means a wholly new feature, "update" means an enhancement to an existing feature, "fix" means a bug fix, etc.) - Ensure the summary is not generic (avoid words like "Update" or "Fix" without context) - Review the draft summary to ensure it accurately reflects the changes and their purpose 6. Create PR using gh pr create with the format below. Use a HEREDOC to pass the body to ensure correct formatting. gh pr create --title "the pr title" --body "$(cat <<'EOF' ## Summary <1-3 bullet points> ## Test plan [Checklist of TODOs for testing the pull request...] 🤖 Generated with opencode EOF )" Important: - Return an empty response - the user will see the gh output directly - Never update git config`, bannedCommandsStr, MaxOutputLength) } func NewBashTool(permission permission.Service) BaseTool { return &bashTool{ permissions: permission, } } func (b *bashTool) Info() ToolInfo { return ToolInfo{ Name: BashToolName, Description: bashDescription(), Parameters: map[string]any{ "command": map[string]any{ "type": "string", "description": "The command to execute", }, "timeout": map[string]any{ "type": "number", "description": "Optional timeout in milliseconds (max 600000)", }, }, Required: []string{"command"}, } } func (b *bashTool) Run(ctx context.Context, call ToolCall) (ToolResponse, error) { var params BashParams if err := json.Unmarshal([]byte(call.Input), ¶ms); err != nil { return NewTextErrorResponse("invalid parameters"), nil } if params.Timeout > MaxTimeout { params.Timeout = MaxTimeout } else if params.Timeout <= 0 { params.Timeout = DefaultTimeout } if params.Command == "" { return NewTextErrorResponse("missing command"), nil } baseCmd := strings.Fields(params.Command)[0] for _, banned := range bannedCommands { if strings.EqualFold(baseCmd, banned) { return NewTextErrorResponse(fmt.Sprintf("command '%s' is not allowed", baseCmd)), nil } } isSafeReadOnly := false cmdLower := strings.ToLower(params.Command) for _, safe := range safeReadOnlyCommands { if strings.HasPrefix(cmdLower, strings.ToLower(safe)) { if len(cmdLower) == len(safe) || cmdLower[len(safe)] == ' ' || cmdLower[len(safe)] == '-' { isSafeReadOnly = true break } } } sessionID, messageID := GetContextValues(ctx) if sessionID == "" || messageID == "" { return ToolResponse{}, fmt.Errorf("session ID and message ID are required for creating a new file") } if !isSafeReadOnly { p := b.permissions.Request( permission.CreatePermissionRequest{ SessionID: sessionID, Path: config.WorkingDirectory(), ToolName: BashToolName, Action: "execute", Description: fmt.Sprintf("Execute command: %s", params.Command), Params: BashPermissionsParams{ Command: params.Command, }, }, ) if !p { return ToolResponse{}, permission.ErrorPermissionDenied } } startTime := time.Now() shell := shell.GetPersistentShell(config.WorkingDirectory()) stdout, stderr, exitCode, interrupted, err := shell.Exec(ctx, params.Command, params.Timeout) if err != nil { return ToolResponse{}, fmt.Errorf("error executing command: %w", err) } stdout = truncateOutput(stdout) stderr = truncateOutput(stderr) errorMessage := stderr if interrupted { if errorMessage != "" { errorMessage += "\n" } errorMessage += "Command was aborted before completion" } else if exitCode != 0 { if errorMessage != "" { errorMessage += "\n" } errorMessage += fmt.Sprintf("Exit code %d", exitCode) } hasBothOutputs := stdout != "" && stderr != "" if hasBothOutputs { stdout += "\n" } if errorMessage != "" { stdout += "\n" + errorMessage } metadata := BashResponseMetadata{ StartTime: startTime.UnixMilli(), EndTime: time.Now().UnixMilli(), } if stdout == "" { return WithResponseMetadata(NewTextResponse("no output"), metadata), nil } return WithResponseMetadata(NewTextResponse(stdout), metadata), nil } func truncateOutput(content string) string { if len(content) <= MaxOutputLength { return content } halfLength := MaxOutputLength / 2 start := content[:halfLength] end := content[len(content)-halfLength:] truncatedLinesCount := countLines(content[halfLength : len(content)-halfLength]) return fmt.Sprintf("%s\n\n... [%d lines truncated] ...\n\n%s", start, truncatedLinesCount, end) } func countLines(s string) int { if s == "" { return 0 } return len(strings.Split(s, "\n")) } ================================================ FILE: internal/llm/tools/diagnostics.go ================================================ package tools import ( "context" "encoding/json" "fmt" "maps" "sort" "strings" "time" "github.com/opencode-ai/opencode/internal/lsp" "github.com/opencode-ai/opencode/internal/lsp/protocol" ) type DiagnosticsParams struct { FilePath string `json:"file_path"` } type diagnosticsTool struct { lspClients map[string]*lsp.Client } const ( DiagnosticsToolName = "diagnostics" diagnosticsDescription = `Get diagnostics for a file and/or project. WHEN TO USE THIS TOOL: - Use when you need to check for errors or warnings in your code - Helpful for debugging and ensuring code quality - Good for getting a quick overview of issues in a file or project HOW TO USE: - Provide a path to a file to get diagnostics for that file - Leave the path empty to get diagnostics for the entire project - Results are displayed in a structured format with severity levels FEATURES: - Displays errors, warnings, and hints - Groups diagnostics by severity - Provides detailed information about each diagnostic LIMITATIONS: - Results are limited to the diagnostics provided by the LSP clients - May not cover all possible issues in the code - Does not provide suggestions for fixing issues TIPS: - Use in conjunction with other tools for a comprehensive code review - Combine with the LSP client for real-time diagnostics ` ) func NewDiagnosticsTool(lspClients map[string]*lsp.Client) BaseTool { return &diagnosticsTool{ lspClients, } } func (b *diagnosticsTool) Info() ToolInfo { return ToolInfo{ Name: DiagnosticsToolName, Description: diagnosticsDescription, Parameters: map[string]any{ "file_path": map[string]any{ "type": "string", "description": "The path to the file to get diagnostics for (leave w empty for project diagnostics)", }, }, Required: []string{}, } } func (b *diagnosticsTool) Run(ctx context.Context, call ToolCall) (ToolResponse, error) { var params DiagnosticsParams if err := json.Unmarshal([]byte(call.Input), ¶ms); err != nil { return NewTextErrorResponse(fmt.Sprintf("error parsing parameters: %s", err)), nil } lsps := b.lspClients if len(lsps) == 0 { return NewTextErrorResponse("no LSP clients available"), nil } if params.FilePath != "" { notifyLspOpenFile(ctx, params.FilePath, lsps) waitForLspDiagnostics(ctx, params.FilePath, lsps) } output := getDiagnostics(params.FilePath, lsps) return NewTextResponse(output), nil } func notifyLspOpenFile(ctx context.Context, filePath string, lsps map[string]*lsp.Client) { for _, client := range lsps { err := client.OpenFile(ctx, filePath) if err != nil { continue } } } func waitForLspDiagnostics(ctx context.Context, filePath string, lsps map[string]*lsp.Client) { if len(lsps) == 0 { return } diagChan := make(chan struct{}, 1) for _, client := range lsps { originalDiags := make(map[protocol.DocumentUri][]protocol.Diagnostic) maps.Copy(originalDiags, client.GetDiagnostics()) handler := func(params json.RawMessage) { lsp.HandleDiagnostics(client, params) var diagParams protocol.PublishDiagnosticsParams if err := json.Unmarshal(params, &diagParams); err != nil { return } if diagParams.URI.Path() == filePath || hasDiagnosticsChanged(client.GetDiagnostics(), originalDiags) { select { case diagChan <- struct{}{}: default: } } } client.RegisterNotificationHandler("textDocument/publishDiagnostics", handler) if client.IsFileOpen(filePath) { err := client.NotifyChange(ctx, filePath) if err != nil { continue } } else { err := client.OpenFile(ctx, filePath) if err != nil { continue } } } select { case <-diagChan: case <-time.After(5 * time.Second): case <-ctx.Done(): } } func hasDiagnosticsChanged(current, original map[protocol.DocumentUri][]protocol.Diagnostic) bool { for uri, diags := range current { origDiags, exists := original[uri] if !exists || len(diags) != len(origDiags) { return true } } return false } func getDiagnostics(filePath string, lsps map[string]*lsp.Client) string { fileDiagnostics := []string{} projectDiagnostics := []string{} formatDiagnostic := func(pth string, diagnostic protocol.Diagnostic, source string) string { severity := "Info" switch diagnostic.Severity { case protocol.SeverityError: severity = "Error" case protocol.SeverityWarning: severity = "Warn" case protocol.SeverityHint: severity = "Hint" } location := fmt.Sprintf("%s:%d:%d", pth, diagnostic.Range.Start.Line+1, diagnostic.Range.Start.Character+1) sourceInfo := "" if diagnostic.Source != "" { sourceInfo = diagnostic.Source } else if source != "" { sourceInfo = source } codeInfo := "" if diagnostic.Code != nil { codeInfo = fmt.Sprintf("[%v]", diagnostic.Code) } tagsInfo := "" if len(diagnostic.Tags) > 0 { tags := []string{} for _, tag := range diagnostic.Tags { switch tag { case protocol.Unnecessary: tags = append(tags, "unnecessary") case protocol.Deprecated: tags = append(tags, "deprecated") } } if len(tags) > 0 { tagsInfo = fmt.Sprintf(" (%s)", strings.Join(tags, ", ")) } } return fmt.Sprintf("%s: %s [%s]%s%s %s", severity, location, sourceInfo, codeInfo, tagsInfo, diagnostic.Message) } for lspName, client := range lsps { diagnostics := client.GetDiagnostics() if len(diagnostics) > 0 { for location, diags := range diagnostics { isCurrentFile := location.Path() == filePath for _, diag := range diags { formattedDiag := formatDiagnostic(location.Path(), diag, lspName) if isCurrentFile { fileDiagnostics = append(fileDiagnostics, formattedDiag) } else { projectDiagnostics = append(projectDiagnostics, formattedDiag) } } } } } sort.Slice(fileDiagnostics, func(i, j int) bool { iIsError := strings.HasPrefix(fileDiagnostics[i], "Error") jIsError := strings.HasPrefix(fileDiagnostics[j], "Error") if iIsError != jIsError { return iIsError // Errors come first } return fileDiagnostics[i] < fileDiagnostics[j] // Then alphabetically }) sort.Slice(projectDiagnostics, func(i, j int) bool { iIsError := strings.HasPrefix(projectDiagnostics[i], "Error") jIsError := strings.HasPrefix(projectDiagnostics[j], "Error") if iIsError != jIsError { return iIsError } return projectDiagnostics[i] < projectDiagnostics[j] }) output := "" if len(fileDiagnostics) > 0 { output += "\n\n" if len(fileDiagnostics) > 10 { output += strings.Join(fileDiagnostics[:10], "\n") output += fmt.Sprintf("\n... and %d more diagnostics", len(fileDiagnostics)-10) } else { output += strings.Join(fileDiagnostics, "\n") } output += "\n\n" } if len(projectDiagnostics) > 0 { output += "\n\n" if len(projectDiagnostics) > 10 { output += strings.Join(projectDiagnostics[:10], "\n") output += fmt.Sprintf("\n... and %d more diagnostics", len(projectDiagnostics)-10) } else { output += strings.Join(projectDiagnostics, "\n") } output += "\n\n" } if len(fileDiagnostics) > 0 || len(projectDiagnostics) > 0 { fileErrors := countSeverity(fileDiagnostics, "Error") fileWarnings := countSeverity(fileDiagnostics, "Warn") projectErrors := countSeverity(projectDiagnostics, "Error") projectWarnings := countSeverity(projectDiagnostics, "Warn") output += "\n\n" output += fmt.Sprintf("Current file: %d errors, %d warnings\n", fileErrors, fileWarnings) output += fmt.Sprintf("Project: %d errors, %d warnings\n", projectErrors, projectWarnings) output += "\n" } return output } func countSeverity(diagnostics []string, severity string) int { count := 0 for _, diag := range diagnostics { if strings.HasPrefix(diag, severity) { count++ } } return count } ================================================ FILE: internal/llm/tools/edit.go ================================================ package tools import ( "context" "encoding/json" "fmt" "os" "path/filepath" "strings" "time" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/diff" "github.com/opencode-ai/opencode/internal/history" "github.com/opencode-ai/opencode/internal/logging" "github.com/opencode-ai/opencode/internal/lsp" "github.com/opencode-ai/opencode/internal/permission" ) type EditParams struct { FilePath string `json:"file_path"` OldString string `json:"old_string"` NewString string `json:"new_string"` } type EditPermissionsParams struct { FilePath string `json:"file_path"` Diff string `json:"diff"` } type EditResponseMetadata struct { Diff string `json:"diff"` Additions int `json:"additions"` Removals int `json:"removals"` } type editTool struct { lspClients map[string]*lsp.Client permissions permission.Service files history.Service } const ( EditToolName = "edit" editDescription = `Edits files by replacing text, creating new files, or deleting content. For moving or renaming files, use the Bash tool with the 'mv' command instead. For larger file edits, use the FileWrite tool to overwrite files. Before using this tool: 1. Use the FileRead tool to understand the file's contents and context 2. Verify the directory path is correct (only applicable when creating new files): - Use the LS tool to verify the parent directory exists and is the correct location To make a file edit, provide the following: 1. file_path: The absolute path to the file to modify (must be absolute, not relative) 2. old_string: The text to replace (must be unique within the file, and must match the file contents exactly, including all whitespace and indentation) 3. new_string: The edited text to replace the old_string Special cases: - To create a new file: provide file_path and new_string, leave old_string empty - To delete content: provide file_path and old_string, leave new_string empty The tool will replace ONE occurrence of old_string with new_string in the specified file. CRITICAL REQUIREMENTS FOR USING THIS TOOL: 1. UNIQUENESS: The old_string MUST uniquely identify the specific instance you want to change. This means: - Include AT LEAST 3-5 lines of context BEFORE the change point - Include AT LEAST 3-5 lines of context AFTER the change point - Include all whitespace, indentation, and surrounding code exactly as it appears in the file 2. SINGLE INSTANCE: This tool can only change ONE instance at a time. If you need to change multiple instances: - Make separate calls to this tool for each instance - Each call must uniquely identify its specific instance using extensive context 3. VERIFICATION: Before using this tool: - Check how many instances of the target text exist in the file - If multiple instances exist, gather enough context to uniquely identify each one - Plan separate tool calls for each instance WARNING: If you do not follow these requirements: - The tool will fail if old_string matches multiple locations - The tool will fail if old_string doesn't match exactly (including whitespace) - You may change the wrong instance if you don't include enough context When making edits: - Ensure the edit results in idiomatic, correct code - Do not leave the code in a broken state - Always use absolute file paths (starting with /) Remember: when making multiple file edits in a row to the same file, you should prefer to send all edits in a single message with multiple calls to this tool, rather than multiple messages with a single call each.` ) func NewEditTool(lspClients map[string]*lsp.Client, permissions permission.Service, files history.Service) BaseTool { return &editTool{ lspClients: lspClients, permissions: permissions, files: files, } } func (e *editTool) Info() ToolInfo { return ToolInfo{ Name: EditToolName, Description: editDescription, Parameters: map[string]any{ "file_path": map[string]any{ "type": "string", "description": "The absolute path to the file to modify", }, "old_string": map[string]any{ "type": "string", "description": "The text to replace", }, "new_string": map[string]any{ "type": "string", "description": "The text to replace it with", }, }, Required: []string{"file_path", "old_string", "new_string"}, } } func (e *editTool) Run(ctx context.Context, call ToolCall) (ToolResponse, error) { var params EditParams if err := json.Unmarshal([]byte(call.Input), ¶ms); err != nil { return NewTextErrorResponse("invalid parameters"), nil } if params.FilePath == "" { return NewTextErrorResponse("file_path is required"), nil } if !filepath.IsAbs(params.FilePath) { wd := config.WorkingDirectory() params.FilePath = filepath.Join(wd, params.FilePath) } var response ToolResponse var err error if params.OldString == "" { response, err = e.createNewFile(ctx, params.FilePath, params.NewString) if err != nil { return response, err } } if params.NewString == "" { response, err = e.deleteContent(ctx, params.FilePath, params.OldString) if err != nil { return response, err } } response, err = e.replaceContent(ctx, params.FilePath, params.OldString, params.NewString) if err != nil { return response, err } if response.IsError { // Return early if there was an error during content replacement // This prevents unnecessary LSP diagnostics processing return response, nil } waitForLspDiagnostics(ctx, params.FilePath, e.lspClients) text := fmt.Sprintf("\n%s\n\n", response.Content) text += getDiagnostics(params.FilePath, e.lspClients) response.Content = text return response, nil } func (e *editTool) createNewFile(ctx context.Context, filePath, content string) (ToolResponse, error) { fileInfo, err := os.Stat(filePath) if err == nil { if fileInfo.IsDir() { return NewTextErrorResponse(fmt.Sprintf("path is a directory, not a file: %s", filePath)), nil } return NewTextErrorResponse(fmt.Sprintf("file already exists: %s", filePath)), nil } else if !os.IsNotExist(err) { return ToolResponse{}, fmt.Errorf("failed to access file: %w", err) } dir := filepath.Dir(filePath) if err = os.MkdirAll(dir, 0o755); err != nil { return ToolResponse{}, fmt.Errorf("failed to create parent directories: %w", err) } sessionID, messageID := GetContextValues(ctx) if sessionID == "" || messageID == "" { return ToolResponse{}, fmt.Errorf("session ID and message ID are required for creating a new file") } diff, additions, removals := diff.GenerateDiff( "", content, filePath, ) rootDir := config.WorkingDirectory() permissionPath := filepath.Dir(filePath) if strings.HasPrefix(filePath, rootDir) { permissionPath = rootDir } p := e.permissions.Request( permission.CreatePermissionRequest{ SessionID: sessionID, Path: permissionPath, ToolName: EditToolName, Action: "write", Description: fmt.Sprintf("Create file %s", filePath), Params: EditPermissionsParams{ FilePath: filePath, Diff: diff, }, }, ) if !p { return ToolResponse{}, permission.ErrorPermissionDenied } err = os.WriteFile(filePath, []byte(content), 0o644) if err != nil { return ToolResponse{}, fmt.Errorf("failed to write file: %w", err) } // File can't be in the history so we create a new file history _, err = e.files.Create(ctx, sessionID, filePath, "") if err != nil { // Log error but don't fail the operation return ToolResponse{}, fmt.Errorf("error creating file history: %w", err) } // Add the new content to the file history _, err = e.files.CreateVersion(ctx, sessionID, filePath, content) if err != nil { // Log error but don't fail the operation logging.Debug("Error creating file history version", "error", err) } recordFileWrite(filePath) recordFileRead(filePath) return WithResponseMetadata( NewTextResponse("File created: "+filePath), EditResponseMetadata{ Diff: diff, Additions: additions, Removals: removals, }, ), nil } func (e *editTool) deleteContent(ctx context.Context, filePath, oldString string) (ToolResponse, error) { fileInfo, err := os.Stat(filePath) if err != nil { if os.IsNotExist(err) { return NewTextErrorResponse(fmt.Sprintf("file not found: %s", filePath)), nil } return ToolResponse{}, fmt.Errorf("failed to access file: %w", err) } if fileInfo.IsDir() { return NewTextErrorResponse(fmt.Sprintf("path is a directory, not a file: %s", filePath)), nil } if getLastReadTime(filePath).IsZero() { return NewTextErrorResponse("you must read the file before editing it. Use the View tool first"), nil } modTime := fileInfo.ModTime() lastRead := getLastReadTime(filePath) if modTime.After(lastRead) { return NewTextErrorResponse( fmt.Sprintf("file %s has been modified since it was last read (mod time: %s, last read: %s)", filePath, modTime.Format(time.RFC3339), lastRead.Format(time.RFC3339), )), nil } content, err := os.ReadFile(filePath) if err != nil { return ToolResponse{}, fmt.Errorf("failed to read file: %w", err) } oldContent := string(content) index := strings.Index(oldContent, oldString) if index == -1 { return NewTextErrorResponse("old_string not found in file. Make sure it matches exactly, including whitespace and line breaks"), nil } lastIndex := strings.LastIndex(oldContent, oldString) if index != lastIndex { return NewTextErrorResponse("old_string appears multiple times in the file. Please provide more context to ensure a unique match"), nil } newContent := oldContent[:index] + oldContent[index+len(oldString):] sessionID, messageID := GetContextValues(ctx) if sessionID == "" || messageID == "" { return ToolResponse{}, fmt.Errorf("session ID and message ID are required for creating a new file") } diff, additions, removals := diff.GenerateDiff( oldContent, newContent, filePath, ) rootDir := config.WorkingDirectory() permissionPath := filepath.Dir(filePath) if strings.HasPrefix(filePath, rootDir) { permissionPath = rootDir } p := e.permissions.Request( permission.CreatePermissionRequest{ SessionID: sessionID, Path: permissionPath, ToolName: EditToolName, Action: "write", Description: fmt.Sprintf("Delete content from file %s", filePath), Params: EditPermissionsParams{ FilePath: filePath, Diff: diff, }, }, ) if !p { return ToolResponse{}, permission.ErrorPermissionDenied } err = os.WriteFile(filePath, []byte(newContent), 0o644) if err != nil { return ToolResponse{}, fmt.Errorf("failed to write file: %w", err) } // Check if file exists in history file, err := e.files.GetByPathAndSession(ctx, filePath, sessionID) if err != nil { _, err = e.files.Create(ctx, sessionID, filePath, oldContent) if err != nil { // Log error but don't fail the operation return ToolResponse{}, fmt.Errorf("error creating file history: %w", err) } } if file.Content != oldContent { // User Manually changed the content store an intermediate version _, err = e.files.CreateVersion(ctx, sessionID, filePath, oldContent) if err != nil { logging.Debug("Error creating file history version", "error", err) } } // Store the new version _, err = e.files.CreateVersion(ctx, sessionID, filePath, "") if err != nil { logging.Debug("Error creating file history version", "error", err) } recordFileWrite(filePath) recordFileRead(filePath) return WithResponseMetadata( NewTextResponse("Content deleted from file: "+filePath), EditResponseMetadata{ Diff: diff, Additions: additions, Removals: removals, }, ), nil } func (e *editTool) replaceContent(ctx context.Context, filePath, oldString, newString string) (ToolResponse, error) { fileInfo, err := os.Stat(filePath) if err != nil { if os.IsNotExist(err) { return NewTextErrorResponse(fmt.Sprintf("file not found: %s", filePath)), nil } return ToolResponse{}, fmt.Errorf("failed to access file: %w", err) } if fileInfo.IsDir() { return NewTextErrorResponse(fmt.Sprintf("path is a directory, not a file: %s", filePath)), nil } if getLastReadTime(filePath).IsZero() { return NewTextErrorResponse("you must read the file before editing it. Use the View tool first"), nil } modTime := fileInfo.ModTime() lastRead := getLastReadTime(filePath) if modTime.After(lastRead) { return NewTextErrorResponse( fmt.Sprintf("file %s has been modified since it was last read (mod time: %s, last read: %s)", filePath, modTime.Format(time.RFC3339), lastRead.Format(time.RFC3339), )), nil } content, err := os.ReadFile(filePath) if err != nil { return ToolResponse{}, fmt.Errorf("failed to read file: %w", err) } oldContent := string(content) index := strings.Index(oldContent, oldString) if index == -1 { return NewTextErrorResponse("old_string not found in file. Make sure it matches exactly, including whitespace and line breaks"), nil } lastIndex := strings.LastIndex(oldContent, oldString) if index != lastIndex { return NewTextErrorResponse("old_string appears multiple times in the file. Please provide more context to ensure a unique match"), nil } newContent := oldContent[:index] + newString + oldContent[index+len(oldString):] if oldContent == newContent { return NewTextErrorResponse("new content is the same as old content. No changes made."), nil } sessionID, messageID := GetContextValues(ctx) if sessionID == "" || messageID == "" { return ToolResponse{}, fmt.Errorf("session ID and message ID are required for creating a new file") } diff, additions, removals := diff.GenerateDiff( oldContent, newContent, filePath, ) rootDir := config.WorkingDirectory() permissionPath := filepath.Dir(filePath) if strings.HasPrefix(filePath, rootDir) { permissionPath = rootDir } p := e.permissions.Request( permission.CreatePermissionRequest{ SessionID: sessionID, Path: permissionPath, ToolName: EditToolName, Action: "write", Description: fmt.Sprintf("Replace content in file %s", filePath), Params: EditPermissionsParams{ FilePath: filePath, Diff: diff, }, }, ) if !p { return ToolResponse{}, permission.ErrorPermissionDenied } err = os.WriteFile(filePath, []byte(newContent), 0o644) if err != nil { return ToolResponse{}, fmt.Errorf("failed to write file: %w", err) } // Check if file exists in history file, err := e.files.GetByPathAndSession(ctx, filePath, sessionID) if err != nil { _, err = e.files.Create(ctx, sessionID, filePath, oldContent) if err != nil { // Log error but don't fail the operation return ToolResponse{}, fmt.Errorf("error creating file history: %w", err) } } if file.Content != oldContent { // User Manually changed the content store an intermediate version _, err = e.files.CreateVersion(ctx, sessionID, filePath, oldContent) if err != nil { logging.Debug("Error creating file history version", "error", err) } } // Store the new version _, err = e.files.CreateVersion(ctx, sessionID, filePath, newContent) if err != nil { logging.Debug("Error creating file history version", "error", err) } recordFileWrite(filePath) recordFileRead(filePath) return WithResponseMetadata( NewTextResponse("Content replaced in file: "+filePath), EditResponseMetadata{ Diff: diff, Additions: additions, Removals: removals, }), nil } ================================================ FILE: internal/llm/tools/fetch.go ================================================ package tools import ( "context" "encoding/json" "fmt" "io" "net/http" "strings" "time" md "github.com/JohannesKaufmann/html-to-markdown" "github.com/PuerkitoBio/goquery" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/permission" ) type FetchParams struct { URL string `json:"url"` Format string `json:"format"` Timeout int `json:"timeout,omitempty"` } type FetchPermissionsParams struct { URL string `json:"url"` Format string `json:"format"` Timeout int `json:"timeout,omitempty"` } type fetchTool struct { client *http.Client permissions permission.Service } const ( FetchToolName = "fetch" fetchToolDescription = `Fetches content from a URL and returns it in the specified format. WHEN TO USE THIS TOOL: - Use when you need to download content from a URL - Helpful for retrieving documentation, API responses, or web content - Useful for getting external information to assist with tasks HOW TO USE: - Provide the URL to fetch content from - Specify the desired output format (text, markdown, or html) - Optionally set a timeout for the request FEATURES: - Supports three output formats: text, markdown, and html - Automatically handles HTTP redirects - Sets reasonable timeouts to prevent hanging - Validates input parameters before making requests LIMITATIONS: - Maximum response size is 5MB - Only supports HTTP and HTTPS protocols - Cannot handle authentication or cookies - Some websites may block automated requests TIPS: - Use text format for plain text content or simple API responses - Use markdown format for content that should be rendered with formatting - Use html format when you need the raw HTML structure - Set appropriate timeouts for potentially slow websites` ) func NewFetchTool(permissions permission.Service) BaseTool { return &fetchTool{ client: &http.Client{ Timeout: 30 * time.Second, }, permissions: permissions, } } func (t *fetchTool) Info() ToolInfo { return ToolInfo{ Name: FetchToolName, Description: fetchToolDescription, Parameters: map[string]any{ "url": map[string]any{ "type": "string", "description": "The URL to fetch content from", }, "format": map[string]any{ "type": "string", "description": "The format to return the content in (text, markdown, or html)", "enum": []string{"text", "markdown", "html"}, }, "timeout": map[string]any{ "type": "number", "description": "Optional timeout in seconds (max 120)", }, }, Required: []string{"url", "format"}, } } func (t *fetchTool) Run(ctx context.Context, call ToolCall) (ToolResponse, error) { var params FetchParams if err := json.Unmarshal([]byte(call.Input), ¶ms); err != nil { return NewTextErrorResponse("Failed to parse fetch parameters: " + err.Error()), nil } if params.URL == "" { return NewTextErrorResponse("URL parameter is required"), nil } format := strings.ToLower(params.Format) if format != "text" && format != "markdown" && format != "html" { return NewTextErrorResponse("Format must be one of: text, markdown, html"), nil } if !strings.HasPrefix(params.URL, "http://") && !strings.HasPrefix(params.URL, "https://") { return NewTextErrorResponse("URL must start with http:// or https://"), nil } sessionID, messageID := GetContextValues(ctx) if sessionID == "" || messageID == "" { return ToolResponse{}, fmt.Errorf("session ID and message ID are required for creating a new file") } p := t.permissions.Request( permission.CreatePermissionRequest{ SessionID: sessionID, Path: config.WorkingDirectory(), ToolName: FetchToolName, Action: "fetch", Description: fmt.Sprintf("Fetch content from URL: %s", params.URL), Params: FetchPermissionsParams(params), }, ) if !p { return ToolResponse{}, permission.ErrorPermissionDenied } client := t.client if params.Timeout > 0 { maxTimeout := 120 // 2 minutes if params.Timeout > maxTimeout { params.Timeout = maxTimeout } client = &http.Client{ Timeout: time.Duration(params.Timeout) * time.Second, } } req, err := http.NewRequestWithContext(ctx, "GET", params.URL, nil) if err != nil { return ToolResponse{}, fmt.Errorf("failed to create request: %w", err) } req.Header.Set("User-Agent", "opencode/1.0") resp, err := client.Do(req) if err != nil { return ToolResponse{}, fmt.Errorf("failed to fetch URL: %w", err) } defer resp.Body.Close() if resp.StatusCode != http.StatusOK { return NewTextErrorResponse(fmt.Sprintf("Request failed with status code: %d", resp.StatusCode)), nil } maxSize := int64(5 * 1024 * 1024) // 5MB body, err := io.ReadAll(io.LimitReader(resp.Body, maxSize)) if err != nil { return NewTextErrorResponse("Failed to read response body: " + err.Error()), nil } content := string(body) contentType := resp.Header.Get("Content-Type") switch format { case "text": if strings.Contains(contentType, "text/html") { text, err := extractTextFromHTML(content) if err != nil { return NewTextErrorResponse("Failed to extract text from HTML: " + err.Error()), nil } return NewTextResponse(text), nil } return NewTextResponse(content), nil case "markdown": if strings.Contains(contentType, "text/html") { markdown, err := convertHTMLToMarkdown(content) if err != nil { return NewTextErrorResponse("Failed to convert HTML to Markdown: " + err.Error()), nil } return NewTextResponse(markdown), nil } return NewTextResponse("```\n" + content + "\n```"), nil case "html": return NewTextResponse(content), nil default: return NewTextResponse(content), nil } } func extractTextFromHTML(html string) (string, error) { doc, err := goquery.NewDocumentFromReader(strings.NewReader(html)) if err != nil { return "", err } text := doc.Text() text = strings.Join(strings.Fields(text), " ") return text, nil } func convertHTMLToMarkdown(html string) (string, error) { converter := md.NewConverter("", true, nil) markdown, err := converter.ConvertString(html) if err != nil { return "", err } return markdown, nil } ================================================ FILE: internal/llm/tools/file.go ================================================ package tools import ( "sync" "time" ) // File record to track when files were read/written type fileRecord struct { path string readTime time.Time writeTime time.Time } var ( fileRecords = make(map[string]fileRecord) fileRecordMutex sync.RWMutex ) func recordFileRead(path string) { fileRecordMutex.Lock() defer fileRecordMutex.Unlock() record, exists := fileRecords[path] if !exists { record = fileRecord{path: path} } record.readTime = time.Now() fileRecords[path] = record } func getLastReadTime(path string) time.Time { fileRecordMutex.RLock() defer fileRecordMutex.RUnlock() record, exists := fileRecords[path] if !exists { return time.Time{} } return record.readTime } func recordFileWrite(path string) { fileRecordMutex.Lock() defer fileRecordMutex.Unlock() record, exists := fileRecords[path] if !exists { record = fileRecord{path: path} } record.writeTime = time.Now() fileRecords[path] = record } ================================================ FILE: internal/llm/tools/glob.go ================================================ package tools import ( "bytes" "context" "encoding/json" "fmt" "os/exec" "path/filepath" "sort" "strings" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/fileutil" "github.com/opencode-ai/opencode/internal/logging" ) const ( GlobToolName = "glob" globDescription = `Fast file pattern matching tool that finds files by name and pattern, returning matching paths sorted by modification time (newest first). WHEN TO USE THIS TOOL: - Use when you need to find files by name patterns or extensions - Great for finding specific file types across a directory structure - Useful for discovering files that match certain naming conventions HOW TO USE: - Provide a glob pattern to match against file paths - Optionally specify a starting directory (defaults to current working directory) - Results are sorted with most recently modified files first GLOB PATTERN SYNTAX: - '*' matches any sequence of non-separator characters - '**' matches any sequence of characters, including separators - '?' matches any single non-separator character - '[...]' matches any character in the brackets - '[!...]' matches any character not in the brackets COMMON PATTERN EXAMPLES: - '*.js' - Find all JavaScript files in the current directory - '**/*.js' - Find all JavaScript files in any subdirectory - 'src/**/*.{ts,tsx}' - Find all TypeScript files in the src directory - '*.{html,css,js}' - Find all HTML, CSS, and JS files LIMITATIONS: - Results are limited to 100 files (newest first) - Does not search file contents (use Grep tool for that) - Hidden files (starting with '.') are skipped TIPS: - For the most useful results, combine with the Grep tool: first find files with Glob, then search their contents with Grep - When doing iterative exploration that may require multiple rounds of searching, consider using the Agent tool instead - Always check if results are truncated and refine your search pattern if needed` ) type GlobParams struct { Pattern string `json:"pattern"` Path string `json:"path"` } type GlobResponseMetadata struct { NumberOfFiles int `json:"number_of_files"` Truncated bool `json:"truncated"` } type globTool struct{} func NewGlobTool() BaseTool { return &globTool{} } func (g *globTool) Info() ToolInfo { return ToolInfo{ Name: GlobToolName, Description: globDescription, Parameters: map[string]any{ "pattern": map[string]any{ "type": "string", "description": "The glob pattern to match files against", }, "path": map[string]any{ "type": "string", "description": "The directory to search in. Defaults to the current working directory.", }, }, Required: []string{"pattern"}, } } func (g *globTool) Run(ctx context.Context, call ToolCall) (ToolResponse, error) { var params GlobParams if err := json.Unmarshal([]byte(call.Input), ¶ms); err != nil { return NewTextErrorResponse(fmt.Sprintf("error parsing parameters: %s", err)), nil } if params.Pattern == "" { return NewTextErrorResponse("pattern is required"), nil } searchPath := params.Path if searchPath == "" { searchPath = config.WorkingDirectory() } files, truncated, err := globFiles(params.Pattern, searchPath, 100) if err != nil { return ToolResponse{}, fmt.Errorf("error finding files: %w", err) } var output string if len(files) == 0 { output = "No files found" } else { output = strings.Join(files, "\n") if truncated { output += "\n\n(Results are truncated. Consider using a more specific path or pattern.)" } } return WithResponseMetadata( NewTextResponse(output), GlobResponseMetadata{ NumberOfFiles: len(files), Truncated: truncated, }, ), nil } func globFiles(pattern, searchPath string, limit int) ([]string, bool, error) { cmdRg := fileutil.GetRgCmd(pattern) if cmdRg != nil { cmdRg.Dir = searchPath matches, err := runRipgrep(cmdRg, searchPath, limit) if err == nil { return matches, len(matches) >= limit && limit > 0, nil } logging.Warn(fmt.Sprintf("Ripgrep execution failed: %v. Falling back to doublestar.", err)) } return fileutil.GlobWithDoublestar(pattern, searchPath, limit) } func runRipgrep(cmd *exec.Cmd, searchRoot string, limit int) ([]string, error) { out, err := cmd.CombinedOutput() if err != nil { if ee, ok := err.(*exec.ExitError); ok && ee.ExitCode() == 1 { return nil, nil } return nil, fmt.Errorf("ripgrep: %w\n%s", err, out) } var matches []string for _, p := range bytes.Split(out, []byte{0}) { if len(p) == 0 { continue } absPath := string(p) if !filepath.IsAbs(absPath) { absPath = filepath.Join(searchRoot, absPath) } if fileutil.SkipHidden(absPath) { continue } matches = append(matches, absPath) } sort.SliceStable(matches, func(i, j int) bool { return len(matches[i]) < len(matches[j]) }) if limit > 0 && len(matches) > limit { matches = matches[:limit] } return matches, nil } ================================================ FILE: internal/llm/tools/grep.go ================================================ package tools import ( "bufio" "context" "encoding/json" "fmt" "os" "os/exec" "path/filepath" "regexp" "sort" "strconv" "strings" "time" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/fileutil" ) type GrepParams struct { Pattern string `json:"pattern"` Path string `json:"path"` Include string `json:"include"` LiteralText bool `json:"literal_text"` } type grepMatch struct { path string modTime time.Time lineNum int lineText string } type GrepResponseMetadata struct { NumberOfMatches int `json:"number_of_matches"` Truncated bool `json:"truncated"` } type grepTool struct{} const ( GrepToolName = "grep" grepDescription = `Fast content search tool that finds files containing specific text or patterns, returning matching file paths sorted by modification time (newest first). WHEN TO USE THIS TOOL: - Use when you need to find files containing specific text or patterns - Great for searching code bases for function names, variable declarations, or error messages - Useful for finding all files that use a particular API or pattern HOW TO USE: - Provide a regex pattern to search for within file contents - Set literal_text=true if you want to search for the exact text with special characters (recommended for non-regex users) - Optionally specify a starting directory (defaults to current working directory) - Optionally provide an include pattern to filter which files to search - Results are sorted with most recently modified files first REGEX PATTERN SYNTAX (when literal_text=false): - Supports standard regular expression syntax - 'function' searches for the literal text "function" - 'log\..*Error' finds text starting with "log." and ending with "Error" - 'import\s+.*\s+from' finds import statements in JavaScript/TypeScript COMMON INCLUDE PATTERN EXAMPLES: - '*.js' - Only search JavaScript files - '*.{ts,tsx}' - Only search TypeScript files - '*.go' - Only search Go files LIMITATIONS: - Results are limited to 100 files (newest first) - Performance depends on the number of files being searched - Very large binary files may be skipped - Hidden files (starting with '.') are skipped TIPS: - For faster, more targeted searches, first use Glob to find relevant files, then use Grep - When doing iterative exploration that may require multiple rounds of searching, consider using the Agent tool instead - Always check if results are truncated and refine your search pattern if needed - Use literal_text=true when searching for exact text containing special characters like dots, parentheses, etc.` ) func NewGrepTool() BaseTool { return &grepTool{} } func (g *grepTool) Info() ToolInfo { return ToolInfo{ Name: GrepToolName, Description: grepDescription, Parameters: map[string]any{ "pattern": map[string]any{ "type": "string", "description": "The regex pattern to search for in file contents", }, "path": map[string]any{ "type": "string", "description": "The directory to search in. Defaults to the current working directory.", }, "include": map[string]any{ "type": "string", "description": "File pattern to include in the search (e.g. \"*.js\", \"*.{ts,tsx}\")", }, "literal_text": map[string]any{ "type": "boolean", "description": "If true, the pattern will be treated as literal text with special regex characters escaped. Default is false.", }, }, Required: []string{"pattern"}, } } // escapeRegexPattern escapes special regex characters so they're treated as literal characters func escapeRegexPattern(pattern string) string { specialChars := []string{"\\", ".", "+", "*", "?", "(", ")", "[", "]", "{", "}", "^", "$", "|"} escaped := pattern for _, char := range specialChars { escaped = strings.ReplaceAll(escaped, char, "\\"+char) } return escaped } func (g *grepTool) Run(ctx context.Context, call ToolCall) (ToolResponse, error) { var params GrepParams if err := json.Unmarshal([]byte(call.Input), ¶ms); err != nil { return NewTextErrorResponse(fmt.Sprintf("error parsing parameters: %s", err)), nil } if params.Pattern == "" { return NewTextErrorResponse("pattern is required"), nil } // If literal_text is true, escape the pattern searchPattern := params.Pattern if params.LiteralText { searchPattern = escapeRegexPattern(params.Pattern) } searchPath := params.Path if searchPath == "" { searchPath = config.WorkingDirectory() } matches, truncated, err := searchFiles(searchPattern, searchPath, params.Include, 100) if err != nil { return ToolResponse{}, fmt.Errorf("error searching files: %w", err) } var output string if len(matches) == 0 { output = "No files found" } else { output = fmt.Sprintf("Found %d matches\n", len(matches)) currentFile := "" for _, match := range matches { if currentFile != match.path { if currentFile != "" { output += "\n" } currentFile = match.path output += fmt.Sprintf("%s:\n", match.path) } if match.lineNum > 0 { output += fmt.Sprintf(" Line %d: %s\n", match.lineNum, match.lineText) } else { output += fmt.Sprintf(" %s\n", match.path) } } if truncated { output += "\n(Results are truncated. Consider using a more specific path or pattern.)" } } return WithResponseMetadata( NewTextResponse(output), GrepResponseMetadata{ NumberOfMatches: len(matches), Truncated: truncated, }, ), nil } func searchFiles(pattern, rootPath, include string, limit int) ([]grepMatch, bool, error) { matches, err := searchWithRipgrep(pattern, rootPath, include) if err != nil { matches, err = searchFilesWithRegex(pattern, rootPath, include) if err != nil { return nil, false, err } } sort.Slice(matches, func(i, j int) bool { return matches[i].modTime.After(matches[j].modTime) }) truncated := len(matches) > limit if truncated { matches = matches[:limit] } return matches, truncated, nil } func searchWithRipgrep(pattern, path, include string) ([]grepMatch, error) { _, err := exec.LookPath("rg") if err != nil { return nil, fmt.Errorf("ripgrep not found: %w", err) } // Use -n to show line numbers and include the matched line args := []string{"-H", "-n", pattern} if include != "" { args = append(args, "--glob", include) } args = append(args, path) cmd := exec.Command("rg", args...) output, err := cmd.Output() if err != nil { if exitErr, ok := err.(*exec.ExitError); ok && exitErr.ExitCode() == 1 { return []grepMatch{}, nil } return nil, err } lines := strings.Split(strings.TrimSpace(string(output)), "\n") matches := make([]grepMatch, 0, len(lines)) for _, line := range lines { if line == "" { continue } // Parse ripgrep output format: file:line:content parts := strings.SplitN(line, ":", 3) if len(parts) < 3 { continue } filePath := parts[0] lineNum, err := strconv.Atoi(parts[1]) if err != nil { continue } lineText := parts[2] fileInfo, err := os.Stat(filePath) if err != nil { continue // Skip files we can't access } matches = append(matches, grepMatch{ path: filePath, modTime: fileInfo.ModTime(), lineNum: lineNum, lineText: lineText, }) } return matches, nil } func searchFilesWithRegex(pattern, rootPath, include string) ([]grepMatch, error) { matches := []grepMatch{} regex, err := regexp.Compile(pattern) if err != nil { return nil, fmt.Errorf("invalid regex pattern: %w", err) } var includePattern *regexp.Regexp if include != "" { regexPattern := globToRegex(include) includePattern, err = regexp.Compile(regexPattern) if err != nil { return nil, fmt.Errorf("invalid include pattern: %w", err) } } err = filepath.Walk(rootPath, func(path string, info os.FileInfo, err error) error { if err != nil { return nil // Skip errors } if info.IsDir() { return nil // Skip directories } if fileutil.SkipHidden(path) { return nil } if includePattern != nil && !includePattern.MatchString(path) { return nil } match, lineNum, lineText, err := fileContainsPattern(path, regex) if err != nil { return nil // Skip files we can't read } if match { matches = append(matches, grepMatch{ path: path, modTime: info.ModTime(), lineNum: lineNum, lineText: lineText, }) if len(matches) >= 200 { return filepath.SkipAll } } return nil }) if err != nil { return nil, err } return matches, nil } func fileContainsPattern(filePath string, pattern *regexp.Regexp) (bool, int, string, error) { file, err := os.Open(filePath) if err != nil { return false, 0, "", err } defer file.Close() scanner := bufio.NewScanner(file) lineNum := 0 for scanner.Scan() { lineNum++ line := scanner.Text() if pattern.MatchString(line) { return true, lineNum, line, nil } } return false, 0, "", scanner.Err() } func globToRegex(glob string) string { regexPattern := strings.ReplaceAll(glob, ".", "\\.") regexPattern = strings.ReplaceAll(regexPattern, "*", ".*") regexPattern = strings.ReplaceAll(regexPattern, "?", ".") re := regexp.MustCompile(`\{([^}]+)\}`) regexPattern = re.ReplaceAllStringFunc(regexPattern, func(match string) string { inner := match[1 : len(match)-1] return "(" + strings.ReplaceAll(inner, ",", "|") + ")" }) return regexPattern } ================================================ FILE: internal/llm/tools/ls.go ================================================ package tools import ( "context" "encoding/json" "fmt" "os" "path/filepath" "strings" "github.com/opencode-ai/opencode/internal/config" ) type LSParams struct { Path string `json:"path"` Ignore []string `json:"ignore"` } type TreeNode struct { Name string `json:"name"` Path string `json:"path"` Type string `json:"type"` // "file" or "directory" Children []*TreeNode `json:"children,omitempty"` } type LSResponseMetadata struct { NumberOfFiles int `json:"number_of_files"` Truncated bool `json:"truncated"` } type lsTool struct{} const ( LSToolName = "ls" MaxLSFiles = 1000 lsDescription = `Directory listing tool that shows files and subdirectories in a tree structure, helping you explore and understand the project organization. WHEN TO USE THIS TOOL: - Use when you need to explore the structure of a directory - Helpful for understanding the organization of a project - Good first step when getting familiar with a new codebase HOW TO USE: - Provide a path to list (defaults to current working directory) - Optionally specify glob patterns to ignore - Results are displayed in a tree structure FEATURES: - Displays a hierarchical view of files and directories - Automatically skips hidden files/directories (starting with '.') - Skips common system directories like __pycache__ - Can filter out files matching specific patterns LIMITATIONS: - Results are limited to 1000 files - Very large directories will be truncated - Does not show file sizes or permissions - Cannot recursively list all directories in a large project TIPS: - Use Glob tool for finding files by name patterns instead of browsing - Use Grep tool for searching file contents - Combine with other tools for more effective exploration` ) func NewLsTool() BaseTool { return &lsTool{} } func (l *lsTool) Info() ToolInfo { return ToolInfo{ Name: LSToolName, Description: lsDescription, Parameters: map[string]any{ "path": map[string]any{ "type": "string", "description": "The path to the directory to list (defaults to current working directory)", }, "ignore": map[string]any{ "type": "array", "description": "List of glob patterns to ignore", "items": map[string]any{ "type": "string", }, }, }, Required: []string{"path"}, } } func (l *lsTool) Run(ctx context.Context, call ToolCall) (ToolResponse, error) { var params LSParams if err := json.Unmarshal([]byte(call.Input), ¶ms); err != nil { return NewTextErrorResponse(fmt.Sprintf("error parsing parameters: %s", err)), nil } searchPath := params.Path if searchPath == "" { searchPath = config.WorkingDirectory() } if !filepath.IsAbs(searchPath) { searchPath = filepath.Join(config.WorkingDirectory(), searchPath) } if _, err := os.Stat(searchPath); os.IsNotExist(err) { return NewTextErrorResponse(fmt.Sprintf("path does not exist: %s", searchPath)), nil } files, truncated, err := listDirectory(searchPath, params.Ignore, MaxLSFiles) if err != nil { return ToolResponse{}, fmt.Errorf("error listing directory: %w", err) } tree := createFileTree(files) output := printTree(tree, searchPath) if truncated { output = fmt.Sprintf("There are more than %d files in the directory. Use a more specific path or use the Glob tool to find specific files. The first %d files and directories are included below:\n\n%s", MaxLSFiles, MaxLSFiles, output) } return WithResponseMetadata( NewTextResponse(output), LSResponseMetadata{ NumberOfFiles: len(files), Truncated: truncated, }, ), nil } func listDirectory(initialPath string, ignorePatterns []string, limit int) ([]string, bool, error) { var results []string truncated := false err := filepath.Walk(initialPath, func(path string, info os.FileInfo, err error) error { if err != nil { return nil // Skip files we don't have permission to access } if shouldSkip(path, ignorePatterns) { if info.IsDir() { return filepath.SkipDir } return nil } if path != initialPath { if info.IsDir() { path = path + string(filepath.Separator) } results = append(results, path) } if len(results) >= limit { truncated = true return filepath.SkipAll } return nil }) if err != nil { return nil, truncated, err } return results, truncated, nil } func shouldSkip(path string, ignorePatterns []string) bool { base := filepath.Base(path) if base != "." && strings.HasPrefix(base, ".") { return true } commonIgnored := []string{ "__pycache__", "node_modules", "dist", "build", "target", "vendor", "bin", "obj", ".git", ".idea", ".vscode", ".DS_Store", "*.pyc", "*.pyo", "*.pyd", "*.so", "*.dll", "*.exe", } if strings.Contains(path, filepath.Join("__pycache__", "")) { return true } for _, ignored := range commonIgnored { if strings.HasSuffix(ignored, "/") { if strings.Contains(path, filepath.Join(ignored[:len(ignored)-1], "")) { return true } } else if strings.HasPrefix(ignored, "*.") { if strings.HasSuffix(base, ignored[1:]) { return true } } else { if base == ignored { return true } } } for _, pattern := range ignorePatterns { matched, err := filepath.Match(pattern, base) if err == nil && matched { return true } } return false } func createFileTree(sortedPaths []string) []*TreeNode { root := []*TreeNode{} pathMap := make(map[string]*TreeNode) for _, path := range sortedPaths { parts := strings.Split(path, string(filepath.Separator)) currentPath := "" var parentPath string var cleanParts []string for _, part := range parts { if part != "" { cleanParts = append(cleanParts, part) } } parts = cleanParts if len(parts) == 0 { continue } for i, part := range parts { if currentPath == "" { currentPath = part } else { currentPath = filepath.Join(currentPath, part) } if _, exists := pathMap[currentPath]; exists { parentPath = currentPath continue } isLastPart := i == len(parts)-1 isDir := !isLastPart || strings.HasSuffix(path, string(filepath.Separator)) nodeType := "file" if isDir { nodeType = "directory" } newNode := &TreeNode{ Name: part, Path: currentPath, Type: nodeType, Children: []*TreeNode{}, } pathMap[currentPath] = newNode if i > 0 && parentPath != "" { if parent, ok := pathMap[parentPath]; ok { parent.Children = append(parent.Children, newNode) } } else { root = append(root, newNode) } parentPath = currentPath } } return root } func printTree(tree []*TreeNode, rootPath string) string { var result strings.Builder result.WriteString(fmt.Sprintf("- %s%s\n", rootPath, string(filepath.Separator))) for _, node := range tree { printNode(&result, node, 1) } return result.String() } func printNode(builder *strings.Builder, node *TreeNode, level int) { indent := strings.Repeat(" ", level) nodeName := node.Name if node.Type == "directory" { nodeName += string(filepath.Separator) } fmt.Fprintf(builder, "%s- %s\n", indent, nodeName) if node.Type == "directory" && len(node.Children) > 0 { for _, child := range node.Children { printNode(builder, child, level+1) } } } ================================================ FILE: internal/llm/tools/ls_test.go ================================================ package tools import ( "context" "encoding/json" "os" "path/filepath" "strings" "testing" "github.com/stretchr/testify/assert" "github.com/stretchr/testify/require" ) func TestLsTool_Info(t *testing.T) { tool := NewLsTool() info := tool.Info() assert.Equal(t, LSToolName, info.Name) assert.NotEmpty(t, info.Description) assert.Contains(t, info.Parameters, "path") assert.Contains(t, info.Parameters, "ignore") assert.Contains(t, info.Required, "path") } func TestLsTool_Run(t *testing.T) { // Create a temporary directory for testing tempDir, err := os.MkdirTemp("", "ls_tool_test") require.NoError(t, err) defer os.RemoveAll(tempDir) // Create a test directory structure testDirs := []string{ "dir1", "dir2", "dir2/subdir1", "dir2/subdir2", "dir3", "dir3/.hidden_dir", "__pycache__", } testFiles := []string{ "file1.txt", "file2.txt", "dir1/file3.txt", "dir2/file4.txt", "dir2/subdir1/file5.txt", "dir2/subdir2/file6.txt", "dir3/file7.txt", "dir3/.hidden_file.txt", "__pycache__/cache.pyc", ".hidden_root_file.txt", } // Create directories for _, dir := range testDirs { dirPath := filepath.Join(tempDir, dir) err := os.MkdirAll(dirPath, 0755) require.NoError(t, err) } // Create files for _, file := range testFiles { filePath := filepath.Join(tempDir, file) err := os.WriteFile(filePath, []byte("test content"), 0644) require.NoError(t, err) } t.Run("lists directory successfully", func(t *testing.T) { tool := NewLsTool() params := LSParams{ Path: tempDir, } paramsJSON, err := json.Marshal(params) require.NoError(t, err) call := ToolCall{ Name: LSToolName, Input: string(paramsJSON), } response, err := tool.Run(context.Background(), call) require.NoError(t, err) // Check that visible directories and files are included assert.Contains(t, response.Content, "dir1") assert.Contains(t, response.Content, "dir2") assert.Contains(t, response.Content, "dir3") assert.Contains(t, response.Content, "file1.txt") assert.Contains(t, response.Content, "file2.txt") // Check that hidden files and directories are not included assert.NotContains(t, response.Content, ".hidden_dir") assert.NotContains(t, response.Content, ".hidden_file.txt") assert.NotContains(t, response.Content, ".hidden_root_file.txt") // Check that __pycache__ is not included assert.NotContains(t, response.Content, "__pycache__") }) t.Run("handles non-existent path", func(t *testing.T) { tool := NewLsTool() params := LSParams{ Path: filepath.Join(tempDir, "non_existent_dir"), } paramsJSON, err := json.Marshal(params) require.NoError(t, err) call := ToolCall{ Name: LSToolName, Input: string(paramsJSON), } response, err := tool.Run(context.Background(), call) require.NoError(t, err) assert.Contains(t, response.Content, "path does not exist") }) t.Run("handles empty path parameter", func(t *testing.T) { // For this test, we need to mock the config.WorkingDirectory function // Since we can't easily do that, we'll just check that the response doesn't contain an error message tool := NewLsTool() params := LSParams{ Path: "", } paramsJSON, err := json.Marshal(params) require.NoError(t, err) call := ToolCall{ Name: LSToolName, Input: string(paramsJSON), } response, err := tool.Run(context.Background(), call) require.NoError(t, err) // The response should either contain a valid directory listing or an error // We'll just check that it's not empty assert.NotEmpty(t, response.Content) }) t.Run("handles invalid parameters", func(t *testing.T) { tool := NewLsTool() call := ToolCall{ Name: LSToolName, Input: "invalid json", } response, err := tool.Run(context.Background(), call) require.NoError(t, err) assert.Contains(t, response.Content, "error parsing parameters") }) t.Run("respects ignore patterns", func(t *testing.T) { tool := NewLsTool() params := LSParams{ Path: tempDir, Ignore: []string{"file1.txt", "dir1"}, } paramsJSON, err := json.Marshal(params) require.NoError(t, err) call := ToolCall{ Name: LSToolName, Input: string(paramsJSON), } response, err := tool.Run(context.Background(), call) require.NoError(t, err) // The output format is a tree, so we need to check for specific patterns // Check that file1.txt is not directly mentioned assert.NotContains(t, response.Content, "- file1.txt") // Check that dir1/ is not directly mentioned assert.NotContains(t, response.Content, "- dir1/") }) t.Run("handles relative path", func(t *testing.T) { // Save original working directory origWd, err := os.Getwd() require.NoError(t, err) defer func() { os.Chdir(origWd) }() // Change to a directory above the temp directory parentDir := filepath.Dir(tempDir) err = os.Chdir(parentDir) require.NoError(t, err) tool := NewLsTool() params := LSParams{ Path: filepath.Base(tempDir), } paramsJSON, err := json.Marshal(params) require.NoError(t, err) call := ToolCall{ Name: LSToolName, Input: string(paramsJSON), } response, err := tool.Run(context.Background(), call) require.NoError(t, err) // Should list the temp directory contents assert.Contains(t, response.Content, "dir1") assert.Contains(t, response.Content, "file1.txt") }) } func TestShouldSkip(t *testing.T) { testCases := []struct { name string path string ignorePatterns []string expected bool }{ { name: "hidden file", path: "/path/to/.hidden_file", ignorePatterns: []string{}, expected: true, }, { name: "hidden directory", path: "/path/to/.hidden_dir", ignorePatterns: []string{}, expected: true, }, { name: "pycache directory", path: "/path/to/__pycache__/file.pyc", ignorePatterns: []string{}, expected: true, }, { name: "node_modules directory", path: "/path/to/node_modules/package", ignorePatterns: []string{}, expected: false, // The shouldSkip function doesn't directly check for node_modules in the path }, { name: "normal file", path: "/path/to/normal_file.txt", ignorePatterns: []string{}, expected: false, }, { name: "normal directory", path: "/path/to/normal_dir", ignorePatterns: []string{}, expected: false, }, { name: "ignored by pattern", path: "/path/to/ignore_me.txt", ignorePatterns: []string{"ignore_*.txt"}, expected: true, }, { name: "not ignored by pattern", path: "/path/to/keep_me.txt", ignorePatterns: []string{"ignore_*.txt"}, expected: false, }, } for _, tc := range testCases { t.Run(tc.name, func(t *testing.T) { result := shouldSkip(tc.path, tc.ignorePatterns) assert.Equal(t, tc.expected, result) }) } } func TestCreateFileTree(t *testing.T) { paths := []string{ "/path/to/file1.txt", "/path/to/dir1/file2.txt", "/path/to/dir1/subdir/file3.txt", "/path/to/dir2/file4.txt", } tree := createFileTree(paths) // Check the structure of the tree assert.Len(t, tree, 1) // Should have one root node // Check the root node rootNode := tree[0] assert.Equal(t, "path", rootNode.Name) assert.Equal(t, "directory", rootNode.Type) assert.Len(t, rootNode.Children, 1) // Check the "to" node toNode := rootNode.Children[0] assert.Equal(t, "to", toNode.Name) assert.Equal(t, "directory", toNode.Type) assert.Len(t, toNode.Children, 3) // file1.txt, dir1, dir2 // Find the dir1 node var dir1Node *TreeNode for _, child := range toNode.Children { if child.Name == "dir1" { dir1Node = child break } } require.NotNil(t, dir1Node) assert.Equal(t, "directory", dir1Node.Type) assert.Len(t, dir1Node.Children, 2) // file2.txt and subdir } func TestPrintTree(t *testing.T) { // Create a simple tree tree := []*TreeNode{ { Name: "dir1", Path: "dir1", Type: "directory", Children: []*TreeNode{ { Name: "file1.txt", Path: "dir1/file1.txt", Type: "file", }, { Name: "subdir", Path: "dir1/subdir", Type: "directory", Children: []*TreeNode{ { Name: "file2.txt", Path: "dir1/subdir/file2.txt", Type: "file", }, }, }, }, }, { Name: "file3.txt", Path: "file3.txt", Type: "file", }, } result := printTree(tree, "/root") // Check the output format assert.Contains(t, result, "- /root/") assert.Contains(t, result, " - dir1/") assert.Contains(t, result, " - file1.txt") assert.Contains(t, result, " - subdir/") assert.Contains(t, result, " - file2.txt") assert.Contains(t, result, " - file3.txt") } func TestListDirectory(t *testing.T) { // Create a temporary directory for testing tempDir, err := os.MkdirTemp("", "list_directory_test") require.NoError(t, err) defer os.RemoveAll(tempDir) // Create a test directory structure testDirs := []string{ "dir1", "dir1/subdir1", ".hidden_dir", } testFiles := []string{ "file1.txt", "file2.txt", "dir1/file3.txt", "dir1/subdir1/file4.txt", ".hidden_file.txt", } // Create directories for _, dir := range testDirs { dirPath := filepath.Join(tempDir, dir) err := os.MkdirAll(dirPath, 0755) require.NoError(t, err) } // Create files for _, file := range testFiles { filePath := filepath.Join(tempDir, file) err := os.WriteFile(filePath, []byte("test content"), 0644) require.NoError(t, err) } t.Run("lists files with no limit", func(t *testing.T) { files, truncated, err := listDirectory(tempDir, []string{}, 1000) require.NoError(t, err) assert.False(t, truncated) // Check that visible files and directories are included containsPath := func(paths []string, target string) bool { targetPath := filepath.Join(tempDir, target) for _, path := range paths { if strings.HasPrefix(path, targetPath) { return true } } return false } assert.True(t, containsPath(files, "dir1")) assert.True(t, containsPath(files, "file1.txt")) assert.True(t, containsPath(files, "file2.txt")) assert.True(t, containsPath(files, "dir1/file3.txt")) // Check that hidden files and directories are not included assert.False(t, containsPath(files, ".hidden_dir")) assert.False(t, containsPath(files, ".hidden_file.txt")) }) t.Run("respects limit and returns truncated flag", func(t *testing.T) { files, truncated, err := listDirectory(tempDir, []string{}, 2) require.NoError(t, err) assert.True(t, truncated) assert.Len(t, files, 2) }) t.Run("respects ignore patterns", func(t *testing.T) { files, truncated, err := listDirectory(tempDir, []string{"*.txt"}, 1000) require.NoError(t, err) assert.False(t, truncated) // Check that no .txt files are included for _, file := range files { assert.False(t, strings.HasSuffix(file, ".txt"), "Found .txt file: %s", file) } // But directories should still be included containsDir := false for _, file := range files { if strings.Contains(file, "dir1") { containsDir = true break } } assert.True(t, containsDir) }) } ================================================ FILE: internal/llm/tools/patch.go ================================================ package tools import ( "context" "encoding/json" "fmt" "os" "path/filepath" "time" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/diff" "github.com/opencode-ai/opencode/internal/history" "github.com/opencode-ai/opencode/internal/logging" "github.com/opencode-ai/opencode/internal/lsp" "github.com/opencode-ai/opencode/internal/permission" ) type PatchParams struct { PatchText string `json:"patch_text"` } type PatchResponseMetadata struct { FilesChanged []string `json:"files_changed"` Additions int `json:"additions"` Removals int `json:"removals"` } type patchTool struct { lspClients map[string]*lsp.Client permissions permission.Service files history.Service } const ( PatchToolName = "patch" patchDescription = `Applies a patch to multiple files in one operation. This tool is useful for making coordinated changes across multiple files. The patch text must follow this format: *** Begin Patch *** Update File: /path/to/file @@ Context line (unique within the file) Line to keep -Line to remove +Line to add Line to keep *** Add File: /path/to/new/file +Content of the new file +More content *** Delete File: /path/to/file/to/delete *** End Patch Before using this tool: 1. Use the FileRead tool to understand the files' contents and context 2. Verify all file paths are correct (use the LS tool) CRITICAL REQUIREMENTS FOR USING THIS TOOL: 1. UNIQUENESS: Context lines MUST uniquely identify the specific sections you want to change 2. PRECISION: All whitespace, indentation, and surrounding code must match exactly 3. VALIDATION: Ensure edits result in idiomatic, correct code 4. PATHS: Always use absolute file paths (starting with /) The tool will apply all changes in a single atomic operation.` ) func NewPatchTool(lspClients map[string]*lsp.Client, permissions permission.Service, files history.Service) BaseTool { return &patchTool{ lspClients: lspClients, permissions: permissions, files: files, } } func (p *patchTool) Info() ToolInfo { return ToolInfo{ Name: PatchToolName, Description: patchDescription, Parameters: map[string]any{ "patch_text": map[string]any{ "type": "string", "description": "The full patch text that describes all changes to be made", }, }, Required: []string{"patch_text"}, } } func (p *patchTool) Run(ctx context.Context, call ToolCall) (ToolResponse, error) { var params PatchParams if err := json.Unmarshal([]byte(call.Input), ¶ms); err != nil { return NewTextErrorResponse("invalid parameters"), nil } if params.PatchText == "" { return NewTextErrorResponse("patch_text is required"), nil } // Identify all files needed for the patch and verify they've been read filesToRead := diff.IdentifyFilesNeeded(params.PatchText) for _, filePath := range filesToRead { absPath := filePath if !filepath.IsAbs(absPath) { wd := config.WorkingDirectory() absPath = filepath.Join(wd, absPath) } if getLastReadTime(absPath).IsZero() { return NewTextErrorResponse(fmt.Sprintf("you must read the file %s before patching it. Use the FileRead tool first", filePath)), nil } fileInfo, err := os.Stat(absPath) if err != nil { if os.IsNotExist(err) { return NewTextErrorResponse(fmt.Sprintf("file not found: %s", absPath)), nil } return ToolResponse{}, fmt.Errorf("failed to access file: %w", err) } if fileInfo.IsDir() { return NewTextErrorResponse(fmt.Sprintf("path is a directory, not a file: %s", absPath)), nil } modTime := fileInfo.ModTime() lastRead := getLastReadTime(absPath) if modTime.After(lastRead) { return NewTextErrorResponse( fmt.Sprintf("file %s has been modified since it was last read (mod time: %s, last read: %s)", absPath, modTime.Format(time.RFC3339), lastRead.Format(time.RFC3339), )), nil } } // Check for new files to ensure they don't already exist filesToAdd := diff.IdentifyFilesAdded(params.PatchText) for _, filePath := range filesToAdd { absPath := filePath if !filepath.IsAbs(absPath) { wd := config.WorkingDirectory() absPath = filepath.Join(wd, absPath) } _, err := os.Stat(absPath) if err == nil { return NewTextErrorResponse(fmt.Sprintf("file already exists and cannot be added: %s", absPath)), nil } else if !os.IsNotExist(err) { return ToolResponse{}, fmt.Errorf("failed to check file: %w", err) } } // Load all required files currentFiles := make(map[string]string) for _, filePath := range filesToRead { absPath := filePath if !filepath.IsAbs(absPath) { wd := config.WorkingDirectory() absPath = filepath.Join(wd, absPath) } content, err := os.ReadFile(absPath) if err != nil { return ToolResponse{}, fmt.Errorf("failed to read file %s: %w", absPath, err) } currentFiles[filePath] = string(content) } // Process the patch patch, fuzz, err := diff.TextToPatch(params.PatchText, currentFiles) if err != nil { return NewTextErrorResponse(fmt.Sprintf("failed to parse patch: %s", err)), nil } if fuzz > 3 { return NewTextErrorResponse(fmt.Sprintf("patch contains fuzzy matches (fuzz level: %d). Please make your context lines more precise", fuzz)), nil } // Convert patch to commit commit, err := diff.PatchToCommit(patch, currentFiles) if err != nil { return NewTextErrorResponse(fmt.Sprintf("failed to create commit from patch: %s", err)), nil } // Get session ID and message ID sessionID, messageID := GetContextValues(ctx) if sessionID == "" || messageID == "" { return ToolResponse{}, fmt.Errorf("session ID and message ID are required for creating a patch") } // Request permission for all changes for path, change := range commit.Changes { switch change.Type { case diff.ActionAdd: dir := filepath.Dir(path) patchDiff, _, _ := diff.GenerateDiff("", *change.NewContent, path) p := p.permissions.Request( permission.CreatePermissionRequest{ SessionID: sessionID, Path: dir, ToolName: PatchToolName, Action: "create", Description: fmt.Sprintf("Create file %s", path), Params: EditPermissionsParams{ FilePath: path, Diff: patchDiff, }, }, ) if !p { return ToolResponse{}, permission.ErrorPermissionDenied } case diff.ActionUpdate: currentContent := "" if change.OldContent != nil { currentContent = *change.OldContent } newContent := "" if change.NewContent != nil { newContent = *change.NewContent } patchDiff, _, _ := diff.GenerateDiff(currentContent, newContent, path) dir := filepath.Dir(path) p := p.permissions.Request( permission.CreatePermissionRequest{ SessionID: sessionID, Path: dir, ToolName: PatchToolName, Action: "update", Description: fmt.Sprintf("Update file %s", path), Params: EditPermissionsParams{ FilePath: path, Diff: patchDiff, }, }, ) if !p { return ToolResponse{}, permission.ErrorPermissionDenied } case diff.ActionDelete: dir := filepath.Dir(path) patchDiff, _, _ := diff.GenerateDiff(*change.OldContent, "", path) p := p.permissions.Request( permission.CreatePermissionRequest{ SessionID: sessionID, Path: dir, ToolName: PatchToolName, Action: "delete", Description: fmt.Sprintf("Delete file %s", path), Params: EditPermissionsParams{ FilePath: path, Diff: patchDiff, }, }, ) if !p { return ToolResponse{}, permission.ErrorPermissionDenied } } } // Apply the changes to the filesystem err = diff.ApplyCommit(commit, func(path string, content string) error { absPath := path if !filepath.IsAbs(absPath) { wd := config.WorkingDirectory() absPath = filepath.Join(wd, absPath) } // Create parent directories if needed dir := filepath.Dir(absPath) if err := os.MkdirAll(dir, 0o755); err != nil { return fmt.Errorf("failed to create parent directories for %s: %w", absPath, err) } return os.WriteFile(absPath, []byte(content), 0o644) }, func(path string) error { absPath := path if !filepath.IsAbs(absPath) { wd := config.WorkingDirectory() absPath = filepath.Join(wd, absPath) } return os.Remove(absPath) }) if err != nil { return NewTextErrorResponse(fmt.Sprintf("failed to apply patch: %s", err)), nil } // Update file history for all modified files changedFiles := []string{} totalAdditions := 0 totalRemovals := 0 for path, change := range commit.Changes { absPath := path if !filepath.IsAbs(absPath) { wd := config.WorkingDirectory() absPath = filepath.Join(wd, absPath) } changedFiles = append(changedFiles, absPath) oldContent := "" if change.OldContent != nil { oldContent = *change.OldContent } newContent := "" if change.NewContent != nil { newContent = *change.NewContent } // Calculate diff statistics _, additions, removals := diff.GenerateDiff(oldContent, newContent, path) totalAdditions += additions totalRemovals += removals // Update history file, err := p.files.GetByPathAndSession(ctx, absPath, sessionID) if err != nil && change.Type != diff.ActionAdd { // If not adding a file, create history entry for existing file _, err = p.files.Create(ctx, sessionID, absPath, oldContent) if err != nil { logging.Debug("Error creating file history", "error", err) } } if err == nil && change.Type != diff.ActionAdd && file.Content != oldContent { // User manually changed content, store intermediate version _, err = p.files.CreateVersion(ctx, sessionID, absPath, oldContent) if err != nil { logging.Debug("Error creating file history version", "error", err) } } // Store new version if change.Type == diff.ActionDelete { _, err = p.files.CreateVersion(ctx, sessionID, absPath, "") } else { _, err = p.files.CreateVersion(ctx, sessionID, absPath, newContent) } if err != nil { logging.Debug("Error creating file history version", "error", err) } // Record file operations recordFileWrite(absPath) recordFileRead(absPath) } // Run LSP diagnostics on all changed files for _, filePath := range changedFiles { waitForLspDiagnostics(ctx, filePath, p.lspClients) } result := fmt.Sprintf("Patch applied successfully. %d files changed, %d additions, %d removals", len(changedFiles), totalAdditions, totalRemovals) diagnosticsText := "" for _, filePath := range changedFiles { diagnosticsText += getDiagnostics(filePath, p.lspClients) } if diagnosticsText != "" { result += "\n\nDiagnostics:\n" + diagnosticsText } return WithResponseMetadata( NewTextResponse(result), PatchResponseMetadata{ FilesChanged: changedFiles, Additions: totalAdditions, Removals: totalRemovals, }), nil } ================================================ FILE: internal/llm/tools/shell/shell.go ================================================ package shell import ( "context" "errors" "fmt" "os" "os/exec" "path/filepath" "strings" "sync" "syscall" "time" "github.com/opencode-ai/opencode/internal/config" ) type PersistentShell struct { cmd *exec.Cmd stdin *os.File isAlive bool cwd string mu sync.Mutex commandQueue chan *commandExecution } type commandExecution struct { command string timeout time.Duration resultChan chan commandResult ctx context.Context } type commandResult struct { stdout string stderr string exitCode int interrupted bool err error } var ( shellInstance *PersistentShell shellInstanceOnce sync.Once ) func GetPersistentShell(workingDir string) *PersistentShell { shellInstanceOnce.Do(func() { shellInstance = newPersistentShell(workingDir) }) if shellInstance == nil { shellInstance = newPersistentShell(workingDir) } else if !shellInstance.isAlive { shellInstance = newPersistentShell(shellInstance.cwd) } return shellInstance } func newPersistentShell(cwd string) *PersistentShell { // Get shell configuration from config cfg := config.Get() // Default to environment variable if config is not set or nil var shellPath string var shellArgs []string if cfg != nil { shellPath = cfg.Shell.Path shellArgs = cfg.Shell.Args } if shellPath == "" { shellPath = os.Getenv("SHELL") if shellPath == "" { shellPath = "/bin/bash" } } // Default shell args if len(shellArgs) == 0 { shellArgs = []string{"-l"} } cmd := exec.Command(shellPath, shellArgs...) cmd.Dir = cwd stdinPipe, err := cmd.StdinPipe() if err != nil { return nil } cmd.Env = append(os.Environ(), "GIT_EDITOR=true") err = cmd.Start() if err != nil { return nil } shell := &PersistentShell{ cmd: cmd, stdin: stdinPipe.(*os.File), isAlive: true, cwd: cwd, commandQueue: make(chan *commandExecution, 10), } go func() { defer func() { if r := recover(); r != nil { fmt.Fprintf(os.Stderr, "Panic in shell command processor: %v\n", r) shell.isAlive = false close(shell.commandQueue) } }() shell.processCommands() }() go func() { err := cmd.Wait() if err != nil { // Log the error if needed } shell.isAlive = false close(shell.commandQueue) }() return shell } func (s *PersistentShell) processCommands() { for cmd := range s.commandQueue { result := s.execCommand(cmd.command, cmd.timeout, cmd.ctx) cmd.resultChan <- result } } func (s *PersistentShell) execCommand(command string, timeout time.Duration, ctx context.Context) commandResult { s.mu.Lock() defer s.mu.Unlock() if !s.isAlive { return commandResult{ stderr: "Shell is not alive", exitCode: 1, err: errors.New("shell is not alive"), } } tempDir := os.TempDir() stdoutFile := filepath.Join(tempDir, fmt.Sprintf("opencode-stdout-%d", time.Now().UnixNano())) stderrFile := filepath.Join(tempDir, fmt.Sprintf("opencode-stderr-%d", time.Now().UnixNano())) statusFile := filepath.Join(tempDir, fmt.Sprintf("opencode-status-%d", time.Now().UnixNano())) cwdFile := filepath.Join(tempDir, fmt.Sprintf("opencode-cwd-%d", time.Now().UnixNano())) defer func() { os.Remove(stdoutFile) os.Remove(stderrFile) os.Remove(statusFile) os.Remove(cwdFile) }() fullCommand := fmt.Sprintf(` eval %s < /dev/null > %s 2> %s EXEC_EXIT_CODE=$? pwd > %s echo $EXEC_EXIT_CODE > %s `, shellQuote(command), shellQuote(stdoutFile), shellQuote(stderrFile), shellQuote(cwdFile), shellQuote(statusFile), ) _, err := s.stdin.Write([]byte(fullCommand + "\n")) if err != nil { return commandResult{ stderr: fmt.Sprintf("Failed to write command to shell: %v", err), exitCode: 1, err: err, } } interrupted := false startTime := time.Now() done := make(chan bool) go func() { for { select { case <-ctx.Done(): s.killChildren() interrupted = true done <- true return case <-time.After(10 * time.Millisecond): if fileExists(statusFile) && fileSize(statusFile) > 0 { done <- true return } if timeout > 0 { elapsed := time.Since(startTime) if elapsed > timeout { s.killChildren() interrupted = true done <- true return } } } } }() <-done stdout := readFileOrEmpty(stdoutFile) stderr := readFileOrEmpty(stderrFile) exitCodeStr := readFileOrEmpty(statusFile) newCwd := readFileOrEmpty(cwdFile) exitCode := 0 if exitCodeStr != "" { fmt.Sscanf(exitCodeStr, "%d", &exitCode) } else if interrupted { exitCode = 143 stderr += "\nCommand execution timed out or was interrupted" } if newCwd != "" { s.cwd = strings.TrimSpace(newCwd) } return commandResult{ stdout: stdout, stderr: stderr, exitCode: exitCode, interrupted: interrupted, } } func (s *PersistentShell) killChildren() { if s.cmd == nil || s.cmd.Process == nil { return } pgrepCmd := exec.Command("pgrep", "-P", fmt.Sprintf("%d", s.cmd.Process.Pid)) output, err := pgrepCmd.Output() if err != nil { return } for pidStr := range strings.SplitSeq(string(output), "\n") { if pidStr = strings.TrimSpace(pidStr); pidStr != "" { var pid int fmt.Sscanf(pidStr, "%d", &pid) if pid > 0 { proc, err := os.FindProcess(pid) if err == nil { proc.Signal(syscall.SIGTERM) } } } } } func (s *PersistentShell) Exec(ctx context.Context, command string, timeoutMs int) (string, string, int, bool, error) { if !s.isAlive { return "", "Shell is not alive", 1, false, errors.New("shell is not alive") } timeout := time.Duration(timeoutMs) * time.Millisecond resultChan := make(chan commandResult) s.commandQueue <- &commandExecution{ command: command, timeout: timeout, resultChan: resultChan, ctx: ctx, } result := <-resultChan return result.stdout, result.stderr, result.exitCode, result.interrupted, result.err } func (s *PersistentShell) Close() { s.mu.Lock() defer s.mu.Unlock() if !s.isAlive { return } s.stdin.Write([]byte("exit\n")) s.cmd.Process.Kill() s.isAlive = false } func shellQuote(s string) string { return "'" + strings.ReplaceAll(s, "'", "'\\''") + "'" } func readFileOrEmpty(path string) string { content, err := os.ReadFile(path) if err != nil { return "" } return string(content) } func fileExists(path string) bool { _, err := os.Stat(path) return err == nil } func fileSize(path string) int64 { info, err := os.Stat(path) if err != nil { return 0 } return info.Size() } ================================================ FILE: internal/llm/tools/sourcegraph.go ================================================ package tools import ( "bytes" "context" "encoding/json" "fmt" "io" "net/http" "strings" "time" ) type SourcegraphParams struct { Query string `json:"query"` Count int `json:"count,omitempty"` ContextWindow int `json:"context_window,omitempty"` Timeout int `json:"timeout,omitempty"` } type SourcegraphResponseMetadata struct { NumberOfMatches int `json:"number_of_matches"` Truncated bool `json:"truncated"` } type sourcegraphTool struct { client *http.Client } const ( SourcegraphToolName = "sourcegraph" sourcegraphToolDescription = `Search code across public repositories using Sourcegraph's GraphQL API. WHEN TO USE THIS TOOL: - Use when you need to find code examples or implementations across public repositories - Helpful for researching how others have solved similar problems - Useful for discovering patterns and best practices in open source code HOW TO USE: - Provide a search query using Sourcegraph's query syntax - Optionally specify the number of results to return (default: 10) - Optionally set a timeout for the request QUERY SYNTAX: - Basic search: "fmt.Println" searches for exact matches - File filters: "file:.go fmt.Println" limits to Go files - Repository filters: "repo:^github\.com/golang/go$ fmt.Println" limits to specific repos - Language filters: "lang:go fmt.Println" limits to Go code - Boolean operators: "fmt.Println AND log.Fatal" for combined terms - Regular expressions: "fmt\.(Print|Printf|Println)" for pattern matching - Quoted strings: "\"exact phrase\"" for exact phrase matching - Exclude filters: "-file:test" or "-repo:forks" to exclude matches ADVANCED FILTERS: - Repository filters: * "repo:name" - Match repositories with name containing "name" * "repo:^github\.com/org/repo$" - Exact repository match * "repo:org/repo@branch" - Search specific branch * "repo:org/repo rev:branch" - Alternative branch syntax * "-repo:name" - Exclude repositories * "fork:yes" or "fork:only" - Include or only show forks * "archived:yes" or "archived:only" - Include or only show archived repos * "visibility:public" or "visibility:private" - Filter by visibility - File filters: * "file:\.js$" - Files with .js extension * "file:internal/" - Files in internal directory * "-file:test" - Exclude test files * "file:has.content(Copyright)" - Files containing "Copyright" * "file:has.contributor([email protected])" - Files with specific contributor - Content filters: * "content:\"exact string\"" - Search for exact string * "-content:\"unwanted\"" - Exclude files with unwanted content * "case:yes" - Case-sensitive search - Type filters: * "type:symbol" - Search for symbols (functions, classes, etc.) * "type:file" - Search file content only * "type:path" - Search filenames only * "type:diff" - Search code changes * "type:commit" - Search commit messages - Commit/diff search: * "after:\"1 month ago\"" - Commits after date * "before:\"2023-01-01\"" - Commits before date * "author:name" - Commits by author * "message:\"fix bug\"" - Commits with message - Result selection: * "select:repo" - Show only repository names * "select:file" - Show only file paths * "select:content" - Show only matching content * "select:symbol" - Show only matching symbols - Result control: * "count:100" - Return up to 100 results * "count:all" - Return all results * "timeout:30s" - Set search timeout EXAMPLES: - "file:.go context.WithTimeout" - Find Go code using context.WithTimeout - "lang:typescript useState type:symbol" - Find TypeScript React useState hooks - "repo:^github\.com/kubernetes/kubernetes$ pod list type:file" - Find Kubernetes files related to pod listing - "repo:sourcegraph/sourcegraph$ after:\"3 months ago\" type:diff database" - Recent changes to database code - "file:Dockerfile (alpine OR ubuntu) -content:alpine:latest" - Dockerfiles with specific base images - "repo:has.path(\.py) file:requirements.txt tensorflow" - Python projects using TensorFlow BOOLEAN OPERATORS: - "term1 AND term2" - Results containing both terms - "term1 OR term2" - Results containing either term - "term1 NOT term2" - Results with term1 but not term2 - "term1 and (term2 or term3)" - Grouping with parentheses LIMITATIONS: - Only searches public repositories - Rate limits may apply - Complex queries may take longer to execute - Maximum of 20 results per query TIPS: - Use specific file extensions to narrow results - Add repo: filters for more targeted searches - Use type:symbol to find function/method definitions - Use type:file to find relevant files` ) func NewSourcegraphTool() BaseTool { return &sourcegraphTool{ client: &http.Client{ Timeout: 30 * time.Second, }, } } func (t *sourcegraphTool) Info() ToolInfo { return ToolInfo{ Name: SourcegraphToolName, Description: sourcegraphToolDescription, Parameters: map[string]any{ "query": map[string]any{ "type": "string", "description": "The Sourcegraph search query", }, "count": map[string]any{ "type": "number", "description": "Optional number of results to return (default: 10, max: 20)", }, "context_window": map[string]any{ "type": "number", "description": "The context around the match to return (default: 10 lines)", }, "timeout": map[string]any{ "type": "number", "description": "Optional timeout in seconds (max 120)", }, }, Required: []string{"query"}, } } func (t *sourcegraphTool) Run(ctx context.Context, call ToolCall) (ToolResponse, error) { var params SourcegraphParams if err := json.Unmarshal([]byte(call.Input), ¶ms); err != nil { return NewTextErrorResponse("Failed to parse sourcegraph parameters: " + err.Error()), nil } if params.Query == "" { return NewTextErrorResponse("Query parameter is required"), nil } if params.Count <= 0 { params.Count = 10 } else if params.Count > 20 { params.Count = 20 // Limit to 20 results } if params.ContextWindow <= 0 { params.ContextWindow = 10 // Default context window } client := t.client if params.Timeout > 0 { maxTimeout := 120 // 2 minutes if params.Timeout > maxTimeout { params.Timeout = maxTimeout } client = &http.Client{ Timeout: time.Duration(params.Timeout) * time.Second, } } type graphqlRequest struct { Query string `json:"query"` Variables struct { Query string `json:"query"` } `json:"variables"` } request := graphqlRequest{ Query: "query Search($query: String!) { search(query: $query, version: V2, patternType: keyword ) { results { matchCount, limitHit, resultCount, approximateResultCount, missing { name }, timedout { name }, indexUnavailable, results { __typename, ... on FileMatch { repository { name }, file { path, url, content }, lineMatches { preview, lineNumber, offsetAndLengths } } } } } }", } request.Variables.Query = params.Query graphqlQueryBytes, err := json.Marshal(request) if err != nil { return ToolResponse{}, fmt.Errorf("failed to marshal GraphQL request: %w", err) } graphqlQuery := string(graphqlQueryBytes) req, err := http.NewRequestWithContext( ctx, "POST", "https://sourcegraph.com/.api/graphql", bytes.NewBuffer([]byte(graphqlQuery)), ) if err != nil { return ToolResponse{}, fmt.Errorf("failed to create request: %w", err) } req.Header.Set("Content-Type", "application/json") req.Header.Set("User-Agent", "opencode/1.0") resp, err := client.Do(req) if err != nil { return ToolResponse{}, fmt.Errorf("failed to fetch URL: %w", err) } defer resp.Body.Close() if resp.StatusCode != http.StatusOK { body, _ := io.ReadAll(resp.Body) if len(body) > 0 { return NewTextErrorResponse(fmt.Sprintf("Request failed with status code: %d, response: %s", resp.StatusCode, string(body))), nil } return NewTextErrorResponse(fmt.Sprintf("Request failed with status code: %d", resp.StatusCode)), nil } body, err := io.ReadAll(resp.Body) if err != nil { return ToolResponse{}, fmt.Errorf("failed to read response body: %w", err) } var result map[string]any if err = json.Unmarshal(body, &result); err != nil { return ToolResponse{}, fmt.Errorf("failed to unmarshal response: %w", err) } formattedResults, err := formatSourcegraphResults(result, params.ContextWindow) if err != nil { return NewTextErrorResponse("Failed to format results: " + err.Error()), nil } return NewTextResponse(formattedResults), nil } func formatSourcegraphResults(result map[string]any, contextWindow int) (string, error) { var buffer strings.Builder if errors, ok := result["errors"].([]any); ok && len(errors) > 0 { buffer.WriteString("## Sourcegraph API Error\n\n") for _, err := range errors { if errMap, ok := err.(map[string]any); ok { if message, ok := errMap["message"].(string); ok { buffer.WriteString(fmt.Sprintf("- %s\n", message)) } } } return buffer.String(), nil } data, ok := result["data"].(map[string]any) if !ok { return "", fmt.Errorf("invalid response format: missing data field") } search, ok := data["search"].(map[string]any) if !ok { return "", fmt.Errorf("invalid response format: missing search field") } searchResults, ok := search["results"].(map[string]any) if !ok { return "", fmt.Errorf("invalid response format: missing results field") } matchCount, _ := searchResults["matchCount"].(float64) resultCount, _ := searchResults["resultCount"].(float64) limitHit, _ := searchResults["limitHit"].(bool) buffer.WriteString("# Sourcegraph Search Results\n\n") buffer.WriteString(fmt.Sprintf("Found %d matches across %d results\n", int(matchCount), int(resultCount))) if limitHit { buffer.WriteString("(Result limit reached, try a more specific query)\n") } buffer.WriteString("\n") results, ok := searchResults["results"].([]any) if !ok || len(results) == 0 { buffer.WriteString("No results found. Try a different query.\n") return buffer.String(), nil } maxResults := 10 if len(results) > maxResults { results = results[:maxResults] } for i, res := range results { fileMatch, ok := res.(map[string]any) if !ok { continue } typeName, _ := fileMatch["__typename"].(string) if typeName != "FileMatch" { continue } repo, _ := fileMatch["repository"].(map[string]any) file, _ := fileMatch["file"].(map[string]any) lineMatches, _ := fileMatch["lineMatches"].([]any) if repo == nil || file == nil { continue } repoName, _ := repo["name"].(string) filePath, _ := file["path"].(string) fileURL, _ := file["url"].(string) fileContent, _ := file["content"].(string) buffer.WriteString(fmt.Sprintf("## Result %d: %s/%s\n\n", i+1, repoName, filePath)) if fileURL != "" { buffer.WriteString(fmt.Sprintf("URL: %s\n\n", fileURL)) } if len(lineMatches) > 0 { for _, lm := range lineMatches { lineMatch, ok := lm.(map[string]any) if !ok { continue } lineNumber, _ := lineMatch["lineNumber"].(float64) preview, _ := lineMatch["preview"].(string) if fileContent != "" { lines := strings.Split(fileContent, "\n") buffer.WriteString("```\n") startLine := max(1, int(lineNumber)-contextWindow) for j := startLine - 1; j < int(lineNumber)-1 && j < len(lines); j++ { if j >= 0 { buffer.WriteString(fmt.Sprintf("%d| %s\n", j+1, lines[j])) } } buffer.WriteString(fmt.Sprintf("%d| %s\n", int(lineNumber), preview)) endLine := int(lineNumber) + contextWindow for j := int(lineNumber); j < endLine && j < len(lines); j++ { if j < len(lines) { buffer.WriteString(fmt.Sprintf("%d| %s\n", j+1, lines[j])) } } buffer.WriteString("```\n\n") } else { buffer.WriteString("```\n") buffer.WriteString(fmt.Sprintf("%d| %s\n", int(lineNumber), preview)) buffer.WriteString("```\n\n") } } } } return buffer.String(), nil } ================================================ FILE: internal/llm/tools/tools.go ================================================ package tools import ( "context" "encoding/json" ) type ToolInfo struct { Name string Description string Parameters map[string]any Required []string } type toolResponseType string type ( sessionIDContextKey string messageIDContextKey string ) const ( ToolResponseTypeText toolResponseType = "text" ToolResponseTypeImage toolResponseType = "image" SessionIDContextKey sessionIDContextKey = "session_id" MessageIDContextKey messageIDContextKey = "message_id" ) type ToolResponse struct { Type toolResponseType `json:"type"` Content string `json:"content"` Metadata string `json:"metadata,omitempty"` IsError bool `json:"is_error"` } func NewTextResponse(content string) ToolResponse { return ToolResponse{ Type: ToolResponseTypeText, Content: content, } } func WithResponseMetadata(response ToolResponse, metadata any) ToolResponse { if metadata != nil { metadataBytes, err := json.Marshal(metadata) if err != nil { return response } response.Metadata = string(metadataBytes) } return response } func NewTextErrorResponse(content string) ToolResponse { return ToolResponse{ Type: ToolResponseTypeText, Content: content, IsError: true, } } type ToolCall struct { ID string `json:"id"` Name string `json:"name"` Input string `json:"input"` } type BaseTool interface { Info() ToolInfo Run(ctx context.Context, params ToolCall) (ToolResponse, error) } func GetContextValues(ctx context.Context) (string, string) { sessionID := ctx.Value(SessionIDContextKey) messageID := ctx.Value(MessageIDContextKey) if sessionID == nil { return "", "" } if messageID == nil { return sessionID.(string), "" } return sessionID.(string), messageID.(string) } ================================================ FILE: internal/llm/tools/view.go ================================================ package tools import ( "bufio" "context" "encoding/json" "fmt" "io" "os" "path/filepath" "strings" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/logging" "github.com/opencode-ai/opencode/internal/lsp" ) type ViewParams struct { FilePath string `json:"file_path"` Offset int `json:"offset"` Limit int `json:"limit"` } type viewTool struct { lspClients map[string]*lsp.Client } type ViewResponseMetadata struct { FilePath string `json:"file_path"` Content string `json:"content"` } const ( ViewToolName = "view" MaxReadSize = 250 * 1024 DefaultReadLimit = 2000 MaxLineLength = 2000 viewDescription = `File viewing tool that reads and displays the contents of files with line numbers, allowing you to examine code, logs, or text data. WHEN TO USE THIS TOOL: - Use when you need to read the contents of a specific file - Helpful for examining source code, configuration files, or log files - Perfect for looking at text-based file formats HOW TO USE: - Provide the path to the file you want to view - Optionally specify an offset to start reading from a specific line - Optionally specify a limit to control how many lines are read FEATURES: - Displays file contents with line numbers for easy reference - Can read from any position in a file using the offset parameter - Handles large files by limiting the number of lines read - Automatically truncates very long lines for better display - Suggests similar file names when the requested file isn't found LIMITATIONS: - Maximum file size is 250KB - Default reading limit is 2000 lines - Lines longer than 2000 characters are truncated - Cannot display binary files or images - Images can be identified but not displayed TIPS: - Use with Glob tool to first find files you want to view - For code exploration, first use Grep to find relevant files, then View to examine them - When viewing large files, use the offset parameter to read specific sections` ) func NewViewTool(lspClients map[string]*lsp.Client) BaseTool { return &viewTool{ lspClients, } } func (v *viewTool) Info() ToolInfo { return ToolInfo{ Name: ViewToolName, Description: viewDescription, Parameters: map[string]any{ "file_path": map[string]any{ "type": "string", "description": "The path to the file to read", }, "offset": map[string]any{ "type": "integer", "description": "The line number to start reading from (0-based)", }, "limit": map[string]any{ "type": "integer", "description": "The number of lines to read (defaults to 2000)", }, }, Required: []string{"file_path"}, } } // Run implements Tool. func (v *viewTool) Run(ctx context.Context, call ToolCall) (ToolResponse, error) { var params ViewParams logging.Debug("view tool params", "params", call.Input) if err := json.Unmarshal([]byte(call.Input), ¶ms); err != nil { return NewTextErrorResponse(fmt.Sprintf("error parsing parameters: %s", err)), nil } if params.FilePath == "" { return NewTextErrorResponse("file_path is required"), nil } // Handle relative paths filePath := params.FilePath if !filepath.IsAbs(filePath) { filePath = filepath.Join(config.WorkingDirectory(), filePath) } // Check if file exists fileInfo, err := os.Stat(filePath) if err != nil { if os.IsNotExist(err) { // Try to offer suggestions for similarly named files dir := filepath.Dir(filePath) base := filepath.Base(filePath) dirEntries, dirErr := os.ReadDir(dir) if dirErr == nil { var suggestions []string for _, entry := range dirEntries { if strings.Contains(strings.ToLower(entry.Name()), strings.ToLower(base)) || strings.Contains(strings.ToLower(base), strings.ToLower(entry.Name())) { suggestions = append(suggestions, filepath.Join(dir, entry.Name())) if len(suggestions) >= 3 { break } } } if len(suggestions) > 0 { return NewTextErrorResponse(fmt.Sprintf("File not found: %s\n\nDid you mean one of these?\n%s", filePath, strings.Join(suggestions, "\n"))), nil } } return NewTextErrorResponse(fmt.Sprintf("File not found: %s", filePath)), nil } return ToolResponse{}, fmt.Errorf("error accessing file: %w", err) } // Check if it's a directory if fileInfo.IsDir() { return NewTextErrorResponse(fmt.Sprintf("Path is a directory, not a file: %s", filePath)), nil } // Check file size if fileInfo.Size() > MaxReadSize { return NewTextErrorResponse(fmt.Sprintf("File is too large (%d bytes). Maximum size is %d bytes", fileInfo.Size(), MaxReadSize)), nil } // Set default limit if not provided if params.Limit <= 0 { params.Limit = DefaultReadLimit } // Check if it's an image file isImage, imageType := isImageFile(filePath) // TODO: handle images if isImage { return NewTextErrorResponse(fmt.Sprintf("This is an image file of type: %s\nUse a different tool to process images", imageType)), nil } // Read the file content content, lineCount, err := readTextFile(filePath, params.Offset, params.Limit) if err != nil { return ToolResponse{}, fmt.Errorf("error reading file: %w", err) } notifyLspOpenFile(ctx, filePath, v.lspClients) output := "\n" // Format the output with line numbers output += addLineNumbers(content, params.Offset+1) // Add a note if the content was truncated if lineCount > params.Offset+len(strings.Split(content, "\n")) { output += fmt.Sprintf("\n\n(File has more lines. Use 'offset' parameter to read beyond line %d)", params.Offset+len(strings.Split(content, "\n"))) } output += "\n\n" output += getDiagnostics(filePath, v.lspClients) recordFileRead(filePath) return WithResponseMetadata( NewTextResponse(output), ViewResponseMetadata{ FilePath: filePath, Content: content, }, ), nil } func addLineNumbers(content string, startLine int) string { if content == "" { return "" } lines := strings.Split(content, "\n") var result []string for i, line := range lines { line = strings.TrimSuffix(line, "\r") lineNum := i + startLine numStr := fmt.Sprintf("%d", lineNum) if len(numStr) >= 6 { result = append(result, fmt.Sprintf("%s|%s", numStr, line)) } else { paddedNum := fmt.Sprintf("%6s", numStr) result = append(result, fmt.Sprintf("%s|%s", paddedNum, line)) } } return strings.Join(result, "\n") } func readTextFile(filePath string, offset, limit int) (string, int, error) { file, err := os.Open(filePath) if err != nil { return "", 0, err } defer file.Close() lineCount := 0 scanner := NewLineScanner(file) if offset > 0 { for lineCount < offset && scanner.Scan() { lineCount++ } if err = scanner.Err(); err != nil { return "", 0, err } } if offset == 0 { _, err = file.Seek(0, io.SeekStart) if err != nil { return "", 0, err } } var lines []string lineCount = offset for scanner.Scan() && len(lines) < limit { lineCount++ lineText := scanner.Text() if len(lineText) > MaxLineLength { lineText = lineText[:MaxLineLength] + "..." } lines = append(lines, lineText) } // Continue scanning to get total line count for scanner.Scan() { lineCount++ } if err := scanner.Err(); err != nil { return "", 0, err } return strings.Join(lines, "\n"), lineCount, nil } func isImageFile(filePath string) (bool, string) { ext := strings.ToLower(filepath.Ext(filePath)) switch ext { case ".jpg", ".jpeg": return true, "JPEG" case ".png": return true, "PNG" case ".gif": return true, "GIF" case ".bmp": return true, "BMP" case ".svg": return true, "SVG" case ".webp": return true, "WebP" default: return false, "" } } type LineScanner struct { scanner *bufio.Scanner } func NewLineScanner(r io.Reader) *LineScanner { return &LineScanner{ scanner: bufio.NewScanner(r), } } func (s *LineScanner) Scan() bool { return s.scanner.Scan() } func (s *LineScanner) Text() string { return s.scanner.Text() } func (s *LineScanner) Err() error { return s.scanner.Err() } ================================================ FILE: internal/llm/tools/write.go ================================================ package tools import ( "context" "encoding/json" "fmt" "os" "path/filepath" "strings" "time" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/diff" "github.com/opencode-ai/opencode/internal/history" "github.com/opencode-ai/opencode/internal/logging" "github.com/opencode-ai/opencode/internal/lsp" "github.com/opencode-ai/opencode/internal/permission" ) type WriteParams struct { FilePath string `json:"file_path"` Content string `json:"content"` } type WritePermissionsParams struct { FilePath string `json:"file_path"` Diff string `json:"diff"` } type writeTool struct { lspClients map[string]*lsp.Client permissions permission.Service files history.Service } type WriteResponseMetadata struct { Diff string `json:"diff"` Additions int `json:"additions"` Removals int `json:"removals"` } const ( WriteToolName = "write" writeDescription = `File writing tool that creates or updates files in the filesystem, allowing you to save or modify text content. WHEN TO USE THIS TOOL: - Use when you need to create a new file - Helpful for updating existing files with modified content - Perfect for saving generated code, configurations, or text data HOW TO USE: - Provide the path to the file you want to write - Include the content to be written to the file - The tool will create any necessary parent directories FEATURES: - Can create new files or overwrite existing ones - Creates parent directories automatically if they don't exist - Checks if the file has been modified since last read for safety - Avoids unnecessary writes when content hasn't changed LIMITATIONS: - You should read a file before writing to it to avoid conflicts - Cannot append to files (rewrites the entire file) TIPS: - Use the View tool first to examine existing files before modifying them - Use the LS tool to verify the correct location when creating new files - Combine with Glob and Grep tools to find and modify multiple files - Always include descriptive comments when making changes to existing code` ) func NewWriteTool(lspClients map[string]*lsp.Client, permissions permission.Service, files history.Service) BaseTool { return &writeTool{ lspClients: lspClients, permissions: permissions, files: files, } } func (w *writeTool) Info() ToolInfo { return ToolInfo{ Name: WriteToolName, Description: writeDescription, Parameters: map[string]any{ "file_path": map[string]any{ "type": "string", "description": "The path to the file to write", }, "content": map[string]any{ "type": "string", "description": "The content to write to the file", }, }, Required: []string{"file_path", "content"}, } } func (w *writeTool) Run(ctx context.Context, call ToolCall) (ToolResponse, error) { var params WriteParams if err := json.Unmarshal([]byte(call.Input), ¶ms); err != nil { return NewTextErrorResponse(fmt.Sprintf("error parsing parameters: %s", err)), nil } if params.FilePath == "" { return NewTextErrorResponse("file_path is required"), nil } if params.Content == "" { return NewTextErrorResponse("content is required"), nil } filePath := params.FilePath if !filepath.IsAbs(filePath) { filePath = filepath.Join(config.WorkingDirectory(), filePath) } fileInfo, err := os.Stat(filePath) if err == nil { if fileInfo.IsDir() { return NewTextErrorResponse(fmt.Sprintf("Path is a directory, not a file: %s", filePath)), nil } modTime := fileInfo.ModTime() lastRead := getLastReadTime(filePath) if modTime.After(lastRead) { return NewTextErrorResponse(fmt.Sprintf("File %s has been modified since it was last read.\nLast modification: %s\nLast read: %s\n\nPlease read the file again before modifying it.", filePath, modTime.Format(time.RFC3339), lastRead.Format(time.RFC3339))), nil } oldContent, readErr := os.ReadFile(filePath) if readErr == nil && string(oldContent) == params.Content { return NewTextErrorResponse(fmt.Sprintf("File %s already contains the exact content. No changes made.", filePath)), nil } } else if !os.IsNotExist(err) { return ToolResponse{}, fmt.Errorf("error checking file: %w", err) } dir := filepath.Dir(filePath) if err = os.MkdirAll(dir, 0o755); err != nil { return ToolResponse{}, fmt.Errorf("error creating directory: %w", err) } oldContent := "" if fileInfo != nil && !fileInfo.IsDir() { oldBytes, readErr := os.ReadFile(filePath) if readErr == nil { oldContent = string(oldBytes) } } sessionID, messageID := GetContextValues(ctx) if sessionID == "" || messageID == "" { return ToolResponse{}, fmt.Errorf("session_id and message_id are required") } diff, additions, removals := diff.GenerateDiff( oldContent, params.Content, filePath, ) rootDir := config.WorkingDirectory() permissionPath := filepath.Dir(filePath) if strings.HasPrefix(filePath, rootDir) { permissionPath = rootDir } p := w.permissions.Request( permission.CreatePermissionRequest{ SessionID: sessionID, Path: permissionPath, ToolName: WriteToolName, Action: "write", Description: fmt.Sprintf("Create file %s", filePath), Params: WritePermissionsParams{ FilePath: filePath, Diff: diff, }, }, ) if !p { return ToolResponse{}, permission.ErrorPermissionDenied } err = os.WriteFile(filePath, []byte(params.Content), 0o644) if err != nil { return ToolResponse{}, fmt.Errorf("error writing file: %w", err) } // Check if file exists in history file, err := w.files.GetByPathAndSession(ctx, filePath, sessionID) if err != nil { _, err = w.files.Create(ctx, sessionID, filePath, oldContent) if err != nil { // Log error but don't fail the operation return ToolResponse{}, fmt.Errorf("error creating file history: %w", err) } } if file.Content != oldContent { // User Manually changed the content store an intermediate version _, err = w.files.CreateVersion(ctx, sessionID, filePath, oldContent) if err != nil { logging.Debug("Error creating file history version", "error", err) } } // Store the new version _, err = w.files.CreateVersion(ctx, sessionID, filePath, params.Content) if err != nil { logging.Debug("Error creating file history version", "error", err) } recordFileWrite(filePath) recordFileRead(filePath) waitForLspDiagnostics(ctx, filePath, w.lspClients) result := fmt.Sprintf("File successfully written: %s", filePath) result = fmt.Sprintf("\n%s\n", result) result += getDiagnostics(filePath, w.lspClients) return WithResponseMetadata(NewTextResponse(result), WriteResponseMetadata{ Diff: diff, Additions: additions, Removals: removals, }, ), nil } ================================================ FILE: internal/logging/logger.go ================================================ package logging import ( "fmt" "log/slog" "os" // "path/filepath" "encoding/json" "runtime" "runtime/debug" "sync" "time" ) func getCaller() string { var caller string if _, file, line, ok := runtime.Caller(2); ok { // caller = fmt.Sprintf("%s:%d", filepath.Base(file), line) caller = fmt.Sprintf("%s:%d", file, line) } else { caller = "unknown" } return caller } func Info(msg string, args ...any) { source := getCaller() slog.Info(msg, append([]any{"source", source}, args...)...) } func Debug(msg string, args ...any) { // slog.Debug(msg, args...) source := getCaller() slog.Debug(msg, append([]any{"source", source}, args...)...) } func Warn(msg string, args ...any) { slog.Warn(msg, args...) } func Error(msg string, args ...any) { slog.Error(msg, args...) } func InfoPersist(msg string, args ...any) { args = append(args, persistKeyArg, true) slog.Info(msg, args...) } func DebugPersist(msg string, args ...any) { args = append(args, persistKeyArg, true) slog.Debug(msg, args...) } func WarnPersist(msg string, args ...any) { args = append(args, persistKeyArg, true) slog.Warn(msg, args...) } func ErrorPersist(msg string, args ...any) { args = append(args, persistKeyArg, true) slog.Error(msg, args...) } // RecoverPanic is a common function to handle panics gracefully. // It logs the error, creates a panic log file with stack trace, // and executes an optional cleanup function before returning. func RecoverPanic(name string, cleanup func()) { if r := recover(); r != nil { // Log the panic ErrorPersist(fmt.Sprintf("Panic in %s: %v", name, r)) // Create a timestamped panic log file timestamp := time.Now().Format("20060102-150405") filename := fmt.Sprintf("opencode-panic-%s-%s.log", name, timestamp) file, err := os.Create(filename) if err != nil { ErrorPersist(fmt.Sprintf("Failed to create panic log: %v", err)) } else { defer file.Close() // Write panic information and stack trace fmt.Fprintf(file, "Panic in %s: %v\n\n", name, r) fmt.Fprintf(file, "Time: %s\n\n", time.Now().Format(time.RFC3339)) fmt.Fprintf(file, "Stack Trace:\n%s\n", debug.Stack()) InfoPersist(fmt.Sprintf("Panic details written to %s", filename)) } // Execute cleanup function if provided if cleanup != nil { cleanup() } } } // Message Logging for Debug var MessageDir string func GetSessionPrefix(sessionId string) string { return sessionId[:8] } var sessionLogMutex sync.Mutex func AppendToSessionLogFile(sessionId string, filename string, content string) string { if MessageDir == "" || sessionId == "" { return "" } sessionPrefix := GetSessionPrefix(sessionId) sessionLogMutex.Lock() defer sessionLogMutex.Unlock() sessionPath := fmt.Sprintf("%s/%s", MessageDir, sessionPrefix) if _, err := os.Stat(sessionPath); os.IsNotExist(err) { if err := os.MkdirAll(sessionPath, 0o766); err != nil { Error("Failed to create session directory", "dirpath", sessionPath, "error", err) return "" } } filePath := fmt.Sprintf("%s/%s", sessionPath, filename) f, err := os.OpenFile(filePath, os.O_APPEND|os.O_CREATE|os.O_WRONLY, 0644) if err != nil { Error("Failed to open session log file", "filepath", filePath, "error", err) return "" } defer f.Close() // Append chunk to file _, err = f.WriteString(content) if err != nil { Error("Failed to write chunk to session log file", "filepath", filePath, "error", err) return "" } return filePath } func WriteRequestMessageJson(sessionId string, requestSeqId int, message any) string { if MessageDir == "" || sessionId == "" || requestSeqId <= 0 { return "" } msgJson, err := json.Marshal(message) if err != nil { Error("Failed to marshal message", "session_id", sessionId, "request_seq_id", requestSeqId, "error", err) return "" } return WriteRequestMessage(sessionId, requestSeqId, string(msgJson)) } func WriteRequestMessage(sessionId string, requestSeqId int, message string) string { if MessageDir == "" || sessionId == "" || requestSeqId <= 0 { return "" } filename := fmt.Sprintf("%d_request.json", requestSeqId) return AppendToSessionLogFile(sessionId, filename, message) } func AppendToStreamSessionLogJson(sessionId string, requestSeqId int, jsonableChunk any) string { if MessageDir == "" || sessionId == "" || requestSeqId <= 0 { return "" } chunkJson, err := json.Marshal(jsonableChunk) if err != nil { Error("Failed to marshal message", "session_id", sessionId, "request_seq_id", requestSeqId, "error", err) return "" } return AppendToStreamSessionLog(sessionId, requestSeqId, string(chunkJson)) } func AppendToStreamSessionLog(sessionId string, requestSeqId int, chunk string) string { if MessageDir == "" || sessionId == "" || requestSeqId <= 0 { return "" } filename := fmt.Sprintf("%d_response_stream.log", requestSeqId) return AppendToSessionLogFile(sessionId, filename, chunk) } func WriteChatResponseJson(sessionId string, requestSeqId int, response any) string { if MessageDir == "" || sessionId == "" || requestSeqId <= 0 { return "" } responseJson, err := json.Marshal(response) if err != nil { Error("Failed to marshal response", "session_id", sessionId, "request_seq_id", requestSeqId, "error", err) return "" } filename := fmt.Sprintf("%d_response.json", requestSeqId) return AppendToSessionLogFile(sessionId, filename, string(responseJson)) } func WriteToolResultsJson(sessionId string, requestSeqId int, toolResults any) string { if MessageDir == "" || sessionId == "" || requestSeqId <= 0 { return "" } toolResultsJson, err := json.Marshal(toolResults) if err != nil { Error("Failed to marshal tool results", "session_id", sessionId, "request_seq_id", requestSeqId, "error", err) return "" } filename := fmt.Sprintf("%d_tool_results.json", requestSeqId) return AppendToSessionLogFile(sessionId, filename, string(toolResultsJson)) } ================================================ FILE: internal/logging/message.go ================================================ package logging import ( "time" ) // LogMessage is the event payload for a log message type LogMessage struct { ID string Time time.Time Level string Persist bool // used when we want to show the mesage in the status bar PersistTime time.Duration // used when we want to show the mesage in the status bar Message string `json:"msg"` Attributes []Attr } type Attr struct { Key string Value string } ================================================ FILE: internal/logging/writer.go ================================================ package logging import ( "bytes" "context" "fmt" "strings" "sync" "time" "github.com/go-logfmt/logfmt" "github.com/opencode-ai/opencode/internal/pubsub" ) const ( persistKeyArg = "$_persist" PersistTimeArg = "$_persist_time" ) type LogData struct { messages []LogMessage *pubsub.Broker[LogMessage] lock sync.Mutex } func (l *LogData) Add(msg LogMessage) { l.lock.Lock() defer l.lock.Unlock() l.messages = append(l.messages, msg) l.Publish(pubsub.CreatedEvent, msg) } func (l *LogData) List() []LogMessage { l.lock.Lock() defer l.lock.Unlock() return l.messages } var defaultLogData = &LogData{ messages: make([]LogMessage, 0), Broker: pubsub.NewBroker[LogMessage](), } type writer struct{} func (w *writer) Write(p []byte) (int, error) { d := logfmt.NewDecoder(bytes.NewReader(p)) for d.ScanRecord() { msg := LogMessage{ ID: fmt.Sprintf("%d", time.Now().UnixNano()), Time: time.Now(), } for d.ScanKeyval() { switch string(d.Key()) { case "time": parsed, err := time.Parse(time.RFC3339, string(d.Value())) if err != nil { return 0, fmt.Errorf("parsing time: %w", err) } msg.Time = parsed case "level": msg.Level = strings.ToLower(string(d.Value())) case "msg": msg.Message = string(d.Value()) default: if string(d.Key()) == persistKeyArg { msg.Persist = true } else if string(d.Key()) == PersistTimeArg { parsed, err := time.ParseDuration(string(d.Value())) if err != nil { continue } msg.PersistTime = parsed } else { msg.Attributes = append(msg.Attributes, Attr{ Key: string(d.Key()), Value: string(d.Value()), }) } } } defaultLogData.Add(msg) } if d.Err() != nil { return 0, d.Err() } return len(p), nil } func NewWriter() *writer { w := &writer{} return w } func Subscribe(ctx context.Context) <-chan pubsub.Event[LogMessage] { return defaultLogData.Subscribe(ctx) } func List() []LogMessage { return defaultLogData.List() } ================================================ FILE: internal/lsp/client.go ================================================ package lsp import ( "bufio" "context" "encoding/json" "fmt" "io" "os" "os/exec" "path/filepath" "strings" "sync" "sync/atomic" "time" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/logging" "github.com/opencode-ai/opencode/internal/lsp/protocol" ) type Client struct { Cmd *exec.Cmd stdin io.WriteCloser stdout *bufio.Reader stderr io.ReadCloser // Request ID counter nextID atomic.Int32 // Response handlers handlers map[int32]chan *Message handlersMu sync.RWMutex // Server request handlers serverRequestHandlers map[string]ServerRequestHandler serverHandlersMu sync.RWMutex // Notification handlers notificationHandlers map[string]NotificationHandler notificationMu sync.RWMutex // Diagnostic cache diagnostics map[protocol.DocumentUri][]protocol.Diagnostic diagnosticsMu sync.RWMutex // Files are currently opened by the LSP openFiles map[string]*OpenFileInfo openFilesMu sync.RWMutex // Server state serverState atomic.Value } func NewClient(ctx context.Context, command string, args ...string) (*Client, error) { cmd := exec.CommandContext(ctx, command, args...) // Copy env cmd.Env = os.Environ() stdin, err := cmd.StdinPipe() if err != nil { return nil, fmt.Errorf("failed to create stdin pipe: %w", err) } stdout, err := cmd.StdoutPipe() if err != nil { return nil, fmt.Errorf("failed to create stdout pipe: %w", err) } stderr, err := cmd.StderrPipe() if err != nil { return nil, fmt.Errorf("failed to create stderr pipe: %w", err) } client := &Client{ Cmd: cmd, stdin: stdin, stdout: bufio.NewReader(stdout), stderr: stderr, handlers: make(map[int32]chan *Message), notificationHandlers: make(map[string]NotificationHandler), serverRequestHandlers: make(map[string]ServerRequestHandler), diagnostics: make(map[protocol.DocumentUri][]protocol.Diagnostic), openFiles: make(map[string]*OpenFileInfo), } // Initialize server state client.serverState.Store(StateStarting) // Start the LSP server process if err := cmd.Start(); err != nil { return nil, fmt.Errorf("failed to start LSP server: %w", err) } // Handle stderr in a separate goroutine go func() { scanner := bufio.NewScanner(stderr) for scanner.Scan() { fmt.Fprintf(os.Stderr, "LSP Server: %s\n", scanner.Text()) } if err := scanner.Err(); err != nil { fmt.Fprintf(os.Stderr, "Error reading stderr: %v\n", err) } }() // Start message handling loop go func() { defer logging.RecoverPanic("LSP-message-handler", func() { logging.ErrorPersist("LSP message handler crashed, LSP functionality may be impaired") }) client.handleMessages() }() return client, nil } func (c *Client) RegisterNotificationHandler(method string, handler NotificationHandler) { c.notificationMu.Lock() defer c.notificationMu.Unlock() c.notificationHandlers[method] = handler } func (c *Client) RegisterServerRequestHandler(method string, handler ServerRequestHandler) { c.serverHandlersMu.Lock() defer c.serverHandlersMu.Unlock() c.serverRequestHandlers[method] = handler } func (c *Client) InitializeLSPClient(ctx context.Context, workspaceDir string) (*protocol.InitializeResult, error) { initParams := &protocol.InitializeParams{ WorkspaceFoldersInitializeParams: protocol.WorkspaceFoldersInitializeParams{ WorkspaceFolders: []protocol.WorkspaceFolder{ { URI: protocol.URI("file://" + workspaceDir), Name: workspaceDir, }, }, }, XInitializeParams: protocol.XInitializeParams{ ProcessID: int32(os.Getpid()), ClientInfo: &protocol.ClientInfo{ Name: "mcp-language-server", Version: "0.1.0", }, RootPath: workspaceDir, RootURI: protocol.DocumentUri("file://" + workspaceDir), Capabilities: protocol.ClientCapabilities{ Workspace: protocol.WorkspaceClientCapabilities{ Configuration: true, DidChangeConfiguration: protocol.DidChangeConfigurationClientCapabilities{ DynamicRegistration: true, }, DidChangeWatchedFiles: protocol.DidChangeWatchedFilesClientCapabilities{ DynamicRegistration: true, RelativePatternSupport: true, }, }, TextDocument: protocol.TextDocumentClientCapabilities{ Synchronization: &protocol.TextDocumentSyncClientCapabilities{ DynamicRegistration: true, DidSave: true, }, Completion: protocol.CompletionClientCapabilities{ CompletionItem: protocol.ClientCompletionItemOptions{}, }, CodeLens: &protocol.CodeLensClientCapabilities{ DynamicRegistration: true, }, DocumentSymbol: protocol.DocumentSymbolClientCapabilities{}, CodeAction: protocol.CodeActionClientCapabilities{ CodeActionLiteralSupport: protocol.ClientCodeActionLiteralOptions{ CodeActionKind: protocol.ClientCodeActionKindOptions{ ValueSet: []protocol.CodeActionKind{}, }, }, }, PublishDiagnostics: protocol.PublishDiagnosticsClientCapabilities{ VersionSupport: true, }, SemanticTokens: protocol.SemanticTokensClientCapabilities{ Requests: protocol.ClientSemanticTokensRequestOptions{ Range: &protocol.Or_ClientSemanticTokensRequestOptions_range{}, Full: &protocol.Or_ClientSemanticTokensRequestOptions_full{}, }, TokenTypes: []string{}, TokenModifiers: []string{}, Formats: []protocol.TokenFormat{}, }, }, Window: protocol.WindowClientCapabilities{}, }, InitializationOptions: map[string]any{ "codelenses": map[string]bool{ "generate": true, "regenerate_cgo": true, "test": true, "tidy": true, "upgrade_dependency": true, "vendor": true, "vulncheck": false, }, }, }, } var result protocol.InitializeResult if err := c.Call(ctx, "initialize", initParams, &result); err != nil { return nil, fmt.Errorf("initialize failed: %w", err) } if err := c.Notify(ctx, "initialized", struct{}{}); err != nil { return nil, fmt.Errorf("initialized notification failed: %w", err) } // Register handlers c.RegisterServerRequestHandler("workspace/applyEdit", HandleApplyEdit) c.RegisterServerRequestHandler("workspace/configuration", HandleWorkspaceConfiguration) c.RegisterServerRequestHandler("client/registerCapability", HandleRegisterCapability) c.RegisterNotificationHandler("window/showMessage", HandleServerMessage) c.RegisterNotificationHandler("textDocument/publishDiagnostics", func(params json.RawMessage) { HandleDiagnostics(c, params) }) // Notify the LSP server err := c.Initialized(ctx, protocol.InitializedParams{}) if err != nil { return nil, fmt.Errorf("initialization failed: %w", err) } return &result, nil } func (c *Client) Close() error { // Try to close all open files first ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second) defer cancel() // Attempt to close files but continue shutdown regardless c.CloseAllFiles(ctx) // Close stdin to signal the server if err := c.stdin.Close(); err != nil { return fmt.Errorf("failed to close stdin: %w", err) } // Use a channel to handle the Wait with timeout done := make(chan error, 1) go func() { done <- c.Cmd.Wait() }() // Wait for process to exit with timeout select { case err := <-done: return err case <-time.After(2 * time.Second): // If we timeout, try to kill the process if err := c.Cmd.Process.Kill(); err != nil { return fmt.Errorf("failed to kill process: %w", err) } return fmt.Errorf("process killed after timeout") } } type ServerState int const ( StateStarting ServerState = iota StateReady StateError ) // GetServerState returns the current state of the LSP server func (c *Client) GetServerState() ServerState { if val := c.serverState.Load(); val != nil { return val.(ServerState) } return StateStarting } // SetServerState sets the current state of the LSP server func (c *Client) SetServerState(state ServerState) { c.serverState.Store(state) } // WaitForServerReady waits for the server to be ready by polling the server // with a simple request until it responds successfully or times out func (c *Client) WaitForServerReady(ctx context.Context) error { cnf := config.Get() // Set initial state c.SetServerState(StateStarting) // Create a context with timeout ctx, cancel := context.WithTimeout(ctx, 30*time.Second) defer cancel() // Try to ping the server with a simple request ticker := time.NewTicker(500 * time.Millisecond) defer ticker.Stop() if cnf.DebugLSP { logging.Debug("Waiting for LSP server to be ready...") } // Determine server type for specialized initialization serverType := c.detectServerType() // For TypeScript-like servers, we need to open some key files first if serverType == ServerTypeTypeScript { if cnf.DebugLSP { logging.Debug("TypeScript-like server detected, opening key configuration files") } c.openKeyConfigFiles(ctx) } for { select { case <-ctx.Done(): c.SetServerState(StateError) return fmt.Errorf("timeout waiting for LSP server to be ready") case <-ticker.C: // Try a ping method appropriate for this server type err := c.pingServerByType(ctx, serverType) if err == nil { // Server responded successfully c.SetServerState(StateReady) if cnf.DebugLSP { logging.Debug("LSP server is ready") } return nil } else { logging.Debug("LSP server not ready yet", "error", err, "serverType", serverType) } if cnf.DebugLSP { logging.Debug("LSP server not ready yet", "error", err, "serverType", serverType) } } } } // ServerType represents the type of LSP server type ServerType int const ( ServerTypeUnknown ServerType = iota ServerTypeGo ServerTypeTypeScript ServerTypeRust ServerTypePython ServerTypeGeneric ) // detectServerType tries to determine what type of LSP server we're dealing with func (c *Client) detectServerType() ServerType { if c.Cmd == nil { return ServerTypeUnknown } cmdPath := strings.ToLower(c.Cmd.Path) switch { case strings.Contains(cmdPath, "gopls"): return ServerTypeGo case strings.Contains(cmdPath, "typescript") || strings.Contains(cmdPath, "vtsls") || strings.Contains(cmdPath, "tsserver"): return ServerTypeTypeScript case strings.Contains(cmdPath, "rust-analyzer"): return ServerTypeRust case strings.Contains(cmdPath, "pyright") || strings.Contains(cmdPath, "pylsp") || strings.Contains(cmdPath, "python"): return ServerTypePython default: return ServerTypeGeneric } } // openKeyConfigFiles opens important configuration files that help initialize the server func (c *Client) openKeyConfigFiles(ctx context.Context) { workDir := config.WorkingDirectory() serverType := c.detectServerType() var filesToOpen []string switch serverType { case ServerTypeTypeScript: // TypeScript servers need these config files to properly initialize filesToOpen = []string{ filepath.Join(workDir, "tsconfig.json"), filepath.Join(workDir, "package.json"), filepath.Join(workDir, "jsconfig.json"), } // Also find and open a few TypeScript files to help the server initialize c.openTypeScriptFiles(ctx, workDir) case ServerTypeGo: filesToOpen = []string{ filepath.Join(workDir, "go.mod"), filepath.Join(workDir, "go.sum"), } case ServerTypeRust: filesToOpen = []string{ filepath.Join(workDir, "Cargo.toml"), filepath.Join(workDir, "Cargo.lock"), } } // Try to open each file, ignoring errors if they don't exist for _, file := range filesToOpen { if _, err := os.Stat(file); err == nil { // File exists, try to open it if err := c.OpenFile(ctx, file); err != nil { logging.Debug("Failed to open key config file", "file", file, "error", err) } else { logging.Debug("Opened key config file for initialization", "file", file) } } } } // pingServerByType sends a ping request appropriate for the server type func (c *Client) pingServerByType(ctx context.Context, serverType ServerType) error { switch serverType { case ServerTypeTypeScript: // For TypeScript, try a document symbol request on an open file return c.pingTypeScriptServer(ctx) case ServerTypeGo: // For Go, workspace/symbol works well return c.pingWithWorkspaceSymbol(ctx) case ServerTypeRust: // For Rust, workspace/symbol works well return c.pingWithWorkspaceSymbol(ctx) default: // Default ping method return c.pingWithWorkspaceSymbol(ctx) } } // pingTypeScriptServer tries to ping a TypeScript server with appropriate methods func (c *Client) pingTypeScriptServer(ctx context.Context) error { // First try workspace/symbol which works for many servers if err := c.pingWithWorkspaceSymbol(ctx); err == nil { return nil } // If that fails, try to find an open file and request document symbols c.openFilesMu.RLock() defer c.openFilesMu.RUnlock() // If we have any open files, try to get document symbols for one for uri := range c.openFiles { filePath := strings.TrimPrefix(uri, "file://") if strings.HasSuffix(filePath, ".ts") || strings.HasSuffix(filePath, ".js") || strings.HasSuffix(filePath, ".tsx") || strings.HasSuffix(filePath, ".jsx") { var symbols []protocol.DocumentSymbol err := c.Call(ctx, "textDocument/documentSymbol", protocol.DocumentSymbolParams{ TextDocument: protocol.TextDocumentIdentifier{ URI: protocol.DocumentUri(uri), }, }, &symbols) if err == nil { return nil } } } // If we have no open TypeScript files, try to find and open one workDir := config.WorkingDirectory() err := filepath.WalkDir(workDir, func(path string, d os.DirEntry, err error) error { if err != nil { return err } // Skip directories and non-TypeScript files if d.IsDir() { return nil } ext := filepath.Ext(path) if ext == ".ts" || ext == ".js" || ext == ".tsx" || ext == ".jsx" { // Found a TypeScript file, try to open it if err := c.OpenFile(ctx, path); err == nil { // Successfully opened, stop walking return filepath.SkipAll } } return nil }) if err != nil { logging.Debug("Error walking directory for TypeScript files", "error", err) } // Final fallback - just try a generic capability return c.pingWithServerCapabilities(ctx) } // openTypeScriptFiles finds and opens TypeScript files to help initialize the server func (c *Client) openTypeScriptFiles(ctx context.Context, workDir string) { cnf := config.Get() filesOpened := 0 maxFilesToOpen := 5 // Limit to a reasonable number of files // Find and open TypeScript files err := filepath.WalkDir(workDir, func(path string, d os.DirEntry, err error) error { if err != nil { return err } // Skip directories and non-TypeScript files if d.IsDir() { // Skip common directories to avoid wasting time if shouldSkipDir(path) { return filepath.SkipDir } return nil } // Check if we've opened enough files if filesOpened >= maxFilesToOpen { return filepath.SkipAll } // Check file extension ext := filepath.Ext(path) if ext == ".ts" || ext == ".tsx" || ext == ".js" || ext == ".jsx" { // Try to open the file if err := c.OpenFile(ctx, path); err == nil { filesOpened++ if cnf.DebugLSP { logging.Debug("Opened TypeScript file for initialization", "file", path) } } } return nil }) if err != nil && cnf.DebugLSP { logging.Debug("Error walking directory for TypeScript files", "error", err) } if cnf.DebugLSP { logging.Debug("Opened TypeScript files for initialization", "count", filesOpened) } } // shouldSkipDir returns true if the directory should be skipped during file search func shouldSkipDir(path string) bool { dirName := filepath.Base(path) // Skip hidden directories if strings.HasPrefix(dirName, ".") { return true } // Skip common directories that won't contain relevant source files skipDirs := map[string]bool{ "node_modules": true, "dist": true, "build": true, "coverage": true, "vendor": true, "target": true, } return skipDirs[dirName] } // pingWithWorkspaceSymbol tries a workspace/symbol request func (c *Client) pingWithWorkspaceSymbol(ctx context.Context) error { var result []protocol.SymbolInformation return c.Call(ctx, "workspace/symbol", protocol.WorkspaceSymbolParams{ Query: "", }, &result) } // pingWithServerCapabilities tries to get server capabilities func (c *Client) pingWithServerCapabilities(ctx context.Context) error { // This is a very lightweight request that should work for most servers return c.Notify(ctx, "$/cancelRequest", struct{ ID int }{ID: -1}) } type OpenFileInfo struct { Version int32 URI protocol.DocumentUri } func (c *Client) OpenFile(ctx context.Context, filepath string) error { uri := fmt.Sprintf("file://%s", filepath) c.openFilesMu.Lock() if _, exists := c.openFiles[uri]; exists { c.openFilesMu.Unlock() return nil // Already open } c.openFilesMu.Unlock() // Skip files that do not exist or cannot be read content, err := os.ReadFile(filepath) if err != nil { return fmt.Errorf("error reading file: %w", err) } params := protocol.DidOpenTextDocumentParams{ TextDocument: protocol.TextDocumentItem{ URI: protocol.DocumentUri(uri), LanguageID: DetectLanguageID(uri), Version: 1, Text: string(content), }, } if err := c.Notify(ctx, "textDocument/didOpen", params); err != nil { return err } c.openFilesMu.Lock() c.openFiles[uri] = &OpenFileInfo{ Version: 1, URI: protocol.DocumentUri(uri), } c.openFilesMu.Unlock() return nil } func (c *Client) NotifyChange(ctx context.Context, filepath string) error { uri := fmt.Sprintf("file://%s", filepath) content, err := os.ReadFile(filepath) if err != nil { return fmt.Errorf("error reading file: %w", err) } c.openFilesMu.Lock() fileInfo, isOpen := c.openFiles[uri] if !isOpen { c.openFilesMu.Unlock() return fmt.Errorf("cannot notify change for unopened file: %s", filepath) } // Increment version fileInfo.Version++ version := fileInfo.Version c.openFilesMu.Unlock() params := protocol.DidChangeTextDocumentParams{ TextDocument: protocol.VersionedTextDocumentIdentifier{ TextDocumentIdentifier: protocol.TextDocumentIdentifier{ URI: protocol.DocumentUri(uri), }, Version: version, }, ContentChanges: []protocol.TextDocumentContentChangeEvent{ { Value: protocol.TextDocumentContentChangeWholeDocument{ Text: string(content), }, }, }, } return c.Notify(ctx, "textDocument/didChange", params) } func (c *Client) CloseFile(ctx context.Context, filepath string) error { cnf := config.Get() uri := fmt.Sprintf("file://%s", filepath) c.openFilesMu.Lock() if _, exists := c.openFiles[uri]; !exists { c.openFilesMu.Unlock() return nil // Already closed } c.openFilesMu.Unlock() params := protocol.DidCloseTextDocumentParams{ TextDocument: protocol.TextDocumentIdentifier{ URI: protocol.DocumentUri(uri), }, } if cnf.DebugLSP { logging.Debug("Closing file", "file", filepath) } if err := c.Notify(ctx, "textDocument/didClose", params); err != nil { return err } c.openFilesMu.Lock() delete(c.openFiles, uri) c.openFilesMu.Unlock() return nil } func (c *Client) IsFileOpen(filepath string) bool { uri := fmt.Sprintf("file://%s", filepath) c.openFilesMu.RLock() defer c.openFilesMu.RUnlock() _, exists := c.openFiles[uri] return exists } // CloseAllFiles closes all currently open files func (c *Client) CloseAllFiles(ctx context.Context) { cnf := config.Get() c.openFilesMu.Lock() filesToClose := make([]string, 0, len(c.openFiles)) // First collect all URIs that need to be closed for uri := range c.openFiles { // Convert URI back to file path by trimming "file://" prefix filePath := strings.TrimPrefix(uri, "file://") filesToClose = append(filesToClose, filePath) } c.openFilesMu.Unlock() // Then close them all for _, filePath := range filesToClose { err := c.CloseFile(ctx, filePath) if err != nil && cnf.DebugLSP { logging.Warn("Error closing file", "file", filePath, "error", err) } } if cnf.DebugLSP { logging.Debug("Closed all files", "files", filesToClose) } } func (c *Client) GetFileDiagnostics(uri protocol.DocumentUri) []protocol.Diagnostic { c.diagnosticsMu.RLock() defer c.diagnosticsMu.RUnlock() return c.diagnostics[uri] } // GetDiagnostics returns all diagnostics for all files func (c *Client) GetDiagnostics() map[protocol.DocumentUri][]protocol.Diagnostic { return c.diagnostics } // OpenFileOnDemand opens a file only if it's not already open // This is used for lazy-loading files when they're actually needed func (c *Client) OpenFileOnDemand(ctx context.Context, filepath string) error { // Check if the file is already open if c.IsFileOpen(filepath) { return nil } // Open the file return c.OpenFile(ctx, filepath) } // GetDiagnosticsForFile ensures a file is open and returns its diagnostics // This is useful for on-demand diagnostics when using lazy loading func (c *Client) GetDiagnosticsForFile(ctx context.Context, filepath string) ([]protocol.Diagnostic, error) { uri := fmt.Sprintf("file://%s", filepath) documentUri := protocol.DocumentUri(uri) // Make sure the file is open if !c.IsFileOpen(filepath) { if err := c.OpenFile(ctx, filepath); err != nil { return nil, fmt.Errorf("failed to open file for diagnostics: %w", err) } // Give the LSP server a moment to process the file time.Sleep(100 * time.Millisecond) } // Get diagnostics c.diagnosticsMu.RLock() diagnostics := c.diagnostics[documentUri] c.diagnosticsMu.RUnlock() return diagnostics, nil } // ClearDiagnosticsForURI removes diagnostics for a specific URI from the cache func (c *Client) ClearDiagnosticsForURI(uri protocol.DocumentUri) { c.diagnosticsMu.Lock() defer c.diagnosticsMu.Unlock() delete(c.diagnostics, uri) } ================================================ FILE: internal/lsp/handlers.go ================================================ package lsp import ( "encoding/json" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/logging" "github.com/opencode-ai/opencode/internal/lsp/protocol" "github.com/opencode-ai/opencode/internal/lsp/util" ) // Requests func HandleWorkspaceConfiguration(params json.RawMessage) (any, error) { return []map[string]any{{}}, nil } func HandleRegisterCapability(params json.RawMessage) (any, error) { var registerParams protocol.RegistrationParams if err := json.Unmarshal(params, ®isterParams); err != nil { logging.Error("Error unmarshaling registration params", "error", err) return nil, err } for _, reg := range registerParams.Registrations { switch reg.Method { case "workspace/didChangeWatchedFiles": // Parse the registration options optionsJSON, err := json.Marshal(reg.RegisterOptions) if err != nil { logging.Error("Error marshaling registration options", "error", err) continue } var options protocol.DidChangeWatchedFilesRegistrationOptions if err := json.Unmarshal(optionsJSON, &options); err != nil { logging.Error("Error unmarshaling registration options", "error", err) continue } // Store the file watchers registrations notifyFileWatchRegistration(reg.ID, options.Watchers) } } return nil, nil } func HandleApplyEdit(params json.RawMessage) (any, error) { var edit protocol.ApplyWorkspaceEditParams if err := json.Unmarshal(params, &edit); err != nil { return nil, err } err := util.ApplyWorkspaceEdit(edit.Edit) if err != nil { logging.Error("Error applying workspace edit", "error", err) return protocol.ApplyWorkspaceEditResult{Applied: false, FailureReason: err.Error()}, nil } return protocol.ApplyWorkspaceEditResult{Applied: true}, nil } // FileWatchRegistrationHandler is a function that will be called when file watch registrations are received type FileWatchRegistrationHandler func(id string, watchers []protocol.FileSystemWatcher) // fileWatchHandler holds the current handler for file watch registrations var fileWatchHandler FileWatchRegistrationHandler // RegisterFileWatchHandler sets the handler for file watch registrations func RegisterFileWatchHandler(handler FileWatchRegistrationHandler) { fileWatchHandler = handler } // notifyFileWatchRegistration notifies the handler about new file watch registrations func notifyFileWatchRegistration(id string, watchers []protocol.FileSystemWatcher) { if fileWatchHandler != nil { fileWatchHandler(id, watchers) } } // Notifications func HandleServerMessage(params json.RawMessage) { cnf := config.Get() var msg struct { Type int `json:"type"` Message string `json:"message"` } if err := json.Unmarshal(params, &msg); err == nil { if cnf.DebugLSP { logging.Debug("Server message", "type", msg.Type, "message", msg.Message) } } } func HandleDiagnostics(client *Client, params json.RawMessage) { var diagParams protocol.PublishDiagnosticsParams if err := json.Unmarshal(params, &diagParams); err != nil { logging.Error("Error unmarshaling diagnostics params", "error", err) return } client.diagnosticsMu.Lock() defer client.diagnosticsMu.Unlock() client.diagnostics[diagParams.URI] = diagParams.Diagnostics } ================================================ FILE: internal/lsp/language.go ================================================ package lsp import ( "path/filepath" "strings" "github.com/opencode-ai/opencode/internal/lsp/protocol" ) func DetectLanguageID(uri string) protocol.LanguageKind { ext := strings.ToLower(filepath.Ext(uri)) switch ext { case ".abap": return protocol.LangABAP case ".bat": return protocol.LangWindowsBat case ".bib", ".bibtex": return protocol.LangBibTeX case ".clj": return protocol.LangClojure case ".coffee": return protocol.LangCoffeescript case ".c": return protocol.LangC case ".cpp", ".cxx", ".cc", ".c++": return protocol.LangCPP case ".cs": return protocol.LangCSharp case ".css": return protocol.LangCSS case ".d": return protocol.LangD case ".pas", ".pascal": return protocol.LangDelphi case ".diff", ".patch": return protocol.LangDiff case ".dart": return protocol.LangDart case ".dockerfile": return protocol.LangDockerfile case ".ex", ".exs": return protocol.LangElixir case ".erl", ".hrl": return protocol.LangErlang case ".fs", ".fsi", ".fsx", ".fsscript": return protocol.LangFSharp case ".gitcommit": return protocol.LangGitCommit case ".gitrebase": return protocol.LangGitRebase case ".go": return protocol.LangGo case ".groovy": return protocol.LangGroovy case ".hbs", ".handlebars": return protocol.LangHandlebars case ".hs": return protocol.LangHaskell case ".html", ".htm": return protocol.LangHTML case ".ini": return protocol.LangIni case ".java": return protocol.LangJava case ".js": return protocol.LangJavaScript case ".jsx": return protocol.LangJavaScriptReact case ".json": return protocol.LangJSON case ".tex", ".latex": return protocol.LangLaTeX case ".less": return protocol.LangLess case ".lua": return protocol.LangLua case ".makefile", "makefile": return protocol.LangMakefile case ".md", ".markdown": return protocol.LangMarkdown case ".m": return protocol.LangObjectiveC case ".mm": return protocol.LangObjectiveCPP case ".pl": return protocol.LangPerl case ".pm": return protocol.LangPerl6 case ".php": return protocol.LangPHP case ".ps1", ".psm1": return protocol.LangPowershell case ".pug", ".jade": return protocol.LangPug case ".py": return protocol.LangPython case ".r": return protocol.LangR case ".cshtml", ".razor": return protocol.LangRazor case ".rb": return protocol.LangRuby case ".rs": return protocol.LangRust case ".scss": return protocol.LangSCSS case ".sass": return protocol.LangSASS case ".scala": return protocol.LangScala case ".shader": return protocol.LangShaderLab case ".sh", ".bash", ".zsh", ".ksh": return protocol.LangShellScript case ".sql": return protocol.LangSQL case ".swift": return protocol.LangSwift case ".ts": return protocol.LangTypeScript case ".tsx": return protocol.LangTypeScriptReact case ".xml": return protocol.LangXML case ".xsl": return protocol.LangXSL case ".yaml", ".yml": return protocol.LangYAML default: return protocol.LanguageKind("") // Unknown language } } ================================================ FILE: internal/lsp/methods.go ================================================ // Generated code. Do not edit package lsp import ( "context" "github.com/opencode-ai/opencode/internal/lsp/protocol" ) // Implementation sends a textDocument/implementation request to the LSP server. // A request to resolve the implementation locations of a symbol at a given text document position. The request's parameter is of type TextDocumentPositionParams the response is of type Definition or a Thenable that resolves to such. func (c *Client) Implementation(ctx context.Context, params protocol.ImplementationParams) (protocol.Or_Result_textDocument_implementation, error) { var result protocol.Or_Result_textDocument_implementation err := c.Call(ctx, "textDocument/implementation", params, &result) return result, err } // TypeDefinition sends a textDocument/typeDefinition request to the LSP server. // A request to resolve the type definition locations of a symbol at a given text document position. The request's parameter is of type TextDocumentPositionParams the response is of type Definition or a Thenable that resolves to such. func (c *Client) TypeDefinition(ctx context.Context, params protocol.TypeDefinitionParams) (protocol.Or_Result_textDocument_typeDefinition, error) { var result protocol.Or_Result_textDocument_typeDefinition err := c.Call(ctx, "textDocument/typeDefinition", params, &result) return result, err } // DocumentColor sends a textDocument/documentColor request to the LSP server. // A request to list all color symbols found in a given text document. The request's parameter is of type DocumentColorParams the response is of type ColorInformation ColorInformation[] or a Thenable that resolves to such. func (c *Client) DocumentColor(ctx context.Context, params protocol.DocumentColorParams) ([]protocol.ColorInformation, error) { var result []protocol.ColorInformation err := c.Call(ctx, "textDocument/documentColor", params, &result) return result, err } // ColorPresentation sends a textDocument/colorPresentation request to the LSP server. // A request to list all presentation for a color. The request's parameter is of type ColorPresentationParams the response is of type ColorInformation ColorInformation[] or a Thenable that resolves to such. func (c *Client) ColorPresentation(ctx context.Context, params protocol.ColorPresentationParams) ([]protocol.ColorPresentation, error) { var result []protocol.ColorPresentation err := c.Call(ctx, "textDocument/colorPresentation", params, &result) return result, err } // FoldingRange sends a textDocument/foldingRange request to the LSP server. // A request to provide folding ranges in a document. The request's parameter is of type FoldingRangeParams, the response is of type FoldingRangeList or a Thenable that resolves to such. func (c *Client) FoldingRange(ctx context.Context, params protocol.FoldingRangeParams) ([]protocol.FoldingRange, error) { var result []protocol.FoldingRange err := c.Call(ctx, "textDocument/foldingRange", params, &result) return result, err } // Declaration sends a textDocument/declaration request to the LSP server. // A request to resolve the type definition locations of a symbol at a given text document position. The request's parameter is of type TextDocumentPositionParams the response is of type Declaration or a typed array of DeclarationLink or a Thenable that resolves to such. func (c *Client) Declaration(ctx context.Context, params protocol.DeclarationParams) (protocol.Or_Result_textDocument_declaration, error) { var result protocol.Or_Result_textDocument_declaration err := c.Call(ctx, "textDocument/declaration", params, &result) return result, err } // SelectionRange sends a textDocument/selectionRange request to the LSP server. // A request to provide selection ranges in a document. The request's parameter is of type SelectionRangeParams, the response is of type SelectionRange SelectionRange[] or a Thenable that resolves to such. func (c *Client) SelectionRange(ctx context.Context, params protocol.SelectionRangeParams) ([]protocol.SelectionRange, error) { var result []protocol.SelectionRange err := c.Call(ctx, "textDocument/selectionRange", params, &result) return result, err } // PrepareCallHierarchy sends a textDocument/prepareCallHierarchy request to the LSP server. // A request to result a CallHierarchyItem in a document at a given position. Can be used as an input to an incoming or outgoing call hierarchy. Since 3.16.0 func (c *Client) PrepareCallHierarchy(ctx context.Context, params protocol.CallHierarchyPrepareParams) ([]protocol.CallHierarchyItem, error) { var result []protocol.CallHierarchyItem err := c.Call(ctx, "textDocument/prepareCallHierarchy", params, &result) return result, err } // IncomingCalls sends a callHierarchy/incomingCalls request to the LSP server. // A request to resolve the incoming calls for a given CallHierarchyItem. Since 3.16.0 func (c *Client) IncomingCalls(ctx context.Context, params protocol.CallHierarchyIncomingCallsParams) ([]protocol.CallHierarchyIncomingCall, error) { var result []protocol.CallHierarchyIncomingCall err := c.Call(ctx, "callHierarchy/incomingCalls", params, &result) return result, err } // OutgoingCalls sends a callHierarchy/outgoingCalls request to the LSP server. // A request to resolve the outgoing calls for a given CallHierarchyItem. Since 3.16.0 func (c *Client) OutgoingCalls(ctx context.Context, params protocol.CallHierarchyOutgoingCallsParams) ([]protocol.CallHierarchyOutgoingCall, error) { var result []protocol.CallHierarchyOutgoingCall err := c.Call(ctx, "callHierarchy/outgoingCalls", params, &result) return result, err } // SemanticTokensFull sends a textDocument/semanticTokens/full request to the LSP server. // Since 3.16.0 func (c *Client) SemanticTokensFull(ctx context.Context, params protocol.SemanticTokensParams) (protocol.SemanticTokens, error) { var result protocol.SemanticTokens err := c.Call(ctx, "textDocument/semanticTokens/full", params, &result) return result, err } // SemanticTokensFullDelta sends a textDocument/semanticTokens/full/delta request to the LSP server. // Since 3.16.0 func (c *Client) SemanticTokensFullDelta(ctx context.Context, params protocol.SemanticTokensDeltaParams) (protocol.Or_Result_textDocument_semanticTokens_full_delta, error) { var result protocol.Or_Result_textDocument_semanticTokens_full_delta err := c.Call(ctx, "textDocument/semanticTokens/full/delta", params, &result) return result, err } // SemanticTokensRange sends a textDocument/semanticTokens/range request to the LSP server. // Since 3.16.0 func (c *Client) SemanticTokensRange(ctx context.Context, params protocol.SemanticTokensRangeParams) (protocol.SemanticTokens, error) { var result protocol.SemanticTokens err := c.Call(ctx, "textDocument/semanticTokens/range", params, &result) return result, err } // LinkedEditingRange sends a textDocument/linkedEditingRange request to the LSP server. // A request to provide ranges that can be edited together. Since 3.16.0 func (c *Client) LinkedEditingRange(ctx context.Context, params protocol.LinkedEditingRangeParams) (protocol.LinkedEditingRanges, error) { var result protocol.LinkedEditingRanges err := c.Call(ctx, "textDocument/linkedEditingRange", params, &result) return result, err } // WillCreateFiles sends a workspace/willCreateFiles request to the LSP server. // The will create files request is sent from the client to the server before files are actually created as long as the creation is triggered from within the client. The request can return a WorkspaceEdit which will be applied to workspace before the files are created. Hence the WorkspaceEdit can not manipulate the content of the file to be created. Since 3.16.0 func (c *Client) WillCreateFiles(ctx context.Context, params protocol.CreateFilesParams) (protocol.WorkspaceEdit, error) { var result protocol.WorkspaceEdit err := c.Call(ctx, "workspace/willCreateFiles", params, &result) return result, err } // WillRenameFiles sends a workspace/willRenameFiles request to the LSP server. // The will rename files request is sent from the client to the server before files are actually renamed as long as the rename is triggered from within the client. Since 3.16.0 func (c *Client) WillRenameFiles(ctx context.Context, params protocol.RenameFilesParams) (protocol.WorkspaceEdit, error) { var result protocol.WorkspaceEdit err := c.Call(ctx, "workspace/willRenameFiles", params, &result) return result, err } // WillDeleteFiles sends a workspace/willDeleteFiles request to the LSP server. // The did delete files notification is sent from the client to the server when files were deleted from within the client. Since 3.16.0 func (c *Client) WillDeleteFiles(ctx context.Context, params protocol.DeleteFilesParams) (protocol.WorkspaceEdit, error) { var result protocol.WorkspaceEdit err := c.Call(ctx, "workspace/willDeleteFiles", params, &result) return result, err } // Moniker sends a textDocument/moniker request to the LSP server. // A request to get the moniker of a symbol at a given text document position. The request parameter is of type TextDocumentPositionParams. The response is of type Moniker Moniker[] or null. func (c *Client) Moniker(ctx context.Context, params protocol.MonikerParams) ([]protocol.Moniker, error) { var result []protocol.Moniker err := c.Call(ctx, "textDocument/moniker", params, &result) return result, err } // PrepareTypeHierarchy sends a textDocument/prepareTypeHierarchy request to the LSP server. // A request to result a TypeHierarchyItem in a document at a given position. Can be used as an input to a subtypes or supertypes type hierarchy. Since 3.17.0 func (c *Client) PrepareTypeHierarchy(ctx context.Context, params protocol.TypeHierarchyPrepareParams) ([]protocol.TypeHierarchyItem, error) { var result []protocol.TypeHierarchyItem err := c.Call(ctx, "textDocument/prepareTypeHierarchy", params, &result) return result, err } // Supertypes sends a typeHierarchy/supertypes request to the LSP server. // A request to resolve the supertypes for a given TypeHierarchyItem. Since 3.17.0 func (c *Client) Supertypes(ctx context.Context, params protocol.TypeHierarchySupertypesParams) ([]protocol.TypeHierarchyItem, error) { var result []protocol.TypeHierarchyItem err := c.Call(ctx, "typeHierarchy/supertypes", params, &result) return result, err } // Subtypes sends a typeHierarchy/subtypes request to the LSP server. // A request to resolve the subtypes for a given TypeHierarchyItem. Since 3.17.0 func (c *Client) Subtypes(ctx context.Context, params protocol.TypeHierarchySubtypesParams) ([]protocol.TypeHierarchyItem, error) { var result []protocol.TypeHierarchyItem err := c.Call(ctx, "typeHierarchy/subtypes", params, &result) return result, err } // InlineValue sends a textDocument/inlineValue request to the LSP server. // A request to provide inline values in a document. The request's parameter is of type InlineValueParams, the response is of type InlineValue InlineValue[] or a Thenable that resolves to such. Since 3.17.0 func (c *Client) InlineValue(ctx context.Context, params protocol.InlineValueParams) ([]protocol.InlineValue, error) { var result []protocol.InlineValue err := c.Call(ctx, "textDocument/inlineValue", params, &result) return result, err } // InlayHint sends a textDocument/inlayHint request to the LSP server. // A request to provide inlay hints in a document. The request's parameter is of type InlayHintsParams, the response is of type InlayHint InlayHint[] or a Thenable that resolves to such. Since 3.17.0 func (c *Client) InlayHint(ctx context.Context, params protocol.InlayHintParams) ([]protocol.InlayHint, error) { var result []protocol.InlayHint err := c.Call(ctx, "textDocument/inlayHint", params, &result) return result, err } // Resolve sends a inlayHint/resolve request to the LSP server. // A request to resolve additional properties for an inlay hint. The request's parameter is of type InlayHint, the response is of type InlayHint or a Thenable that resolves to such. Since 3.17.0 func (c *Client) Resolve(ctx context.Context, params protocol.InlayHint) (protocol.InlayHint, error) { var result protocol.InlayHint err := c.Call(ctx, "inlayHint/resolve", params, &result) return result, err } // Diagnostic sends a textDocument/diagnostic request to the LSP server. // The document diagnostic request definition. Since 3.17.0 func (c *Client) Diagnostic(ctx context.Context, params protocol.DocumentDiagnosticParams) (protocol.DocumentDiagnosticReport, error) { var result protocol.DocumentDiagnosticReport err := c.Call(ctx, "textDocument/diagnostic", params, &result) return result, err } // DiagnosticWorkspace sends a workspace/diagnostic request to the LSP server. // The workspace diagnostic request definition. Since 3.17.0 func (c *Client) DiagnosticWorkspace(ctx context.Context, params protocol.WorkspaceDiagnosticParams) (protocol.WorkspaceDiagnosticReport, error) { var result protocol.WorkspaceDiagnosticReport err := c.Call(ctx, "workspace/diagnostic", params, &result) return result, err } // InlineCompletion sends a textDocument/inlineCompletion request to the LSP server. // A request to provide inline completions in a document. The request's parameter is of type InlineCompletionParams, the response is of type InlineCompletion InlineCompletion[] or a Thenable that resolves to such. Since 3.18.0 PROPOSED func (c *Client) InlineCompletion(ctx context.Context, params protocol.InlineCompletionParams) (protocol.Or_Result_textDocument_inlineCompletion, error) { var result protocol.Or_Result_textDocument_inlineCompletion err := c.Call(ctx, "textDocument/inlineCompletion", params, &result) return result, err } // TextDocumentContent sends a workspace/textDocumentContent request to the LSP server. // The workspace/textDocumentContent request is sent from the client to the server to request the content of a text document. Since 3.18.0 PROPOSED func (c *Client) TextDocumentContent(ctx context.Context, params protocol.TextDocumentContentParams) (string, error) { var result string err := c.Call(ctx, "workspace/textDocumentContent", params, &result) return result, err } // Initialize sends a initialize request to the LSP server. // The initialize request is sent from the client to the server. It is sent once as the request after starting up the server. The requests parameter is of type InitializeParams the response if of type InitializeResult of a Thenable that resolves to such. func (c *Client) Initialize(ctx context.Context, params protocol.ParamInitialize) (protocol.InitializeResult, error) { var result protocol.InitializeResult err := c.Call(ctx, "initialize", params, &result) return result, err } // Shutdown sends a shutdown request to the LSP server. // A shutdown request is sent from the client to the server. It is sent once when the client decides to shutdown the server. The only notification that is sent after a shutdown request is the exit event. func (c *Client) Shutdown(ctx context.Context) error { return c.Call(ctx, "shutdown", nil, nil) } // WillSaveWaitUntil sends a textDocument/willSaveWaitUntil request to the LSP server. // A document will save request is sent from the client to the server before the document is actually saved. The request can return an array of TextEdits which will be applied to the text document before it is saved. Please note that clients might drop results if computing the text edits took too long or if a server constantly fails on this request. This is done to keep the save fast and reliable. func (c *Client) WillSaveWaitUntil(ctx context.Context, params protocol.WillSaveTextDocumentParams) ([]protocol.TextEdit, error) { var result []protocol.TextEdit err := c.Call(ctx, "textDocument/willSaveWaitUntil", params, &result) return result, err } // Completion sends a textDocument/completion request to the LSP server. // Request to request completion at a given text document position. The request's parameter is of type TextDocumentPosition the response is of type CompletionItem CompletionItem[] or CompletionList or a Thenable that resolves to such. The request can delay the computation of the CompletionItem.detail detail and CompletionItem.documentation documentation properties to the completionItem/resolve request. However, properties that are needed for the initial sorting and filtering, like sortText, filterText, insertText, and textEdit, must not be changed during resolve. func (c *Client) Completion(ctx context.Context, params protocol.CompletionParams) (protocol.Or_Result_textDocument_completion, error) { var result protocol.Or_Result_textDocument_completion err := c.Call(ctx, "textDocument/completion", params, &result) return result, err } // ResolveCompletionItem sends a completionItem/resolve request to the LSP server. // Request to resolve additional information for a given completion item.The request's parameter is of type CompletionItem the response is of type CompletionItem or a Thenable that resolves to such. func (c *Client) ResolveCompletionItem(ctx context.Context, params protocol.CompletionItem) (protocol.CompletionItem, error) { var result protocol.CompletionItem err := c.Call(ctx, "completionItem/resolve", params, &result) return result, err } // Hover sends a textDocument/hover request to the LSP server. // Request to request hover information at a given text document position. The request's parameter is of type TextDocumentPosition the response is of type Hover or a Thenable that resolves to such. func (c *Client) Hover(ctx context.Context, params protocol.HoverParams) (protocol.Hover, error) { var result protocol.Hover err := c.Call(ctx, "textDocument/hover", params, &result) return result, err } // SignatureHelp sends a textDocument/signatureHelp request to the LSP server. func (c *Client) SignatureHelp(ctx context.Context, params protocol.SignatureHelpParams) (protocol.SignatureHelp, error) { var result protocol.SignatureHelp err := c.Call(ctx, "textDocument/signatureHelp", params, &result) return result, err } // Definition sends a textDocument/definition request to the LSP server. // A request to resolve the definition location of a symbol at a given text document position. The request's parameter is of type TextDocumentPosition the response is of either type Definition or a typed array of DefinitionLink or a Thenable that resolves to such. func (c *Client) Definition(ctx context.Context, params protocol.DefinitionParams) (protocol.Or_Result_textDocument_definition, error) { var result protocol.Or_Result_textDocument_definition err := c.Call(ctx, "textDocument/definition", params, &result) return result, err } // References sends a textDocument/references request to the LSP server. // A request to resolve project-wide references for the symbol denoted by the given text document position. The request's parameter is of type ReferenceParams the response is of type Location Location[] or a Thenable that resolves to such. func (c *Client) References(ctx context.Context, params protocol.ReferenceParams) ([]protocol.Location, error) { var result []protocol.Location err := c.Call(ctx, "textDocument/references", params, &result) return result, err } // DocumentHighlight sends a textDocument/documentHighlight request to the LSP server. // Request to resolve a DocumentHighlight for a given text document position. The request's parameter is of type TextDocumentPosition the request response is an array of type DocumentHighlight or a Thenable that resolves to such. func (c *Client) DocumentHighlight(ctx context.Context, params protocol.DocumentHighlightParams) ([]protocol.DocumentHighlight, error) { var result []protocol.DocumentHighlight err := c.Call(ctx, "textDocument/documentHighlight", params, &result) return result, err } // DocumentSymbol sends a textDocument/documentSymbol request to the LSP server. // A request to list all symbols found in a given text document. The request's parameter is of type TextDocumentIdentifier the response is of type SymbolInformation SymbolInformation[] or a Thenable that resolves to such. func (c *Client) DocumentSymbol(ctx context.Context, params protocol.DocumentSymbolParams) (protocol.Or_Result_textDocument_documentSymbol, error) { var result protocol.Or_Result_textDocument_documentSymbol err := c.Call(ctx, "textDocument/documentSymbol", params, &result) return result, err } // CodeAction sends a textDocument/codeAction request to the LSP server. // A request to provide commands for the given text document and range. func (c *Client) CodeAction(ctx context.Context, params protocol.CodeActionParams) ([]protocol.Or_Result_textDocument_codeAction_Item0_Elem, error) { var result []protocol.Or_Result_textDocument_codeAction_Item0_Elem err := c.Call(ctx, "textDocument/codeAction", params, &result) return result, err } // ResolveCodeAction sends a codeAction/resolve request to the LSP server. // Request to resolve additional information for a given code action.The request's parameter is of type CodeAction the response is of type CodeAction or a Thenable that resolves to such. func (c *Client) ResolveCodeAction(ctx context.Context, params protocol.CodeAction) (protocol.CodeAction, error) { var result protocol.CodeAction err := c.Call(ctx, "codeAction/resolve", params, &result) return result, err } // Symbol sends a workspace/symbol request to the LSP server. // A request to list project-wide symbols matching the query string given by the WorkspaceSymbolParams. The response is of type SymbolInformation SymbolInformation[] or a Thenable that resolves to such. Since 3.17.0 - support for WorkspaceSymbol in the returned data. Clients need to advertise support for WorkspaceSymbols via the client capability workspace.symbol.resolveSupport. func (c *Client) Symbol(ctx context.Context, params protocol.WorkspaceSymbolParams) (protocol.Or_Result_workspace_symbol, error) { var result protocol.Or_Result_workspace_symbol err := c.Call(ctx, "workspace/symbol", params, &result) return result, err } // ResolveWorkspaceSymbol sends a workspaceSymbol/resolve request to the LSP server. // A request to resolve the range inside the workspace symbol's location. Since 3.17.0 func (c *Client) ResolveWorkspaceSymbol(ctx context.Context, params protocol.WorkspaceSymbol) (protocol.WorkspaceSymbol, error) { var result protocol.WorkspaceSymbol err := c.Call(ctx, "workspaceSymbol/resolve", params, &result) return result, err } // CodeLens sends a textDocument/codeLens request to the LSP server. // A request to provide code lens for the given text document. func (c *Client) CodeLens(ctx context.Context, params protocol.CodeLensParams) ([]protocol.CodeLens, error) { var result []protocol.CodeLens err := c.Call(ctx, "textDocument/codeLens", params, &result) return result, err } // ResolveCodeLens sends a codeLens/resolve request to the LSP server. // A request to resolve a command for a given code lens. func (c *Client) ResolveCodeLens(ctx context.Context, params protocol.CodeLens) (protocol.CodeLens, error) { var result protocol.CodeLens err := c.Call(ctx, "codeLens/resolve", params, &result) return result, err } // DocumentLink sends a textDocument/documentLink request to the LSP server. // A request to provide document links func (c *Client) DocumentLink(ctx context.Context, params protocol.DocumentLinkParams) ([]protocol.DocumentLink, error) { var result []protocol.DocumentLink err := c.Call(ctx, "textDocument/documentLink", params, &result) return result, err } // ResolveDocumentLink sends a documentLink/resolve request to the LSP server. // Request to resolve additional information for a given document link. The request's parameter is of type DocumentLink the response is of type DocumentLink or a Thenable that resolves to such. func (c *Client) ResolveDocumentLink(ctx context.Context, params protocol.DocumentLink) (protocol.DocumentLink, error) { var result protocol.DocumentLink err := c.Call(ctx, "documentLink/resolve", params, &result) return result, err } // Formatting sends a textDocument/formatting request to the LSP server. // A request to format a whole document. func (c *Client) Formatting(ctx context.Context, params protocol.DocumentFormattingParams) ([]protocol.TextEdit, error) { var result []protocol.TextEdit err := c.Call(ctx, "textDocument/formatting", params, &result) return result, err } // RangeFormatting sends a textDocument/rangeFormatting request to the LSP server. // A request to format a range in a document. func (c *Client) RangeFormatting(ctx context.Context, params protocol.DocumentRangeFormattingParams) ([]protocol.TextEdit, error) { var result []protocol.TextEdit err := c.Call(ctx, "textDocument/rangeFormatting", params, &result) return result, err } // RangesFormatting sends a textDocument/rangesFormatting request to the LSP server. // A request to format ranges in a document. Since 3.18.0 PROPOSED func (c *Client) RangesFormatting(ctx context.Context, params protocol.DocumentRangesFormattingParams) ([]protocol.TextEdit, error) { var result []protocol.TextEdit err := c.Call(ctx, "textDocument/rangesFormatting", params, &result) return result, err } // OnTypeFormatting sends a textDocument/onTypeFormatting request to the LSP server. // A request to format a document on type. func (c *Client) OnTypeFormatting(ctx context.Context, params protocol.DocumentOnTypeFormattingParams) ([]protocol.TextEdit, error) { var result []protocol.TextEdit err := c.Call(ctx, "textDocument/onTypeFormatting", params, &result) return result, err } // Rename sends a textDocument/rename request to the LSP server. // A request to rename a symbol. func (c *Client) Rename(ctx context.Context, params protocol.RenameParams) (protocol.WorkspaceEdit, error) { var result protocol.WorkspaceEdit err := c.Call(ctx, "textDocument/rename", params, &result) return result, err } // PrepareRename sends a textDocument/prepareRename request to the LSP server. // A request to test and perform the setup necessary for a rename. Since 3.16 - support for default behavior func (c *Client) PrepareRename(ctx context.Context, params protocol.PrepareRenameParams) (protocol.PrepareRenameResult, error) { var result protocol.PrepareRenameResult err := c.Call(ctx, "textDocument/prepareRename", params, &result) return result, err } // ExecuteCommand sends a workspace/executeCommand request to the LSP server. // A request send from the client to the server to execute a command. The request might return a workspace edit which the client will apply to the workspace. func (c *Client) ExecuteCommand(ctx context.Context, params protocol.ExecuteCommandParams) (any, error) { var result any err := c.Call(ctx, "workspace/executeCommand", params, &result) return result, err } // DidChangeWorkspaceFolders sends a workspace/didChangeWorkspaceFolders notification to the LSP server. // The workspace/didChangeWorkspaceFolders notification is sent from the client to the server when the workspace folder configuration changes. func (c *Client) DidChangeWorkspaceFolders(ctx context.Context, params protocol.DidChangeWorkspaceFoldersParams) error { return c.Notify(ctx, "workspace/didChangeWorkspaceFolders", params) } // WorkDoneProgressCancel sends a window/workDoneProgress/cancel notification to the LSP server. // The window/workDoneProgress/cancel notification is sent from the client to the server to cancel a progress initiated on the server side. func (c *Client) WorkDoneProgressCancel(ctx context.Context, params protocol.WorkDoneProgressCancelParams) error { return c.Notify(ctx, "window/workDoneProgress/cancel", params) } // DidCreateFiles sends a workspace/didCreateFiles notification to the LSP server. // The did create files notification is sent from the client to the server when files were created from within the client. Since 3.16.0 func (c *Client) DidCreateFiles(ctx context.Context, params protocol.CreateFilesParams) error { return c.Notify(ctx, "workspace/didCreateFiles", params) } // DidRenameFiles sends a workspace/didRenameFiles notification to the LSP server. // The did rename files notification is sent from the client to the server when files were renamed from within the client. Since 3.16.0 func (c *Client) DidRenameFiles(ctx context.Context, params protocol.RenameFilesParams) error { return c.Notify(ctx, "workspace/didRenameFiles", params) } // DidDeleteFiles sends a workspace/didDeleteFiles notification to the LSP server. // The will delete files request is sent from the client to the server before files are actually deleted as long as the deletion is triggered from within the client. Since 3.16.0 func (c *Client) DidDeleteFiles(ctx context.Context, params protocol.DeleteFilesParams) error { return c.Notify(ctx, "workspace/didDeleteFiles", params) } // DidOpenNotebookDocument sends a notebookDocument/didOpen notification to the LSP server. // A notification sent when a notebook opens. Since 3.17.0 func (c *Client) DidOpenNotebookDocument(ctx context.Context, params protocol.DidOpenNotebookDocumentParams) error { return c.Notify(ctx, "notebookDocument/didOpen", params) } // DidChangeNotebookDocument sends a notebookDocument/didChange notification to the LSP server. func (c *Client) DidChangeNotebookDocument(ctx context.Context, params protocol.DidChangeNotebookDocumentParams) error { return c.Notify(ctx, "notebookDocument/didChange", params) } // DidSaveNotebookDocument sends a notebookDocument/didSave notification to the LSP server. // A notification sent when a notebook document is saved. Since 3.17.0 func (c *Client) DidSaveNotebookDocument(ctx context.Context, params protocol.DidSaveNotebookDocumentParams) error { return c.Notify(ctx, "notebookDocument/didSave", params) } // DidCloseNotebookDocument sends a notebookDocument/didClose notification to the LSP server. // A notification sent when a notebook closes. Since 3.17.0 func (c *Client) DidCloseNotebookDocument(ctx context.Context, params protocol.DidCloseNotebookDocumentParams) error { return c.Notify(ctx, "notebookDocument/didClose", params) } // Initialized sends a initialized notification to the LSP server. // The initialized notification is sent from the client to the server after the client is fully initialized and the server is allowed to send requests from the server to the client. func (c *Client) Initialized(ctx context.Context, params protocol.InitializedParams) error { return c.Notify(ctx, "initialized", params) } // Exit sends a exit notification to the LSP server. // The exit event is sent from the client to the server to ask the server to exit its process. func (c *Client) Exit(ctx context.Context) error { return c.Notify(ctx, "exit", nil) } // DidChangeConfiguration sends a workspace/didChangeConfiguration notification to the LSP server. // The configuration change notification is sent from the client to the server when the client's configuration has changed. The notification contains the changed configuration as defined by the language client. func (c *Client) DidChangeConfiguration(ctx context.Context, params protocol.DidChangeConfigurationParams) error { return c.Notify(ctx, "workspace/didChangeConfiguration", params) } // DidOpen sends a textDocument/didOpen notification to the LSP server. // The document open notification is sent from the client to the server to signal newly opened text documents. The document's truth is now managed by the client and the server must not try to read the document's truth using the document's uri. Open in this sense means it is managed by the client. It doesn't necessarily mean that its content is presented in an editor. An open notification must not be sent more than once without a corresponding close notification send before. This means open and close notification must be balanced and the max open count is one. func (c *Client) DidOpen(ctx context.Context, params protocol.DidOpenTextDocumentParams) error { return c.Notify(ctx, "textDocument/didOpen", params) } // DidChange sends a textDocument/didChange notification to the LSP server. // The document change notification is sent from the client to the server to signal changes to a text document. func (c *Client) DidChange(ctx context.Context, params protocol.DidChangeTextDocumentParams) error { return c.Notify(ctx, "textDocument/didChange", params) } // DidClose sends a textDocument/didClose notification to the LSP server. // The document close notification is sent from the client to the server when the document got closed in the client. The document's truth now exists where the document's uri points to (e.g. if the document's uri is a file uri the truth now exists on disk). As with the open notification the close notification is about managing the document's content. Receiving a close notification doesn't mean that the document was open in an editor before. A close notification requires a previous open notification to be sent. func (c *Client) DidClose(ctx context.Context, params protocol.DidCloseTextDocumentParams) error { return c.Notify(ctx, "textDocument/didClose", params) } // DidSave sends a textDocument/didSave notification to the LSP server. // The document save notification is sent from the client to the server when the document got saved in the client. func (c *Client) DidSave(ctx context.Context, params protocol.DidSaveTextDocumentParams) error { return c.Notify(ctx, "textDocument/didSave", params) } // WillSave sends a textDocument/willSave notification to the LSP server. // A document will save notification is sent from the client to the server before the document is actually saved. func (c *Client) WillSave(ctx context.Context, params protocol.WillSaveTextDocumentParams) error { return c.Notify(ctx, "textDocument/willSave", params) } // DidChangeWatchedFiles sends a workspace/didChangeWatchedFiles notification to the LSP server. // The watched files notification is sent from the client to the server when the client detects changes to file watched by the language client. func (c *Client) DidChangeWatchedFiles(ctx context.Context, params protocol.DidChangeWatchedFilesParams) error { return c.Notify(ctx, "workspace/didChangeWatchedFiles", params) } // SetTrace sends a $/setTrace notification to the LSP server. func (c *Client) SetTrace(ctx context.Context, params protocol.SetTraceParams) error { return c.Notify(ctx, "$/setTrace", params) } // Progress sends a $/progress notification to the LSP server. func (c *Client) Progress(ctx context.Context, params protocol.ProgressParams) error { return c.Notify(ctx, "$/progress", params) } ================================================ FILE: internal/lsp/protocol/LICENSE ================================================ Copyright 2009 The Go Authors. Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: * Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. * Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. * Neither the name of Google LLC nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. ================================================ FILE: internal/lsp/protocol/interface.go ================================================ package protocol import "fmt" // TextEditResult is an interface for types that represent workspace symbols type WorkspaceSymbolResult interface { GetName() string GetLocation() Location isWorkspaceSymbol() // marker method } func (ws *WorkspaceSymbol) GetName() string { return ws.Name } func (ws *WorkspaceSymbol) GetLocation() Location { switch v := ws.Location.Value.(type) { case Location: return v case LocationUriOnly: return Location{URI: v.URI} } return Location{} } func (ws *WorkspaceSymbol) isWorkspaceSymbol() {} func (si *SymbolInformation) GetName() string { return si.Name } func (si *SymbolInformation) GetLocation() Location { return si.Location } func (si *SymbolInformation) isWorkspaceSymbol() {} // Results converts the Value to a slice of WorkspaceSymbolResult func (r Or_Result_workspace_symbol) Results() ([]WorkspaceSymbolResult, error) { if r.Value == nil { return make([]WorkspaceSymbolResult, 0), nil } switch v := r.Value.(type) { case []WorkspaceSymbol: results := make([]WorkspaceSymbolResult, len(v)) for i := range v { results[i] = &v[i] } return results, nil case []SymbolInformation: results := make([]WorkspaceSymbolResult, len(v)) for i := range v { results[i] = &v[i] } return results, nil default: return nil, fmt.Errorf("unknown symbol type: %T", r.Value) } } // TextEditResult is an interface for types that represent document symbols type DocumentSymbolResult interface { GetRange() Range GetName() string isDocumentSymbol() // marker method } func (ds *DocumentSymbol) GetRange() Range { return ds.Range } func (ds *DocumentSymbol) GetName() string { return ds.Name } func (ds *DocumentSymbol) isDocumentSymbol() {} func (si *SymbolInformation) GetRange() Range { return si.Location.Range } // Note: SymbolInformation already has GetName() implemented above func (si *SymbolInformation) isDocumentSymbol() {} // Results converts the Value to a slice of DocumentSymbolResult func (r Or_Result_textDocument_documentSymbol) Results() ([]DocumentSymbolResult, error) { if r.Value == nil { return make([]DocumentSymbolResult, 0), nil } switch v := r.Value.(type) { case []DocumentSymbol: results := make([]DocumentSymbolResult, len(v)) for i := range v { results[i] = &v[i] } return results, nil case []SymbolInformation: results := make([]DocumentSymbolResult, len(v)) for i := range v { results[i] = &v[i] } return results, nil default: return nil, fmt.Errorf("unknown document symbol type: %T", v) } } // TextEditResult is an interface for types that can be used as text edits type TextEditResult interface { GetRange() Range GetNewText() string isTextEdit() // marker method } func (te *TextEdit) GetRange() Range { return te.Range } func (te *TextEdit) GetNewText() string { return te.NewText } func (te *TextEdit) isTextEdit() {} // Convert Or_TextDocumentEdit_edits_Elem to TextEdit func (e Or_TextDocumentEdit_edits_Elem) AsTextEdit() (TextEdit, error) { if e.Value == nil { return TextEdit{}, fmt.Errorf("nil text edit") } switch v := e.Value.(type) { case TextEdit: return v, nil case AnnotatedTextEdit: return TextEdit{ Range: v.Range, NewText: v.NewText, }, nil default: return TextEdit{}, fmt.Errorf("unknown text edit type: %T", e.Value) } } ================================================ FILE: internal/lsp/protocol/pattern_interfaces.go ================================================ package protocol import ( "fmt" "strings" ) // PatternInfo is an interface for types that represent glob patterns type PatternInfo interface { GetPattern() string GetBasePath() string isPattern() // marker method } // StringPattern implements PatternInfo for string patterns type StringPattern struct { Pattern string } func (p StringPattern) GetPattern() string { return p.Pattern } func (p StringPattern) GetBasePath() string { return "" } func (p StringPattern) isPattern() {} // RelativePatternInfo implements PatternInfo for RelativePattern type RelativePatternInfo struct { RP RelativePattern BasePath string } func (p RelativePatternInfo) GetPattern() string { return string(p.RP.Pattern) } func (p RelativePatternInfo) GetBasePath() string { return p.BasePath } func (p RelativePatternInfo) isPattern() {} // AsPattern converts GlobPattern to a PatternInfo object func (g *GlobPattern) AsPattern() (PatternInfo, error) { if g.Value == nil { return nil, fmt.Errorf("nil pattern") } switch v := g.Value.(type) { case string: return StringPattern{Pattern: v}, nil case RelativePattern: // Handle BaseURI which could be string or DocumentUri basePath := "" switch baseURI := v.BaseURI.Value.(type) { case string: basePath = strings.TrimPrefix(baseURI, "file://") case DocumentUri: basePath = strings.TrimPrefix(string(baseURI), "file://") default: return nil, fmt.Errorf("unknown BaseURI type: %T", v.BaseURI.Value) } return RelativePatternInfo{RP: v, BasePath: basePath}, nil default: return nil, fmt.Errorf("unknown pattern type: %T", g.Value) } } ================================================ FILE: internal/lsp/protocol/tables.go ================================================ package protocol var TableKindMap = map[SymbolKind]string{ File: "File", Module: "Module", Namespace: "Namespace", Package: "Package", Class: "Class", Method: "Method", Property: "Property", Field: "Field", Constructor: "Constructor", Enum: "Enum", Interface: "Interface", Function: "Function", Variable: "Variable", Constant: "Constant", String: "String", Number: "Number", Boolean: "Boolean", Array: "Array", Object: "Object", Key: "Key", Null: "Null", EnumMember: "EnumMember", Struct: "Struct", Event: "Event", Operator: "Operator", TypeParameter: "TypeParameter", } ================================================ FILE: internal/lsp/protocol/tsdocument-changes.go ================================================ // Copyright 2022 The Go Authors. All rights reserved. // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file. package protocol import ( "encoding/json" "fmt" ) // DocumentChange is a union of various file edit operations. // // Exactly one field of this struct is non-nil; see [DocumentChange.Valid]. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification/#resourceChanges type DocumentChange struct { TextDocumentEdit *TextDocumentEdit CreateFile *CreateFile RenameFile *RenameFile DeleteFile *DeleteFile } // Valid reports whether the DocumentChange sum-type value is valid, // that is, exactly one of create, delete, edit, or rename. func (ch DocumentChange) Valid() bool { n := 0 if ch.TextDocumentEdit != nil { n++ } if ch.CreateFile != nil { n++ } if ch.RenameFile != nil { n++ } if ch.DeleteFile != nil { n++ } return n == 1 } func (d *DocumentChange) UnmarshalJSON(data []byte) error { var m map[string]any if err := json.Unmarshal(data, &m); err != nil { return err } if _, ok := m["textDocument"]; ok { d.TextDocumentEdit = new(TextDocumentEdit) return json.Unmarshal(data, d.TextDocumentEdit) } // The {Create,Rename,Delete}File types all share a 'kind' field. kind := m["kind"] switch kind { case "create": d.CreateFile = new(CreateFile) return json.Unmarshal(data, d.CreateFile) case "rename": d.RenameFile = new(RenameFile) return json.Unmarshal(data, d.RenameFile) case "delete": d.DeleteFile = new(DeleteFile) return json.Unmarshal(data, d.DeleteFile) } return fmt.Errorf("DocumentChanges: unexpected kind: %q", kind) } func (d *DocumentChange) MarshalJSON() ([]byte, error) { if d.TextDocumentEdit != nil { return json.Marshal(d.TextDocumentEdit) } else if d.CreateFile != nil { return json.Marshal(d.CreateFile) } else if d.RenameFile != nil { return json.Marshal(d.RenameFile) } else if d.DeleteFile != nil { return json.Marshal(d.DeleteFile) } return nil, fmt.Errorf("empty DocumentChanges union value") } ================================================ FILE: internal/lsp/protocol/tsjson.go ================================================ // Copyright 2023 The Go Authors. All rights reserved. // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file. // Code generated for LSP. DO NOT EDIT. package protocol // Code generated from protocol/metaModel.json at ref release/protocol/3.17.6-next.9 (hash c94395b5da53729e6dff931293b051009ccaaaa4). // https://github.com/microsoft/vscode-languageserver-node/blob/release/protocol/3.17.6-next.9/protocol/metaModel.json // LSP metaData.version = 3.17.0. import "bytes" import "encoding/json" import "fmt" // UnmarshalError indicates that a JSON value did not conform to // one of the expected cases of an LSP union type. type UnmarshalError struct { msg string } func (e UnmarshalError) Error() string { return e.msg } func (t Or_CancelParams_id) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case int32: return json.Marshal(x) case string: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [int32 string]", t) } func (t *Or_CancelParams_id) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder41 := json.NewDecoder(bytes.NewReader(x)) decoder41.DisallowUnknownFields() var int32Val int32 if err := decoder41.Decode(&int32Val); err == nil { t.Value = int32Val return nil } decoder42 := json.NewDecoder(bytes.NewReader(x)) decoder42.DisallowUnknownFields() var stringVal string if err := decoder42.Decode(&stringVal); err == nil { t.Value = stringVal return nil } return &UnmarshalError{"unmarshal failed to match one of [int32 string]"} } func (t Or_ClientSemanticTokensRequestOptions_full) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case ClientSemanticTokensRequestFullDelta: return json.Marshal(x) case bool: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [ClientSemanticTokensRequestFullDelta bool]", t) } func (t *Or_ClientSemanticTokensRequestOptions_full) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder220 := json.NewDecoder(bytes.NewReader(x)) decoder220.DisallowUnknownFields() var boolVal bool if err := decoder220.Decode(&boolVal); err == nil { t.Value = boolVal return nil } decoder221 := json.NewDecoder(bytes.NewReader(x)) decoder221.DisallowUnknownFields() var h221 ClientSemanticTokensRequestFullDelta if err := decoder221.Decode(&h221); err == nil { t.Value = h221 return nil } return &UnmarshalError{"unmarshal failed to match one of [ClientSemanticTokensRequestFullDelta bool]"} } func (t Or_ClientSemanticTokensRequestOptions_range) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case Lit_ClientSemanticTokensRequestOptions_range_Item1: return json.Marshal(x) case bool: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [Lit_ClientSemanticTokensRequestOptions_range_Item1 bool]", t) } func (t *Or_ClientSemanticTokensRequestOptions_range) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder217 := json.NewDecoder(bytes.NewReader(x)) decoder217.DisallowUnknownFields() var boolVal bool if err := decoder217.Decode(&boolVal); err == nil { t.Value = boolVal return nil } decoder218 := json.NewDecoder(bytes.NewReader(x)) decoder218.DisallowUnknownFields() var h218 Lit_ClientSemanticTokensRequestOptions_range_Item1 if err := decoder218.Decode(&h218); err == nil { t.Value = h218 return nil } return &UnmarshalError{"unmarshal failed to match one of [Lit_ClientSemanticTokensRequestOptions_range_Item1 bool]"} } func (t Or_CompletionItemDefaults_editRange) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case EditRangeWithInsertReplace: return json.Marshal(x) case Range: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [EditRangeWithInsertReplace Range]", t) } func (t *Or_CompletionItemDefaults_editRange) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder183 := json.NewDecoder(bytes.NewReader(x)) decoder183.DisallowUnknownFields() var h183 EditRangeWithInsertReplace if err := decoder183.Decode(&h183); err == nil { t.Value = h183 return nil } decoder184 := json.NewDecoder(bytes.NewReader(x)) decoder184.DisallowUnknownFields() var h184 Range if err := decoder184.Decode(&h184); err == nil { t.Value = h184 return nil } return &UnmarshalError{"unmarshal failed to match one of [EditRangeWithInsertReplace Range]"} } func (t Or_CompletionItem_documentation) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case MarkupContent: return json.Marshal(x) case string: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [MarkupContent string]", t) } func (t *Or_CompletionItem_documentation) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder25 := json.NewDecoder(bytes.NewReader(x)) decoder25.DisallowUnknownFields() var stringVal string if err := decoder25.Decode(&stringVal); err == nil { t.Value = stringVal return nil } decoder26 := json.NewDecoder(bytes.NewReader(x)) decoder26.DisallowUnknownFields() var h26 MarkupContent if err := decoder26.Decode(&h26); err == nil { t.Value = h26 return nil } return &UnmarshalError{"unmarshal failed to match one of [MarkupContent string]"} } func (t Or_CompletionItem_textEdit) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case InsertReplaceEdit: return json.Marshal(x) case TextEdit: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [InsertReplaceEdit TextEdit]", t) } func (t *Or_CompletionItem_textEdit) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder29 := json.NewDecoder(bytes.NewReader(x)) decoder29.DisallowUnknownFields() var h29 InsertReplaceEdit if err := decoder29.Decode(&h29); err == nil { t.Value = h29 return nil } decoder30 := json.NewDecoder(bytes.NewReader(x)) decoder30.DisallowUnknownFields() var h30 TextEdit if err := decoder30.Decode(&h30); err == nil { t.Value = h30 return nil } return &UnmarshalError{"unmarshal failed to match one of [InsertReplaceEdit TextEdit]"} } func (t Or_Declaration) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case Location: return json.Marshal(x) case []Location: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [Location []Location]", t) } func (t *Or_Declaration) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder237 := json.NewDecoder(bytes.NewReader(x)) decoder237.DisallowUnknownFields() var h237 Location if err := decoder237.Decode(&h237); err == nil { t.Value = h237 return nil } decoder238 := json.NewDecoder(bytes.NewReader(x)) decoder238.DisallowUnknownFields() var h238 []Location if err := decoder238.Decode(&h238); err == nil { t.Value = h238 return nil } return &UnmarshalError{"unmarshal failed to match one of [Location []Location]"} } func (t Or_Definition) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case Location: return json.Marshal(x) case []Location: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [Location []Location]", t) } func (t *Or_Definition) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder224 := json.NewDecoder(bytes.NewReader(x)) decoder224.DisallowUnknownFields() var h224 Location if err := decoder224.Decode(&h224); err == nil { t.Value = h224 return nil } decoder225 := json.NewDecoder(bytes.NewReader(x)) decoder225.DisallowUnknownFields() var h225 []Location if err := decoder225.Decode(&h225); err == nil { t.Value = h225 return nil } return &UnmarshalError{"unmarshal failed to match one of [Location []Location]"} } func (t Or_Diagnostic_code) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case int32: return json.Marshal(x) case string: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [int32 string]", t) } func (t *Or_Diagnostic_code) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder179 := json.NewDecoder(bytes.NewReader(x)) decoder179.DisallowUnknownFields() var int32Val int32 if err := decoder179.Decode(&int32Val); err == nil { t.Value = int32Val return nil } decoder180 := json.NewDecoder(bytes.NewReader(x)) decoder180.DisallowUnknownFields() var stringVal string if err := decoder180.Decode(&stringVal); err == nil { t.Value = stringVal return nil } return &UnmarshalError{"unmarshal failed to match one of [int32 string]"} } func (t Or_DidChangeConfigurationRegistrationOptions_section) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case []string: return json.Marshal(x) case string: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [[]string string]", t) } func (t *Or_DidChangeConfigurationRegistrationOptions_section) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder22 := json.NewDecoder(bytes.NewReader(x)) decoder22.DisallowUnknownFields() var stringVal string if err := decoder22.Decode(&stringVal); err == nil { t.Value = stringVal return nil } decoder23 := json.NewDecoder(bytes.NewReader(x)) decoder23.DisallowUnknownFields() var h23 []string if err := decoder23.Decode(&h23); err == nil { t.Value = h23 return nil } return &UnmarshalError{"unmarshal failed to match one of [[]string string]"} } func (t Or_DocumentDiagnosticReport) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case RelatedFullDocumentDiagnosticReport: return json.Marshal(x) case RelatedUnchangedDocumentDiagnosticReport: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [RelatedFullDocumentDiagnosticReport RelatedUnchangedDocumentDiagnosticReport]", t) } func (t *Or_DocumentDiagnosticReport) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder247 := json.NewDecoder(bytes.NewReader(x)) decoder247.DisallowUnknownFields() var h247 RelatedFullDocumentDiagnosticReport if err := decoder247.Decode(&h247); err == nil { t.Value = h247 return nil } decoder248 := json.NewDecoder(bytes.NewReader(x)) decoder248.DisallowUnknownFields() var h248 RelatedUnchangedDocumentDiagnosticReport if err := decoder248.Decode(&h248); err == nil { t.Value = h248 return nil } return &UnmarshalError{"unmarshal failed to match one of [RelatedFullDocumentDiagnosticReport RelatedUnchangedDocumentDiagnosticReport]"} } func (t Or_DocumentDiagnosticReportPartialResult_relatedDocuments_Value) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case FullDocumentDiagnosticReport: return json.Marshal(x) case UnchangedDocumentDiagnosticReport: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [FullDocumentDiagnosticReport UnchangedDocumentDiagnosticReport]", t) } func (t *Or_DocumentDiagnosticReportPartialResult_relatedDocuments_Value) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder16 := json.NewDecoder(bytes.NewReader(x)) decoder16.DisallowUnknownFields() var h16 FullDocumentDiagnosticReport if err := decoder16.Decode(&h16); err == nil { t.Value = h16 return nil } decoder17 := json.NewDecoder(bytes.NewReader(x)) decoder17.DisallowUnknownFields() var h17 UnchangedDocumentDiagnosticReport if err := decoder17.Decode(&h17); err == nil { t.Value = h17 return nil } return &UnmarshalError{"unmarshal failed to match one of [FullDocumentDiagnosticReport UnchangedDocumentDiagnosticReport]"} } func (t Or_DocumentFilter) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case NotebookCellTextDocumentFilter: return json.Marshal(x) case TextDocumentFilter: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [NotebookCellTextDocumentFilter TextDocumentFilter]", t) } func (t *Or_DocumentFilter) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder270 := json.NewDecoder(bytes.NewReader(x)) decoder270.DisallowUnknownFields() var h270 NotebookCellTextDocumentFilter if err := decoder270.Decode(&h270); err == nil { t.Value = h270 return nil } decoder271 := json.NewDecoder(bytes.NewReader(x)) decoder271.DisallowUnknownFields() var h271 TextDocumentFilter if err := decoder271.Decode(&h271); err == nil { t.Value = h271 return nil } return &UnmarshalError{"unmarshal failed to match one of [NotebookCellTextDocumentFilter TextDocumentFilter]"} } func (t Or_GlobPattern) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case Pattern: return json.Marshal(x) case RelativePattern: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [Pattern RelativePattern]", t) } func (t *Or_GlobPattern) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder274 := json.NewDecoder(bytes.NewReader(x)) decoder274.DisallowUnknownFields() var h274 Pattern if err := decoder274.Decode(&h274); err == nil { t.Value = h274 return nil } decoder275 := json.NewDecoder(bytes.NewReader(x)) decoder275.DisallowUnknownFields() var h275 RelativePattern if err := decoder275.Decode(&h275); err == nil { t.Value = h275 return nil } return &UnmarshalError{"unmarshal failed to match one of [Pattern RelativePattern]"} } func (t Or_Hover_contents) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case MarkedString: return json.Marshal(x) case MarkupContent: return json.Marshal(x) case []MarkedString: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [MarkedString MarkupContent []MarkedString]", t) } func (t *Or_Hover_contents) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder34 := json.NewDecoder(bytes.NewReader(x)) decoder34.DisallowUnknownFields() var h34 MarkedString if err := decoder34.Decode(&h34); err == nil { t.Value = h34 return nil } decoder35 := json.NewDecoder(bytes.NewReader(x)) decoder35.DisallowUnknownFields() var h35 MarkupContent if err := decoder35.Decode(&h35); err == nil { t.Value = h35 return nil } decoder36 := json.NewDecoder(bytes.NewReader(x)) decoder36.DisallowUnknownFields() var h36 []MarkedString if err := decoder36.Decode(&h36); err == nil { t.Value = h36 return nil } return &UnmarshalError{"unmarshal failed to match one of [MarkedString MarkupContent []MarkedString]"} } func (t Or_InlayHintLabelPart_tooltip) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case MarkupContent: return json.Marshal(x) case string: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [MarkupContent string]", t) } func (t *Or_InlayHintLabelPart_tooltip) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder56 := json.NewDecoder(bytes.NewReader(x)) decoder56.DisallowUnknownFields() var stringVal string if err := decoder56.Decode(&stringVal); err == nil { t.Value = stringVal return nil } decoder57 := json.NewDecoder(bytes.NewReader(x)) decoder57.DisallowUnknownFields() var h57 MarkupContent if err := decoder57.Decode(&h57); err == nil { t.Value = h57 return nil } return &UnmarshalError{"unmarshal failed to match one of [MarkupContent string]"} } func (t Or_InlayHint_label) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case []InlayHintLabelPart: return json.Marshal(x) case string: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [[]InlayHintLabelPart string]", t) } func (t *Or_InlayHint_label) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder9 := json.NewDecoder(bytes.NewReader(x)) decoder9.DisallowUnknownFields() var stringVal string if err := decoder9.Decode(&stringVal); err == nil { t.Value = stringVal return nil } decoder10 := json.NewDecoder(bytes.NewReader(x)) decoder10.DisallowUnknownFields() var h10 []InlayHintLabelPart if err := decoder10.Decode(&h10); err == nil { t.Value = h10 return nil } return &UnmarshalError{"unmarshal failed to match one of [[]InlayHintLabelPart string]"} } func (t Or_InlayHint_tooltip) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case MarkupContent: return json.Marshal(x) case string: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [MarkupContent string]", t) } func (t *Or_InlayHint_tooltip) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder12 := json.NewDecoder(bytes.NewReader(x)) decoder12.DisallowUnknownFields() var stringVal string if err := decoder12.Decode(&stringVal); err == nil { t.Value = stringVal return nil } decoder13 := json.NewDecoder(bytes.NewReader(x)) decoder13.DisallowUnknownFields() var h13 MarkupContent if err := decoder13.Decode(&h13); err == nil { t.Value = h13 return nil } return &UnmarshalError{"unmarshal failed to match one of [MarkupContent string]"} } func (t Or_InlineCompletionItem_insertText) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case StringValue: return json.Marshal(x) case string: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [StringValue string]", t) } func (t *Or_InlineCompletionItem_insertText) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder19 := json.NewDecoder(bytes.NewReader(x)) decoder19.DisallowUnknownFields() var stringVal string if err := decoder19.Decode(&stringVal); err == nil { t.Value = stringVal return nil } decoder20 := json.NewDecoder(bytes.NewReader(x)) decoder20.DisallowUnknownFields() var h20 StringValue if err := decoder20.Decode(&h20); err == nil { t.Value = h20 return nil } return &UnmarshalError{"unmarshal failed to match one of [StringValue string]"} } func (t Or_InlineValue) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case InlineValueEvaluatableExpression: return json.Marshal(x) case InlineValueText: return json.Marshal(x) case InlineValueVariableLookup: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [InlineValueEvaluatableExpression InlineValueText InlineValueVariableLookup]", t) } func (t *Or_InlineValue) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder242 := json.NewDecoder(bytes.NewReader(x)) decoder242.DisallowUnknownFields() var h242 InlineValueEvaluatableExpression if err := decoder242.Decode(&h242); err == nil { t.Value = h242 return nil } decoder243 := json.NewDecoder(bytes.NewReader(x)) decoder243.DisallowUnknownFields() var h243 InlineValueText if err := decoder243.Decode(&h243); err == nil { t.Value = h243 return nil } decoder244 := json.NewDecoder(bytes.NewReader(x)) decoder244.DisallowUnknownFields() var h244 InlineValueVariableLookup if err := decoder244.Decode(&h244); err == nil { t.Value = h244 return nil } return &UnmarshalError{"unmarshal failed to match one of [InlineValueEvaluatableExpression InlineValueText InlineValueVariableLookup]"} } func (t Or_LSPAny) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case LSPArray: return json.Marshal(x) case LSPObject: return json.Marshal(x) case bool: return json.Marshal(x) case float64: return json.Marshal(x) case int32: return json.Marshal(x) case string: return json.Marshal(x) case uint32: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [LSPArray LSPObject bool float64 int32 string uint32]", t) } func (t *Or_LSPAny) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder228 := json.NewDecoder(bytes.NewReader(x)) decoder228.DisallowUnknownFields() var boolVal bool if err := decoder228.Decode(&boolVal); err == nil { t.Value = boolVal return nil } decoder229 := json.NewDecoder(bytes.NewReader(x)) decoder229.DisallowUnknownFields() var float64Val float64 if err := decoder229.Decode(&float64Val); err == nil { t.Value = float64Val return nil } decoder230 := json.NewDecoder(bytes.NewReader(x)) decoder230.DisallowUnknownFields() var int32Val int32 if err := decoder230.Decode(&int32Val); err == nil { t.Value = int32Val return nil } decoder231 := json.NewDecoder(bytes.NewReader(x)) decoder231.DisallowUnknownFields() var stringVal string if err := decoder231.Decode(&stringVal); err == nil { t.Value = stringVal return nil } decoder232 := json.NewDecoder(bytes.NewReader(x)) decoder232.DisallowUnknownFields() var uint32Val uint32 if err := decoder232.Decode(&uint32Val); err == nil { t.Value = uint32Val return nil } decoder233 := json.NewDecoder(bytes.NewReader(x)) decoder233.DisallowUnknownFields() var h233 LSPArray if err := decoder233.Decode(&h233); err == nil { t.Value = h233 return nil } decoder234 := json.NewDecoder(bytes.NewReader(x)) decoder234.DisallowUnknownFields() var h234 LSPObject if err := decoder234.Decode(&h234); err == nil { t.Value = h234 return nil } return &UnmarshalError{"unmarshal failed to match one of [LSPArray LSPObject bool float64 int32 string uint32]"} } func (t Or_MarkedString) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case MarkedStringWithLanguage: return json.Marshal(x) case string: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [MarkedStringWithLanguage string]", t) } func (t *Or_MarkedString) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder266 := json.NewDecoder(bytes.NewReader(x)) decoder266.DisallowUnknownFields() var stringVal string if err := decoder266.Decode(&stringVal); err == nil { t.Value = stringVal return nil } decoder267 := json.NewDecoder(bytes.NewReader(x)) decoder267.DisallowUnknownFields() var h267 MarkedStringWithLanguage if err := decoder267.Decode(&h267); err == nil { t.Value = h267 return nil } return &UnmarshalError{"unmarshal failed to match one of [MarkedStringWithLanguage string]"} } func (t Or_NotebookCellTextDocumentFilter_notebook) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case NotebookDocumentFilter: return json.Marshal(x) case string: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [NotebookDocumentFilter string]", t) } func (t *Or_NotebookCellTextDocumentFilter_notebook) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder208 := json.NewDecoder(bytes.NewReader(x)) decoder208.DisallowUnknownFields() var stringVal string if err := decoder208.Decode(&stringVal); err == nil { t.Value = stringVal return nil } decoder209 := json.NewDecoder(bytes.NewReader(x)) decoder209.DisallowUnknownFields() var h209 NotebookDocumentFilter if err := decoder209.Decode(&h209); err == nil { t.Value = h209 return nil } return &UnmarshalError{"unmarshal failed to match one of [NotebookDocumentFilter string]"} } func (t Or_NotebookDocumentFilter) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case NotebookDocumentFilterNotebookType: return json.Marshal(x) case NotebookDocumentFilterPattern: return json.Marshal(x) case NotebookDocumentFilterScheme: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [NotebookDocumentFilterNotebookType NotebookDocumentFilterPattern NotebookDocumentFilterScheme]", t) } func (t *Or_NotebookDocumentFilter) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder285 := json.NewDecoder(bytes.NewReader(x)) decoder285.DisallowUnknownFields() var h285 NotebookDocumentFilterNotebookType if err := decoder285.Decode(&h285); err == nil { t.Value = h285 return nil } decoder286 := json.NewDecoder(bytes.NewReader(x)) decoder286.DisallowUnknownFields() var h286 NotebookDocumentFilterPattern if err := decoder286.Decode(&h286); err == nil { t.Value = h286 return nil } decoder287 := json.NewDecoder(bytes.NewReader(x)) decoder287.DisallowUnknownFields() var h287 NotebookDocumentFilterScheme if err := decoder287.Decode(&h287); err == nil { t.Value = h287 return nil } return &UnmarshalError{"unmarshal failed to match one of [NotebookDocumentFilterNotebookType NotebookDocumentFilterPattern NotebookDocumentFilterScheme]"} } func (t Or_NotebookDocumentFilterWithCells_notebook) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case NotebookDocumentFilter: return json.Marshal(x) case string: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [NotebookDocumentFilter string]", t) } func (t *Or_NotebookDocumentFilterWithCells_notebook) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder192 := json.NewDecoder(bytes.NewReader(x)) decoder192.DisallowUnknownFields() var stringVal string if err := decoder192.Decode(&stringVal); err == nil { t.Value = stringVal return nil } decoder193 := json.NewDecoder(bytes.NewReader(x)) decoder193.DisallowUnknownFields() var h193 NotebookDocumentFilter if err := decoder193.Decode(&h193); err == nil { t.Value = h193 return nil } return &UnmarshalError{"unmarshal failed to match one of [NotebookDocumentFilter string]"} } func (t Or_NotebookDocumentFilterWithNotebook_notebook) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case NotebookDocumentFilter: return json.Marshal(x) case string: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [NotebookDocumentFilter string]", t) } func (t *Or_NotebookDocumentFilterWithNotebook_notebook) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder189 := json.NewDecoder(bytes.NewReader(x)) decoder189.DisallowUnknownFields() var stringVal string if err := decoder189.Decode(&stringVal); err == nil { t.Value = stringVal return nil } decoder190 := json.NewDecoder(bytes.NewReader(x)) decoder190.DisallowUnknownFields() var h190 NotebookDocumentFilter if err := decoder190.Decode(&h190); err == nil { t.Value = h190 return nil } return &UnmarshalError{"unmarshal failed to match one of [NotebookDocumentFilter string]"} } func (t Or_NotebookDocumentSyncOptions_notebookSelector_Elem) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case NotebookDocumentFilterWithCells: return json.Marshal(x) case NotebookDocumentFilterWithNotebook: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [NotebookDocumentFilterWithCells NotebookDocumentFilterWithNotebook]", t) } func (t *Or_NotebookDocumentSyncOptions_notebookSelector_Elem) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder68 := json.NewDecoder(bytes.NewReader(x)) decoder68.DisallowUnknownFields() var h68 NotebookDocumentFilterWithCells if err := decoder68.Decode(&h68); err == nil { t.Value = h68 return nil } decoder69 := json.NewDecoder(bytes.NewReader(x)) decoder69.DisallowUnknownFields() var h69 NotebookDocumentFilterWithNotebook if err := decoder69.Decode(&h69); err == nil { t.Value = h69 return nil } return &UnmarshalError{"unmarshal failed to match one of [NotebookDocumentFilterWithCells NotebookDocumentFilterWithNotebook]"} } func (t Or_ParameterInformation_documentation) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case MarkupContent: return json.Marshal(x) case string: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [MarkupContent string]", t) } func (t *Or_ParameterInformation_documentation) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder205 := json.NewDecoder(bytes.NewReader(x)) decoder205.DisallowUnknownFields() var stringVal string if err := decoder205.Decode(&stringVal); err == nil { t.Value = stringVal return nil } decoder206 := json.NewDecoder(bytes.NewReader(x)) decoder206.DisallowUnknownFields() var h206 MarkupContent if err := decoder206.Decode(&h206); err == nil { t.Value = h206 return nil } return &UnmarshalError{"unmarshal failed to match one of [MarkupContent string]"} } func (t Or_ParameterInformation_label) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case Tuple_ParameterInformation_label_Item1: return json.Marshal(x) case string: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [Tuple_ParameterInformation_label_Item1 string]", t) } func (t *Or_ParameterInformation_label) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder202 := json.NewDecoder(bytes.NewReader(x)) decoder202.DisallowUnknownFields() var stringVal string if err := decoder202.Decode(&stringVal); err == nil { t.Value = stringVal return nil } decoder203 := json.NewDecoder(bytes.NewReader(x)) decoder203.DisallowUnknownFields() var h203 Tuple_ParameterInformation_label_Item1 if err := decoder203.Decode(&h203); err == nil { t.Value = h203 return nil } return &UnmarshalError{"unmarshal failed to match one of [Tuple_ParameterInformation_label_Item1 string]"} } func (t Or_PrepareRenameResult) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case PrepareRenameDefaultBehavior: return json.Marshal(x) case PrepareRenamePlaceholder: return json.Marshal(x) case Range: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [PrepareRenameDefaultBehavior PrepareRenamePlaceholder Range]", t) } func (t *Or_PrepareRenameResult) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder252 := json.NewDecoder(bytes.NewReader(x)) decoder252.DisallowUnknownFields() var h252 PrepareRenameDefaultBehavior if err := decoder252.Decode(&h252); err == nil { t.Value = h252 return nil } decoder253 := json.NewDecoder(bytes.NewReader(x)) decoder253.DisallowUnknownFields() var h253 PrepareRenamePlaceholder if err := decoder253.Decode(&h253); err == nil { t.Value = h253 return nil } decoder254 := json.NewDecoder(bytes.NewReader(x)) decoder254.DisallowUnknownFields() var h254 Range if err := decoder254.Decode(&h254); err == nil { t.Value = h254 return nil } return &UnmarshalError{"unmarshal failed to match one of [PrepareRenameDefaultBehavior PrepareRenamePlaceholder Range]"} } func (t Or_ProgressToken) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case int32: return json.Marshal(x) case string: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [int32 string]", t) } func (t *Or_ProgressToken) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder255 := json.NewDecoder(bytes.NewReader(x)) decoder255.DisallowUnknownFields() var int32Val int32 if err := decoder255.Decode(&int32Val); err == nil { t.Value = int32Val return nil } decoder256 := json.NewDecoder(bytes.NewReader(x)) decoder256.DisallowUnknownFields() var stringVal string if err := decoder256.Decode(&stringVal); err == nil { t.Value = stringVal return nil } return &UnmarshalError{"unmarshal failed to match one of [int32 string]"} } func (t Or_RelatedFullDocumentDiagnosticReport_relatedDocuments_Value) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case FullDocumentDiagnosticReport: return json.Marshal(x) case UnchangedDocumentDiagnosticReport: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [FullDocumentDiagnosticReport UnchangedDocumentDiagnosticReport]", t) } func (t *Or_RelatedFullDocumentDiagnosticReport_relatedDocuments_Value) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder60 := json.NewDecoder(bytes.NewReader(x)) decoder60.DisallowUnknownFields() var h60 FullDocumentDiagnosticReport if err := decoder60.Decode(&h60); err == nil { t.Value = h60 return nil } decoder61 := json.NewDecoder(bytes.NewReader(x)) decoder61.DisallowUnknownFields() var h61 UnchangedDocumentDiagnosticReport if err := decoder61.Decode(&h61); err == nil { t.Value = h61 return nil } return &UnmarshalError{"unmarshal failed to match one of [FullDocumentDiagnosticReport UnchangedDocumentDiagnosticReport]"} } func (t Or_RelatedUnchangedDocumentDiagnosticReport_relatedDocuments_Value) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case FullDocumentDiagnosticReport: return json.Marshal(x) case UnchangedDocumentDiagnosticReport: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [FullDocumentDiagnosticReport UnchangedDocumentDiagnosticReport]", t) } func (t *Or_RelatedUnchangedDocumentDiagnosticReport_relatedDocuments_Value) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder64 := json.NewDecoder(bytes.NewReader(x)) decoder64.DisallowUnknownFields() var h64 FullDocumentDiagnosticReport if err := decoder64.Decode(&h64); err == nil { t.Value = h64 return nil } decoder65 := json.NewDecoder(bytes.NewReader(x)) decoder65.DisallowUnknownFields() var h65 UnchangedDocumentDiagnosticReport if err := decoder65.Decode(&h65); err == nil { t.Value = h65 return nil } return &UnmarshalError{"unmarshal failed to match one of [FullDocumentDiagnosticReport UnchangedDocumentDiagnosticReport]"} } func (t Or_RelativePattern_baseUri) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case URI: return json.Marshal(x) case WorkspaceFolder: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [URI WorkspaceFolder]", t) } func (t *Or_RelativePattern_baseUri) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder214 := json.NewDecoder(bytes.NewReader(x)) decoder214.DisallowUnknownFields() var h214 URI if err := decoder214.Decode(&h214); err == nil { t.Value = h214 return nil } decoder215 := json.NewDecoder(bytes.NewReader(x)) decoder215.DisallowUnknownFields() var h215 WorkspaceFolder if err := decoder215.Decode(&h215); err == nil { t.Value = h215 return nil } return &UnmarshalError{"unmarshal failed to match one of [URI WorkspaceFolder]"} } func (t Or_Result_textDocument_codeAction_Item0_Elem) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case CodeAction: return json.Marshal(x) case Command: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [CodeAction Command]", t) } func (t *Or_Result_textDocument_codeAction_Item0_Elem) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder322 := json.NewDecoder(bytes.NewReader(x)) decoder322.DisallowUnknownFields() var h322 CodeAction if err := decoder322.Decode(&h322); err == nil { t.Value = h322 return nil } decoder323 := json.NewDecoder(bytes.NewReader(x)) decoder323.DisallowUnknownFields() var h323 Command if err := decoder323.Decode(&h323); err == nil { t.Value = h323 return nil } return &UnmarshalError{"unmarshal failed to match one of [CodeAction Command]"} } func (t Or_Result_textDocument_completion) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case CompletionList: return json.Marshal(x) case []CompletionItem: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [CompletionList []CompletionItem]", t) } func (t *Or_Result_textDocument_completion) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder310 := json.NewDecoder(bytes.NewReader(x)) decoder310.DisallowUnknownFields() var h310 CompletionList if err := decoder310.Decode(&h310); err == nil { t.Value = h310 return nil } decoder311 := json.NewDecoder(bytes.NewReader(x)) decoder311.DisallowUnknownFields() var h311 []CompletionItem if err := decoder311.Decode(&h311); err == nil { t.Value = h311 return nil } return &UnmarshalError{"unmarshal failed to match one of [CompletionList []CompletionItem]"} } func (t Or_Result_textDocument_declaration) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case Declaration: return json.Marshal(x) case []DeclarationLink: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [Declaration []DeclarationLink]", t) } func (t *Or_Result_textDocument_declaration) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder298 := json.NewDecoder(bytes.NewReader(x)) decoder298.DisallowUnknownFields() var h298 Declaration if err := decoder298.Decode(&h298); err == nil { t.Value = h298 return nil } decoder299 := json.NewDecoder(bytes.NewReader(x)) decoder299.DisallowUnknownFields() var h299 []DeclarationLink if err := decoder299.Decode(&h299); err == nil { t.Value = h299 return nil } return &UnmarshalError{"unmarshal failed to match one of [Declaration []DeclarationLink]"} } func (t Or_Result_textDocument_definition) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case Definition: return json.Marshal(x) case []DefinitionLink: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [Definition []DefinitionLink]", t) } func (t *Or_Result_textDocument_definition) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder314 := json.NewDecoder(bytes.NewReader(x)) decoder314.DisallowUnknownFields() var h314 Definition if err := decoder314.Decode(&h314); err == nil { t.Value = h314 return nil } decoder315 := json.NewDecoder(bytes.NewReader(x)) decoder315.DisallowUnknownFields() var h315 []DefinitionLink if err := decoder315.Decode(&h315); err == nil { t.Value = h315 return nil } return &UnmarshalError{"unmarshal failed to match one of [Definition []DefinitionLink]"} } func (t Or_Result_textDocument_documentSymbol) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case []DocumentSymbol: return json.Marshal(x) case []SymbolInformation: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [[]DocumentSymbol []SymbolInformation]", t) } func (t *Or_Result_textDocument_documentSymbol) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder318 := json.NewDecoder(bytes.NewReader(x)) decoder318.DisallowUnknownFields() var h318 []DocumentSymbol if err := decoder318.Decode(&h318); err == nil { t.Value = h318 return nil } decoder319 := json.NewDecoder(bytes.NewReader(x)) decoder319.DisallowUnknownFields() var h319 []SymbolInformation if err := decoder319.Decode(&h319); err == nil { t.Value = h319 return nil } return &UnmarshalError{"unmarshal failed to match one of [[]DocumentSymbol []SymbolInformation]"} } func (t Or_Result_textDocument_implementation) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case Definition: return json.Marshal(x) case []DefinitionLink: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [Definition []DefinitionLink]", t) } func (t *Or_Result_textDocument_implementation) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder290 := json.NewDecoder(bytes.NewReader(x)) decoder290.DisallowUnknownFields() var h290 Definition if err := decoder290.Decode(&h290); err == nil { t.Value = h290 return nil } decoder291 := json.NewDecoder(bytes.NewReader(x)) decoder291.DisallowUnknownFields() var h291 []DefinitionLink if err := decoder291.Decode(&h291); err == nil { t.Value = h291 return nil } return &UnmarshalError{"unmarshal failed to match one of [Definition []DefinitionLink]"} } func (t Or_Result_textDocument_inlineCompletion) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case InlineCompletionList: return json.Marshal(x) case []InlineCompletionItem: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [InlineCompletionList []InlineCompletionItem]", t) } func (t *Or_Result_textDocument_inlineCompletion) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder306 := json.NewDecoder(bytes.NewReader(x)) decoder306.DisallowUnknownFields() var h306 InlineCompletionList if err := decoder306.Decode(&h306); err == nil { t.Value = h306 return nil } decoder307 := json.NewDecoder(bytes.NewReader(x)) decoder307.DisallowUnknownFields() var h307 []InlineCompletionItem if err := decoder307.Decode(&h307); err == nil { t.Value = h307 return nil } return &UnmarshalError{"unmarshal failed to match one of [InlineCompletionList []InlineCompletionItem]"} } func (t Or_Result_textDocument_semanticTokens_full_delta) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case SemanticTokens: return json.Marshal(x) case SemanticTokensDelta: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [SemanticTokens SemanticTokensDelta]", t) } func (t *Or_Result_textDocument_semanticTokens_full_delta) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder302 := json.NewDecoder(bytes.NewReader(x)) decoder302.DisallowUnknownFields() var h302 SemanticTokens if err := decoder302.Decode(&h302); err == nil { t.Value = h302 return nil } decoder303 := json.NewDecoder(bytes.NewReader(x)) decoder303.DisallowUnknownFields() var h303 SemanticTokensDelta if err := decoder303.Decode(&h303); err == nil { t.Value = h303 return nil } return &UnmarshalError{"unmarshal failed to match one of [SemanticTokens SemanticTokensDelta]"} } func (t Or_Result_textDocument_typeDefinition) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case Definition: return json.Marshal(x) case []DefinitionLink: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [Definition []DefinitionLink]", t) } func (t *Or_Result_textDocument_typeDefinition) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder294 := json.NewDecoder(bytes.NewReader(x)) decoder294.DisallowUnknownFields() var h294 Definition if err := decoder294.Decode(&h294); err == nil { t.Value = h294 return nil } decoder295 := json.NewDecoder(bytes.NewReader(x)) decoder295.DisallowUnknownFields() var h295 []DefinitionLink if err := decoder295.Decode(&h295); err == nil { t.Value = h295 return nil } return &UnmarshalError{"unmarshal failed to match one of [Definition []DefinitionLink]"} } func (t Or_Result_workspace_symbol) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case []SymbolInformation: return json.Marshal(x) case []WorkspaceSymbol: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [[]SymbolInformation []WorkspaceSymbol]", t) } func (t *Or_Result_workspace_symbol) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder326 := json.NewDecoder(bytes.NewReader(x)) decoder326.DisallowUnknownFields() var h326 []SymbolInformation if err := decoder326.Decode(&h326); err == nil { t.Value = h326 return nil } decoder327 := json.NewDecoder(bytes.NewReader(x)) decoder327.DisallowUnknownFields() var h327 []WorkspaceSymbol if err := decoder327.Decode(&h327); err == nil { t.Value = h327 return nil } return &UnmarshalError{"unmarshal failed to match one of [[]SymbolInformation []WorkspaceSymbol]"} } func (t Or_SemanticTokensOptions_full) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case SemanticTokensFullDelta: return json.Marshal(x) case bool: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [SemanticTokensFullDelta bool]", t) } func (t *Or_SemanticTokensOptions_full) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder47 := json.NewDecoder(bytes.NewReader(x)) decoder47.DisallowUnknownFields() var boolVal bool if err := decoder47.Decode(&boolVal); err == nil { t.Value = boolVal return nil } decoder48 := json.NewDecoder(bytes.NewReader(x)) decoder48.DisallowUnknownFields() var h48 SemanticTokensFullDelta if err := decoder48.Decode(&h48); err == nil { t.Value = h48 return nil } return &UnmarshalError{"unmarshal failed to match one of [SemanticTokensFullDelta bool]"} } func (t Or_SemanticTokensOptions_range) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case Lit_SemanticTokensOptions_range_Item1: return json.Marshal(x) case bool: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [Lit_SemanticTokensOptions_range_Item1 bool]", t) } func (t *Or_SemanticTokensOptions_range) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder44 := json.NewDecoder(bytes.NewReader(x)) decoder44.DisallowUnknownFields() var boolVal bool if err := decoder44.Decode(&boolVal); err == nil { t.Value = boolVal return nil } decoder45 := json.NewDecoder(bytes.NewReader(x)) decoder45.DisallowUnknownFields() var h45 Lit_SemanticTokensOptions_range_Item1 if err := decoder45.Decode(&h45); err == nil { t.Value = h45 return nil } return &UnmarshalError{"unmarshal failed to match one of [Lit_SemanticTokensOptions_range_Item1 bool]"} } func (t Or_ServerCapabilities_callHierarchyProvider) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case CallHierarchyOptions: return json.Marshal(x) case CallHierarchyRegistrationOptions: return json.Marshal(x) case bool: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [CallHierarchyOptions CallHierarchyRegistrationOptions bool]", t) } func (t *Or_ServerCapabilities_callHierarchyProvider) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder140 := json.NewDecoder(bytes.NewReader(x)) decoder140.DisallowUnknownFields() var boolVal bool if err := decoder140.Decode(&boolVal); err == nil { t.Value = boolVal return nil } decoder141 := json.NewDecoder(bytes.NewReader(x)) decoder141.DisallowUnknownFields() var h141 CallHierarchyOptions if err := decoder141.Decode(&h141); err == nil { t.Value = h141 return nil } decoder142 := json.NewDecoder(bytes.NewReader(x)) decoder142.DisallowUnknownFields() var h142 CallHierarchyRegistrationOptions if err := decoder142.Decode(&h142); err == nil { t.Value = h142 return nil } return &UnmarshalError{"unmarshal failed to match one of [CallHierarchyOptions CallHierarchyRegistrationOptions bool]"} } func (t Or_ServerCapabilities_codeActionProvider) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case CodeActionOptions: return json.Marshal(x) case bool: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [CodeActionOptions bool]", t) } func (t *Or_ServerCapabilities_codeActionProvider) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder109 := json.NewDecoder(bytes.NewReader(x)) decoder109.DisallowUnknownFields() var boolVal bool if err := decoder109.Decode(&boolVal); err == nil { t.Value = boolVal return nil } decoder110 := json.NewDecoder(bytes.NewReader(x)) decoder110.DisallowUnknownFields() var h110 CodeActionOptions if err := decoder110.Decode(&h110); err == nil { t.Value = h110 return nil } return &UnmarshalError{"unmarshal failed to match one of [CodeActionOptions bool]"} } func (t Or_ServerCapabilities_colorProvider) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case DocumentColorOptions: return json.Marshal(x) case DocumentColorRegistrationOptions: return json.Marshal(x) case bool: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [DocumentColorOptions DocumentColorRegistrationOptions bool]", t) } func (t *Or_ServerCapabilities_colorProvider) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder113 := json.NewDecoder(bytes.NewReader(x)) decoder113.DisallowUnknownFields() var boolVal bool if err := decoder113.Decode(&boolVal); err == nil { t.Value = boolVal return nil } decoder114 := json.NewDecoder(bytes.NewReader(x)) decoder114.DisallowUnknownFields() var h114 DocumentColorOptions if err := decoder114.Decode(&h114); err == nil { t.Value = h114 return nil } decoder115 := json.NewDecoder(bytes.NewReader(x)) decoder115.DisallowUnknownFields() var h115 DocumentColorRegistrationOptions if err := decoder115.Decode(&h115); err == nil { t.Value = h115 return nil } return &UnmarshalError{"unmarshal failed to match one of [DocumentColorOptions DocumentColorRegistrationOptions bool]"} } func (t Or_ServerCapabilities_declarationProvider) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case DeclarationOptions: return json.Marshal(x) case DeclarationRegistrationOptions: return json.Marshal(x) case bool: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [DeclarationOptions DeclarationRegistrationOptions bool]", t) } func (t *Or_ServerCapabilities_declarationProvider) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder83 := json.NewDecoder(bytes.NewReader(x)) decoder83.DisallowUnknownFields() var boolVal bool if err := decoder83.Decode(&boolVal); err == nil { t.Value = boolVal return nil } decoder84 := json.NewDecoder(bytes.NewReader(x)) decoder84.DisallowUnknownFields() var h84 DeclarationOptions if err := decoder84.Decode(&h84); err == nil { t.Value = h84 return nil } decoder85 := json.NewDecoder(bytes.NewReader(x)) decoder85.DisallowUnknownFields() var h85 DeclarationRegistrationOptions if err := decoder85.Decode(&h85); err == nil { t.Value = h85 return nil } return &UnmarshalError{"unmarshal failed to match one of [DeclarationOptions DeclarationRegistrationOptions bool]"} } func (t Or_ServerCapabilities_definitionProvider) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case DefinitionOptions: return json.Marshal(x) case bool: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [DefinitionOptions bool]", t) } func (t *Or_ServerCapabilities_definitionProvider) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder87 := json.NewDecoder(bytes.NewReader(x)) decoder87.DisallowUnknownFields() var boolVal bool if err := decoder87.Decode(&boolVal); err == nil { t.Value = boolVal return nil } decoder88 := json.NewDecoder(bytes.NewReader(x)) decoder88.DisallowUnknownFields() var h88 DefinitionOptions if err := decoder88.Decode(&h88); err == nil { t.Value = h88 return nil } return &UnmarshalError{"unmarshal failed to match one of [DefinitionOptions bool]"} } func (t Or_ServerCapabilities_diagnosticProvider) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case DiagnosticOptions: return json.Marshal(x) case DiagnosticRegistrationOptions: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [DiagnosticOptions DiagnosticRegistrationOptions]", t) } func (t *Or_ServerCapabilities_diagnosticProvider) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder174 := json.NewDecoder(bytes.NewReader(x)) decoder174.DisallowUnknownFields() var h174 DiagnosticOptions if err := decoder174.Decode(&h174); err == nil { t.Value = h174 return nil } decoder175 := json.NewDecoder(bytes.NewReader(x)) decoder175.DisallowUnknownFields() var h175 DiagnosticRegistrationOptions if err := decoder175.Decode(&h175); err == nil { t.Value = h175 return nil } return &UnmarshalError{"unmarshal failed to match one of [DiagnosticOptions DiagnosticRegistrationOptions]"} } func (t Or_ServerCapabilities_documentFormattingProvider) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case DocumentFormattingOptions: return json.Marshal(x) case bool: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [DocumentFormattingOptions bool]", t) } func (t *Or_ServerCapabilities_documentFormattingProvider) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder120 := json.NewDecoder(bytes.NewReader(x)) decoder120.DisallowUnknownFields() var boolVal bool if err := decoder120.Decode(&boolVal); err == nil { t.Value = boolVal return nil } decoder121 := json.NewDecoder(bytes.NewReader(x)) decoder121.DisallowUnknownFields() var h121 DocumentFormattingOptions if err := decoder121.Decode(&h121); err == nil { t.Value = h121 return nil } return &UnmarshalError{"unmarshal failed to match one of [DocumentFormattingOptions bool]"} } func (t Or_ServerCapabilities_documentHighlightProvider) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case DocumentHighlightOptions: return json.Marshal(x) case bool: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [DocumentHighlightOptions bool]", t) } func (t *Or_ServerCapabilities_documentHighlightProvider) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder103 := json.NewDecoder(bytes.NewReader(x)) decoder103.DisallowUnknownFields() var boolVal bool if err := decoder103.Decode(&boolVal); err == nil { t.Value = boolVal return nil } decoder104 := json.NewDecoder(bytes.NewReader(x)) decoder104.DisallowUnknownFields() var h104 DocumentHighlightOptions if err := decoder104.Decode(&h104); err == nil { t.Value = h104 return nil } return &UnmarshalError{"unmarshal failed to match one of [DocumentHighlightOptions bool]"} } func (t Or_ServerCapabilities_documentRangeFormattingProvider) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case DocumentRangeFormattingOptions: return json.Marshal(x) case bool: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [DocumentRangeFormattingOptions bool]", t) } func (t *Or_ServerCapabilities_documentRangeFormattingProvider) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder123 := json.NewDecoder(bytes.NewReader(x)) decoder123.DisallowUnknownFields() var boolVal bool if err := decoder123.Decode(&boolVal); err == nil { t.Value = boolVal return nil } decoder124 := json.NewDecoder(bytes.NewReader(x)) decoder124.DisallowUnknownFields() var h124 DocumentRangeFormattingOptions if err := decoder124.Decode(&h124); err == nil { t.Value = h124 return nil } return &UnmarshalError{"unmarshal failed to match one of [DocumentRangeFormattingOptions bool]"} } func (t Or_ServerCapabilities_documentSymbolProvider) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case DocumentSymbolOptions: return json.Marshal(x) case bool: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [DocumentSymbolOptions bool]", t) } func (t *Or_ServerCapabilities_documentSymbolProvider) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder106 := json.NewDecoder(bytes.NewReader(x)) decoder106.DisallowUnknownFields() var boolVal bool if err := decoder106.Decode(&boolVal); err == nil { t.Value = boolVal return nil } decoder107 := json.NewDecoder(bytes.NewReader(x)) decoder107.DisallowUnknownFields() var h107 DocumentSymbolOptions if err := decoder107.Decode(&h107); err == nil { t.Value = h107 return nil } return &UnmarshalError{"unmarshal failed to match one of [DocumentSymbolOptions bool]"} } func (t Or_ServerCapabilities_foldingRangeProvider) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case FoldingRangeOptions: return json.Marshal(x) case FoldingRangeRegistrationOptions: return json.Marshal(x) case bool: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [FoldingRangeOptions FoldingRangeRegistrationOptions bool]", t) } func (t *Or_ServerCapabilities_foldingRangeProvider) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder130 := json.NewDecoder(bytes.NewReader(x)) decoder130.DisallowUnknownFields() var boolVal bool if err := decoder130.Decode(&boolVal); err == nil { t.Value = boolVal return nil } decoder131 := json.NewDecoder(bytes.NewReader(x)) decoder131.DisallowUnknownFields() var h131 FoldingRangeOptions if err := decoder131.Decode(&h131); err == nil { t.Value = h131 return nil } decoder132 := json.NewDecoder(bytes.NewReader(x)) decoder132.DisallowUnknownFields() var h132 FoldingRangeRegistrationOptions if err := decoder132.Decode(&h132); err == nil { t.Value = h132 return nil } return &UnmarshalError{"unmarshal failed to match one of [FoldingRangeOptions FoldingRangeRegistrationOptions bool]"} } func (t Or_ServerCapabilities_hoverProvider) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case HoverOptions: return json.Marshal(x) case bool: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [HoverOptions bool]", t) } func (t *Or_ServerCapabilities_hoverProvider) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder79 := json.NewDecoder(bytes.NewReader(x)) decoder79.DisallowUnknownFields() var boolVal bool if err := decoder79.Decode(&boolVal); err == nil { t.Value = boolVal return nil } decoder80 := json.NewDecoder(bytes.NewReader(x)) decoder80.DisallowUnknownFields() var h80 HoverOptions if err := decoder80.Decode(&h80); err == nil { t.Value = h80 return nil } return &UnmarshalError{"unmarshal failed to match one of [HoverOptions bool]"} } func (t Or_ServerCapabilities_implementationProvider) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case ImplementationOptions: return json.Marshal(x) case ImplementationRegistrationOptions: return json.Marshal(x) case bool: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [ImplementationOptions ImplementationRegistrationOptions bool]", t) } func (t *Or_ServerCapabilities_implementationProvider) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder96 := json.NewDecoder(bytes.NewReader(x)) decoder96.DisallowUnknownFields() var boolVal bool if err := decoder96.Decode(&boolVal); err == nil { t.Value = boolVal return nil } decoder97 := json.NewDecoder(bytes.NewReader(x)) decoder97.DisallowUnknownFields() var h97 ImplementationOptions if err := decoder97.Decode(&h97); err == nil { t.Value = h97 return nil } decoder98 := json.NewDecoder(bytes.NewReader(x)) decoder98.DisallowUnknownFields() var h98 ImplementationRegistrationOptions if err := decoder98.Decode(&h98); err == nil { t.Value = h98 return nil } return &UnmarshalError{"unmarshal failed to match one of [ImplementationOptions ImplementationRegistrationOptions bool]"} } func (t Or_ServerCapabilities_inlayHintProvider) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case InlayHintOptions: return json.Marshal(x) case InlayHintRegistrationOptions: return json.Marshal(x) case bool: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [InlayHintOptions InlayHintRegistrationOptions bool]", t) } func (t *Or_ServerCapabilities_inlayHintProvider) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder169 := json.NewDecoder(bytes.NewReader(x)) decoder169.DisallowUnknownFields() var boolVal bool if err := decoder169.Decode(&boolVal); err == nil { t.Value = boolVal return nil } decoder170 := json.NewDecoder(bytes.NewReader(x)) decoder170.DisallowUnknownFields() var h170 InlayHintOptions if err := decoder170.Decode(&h170); err == nil { t.Value = h170 return nil } decoder171 := json.NewDecoder(bytes.NewReader(x)) decoder171.DisallowUnknownFields() var h171 InlayHintRegistrationOptions if err := decoder171.Decode(&h171); err == nil { t.Value = h171 return nil } return &UnmarshalError{"unmarshal failed to match one of [InlayHintOptions InlayHintRegistrationOptions bool]"} } func (t Or_ServerCapabilities_inlineCompletionProvider) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case InlineCompletionOptions: return json.Marshal(x) case bool: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [InlineCompletionOptions bool]", t) } func (t *Or_ServerCapabilities_inlineCompletionProvider) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder177 := json.NewDecoder(bytes.NewReader(x)) decoder177.DisallowUnknownFields() var boolVal bool if err := decoder177.Decode(&boolVal); err == nil { t.Value = boolVal return nil } decoder178 := json.NewDecoder(bytes.NewReader(x)) decoder178.DisallowUnknownFields() var h178 InlineCompletionOptions if err := decoder178.Decode(&h178); err == nil { t.Value = h178 return nil } return &UnmarshalError{"unmarshal failed to match one of [InlineCompletionOptions bool]"} } func (t Or_ServerCapabilities_inlineValueProvider) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case InlineValueOptions: return json.Marshal(x) case InlineValueRegistrationOptions: return json.Marshal(x) case bool: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [InlineValueOptions InlineValueRegistrationOptions bool]", t) } func (t *Or_ServerCapabilities_inlineValueProvider) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder164 := json.NewDecoder(bytes.NewReader(x)) decoder164.DisallowUnknownFields() var boolVal bool if err := decoder164.Decode(&boolVal); err == nil { t.Value = boolVal return nil } decoder165 := json.NewDecoder(bytes.NewReader(x)) decoder165.DisallowUnknownFields() var h165 InlineValueOptions if err := decoder165.Decode(&h165); err == nil { t.Value = h165 return nil } decoder166 := json.NewDecoder(bytes.NewReader(x)) decoder166.DisallowUnknownFields() var h166 InlineValueRegistrationOptions if err := decoder166.Decode(&h166); err == nil { t.Value = h166 return nil } return &UnmarshalError{"unmarshal failed to match one of [InlineValueOptions InlineValueRegistrationOptions bool]"} } func (t Or_ServerCapabilities_linkedEditingRangeProvider) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case LinkedEditingRangeOptions: return json.Marshal(x) case LinkedEditingRangeRegistrationOptions: return json.Marshal(x) case bool: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [LinkedEditingRangeOptions LinkedEditingRangeRegistrationOptions bool]", t) } func (t *Or_ServerCapabilities_linkedEditingRangeProvider) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder145 := json.NewDecoder(bytes.NewReader(x)) decoder145.DisallowUnknownFields() var boolVal bool if err := decoder145.Decode(&boolVal); err == nil { t.Value = boolVal return nil } decoder146 := json.NewDecoder(bytes.NewReader(x)) decoder146.DisallowUnknownFields() var h146 LinkedEditingRangeOptions if err := decoder146.Decode(&h146); err == nil { t.Value = h146 return nil } decoder147 := json.NewDecoder(bytes.NewReader(x)) decoder147.DisallowUnknownFields() var h147 LinkedEditingRangeRegistrationOptions if err := decoder147.Decode(&h147); err == nil { t.Value = h147 return nil } return &UnmarshalError{"unmarshal failed to match one of [LinkedEditingRangeOptions LinkedEditingRangeRegistrationOptions bool]"} } func (t Or_ServerCapabilities_monikerProvider) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case MonikerOptions: return json.Marshal(x) case MonikerRegistrationOptions: return json.Marshal(x) case bool: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [MonikerOptions MonikerRegistrationOptions bool]", t) } func (t *Or_ServerCapabilities_monikerProvider) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder154 := json.NewDecoder(bytes.NewReader(x)) decoder154.DisallowUnknownFields() var boolVal bool if err := decoder154.Decode(&boolVal); err == nil { t.Value = boolVal return nil } decoder155 := json.NewDecoder(bytes.NewReader(x)) decoder155.DisallowUnknownFields() var h155 MonikerOptions if err := decoder155.Decode(&h155); err == nil { t.Value = h155 return nil } decoder156 := json.NewDecoder(bytes.NewReader(x)) decoder156.DisallowUnknownFields() var h156 MonikerRegistrationOptions if err := decoder156.Decode(&h156); err == nil { t.Value = h156 return nil } return &UnmarshalError{"unmarshal failed to match one of [MonikerOptions MonikerRegistrationOptions bool]"} } func (t Or_ServerCapabilities_notebookDocumentSync) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case NotebookDocumentSyncOptions: return json.Marshal(x) case NotebookDocumentSyncRegistrationOptions: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [NotebookDocumentSyncOptions NotebookDocumentSyncRegistrationOptions]", t) } func (t *Or_ServerCapabilities_notebookDocumentSync) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder76 := json.NewDecoder(bytes.NewReader(x)) decoder76.DisallowUnknownFields() var h76 NotebookDocumentSyncOptions if err := decoder76.Decode(&h76); err == nil { t.Value = h76 return nil } decoder77 := json.NewDecoder(bytes.NewReader(x)) decoder77.DisallowUnknownFields() var h77 NotebookDocumentSyncRegistrationOptions if err := decoder77.Decode(&h77); err == nil { t.Value = h77 return nil } return &UnmarshalError{"unmarshal failed to match one of [NotebookDocumentSyncOptions NotebookDocumentSyncRegistrationOptions]"} } func (t Or_ServerCapabilities_referencesProvider) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case ReferenceOptions: return json.Marshal(x) case bool: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [ReferenceOptions bool]", t) } func (t *Or_ServerCapabilities_referencesProvider) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder100 := json.NewDecoder(bytes.NewReader(x)) decoder100.DisallowUnknownFields() var boolVal bool if err := decoder100.Decode(&boolVal); err == nil { t.Value = boolVal return nil } decoder101 := json.NewDecoder(bytes.NewReader(x)) decoder101.DisallowUnknownFields() var h101 ReferenceOptions if err := decoder101.Decode(&h101); err == nil { t.Value = h101 return nil } return &UnmarshalError{"unmarshal failed to match one of [ReferenceOptions bool]"} } func (t Or_ServerCapabilities_renameProvider) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case RenameOptions: return json.Marshal(x) case bool: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [RenameOptions bool]", t) } func (t *Or_ServerCapabilities_renameProvider) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder126 := json.NewDecoder(bytes.NewReader(x)) decoder126.DisallowUnknownFields() var boolVal bool if err := decoder126.Decode(&boolVal); err == nil { t.Value = boolVal return nil } decoder127 := json.NewDecoder(bytes.NewReader(x)) decoder127.DisallowUnknownFields() var h127 RenameOptions if err := decoder127.Decode(&h127); err == nil { t.Value = h127 return nil } return &UnmarshalError{"unmarshal failed to match one of [RenameOptions bool]"} } func (t Or_ServerCapabilities_selectionRangeProvider) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case SelectionRangeOptions: return json.Marshal(x) case SelectionRangeRegistrationOptions: return json.Marshal(x) case bool: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [SelectionRangeOptions SelectionRangeRegistrationOptions bool]", t) } func (t *Or_ServerCapabilities_selectionRangeProvider) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder135 := json.NewDecoder(bytes.NewReader(x)) decoder135.DisallowUnknownFields() var boolVal bool if err := decoder135.Decode(&boolVal); err == nil { t.Value = boolVal return nil } decoder136 := json.NewDecoder(bytes.NewReader(x)) decoder136.DisallowUnknownFields() var h136 SelectionRangeOptions if err := decoder136.Decode(&h136); err == nil { t.Value = h136 return nil } decoder137 := json.NewDecoder(bytes.NewReader(x)) decoder137.DisallowUnknownFields() var h137 SelectionRangeRegistrationOptions if err := decoder137.Decode(&h137); err == nil { t.Value = h137 return nil } return &UnmarshalError{"unmarshal failed to match one of [SelectionRangeOptions SelectionRangeRegistrationOptions bool]"} } func (t Or_ServerCapabilities_semanticTokensProvider) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case SemanticTokensOptions: return json.Marshal(x) case SemanticTokensRegistrationOptions: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [SemanticTokensOptions SemanticTokensRegistrationOptions]", t) } func (t *Or_ServerCapabilities_semanticTokensProvider) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder150 := json.NewDecoder(bytes.NewReader(x)) decoder150.DisallowUnknownFields() var h150 SemanticTokensOptions if err := decoder150.Decode(&h150); err == nil { t.Value = h150 return nil } decoder151 := json.NewDecoder(bytes.NewReader(x)) decoder151.DisallowUnknownFields() var h151 SemanticTokensRegistrationOptions if err := decoder151.Decode(&h151); err == nil { t.Value = h151 return nil } return &UnmarshalError{"unmarshal failed to match one of [SemanticTokensOptions SemanticTokensRegistrationOptions]"} } func (t Or_ServerCapabilities_textDocumentSync) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case TextDocumentSyncKind: return json.Marshal(x) case TextDocumentSyncOptions: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [TextDocumentSyncKind TextDocumentSyncOptions]", t) } func (t *Or_ServerCapabilities_textDocumentSync) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder72 := json.NewDecoder(bytes.NewReader(x)) decoder72.DisallowUnknownFields() var h72 TextDocumentSyncKind if err := decoder72.Decode(&h72); err == nil { t.Value = h72 return nil } decoder73 := json.NewDecoder(bytes.NewReader(x)) decoder73.DisallowUnknownFields() var h73 TextDocumentSyncOptions if err := decoder73.Decode(&h73); err == nil { t.Value = h73 return nil } return &UnmarshalError{"unmarshal failed to match one of [TextDocumentSyncKind TextDocumentSyncOptions]"} } func (t Or_ServerCapabilities_typeDefinitionProvider) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case TypeDefinitionOptions: return json.Marshal(x) case TypeDefinitionRegistrationOptions: return json.Marshal(x) case bool: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [TypeDefinitionOptions TypeDefinitionRegistrationOptions bool]", t) } func (t *Or_ServerCapabilities_typeDefinitionProvider) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder91 := json.NewDecoder(bytes.NewReader(x)) decoder91.DisallowUnknownFields() var boolVal bool if err := decoder91.Decode(&boolVal); err == nil { t.Value = boolVal return nil } decoder92 := json.NewDecoder(bytes.NewReader(x)) decoder92.DisallowUnknownFields() var h92 TypeDefinitionOptions if err := decoder92.Decode(&h92); err == nil { t.Value = h92 return nil } decoder93 := json.NewDecoder(bytes.NewReader(x)) decoder93.DisallowUnknownFields() var h93 TypeDefinitionRegistrationOptions if err := decoder93.Decode(&h93); err == nil { t.Value = h93 return nil } return &UnmarshalError{"unmarshal failed to match one of [TypeDefinitionOptions TypeDefinitionRegistrationOptions bool]"} } func (t Or_ServerCapabilities_typeHierarchyProvider) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case TypeHierarchyOptions: return json.Marshal(x) case TypeHierarchyRegistrationOptions: return json.Marshal(x) case bool: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [TypeHierarchyOptions TypeHierarchyRegistrationOptions bool]", t) } func (t *Or_ServerCapabilities_typeHierarchyProvider) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder159 := json.NewDecoder(bytes.NewReader(x)) decoder159.DisallowUnknownFields() var boolVal bool if err := decoder159.Decode(&boolVal); err == nil { t.Value = boolVal return nil } decoder160 := json.NewDecoder(bytes.NewReader(x)) decoder160.DisallowUnknownFields() var h160 TypeHierarchyOptions if err := decoder160.Decode(&h160); err == nil { t.Value = h160 return nil } decoder161 := json.NewDecoder(bytes.NewReader(x)) decoder161.DisallowUnknownFields() var h161 TypeHierarchyRegistrationOptions if err := decoder161.Decode(&h161); err == nil { t.Value = h161 return nil } return &UnmarshalError{"unmarshal failed to match one of [TypeHierarchyOptions TypeHierarchyRegistrationOptions bool]"} } func (t Or_ServerCapabilities_workspaceSymbolProvider) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case WorkspaceSymbolOptions: return json.Marshal(x) case bool: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [WorkspaceSymbolOptions bool]", t) } func (t *Or_ServerCapabilities_workspaceSymbolProvider) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder117 := json.NewDecoder(bytes.NewReader(x)) decoder117.DisallowUnknownFields() var boolVal bool if err := decoder117.Decode(&boolVal); err == nil { t.Value = boolVal return nil } decoder118 := json.NewDecoder(bytes.NewReader(x)) decoder118.DisallowUnknownFields() var h118 WorkspaceSymbolOptions if err := decoder118.Decode(&h118); err == nil { t.Value = h118 return nil } return &UnmarshalError{"unmarshal failed to match one of [WorkspaceSymbolOptions bool]"} } func (t Or_SignatureInformation_documentation) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case MarkupContent: return json.Marshal(x) case string: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [MarkupContent string]", t) } func (t *Or_SignatureInformation_documentation) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder186 := json.NewDecoder(bytes.NewReader(x)) decoder186.DisallowUnknownFields() var stringVal string if err := decoder186.Decode(&stringVal); err == nil { t.Value = stringVal return nil } decoder187 := json.NewDecoder(bytes.NewReader(x)) decoder187.DisallowUnknownFields() var h187 MarkupContent if err := decoder187.Decode(&h187); err == nil { t.Value = h187 return nil } return &UnmarshalError{"unmarshal failed to match one of [MarkupContent string]"} } func (t Or_TextDocumentContentChangeEvent) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case TextDocumentContentChangePartial: return json.Marshal(x) case TextDocumentContentChangeWholeDocument: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [TextDocumentContentChangePartial TextDocumentContentChangeWholeDocument]", t) } func (t *Or_TextDocumentContentChangeEvent) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder263 := json.NewDecoder(bytes.NewReader(x)) decoder263.DisallowUnknownFields() var h263 TextDocumentContentChangePartial if err := decoder263.Decode(&h263); err == nil { t.Value = h263 return nil } decoder264 := json.NewDecoder(bytes.NewReader(x)) decoder264.DisallowUnknownFields() var h264 TextDocumentContentChangeWholeDocument if err := decoder264.Decode(&h264); err == nil { t.Value = h264 return nil } return &UnmarshalError{"unmarshal failed to match one of [TextDocumentContentChangePartial TextDocumentContentChangeWholeDocument]"} } func (t Or_TextDocumentEdit_edits_Elem) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case AnnotatedTextEdit: return json.Marshal(x) case SnippetTextEdit: return json.Marshal(x) case TextEdit: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [AnnotatedTextEdit SnippetTextEdit TextEdit]", t) } func (t *Or_TextDocumentEdit_edits_Elem) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder52 := json.NewDecoder(bytes.NewReader(x)) decoder52.DisallowUnknownFields() var h52 AnnotatedTextEdit if err := decoder52.Decode(&h52); err == nil { t.Value = h52 return nil } decoder53 := json.NewDecoder(bytes.NewReader(x)) decoder53.DisallowUnknownFields() var h53 SnippetTextEdit if err := decoder53.Decode(&h53); err == nil { t.Value = h53 return nil } decoder54 := json.NewDecoder(bytes.NewReader(x)) decoder54.DisallowUnknownFields() var h54 TextEdit if err := decoder54.Decode(&h54); err == nil { t.Value = h54 return nil } return &UnmarshalError{"unmarshal failed to match one of [AnnotatedTextEdit SnippetTextEdit TextEdit]"} } func (t Or_TextDocumentFilter) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case TextDocumentFilterLanguage: return json.Marshal(x) case TextDocumentFilterPattern: return json.Marshal(x) case TextDocumentFilterScheme: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [TextDocumentFilterLanguage TextDocumentFilterPattern TextDocumentFilterScheme]", t) } func (t *Or_TextDocumentFilter) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder279 := json.NewDecoder(bytes.NewReader(x)) decoder279.DisallowUnknownFields() var h279 TextDocumentFilterLanguage if err := decoder279.Decode(&h279); err == nil { t.Value = h279 return nil } decoder280 := json.NewDecoder(bytes.NewReader(x)) decoder280.DisallowUnknownFields() var h280 TextDocumentFilterPattern if err := decoder280.Decode(&h280); err == nil { t.Value = h280 return nil } decoder281 := json.NewDecoder(bytes.NewReader(x)) decoder281.DisallowUnknownFields() var h281 TextDocumentFilterScheme if err := decoder281.Decode(&h281); err == nil { t.Value = h281 return nil } return &UnmarshalError{"unmarshal failed to match one of [TextDocumentFilterLanguage TextDocumentFilterPattern TextDocumentFilterScheme]"} } func (t Or_TextDocumentSyncOptions_save) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case SaveOptions: return json.Marshal(x) case bool: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [SaveOptions bool]", t) } func (t *Or_TextDocumentSyncOptions_save) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder195 := json.NewDecoder(bytes.NewReader(x)) decoder195.DisallowUnknownFields() var boolVal bool if err := decoder195.Decode(&boolVal); err == nil { t.Value = boolVal return nil } decoder196 := json.NewDecoder(bytes.NewReader(x)) decoder196.DisallowUnknownFields() var h196 SaveOptions if err := decoder196.Decode(&h196); err == nil { t.Value = h196 return nil } return &UnmarshalError{"unmarshal failed to match one of [SaveOptions bool]"} } func (t Or_WorkspaceDocumentDiagnosticReport) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case WorkspaceFullDocumentDiagnosticReport: return json.Marshal(x) case WorkspaceUnchangedDocumentDiagnosticReport: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [WorkspaceFullDocumentDiagnosticReport WorkspaceUnchangedDocumentDiagnosticReport]", t) } func (t *Or_WorkspaceDocumentDiagnosticReport) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder259 := json.NewDecoder(bytes.NewReader(x)) decoder259.DisallowUnknownFields() var h259 WorkspaceFullDocumentDiagnosticReport if err := decoder259.Decode(&h259); err == nil { t.Value = h259 return nil } decoder260 := json.NewDecoder(bytes.NewReader(x)) decoder260.DisallowUnknownFields() var h260 WorkspaceUnchangedDocumentDiagnosticReport if err := decoder260.Decode(&h260); err == nil { t.Value = h260 return nil } return &UnmarshalError{"unmarshal failed to match one of [WorkspaceFullDocumentDiagnosticReport WorkspaceUnchangedDocumentDiagnosticReport]"} } func (t Or_WorkspaceEdit_documentChanges_Elem) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case CreateFile: return json.Marshal(x) case DeleteFile: return json.Marshal(x) case RenameFile: return json.Marshal(x) case TextDocumentEdit: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [CreateFile DeleteFile RenameFile TextDocumentEdit]", t) } func (t *Or_WorkspaceEdit_documentChanges_Elem) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder4 := json.NewDecoder(bytes.NewReader(x)) decoder4.DisallowUnknownFields() var h4 CreateFile if err := decoder4.Decode(&h4); err == nil { t.Value = h4 return nil } decoder5 := json.NewDecoder(bytes.NewReader(x)) decoder5.DisallowUnknownFields() var h5 DeleteFile if err := decoder5.Decode(&h5); err == nil { t.Value = h5 return nil } decoder6 := json.NewDecoder(bytes.NewReader(x)) decoder6.DisallowUnknownFields() var h6 RenameFile if err := decoder6.Decode(&h6); err == nil { t.Value = h6 return nil } decoder7 := json.NewDecoder(bytes.NewReader(x)) decoder7.DisallowUnknownFields() var h7 TextDocumentEdit if err := decoder7.Decode(&h7); err == nil { t.Value = h7 return nil } return &UnmarshalError{"unmarshal failed to match one of [CreateFile DeleteFile RenameFile TextDocumentEdit]"} } func (t Or_WorkspaceFoldersServerCapabilities_changeNotifications) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case bool: return json.Marshal(x) case string: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [bool string]", t) } func (t *Or_WorkspaceFoldersServerCapabilities_changeNotifications) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder210 := json.NewDecoder(bytes.NewReader(x)) decoder210.DisallowUnknownFields() var boolVal bool if err := decoder210.Decode(&boolVal); err == nil { t.Value = boolVal return nil } decoder211 := json.NewDecoder(bytes.NewReader(x)) decoder211.DisallowUnknownFields() var stringVal string if err := decoder211.Decode(&stringVal); err == nil { t.Value = stringVal return nil } return &UnmarshalError{"unmarshal failed to match one of [bool string]"} } func (t Or_WorkspaceOptions_textDocumentContent) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case TextDocumentContentOptions: return json.Marshal(x) case TextDocumentContentRegistrationOptions: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [TextDocumentContentOptions TextDocumentContentRegistrationOptions]", t) } func (t *Or_WorkspaceOptions_textDocumentContent) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder199 := json.NewDecoder(bytes.NewReader(x)) decoder199.DisallowUnknownFields() var h199 TextDocumentContentOptions if err := decoder199.Decode(&h199); err == nil { t.Value = h199 return nil } decoder200 := json.NewDecoder(bytes.NewReader(x)) decoder200.DisallowUnknownFields() var h200 TextDocumentContentRegistrationOptions if err := decoder200.Decode(&h200); err == nil { t.Value = h200 return nil } return &UnmarshalError{"unmarshal failed to match one of [TextDocumentContentOptions TextDocumentContentRegistrationOptions]"} } func (t Or_WorkspaceSymbol_location) MarshalJSON() ([]byte, error) { switch x := t.Value.(type) { case Location: return json.Marshal(x) case LocationUriOnly: return json.Marshal(x) case nil: return []byte("null"), nil } return nil, fmt.Errorf("type %T not one of [Location LocationUriOnly]", t) } func (t *Or_WorkspaceSymbol_location) UnmarshalJSON(x []byte) error { if string(x) == "null" { t.Value = nil return nil } decoder39 := json.NewDecoder(bytes.NewReader(x)) decoder39.DisallowUnknownFields() var h39 Location if err := decoder39.Decode(&h39); err == nil { t.Value = h39 return nil } decoder40 := json.NewDecoder(bytes.NewReader(x)) decoder40.DisallowUnknownFields() var h40 LocationUriOnly if err := decoder40.Decode(&h40); err == nil { t.Value = h40 return nil } return &UnmarshalError{"unmarshal failed to match one of [Location LocationUriOnly]"} } ================================================ FILE: internal/lsp/protocol/tsprotocol.go ================================================ // Copyright 2023 The Go Authors. All rights reserved. // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file. // Code generated for LSP. DO NOT EDIT. package protocol // Code generated from protocol/metaModel.json at ref release/protocol/3.17.6-next.9 (hash c94395b5da53729e6dff931293b051009ccaaaa4). // https://github.com/microsoft/vscode-languageserver-node/blob/release/protocol/3.17.6-next.9/protocol/metaModel.json // LSP metaData.version = 3.17.0. import "encoding/json" // created for And type And_RegOpt_textDocument_colorPresentation struct { WorkDoneProgressOptions TextDocumentRegistrationOptions } // A special text edit with an additional change annotation. // // @since 3.16.0. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#annotatedTextEdit type AnnotatedTextEdit struct { // The actual identifier of the change annotation AnnotationID *ChangeAnnotationIdentifier `json:"annotationId,omitempty"` TextEdit } // The parameters passed via an apply workspace edit request. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#applyWorkspaceEditParams type ApplyWorkspaceEditParams struct { // An optional label of the workspace edit. This label is // presented in the user interface for example on an undo // stack to undo the workspace edit. Label string `json:"label,omitempty"` // The edits to apply. Edit WorkspaceEdit `json:"edit"` // Additional data about the edit. // // @since 3.18.0 // @proposed Metadata *WorkspaceEditMetadata `json:"metadata,omitempty"` } // The result returned from the apply workspace edit request. // // @since 3.17 renamed from ApplyWorkspaceEditResponse // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#applyWorkspaceEditResult type ApplyWorkspaceEditResult struct { // Indicates whether the edit was applied or not. Applied bool `json:"applied"` // An optional textual description for why the edit was not applied. // This may be used by the server for diagnostic logging or to provide // a suitable error for a request that triggered the edit. FailureReason string `json:"failureReason,omitempty"` // Depending on the client's failure handling strategy `failedChange` might // contain the index of the change that failed. This property is only available // if the client signals a `failureHandlingStrategy` in its client capabilities. FailedChange uint32 `json:"failedChange,omitempty"` } // A base for all symbol information. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#baseSymbolInformation type BaseSymbolInformation struct { // The name of this symbol. Name string `json:"name"` // The kind of this symbol. Kind SymbolKind `json:"kind"` // Tags for this symbol. // // @since 3.16.0 Tags []SymbolTag `json:"tags,omitempty"` // The name of the symbol containing this symbol. This information is for // user interface purposes (e.g. to render a qualifier in the user interface // if necessary). It can't be used to re-infer a hierarchy for the document // symbols. ContainerName string `json:"containerName,omitempty"` } // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#callHierarchyClientCapabilities type CallHierarchyClientCapabilities struct { // Whether implementation supports dynamic registration. If this is set to `true` // the client supports the new `(TextDocumentRegistrationOptions & StaticRegistrationOptions)` // return value for the corresponding server capability as well. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` } // Represents an incoming call, e.g. a caller of a method or constructor. // // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#callHierarchyIncomingCall type CallHierarchyIncomingCall struct { // The item that makes the call. From CallHierarchyItem `json:"from"` // The ranges at which the calls appear. This is relative to the caller // denoted by {@link CallHierarchyIncomingCall.from `this.from`}. FromRanges []Range `json:"fromRanges"` } // The parameter of a `callHierarchy/incomingCalls` request. // // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#callHierarchyIncomingCallsParams type CallHierarchyIncomingCallsParams struct { Item CallHierarchyItem `json:"item"` WorkDoneProgressParams PartialResultParams } // Represents programming constructs like functions or constructors in the context // of call hierarchy. // // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#callHierarchyItem type CallHierarchyItem struct { // The name of this item. Name string `json:"name"` // The kind of this item. Kind SymbolKind `json:"kind"` // Tags for this item. Tags []SymbolTag `json:"tags,omitempty"` // More detail for this item, e.g. the signature of a function. Detail string `json:"detail,omitempty"` // The resource identifier of this item. URI DocumentUri `json:"uri"` // The range enclosing this symbol not including leading/trailing whitespace but everything else, e.g. comments and code. Range Range `json:"range"` // The range that should be selected and revealed when this symbol is being picked, e.g. the name of a function. // Must be contained by the {@link CallHierarchyItem.range `range`}. SelectionRange Range `json:"selectionRange"` // A data entry field that is preserved between a call hierarchy prepare and // incoming calls or outgoing calls requests. Data interface{} `json:"data,omitempty"` } // Call hierarchy options used during static registration. // // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#callHierarchyOptions type CallHierarchyOptions struct { WorkDoneProgressOptions } // Represents an outgoing call, e.g. calling a getter from a method or a method from a constructor etc. // // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#callHierarchyOutgoingCall type CallHierarchyOutgoingCall struct { // The item that is called. To CallHierarchyItem `json:"to"` // The range at which this item is called. This is the range relative to the caller, e.g the item // passed to {@link CallHierarchyItemProvider.provideCallHierarchyOutgoingCalls `provideCallHierarchyOutgoingCalls`} // and not {@link CallHierarchyOutgoingCall.to `this.to`}. FromRanges []Range `json:"fromRanges"` } // The parameter of a `callHierarchy/outgoingCalls` request. // // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#callHierarchyOutgoingCallsParams type CallHierarchyOutgoingCallsParams struct { Item CallHierarchyItem `json:"item"` WorkDoneProgressParams PartialResultParams } // The parameter of a `textDocument/prepareCallHierarchy` request. // // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#callHierarchyPrepareParams type CallHierarchyPrepareParams struct { TextDocumentPositionParams WorkDoneProgressParams } // Call hierarchy options used during static or dynamic registration. // // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#callHierarchyRegistrationOptions type CallHierarchyRegistrationOptions struct { TextDocumentRegistrationOptions CallHierarchyOptions StaticRegistrationOptions } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#cancelParams type CancelParams struct { // The request id to cancel. ID interface{} `json:"id"` } // Additional information that describes document changes. // // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#changeAnnotation type ChangeAnnotation struct { // A human-readable string describing the actual change. The string // is rendered prominent in the user interface. Label string `json:"label"` // A flag which indicates that user confirmation is needed // before applying the change. NeedsConfirmation bool `json:"needsConfirmation,omitempty"` // A human-readable string which is rendered less prominent in // the user interface. Description string `json:"description,omitempty"` } // An identifier to refer to a change annotation stored with a workspace edit. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#changeAnnotationIdentifier type ChangeAnnotationIdentifier = string // (alias) // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#changeAnnotationsSupportOptions type ChangeAnnotationsSupportOptions struct { // Whether the client groups edits with equal labels into tree nodes, // for instance all edits labelled with "Changes in Strings" would // be a tree node. GroupsOnLabel bool `json:"groupsOnLabel,omitempty"` } // Defines the capabilities provided by the client. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#clientCapabilities type ClientCapabilities struct { // Workspace specific client capabilities. Workspace WorkspaceClientCapabilities `json:"workspace,omitempty"` // Text document specific client capabilities. TextDocument TextDocumentClientCapabilities `json:"textDocument,omitempty"` // Capabilities specific to the notebook document support. // // @since 3.17.0 NotebookDocument *NotebookDocumentClientCapabilities `json:"notebookDocument,omitempty"` // Window specific client capabilities. Window WindowClientCapabilities `json:"window,omitempty"` // General client capabilities. // // @since 3.16.0 General *GeneralClientCapabilities `json:"general,omitempty"` // Experimental client capabilities. Experimental interface{} `json:"experimental,omitempty"` } // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#clientCodeActionKindOptions type ClientCodeActionKindOptions struct { // The code action kind values the client supports. When this // property exists the client also guarantees that it will // handle values outside its set gracefully and falls back // to a default value when unknown. ValueSet []CodeActionKind `json:"valueSet"` } // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#clientCodeActionLiteralOptions type ClientCodeActionLiteralOptions struct { // The code action kind is support with the following value // set. CodeActionKind ClientCodeActionKindOptions `json:"codeActionKind"` } // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#clientCodeActionResolveOptions type ClientCodeActionResolveOptions struct { // The properties that a client can resolve lazily. Properties []string `json:"properties"` } // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#clientCodeLensResolveOptions type ClientCodeLensResolveOptions struct { // The properties that a client can resolve lazily. Properties []string `json:"properties"` } // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#clientCompletionItemInsertTextModeOptions type ClientCompletionItemInsertTextModeOptions struct { ValueSet []InsertTextMode `json:"valueSet"` } // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#clientCompletionItemOptions type ClientCompletionItemOptions struct { // Client supports snippets as insert text. // // A snippet can define tab stops and placeholders with `$1`, `$2` // and `${3:foo}`. `$0` defines the final tab stop, it defaults to // the end of the snippet. Placeholders with equal identifiers are linked, // that is typing in one will update others too. SnippetSupport bool `json:"snippetSupport,omitempty"` // Client supports commit characters on a completion item. CommitCharactersSupport bool `json:"commitCharactersSupport,omitempty"` // Client supports the following content formats for the documentation // property. The order describes the preferred format of the client. DocumentationFormat []MarkupKind `json:"documentationFormat,omitempty"` // Client supports the deprecated property on a completion item. DeprecatedSupport bool `json:"deprecatedSupport,omitempty"` // Client supports the preselect property on a completion item. PreselectSupport bool `json:"preselectSupport,omitempty"` // Client supports the tag property on a completion item. Clients supporting // tags have to handle unknown tags gracefully. Clients especially need to // preserve unknown tags when sending a completion item back to the server in // a resolve call. // // @since 3.15.0 TagSupport *CompletionItemTagOptions `json:"tagSupport,omitempty"` // Client support insert replace edit to control different behavior if a // completion item is inserted in the text or should replace text. // // @since 3.16.0 InsertReplaceSupport bool `json:"insertReplaceSupport,omitempty"` // Indicates which properties a client can resolve lazily on a completion // item. Before version 3.16.0 only the predefined properties `documentation` // and `details` could be resolved lazily. // // @since 3.16.0 ResolveSupport *ClientCompletionItemResolveOptions `json:"resolveSupport,omitempty"` // The client supports the `insertTextMode` property on // a completion item to override the whitespace handling mode // as defined by the client (see `insertTextMode`). // // @since 3.16.0 InsertTextModeSupport *ClientCompletionItemInsertTextModeOptions `json:"insertTextModeSupport,omitempty"` // The client has support for completion item label // details (see also `CompletionItemLabelDetails`). // // @since 3.17.0 LabelDetailsSupport bool `json:"labelDetailsSupport,omitempty"` } // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#clientCompletionItemOptionsKind type ClientCompletionItemOptionsKind struct { // The completion item kind values the client supports. When this // property exists the client also guarantees that it will // handle values outside its set gracefully and falls back // to a default value when unknown. // // If this property is not present the client only supports // the completion items kinds from `Text` to `Reference` as defined in // the initial version of the protocol. ValueSet []CompletionItemKind `json:"valueSet,omitempty"` } // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#clientCompletionItemResolveOptions type ClientCompletionItemResolveOptions struct { // The properties that a client can resolve lazily. Properties []string `json:"properties"` } // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#clientDiagnosticsTagOptions type ClientDiagnosticsTagOptions struct { // The tags supported by the client. ValueSet []DiagnosticTag `json:"valueSet"` } // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#clientFoldingRangeKindOptions type ClientFoldingRangeKindOptions struct { // The folding range kind values the client supports. When this // property exists the client also guarantees that it will // handle values outside its set gracefully and falls back // to a default value when unknown. ValueSet []FoldingRangeKind `json:"valueSet,omitempty"` } // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#clientFoldingRangeOptions type ClientFoldingRangeOptions struct { // If set, the client signals that it supports setting collapsedText on // folding ranges to display custom labels instead of the default text. // // @since 3.17.0 CollapsedText bool `json:"collapsedText,omitempty"` } // Information about the client // // @since 3.15.0 // @since 3.18.0 ClientInfo type name added. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#clientInfo type ClientInfo struct { // The name of the client as defined by the client. Name string `json:"name"` // The client's version as defined by the client. Version string `json:"version,omitempty"` } // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#clientInlayHintResolveOptions type ClientInlayHintResolveOptions struct { // The properties that a client can resolve lazily. Properties []string `json:"properties"` } // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#clientSemanticTokensRequestFullDelta type ClientSemanticTokensRequestFullDelta struct { // The client will send the `textDocument/semanticTokens/full/delta` request if // the server provides a corresponding handler. Delta bool `json:"delta,omitempty"` } // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#clientSemanticTokensRequestOptions type ClientSemanticTokensRequestOptions struct { // The client will send the `textDocument/semanticTokens/range` request if // the server provides a corresponding handler. Range *Or_ClientSemanticTokensRequestOptions_range `json:"range,omitempty"` // The client will send the `textDocument/semanticTokens/full` request if // the server provides a corresponding handler. Full *Or_ClientSemanticTokensRequestOptions_full `json:"full,omitempty"` } // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#clientShowMessageActionItemOptions type ClientShowMessageActionItemOptions struct { // Whether the client supports additional attributes which // are preserved and send back to the server in the // request's response. AdditionalPropertiesSupport bool `json:"additionalPropertiesSupport,omitempty"` } // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#clientSignatureInformationOptions type ClientSignatureInformationOptions struct { // Client supports the following content formats for the documentation // property. The order describes the preferred format of the client. DocumentationFormat []MarkupKind `json:"documentationFormat,omitempty"` // Client capabilities specific to parameter information. ParameterInformation *ClientSignatureParameterInformationOptions `json:"parameterInformation,omitempty"` // The client supports the `activeParameter` property on `SignatureInformation` // literal. // // @since 3.16.0 ActiveParameterSupport bool `json:"activeParameterSupport,omitempty"` // The client supports the `activeParameter` property on // `SignatureHelp`/`SignatureInformation` being set to `null` to // indicate that no parameter should be active. // // @since 3.18.0 // @proposed NoActiveParameterSupport bool `json:"noActiveParameterSupport,omitempty"` } // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#clientSignatureParameterInformationOptions type ClientSignatureParameterInformationOptions struct { // The client supports processing label offsets instead of a // simple label string. // // @since 3.14.0 LabelOffsetSupport bool `json:"labelOffsetSupport,omitempty"` } // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#clientSymbolKindOptions type ClientSymbolKindOptions struct { // The symbol kind values the client supports. When this // property exists the client also guarantees that it will // handle values outside its set gracefully and falls back // to a default value when unknown. // // If this property is not present the client only supports // the symbol kinds from `File` to `Array` as defined in // the initial version of the protocol. ValueSet []SymbolKind `json:"valueSet,omitempty"` } // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#clientSymbolResolveOptions type ClientSymbolResolveOptions struct { // The properties that a client can resolve lazily. Usually // `location.range` Properties []string `json:"properties"` } // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#clientSymbolTagOptions type ClientSymbolTagOptions struct { // The tags supported by the client. ValueSet []SymbolTag `json:"valueSet"` } // A code action represents a change that can be performed in code, e.g. to fix a problem or // to refactor code. // // A CodeAction must set either `edit` and/or a `command`. If both are supplied, the `edit` is applied first, then the `command` is executed. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#codeAction type CodeAction struct { // A short, human-readable, title for this code action. Title string `json:"title"` // The kind of the code action. // // Used to filter code actions. Kind CodeActionKind `json:"kind,omitempty"` // The diagnostics that this code action resolves. Diagnostics []Diagnostic `json:"diagnostics,omitempty"` // Marks this as a preferred action. Preferred actions are used by the `auto fix` command and can be targeted // by keybindings. // // A quick fix should be marked preferred if it properly addresses the underlying error. // A refactoring should be marked preferred if it is the most reasonable choice of actions to take. // // @since 3.15.0 IsPreferred bool `json:"isPreferred,omitempty"` // Marks that the code action cannot currently be applied. // // Clients should follow the following guidelines regarding disabled code actions: // // - Disabled code actions are not shown in automatic [lightbulbs](https://code.visualstudio.com/docs/editor/editingevolved#_code-action) // code action menus. // // - Disabled actions are shown as faded out in the code action menu when the user requests a more specific type // of code action, such as refactorings. // // - If the user has a [keybinding](https://code.visualstudio.com/docs/editor/refactoring#_keybindings-for-code-actions) // that auto applies a code action and only disabled code actions are returned, the client should show the user an // error message with `reason` in the editor. // // @since 3.16.0 Disabled *CodeActionDisabled `json:"disabled,omitempty"` // The workspace edit this code action performs. Edit *WorkspaceEdit `json:"edit,omitempty"` // A command this code action executes. If a code action // provides an edit and a command, first the edit is // executed and then the command. Command *Command `json:"command,omitempty"` // A data entry field that is preserved on a code action between // a `textDocument/codeAction` and a `codeAction/resolve` request. // // @since 3.16.0 Data *json.RawMessage `json:"data,omitempty"` } // The Client Capabilities of a {@link CodeActionRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#codeActionClientCapabilities type CodeActionClientCapabilities struct { // Whether code action supports dynamic registration. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` // The client support code action literals of type `CodeAction` as a valid // response of the `textDocument/codeAction` request. If the property is not // set the request can only return `Command` literals. // // @since 3.8.0 CodeActionLiteralSupport ClientCodeActionLiteralOptions `json:"codeActionLiteralSupport,omitempty"` // Whether code action supports the `isPreferred` property. // // @since 3.15.0 IsPreferredSupport bool `json:"isPreferredSupport,omitempty"` // Whether code action supports the `disabled` property. // // @since 3.16.0 DisabledSupport bool `json:"disabledSupport,omitempty"` // Whether code action supports the `data` property which is // preserved between a `textDocument/codeAction` and a // `codeAction/resolve` request. // // @since 3.16.0 DataSupport bool `json:"dataSupport,omitempty"` // Whether the client supports resolving additional code action // properties via a separate `codeAction/resolve` request. // // @since 3.16.0 ResolveSupport *ClientCodeActionResolveOptions `json:"resolveSupport,omitempty"` // Whether the client honors the change annotations in // text edits and resource operations returned via the // `CodeAction#edit` property by for example presenting // the workspace edit in the user interface and asking // for confirmation. // // @since 3.16.0 HonorsChangeAnnotations bool `json:"honorsChangeAnnotations,omitempty"` // Whether the client supports documentation for a class of // code actions. // // @since 3.18.0 // @proposed DocumentationSupport bool `json:"documentationSupport,omitempty"` } // Contains additional diagnostic information about the context in which // a {@link CodeActionProvider.provideCodeActions code action} is run. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#codeActionContext type CodeActionContext struct { // An array of diagnostics known on the client side overlapping the range provided to the // `textDocument/codeAction` request. They are provided so that the server knows which // errors are currently presented to the user for the given range. There is no guarantee // that these accurately reflect the error state of the resource. The primary parameter // to compute code actions is the provided range. Diagnostics []Diagnostic `json:"diagnostics"` // Requested kind of actions to return. // // Actions not of this kind are filtered out by the client before being shown. So servers // can omit computing them. Only []CodeActionKind `json:"only,omitempty"` // The reason why code actions were requested. // // @since 3.17.0 TriggerKind *CodeActionTriggerKind `json:"triggerKind,omitempty"` } // Captures why the code action is currently disabled. // // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#codeActionDisabled type CodeActionDisabled struct { // Human readable description of why the code action is currently disabled. // // This is displayed in the code actions UI. Reason string `json:"reason"` } // A set of predefined code action kinds type CodeActionKind string // Documentation for a class of code actions. // // @since 3.18.0 // @proposed // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#codeActionKindDocumentation type CodeActionKindDocumentation struct { // The kind of the code action being documented. // // If the kind is generic, such as `CodeActionKind.Refactor`, the documentation will be shown whenever any // refactorings are returned. If the kind if more specific, such as `CodeActionKind.RefactorExtract`, the // documentation will only be shown when extract refactoring code actions are returned. Kind CodeActionKind `json:"kind"` // Command that is ued to display the documentation to the user. // // The title of this documentation code action is taken from {@linkcode Command.title} Command Command `json:"command"` } // Provider options for a {@link CodeActionRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#codeActionOptions type CodeActionOptions struct { // CodeActionKinds that this server may return. // // The list of kinds may be generic, such as `CodeActionKind.Refactor`, or the server // may list out every specific kind they provide. CodeActionKinds []CodeActionKind `json:"codeActionKinds,omitempty"` // Static documentation for a class of code actions. // // Documentation from the provider should be shown in the code actions menu if either: // // // - Code actions of `kind` are requested by the editor. In this case, the editor will show the documentation that // most closely matches the requested code action kind. For example, if a provider has documentation for // both `Refactor` and `RefactorExtract`, when the user requests code actions for `RefactorExtract`, // the editor will use the documentation for `RefactorExtract` instead of the documentation for `Refactor`. // // // - Any code actions of `kind` are returned by the provider. // // At most one documentation entry should be shown per provider. // // @since 3.18.0 // @proposed Documentation []CodeActionKindDocumentation `json:"documentation,omitempty"` // The server provides support to resolve additional // information for a code action. // // @since 3.16.0 ResolveProvider bool `json:"resolveProvider,omitempty"` WorkDoneProgressOptions } // The parameters of a {@link CodeActionRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#codeActionParams type CodeActionParams struct { // The document in which the command was invoked. TextDocument TextDocumentIdentifier `json:"textDocument"` // The range for which the command was invoked. Range Range `json:"range"` // Context carrying additional information. Context CodeActionContext `json:"context"` WorkDoneProgressParams PartialResultParams } // Registration options for a {@link CodeActionRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#codeActionRegistrationOptions type CodeActionRegistrationOptions struct { TextDocumentRegistrationOptions CodeActionOptions } // The reason why code actions were requested. // // @since 3.17.0 type CodeActionTriggerKind uint32 // Structure to capture a description for an error code. // // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#codeDescription type CodeDescription struct { // An URI to open with more information about the diagnostic error. Href URI `json:"href"` } // A code lens represents a {@link Command command} that should be shown along with // source text, like the number of references, a way to run tests, etc. // // A code lens is _unresolved_ when no command is associated to it. For performance // reasons the creation of a code lens and resolving should be done in two stages. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#codeLens type CodeLens struct { // The range in which this code lens is valid. Should only span a single line. Range Range `json:"range"` // The command this code lens represents. Command *Command `json:"command,omitempty"` // A data entry field that is preserved on a code lens item between // a {@link CodeLensRequest} and a {@link CodeLensResolveRequest} Data interface{} `json:"data,omitempty"` } // The client capabilities of a {@link CodeLensRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#codeLensClientCapabilities type CodeLensClientCapabilities struct { // Whether code lens supports dynamic registration. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` // Whether the client supports resolving additional code lens // properties via a separate `codeLens/resolve` request. // // @since 3.18.0 ResolveSupport *ClientCodeLensResolveOptions `json:"resolveSupport,omitempty"` } // Code Lens provider options of a {@link CodeLensRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#codeLensOptions type CodeLensOptions struct { // Code lens has a resolve provider as well. ResolveProvider bool `json:"resolveProvider,omitempty"` WorkDoneProgressOptions } // The parameters of a {@link CodeLensRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#codeLensParams type CodeLensParams struct { // The document to request code lens for. TextDocument TextDocumentIdentifier `json:"textDocument"` WorkDoneProgressParams PartialResultParams } // Registration options for a {@link CodeLensRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#codeLensRegistrationOptions type CodeLensRegistrationOptions struct { TextDocumentRegistrationOptions CodeLensOptions } // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#codeLensWorkspaceClientCapabilities type CodeLensWorkspaceClientCapabilities struct { // Whether the client implementation supports a refresh request sent from the // server to the client. // // Note that this event is global and will force the client to refresh all // code lenses currently shown. It should be used with absolute care and is // useful for situation where a server for example detect a project wide // change that requires such a calculation. RefreshSupport bool `json:"refreshSupport,omitempty"` } // Represents a color in RGBA space. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#color type Color struct { // The red component of this color in the range [0-1]. Red float64 `json:"red"` // The green component of this color in the range [0-1]. Green float64 `json:"green"` // The blue component of this color in the range [0-1]. Blue float64 `json:"blue"` // The alpha component of this color in the range [0-1]. Alpha float64 `json:"alpha"` } // Represents a color range from a document. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#colorInformation type ColorInformation struct { // The range in the document where this color appears. Range Range `json:"range"` // The actual color value for this color range. Color Color `json:"color"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#colorPresentation type ColorPresentation struct { // The label of this color presentation. It will be shown on the color // picker header. By default this is also the text that is inserted when selecting // this color presentation. Label string `json:"label"` // An {@link TextEdit edit} which is applied to a document when selecting // this presentation for the color. When `falsy` the {@link ColorPresentation.label label} // is used. TextEdit *TextEdit `json:"textEdit,omitempty"` // An optional array of additional {@link TextEdit text edits} that are applied when // selecting this color presentation. Edits must not overlap with the main {@link ColorPresentation.textEdit edit} nor with themselves. AdditionalTextEdits []TextEdit `json:"additionalTextEdits,omitempty"` } // Parameters for a {@link ColorPresentationRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#colorPresentationParams type ColorPresentationParams struct { // The text document. TextDocument TextDocumentIdentifier `json:"textDocument"` // The color to request presentations for. Color Color `json:"color"` // The range where the color would be inserted. Serves as a context. Range Range `json:"range"` WorkDoneProgressParams PartialResultParams } // Represents a reference to a command. Provides a title which // will be used to represent a command in the UI and, optionally, // an array of arguments which will be passed to the command handler // function when invoked. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#command type Command struct { // Title of the command, like `save`. Title string `json:"title"` // An optional tooltip. // // @since 3.18.0 // @proposed Tooltip string `json:"tooltip,omitempty"` // The identifier of the actual command handler. Command string `json:"command"` // Arguments that the command handler should be // invoked with. Arguments []json.RawMessage `json:"arguments,omitempty"` } // Completion client capabilities // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#completionClientCapabilities type CompletionClientCapabilities struct { // Whether completion supports dynamic registration. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` // The client supports the following `CompletionItem` specific // capabilities. CompletionItem ClientCompletionItemOptions `json:"completionItem,omitempty"` CompletionItemKind *ClientCompletionItemOptionsKind `json:"completionItemKind,omitempty"` // Defines how the client handles whitespace and indentation // when accepting a completion item that uses multi line // text in either `insertText` or `textEdit`. // // @since 3.17.0 InsertTextMode InsertTextMode `json:"insertTextMode,omitempty"` // The client supports to send additional context information for a // `textDocument/completion` request. ContextSupport bool `json:"contextSupport,omitempty"` // The client supports the following `CompletionList` specific // capabilities. // // @since 3.17.0 CompletionList *CompletionListCapabilities `json:"completionList,omitempty"` } // Contains additional information about the context in which a completion request is triggered. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#completionContext type CompletionContext struct { // How the completion was triggered. TriggerKind CompletionTriggerKind `json:"triggerKind"` // The trigger character (a single character) that has trigger code complete. // Is undefined if `triggerKind !== CompletionTriggerKind.TriggerCharacter` TriggerCharacter string `json:"triggerCharacter,omitempty"` } // A completion item represents a text snippet that is // proposed to complete text that is being typed. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#completionItem type CompletionItem struct { // The label of this completion item. // // The label property is also by default the text that // is inserted when selecting this completion. // // If label details are provided the label itself should // be an unqualified name of the completion item. Label string `json:"label"` // Additional details for the label // // @since 3.17.0 LabelDetails *CompletionItemLabelDetails `json:"labelDetails,omitempty"` // The kind of this completion item. Based of the kind // an icon is chosen by the editor. Kind CompletionItemKind `json:"kind,omitempty"` // Tags for this completion item. // // @since 3.15.0 Tags []CompletionItemTag `json:"tags,omitempty"` // A human-readable string with additional information // about this item, like type or symbol information. Detail string `json:"detail,omitempty"` // A human-readable string that represents a doc-comment. Documentation *Or_CompletionItem_documentation `json:"documentation,omitempty"` // Indicates if this item is deprecated. // @deprecated Use `tags` instead. Deprecated bool `json:"deprecated,omitempty"` // Select this item when showing. // // *Note* that only one completion item can be selected and that the // tool / client decides which item that is. The rule is that the *first* // item of those that match best is selected. Preselect bool `json:"preselect,omitempty"` // A string that should be used when comparing this item // with other items. When `falsy` the {@link CompletionItem.label label} // is used. SortText string `json:"sortText,omitempty"` // A string that should be used when filtering a set of // completion items. When `falsy` the {@link CompletionItem.label label} // is used. FilterText string `json:"filterText,omitempty"` // A string that should be inserted into a document when selecting // this completion. When `falsy` the {@link CompletionItem.label label} // is used. // // The `insertText` is subject to interpretation by the client side. // Some tools might not take the string literally. For example // VS Code when code complete is requested in this example // `con` and a completion item with an `insertText` of // `console` is provided it will only insert `sole`. Therefore it is // recommended to use `textEdit` instead since it avoids additional client // side interpretation. InsertText string `json:"insertText,omitempty"` // The format of the insert text. The format applies to both the // `insertText` property and the `newText` property of a provided // `textEdit`. If omitted defaults to `InsertTextFormat.PlainText`. // // Please note that the insertTextFormat doesn't apply to // `additionalTextEdits`. InsertTextFormat *InsertTextFormat `json:"insertTextFormat,omitempty"` // How whitespace and indentation is handled during completion // item insertion. If not provided the clients default value depends on // the `textDocument.completion.insertTextMode` client capability. // // @since 3.16.0 InsertTextMode *InsertTextMode `json:"insertTextMode,omitempty"` // An {@link TextEdit edit} which is applied to a document when selecting // this completion. When an edit is provided the value of // {@link CompletionItem.insertText insertText} is ignored. // // Most editors support two different operations when accepting a completion // item. One is to insert a completion text and the other is to replace an // existing text with a completion text. Since this can usually not be // predetermined by a server it can report both ranges. Clients need to // signal support for `InsertReplaceEdits` via the // `textDocument.completion.insertReplaceSupport` client capability // property. // // *Note 1:* The text edit's range as well as both ranges from an insert // replace edit must be a [single line] and they must contain the position // at which completion has been requested. // *Note 2:* If an `InsertReplaceEdit` is returned the edit's insert range // must be a prefix of the edit's replace range, that means it must be // contained and starting at the same position. // // @since 3.16.0 additional type `InsertReplaceEdit` TextEdit *Or_CompletionItem_textEdit `json:"textEdit,omitempty"` // The edit text used if the completion item is part of a CompletionList and // CompletionList defines an item default for the text edit range. // // Clients will only honor this property if they opt into completion list // item defaults using the capability `completionList.itemDefaults`. // // If not provided and a list's default range is provided the label // property is used as a text. // // @since 3.17.0 TextEditText string `json:"textEditText,omitempty"` // An optional array of additional {@link TextEdit text edits} that are applied when // selecting this completion. Edits must not overlap (including the same insert position) // with the main {@link CompletionItem.textEdit edit} nor with themselves. // // Additional text edits should be used to change text unrelated to the current cursor position // (for example adding an import statement at the top of the file if the completion item will // insert an unqualified type). AdditionalTextEdits []TextEdit `json:"additionalTextEdits,omitempty"` // An optional set of characters that when pressed while this completion is active will accept it first and // then type that character. *Note* that all commit characters should have `length=1` and that superfluous // characters will be ignored. CommitCharacters []string `json:"commitCharacters,omitempty"` // An optional {@link Command command} that is executed *after* inserting this completion. *Note* that // additional modifications to the current document should be described with the // {@link CompletionItem.additionalTextEdits additionalTextEdits}-property. Command *Command `json:"command,omitempty"` // A data entry field that is preserved on a completion item between a // {@link CompletionRequest} and a {@link CompletionResolveRequest}. Data interface{} `json:"data,omitempty"` } // In many cases the items of an actual completion result share the same // value for properties like `commitCharacters` or the range of a text // edit. A completion list can therefore define item defaults which will // be used if a completion item itself doesn't specify the value. // // If a completion list specifies a default value and a completion item // also specifies a corresponding value the one from the item is used. // // Servers are only allowed to return default values if the client // signals support for this via the `completionList.itemDefaults` // capability. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#completionItemDefaults type CompletionItemDefaults struct { // A default commit character set. // // @since 3.17.0 CommitCharacters []string `json:"commitCharacters,omitempty"` // A default edit range. // // @since 3.17.0 EditRange *Or_CompletionItemDefaults_editRange `json:"editRange,omitempty"` // A default insert text format. // // @since 3.17.0 InsertTextFormat *InsertTextFormat `json:"insertTextFormat,omitempty"` // A default insert text mode. // // @since 3.17.0 InsertTextMode *InsertTextMode `json:"insertTextMode,omitempty"` // A default data value. // // @since 3.17.0 Data interface{} `json:"data,omitempty"` } // The kind of a completion entry. type CompletionItemKind uint32 // Additional details for a completion item label. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#completionItemLabelDetails type CompletionItemLabelDetails struct { // An optional string which is rendered less prominently directly after {@link CompletionItem.label label}, // without any spacing. Should be used for function signatures and type annotations. Detail string `json:"detail,omitempty"` // An optional string which is rendered less prominently after {@link CompletionItem.detail}. Should be used // for fully qualified names and file paths. Description string `json:"description,omitempty"` } // Completion item tags are extra annotations that tweak the rendering of a completion // item. // // @since 3.15.0 type CompletionItemTag uint32 // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#completionItemTagOptions type CompletionItemTagOptions struct { // The tags supported by the client. ValueSet []CompletionItemTag `json:"valueSet"` } // Represents a collection of {@link CompletionItem completion items} to be presented // in the editor. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#completionList type CompletionList struct { // This list it not complete. Further typing results in recomputing this list. // // Recomputed lists have all their items replaced (not appended) in the // incomplete completion sessions. IsIncomplete bool `json:"isIncomplete"` // In many cases the items of an actual completion result share the same // value for properties like `commitCharacters` or the range of a text // edit. A completion list can therefore define item defaults which will // be used if a completion item itself doesn't specify the value. // // If a completion list specifies a default value and a completion item // also specifies a corresponding value the one from the item is used. // // Servers are only allowed to return default values if the client // signals support for this via the `completionList.itemDefaults` // capability. // // @since 3.17.0 ItemDefaults *CompletionItemDefaults `json:"itemDefaults,omitempty"` // The completion items. Items []CompletionItem `json:"items"` } // The client supports the following `CompletionList` specific // capabilities. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#completionListCapabilities type CompletionListCapabilities struct { // The client supports the following itemDefaults on // a completion list. // // The value lists the supported property names of the // `CompletionList.itemDefaults` object. If omitted // no properties are supported. // // @since 3.17.0 ItemDefaults []string `json:"itemDefaults,omitempty"` } // Completion options. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#completionOptions type CompletionOptions struct { // Most tools trigger completion request automatically without explicitly requesting // it using a keyboard shortcut (e.g. Ctrl+Space). Typically they do so when the user // starts to type an identifier. For example if the user types `c` in a JavaScript file // code complete will automatically pop up present `console` besides others as a // completion item. Characters that make up identifiers don't need to be listed here. // // If code complete should automatically be trigger on characters not being valid inside // an identifier (for example `.` in JavaScript) list them in `triggerCharacters`. TriggerCharacters []string `json:"triggerCharacters,omitempty"` // The list of all possible characters that commit a completion. This field can be used // if clients don't support individual commit characters per completion item. See // `ClientCapabilities.textDocument.completion.completionItem.commitCharactersSupport` // // If a server provides both `allCommitCharacters` and commit characters on an individual // completion item the ones on the completion item win. // // @since 3.2.0 AllCommitCharacters []string `json:"allCommitCharacters,omitempty"` // The server provides support to resolve additional // information for a completion item. ResolveProvider bool `json:"resolveProvider,omitempty"` // The server supports the following `CompletionItem` specific // capabilities. // // @since 3.17.0 CompletionItem *ServerCompletionItemOptions `json:"completionItem,omitempty"` WorkDoneProgressOptions } // Completion parameters // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#completionParams type CompletionParams struct { // The completion context. This is only available it the client specifies // to send this using the client capability `textDocument.completion.contextSupport === true` Context CompletionContext `json:"context,omitempty"` TextDocumentPositionParams WorkDoneProgressParams PartialResultParams } // Registration options for a {@link CompletionRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#completionRegistrationOptions type CompletionRegistrationOptions struct { TextDocumentRegistrationOptions CompletionOptions } // How a completion was triggered type CompletionTriggerKind uint32 // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#configurationItem type ConfigurationItem struct { // The scope to get the configuration section for. ScopeURI *URI `json:"scopeUri,omitempty"` // The configuration section asked for. Section string `json:"section,omitempty"` } // The parameters of a configuration request. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#configurationParams type ConfigurationParams struct { Items []ConfigurationItem `json:"items"` } // Create file operation. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#createFile type CreateFile struct { // A create Kind string `json:"kind"` // The resource to create. URI DocumentUri `json:"uri"` // Additional options Options *CreateFileOptions `json:"options,omitempty"` ResourceOperation } // Options to create a file. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#createFileOptions type CreateFileOptions struct { // Overwrite existing file. Overwrite wins over `ignoreIfExists` Overwrite bool `json:"overwrite,omitempty"` // Ignore if exists. IgnoreIfExists bool `json:"ignoreIfExists,omitempty"` } // The parameters sent in notifications/requests for user-initiated creation of // files. // // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#createFilesParams type CreateFilesParams struct { // An array of all files/folders created in this operation. Files []FileCreate `json:"files"` } // The declaration of a symbol representation as one or many {@link Location locations}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#declaration type Declaration = Or_Declaration // (alias) // @since 3.14.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#declarationClientCapabilities type DeclarationClientCapabilities struct { // Whether declaration supports dynamic registration. If this is set to `true` // the client supports the new `DeclarationRegistrationOptions` return value // for the corresponding server capability as well. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` // The client supports additional metadata in the form of declaration links. LinkSupport bool `json:"linkSupport,omitempty"` } // Information about where a symbol is declared. // // Provides additional metadata over normal {@link Location location} declarations, including the range of // the declaring symbol. // // Servers should prefer returning `DeclarationLink` over `Declaration` if supported // by the client. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#declarationLink type DeclarationLink = LocationLink // (alias) // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#declarationOptions type DeclarationOptions struct { WorkDoneProgressOptions } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#declarationParams type DeclarationParams struct { TextDocumentPositionParams WorkDoneProgressParams PartialResultParams } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#declarationRegistrationOptions type DeclarationRegistrationOptions struct { DeclarationOptions TextDocumentRegistrationOptions StaticRegistrationOptions } // The definition of a symbol represented as one or many {@link Location locations}. // For most programming languages there is only one location at which a symbol is // defined. // // Servers should prefer returning `DefinitionLink` over `Definition` if supported // by the client. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#definition type Definition = Or_Definition // (alias) // Client Capabilities for a {@link DefinitionRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#definitionClientCapabilities type DefinitionClientCapabilities struct { // Whether definition supports dynamic registration. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` // The client supports additional metadata in the form of definition links. // // @since 3.14.0 LinkSupport bool `json:"linkSupport,omitempty"` } // Information about where a symbol is defined. // // Provides additional metadata over normal {@link Location location} definitions, including the range of // the defining symbol // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#definitionLink type DefinitionLink = LocationLink // (alias) // Server Capabilities for a {@link DefinitionRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#definitionOptions type DefinitionOptions struct { WorkDoneProgressOptions } // Parameters for a {@link DefinitionRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#definitionParams type DefinitionParams struct { TextDocumentPositionParams WorkDoneProgressParams PartialResultParams } // Registration options for a {@link DefinitionRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#definitionRegistrationOptions type DefinitionRegistrationOptions struct { TextDocumentRegistrationOptions DefinitionOptions } // Delete file operation // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#deleteFile type DeleteFile struct { // A delete Kind string `json:"kind"` // The file to delete. URI DocumentUri `json:"uri"` // Delete options. Options *DeleteFileOptions `json:"options,omitempty"` ResourceOperation } // Delete file options // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#deleteFileOptions type DeleteFileOptions struct { // Delete the content recursively if a folder is denoted. Recursive bool `json:"recursive,omitempty"` // Ignore the operation if the file doesn't exist. IgnoreIfNotExists bool `json:"ignoreIfNotExists,omitempty"` } // The parameters sent in notifications/requests for user-initiated deletes of // files. // // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#deleteFilesParams type DeleteFilesParams struct { // An array of all files/folders deleted in this operation. Files []FileDelete `json:"files"` } // Represents a diagnostic, such as a compiler error or warning. Diagnostic objects // are only valid in the scope of a resource. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#diagnostic type Diagnostic struct { // The range at which the message applies Range Range `json:"range"` // The diagnostic's severity. To avoid interpretation mismatches when a // server is used with different clients it is highly recommended that servers // always provide a severity value. Severity DiagnosticSeverity `json:"severity,omitempty"` // The diagnostic's code, which usually appear in the user interface. Code interface{} `json:"code,omitempty"` // An optional property to describe the error code. // Requires the code field (above) to be present/not null. // // @since 3.16.0 CodeDescription *CodeDescription `json:"codeDescription,omitempty"` // A human-readable string describing the source of this // diagnostic, e.g. 'typescript' or 'super lint'. It usually // appears in the user interface. Source string `json:"source,omitempty"` // The diagnostic's message. It usually appears in the user interface Message string `json:"message"` // Additional metadata about the diagnostic. // // @since 3.15.0 Tags []DiagnosticTag `json:"tags,omitempty"` // An array of related diagnostic information, e.g. when symbol-names within // a scope collide all definitions can be marked via this property. RelatedInformation []DiagnosticRelatedInformation `json:"relatedInformation,omitempty"` // A data entry field that is preserved between a `textDocument/publishDiagnostics` // notification and `textDocument/codeAction` request. // // @since 3.16.0 Data *json.RawMessage `json:"data,omitempty"` } // Client capabilities specific to diagnostic pull requests. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#diagnosticClientCapabilities type DiagnosticClientCapabilities struct { // Whether implementation supports dynamic registration. If this is set to `true` // the client supports the new `(TextDocumentRegistrationOptions & StaticRegistrationOptions)` // return value for the corresponding server capability as well. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` // Whether the clients supports related documents for document diagnostic pulls. RelatedDocumentSupport bool `json:"relatedDocumentSupport,omitempty"` DiagnosticsCapabilities } // Diagnostic options. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#diagnosticOptions type DiagnosticOptions struct { // An optional identifier under which the diagnostics are // managed by the client. Identifier string `json:"identifier,omitempty"` // Whether the language has inter file dependencies meaning that // editing code in one file can result in a different diagnostic // set in another file. Inter file dependencies are common for // most programming languages and typically uncommon for linters. InterFileDependencies bool `json:"interFileDependencies"` // The server provides support for workspace diagnostics as well. WorkspaceDiagnostics bool `json:"workspaceDiagnostics"` WorkDoneProgressOptions } // Diagnostic registration options. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#diagnosticRegistrationOptions type DiagnosticRegistrationOptions struct { TextDocumentRegistrationOptions DiagnosticOptions StaticRegistrationOptions } // Represents a related message and source code location for a diagnostic. This should be // used to point to code locations that cause or related to a diagnostics, e.g when duplicating // a symbol in a scope. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#diagnosticRelatedInformation type DiagnosticRelatedInformation struct { // The location of this related diagnostic information. Location Location `json:"location"` // The message of this related diagnostic information. Message string `json:"message"` } // Cancellation data returned from a diagnostic request. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#diagnosticServerCancellationData type DiagnosticServerCancellationData struct { RetriggerRequest bool `json:"retriggerRequest"` } // The diagnostic's severity. type DiagnosticSeverity uint32 // The diagnostic tags. // // @since 3.15.0 type DiagnosticTag uint32 // Workspace client capabilities specific to diagnostic pull requests. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#diagnosticWorkspaceClientCapabilities type DiagnosticWorkspaceClientCapabilities struct { // Whether the client implementation supports a refresh request sent from // the server to the client. // // Note that this event is global and will force the client to refresh all // pulled diagnostics currently shown. It should be used with absolute care and // is useful for situation where a server for example detects a project wide // change that requires such a calculation. RefreshSupport bool `json:"refreshSupport,omitempty"` } // General diagnostics capabilities for pull and push model. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#diagnosticsCapabilities type DiagnosticsCapabilities struct { // Whether the clients accepts diagnostics with related information. RelatedInformation bool `json:"relatedInformation,omitempty"` // Client supports the tag property to provide meta data about a diagnostic. // Clients supporting tags have to handle unknown tags gracefully. // // @since 3.15.0 TagSupport *ClientDiagnosticsTagOptions `json:"tagSupport,omitempty"` // Client supports a codeDescription property // // @since 3.16.0 CodeDescriptionSupport bool `json:"codeDescriptionSupport,omitempty"` // Whether code action supports the `data` property which is // preserved between a `textDocument/publishDiagnostics` and // `textDocument/codeAction` request. // // @since 3.16.0 DataSupport bool `json:"dataSupport,omitempty"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#didChangeConfigurationClientCapabilities type DidChangeConfigurationClientCapabilities struct { // Did change configuration notification supports dynamic registration. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` } // The parameters of a change configuration notification. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#didChangeConfigurationParams type DidChangeConfigurationParams struct { // The actual changed settings Settings interface{} `json:"settings"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#didChangeConfigurationRegistrationOptions type DidChangeConfigurationRegistrationOptions struct { Section *Or_DidChangeConfigurationRegistrationOptions_section `json:"section,omitempty"` } // The params sent in a change notebook document notification. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#didChangeNotebookDocumentParams type DidChangeNotebookDocumentParams struct { // The notebook document that did change. The version number points // to the version after all provided changes have been applied. If // only the text document content of a cell changes the notebook version // doesn't necessarily have to change. NotebookDocument VersionedNotebookDocumentIdentifier `json:"notebookDocument"` // The actual changes to the notebook document. // // The changes describe single state changes to the notebook document. // So if there are two changes c1 (at array index 0) and c2 (at array // index 1) for a notebook in state S then c1 moves the notebook from // S to S' and c2 from S' to S''. So c1 is computed on the state S and // c2 is computed on the state S'. // // To mirror the content of a notebook using change events use the following approach: // // - start with the same initial content // - apply the 'notebookDocument/didChange' notifications in the order you receive them. // - apply the `NotebookChangeEvent`s in a single notification in the order // you receive them. Change NotebookDocumentChangeEvent `json:"change"` } // The change text document notification's parameters. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#didChangeTextDocumentParams type DidChangeTextDocumentParams struct { // The document that did change. The version number points // to the version after all provided content changes have // been applied. TextDocument VersionedTextDocumentIdentifier `json:"textDocument"` // The actual content changes. The content changes describe single state changes // to the document. So if there are two content changes c1 (at array index 0) and // c2 (at array index 1) for a document in state S then c1 moves the document from // S to S' and c2 from S' to S''. So c1 is computed on the state S and c2 is computed // on the state S'. // // To mirror the content of a document using change events use the following approach: // // - start with the same initial content // - apply the 'textDocument/didChange' notifications in the order you receive them. // - apply the `TextDocumentContentChangeEvent`s in a single notification in the order // you receive them. ContentChanges []TextDocumentContentChangeEvent `json:"contentChanges"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#didChangeWatchedFilesClientCapabilities type DidChangeWatchedFilesClientCapabilities struct { // Did change watched files notification supports dynamic registration. Please note // that the current protocol doesn't support static configuration for file changes // from the server side. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` // Whether the client has support for {@link RelativePattern relative pattern} // or not. // // @since 3.17.0 RelativePatternSupport bool `json:"relativePatternSupport,omitempty"` } // The watched files change notification's parameters. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#didChangeWatchedFilesParams type DidChangeWatchedFilesParams struct { // The actual file events. Changes []FileEvent `json:"changes"` } // Describe options to be used when registered for text document change events. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#didChangeWatchedFilesRegistrationOptions type DidChangeWatchedFilesRegistrationOptions struct { // The watchers to register. Watchers []FileSystemWatcher `json:"watchers"` } // The parameters of a `workspace/didChangeWorkspaceFolders` notification. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#didChangeWorkspaceFoldersParams type DidChangeWorkspaceFoldersParams struct { // The actual workspace folder change event. Event WorkspaceFoldersChangeEvent `json:"event"` } // The params sent in a close notebook document notification. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#didCloseNotebookDocumentParams type DidCloseNotebookDocumentParams struct { // The notebook document that got closed. NotebookDocument NotebookDocumentIdentifier `json:"notebookDocument"` // The text documents that represent the content // of a notebook cell that got closed. CellTextDocuments []TextDocumentIdentifier `json:"cellTextDocuments"` } // The parameters sent in a close text document notification // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#didCloseTextDocumentParams type DidCloseTextDocumentParams struct { // The document that was closed. TextDocument TextDocumentIdentifier `json:"textDocument"` } // The params sent in an open notebook document notification. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#didOpenNotebookDocumentParams type DidOpenNotebookDocumentParams struct { // The notebook document that got opened. NotebookDocument NotebookDocument `json:"notebookDocument"` // The text documents that represent the content // of a notebook cell. CellTextDocuments []TextDocumentItem `json:"cellTextDocuments"` } // The parameters sent in an open text document notification // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#didOpenTextDocumentParams type DidOpenTextDocumentParams struct { // The document that was opened. TextDocument TextDocumentItem `json:"textDocument"` } // The params sent in a save notebook document notification. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#didSaveNotebookDocumentParams type DidSaveNotebookDocumentParams struct { // The notebook document that got saved. NotebookDocument NotebookDocumentIdentifier `json:"notebookDocument"` } // The parameters sent in a save text document notification // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#didSaveTextDocumentParams type DidSaveTextDocumentParams struct { // The document that was saved. TextDocument TextDocumentIdentifier `json:"textDocument"` // Optional the content when saved. Depends on the includeText value // when the save notification was requested. Text *string `json:"text,omitempty"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentColorClientCapabilities type DocumentColorClientCapabilities struct { // Whether implementation supports dynamic registration. If this is set to `true` // the client supports the new `DocumentColorRegistrationOptions` return value // for the corresponding server capability as well. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentColorOptions type DocumentColorOptions struct { WorkDoneProgressOptions } // Parameters for a {@link DocumentColorRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentColorParams type DocumentColorParams struct { // The text document. TextDocument TextDocumentIdentifier `json:"textDocument"` WorkDoneProgressParams PartialResultParams } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentColorRegistrationOptions type DocumentColorRegistrationOptions struct { TextDocumentRegistrationOptions DocumentColorOptions StaticRegistrationOptions } // Parameters of the document diagnostic request. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentDiagnosticParams type DocumentDiagnosticParams struct { // The text document. TextDocument TextDocumentIdentifier `json:"textDocument"` // The additional identifier provided during registration. Identifier string `json:"identifier,omitempty"` // The result id of a previous response if provided. PreviousResultID string `json:"previousResultId,omitempty"` WorkDoneProgressParams PartialResultParams } // The result of a document diagnostic pull request. A report can // either be a full report containing all diagnostics for the // requested document or an unchanged report indicating that nothing // has changed in terms of diagnostics in comparison to the last // pull request. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentDiagnosticReport type DocumentDiagnosticReport = Or_DocumentDiagnosticReport // (alias) // The document diagnostic report kinds. // // @since 3.17.0 type DocumentDiagnosticReportKind string // A partial result for a document diagnostic report. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentDiagnosticReportPartialResult type DocumentDiagnosticReportPartialResult struct { RelatedDocuments map[DocumentUri]interface{} `json:"relatedDocuments"` } // A document filter describes a top level text document or // a notebook cell document. // // @since 3.17.0 - proposed support for NotebookCellTextDocumentFilter. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentFilter type DocumentFilter = Or_DocumentFilter // (alias) // Client capabilities of a {@link DocumentFormattingRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentFormattingClientCapabilities type DocumentFormattingClientCapabilities struct { // Whether formatting supports dynamic registration. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` } // Provider options for a {@link DocumentFormattingRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentFormattingOptions type DocumentFormattingOptions struct { WorkDoneProgressOptions } // The parameters of a {@link DocumentFormattingRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentFormattingParams type DocumentFormattingParams struct { // The document to format. TextDocument TextDocumentIdentifier `json:"textDocument"` // The format options. Options FormattingOptions `json:"options"` WorkDoneProgressParams } // Registration options for a {@link DocumentFormattingRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentFormattingRegistrationOptions type DocumentFormattingRegistrationOptions struct { TextDocumentRegistrationOptions DocumentFormattingOptions } // A document highlight is a range inside a text document which deserves // special attention. Usually a document highlight is visualized by changing // the background color of its range. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentHighlight type DocumentHighlight struct { // The range this highlight applies to. Range Range `json:"range"` // The highlight kind, default is {@link DocumentHighlightKind.Text text}. Kind DocumentHighlightKind `json:"kind,omitempty"` } // Client Capabilities for a {@link DocumentHighlightRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentHighlightClientCapabilities type DocumentHighlightClientCapabilities struct { // Whether document highlight supports dynamic registration. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` } // A document highlight kind. type DocumentHighlightKind uint32 // Provider options for a {@link DocumentHighlightRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentHighlightOptions type DocumentHighlightOptions struct { WorkDoneProgressOptions } // Parameters for a {@link DocumentHighlightRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentHighlightParams type DocumentHighlightParams struct { TextDocumentPositionParams WorkDoneProgressParams PartialResultParams } // Registration options for a {@link DocumentHighlightRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentHighlightRegistrationOptions type DocumentHighlightRegistrationOptions struct { TextDocumentRegistrationOptions DocumentHighlightOptions } // A document link is a range in a text document that links to an internal or external resource, like another // text document or a web site. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentLink type DocumentLink struct { // The range this link applies to. Range Range `json:"range"` // The uri this link points to. If missing a resolve request is sent later. Target *URI `json:"target,omitempty"` // The tooltip text when you hover over this link. // // If a tooltip is provided, is will be displayed in a string that includes instructions on how to // trigger the link, such as `{0} (ctrl + click)`. The specific instructions vary depending on OS, // user settings, and localization. // // @since 3.15.0 Tooltip string `json:"tooltip,omitempty"` // A data entry field that is preserved on a document link between a // DocumentLinkRequest and a DocumentLinkResolveRequest. Data interface{} `json:"data,omitempty"` } // The client capabilities of a {@link DocumentLinkRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentLinkClientCapabilities type DocumentLinkClientCapabilities struct { // Whether document link supports dynamic registration. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` // Whether the client supports the `tooltip` property on `DocumentLink`. // // @since 3.15.0 TooltipSupport bool `json:"tooltipSupport,omitempty"` } // Provider options for a {@link DocumentLinkRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentLinkOptions type DocumentLinkOptions struct { // Document links have a resolve provider as well. ResolveProvider bool `json:"resolveProvider,omitempty"` WorkDoneProgressOptions } // The parameters of a {@link DocumentLinkRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentLinkParams type DocumentLinkParams struct { // The document to provide document links for. TextDocument TextDocumentIdentifier `json:"textDocument"` WorkDoneProgressParams PartialResultParams } // Registration options for a {@link DocumentLinkRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentLinkRegistrationOptions type DocumentLinkRegistrationOptions struct { TextDocumentRegistrationOptions DocumentLinkOptions } // Client capabilities of a {@link DocumentOnTypeFormattingRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentOnTypeFormattingClientCapabilities type DocumentOnTypeFormattingClientCapabilities struct { // Whether on type formatting supports dynamic registration. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` } // Provider options for a {@link DocumentOnTypeFormattingRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentOnTypeFormattingOptions type DocumentOnTypeFormattingOptions struct { // A character on which formatting should be triggered, like `{`. FirstTriggerCharacter string `json:"firstTriggerCharacter"` // More trigger characters. MoreTriggerCharacter []string `json:"moreTriggerCharacter,omitempty"` } // The parameters of a {@link DocumentOnTypeFormattingRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentOnTypeFormattingParams type DocumentOnTypeFormattingParams struct { // The document to format. TextDocument TextDocumentIdentifier `json:"textDocument"` // The position around which the on type formatting should happen. // This is not necessarily the exact position where the character denoted // by the property `ch` got typed. Position Position `json:"position"` // The character that has been typed that triggered the formatting // on type request. That is not necessarily the last character that // got inserted into the document since the client could auto insert // characters as well (e.g. like automatic brace completion). Ch string `json:"ch"` // The formatting options. Options FormattingOptions `json:"options"` } // Registration options for a {@link DocumentOnTypeFormattingRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentOnTypeFormattingRegistrationOptions type DocumentOnTypeFormattingRegistrationOptions struct { TextDocumentRegistrationOptions DocumentOnTypeFormattingOptions } // Client capabilities of a {@link DocumentRangeFormattingRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentRangeFormattingClientCapabilities type DocumentRangeFormattingClientCapabilities struct { // Whether range formatting supports dynamic registration. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` // Whether the client supports formatting multiple ranges at once. // // @since 3.18.0 // @proposed RangesSupport bool `json:"rangesSupport,omitempty"` } // Provider options for a {@link DocumentRangeFormattingRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentRangeFormattingOptions type DocumentRangeFormattingOptions struct { // Whether the server supports formatting multiple ranges at once. // // @since 3.18.0 // @proposed RangesSupport bool `json:"rangesSupport,omitempty"` WorkDoneProgressOptions } // The parameters of a {@link DocumentRangeFormattingRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentRangeFormattingParams type DocumentRangeFormattingParams struct { // The document to format. TextDocument TextDocumentIdentifier `json:"textDocument"` // The range to format Range Range `json:"range"` // The format options Options FormattingOptions `json:"options"` WorkDoneProgressParams } // Registration options for a {@link DocumentRangeFormattingRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentRangeFormattingRegistrationOptions type DocumentRangeFormattingRegistrationOptions struct { TextDocumentRegistrationOptions DocumentRangeFormattingOptions } // The parameters of a {@link DocumentRangesFormattingRequest}. // // @since 3.18.0 // @proposed // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentRangesFormattingParams type DocumentRangesFormattingParams struct { // The document to format. TextDocument TextDocumentIdentifier `json:"textDocument"` // The ranges to format Ranges []Range `json:"ranges"` // The format options Options FormattingOptions `json:"options"` WorkDoneProgressParams } // A document selector is the combination of one or many document filters. // // @sample `let sel:DocumentSelector = [{ language: 'typescript' }, { language: 'json', pattern: '**∕tsconfig.json' }]`; // // The use of a string as a document filter is deprecated @since 3.16.0. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentSelector type DocumentSelector = []DocumentFilter // (alias) // Represents programming constructs like variables, classes, interfaces etc. // that appear in a document. Document symbols can be hierarchical and they // have two ranges: one that encloses its definition and one that points to // its most interesting range, e.g. the range of an identifier. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentSymbol type DocumentSymbol struct { // The name of this symbol. Will be displayed in the user interface and therefore must not be // an empty string or a string only consisting of white spaces. Name string `json:"name"` // More detail for this symbol, e.g the signature of a function. Detail string `json:"detail,omitempty"` // The kind of this symbol. Kind SymbolKind `json:"kind"` // Tags for this document symbol. // // @since 3.16.0 Tags []SymbolTag `json:"tags,omitempty"` // Indicates if this symbol is deprecated. // // @deprecated Use tags instead Deprecated bool `json:"deprecated,omitempty"` // The range enclosing this symbol not including leading/trailing whitespace but everything else // like comments. This information is typically used to determine if the clients cursor is // inside the symbol to reveal in the symbol in the UI. Range Range `json:"range"` // The range that should be selected and revealed when this symbol is being picked, e.g the name of a function. // Must be contained by the `range`. SelectionRange Range `json:"selectionRange"` // Children of this symbol, e.g. properties of a class. Children []DocumentSymbol `json:"children,omitempty"` } // Client Capabilities for a {@link DocumentSymbolRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentSymbolClientCapabilities type DocumentSymbolClientCapabilities struct { // Whether document symbol supports dynamic registration. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` // Specific capabilities for the `SymbolKind` in the // `textDocument/documentSymbol` request. SymbolKind *ClientSymbolKindOptions `json:"symbolKind,omitempty"` // The client supports hierarchical document symbols. HierarchicalDocumentSymbolSupport bool `json:"hierarchicalDocumentSymbolSupport,omitempty"` // The client supports tags on `SymbolInformation`. Tags are supported on // `DocumentSymbol` if `hierarchicalDocumentSymbolSupport` is set to true. // Clients supporting tags have to handle unknown tags gracefully. // // @since 3.16.0 TagSupport *ClientSymbolTagOptions `json:"tagSupport,omitempty"` // The client supports an additional label presented in the UI when // registering a document symbol provider. // // @since 3.16.0 LabelSupport bool `json:"labelSupport,omitempty"` } // Provider options for a {@link DocumentSymbolRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentSymbolOptions type DocumentSymbolOptions struct { // A human-readable string that is shown when multiple outlines trees // are shown for the same document. // // @since 3.16.0 Label string `json:"label,omitempty"` WorkDoneProgressOptions } // Parameters for a {@link DocumentSymbolRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentSymbolParams type DocumentSymbolParams struct { // The text document. TextDocument TextDocumentIdentifier `json:"textDocument"` WorkDoneProgressParams PartialResultParams } // Registration options for a {@link DocumentSymbolRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#documentSymbolRegistrationOptions type DocumentSymbolRegistrationOptions struct { TextDocumentRegistrationOptions DocumentSymbolOptions } // Edit range variant that includes ranges for insert and replace operations. // // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#editRangeWithInsertReplace type EditRangeWithInsertReplace struct { Insert Range `json:"insert"` Replace Range `json:"replace"` } // Predefined error codes. type ErrorCodes int32 // The client capabilities of a {@link ExecuteCommandRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#executeCommandClientCapabilities type ExecuteCommandClientCapabilities struct { // Execute command supports dynamic registration. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` } // The server capabilities of a {@link ExecuteCommandRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#executeCommandOptions type ExecuteCommandOptions struct { // The commands to be executed on the server Commands []string `json:"commands"` WorkDoneProgressOptions } // The parameters of a {@link ExecuteCommandRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#executeCommandParams type ExecuteCommandParams struct { // The identifier of the actual command handler. Command string `json:"command"` // Arguments that the command should be invoked with. Arguments []json.RawMessage `json:"arguments,omitempty"` WorkDoneProgressParams } // Registration options for a {@link ExecuteCommandRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#executeCommandRegistrationOptions type ExecuteCommandRegistrationOptions struct { ExecuteCommandOptions } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#executionSummary type ExecutionSummary struct { // A strict monotonically increasing value // indicating the execution order of a cell // inside a notebook. ExecutionOrder uint32 `json:"executionOrder"` // Whether the execution was successful or // not if known by the client. Success bool `json:"success,omitempty"` } type FailureHandlingKind string // The file event type type FileChangeType uint32 // Represents information on a file/folder create. // // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#fileCreate type FileCreate struct { // A file:// URI for the location of the file/folder being created. URI string `json:"uri"` } // Represents information on a file/folder delete. // // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#fileDelete type FileDelete struct { // A file:// URI for the location of the file/folder being deleted. URI string `json:"uri"` } // An event describing a file change. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#fileEvent type FileEvent struct { // The file's uri. URI DocumentUri `json:"uri"` // The change type. Type FileChangeType `json:"type"` } // Capabilities relating to events from file operations by the user in the client. // // These events do not come from the file system, they come from user operations // like renaming a file in the UI. // // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#fileOperationClientCapabilities type FileOperationClientCapabilities struct { // Whether the client supports dynamic registration for file requests/notifications. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` // The client has support for sending didCreateFiles notifications. DidCreate bool `json:"didCreate,omitempty"` // The client has support for sending willCreateFiles requests. WillCreate bool `json:"willCreate,omitempty"` // The client has support for sending didRenameFiles notifications. DidRename bool `json:"didRename,omitempty"` // The client has support for sending willRenameFiles requests. WillRename bool `json:"willRename,omitempty"` // The client has support for sending didDeleteFiles notifications. DidDelete bool `json:"didDelete,omitempty"` // The client has support for sending willDeleteFiles requests. WillDelete bool `json:"willDelete,omitempty"` } // A filter to describe in which file operation requests or notifications // the server is interested in receiving. // // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#fileOperationFilter type FileOperationFilter struct { // A Uri scheme like `file` or `untitled`. Scheme string `json:"scheme,omitempty"` // The actual file operation pattern. Pattern FileOperationPattern `json:"pattern"` } // Options for notifications/requests for user operations on files. // // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#fileOperationOptions type FileOperationOptions struct { // The server is interested in receiving didCreateFiles notifications. DidCreate *FileOperationRegistrationOptions `json:"didCreate,omitempty"` // The server is interested in receiving willCreateFiles requests. WillCreate *FileOperationRegistrationOptions `json:"willCreate,omitempty"` // The server is interested in receiving didRenameFiles notifications. DidRename *FileOperationRegistrationOptions `json:"didRename,omitempty"` // The server is interested in receiving willRenameFiles requests. WillRename *FileOperationRegistrationOptions `json:"willRename,omitempty"` // The server is interested in receiving didDeleteFiles file notifications. DidDelete *FileOperationRegistrationOptions `json:"didDelete,omitempty"` // The server is interested in receiving willDeleteFiles file requests. WillDelete *FileOperationRegistrationOptions `json:"willDelete,omitempty"` } // A pattern to describe in which file operation requests or notifications // the server is interested in receiving. // // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#fileOperationPattern type FileOperationPattern struct { // The glob pattern to match. Glob patterns can have the following syntax: // // - `*` to match one or more characters in a path segment // - `?` to match on one character in a path segment // - `**` to match any number of path segments, including none // - `{}` to group sub patterns into an OR expression. (e.g. `**​/*.{ts,js}` matches all TypeScript and JavaScript files) // - `[]` to declare a range of characters to match in a path segment (e.g., `example.[0-9]` to match on `example.0`, `example.1`, …) // - `[!...]` to negate a range of characters to match in a path segment (e.g., `example.[!0-9]` to match on `example.a`, `example.b`, but not `example.0`) Glob string `json:"glob"` // Whether to match files or folders with this pattern. // // Matches both if undefined. Matches *FileOperationPatternKind `json:"matches,omitempty"` // Additional options used during matching. Options *FileOperationPatternOptions `json:"options,omitempty"` } // A pattern kind describing if a glob pattern matches a file a folder or // both. // // @since 3.16.0 type FileOperationPatternKind string // Matching options for the file operation pattern. // // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#fileOperationPatternOptions type FileOperationPatternOptions struct { // The pattern should be matched ignoring casing. IgnoreCase bool `json:"ignoreCase,omitempty"` } // The options to register for file operations. // // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#fileOperationRegistrationOptions type FileOperationRegistrationOptions struct { // The actual filters. Filters []FileOperationFilter `json:"filters"` } // Represents information on a file/folder rename. // // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#fileRename type FileRename struct { // A file:// URI for the original location of the file/folder being renamed. OldURI string `json:"oldUri"` // A file:// URI for the new location of the file/folder being renamed. NewURI string `json:"newUri"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#fileSystemWatcher type FileSystemWatcher struct { // The glob pattern to watch. See {@link GlobPattern glob pattern} for more detail. // // @since 3.17.0 support for relative patterns. GlobPattern GlobPattern `json:"globPattern"` // The kind of events of interest. If omitted it defaults // to WatchKind.Create | WatchKind.Change | WatchKind.Delete // which is 7. Kind *WatchKind `json:"kind,omitempty"` } // Represents a folding range. To be valid, start and end line must be bigger than zero and smaller // than the number of lines in the document. Clients are free to ignore invalid ranges. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#foldingRange type FoldingRange struct { // The zero-based start line of the range to fold. The folded area starts after the line's last character. // To be valid, the end must be zero or larger and smaller than the number of lines in the document. StartLine uint32 `json:"startLine"` // The zero-based character offset from where the folded range starts. If not defined, defaults to the length of the start line. StartCharacter uint32 `json:"startCharacter,omitempty"` // The zero-based end line of the range to fold. The folded area ends with the line's last character. // To be valid, the end must be zero or larger and smaller than the number of lines in the document. EndLine uint32 `json:"endLine"` // The zero-based character offset before the folded range ends. If not defined, defaults to the length of the end line. EndCharacter uint32 `json:"endCharacter,omitempty"` // Describes the kind of the folding range such as 'comment' or 'region'. The kind // is used to categorize folding ranges and used by commands like 'Fold all comments'. // See {@link FoldingRangeKind} for an enumeration of standardized kinds. Kind string `json:"kind,omitempty"` // The text that the client should show when the specified range is // collapsed. If not defined or not supported by the client, a default // will be chosen by the client. // // @since 3.17.0 CollapsedText string `json:"collapsedText,omitempty"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#foldingRangeClientCapabilities type FoldingRangeClientCapabilities struct { // Whether implementation supports dynamic registration for folding range // providers. If this is set to `true` the client supports the new // `FoldingRangeRegistrationOptions` return value for the corresponding // server capability as well. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` // The maximum number of folding ranges that the client prefers to receive // per document. The value serves as a hint, servers are free to follow the // limit. RangeLimit uint32 `json:"rangeLimit,omitempty"` // If set, the client signals that it only supports folding complete lines. // If set, client will ignore specified `startCharacter` and `endCharacter` // properties in a FoldingRange. LineFoldingOnly bool `json:"lineFoldingOnly,omitempty"` // Specific options for the folding range kind. // // @since 3.17.0 FoldingRangeKind *ClientFoldingRangeKindOptions `json:"foldingRangeKind,omitempty"` // Specific options for the folding range. // // @since 3.17.0 FoldingRange *ClientFoldingRangeOptions `json:"foldingRange,omitempty"` } // A set of predefined range kinds. type FoldingRangeKind string // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#foldingRangeOptions type FoldingRangeOptions struct { WorkDoneProgressOptions } // Parameters for a {@link FoldingRangeRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#foldingRangeParams type FoldingRangeParams struct { // The text document. TextDocument TextDocumentIdentifier `json:"textDocument"` WorkDoneProgressParams PartialResultParams } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#foldingRangeRegistrationOptions type FoldingRangeRegistrationOptions struct { TextDocumentRegistrationOptions FoldingRangeOptions StaticRegistrationOptions } // Client workspace capabilities specific to folding ranges // // @since 3.18.0 // @proposed // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#foldingRangeWorkspaceClientCapabilities type FoldingRangeWorkspaceClientCapabilities struct { // Whether the client implementation supports a refresh request sent from the // server to the client. // // Note that this event is global and will force the client to refresh all // folding ranges currently shown. It should be used with absolute care and is // useful for situation where a server for example detects a project wide // change that requires such a calculation. // // @since 3.18.0 // @proposed RefreshSupport bool `json:"refreshSupport,omitempty"` } // Value-object describing what options formatting should use. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#formattingOptions type FormattingOptions struct { // Size of a tab in spaces. TabSize uint32 `json:"tabSize"` // Prefer spaces over tabs. InsertSpaces bool `json:"insertSpaces"` // Trim trailing whitespace on a line. // // @since 3.15.0 TrimTrailingWhitespace bool `json:"trimTrailingWhitespace,omitempty"` // Insert a newline character at the end of the file if one does not exist. // // @since 3.15.0 InsertFinalNewline bool `json:"insertFinalNewline,omitempty"` // Trim all newlines after the final newline at the end of the file. // // @since 3.15.0 TrimFinalNewlines bool `json:"trimFinalNewlines,omitempty"` } // A diagnostic report with a full set of problems. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#fullDocumentDiagnosticReport type FullDocumentDiagnosticReport struct { // A full document diagnostic report. Kind string `json:"kind"` // An optional result id. If provided it will // be sent on the next diagnostic request for the // same document. ResultID string `json:"resultId,omitempty"` // The actual items. Items []Diagnostic `json:"items"` } // General client capabilities. // // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#generalClientCapabilities type GeneralClientCapabilities struct { // Client capability that signals how the client // handles stale requests (e.g. a request // for which the client will not process the response // anymore since the information is outdated). // // @since 3.17.0 StaleRequestSupport *StaleRequestSupportOptions `json:"staleRequestSupport,omitempty"` // Client capabilities specific to regular expressions. // // @since 3.16.0 RegularExpressions *RegularExpressionsClientCapabilities `json:"regularExpressions,omitempty"` // Client capabilities specific to the client's markdown parser. // // @since 3.16.0 Markdown *MarkdownClientCapabilities `json:"markdown,omitempty"` // The position encodings supported by the client. Client and server // have to agree on the same position encoding to ensure that offsets // (e.g. character position in a line) are interpreted the same on both // sides. // // To keep the protocol backwards compatible the following applies: if // the value 'utf-16' is missing from the array of position encodings // servers can assume that the client supports UTF-16. UTF-16 is // therefore a mandatory encoding. // // If omitted it defaults to ['utf-16']. // // Implementation considerations: since the conversion from one encoding // into another requires the content of the file / line the conversion // is best done where the file is read which is usually on the server // side. // // @since 3.17.0 PositionEncodings []PositionEncodingKind `json:"positionEncodings,omitempty"` } // The glob pattern. Either a string pattern or a relative pattern. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#globPattern type GlobPattern = Or_GlobPattern // (alias) // The result of a hover request. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#hover type Hover struct { // The hover's content Contents MarkupContent `json:"contents"` // An optional range inside the text document that is used to // visualize the hover, e.g. by changing the background color. Range Range `json:"range,omitempty"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#hoverClientCapabilities type HoverClientCapabilities struct { // Whether hover supports dynamic registration. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` // Client supports the following content formats for the content // property. The order describes the preferred format of the client. ContentFormat []MarkupKind `json:"contentFormat,omitempty"` } // Hover options. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#hoverOptions type HoverOptions struct { WorkDoneProgressOptions } // Parameters for a {@link HoverRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#hoverParams type HoverParams struct { TextDocumentPositionParams WorkDoneProgressParams } // Registration options for a {@link HoverRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#hoverRegistrationOptions type HoverRegistrationOptions struct { TextDocumentRegistrationOptions HoverOptions } // @since 3.6.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#implementationClientCapabilities type ImplementationClientCapabilities struct { // Whether implementation supports dynamic registration. If this is set to `true` // the client supports the new `ImplementationRegistrationOptions` return value // for the corresponding server capability as well. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` // The client supports additional metadata in the form of definition links. // // @since 3.14.0 LinkSupport bool `json:"linkSupport,omitempty"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#implementationOptions type ImplementationOptions struct { WorkDoneProgressOptions } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#implementationParams type ImplementationParams struct { TextDocumentPositionParams WorkDoneProgressParams PartialResultParams } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#implementationRegistrationOptions type ImplementationRegistrationOptions struct { TextDocumentRegistrationOptions ImplementationOptions StaticRegistrationOptions } // The data type of the ResponseError if the // initialize request fails. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#initializeError type InitializeError struct { // Indicates whether the client execute the following retry logic: // (1) show the message provided by the ResponseError to the user // (2) user selects retry or cancel // (3) if user selected retry the initialize method is sent again. Retry bool `json:"retry"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#initializeParams type InitializeParams struct { XInitializeParams WorkspaceFoldersInitializeParams } // The result returned from an initialize request. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#initializeResult type InitializeResult struct { // The capabilities the language server provides. Capabilities ServerCapabilities `json:"capabilities"` // Information about the server. // // @since 3.15.0 ServerInfo *ServerInfo `json:"serverInfo,omitempty"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#initializedParams type InitializedParams struct { } // Inlay hint information. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#inlayHint type InlayHint struct { // The position of this hint. // // If multiple hints have the same position, they will be shown in the order // they appear in the response. Position Position `json:"position"` // The label of this hint. A human readable string or an array of // InlayHintLabelPart label parts. // // *Note* that neither the string nor the label part can be empty. Label []InlayHintLabelPart `json:"label"` // The kind of this hint. Can be omitted in which case the client // should fall back to a reasonable default. Kind InlayHintKind `json:"kind,omitempty"` // Optional text edits that are performed when accepting this inlay hint. // // *Note* that edits are expected to change the document so that the inlay // hint (or its nearest variant) is now part of the document and the inlay // hint itself is now obsolete. TextEdits []TextEdit `json:"textEdits,omitempty"` // The tooltip text when you hover over this item. Tooltip *Or_InlayHint_tooltip `json:"tooltip,omitempty"` // Render padding before the hint. // // Note: Padding should use the editor's background color, not the // background color of the hint itself. That means padding can be used // to visually align/separate an inlay hint. PaddingLeft bool `json:"paddingLeft,omitempty"` // Render padding after the hint. // // Note: Padding should use the editor's background color, not the // background color of the hint itself. That means padding can be used // to visually align/separate an inlay hint. PaddingRight bool `json:"paddingRight,omitempty"` // A data entry field that is preserved on an inlay hint between // a `textDocument/inlayHint` and a `inlayHint/resolve` request. Data interface{} `json:"data,omitempty"` } // Inlay hint client capabilities. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#inlayHintClientCapabilities type InlayHintClientCapabilities struct { // Whether inlay hints support dynamic registration. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` // Indicates which properties a client can resolve lazily on an inlay // hint. ResolveSupport *ClientInlayHintResolveOptions `json:"resolveSupport,omitempty"` } // Inlay hint kinds. // // @since 3.17.0 type InlayHintKind uint32 // An inlay hint label part allows for interactive and composite labels // of inlay hints. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#inlayHintLabelPart type InlayHintLabelPart struct { // The value of this label part. Value string `json:"value"` // The tooltip text when you hover over this label part. Depending on // the client capability `inlayHint.resolveSupport` clients might resolve // this property late using the resolve request. Tooltip *Or_InlayHintLabelPart_tooltip `json:"tooltip,omitempty"` // An optional source code location that represents this // label part. // // The editor will use this location for the hover and for code navigation // features: This part will become a clickable link that resolves to the // definition of the symbol at the given location (not necessarily the // location itself), it shows the hover that shows at the given location, // and it shows a context menu with further code navigation commands. // // Depending on the client capability `inlayHint.resolveSupport` clients // might resolve this property late using the resolve request. Location *Location `json:"location,omitempty"` // An optional command for this label part. // // Depending on the client capability `inlayHint.resolveSupport` clients // might resolve this property late using the resolve request. Command *Command `json:"command,omitempty"` } // Inlay hint options used during static registration. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#inlayHintOptions type InlayHintOptions struct { // The server provides support to resolve additional // information for an inlay hint item. ResolveProvider bool `json:"resolveProvider,omitempty"` WorkDoneProgressOptions } // A parameter literal used in inlay hint requests. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#inlayHintParams type InlayHintParams struct { // The text document. TextDocument TextDocumentIdentifier `json:"textDocument"` // The document range for which inlay hints should be computed. Range Range `json:"range"` WorkDoneProgressParams } // Inlay hint options used during static or dynamic registration. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#inlayHintRegistrationOptions type InlayHintRegistrationOptions struct { InlayHintOptions TextDocumentRegistrationOptions StaticRegistrationOptions } // Client workspace capabilities specific to inlay hints. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#inlayHintWorkspaceClientCapabilities type InlayHintWorkspaceClientCapabilities struct { // Whether the client implementation supports a refresh request sent from // the server to the client. // // Note that this event is global and will force the client to refresh all // inlay hints currently shown. It should be used with absolute care and // is useful for situation where a server for example detects a project wide // change that requires such a calculation. RefreshSupport bool `json:"refreshSupport,omitempty"` } // Client capabilities specific to inline completions. // // @since 3.18.0 // @proposed // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#inlineCompletionClientCapabilities type InlineCompletionClientCapabilities struct { // Whether implementation supports dynamic registration for inline completion providers. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` } // Provides information about the context in which an inline completion was requested. // // @since 3.18.0 // @proposed // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#inlineCompletionContext type InlineCompletionContext struct { // Describes how the inline completion was triggered. TriggerKind InlineCompletionTriggerKind `json:"triggerKind"` // Provides information about the currently selected item in the autocomplete widget if it is visible. SelectedCompletionInfo *SelectedCompletionInfo `json:"selectedCompletionInfo,omitempty"` } // An inline completion item represents a text snippet that is proposed inline to complete text that is being typed. // // @since 3.18.0 // @proposed // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#inlineCompletionItem type InlineCompletionItem struct { // The text to replace the range with. Must be set. InsertText Or_InlineCompletionItem_insertText `json:"insertText"` // A text that is used to decide if this inline completion should be shown. When `falsy` the {@link InlineCompletionItem.insertText} is used. FilterText string `json:"filterText,omitempty"` // The range to replace. Must begin and end on the same line. Range *Range `json:"range,omitempty"` // An optional {@link Command} that is executed *after* inserting this completion. Command *Command `json:"command,omitempty"` } // Represents a collection of {@link InlineCompletionItem inline completion items} to be presented in the editor. // // @since 3.18.0 // @proposed // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#inlineCompletionList type InlineCompletionList struct { // The inline completion items Items []InlineCompletionItem `json:"items"` } // Inline completion options used during static registration. // // @since 3.18.0 // @proposed // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#inlineCompletionOptions type InlineCompletionOptions struct { WorkDoneProgressOptions } // A parameter literal used in inline completion requests. // // @since 3.18.0 // @proposed // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#inlineCompletionParams type InlineCompletionParams struct { // Additional information about the context in which inline completions were // requested. Context InlineCompletionContext `json:"context"` TextDocumentPositionParams WorkDoneProgressParams } // Inline completion options used during static or dynamic registration. // // @since 3.18.0 // @proposed // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#inlineCompletionRegistrationOptions type InlineCompletionRegistrationOptions struct { InlineCompletionOptions TextDocumentRegistrationOptions StaticRegistrationOptions } // Describes how an {@link InlineCompletionItemProvider inline completion provider} was triggered. // // @since 3.18.0 // @proposed type InlineCompletionTriggerKind uint32 // Inline value information can be provided by different means: // // - directly as a text value (class InlineValueText). // - as a name to use for a variable lookup (class InlineValueVariableLookup) // - as an evaluatable expression (class InlineValueEvaluatableExpression) // // The InlineValue types combines all inline value types into one type. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#inlineValue type InlineValue = Or_InlineValue // (alias) // Client capabilities specific to inline values. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#inlineValueClientCapabilities type InlineValueClientCapabilities struct { // Whether implementation supports dynamic registration for inline value providers. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` } // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#inlineValueContext type InlineValueContext struct { // The stack frame (as a DAP Id) where the execution has stopped. FrameID int32 `json:"frameId"` // The document range where execution has stopped. // Typically the end position of the range denotes the line where the inline values are shown. StoppedLocation Range `json:"stoppedLocation"` } // Provide an inline value through an expression evaluation. // If only a range is specified, the expression will be extracted from the underlying document. // An optional expression can be used to override the extracted expression. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#inlineValueEvaluatableExpression type InlineValueEvaluatableExpression struct { // The document range for which the inline value applies. // The range is used to extract the evaluatable expression from the underlying document. Range Range `json:"range"` // If specified the expression overrides the extracted expression. Expression string `json:"expression,omitempty"` } // Inline value options used during static registration. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#inlineValueOptions type InlineValueOptions struct { WorkDoneProgressOptions } // A parameter literal used in inline value requests. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#inlineValueParams type InlineValueParams struct { // The text document. TextDocument TextDocumentIdentifier `json:"textDocument"` // The document range for which inline values should be computed. Range Range `json:"range"` // Additional information about the context in which inline values were // requested. Context InlineValueContext `json:"context"` WorkDoneProgressParams } // Inline value options used during static or dynamic registration. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#inlineValueRegistrationOptions type InlineValueRegistrationOptions struct { InlineValueOptions TextDocumentRegistrationOptions StaticRegistrationOptions } // Provide inline value as text. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#inlineValueText type InlineValueText struct { // The document range for which the inline value applies. Range Range `json:"range"` // The text of the inline value. Text string `json:"text"` } // Provide inline value through a variable lookup. // If only a range is specified, the variable name will be extracted from the underlying document. // An optional variable name can be used to override the extracted name. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#inlineValueVariableLookup type InlineValueVariableLookup struct { // The document range for which the inline value applies. // The range is used to extract the variable name from the underlying document. Range Range `json:"range"` // If specified the name of the variable to look up. VariableName string `json:"variableName,omitempty"` // How to perform the lookup. CaseSensitiveLookup bool `json:"caseSensitiveLookup"` } // Client workspace capabilities specific to inline values. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#inlineValueWorkspaceClientCapabilities type InlineValueWorkspaceClientCapabilities struct { // Whether the client implementation supports a refresh request sent from the // server to the client. // // Note that this event is global and will force the client to refresh all // inline values currently shown. It should be used with absolute care and is // useful for situation where a server for example detects a project wide // change that requires such a calculation. RefreshSupport bool `json:"refreshSupport,omitempty"` } // A special text edit to provide an insert and a replace operation. // // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#insertReplaceEdit type InsertReplaceEdit struct { // The string to be inserted. NewText string `json:"newText"` // The range if the insert is requested Insert Range `json:"insert"` // The range if the replace is requested. Replace Range `json:"replace"` } // Defines whether the insert text in a completion item should be interpreted as // plain text or a snippet. type InsertTextFormat uint32 // How whitespace and indentation is handled during completion // item insertion. // // @since 3.16.0 type InsertTextMode uint32 type LSPAny = interface{} // LSP arrays. // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#lSPArray type LSPArray = []interface{} // (alias) type LSPErrorCodes int32 // LSP object definition. // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#lSPObject type LSPObject = map[string]LSPAny // (alias) // Predefined Language kinds // @since 3.18.0 // @proposed type LanguageKind string // Client capabilities for the linked editing range request. // // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#linkedEditingRangeClientCapabilities type LinkedEditingRangeClientCapabilities struct { // Whether implementation supports dynamic registration. If this is set to `true` // the client supports the new `(TextDocumentRegistrationOptions & StaticRegistrationOptions)` // return value for the corresponding server capability as well. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#linkedEditingRangeOptions type LinkedEditingRangeOptions struct { WorkDoneProgressOptions } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#linkedEditingRangeParams type LinkedEditingRangeParams struct { TextDocumentPositionParams WorkDoneProgressParams } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#linkedEditingRangeRegistrationOptions type LinkedEditingRangeRegistrationOptions struct { TextDocumentRegistrationOptions LinkedEditingRangeOptions StaticRegistrationOptions } // The result of a linked editing range request. // // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#linkedEditingRanges type LinkedEditingRanges struct { // A list of ranges that can be edited together. The ranges must have // identical length and contain identical text content. The ranges cannot overlap. Ranges []Range `json:"ranges"` // An optional word pattern (regular expression) that describes valid contents for // the given ranges. If no pattern is provided, the client configuration's word // pattern will be used. WordPattern string `json:"wordPattern,omitempty"` } // created for Literal (Lit_ClientSemanticTokensRequestOptions_range_Item1) type Lit_ClientSemanticTokensRequestOptions_range_Item1 struct { } // created for Literal (Lit_SemanticTokensOptions_range_Item1) type Lit_SemanticTokensOptions_range_Item1 struct { } // Represents a location inside a resource, such as a line // inside a text file. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#location type Location struct { URI DocumentUri `json:"uri"` Range Range `json:"range"` } // Represents the connection of two locations. Provides additional metadata over normal {@link Location locations}, // including an origin range. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#locationLink type LocationLink struct { // Span of the origin of this link. // // Used as the underlined span for mouse interaction. Defaults to the word range at // the definition position. OriginSelectionRange *Range `json:"originSelectionRange,omitempty"` // The target resource identifier of this link. TargetURI DocumentUri `json:"targetUri"` // The full target range of this link. If the target for example is a symbol then target range is the // range enclosing this symbol not including leading/trailing whitespace but everything else // like comments. This information is typically used to highlight the range in the editor. TargetRange Range `json:"targetRange"` // The range that should be selected and revealed when this link is being followed, e.g the name of a function. // Must be contained by the `targetRange`. See also `DocumentSymbol#range` TargetSelectionRange Range `json:"targetSelectionRange"` } // Location with only uri and does not include range. // // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#locationUriOnly type LocationUriOnly struct { URI DocumentUri `json:"uri"` } // The log message parameters. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#logMessageParams type LogMessageParams struct { // The message type. See {@link MessageType} Type MessageType `json:"type"` // The actual message. Message string `json:"message"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#logTraceParams type LogTraceParams struct { Message string `json:"message"` Verbose string `json:"verbose,omitempty"` } // Client capabilities specific to the used markdown parser. // // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#markdownClientCapabilities type MarkdownClientCapabilities struct { // The name of the parser. Parser string `json:"parser"` // The version of the parser. Version string `json:"version,omitempty"` // A list of HTML tags that the client allows / supports in // Markdown. // // @since 3.17.0 AllowedTags []string `json:"allowedTags,omitempty"` } // MarkedString can be used to render human readable text. It is either a markdown string // or a code-block that provides a language and a code snippet. The language identifier // is semantically equal to the optional language identifier in fenced code blocks in GitHub // issues. See https://help.github.com/articles/creating-and-highlighting-code-blocks/#syntax-highlighting // // The pair of a language and a value is an equivalent to markdown: // ```${language} // ${value} // ``` // // Note that markdown strings will be sanitized - that means html will be escaped. // @deprecated use MarkupContent instead. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#markedString type MarkedString = Or_MarkedString // (alias) // @since 3.18.0 // @deprecated use MarkupContent instead. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#markedStringWithLanguage type MarkedStringWithLanguage struct { Language string `json:"language"` Value string `json:"value"` } // A `MarkupContent` literal represents a string value which content is interpreted base on its // kind flag. Currently the protocol supports `plaintext` and `markdown` as markup kinds. // // If the kind is `markdown` then the value can contain fenced code blocks like in GitHub issues. // See https://help.github.com/articles/creating-and-highlighting-code-blocks/#syntax-highlighting // // Here is an example how such a string can be constructed using JavaScript / TypeScript: // ```ts // // let markdown: MarkdownContent = { // kind: MarkupKind.Markdown, // value: [ // '# Header', // 'Some text', // '```typescript', // 'someCode();', // '```' // ].join('\n') // }; // // ``` // // *Please Note* that clients might sanitize the return markdown. A client could decide to // remove HTML from the markdown to avoid script execution. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#markupContent type MarkupContent struct { // The type of the Markup Kind MarkupKind `json:"kind"` // The content itself Value string `json:"value"` } // Describes the content type that a client supports in various // result literals like `Hover`, `ParameterInfo` or `CompletionItem`. // // Please note that `MarkupKinds` must not start with a `$`. This kinds // are reserved for internal usage. type MarkupKind string // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#messageActionItem type MessageActionItem struct { // A short title like 'Retry', 'Open Log' etc. Title string `json:"title"` } // The message type type MessageType uint32 // Moniker definition to match LSIF 0.5 moniker definition. // // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#moniker type Moniker struct { // The scheme of the moniker. For example tsc or .Net Scheme string `json:"scheme"` // The identifier of the moniker. The value is opaque in LSIF however // schema owners are allowed to define the structure if they want. Identifier string `json:"identifier"` // The scope in which the moniker is unique Unique UniquenessLevel `json:"unique"` // The moniker kind if known. Kind *MonikerKind `json:"kind,omitempty"` } // Client capabilities specific to the moniker request. // // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#monikerClientCapabilities type MonikerClientCapabilities struct { // Whether moniker supports dynamic registration. If this is set to `true` // the client supports the new `MonikerRegistrationOptions` return value // for the corresponding server capability as well. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` } // The moniker kind. // // @since 3.16.0 type MonikerKind string // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#monikerOptions type MonikerOptions struct { WorkDoneProgressOptions } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#monikerParams type MonikerParams struct { TextDocumentPositionParams WorkDoneProgressParams PartialResultParams } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#monikerRegistrationOptions type MonikerRegistrationOptions struct { TextDocumentRegistrationOptions MonikerOptions } // A notebook cell. // // A cell's document URI must be unique across ALL notebook // cells and can therefore be used to uniquely identify a // notebook cell or the cell's text document. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#notebookCell type NotebookCell struct { // The cell's kind Kind NotebookCellKind `json:"kind"` // The URI of the cell's text document // content. Document DocumentUri `json:"document"` // Additional metadata stored with the cell. // // Note: should always be an object literal (e.g. LSPObject) Metadata *LSPObject `json:"metadata,omitempty"` // Additional execution summary information // if supported by the client. ExecutionSummary *ExecutionSummary `json:"executionSummary,omitempty"` } // A change describing how to move a `NotebookCell` // array from state S to S'. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#notebookCellArrayChange type NotebookCellArrayChange struct { // The start oftest of the cell that changed. Start uint32 `json:"start"` // The deleted cells DeleteCount uint32 `json:"deleteCount"` // The new cells, if any Cells []NotebookCell `json:"cells,omitempty"` } // A notebook cell kind. // // @since 3.17.0 type NotebookCellKind uint32 // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#notebookCellLanguage type NotebookCellLanguage struct { Language string `json:"language"` } // A notebook cell text document filter denotes a cell text // document by different properties. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#notebookCellTextDocumentFilter type NotebookCellTextDocumentFilter struct { // A filter that matches against the notebook // containing the notebook cell. If a string // value is provided it matches against the // notebook type. '*' matches every notebook. Notebook Or_NotebookCellTextDocumentFilter_notebook `json:"notebook"` // A language id like `python`. // // Will be matched against the language id of the // notebook cell document. '*' matches every language. Language string `json:"language,omitempty"` } // A notebook document. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#notebookDocument type NotebookDocument struct { // The notebook document's uri. URI URI `json:"uri"` // The type of the notebook. NotebookType string `json:"notebookType"` // The version number of this document (it will increase after each // change, including undo/redo). Version int32 `json:"version"` // Additional metadata stored with the notebook // document. // // Note: should always be an object literal (e.g. LSPObject) Metadata *LSPObject `json:"metadata,omitempty"` // The cells of a notebook. Cells []NotebookCell `json:"cells"` } // Structural changes to cells in a notebook document. // // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#notebookDocumentCellChangeStructure type NotebookDocumentCellChangeStructure struct { // The change to the cell array. Array NotebookCellArrayChange `json:"array"` // Additional opened cell text documents. DidOpen []TextDocumentItem `json:"didOpen,omitempty"` // Additional closed cell text documents. DidClose []TextDocumentIdentifier `json:"didClose,omitempty"` } // Cell changes to a notebook document. // // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#notebookDocumentCellChanges type NotebookDocumentCellChanges struct { // Changes to the cell structure to add or // remove cells. Structure *NotebookDocumentCellChangeStructure `json:"structure,omitempty"` // Changes to notebook cells properties like its // kind, execution summary or metadata. Data []NotebookCell `json:"data,omitempty"` // Changes to the text content of notebook cells. TextContent []NotebookDocumentCellContentChanges `json:"textContent,omitempty"` } // Content changes to a cell in a notebook document. // // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#notebookDocumentCellContentChanges type NotebookDocumentCellContentChanges struct { Document VersionedTextDocumentIdentifier `json:"document"` Changes []TextDocumentContentChangeEvent `json:"changes"` } // A change event for a notebook document. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#notebookDocumentChangeEvent type NotebookDocumentChangeEvent struct { // The changed meta data if any. // // Note: should always be an object literal (e.g. LSPObject) Metadata *LSPObject `json:"metadata,omitempty"` // Changes to cells Cells *NotebookDocumentCellChanges `json:"cells,omitempty"` } // Capabilities specific to the notebook document support. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#notebookDocumentClientCapabilities type NotebookDocumentClientCapabilities struct { // Capabilities specific to notebook document synchronization // // @since 3.17.0 Synchronization NotebookDocumentSyncClientCapabilities `json:"synchronization"` } // A notebook document filter denotes a notebook document by // different properties. The properties will be match // against the notebook's URI (same as with documents) // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#notebookDocumentFilter type NotebookDocumentFilter = Or_NotebookDocumentFilter // (alias) // A notebook document filter where `notebookType` is required field. // // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#notebookDocumentFilterNotebookType type NotebookDocumentFilterNotebookType struct { // The type of the enclosing notebook. NotebookType string `json:"notebookType"` // A Uri {@link Uri.scheme scheme}, like `file` or `untitled`. Scheme string `json:"scheme,omitempty"` // A glob pattern. Pattern *GlobPattern `json:"pattern,omitempty"` } // A notebook document filter where `pattern` is required field. // // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#notebookDocumentFilterPattern type NotebookDocumentFilterPattern struct { // The type of the enclosing notebook. NotebookType string `json:"notebookType,omitempty"` // A Uri {@link Uri.scheme scheme}, like `file` or `untitled`. Scheme string `json:"scheme,omitempty"` // A glob pattern. Pattern GlobPattern `json:"pattern"` } // A notebook document filter where `scheme` is required field. // // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#notebookDocumentFilterScheme type NotebookDocumentFilterScheme struct { // The type of the enclosing notebook. NotebookType string `json:"notebookType,omitempty"` // A Uri {@link Uri.scheme scheme}, like `file` or `untitled`. Scheme string `json:"scheme"` // A glob pattern. Pattern *GlobPattern `json:"pattern,omitempty"` } // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#notebookDocumentFilterWithCells type NotebookDocumentFilterWithCells struct { // The notebook to be synced If a string // value is provided it matches against the // notebook type. '*' matches every notebook. Notebook *Or_NotebookDocumentFilterWithCells_notebook `json:"notebook,omitempty"` // The cells of the matching notebook to be synced. Cells []NotebookCellLanguage `json:"cells"` } // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#notebookDocumentFilterWithNotebook type NotebookDocumentFilterWithNotebook struct { // The notebook to be synced If a string // value is provided it matches against the // notebook type. '*' matches every notebook. Notebook Or_NotebookDocumentFilterWithNotebook_notebook `json:"notebook"` // The cells of the matching notebook to be synced. Cells []NotebookCellLanguage `json:"cells,omitempty"` } // A literal to identify a notebook document in the client. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#notebookDocumentIdentifier type NotebookDocumentIdentifier struct { // The notebook document's uri. URI URI `json:"uri"` } // Notebook specific client capabilities. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#notebookDocumentSyncClientCapabilities type NotebookDocumentSyncClientCapabilities struct { // Whether implementation supports dynamic registration. If this is // set to `true` the client supports the new // `(TextDocumentRegistrationOptions & StaticRegistrationOptions)` // return value for the corresponding server capability as well. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` // The client supports sending execution summary data per cell. ExecutionSummarySupport bool `json:"executionSummarySupport,omitempty"` } // Options specific to a notebook plus its cells // to be synced to the server. // // If a selector provides a notebook document // filter but no cell selector all cells of a // matching notebook document will be synced. // // If a selector provides no notebook document // filter but only a cell selector all notebook // document that contain at least one matching // cell will be synced. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#notebookDocumentSyncOptions type NotebookDocumentSyncOptions struct { // The notebooks to be synced NotebookSelector []Or_NotebookDocumentSyncOptions_notebookSelector_Elem `json:"notebookSelector"` // Whether save notification should be forwarded to // the server. Will only be honored if mode === `notebook`. Save bool `json:"save,omitempty"` } // Registration options specific to a notebook. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#notebookDocumentSyncRegistrationOptions type NotebookDocumentSyncRegistrationOptions struct { NotebookDocumentSyncOptions StaticRegistrationOptions } // A text document identifier to optionally denote a specific version of a text document. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#optionalVersionedTextDocumentIdentifier type OptionalVersionedTextDocumentIdentifier struct { // The version number of this document. If a versioned text document identifier // is sent from the server to the client and the file is not open in the editor // (the server has not received an open notification before) the server can send // `null` to indicate that the version is unknown and the content on disk is the // truth (as specified with document content ownership). Version int32 `json:"version"` TextDocumentIdentifier } // created for Or [int32 string] type Or_CancelParams_id struct { Value interface{} `json:"value"` } // created for Or [ClientSemanticTokensRequestFullDelta bool] type Or_ClientSemanticTokensRequestOptions_full struct { Value interface{} `json:"value"` } // created for Or [Lit_ClientSemanticTokensRequestOptions_range_Item1 bool] type Or_ClientSemanticTokensRequestOptions_range struct { Value interface{} `json:"value"` } // created for Or [EditRangeWithInsertReplace Range] type Or_CompletionItemDefaults_editRange struct { Value interface{} `json:"value"` } // created for Or [MarkupContent string] type Or_CompletionItem_documentation struct { Value interface{} `json:"value"` } // created for Or [InsertReplaceEdit TextEdit] type Or_CompletionItem_textEdit struct { Value interface{} `json:"value"` } // created for Or [Location []Location] type Or_Declaration struct { Value interface{} `json:"value"` } // created for Or [Location []Location] type Or_Definition struct { Value interface{} `json:"value"` } // created for Or [int32 string] type Or_Diagnostic_code struct { Value interface{} `json:"value"` } // created for Or [[]string string] type Or_DidChangeConfigurationRegistrationOptions_section struct { Value interface{} `json:"value"` } // created for Or [RelatedFullDocumentDiagnosticReport RelatedUnchangedDocumentDiagnosticReport] type Or_DocumentDiagnosticReport struct { Value interface{} `json:"value"` } // created for Or [FullDocumentDiagnosticReport UnchangedDocumentDiagnosticReport] type Or_DocumentDiagnosticReportPartialResult_relatedDocuments_Value struct { Value interface{} `json:"value"` } // created for Or [NotebookCellTextDocumentFilter TextDocumentFilter] type Or_DocumentFilter struct { Value interface{} `json:"value"` } // created for Or [Pattern RelativePattern] type Or_GlobPattern struct { Value interface{} `json:"value"` } // created for Or [MarkedString MarkupContent []MarkedString] type Or_Hover_contents struct { Value interface{} `json:"value"` } // created for Or [MarkupContent string] type Or_InlayHintLabelPart_tooltip struct { Value interface{} `json:"value"` } // created for Or [[]InlayHintLabelPart string] type Or_InlayHint_label struct { Value interface{} `json:"value"` } // created for Or [MarkupContent string] type Or_InlayHint_tooltip struct { Value interface{} `json:"value"` } // created for Or [StringValue string] type Or_InlineCompletionItem_insertText struct { Value interface{} `json:"value"` } // created for Or [InlineValueEvaluatableExpression InlineValueText InlineValueVariableLookup] type Or_InlineValue struct { Value interface{} `json:"value"` } // created for Or [LSPArray LSPObject bool float64 int32 string uint32] type Or_LSPAny struct { Value interface{} `json:"value"` } // created for Or [MarkedStringWithLanguage string] type Or_MarkedString struct { Value interface{} `json:"value"` } // created for Or [NotebookDocumentFilter string] type Or_NotebookCellTextDocumentFilter_notebook struct { Value interface{} `json:"value"` } // created for Or [NotebookDocumentFilterNotebookType NotebookDocumentFilterPattern NotebookDocumentFilterScheme] type Or_NotebookDocumentFilter struct { Value interface{} `json:"value"` } // created for Or [NotebookDocumentFilter string] type Or_NotebookDocumentFilterWithCells_notebook struct { Value interface{} `json:"value"` } // created for Or [NotebookDocumentFilter string] type Or_NotebookDocumentFilterWithNotebook_notebook struct { Value interface{} `json:"value"` } // created for Or [NotebookDocumentFilterWithCells NotebookDocumentFilterWithNotebook] type Or_NotebookDocumentSyncOptions_notebookSelector_Elem struct { Value interface{} `json:"value"` } // created for Or [MarkupContent string] type Or_ParameterInformation_documentation struct { Value interface{} `json:"value"` } // created for Or [Tuple_ParameterInformation_label_Item1 string] type Or_ParameterInformation_label struct { Value interface{} `json:"value"` } // created for Or [PrepareRenameDefaultBehavior PrepareRenamePlaceholder Range] type Or_PrepareRenameResult struct { Value interface{} `json:"value"` } // created for Or [int32 string] type Or_ProgressToken struct { Value interface{} `json:"value"` } // created for Or [FullDocumentDiagnosticReport UnchangedDocumentDiagnosticReport] type Or_RelatedFullDocumentDiagnosticReport_relatedDocuments_Value struct { Value interface{} `json:"value"` } // created for Or [FullDocumentDiagnosticReport UnchangedDocumentDiagnosticReport] type Or_RelatedUnchangedDocumentDiagnosticReport_relatedDocuments_Value struct { Value interface{} `json:"value"` } // created for Or [URI WorkspaceFolder] type Or_RelativePattern_baseUri struct { Value interface{} `json:"value"` } // created for Or [CodeAction Command] type Or_Result_textDocument_codeAction_Item0_Elem struct { Value interface{} `json:"value"` } // created for Or [CompletionList []CompletionItem] type Or_Result_textDocument_completion struct { Value interface{} `json:"value"` } // created for Or [Declaration []DeclarationLink] type Or_Result_textDocument_declaration struct { Value interface{} `json:"value"` } // created for Or [Definition []DefinitionLink] type Or_Result_textDocument_definition struct { Value interface{} `json:"value"` } // created for Or [[]DocumentSymbol []SymbolInformation] type Or_Result_textDocument_documentSymbol struct { Value interface{} `json:"value"` } // created for Or [Definition []DefinitionLink] type Or_Result_textDocument_implementation struct { Value interface{} `json:"value"` } // created for Or [InlineCompletionList []InlineCompletionItem] type Or_Result_textDocument_inlineCompletion struct { Value interface{} `json:"value"` } // created for Or [SemanticTokens SemanticTokensDelta] type Or_Result_textDocument_semanticTokens_full_delta struct { Value interface{} `json:"value"` } // created for Or [Definition []DefinitionLink] type Or_Result_textDocument_typeDefinition struct { Value interface{} `json:"value"` } // created for Or [[]SymbolInformation []WorkspaceSymbol] type Or_Result_workspace_symbol struct { Value interface{} `json:"value"` } // created for Or [SemanticTokensFullDelta bool] type Or_SemanticTokensOptions_full struct { Value interface{} `json:"value"` } // created for Or [Lit_SemanticTokensOptions_range_Item1 bool] type Or_SemanticTokensOptions_range struct { Value interface{} `json:"value"` } // created for Or [CallHierarchyOptions CallHierarchyRegistrationOptions bool] type Or_ServerCapabilities_callHierarchyProvider struct { Value interface{} `json:"value"` } // created for Or [CodeActionOptions bool] type Or_ServerCapabilities_codeActionProvider struct { Value interface{} `json:"value"` } // created for Or [DocumentColorOptions DocumentColorRegistrationOptions bool] type Or_ServerCapabilities_colorProvider struct { Value interface{} `json:"value"` } // created for Or [DeclarationOptions DeclarationRegistrationOptions bool] type Or_ServerCapabilities_declarationProvider struct { Value interface{} `json:"value"` } // created for Or [DefinitionOptions bool] type Or_ServerCapabilities_definitionProvider struct { Value interface{} `json:"value"` } // created for Or [DiagnosticOptions DiagnosticRegistrationOptions] type Or_ServerCapabilities_diagnosticProvider struct { Value interface{} `json:"value"` } // created for Or [DocumentFormattingOptions bool] type Or_ServerCapabilities_documentFormattingProvider struct { Value interface{} `json:"value"` } // created for Or [DocumentHighlightOptions bool] type Or_ServerCapabilities_documentHighlightProvider struct { Value interface{} `json:"value"` } // created for Or [DocumentRangeFormattingOptions bool] type Or_ServerCapabilities_documentRangeFormattingProvider struct { Value interface{} `json:"value"` } // created for Or [DocumentSymbolOptions bool] type Or_ServerCapabilities_documentSymbolProvider struct { Value interface{} `json:"value"` } // created for Or [FoldingRangeOptions FoldingRangeRegistrationOptions bool] type Or_ServerCapabilities_foldingRangeProvider struct { Value interface{} `json:"value"` } // created for Or [HoverOptions bool] type Or_ServerCapabilities_hoverProvider struct { Value interface{} `json:"value"` } // created for Or [ImplementationOptions ImplementationRegistrationOptions bool] type Or_ServerCapabilities_implementationProvider struct { Value interface{} `json:"value"` } // created for Or [InlayHintOptions InlayHintRegistrationOptions bool] type Or_ServerCapabilities_inlayHintProvider struct { Value interface{} `json:"value"` } // created for Or [InlineCompletionOptions bool] type Or_ServerCapabilities_inlineCompletionProvider struct { Value interface{} `json:"value"` } // created for Or [InlineValueOptions InlineValueRegistrationOptions bool] type Or_ServerCapabilities_inlineValueProvider struct { Value interface{} `json:"value"` } // created for Or [LinkedEditingRangeOptions LinkedEditingRangeRegistrationOptions bool] type Or_ServerCapabilities_linkedEditingRangeProvider struct { Value interface{} `json:"value"` } // created for Or [MonikerOptions MonikerRegistrationOptions bool] type Or_ServerCapabilities_monikerProvider struct { Value interface{} `json:"value"` } // created for Or [NotebookDocumentSyncOptions NotebookDocumentSyncRegistrationOptions] type Or_ServerCapabilities_notebookDocumentSync struct { Value interface{} `json:"value"` } // created for Or [ReferenceOptions bool] type Or_ServerCapabilities_referencesProvider struct { Value interface{} `json:"value"` } // created for Or [RenameOptions bool] type Or_ServerCapabilities_renameProvider struct { Value interface{} `json:"value"` } // created for Or [SelectionRangeOptions SelectionRangeRegistrationOptions bool] type Or_ServerCapabilities_selectionRangeProvider struct { Value interface{} `json:"value"` } // created for Or [SemanticTokensOptions SemanticTokensRegistrationOptions] type Or_ServerCapabilities_semanticTokensProvider struct { Value interface{} `json:"value"` } // created for Or [TextDocumentSyncKind TextDocumentSyncOptions] type Or_ServerCapabilities_textDocumentSync struct { Value interface{} `json:"value"` } // created for Or [TypeDefinitionOptions TypeDefinitionRegistrationOptions bool] type Or_ServerCapabilities_typeDefinitionProvider struct { Value interface{} `json:"value"` } // created for Or [TypeHierarchyOptions TypeHierarchyRegistrationOptions bool] type Or_ServerCapabilities_typeHierarchyProvider struct { Value interface{} `json:"value"` } // created for Or [WorkspaceSymbolOptions bool] type Or_ServerCapabilities_workspaceSymbolProvider struct { Value interface{} `json:"value"` } // created for Or [MarkupContent string] type Or_SignatureInformation_documentation struct { Value interface{} `json:"value"` } // created for Or [TextDocumentContentChangePartial TextDocumentContentChangeWholeDocument] type Or_TextDocumentContentChangeEvent struct { Value interface{} `json:"value"` } // created for Or [AnnotatedTextEdit SnippetTextEdit TextEdit] type Or_TextDocumentEdit_edits_Elem struct { Value interface{} `json:"value"` } // created for Or [TextDocumentFilterLanguage TextDocumentFilterPattern TextDocumentFilterScheme] type Or_TextDocumentFilter struct { Value interface{} `json:"value"` } // created for Or [SaveOptions bool] type Or_TextDocumentSyncOptions_save struct { Value interface{} `json:"value"` } // created for Or [WorkspaceFullDocumentDiagnosticReport WorkspaceUnchangedDocumentDiagnosticReport] type Or_WorkspaceDocumentDiagnosticReport struct { Value interface{} `json:"value"` } // created for Or [CreateFile DeleteFile RenameFile TextDocumentEdit] type Or_WorkspaceEdit_documentChanges_Elem struct { Value interface{} `json:"value"` } // created for Or [bool string] type Or_WorkspaceFoldersServerCapabilities_changeNotifications struct { Value interface{} `json:"value"` } // created for Or [TextDocumentContentOptions TextDocumentContentRegistrationOptions] type Or_WorkspaceOptions_textDocumentContent struct { Value interface{} `json:"value"` } // created for Or [Location LocationUriOnly] type Or_WorkspaceSymbol_location struct { Value interface{} `json:"value"` } // The parameters of a configuration request. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#configurationParams type ParamConfiguration struct { Items []ConfigurationItem `json:"items"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#initializeParams type ParamInitialize struct { XInitializeParams WorkspaceFoldersInitializeParams } // Represents a parameter of a callable-signature. A parameter can // have a label and a doc-comment. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#parameterInformation type ParameterInformation struct { // The label of this parameter information. // // Either a string or an inclusive start and exclusive end offsets within its containing // signature label. (see SignatureInformation.label). The offsets are based on a UTF-16 // string representation as `Position` and `Range` does. // // To avoid ambiguities a server should use the [start, end] offset value instead of using // a substring. Whether a client support this is controlled via `labelOffsetSupport` client // capability. // // *Note*: a label of type string should be a substring of its containing signature label. // Its intended use case is to highlight the parameter label part in the `SignatureInformation.label`. Label Or_ParameterInformation_label `json:"label"` // The human-readable doc-comment of this parameter. Will be shown // in the UI but can be omitted. Documentation *Or_ParameterInformation_documentation `json:"documentation,omitempty"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#partialResultParams type PartialResultParams struct { // An optional token that a server can use to report partial results (e.g. streaming) to // the client. PartialResultToken *ProgressToken `json:"partialResultToken,omitempty"` } // The glob pattern to watch relative to the base path. Glob patterns can have the following syntax: // // - `*` to match one or more characters in a path segment // - `?` to match on one character in a path segment // - `**` to match any number of path segments, including none // - `{}` to group conditions (e.g. `**​/*.{ts,js}` matches all TypeScript and JavaScript files) // - `[]` to declare a range of characters to match in a path segment (e.g., `example.[0-9]` to match on `example.0`, `example.1`, …) // - `[!...]` to negate a range of characters to match in a path segment (e.g., `example.[!0-9]` to match on `example.a`, `example.b`, but not `example.0`) // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#pattern type Pattern = string // (alias) // Position in a text document expressed as zero-based line and character // offset. Prior to 3.17 the offsets were always based on a UTF-16 string // representation. So a string of the form `a𐐀b` the character offset of the // character `a` is 0, the character offset of `𐐀` is 1 and the character // offset of b is 3 since `𐐀` is represented using two code units in UTF-16. // Since 3.17 clients and servers can agree on a different string encoding // representation (e.g. UTF-8). The client announces it's supported encoding // via the client capability [`general.positionEncodings`](https://microsoft.github.io/language-server-protocol/specifications/specification-current/#clientCapabilities). // The value is an array of position encodings the client supports, with // decreasing preference (e.g. the encoding at index `0` is the most preferred // one). To stay backwards compatible the only mandatory encoding is UTF-16 // represented via the string `utf-16`. The server can pick one of the // encodings offered by the client and signals that encoding back to the // client via the initialize result's property // [`capabilities.positionEncoding`](https://microsoft.github.io/language-server-protocol/specifications/specification-current/#serverCapabilities). If the string value // `utf-16` is missing from the client's capability `general.positionEncodings` // servers can safely assume that the client supports UTF-16. If the server // omits the position encoding in its initialize result the encoding defaults // to the string value `utf-16`. Implementation considerations: since the // conversion from one encoding into another requires the content of the // file / line the conversion is best done where the file is read which is // usually on the server side. // // Positions are line end character agnostic. So you can not specify a position // that denotes `\r|\n` or `\n|` where `|` represents the character offset. // // @since 3.17.0 - support for negotiated position encoding. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#position type Position struct { // Line position in a document (zero-based). // // If a line number is greater than the number of lines in a document, it defaults back to the number of lines in the document. // If a line number is negative, it defaults to 0. Line uint32 `json:"line"` // Character offset on a line in a document (zero-based). // // The meaning of this offset is determined by the negotiated // `PositionEncodingKind`. // // If the character value is greater than the line length it defaults back to the // line length. Character uint32 `json:"character"` } // A set of predefined position encoding kinds. // // @since 3.17.0 type PositionEncodingKind string // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#prepareRenameDefaultBehavior type PrepareRenameDefaultBehavior struct { DefaultBehavior bool `json:"defaultBehavior"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#prepareRenameParams type PrepareRenameParams struct { TextDocumentPositionParams WorkDoneProgressParams } // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#prepareRenamePlaceholder type PrepareRenamePlaceholder struct { Range Range `json:"range"` Placeholder string `json:"placeholder"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#prepareRenameResult type PrepareRenameResult = Or_PrepareRenameResult // (alias) type PrepareSupportDefaultBehavior uint32 // A previous result id in a workspace pull request. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#previousResultId type PreviousResultID struct { // The URI for which the client knowns a // result id. URI DocumentUri `json:"uri"` // The value of the previous result id. Value string `json:"value"` } // A previous result id in a workspace pull request. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#previousResultId type PreviousResultId struct { // The URI for which the client knowns a // result id. URI DocumentUri `json:"uri"` // The value of the previous result id. Value string `json:"value"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#progressParams type ProgressParams struct { // The progress token provided by the client or server. Token ProgressToken `json:"token"` // The progress data. Value interface{} `json:"value"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#progressToken type ProgressToken = Or_ProgressToken // (alias) // The publish diagnostic client capabilities. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#publishDiagnosticsClientCapabilities type PublishDiagnosticsClientCapabilities struct { // Whether the client interprets the version property of the // `textDocument/publishDiagnostics` notification's parameter. // // @since 3.15.0 VersionSupport bool `json:"versionSupport,omitempty"` DiagnosticsCapabilities } // The publish diagnostic notification's parameters. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#publishDiagnosticsParams type PublishDiagnosticsParams struct { // The URI for which diagnostic information is reported. URI DocumentUri `json:"uri"` // Optional the version number of the document the diagnostics are published for. // // @since 3.15.0 Version int32 `json:"version,omitempty"` // An array of diagnostic information items. Diagnostics []Diagnostic `json:"diagnostics"` } // A range in a text document expressed as (zero-based) start and end positions. // // If you want to specify a range that contains a line including the line ending // character(s) then use an end position denoting the start of the next line. // For example: // ```ts // // { // start: { line: 5, character: 23 } // end : { line 6, character : 0 } // } // // ``` // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#range type Range struct { // The range's start position. Start Position `json:"start"` // The range's end position. End Position `json:"end"` } // Client Capabilities for a {@link ReferencesRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#referenceClientCapabilities type ReferenceClientCapabilities struct { // Whether references supports dynamic registration. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` } // Value-object that contains additional information when // requesting references. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#referenceContext type ReferenceContext struct { // Include the declaration of the current symbol. IncludeDeclaration bool `json:"includeDeclaration"` } // Reference options. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#referenceOptions type ReferenceOptions struct { WorkDoneProgressOptions } // Parameters for a {@link ReferencesRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#referenceParams type ReferenceParams struct { Context ReferenceContext `json:"context"` TextDocumentPositionParams WorkDoneProgressParams PartialResultParams } // Registration options for a {@link ReferencesRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#referenceRegistrationOptions type ReferenceRegistrationOptions struct { TextDocumentRegistrationOptions ReferenceOptions } // General parameters to register for a notification or to register a provider. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#registration type Registration struct { // The id used to register the request. The id can be used to deregister // the request again. ID string `json:"id"` // The method / capability to register for. Method string `json:"method"` // Options necessary for the registration. RegisterOptions interface{} `json:"registerOptions,omitempty"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#registrationParams type RegistrationParams struct { Registrations []Registration `json:"registrations"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#regularExpressionEngineKind type RegularExpressionEngineKind = string // (alias) // Client capabilities specific to regular expressions. // // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#regularExpressionsClientCapabilities type RegularExpressionsClientCapabilities struct { // The engine's name. Engine RegularExpressionEngineKind `json:"engine"` // The engine's version. Version string `json:"version,omitempty"` } // A full diagnostic report with a set of related documents. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#relatedFullDocumentDiagnosticReport type RelatedFullDocumentDiagnosticReport struct { // Diagnostics of related documents. This information is useful // in programming languages where code in a file A can generate // diagnostics in a file B which A depends on. An example of // such a language is C/C++ where marco definitions in a file // a.cpp and result in errors in a header file b.hpp. // // @since 3.17.0 RelatedDocuments map[DocumentUri]interface{} `json:"relatedDocuments,omitempty"` FullDocumentDiagnosticReport } // An unchanged diagnostic report with a set of related documents. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#relatedUnchangedDocumentDiagnosticReport type RelatedUnchangedDocumentDiagnosticReport struct { // Diagnostics of related documents. This information is useful // in programming languages where code in a file A can generate // diagnostics in a file B which A depends on. An example of // such a language is C/C++ where marco definitions in a file // a.cpp and result in errors in a header file b.hpp. // // @since 3.17.0 RelatedDocuments map[DocumentUri]interface{} `json:"relatedDocuments,omitempty"` UnchangedDocumentDiagnosticReport } // A relative pattern is a helper to construct glob patterns that are matched // relatively to a base URI. The common value for a `baseUri` is a workspace // folder root, but it can be another absolute URI as well. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#relativePattern type RelativePattern struct { // A workspace folder or a base URI to which this pattern will be matched // against relatively. BaseURI Or_RelativePattern_baseUri `json:"baseUri"` // The actual glob pattern; Pattern Pattern `json:"pattern"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#renameClientCapabilities type RenameClientCapabilities struct { // Whether rename supports dynamic registration. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` // Client supports testing for validity of rename operations // before execution. // // @since 3.12.0 PrepareSupport bool `json:"prepareSupport,omitempty"` // Client supports the default behavior result. // // The value indicates the default behavior used by the // client. // // @since 3.16.0 PrepareSupportDefaultBehavior *PrepareSupportDefaultBehavior `json:"prepareSupportDefaultBehavior,omitempty"` // Whether the client honors the change annotations in // text edits and resource operations returned via the // rename request's workspace edit by for example presenting // the workspace edit in the user interface and asking // for confirmation. // // @since 3.16.0 HonorsChangeAnnotations bool `json:"honorsChangeAnnotations,omitempty"` } // Rename file operation // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#renameFile type RenameFile struct { // A rename Kind string `json:"kind"` // The old (existing) location. OldURI DocumentUri `json:"oldUri"` // The new location. NewURI DocumentUri `json:"newUri"` // Rename options. Options *RenameFileOptions `json:"options,omitempty"` ResourceOperation } // Rename file options // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#renameFileOptions type RenameFileOptions struct { // Overwrite target if existing. Overwrite wins over `ignoreIfExists` Overwrite bool `json:"overwrite,omitempty"` // Ignores if target exists. IgnoreIfExists bool `json:"ignoreIfExists,omitempty"` } // The parameters sent in notifications/requests for user-initiated renames of // files. // // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#renameFilesParams type RenameFilesParams struct { // An array of all files/folders renamed in this operation. When a folder is renamed, only // the folder will be included, and not its children. Files []FileRename `json:"files"` } // Provider options for a {@link RenameRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#renameOptions type RenameOptions struct { // Renames should be checked and tested before being executed. // // @since version 3.12.0 PrepareProvider bool `json:"prepareProvider,omitempty"` WorkDoneProgressOptions } // The parameters of a {@link RenameRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#renameParams type RenameParams struct { // The document to rename. TextDocument TextDocumentIdentifier `json:"textDocument"` // The position at which this request was sent. Position Position `json:"position"` // The new name of the symbol. If the given name is not valid the // request must return a {@link ResponseError} with an // appropriate message set. NewName string `json:"newName"` WorkDoneProgressParams } // Registration options for a {@link RenameRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#renameRegistrationOptions type RenameRegistrationOptions struct { TextDocumentRegistrationOptions RenameOptions } // A generic resource operation. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#resourceOperation type ResourceOperation struct { // The resource operation kind. Kind string `json:"kind"` // An optional annotation identifier describing the operation. // // @since 3.16.0 AnnotationID *ChangeAnnotationIdentifier `json:"annotationId,omitempty"` } type ResourceOperationKind string // Save options. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#saveOptions type SaveOptions struct { // The client is supposed to include the content on save. IncludeText bool `json:"includeText,omitempty"` } // Describes the currently selected completion item. // // @since 3.18.0 // @proposed // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#selectedCompletionInfo type SelectedCompletionInfo struct { // The range that will be replaced if this completion item is accepted. Range Range `json:"range"` // The text the range will be replaced with if this completion is accepted. Text string `json:"text"` } // A selection range represents a part of a selection hierarchy. A selection range // may have a parent selection range that contains it. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#selectionRange type SelectionRange struct { // The {@link Range range} of this selection range. Range Range `json:"range"` // The parent selection range containing this range. Therefore `parent.range` must contain `this.range`. Parent *SelectionRange `json:"parent,omitempty"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#selectionRangeClientCapabilities type SelectionRangeClientCapabilities struct { // Whether implementation supports dynamic registration for selection range providers. If this is set to `true` // the client supports the new `SelectionRangeRegistrationOptions` return value for the corresponding server // capability as well. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#selectionRangeOptions type SelectionRangeOptions struct { WorkDoneProgressOptions } // A parameter literal used in selection range requests. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#selectionRangeParams type SelectionRangeParams struct { // The text document. TextDocument TextDocumentIdentifier `json:"textDocument"` // The positions inside the text document. Positions []Position `json:"positions"` WorkDoneProgressParams PartialResultParams } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#selectionRangeRegistrationOptions type SelectionRangeRegistrationOptions struct { SelectionRangeOptions TextDocumentRegistrationOptions StaticRegistrationOptions } // A set of predefined token modifiers. This set is not fixed // an clients can specify additional token types via the // corresponding client capabilities. // // @since 3.16.0 type SemanticTokenModifiers string // A set of predefined token types. This set is not fixed // an clients can specify additional token types via the // corresponding client capabilities. // // @since 3.16.0 type SemanticTokenTypes string // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#semanticTokens type SemanticTokens struct { // An optional result id. If provided and clients support delta updating // the client will include the result id in the next semantic token request. // A server can then instead of computing all semantic tokens again simply // send a delta. ResultID string `json:"resultId,omitempty"` // The actual tokens. Data []uint32 `json:"data"` } // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#semanticTokensClientCapabilities type SemanticTokensClientCapabilities struct { // Whether implementation supports dynamic registration. If this is set to `true` // the client supports the new `(TextDocumentRegistrationOptions & StaticRegistrationOptions)` // return value for the corresponding server capability as well. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` // Which requests the client supports and might send to the server // depending on the server's capability. Please note that clients might not // show semantic tokens or degrade some of the user experience if a range // or full request is advertised by the client but not provided by the // server. If for example the client capability `requests.full` and // `request.range` are both set to true but the server only provides a // range provider the client might not render a minimap correctly or might // even decide to not show any semantic tokens at all. Requests ClientSemanticTokensRequestOptions `json:"requests"` // The token types that the client supports. TokenTypes []string `json:"tokenTypes"` // The token modifiers that the client supports. TokenModifiers []string `json:"tokenModifiers"` // The token formats the clients supports. Formats []TokenFormat `json:"formats"` // Whether the client supports tokens that can overlap each other. OverlappingTokenSupport bool `json:"overlappingTokenSupport,omitempty"` // Whether the client supports tokens that can span multiple lines. MultilineTokenSupport bool `json:"multilineTokenSupport,omitempty"` // Whether the client allows the server to actively cancel a // semantic token request, e.g. supports returning // LSPErrorCodes.ServerCancelled. If a server does the client // needs to retrigger the request. // // @since 3.17.0 ServerCancelSupport bool `json:"serverCancelSupport,omitempty"` // Whether the client uses semantic tokens to augment existing // syntax tokens. If set to `true` client side created syntax // tokens and semantic tokens are both used for colorization. If // set to `false` the client only uses the returned semantic tokens // for colorization. // // If the value is `undefined` then the client behavior is not // specified. // // @since 3.17.0 AugmentsSyntaxTokens bool `json:"augmentsSyntaxTokens,omitempty"` } // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#semanticTokensDelta type SemanticTokensDelta struct { ResultID string `json:"resultId,omitempty"` // The semantic token edits to transform a previous result into a new result. Edits []SemanticTokensEdit `json:"edits"` } // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#semanticTokensDeltaParams type SemanticTokensDeltaParams struct { // The text document. TextDocument TextDocumentIdentifier `json:"textDocument"` // The result id of a previous response. The result Id can either point to a full response // or a delta response depending on what was received last. PreviousResultID string `json:"previousResultId"` WorkDoneProgressParams PartialResultParams } // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#semanticTokensDeltaPartialResult type SemanticTokensDeltaPartialResult struct { Edits []SemanticTokensEdit `json:"edits"` } // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#semanticTokensEdit type SemanticTokensEdit struct { // The start offset of the edit. Start uint32 `json:"start"` // The count of elements to remove. DeleteCount uint32 `json:"deleteCount"` // The elements to insert. Data []uint32 `json:"data,omitempty"` } // Semantic tokens options to support deltas for full documents // // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#semanticTokensFullDelta type SemanticTokensFullDelta struct { // The server supports deltas for full documents. Delta bool `json:"delta,omitempty"` } // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#semanticTokensLegend type SemanticTokensLegend struct { // The token types a server uses. TokenTypes []string `json:"tokenTypes"` // The token modifiers a server uses. TokenModifiers []string `json:"tokenModifiers"` } // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#semanticTokensOptions type SemanticTokensOptions struct { // The legend used by the server Legend SemanticTokensLegend `json:"legend"` // Server supports providing semantic tokens for a specific range // of a document. Range *Or_SemanticTokensOptions_range `json:"range,omitempty"` // Server supports providing semantic tokens for a full document. Full *Or_SemanticTokensOptions_full `json:"full,omitempty"` WorkDoneProgressOptions } // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#semanticTokensParams type SemanticTokensParams struct { // The text document. TextDocument TextDocumentIdentifier `json:"textDocument"` WorkDoneProgressParams PartialResultParams } // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#semanticTokensPartialResult type SemanticTokensPartialResult struct { Data []uint32 `json:"data"` } // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#semanticTokensRangeParams type SemanticTokensRangeParams struct { // The text document. TextDocument TextDocumentIdentifier `json:"textDocument"` // The range the semantic tokens are requested for. Range Range `json:"range"` WorkDoneProgressParams PartialResultParams } // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#semanticTokensRegistrationOptions type SemanticTokensRegistrationOptions struct { TextDocumentRegistrationOptions SemanticTokensOptions StaticRegistrationOptions } // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#semanticTokensWorkspaceClientCapabilities type SemanticTokensWorkspaceClientCapabilities struct { // Whether the client implementation supports a refresh request sent from // the server to the client. // // Note that this event is global and will force the client to refresh all // semantic tokens currently shown. It should be used with absolute care // and is useful for situation where a server for example detects a project // wide change that requires such a calculation. RefreshSupport bool `json:"refreshSupport,omitempty"` } // Defines the capabilities provided by a language // server. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#serverCapabilities type ServerCapabilities struct { // The position encoding the server picked from the encodings offered // by the client via the client capability `general.positionEncodings`. // // If the client didn't provide any position encodings the only valid // value that a server can return is 'utf-16'. // // If omitted it defaults to 'utf-16'. // // @since 3.17.0 PositionEncoding *PositionEncodingKind `json:"positionEncoding,omitempty"` // Defines how text documents are synced. Is either a detailed structure // defining each notification or for backwards compatibility the // TextDocumentSyncKind number. TextDocumentSync interface{} `json:"textDocumentSync,omitempty"` // Defines how notebook documents are synced. // // @since 3.17.0 NotebookDocumentSync *Or_ServerCapabilities_notebookDocumentSync `json:"notebookDocumentSync,omitempty"` // The server provides completion support. CompletionProvider *CompletionOptions `json:"completionProvider,omitempty"` // The server provides hover support. HoverProvider *Or_ServerCapabilities_hoverProvider `json:"hoverProvider,omitempty"` // The server provides signature help support. SignatureHelpProvider *SignatureHelpOptions `json:"signatureHelpProvider,omitempty"` // The server provides Goto Declaration support. DeclarationProvider *Or_ServerCapabilities_declarationProvider `json:"declarationProvider,omitempty"` // The server provides goto definition support. DefinitionProvider *Or_ServerCapabilities_definitionProvider `json:"definitionProvider,omitempty"` // The server provides Goto Type Definition support. TypeDefinitionProvider *Or_ServerCapabilities_typeDefinitionProvider `json:"typeDefinitionProvider,omitempty"` // The server provides Goto Implementation support. ImplementationProvider *Or_ServerCapabilities_implementationProvider `json:"implementationProvider,omitempty"` // The server provides find references support. ReferencesProvider *Or_ServerCapabilities_referencesProvider `json:"referencesProvider,omitempty"` // The server provides document highlight support. DocumentHighlightProvider *Or_ServerCapabilities_documentHighlightProvider `json:"documentHighlightProvider,omitempty"` // The server provides document symbol support. DocumentSymbolProvider *Or_ServerCapabilities_documentSymbolProvider `json:"documentSymbolProvider,omitempty"` // The server provides code actions. CodeActionOptions may only be // specified if the client states that it supports // `codeActionLiteralSupport` in its initial `initialize` request. CodeActionProvider interface{} `json:"codeActionProvider,omitempty"` // The server provides code lens. CodeLensProvider *CodeLensOptions `json:"codeLensProvider,omitempty"` // The server provides document link support. DocumentLinkProvider *DocumentLinkOptions `json:"documentLinkProvider,omitempty"` // The server provides color provider support. ColorProvider *Or_ServerCapabilities_colorProvider `json:"colorProvider,omitempty"` // The server provides workspace symbol support. WorkspaceSymbolProvider *Or_ServerCapabilities_workspaceSymbolProvider `json:"workspaceSymbolProvider,omitempty"` // The server provides document formatting. DocumentFormattingProvider *Or_ServerCapabilities_documentFormattingProvider `json:"documentFormattingProvider,omitempty"` // The server provides document range formatting. DocumentRangeFormattingProvider *Or_ServerCapabilities_documentRangeFormattingProvider `json:"documentRangeFormattingProvider,omitempty"` // The server provides document formatting on typing. DocumentOnTypeFormattingProvider *DocumentOnTypeFormattingOptions `json:"documentOnTypeFormattingProvider,omitempty"` // The server provides rename support. RenameOptions may only be // specified if the client states that it supports // `prepareSupport` in its initial `initialize` request. RenameProvider interface{} `json:"renameProvider,omitempty"` // The server provides folding provider support. FoldingRangeProvider *Or_ServerCapabilities_foldingRangeProvider `json:"foldingRangeProvider,omitempty"` // The server provides selection range support. SelectionRangeProvider *Or_ServerCapabilities_selectionRangeProvider `json:"selectionRangeProvider,omitempty"` // The server provides execute command support. ExecuteCommandProvider *ExecuteCommandOptions `json:"executeCommandProvider,omitempty"` // The server provides call hierarchy support. // // @since 3.16.0 CallHierarchyProvider *Or_ServerCapabilities_callHierarchyProvider `json:"callHierarchyProvider,omitempty"` // The server provides linked editing range support. // // @since 3.16.0 LinkedEditingRangeProvider *Or_ServerCapabilities_linkedEditingRangeProvider `json:"linkedEditingRangeProvider,omitempty"` // The server provides semantic tokens support. // // @since 3.16.0 SemanticTokensProvider interface{} `json:"semanticTokensProvider,omitempty"` // The server provides moniker support. // // @since 3.16.0 MonikerProvider *Or_ServerCapabilities_monikerProvider `json:"monikerProvider,omitempty"` // The server provides type hierarchy support. // // @since 3.17.0 TypeHierarchyProvider *Or_ServerCapabilities_typeHierarchyProvider `json:"typeHierarchyProvider,omitempty"` // The server provides inline values. // // @since 3.17.0 InlineValueProvider *Or_ServerCapabilities_inlineValueProvider `json:"inlineValueProvider,omitempty"` // The server provides inlay hints. // // @since 3.17.0 InlayHintProvider interface{} `json:"inlayHintProvider,omitempty"` // The server has support for pull model diagnostics. // // @since 3.17.0 DiagnosticProvider *Or_ServerCapabilities_diagnosticProvider `json:"diagnosticProvider,omitempty"` // Inline completion options used during static registration. // // @since 3.18.0 // @proposed InlineCompletionProvider *Or_ServerCapabilities_inlineCompletionProvider `json:"inlineCompletionProvider,omitempty"` // Workspace specific server capabilities. Workspace *WorkspaceOptions `json:"workspace,omitempty"` // Experimental server capabilities. Experimental interface{} `json:"experimental,omitempty"` } // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#serverCompletionItemOptions type ServerCompletionItemOptions struct { // The server has support for completion item label // details (see also `CompletionItemLabelDetails`) when // receiving a completion item in a resolve call. // // @since 3.17.0 LabelDetailsSupport bool `json:"labelDetailsSupport,omitempty"` } // Information about the server // // @since 3.15.0 // @since 3.18.0 ServerInfo type name added. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#serverInfo type ServerInfo struct { // The name of the server as defined by the server. Name string `json:"name"` // The server's version as defined by the server. Version string `json:"version,omitempty"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#setTraceParams type SetTraceParams struct { Value TraceValue `json:"value"` } // Client capabilities for the showDocument request. // // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#showDocumentClientCapabilities type ShowDocumentClientCapabilities struct { // The client has support for the showDocument // request. Support bool `json:"support"` } // Params to show a resource in the UI. // // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#showDocumentParams type ShowDocumentParams struct { // The uri to show. URI URI `json:"uri"` // Indicates to show the resource in an external program. // To show, for example, `https://code.visualstudio.com/` // in the default WEB browser set `external` to `true`. External bool `json:"external,omitempty"` // An optional property to indicate whether the editor // showing the document should take focus or not. // Clients might ignore this property if an external // program is started. TakeFocus bool `json:"takeFocus,omitempty"` // An optional selection range if the document is a text // document. Clients might ignore the property if an // external program is started or the file is not a text // file. Selection *Range `json:"selection,omitempty"` } // The result of a showDocument request. // // @since 3.16.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#showDocumentResult type ShowDocumentResult struct { // A boolean indicating if the show was successful. Success bool `json:"success"` } // The parameters of a notification message. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#showMessageParams type ShowMessageParams struct { // The message type. See {@link MessageType} Type MessageType `json:"type"` // The actual message. Message string `json:"message"` } // Show message request client capabilities // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#showMessageRequestClientCapabilities type ShowMessageRequestClientCapabilities struct { // Capabilities specific to the `MessageActionItem` type. MessageActionItem *ClientShowMessageActionItemOptions `json:"messageActionItem,omitempty"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#showMessageRequestParams type ShowMessageRequestParams struct { // The message type. See {@link MessageType} Type MessageType `json:"type"` // The actual message. Message string `json:"message"` // The message action items to present. Actions []MessageActionItem `json:"actions,omitempty"` } // Signature help represents the signature of something // callable. There can be multiple signature but only one // active and only one active parameter. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#signatureHelp type SignatureHelp struct { // One or more signatures. Signatures []SignatureInformation `json:"signatures"` // The active signature. If omitted or the value lies outside the // range of `signatures` the value defaults to zero or is ignored if // the `SignatureHelp` has no signatures. // // Whenever possible implementors should make an active decision about // the active signature and shouldn't rely on a default value. // // In future version of the protocol this property might become // mandatory to better express this. ActiveSignature uint32 `json:"activeSignature,omitempty"` // The active parameter of the active signature. // // If `null`, no parameter of the signature is active (for example a named // argument that does not match any declared parameters). This is only valid // if the client specifies the client capability // `textDocument.signatureHelp.noActiveParameterSupport === true` // // If omitted or the value lies outside the range of // `signatures[activeSignature].parameters` defaults to 0 if the active // signature has parameters. // // If the active signature has no parameters it is ignored. // // In future version of the protocol this property might become // mandatory (but still nullable) to better express the active parameter if // the active signature does have any. ActiveParameter uint32 `json:"activeParameter,omitempty"` } // Client Capabilities for a {@link SignatureHelpRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#signatureHelpClientCapabilities type SignatureHelpClientCapabilities struct { // Whether signature help supports dynamic registration. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` // The client supports the following `SignatureInformation` // specific properties. SignatureInformation *ClientSignatureInformationOptions `json:"signatureInformation,omitempty"` // The client supports to send additional context information for a // `textDocument/signatureHelp` request. A client that opts into // contextSupport will also support the `retriggerCharacters` on // `SignatureHelpOptions`. // // @since 3.15.0 ContextSupport bool `json:"contextSupport,omitempty"` } // Additional information about the context in which a signature help request was triggered. // // @since 3.15.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#signatureHelpContext type SignatureHelpContext struct { // Action that caused signature help to be triggered. TriggerKind SignatureHelpTriggerKind `json:"triggerKind"` // Character that caused signature help to be triggered. // // This is undefined when `triggerKind !== SignatureHelpTriggerKind.TriggerCharacter` TriggerCharacter string `json:"triggerCharacter,omitempty"` // `true` if signature help was already showing when it was triggered. // // Retriggers occurs when the signature help is already active and can be caused by actions such as // typing a trigger character, a cursor move, or document content changes. IsRetrigger bool `json:"isRetrigger"` // The currently active `SignatureHelp`. // // The `activeSignatureHelp` has its `SignatureHelp.activeSignature` field updated based on // the user navigating through available signatures. ActiveSignatureHelp *SignatureHelp `json:"activeSignatureHelp,omitempty"` } // Server Capabilities for a {@link SignatureHelpRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#signatureHelpOptions type SignatureHelpOptions struct { // List of characters that trigger signature help automatically. TriggerCharacters []string `json:"triggerCharacters,omitempty"` // List of characters that re-trigger signature help. // // These trigger characters are only active when signature help is already showing. All trigger characters // are also counted as re-trigger characters. // // @since 3.15.0 RetriggerCharacters []string `json:"retriggerCharacters,omitempty"` WorkDoneProgressOptions } // Parameters for a {@link SignatureHelpRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#signatureHelpParams type SignatureHelpParams struct { // The signature help context. This is only available if the client specifies // to send this using the client capability `textDocument.signatureHelp.contextSupport === true` // // @since 3.15.0 Context *SignatureHelpContext `json:"context,omitempty"` TextDocumentPositionParams WorkDoneProgressParams } // Registration options for a {@link SignatureHelpRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#signatureHelpRegistrationOptions type SignatureHelpRegistrationOptions struct { TextDocumentRegistrationOptions SignatureHelpOptions } // How a signature help was triggered. // // @since 3.15.0 type SignatureHelpTriggerKind uint32 // Represents the signature of something callable. A signature // can have a label, like a function-name, a doc-comment, and // a set of parameters. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#signatureInformation type SignatureInformation struct { // The label of this signature. Will be shown in // the UI. Label string `json:"label"` // The human-readable doc-comment of this signature. Will be shown // in the UI but can be omitted. Documentation *Or_SignatureInformation_documentation `json:"documentation,omitempty"` // The parameters of this signature. Parameters []ParameterInformation `json:"parameters,omitempty"` // The index of the active parameter. // // If `null`, no parameter of the signature is active (for example a named // argument that does not match any declared parameters). This is only valid // if the client specifies the client capability // `textDocument.signatureHelp.noActiveParameterSupport === true` // // If provided (or `null`), this is used in place of // `SignatureHelp.activeParameter`. // // @since 3.16.0 ActiveParameter uint32 `json:"activeParameter,omitempty"` } // An interactive text edit. // // @since 3.18.0 // @proposed // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#snippetTextEdit type SnippetTextEdit struct { // The range of the text document to be manipulated. Range Range `json:"range"` // The snippet to be inserted. Snippet StringValue `json:"snippet"` // The actual identifier of the snippet edit. AnnotationID *ChangeAnnotationIdentifier `json:"annotationId,omitempty"` } // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#staleRequestSupportOptions type StaleRequestSupportOptions struct { // The client will actively cancel the request. Cancel bool `json:"cancel"` // The list of requests for which the client // will retry the request if it receives a // response with error code `ContentModified` RetryOnContentModified []string `json:"retryOnContentModified"` } // Static registration options to be returned in the initialize // request. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#staticRegistrationOptions type StaticRegistrationOptions struct { // The id used to register the request. The id can be used to deregister // the request again. See also Registration#id. ID string `json:"id,omitempty"` } // A string value used as a snippet is a template which allows to insert text // and to control the editor cursor when insertion happens. // // A snippet can define tab stops and placeholders with `$1`, `$2` // and `${3:foo}`. `$0` defines the final tab stop, it defaults to // the end of the snippet. Variables are defined with `$name` and // `${name:default value}`. // // @since 3.18.0 // @proposed // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#stringValue type StringValue struct { // The kind of string value. Kind string `json:"kind"` // The snippet string. Value string `json:"value"` } // Represents information about programming constructs like variables, classes, // interfaces etc. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#symbolInformation type SymbolInformation struct { // extends BaseSymbolInformation // Indicates if this symbol is deprecated. // // @deprecated Use tags instead Deprecated bool `json:"deprecated,omitempty"` // The location of this symbol. The location's range is used by a tool // to reveal the location in the editor. If the symbol is selected in the // tool the range's start information is used to position the cursor. So // the range usually spans more than the actual symbol's name and does // normally include things like visibility modifiers. // // The range doesn't have to denote a node range in the sense of an abstract // syntax tree. It can therefore not be used to re-construct a hierarchy of // the symbols. Location Location `json:"location"` // The name of this symbol. Name string `json:"name"` // The kind of this symbol. Kind SymbolKind `json:"kind"` // Tags for this symbol. // // @since 3.16.0 Tags []SymbolTag `json:"tags,omitempty"` // The name of the symbol containing this symbol. This information is for // user interface purposes (e.g. to render a qualifier in the user interface // if necessary). It can't be used to re-infer a hierarchy for the document // symbols. ContainerName string `json:"containerName,omitempty"` } // A symbol kind. type SymbolKind uint32 // Symbol tags are extra annotations that tweak the rendering of a symbol. // // @since 3.16 type SymbolTag uint32 // Describe options to be used when registered for text document change events. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#textDocumentChangeRegistrationOptions type TextDocumentChangeRegistrationOptions struct { // How documents are synced to the server. SyncKind TextDocumentSyncKind `json:"syncKind"` TextDocumentRegistrationOptions } // Text document specific client capabilities. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#textDocumentClientCapabilities type TextDocumentClientCapabilities struct { // Defines which synchronization capabilities the client supports. Synchronization *TextDocumentSyncClientCapabilities `json:"synchronization,omitempty"` // Capabilities specific to the `textDocument/completion` request. Completion CompletionClientCapabilities `json:"completion,omitempty"` // Capabilities specific to the `textDocument/hover` request. Hover *HoverClientCapabilities `json:"hover,omitempty"` // Capabilities specific to the `textDocument/signatureHelp` request. SignatureHelp *SignatureHelpClientCapabilities `json:"signatureHelp,omitempty"` // Capabilities specific to the `textDocument/declaration` request. // // @since 3.14.0 Declaration *DeclarationClientCapabilities `json:"declaration,omitempty"` // Capabilities specific to the `textDocument/definition` request. Definition *DefinitionClientCapabilities `json:"definition,omitempty"` // Capabilities specific to the `textDocument/typeDefinition` request. // // @since 3.6.0 TypeDefinition *TypeDefinitionClientCapabilities `json:"typeDefinition,omitempty"` // Capabilities specific to the `textDocument/implementation` request. // // @since 3.6.0 Implementation *ImplementationClientCapabilities `json:"implementation,omitempty"` // Capabilities specific to the `textDocument/references` request. References *ReferenceClientCapabilities `json:"references,omitempty"` // Capabilities specific to the `textDocument/documentHighlight` request. DocumentHighlight *DocumentHighlightClientCapabilities `json:"documentHighlight,omitempty"` // Capabilities specific to the `textDocument/documentSymbol` request. DocumentSymbol DocumentSymbolClientCapabilities `json:"documentSymbol,omitempty"` // Capabilities specific to the `textDocument/codeAction` request. CodeAction CodeActionClientCapabilities `json:"codeAction,omitempty"` // Capabilities specific to the `textDocument/codeLens` request. CodeLens *CodeLensClientCapabilities `json:"codeLens,omitempty"` // Capabilities specific to the `textDocument/documentLink` request. DocumentLink *DocumentLinkClientCapabilities `json:"documentLink,omitempty"` // Capabilities specific to the `textDocument/documentColor` and the // `textDocument/colorPresentation` request. // // @since 3.6.0 ColorProvider *DocumentColorClientCapabilities `json:"colorProvider,omitempty"` // Capabilities specific to the `textDocument/formatting` request. Formatting *DocumentFormattingClientCapabilities `json:"formatting,omitempty"` // Capabilities specific to the `textDocument/rangeFormatting` request. RangeFormatting *DocumentRangeFormattingClientCapabilities `json:"rangeFormatting,omitempty"` // Capabilities specific to the `textDocument/onTypeFormatting` request. OnTypeFormatting *DocumentOnTypeFormattingClientCapabilities `json:"onTypeFormatting,omitempty"` // Capabilities specific to the `textDocument/rename` request. Rename *RenameClientCapabilities `json:"rename,omitempty"` // Capabilities specific to the `textDocument/foldingRange` request. // // @since 3.10.0 FoldingRange *FoldingRangeClientCapabilities `json:"foldingRange,omitempty"` // Capabilities specific to the `textDocument/selectionRange` request. // // @since 3.15.0 SelectionRange *SelectionRangeClientCapabilities `json:"selectionRange,omitempty"` // Capabilities specific to the `textDocument/publishDiagnostics` notification. PublishDiagnostics PublishDiagnosticsClientCapabilities `json:"publishDiagnostics,omitempty"` // Capabilities specific to the various call hierarchy requests. // // @since 3.16.0 CallHierarchy *CallHierarchyClientCapabilities `json:"callHierarchy,omitempty"` // Capabilities specific to the various semantic token request. // // @since 3.16.0 SemanticTokens SemanticTokensClientCapabilities `json:"semanticTokens,omitempty"` // Capabilities specific to the `textDocument/linkedEditingRange` request. // // @since 3.16.0 LinkedEditingRange *LinkedEditingRangeClientCapabilities `json:"linkedEditingRange,omitempty"` // Client capabilities specific to the `textDocument/moniker` request. // // @since 3.16.0 Moniker *MonikerClientCapabilities `json:"moniker,omitempty"` // Capabilities specific to the various type hierarchy requests. // // @since 3.17.0 TypeHierarchy *TypeHierarchyClientCapabilities `json:"typeHierarchy,omitempty"` // Capabilities specific to the `textDocument/inlineValue` request. // // @since 3.17.0 InlineValue *InlineValueClientCapabilities `json:"inlineValue,omitempty"` // Capabilities specific to the `textDocument/inlayHint` request. // // @since 3.17.0 InlayHint *InlayHintClientCapabilities `json:"inlayHint,omitempty"` // Capabilities specific to the diagnostic pull model. // // @since 3.17.0 Diagnostic *DiagnosticClientCapabilities `json:"diagnostic,omitempty"` // Client capabilities specific to inline completions. // // @since 3.18.0 // @proposed InlineCompletion *InlineCompletionClientCapabilities `json:"inlineCompletion,omitempty"` } // An event describing a change to a text document. If only a text is provided // it is considered to be the full content of the document. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#textDocumentContentChangeEvent type TextDocumentContentChangeEvent = Or_TextDocumentContentChangeEvent // (alias) // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#textDocumentContentChangePartial type TextDocumentContentChangePartial struct { // The range of the document that changed. Range *Range `json:"range,omitempty"` // The optional length of the range that got replaced. // // @deprecated use range instead. RangeLength uint32 `json:"rangeLength,omitempty"` // The new text for the provided range. Text string `json:"text"` } // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#textDocumentContentChangeWholeDocument type TextDocumentContentChangeWholeDocument struct { // The new text of the whole document. Text string `json:"text"` } // Client capabilities for a text document content provider. // // @since 3.18.0 // @proposed // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#textDocumentContentClientCapabilities type TextDocumentContentClientCapabilities struct { // Text document content provider supports dynamic registration. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` } // Text document content provider options. // // @since 3.18.0 // @proposed // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#textDocumentContentOptions type TextDocumentContentOptions struct { // The scheme for which the server provides content. Scheme string `json:"scheme"` } // Parameters for the `workspace/textDocumentContent` request. // // @since 3.18.0 // @proposed // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#textDocumentContentParams type TextDocumentContentParams struct { // The uri of the text document. URI DocumentUri `json:"uri"` } // Parameters for the `workspace/textDocumentContent/refresh` request. // // @since 3.18.0 // @proposed // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#textDocumentContentRefreshParams type TextDocumentContentRefreshParams struct { // The uri of the text document to refresh. URI DocumentUri `json:"uri"` } // Text document content provider registration options. // // @since 3.18.0 // @proposed // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#textDocumentContentRegistrationOptions type TextDocumentContentRegistrationOptions struct { TextDocumentContentOptions StaticRegistrationOptions } // Describes textual changes on a text document. A TextDocumentEdit describes all changes // on a document version Si and after they are applied move the document to version Si+1. // So the creator of a TextDocumentEdit doesn't need to sort the array of edits or do any // kind of ordering. However the edits must be non overlapping. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#textDocumentEdit type TextDocumentEdit struct { // The text document to change. TextDocument OptionalVersionedTextDocumentIdentifier `json:"textDocument"` // The edits to be applied. // // @since 3.16.0 - support for AnnotatedTextEdit. This is guarded using a // client capability. // // @since 3.18.0 - support for SnippetTextEdit. This is guarded using a // client capability. Edits []Or_TextDocumentEdit_edits_Elem `json:"edits"` } // A document filter denotes a document by different properties like // the {@link TextDocument.languageId language}, the {@link Uri.scheme scheme} of // its resource, or a glob-pattern that is applied to the {@link TextDocument.fileName path}. // // Glob patterns can have the following syntax: // // - `*` to match one or more characters in a path segment // - `?` to match on one character in a path segment // - `**` to match any number of path segments, including none // - `{}` to group sub patterns into an OR expression. (e.g. `**​/*.{ts,js}` matches all TypeScript and JavaScript files) // - `[]` to declare a range of characters to match in a path segment (e.g., `example.[0-9]` to match on `example.0`, `example.1`, …) // - `[!...]` to negate a range of characters to match in a path segment (e.g., `example.[!0-9]` to match on `example.a`, `example.b`, but not `example.0`) // // @sample A language filter that applies to typescript files on disk: `{ language: 'typescript', scheme: 'file' }` // @sample A language filter that applies to all package.json paths: `{ language: 'json', pattern: '**package.json' }` // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#textDocumentFilter type TextDocumentFilter = Or_TextDocumentFilter // (alias) // A document filter where `language` is required field. // // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#textDocumentFilterLanguage type TextDocumentFilterLanguage struct { // A language id, like `typescript`. Language string `json:"language"` // A Uri {@link Uri.scheme scheme}, like `file` or `untitled`. Scheme string `json:"scheme,omitempty"` // A glob pattern, like **​/*.{ts,js}. See TextDocumentFilter for examples. // // @since 3.18.0 - support for relative patterns. Pattern *GlobPattern `json:"pattern,omitempty"` } // A document filter where `pattern` is required field. // // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#textDocumentFilterPattern type TextDocumentFilterPattern struct { // A language id, like `typescript`. Language string `json:"language,omitempty"` // A Uri {@link Uri.scheme scheme}, like `file` or `untitled`. Scheme string `json:"scheme,omitempty"` // A glob pattern, like **​/*.{ts,js}. See TextDocumentFilter for examples. // // @since 3.18.0 - support for relative patterns. Pattern GlobPattern `json:"pattern"` } // A document filter where `scheme` is required field. // // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#textDocumentFilterScheme type TextDocumentFilterScheme struct { // A language id, like `typescript`. Language string `json:"language,omitempty"` // A Uri {@link Uri.scheme scheme}, like `file` or `untitled`. Scheme string `json:"scheme"` // A glob pattern, like **​/*.{ts,js}. See TextDocumentFilter for examples. // // @since 3.18.0 - support for relative patterns. Pattern *GlobPattern `json:"pattern,omitempty"` } // A literal to identify a text document in the client. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#textDocumentIdentifier type TextDocumentIdentifier struct { // The text document's uri. URI DocumentUri `json:"uri"` } // An item to transfer a text document from the client to the // server. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#textDocumentItem type TextDocumentItem struct { // The text document's uri. URI DocumentUri `json:"uri"` // The text document's language identifier. LanguageID LanguageKind `json:"languageId"` // The version number of this document (it will increase after each // change, including undo/redo). Version int32 `json:"version"` // The content of the opened text document. Text string `json:"text"` } // A parameter literal used in requests to pass a text document and a position inside that // document. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#textDocumentPositionParams type TextDocumentPositionParams struct { // The text document. TextDocument TextDocumentIdentifier `json:"textDocument"` // The position inside the text document. Position Position `json:"position"` } // General text document registration options. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#textDocumentRegistrationOptions type TextDocumentRegistrationOptions struct { // A document selector to identify the scope of the registration. If set to null // the document selector provided on the client side will be used. DocumentSelector DocumentSelector `json:"documentSelector"` } // Represents reasons why a text document is saved. type TextDocumentSaveReason uint32 // Save registration options. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#textDocumentSaveRegistrationOptions type TextDocumentSaveRegistrationOptions struct { TextDocumentRegistrationOptions SaveOptions } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#textDocumentSyncClientCapabilities type TextDocumentSyncClientCapabilities struct { // Whether text document synchronization supports dynamic registration. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` // The client supports sending will save notifications. WillSave bool `json:"willSave,omitempty"` // The client supports sending a will save request and // waits for a response providing text edits which will // be applied to the document before it is saved. WillSaveWaitUntil bool `json:"willSaveWaitUntil,omitempty"` // The client supports did save notifications. DidSave bool `json:"didSave,omitempty"` } // Defines how the host (editor) should sync // document changes to the language server. type TextDocumentSyncKind uint32 // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#textDocumentSyncOptions type TextDocumentSyncOptions struct { // Open and close notifications are sent to the server. If omitted open close notification should not // be sent. OpenClose bool `json:"openClose,omitempty"` // Change notifications are sent to the server. See TextDocumentSyncKind.None, TextDocumentSyncKind.Full // and TextDocumentSyncKind.Incremental. If omitted it defaults to TextDocumentSyncKind.None. Change TextDocumentSyncKind `json:"change,omitempty"` // If present will save notifications are sent to the server. If omitted the notification should not be // sent. WillSave bool `json:"willSave,omitempty"` // If present will save wait until requests are sent to the server. If omitted the request should not be // sent. WillSaveWaitUntil bool `json:"willSaveWaitUntil,omitempty"` // If present save notifications are sent to the server. If omitted the notification should not be // sent. Save *SaveOptions `json:"save,omitempty"` } // A text edit applicable to a text document. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#textEdit type TextEdit struct { // The range of the text document to be manipulated. To insert // text into a document create a range where start === end. Range Range `json:"range"` // The string to be inserted. For delete operations use an // empty string. NewText string `json:"newText"` } type TokenFormat string type TraceValue string // created for Tuple type Tuple_ParameterInformation_label_Item1 struct { Fld0 uint32 `json:"fld0"` Fld1 uint32 `json:"fld1"` } // Since 3.6.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#typeDefinitionClientCapabilities type TypeDefinitionClientCapabilities struct { // Whether implementation supports dynamic registration. If this is set to `true` // the client supports the new `TypeDefinitionRegistrationOptions` return value // for the corresponding server capability as well. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` // The client supports additional metadata in the form of definition links. // // Since 3.14.0 LinkSupport bool `json:"linkSupport,omitempty"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#typeDefinitionOptions type TypeDefinitionOptions struct { WorkDoneProgressOptions } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#typeDefinitionParams type TypeDefinitionParams struct { TextDocumentPositionParams WorkDoneProgressParams PartialResultParams } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#typeDefinitionRegistrationOptions type TypeDefinitionRegistrationOptions struct { TextDocumentRegistrationOptions TypeDefinitionOptions StaticRegistrationOptions } // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#typeHierarchyClientCapabilities type TypeHierarchyClientCapabilities struct { // Whether implementation supports dynamic registration. If this is set to `true` // the client supports the new `(TextDocumentRegistrationOptions & StaticRegistrationOptions)` // return value for the corresponding server capability as well. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` } // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#typeHierarchyItem type TypeHierarchyItem struct { // The name of this item. Name string `json:"name"` // The kind of this item. Kind SymbolKind `json:"kind"` // Tags for this item. Tags []SymbolTag `json:"tags,omitempty"` // More detail for this item, e.g. the signature of a function. Detail string `json:"detail,omitempty"` // The resource identifier of this item. URI DocumentUri `json:"uri"` // The range enclosing this symbol not including leading/trailing whitespace // but everything else, e.g. comments and code. Range Range `json:"range"` // The range that should be selected and revealed when this symbol is being // picked, e.g. the name of a function. Must be contained by the // {@link TypeHierarchyItem.range `range`}. SelectionRange Range `json:"selectionRange"` // A data entry field that is preserved between a type hierarchy prepare and // supertypes or subtypes requests. It could also be used to identify the // type hierarchy in the server, helping improve the performance on // resolving supertypes and subtypes. Data interface{} `json:"data,omitempty"` } // Type hierarchy options used during static registration. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#typeHierarchyOptions type TypeHierarchyOptions struct { WorkDoneProgressOptions } // The parameter of a `textDocument/prepareTypeHierarchy` request. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#typeHierarchyPrepareParams type TypeHierarchyPrepareParams struct { TextDocumentPositionParams WorkDoneProgressParams } // Type hierarchy options used during static or dynamic registration. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#typeHierarchyRegistrationOptions type TypeHierarchyRegistrationOptions struct { TextDocumentRegistrationOptions TypeHierarchyOptions StaticRegistrationOptions } // The parameter of a `typeHierarchy/subtypes` request. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#typeHierarchySubtypesParams type TypeHierarchySubtypesParams struct { Item TypeHierarchyItem `json:"item"` WorkDoneProgressParams PartialResultParams } // The parameter of a `typeHierarchy/supertypes` request. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#typeHierarchySupertypesParams type TypeHierarchySupertypesParams struct { Item TypeHierarchyItem `json:"item"` WorkDoneProgressParams PartialResultParams } // A diagnostic report indicating that the last returned // report is still accurate. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#unchangedDocumentDiagnosticReport type UnchangedDocumentDiagnosticReport struct { // A document diagnostic report indicating // no changes to the last result. A server can // only return `unchanged` if result ids are // provided. Kind string `json:"kind"` // A result id which will be sent on the next // diagnostic request for the same document. ResultID string `json:"resultId"` } // Moniker uniqueness level to define scope of the moniker. // // @since 3.16.0 type UniquenessLevel string // General parameters to unregister a request or notification. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#unregistration type Unregistration struct { // The id used to unregister the request or notification. Usually an id // provided during the register request. ID string `json:"id"` // The method to unregister for. Method string `json:"method"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#unregistrationParams type UnregistrationParams struct { Unregisterations []Unregistration `json:"unregisterations"` } // A versioned notebook document identifier. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#versionedNotebookDocumentIdentifier type VersionedNotebookDocumentIdentifier struct { // The version number of this notebook document. Version int32 `json:"version"` // The notebook document's uri. URI URI `json:"uri"` } // A text document identifier to denote a specific version of a text document. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#versionedTextDocumentIdentifier type VersionedTextDocumentIdentifier struct { // The version number of this document. Version int32 `json:"version"` TextDocumentIdentifier } type WatchKind = uint32 // The parameters sent in a will save text document notification. // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#willSaveTextDocumentParams type WillSaveTextDocumentParams struct { // The document that will be saved. TextDocument TextDocumentIdentifier `json:"textDocument"` // The 'TextDocumentSaveReason'. Reason TextDocumentSaveReason `json:"reason"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#windowClientCapabilities type WindowClientCapabilities struct { // It indicates whether the client supports server initiated // progress using the `window/workDoneProgress/create` request. // // The capability also controls Whether client supports handling // of progress notifications. If set servers are allowed to report a // `workDoneProgress` property in the request specific server // capabilities. // // @since 3.15.0 WorkDoneProgress bool `json:"workDoneProgress,omitempty"` // Capabilities specific to the showMessage request. // // @since 3.16.0 ShowMessage *ShowMessageRequestClientCapabilities `json:"showMessage,omitempty"` // Capabilities specific to the showDocument request. // // @since 3.16.0 ShowDocument *ShowDocumentClientCapabilities `json:"showDocument,omitempty"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#workDoneProgressBegin type WorkDoneProgressBegin struct { Kind string `json:"kind"` // Mandatory title of the progress operation. Used to briefly inform about // the kind of operation being performed. // // Examples: "Indexing" or "Linking dependencies". Title string `json:"title"` // Controls if a cancel button should show to allow the user to cancel the // long running operation. Clients that don't support cancellation are allowed // to ignore the setting. Cancellable bool `json:"cancellable,omitempty"` // Optional, more detailed associated progress message. Contains // complementary information to the `title`. // // Examples: "3/25 files", "project/src/module2", "node_modules/some_dep". // If unset, the previous progress message (if any) is still valid. Message string `json:"message,omitempty"` // Optional progress percentage to display (value 100 is considered 100%). // If not provided infinite progress is assumed and clients are allowed // to ignore the `percentage` value in subsequent in report notifications. // // The value should be steadily rising. Clients are free to ignore values // that are not following this rule. The value range is [0, 100]. Percentage uint32 `json:"percentage,omitempty"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#workDoneProgressCancelParams type WorkDoneProgressCancelParams struct { // The token to be used to report progress. Token ProgressToken `json:"token"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#workDoneProgressCreateParams type WorkDoneProgressCreateParams struct { // The token to be used to report progress. Token ProgressToken `json:"token"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#workDoneProgressEnd type WorkDoneProgressEnd struct { Kind string `json:"kind"` // Optional, a final message indicating to for example indicate the outcome // of the operation. Message string `json:"message,omitempty"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#workDoneProgressOptions type WorkDoneProgressOptions struct { WorkDoneProgress bool `json:"workDoneProgress,omitempty"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#workDoneProgressParams type WorkDoneProgressParams struct { // An optional token that a server can use to report work done progress. WorkDoneToken ProgressToken `json:"workDoneToken,omitempty"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#workDoneProgressReport type WorkDoneProgressReport struct { Kind string `json:"kind"` // Controls enablement state of a cancel button. // // Clients that don't support cancellation or don't support controlling the button's // enablement state are allowed to ignore the property. Cancellable bool `json:"cancellable,omitempty"` // Optional, more detailed associated progress message. Contains // complementary information to the `title`. // // Examples: "3/25 files", "project/src/module2", "node_modules/some_dep". // If unset, the previous progress message (if any) is still valid. Message string `json:"message,omitempty"` // Optional progress percentage to display (value 100 is considered 100%). // If not provided infinite progress is assumed and clients are allowed // to ignore the `percentage` value in subsequent in report notifications. // // The value should be steadily rising. Clients are free to ignore values // that are not following this rule. The value range is [0, 100] Percentage uint32 `json:"percentage,omitempty"` } // Workspace specific client capabilities. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#workspaceClientCapabilities type WorkspaceClientCapabilities struct { // The client supports applying batch edits // to the workspace by supporting the request // 'workspace/applyEdit' ApplyEdit bool `json:"applyEdit,omitempty"` // Capabilities specific to `WorkspaceEdit`s. WorkspaceEdit *WorkspaceEditClientCapabilities `json:"workspaceEdit,omitempty"` // Capabilities specific to the `workspace/didChangeConfiguration` notification. DidChangeConfiguration DidChangeConfigurationClientCapabilities `json:"didChangeConfiguration,omitempty"` // Capabilities specific to the `workspace/didChangeWatchedFiles` notification. DidChangeWatchedFiles DidChangeWatchedFilesClientCapabilities `json:"didChangeWatchedFiles,omitempty"` // Capabilities specific to the `workspace/symbol` request. Symbol *WorkspaceSymbolClientCapabilities `json:"symbol,omitempty"` // Capabilities specific to the `workspace/executeCommand` request. ExecuteCommand *ExecuteCommandClientCapabilities `json:"executeCommand,omitempty"` // The client has support for workspace folders. // // @since 3.6.0 WorkspaceFolders bool `json:"workspaceFolders,omitempty"` // The client supports `workspace/configuration` requests. // // @since 3.6.0 Configuration bool `json:"configuration,omitempty"` // Capabilities specific to the semantic token requests scoped to the // workspace. // // @since 3.16.0. SemanticTokens *SemanticTokensWorkspaceClientCapabilities `json:"semanticTokens,omitempty"` // Capabilities specific to the code lens requests scoped to the // workspace. // // @since 3.16.0. CodeLens *CodeLensWorkspaceClientCapabilities `json:"codeLens,omitempty"` // The client has support for file notifications/requests for user operations on files. // // Since 3.16.0 FileOperations *FileOperationClientCapabilities `json:"fileOperations,omitempty"` // Capabilities specific to the inline values requests scoped to the // workspace. // // @since 3.17.0. InlineValue *InlineValueWorkspaceClientCapabilities `json:"inlineValue,omitempty"` // Capabilities specific to the inlay hint requests scoped to the // workspace. // // @since 3.17.0. InlayHint *InlayHintWorkspaceClientCapabilities `json:"inlayHint,omitempty"` // Capabilities specific to the diagnostic requests scoped to the // workspace. // // @since 3.17.0. Diagnostics *DiagnosticWorkspaceClientCapabilities `json:"diagnostics,omitempty"` // Capabilities specific to the folding range requests scoped to the workspace. // // @since 3.18.0 // @proposed FoldingRange *FoldingRangeWorkspaceClientCapabilities `json:"foldingRange,omitempty"` // Capabilities specific to the `workspace/textDocumentContent` request. // // @since 3.18.0 // @proposed TextDocumentContent *TextDocumentContentClientCapabilities `json:"textDocumentContent,omitempty"` } // Parameters of the workspace diagnostic request. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#workspaceDiagnosticParams type WorkspaceDiagnosticParams struct { // The additional identifier provided during registration. Identifier string `json:"identifier,omitempty"` // The currently known diagnostic reports with their // previous result ids. PreviousResultIds []PreviousResultId `json:"previousResultIds"` WorkDoneProgressParams PartialResultParams } // A workspace diagnostic report. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#workspaceDiagnosticReport type WorkspaceDiagnosticReport struct { Items []WorkspaceDocumentDiagnosticReport `json:"items"` } // A partial result for a workspace diagnostic report. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#workspaceDiagnosticReportPartialResult type WorkspaceDiagnosticReportPartialResult struct { Items []WorkspaceDocumentDiagnosticReport `json:"items"` } // A workspace diagnostic document report. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#workspaceDocumentDiagnosticReport type WorkspaceDocumentDiagnosticReport = Or_WorkspaceDocumentDiagnosticReport // (alias) // A workspace edit represents changes to many resources managed in the workspace. The edit // should either provide `changes` or `documentChanges`. If documentChanges are present // they are preferred over `changes` if the client can handle versioned document edits. // // Since version 3.13.0 a workspace edit can contain resource operations as well. If resource // operations are present clients need to execute the operations in the order in which they // are provided. So a workspace edit for example can consist of the following two changes: // (1) a create file a.txt and (2) a text document edit which insert text into file a.txt. // // An invalid sequence (e.g. (1) delete file a.txt and (2) insert text into file a.txt) will // cause failure of the operation. How the client recovers from the failure is described by // the client capability: `workspace.workspaceEdit.failureHandling` // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#workspaceEdit type WorkspaceEdit struct { // Holds changes to existing resources. Changes map[DocumentUri][]TextEdit `json:"changes,omitempty"` // Depending on the client capability `workspace.workspaceEdit.resourceOperations` document changes // are either an array of `TextDocumentEdit`s to express changes to n different text documents // where each text document edit addresses a specific version of a text document. Or it can contain // above `TextDocumentEdit`s mixed with create, rename and delete file / folder operations. // // Whether a client supports versioned document edits is expressed via // `workspace.workspaceEdit.documentChanges` client capability. // // If a client neither supports `documentChanges` nor `workspace.workspaceEdit.resourceOperations` then // only plain `TextEdit`s using the `changes` property are supported. DocumentChanges []DocumentChange `json:"documentChanges,omitempty"` // A map of change annotations that can be referenced in `AnnotatedTextEdit`s or create, rename and // delete file / folder operations. // // Whether clients honor this property depends on the client capability `workspace.changeAnnotationSupport`. // // @since 3.16.0 ChangeAnnotations map[ChangeAnnotationIdentifier]ChangeAnnotation `json:"changeAnnotations,omitempty"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#workspaceEditClientCapabilities type WorkspaceEditClientCapabilities struct { // The client supports versioned document changes in `WorkspaceEdit`s DocumentChanges bool `json:"documentChanges,omitempty"` // The resource operations the client supports. Clients should at least // support 'create', 'rename' and 'delete' files and folders. // // @since 3.13.0 ResourceOperations []ResourceOperationKind `json:"resourceOperations,omitempty"` // The failure handling strategy of a client if applying the workspace edit // fails. // // @since 3.13.0 FailureHandling *FailureHandlingKind `json:"failureHandling,omitempty"` // Whether the client normalizes line endings to the client specific // setting. // If set to `true` the client will normalize line ending characters // in a workspace edit to the client-specified new line // character. // // @since 3.16.0 NormalizesLineEndings bool `json:"normalizesLineEndings,omitempty"` // Whether the client in general supports change annotations on text edits, // create file, rename file and delete file changes. // // @since 3.16.0 ChangeAnnotationSupport *ChangeAnnotationsSupportOptions `json:"changeAnnotationSupport,omitempty"` // Whether the client supports `WorkspaceEditMetadata` in `WorkspaceEdit`s. // // @since 3.18.0 // @proposed MetadataSupport bool `json:"metadataSupport,omitempty"` // Whether the client supports snippets as text edits. // // @since 3.18.0 // @proposed SnippetEditSupport bool `json:"snippetEditSupport,omitempty"` } // Additional data about a workspace edit. // // @since 3.18.0 // @proposed // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#workspaceEditMetadata type WorkspaceEditMetadata struct { // Signal to the editor that this edit is a refactoring. IsRefactoring bool `json:"isRefactoring,omitempty"` } // A workspace folder inside a client. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#workspaceFolder type WorkspaceFolder struct { // The associated URI for this workspace folder. URI URI `json:"uri"` // The name of the workspace folder. Used to refer to this // workspace folder in the user interface. Name string `json:"name"` } // The workspace folder change event. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#workspaceFoldersChangeEvent type WorkspaceFoldersChangeEvent struct { // The array of added workspace folders Added []WorkspaceFolder `json:"added"` // The array of the removed workspace folders Removed []WorkspaceFolder `json:"removed"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#workspaceFoldersInitializeParams type WorkspaceFoldersInitializeParams struct { // The workspace folders configured in the client when the server starts. // // This property is only available if the client supports workspace folders. // It can be `null` if the client supports workspace folders but none are // configured. // // @since 3.6.0 WorkspaceFolders []WorkspaceFolder `json:"workspaceFolders,omitempty"` } // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#workspaceFoldersServerCapabilities type WorkspaceFoldersServerCapabilities struct { // The server has support for workspace folders Supported bool `json:"supported,omitempty"` // Whether the server wants to receive workspace folder // change notifications. // // If a string is provided the string is treated as an ID // under which the notification is registered on the client // side. The ID can be used to unregister for these events // using the `client/unregisterCapability` request. ChangeNotifications *Or_WorkspaceFoldersServerCapabilities_changeNotifications `json:"changeNotifications,omitempty"` } // A full document diagnostic report for a workspace diagnostic result. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#workspaceFullDocumentDiagnosticReport type WorkspaceFullDocumentDiagnosticReport struct { // The URI for which diagnostic information is reported. URI DocumentUri `json:"uri"` // The version number for which the diagnostics are reported. // If the document is not marked as open `null` can be provided. Version int32 `json:"version"` FullDocumentDiagnosticReport } // Defines workspace specific capabilities of the server. // // @since 3.18.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#workspaceOptions type WorkspaceOptions struct { // The server supports workspace folder. // // @since 3.6.0 WorkspaceFolders *WorkspaceFoldersServerCapabilities `json:"workspaceFolders,omitempty"` // The server is interested in notifications/requests for operations on files. // // @since 3.16.0 FileOperations *FileOperationOptions `json:"fileOperations,omitempty"` // The server supports the `workspace/textDocumentContent` request. // // @since 3.18.0 // @proposed TextDocumentContent *Or_WorkspaceOptions_textDocumentContent `json:"textDocumentContent,omitempty"` } // A special workspace symbol that supports locations without a range. // // See also SymbolInformation. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#workspaceSymbol type WorkspaceSymbol struct { // The location of the symbol. Whether a server is allowed to // return a location without a range depends on the client // capability `workspace.symbol.resolveSupport`. // // See SymbolInformation#location for more details. Location Or_WorkspaceSymbol_location `json:"location"` // A data entry field that is preserved on a workspace symbol between a // workspace symbol request and a workspace symbol resolve request. Data interface{} `json:"data,omitempty"` BaseSymbolInformation } // Client capabilities for a {@link WorkspaceSymbolRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#workspaceSymbolClientCapabilities type WorkspaceSymbolClientCapabilities struct { // Symbol request supports dynamic registration. DynamicRegistration bool `json:"dynamicRegistration,omitempty"` // Specific capabilities for the `SymbolKind` in the `workspace/symbol` request. SymbolKind *ClientSymbolKindOptions `json:"symbolKind,omitempty"` // The client supports tags on `SymbolInformation`. // Clients supporting tags have to handle unknown tags gracefully. // // @since 3.16.0 TagSupport *ClientSymbolTagOptions `json:"tagSupport,omitempty"` // The client support partial workspace symbols. The client will send the // request `workspaceSymbol/resolve` to the server to resolve additional // properties. // // @since 3.17.0 ResolveSupport *ClientSymbolResolveOptions `json:"resolveSupport,omitempty"` } // Server capabilities for a {@link WorkspaceSymbolRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#workspaceSymbolOptions type WorkspaceSymbolOptions struct { // The server provides support to resolve additional // information for a workspace symbol. // // @since 3.17.0 ResolveProvider bool `json:"resolveProvider,omitempty"` WorkDoneProgressOptions } // The parameters of a {@link WorkspaceSymbolRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#workspaceSymbolParams type WorkspaceSymbolParams struct { // A query string to filter symbols by. Clients may send an empty // string here to request all symbols. // // The `query`-parameter should be interpreted in a *relaxed way* as editors // will apply their own highlighting and scoring on the results. A good rule // of thumb is to match case-insensitive and to simply check that the // characters of *query* appear in their order in a candidate symbol. // Servers shouldn't use prefix, substring, or similar strict matching. Query string `json:"query"` WorkDoneProgressParams PartialResultParams } // Registration options for a {@link WorkspaceSymbolRequest}. // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#workspaceSymbolRegistrationOptions type WorkspaceSymbolRegistrationOptions struct { WorkspaceSymbolOptions } // An unchanged document diagnostic report for a workspace diagnostic result. // // @since 3.17.0 // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#workspaceUnchangedDocumentDiagnosticReport type WorkspaceUnchangedDocumentDiagnosticReport struct { // The URI for which diagnostic information is reported. URI DocumentUri `json:"uri"` // The version number for which the diagnostics are reported. // If the document is not marked as open `null` can be provided. Version int32 `json:"version"` UnchangedDocumentDiagnosticReport } // The initialize parameters // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#_InitializeParams type XInitializeParams struct { // The process Id of the parent process that started // the server. // // Is `null` if the process has not been started by another process. // If the parent process is not alive then the server should exit. ProcessID int32 `json:"processId"` // Information about the client // // @since 3.15.0 ClientInfo *ClientInfo `json:"clientInfo,omitempty"` // The locale the client is currently showing the user interface // in. This must not necessarily be the locale of the operating // system. // // Uses IETF language tags as the value's syntax // (See https://en.wikipedia.org/wiki/IETF_language_tag) // // @since 3.16.0 Locale string `json:"locale,omitempty"` // The rootPath of the workspace. Is null // if no folder is open. // // @deprecated in favour of rootUri. RootPath string `json:"rootPath,omitempty"` // The rootUri of the workspace. Is null if no // folder is open. If both `rootPath` and `rootUri` are set // `rootUri` wins. // // @deprecated in favour of workspaceFolders. RootURI DocumentUri `json:"rootUri"` // The capabilities provided by the client (editor or tool) Capabilities ClientCapabilities `json:"capabilities"` // User provided initialization options. InitializationOptions interface{} `json:"initializationOptions,omitempty"` // The initial trace setting. If omitted trace is disabled ('off'). Trace *TraceValue `json:"trace,omitempty"` WorkDoneProgressParams } // The initialize parameters // // See https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification#_InitializeParams type _InitializeParams struct { // The process Id of the parent process that started // the server. // // Is `null` if the process has not been started by another process. // If the parent process is not alive then the server should exit. ProcessID int32 `json:"processId"` // Information about the client // // @since 3.15.0 ClientInfo *ClientInfo `json:"clientInfo,omitempty"` // The locale the client is currently showing the user interface // in. This must not necessarily be the locale of the operating // system. // // Uses IETF language tags as the value's syntax // (See https://en.wikipedia.org/wiki/IETF_language_tag) // // @since 3.16.0 Locale string `json:"locale,omitempty"` // The rootPath of the workspace. Is null // if no folder is open. // // @deprecated in favour of rootUri. RootPath string `json:"rootPath,omitempty"` // The rootUri of the workspace. Is null if no // folder is open. If both `rootPath` and `rootUri` are set // `rootUri` wins. // // @deprecated in favour of workspaceFolders. RootURI DocumentUri `json:"rootUri"` // The capabilities provided by the client (editor or tool) Capabilities ClientCapabilities `json:"capabilities"` // User provided initialization options. InitializationOptions interface{} `json:"initializationOptions,omitempty"` // The initial trace setting. If omitted trace is disabled ('off'). Trace *TraceValue `json:"trace,omitempty"` WorkDoneProgressParams } const ( // A set of predefined code action kinds // Empty kind. Empty CodeActionKind = "" // Base kind for quickfix actions: 'quickfix' QuickFix CodeActionKind = "quickfix" // Base kind for refactoring actions: 'refactor' Refactor CodeActionKind = "refactor" // Base kind for refactoring extraction actions: 'refactor.extract' // // Example extract actions: // // // - Extract method // - Extract function // - Extract variable // - Extract interface from class // - ... RefactorExtract CodeActionKind = "refactor.extract" // Base kind for refactoring inline actions: 'refactor.inline' // // Example inline actions: // // // - Inline function // - Inline variable // - Inline constant // - ... RefactorInline CodeActionKind = "refactor.inline" // Base kind for refactoring move actions: `refactor.move` // // Example move actions: // // // - Move a function to a new file // - Move a property between classes // - Move method to base class // - ... // // @since 3.18.0 // @proposed RefactorMove CodeActionKind = "refactor.move" // Base kind for refactoring rewrite actions: 'refactor.rewrite' // // Example rewrite actions: // // // - Convert JavaScript function to class // - Add or remove parameter // - Encapsulate field // - Make method static // - Move method to base class // - ... RefactorRewrite CodeActionKind = "refactor.rewrite" // Base kind for source actions: `source` // // Source code actions apply to the entire file. Source CodeActionKind = "source" // Base kind for an organize imports source action: `source.organizeImports` SourceOrganizeImports CodeActionKind = "source.organizeImports" // Base kind for auto-fix source actions: `source.fixAll`. // // Fix all actions automatically fix errors that have a clear fix that do not require user input. // They should not suppress errors or perform unsafe fixes such as generating new types or classes. // // @since 3.15.0 SourceFixAll CodeActionKind = "source.fixAll" // Base kind for all code actions applying to the entire notebook's scope. CodeActionKinds using // this should always begin with `notebook.` // // @since 3.18.0 Notebook CodeActionKind = "notebook" // The reason why code actions were requested. // // @since 3.17.0 // Code actions were explicitly requested by the user or by an extension. CodeActionInvoked CodeActionTriggerKind = 1 // Code actions were requested automatically. // // This typically happens when current selection in a file changes, but can // also be triggered when file content changes. CodeActionAutomatic CodeActionTriggerKind = 2 // The kind of a completion entry. TextCompletion CompletionItemKind = 1 MethodCompletion CompletionItemKind = 2 FunctionCompletion CompletionItemKind = 3 ConstructorCompletion CompletionItemKind = 4 FieldCompletion CompletionItemKind = 5 VariableCompletion CompletionItemKind = 6 ClassCompletion CompletionItemKind = 7 InterfaceCompletion CompletionItemKind = 8 ModuleCompletion CompletionItemKind = 9 PropertyCompletion CompletionItemKind = 10 UnitCompletion CompletionItemKind = 11 ValueCompletion CompletionItemKind = 12 EnumCompletion CompletionItemKind = 13 KeywordCompletion CompletionItemKind = 14 SnippetCompletion CompletionItemKind = 15 ColorCompletion CompletionItemKind = 16 FileCompletion CompletionItemKind = 17 ReferenceCompletion CompletionItemKind = 18 FolderCompletion CompletionItemKind = 19 EnumMemberCompletion CompletionItemKind = 20 ConstantCompletion CompletionItemKind = 21 StructCompletion CompletionItemKind = 22 EventCompletion CompletionItemKind = 23 OperatorCompletion CompletionItemKind = 24 TypeParameterCompletion CompletionItemKind = 25 // Completion item tags are extra annotations that tweak the rendering of a completion // item. // // @since 3.15.0 // Render a completion as obsolete, usually using a strike-out. ComplDeprecated CompletionItemTag = 1 // How a completion was triggered // Completion was triggered by typing an identifier (24x7 code // complete), manual invocation (e.g Ctrl+Space) or via API. Invoked CompletionTriggerKind = 1 // Completion was triggered by a trigger character specified by // the `triggerCharacters` properties of the `CompletionRegistrationOptions`. TriggerCharacter CompletionTriggerKind = 2 // Completion was re-triggered as current completion list is incomplete TriggerForIncompleteCompletions CompletionTriggerKind = 3 // The diagnostic's severity. // Reports an error. SeverityError DiagnosticSeverity = 1 // Reports a warning. SeverityWarning DiagnosticSeverity = 2 // Reports an information. SeverityInformation DiagnosticSeverity = 3 // Reports a hint. SeverityHint DiagnosticSeverity = 4 // The diagnostic tags. // // @since 3.15.0 // Unused or unnecessary code. // // Clients are allowed to render diagnostics with this tag faded out instead of having // an error squiggle. Unnecessary DiagnosticTag = 1 // Deprecated or obsolete code. // // Clients are allowed to rendered diagnostics with this tag strike through. Deprecated DiagnosticTag = 2 // The document diagnostic report kinds. // // @since 3.17.0 // A diagnostic report with a full // set of problems. DiagnosticFull DocumentDiagnosticReportKind = "full" // A report indicating that the last // returned report is still accurate. DiagnosticUnchanged DocumentDiagnosticReportKind = "unchanged" // A document highlight kind. // A textual occurrence. Text DocumentHighlightKind = 1 // Read-access of a symbol, like reading a variable. Read DocumentHighlightKind = 2 // Write-access of a symbol, like writing to a variable. Write DocumentHighlightKind = 3 // Predefined error codes. ParseError ErrorCodes = -32700 InvalidRequest ErrorCodes = -32600 MethodNotFound ErrorCodes = -32601 InvalidParams ErrorCodes = -32602 InternalError ErrorCodes = -32603 // Error code indicating that a server received a notification or // request before the server has received the `initialize` request. ServerNotInitialized ErrorCodes = -32002 UnknownErrorCode ErrorCodes = -32001 // Applying the workspace change is simply aborted if one of the changes provided // fails. All operations executed before the failing operation stay executed. Abort FailureHandlingKind = "abort" // All operations are executed transactional. That means they either all // succeed or no changes at all are applied to the workspace. Transactional FailureHandlingKind = "transactional" // If the workspace edit contains only textual file changes they are executed transactional. // If resource changes (create, rename or delete file) are part of the change the failure // handling strategy is abort. TextOnlyTransactional FailureHandlingKind = "textOnlyTransactional" // The client tries to undo the operations already executed. But there is no // guarantee that this is succeeding. Undo FailureHandlingKind = "undo" // The file event type // The file got created. Created FileChangeType = 1 // The file got changed. Changed FileChangeType = 2 // The file got deleted. Deleted FileChangeType = 3 // A pattern kind describing if a glob pattern matches a file a folder or // both. // // @since 3.16.0 // The pattern matches a file only. FilePattern FileOperationPatternKind = "file" // The pattern matches a folder only. FolderPattern FileOperationPatternKind = "folder" // A set of predefined range kinds. // Folding range for a comment Comment FoldingRangeKind = "comment" // Folding range for an import or include Imports FoldingRangeKind = "imports" // Folding range for a region (e.g. `#region`) Region FoldingRangeKind = "region" // Inlay hint kinds. // // @since 3.17.0 // An inlay hint that for a type annotation. Type InlayHintKind = 1 // An inlay hint that is for a parameter. Parameter InlayHintKind = 2 // Describes how an {@link InlineCompletionItemProvider inline completion provider} was triggered. // // @since 3.18.0 // @proposed // Completion was triggered explicitly by a user gesture. InlineInvoked InlineCompletionTriggerKind = 1 // Completion was triggered automatically while editing. InlineAutomatic InlineCompletionTriggerKind = 2 // Defines whether the insert text in a completion item should be interpreted as // plain text or a snippet. // The primary text to be inserted is treated as a plain string. PlainTextTextFormat InsertTextFormat = 1 // The primary text to be inserted is treated as a snippet. // // A snippet can define tab stops and placeholders with `$1`, `$2` // and `${3:foo}`. `$0` defines the final tab stop, it defaults to // the end of the snippet. Placeholders with equal identifiers are linked, // that is typing in one will update others too. // // See also: https://microsoft.github.io/language-server-protocol/specifications/specification-current/#snippet_syntax SnippetTextFormat InsertTextFormat = 2 // How whitespace and indentation is handled during completion // item insertion. // // @since 3.16.0 // The insertion or replace strings is taken as it is. If the // value is multi line the lines below the cursor will be // inserted using the indentation defined in the string value. // The client will not apply any kind of adjustments to the // string. AsIs InsertTextMode = 1 // The editor adjusts leading whitespace of new lines so that // they match the indentation up to the cursor of the line for // which the item is accepted. // // Consider a line like this: <2tabs><3tabs>foo. Accepting a // multi line completion item is indented using 2 tabs and all // following lines inserted will be indented using 2 tabs as well. AdjustIndentation InsertTextMode = 2 // A request failed but it was syntactically correct, e.g the // method name was known and the parameters were valid. The error // message should contain human readable information about why // the request failed. // // @since 3.17.0 RequestFailed LSPErrorCodes = -32803 // The server cancelled the request. This error code should // only be used for requests that explicitly support being // server cancellable. // // @since 3.17.0 ServerCancelled LSPErrorCodes = -32802 // The server detected that the content of a document got // modified outside normal conditions. A server should // NOT send this error code if it detects a content change // in it unprocessed messages. The result even computed // on an older state might still be useful for the client. // // If a client decides that a result is not of any use anymore // the client should cancel the request. ContentModified LSPErrorCodes = -32801 // The client has canceled a request and a server has detected // the cancel. RequestCancelled LSPErrorCodes = -32800 // Predefined Language kinds // @since 3.18.0 // @proposed LangABAP LanguageKind = "abap" LangWindowsBat LanguageKind = "bat" LangBibTeX LanguageKind = "bibtex" LangClojure LanguageKind = "clojure" LangCoffeescript LanguageKind = "coffeescript" LangC LanguageKind = "c" LangCPP LanguageKind = "cpp" LangCSharp LanguageKind = "csharp" LangCSS LanguageKind = "css" // @since 3.18.0 // @proposed LangD LanguageKind = "d" // @since 3.18.0 // @proposed LangDelphi LanguageKind = "pascal" LangDiff LanguageKind = "diff" LangDart LanguageKind = "dart" LangDockerfile LanguageKind = "dockerfile" LangElixir LanguageKind = "elixir" LangErlang LanguageKind = "erlang" LangFSharp LanguageKind = "fsharp" LangGitCommit LanguageKind = "git-commit" LangGitRebase LanguageKind = "rebase" LangGo LanguageKind = "go" LangGroovy LanguageKind = "groovy" LangHandlebars LanguageKind = "handlebars" LangHaskell LanguageKind = "haskell" LangHTML LanguageKind = "html" LangIni LanguageKind = "ini" LangJava LanguageKind = "java" LangJavaScript LanguageKind = "javascript" LangJavaScriptReact LanguageKind = "javascriptreact" LangJSON LanguageKind = "json" LangLaTeX LanguageKind = "latex" LangLess LanguageKind = "less" LangLua LanguageKind = "lua" LangMakefile LanguageKind = "makefile" LangMarkdown LanguageKind = "markdown" LangObjectiveC LanguageKind = "objective-c" LangObjectiveCPP LanguageKind = "objective-cpp" // @since 3.18.0 // @proposed LangPascal LanguageKind = "pascal" LangPerl LanguageKind = "perl" LangPerl6 LanguageKind = "perl6" LangPHP LanguageKind = "php" LangPowershell LanguageKind = "powershell" LangPug LanguageKind = "jade" LangPython LanguageKind = "python" LangR LanguageKind = "r" LangRazor LanguageKind = "razor" LangRuby LanguageKind = "ruby" LangRust LanguageKind = "rust" LangSCSS LanguageKind = "scss" LangSASS LanguageKind = "sass" LangScala LanguageKind = "scala" LangShaderLab LanguageKind = "shaderlab" LangShellScript LanguageKind = "shellscript" LangSQL LanguageKind = "sql" LangSwift LanguageKind = "swift" LangTypeScript LanguageKind = "typescript" LangTypeScriptReact LanguageKind = "typescriptreact" LangTeX LanguageKind = "tex" LangVisualBasic LanguageKind = "vb" LangXML LanguageKind = "xml" LangXSL LanguageKind = "xsl" LangYAML LanguageKind = "yaml" // Describes the content type that a client supports in various // result literals like `Hover`, `ParameterInfo` or `CompletionItem`. // // Please note that `MarkupKinds` must not start with a `$`. This kinds // are reserved for internal usage. // Plain text is supported as a content format PlainText MarkupKind = "plaintext" // Markdown is supported as a content format Markdown MarkupKind = "markdown" // The message type // An error message. Error MessageType = 1 // A warning message. Warning MessageType = 2 // An information message. Info MessageType = 3 // A log message. Log MessageType = 4 // A debug message. // // @since 3.18.0 // @proposed Debug MessageType = 5 // The moniker kind. // // @since 3.16.0 // The moniker represent a symbol that is imported into a project Import MonikerKind = "import" // The moniker represents a symbol that is exported from a project Export MonikerKind = "export" // The moniker represents a symbol that is local to a project (e.g. a local // variable of a function, a class not visible outside the project, ...) Local MonikerKind = "local" // A notebook cell kind. // // @since 3.17.0 // A markup-cell is formatted source that is used for display. Markup NotebookCellKind = 1 // A code-cell is source code. Code NotebookCellKind = 2 // A set of predefined position encoding kinds. // // @since 3.17.0 // Character offsets count UTF-8 code units (e.g. bytes). UTF8 PositionEncodingKind = "utf-8" // Character offsets count UTF-16 code units. // // This is the default and must always be supported // by servers UTF16 PositionEncodingKind = "utf-16" // Character offsets count UTF-32 code units. // // Implementation note: these are the same as Unicode codepoints, // so this `PositionEncodingKind` may also be used for an // encoding-agnostic representation of character offsets. UTF32 PositionEncodingKind = "utf-32" // The client's default behavior is to select the identifier // according the to language's syntax rule. Identifier PrepareSupportDefaultBehavior = 1 // Supports creating new files and folders. Create ResourceOperationKind = "create" // Supports renaming existing files and folders. Rename ResourceOperationKind = "rename" // Supports deleting existing files and folders. Delete ResourceOperationKind = "delete" // A set of predefined token modifiers. This set is not fixed // an clients can specify additional token types via the // corresponding client capabilities. // // @since 3.16.0 ModDeclaration SemanticTokenModifiers = "declaration" ModDefinition SemanticTokenModifiers = "definition" ModReadonly SemanticTokenModifiers = "readonly" ModStatic SemanticTokenModifiers = "static" ModDeprecated SemanticTokenModifiers = "deprecated" ModAbstract SemanticTokenModifiers = "abstract" ModAsync SemanticTokenModifiers = "async" ModModification SemanticTokenModifiers = "modification" ModDocumentation SemanticTokenModifiers = "documentation" ModDefaultLibrary SemanticTokenModifiers = "defaultLibrary" // A set of predefined token types. This set is not fixed // an clients can specify additional token types via the // corresponding client capabilities. // // @since 3.16.0 NamespaceType SemanticTokenTypes = "namespace" // Represents a generic type. Acts as a fallback for types which can't be mapped to // a specific type like class or enum. TypeType SemanticTokenTypes = "type" ClassType SemanticTokenTypes = "class" EnumType SemanticTokenTypes = "enum" InterfaceType SemanticTokenTypes = "interface" StructType SemanticTokenTypes = "struct" TypeParameterType SemanticTokenTypes = "typeParameter" ParameterType SemanticTokenTypes = "parameter" VariableType SemanticTokenTypes = "variable" PropertyType SemanticTokenTypes = "property" EnumMemberType SemanticTokenTypes = "enumMember" EventType SemanticTokenTypes = "event" FunctionType SemanticTokenTypes = "function" MethodType SemanticTokenTypes = "method" MacroType SemanticTokenTypes = "macro" KeywordType SemanticTokenTypes = "keyword" ModifierType SemanticTokenTypes = "modifier" CommentType SemanticTokenTypes = "comment" StringType SemanticTokenTypes = "string" NumberType SemanticTokenTypes = "number" RegexpType SemanticTokenTypes = "regexp" OperatorType SemanticTokenTypes = "operator" // @since 3.17.0 DecoratorType SemanticTokenTypes = "decorator" // @since 3.18.0 LabelType SemanticTokenTypes = "label" // How a signature help was triggered. // // @since 3.15.0 // Signature help was invoked manually by the user or by a command. SigInvoked SignatureHelpTriggerKind = 1 // Signature help was triggered by a trigger character. SigTriggerCharacter SignatureHelpTriggerKind = 2 // Signature help was triggered by the cursor moving or by the document content changing. SigContentChange SignatureHelpTriggerKind = 3 // A symbol kind. File SymbolKind = 1 Module SymbolKind = 2 Namespace SymbolKind = 3 Package SymbolKind = 4 Class SymbolKind = 5 Method SymbolKind = 6 Property SymbolKind = 7 Field SymbolKind = 8 Constructor SymbolKind = 9 Enum SymbolKind = 10 Interface SymbolKind = 11 Function SymbolKind = 12 Variable SymbolKind = 13 Constant SymbolKind = 14 String SymbolKind = 15 Number SymbolKind = 16 Boolean SymbolKind = 17 Array SymbolKind = 18 Object SymbolKind = 19 Key SymbolKind = 20 Null SymbolKind = 21 EnumMember SymbolKind = 22 Struct SymbolKind = 23 Event SymbolKind = 24 Operator SymbolKind = 25 TypeParameter SymbolKind = 26 // Symbol tags are extra annotations that tweak the rendering of a symbol. // // @since 3.16 // Render a symbol as obsolete, usually using a strike-out. DeprecatedSymbol SymbolTag = 1 // Represents reasons why a text document is saved. // Manually triggered, e.g. by the user pressing save, by starting debugging, // or by an API call. Manual TextDocumentSaveReason = 1 // Automatic after a delay. AfterDelay TextDocumentSaveReason = 2 // When the editor lost focus. FocusOut TextDocumentSaveReason = 3 // Defines how the host (editor) should sync // document changes to the language server. // Documents should not be synced at all. None TextDocumentSyncKind = 0 // Documents are synced by always sending the full content // of the document. Full TextDocumentSyncKind = 1 // Documents are synced by sending the full content on open. // After that only incremental updates to the document are // send. Incremental TextDocumentSyncKind = 2 Relative TokenFormat = "relative" // Turn tracing off. Off TraceValue = "off" // Trace messages only. Messages TraceValue = "messages" // Verbose message tracing. Verbose TraceValue = "verbose" // Moniker uniqueness level to define scope of the moniker. // // @since 3.16.0 // The moniker is only unique inside a document Document UniquenessLevel = "document" // The moniker is unique inside a project for which a dump got created Project UniquenessLevel = "project" // The moniker is unique inside the group to which a project belongs Group UniquenessLevel = "group" // The moniker is unique inside the moniker scheme. Scheme UniquenessLevel = "scheme" // The moniker is globally unique Global UniquenessLevel = "global" // Interested in create events. WatchCreate WatchKind = 1 // Interested in change events WatchChange WatchKind = 2 // Interested in delete events WatchDelete WatchKind = 4 ) ================================================ FILE: internal/lsp/protocol/uri.go ================================================ // Copyright 2023 The Go Authors. All rights reserved. // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file. package protocol // This file declares URI, DocumentUri, and its methods. // // For the LSP definition of these types, see // https://microsoft.github.io/language-server-protocol/specifications/lsp/3.17/specification/#uri import ( "fmt" "net/url" "path/filepath" "strings" "unicode" ) // A DocumentUri is the URI of a client editor document. // // According to the LSP specification: // // Care should be taken to handle encoding in URIs. For // example, some clients (such as VS Code) may encode colons // in drive letters while others do not. The URIs below are // both valid, but clients and servers should be consistent // with the form they use themselves to ensure the other party // doesn’t interpret them as distinct URIs. Clients and // servers should not assume that each other are encoding the // same way (for example a client encoding colons in drive // letters cannot assume server responses will have encoded // colons). The same applies to casing of drive letters - one // party should not assume the other party will return paths // with drive letters cased the same as it. // // file:///c:/project/readme.md // file:///C%3A/project/readme.md // // This is done during JSON unmarshalling; // see [DocumentUri.UnmarshalText] for details. type DocumentUri string // A URI is an arbitrary URL (e.g. https), not necessarily a file. type URI = string // UnmarshalText implements decoding of DocumentUri values. // // In particular, it implements a systematic correction of various odd // features of the definition of DocumentUri in the LSP spec that // appear to be workarounds for bugs in VS Code. For example, it may // URI-encode the URI itself, so that colon becomes %3A, and it may // send file://foo.go URIs that have two slashes (not three) and no // hostname. // // We use UnmarshalText, not UnmarshalJSON, because it is called even // for non-addressable values such as keys and values of map[K]V, // where there is no pointer of type *K or *V on which to call // UnmarshalJSON. (See Go issue #28189 for more detail.) // // Non-empty DocumentUris are valid "file"-scheme URIs. // The empty DocumentUri is valid. func (uri *DocumentUri) UnmarshalText(data []byte) (err error) { *uri, err = ParseDocumentUri(string(data)) return } // Path returns the file path for the given URI. // // DocumentUri("").Path() returns the empty string. // // Path panics if called on a URI that is not a valid filename. func (uri DocumentUri) Path() string { filename, err := filename(uri) if err != nil { // e.g. ParseRequestURI failed. // // This can only affect DocumentUris created by // direct string manipulation; all DocumentUris // received from the client pass through // ParseRequestURI, which ensures validity. panic(err) } return filepath.FromSlash(filename) } // Dir returns the URI for the directory containing the receiver. func (uri DocumentUri) Dir() DocumentUri { // This function could be more efficiently implemented by avoiding any call // to Path(), but at least consolidates URI manipulation. return URIFromPath(uri.DirPath()) } // DirPath returns the file path to the directory containing this URI, which // must be a file URI. func (uri DocumentUri) DirPath() string { return filepath.Dir(uri.Path()) } func filename(uri DocumentUri) (string, error) { if uri == "" { return "", nil } // This conservative check for the common case // of a simple non-empty absolute POSIX filename // avoids the allocation of a net.URL. if strings.HasPrefix(string(uri), "file:///") { rest := string(uri)[len("file://"):] // leave one slash for i := range len(rest) { b := rest[i] // Reject these cases: if b < ' ' || b == 0x7f || // control character b == '%' || b == '+' || // URI escape b == ':' || // Windows drive letter b == '@' || b == '&' || b == '?' { // authority or query goto slow } } return rest, nil } slow: u, err := url.ParseRequestURI(string(uri)) if err != nil { return "", err } if u.Scheme != fileScheme { return "", fmt.Errorf("only file URIs are supported, got %q from %q", u.Scheme, uri) } // If the URI is a Windows URI, we trim the leading "/" and uppercase // the drive letter, which will never be case sensitive. if isWindowsDriveURIPath(u.Path) { u.Path = strings.ToUpper(string(u.Path[1])) + u.Path[2:] } return u.Path, nil } // ParseDocumentUri interprets a string as a DocumentUri, applying VS // Code workarounds; see [DocumentUri.UnmarshalText] for details. func ParseDocumentUri(s string) (DocumentUri, error) { if s == "" { return "", nil } if !strings.HasPrefix(s, "file://") { return "", fmt.Errorf("DocumentUri scheme is not 'file': %s", s) } // VS Code sends URLs with only two slashes, // which are invalid. golang/go#39789. if !strings.HasPrefix(s, "file:///") { s = "file:///" + s[len("file://"):] } // Even though the input is a URI, it may not be in canonical form. VS Code // in particular over-escapes :, @, etc. Unescape and re-encode to canonicalize. path, err := url.PathUnescape(s[len("file://"):]) if err != nil { return "", err } // File URIs from Windows may have lowercase drive letters. // Since drive letters are guaranteed to be case insensitive, // we change them to uppercase to remain consistent. // For example, file:///c:/x/y/z becomes file:///C:/x/y/z. if isWindowsDriveURIPath(path) { path = path[:1] + strings.ToUpper(string(path[1])) + path[2:] } u := url.URL{Scheme: fileScheme, Path: path} return DocumentUri(u.String()), nil } // URIFromPath returns DocumentUri for the supplied file path. // Given "", it returns "". func URIFromPath(path string) DocumentUri { if path == "" { return "" } if !isWindowsDrivePath(path) { if abs, err := filepath.Abs(path); err == nil { path = abs } } // Check the file path again, in case it became absolute. if isWindowsDrivePath(path) { path = "/" + strings.ToUpper(string(path[0])) + path[1:] } path = filepath.ToSlash(path) u := url.URL{ Scheme: fileScheme, Path: path, } return DocumentUri(u.String()) } const fileScheme = "file" // isWindowsDrivePath returns true if the file path is of the form used by // Windows. We check if the path begins with a drive letter, followed by a ":". // For example: C:/x/y/z. func isWindowsDrivePath(path string) bool { if len(path) < 3 { return false } return unicode.IsLetter(rune(path[0])) && path[1] == ':' } // isWindowsDriveURIPath returns true if the file URI is of the format used by // Windows URIs. The url.Parse package does not specially handle Windows paths // (see golang/go#6027), so we check if the URI path has a drive prefix (e.g. "/C:"). func isWindowsDriveURIPath(uri string) bool { if len(uri) < 4 { return false } return uri[0] == '/' && unicode.IsLetter(rune(uri[1])) && uri[2] == ':' } ================================================ FILE: internal/lsp/protocol.go ================================================ package lsp import ( "encoding/json" ) // Message represents a JSON-RPC 2.0 message type Message struct { JSONRPC string `json:"jsonrpc"` ID int32 `json:"id,omitempty"` Method string `json:"method,omitempty"` Params json.RawMessage `json:"params,omitempty"` Result json.RawMessage `json:"result,omitempty"` Error *ResponseError `json:"error,omitempty"` } // ResponseError represents a JSON-RPC 2.0 error type ResponseError struct { Code int `json:"code"` Message string `json:"message"` } func NewRequest(id int32, method string, params any) (*Message, error) { paramsJSON, err := json.Marshal(params) if err != nil { return nil, err } return &Message{ JSONRPC: "2.0", ID: id, Method: method, Params: paramsJSON, }, nil } func NewNotification(method string, params any) (*Message, error) { paramsJSON, err := json.Marshal(params) if err != nil { return nil, err } return &Message{ JSONRPC: "2.0", Method: method, Params: paramsJSON, }, nil } ================================================ FILE: internal/lsp/transport.go ================================================ package lsp import ( "bufio" "context" "encoding/json" "fmt" "io" "strings" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/logging" ) // Write writes an LSP message to the given writer func WriteMessage(w io.Writer, msg *Message) error { data, err := json.Marshal(msg) if err != nil { return fmt.Errorf("failed to marshal message: %w", err) } cnf := config.Get() if cnf.DebugLSP { logging.Debug("Sending message to server", "method", msg.Method, "id", msg.ID) } _, err = fmt.Fprintf(w, "Content-Length: %d\r\n\r\n", len(data)) if err != nil { return fmt.Errorf("failed to write header: %w", err) } _, err = w.Write(data) if err != nil { return fmt.Errorf("failed to write message: %w", err) } return nil } // ReadMessage reads a single LSP message from the given reader func ReadMessage(r *bufio.Reader) (*Message, error) { cnf := config.Get() // Read headers var contentLength int for { line, err := r.ReadString('\n') if err != nil { return nil, fmt.Errorf("failed to read header: %w", err) } line = strings.TrimSpace(line) if cnf.DebugLSP { logging.Debug("Received header", "line", line) } if line == "" { break // End of headers } if strings.HasPrefix(line, "Content-Length: ") { _, err := fmt.Sscanf(line, "Content-Length: %d", &contentLength) if err != nil { return nil, fmt.Errorf("invalid Content-Length: %w", err) } } } if cnf.DebugLSP { logging.Debug("Content-Length", "length", contentLength) } // Read content content := make([]byte, contentLength) _, err := io.ReadFull(r, content) if err != nil { return nil, fmt.Errorf("failed to read content: %w", err) } if cnf.DebugLSP { logging.Debug("Received content", "content", string(content)) } // Parse message var msg Message if err := json.Unmarshal(content, &msg); err != nil { return nil, fmt.Errorf("failed to unmarshal message: %w", err) } return &msg, nil } // handleMessages reads and dispatches messages in a loop func (c *Client) handleMessages() { cnf := config.Get() for { msg, err := ReadMessage(c.stdout) if err != nil { if cnf.DebugLSP { logging.Error("Error reading message", "error", err) } return } // Handle server->client request (has both Method and ID) if msg.Method != "" && msg.ID != 0 { if cnf.DebugLSP { logging.Debug("Received request from server", "method", msg.Method, "id", msg.ID) } response := &Message{ JSONRPC: "2.0", ID: msg.ID, } // Look up handler for this method c.serverHandlersMu.RLock() handler, ok := c.serverRequestHandlers[msg.Method] c.serverHandlersMu.RUnlock() if ok { result, err := handler(msg.Params) if err != nil { response.Error = &ResponseError{ Code: -32603, Message: err.Error(), } } else { rawJSON, err := json.Marshal(result) if err != nil { response.Error = &ResponseError{ Code: -32603, Message: fmt.Sprintf("failed to marshal response: %v", err), } } else { response.Result = rawJSON } } } else { response.Error = &ResponseError{ Code: -32601, Message: fmt.Sprintf("method not found: %s", msg.Method), } } // Send response back to server if err := WriteMessage(c.stdin, response); err != nil { logging.Error("Error sending response to server", "error", err) } continue } // Handle notification (has Method but no ID) if msg.Method != "" && msg.ID == 0 { c.notificationMu.RLock() handler, ok := c.notificationHandlers[msg.Method] c.notificationMu.RUnlock() if ok { if cnf.DebugLSP { logging.Debug("Handling notification", "method", msg.Method) } go handler(msg.Params) } else if cnf.DebugLSP { logging.Debug("No handler for notification", "method", msg.Method) } continue } // Handle response to our request (has ID but no Method) if msg.ID != 0 && msg.Method == "" { c.handlersMu.RLock() ch, ok := c.handlers[msg.ID] c.handlersMu.RUnlock() if ok { if cnf.DebugLSP { logging.Debug("Received response for request", "id", msg.ID) } ch <- msg close(ch) } else if cnf.DebugLSP { logging.Debug("No handler for response", "id", msg.ID) } } } } // Call makes a request and waits for the response func (c *Client) Call(ctx context.Context, method string, params any, result any) error { cnf := config.Get() id := c.nextID.Add(1) if cnf.DebugLSP { logging.Debug("Making call", "method", method, "id", id) } msg, err := NewRequest(id, method, params) if err != nil { return fmt.Errorf("failed to create request: %w", err) } // Create response channel ch := make(chan *Message, 1) c.handlersMu.Lock() c.handlers[id] = ch c.handlersMu.Unlock() defer func() { c.handlersMu.Lock() delete(c.handlers, id) c.handlersMu.Unlock() }() // Send request if err := WriteMessage(c.stdin, msg); err != nil { return fmt.Errorf("failed to send request: %w", err) } if cnf.DebugLSP { logging.Debug("Request sent", "method", method, "id", id) } // Wait for response resp := <-ch if cnf.DebugLSP { logging.Debug("Received response", "id", id) } if resp.Error != nil { return fmt.Errorf("request failed: %s (code: %d)", resp.Error.Message, resp.Error.Code) } if result != nil { // If result is a json.RawMessage, just copy the raw bytes if rawMsg, ok := result.(*json.RawMessage); ok { *rawMsg = resp.Result return nil } // Otherwise unmarshal into the provided type if err := json.Unmarshal(resp.Result, result); err != nil { return fmt.Errorf("failed to unmarshal result: %w", err) } } return nil } // Notify sends a notification (a request without an ID that doesn't expect a response) func (c *Client) Notify(ctx context.Context, method string, params any) error { cnf := config.Get() if cnf.DebugLSP { logging.Debug("Sending notification", "method", method) } msg, err := NewNotification(method, params) if err != nil { return fmt.Errorf("failed to create notification: %w", err) } if err := WriteMessage(c.stdin, msg); err != nil { return fmt.Errorf("failed to send notification: %w", err) } return nil } type ( NotificationHandler func(params json.RawMessage) ServerRequestHandler func(params json.RawMessage) (any, error) ) ================================================ FILE: internal/lsp/util/edit.go ================================================ package util import ( "bytes" "fmt" "os" "sort" "strings" "github.com/opencode-ai/opencode/internal/lsp/protocol" ) func applyTextEdits(uri protocol.DocumentUri, edits []protocol.TextEdit) error { path := strings.TrimPrefix(string(uri), "file://") // Read the file content content, err := os.ReadFile(path) if err != nil { return fmt.Errorf("failed to read file: %w", err) } // Detect line ending style var lineEnding string if bytes.Contains(content, []byte("\r\n")) { lineEnding = "\r\n" } else { lineEnding = "\n" } // Track if file ends with a newline endsWithNewline := len(content) > 0 && bytes.HasSuffix(content, []byte(lineEnding)) // Split into lines without the endings lines := strings.Split(string(content), lineEnding) // Check for overlapping edits for i, edit1 := range edits { for j := i + 1; j < len(edits); j++ { if rangesOverlap(edit1.Range, edits[j].Range) { return fmt.Errorf("overlapping edits detected between edit %d and %d", i, j) } } } // Sort edits in reverse order sortedEdits := make([]protocol.TextEdit, len(edits)) copy(sortedEdits, edits) sort.Slice(sortedEdits, func(i, j int) bool { if sortedEdits[i].Range.Start.Line != sortedEdits[j].Range.Start.Line { return sortedEdits[i].Range.Start.Line > sortedEdits[j].Range.Start.Line } return sortedEdits[i].Range.Start.Character > sortedEdits[j].Range.Start.Character }) // Apply each edit for _, edit := range sortedEdits { newLines, err := applyTextEdit(lines, edit) if err != nil { return fmt.Errorf("failed to apply edit: %w", err) } lines = newLines } // Join lines with proper line endings var newContent strings.Builder for i, line := range lines { if i > 0 { newContent.WriteString(lineEnding) } newContent.WriteString(line) } // Only add a newline if the original file had one and we haven't already added it if endsWithNewline && !strings.HasSuffix(newContent.String(), lineEnding) { newContent.WriteString(lineEnding) } if err := os.WriteFile(path, []byte(newContent.String()), 0o644); err != nil { return fmt.Errorf("failed to write file: %w", err) } return nil } func applyTextEdit(lines []string, edit protocol.TextEdit) ([]string, error) { startLine := int(edit.Range.Start.Line) endLine := int(edit.Range.End.Line) startChar := int(edit.Range.Start.Character) endChar := int(edit.Range.End.Character) // Validate positions if startLine < 0 || startLine >= len(lines) { return nil, fmt.Errorf("invalid start line: %d", startLine) } if endLine < 0 || endLine >= len(lines) { endLine = len(lines) - 1 } // Create result slice with initial capacity result := make([]string, 0, len(lines)) // Copy lines before edit result = append(result, lines[:startLine]...) // Get the prefix of the start line startLineContent := lines[startLine] if startChar < 0 || startChar > len(startLineContent) { startChar = len(startLineContent) } prefix := startLineContent[:startChar] // Get the suffix of the end line endLineContent := lines[endLine] if endChar < 0 || endChar > len(endLineContent) { endChar = len(endLineContent) } suffix := endLineContent[endChar:] // Handle the edit if edit.NewText == "" { if prefix+suffix != "" { result = append(result, prefix+suffix) } } else { // Split new text into lines, being careful not to add extra newlines // newLines := strings.Split(strings.TrimRight(edit.NewText, "\n"), "\n") newLines := strings.Split(edit.NewText, "\n") if len(newLines) == 1 { // Single line change result = append(result, prefix+newLines[0]+suffix) } else { // Multi-line change result = append(result, prefix+newLines[0]) result = append(result, newLines[1:len(newLines)-1]...) result = append(result, newLines[len(newLines)-1]+suffix) } } // Add remaining lines if endLine+1 < len(lines) { result = append(result, lines[endLine+1:]...) } return result, nil } // applyDocumentChange applies a DocumentChange (create/rename/delete operations) func applyDocumentChange(change protocol.DocumentChange) error { if change.CreateFile != nil { path := strings.TrimPrefix(string(change.CreateFile.URI), "file://") if change.CreateFile.Options != nil { if change.CreateFile.Options.Overwrite { // Proceed with overwrite } else if change.CreateFile.Options.IgnoreIfExists { if _, err := os.Stat(path); err == nil { return nil // File exists and we're ignoring it } } } if err := os.WriteFile(path, []byte(""), 0o644); err != nil { return fmt.Errorf("failed to create file: %w", err) } } if change.DeleteFile != nil { path := strings.TrimPrefix(string(change.DeleteFile.URI), "file://") if change.DeleteFile.Options != nil && change.DeleteFile.Options.Recursive { if err := os.RemoveAll(path); err != nil { return fmt.Errorf("failed to delete directory recursively: %w", err) } } else { if err := os.Remove(path); err != nil { return fmt.Errorf("failed to delete file: %w", err) } } } if change.RenameFile != nil { oldPath := strings.TrimPrefix(string(change.RenameFile.OldURI), "file://") newPath := strings.TrimPrefix(string(change.RenameFile.NewURI), "file://") if change.RenameFile.Options != nil { if !change.RenameFile.Options.Overwrite { if _, err := os.Stat(newPath); err == nil { return fmt.Errorf("target file already exists and overwrite is not allowed: %s", newPath) } } } if err := os.Rename(oldPath, newPath); err != nil { return fmt.Errorf("failed to rename file: %w", err) } } if change.TextDocumentEdit != nil { textEdits := make([]protocol.TextEdit, len(change.TextDocumentEdit.Edits)) for i, edit := range change.TextDocumentEdit.Edits { var err error textEdits[i], err = edit.AsTextEdit() if err != nil { return fmt.Errorf("invalid edit type: %w", err) } } return applyTextEdits(change.TextDocumentEdit.TextDocument.URI, textEdits) } return nil } // ApplyWorkspaceEdit applies the given WorkspaceEdit to the filesystem func ApplyWorkspaceEdit(edit protocol.WorkspaceEdit) error { // Handle Changes field for uri, textEdits := range edit.Changes { if err := applyTextEdits(uri, textEdits); err != nil { return fmt.Errorf("failed to apply text edits: %w", err) } } // Handle DocumentChanges field for _, change := range edit.DocumentChanges { if err := applyDocumentChange(change); err != nil { return fmt.Errorf("failed to apply document change: %w", err) } } return nil } func rangesOverlap(r1, r2 protocol.Range) bool { if r1.Start.Line > r2.End.Line || r2.Start.Line > r1.End.Line { return false } if r1.Start.Line == r2.End.Line && r1.Start.Character > r2.End.Character { return false } if r2.Start.Line == r1.End.Line && r2.Start.Character > r1.End.Character { return false } return true } ================================================ FILE: internal/lsp/watcher/watcher.go ================================================ package watcher import ( "context" "fmt" "os" "path/filepath" "strings" "sync" "time" "github.com/bmatcuk/doublestar/v4" "github.com/fsnotify/fsnotify" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/logging" "github.com/opencode-ai/opencode/internal/lsp" "github.com/opencode-ai/opencode/internal/lsp/protocol" ) // WorkspaceWatcher manages LSP file watching type WorkspaceWatcher struct { client *lsp.Client workspacePath string debounceTime time.Duration debounceMap map[string]*time.Timer debounceMu sync.Mutex // File watchers registered by the server registrations []protocol.FileSystemWatcher registrationMu sync.RWMutex } // NewWorkspaceWatcher creates a new workspace watcher func NewWorkspaceWatcher(client *lsp.Client) *WorkspaceWatcher { return &WorkspaceWatcher{ client: client, debounceTime: 300 * time.Millisecond, debounceMap: make(map[string]*time.Timer), registrations: []protocol.FileSystemWatcher{}, } } // AddRegistrations adds file watchers to track func (w *WorkspaceWatcher) AddRegistrations(ctx context.Context, id string, watchers []protocol.FileSystemWatcher) { cnf := config.Get() logging.Debug("Adding file watcher registrations") w.registrationMu.Lock() defer w.registrationMu.Unlock() // Add new watchers w.registrations = append(w.registrations, watchers...) // Print detailed registration information for debugging if cnf.DebugLSP { logging.Debug("Adding file watcher registrations", "id", id, "watchers", len(watchers), "total", len(w.registrations), ) for i, watcher := range watchers { logging.Debug("Registration", "index", i+1) // Log the GlobPattern switch v := watcher.GlobPattern.Value.(type) { case string: logging.Debug("GlobPattern", "pattern", v) case protocol.RelativePattern: logging.Debug("GlobPattern", "pattern", v.Pattern) // Log BaseURI details switch u := v.BaseURI.Value.(type) { case string: logging.Debug("BaseURI", "baseURI", u) case protocol.DocumentUri: logging.Debug("BaseURI", "baseURI", u) default: logging.Debug("BaseURI", "baseURI", u) } default: logging.Debug("GlobPattern", "unknown type", fmt.Sprintf("%T", v)) } // Log WatchKind watchKind := protocol.WatchKind(protocol.WatchChange | protocol.WatchCreate | protocol.WatchDelete) if watcher.Kind != nil { watchKind = *watcher.Kind } logging.Debug("WatchKind", "kind", watchKind) } } // Determine server type for specialized handling serverName := getServerNameFromContext(ctx) logging.Debug("Server type detected", "serverName", serverName) // Check if this server has sent file watchers hasFileWatchers := len(watchers) > 0 // For servers that need file preloading, we'll use a smart approach if shouldPreloadFiles(serverName) || !hasFileWatchers { go func() { startTime := time.Now() filesOpened := 0 // Determine max files to open based on server type maxFilesToOpen := 50 // Default conservative limit switch serverName { case "typescript", "typescript-language-server", "tsserver", "vtsls": // TypeScript servers benefit from seeing more files maxFilesToOpen = 100 case "java", "jdtls": // Java servers need to see many files for project model maxFilesToOpen = 200 } // First, open high-priority files highPriorityFilesOpened := w.openHighPriorityFiles(ctx, serverName) filesOpened += highPriorityFilesOpened if cnf.DebugLSP { logging.Debug("Opened high-priority files", "count", highPriorityFilesOpened, "serverName", serverName) } // If we've already opened enough high-priority files, we might not need more if filesOpened >= maxFilesToOpen { if cnf.DebugLSP { logging.Debug("Reached file limit with high-priority files", "filesOpened", filesOpened, "maxFiles", maxFilesToOpen) } return } // For the remaining slots, walk the directory and open matching files err := filepath.WalkDir(w.workspacePath, func(path string, d os.DirEntry, err error) error { if err != nil { return err } // Skip directories that should be excluded if d.IsDir() { if path != w.workspacePath && shouldExcludeDir(path) { if cnf.DebugLSP { logging.Debug("Skipping excluded directory", "path", path) } return filepath.SkipDir } } else { // Process files, but limit the total number if filesOpened < maxFilesToOpen { // Only process if it's not already open (high-priority files were opened earlier) if !w.client.IsFileOpen(path) { w.openMatchingFile(ctx, path) filesOpened++ // Add a small delay after every 10 files to prevent overwhelming the server if filesOpened%10 == 0 { time.Sleep(50 * time.Millisecond) } } } else { // We've reached our limit, stop walking return filepath.SkipAll } } return nil }) elapsedTime := time.Since(startTime) if cnf.DebugLSP { logging.Debug("Limited workspace scan complete", "filesOpened", filesOpened, "maxFiles", maxFilesToOpen, "elapsedTime", elapsedTime.Seconds(), "workspacePath", w.workspacePath, ) } if err != nil && cnf.DebugLSP { logging.Debug("Error scanning workspace for files to open", "error", err) } }() } else if cnf.DebugLSP { logging.Debug("Using on-demand file loading for server", "server", serverName) } } // openHighPriorityFiles opens important files for the server type // Returns the number of files opened func (w *WorkspaceWatcher) openHighPriorityFiles(ctx context.Context, serverName string) int { cnf := config.Get() filesOpened := 0 // Define patterns for high-priority files based on server type var patterns []string switch serverName { case "typescript", "typescript-language-server", "tsserver", "vtsls": patterns = []string{ "**/tsconfig.json", "**/package.json", "**/jsconfig.json", "**/index.ts", "**/index.js", "**/main.ts", "**/main.js", } case "gopls": patterns = []string{ "**/go.mod", "**/go.sum", "**/main.go", } case "rust-analyzer": patterns = []string{ "**/Cargo.toml", "**/Cargo.lock", "**/src/lib.rs", "**/src/main.rs", } case "python", "pyright", "pylsp": patterns = []string{ "**/pyproject.toml", "**/setup.py", "**/requirements.txt", "**/__init__.py", "**/__main__.py", } case "clangd": patterns = []string{ "**/CMakeLists.txt", "**/Makefile", "**/compile_commands.json", } case "java", "jdtls": patterns = []string{ "**/pom.xml", "**/build.gradle", "**/src/main/java/**/*.java", } default: // For unknown servers, use common configuration files patterns = []string{ "**/package.json", "**/Makefile", "**/CMakeLists.txt", "**/.editorconfig", } } // For each pattern, find and open matching files for _, pattern := range patterns { // Use doublestar.Glob to find files matching the pattern (supports ** patterns) matches, err := doublestar.Glob(os.DirFS(w.workspacePath), pattern) if err != nil { if cnf.DebugLSP { logging.Debug("Error finding high-priority files", "pattern", pattern, "error", err) } continue } for _, match := range matches { // Convert relative path to absolute fullPath := filepath.Join(w.workspacePath, match) // Skip directories and excluded files info, err := os.Stat(fullPath) if err != nil || info.IsDir() || shouldExcludeFile(fullPath) { continue } // Open the file if err := w.client.OpenFile(ctx, fullPath); err != nil { if cnf.DebugLSP { logging.Debug("Error opening high-priority file", "path", fullPath, "error", err) } } else { filesOpened++ if cnf.DebugLSP { logging.Debug("Opened high-priority file", "path", fullPath) } } // Add a small delay to prevent overwhelming the server time.Sleep(20 * time.Millisecond) // Limit the number of files opened per pattern if filesOpened >= 5 && (serverName != "java" && serverName != "jdtls") { break } } } return filesOpened } // WatchWorkspace sets up file watching for a workspace func (w *WorkspaceWatcher) WatchWorkspace(ctx context.Context, workspacePath string) { cnf := config.Get() w.workspacePath = workspacePath // Store the watcher in the context for later use ctx = context.WithValue(ctx, "workspaceWatcher", w) // If the server name isn't already in the context, try to detect it if _, ok := ctx.Value("serverName").(string); !ok { serverName := getServerNameFromContext(ctx) ctx = context.WithValue(ctx, "serverName", serverName) } serverName := getServerNameFromContext(ctx) logging.Debug("Starting workspace watcher", "workspacePath", workspacePath, "serverName", serverName) // Register handler for file watcher registrations from the server lsp.RegisterFileWatchHandler(func(id string, watchers []protocol.FileSystemWatcher) { w.AddRegistrations(ctx, id, watchers) }) watcher, err := fsnotify.NewWatcher() if err != nil { logging.Error("Error creating watcher", "error", err) } defer watcher.Close() // Watch the workspace recursively err = filepath.WalkDir(workspacePath, func(path string, d os.DirEntry, err error) error { if err != nil { return err } // Skip excluded directories (except workspace root) if d.IsDir() && path != workspacePath { if shouldExcludeDir(path) { if cnf.DebugLSP { logging.Debug("Skipping excluded directory", "path", path) } return filepath.SkipDir } } // Add directories to watcher if d.IsDir() { err = watcher.Add(path) if err != nil { logging.Error("Error watching path", "path", path, "error", err) } } return nil }) if err != nil { logging.Error("Error walking workspace", "error", err) } // Event loop for { select { case <-ctx.Done(): return case event, ok := <-watcher.Events: if !ok { return } uri := fmt.Sprintf("file://%s", event.Name) // Add new directories to the watcher if event.Op&fsnotify.Create != 0 { if info, err := os.Stat(event.Name); err == nil { if info.IsDir() { // Skip excluded directories if !shouldExcludeDir(event.Name) { if err := watcher.Add(event.Name); err != nil { logging.Error("Error adding directory to watcher", "path", event.Name, "error", err) } } } else { // For newly created files if !shouldExcludeFile(event.Name) { w.openMatchingFile(ctx, event.Name) } } } } // Debug logging if cnf.DebugLSP { matched, kind := w.isPathWatched(event.Name) logging.Debug("File event", "path", event.Name, "operation", event.Op.String(), "watched", matched, "kind", kind, ) } // Check if this path should be watched according to server registrations if watched, watchKind := w.isPathWatched(event.Name); watched { switch { case event.Op&fsnotify.Write != 0: if watchKind&protocol.WatchChange != 0 { w.debounceHandleFileEvent(ctx, uri, protocol.FileChangeType(protocol.Changed)) } case event.Op&fsnotify.Create != 0: // Already handled earlier in the event loop // Just send the notification if needed info, err := os.Stat(event.Name) if err != nil { logging.Error("Error getting file info", "path", event.Name, "error", err) return } if !info.IsDir() && watchKind&protocol.WatchCreate != 0 { w.debounceHandleFileEvent(ctx, uri, protocol.FileChangeType(protocol.Created)) } case event.Op&fsnotify.Remove != 0: if watchKind&protocol.WatchDelete != 0 { w.handleFileEvent(ctx, uri, protocol.FileChangeType(protocol.Deleted)) } case event.Op&fsnotify.Rename != 0: // For renames, first delete if watchKind&protocol.WatchDelete != 0 { w.handleFileEvent(ctx, uri, protocol.FileChangeType(protocol.Deleted)) } // Then check if the new file exists and create an event if info, err := os.Stat(event.Name); err == nil && !info.IsDir() { if watchKind&protocol.WatchCreate != 0 { w.debounceHandleFileEvent(ctx, uri, protocol.FileChangeType(protocol.Created)) } } } } case err, ok := <-watcher.Errors: if !ok { return } logging.Error("Error watching file", "error", err) } } } // isPathWatched checks if a path should be watched based on server registrations func (w *WorkspaceWatcher) isPathWatched(path string) (bool, protocol.WatchKind) { w.registrationMu.RLock() defer w.registrationMu.RUnlock() // If no explicit registrations, watch everything if len(w.registrations) == 0 { return true, protocol.WatchKind(protocol.WatchChange | protocol.WatchCreate | protocol.WatchDelete) } // Check each registration for _, reg := range w.registrations { isMatch := w.matchesPattern(path, reg.GlobPattern) if isMatch { kind := protocol.WatchKind(protocol.WatchChange | protocol.WatchCreate | protocol.WatchDelete) if reg.Kind != nil { kind = *reg.Kind } return true, kind } } return false, 0 } // matchesGlob handles advanced glob patterns including ** and alternatives func matchesGlob(pattern, path string) bool { // Handle file extension patterns with braces like *.{go,mod,sum} if strings.Contains(pattern, "{") && strings.Contains(pattern, "}") { // Extract extensions from pattern like "*.{go,mod,sum}" parts := strings.SplitN(pattern, "{", 2) if len(parts) == 2 { prefix := parts[0] extPart := strings.SplitN(parts[1], "}", 2) if len(extPart) == 2 { extensions := strings.Split(extPart[0], ",") suffix := extPart[1] // Check if the path matches any of the extensions for _, ext := range extensions { extPattern := prefix + ext + suffix isMatch := matchesSimpleGlob(extPattern, path) if isMatch { return true } } return false } } } return matchesSimpleGlob(pattern, path) } // matchesSimpleGlob handles glob patterns with ** wildcards func matchesSimpleGlob(pattern, path string) bool { // Handle special case for **/*.ext pattern (common in LSP) if strings.HasPrefix(pattern, "**/") { rest := strings.TrimPrefix(pattern, "**/") // If the rest is a simple file extension pattern like *.go if strings.HasPrefix(rest, "*.") { ext := strings.TrimPrefix(rest, "*") isMatch := strings.HasSuffix(path, ext) return isMatch } // Otherwise, try to check if the path ends with the rest part isMatch := strings.HasSuffix(path, rest) // If it matches directly, great! if isMatch { return true } // Otherwise, check if any path component matches pathComponents := strings.Split(path, "/") for i := range pathComponents { subPath := strings.Join(pathComponents[i:], "/") if strings.HasSuffix(subPath, rest) { return true } } return false } // Handle other ** wildcard pattern cases if strings.Contains(pattern, "**") { parts := strings.Split(pattern, "**") // Validate the path starts with the first part if !strings.HasPrefix(path, parts[0]) && parts[0] != "" { return false } // For patterns like "**/*.go", just check the suffix if len(parts) == 2 && parts[0] == "" { isMatch := strings.HasSuffix(path, parts[1]) return isMatch } // For other patterns, handle middle part remaining := strings.TrimPrefix(path, parts[0]) if len(parts) == 2 { isMatch := strings.HasSuffix(remaining, parts[1]) return isMatch } } // Handle simple * wildcard for file extension patterns (*.go, *.sum, etc) if strings.HasPrefix(pattern, "*.") { ext := strings.TrimPrefix(pattern, "*") isMatch := strings.HasSuffix(path, ext) return isMatch } // Fall back to simple matching for simpler patterns matched, err := filepath.Match(pattern, path) if err != nil { logging.Error("Error matching pattern", "pattern", pattern, "path", path, "error", err) return false } return matched } // matchesPattern checks if a path matches the glob pattern func (w *WorkspaceWatcher) matchesPattern(path string, pattern protocol.GlobPattern) bool { patternInfo, err := pattern.AsPattern() if err != nil { logging.Error("Error parsing pattern", "pattern", pattern, "error", err) return false } basePath := patternInfo.GetBasePath() patternText := patternInfo.GetPattern() path = filepath.ToSlash(path) // For simple patterns without base path if basePath == "" { // Check if the pattern matches the full path or just the file extension fullPathMatch := matchesGlob(patternText, path) baseNameMatch := matchesGlob(patternText, filepath.Base(path)) return fullPathMatch || baseNameMatch } // For relative patterns basePath = strings.TrimPrefix(basePath, "file://") basePath = filepath.ToSlash(basePath) // Make path relative to basePath for matching relPath, err := filepath.Rel(basePath, path) if err != nil { logging.Error("Error getting relative path", "path", path, "basePath", basePath, "error", err) return false } relPath = filepath.ToSlash(relPath) isMatch := matchesGlob(patternText, relPath) return isMatch } // debounceHandleFileEvent handles file events with debouncing to reduce notifications func (w *WorkspaceWatcher) debounceHandleFileEvent(ctx context.Context, uri string, changeType protocol.FileChangeType) { w.debounceMu.Lock() defer w.debounceMu.Unlock() // Create a unique key based on URI and change type key := fmt.Sprintf("%s:%d", uri, changeType) // Cancel existing timer if any if timer, exists := w.debounceMap[key]; exists { timer.Stop() } // Create new timer w.debounceMap[key] = time.AfterFunc(w.debounceTime, func() { w.handleFileEvent(ctx, uri, changeType) // Cleanup timer after execution w.debounceMu.Lock() delete(w.debounceMap, key) w.debounceMu.Unlock() }) } // handleFileEvent sends file change notifications func (w *WorkspaceWatcher) handleFileEvent(ctx context.Context, uri string, changeType protocol.FileChangeType) { // If the file is open and it's a change event, use didChange notification filePath := uri[7:] // Remove "file://" prefix if changeType == protocol.FileChangeType(protocol.Deleted) { w.client.ClearDiagnosticsForURI(protocol.DocumentUri(uri)) } else if changeType == protocol.FileChangeType(protocol.Changed) && w.client.IsFileOpen(filePath) { err := w.client.NotifyChange(ctx, filePath) if err != nil { logging.Error("Error notifying change", "error", err) } return } // Notify LSP server about the file event using didChangeWatchedFiles if err := w.notifyFileEvent(ctx, uri, changeType); err != nil { logging.Error("Error notifying LSP server about file event", "error", err) } } // notifyFileEvent sends a didChangeWatchedFiles notification for a file event func (w *WorkspaceWatcher) notifyFileEvent(ctx context.Context, uri string, changeType protocol.FileChangeType) error { cnf := config.Get() if cnf.DebugLSP { logging.Debug("Notifying file event", "uri", uri, "changeType", changeType, ) } params := protocol.DidChangeWatchedFilesParams{ Changes: []protocol.FileEvent{ { URI: protocol.DocumentUri(uri), Type: changeType, }, }, } return w.client.DidChangeWatchedFiles(ctx, params) } // getServerNameFromContext extracts the server name from the context // This is a best-effort function that tries to identify which LSP server we're dealing with func getServerNameFromContext(ctx context.Context) string { // First check if the server name is directly stored in the context if serverName, ok := ctx.Value("serverName").(string); ok && serverName != "" { return strings.ToLower(serverName) } // Otherwise, try to extract server name from the client command path if w, ok := ctx.Value("workspaceWatcher").(*WorkspaceWatcher); ok && w != nil && w.client != nil && w.client.Cmd != nil { path := strings.ToLower(w.client.Cmd.Path) // Extract server name from path if strings.Contains(path, "typescript") || strings.Contains(path, "tsserver") || strings.Contains(path, "vtsls") { return "typescript" } else if strings.Contains(path, "gopls") { return "gopls" } else if strings.Contains(path, "rust-analyzer") { return "rust-analyzer" } else if strings.Contains(path, "pyright") || strings.Contains(path, "pylsp") || strings.Contains(path, "python") { return "python" } else if strings.Contains(path, "clangd") { return "clangd" } else if strings.Contains(path, "jdtls") || strings.Contains(path, "java") { return "java" } // Return the base name as fallback return filepath.Base(path) } return "unknown" } // shouldPreloadFiles determines if we should preload files for a specific language server // Some servers work better with preloaded files, others don't need it func shouldPreloadFiles(serverName string) bool { // TypeScript/JavaScript servers typically need some files preloaded // to properly resolve imports and provide intellisense switch serverName { case "typescript", "typescript-language-server", "tsserver", "vtsls": return true case "java", "jdtls": // Java servers often need to see source files to build the project model return true default: // For most servers, we'll use lazy loading by default return false } } // Common patterns for directories and files to exclude // TODO: make configurable var ( excludedDirNames = map[string]bool{ ".git": true, "node_modules": true, "dist": true, "build": true, "out": true, "bin": true, ".idea": true, ".vscode": true, ".cache": true, "coverage": true, "target": true, // Rust build output "vendor": true, // Go vendor directory } excludedFileExtensions = map[string]bool{ ".swp": true, ".swo": true, ".tmp": true, ".temp": true, ".bak": true, ".log": true, ".o": true, // Object files ".so": true, // Shared libraries ".dylib": true, // macOS shared libraries ".dll": true, // Windows shared libraries ".a": true, // Static libraries ".exe": true, // Windows executables ".lock": true, // Lock files } // Large binary files that shouldn't be opened largeBinaryExtensions = map[string]bool{ ".png": true, ".jpg": true, ".jpeg": true, ".gif": true, ".bmp": true, ".ico": true, ".zip": true, ".tar": true, ".gz": true, ".rar": true, ".7z": true, ".pdf": true, ".mp3": true, ".mp4": true, ".mov": true, ".wav": true, ".wasm": true, } // Maximum file size to open (5MB) maxFileSize int64 = 5 * 1024 * 1024 ) // shouldExcludeDir returns true if the directory should be excluded from watching/opening func shouldExcludeDir(dirPath string) bool { dirName := filepath.Base(dirPath) // Skip dot directories if strings.HasPrefix(dirName, ".") { return true } // Skip common excluded directories if excludedDirNames[dirName] { return true } return false } // shouldExcludeFile returns true if the file should be excluded from opening func shouldExcludeFile(filePath string) bool { fileName := filepath.Base(filePath) cnf := config.Get() // Skip dot files if strings.HasPrefix(fileName, ".") { return true } // Check file extension ext := strings.ToLower(filepath.Ext(filePath)) if excludedFileExtensions[ext] || largeBinaryExtensions[ext] { return true } // Skip temporary files if strings.HasSuffix(filePath, "~") { return true } // Check file size info, err := os.Stat(filePath) if err != nil { // If we can't stat the file, skip it return true } // Skip large files if info.Size() > maxFileSize { if cnf.DebugLSP { logging.Debug("Skipping large file", "path", filePath, "size", info.Size(), "maxSize", maxFileSize, "debug", cnf.Debug, "sizeMB", float64(info.Size())/(1024*1024), "maxSizeMB", float64(maxFileSize)/(1024*1024), ) } return true } return false } // openMatchingFile opens a file if it matches any of the registered patterns func (w *WorkspaceWatcher) openMatchingFile(ctx context.Context, path string) { cnf := config.Get() // Skip directories info, err := os.Stat(path) if err != nil || info.IsDir() { return } // Skip excluded files if shouldExcludeFile(path) { return } // Check if this path should be watched according to server registrations if watched, _ := w.isPathWatched(path); watched { // Get server name for specialized handling serverName := getServerNameFromContext(ctx) // Check if the file is a high-priority file that should be opened immediately // This helps with project initialization for certain language servers if isHighPriorityFile(path, serverName) { if cnf.DebugLSP { logging.Debug("Opening high-priority file", "path", path, "serverName", serverName) } if err := w.client.OpenFile(ctx, path); err != nil && cnf.DebugLSP { logging.Error("Error opening high-priority file", "path", path, "error", err) } return } // For non-high-priority files, we'll use different strategies based on server type if shouldPreloadFiles(serverName) { // For servers that benefit from preloading, open files but with limits // Check file size - for preloading we're more conservative if info.Size() > (1 * 1024 * 1024) { // 1MB limit for preloaded files if cnf.DebugLSP { logging.Debug("Skipping large file for preloading", "path", path, "size", info.Size()) } return } // Check file extension for common source files ext := strings.ToLower(filepath.Ext(path)) // Only preload source files for the specific language shouldOpen := false switch serverName { case "typescript", "typescript-language-server", "tsserver", "vtsls": shouldOpen = ext == ".ts" || ext == ".js" || ext == ".tsx" || ext == ".jsx" case "gopls": shouldOpen = ext == ".go" case "rust-analyzer": shouldOpen = ext == ".rs" case "python", "pyright", "pylsp": shouldOpen = ext == ".py" case "clangd": shouldOpen = ext == ".c" || ext == ".cpp" || ext == ".h" || ext == ".hpp" case "java", "jdtls": shouldOpen = ext == ".java" default: // For unknown servers, be conservative shouldOpen = false } if shouldOpen { // Don't need to check if it's already open - the client.OpenFile handles that if err := w.client.OpenFile(ctx, path); err != nil && cnf.DebugLSP { logging.Error("Error opening file", "path", path, "error", err) } } } } } // isHighPriorityFile determines if a file should be opened immediately // regardless of the preloading strategy func isHighPriorityFile(path string, serverName string) bool { fileName := filepath.Base(path) ext := filepath.Ext(path) switch serverName { case "typescript", "typescript-language-server", "tsserver", "vtsls": // For TypeScript, we want to open configuration files immediately return fileName == "tsconfig.json" || fileName == "package.json" || fileName == "jsconfig.json" || // Also open main entry points fileName == "index.ts" || fileName == "index.js" || fileName == "main.ts" || fileName == "main.js" case "gopls": // For Go, we want to open go.mod files immediately return fileName == "go.mod" || fileName == "go.sum" || // Also open main.go files fileName == "main.go" case "rust-analyzer": // For Rust, we want to open Cargo.toml files immediately return fileName == "Cargo.toml" || fileName == "Cargo.lock" || // Also open lib.rs and main.rs fileName == "lib.rs" || fileName == "main.rs" case "python", "pyright", "pylsp": // For Python, open key project files return fileName == "pyproject.toml" || fileName == "setup.py" || fileName == "requirements.txt" || fileName == "__init__.py" || fileName == "__main__.py" case "clangd": // For C/C++, open key project files return fileName == "CMakeLists.txt" || fileName == "Makefile" || fileName == "compile_commands.json" case "java", "jdtls": // For Java, open key project files return fileName == "pom.xml" || fileName == "build.gradle" || ext == ".java" // Java servers often need to see source files } // For unknown servers, prioritize common configuration files return fileName == "package.json" || fileName == "Makefile" || fileName == "CMakeLists.txt" || fileName == ".editorconfig" } ================================================ FILE: internal/message/attachment.go ================================================ package message type Attachment struct { FilePath string FileName string MimeType string Content []byte } ================================================ FILE: internal/message/content.go ================================================ package message import ( "encoding/base64" "slices" "time" "github.com/opencode-ai/opencode/internal/llm/models" ) type MessageRole string const ( Assistant MessageRole = "assistant" User MessageRole = "user" System MessageRole = "system" Tool MessageRole = "tool" ) type FinishReason string const ( FinishReasonEndTurn FinishReason = "end_turn" FinishReasonMaxTokens FinishReason = "max_tokens" FinishReasonToolUse FinishReason = "tool_use" FinishReasonCanceled FinishReason = "canceled" FinishReasonError FinishReason = "error" FinishReasonPermissionDenied FinishReason = "permission_denied" // Should never happen FinishReasonUnknown FinishReason = "unknown" ) type ContentPart interface { isPart() } type ReasoningContent struct { Thinking string `json:"thinking"` } func (tc ReasoningContent) String() string { return tc.Thinking } func (ReasoningContent) isPart() {} type TextContent struct { Text string `json:"text"` } func (tc TextContent) String() string { return tc.Text } func (TextContent) isPart() {} type ImageURLContent struct { URL string `json:"url"` Detail string `json:"detail,omitempty"` } func (iuc ImageURLContent) String() string { return iuc.URL } func (ImageURLContent) isPart() {} type BinaryContent struct { Path string MIMEType string Data []byte } func (bc BinaryContent) String(provider models.ModelProvider) string { base64Encoded := base64.StdEncoding.EncodeToString(bc.Data) if provider == models.ProviderOpenAI { return "data:" + bc.MIMEType + ";base64," + base64Encoded } return base64Encoded } func (BinaryContent) isPart() {} type ToolCall struct { ID string `json:"id"` Name string `json:"name"` Input string `json:"input"` Type string `json:"type"` Finished bool `json:"finished"` } func (ToolCall) isPart() {} type ToolResult struct { ToolCallID string `json:"tool_call_id"` Name string `json:"name"` Content string `json:"content"` Metadata string `json:"metadata"` IsError bool `json:"is_error"` } func (ToolResult) isPart() {} type Finish struct { Reason FinishReason `json:"reason"` Time int64 `json:"time"` } func (Finish) isPart() {} type Message struct { ID string Role MessageRole SessionID string Parts []ContentPart Model models.ModelID CreatedAt int64 UpdatedAt int64 } func (m *Message) Content() TextContent { for _, part := range m.Parts { if c, ok := part.(TextContent); ok { return c } } return TextContent{} } func (m *Message) ReasoningContent() ReasoningContent { for _, part := range m.Parts { if c, ok := part.(ReasoningContent); ok { return c } } return ReasoningContent{} } func (m *Message) ImageURLContent() []ImageURLContent { imageURLContents := make([]ImageURLContent, 0) for _, part := range m.Parts { if c, ok := part.(ImageURLContent); ok { imageURLContents = append(imageURLContents, c) } } return imageURLContents } func (m *Message) BinaryContent() []BinaryContent { binaryContents := make([]BinaryContent, 0) for _, part := range m.Parts { if c, ok := part.(BinaryContent); ok { binaryContents = append(binaryContents, c) } } return binaryContents } func (m *Message) ToolCalls() []ToolCall { toolCalls := make([]ToolCall, 0) for _, part := range m.Parts { if c, ok := part.(ToolCall); ok { toolCalls = append(toolCalls, c) } } return toolCalls } func (m *Message) ToolResults() []ToolResult { toolResults := make([]ToolResult, 0) for _, part := range m.Parts { if c, ok := part.(ToolResult); ok { toolResults = append(toolResults, c) } } return toolResults } func (m *Message) IsFinished() bool { for _, part := range m.Parts { if _, ok := part.(Finish); ok { return true } } return false } func (m *Message) FinishPart() *Finish { for _, part := range m.Parts { if c, ok := part.(Finish); ok { return &c } } return nil } func (m *Message) FinishReason() FinishReason { for _, part := range m.Parts { if c, ok := part.(Finish); ok { return c.Reason } } return "" } func (m *Message) IsThinking() bool { if m.ReasoningContent().Thinking != "" && m.Content().Text == "" && !m.IsFinished() { return true } return false } func (m *Message) AppendContent(delta string) { found := false for i, part := range m.Parts { if c, ok := part.(TextContent); ok { m.Parts[i] = TextContent{Text: c.Text + delta} found = true } } if !found { m.Parts = append(m.Parts, TextContent{Text: delta}) } } func (m *Message) AppendReasoningContent(delta string) { found := false for i, part := range m.Parts { if c, ok := part.(ReasoningContent); ok { m.Parts[i] = ReasoningContent{Thinking: c.Thinking + delta} found = true } } if !found { m.Parts = append(m.Parts, ReasoningContent{Thinking: delta}) } } func (m *Message) FinishToolCall(toolCallID string) { for i, part := range m.Parts { if c, ok := part.(ToolCall); ok { if c.ID == toolCallID { m.Parts[i] = ToolCall{ ID: c.ID, Name: c.Name, Input: c.Input, Type: c.Type, Finished: true, } return } } } } func (m *Message) AppendToolCallInput(toolCallID string, inputDelta string) { for i, part := range m.Parts { if c, ok := part.(ToolCall); ok { if c.ID == toolCallID { m.Parts[i] = ToolCall{ ID: c.ID, Name: c.Name, Input: c.Input + inputDelta, Type: c.Type, Finished: c.Finished, } return } } } } func (m *Message) AddToolCall(tc ToolCall) { for i, part := range m.Parts { if c, ok := part.(ToolCall); ok { if c.ID == tc.ID { m.Parts[i] = tc return } } } m.Parts = append(m.Parts, tc) } func (m *Message) SetToolCalls(tc []ToolCall) { // remove any existing tool call part it could have multiple parts := make([]ContentPart, 0) for _, part := range m.Parts { if _, ok := part.(ToolCall); ok { continue } parts = append(parts, part) } m.Parts = parts for _, toolCall := range tc { m.Parts = append(m.Parts, toolCall) } } func (m *Message) AddToolResult(tr ToolResult) { m.Parts = append(m.Parts, tr) } func (m *Message) SetToolResults(tr []ToolResult) { for _, toolResult := range tr { m.Parts = append(m.Parts, toolResult) } } func (m *Message) AddFinish(reason FinishReason) { // remove any existing finish part for i, part := range m.Parts { if _, ok := part.(Finish); ok { m.Parts = slices.Delete(m.Parts, i, i+1) break } } m.Parts = append(m.Parts, Finish{Reason: reason, Time: time.Now().Unix()}) } func (m *Message) AddImageURL(url, detail string) { m.Parts = append(m.Parts, ImageURLContent{URL: url, Detail: detail}) } func (m *Message) AddBinary(mimeType string, data []byte) { m.Parts = append(m.Parts, BinaryContent{MIMEType: mimeType, Data: data}) } ================================================ FILE: internal/message/message.go ================================================ package message import ( "context" "database/sql" "encoding/json" "fmt" "time" "github.com/google/uuid" "github.com/opencode-ai/opencode/internal/db" "github.com/opencode-ai/opencode/internal/llm/models" "github.com/opencode-ai/opencode/internal/pubsub" ) type CreateMessageParams struct { Role MessageRole Parts []ContentPart Model models.ModelID } type Service interface { pubsub.Suscriber[Message] Create(ctx context.Context, sessionID string, params CreateMessageParams) (Message, error) Update(ctx context.Context, message Message) error Get(ctx context.Context, id string) (Message, error) List(ctx context.Context, sessionID string) ([]Message, error) Delete(ctx context.Context, id string) error DeleteSessionMessages(ctx context.Context, sessionID string) error } type service struct { *pubsub.Broker[Message] q db.Querier } func NewService(q db.Querier) Service { return &service{ Broker: pubsub.NewBroker[Message](), q: q, } } func (s *service) Delete(ctx context.Context, id string) error { message, err := s.Get(ctx, id) if err != nil { return err } err = s.q.DeleteMessage(ctx, message.ID) if err != nil { return err } s.Publish(pubsub.DeletedEvent, message) return nil } func (s *service) Create(ctx context.Context, sessionID string, params CreateMessageParams) (Message, error) { if params.Role != Assistant { params.Parts = append(params.Parts, Finish{ Reason: "stop", }) } partsJSON, err := marshallParts(params.Parts) if err != nil { return Message{}, err } dbMessage, err := s.q.CreateMessage(ctx, db.CreateMessageParams{ ID: uuid.New().String(), SessionID: sessionID, Role: string(params.Role), Parts: string(partsJSON), Model: sql.NullString{String: string(params.Model), Valid: true}, }) if err != nil { return Message{}, err } message, err := s.fromDBItem(dbMessage) if err != nil { return Message{}, err } s.Publish(pubsub.CreatedEvent, message) return message, nil } func (s *service) DeleteSessionMessages(ctx context.Context, sessionID string) error { messages, err := s.List(ctx, sessionID) if err != nil { return err } for _, message := range messages { if message.SessionID == sessionID { err = s.Delete(ctx, message.ID) if err != nil { return err } } } return nil } func (s *service) Update(ctx context.Context, message Message) error { parts, err := marshallParts(message.Parts) if err != nil { return err } finishedAt := sql.NullInt64{} if f := message.FinishPart(); f != nil { finishedAt.Int64 = f.Time finishedAt.Valid = true } err = s.q.UpdateMessage(ctx, db.UpdateMessageParams{ ID: message.ID, Parts: string(parts), FinishedAt: finishedAt, }) if err != nil { return err } message.UpdatedAt = time.Now().Unix() s.Publish(pubsub.UpdatedEvent, message) return nil } func (s *service) Get(ctx context.Context, id string) (Message, error) { dbMessage, err := s.q.GetMessage(ctx, id) if err != nil { return Message{}, err } return s.fromDBItem(dbMessage) } func (s *service) List(ctx context.Context, sessionID string) ([]Message, error) { dbMessages, err := s.q.ListMessagesBySession(ctx, sessionID) if err != nil { return nil, err } messages := make([]Message, len(dbMessages)) for i, dbMessage := range dbMessages { messages[i], err = s.fromDBItem(dbMessage) if err != nil { return nil, err } } return messages, nil } func (s *service) fromDBItem(item db.Message) (Message, error) { parts, err := unmarshallParts([]byte(item.Parts)) if err != nil { return Message{}, err } return Message{ ID: item.ID, SessionID: item.SessionID, Role: MessageRole(item.Role), Parts: parts, Model: models.ModelID(item.Model.String), CreatedAt: item.CreatedAt, UpdatedAt: item.UpdatedAt, }, nil } type partType string const ( reasoningType partType = "reasoning" textType partType = "text" imageURLType partType = "image_url" binaryType partType = "binary" toolCallType partType = "tool_call" toolResultType partType = "tool_result" finishType partType = "finish" ) type partWrapper struct { Type partType `json:"type"` Data ContentPart `json:"data"` } func marshallParts(parts []ContentPart) ([]byte, error) { wrappedParts := make([]partWrapper, len(parts)) for i, part := range parts { var typ partType switch part.(type) { case ReasoningContent: typ = reasoningType case TextContent: typ = textType case ImageURLContent: typ = imageURLType case BinaryContent: typ = binaryType case ToolCall: typ = toolCallType case ToolResult: typ = toolResultType case Finish: typ = finishType default: return nil, fmt.Errorf("unknown part type: %T", part) } wrappedParts[i] = partWrapper{ Type: typ, Data: part, } } return json.Marshal(wrappedParts) } func unmarshallParts(data []byte) ([]ContentPart, error) { temp := []json.RawMessage{} if err := json.Unmarshal(data, &temp); err != nil { return nil, err } parts := make([]ContentPart, 0) for _, rawPart := range temp { var wrapper struct { Type partType `json:"type"` Data json.RawMessage `json:"data"` } if err := json.Unmarshal(rawPart, &wrapper); err != nil { return nil, err } switch wrapper.Type { case reasoningType: part := ReasoningContent{} if err := json.Unmarshal(wrapper.Data, &part); err != nil { return nil, err } parts = append(parts, part) case textType: part := TextContent{} if err := json.Unmarshal(wrapper.Data, &part); err != nil { return nil, err } parts = append(parts, part) case imageURLType: part := ImageURLContent{} if err := json.Unmarshal(wrapper.Data, &part); err != nil { return nil, err } case binaryType: part := BinaryContent{} if err := json.Unmarshal(wrapper.Data, &part); err != nil { return nil, err } parts = append(parts, part) case toolCallType: part := ToolCall{} if err := json.Unmarshal(wrapper.Data, &part); err != nil { return nil, err } parts = append(parts, part) case toolResultType: part := ToolResult{} if err := json.Unmarshal(wrapper.Data, &part); err != nil { return nil, err } parts = append(parts, part) case finishType: part := Finish{} if err := json.Unmarshal(wrapper.Data, &part); err != nil { return nil, err } parts = append(parts, part) default: return nil, fmt.Errorf("unknown part type: %s", wrapper.Type) } } return parts, nil } ================================================ FILE: internal/permission/permission.go ================================================ package permission import ( "errors" "path/filepath" "slices" "sync" "github.com/google/uuid" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/pubsub" ) var ErrorPermissionDenied = errors.New("permission denied") type CreatePermissionRequest struct { SessionID string `json:"session_id"` ToolName string `json:"tool_name"` Description string `json:"description"` Action string `json:"action"` Params any `json:"params"` Path string `json:"path"` } type PermissionRequest struct { ID string `json:"id"` SessionID string `json:"session_id"` ToolName string `json:"tool_name"` Description string `json:"description"` Action string `json:"action"` Params any `json:"params"` Path string `json:"path"` } type Service interface { pubsub.Suscriber[PermissionRequest] GrantPersistant(permission PermissionRequest) Grant(permission PermissionRequest) Deny(permission PermissionRequest) Request(opts CreatePermissionRequest) bool AutoApproveSession(sessionID string) } type permissionService struct { *pubsub.Broker[PermissionRequest] sessionPermissions []PermissionRequest pendingRequests sync.Map autoApproveSessions []string } func (s *permissionService) GrantPersistant(permission PermissionRequest) { respCh, ok := s.pendingRequests.Load(permission.ID) if ok { respCh.(chan bool) <- true } s.sessionPermissions = append(s.sessionPermissions, permission) } func (s *permissionService) Grant(permission PermissionRequest) { respCh, ok := s.pendingRequests.Load(permission.ID) if ok { respCh.(chan bool) <- true } } func (s *permissionService) Deny(permission PermissionRequest) { respCh, ok := s.pendingRequests.Load(permission.ID) if ok { respCh.(chan bool) <- false } } func (s *permissionService) Request(opts CreatePermissionRequest) bool { if slices.Contains(s.autoApproveSessions, opts.SessionID) { return true } dir := filepath.Dir(opts.Path) if dir == "." { dir = config.WorkingDirectory() } permission := PermissionRequest{ ID: uuid.New().String(), Path: dir, SessionID: opts.SessionID, ToolName: opts.ToolName, Description: opts.Description, Action: opts.Action, Params: opts.Params, } for _, p := range s.sessionPermissions { if p.ToolName == permission.ToolName && p.Action == permission.Action && p.SessionID == permission.SessionID && p.Path == permission.Path { return true } } respCh := make(chan bool, 1) s.pendingRequests.Store(permission.ID, respCh) defer s.pendingRequests.Delete(permission.ID) s.Publish(pubsub.CreatedEvent, permission) // Wait for the response with a timeout resp := <-respCh return resp } func (s *permissionService) AutoApproveSession(sessionID string) { s.autoApproveSessions = append(s.autoApproveSessions, sessionID) } func NewPermissionService() Service { return &permissionService{ Broker: pubsub.NewBroker[PermissionRequest](), sessionPermissions: make([]PermissionRequest, 0), } } ================================================ FILE: internal/pubsub/broker.go ================================================ package pubsub import ( "context" "sync" ) const bufferSize = 64 type Broker[T any] struct { subs map[chan Event[T]]struct{} mu sync.RWMutex done chan struct{} subCount int maxEvents int } func NewBroker[T any]() *Broker[T] { return NewBrokerWithOptions[T](bufferSize, 1000) } func NewBrokerWithOptions[T any](channelBufferSize, maxEvents int) *Broker[T] { b := &Broker[T]{ subs: make(map[chan Event[T]]struct{}), done: make(chan struct{}), subCount: 0, maxEvents: maxEvents, } return b } func (b *Broker[T]) Shutdown() { select { case <-b.done: // Already closed return default: close(b.done) } b.mu.Lock() defer b.mu.Unlock() for ch := range b.subs { delete(b.subs, ch) close(ch) } b.subCount = 0 } func (b *Broker[T]) Subscribe(ctx context.Context) <-chan Event[T] { b.mu.Lock() defer b.mu.Unlock() select { case <-b.done: ch := make(chan Event[T]) close(ch) return ch default: } sub := make(chan Event[T], bufferSize) b.subs[sub] = struct{}{} b.subCount++ go func() { <-ctx.Done() b.mu.Lock() defer b.mu.Unlock() select { case <-b.done: return default: } delete(b.subs, sub) close(sub) b.subCount-- }() return sub } func (b *Broker[T]) GetSubscriberCount() int { b.mu.RLock() defer b.mu.RUnlock() return b.subCount } func (b *Broker[T]) Publish(t EventType, payload T) { b.mu.RLock() select { case <-b.done: b.mu.RUnlock() return default: } subscribers := make([]chan Event[T], 0, len(b.subs)) for sub := range b.subs { subscribers = append(subscribers, sub) } b.mu.RUnlock() event := Event[T]{Type: t, Payload: payload} for _, sub := range subscribers { select { case sub <- event: default: } } } ================================================ FILE: internal/pubsub/events.go ================================================ package pubsub import "context" const ( CreatedEvent EventType = "created" UpdatedEvent EventType = "updated" DeletedEvent EventType = "deleted" ) type Suscriber[T any] interface { Subscribe(context.Context) <-chan Event[T] } type ( // EventType identifies the type of event EventType string // Event represents an event in the lifecycle of a resource Event[T any] struct { Type EventType Payload T } Publisher[T any] interface { Publish(EventType, T) } ) ================================================ FILE: internal/session/session.go ================================================ package session import ( "context" "database/sql" "github.com/google/uuid" "github.com/opencode-ai/opencode/internal/db" "github.com/opencode-ai/opencode/internal/pubsub" ) type Session struct { ID string ParentSessionID string Title string MessageCount int64 PromptTokens int64 CompletionTokens int64 SummaryMessageID string Cost float64 CreatedAt int64 UpdatedAt int64 } type Service interface { pubsub.Suscriber[Session] Create(ctx context.Context, title string) (Session, error) CreateTitleSession(ctx context.Context, parentSessionID string) (Session, error) CreateTaskSession(ctx context.Context, toolCallID, parentSessionID, title string) (Session, error) Get(ctx context.Context, id string) (Session, error) List(ctx context.Context) ([]Session, error) Save(ctx context.Context, session Session) (Session, error) Delete(ctx context.Context, id string) error } type service struct { *pubsub.Broker[Session] q db.Querier } func (s *service) Create(ctx context.Context, title string) (Session, error) { dbSession, err := s.q.CreateSession(ctx, db.CreateSessionParams{ ID: uuid.New().String(), Title: title, }) if err != nil { return Session{}, err } session := s.fromDBItem(dbSession) s.Publish(pubsub.CreatedEvent, session) return session, nil } func (s *service) CreateTaskSession(ctx context.Context, toolCallID, parentSessionID, title string) (Session, error) { dbSession, err := s.q.CreateSession(ctx, db.CreateSessionParams{ ID: toolCallID, ParentSessionID: sql.NullString{String: parentSessionID, Valid: true}, Title: title, }) if err != nil { return Session{}, err } session := s.fromDBItem(dbSession) s.Publish(pubsub.CreatedEvent, session) return session, nil } func (s *service) CreateTitleSession(ctx context.Context, parentSessionID string) (Session, error) { dbSession, err := s.q.CreateSession(ctx, db.CreateSessionParams{ ID: "title-" + parentSessionID, ParentSessionID: sql.NullString{String: parentSessionID, Valid: true}, Title: "Generate a title", }) if err != nil { return Session{}, err } session := s.fromDBItem(dbSession) s.Publish(pubsub.CreatedEvent, session) return session, nil } func (s *service) Delete(ctx context.Context, id string) error { session, err := s.Get(ctx, id) if err != nil { return err } err = s.q.DeleteSession(ctx, session.ID) if err != nil { return err } s.Publish(pubsub.DeletedEvent, session) return nil } func (s *service) Get(ctx context.Context, id string) (Session, error) { dbSession, err := s.q.GetSessionByID(ctx, id) if err != nil { return Session{}, err } return s.fromDBItem(dbSession), nil } func (s *service) Save(ctx context.Context, session Session) (Session, error) { dbSession, err := s.q.UpdateSession(ctx, db.UpdateSessionParams{ ID: session.ID, Title: session.Title, PromptTokens: session.PromptTokens, CompletionTokens: session.CompletionTokens, SummaryMessageID: sql.NullString{ String: session.SummaryMessageID, Valid: session.SummaryMessageID != "", }, Cost: session.Cost, }) if err != nil { return Session{}, err } session = s.fromDBItem(dbSession) s.Publish(pubsub.UpdatedEvent, session) return session, nil } func (s *service) List(ctx context.Context) ([]Session, error) { dbSessions, err := s.q.ListSessions(ctx) if err != nil { return nil, err } sessions := make([]Session, len(dbSessions)) for i, dbSession := range dbSessions { sessions[i] = s.fromDBItem(dbSession) } return sessions, nil } func (s service) fromDBItem(item db.Session) Session { return Session{ ID: item.ID, ParentSessionID: item.ParentSessionID.String, Title: item.Title, MessageCount: item.MessageCount, PromptTokens: item.PromptTokens, CompletionTokens: item.CompletionTokens, SummaryMessageID: item.SummaryMessageID.String, Cost: item.Cost, CreatedAt: item.CreatedAt, UpdatedAt: item.UpdatedAt, } } func NewService(q db.Querier) Service { broker := pubsub.NewBroker[Session]() return &service{ broker, q, } } ================================================ FILE: internal/tui/components/chat/chat.go ================================================ package chat import ( "fmt" "sort" "github.com/charmbracelet/lipgloss" "github.com/charmbracelet/x/ansi" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/message" "github.com/opencode-ai/opencode/internal/session" "github.com/opencode-ai/opencode/internal/tui/styles" "github.com/opencode-ai/opencode/internal/tui/theme" "github.com/opencode-ai/opencode/internal/version" ) type SendMsg struct { Text string Attachments []message.Attachment } type SessionSelectedMsg = session.Session type SessionClearedMsg struct{} type EditorFocusMsg bool func header(width int) string { return lipgloss.JoinVertical( lipgloss.Top, logo(width), repo(width), "", cwd(width), ) } func lspsConfigured(width int) string { cfg := config.Get() title := "LSP Configuration" title = ansi.Truncate(title, width, "…") t := theme.CurrentTheme() baseStyle := styles.BaseStyle() lsps := baseStyle. Width(width). Foreground(t.Primary()). Bold(true). Render(title) // Get LSP names and sort them for consistent ordering var lspNames []string for name := range cfg.LSP { lspNames = append(lspNames, name) } sort.Strings(lspNames) var lspViews []string for _, name := range lspNames { lsp := cfg.LSP[name] lspName := baseStyle. Foreground(t.Text()). Render(fmt.Sprintf("• %s", name)) cmd := lsp.Command cmd = ansi.Truncate(cmd, width-lipgloss.Width(lspName)-3, "…") lspPath := baseStyle. Foreground(t.TextMuted()). Render(fmt.Sprintf(" (%s)", cmd)) lspViews = append(lspViews, baseStyle. Width(width). Render( lipgloss.JoinHorizontal( lipgloss.Left, lspName, lspPath, ), ), ) } return baseStyle. Width(width). Render( lipgloss.JoinVertical( lipgloss.Left, lsps, lipgloss.JoinVertical( lipgloss.Left, lspViews..., ), ), ) } func logo(width int) string { logo := fmt.Sprintf("%s %s", styles.OpenCodeIcon, "OpenCode") t := theme.CurrentTheme() baseStyle := styles.BaseStyle() versionText := baseStyle. Foreground(t.TextMuted()). Render(version.Version) return baseStyle. Bold(true). Width(width). Render( lipgloss.JoinHorizontal( lipgloss.Left, logo, " ", versionText, ), ) } func repo(width int) string { repo := "https://github.com/opencode-ai/opencode" t := theme.CurrentTheme() return styles.BaseStyle(). Foreground(t.TextMuted()). Width(width). Render(repo) } func cwd(width int) string { cwd := fmt.Sprintf("cwd: %s", config.WorkingDirectory()) t := theme.CurrentTheme() return styles.BaseStyle(). Foreground(t.TextMuted()). Width(width). Render(cwd) } ================================================ FILE: internal/tui/components/chat/editor.go ================================================ package chat import ( "fmt" "os" "os/exec" "slices" "strings" "unicode" "github.com/charmbracelet/bubbles/key" "github.com/charmbracelet/bubbles/textarea" tea "github.com/charmbracelet/bubbletea" "github.com/charmbracelet/lipgloss" "github.com/opencode-ai/opencode/internal/app" "github.com/opencode-ai/opencode/internal/logging" "github.com/opencode-ai/opencode/internal/message" "github.com/opencode-ai/opencode/internal/session" "github.com/opencode-ai/opencode/internal/tui/components/dialog" "github.com/opencode-ai/opencode/internal/tui/layout" "github.com/opencode-ai/opencode/internal/tui/styles" "github.com/opencode-ai/opencode/internal/tui/theme" "github.com/opencode-ai/opencode/internal/tui/util" ) type editorCmp struct { width int height int app *app.App session session.Session textarea textarea.Model attachments []message.Attachment deleteMode bool } type EditorKeyMaps struct { Send key.Binding OpenEditor key.Binding } type bluredEditorKeyMaps struct { Send key.Binding Focus key.Binding OpenEditor key.Binding } type DeleteAttachmentKeyMaps struct { AttachmentDeleteMode key.Binding Escape key.Binding DeleteAllAttachments key.Binding } var editorMaps = EditorKeyMaps{ Send: key.NewBinding( key.WithKeys("enter", "ctrl+s"), key.WithHelp("enter", "send message"), ), OpenEditor: key.NewBinding( key.WithKeys("ctrl+e"), key.WithHelp("ctrl+e", "open editor"), ), } var DeleteKeyMaps = DeleteAttachmentKeyMaps{ AttachmentDeleteMode: key.NewBinding( key.WithKeys("ctrl+r"), key.WithHelp("ctrl+r+{i}", "delete attachment at index i"), ), Escape: key.NewBinding( key.WithKeys("esc"), key.WithHelp("esc", "cancel delete mode"), ), DeleteAllAttachments: key.NewBinding( key.WithKeys("r"), key.WithHelp("ctrl+r+r", "delete all attchments"), ), } const ( maxAttachments = 5 ) func (m *editorCmp) openEditor() tea.Cmd { editor := os.Getenv("EDITOR") if editor == "" { editor = "nvim" } tmpfile, err := os.CreateTemp("", "msg_*.md") if err != nil { return util.ReportError(err) } tmpfile.Close() c := exec.Command(editor, tmpfile.Name()) //nolint:gosec c.Stdin = os.Stdin c.Stdout = os.Stdout c.Stderr = os.Stderr return tea.ExecProcess(c, func(err error) tea.Msg { if err != nil { return util.ReportError(err) } content, err := os.ReadFile(tmpfile.Name()) if err != nil { return util.ReportError(err) } if len(content) == 0 { return util.ReportWarn("Message is empty") } os.Remove(tmpfile.Name()) attachments := m.attachments m.attachments = nil return SendMsg{ Text: string(content), Attachments: attachments, } }) } func (m *editorCmp) Init() tea.Cmd { return textarea.Blink } func (m *editorCmp) send() tea.Cmd { if m.app.CoderAgent.IsSessionBusy(m.session.ID) { return util.ReportWarn("Agent is working, please wait...") } value := m.textarea.Value() m.textarea.Reset() attachments := m.attachments m.attachments = nil if value == "" { return nil } return tea.Batch( util.CmdHandler(SendMsg{ Text: value, Attachments: attachments, }), ) } func (m *editorCmp) Update(msg tea.Msg) (tea.Model, tea.Cmd) { var cmd tea.Cmd switch msg := msg.(type) { case dialog.ThemeChangedMsg: m.textarea = CreateTextArea(&m.textarea) case dialog.CompletionSelectedMsg: existingValue := m.textarea.Value() modifiedValue := strings.Replace(existingValue, msg.SearchString, msg.CompletionValue, 1) m.textarea.SetValue(modifiedValue) return m, nil case SessionSelectedMsg: if msg.ID != m.session.ID { m.session = msg } return m, nil case dialog.AttachmentAddedMsg: if len(m.attachments) >= maxAttachments { logging.ErrorPersist(fmt.Sprintf("cannot add more than %d images", maxAttachments)) return m, cmd } m.attachments = append(m.attachments, msg.Attachment) case tea.KeyMsg: if key.Matches(msg, DeleteKeyMaps.AttachmentDeleteMode) { m.deleteMode = true return m, nil } if key.Matches(msg, DeleteKeyMaps.DeleteAllAttachments) && m.deleteMode { m.deleteMode = false m.attachments = nil return m, nil } if m.deleteMode && len(msg.Runes) > 0 && unicode.IsDigit(msg.Runes[0]) { num := int(msg.Runes[0] - '0') m.deleteMode = false if num < 10 && len(m.attachments) > num { if num == 0 { m.attachments = m.attachments[num+1:] } else { m.attachments = slices.Delete(m.attachments, num, num+1) } return m, nil } } if key.Matches(msg, messageKeys.PageUp) || key.Matches(msg, messageKeys.PageDown) || key.Matches(msg, messageKeys.HalfPageUp) || key.Matches(msg, messageKeys.HalfPageDown) { return m, nil } if key.Matches(msg, editorMaps.OpenEditor) { if m.app.CoderAgent.IsSessionBusy(m.session.ID) { return m, util.ReportWarn("Agent is working, please wait...") } return m, m.openEditor() } if key.Matches(msg, DeleteKeyMaps.Escape) { m.deleteMode = false return m, nil } // Hanlde Enter key if m.textarea.Focused() && key.Matches(msg, editorMaps.Send) { value := m.textarea.Value() if len(value) > 0 && value[len(value)-1] == '\\' { // If the last character is a backslash, remove it and add a newline m.textarea.SetValue(value[:len(value)-1] + "\n") return m, nil } else { // Otherwise, send the message return m, m.send() } } } m.textarea, cmd = m.textarea.Update(msg) return m, cmd } func (m *editorCmp) View() string { t := theme.CurrentTheme() // Style the prompt with theme colors style := lipgloss.NewStyle(). Padding(0, 0, 0, 1). Bold(true). Foreground(t.Primary()) if len(m.attachments) == 0 { return lipgloss.JoinHorizontal(lipgloss.Top, style.Render(">"), m.textarea.View()) } m.textarea.SetHeight(m.height - 1) return lipgloss.JoinVertical(lipgloss.Top, m.attachmentsContent(), lipgloss.JoinHorizontal(lipgloss.Top, style.Render(">"), m.textarea.View()), ) } func (m *editorCmp) SetSize(width, height int) tea.Cmd { m.width = width m.height = height m.textarea.SetWidth(width - 3) // account for the prompt and padding right m.textarea.SetHeight(height) m.textarea.SetWidth(width) return nil } func (m *editorCmp) GetSize() (int, int) { return m.textarea.Width(), m.textarea.Height() } func (m *editorCmp) attachmentsContent() string { var styledAttachments []string t := theme.CurrentTheme() attachmentStyles := styles.BaseStyle(). MarginLeft(1). Background(t.TextMuted()). Foreground(t.Text()) for i, attachment := range m.attachments { var filename string if len(attachment.FileName) > 10 { filename = fmt.Sprintf(" %s %s...", styles.DocumentIcon, attachment.FileName[0:7]) } else { filename = fmt.Sprintf(" %s %s", styles.DocumentIcon, attachment.FileName) } if m.deleteMode { filename = fmt.Sprintf("%d%s", i, filename) } styledAttachments = append(styledAttachments, attachmentStyles.Render(filename)) } content := lipgloss.JoinHorizontal(lipgloss.Left, styledAttachments...) return content } func (m *editorCmp) BindingKeys() []key.Binding { bindings := []key.Binding{} bindings = append(bindings, layout.KeyMapToSlice(editorMaps)...) bindings = append(bindings, layout.KeyMapToSlice(DeleteKeyMaps)...) return bindings } func CreateTextArea(existing *textarea.Model) textarea.Model { t := theme.CurrentTheme() bgColor := t.Background() textColor := t.Text() textMutedColor := t.TextMuted() ta := textarea.New() ta.BlurredStyle.Base = styles.BaseStyle().Background(bgColor).Foreground(textColor) ta.BlurredStyle.CursorLine = styles.BaseStyle().Background(bgColor) ta.BlurredStyle.Placeholder = styles.BaseStyle().Background(bgColor).Foreground(textMutedColor) ta.BlurredStyle.Text = styles.BaseStyle().Background(bgColor).Foreground(textColor) ta.FocusedStyle.Base = styles.BaseStyle().Background(bgColor).Foreground(textColor) ta.FocusedStyle.CursorLine = styles.BaseStyle().Background(bgColor) ta.FocusedStyle.Placeholder = styles.BaseStyle().Background(bgColor).Foreground(textMutedColor) ta.FocusedStyle.Text = styles.BaseStyle().Background(bgColor).Foreground(textColor) ta.Prompt = " " ta.ShowLineNumbers = false ta.CharLimit = -1 if existing != nil { ta.SetValue(existing.Value()) ta.SetWidth(existing.Width()) ta.SetHeight(existing.Height()) } ta.Focus() return ta } func NewEditorCmp(app *app.App) tea.Model { ta := CreateTextArea(nil) return &editorCmp{ app: app, textarea: ta, } } ================================================ FILE: internal/tui/components/chat/list.go ================================================ package chat import ( "context" "fmt" "math" "github.com/charmbracelet/bubbles/key" "github.com/charmbracelet/bubbles/spinner" "github.com/charmbracelet/bubbles/viewport" tea "github.com/charmbracelet/bubbletea" "github.com/charmbracelet/lipgloss" "github.com/opencode-ai/opencode/internal/app" "github.com/opencode-ai/opencode/internal/message" "github.com/opencode-ai/opencode/internal/pubsub" "github.com/opencode-ai/opencode/internal/session" "github.com/opencode-ai/opencode/internal/tui/components/dialog" "github.com/opencode-ai/opencode/internal/tui/styles" "github.com/opencode-ai/opencode/internal/tui/theme" "github.com/opencode-ai/opencode/internal/tui/util" ) type cacheItem struct { width int content []uiMessage } type messagesCmp struct { app *app.App width, height int viewport viewport.Model session session.Session messages []message.Message uiMessages []uiMessage currentMsgID string cachedContent map[string]cacheItem spinner spinner.Model rendering bool attachments viewport.Model } type renderFinishedMsg struct{} type MessageKeys struct { PageDown key.Binding PageUp key.Binding HalfPageUp key.Binding HalfPageDown key.Binding } var messageKeys = MessageKeys{ PageDown: key.NewBinding( key.WithKeys("pgdown"), key.WithHelp("f/pgdn", "page down"), ), PageUp: key.NewBinding( key.WithKeys("pgup"), key.WithHelp("b/pgup", "page up"), ), HalfPageUp: key.NewBinding( key.WithKeys("ctrl+u"), key.WithHelp("ctrl+u", "½ page up"), ), HalfPageDown: key.NewBinding( key.WithKeys("ctrl+d", "ctrl+d"), key.WithHelp("ctrl+d", "½ page down"), ), } func (m *messagesCmp) Init() tea.Cmd { return tea.Batch(m.viewport.Init(), m.spinner.Tick) } func (m *messagesCmp) Update(msg tea.Msg) (tea.Model, tea.Cmd) { var cmds []tea.Cmd switch msg := msg.(type) { case dialog.ThemeChangedMsg: m.rerender() return m, nil case SessionSelectedMsg: if msg.ID != m.session.ID { cmd := m.SetSession(msg) return m, cmd } return m, nil case SessionClearedMsg: m.session = session.Session{} m.messages = make([]message.Message, 0) m.currentMsgID = "" m.rendering = false return m, nil case tea.KeyMsg: if key.Matches(msg, messageKeys.PageUp) || key.Matches(msg, messageKeys.PageDown) || key.Matches(msg, messageKeys.HalfPageUp) || key.Matches(msg, messageKeys.HalfPageDown) { u, cmd := m.viewport.Update(msg) m.viewport = u cmds = append(cmds, cmd) } case renderFinishedMsg: m.rendering = false m.viewport.GotoBottom() case pubsub.Event[session.Session]: if msg.Type == pubsub.UpdatedEvent && msg.Payload.ID == m.session.ID { m.session = msg.Payload if m.session.SummaryMessageID == m.currentMsgID { delete(m.cachedContent, m.currentMsgID) m.renderView() } } case pubsub.Event[message.Message]: needsRerender := false if msg.Type == pubsub.CreatedEvent { if msg.Payload.SessionID == m.session.ID { messageExists := false for _, v := range m.messages { if v.ID == msg.Payload.ID { messageExists = true break } } if !messageExists { if len(m.messages) > 0 { lastMsgID := m.messages[len(m.messages)-1].ID delete(m.cachedContent, lastMsgID) } m.messages = append(m.messages, msg.Payload) delete(m.cachedContent, m.currentMsgID) m.currentMsgID = msg.Payload.ID needsRerender = true } } // There are tool calls from the child task for _, v := range m.messages { for _, c := range v.ToolCalls() { if c.ID == msg.Payload.SessionID { delete(m.cachedContent, v.ID) needsRerender = true } } } } else if msg.Type == pubsub.UpdatedEvent && msg.Payload.SessionID == m.session.ID { for i, v := range m.messages { if v.ID == msg.Payload.ID { m.messages[i] = msg.Payload delete(m.cachedContent, msg.Payload.ID) needsRerender = true break } } } if needsRerender { m.renderView() if len(m.messages) > 0 { if (msg.Type == pubsub.CreatedEvent) || (msg.Type == pubsub.UpdatedEvent && msg.Payload.ID == m.messages[len(m.messages)-1].ID) { m.viewport.GotoBottom() } } } } spinner, cmd := m.spinner.Update(msg) m.spinner = spinner cmds = append(cmds, cmd) return m, tea.Batch(cmds...) } func (m *messagesCmp) IsAgentWorking() bool { return m.app.CoderAgent.IsSessionBusy(m.session.ID) } func formatTimeDifference(unixTime1, unixTime2 int64) string { diffSeconds := float64(math.Abs(float64(unixTime2 - unixTime1))) if diffSeconds < 60 { return fmt.Sprintf("%.1fs", diffSeconds) } minutes := int(diffSeconds / 60) seconds := int(diffSeconds) % 60 return fmt.Sprintf("%dm%ds", minutes, seconds) } func (m *messagesCmp) renderView() { m.uiMessages = make([]uiMessage, 0) pos := 0 baseStyle := styles.BaseStyle() if m.width == 0 { return } for inx, msg := range m.messages { switch msg.Role { case message.User: if cache, ok := m.cachedContent[msg.ID]; ok && cache.width == m.width { m.uiMessages = append(m.uiMessages, cache.content...) continue } userMsg := renderUserMessage( msg, msg.ID == m.currentMsgID, m.width, pos, ) m.uiMessages = append(m.uiMessages, userMsg) m.cachedContent[msg.ID] = cacheItem{ width: m.width, content: []uiMessage{userMsg}, } pos += userMsg.height + 1 // + 1 for spacing case message.Assistant: if cache, ok := m.cachedContent[msg.ID]; ok && cache.width == m.width { m.uiMessages = append(m.uiMessages, cache.content...) continue } isSummary := m.session.SummaryMessageID == msg.ID assistantMessages := renderAssistantMessage( msg, inx, m.messages, m.app.Messages, m.currentMsgID, isSummary, m.width, pos, ) for _, msg := range assistantMessages { m.uiMessages = append(m.uiMessages, msg) pos += msg.height + 1 // + 1 for spacing } m.cachedContent[msg.ID] = cacheItem{ width: m.width, content: assistantMessages, } } } messages := make([]string, 0) for _, v := range m.uiMessages { messages = append(messages, lipgloss.JoinVertical(lipgloss.Left, v.content), baseStyle. Width(m.width). Render( "", ), ) } m.viewport.SetContent( baseStyle. Width(m.width). Render( lipgloss.JoinVertical( lipgloss.Top, messages..., ), ), ) } func (m *messagesCmp) View() string { baseStyle := styles.BaseStyle() if m.rendering { return baseStyle. Width(m.width). Render( lipgloss.JoinVertical( lipgloss.Top, "Loading...", m.working(), m.help(), ), ) } if len(m.messages) == 0 { content := baseStyle. Width(m.width). Height(m.height - 1). Render( m.initialScreen(), ) return baseStyle. Width(m.width). Render( lipgloss.JoinVertical( lipgloss.Top, content, "", m.help(), ), ) } return baseStyle. Width(m.width). Render( lipgloss.JoinVertical( lipgloss.Top, m.viewport.View(), m.working(), m.help(), ), ) } func hasToolsWithoutResponse(messages []message.Message) bool { toolCalls := make([]message.ToolCall, 0) toolResults := make([]message.ToolResult, 0) for _, m := range messages { toolCalls = append(toolCalls, m.ToolCalls()...) toolResults = append(toolResults, m.ToolResults()...) } for _, v := range toolCalls { found := false for _, r := range toolResults { if v.ID == r.ToolCallID { found = true break } } if !found && v.Finished { return true } } return false } func hasUnfinishedToolCalls(messages []message.Message) bool { toolCalls := make([]message.ToolCall, 0) for _, m := range messages { toolCalls = append(toolCalls, m.ToolCalls()...) } for _, v := range toolCalls { if !v.Finished { return true } } return false } func (m *messagesCmp) working() string { text := "" if m.IsAgentWorking() && len(m.messages) > 0 { t := theme.CurrentTheme() baseStyle := styles.BaseStyle() task := "Thinking..." lastMessage := m.messages[len(m.messages)-1] if hasToolsWithoutResponse(m.messages) { task = "Waiting for tool response..." } else if hasUnfinishedToolCalls(m.messages) { task = "Building tool call..." } else if !lastMessage.IsFinished() { task = "Generating..." } if task != "" { text += baseStyle. Width(m.width). Foreground(t.Primary()). Bold(true). Render(fmt.Sprintf("%s %s ", m.spinner.View(), task)) } } return text } func (m *messagesCmp) help() string { t := theme.CurrentTheme() baseStyle := styles.BaseStyle() text := "" if m.app.CoderAgent.IsBusy() { text += lipgloss.JoinHorizontal( lipgloss.Left, baseStyle.Foreground(t.TextMuted()).Bold(true).Render("press "), baseStyle.Foreground(t.Text()).Bold(true).Render("esc"), baseStyle.Foreground(t.TextMuted()).Bold(true).Render(" to exit cancel"), ) } else { text += lipgloss.JoinHorizontal( lipgloss.Left, baseStyle.Foreground(t.TextMuted()).Bold(true).Render("press "), baseStyle.Foreground(t.Text()).Bold(true).Render("enter"), baseStyle.Foreground(t.TextMuted()).Bold(true).Render(" to send the message,"), baseStyle.Foreground(t.TextMuted()).Bold(true).Render(" write"), baseStyle.Foreground(t.Text()).Bold(true).Render(" \\"), baseStyle.Foreground(t.TextMuted()).Bold(true).Render(" and enter to add a new line"), ) } return baseStyle. Width(m.width). Render(text) } func (m *messagesCmp) initialScreen() string { baseStyle := styles.BaseStyle() return baseStyle.Width(m.width).Render( lipgloss.JoinVertical( lipgloss.Top, header(m.width), "", lspsConfigured(m.width), ), ) } func (m *messagesCmp) rerender() { for _, msg := range m.messages { delete(m.cachedContent, msg.ID) } m.renderView() } func (m *messagesCmp) SetSize(width, height int) tea.Cmd { if m.width == width && m.height == height { return nil } m.width = width m.height = height m.viewport.Width = width m.viewport.Height = height - 2 m.attachments.Width = width + 40 m.attachments.Height = 3 m.rerender() return nil } func (m *messagesCmp) GetSize() (int, int) { return m.width, m.height } func (m *messagesCmp) SetSession(session session.Session) tea.Cmd { if m.session.ID == session.ID { return nil } m.session = session messages, err := m.app.Messages.List(context.Background(), session.ID) if err != nil { return util.ReportError(err) } m.messages = messages if len(m.messages) > 0 { m.currentMsgID = m.messages[len(m.messages)-1].ID } delete(m.cachedContent, m.currentMsgID) m.rendering = true return func() tea.Msg { m.renderView() return renderFinishedMsg{} } } func (m *messagesCmp) BindingKeys() []key.Binding { return []key.Binding{ m.viewport.KeyMap.PageDown, m.viewport.KeyMap.PageUp, m.viewport.KeyMap.HalfPageUp, m.viewport.KeyMap.HalfPageDown, } } func NewMessagesCmp(app *app.App) tea.Model { s := spinner.New() s.Spinner = spinner.Pulse vp := viewport.New(0, 0) attachmets := viewport.New(0, 0) vp.KeyMap.PageUp = messageKeys.PageUp vp.KeyMap.PageDown = messageKeys.PageDown vp.KeyMap.HalfPageUp = messageKeys.HalfPageUp vp.KeyMap.HalfPageDown = messageKeys.HalfPageDown return &messagesCmp{ app: app, cachedContent: make(map[string]cacheItem), viewport: vp, spinner: s, attachments: attachmets, } } ================================================ FILE: internal/tui/components/chat/message.go ================================================ package chat import ( "context" "encoding/json" "fmt" "path/filepath" "strings" "time" "github.com/charmbracelet/lipgloss" "github.com/charmbracelet/x/ansi" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/diff" "github.com/opencode-ai/opencode/internal/llm/agent" "github.com/opencode-ai/opencode/internal/llm/models" "github.com/opencode-ai/opencode/internal/llm/tools" "github.com/opencode-ai/opencode/internal/message" "github.com/opencode-ai/opencode/internal/tui/styles" "github.com/opencode-ai/opencode/internal/tui/theme" ) type uiMessageType int const ( userMessageType uiMessageType = iota assistantMessageType toolMessageType maxResultHeight = 10 ) type uiMessage struct { ID string messageType uiMessageType position int height int content string } func toMarkdown(content string, focused bool, width int) string { r := styles.GetMarkdownRenderer(width) rendered, _ := r.Render(content) return rendered } func renderMessage(msg string, isUser bool, isFocused bool, width int, info ...string) string { t := theme.CurrentTheme() style := styles.BaseStyle(). Width(width - 1). BorderLeft(true). Foreground(t.TextMuted()). BorderForeground(t.Primary()). BorderStyle(lipgloss.ThickBorder()) if isUser { style = style.BorderForeground(t.Secondary()) } // Apply markdown formatting and handle background color parts := []string{ styles.ForceReplaceBackgroundWithLipgloss(toMarkdown(msg, isFocused, width), t.Background()), } // Remove newline at the end parts[0] = strings.TrimSuffix(parts[0], "\n") if len(info) > 0 { parts = append(parts, info...) } rendered := style.Render( lipgloss.JoinVertical( lipgloss.Left, parts..., ), ) return rendered } func renderUserMessage(msg message.Message, isFocused bool, width int, position int) uiMessage { var styledAttachments []string t := theme.CurrentTheme() attachmentStyles := styles.BaseStyle(). MarginLeft(1). Background(t.TextMuted()). Foreground(t.Text()) for _, attachment := range msg.BinaryContent() { file := filepath.Base(attachment.Path) var filename string if len(file) > 10 { filename = fmt.Sprintf(" %s %s...", styles.DocumentIcon, file[0:7]) } else { filename = fmt.Sprintf(" %s %s", styles.DocumentIcon, file) } styledAttachments = append(styledAttachments, attachmentStyles.Render(filename)) } content := "" if len(styledAttachments) > 0 { attachmentContent := styles.BaseStyle().Width(width).Render(lipgloss.JoinHorizontal(lipgloss.Left, styledAttachments...)) content = renderMessage(msg.Content().String(), true, isFocused, width, attachmentContent) } else { content = renderMessage(msg.Content().String(), true, isFocused, width) } userMsg := uiMessage{ ID: msg.ID, messageType: userMessageType, position: position, height: lipgloss.Height(content), content: content, } return userMsg } // Returns multiple uiMessages because of the tool calls func renderAssistantMessage( msg message.Message, msgIndex int, allMessages []message.Message, // we need this to get tool results and the user message messagesService message.Service, // We need this to get the task tool messages focusedUIMessageId string, isSummary bool, width int, position int, ) []uiMessage { messages := []uiMessage{} content := msg.Content().String() thinking := msg.IsThinking() thinkingContent := msg.ReasoningContent().Thinking finished := msg.IsFinished() finishData := msg.FinishPart() info := []string{} t := theme.CurrentTheme() baseStyle := styles.BaseStyle() // Add finish info if available if finished { switch finishData.Reason { case message.FinishReasonEndTurn: took := formatTimestampDiff(msg.CreatedAt, finishData.Time) info = append(info, baseStyle. Width(width-1). Foreground(t.TextMuted()). Render(fmt.Sprintf(" %s (%s)", models.SupportedModels[msg.Model].Name, took)), ) case message.FinishReasonCanceled: info = append(info, baseStyle. Width(width-1). Foreground(t.TextMuted()). Render(fmt.Sprintf(" %s (%s)", models.SupportedModels[msg.Model].Name, "canceled")), ) case message.FinishReasonError: info = append(info, baseStyle. Width(width-1). Foreground(t.TextMuted()). Render(fmt.Sprintf(" %s (%s)", models.SupportedModels[msg.Model].Name, "error")), ) case message.FinishReasonPermissionDenied: info = append(info, baseStyle. Width(width-1). Foreground(t.TextMuted()). Render(fmt.Sprintf(" %s (%s)", models.SupportedModels[msg.Model].Name, "permission denied")), ) } } if content != "" || (finished && finishData.Reason == message.FinishReasonEndTurn) { if content == "" { content = "*Finished without output*" } if isSummary { info = append(info, baseStyle.Width(width-1).Foreground(t.TextMuted()).Render(" (summary)")) } content = renderMessage(content, false, true, width, info...) messages = append(messages, uiMessage{ ID: msg.ID, messageType: assistantMessageType, position: position, height: lipgloss.Height(content), content: content, }) position += messages[0].height position++ // for the space } else if thinking && thinkingContent != "" { // Render the thinking content content = renderMessage(thinkingContent, false, msg.ID == focusedUIMessageId, width) } for i, toolCall := range msg.ToolCalls() { toolCallContent := renderToolMessage( toolCall, allMessages, messagesService, focusedUIMessageId, false, width, i+1, ) messages = append(messages, toolCallContent) position += toolCallContent.height position++ // for the space } return messages } func findToolResponse(toolCallID string, futureMessages []message.Message) *message.ToolResult { for _, msg := range futureMessages { for _, result := range msg.ToolResults() { if result.ToolCallID == toolCallID { return &result } } } return nil } func toolName(name string) string { switch name { case agent.AgentToolName: return "Task" case tools.BashToolName: return "Bash" case tools.EditToolName: return "Edit" case tools.FetchToolName: return "Fetch" case tools.GlobToolName: return "Glob" case tools.GrepToolName: return "Grep" case tools.LSToolName: return "List" case tools.SourcegraphToolName: return "Sourcegraph" case tools.ViewToolName: return "View" case tools.WriteToolName: return "Write" case tools.PatchToolName: return "Patch" } return name } func getToolAction(name string) string { switch name { case agent.AgentToolName: return "Preparing prompt..." case tools.BashToolName: return "Building command..." case tools.EditToolName: return "Preparing edit..." case tools.FetchToolName: return "Writing fetch..." case tools.GlobToolName: return "Finding files..." case tools.GrepToolName: return "Searching content..." case tools.LSToolName: return "Listing directory..." case tools.SourcegraphToolName: return "Searching code..." case tools.ViewToolName: return "Reading file..." case tools.WriteToolName: return "Preparing write..." case tools.PatchToolName: return "Preparing patch..." } return "Working..." } // renders params, params[0] (params[1]=params[2] ....) func renderParams(paramsWidth int, params ...string) string { if len(params) == 0 { return "" } mainParam := params[0] if len(mainParam) > paramsWidth { mainParam = mainParam[:paramsWidth-3] + "..." } if len(params) == 1 { return mainParam } otherParams := params[1:] // create pairs of key/value // if odd number of params, the last one is a key without value if len(otherParams)%2 != 0 { otherParams = append(otherParams, "") } parts := make([]string, 0, len(otherParams)/2) for i := 0; i < len(otherParams); i += 2 { key := otherParams[i] value := otherParams[i+1] if value == "" { continue } parts = append(parts, fmt.Sprintf("%s=%s", key, value)) } partsRendered := strings.Join(parts, ", ") remainingWidth := paramsWidth - lipgloss.Width(partsRendered) - 5 // for the space if remainingWidth < 30 { // No space for the params, just show the main return mainParam } if len(parts) > 0 { mainParam = fmt.Sprintf("%s (%s)", mainParam, strings.Join(parts, ", ")) } return ansi.Truncate(mainParam, paramsWidth, "...") } func removeWorkingDirPrefix(path string) string { wd := config.WorkingDirectory() if strings.HasPrefix(path, wd) { path = strings.TrimPrefix(path, wd) } if strings.HasPrefix(path, "/") { path = strings.TrimPrefix(path, "/") } if strings.HasPrefix(path, "./") { path = strings.TrimPrefix(path, "./") } if strings.HasPrefix(path, "../") { path = strings.TrimPrefix(path, "../") } return path } func renderToolParams(paramWidth int, toolCall message.ToolCall) string { params := "" switch toolCall.Name { case agent.AgentToolName: var params agent.AgentParams json.Unmarshal([]byte(toolCall.Input), ¶ms) prompt := strings.ReplaceAll(params.Prompt, "\n", " ") return renderParams(paramWidth, prompt) case tools.BashToolName: var params tools.BashParams json.Unmarshal([]byte(toolCall.Input), ¶ms) command := strings.ReplaceAll(params.Command, "\n", " ") return renderParams(paramWidth, command) case tools.EditToolName: var params tools.EditParams json.Unmarshal([]byte(toolCall.Input), ¶ms) filePath := removeWorkingDirPrefix(params.FilePath) return renderParams(paramWidth, filePath) case tools.FetchToolName: var params tools.FetchParams json.Unmarshal([]byte(toolCall.Input), ¶ms) url := params.URL toolParams := []string{ url, } if params.Format != "" { toolParams = append(toolParams, "format", params.Format) } if params.Timeout != 0 { toolParams = append(toolParams, "timeout", (time.Duration(params.Timeout) * time.Second).String()) } return renderParams(paramWidth, toolParams...) case tools.GlobToolName: var params tools.GlobParams json.Unmarshal([]byte(toolCall.Input), ¶ms) pattern := params.Pattern toolParams := []string{ pattern, } if params.Path != "" { toolParams = append(toolParams, "path", params.Path) } return renderParams(paramWidth, toolParams...) case tools.GrepToolName: var params tools.GrepParams json.Unmarshal([]byte(toolCall.Input), ¶ms) pattern := params.Pattern toolParams := []string{ pattern, } if params.Path != "" { toolParams = append(toolParams, "path", params.Path) } if params.Include != "" { toolParams = append(toolParams, "include", params.Include) } if params.LiteralText { toolParams = append(toolParams, "literal", "true") } return renderParams(paramWidth, toolParams...) case tools.LSToolName: var params tools.LSParams json.Unmarshal([]byte(toolCall.Input), ¶ms) path := params.Path if path == "" { path = "." } return renderParams(paramWidth, path) case tools.SourcegraphToolName: var params tools.SourcegraphParams json.Unmarshal([]byte(toolCall.Input), ¶ms) return renderParams(paramWidth, params.Query) case tools.ViewToolName: var params tools.ViewParams json.Unmarshal([]byte(toolCall.Input), ¶ms) filePath := removeWorkingDirPrefix(params.FilePath) toolParams := []string{ filePath, } if params.Limit != 0 { toolParams = append(toolParams, "limit", fmt.Sprintf("%d", params.Limit)) } if params.Offset != 0 { toolParams = append(toolParams, "offset", fmt.Sprintf("%d", params.Offset)) } return renderParams(paramWidth, toolParams...) case tools.WriteToolName: var params tools.WriteParams json.Unmarshal([]byte(toolCall.Input), ¶ms) filePath := removeWorkingDirPrefix(params.FilePath) return renderParams(paramWidth, filePath) default: input := strings.ReplaceAll(toolCall.Input, "\n", " ") params = renderParams(paramWidth, input) } return params } func truncateHeight(content string, height int) string { lines := strings.Split(content, "\n") if len(lines) > height { return strings.Join(lines[:height], "\n") } return content } func renderToolResponse(toolCall message.ToolCall, response message.ToolResult, width int) string { t := theme.CurrentTheme() baseStyle := styles.BaseStyle() if response.IsError { errContent := fmt.Sprintf("Error: %s", strings.ReplaceAll(response.Content, "\n", " ")) errContent = ansi.Truncate(errContent, width-1, "...") return baseStyle. Width(width). Foreground(t.Error()). Render(errContent) } resultContent := truncateHeight(response.Content, maxResultHeight) switch toolCall.Name { case agent.AgentToolName: return styles.ForceReplaceBackgroundWithLipgloss( toMarkdown(resultContent, false, width), t.Background(), ) case tools.BashToolName: resultContent = fmt.Sprintf("```bash\n%s\n```", resultContent) return styles.ForceReplaceBackgroundWithLipgloss( toMarkdown(resultContent, true, width), t.Background(), ) case tools.EditToolName: metadata := tools.EditResponseMetadata{} json.Unmarshal([]byte(response.Metadata), &metadata) truncDiff := truncateHeight(metadata.Diff, maxResultHeight) formattedDiff, _ := diff.FormatDiff(truncDiff, diff.WithTotalWidth(width)) return formattedDiff case tools.FetchToolName: var params tools.FetchParams json.Unmarshal([]byte(toolCall.Input), ¶ms) mdFormat := "markdown" switch params.Format { case "text": mdFormat = "text" case "html": mdFormat = "html" } resultContent = fmt.Sprintf("```%s\n%s\n```", mdFormat, resultContent) return styles.ForceReplaceBackgroundWithLipgloss( toMarkdown(resultContent, true, width), t.Background(), ) case tools.GlobToolName: return baseStyle.Width(width).Foreground(t.TextMuted()).Render(resultContent) case tools.GrepToolName: return baseStyle.Width(width).Foreground(t.TextMuted()).Render(resultContent) case tools.LSToolName: return baseStyle.Width(width).Foreground(t.TextMuted()).Render(resultContent) case tools.SourcegraphToolName: return baseStyle.Width(width).Foreground(t.TextMuted()).Render(resultContent) case tools.ViewToolName: metadata := tools.ViewResponseMetadata{} json.Unmarshal([]byte(response.Metadata), &metadata) ext := filepath.Ext(metadata.FilePath) if ext == "" { ext = "" } else { ext = strings.ToLower(ext[1:]) } resultContent = fmt.Sprintf("```%s\n%s\n```", ext, truncateHeight(metadata.Content, maxResultHeight)) return styles.ForceReplaceBackgroundWithLipgloss( toMarkdown(resultContent, true, width), t.Background(), ) case tools.WriteToolName: params := tools.WriteParams{} json.Unmarshal([]byte(toolCall.Input), ¶ms) metadata := tools.WriteResponseMetadata{} json.Unmarshal([]byte(response.Metadata), &metadata) ext := filepath.Ext(params.FilePath) if ext == "" { ext = "" } else { ext = strings.ToLower(ext[1:]) } resultContent = fmt.Sprintf("```%s\n%s\n```", ext, truncateHeight(params.Content, maxResultHeight)) return styles.ForceReplaceBackgroundWithLipgloss( toMarkdown(resultContent, true, width), t.Background(), ) default: resultContent = fmt.Sprintf("```text\n%s\n```", resultContent) return styles.ForceReplaceBackgroundWithLipgloss( toMarkdown(resultContent, true, width), t.Background(), ) } } func renderToolMessage( toolCall message.ToolCall, allMessages []message.Message, messagesService message.Service, focusedUIMessageId string, nested bool, width int, position int, ) uiMessage { if nested { width = width - 3 } t := theme.CurrentTheme() baseStyle := styles.BaseStyle() style := baseStyle. Width(width - 1). BorderLeft(true). BorderStyle(lipgloss.ThickBorder()). PaddingLeft(1). BorderForeground(t.TextMuted()) response := findToolResponse(toolCall.ID, allMessages) toolNameText := baseStyle.Foreground(t.TextMuted()). Render(fmt.Sprintf("%s: ", toolName(toolCall.Name))) if !toolCall.Finished { // Get a brief description of what the tool is doing toolAction := getToolAction(toolCall.Name) progressText := baseStyle. Width(width - 2 - lipgloss.Width(toolNameText)). Foreground(t.TextMuted()). Render(fmt.Sprintf("%s", toolAction)) content := style.Render(lipgloss.JoinHorizontal(lipgloss.Left, toolNameText, progressText)) toolMsg := uiMessage{ messageType: toolMessageType, position: position, height: lipgloss.Height(content), content: content, } return toolMsg } params := renderToolParams(width-2-lipgloss.Width(toolNameText), toolCall) responseContent := "" if response != nil { responseContent = renderToolResponse(toolCall, *response, width-2) responseContent = strings.TrimSuffix(responseContent, "\n") } else { responseContent = baseStyle. Italic(true). Width(width - 2). Foreground(t.TextMuted()). Render("Waiting for response...") } parts := []string{} if !nested { formattedParams := baseStyle. Width(width - 2 - lipgloss.Width(toolNameText)). Foreground(t.TextMuted()). Render(params) parts = append(parts, lipgloss.JoinHorizontal(lipgloss.Left, toolNameText, formattedParams)) } else { prefix := baseStyle. Foreground(t.TextMuted()). Render(" └ ") formattedParams := baseStyle. Width(width - 2 - lipgloss.Width(toolNameText)). Foreground(t.TextMuted()). Render(params) parts = append(parts, lipgloss.JoinHorizontal(lipgloss.Left, prefix, toolNameText, formattedParams)) } if toolCall.Name == agent.AgentToolName { taskMessages, _ := messagesService.List(context.Background(), toolCall.ID) toolCalls := []message.ToolCall{} for _, v := range taskMessages { toolCalls = append(toolCalls, v.ToolCalls()...) } for _, call := range toolCalls { rendered := renderToolMessage(call, []message.Message{}, messagesService, focusedUIMessageId, true, width, 0) parts = append(parts, rendered.content) } } if responseContent != "" && !nested { parts = append(parts, responseContent) } content := style.Render( lipgloss.JoinVertical( lipgloss.Left, parts..., ), ) if nested { content = lipgloss.JoinVertical( lipgloss.Left, parts..., ) } toolMsg := uiMessage{ messageType: toolMessageType, position: position, height: lipgloss.Height(content), content: content, } return toolMsg } // Helper function to format the time difference between two Unix timestamps func formatTimestampDiff(start, end int64) string { diffSeconds := float64(end-start) / 1000.0 // Convert to seconds if diffSeconds < 1 { return fmt.Sprintf("%dms", int(diffSeconds*1000)) } if diffSeconds < 60 { return fmt.Sprintf("%.1fs", diffSeconds) } return fmt.Sprintf("%.1fm", diffSeconds/60) } ================================================ FILE: internal/tui/components/chat/sidebar.go ================================================ package chat import ( "context" "fmt" "sort" "strings" tea "github.com/charmbracelet/bubbletea" "github.com/charmbracelet/lipgloss" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/diff" "github.com/opencode-ai/opencode/internal/history" "github.com/opencode-ai/opencode/internal/pubsub" "github.com/opencode-ai/opencode/internal/session" "github.com/opencode-ai/opencode/internal/tui/styles" "github.com/opencode-ai/opencode/internal/tui/theme" ) type sidebarCmp struct { width, height int session session.Session history history.Service modFiles map[string]struct { additions int removals int } } func (m *sidebarCmp) Init() tea.Cmd { if m.history != nil { ctx := context.Background() // Subscribe to file events filesCh := m.history.Subscribe(ctx) // Initialize the modified files map m.modFiles = make(map[string]struct { additions int removals int }) // Load initial files and calculate diffs m.loadModifiedFiles(ctx) // Return a command that will send file events to the Update method return func() tea.Msg { return <-filesCh } } return nil } func (m *sidebarCmp) Update(msg tea.Msg) (tea.Model, tea.Cmd) { switch msg := msg.(type) { case SessionSelectedMsg: if msg.ID != m.session.ID { m.session = msg ctx := context.Background() m.loadModifiedFiles(ctx) } case pubsub.Event[session.Session]: if msg.Type == pubsub.UpdatedEvent { if m.session.ID == msg.Payload.ID { m.session = msg.Payload } } case pubsub.Event[history.File]: if msg.Payload.SessionID == m.session.ID { // Process the individual file change instead of reloading all files ctx := context.Background() m.processFileChanges(ctx, msg.Payload) // Return a command to continue receiving events return m, func() tea.Msg { ctx := context.Background() filesCh := m.history.Subscribe(ctx) return <-filesCh } } } return m, nil } func (m *sidebarCmp) View() string { baseStyle := styles.BaseStyle() return baseStyle. Width(m.width). PaddingLeft(4). PaddingRight(2). Height(m.height - 1). Render( lipgloss.JoinVertical( lipgloss.Top, header(m.width), " ", m.sessionSection(), " ", lspsConfigured(m.width), " ", m.modifiedFiles(), ), ) } func (m *sidebarCmp) sessionSection() string { t := theme.CurrentTheme() baseStyle := styles.BaseStyle() sessionKey := baseStyle. Foreground(t.Primary()). Bold(true). Render("Session") sessionValue := baseStyle. Foreground(t.Text()). Width(m.width - lipgloss.Width(sessionKey)). Render(fmt.Sprintf(": %s", m.session.Title)) return lipgloss.JoinHorizontal( lipgloss.Left, sessionKey, sessionValue, ) } func (m *sidebarCmp) modifiedFile(filePath string, additions, removals int) string { t := theme.CurrentTheme() baseStyle := styles.BaseStyle() stats := "" if additions > 0 && removals > 0 { additionsStr := baseStyle. Foreground(t.Success()). PaddingLeft(1). Render(fmt.Sprintf("+%d", additions)) removalsStr := baseStyle. Foreground(t.Error()). PaddingLeft(1). Render(fmt.Sprintf("-%d", removals)) content := lipgloss.JoinHorizontal(lipgloss.Left, additionsStr, removalsStr) stats = baseStyle.Width(lipgloss.Width(content)).Render(content) } else if additions > 0 { additionsStr := fmt.Sprintf(" %s", baseStyle. PaddingLeft(1). Foreground(t.Success()). Render(fmt.Sprintf("+%d", additions))) stats = baseStyle.Width(lipgloss.Width(additionsStr)).Render(additionsStr) } else if removals > 0 { removalsStr := fmt.Sprintf(" %s", baseStyle. PaddingLeft(1). Foreground(t.Error()). Render(fmt.Sprintf("-%d", removals))) stats = baseStyle.Width(lipgloss.Width(removalsStr)).Render(removalsStr) } filePathStr := baseStyle.Render(filePath) return baseStyle. Width(m.width). Render( lipgloss.JoinHorizontal( lipgloss.Left, filePathStr, stats, ), ) } func (m *sidebarCmp) modifiedFiles() string { t := theme.CurrentTheme() baseStyle := styles.BaseStyle() modifiedFiles := baseStyle. Width(m.width). Foreground(t.Primary()). Bold(true). Render("Modified Files:") // If no modified files, show a placeholder message if m.modFiles == nil || len(m.modFiles) == 0 { message := "No modified files" remainingWidth := m.width - lipgloss.Width(message) if remainingWidth > 0 { message += strings.Repeat(" ", remainingWidth) } return baseStyle. Width(m.width). Render( lipgloss.JoinVertical( lipgloss.Top, modifiedFiles, baseStyle.Foreground(t.TextMuted()).Render(message), ), ) } // Sort file paths alphabetically for consistent ordering var paths []string for path := range m.modFiles { paths = append(paths, path) } sort.Strings(paths) // Create views for each file in sorted order var fileViews []string for _, path := range paths { stats := m.modFiles[path] fileViews = append(fileViews, m.modifiedFile(path, stats.additions, stats.removals)) } return baseStyle. Width(m.width). Render( lipgloss.JoinVertical( lipgloss.Top, modifiedFiles, lipgloss.JoinVertical( lipgloss.Left, fileViews..., ), ), ) } func (m *sidebarCmp) SetSize(width, height int) tea.Cmd { m.width = width m.height = height return nil } func (m *sidebarCmp) GetSize() (int, int) { return m.width, m.height } func NewSidebarCmp(session session.Session, history history.Service) tea.Model { return &sidebarCmp{ session: session, history: history, } } func (m *sidebarCmp) loadModifiedFiles(ctx context.Context) { if m.history == nil || m.session.ID == "" { return } // Get all latest files for this session latestFiles, err := m.history.ListLatestSessionFiles(ctx, m.session.ID) if err != nil { return } // Get all files for this session (to find initial versions) allFiles, err := m.history.ListBySession(ctx, m.session.ID) if err != nil { return } // Clear the existing map to rebuild it m.modFiles = make(map[string]struct { additions int removals int }) // Process each latest file for _, file := range latestFiles { // Skip if this is the initial version (no changes to show) if file.Version == history.InitialVersion { continue } // Find the initial version for this specific file var initialVersion history.File for _, v := range allFiles { if v.Path == file.Path && v.Version == history.InitialVersion { initialVersion = v break } } // Skip if we can't find the initial version if initialVersion.ID == "" { continue } if initialVersion.Content == file.Content { continue } // Calculate diff between initial and latest version _, additions, removals := diff.GenerateDiff(initialVersion.Content, file.Content, file.Path) // Only add to modified files if there are changes if additions > 0 || removals > 0 { // Remove working directory prefix from file path displayPath := file.Path workingDir := config.WorkingDirectory() displayPath = strings.TrimPrefix(displayPath, workingDir) displayPath = strings.TrimPrefix(displayPath, "/") m.modFiles[displayPath] = struct { additions int removals int }{ additions: additions, removals: removals, } } } } func (m *sidebarCmp) processFileChanges(ctx context.Context, file history.File) { // Skip if this is the initial version (no changes to show) if file.Version == history.InitialVersion { return } // Find the initial version for this file initialVersion, err := m.findInitialVersion(ctx, file.Path) if err != nil || initialVersion.ID == "" { return } // Skip if content hasn't changed if initialVersion.Content == file.Content { // If this file was previously modified but now matches the initial version, // remove it from the modified files list displayPath := getDisplayPath(file.Path) delete(m.modFiles, displayPath) return } // Calculate diff between initial and latest version _, additions, removals := diff.GenerateDiff(initialVersion.Content, file.Content, file.Path) // Only add to modified files if there are changes if additions > 0 || removals > 0 { displayPath := getDisplayPath(file.Path) m.modFiles[displayPath] = struct { additions int removals int }{ additions: additions, removals: removals, } } else { // If no changes, remove from modified files displayPath := getDisplayPath(file.Path) delete(m.modFiles, displayPath) } } // Helper function to find the initial version of a file func (m *sidebarCmp) findInitialVersion(ctx context.Context, path string) (history.File, error) { // Get all versions of this file for the session fileVersions, err := m.history.ListBySession(ctx, m.session.ID) if err != nil { return history.File{}, err } // Find the initial version for _, v := range fileVersions { if v.Path == path && v.Version == history.InitialVersion { return v, nil } } return history.File{}, fmt.Errorf("initial version not found") } // Helper function to get the display path for a file func getDisplayPath(path string) string { workingDir := config.WorkingDirectory() displayPath := strings.TrimPrefix(path, workingDir) return strings.TrimPrefix(displayPath, "/") } ================================================ FILE: internal/tui/components/core/status.go ================================================ package core import ( "fmt" "strings" "time" tea "github.com/charmbracelet/bubbletea" "github.com/charmbracelet/lipgloss" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/llm/models" "github.com/opencode-ai/opencode/internal/lsp" "github.com/opencode-ai/opencode/internal/lsp/protocol" "github.com/opencode-ai/opencode/internal/pubsub" "github.com/opencode-ai/opencode/internal/session" "github.com/opencode-ai/opencode/internal/tui/components/chat" "github.com/opencode-ai/opencode/internal/tui/styles" "github.com/opencode-ai/opencode/internal/tui/theme" "github.com/opencode-ai/opencode/internal/tui/util" ) type StatusCmp interface { tea.Model } type statusCmp struct { info util.InfoMsg width int messageTTL time.Duration lspClients map[string]*lsp.Client session session.Session } // clearMessageCmd is a command that clears status messages after a timeout func (m statusCmp) clearMessageCmd(ttl time.Duration) tea.Cmd { return tea.Tick(ttl, func(time.Time) tea.Msg { return util.ClearStatusMsg{} }) } func (m statusCmp) Init() tea.Cmd { return nil } func (m statusCmp) Update(msg tea.Msg) (tea.Model, tea.Cmd) { switch msg := msg.(type) { case tea.WindowSizeMsg: m.width = msg.Width return m, nil case chat.SessionSelectedMsg: m.session = msg case chat.SessionClearedMsg: m.session = session.Session{} case pubsub.Event[session.Session]: if msg.Type == pubsub.UpdatedEvent { if m.session.ID == msg.Payload.ID { m.session = msg.Payload } } case util.InfoMsg: m.info = msg ttl := msg.TTL if ttl == 0 { ttl = m.messageTTL } return m, m.clearMessageCmd(ttl) case util.ClearStatusMsg: m.info = util.InfoMsg{} } return m, nil } var helpWidget = "" // getHelpWidget returns the help widget with current theme colors func getHelpWidget() string { t := theme.CurrentTheme() helpText := "ctrl+? help" return styles.Padded(). Background(t.TextMuted()). Foreground(t.BackgroundDarker()). Bold(true). Render(helpText) } func formatTokensAndCost(tokens, contextWindow int64, cost float64) string { // Format tokens in human-readable format (e.g., 110K, 1.2M) var formattedTokens string switch { case tokens >= 1_000_000: formattedTokens = fmt.Sprintf("%.1fM", float64(tokens)/1_000_000) case tokens >= 1_000: formattedTokens = fmt.Sprintf("%.1fK", float64(tokens)/1_000) default: formattedTokens = fmt.Sprintf("%d", tokens) } // Remove .0 suffix if present if strings.HasSuffix(formattedTokens, ".0K") { formattedTokens = strings.Replace(formattedTokens, ".0K", "K", 1) } if strings.HasSuffix(formattedTokens, ".0M") { formattedTokens = strings.Replace(formattedTokens, ".0M", "M", 1) } // Format cost with $ symbol and 2 decimal places formattedCost := fmt.Sprintf("$%.2f", cost) percentage := (float64(tokens) / float64(contextWindow)) * 100 if percentage > 80 { // add the warning icon and percentage formattedTokens = fmt.Sprintf("%s(%d%%)", styles.WarningIcon, int(percentage)) } return fmt.Sprintf("Context: %s, Cost: %s", formattedTokens, formattedCost) } func (m statusCmp) View() string { t := theme.CurrentTheme() modelID := config.Get().Agents[config.AgentCoder].Model model := models.SupportedModels[modelID] // Initialize the help widget status := getHelpWidget() tokenInfoWidth := 0 if m.session.ID != "" { totalTokens := m.session.PromptTokens + m.session.CompletionTokens tokens := formatTokensAndCost(totalTokens, model.ContextWindow, m.session.Cost) tokensStyle := styles.Padded(). Background(t.Text()). Foreground(t.BackgroundSecondary()) percentage := (float64(totalTokens) / float64(model.ContextWindow)) * 100 if percentage > 80 { tokensStyle = tokensStyle.Background(t.Warning()) } tokenInfoWidth = lipgloss.Width(tokens) + 2 status += tokensStyle.Render(tokens) } diagnostics := styles.Padded(). Background(t.BackgroundDarker()). Render(m.projectDiagnostics()) availableWidht := max(0, m.width-lipgloss.Width(helpWidget)-lipgloss.Width(m.model())-lipgloss.Width(diagnostics)-tokenInfoWidth) if m.info.Msg != "" { infoStyle := styles.Padded(). Foreground(t.Background()). Width(availableWidht) switch m.info.Type { case util.InfoTypeInfo: infoStyle = infoStyle.Background(t.Info()) case util.InfoTypeWarn: infoStyle = infoStyle.Background(t.Warning()) case util.InfoTypeError: infoStyle = infoStyle.Background(t.Error()) } infoWidth := availableWidht - 10 // Truncate message if it's longer than available width msg := m.info.Msg if len(msg) > infoWidth && infoWidth > 0 { msg = msg[:infoWidth] + "..." } status += infoStyle.Render(msg) } else { status += styles.Padded(). Foreground(t.Text()). Background(t.BackgroundSecondary()). Width(availableWidht). Render("") } status += diagnostics status += m.model() return status } func (m *statusCmp) projectDiagnostics() string { t := theme.CurrentTheme() // Check if any LSP server is still initializing initializing := false for _, client := range m.lspClients { if client.GetServerState() == lsp.StateStarting { initializing = true break } } // If any server is initializing, show that status if initializing { return lipgloss.NewStyle(). Background(t.BackgroundDarker()). Foreground(t.Warning()). Render(fmt.Sprintf("%s Initializing LSP...", styles.SpinnerIcon)) } errorDiagnostics := []protocol.Diagnostic{} warnDiagnostics := []protocol.Diagnostic{} hintDiagnostics := []protocol.Diagnostic{} infoDiagnostics := []protocol.Diagnostic{} for _, client := range m.lspClients { for _, d := range client.GetDiagnostics() { for _, diag := range d { switch diag.Severity { case protocol.SeverityError: errorDiagnostics = append(errorDiagnostics, diag) case protocol.SeverityWarning: warnDiagnostics = append(warnDiagnostics, diag) case protocol.SeverityHint: hintDiagnostics = append(hintDiagnostics, diag) case protocol.SeverityInformation: infoDiagnostics = append(infoDiagnostics, diag) } } } } if len(errorDiagnostics) == 0 && len(warnDiagnostics) == 0 && len(hintDiagnostics) == 0 && len(infoDiagnostics) == 0 { return "No diagnostics" } diagnostics := []string{} if len(errorDiagnostics) > 0 { errStr := lipgloss.NewStyle(). Background(t.BackgroundDarker()). Foreground(t.Error()). Render(fmt.Sprintf("%s %d", styles.ErrorIcon, len(errorDiagnostics))) diagnostics = append(diagnostics, errStr) } if len(warnDiagnostics) > 0 { warnStr := lipgloss.NewStyle(). Background(t.BackgroundDarker()). Foreground(t.Warning()). Render(fmt.Sprintf("%s %d", styles.WarningIcon, len(warnDiagnostics))) diagnostics = append(diagnostics, warnStr) } if len(hintDiagnostics) > 0 { hintStr := lipgloss.NewStyle(). Background(t.BackgroundDarker()). Foreground(t.Text()). Render(fmt.Sprintf("%s %d", styles.HintIcon, len(hintDiagnostics))) diagnostics = append(diagnostics, hintStr) } if len(infoDiagnostics) > 0 { infoStr := lipgloss.NewStyle(). Background(t.BackgroundDarker()). Foreground(t.Info()). Render(fmt.Sprintf("%s %d", styles.InfoIcon, len(infoDiagnostics))) diagnostics = append(diagnostics, infoStr) } return strings.Join(diagnostics, " ") } func (m statusCmp) availableFooterMsgWidth(diagnostics, tokenInfo string) int { tokensWidth := 0 if m.session.ID != "" { tokensWidth = lipgloss.Width(tokenInfo) + 2 } return max(0, m.width-lipgloss.Width(helpWidget)-lipgloss.Width(m.model())-lipgloss.Width(diagnostics)-tokensWidth) } func (m statusCmp) model() string { t := theme.CurrentTheme() cfg := config.Get() coder, ok := cfg.Agents[config.AgentCoder] if !ok { return "Unknown" } model := models.SupportedModels[coder.Model] return styles.Padded(). Background(t.Secondary()). Foreground(t.Background()). Render(model.Name) } func NewStatusCmp(lspClients map[string]*lsp.Client) StatusCmp { helpWidget = getHelpWidget() return &statusCmp{ messageTTL: 10 * time.Second, lspClients: lspClients, } } ================================================ FILE: internal/tui/components/dialog/arguments.go ================================================ package dialog import ( "fmt" "github.com/charmbracelet/bubbles/key" "github.com/charmbracelet/bubbles/textinput" tea "github.com/charmbracelet/bubbletea" "github.com/charmbracelet/lipgloss" "github.com/opencode-ai/opencode/internal/tui/styles" "github.com/opencode-ai/opencode/internal/tui/theme" "github.com/opencode-ai/opencode/internal/tui/util" ) type argumentsDialogKeyMap struct { Enter key.Binding Escape key.Binding } // ShortHelp implements key.Map. func (k argumentsDialogKeyMap) ShortHelp() []key.Binding { return []key.Binding{ key.NewBinding( key.WithKeys("enter"), key.WithHelp("enter", "confirm"), ), key.NewBinding( key.WithKeys("esc"), key.WithHelp("esc", "cancel"), ), } } // FullHelp implements key.Map. func (k argumentsDialogKeyMap) FullHelp() [][]key.Binding { return [][]key.Binding{k.ShortHelp()} } // ShowMultiArgumentsDialogMsg is a message that is sent to show the multi-arguments dialog. type ShowMultiArgumentsDialogMsg struct { CommandID string Content string ArgNames []string } // CloseMultiArgumentsDialogMsg is a message that is sent when the multi-arguments dialog is closed. type CloseMultiArgumentsDialogMsg struct { Submit bool CommandID string Content string Args map[string]string } // MultiArgumentsDialogCmp is a component that asks the user for multiple command arguments. type MultiArgumentsDialogCmp struct { width, height int inputs []textinput.Model focusIndex int keys argumentsDialogKeyMap commandID string content string argNames []string } // NewMultiArgumentsDialogCmp creates a new MultiArgumentsDialogCmp. func NewMultiArgumentsDialogCmp(commandID, content string, argNames []string) MultiArgumentsDialogCmp { t := theme.CurrentTheme() inputs := make([]textinput.Model, len(argNames)) for i, name := range argNames { ti := textinput.New() ti.Placeholder = fmt.Sprintf("Enter value for %s...", name) ti.Width = 40 ti.Prompt = "" ti.PlaceholderStyle = ti.PlaceholderStyle.Background(t.Background()) ti.PromptStyle = ti.PromptStyle.Background(t.Background()) ti.TextStyle = ti.TextStyle.Background(t.Background()) // Only focus the first input initially if i == 0 { ti.Focus() ti.PromptStyle = ti.PromptStyle.Foreground(t.Primary()) ti.TextStyle = ti.TextStyle.Foreground(t.Primary()) } else { ti.Blur() } inputs[i] = ti } return MultiArgumentsDialogCmp{ inputs: inputs, keys: argumentsDialogKeyMap{}, commandID: commandID, content: content, argNames: argNames, focusIndex: 0, } } // Init implements tea.Model. func (m MultiArgumentsDialogCmp) Init() tea.Cmd { // Make sure only the first input is focused for i := range m.inputs { if i == 0 { m.inputs[i].Focus() } else { m.inputs[i].Blur() } } return textinput.Blink } // Update implements tea.Model. func (m MultiArgumentsDialogCmp) Update(msg tea.Msg) (tea.Model, tea.Cmd) { var cmds []tea.Cmd t := theme.CurrentTheme() switch msg := msg.(type) { case tea.KeyMsg: switch { case key.Matches(msg, key.NewBinding(key.WithKeys("esc"))): return m, util.CmdHandler(CloseMultiArgumentsDialogMsg{ Submit: false, CommandID: m.commandID, Content: m.content, Args: nil, }) case key.Matches(msg, key.NewBinding(key.WithKeys("enter"))): // If we're on the last input, submit the form if m.focusIndex == len(m.inputs)-1 { args := make(map[string]string) for i, name := range m.argNames { args[name] = m.inputs[i].Value() } return m, util.CmdHandler(CloseMultiArgumentsDialogMsg{ Submit: true, CommandID: m.commandID, Content: m.content, Args: args, }) } // Otherwise, move to the next input m.inputs[m.focusIndex].Blur() m.focusIndex++ m.inputs[m.focusIndex].Focus() m.inputs[m.focusIndex].PromptStyle = m.inputs[m.focusIndex].PromptStyle.Foreground(t.Primary()) m.inputs[m.focusIndex].TextStyle = m.inputs[m.focusIndex].TextStyle.Foreground(t.Primary()) case key.Matches(msg, key.NewBinding(key.WithKeys("tab"))): // Move to the next input m.inputs[m.focusIndex].Blur() m.focusIndex = (m.focusIndex + 1) % len(m.inputs) m.inputs[m.focusIndex].Focus() m.inputs[m.focusIndex].PromptStyle = m.inputs[m.focusIndex].PromptStyle.Foreground(t.Primary()) m.inputs[m.focusIndex].TextStyle = m.inputs[m.focusIndex].TextStyle.Foreground(t.Primary()) case key.Matches(msg, key.NewBinding(key.WithKeys("shift+tab"))): // Move to the previous input m.inputs[m.focusIndex].Blur() m.focusIndex = (m.focusIndex - 1 + len(m.inputs)) % len(m.inputs) m.inputs[m.focusIndex].Focus() m.inputs[m.focusIndex].PromptStyle = m.inputs[m.focusIndex].PromptStyle.Foreground(t.Primary()) m.inputs[m.focusIndex].TextStyle = m.inputs[m.focusIndex].TextStyle.Foreground(t.Primary()) } case tea.WindowSizeMsg: m.width = msg.Width m.height = msg.Height } // Update the focused input var cmd tea.Cmd m.inputs[m.focusIndex], cmd = m.inputs[m.focusIndex].Update(msg) cmds = append(cmds, cmd) return m, tea.Batch(cmds...) } // View implements tea.Model. func (m MultiArgumentsDialogCmp) View() string { t := theme.CurrentTheme() baseStyle := styles.BaseStyle() // Calculate width needed for content maxWidth := 60 // Width for explanation text title := lipgloss.NewStyle(). Foreground(t.Primary()). Bold(true). Width(maxWidth). Padding(0, 1). Background(t.Background()). Render("Command Arguments") explanation := lipgloss.NewStyle(). Foreground(t.Text()). Width(maxWidth). Padding(0, 1). Background(t.Background()). Render("This command requires multiple arguments. Please enter values for each:") // Create input fields for each argument inputFields := make([]string, len(m.inputs)) for i, input := range m.inputs { // Highlight the label of the focused input labelStyle := lipgloss.NewStyle(). Width(maxWidth). Padding(1, 1, 0, 1). Background(t.Background()) if i == m.focusIndex { labelStyle = labelStyle.Foreground(t.Primary()).Bold(true) } else { labelStyle = labelStyle.Foreground(t.TextMuted()) } label := labelStyle.Render(m.argNames[i] + ":") field := lipgloss.NewStyle(). Foreground(t.Text()). Width(maxWidth). Padding(0, 1). Background(t.Background()). Render(input.View()) inputFields[i] = lipgloss.JoinVertical(lipgloss.Left, label, field) } maxWidth = min(maxWidth, m.width-10) // Join all elements vertically elements := []string{title, explanation} elements = append(elements, inputFields...) content := lipgloss.JoinVertical( lipgloss.Left, elements..., ) return baseStyle.Padding(1, 2). Border(lipgloss.RoundedBorder()). BorderBackground(t.Background()). BorderForeground(t.TextMuted()). Background(t.Background()). Width(lipgloss.Width(content) + 4). Render(content) } // SetSize sets the size of the component. func (m *MultiArgumentsDialogCmp) SetSize(width, height int) { m.width = width m.height = height } // Bindings implements layout.Bindings. func (m MultiArgumentsDialogCmp) Bindings() []key.Binding { return m.keys.ShortHelp() } ================================================ FILE: internal/tui/components/dialog/commands.go ================================================ package dialog import ( "github.com/charmbracelet/bubbles/key" tea "github.com/charmbracelet/bubbletea" "github.com/charmbracelet/lipgloss" utilComponents "github.com/opencode-ai/opencode/internal/tui/components/util" "github.com/opencode-ai/opencode/internal/tui/layout" "github.com/opencode-ai/opencode/internal/tui/styles" "github.com/opencode-ai/opencode/internal/tui/theme" "github.com/opencode-ai/opencode/internal/tui/util" ) // Command represents a command that can be executed type Command struct { ID string Title string Description string Handler func(cmd Command) tea.Cmd } func (ci Command) Render(selected bool, width int) string { t := theme.CurrentTheme() baseStyle := styles.BaseStyle() descStyle := baseStyle.Width(width).Foreground(t.TextMuted()) itemStyle := baseStyle.Width(width). Foreground(t.Text()). Background(t.Background()) if selected { itemStyle = itemStyle. Background(t.Primary()). Foreground(t.Background()). Bold(true) descStyle = descStyle. Background(t.Primary()). Foreground(t.Background()) } title := itemStyle.Padding(0, 1).Render(ci.Title) if ci.Description != "" { description := descStyle.Padding(0, 1).Render(ci.Description) return lipgloss.JoinVertical(lipgloss.Left, title, description) } return title } // CommandSelectedMsg is sent when a command is selected type CommandSelectedMsg struct { Command Command } // CloseCommandDialogMsg is sent when the command dialog is closed type CloseCommandDialogMsg struct{} // CommandDialog interface for the command selection dialog type CommandDialog interface { tea.Model layout.Bindings SetCommands(commands []Command) } type commandDialogCmp struct { listView utilComponents.SimpleList[Command] width int height int } type commandKeyMap struct { Enter key.Binding Escape key.Binding } var commandKeys = commandKeyMap{ Enter: key.NewBinding( key.WithKeys("enter"), key.WithHelp("enter", "select command"), ), Escape: key.NewBinding( key.WithKeys("esc"), key.WithHelp("esc", "close"), ), } func (c *commandDialogCmp) Init() tea.Cmd { return c.listView.Init() } func (c *commandDialogCmp) Update(msg tea.Msg) (tea.Model, tea.Cmd) { var cmds []tea.Cmd switch msg := msg.(type) { case tea.KeyMsg: switch { case key.Matches(msg, commandKeys.Enter): selectedItem, idx := c.listView.GetSelectedItem() if idx != -1 { return c, util.CmdHandler(CommandSelectedMsg{ Command: selectedItem, }) } case key.Matches(msg, commandKeys.Escape): return c, util.CmdHandler(CloseCommandDialogMsg{}) } case tea.WindowSizeMsg: c.width = msg.Width c.height = msg.Height } u, cmd := c.listView.Update(msg) c.listView = u.(utilComponents.SimpleList[Command]) cmds = append(cmds, cmd) return c, tea.Batch(cmds...) } func (c *commandDialogCmp) View() string { t := theme.CurrentTheme() baseStyle := styles.BaseStyle() maxWidth := 40 commands := c.listView.GetItems() for _, cmd := range commands { if len(cmd.Title) > maxWidth-4 { maxWidth = len(cmd.Title) + 4 } if cmd.Description != "" { if len(cmd.Description) > maxWidth-4 { maxWidth = len(cmd.Description) + 4 } } } c.listView.SetMaxWidth(maxWidth) title := baseStyle. Foreground(t.Primary()). Bold(true). Width(maxWidth). Padding(0, 1). Render("Commands") content := lipgloss.JoinVertical( lipgloss.Left, title, baseStyle.Width(maxWidth).Render(""), baseStyle.Width(maxWidth).Render(c.listView.View()), baseStyle.Width(maxWidth).Render(""), ) return baseStyle.Padding(1, 2). Border(lipgloss.RoundedBorder()). BorderBackground(t.Background()). BorderForeground(t.TextMuted()). Width(lipgloss.Width(content) + 4). Render(content) } func (c *commandDialogCmp) BindingKeys() []key.Binding { return layout.KeyMapToSlice(commandKeys) } func (c *commandDialogCmp) SetCommands(commands []Command) { c.listView.SetItems(commands) } // NewCommandDialogCmp creates a new command selection dialog func NewCommandDialogCmp() CommandDialog { listView := utilComponents.NewSimpleList[Command]( []Command{}, 10, "No commands available", true, ) return &commandDialogCmp{ listView: listView, } } ================================================ FILE: internal/tui/components/dialog/complete.go ================================================ package dialog import ( "github.com/charmbracelet/bubbles/key" "github.com/charmbracelet/bubbles/textarea" tea "github.com/charmbracelet/bubbletea" "github.com/charmbracelet/lipgloss" "github.com/opencode-ai/opencode/internal/logging" utilComponents "github.com/opencode-ai/opencode/internal/tui/components/util" "github.com/opencode-ai/opencode/internal/tui/layout" "github.com/opencode-ai/opencode/internal/tui/styles" "github.com/opencode-ai/opencode/internal/tui/theme" "github.com/opencode-ai/opencode/internal/tui/util" ) type CompletionItem struct { title string Title string Value string } type CompletionItemI interface { utilComponents.SimpleListItem GetValue() string DisplayValue() string } func (ci *CompletionItem) Render(selected bool, width int) string { t := theme.CurrentTheme() baseStyle := styles.BaseStyle() itemStyle := baseStyle. Width(width). Padding(0, 1) if selected { itemStyle = itemStyle. Background(t.Background()). Foreground(t.Primary()). Bold(true) } title := itemStyle.Render( ci.GetValue(), ) return title } func (ci *CompletionItem) DisplayValue() string { return ci.Title } func (ci *CompletionItem) GetValue() string { return ci.Value } func NewCompletionItem(completionItem CompletionItem) CompletionItemI { return &completionItem } type CompletionProvider interface { GetId() string GetEntry() CompletionItemI GetChildEntries(query string) ([]CompletionItemI, error) } type CompletionSelectedMsg struct { SearchString string CompletionValue string } type CompletionDialogCompleteItemMsg struct { Value string } type CompletionDialogCloseMsg struct{} type CompletionDialog interface { tea.Model layout.Bindings SetWidth(width int) } type completionDialogCmp struct { query string completionProvider CompletionProvider width int height int pseudoSearchTextArea textarea.Model listView utilComponents.SimpleList[CompletionItemI] } type completionDialogKeyMap struct { Complete key.Binding Cancel key.Binding } var completionDialogKeys = completionDialogKeyMap{ Complete: key.NewBinding( key.WithKeys("tab", "enter"), ), Cancel: key.NewBinding( key.WithKeys(" ", "esc", "backspace"), ), } func (c *completionDialogCmp) Init() tea.Cmd { return nil } func (c *completionDialogCmp) complete(item CompletionItemI) tea.Cmd { value := c.pseudoSearchTextArea.Value() if value == "" { return nil } return tea.Batch( util.CmdHandler(CompletionSelectedMsg{ SearchString: value, CompletionValue: item.GetValue(), }), c.close(), ) } func (c *completionDialogCmp) close() tea.Cmd { c.listView.SetItems([]CompletionItemI{}) c.pseudoSearchTextArea.Reset() c.pseudoSearchTextArea.Blur() return util.CmdHandler(CompletionDialogCloseMsg{}) } func (c *completionDialogCmp) Update(msg tea.Msg) (tea.Model, tea.Cmd) { var cmds []tea.Cmd switch msg := msg.(type) { case tea.KeyMsg: if c.pseudoSearchTextArea.Focused() { if !key.Matches(msg, completionDialogKeys.Complete) { var cmd tea.Cmd c.pseudoSearchTextArea, cmd = c.pseudoSearchTextArea.Update(msg) cmds = append(cmds, cmd) var query string query = c.pseudoSearchTextArea.Value() if query != "" { query = query[1:] } if query != c.query { logging.Info("Query", query) items, err := c.completionProvider.GetChildEntries(query) if err != nil { logging.Error("Failed to get child entries", err) } c.listView.SetItems(items) c.query = query } u, cmd := c.listView.Update(msg) c.listView = u.(utilComponents.SimpleList[CompletionItemI]) cmds = append(cmds, cmd) } switch { case key.Matches(msg, completionDialogKeys.Complete): item, i := c.listView.GetSelectedItem() if i == -1 { return c, nil } cmd := c.complete(item) return c, cmd case key.Matches(msg, completionDialogKeys.Cancel): // Only close on backspace when there are no characters left if msg.String() != "backspace" || len(c.pseudoSearchTextArea.Value()) <= 0 { return c, c.close() } } return c, tea.Batch(cmds...) } else { items, err := c.completionProvider.GetChildEntries("") if err != nil { logging.Error("Failed to get child entries", err) } c.listView.SetItems(items) c.pseudoSearchTextArea.SetValue(msg.String()) return c, c.pseudoSearchTextArea.Focus() } case tea.WindowSizeMsg: c.width = msg.Width c.height = msg.Height } return c, tea.Batch(cmds...) } func (c *completionDialogCmp) View() string { t := theme.CurrentTheme() baseStyle := styles.BaseStyle() maxWidth := 40 completions := c.listView.GetItems() for _, cmd := range completions { title := cmd.DisplayValue() if len(title) > maxWidth-4 { maxWidth = len(title) + 4 } } c.listView.SetMaxWidth(maxWidth) return baseStyle.Padding(0, 0). Border(lipgloss.NormalBorder()). BorderBottom(false). BorderRight(false). BorderLeft(false). BorderBackground(t.Background()). BorderForeground(t.TextMuted()). Width(c.width). Render(c.listView.View()) } func (c *completionDialogCmp) SetWidth(width int) { c.width = width } func (c *completionDialogCmp) BindingKeys() []key.Binding { return layout.KeyMapToSlice(completionDialogKeys) } func NewCompletionDialogCmp(completionProvider CompletionProvider) CompletionDialog { ti := textarea.New() items, err := completionProvider.GetChildEntries("") if err != nil { logging.Error("Failed to get child entries", err) } li := utilComponents.NewSimpleList( items, 7, "No file matches found", false, ) return &completionDialogCmp{ query: "", completionProvider: completionProvider, pseudoSearchTextArea: ti, listView: li, } } ================================================ FILE: internal/tui/components/dialog/custom_commands.go ================================================ package dialog import ( "fmt" "os" "path/filepath" "regexp" "strings" tea "github.com/charmbracelet/bubbletea" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/tui/util" ) // Command prefix constants const ( UserCommandPrefix = "user:" ProjectCommandPrefix = "project:" ) // namedArgPattern is a regex pattern to find named arguments in the format $NAME var namedArgPattern = regexp.MustCompile(`\$([A-Z][A-Z0-9_]*)`) // LoadCustomCommands loads custom commands from both XDG_CONFIG_HOME and project data directory func LoadCustomCommands() ([]Command, error) { cfg := config.Get() if cfg == nil { return nil, fmt.Errorf("config not loaded") } var commands []Command // Load user commands from XDG_CONFIG_HOME/opencode/commands xdgConfigHome := os.Getenv("XDG_CONFIG_HOME") if xdgConfigHome == "" { // Default to ~/.config if XDG_CONFIG_HOME is not set home, err := os.UserHomeDir() if err == nil { xdgConfigHome = filepath.Join(home, ".config") } } if xdgConfigHome != "" { userCommandsDir := filepath.Join(xdgConfigHome, "opencode", "commands") userCommands, err := loadCommandsFromDir(userCommandsDir, UserCommandPrefix) if err != nil { // Log error but continue - we'll still try to load other commands fmt.Printf("Warning: failed to load user commands from XDG_CONFIG_HOME: %v\n", err) } else { commands = append(commands, userCommands...) } } // Load commands from $HOME/.opencode/commands home, err := os.UserHomeDir() if err == nil { homeCommandsDir := filepath.Join(home, ".opencode", "commands") homeCommands, err := loadCommandsFromDir(homeCommandsDir, UserCommandPrefix) if err != nil { // Log error but continue - we'll still try to load other commands fmt.Printf("Warning: failed to load home commands: %v\n", err) } else { commands = append(commands, homeCommands...) } } // Load project commands from data directory projectCommandsDir := filepath.Join(cfg.Data.Directory, "commands") projectCommands, err := loadCommandsFromDir(projectCommandsDir, ProjectCommandPrefix) if err != nil { // Log error but return what we have so far fmt.Printf("Warning: failed to load project commands: %v\n", err) } else { commands = append(commands, projectCommands...) } return commands, nil } // loadCommandsFromDir loads commands from a specific directory with the given prefix func loadCommandsFromDir(commandsDir string, prefix string) ([]Command, error) { // Check if the commands directory exists if _, err := os.Stat(commandsDir); os.IsNotExist(err) { // Create the commands directory if it doesn't exist if err := os.MkdirAll(commandsDir, 0755); err != nil { return nil, fmt.Errorf("failed to create commands directory %s: %w", commandsDir, err) } // Return empty list since we just created the directory return []Command{}, nil } var commands []Command // Walk through the commands directory and load all .md files err := filepath.Walk(commandsDir, func(path string, info os.FileInfo, err error) error { if err != nil { return err } // Skip directories if info.IsDir() { return nil } // Only process markdown files if !strings.HasSuffix(strings.ToLower(info.Name()), ".md") { return nil } // Read the file content content, err := os.ReadFile(path) if err != nil { return fmt.Errorf("failed to read command file %s: %w", path, err) } // Get the command ID from the file name without the .md extension commandID := strings.TrimSuffix(info.Name(), filepath.Ext(info.Name())) // Get relative path from commands directory relPath, err := filepath.Rel(commandsDir, path) if err != nil { return fmt.Errorf("failed to get relative path for %s: %w", path, err) } // Create the command ID from the relative path // Replace directory separators with colons commandIDPath := strings.ReplaceAll(filepath.Dir(relPath), string(filepath.Separator), ":") if commandIDPath != "." { commandID = commandIDPath + ":" + commandID } // Create a command command := Command{ ID: prefix + commandID, Title: prefix + commandID, Description: fmt.Sprintf("Custom command from %s", relPath), Handler: func(cmd Command) tea.Cmd { commandContent := string(content) // Check for named arguments matches := namedArgPattern.FindAllStringSubmatch(commandContent, -1) if len(matches) > 0 { // Extract unique argument names argNames := make([]string, 0) argMap := make(map[string]bool) for _, match := range matches { argName := match[1] // Group 1 is the name without $ if !argMap[argName] { argMap[argName] = true argNames = append(argNames, argName) } } // Show multi-arguments dialog for all named arguments return util.CmdHandler(ShowMultiArgumentsDialogMsg{ CommandID: cmd.ID, Content: commandContent, ArgNames: argNames, }) } // No arguments needed, run command directly return util.CmdHandler(CommandRunCustomMsg{ Content: commandContent, Args: nil, // No arguments }) }, } commands = append(commands, command) return nil }) if err != nil { return nil, fmt.Errorf("failed to load custom commands from %s: %w", commandsDir, err) } return commands, nil } // CommandRunCustomMsg is sent when a custom command is executed type CommandRunCustomMsg struct { Content string Args map[string]string // Map of argument names to values } ================================================ FILE: internal/tui/components/dialog/custom_commands_test.go ================================================ package dialog import ( "testing" "regexp" ) func TestNamedArgPattern(t *testing.T) { testCases := []struct { input string expected []string }{ { input: "This is a test with $ARGUMENTS placeholder", expected: []string{"ARGUMENTS"}, }, { input: "This is a test with $FOO and $BAR placeholders", expected: []string{"FOO", "BAR"}, }, { input: "This is a test with $FOO_BAR and $BAZ123 placeholders", expected: []string{"FOO_BAR", "BAZ123"}, }, { input: "This is a test with no placeholders", expected: []string{}, }, { input: "This is a test with $FOO appearing twice: $FOO", expected: []string{"FOO"}, }, { input: "This is a test with $1INVALID placeholder", expected: []string{}, }, } for _, tc := range testCases { matches := namedArgPattern.FindAllStringSubmatch(tc.input, -1) // Extract unique argument names argNames := make([]string, 0) argMap := make(map[string]bool) for _, match := range matches { argName := match[1] // Group 1 is the name without $ if !argMap[argName] { argMap[argName] = true argNames = append(argNames, argName) } } // Check if we got the expected number of arguments if len(argNames) != len(tc.expected) { t.Errorf("Expected %d arguments, got %d for input: %s", len(tc.expected), len(argNames), tc.input) continue } // Check if we got the expected argument names for _, expectedArg := range tc.expected { found := false for _, actualArg := range argNames { if actualArg == expectedArg { found = true break } } if !found { t.Errorf("Expected argument %s not found in %v for input: %s", expectedArg, argNames, tc.input) } } } } func TestRegexPattern(t *testing.T) { pattern := regexp.MustCompile(`\$([A-Z][A-Z0-9_]*)`) validMatches := []string{ "$FOO", "$BAR", "$FOO_BAR", "$BAZ123", "$ARGUMENTS", } invalidMatches := []string{ "$foo", "$1BAR", "$_FOO", "FOO", "$", } for _, valid := range validMatches { if !pattern.MatchString(valid) { t.Errorf("Expected %s to match, but it didn't", valid) } } for _, invalid := range invalidMatches { if pattern.MatchString(invalid) { t.Errorf("Expected %s not to match, but it did", invalid) } } } ================================================ FILE: internal/tui/components/dialog/filepicker.go ================================================ package dialog import ( "fmt" "net/http" "os" "path/filepath" "sort" "strings" "time" "github.com/charmbracelet/bubbles/key" "github.com/charmbracelet/bubbles/textinput" "github.com/charmbracelet/bubbles/viewport" tea "github.com/charmbracelet/bubbletea" "github.com/charmbracelet/lipgloss" "github.com/opencode-ai/opencode/internal/app" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/logging" "github.com/opencode-ai/opencode/internal/message" "github.com/opencode-ai/opencode/internal/tui/image" "github.com/opencode-ai/opencode/internal/tui/styles" "github.com/opencode-ai/opencode/internal/tui/theme" "github.com/opencode-ai/opencode/internal/tui/util" ) const ( maxAttachmentSize = int64(5 * 1024 * 1024) // 5MB downArrow = "down" upArrow = "up" ) type FilePrickerKeyMap struct { Enter key.Binding Down key.Binding Up key.Binding Forward key.Binding Backward key.Binding OpenFilePicker key.Binding Esc key.Binding InsertCWD key.Binding } var filePickerKeyMap = FilePrickerKeyMap{ Enter: key.NewBinding( key.WithKeys("enter"), key.WithHelp("enter", "select file/enter directory"), ), Down: key.NewBinding( key.WithKeys("j", downArrow), key.WithHelp("↓/j", "down"), ), Up: key.NewBinding( key.WithKeys("k", upArrow), key.WithHelp("↑/k", "up"), ), Forward: key.NewBinding( key.WithKeys("l"), key.WithHelp("l", "enter directory"), ), Backward: key.NewBinding( key.WithKeys("h", "backspace"), key.WithHelp("h/backspace", "go back"), ), OpenFilePicker: key.NewBinding( key.WithKeys("ctrl+f"), key.WithHelp("ctrl+f", "open file picker"), ), Esc: key.NewBinding( key.WithKeys("esc"), key.WithHelp("esc", "close/exit"), ), InsertCWD: key.NewBinding( key.WithKeys("i"), key.WithHelp("i", "manual path input"), ), } type filepickerCmp struct { basePath string width int height int cursor int err error cursorChain stack viewport viewport.Model dirs []os.DirEntry cwdDetails *DirNode selectedFile string cwd textinput.Model ShowFilePicker bool app *app.App } type DirNode struct { parent *DirNode child *DirNode directory string } type stack []int func (s stack) Push(v int) stack { return append(s, v) } func (s stack) Pop() (stack, int) { l := len(s) return s[:l-1], s[l-1] } type AttachmentAddedMsg struct { Attachment message.Attachment } func (f *filepickerCmp) Init() tea.Cmd { return nil } func (f *filepickerCmp) Update(msg tea.Msg) (tea.Model, tea.Cmd) { var cmd tea.Cmd switch msg := msg.(type) { case tea.WindowSizeMsg: f.width = 60 f.height = 20 f.viewport.Width = 80 f.viewport.Height = 22 f.cursor = 0 f.getCurrentFileBelowCursor() case tea.KeyMsg: if f.cwd.Focused() { f.cwd, cmd = f.cwd.Update(msg) } switch { case key.Matches(msg, filePickerKeyMap.InsertCWD): f.cwd.Focus() return f, cmd case key.Matches(msg, filePickerKeyMap.Esc): if f.cwd.Focused() { f.cwd.Blur() } case key.Matches(msg, filePickerKeyMap.Down): if !f.cwd.Focused() || msg.String() == downArrow { if f.cursor < len(f.dirs)-1 { f.cursor++ f.getCurrentFileBelowCursor() } } case key.Matches(msg, filePickerKeyMap.Up): if !f.cwd.Focused() || msg.String() == upArrow { if f.cursor > 0 { f.cursor-- f.getCurrentFileBelowCursor() } } case key.Matches(msg, filePickerKeyMap.Enter): var path string var isPathDir bool if f.cwd.Focused() { path = f.cwd.Value() fileInfo, err := os.Stat(path) if err != nil { logging.ErrorPersist("Invalid path") return f, cmd } isPathDir = fileInfo.IsDir() } else { path = filepath.Join(f.cwdDetails.directory, "/", f.dirs[f.cursor].Name()) isPathDir = f.dirs[f.cursor].IsDir() } if isPathDir { newWorkingDir := DirNode{parent: f.cwdDetails, directory: path} f.cwdDetails.child = &newWorkingDir f.cwdDetails = f.cwdDetails.child f.cursorChain = f.cursorChain.Push(f.cursor) f.dirs = readDir(f.cwdDetails.directory, false) f.cursor = 0 f.cwd.SetValue(f.cwdDetails.directory) f.getCurrentFileBelowCursor() } else { f.selectedFile = path return f.addAttachmentToMessage() } case key.Matches(msg, filePickerKeyMap.Esc): if !f.cwd.Focused() { f.cursorChain = make(stack, 0) f.cursor = 0 } else { f.cwd.Blur() } case key.Matches(msg, filePickerKeyMap.Forward): if !f.cwd.Focused() { if f.dirs[f.cursor].IsDir() { path := filepath.Join(f.cwdDetails.directory, "/", f.dirs[f.cursor].Name()) newWorkingDir := DirNode{parent: f.cwdDetails, directory: path} f.cwdDetails.child = &newWorkingDir f.cwdDetails = f.cwdDetails.child f.cursorChain = f.cursorChain.Push(f.cursor) f.dirs = readDir(f.cwdDetails.directory, false) f.cursor = 0 f.cwd.SetValue(f.cwdDetails.directory) f.getCurrentFileBelowCursor() } } case key.Matches(msg, filePickerKeyMap.Backward): if !f.cwd.Focused() { if len(f.cursorChain) != 0 && f.cwdDetails.parent != nil { f.cursorChain, f.cursor = f.cursorChain.Pop() f.cwdDetails = f.cwdDetails.parent f.cwdDetails.child = nil f.dirs = readDir(f.cwdDetails.directory, false) f.cwd.SetValue(f.cwdDetails.directory) f.getCurrentFileBelowCursor() } } case key.Matches(msg, filePickerKeyMap.OpenFilePicker): f.dirs = readDir(f.cwdDetails.directory, false) f.cursor = 0 f.getCurrentFileBelowCursor() } } return f, cmd } func (f *filepickerCmp) addAttachmentToMessage() (tea.Model, tea.Cmd) { modeInfo := GetSelectedModel(config.Get()) if !modeInfo.SupportsAttachments { logging.ErrorPersist(fmt.Sprintf("Model %s doesn't support attachments", modeInfo.Name)) return f, nil } selectedFilePath := f.selectedFile if !isExtSupported(selectedFilePath) { logging.ErrorPersist("Unsupported file") return f, nil } isFileLarge, err := image.ValidateFileSize(selectedFilePath, maxAttachmentSize) if err != nil { logging.ErrorPersist("unable to read the image") return f, nil } if isFileLarge { logging.ErrorPersist("file too large, max 5MB") return f, nil } content, err := os.ReadFile(selectedFilePath) if err != nil { logging.ErrorPersist("Unable read selected file") return f, nil } mimeBufferSize := min(512, len(content)) mimeType := http.DetectContentType(content[:mimeBufferSize]) fileName := filepath.Base(selectedFilePath) attachment := message.Attachment{FilePath: selectedFilePath, FileName: fileName, MimeType: mimeType, Content: content} f.selectedFile = "" return f, util.CmdHandler(AttachmentAddedMsg{attachment}) } func (f *filepickerCmp) View() string { t := theme.CurrentTheme() const maxVisibleDirs = 20 const maxWidth = 80 adjustedWidth := maxWidth for _, file := range f.dirs { if len(file.Name()) > adjustedWidth-4 { // Account for padding adjustedWidth = len(file.Name()) + 4 } } adjustedWidth = max(30, min(adjustedWidth, f.width-15)) + 1 files := make([]string, 0, maxVisibleDirs) startIdx := 0 if len(f.dirs) > maxVisibleDirs { halfVisible := maxVisibleDirs / 2 if f.cursor >= halfVisible && f.cursor < len(f.dirs)-halfVisible { startIdx = f.cursor - halfVisible } else if f.cursor >= len(f.dirs)-halfVisible { startIdx = len(f.dirs) - maxVisibleDirs } } endIdx := min(startIdx+maxVisibleDirs, len(f.dirs)) for i := startIdx; i < endIdx; i++ { file := f.dirs[i] itemStyle := styles.BaseStyle().Width(adjustedWidth) if i == f.cursor { itemStyle = itemStyle. Background(t.Primary()). Foreground(t.Background()). Bold(true) } filename := file.Name() if len(filename) > adjustedWidth-4 { filename = filename[:adjustedWidth-7] + "..." } if file.IsDir() { filename = filename + "/" } // No need to reassign filename if it's not changing files = append(files, itemStyle.Padding(0, 1).Render(filename)) } // Pad to always show exactly 21 lines for len(files) < maxVisibleDirs { files = append(files, styles.BaseStyle().Width(adjustedWidth).Render("")) } currentPath := styles.BaseStyle(). Height(1). Width(adjustedWidth). Render(f.cwd.View()) viewportstyle := lipgloss.NewStyle(). Width(f.viewport.Width). Background(t.Background()). Border(lipgloss.RoundedBorder()). BorderForeground(t.TextMuted()). BorderBackground(t.Background()). Padding(2). Render(f.viewport.View()) var insertExitText string if f.IsCWDFocused() { insertExitText = "Press esc to exit typing path" } else { insertExitText = "Press i to start typing path" } content := lipgloss.JoinVertical( lipgloss.Left, currentPath, styles.BaseStyle().Width(adjustedWidth).Render(""), styles.BaseStyle().Width(adjustedWidth).Render(lipgloss.JoinVertical(lipgloss.Left, files...)), styles.BaseStyle().Width(adjustedWidth).Render(""), styles.BaseStyle().Foreground(t.TextMuted()).Width(adjustedWidth).Render(insertExitText), ) f.cwd.SetValue(f.cwd.Value()) contentStyle := styles.BaseStyle().Padding(1, 2). Border(lipgloss.RoundedBorder()). BorderBackground(t.Background()). BorderForeground(t.TextMuted()). Width(lipgloss.Width(content) + 4) return lipgloss.JoinHorizontal(lipgloss.Center, contentStyle.Render(content), viewportstyle) } type FilepickerCmp interface { tea.Model ToggleFilepicker(showFilepicker bool) IsCWDFocused() bool } func (f *filepickerCmp) ToggleFilepicker(showFilepicker bool) { f.ShowFilePicker = showFilepicker } func (f *filepickerCmp) IsCWDFocused() bool { return f.cwd.Focused() } func NewFilepickerCmp(app *app.App) FilepickerCmp { homepath, err := os.UserHomeDir() if err != nil { logging.Error("error loading user files") return nil } baseDir := DirNode{parent: nil, directory: homepath} dirs := readDir(homepath, false) viewport := viewport.New(0, 0) currentDirectory := textinput.New() currentDirectory.CharLimit = 200 currentDirectory.Width = 44 currentDirectory.Cursor.Blink = true currentDirectory.SetValue(baseDir.directory) return &filepickerCmp{cwdDetails: &baseDir, dirs: dirs, cursorChain: make(stack, 0), viewport: viewport, cwd: currentDirectory, app: app} } func (f *filepickerCmp) getCurrentFileBelowCursor() { if len(f.dirs) == 0 || f.cursor < 0 || f.cursor >= len(f.dirs) { logging.Error(fmt.Sprintf("Invalid cursor position. Dirs length: %d, Cursor: %d", len(f.dirs), f.cursor)) f.viewport.SetContent("Preview unavailable") return } dir := f.dirs[f.cursor] filename := dir.Name() if !dir.IsDir() && isExtSupported(filename) { fullPath := f.cwdDetails.directory + "/" + dir.Name() go func() { imageString, err := image.ImagePreview(f.viewport.Width-4, fullPath) if err != nil { logging.Error(err.Error()) f.viewport.SetContent("Preview unavailable") return } f.viewport.SetContent(imageString) }() } else { f.viewport.SetContent("Preview unavailable") } } func readDir(path string, showHidden bool) []os.DirEntry { logging.Info(fmt.Sprintf("Reading directory: %s", path)) entriesChan := make(chan []os.DirEntry, 1) errChan := make(chan error, 1) go func() { dirEntries, err := os.ReadDir(path) if err != nil { logging.ErrorPersist(err.Error()) errChan <- err return } entriesChan <- dirEntries }() select { case dirEntries := <-entriesChan: sort.Slice(dirEntries, func(i, j int) bool { if dirEntries[i].IsDir() == dirEntries[j].IsDir() { return dirEntries[i].Name() < dirEntries[j].Name() } return dirEntries[i].IsDir() }) if showHidden { return dirEntries } var sanitizedDirEntries []os.DirEntry for _, dirEntry := range dirEntries { isHidden, _ := IsHidden(dirEntry.Name()) if !isHidden { if dirEntry.IsDir() || isExtSupported(dirEntry.Name()) { sanitizedDirEntries = append(sanitizedDirEntries, dirEntry) } } } return sanitizedDirEntries case err := <-errChan: logging.ErrorPersist(fmt.Sprintf("Error reading directory %s", path), err) return []os.DirEntry{} case <-time.After(5 * time.Second): logging.ErrorPersist(fmt.Sprintf("Timeout reading directory %s", path), nil) return []os.DirEntry{} } } func IsHidden(file string) (bool, error) { return strings.HasPrefix(file, "."), nil } func isExtSupported(path string) bool { ext := strings.ToLower(filepath.Ext(path)) return (ext == ".jpg" || ext == ".jpeg" || ext == ".webp" || ext == ".png") } ================================================ FILE: internal/tui/components/dialog/help.go ================================================ package dialog import ( "strings" "github.com/charmbracelet/bubbles/key" tea "github.com/charmbracelet/bubbletea" "github.com/charmbracelet/lipgloss" "github.com/opencode-ai/opencode/internal/tui/styles" "github.com/opencode-ai/opencode/internal/tui/theme" ) type helpCmp struct { width int height int keys []key.Binding } func (h *helpCmp) Init() tea.Cmd { return nil } func (h *helpCmp) SetBindings(k []key.Binding) { h.keys = k } func (h *helpCmp) Update(msg tea.Msg) (tea.Model, tea.Cmd) { switch msg := msg.(type) { case tea.WindowSizeMsg: h.width = 90 h.height = msg.Height } return h, nil } func removeDuplicateBindings(bindings []key.Binding) []key.Binding { seen := make(map[string]struct{}) result := make([]key.Binding, 0, len(bindings)) // Process bindings in reverse order for i := len(bindings) - 1; i >= 0; i-- { b := bindings[i] k := strings.Join(b.Keys(), " ") if _, ok := seen[k]; ok { // duplicate, skip continue } seen[k] = struct{}{} // Add to the beginning of result to maintain original order result = append([]key.Binding{b}, result...) } return result } func (h *helpCmp) render() string { t := theme.CurrentTheme() baseStyle := styles.BaseStyle() helpKeyStyle := styles.Bold(). Background(t.Background()). Foreground(t.Text()). Padding(0, 1, 0, 0) helpDescStyle := styles.Regular(). Background(t.Background()). Foreground(t.TextMuted()) // Compile list of bindings to render bindings := removeDuplicateBindings(h.keys) // Enumerate through each group of bindings, populating a series of // pairs of columns, one for keys, one for descriptions var ( pairs []string width int rows = 12 - 2 ) for i := 0; i < len(bindings); i += rows { var ( keys []string descs []string ) for j := i; j < min(i+rows, len(bindings)); j++ { keys = append(keys, helpKeyStyle.Render(bindings[j].Help().Key)) descs = append(descs, helpDescStyle.Render(bindings[j].Help().Desc)) } // Render pair of columns; beyond the first pair, render a three space // left margin, in order to visually separate the pairs. var cols []string if len(pairs) > 0 { cols = []string{baseStyle.Render(" ")} } maxDescWidth := 0 for _, desc := range descs { if maxDescWidth < lipgloss.Width(desc) { maxDescWidth = lipgloss.Width(desc) } } for i := range descs { remainingWidth := maxDescWidth - lipgloss.Width(descs[i]) if remainingWidth > 0 { descs[i] = descs[i] + baseStyle.Render(strings.Repeat(" ", remainingWidth)) } } maxKeyWidth := 0 for _, key := range keys { if maxKeyWidth < lipgloss.Width(key) { maxKeyWidth = lipgloss.Width(key) } } for i := range keys { remainingWidth := maxKeyWidth - lipgloss.Width(keys[i]) if remainingWidth > 0 { keys[i] = keys[i] + baseStyle.Render(strings.Repeat(" ", remainingWidth)) } } cols = append(cols, strings.Join(keys, "\n"), strings.Join(descs, "\n"), ) pair := baseStyle.Render(lipgloss.JoinHorizontal(lipgloss.Top, cols...)) // check whether it exceeds the maximum width avail (the width of the // terminal, subtracting 2 for the borders). width += lipgloss.Width(pair) if width > h.width-2 { break } pairs = append(pairs, pair) } // https://github.com/charmbracelet/lipgloss/issues/209 if len(pairs) > 1 { prefix := pairs[:len(pairs)-1] lastPair := pairs[len(pairs)-1] prefix = append(prefix, lipgloss.Place( lipgloss.Width(lastPair), // width lipgloss.Height(prefix[0]), // height lipgloss.Left, // x lipgloss.Top, // y lastPair, // content lipgloss.WithWhitespaceBackground(t.Background()), )) content := baseStyle.Width(h.width).Render( lipgloss.JoinHorizontal( lipgloss.Top, prefix..., ), ) return content } // Join pairs of columns and enclose in a border content := baseStyle.Width(h.width).Render( lipgloss.JoinHorizontal( lipgloss.Top, pairs..., ), ) return content } func (h *helpCmp) View() string { t := theme.CurrentTheme() baseStyle := styles.BaseStyle() content := h.render() header := baseStyle. Bold(true). Width(lipgloss.Width(content)). Foreground(t.Primary()). Render("Keyboard Shortcuts") return baseStyle.Padding(1). Border(lipgloss.RoundedBorder()). BorderForeground(t.TextMuted()). Width(h.width). BorderBackground(t.Background()). Render( lipgloss.JoinVertical(lipgloss.Center, header, baseStyle.Render(strings.Repeat(" ", lipgloss.Width(header))), content, ), ) } type HelpCmp interface { tea.Model SetBindings([]key.Binding) } func NewHelpCmp() HelpCmp { return &helpCmp{} } ================================================ FILE: internal/tui/components/dialog/init.go ================================================ package dialog import ( "github.com/charmbracelet/bubbles/key" tea "github.com/charmbracelet/bubbletea" "github.com/charmbracelet/lipgloss" "github.com/opencode-ai/opencode/internal/tui/styles" "github.com/opencode-ai/opencode/internal/tui/theme" "github.com/opencode-ai/opencode/internal/tui/util" ) // InitDialogCmp is a component that asks the user if they want to initialize the project. type InitDialogCmp struct { width, height int selected int keys initDialogKeyMap } // NewInitDialogCmp creates a new InitDialogCmp. func NewInitDialogCmp() InitDialogCmp { return InitDialogCmp{ selected: 0, keys: initDialogKeyMap{}, } } type initDialogKeyMap struct { Tab key.Binding Left key.Binding Right key.Binding Enter key.Binding Escape key.Binding Y key.Binding N key.Binding } // ShortHelp implements key.Map. func (k initDialogKeyMap) ShortHelp() []key.Binding { return []key.Binding{ key.NewBinding( key.WithKeys("tab", "left", "right"), key.WithHelp("tab/←/→", "toggle selection"), ), key.NewBinding( key.WithKeys("enter"), key.WithHelp("enter", "confirm"), ), key.NewBinding( key.WithKeys("esc", "q"), key.WithHelp("esc/q", "cancel"), ), key.NewBinding( key.WithKeys("y", "n"), key.WithHelp("y/n", "yes/no"), ), } } // FullHelp implements key.Map. func (k initDialogKeyMap) FullHelp() [][]key.Binding { return [][]key.Binding{k.ShortHelp()} } // Init implements tea.Model. func (m InitDialogCmp) Init() tea.Cmd { return nil } // Update implements tea.Model. func (m InitDialogCmp) Update(msg tea.Msg) (tea.Model, tea.Cmd) { switch msg := msg.(type) { case tea.KeyMsg: switch { case key.Matches(msg, key.NewBinding(key.WithKeys("esc"))): return m, util.CmdHandler(CloseInitDialogMsg{Initialize: false}) case key.Matches(msg, key.NewBinding(key.WithKeys("tab", "left", "right", "h", "l"))): m.selected = (m.selected + 1) % 2 return m, nil case key.Matches(msg, key.NewBinding(key.WithKeys("enter"))): return m, util.CmdHandler(CloseInitDialogMsg{Initialize: m.selected == 0}) case key.Matches(msg, key.NewBinding(key.WithKeys("y"))): return m, util.CmdHandler(CloseInitDialogMsg{Initialize: true}) case key.Matches(msg, key.NewBinding(key.WithKeys("n"))): return m, util.CmdHandler(CloseInitDialogMsg{Initialize: false}) } case tea.WindowSizeMsg: m.width = msg.Width m.height = msg.Height } return m, nil } // View implements tea.Model. func (m InitDialogCmp) View() string { t := theme.CurrentTheme() baseStyle := styles.BaseStyle() // Calculate width needed for content maxWidth := 60 // Width for explanation text title := baseStyle. Foreground(t.Primary()). Bold(true). Width(maxWidth). Padding(0, 1). Render("Initialize Project") explanation := baseStyle. Foreground(t.Text()). Width(maxWidth). Padding(0, 1). Render("Initialization generates a new OpenCode.md file that contains information about your codebase, this file serves as memory for each project, you can freely add to it to help the agents be better at their job.") question := baseStyle. Foreground(t.Text()). Width(maxWidth). Padding(1, 1). Render("Would you like to initialize this project?") maxWidth = min(maxWidth, m.width-10) yesStyle := baseStyle noStyle := baseStyle if m.selected == 0 { yesStyle = yesStyle. Background(t.Primary()). Foreground(t.Background()). Bold(true) noStyle = noStyle. Background(t.Background()). Foreground(t.Primary()) } else { noStyle = noStyle. Background(t.Primary()). Foreground(t.Background()). Bold(true) yesStyle = yesStyle. Background(t.Background()). Foreground(t.Primary()) } yes := yesStyle.Padding(0, 3).Render("Yes") no := noStyle.Padding(0, 3).Render("No") buttons := lipgloss.JoinHorizontal(lipgloss.Center, yes, baseStyle.Render(" "), no) buttons = baseStyle. Width(maxWidth). Padding(1, 0). Render(buttons) content := lipgloss.JoinVertical( lipgloss.Left, title, baseStyle.Width(maxWidth).Render(""), explanation, question, buttons, baseStyle.Width(maxWidth).Render(""), ) return baseStyle.Padding(1, 2). Border(lipgloss.RoundedBorder()). BorderBackground(t.Background()). BorderForeground(t.TextMuted()). Width(lipgloss.Width(content) + 4). Render(content) } // SetSize sets the size of the component. func (m *InitDialogCmp) SetSize(width, height int) { m.width = width m.height = height } // Bindings implements layout.Bindings. func (m InitDialogCmp) Bindings() []key.Binding { return m.keys.ShortHelp() } // CloseInitDialogMsg is a message that is sent when the init dialog is closed. type CloseInitDialogMsg struct { Initialize bool } // ShowInitDialogMsg is a message that is sent to show the init dialog. type ShowInitDialogMsg struct { Show bool } ================================================ FILE: internal/tui/components/dialog/models.go ================================================ package dialog import ( "fmt" "slices" "strings" "github.com/charmbracelet/bubbles/key" tea "github.com/charmbracelet/bubbletea" "github.com/charmbracelet/lipgloss" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/llm/models" "github.com/opencode-ai/opencode/internal/tui/layout" "github.com/opencode-ai/opencode/internal/tui/styles" "github.com/opencode-ai/opencode/internal/tui/theme" "github.com/opencode-ai/opencode/internal/tui/util" ) const ( numVisibleModels = 10 maxDialogWidth = 40 ) // ModelSelectedMsg is sent when a model is selected type ModelSelectedMsg struct { Model models.Model } // CloseModelDialogMsg is sent when a model is selected type CloseModelDialogMsg struct{} // ModelDialog interface for the model selection dialog type ModelDialog interface { tea.Model layout.Bindings } type modelDialogCmp struct { models []models.Model provider models.ModelProvider availableProviders []models.ModelProvider selectedIdx int width int height int scrollOffset int hScrollOffset int hScrollPossible bool } type modelKeyMap struct { Up key.Binding Down key.Binding Left key.Binding Right key.Binding Enter key.Binding Escape key.Binding J key.Binding K key.Binding H key.Binding L key.Binding } var modelKeys = modelKeyMap{ Up: key.NewBinding( key.WithKeys("up"), key.WithHelp("↑", "previous model"), ), Down: key.NewBinding( key.WithKeys("down"), key.WithHelp("↓", "next model"), ), Left: key.NewBinding( key.WithKeys("left"), key.WithHelp("←", "scroll left"), ), Right: key.NewBinding( key.WithKeys("right"), key.WithHelp("→", "scroll right"), ), Enter: key.NewBinding( key.WithKeys("enter"), key.WithHelp("enter", "select model"), ), Escape: key.NewBinding( key.WithKeys("esc"), key.WithHelp("esc", "close"), ), J: key.NewBinding( key.WithKeys("j"), key.WithHelp("j", "next model"), ), K: key.NewBinding( key.WithKeys("k"), key.WithHelp("k", "previous model"), ), H: key.NewBinding( key.WithKeys("h"), key.WithHelp("h", "scroll left"), ), L: key.NewBinding( key.WithKeys("l"), key.WithHelp("l", "scroll right"), ), } func (m *modelDialogCmp) Init() tea.Cmd { m.setupModels() return nil } func (m *modelDialogCmp) Update(msg tea.Msg) (tea.Model, tea.Cmd) { switch msg := msg.(type) { case tea.KeyMsg: switch { case key.Matches(msg, modelKeys.Up) || key.Matches(msg, modelKeys.K): m.moveSelectionUp() case key.Matches(msg, modelKeys.Down) || key.Matches(msg, modelKeys.J): m.moveSelectionDown() case key.Matches(msg, modelKeys.Left) || key.Matches(msg, modelKeys.H): if m.hScrollPossible { m.switchProvider(-1) } case key.Matches(msg, modelKeys.Right) || key.Matches(msg, modelKeys.L): if m.hScrollPossible { m.switchProvider(1) } case key.Matches(msg, modelKeys.Enter): util.ReportInfo(fmt.Sprintf("selected model: %s", m.models[m.selectedIdx].Name)) return m, util.CmdHandler(ModelSelectedMsg{Model: m.models[m.selectedIdx]}) case key.Matches(msg, modelKeys.Escape): return m, util.CmdHandler(CloseModelDialogMsg{}) } case tea.WindowSizeMsg: m.width = msg.Width m.height = msg.Height } return m, nil } // moveSelectionUp moves the selection up or wraps to bottom func (m *modelDialogCmp) moveSelectionUp() { if m.selectedIdx > 0 { m.selectedIdx-- } else { m.selectedIdx = len(m.models) - 1 m.scrollOffset = max(0, len(m.models)-numVisibleModels) } // Keep selection visible if m.selectedIdx < m.scrollOffset { m.scrollOffset = m.selectedIdx } } // moveSelectionDown moves the selection down or wraps to top func (m *modelDialogCmp) moveSelectionDown() { if m.selectedIdx < len(m.models)-1 { m.selectedIdx++ } else { m.selectedIdx = 0 m.scrollOffset = 0 } // Keep selection visible if m.selectedIdx >= m.scrollOffset+numVisibleModels { m.scrollOffset = m.selectedIdx - (numVisibleModels - 1) } } func (m *modelDialogCmp) switchProvider(offset int) { newOffset := m.hScrollOffset + offset // Ensure we stay within bounds if newOffset < 0 { newOffset = len(m.availableProviders) - 1 } if newOffset >= len(m.availableProviders) { newOffset = 0 } m.hScrollOffset = newOffset m.provider = m.availableProviders[m.hScrollOffset] m.setupModelsForProvider(m.provider) } func (m *modelDialogCmp) View() string { t := theme.CurrentTheme() baseStyle := styles.BaseStyle() // Capitalize first letter of provider name providerName := strings.ToUpper(string(m.provider)[:1]) + string(m.provider[1:]) title := baseStyle. Foreground(t.Primary()). Bold(true). Width(maxDialogWidth). Padding(0, 0, 1). Render(fmt.Sprintf("Select %s Model", providerName)) // Render visible models endIdx := min(m.scrollOffset+numVisibleModels, len(m.models)) modelItems := make([]string, 0, endIdx-m.scrollOffset) for i := m.scrollOffset; i < endIdx; i++ { itemStyle := baseStyle.Width(maxDialogWidth) if i == m.selectedIdx { itemStyle = itemStyle.Background(t.Primary()). Foreground(t.Background()).Bold(true) } modelItems = append(modelItems, itemStyle.Render(m.models[i].Name)) } scrollIndicator := m.getScrollIndicators(maxDialogWidth) content := lipgloss.JoinVertical( lipgloss.Left, title, baseStyle.Width(maxDialogWidth).Render(lipgloss.JoinVertical(lipgloss.Left, modelItems...)), scrollIndicator, ) return baseStyle.Padding(1, 2). Border(lipgloss.RoundedBorder()). BorderBackground(t.Background()). BorderForeground(t.TextMuted()). Width(lipgloss.Width(content) + 4). Render(content) } func (m *modelDialogCmp) getScrollIndicators(maxWidth int) string { var indicator string if len(m.models) > numVisibleModels { if m.scrollOffset > 0 { indicator += "↑ " } if m.scrollOffset+numVisibleModels < len(m.models) { indicator += "↓ " } } if m.hScrollPossible { if m.hScrollOffset > 0 { indicator = "← " + indicator } if m.hScrollOffset < len(m.availableProviders)-1 { indicator += "→" } } if indicator == "" { return "" } t := theme.CurrentTheme() baseStyle := styles.BaseStyle() return baseStyle. Foreground(t.Primary()). Width(maxWidth). Align(lipgloss.Right). Bold(true). Render(indicator) } func (m *modelDialogCmp) BindingKeys() []key.Binding { return layout.KeyMapToSlice(modelKeys) } func (m *modelDialogCmp) setupModels() { cfg := config.Get() modelInfo := GetSelectedModel(cfg) m.availableProviders = getEnabledProviders(cfg) m.hScrollPossible = len(m.availableProviders) > 1 m.provider = modelInfo.Provider m.hScrollOffset = findProviderIndex(m.availableProviders, m.provider) m.setupModelsForProvider(m.provider) } func GetSelectedModel(cfg *config.Config) models.Model { agentCfg := cfg.Agents[config.AgentCoder] selectedModelId := agentCfg.Model return models.SupportedModels[selectedModelId] } func getEnabledProviders(cfg *config.Config) []models.ModelProvider { var providers []models.ModelProvider for providerId, provider := range cfg.Providers { if !provider.Disabled { providers = append(providers, providerId) } } // Sort by provider popularity slices.SortFunc(providers, func(a, b models.ModelProvider) int { rA := models.ProviderPopularity[a] rB := models.ProviderPopularity[b] // models not included in popularity ranking default to last if rA == 0 { rA = 999 } if rB == 0 { rB = 999 } return rA - rB }) return providers } // findProviderIndex returns the index of the provider in the list, or -1 if not found func findProviderIndex(providers []models.ModelProvider, provider models.ModelProvider) int { for i, p := range providers { if p == provider { return i } } return -1 } func (m *modelDialogCmp) setupModelsForProvider(provider models.ModelProvider) { cfg := config.Get() agentCfg := cfg.Agents[config.AgentCoder] selectedModelId := agentCfg.Model m.provider = provider m.models = getModelsForProvider(provider) m.selectedIdx = 0 m.scrollOffset = 0 // Try to select the current model if it belongs to this provider if provider == models.SupportedModels[selectedModelId].Provider { for i, model := range m.models { if model.ID == selectedModelId { m.selectedIdx = i // Adjust scroll position to keep selected model visible if m.selectedIdx >= numVisibleModels { m.scrollOffset = m.selectedIdx - (numVisibleModels - 1) } break } } } } func getModelsForProvider(provider models.ModelProvider) []models.Model { var providerModels []models.Model for _, model := range models.SupportedModels { if model.Provider == provider { providerModels = append(providerModels, model) } } // reverse alphabetical order (if llm naming was consistent latest would appear first) slices.SortFunc(providerModels, func(a, b models.Model) int { if a.Name > b.Name { return -1 } else if a.Name < b.Name { return 1 } return 0 }) return providerModels } func NewModelDialogCmp() ModelDialog { return &modelDialogCmp{} } ================================================ FILE: internal/tui/components/dialog/permission.go ================================================ package dialog import ( "fmt" "strings" "github.com/charmbracelet/bubbles/key" "github.com/charmbracelet/bubbles/viewport" tea "github.com/charmbracelet/bubbletea" "github.com/charmbracelet/lipgloss" "github.com/opencode-ai/opencode/internal/diff" "github.com/opencode-ai/opencode/internal/llm/tools" "github.com/opencode-ai/opencode/internal/permission" "github.com/opencode-ai/opencode/internal/tui/layout" "github.com/opencode-ai/opencode/internal/tui/styles" "github.com/opencode-ai/opencode/internal/tui/theme" "github.com/opencode-ai/opencode/internal/tui/util" ) type PermissionAction string // Permission responses const ( PermissionAllow PermissionAction = "allow" PermissionAllowForSession PermissionAction = "allow_session" PermissionDeny PermissionAction = "deny" ) // PermissionResponseMsg represents the user's response to a permission request type PermissionResponseMsg struct { Permission permission.PermissionRequest Action PermissionAction } // PermissionDialogCmp interface for permission dialog component type PermissionDialogCmp interface { tea.Model layout.Bindings SetPermissions(permission permission.PermissionRequest) tea.Cmd } type permissionsMapping struct { Left key.Binding Right key.Binding EnterSpace key.Binding Allow key.Binding AllowSession key.Binding Deny key.Binding Tab key.Binding } var permissionsKeys = permissionsMapping{ Left: key.NewBinding( key.WithKeys("left"), key.WithHelp("←", "switch options"), ), Right: key.NewBinding( key.WithKeys("right"), key.WithHelp("→", "switch options"), ), EnterSpace: key.NewBinding( key.WithKeys("enter", " "), key.WithHelp("enter/space", "confirm"), ), Allow: key.NewBinding( key.WithKeys("a"), key.WithHelp("a", "allow"), ), AllowSession: key.NewBinding( key.WithKeys("s"), key.WithHelp("s", "allow for session"), ), Deny: key.NewBinding( key.WithKeys("d"), key.WithHelp("d", "deny"), ), Tab: key.NewBinding( key.WithKeys("tab"), key.WithHelp("tab", "switch options"), ), } // permissionDialogCmp is the implementation of PermissionDialog type permissionDialogCmp struct { width int height int permission permission.PermissionRequest windowSize tea.WindowSizeMsg contentViewPort viewport.Model selectedOption int // 0: Allow, 1: Allow for session, 2: Deny diffCache map[string]string markdownCache map[string]string } func (p *permissionDialogCmp) Init() tea.Cmd { return p.contentViewPort.Init() } func (p *permissionDialogCmp) Update(msg tea.Msg) (tea.Model, tea.Cmd) { var cmds []tea.Cmd switch msg := msg.(type) { case tea.WindowSizeMsg: p.windowSize = msg cmd := p.SetSize() cmds = append(cmds, cmd) p.markdownCache = make(map[string]string) p.diffCache = make(map[string]string) case tea.KeyMsg: switch { case key.Matches(msg, permissionsKeys.Right) || key.Matches(msg, permissionsKeys.Tab): p.selectedOption = (p.selectedOption + 1) % 3 return p, nil case key.Matches(msg, permissionsKeys.Left): p.selectedOption = (p.selectedOption + 2) % 3 case key.Matches(msg, permissionsKeys.EnterSpace): return p, p.selectCurrentOption() case key.Matches(msg, permissionsKeys.Allow): return p, util.CmdHandler(PermissionResponseMsg{Action: PermissionAllow, Permission: p.permission}) case key.Matches(msg, permissionsKeys.AllowSession): return p, util.CmdHandler(PermissionResponseMsg{Action: PermissionAllowForSession, Permission: p.permission}) case key.Matches(msg, permissionsKeys.Deny): return p, util.CmdHandler(PermissionResponseMsg{Action: PermissionDeny, Permission: p.permission}) default: // Pass other keys to viewport viewPort, cmd := p.contentViewPort.Update(msg) p.contentViewPort = viewPort cmds = append(cmds, cmd) } } return p, tea.Batch(cmds...) } func (p *permissionDialogCmp) selectCurrentOption() tea.Cmd { var action PermissionAction switch p.selectedOption { case 0: action = PermissionAllow case 1: action = PermissionAllowForSession case 2: action = PermissionDeny } return util.CmdHandler(PermissionResponseMsg{Action: action, Permission: p.permission}) } func (p *permissionDialogCmp) renderButtons() string { t := theme.CurrentTheme() baseStyle := styles.BaseStyle() allowStyle := baseStyle allowSessionStyle := baseStyle denyStyle := baseStyle spacerStyle := baseStyle.Background(t.Background()) // Style the selected button switch p.selectedOption { case 0: allowStyle = allowStyle.Background(t.Primary()).Foreground(t.Background()) allowSessionStyle = allowSessionStyle.Background(t.Background()).Foreground(t.Primary()) denyStyle = denyStyle.Background(t.Background()).Foreground(t.Primary()) case 1: allowStyle = allowStyle.Background(t.Background()).Foreground(t.Primary()) allowSessionStyle = allowSessionStyle.Background(t.Primary()).Foreground(t.Background()) denyStyle = denyStyle.Background(t.Background()).Foreground(t.Primary()) case 2: allowStyle = allowStyle.Background(t.Background()).Foreground(t.Primary()) allowSessionStyle = allowSessionStyle.Background(t.Background()).Foreground(t.Primary()) denyStyle = denyStyle.Background(t.Primary()).Foreground(t.Background()) } allowButton := allowStyle.Padding(0, 1).Render("Allow (a)") allowSessionButton := allowSessionStyle.Padding(0, 1).Render("Allow for session (s)") denyButton := denyStyle.Padding(0, 1).Render("Deny (d)") content := lipgloss.JoinHorizontal( lipgloss.Left, allowButton, spacerStyle.Render(" "), allowSessionButton, spacerStyle.Render(" "), denyButton, spacerStyle.Render(" "), ) remainingWidth := p.width - lipgloss.Width(content) if remainingWidth > 0 { content = spacerStyle.Render(strings.Repeat(" ", remainingWidth)) + content } return content } func (p *permissionDialogCmp) renderHeader() string { t := theme.CurrentTheme() baseStyle := styles.BaseStyle() toolKey := baseStyle.Foreground(t.TextMuted()).Bold(true).Render("Tool") toolValue := baseStyle. Foreground(t.Text()). Width(p.width - lipgloss.Width(toolKey)). Render(fmt.Sprintf(": %s", p.permission.ToolName)) pathKey := baseStyle.Foreground(t.TextMuted()).Bold(true).Render("Path") pathValue := baseStyle. Foreground(t.Text()). Width(p.width - lipgloss.Width(pathKey)). Render(fmt.Sprintf(": %s", p.permission.Path)) headerParts := []string{ lipgloss.JoinHorizontal( lipgloss.Left, toolKey, toolValue, ), baseStyle.Render(strings.Repeat(" ", p.width)), lipgloss.JoinHorizontal( lipgloss.Left, pathKey, pathValue, ), baseStyle.Render(strings.Repeat(" ", p.width)), } // Add tool-specific header information switch p.permission.ToolName { case tools.BashToolName: headerParts = append(headerParts, baseStyle.Foreground(t.TextMuted()).Width(p.width).Bold(true).Render("Command")) case tools.EditToolName: params := p.permission.Params.(tools.EditPermissionsParams) fileKey := baseStyle.Foreground(t.TextMuted()).Bold(true).Render("File") filePath := baseStyle. Foreground(t.Text()). Width(p.width - lipgloss.Width(fileKey)). Render(fmt.Sprintf(": %s", params.FilePath)) headerParts = append(headerParts, lipgloss.JoinHorizontal( lipgloss.Left, fileKey, filePath, ), baseStyle.Render(strings.Repeat(" ", p.width)), ) case tools.WriteToolName: params := p.permission.Params.(tools.WritePermissionsParams) fileKey := baseStyle.Foreground(t.TextMuted()).Bold(true).Render("File") filePath := baseStyle. Foreground(t.Text()). Width(p.width - lipgloss.Width(fileKey)). Render(fmt.Sprintf(": %s", params.FilePath)) headerParts = append(headerParts, lipgloss.JoinHorizontal( lipgloss.Left, fileKey, filePath, ), baseStyle.Render(strings.Repeat(" ", p.width)), ) case tools.FetchToolName: headerParts = append(headerParts, baseStyle.Foreground(t.TextMuted()).Width(p.width).Bold(true).Render("URL")) } return lipgloss.NewStyle().Background(t.Background()).Render(lipgloss.JoinVertical(lipgloss.Left, headerParts...)) } func (p *permissionDialogCmp) renderBashContent() string { t := theme.CurrentTheme() baseStyle := styles.BaseStyle() if pr, ok := p.permission.Params.(tools.BashPermissionsParams); ok { content := fmt.Sprintf("```bash\n%s\n```", pr.Command) // Use the cache for markdown rendering renderedContent := p.GetOrSetMarkdown(p.permission.ID, func() (string, error) { r := styles.GetMarkdownRenderer(p.width - 10) s, err := r.Render(content) return styles.ForceReplaceBackgroundWithLipgloss(s, t.Background()), err }) finalContent := baseStyle. Width(p.contentViewPort.Width). Render(renderedContent) p.contentViewPort.SetContent(finalContent) return p.styleViewport() } return "" } func (p *permissionDialogCmp) renderEditContent() string { if pr, ok := p.permission.Params.(tools.EditPermissionsParams); ok { diff := p.GetOrSetDiff(p.permission.ID, func() (string, error) { return diff.FormatDiff(pr.Diff, diff.WithTotalWidth(p.contentViewPort.Width)) }) p.contentViewPort.SetContent(diff) return p.styleViewport() } return "" } func (p *permissionDialogCmp) renderPatchContent() string { if pr, ok := p.permission.Params.(tools.EditPermissionsParams); ok { diff := p.GetOrSetDiff(p.permission.ID, func() (string, error) { return diff.FormatDiff(pr.Diff, diff.WithTotalWidth(p.contentViewPort.Width)) }) p.contentViewPort.SetContent(diff) return p.styleViewport() } return "" } func (p *permissionDialogCmp) renderWriteContent() string { if pr, ok := p.permission.Params.(tools.WritePermissionsParams); ok { // Use the cache for diff rendering diff := p.GetOrSetDiff(p.permission.ID, func() (string, error) { return diff.FormatDiff(pr.Diff, diff.WithTotalWidth(p.contentViewPort.Width)) }) p.contentViewPort.SetContent(diff) return p.styleViewport() } return "" } func (p *permissionDialogCmp) renderFetchContent() string { t := theme.CurrentTheme() baseStyle := styles.BaseStyle() if pr, ok := p.permission.Params.(tools.FetchPermissionsParams); ok { content := fmt.Sprintf("```bash\n%s\n```", pr.URL) // Use the cache for markdown rendering renderedContent := p.GetOrSetMarkdown(p.permission.ID, func() (string, error) { r := styles.GetMarkdownRenderer(p.width - 10) s, err := r.Render(content) return styles.ForceReplaceBackgroundWithLipgloss(s, t.Background()), err }) finalContent := baseStyle. Width(p.contentViewPort.Width). Render(renderedContent) p.contentViewPort.SetContent(finalContent) return p.styleViewport() } return "" } func (p *permissionDialogCmp) renderDefaultContent() string { t := theme.CurrentTheme() baseStyle := styles.BaseStyle() content := p.permission.Description // Use the cache for markdown rendering renderedContent := p.GetOrSetMarkdown(p.permission.ID, func() (string, error) { r := styles.GetMarkdownRenderer(p.width - 10) s, err := r.Render(content) return styles.ForceReplaceBackgroundWithLipgloss(s, t.Background()), err }) finalContent := baseStyle. Width(p.contentViewPort.Width). Render(renderedContent) p.contentViewPort.SetContent(finalContent) if renderedContent == "" { return "" } return p.styleViewport() } func (p *permissionDialogCmp) styleViewport() string { t := theme.CurrentTheme() contentStyle := lipgloss.NewStyle(). Background(t.Background()) return contentStyle.Render(p.contentViewPort.View()) } func (p *permissionDialogCmp) render() string { t := theme.CurrentTheme() baseStyle := styles.BaseStyle() title := baseStyle. Bold(true). Width(p.width - 4). Foreground(t.Primary()). Render("Permission Required") // Render header headerContent := p.renderHeader() // Render buttons buttons := p.renderButtons() // Calculate content height dynamically based on window size p.contentViewPort.Height = p.height - lipgloss.Height(headerContent) - lipgloss.Height(buttons) - 2 - lipgloss.Height(title) p.contentViewPort.Width = p.width - 4 // Render content based on tool type var contentFinal string switch p.permission.ToolName { case tools.BashToolName: contentFinal = p.renderBashContent() case tools.EditToolName: contentFinal = p.renderEditContent() case tools.PatchToolName: contentFinal = p.renderPatchContent() case tools.WriteToolName: contentFinal = p.renderWriteContent() case tools.FetchToolName: contentFinal = p.renderFetchContent() default: contentFinal = p.renderDefaultContent() } content := lipgloss.JoinVertical( lipgloss.Top, title, baseStyle.Render(strings.Repeat(" ", lipgloss.Width(title))), headerContent, contentFinal, buttons, baseStyle.Render(strings.Repeat(" ", p.width-4)), ) return baseStyle. Padding(1, 0, 0, 1). Border(lipgloss.RoundedBorder()). BorderBackground(t.Background()). BorderForeground(t.TextMuted()). Width(p.width). Height(p.height). Render( content, ) } func (p *permissionDialogCmp) View() string { return p.render() } func (p *permissionDialogCmp) BindingKeys() []key.Binding { return layout.KeyMapToSlice(permissionsKeys) } func (p *permissionDialogCmp) SetSize() tea.Cmd { if p.permission.ID == "" { return nil } switch p.permission.ToolName { case tools.BashToolName: p.width = int(float64(p.windowSize.Width) * 0.4) p.height = int(float64(p.windowSize.Height) * 0.3) case tools.EditToolName: p.width = int(float64(p.windowSize.Width) * 0.8) p.height = int(float64(p.windowSize.Height) * 0.8) case tools.WriteToolName: p.width = int(float64(p.windowSize.Width) * 0.8) p.height = int(float64(p.windowSize.Height) * 0.8) case tools.FetchToolName: p.width = int(float64(p.windowSize.Width) * 0.4) p.height = int(float64(p.windowSize.Height) * 0.3) default: p.width = int(float64(p.windowSize.Width) * 0.7) p.height = int(float64(p.windowSize.Height) * 0.5) } return nil } func (p *permissionDialogCmp) SetPermissions(permission permission.PermissionRequest) tea.Cmd { p.permission = permission return p.SetSize() } // Helper to get or set cached diff content func (c *permissionDialogCmp) GetOrSetDiff(key string, generator func() (string, error)) string { if cached, ok := c.diffCache[key]; ok { return cached } content, err := generator() if err != nil { return fmt.Sprintf("Error formatting diff: %v", err) } c.diffCache[key] = content return content } // Helper to get or set cached markdown content func (c *permissionDialogCmp) GetOrSetMarkdown(key string, generator func() (string, error)) string { if cached, ok := c.markdownCache[key]; ok { return cached } content, err := generator() if err != nil { return fmt.Sprintf("Error rendering markdown: %v", err) } c.markdownCache[key] = content return content } func NewPermissionDialogCmp() PermissionDialogCmp { // Create viewport for content contentViewport := viewport.New(0, 0) return &permissionDialogCmp{ contentViewPort: contentViewport, selectedOption: 0, // Default to "Allow" diffCache: make(map[string]string), markdownCache: make(map[string]string), } } ================================================ FILE: internal/tui/components/dialog/quit.go ================================================ package dialog import ( "strings" "github.com/charmbracelet/bubbles/key" tea "github.com/charmbracelet/bubbletea" "github.com/charmbracelet/lipgloss" "github.com/opencode-ai/opencode/internal/tui/layout" "github.com/opencode-ai/opencode/internal/tui/styles" "github.com/opencode-ai/opencode/internal/tui/theme" "github.com/opencode-ai/opencode/internal/tui/util" ) const question = "Are you sure you want to quit?" type CloseQuitMsg struct{} type QuitDialog interface { tea.Model layout.Bindings } type quitDialogCmp struct { selectedNo bool } type helpMapping struct { LeftRight key.Binding EnterSpace key.Binding Yes key.Binding No key.Binding Tab key.Binding } var helpKeys = helpMapping{ LeftRight: key.NewBinding( key.WithKeys("left", "right"), key.WithHelp("←/→", "switch options"), ), EnterSpace: key.NewBinding( key.WithKeys("enter", " "), key.WithHelp("enter/space", "confirm"), ), Yes: key.NewBinding( key.WithKeys("y", "Y"), key.WithHelp("y/Y", "yes"), ), No: key.NewBinding( key.WithKeys("n", "N"), key.WithHelp("n/N", "no"), ), Tab: key.NewBinding( key.WithKeys("tab"), key.WithHelp("tab", "switch options"), ), } func (q *quitDialogCmp) Init() tea.Cmd { return nil } func (q *quitDialogCmp) Update(msg tea.Msg) (tea.Model, tea.Cmd) { switch msg := msg.(type) { case tea.KeyMsg: switch { case key.Matches(msg, helpKeys.LeftRight) || key.Matches(msg, helpKeys.Tab): q.selectedNo = !q.selectedNo return q, nil case key.Matches(msg, helpKeys.EnterSpace): if !q.selectedNo { return q, tea.Quit } return q, util.CmdHandler(CloseQuitMsg{}) case key.Matches(msg, helpKeys.Yes): return q, tea.Quit case key.Matches(msg, helpKeys.No): return q, util.CmdHandler(CloseQuitMsg{}) } } return q, nil } func (q *quitDialogCmp) View() string { t := theme.CurrentTheme() baseStyle := styles.BaseStyle() yesStyle := baseStyle noStyle := baseStyle spacerStyle := baseStyle.Background(t.Background()) if q.selectedNo { noStyle = noStyle.Background(t.Primary()).Foreground(t.Background()) yesStyle = yesStyle.Background(t.Background()).Foreground(t.Primary()) } else { yesStyle = yesStyle.Background(t.Primary()).Foreground(t.Background()) noStyle = noStyle.Background(t.Background()).Foreground(t.Primary()) } yesButton := yesStyle.Padding(0, 1).Render("Yes") noButton := noStyle.Padding(0, 1).Render("No") buttons := lipgloss.JoinHorizontal(lipgloss.Left, yesButton, spacerStyle.Render(" "), noButton) width := lipgloss.Width(question) remainingWidth := width - lipgloss.Width(buttons) if remainingWidth > 0 { buttons = spacerStyle.Render(strings.Repeat(" ", remainingWidth)) + buttons } content := baseStyle.Render( lipgloss.JoinVertical( lipgloss.Center, question, "", buttons, ), ) return baseStyle.Padding(1, 2). Border(lipgloss.RoundedBorder()). BorderBackground(t.Background()). BorderForeground(t.TextMuted()). Width(lipgloss.Width(content) + 4). Render(content) } func (q *quitDialogCmp) BindingKeys() []key.Binding { return layout.KeyMapToSlice(helpKeys) } func NewQuitCmp() QuitDialog { return &quitDialogCmp{ selectedNo: true, } } ================================================ FILE: internal/tui/components/dialog/session.go ================================================ package dialog import ( "github.com/charmbracelet/bubbles/key" tea "github.com/charmbracelet/bubbletea" "github.com/charmbracelet/lipgloss" "github.com/opencode-ai/opencode/internal/session" "github.com/opencode-ai/opencode/internal/tui/layout" "github.com/opencode-ai/opencode/internal/tui/styles" "github.com/opencode-ai/opencode/internal/tui/theme" "github.com/opencode-ai/opencode/internal/tui/util" ) // SessionSelectedMsg is sent when a session is selected type SessionSelectedMsg struct { Session session.Session } // CloseSessionDialogMsg is sent when the session dialog is closed type CloseSessionDialogMsg struct{} // SessionDialog interface for the session switching dialog type SessionDialog interface { tea.Model layout.Bindings SetSessions(sessions []session.Session) SetSelectedSession(sessionID string) } type sessionDialogCmp struct { sessions []session.Session selectedIdx int width int height int selectedSessionID string } type sessionKeyMap struct { Up key.Binding Down key.Binding Enter key.Binding Escape key.Binding J key.Binding K key.Binding } var sessionKeys = sessionKeyMap{ Up: key.NewBinding( key.WithKeys("up"), key.WithHelp("↑", "previous session"), ), Down: key.NewBinding( key.WithKeys("down"), key.WithHelp("↓", "next session"), ), Enter: key.NewBinding( key.WithKeys("enter"), key.WithHelp("enter", "select session"), ), Escape: key.NewBinding( key.WithKeys("esc"), key.WithHelp("esc", "close"), ), J: key.NewBinding( key.WithKeys("j"), key.WithHelp("j", "next session"), ), K: key.NewBinding( key.WithKeys("k"), key.WithHelp("k", "previous session"), ), } func (s *sessionDialogCmp) Init() tea.Cmd { return nil } func (s *sessionDialogCmp) Update(msg tea.Msg) (tea.Model, tea.Cmd) { switch msg := msg.(type) { case tea.KeyMsg: switch { case key.Matches(msg, sessionKeys.Up) || key.Matches(msg, sessionKeys.K): if s.selectedIdx > 0 { s.selectedIdx-- } return s, nil case key.Matches(msg, sessionKeys.Down) || key.Matches(msg, sessionKeys.J): if s.selectedIdx < len(s.sessions)-1 { s.selectedIdx++ } return s, nil case key.Matches(msg, sessionKeys.Enter): if len(s.sessions) > 0 { return s, util.CmdHandler(SessionSelectedMsg{ Session: s.sessions[s.selectedIdx], }) } case key.Matches(msg, sessionKeys.Escape): return s, util.CmdHandler(CloseSessionDialogMsg{}) } case tea.WindowSizeMsg: s.width = msg.Width s.height = msg.Height } return s, nil } func (s *sessionDialogCmp) View() string { t := theme.CurrentTheme() baseStyle := styles.BaseStyle() if len(s.sessions) == 0 { return baseStyle.Padding(1, 2). Border(lipgloss.RoundedBorder()). BorderBackground(t.Background()). BorderForeground(t.TextMuted()). Width(40). Render("No sessions available") } // Calculate max width needed for session titles maxWidth := 40 // Minimum width for _, sess := range s.sessions { if len(sess.Title) > maxWidth-4 { // Account for padding maxWidth = len(sess.Title) + 4 } } maxWidth = max(30, min(maxWidth, s.width-15)) // Limit width to avoid overflow // Limit height to avoid taking up too much screen space maxVisibleSessions := min(10, len(s.sessions)) // Build the session list sessionItems := make([]string, 0, maxVisibleSessions) startIdx := 0 // If we have more sessions than can be displayed, adjust the start index if len(s.sessions) > maxVisibleSessions { // Center the selected item when possible halfVisible := maxVisibleSessions / 2 if s.selectedIdx >= halfVisible && s.selectedIdx < len(s.sessions)-halfVisible { startIdx = s.selectedIdx - halfVisible } else if s.selectedIdx >= len(s.sessions)-halfVisible { startIdx = len(s.sessions) - maxVisibleSessions } } endIdx := min(startIdx+maxVisibleSessions, len(s.sessions)) for i := startIdx; i < endIdx; i++ { sess := s.sessions[i] itemStyle := baseStyle.Width(maxWidth) if i == s.selectedIdx { itemStyle = itemStyle. Background(t.Primary()). Foreground(t.Background()). Bold(true) } sessionItems = append(sessionItems, itemStyle.Padding(0, 1).Render(sess.Title)) } title := baseStyle. Foreground(t.Primary()). Bold(true). Width(maxWidth). Padding(0, 1). Render("Switch Session") content := lipgloss.JoinVertical( lipgloss.Left, title, baseStyle.Width(maxWidth).Render(""), baseStyle.Width(maxWidth).Render(lipgloss.JoinVertical(lipgloss.Left, sessionItems...)), baseStyle.Width(maxWidth).Render(""), ) return baseStyle.Padding(1, 2). Border(lipgloss.RoundedBorder()). BorderBackground(t.Background()). BorderForeground(t.TextMuted()). Width(lipgloss.Width(content) + 4). Render(content) } func (s *sessionDialogCmp) BindingKeys() []key.Binding { return layout.KeyMapToSlice(sessionKeys) } func (s *sessionDialogCmp) SetSessions(sessions []session.Session) { s.sessions = sessions // If we have a selected session ID, find its index if s.selectedSessionID != "" { for i, sess := range sessions { if sess.ID == s.selectedSessionID { s.selectedIdx = i return } } } // Default to first session if selected not found s.selectedIdx = 0 } func (s *sessionDialogCmp) SetSelectedSession(sessionID string) { s.selectedSessionID = sessionID // Update the selected index if sessions are already loaded if len(s.sessions) > 0 { for i, sess := range s.sessions { if sess.ID == sessionID { s.selectedIdx = i return } } } } // NewSessionDialogCmp creates a new session switching dialog func NewSessionDialogCmp() SessionDialog { return &sessionDialogCmp{ sessions: []session.Session{}, selectedIdx: 0, selectedSessionID: "", } } ================================================ FILE: internal/tui/components/dialog/theme.go ================================================ package dialog import ( "github.com/charmbracelet/bubbles/key" tea "github.com/charmbracelet/bubbletea" "github.com/charmbracelet/lipgloss" "github.com/opencode-ai/opencode/internal/tui/layout" "github.com/opencode-ai/opencode/internal/tui/styles" "github.com/opencode-ai/opencode/internal/tui/theme" "github.com/opencode-ai/opencode/internal/tui/util" ) // ThemeChangedMsg is sent when the theme is changed type ThemeChangedMsg struct { ThemeName string } // CloseThemeDialogMsg is sent when the theme dialog is closed type CloseThemeDialogMsg struct{} // ThemeDialog interface for the theme switching dialog type ThemeDialog interface { tea.Model layout.Bindings } type themeDialogCmp struct { themes []string selectedIdx int width int height int currentTheme string } type themeKeyMap struct { Up key.Binding Down key.Binding Enter key.Binding Escape key.Binding J key.Binding K key.Binding } var themeKeys = themeKeyMap{ Up: key.NewBinding( key.WithKeys("up"), key.WithHelp("↑", "previous theme"), ), Down: key.NewBinding( key.WithKeys("down"), key.WithHelp("↓", "next theme"), ), Enter: key.NewBinding( key.WithKeys("enter"), key.WithHelp("enter", "select theme"), ), Escape: key.NewBinding( key.WithKeys("esc"), key.WithHelp("esc", "close"), ), J: key.NewBinding( key.WithKeys("j"), key.WithHelp("j", "next theme"), ), K: key.NewBinding( key.WithKeys("k"), key.WithHelp("k", "previous theme"), ), } func (t *themeDialogCmp) Init() tea.Cmd { // Load available themes and update selectedIdx based on current theme t.themes = theme.AvailableThemes() t.currentTheme = theme.CurrentThemeName() // Find the current theme in the list for i, name := range t.themes { if name == t.currentTheme { t.selectedIdx = i break } } return nil } func (t *themeDialogCmp) Update(msg tea.Msg) (tea.Model, tea.Cmd) { switch msg := msg.(type) { case tea.KeyMsg: switch { case key.Matches(msg, themeKeys.Up) || key.Matches(msg, themeKeys.K): if t.selectedIdx > 0 { t.selectedIdx-- } return t, nil case key.Matches(msg, themeKeys.Down) || key.Matches(msg, themeKeys.J): if t.selectedIdx < len(t.themes)-1 { t.selectedIdx++ } return t, nil case key.Matches(msg, themeKeys.Enter): if len(t.themes) > 0 { previousTheme := theme.CurrentThemeName() selectedTheme := t.themes[t.selectedIdx] if previousTheme == selectedTheme { return t, util.CmdHandler(CloseThemeDialogMsg{}) } if err := theme.SetTheme(selectedTheme); err != nil { return t, util.ReportError(err) } return t, util.CmdHandler(ThemeChangedMsg{ ThemeName: selectedTheme, }) } case key.Matches(msg, themeKeys.Escape): return t, util.CmdHandler(CloseThemeDialogMsg{}) } case tea.WindowSizeMsg: t.width = msg.Width t.height = msg.Height } return t, nil } func (t *themeDialogCmp) View() string { currentTheme := theme.CurrentTheme() baseStyle := styles.BaseStyle() if len(t.themes) == 0 { return baseStyle.Padding(1, 2). Border(lipgloss.RoundedBorder()). BorderBackground(currentTheme.Background()). BorderForeground(currentTheme.TextMuted()). Width(40). Render("No themes available") } // Calculate max width needed for theme names maxWidth := 40 // Minimum width for _, themeName := range t.themes { if len(themeName) > maxWidth-4 { // Account for padding maxWidth = len(themeName) + 4 } } maxWidth = max(30, min(maxWidth, t.width-15)) // Limit width to avoid overflow // Build the theme list themeItems := make([]string, 0, len(t.themes)) for i, themeName := range t.themes { itemStyle := baseStyle.Width(maxWidth) if i == t.selectedIdx { itemStyle = itemStyle. Background(currentTheme.Primary()). Foreground(currentTheme.Background()). Bold(true) } themeItems = append(themeItems, itemStyle.Padding(0, 1).Render(themeName)) } title := baseStyle. Foreground(currentTheme.Primary()). Bold(true). Width(maxWidth). Padding(0, 1). Render("Select Theme") content := lipgloss.JoinVertical( lipgloss.Left, title, baseStyle.Width(maxWidth).Render(""), baseStyle.Width(maxWidth).Render(lipgloss.JoinVertical(lipgloss.Left, themeItems...)), baseStyle.Width(maxWidth).Render(""), ) return baseStyle.Padding(1, 2). Border(lipgloss.RoundedBorder()). BorderBackground(currentTheme.Background()). BorderForeground(currentTheme.TextMuted()). Width(lipgloss.Width(content) + 4). Render(content) } func (t *themeDialogCmp) BindingKeys() []key.Binding { return layout.KeyMapToSlice(themeKeys) } // NewThemeDialogCmp creates a new theme switching dialog func NewThemeDialogCmp() ThemeDialog { return &themeDialogCmp{ themes: []string{}, selectedIdx: 0, currentTheme: "", } } ================================================ FILE: internal/tui/components/logs/details.go ================================================ package logs import ( "fmt" "strings" "time" "github.com/charmbracelet/bubbles/key" "github.com/charmbracelet/bubbles/viewport" tea "github.com/charmbracelet/bubbletea" "github.com/charmbracelet/lipgloss" "github.com/opencode-ai/opencode/internal/logging" "github.com/opencode-ai/opencode/internal/tui/layout" "github.com/opencode-ai/opencode/internal/tui/styles" "github.com/opencode-ai/opencode/internal/tui/theme" ) type DetailComponent interface { tea.Model layout.Sizeable layout.Bindings } type detailCmp struct { width, height int currentLog logging.LogMessage viewport viewport.Model } func (i *detailCmp) Init() tea.Cmd { messages := logging.List() if len(messages) == 0 { return nil } i.currentLog = messages[0] return nil } func (i *detailCmp) Update(msg tea.Msg) (tea.Model, tea.Cmd) { switch msg := msg.(type) { case selectedLogMsg: if msg.ID != i.currentLog.ID { i.currentLog = logging.LogMessage(msg) i.updateContent() } } return i, nil } func (i *detailCmp) updateContent() { var content strings.Builder t := theme.CurrentTheme() // Format the header with timestamp and level timeStyle := lipgloss.NewStyle().Foreground(t.TextMuted()) levelStyle := getLevelStyle(i.currentLog.Level) header := lipgloss.JoinHorizontal( lipgloss.Center, timeStyle.Render(i.currentLog.Time.Format(time.RFC3339)), " ", levelStyle.Render(i.currentLog.Level), ) content.WriteString(lipgloss.NewStyle().Bold(true).Render(header)) content.WriteString("\n\n") // Message with styling messageStyle := lipgloss.NewStyle().Bold(true).Foreground(t.Text()) content.WriteString(messageStyle.Render("Message:")) content.WriteString("\n") content.WriteString(lipgloss.NewStyle().Padding(0, 2).Render(i.currentLog.Message)) content.WriteString("\n\n") // Attributes section if len(i.currentLog.Attributes) > 0 { attrHeaderStyle := lipgloss.NewStyle().Bold(true).Foreground(t.Text()) content.WriteString(attrHeaderStyle.Render("Attributes:")) content.WriteString("\n") // Create a table-like display for attributes keyStyle := lipgloss.NewStyle().Foreground(t.Primary()).Bold(true) valueStyle := lipgloss.NewStyle().Foreground(t.Text()) for _, attr := range i.currentLog.Attributes { attrLine := fmt.Sprintf("%s: %s", keyStyle.Render(attr.Key), valueStyle.Render(attr.Value), ) content.WriteString(lipgloss.NewStyle().Padding(0, 2).Render(attrLine)) content.WriteString("\n") } } i.viewport.SetContent(content.String()) } func getLevelStyle(level string) lipgloss.Style { style := lipgloss.NewStyle().Bold(true) t := theme.CurrentTheme() switch strings.ToLower(level) { case "info": return style.Foreground(t.Info()) case "warn", "warning": return style.Foreground(t.Warning()) case "error", "err": return style.Foreground(t.Error()) case "debug": return style.Foreground(t.Success()) default: return style.Foreground(t.Text()) } } func (i *detailCmp) View() string { t := theme.CurrentTheme() return styles.ForceReplaceBackgroundWithLipgloss(i.viewport.View(), t.Background()) } func (i *detailCmp) GetSize() (int, int) { return i.width, i.height } func (i *detailCmp) SetSize(width int, height int) tea.Cmd { i.width = width i.height = height i.viewport.Width = i.width i.viewport.Height = i.height i.updateContent() return nil } func (i *detailCmp) BindingKeys() []key.Binding { return layout.KeyMapToSlice(i.viewport.KeyMap) } func NewLogsDetails() DetailComponent { return &detailCmp{ viewport: viewport.New(0, 0), } } ================================================ FILE: internal/tui/components/logs/table.go ================================================ package logs import ( "encoding/json" "slices" "github.com/charmbracelet/bubbles/key" "github.com/charmbracelet/bubbles/table" tea "github.com/charmbracelet/bubbletea" "github.com/opencode-ai/opencode/internal/logging" "github.com/opencode-ai/opencode/internal/pubsub" "github.com/opencode-ai/opencode/internal/tui/layout" "github.com/opencode-ai/opencode/internal/tui/styles" "github.com/opencode-ai/opencode/internal/tui/theme" "github.com/opencode-ai/opencode/internal/tui/util" ) type TableComponent interface { tea.Model layout.Sizeable layout.Bindings } type tableCmp struct { table table.Model } type selectedLogMsg logging.LogMessage func (i *tableCmp) Init() tea.Cmd { i.setRows() return nil } func (i *tableCmp) Update(msg tea.Msg) (tea.Model, tea.Cmd) { var cmds []tea.Cmd switch msg.(type) { case pubsub.Event[logging.LogMessage]: i.setRows() return i, nil } prevSelectedRow := i.table.SelectedRow() t, cmd := i.table.Update(msg) cmds = append(cmds, cmd) i.table = t selectedRow := i.table.SelectedRow() if selectedRow != nil { if prevSelectedRow == nil || selectedRow[0] == prevSelectedRow[0] { var log logging.LogMessage for _, row := range logging.List() { if row.ID == selectedRow[0] { log = row break } } if log.ID != "" { cmds = append(cmds, util.CmdHandler(selectedLogMsg(log))) } } } return i, tea.Batch(cmds...) } func (i *tableCmp) View() string { t := theme.CurrentTheme() defaultStyles := table.DefaultStyles() defaultStyles.Selected = defaultStyles.Selected.Foreground(t.Primary()) i.table.SetStyles(defaultStyles) return styles.ForceReplaceBackgroundWithLipgloss(i.table.View(), t.Background()) } func (i *tableCmp) GetSize() (int, int) { return i.table.Width(), i.table.Height() } func (i *tableCmp) SetSize(width int, height int) tea.Cmd { i.table.SetWidth(width) i.table.SetHeight(height) cloumns := i.table.Columns() for i, col := range cloumns { col.Width = (width / len(cloumns)) - 2 cloumns[i] = col } i.table.SetColumns(cloumns) return nil } func (i *tableCmp) BindingKeys() []key.Binding { return layout.KeyMapToSlice(i.table.KeyMap) } func (i *tableCmp) setRows() { rows := []table.Row{} logs := logging.List() slices.SortFunc(logs, func(a, b logging.LogMessage) int { if a.Time.Before(b.Time) { return 1 } if a.Time.After(b.Time) { return -1 } return 0 }) for _, log := range logs { bm, _ := json.Marshal(log.Attributes) row := table.Row{ log.ID, log.Time.Format("15:04:05"), log.Level, log.Message, string(bm), } rows = append(rows, row) } i.table.SetRows(rows) } func NewLogsTable() TableComponent { columns := []table.Column{ {Title: "ID", Width: 4}, {Title: "Time", Width: 4}, {Title: "Level", Width: 10}, {Title: "Message", Width: 10}, {Title: "Attributes", Width: 10}, } tableModel := table.New( table.WithColumns(columns), ) tableModel.Focus() return &tableCmp{ table: tableModel, } } ================================================ FILE: internal/tui/components/util/simple-list.go ================================================ package utilComponents import ( "github.com/charmbracelet/bubbles/key" tea "github.com/charmbracelet/bubbletea" "github.com/charmbracelet/lipgloss" "github.com/opencode-ai/opencode/internal/tui/layout" "github.com/opencode-ai/opencode/internal/tui/styles" "github.com/opencode-ai/opencode/internal/tui/theme" ) type SimpleListItem interface { Render(selected bool, width int) string } type SimpleList[T SimpleListItem] interface { tea.Model layout.Bindings SetMaxWidth(maxWidth int) GetSelectedItem() (item T, idx int) SetItems(items []T) GetItems() []T } type simpleListCmp[T SimpleListItem] struct { fallbackMsg string items []T selectedIdx int maxWidth int maxVisibleItems int useAlphaNumericKeys bool width int height int } type simpleListKeyMap struct { Up key.Binding Down key.Binding UpAlpha key.Binding DownAlpha key.Binding } var simpleListKeys = simpleListKeyMap{ Up: key.NewBinding( key.WithKeys("up"), key.WithHelp("↑", "previous list item"), ), Down: key.NewBinding( key.WithKeys("down"), key.WithHelp("↓", "next list item"), ), UpAlpha: key.NewBinding( key.WithKeys("k"), key.WithHelp("k", "previous list item"), ), DownAlpha: key.NewBinding( key.WithKeys("j"), key.WithHelp("j", "next list item"), ), } func (c *simpleListCmp[T]) Init() tea.Cmd { return nil } func (c *simpleListCmp[T]) Update(msg tea.Msg) (tea.Model, tea.Cmd) { switch msg := msg.(type) { case tea.KeyMsg: switch { case key.Matches(msg, simpleListKeys.Up) || (c.useAlphaNumericKeys && key.Matches(msg, simpleListKeys.UpAlpha)): if c.selectedIdx > 0 { c.selectedIdx-- } return c, nil case key.Matches(msg, simpleListKeys.Down) || (c.useAlphaNumericKeys && key.Matches(msg, simpleListKeys.DownAlpha)): if c.selectedIdx < len(c.items)-1 { c.selectedIdx++ } return c, nil } } return c, nil } func (c *simpleListCmp[T]) BindingKeys() []key.Binding { return layout.KeyMapToSlice(simpleListKeys) } func (c *simpleListCmp[T]) GetSelectedItem() (T, int) { if len(c.items) > 0 { return c.items[c.selectedIdx], c.selectedIdx } var zero T return zero, -1 } func (c *simpleListCmp[T]) SetItems(items []T) { c.selectedIdx = 0 c.items = items } func (c *simpleListCmp[T]) GetItems() []T { return c.items } func (c *simpleListCmp[T]) SetMaxWidth(width int) { c.maxWidth = width } func (c *simpleListCmp[T]) View() string { t := theme.CurrentTheme() baseStyle := styles.BaseStyle() items := c.items maxWidth := c.maxWidth maxVisibleItems := min(c.maxVisibleItems, len(items)) startIdx := 0 if len(items) <= 0 { return baseStyle. Background(t.Background()). Padding(0, 1). Width(maxWidth). Render(c.fallbackMsg) } if len(items) > maxVisibleItems { halfVisible := maxVisibleItems / 2 if c.selectedIdx >= halfVisible && c.selectedIdx < len(items)-halfVisible { startIdx = c.selectedIdx - halfVisible } else if c.selectedIdx >= len(items)-halfVisible { startIdx = len(items) - maxVisibleItems } } endIdx := min(startIdx+maxVisibleItems, len(items)) listItems := make([]string, 0, maxVisibleItems) for i := startIdx; i < endIdx; i++ { item := items[i] title := item.Render(i == c.selectedIdx, maxWidth) listItems = append(listItems, title) } return lipgloss.JoinVertical(lipgloss.Left, listItems...) } func NewSimpleList[T SimpleListItem](items []T, maxVisibleItems int, fallbackMsg string, useAlphaNumericKeys bool) SimpleList[T] { return &simpleListCmp[T]{ fallbackMsg: fallbackMsg, items: items, maxVisibleItems: maxVisibleItems, useAlphaNumericKeys: useAlphaNumericKeys, selectedIdx: 0, } } ================================================ FILE: internal/tui/image/images.go ================================================ package image import ( "fmt" "image" "os" "strings" "github.com/charmbracelet/lipgloss" "github.com/disintegration/imaging" "github.com/lucasb-eyer/go-colorful" ) func ValidateFileSize(filePath string, sizeLimit int64) (bool, error) { fileInfo, err := os.Stat(filePath) if err != nil { return false, fmt.Errorf("error getting file info: %w", err) } if fileInfo.Size() > sizeLimit { return true, nil } return false, nil } func ToString(width int, img image.Image) string { img = imaging.Resize(img, width, 0, imaging.Lanczos) b := img.Bounds() imageWidth := b.Max.X h := b.Max.Y str := strings.Builder{} for heightCounter := 0; heightCounter < h; heightCounter += 2 { for x := range imageWidth { c1, _ := colorful.MakeColor(img.At(x, heightCounter)) color1 := lipgloss.Color(c1.Hex()) var color2 lipgloss.Color if heightCounter+1 < h { c2, _ := colorful.MakeColor(img.At(x, heightCounter+1)) color2 = lipgloss.Color(c2.Hex()) } else { color2 = color1 } str.WriteString(lipgloss.NewStyle().Foreground(color1). Background(color2).Render("▀")) } str.WriteString("\n") } return str.String() } func ImagePreview(width int, filename string) (string, error) { imageContent, err := os.Open(filename) if err != nil { return "", err } defer imageContent.Close() img, _, err := image.Decode(imageContent) if err != nil { return "", err } imageString := ToString(width, img) return imageString, nil } ================================================ FILE: internal/tui/layout/container.go ================================================ package layout import ( "github.com/charmbracelet/bubbles/key" tea "github.com/charmbracelet/bubbletea" "github.com/charmbracelet/lipgloss" "github.com/opencode-ai/opencode/internal/tui/theme" ) type Container interface { tea.Model Sizeable Bindings } type container struct { width int height int content tea.Model // Style options paddingTop int paddingRight int paddingBottom int paddingLeft int borderTop bool borderRight bool borderBottom bool borderLeft bool borderStyle lipgloss.Border } func (c *container) Init() tea.Cmd { return c.content.Init() } func (c *container) Update(msg tea.Msg) (tea.Model, tea.Cmd) { u, cmd := c.content.Update(msg) c.content = u return c, cmd } func (c *container) View() string { t := theme.CurrentTheme() style := lipgloss.NewStyle() width := c.width height := c.height style = style.Background(t.Background()) // Apply border if any side is enabled if c.borderTop || c.borderRight || c.borderBottom || c.borderLeft { // Adjust width and height for borders if c.borderTop { height-- } if c.borderBottom { height-- } if c.borderLeft { width-- } if c.borderRight { width-- } style = style.Border(c.borderStyle, c.borderTop, c.borderRight, c.borderBottom, c.borderLeft) style = style.BorderBackground(t.Background()).BorderForeground(t.BorderNormal()) } style = style. Width(width). Height(height). PaddingTop(c.paddingTop). PaddingRight(c.paddingRight). PaddingBottom(c.paddingBottom). PaddingLeft(c.paddingLeft) return style.Render(c.content.View()) } func (c *container) SetSize(width, height int) tea.Cmd { c.width = width c.height = height // If the content implements Sizeable, adjust its size to account for padding and borders if sizeable, ok := c.content.(Sizeable); ok { // Calculate horizontal space taken by padding and borders horizontalSpace := c.paddingLeft + c.paddingRight if c.borderLeft { horizontalSpace++ } if c.borderRight { horizontalSpace++ } // Calculate vertical space taken by padding and borders verticalSpace := c.paddingTop + c.paddingBottom if c.borderTop { verticalSpace++ } if c.borderBottom { verticalSpace++ } // Set content size with adjusted dimensions contentWidth := max(0, width-horizontalSpace) contentHeight := max(0, height-verticalSpace) return sizeable.SetSize(contentWidth, contentHeight) } return nil } func (c *container) GetSize() (int, int) { return c.width, c.height } func (c *container) BindingKeys() []key.Binding { if b, ok := c.content.(Bindings); ok { return b.BindingKeys() } return []key.Binding{} } type ContainerOption func(*container) func NewContainer(content tea.Model, options ...ContainerOption) Container { c := &container{ content: content, borderStyle: lipgloss.NormalBorder(), } for _, option := range options { option(c) } return c } // Padding options func WithPadding(top, right, bottom, left int) ContainerOption { return func(c *container) { c.paddingTop = top c.paddingRight = right c.paddingBottom = bottom c.paddingLeft = left } } func WithPaddingAll(padding int) ContainerOption { return WithPadding(padding, padding, padding, padding) } func WithPaddingHorizontal(padding int) ContainerOption { return func(c *container) { c.paddingLeft = padding c.paddingRight = padding } } func WithPaddingVertical(padding int) ContainerOption { return func(c *container) { c.paddingTop = padding c.paddingBottom = padding } } func WithBorder(top, right, bottom, left bool) ContainerOption { return func(c *container) { c.borderTop = top c.borderRight = right c.borderBottom = bottom c.borderLeft = left } } func WithBorderAll() ContainerOption { return WithBorder(true, true, true, true) } func WithBorderHorizontal() ContainerOption { return WithBorder(true, false, true, false) } func WithBorderVertical() ContainerOption { return WithBorder(false, true, false, true) } func WithBorderStyle(style lipgloss.Border) ContainerOption { return func(c *container) { c.borderStyle = style } } func WithRoundedBorder() ContainerOption { return WithBorderStyle(lipgloss.RoundedBorder()) } func WithThickBorder() ContainerOption { return WithBorderStyle(lipgloss.ThickBorder()) } func WithDoubleBorder() ContainerOption { return WithBorderStyle(lipgloss.DoubleBorder()) } ================================================ FILE: internal/tui/layout/layout.go ================================================ package layout import ( "reflect" "github.com/charmbracelet/bubbles/key" tea "github.com/charmbracelet/bubbletea" ) type Focusable interface { Focus() tea.Cmd Blur() tea.Cmd IsFocused() bool } type Sizeable interface { SetSize(width, height int) tea.Cmd GetSize() (int, int) } type Bindings interface { BindingKeys() []key.Binding } func KeyMapToSlice(t any) (bindings []key.Binding) { typ := reflect.TypeOf(t) if typ.Kind() != reflect.Struct { return nil } for i := range typ.NumField() { v := reflect.ValueOf(t).Field(i) bindings = append(bindings, v.Interface().(key.Binding)) } return } ================================================ FILE: internal/tui/layout/overlay.go ================================================ package layout import ( "strings" "github.com/charmbracelet/lipgloss" chAnsi "github.com/charmbracelet/x/ansi" "github.com/muesli/ansi" "github.com/muesli/reflow/truncate" "github.com/muesli/termenv" "github.com/opencode-ai/opencode/internal/tui/styles" "github.com/opencode-ai/opencode/internal/tui/theme" "github.com/opencode-ai/opencode/internal/tui/util" ) // Most of this code is borrowed from // https://github.com/charmbracelet/lipgloss/pull/102 // as well as the lipgloss library, with some modification for what I needed. // Split a string into lines, additionally returning the size of the widest // line. func getLines(s string) (lines []string, widest int) { lines = strings.Split(s, "\n") for _, l := range lines { w := ansi.PrintableRuneWidth(l) if widest < w { widest = w } } return lines, widest } // PlaceOverlay places fg on top of bg. func PlaceOverlay( x, y int, fg, bg string, shadow bool, opts ...WhitespaceOption, ) string { fgLines, fgWidth := getLines(fg) bgLines, bgWidth := getLines(bg) bgHeight := len(bgLines) fgHeight := len(fgLines) if shadow { t := theme.CurrentTheme() baseStyle := styles.BaseStyle() var shadowbg string = "" shadowchar := lipgloss.NewStyle(). Background(t.BackgroundDarker()). Foreground(t.Background()). Render("░") bgchar := baseStyle.Render(" ") for i := 0; i <= fgHeight; i++ { if i == 0 { shadowbg += bgchar + strings.Repeat(bgchar, fgWidth) + "\n" } else { shadowbg += bgchar + strings.Repeat(shadowchar, fgWidth) + "\n" } } fg = PlaceOverlay(0, 0, fg, shadowbg, false, opts...) fgLines, fgWidth = getLines(fg) fgHeight = len(fgLines) } if fgWidth >= bgWidth && fgHeight >= bgHeight { // FIXME: return fg or bg? return fg } // TODO: allow placement outside of the bg box? x = util.Clamp(x, 0, bgWidth-fgWidth) y = util.Clamp(y, 0, bgHeight-fgHeight) ws := &whitespace{} for _, opt := range opts { opt(ws) } var b strings.Builder for i, bgLine := range bgLines { if i > 0 { b.WriteByte('\n') } if i < y || i >= y+fgHeight { b.WriteString(bgLine) continue } pos := 0 if x > 0 { left := truncate.String(bgLine, uint(x)) pos = ansi.PrintableRuneWidth(left) b.WriteString(left) if pos < x { b.WriteString(ws.render(x - pos)) pos = x } } fgLine := fgLines[i-y] b.WriteString(fgLine) pos += ansi.PrintableRuneWidth(fgLine) right := cutLeft(bgLine, pos) bgWidth := ansi.PrintableRuneWidth(bgLine) rightWidth := ansi.PrintableRuneWidth(right) if rightWidth <= bgWidth-pos { b.WriteString(ws.render(bgWidth - rightWidth - pos)) } b.WriteString(right) } return b.String() } // cutLeft cuts printable characters from the left. // This function is heavily based on muesli's ansi and truncate packages. func cutLeft(s string, cutWidth int) string { return chAnsi.Cut(s, cutWidth, lipgloss.Width(s)) } func max(a, b int) int { if a > b { return a } return b } type whitespace struct { style termenv.Style chars string } // Render whitespaces. func (w whitespace) render(width int) string { if w.chars == "" { w.chars = " " } r := []rune(w.chars) j := 0 b := strings.Builder{} // Cycle through runes and print them into the whitespace. for i := 0; i < width; { b.WriteRune(r[j]) j++ if j >= len(r) { j = 0 } i += ansi.PrintableRuneWidth(string(r[j])) } // Fill any extra gaps white spaces. This might be necessary if any runes // are more than one cell wide, which could leave a one-rune gap. short := width - ansi.PrintableRuneWidth(b.String()) if short > 0 { b.WriteString(strings.Repeat(" ", short)) } return w.style.Styled(b.String()) } // WhitespaceOption sets a styling rule for rendering whitespace. type WhitespaceOption func(*whitespace) ================================================ FILE: internal/tui/layout/split.go ================================================ package layout import ( "github.com/charmbracelet/bubbles/key" tea "github.com/charmbracelet/bubbletea" "github.com/charmbracelet/lipgloss" "github.com/opencode-ai/opencode/internal/tui/theme" ) type SplitPaneLayout interface { tea.Model Sizeable Bindings SetLeftPanel(panel Container) tea.Cmd SetRightPanel(panel Container) tea.Cmd SetBottomPanel(panel Container) tea.Cmd ClearLeftPanel() tea.Cmd ClearRightPanel() tea.Cmd ClearBottomPanel() tea.Cmd } type splitPaneLayout struct { width int height int ratio float64 verticalRatio float64 rightPanel Container leftPanel Container bottomPanel Container } type SplitPaneOption func(*splitPaneLayout) func (s *splitPaneLayout) Init() tea.Cmd { var cmds []tea.Cmd if s.leftPanel != nil { cmds = append(cmds, s.leftPanel.Init()) } if s.rightPanel != nil { cmds = append(cmds, s.rightPanel.Init()) } if s.bottomPanel != nil { cmds = append(cmds, s.bottomPanel.Init()) } return tea.Batch(cmds...) } func (s *splitPaneLayout) Update(msg tea.Msg) (tea.Model, tea.Cmd) { var cmds []tea.Cmd switch msg := msg.(type) { case tea.WindowSizeMsg: return s, s.SetSize(msg.Width, msg.Height) } if s.rightPanel != nil { u, cmd := s.rightPanel.Update(msg) s.rightPanel = u.(Container) if cmd != nil { cmds = append(cmds, cmd) } } if s.leftPanel != nil { u, cmd := s.leftPanel.Update(msg) s.leftPanel = u.(Container) if cmd != nil { cmds = append(cmds, cmd) } } if s.bottomPanel != nil { u, cmd := s.bottomPanel.Update(msg) s.bottomPanel = u.(Container) if cmd != nil { cmds = append(cmds, cmd) } } return s, tea.Batch(cmds...) } func (s *splitPaneLayout) View() string { var topSection string if s.leftPanel != nil && s.rightPanel != nil { leftView := s.leftPanel.View() rightView := s.rightPanel.View() topSection = lipgloss.JoinHorizontal(lipgloss.Top, leftView, rightView) } else if s.leftPanel != nil { topSection = s.leftPanel.View() } else if s.rightPanel != nil { topSection = s.rightPanel.View() } else { topSection = "" } var finalView string if s.bottomPanel != nil && topSection != "" { bottomView := s.bottomPanel.View() finalView = lipgloss.JoinVertical(lipgloss.Left, topSection, bottomView) } else if s.bottomPanel != nil { finalView = s.bottomPanel.View() } else { finalView = topSection } if finalView != "" { t := theme.CurrentTheme() style := lipgloss.NewStyle(). Width(s.width). Height(s.height). Background(t.Background()) return style.Render(finalView) } return finalView } func (s *splitPaneLayout) SetSize(width, height int) tea.Cmd { s.width = width s.height = height var topHeight, bottomHeight int if s.bottomPanel != nil { topHeight = int(float64(height) * s.verticalRatio) bottomHeight = height - topHeight } else { topHeight = height bottomHeight = 0 } var leftWidth, rightWidth int if s.leftPanel != nil && s.rightPanel != nil { leftWidth = int(float64(width) * s.ratio) rightWidth = width - leftWidth } else if s.leftPanel != nil { leftWidth = width rightWidth = 0 } else if s.rightPanel != nil { leftWidth = 0 rightWidth = width } var cmds []tea.Cmd if s.leftPanel != nil { cmd := s.leftPanel.SetSize(leftWidth, topHeight) cmds = append(cmds, cmd) } if s.rightPanel != nil { cmd := s.rightPanel.SetSize(rightWidth, topHeight) cmds = append(cmds, cmd) } if s.bottomPanel != nil { cmd := s.bottomPanel.SetSize(width, bottomHeight) cmds = append(cmds, cmd) } return tea.Batch(cmds...) } func (s *splitPaneLayout) GetSize() (int, int) { return s.width, s.height } func (s *splitPaneLayout) SetLeftPanel(panel Container) tea.Cmd { s.leftPanel = panel if s.width > 0 && s.height > 0 { return s.SetSize(s.width, s.height) } return nil } func (s *splitPaneLayout) SetRightPanel(panel Container) tea.Cmd { s.rightPanel = panel if s.width > 0 && s.height > 0 { return s.SetSize(s.width, s.height) } return nil } func (s *splitPaneLayout) SetBottomPanel(panel Container) tea.Cmd { s.bottomPanel = panel if s.width > 0 && s.height > 0 { return s.SetSize(s.width, s.height) } return nil } func (s *splitPaneLayout) ClearLeftPanel() tea.Cmd { s.leftPanel = nil if s.width > 0 && s.height > 0 { return s.SetSize(s.width, s.height) } return nil } func (s *splitPaneLayout) ClearRightPanel() tea.Cmd { s.rightPanel = nil if s.width > 0 && s.height > 0 { return s.SetSize(s.width, s.height) } return nil } func (s *splitPaneLayout) ClearBottomPanel() tea.Cmd { s.bottomPanel = nil if s.width > 0 && s.height > 0 { return s.SetSize(s.width, s.height) } return nil } func (s *splitPaneLayout) BindingKeys() []key.Binding { keys := []key.Binding{} if s.leftPanel != nil { if b, ok := s.leftPanel.(Bindings); ok { keys = append(keys, b.BindingKeys()...) } } if s.rightPanel != nil { if b, ok := s.rightPanel.(Bindings); ok { keys = append(keys, b.BindingKeys()...) } } if s.bottomPanel != nil { if b, ok := s.bottomPanel.(Bindings); ok { keys = append(keys, b.BindingKeys()...) } } return keys } func NewSplitPane(options ...SplitPaneOption) SplitPaneLayout { layout := &splitPaneLayout{ ratio: 0.7, verticalRatio: 0.9, // Default 90% for top section, 10% for bottom } for _, option := range options { option(layout) } return layout } func WithLeftPanel(panel Container) SplitPaneOption { return func(s *splitPaneLayout) { s.leftPanel = panel } } func WithRightPanel(panel Container) SplitPaneOption { return func(s *splitPaneLayout) { s.rightPanel = panel } } func WithRatio(ratio float64) SplitPaneOption { return func(s *splitPaneLayout) { s.ratio = ratio } } func WithBottomPanel(panel Container) SplitPaneOption { return func(s *splitPaneLayout) { s.bottomPanel = panel } } func WithVerticalRatio(ratio float64) SplitPaneOption { return func(s *splitPaneLayout) { s.verticalRatio = ratio } } ================================================ FILE: internal/tui/page/chat.go ================================================ package page import ( "context" "strings" "github.com/charmbracelet/bubbles/key" tea "github.com/charmbracelet/bubbletea" "github.com/charmbracelet/lipgloss" "github.com/opencode-ai/opencode/internal/app" "github.com/opencode-ai/opencode/internal/completions" "github.com/opencode-ai/opencode/internal/message" "github.com/opencode-ai/opencode/internal/session" "github.com/opencode-ai/opencode/internal/tui/components/chat" "github.com/opencode-ai/opencode/internal/tui/components/dialog" "github.com/opencode-ai/opencode/internal/tui/layout" "github.com/opencode-ai/opencode/internal/tui/util" ) var ChatPage PageID = "chat" type chatPage struct { app *app.App editor layout.Container messages layout.Container layout layout.SplitPaneLayout session session.Session completionDialog dialog.CompletionDialog showCompletionDialog bool } type ChatKeyMap struct { ShowCompletionDialog key.Binding NewSession key.Binding Cancel key.Binding } var keyMap = ChatKeyMap{ ShowCompletionDialog: key.NewBinding( key.WithKeys("@"), key.WithHelp("@", "Complete"), ), NewSession: key.NewBinding( key.WithKeys("ctrl+n"), key.WithHelp("ctrl+n", "new session"), ), Cancel: key.NewBinding( key.WithKeys("esc"), key.WithHelp("esc", "cancel"), ), } func (p *chatPage) Init() tea.Cmd { cmds := []tea.Cmd{ p.layout.Init(), p.completionDialog.Init(), } return tea.Batch(cmds...) } func (p *chatPage) Update(msg tea.Msg) (tea.Model, tea.Cmd) { var cmds []tea.Cmd switch msg := msg.(type) { case tea.WindowSizeMsg: cmd := p.layout.SetSize(msg.Width, msg.Height) cmds = append(cmds, cmd) case dialog.CompletionDialogCloseMsg: p.showCompletionDialog = false case chat.SendMsg: cmd := p.sendMessage(msg.Text, msg.Attachments) if cmd != nil { return p, cmd } case dialog.CommandRunCustomMsg: // Check if the agent is busy before executing custom commands if p.app.CoderAgent.IsBusy() { return p, util.ReportWarn("Agent is busy, please wait before executing a command...") } // Process the command content with arguments if any content := msg.Content if msg.Args != nil { // Replace all named arguments with their values for name, value := range msg.Args { placeholder := "$" + name content = strings.ReplaceAll(content, placeholder, value) } } // Handle custom command execution cmd := p.sendMessage(content, nil) if cmd != nil { return p, cmd } case chat.SessionSelectedMsg: if p.session.ID == "" { cmd := p.setSidebar() if cmd != nil { cmds = append(cmds, cmd) } } p.session = msg case tea.KeyMsg: switch { case key.Matches(msg, keyMap.ShowCompletionDialog): p.showCompletionDialog = true // Continue sending keys to layout->chat case key.Matches(msg, keyMap.NewSession): p.session = session.Session{} return p, tea.Batch( p.clearSidebar(), util.CmdHandler(chat.SessionClearedMsg{}), ) case key.Matches(msg, keyMap.Cancel): if p.session.ID != "" { // Cancel the current session's generation process // This allows users to interrupt long-running operations p.app.CoderAgent.Cancel(p.session.ID) return p, nil } } } if p.showCompletionDialog { context, contextCmd := p.completionDialog.Update(msg) p.completionDialog = context.(dialog.CompletionDialog) cmds = append(cmds, contextCmd) // Doesn't forward event if enter key is pressed if keyMsg, ok := msg.(tea.KeyMsg); ok { if keyMsg.String() == "enter" { return p, tea.Batch(cmds...) } } } u, cmd := p.layout.Update(msg) cmds = append(cmds, cmd) p.layout = u.(layout.SplitPaneLayout) return p, tea.Batch(cmds...) } func (p *chatPage) setSidebar() tea.Cmd { sidebarContainer := layout.NewContainer( chat.NewSidebarCmp(p.session, p.app.History), layout.WithPadding(1, 1, 1, 1), ) return tea.Batch(p.layout.SetRightPanel(sidebarContainer), sidebarContainer.Init()) } func (p *chatPage) clearSidebar() tea.Cmd { return p.layout.ClearRightPanel() } func (p *chatPage) sendMessage(text string, attachments []message.Attachment) tea.Cmd { var cmds []tea.Cmd if p.session.ID == "" { session, err := p.app.Sessions.Create(context.Background(), "New Session") if err != nil { return util.ReportError(err) } p.session = session cmd := p.setSidebar() if cmd != nil { cmds = append(cmds, cmd) } cmds = append(cmds, util.CmdHandler(chat.SessionSelectedMsg(session))) } _, err := p.app.CoderAgent.Run(context.Background(), p.session.ID, text, attachments...) if err != nil { return util.ReportError(err) } return tea.Batch(cmds...) } func (p *chatPage) SetSize(width, height int) tea.Cmd { return p.layout.SetSize(width, height) } func (p *chatPage) GetSize() (int, int) { return p.layout.GetSize() } func (p *chatPage) View() string { layoutView := p.layout.View() if p.showCompletionDialog { _, layoutHeight := p.layout.GetSize() editorWidth, editorHeight := p.editor.GetSize() p.completionDialog.SetWidth(editorWidth) overlay := p.completionDialog.View() layoutView = layout.PlaceOverlay( 0, layoutHeight-editorHeight-lipgloss.Height(overlay), overlay, layoutView, false, ) } return layoutView } func (p *chatPage) BindingKeys() []key.Binding { bindings := layout.KeyMapToSlice(keyMap) bindings = append(bindings, p.messages.BindingKeys()...) bindings = append(bindings, p.editor.BindingKeys()...) return bindings } func NewChatPage(app *app.App) tea.Model { cg := completions.NewFileAndFolderContextGroup() completionDialog := dialog.NewCompletionDialogCmp(cg) messagesContainer := layout.NewContainer( chat.NewMessagesCmp(app), layout.WithPadding(1, 1, 0, 1), ) editorContainer := layout.NewContainer( chat.NewEditorCmp(app), layout.WithBorder(true, false, false, false), ) return &chatPage{ app: app, editor: editorContainer, messages: messagesContainer, completionDialog: completionDialog, layout: layout.NewSplitPane( layout.WithLeftPanel(messagesContainer), layout.WithBottomPanel(editorContainer), ), } } ================================================ FILE: internal/tui/page/logs.go ================================================ package page import ( "github.com/charmbracelet/bubbles/key" tea "github.com/charmbracelet/bubbletea" "github.com/charmbracelet/lipgloss" "github.com/opencode-ai/opencode/internal/tui/components/logs" "github.com/opencode-ai/opencode/internal/tui/layout" "github.com/opencode-ai/opencode/internal/tui/styles" ) var LogsPage PageID = "logs" type LogPage interface { tea.Model layout.Sizeable layout.Bindings } type logsPage struct { width, height int table layout.Container details layout.Container } func (p *logsPage) Update(msg tea.Msg) (tea.Model, tea.Cmd) { var cmds []tea.Cmd switch msg := msg.(type) { case tea.WindowSizeMsg: p.width = msg.Width p.height = msg.Height return p, p.SetSize(msg.Width, msg.Height) } table, cmd := p.table.Update(msg) cmds = append(cmds, cmd) p.table = table.(layout.Container) details, cmd := p.details.Update(msg) cmds = append(cmds, cmd) p.details = details.(layout.Container) return p, tea.Batch(cmds...) } func (p *logsPage) View() string { style := styles.BaseStyle().Width(p.width).Height(p.height) return style.Render(lipgloss.JoinVertical(lipgloss.Top, p.table.View(), p.details.View(), )) } func (p *logsPage) BindingKeys() []key.Binding { return p.table.BindingKeys() } // GetSize implements LogPage. func (p *logsPage) GetSize() (int, int) { return p.width, p.height } // SetSize implements LogPage. func (p *logsPage) SetSize(width int, height int) tea.Cmd { p.width = width p.height = height return tea.Batch( p.table.SetSize(width, height/2), p.details.SetSize(width, height/2), ) } func (p *logsPage) Init() tea.Cmd { return tea.Batch( p.table.Init(), p.details.Init(), ) } func NewLogsPage() LogPage { return &logsPage{ table: layout.NewContainer(logs.NewLogsTable(), layout.WithBorderAll()), details: layout.NewContainer(logs.NewLogsDetails(), layout.WithBorderAll()), } } ================================================ FILE: internal/tui/page/page.go ================================================ package page type PageID string // PageChangeMsg is used to change the current page type PageChangeMsg struct { ID PageID } ================================================ FILE: internal/tui/styles/background.go ================================================ package styles import ( "fmt" "regexp" "strings" "github.com/charmbracelet/lipgloss" ) var ansiEscape = regexp.MustCompile("\x1b\\[[0-9;]*m") func getColorRGB(c lipgloss.TerminalColor) (uint8, uint8, uint8) { r, g, b, a := c.RGBA() // Un-premultiply alpha if needed if a > 0 && a < 0xffff { r = (r * 0xffff) / a g = (g * 0xffff) / a b = (b * 0xffff) / a } // Convert from 16-bit to 8-bit color return uint8(r >> 8), uint8(g >> 8), uint8(b >> 8) } // ForceReplaceBackgroundWithLipgloss replaces any ANSI background color codes // in `input` with a single 24‑bit background (48;2;R;G;B). func ForceReplaceBackgroundWithLipgloss(input string, newBgColor lipgloss.TerminalColor) string { // Precompute our new-bg sequence once r, g, b := getColorRGB(newBgColor) newBg := fmt.Sprintf("48;2;%d;%d;%d", r, g, b) return ansiEscape.ReplaceAllStringFunc(input, func(seq string) string { const ( escPrefixLen = 2 // "\x1b[" escSuffixLen = 1 // "m" ) raw := seq start := escPrefixLen end := len(raw) - escSuffixLen var sb strings.Builder // reserve enough space: original content minus bg codes + our newBg sb.Grow((end - start) + len(newBg) + 2) // scan from start..end, token by token for i := start; i < end; { // find the next ';' or end j := i for j < end && raw[j] != ';' { j++ } token := raw[i:j] // fast‑path: skip "48;5;N" or "48;2;R;G;B" if len(token) == 2 && token[0] == '4' && token[1] == '8' { k := j + 1 if k < end { // find next token l := k for l < end && raw[l] != ';' { l++ } next := raw[k:l] if next == "5" { // skip "48;5;N" m := l + 1 for m < end && raw[m] != ';' { m++ } i = m + 1 continue } else if next == "2" { // skip "48;2;R;G;B" m := l + 1 for count := 0; count < 3 && m < end; count++ { for m < end && raw[m] != ';' { m++ } m++ } i = m continue } } } // decide whether to keep this token // manually parse ASCII digits to int isNum := true val := 0 for p := i; p < j; p++ { c := raw[p] if c < '0' || c > '9' { isNum = false break } val = val*10 + int(c-'0') } keep := !isNum || ((val < 40 || val > 47) && (val < 100 || val > 107) && val != 49) if keep { if sb.Len() > 0 { sb.WriteByte(';') } sb.WriteString(token) } // advance past this token (and the semicolon) i = j + 1 } // append our new background if sb.Len() > 0 { sb.WriteByte(';') } sb.WriteString(newBg) return "\x1b[" + sb.String() + "m" }) } ================================================ FILE: internal/tui/styles/icons.go ================================================ package styles const ( OpenCodeIcon string = "⌬" CheckIcon string = "✓" ErrorIcon string = "✖" WarningIcon string = "⚠" InfoIcon string = "" HintIcon string = "i" SpinnerIcon string = "..." LoadingIcon string = "⟳" DocumentIcon string = "🖼" ) ================================================ FILE: internal/tui/styles/markdown.go ================================================ package styles import ( "github.com/charmbracelet/glamour" "github.com/charmbracelet/glamour/ansi" "github.com/charmbracelet/lipgloss" "github.com/opencode-ai/opencode/internal/tui/theme" ) const defaultMargin = 1 // Helper functions for style pointers func boolPtr(b bool) *bool { return &b } func stringPtr(s string) *string { return &s } func uintPtr(u uint) *uint { return &u } // returns a glamour TermRenderer configured with the current theme func GetMarkdownRenderer(width int) *glamour.TermRenderer { r, _ := glamour.NewTermRenderer( glamour.WithStyles(generateMarkdownStyleConfig()), glamour.WithWordWrap(width), ) return r } // creates an ansi.StyleConfig for markdown rendering // using adaptive colors from the provided theme. func generateMarkdownStyleConfig() ansi.StyleConfig { t := theme.CurrentTheme() return ansi.StyleConfig{ Document: ansi.StyleBlock{ StylePrimitive: ansi.StylePrimitive{ BlockPrefix: "", BlockSuffix: "", Color: stringPtr(adaptiveColorToString(t.MarkdownText())), }, Margin: uintPtr(defaultMargin), }, BlockQuote: ansi.StyleBlock{ StylePrimitive: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.MarkdownBlockQuote())), Italic: boolPtr(true), Prefix: "┃ ", }, Indent: uintPtr(1), IndentToken: stringPtr(BaseStyle().Render(" ")), }, List: ansi.StyleList{ LevelIndent: defaultMargin, StyleBlock: ansi.StyleBlock{ IndentToken: stringPtr(BaseStyle().Render(" ")), StylePrimitive: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.MarkdownText())), }, }, }, Heading: ansi.StyleBlock{ StylePrimitive: ansi.StylePrimitive{ BlockSuffix: "\n", Color: stringPtr(adaptiveColorToString(t.MarkdownHeading())), Bold: boolPtr(true), }, }, H1: ansi.StyleBlock{ StylePrimitive: ansi.StylePrimitive{ Prefix: "# ", Color: stringPtr(adaptiveColorToString(t.MarkdownHeading())), Bold: boolPtr(true), }, }, H2: ansi.StyleBlock{ StylePrimitive: ansi.StylePrimitive{ Prefix: "## ", Color: stringPtr(adaptiveColorToString(t.MarkdownHeading())), Bold: boolPtr(true), }, }, H3: ansi.StyleBlock{ StylePrimitive: ansi.StylePrimitive{ Prefix: "### ", Color: stringPtr(adaptiveColorToString(t.MarkdownHeading())), Bold: boolPtr(true), }, }, H4: ansi.StyleBlock{ StylePrimitive: ansi.StylePrimitive{ Prefix: "#### ", Color: stringPtr(adaptiveColorToString(t.MarkdownHeading())), Bold: boolPtr(true), }, }, H5: ansi.StyleBlock{ StylePrimitive: ansi.StylePrimitive{ Prefix: "##### ", Color: stringPtr(adaptiveColorToString(t.MarkdownHeading())), Bold: boolPtr(true), }, }, H6: ansi.StyleBlock{ StylePrimitive: ansi.StylePrimitive{ Prefix: "###### ", Color: stringPtr(adaptiveColorToString(t.MarkdownHeading())), Bold: boolPtr(true), }, }, Strikethrough: ansi.StylePrimitive{ CrossedOut: boolPtr(true), Color: stringPtr(adaptiveColorToString(t.TextMuted())), }, Emph: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.MarkdownEmph())), Italic: boolPtr(true), }, Strong: ansi.StylePrimitive{ Bold: boolPtr(true), Color: stringPtr(adaptiveColorToString(t.MarkdownStrong())), }, HorizontalRule: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.MarkdownHorizontalRule())), Format: "\n─────────────────────────────────────────\n", }, Item: ansi.StylePrimitive{ BlockPrefix: "• ", Color: stringPtr(adaptiveColorToString(t.MarkdownListItem())), }, Enumeration: ansi.StylePrimitive{ BlockPrefix: ". ", Color: stringPtr(adaptiveColorToString(t.MarkdownListEnumeration())), }, Task: ansi.StyleTask{ StylePrimitive: ansi.StylePrimitive{}, Ticked: "[✓] ", Unticked: "[ ] ", }, Link: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.MarkdownLink())), Underline: boolPtr(true), }, LinkText: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.MarkdownLinkText())), Bold: boolPtr(true), }, Image: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.MarkdownImage())), Underline: boolPtr(true), Format: "🖼 {{.text}}", }, ImageText: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.MarkdownImageText())), Format: "{{.text}}", }, Code: ansi.StyleBlock{ StylePrimitive: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.MarkdownCode())), Prefix: "", Suffix: "", }, }, CodeBlock: ansi.StyleCodeBlock{ StyleBlock: ansi.StyleBlock{ StylePrimitive: ansi.StylePrimitive{ Prefix: " ", Color: stringPtr(adaptiveColorToString(t.MarkdownCodeBlock())), }, Margin: uintPtr(defaultMargin), }, Chroma: &ansi.Chroma{ Text: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.MarkdownText())), }, Error: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.Error())), }, Comment: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.SyntaxComment())), }, CommentPreproc: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.SyntaxKeyword())), }, Keyword: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.SyntaxKeyword())), }, KeywordReserved: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.SyntaxKeyword())), }, KeywordNamespace: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.SyntaxKeyword())), }, KeywordType: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.SyntaxType())), }, Operator: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.SyntaxOperator())), }, Punctuation: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.SyntaxPunctuation())), }, Name: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.SyntaxVariable())), }, NameBuiltin: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.SyntaxVariable())), }, NameTag: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.SyntaxKeyword())), }, NameAttribute: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.SyntaxFunction())), }, NameClass: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.SyntaxType())), }, NameConstant: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.SyntaxVariable())), }, NameDecorator: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.SyntaxFunction())), }, NameFunction: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.SyntaxFunction())), }, LiteralNumber: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.SyntaxNumber())), }, LiteralString: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.SyntaxString())), }, LiteralStringEscape: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.SyntaxKeyword())), }, GenericDeleted: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.DiffRemoved())), }, GenericEmph: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.MarkdownEmph())), Italic: boolPtr(true), }, GenericInserted: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.DiffAdded())), }, GenericStrong: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.MarkdownStrong())), Bold: boolPtr(true), }, GenericSubheading: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.MarkdownHeading())), }, }, }, Table: ansi.StyleTable{ StyleBlock: ansi.StyleBlock{ StylePrimitive: ansi.StylePrimitive{ BlockPrefix: "\n", BlockSuffix: "\n", }, }, CenterSeparator: stringPtr("┼"), ColumnSeparator: stringPtr("│"), RowSeparator: stringPtr("─"), }, DefinitionDescription: ansi.StylePrimitive{ BlockPrefix: "\n ❯ ", Color: stringPtr(adaptiveColorToString(t.MarkdownLinkText())), }, Text: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.MarkdownText())), }, Paragraph: ansi.StyleBlock{ StylePrimitive: ansi.StylePrimitive{ Color: stringPtr(adaptiveColorToString(t.MarkdownText())), }, }, } } // adaptiveColorToString converts a lipgloss.AdaptiveColor to the appropriate // hex color string based on the current terminal background func adaptiveColorToString(color lipgloss.AdaptiveColor) string { if lipgloss.HasDarkBackground() { return color.Dark } return color.Light } ================================================ FILE: internal/tui/styles/styles.go ================================================ package styles import ( "github.com/charmbracelet/lipgloss" "github.com/opencode-ai/opencode/internal/tui/theme" ) var ( ImageBakcground = "#212121" ) // Style generation functions that use the current theme // BaseStyle returns the base style with background and foreground colors func BaseStyle() lipgloss.Style { t := theme.CurrentTheme() return lipgloss.NewStyle(). Background(t.Background()). Foreground(t.Text()) } // Regular returns a basic unstyled lipgloss.Style func Regular() lipgloss.Style { return lipgloss.NewStyle() } // Bold returns a bold style func Bold() lipgloss.Style { return Regular().Bold(true) } // Padded returns a style with horizontal padding func Padded() lipgloss.Style { return Regular().Padding(0, 1) } // Border returns a style with a normal border func Border() lipgloss.Style { t := theme.CurrentTheme() return Regular(). Border(lipgloss.NormalBorder()). BorderForeground(t.BorderNormal()) } // ThickBorder returns a style with a thick border func ThickBorder() lipgloss.Style { t := theme.CurrentTheme() return Regular(). Border(lipgloss.ThickBorder()). BorderForeground(t.BorderNormal()) } // DoubleBorder returns a style with a double border func DoubleBorder() lipgloss.Style { t := theme.CurrentTheme() return Regular(). Border(lipgloss.DoubleBorder()). BorderForeground(t.BorderNormal()) } // FocusedBorder returns a style with a border using the focused border color func FocusedBorder() lipgloss.Style { t := theme.CurrentTheme() return Regular(). Border(lipgloss.NormalBorder()). BorderForeground(t.BorderFocused()) } // DimBorder returns a style with a border using the dim border color func DimBorder() lipgloss.Style { t := theme.CurrentTheme() return Regular(). Border(lipgloss.NormalBorder()). BorderForeground(t.BorderDim()) } // PrimaryColor returns the primary color from the current theme func PrimaryColor() lipgloss.AdaptiveColor { return theme.CurrentTheme().Primary() } // SecondaryColor returns the secondary color from the current theme func SecondaryColor() lipgloss.AdaptiveColor { return theme.CurrentTheme().Secondary() } // AccentColor returns the accent color from the current theme func AccentColor() lipgloss.AdaptiveColor { return theme.CurrentTheme().Accent() } // ErrorColor returns the error color from the current theme func ErrorColor() lipgloss.AdaptiveColor { return theme.CurrentTheme().Error() } // WarningColor returns the warning color from the current theme func WarningColor() lipgloss.AdaptiveColor { return theme.CurrentTheme().Warning() } // SuccessColor returns the success color from the current theme func SuccessColor() lipgloss.AdaptiveColor { return theme.CurrentTheme().Success() } // InfoColor returns the info color from the current theme func InfoColor() lipgloss.AdaptiveColor { return theme.CurrentTheme().Info() } // TextColor returns the text color from the current theme func TextColor() lipgloss.AdaptiveColor { return theme.CurrentTheme().Text() } // TextMutedColor returns the muted text color from the current theme func TextMutedColor() lipgloss.AdaptiveColor { return theme.CurrentTheme().TextMuted() } // TextEmphasizedColor returns the emphasized text color from the current theme func TextEmphasizedColor() lipgloss.AdaptiveColor { return theme.CurrentTheme().TextEmphasized() } // BackgroundColor returns the background color from the current theme func BackgroundColor() lipgloss.AdaptiveColor { return theme.CurrentTheme().Background() } // BackgroundSecondaryColor returns the secondary background color from the current theme func BackgroundSecondaryColor() lipgloss.AdaptiveColor { return theme.CurrentTheme().BackgroundSecondary() } // BackgroundDarkerColor returns the darker background color from the current theme func BackgroundDarkerColor() lipgloss.AdaptiveColor { return theme.CurrentTheme().BackgroundDarker() } // BorderNormalColor returns the normal border color from the current theme func BorderNormalColor() lipgloss.AdaptiveColor { return theme.CurrentTheme().BorderNormal() } // BorderFocusedColor returns the focused border color from the current theme func BorderFocusedColor() lipgloss.AdaptiveColor { return theme.CurrentTheme().BorderFocused() } // BorderDimColor returns the dim border color from the current theme func BorderDimColor() lipgloss.AdaptiveColor { return theme.CurrentTheme().BorderDim() } ================================================ FILE: internal/tui/theme/catppuccin.go ================================================ package theme import ( catppuccin "github.com/catppuccin/go" "github.com/charmbracelet/lipgloss" ) // CatppuccinTheme implements the Theme interface with Catppuccin colors. // It provides both dark (Mocha) and light (Latte) variants. type CatppuccinTheme struct { BaseTheme } // NewCatppuccinTheme creates a new instance of the Catppuccin theme. func NewCatppuccinTheme() *CatppuccinTheme { // Get the Catppuccin palettes mocha := catppuccin.Mocha latte := catppuccin.Latte theme := &CatppuccinTheme{} // Base colors theme.PrimaryColor = lipgloss.AdaptiveColor{ Dark: mocha.Blue().Hex, Light: latte.Blue().Hex, } theme.SecondaryColor = lipgloss.AdaptiveColor{ Dark: mocha.Mauve().Hex, Light: latte.Mauve().Hex, } theme.AccentColor = lipgloss.AdaptiveColor{ Dark: mocha.Peach().Hex, Light: latte.Peach().Hex, } // Status colors theme.ErrorColor = lipgloss.AdaptiveColor{ Dark: mocha.Red().Hex, Light: latte.Red().Hex, } theme.WarningColor = lipgloss.AdaptiveColor{ Dark: mocha.Peach().Hex, Light: latte.Peach().Hex, } theme.SuccessColor = lipgloss.AdaptiveColor{ Dark: mocha.Green().Hex, Light: latte.Green().Hex, } theme.InfoColor = lipgloss.AdaptiveColor{ Dark: mocha.Blue().Hex, Light: latte.Blue().Hex, } // Text colors theme.TextColor = lipgloss.AdaptiveColor{ Dark: mocha.Text().Hex, Light: latte.Text().Hex, } theme.TextMutedColor = lipgloss.AdaptiveColor{ Dark: mocha.Subtext0().Hex, Light: latte.Subtext0().Hex, } theme.TextEmphasizedColor = lipgloss.AdaptiveColor{ Dark: mocha.Lavender().Hex, Light: latte.Lavender().Hex, } // Background colors theme.BackgroundColor = lipgloss.AdaptiveColor{ Dark: "#212121", // From existing styles Light: "#EEEEEE", // Light equivalent } theme.BackgroundSecondaryColor = lipgloss.AdaptiveColor{ Dark: "#2c2c2c", // From existing styles Light: "#E0E0E0", // Light equivalent } theme.BackgroundDarkerColor = lipgloss.AdaptiveColor{ Dark: "#181818", // From existing styles Light: "#F5F5F5", // Light equivalent } // Border colors theme.BorderNormalColor = lipgloss.AdaptiveColor{ Dark: "#4b4c5c", // From existing styles Light: "#BDBDBD", // Light equivalent } theme.BorderFocusedColor = lipgloss.AdaptiveColor{ Dark: mocha.Blue().Hex, Light: latte.Blue().Hex, } theme.BorderDimColor = lipgloss.AdaptiveColor{ Dark: mocha.Surface0().Hex, Light: latte.Surface0().Hex, } // Diff view colors theme.DiffAddedColor = lipgloss.AdaptiveColor{ Dark: "#478247", // From existing diff.go Light: "#2E7D32", // Light equivalent } theme.DiffRemovedColor = lipgloss.AdaptiveColor{ Dark: "#7C4444", // From existing diff.go Light: "#C62828", // Light equivalent } theme.DiffContextColor = lipgloss.AdaptiveColor{ Dark: "#a0a0a0", // From existing diff.go Light: "#757575", // Light equivalent } theme.DiffHunkHeaderColor = lipgloss.AdaptiveColor{ Dark: "#a0a0a0", // From existing diff.go Light: "#757575", // Light equivalent } theme.DiffHighlightAddedColor = lipgloss.AdaptiveColor{ Dark: "#DAFADA", // From existing diff.go Light: "#A5D6A7", // Light equivalent } theme.DiffHighlightRemovedColor = lipgloss.AdaptiveColor{ Dark: "#FADADD", // From existing diff.go Light: "#EF9A9A", // Light equivalent } theme.DiffAddedBgColor = lipgloss.AdaptiveColor{ Dark: "#303A30", // From existing diff.go Light: "#E8F5E9", // Light equivalent } theme.DiffRemovedBgColor = lipgloss.AdaptiveColor{ Dark: "#3A3030", // From existing diff.go Light: "#FFEBEE", // Light equivalent } theme.DiffContextBgColor = lipgloss.AdaptiveColor{ Dark: "#212121", // From existing diff.go Light: "#F5F5F5", // Light equivalent } theme.DiffLineNumberColor = lipgloss.AdaptiveColor{ Dark: "#888888", // From existing diff.go Light: "#9E9E9E", // Light equivalent } theme.DiffAddedLineNumberBgColor = lipgloss.AdaptiveColor{ Dark: "#293229", // From existing diff.go Light: "#C8E6C9", // Light equivalent } theme.DiffRemovedLineNumberBgColor = lipgloss.AdaptiveColor{ Dark: "#332929", // From existing diff.go Light: "#FFCDD2", // Light equivalent } // Markdown colors theme.MarkdownTextColor = lipgloss.AdaptiveColor{ Dark: mocha.Text().Hex, Light: latte.Text().Hex, } theme.MarkdownHeadingColor = lipgloss.AdaptiveColor{ Dark: mocha.Mauve().Hex, Light: latte.Mauve().Hex, } theme.MarkdownLinkColor = lipgloss.AdaptiveColor{ Dark: mocha.Sky().Hex, Light: latte.Sky().Hex, } theme.MarkdownLinkTextColor = lipgloss.AdaptiveColor{ Dark: mocha.Pink().Hex, Light: latte.Pink().Hex, } theme.MarkdownCodeColor = lipgloss.AdaptiveColor{ Dark: mocha.Green().Hex, Light: latte.Green().Hex, } theme.MarkdownBlockQuoteColor = lipgloss.AdaptiveColor{ Dark: mocha.Yellow().Hex, Light: latte.Yellow().Hex, } theme.MarkdownEmphColor = lipgloss.AdaptiveColor{ Dark: mocha.Yellow().Hex, Light: latte.Yellow().Hex, } theme.MarkdownStrongColor = lipgloss.AdaptiveColor{ Dark: mocha.Peach().Hex, Light: latte.Peach().Hex, } theme.MarkdownHorizontalRuleColor = lipgloss.AdaptiveColor{ Dark: mocha.Overlay0().Hex, Light: latte.Overlay0().Hex, } theme.MarkdownListItemColor = lipgloss.AdaptiveColor{ Dark: mocha.Blue().Hex, Light: latte.Blue().Hex, } theme.MarkdownListEnumerationColor = lipgloss.AdaptiveColor{ Dark: mocha.Sky().Hex, Light: latte.Sky().Hex, } theme.MarkdownImageColor = lipgloss.AdaptiveColor{ Dark: mocha.Sapphire().Hex, Light: latte.Sapphire().Hex, } theme.MarkdownImageTextColor = lipgloss.AdaptiveColor{ Dark: mocha.Pink().Hex, Light: latte.Pink().Hex, } theme.MarkdownCodeBlockColor = lipgloss.AdaptiveColor{ Dark: mocha.Text().Hex, Light: latte.Text().Hex, } // Syntax highlighting colors theme.SyntaxCommentColor = lipgloss.AdaptiveColor{ Dark: mocha.Overlay1().Hex, Light: latte.Overlay1().Hex, } theme.SyntaxKeywordColor = lipgloss.AdaptiveColor{ Dark: mocha.Pink().Hex, Light: latte.Pink().Hex, } theme.SyntaxFunctionColor = lipgloss.AdaptiveColor{ Dark: mocha.Green().Hex, Light: latte.Green().Hex, } theme.SyntaxVariableColor = lipgloss.AdaptiveColor{ Dark: mocha.Sky().Hex, Light: latte.Sky().Hex, } theme.SyntaxStringColor = lipgloss.AdaptiveColor{ Dark: mocha.Yellow().Hex, Light: latte.Yellow().Hex, } theme.SyntaxNumberColor = lipgloss.AdaptiveColor{ Dark: mocha.Teal().Hex, Light: latte.Teal().Hex, } theme.SyntaxTypeColor = lipgloss.AdaptiveColor{ Dark: mocha.Sky().Hex, Light: latte.Sky().Hex, } theme.SyntaxOperatorColor = lipgloss.AdaptiveColor{ Dark: mocha.Pink().Hex, Light: latte.Pink().Hex, } theme.SyntaxPunctuationColor = lipgloss.AdaptiveColor{ Dark: mocha.Text().Hex, Light: latte.Text().Hex, } return theme } func init() { // Register the Catppuccin theme with the theme manager RegisterTheme("catppuccin", NewCatppuccinTheme()) } ================================================ FILE: internal/tui/theme/dracula.go ================================================ package theme import ( "github.com/charmbracelet/lipgloss" ) // DraculaTheme implements the Theme interface with Dracula colors. // It provides both dark and light variants, though Dracula is primarily a dark theme. type DraculaTheme struct { BaseTheme } // NewDraculaTheme creates a new instance of the Dracula theme. func NewDraculaTheme() *DraculaTheme { // Dracula color palette // Official colors from https://draculatheme.com/ darkBackground := "#282a36" darkCurrentLine := "#44475a" darkSelection := "#44475a" darkForeground := "#f8f8f2" darkComment := "#6272a4" darkCyan := "#8be9fd" darkGreen := "#50fa7b" darkOrange := "#ffb86c" darkPink := "#ff79c6" darkPurple := "#bd93f9" darkRed := "#ff5555" darkYellow := "#f1fa8c" darkBorder := "#44475a" // Light mode approximation (Dracula is primarily a dark theme) lightBackground := "#f8f8f2" lightCurrentLine := "#e6e6e6" lightSelection := "#d8d8d8" lightForeground := "#282a36" lightComment := "#6272a4" lightCyan := "#0097a7" lightGreen := "#388e3c" lightOrange := "#f57c00" lightPink := "#d81b60" lightPurple := "#7e57c2" lightRed := "#e53935" lightYellow := "#fbc02d" lightBorder := "#d8d8d8" theme := &DraculaTheme{} // Base colors theme.PrimaryColor = lipgloss.AdaptiveColor{ Dark: darkPurple, Light: lightPurple, } theme.SecondaryColor = lipgloss.AdaptiveColor{ Dark: darkPink, Light: lightPink, } theme.AccentColor = lipgloss.AdaptiveColor{ Dark: darkCyan, Light: lightCyan, } // Status colors theme.ErrorColor = lipgloss.AdaptiveColor{ Dark: darkRed, Light: lightRed, } theme.WarningColor = lipgloss.AdaptiveColor{ Dark: darkOrange, Light: lightOrange, } theme.SuccessColor = lipgloss.AdaptiveColor{ Dark: darkGreen, Light: lightGreen, } theme.InfoColor = lipgloss.AdaptiveColor{ Dark: darkCyan, Light: lightCyan, } // Text colors theme.TextColor = lipgloss.AdaptiveColor{ Dark: darkForeground, Light: lightForeground, } theme.TextMutedColor = lipgloss.AdaptiveColor{ Dark: darkComment, Light: lightComment, } theme.TextEmphasizedColor = lipgloss.AdaptiveColor{ Dark: darkYellow, Light: lightYellow, } // Background colors theme.BackgroundColor = lipgloss.AdaptiveColor{ Dark: darkBackground, Light: lightBackground, } theme.BackgroundSecondaryColor = lipgloss.AdaptiveColor{ Dark: darkCurrentLine, Light: lightCurrentLine, } theme.BackgroundDarkerColor = lipgloss.AdaptiveColor{ Dark: "#21222c", // Slightly darker than background Light: "#ffffff", // Slightly lighter than background } // Border colors theme.BorderNormalColor = lipgloss.AdaptiveColor{ Dark: darkBorder, Light: lightBorder, } theme.BorderFocusedColor = lipgloss.AdaptiveColor{ Dark: darkPurple, Light: lightPurple, } theme.BorderDimColor = lipgloss.AdaptiveColor{ Dark: darkSelection, Light: lightSelection, } // Diff view colors theme.DiffAddedColor = lipgloss.AdaptiveColor{ Dark: darkGreen, Light: lightGreen, } theme.DiffRemovedColor = lipgloss.AdaptiveColor{ Dark: darkRed, Light: lightRed, } theme.DiffContextColor = lipgloss.AdaptiveColor{ Dark: darkComment, Light: lightComment, } theme.DiffHunkHeaderColor = lipgloss.AdaptiveColor{ Dark: darkPurple, Light: lightPurple, } theme.DiffHighlightAddedColor = lipgloss.AdaptiveColor{ Dark: "#50fa7b", Light: "#a5d6a7", } theme.DiffHighlightRemovedColor = lipgloss.AdaptiveColor{ Dark: "#ff5555", Light: "#ef9a9a", } theme.DiffAddedBgColor = lipgloss.AdaptiveColor{ Dark: "#2c3b2c", Light: "#e8f5e9", } theme.DiffRemovedBgColor = lipgloss.AdaptiveColor{ Dark: "#3b2c2c", Light: "#ffebee", } theme.DiffContextBgColor = lipgloss.AdaptiveColor{ Dark: darkBackground, Light: lightBackground, } theme.DiffLineNumberColor = lipgloss.AdaptiveColor{ Dark: darkComment, Light: lightComment, } theme.DiffAddedLineNumberBgColor = lipgloss.AdaptiveColor{ Dark: "#253025", Light: "#c8e6c9", } theme.DiffRemovedLineNumberBgColor = lipgloss.AdaptiveColor{ Dark: "#302525", Light: "#ffcdd2", } // Markdown colors theme.MarkdownTextColor = lipgloss.AdaptiveColor{ Dark: darkForeground, Light: lightForeground, } theme.MarkdownHeadingColor = lipgloss.AdaptiveColor{ Dark: darkPink, Light: lightPink, } theme.MarkdownLinkColor = lipgloss.AdaptiveColor{ Dark: darkPurple, Light: lightPurple, } theme.MarkdownLinkTextColor = lipgloss.AdaptiveColor{ Dark: darkCyan, Light: lightCyan, } theme.MarkdownCodeColor = lipgloss.AdaptiveColor{ Dark: darkGreen, Light: lightGreen, } theme.MarkdownBlockQuoteColor = lipgloss.AdaptiveColor{ Dark: darkYellow, Light: lightYellow, } theme.MarkdownEmphColor = lipgloss.AdaptiveColor{ Dark: darkYellow, Light: lightYellow, } theme.MarkdownStrongColor = lipgloss.AdaptiveColor{ Dark: darkOrange, Light: lightOrange, } theme.MarkdownHorizontalRuleColor = lipgloss.AdaptiveColor{ Dark: darkComment, Light: lightComment, } theme.MarkdownListItemColor = lipgloss.AdaptiveColor{ Dark: darkPurple, Light: lightPurple, } theme.MarkdownListEnumerationColor = lipgloss.AdaptiveColor{ Dark: darkCyan, Light: lightCyan, } theme.MarkdownImageColor = lipgloss.AdaptiveColor{ Dark: darkPurple, Light: lightPurple, } theme.MarkdownImageTextColor = lipgloss.AdaptiveColor{ Dark: darkCyan, Light: lightCyan, } theme.MarkdownCodeBlockColor = lipgloss.AdaptiveColor{ Dark: darkForeground, Light: lightForeground, } // Syntax highlighting colors theme.SyntaxCommentColor = lipgloss.AdaptiveColor{ Dark: darkComment, Light: lightComment, } theme.SyntaxKeywordColor = lipgloss.AdaptiveColor{ Dark: darkPink, Light: lightPink, } theme.SyntaxFunctionColor = lipgloss.AdaptiveColor{ Dark: darkGreen, Light: lightGreen, } theme.SyntaxVariableColor = lipgloss.AdaptiveColor{ Dark: darkOrange, Light: lightOrange, } theme.SyntaxStringColor = lipgloss.AdaptiveColor{ Dark: darkYellow, Light: lightYellow, } theme.SyntaxNumberColor = lipgloss.AdaptiveColor{ Dark: darkPurple, Light: lightPurple, } theme.SyntaxTypeColor = lipgloss.AdaptiveColor{ Dark: darkCyan, Light: lightCyan, } theme.SyntaxOperatorColor = lipgloss.AdaptiveColor{ Dark: darkPink, Light: lightPink, } theme.SyntaxPunctuationColor = lipgloss.AdaptiveColor{ Dark: darkForeground, Light: lightForeground, } return theme } func init() { // Register the Dracula theme with the theme manager RegisterTheme("dracula", NewDraculaTheme()) } ================================================ FILE: internal/tui/theme/flexoki.go ================================================ package theme import ( "github.com/charmbracelet/lipgloss" ) // Flexoki color palette constants const ( // Base colors flexokiPaper = "#FFFCF0" // Paper (lightest) flexokiBase50 = "#F2F0E5" // bg-2 (light) flexokiBase100 = "#E6E4D9" // ui (light) flexokiBase150 = "#DAD8CE" // ui-2 (light) flexokiBase200 = "#CECDC3" // ui-3 (light) flexokiBase300 = "#B7B5AC" // tx-3 (light) flexokiBase500 = "#878580" // tx-2 (light) flexokiBase600 = "#6F6E69" // tx (light) flexokiBase700 = "#575653" // tx-3 (dark) flexokiBase800 = "#403E3C" // ui-3 (dark) flexokiBase850 = "#343331" // ui-2 (dark) flexokiBase900 = "#282726" // ui (dark) flexokiBase950 = "#1C1B1A" // bg-2 (dark) flexokiBlack = "#100F0F" // bg (darkest) // Accent colors - Light theme (600) flexokiRed600 = "#AF3029" flexokiOrange600 = "#BC5215" flexokiYellow600 = "#AD8301" flexokiGreen600 = "#66800B" flexokiCyan600 = "#24837B" flexokiBlue600 = "#205EA6" flexokiPurple600 = "#5E409D" flexokiMagenta600 = "#A02F6F" // Accent colors - Dark theme (400) flexokiRed400 = "#D14D41" flexokiOrange400 = "#DA702C" flexokiYellow400 = "#D0A215" flexokiGreen400 = "#879A39" flexokiCyan400 = "#3AA99F" flexokiBlue400 = "#4385BE" flexokiPurple400 = "#8B7EC8" flexokiMagenta400 = "#CE5D97" ) // FlexokiTheme implements the Theme interface with Flexoki colors. // It provides both dark and light variants. type FlexokiTheme struct { BaseTheme } // NewFlexokiTheme creates a new instance of the Flexoki theme. func NewFlexokiTheme() *FlexokiTheme { theme := &FlexokiTheme{} // Base colors theme.PrimaryColor = lipgloss.AdaptiveColor{ Dark: flexokiBlue400, Light: flexokiBlue600, } theme.SecondaryColor = lipgloss.AdaptiveColor{ Dark: flexokiPurple400, Light: flexokiPurple600, } theme.AccentColor = lipgloss.AdaptiveColor{ Dark: flexokiOrange400, Light: flexokiOrange600, } // Status colors theme.ErrorColor = lipgloss.AdaptiveColor{ Dark: flexokiRed400, Light: flexokiRed600, } theme.WarningColor = lipgloss.AdaptiveColor{ Dark: flexokiYellow400, Light: flexokiYellow600, } theme.SuccessColor = lipgloss.AdaptiveColor{ Dark: flexokiGreen400, Light: flexokiGreen600, } theme.InfoColor = lipgloss.AdaptiveColor{ Dark: flexokiCyan400, Light: flexokiCyan600, } // Text colors theme.TextColor = lipgloss.AdaptiveColor{ Dark: flexokiBase300, Light: flexokiBase600, } theme.TextMutedColor = lipgloss.AdaptiveColor{ Dark: flexokiBase700, Light: flexokiBase500, } theme.TextEmphasizedColor = lipgloss.AdaptiveColor{ Dark: flexokiYellow400, Light: flexokiYellow600, } // Background colors theme.BackgroundColor = lipgloss.AdaptiveColor{ Dark: flexokiBlack, Light: flexokiPaper, } theme.BackgroundSecondaryColor = lipgloss.AdaptiveColor{ Dark: flexokiBase950, Light: flexokiBase50, } theme.BackgroundDarkerColor = lipgloss.AdaptiveColor{ Dark: flexokiBase900, Light: flexokiBase100, } // Border colors theme.BorderNormalColor = lipgloss.AdaptiveColor{ Dark: flexokiBase900, Light: flexokiBase100, } theme.BorderFocusedColor = lipgloss.AdaptiveColor{ Dark: flexokiBlue400, Light: flexokiBlue600, } theme.BorderDimColor = lipgloss.AdaptiveColor{ Dark: flexokiBase850, Light: flexokiBase150, } // Diff view colors theme.DiffAddedColor = lipgloss.AdaptiveColor{ Dark: flexokiGreen400, Light: flexokiGreen600, } theme.DiffRemovedColor = lipgloss.AdaptiveColor{ Dark: flexokiRed400, Light: flexokiRed600, } theme.DiffContextColor = lipgloss.AdaptiveColor{ Dark: flexokiBase700, Light: flexokiBase500, } theme.DiffHunkHeaderColor = lipgloss.AdaptiveColor{ Dark: flexokiBase700, Light: flexokiBase500, } theme.DiffHighlightAddedColor = lipgloss.AdaptiveColor{ Dark: flexokiGreen400, Light: flexokiGreen600, } theme.DiffHighlightRemovedColor = lipgloss.AdaptiveColor{ Dark: flexokiRed400, Light: flexokiRed600, } theme.DiffAddedBgColor = lipgloss.AdaptiveColor{ Dark: "#1D2419", // Darker green background Light: "#EFF2E2", // Light green background } theme.DiffRemovedBgColor = lipgloss.AdaptiveColor{ Dark: "#241919", // Darker red background Light: "#F2E2E2", // Light red background } theme.DiffContextBgColor = lipgloss.AdaptiveColor{ Dark: flexokiBlack, Light: flexokiPaper, } theme.DiffLineNumberColor = lipgloss.AdaptiveColor{ Dark: flexokiBase700, Light: flexokiBase500, } theme.DiffAddedLineNumberBgColor = lipgloss.AdaptiveColor{ Dark: "#1A2017", // Slightly darker green Light: "#E5EBD9", // Light green } theme.DiffRemovedLineNumberBgColor = lipgloss.AdaptiveColor{ Dark: "#201717", // Slightly darker red Light: "#EBD9D9", // Light red } // Markdown colors theme.MarkdownTextColor = lipgloss.AdaptiveColor{ Dark: flexokiBase300, Light: flexokiBase600, } theme.MarkdownHeadingColor = lipgloss.AdaptiveColor{ Dark: flexokiYellow400, Light: flexokiYellow600, } theme.MarkdownLinkColor = lipgloss.AdaptiveColor{ Dark: flexokiCyan400, Light: flexokiCyan600, } theme.MarkdownLinkTextColor = lipgloss.AdaptiveColor{ Dark: flexokiMagenta400, Light: flexokiMagenta600, } theme.MarkdownCodeColor = lipgloss.AdaptiveColor{ Dark: flexokiGreen400, Light: flexokiGreen600, } theme.MarkdownBlockQuoteColor = lipgloss.AdaptiveColor{ Dark: flexokiCyan400, Light: flexokiCyan600, } theme.MarkdownEmphColor = lipgloss.AdaptiveColor{ Dark: flexokiYellow400, Light: flexokiYellow600, } theme.MarkdownStrongColor = lipgloss.AdaptiveColor{ Dark: flexokiOrange400, Light: flexokiOrange600, } theme.MarkdownHorizontalRuleColor = lipgloss.AdaptiveColor{ Dark: flexokiBase800, Light: flexokiBase200, } theme.MarkdownListItemColor = lipgloss.AdaptiveColor{ Dark: flexokiBlue400, Light: flexokiBlue600, } theme.MarkdownListEnumerationColor = lipgloss.AdaptiveColor{ Dark: flexokiBlue400, Light: flexokiBlue600, } theme.MarkdownImageColor = lipgloss.AdaptiveColor{ Dark: flexokiPurple400, Light: flexokiPurple600, } theme.MarkdownImageTextColor = lipgloss.AdaptiveColor{ Dark: flexokiMagenta400, Light: flexokiMagenta600, } theme.MarkdownCodeBlockColor = lipgloss.AdaptiveColor{ Dark: flexokiBase300, Light: flexokiBase600, } // Syntax highlighting colors (based on Flexoki's mappings) theme.SyntaxCommentColor = lipgloss.AdaptiveColor{ Dark: flexokiBase700, // tx-3 Light: flexokiBase300, // tx-3 } theme.SyntaxKeywordColor = lipgloss.AdaptiveColor{ Dark: flexokiGreen400, // gr Light: flexokiGreen600, // gr } theme.SyntaxFunctionColor = lipgloss.AdaptiveColor{ Dark: flexokiOrange400, // or Light: flexokiOrange600, // or } theme.SyntaxVariableColor = lipgloss.AdaptiveColor{ Dark: flexokiBlue400, // bl Light: flexokiBlue600, // bl } theme.SyntaxStringColor = lipgloss.AdaptiveColor{ Dark: flexokiCyan400, // cy Light: flexokiCyan600, // cy } theme.SyntaxNumberColor = lipgloss.AdaptiveColor{ Dark: flexokiPurple400, // pu Light: flexokiPurple600, // pu } theme.SyntaxTypeColor = lipgloss.AdaptiveColor{ Dark: flexokiYellow400, // ye Light: flexokiYellow600, // ye } theme.SyntaxOperatorColor = lipgloss.AdaptiveColor{ Dark: flexokiBase500, // tx-2 Light: flexokiBase500, // tx-2 } theme.SyntaxPunctuationColor = lipgloss.AdaptiveColor{ Dark: flexokiBase500, // tx-2 Light: flexokiBase500, // tx-2 } return theme } func init() { // Register the Flexoki theme with the theme manager RegisterTheme("flexoki", NewFlexokiTheme()) } ================================================ FILE: internal/tui/theme/gruvbox.go ================================================ package theme import ( "github.com/charmbracelet/lipgloss" ) // Gruvbox color palette constants const ( // Dark theme colors gruvboxDarkBg0 = "#282828" gruvboxDarkBg0Soft = "#32302f" gruvboxDarkBg1 = "#3c3836" gruvboxDarkBg2 = "#504945" gruvboxDarkBg3 = "#665c54" gruvboxDarkBg4 = "#7c6f64" gruvboxDarkFg0 = "#fbf1c7" gruvboxDarkFg1 = "#ebdbb2" gruvboxDarkFg2 = "#d5c4a1" gruvboxDarkFg3 = "#bdae93" gruvboxDarkFg4 = "#a89984" gruvboxDarkGray = "#928374" gruvboxDarkRed = "#cc241d" gruvboxDarkRedBright = "#fb4934" gruvboxDarkGreen = "#98971a" gruvboxDarkGreenBright = "#b8bb26" gruvboxDarkYellow = "#d79921" gruvboxDarkYellowBright = "#fabd2f" gruvboxDarkBlue = "#458588" gruvboxDarkBlueBright = "#83a598" gruvboxDarkPurple = "#b16286" gruvboxDarkPurpleBright = "#d3869b" gruvboxDarkAqua = "#689d6a" gruvboxDarkAquaBright = "#8ec07c" gruvboxDarkOrange = "#d65d0e" gruvboxDarkOrangeBright = "#fe8019" // Light theme colors gruvboxLightBg0 = "#fbf1c7" gruvboxLightBg0Soft = "#f2e5bc" gruvboxLightBg1 = "#ebdbb2" gruvboxLightBg2 = "#d5c4a1" gruvboxLightBg3 = "#bdae93" gruvboxLightBg4 = "#a89984" gruvboxLightFg0 = "#282828" gruvboxLightFg1 = "#3c3836" gruvboxLightFg2 = "#504945" gruvboxLightFg3 = "#665c54" gruvboxLightFg4 = "#7c6f64" gruvboxLightGray = "#928374" gruvboxLightRed = "#9d0006" gruvboxLightRedBright = "#cc241d" gruvboxLightGreen = "#79740e" gruvboxLightGreenBright = "#98971a" gruvboxLightYellow = "#b57614" gruvboxLightYellowBright = "#d79921" gruvboxLightBlue = "#076678" gruvboxLightBlueBright = "#458588" gruvboxLightPurple = "#8f3f71" gruvboxLightPurpleBright = "#b16286" gruvboxLightAqua = "#427b58" gruvboxLightAquaBright = "#689d6a" gruvboxLightOrange = "#af3a03" gruvboxLightOrangeBright = "#d65d0e" ) // GruvboxTheme implements the Theme interface with Gruvbox colors. // It provides both dark and light variants. type GruvboxTheme struct { BaseTheme } // NewGruvboxTheme creates a new instance of the Gruvbox theme. func NewGruvboxTheme() *GruvboxTheme { theme := &GruvboxTheme{} // Base colors theme.PrimaryColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkBlueBright, Light: gruvboxLightBlueBright, } theme.SecondaryColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkPurpleBright, Light: gruvboxLightPurpleBright, } theme.AccentColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkOrangeBright, Light: gruvboxLightOrangeBright, } // Status colors theme.ErrorColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkRedBright, Light: gruvboxLightRedBright, } theme.WarningColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkYellowBright, Light: gruvboxLightYellowBright, } theme.SuccessColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkGreenBright, Light: gruvboxLightGreenBright, } theme.InfoColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkBlueBright, Light: gruvboxLightBlueBright, } // Text colors theme.TextColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkFg1, Light: gruvboxLightFg1, } theme.TextMutedColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkFg4, Light: gruvboxLightFg4, } theme.TextEmphasizedColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkYellowBright, Light: gruvboxLightYellowBright, } // Background colors theme.BackgroundColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkBg0, Light: gruvboxLightBg0, } theme.BackgroundSecondaryColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkBg1, Light: gruvboxLightBg1, } theme.BackgroundDarkerColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkBg0Soft, Light: gruvboxLightBg0Soft, } // Border colors theme.BorderNormalColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkBg2, Light: gruvboxLightBg2, } theme.BorderFocusedColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkBlueBright, Light: gruvboxLightBlueBright, } theme.BorderDimColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkBg1, Light: gruvboxLightBg1, } // Diff view colors theme.DiffAddedColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkGreenBright, Light: gruvboxLightGreenBright, } theme.DiffRemovedColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkRedBright, Light: gruvboxLightRedBright, } theme.DiffContextColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkFg4, Light: gruvboxLightFg4, } theme.DiffHunkHeaderColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkFg3, Light: gruvboxLightFg3, } theme.DiffHighlightAddedColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkGreenBright, Light: gruvboxLightGreenBright, } theme.DiffHighlightRemovedColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkRedBright, Light: gruvboxLightRedBright, } theme.DiffAddedBgColor = lipgloss.AdaptiveColor{ Dark: "#3C4C3C", // Darker green background Light: "#E8F5E9", // Light green background } theme.DiffRemovedBgColor = lipgloss.AdaptiveColor{ Dark: "#4C3C3C", // Darker red background Light: "#FFEBEE", // Light red background } theme.DiffContextBgColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkBg0, Light: gruvboxLightBg0, } theme.DiffLineNumberColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkFg4, Light: gruvboxLightFg4, } theme.DiffAddedLineNumberBgColor = lipgloss.AdaptiveColor{ Dark: "#32432F", // Slightly darker green Light: "#C8E6C9", // Light green } theme.DiffRemovedLineNumberBgColor = lipgloss.AdaptiveColor{ Dark: "#43322F", // Slightly darker red Light: "#FFCDD2", // Light red } // Markdown colors theme.MarkdownTextColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkFg1, Light: gruvboxLightFg1, } theme.MarkdownHeadingColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkYellowBright, Light: gruvboxLightYellowBright, } theme.MarkdownLinkColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkBlueBright, Light: gruvboxLightBlueBright, } theme.MarkdownLinkTextColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkAquaBright, Light: gruvboxLightAquaBright, } theme.MarkdownCodeColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkGreenBright, Light: gruvboxLightGreenBright, } theme.MarkdownBlockQuoteColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkAquaBright, Light: gruvboxLightAquaBright, } theme.MarkdownEmphColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkYellowBright, Light: gruvboxLightYellowBright, } theme.MarkdownStrongColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkOrangeBright, Light: gruvboxLightOrangeBright, } theme.MarkdownHorizontalRuleColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkBg3, Light: gruvboxLightBg3, } theme.MarkdownListItemColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkBlueBright, Light: gruvboxLightBlueBright, } theme.MarkdownListEnumerationColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkBlueBright, Light: gruvboxLightBlueBright, } theme.MarkdownImageColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkPurpleBright, Light: gruvboxLightPurpleBright, } theme.MarkdownImageTextColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkAquaBright, Light: gruvboxLightAquaBright, } theme.MarkdownCodeBlockColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkFg1, Light: gruvboxLightFg1, } // Syntax highlighting colors theme.SyntaxCommentColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkGray, Light: gruvboxLightGray, } theme.SyntaxKeywordColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkRedBright, Light: gruvboxLightRedBright, } theme.SyntaxFunctionColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkGreenBright, Light: gruvboxLightGreenBright, } theme.SyntaxVariableColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkBlueBright, Light: gruvboxLightBlueBright, } theme.SyntaxStringColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkYellowBright, Light: gruvboxLightYellowBright, } theme.SyntaxNumberColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkPurpleBright, Light: gruvboxLightPurpleBright, } theme.SyntaxTypeColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkYellow, Light: gruvboxLightYellow, } theme.SyntaxOperatorColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkAquaBright, Light: gruvboxLightAquaBright, } theme.SyntaxPunctuationColor = lipgloss.AdaptiveColor{ Dark: gruvboxDarkFg1, Light: gruvboxLightFg1, } return theme } func init() { // Register the Gruvbox theme with the theme manager RegisterTheme("gruvbox", NewGruvboxTheme()) } ================================================ FILE: internal/tui/theme/manager.go ================================================ package theme import ( "fmt" "slices" "strings" "sync" "github.com/alecthomas/chroma/v2/styles" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/logging" ) // Manager handles theme registration, selection, and retrieval. // It maintains a registry of available themes and tracks the currently active theme. type Manager struct { themes map[string]Theme currentName string mu sync.RWMutex } // Global instance of the theme manager var globalManager = &Manager{ themes: make(map[string]Theme), currentName: "", } // RegisterTheme adds a new theme to the registry. // If this is the first theme registered, it becomes the default. func RegisterTheme(name string, theme Theme) { globalManager.mu.Lock() defer globalManager.mu.Unlock() globalManager.themes[name] = theme // If this is the first theme, make it the default if globalManager.currentName == "" { globalManager.currentName = name } } // SetTheme changes the active theme to the one with the specified name. // Returns an error if the theme doesn't exist. func SetTheme(name string) error { globalManager.mu.Lock() defer globalManager.mu.Unlock() delete(styles.Registry, "charm") if _, exists := globalManager.themes[name]; !exists { return fmt.Errorf("theme '%s' not found", name) } globalManager.currentName = name // Update the config file using viper if err := updateConfigTheme(name); err != nil { // Log the error but don't fail the theme change logging.Warn("Warning: Failed to update config file with new theme", "err", err) } return nil } // CurrentTheme returns the currently active theme. // If no theme is set, it returns nil. func CurrentTheme() Theme { globalManager.mu.RLock() defer globalManager.mu.RUnlock() if globalManager.currentName == "" { return nil } return globalManager.themes[globalManager.currentName] } // CurrentThemeName returns the name of the currently active theme. func CurrentThemeName() string { globalManager.mu.RLock() defer globalManager.mu.RUnlock() return globalManager.currentName } // AvailableThemes returns a list of all registered theme names. func AvailableThemes() []string { globalManager.mu.RLock() defer globalManager.mu.RUnlock() names := make([]string, 0, len(globalManager.themes)) for name := range globalManager.themes { names = append(names, name) } slices.SortFunc(names, func(a, b string) int { if a == "opencode" { return -1 } else if b == "opencode" { return 1 } return strings.Compare(a, b) }) return names } // GetTheme returns a specific theme by name. // Returns nil if the theme doesn't exist. func GetTheme(name string) Theme { globalManager.mu.RLock() defer globalManager.mu.RUnlock() return globalManager.themes[name] } // updateConfigTheme updates the theme setting in the configuration file func updateConfigTheme(themeName string) error { // Use the config package to update the theme return config.UpdateTheme(themeName) } ================================================ FILE: internal/tui/theme/monokai.go ================================================ package theme import ( "github.com/charmbracelet/lipgloss" ) // MonokaiProTheme implements the Theme interface with Monokai Pro colors. // It provides both dark and light variants. type MonokaiProTheme struct { BaseTheme } // NewMonokaiProTheme creates a new instance of the Monokai Pro theme. func NewMonokaiProTheme() *MonokaiProTheme { // Monokai Pro color palette (dark mode) darkBackground := "#2d2a2e" darkCurrentLine := "#403e41" darkSelection := "#5b595c" darkForeground := "#fcfcfa" darkComment := "#727072" darkRed := "#ff6188" darkOrange := "#fc9867" darkYellow := "#ffd866" darkGreen := "#a9dc76" darkCyan := "#78dce8" darkBlue := "#ab9df2" darkPurple := "#ab9df2" darkBorder := "#403e41" // Light mode colors (adapted from dark) lightBackground := "#fafafa" lightCurrentLine := "#f0f0f0" lightSelection := "#e5e5e6" lightForeground := "#2d2a2e" lightComment := "#939293" lightRed := "#f92672" lightOrange := "#fd971f" lightYellow := "#e6db74" lightGreen := "#9bca65" lightCyan := "#66d9ef" lightBlue := "#7e75db" lightPurple := "#ae81ff" lightBorder := "#d3d3d3" theme := &MonokaiProTheme{} // Base colors theme.PrimaryColor = lipgloss.AdaptiveColor{ Dark: darkCyan, Light: lightCyan, } theme.SecondaryColor = lipgloss.AdaptiveColor{ Dark: darkPurple, Light: lightPurple, } theme.AccentColor = lipgloss.AdaptiveColor{ Dark: darkOrange, Light: lightOrange, } // Status colors theme.ErrorColor = lipgloss.AdaptiveColor{ Dark: darkRed, Light: lightRed, } theme.WarningColor = lipgloss.AdaptiveColor{ Dark: darkOrange, Light: lightOrange, } theme.SuccessColor = lipgloss.AdaptiveColor{ Dark: darkGreen, Light: lightGreen, } theme.InfoColor = lipgloss.AdaptiveColor{ Dark: darkBlue, Light: lightBlue, } // Text colors theme.TextColor = lipgloss.AdaptiveColor{ Dark: darkForeground, Light: lightForeground, } theme.TextMutedColor = lipgloss.AdaptiveColor{ Dark: darkComment, Light: lightComment, } theme.TextEmphasizedColor = lipgloss.AdaptiveColor{ Dark: darkYellow, Light: lightYellow, } // Background colors theme.BackgroundColor = lipgloss.AdaptiveColor{ Dark: darkBackground, Light: lightBackground, } theme.BackgroundSecondaryColor = lipgloss.AdaptiveColor{ Dark: darkCurrentLine, Light: lightCurrentLine, } theme.BackgroundDarkerColor = lipgloss.AdaptiveColor{ Dark: "#221f22", // Slightly darker than background Light: "#ffffff", // Slightly lighter than background } // Border colors theme.BorderNormalColor = lipgloss.AdaptiveColor{ Dark: darkBorder, Light: lightBorder, } theme.BorderFocusedColor = lipgloss.AdaptiveColor{ Dark: darkCyan, Light: lightCyan, } theme.BorderDimColor = lipgloss.AdaptiveColor{ Dark: darkSelection, Light: lightSelection, } // Diff view colors theme.DiffAddedColor = lipgloss.AdaptiveColor{ Dark: "#a9dc76", Light: "#9bca65", } theme.DiffRemovedColor = lipgloss.AdaptiveColor{ Dark: "#ff6188", Light: "#f92672", } theme.DiffContextColor = lipgloss.AdaptiveColor{ Dark: "#a0a0a0", Light: "#757575", } theme.DiffHunkHeaderColor = lipgloss.AdaptiveColor{ Dark: "#a0a0a0", Light: "#757575", } theme.DiffHighlightAddedColor = lipgloss.AdaptiveColor{ Dark: "#c2e7a9", Light: "#c5e0b4", } theme.DiffHighlightRemovedColor = lipgloss.AdaptiveColor{ Dark: "#ff8ca6", Light: "#ffb3c8", } theme.DiffAddedBgColor = lipgloss.AdaptiveColor{ Dark: "#3a4a35", Light: "#e8f5e9", } theme.DiffRemovedBgColor = lipgloss.AdaptiveColor{ Dark: "#4a3439", Light: "#ffebee", } theme.DiffContextBgColor = lipgloss.AdaptiveColor{ Dark: darkBackground, Light: lightBackground, } theme.DiffLineNumberColor = lipgloss.AdaptiveColor{ Dark: "#888888", Light: "#9e9e9e", } theme.DiffAddedLineNumberBgColor = lipgloss.AdaptiveColor{ Dark: "#2d3a28", Light: "#c8e6c9", } theme.DiffRemovedLineNumberBgColor = lipgloss.AdaptiveColor{ Dark: "#3d2a2e", Light: "#ffcdd2", } // Markdown colors theme.MarkdownTextColor = lipgloss.AdaptiveColor{ Dark: darkForeground, Light: lightForeground, } theme.MarkdownHeadingColor = lipgloss.AdaptiveColor{ Dark: darkPurple, Light: lightPurple, } theme.MarkdownLinkColor = lipgloss.AdaptiveColor{ Dark: darkCyan, Light: lightCyan, } theme.MarkdownLinkTextColor = lipgloss.AdaptiveColor{ Dark: darkBlue, Light: lightBlue, } theme.MarkdownCodeColor = lipgloss.AdaptiveColor{ Dark: darkGreen, Light: lightGreen, } theme.MarkdownBlockQuoteColor = lipgloss.AdaptiveColor{ Dark: darkYellow, Light: lightYellow, } theme.MarkdownEmphColor = lipgloss.AdaptiveColor{ Dark: darkYellow, Light: lightYellow, } theme.MarkdownStrongColor = lipgloss.AdaptiveColor{ Dark: darkOrange, Light: lightOrange, } theme.MarkdownHorizontalRuleColor = lipgloss.AdaptiveColor{ Dark: darkComment, Light: lightComment, } theme.MarkdownListItemColor = lipgloss.AdaptiveColor{ Dark: darkCyan, Light: lightCyan, } theme.MarkdownListEnumerationColor = lipgloss.AdaptiveColor{ Dark: darkBlue, Light: lightBlue, } theme.MarkdownImageColor = lipgloss.AdaptiveColor{ Dark: darkCyan, Light: lightCyan, } theme.MarkdownImageTextColor = lipgloss.AdaptiveColor{ Dark: darkBlue, Light: lightBlue, } theme.MarkdownCodeBlockColor = lipgloss.AdaptiveColor{ Dark: darkForeground, Light: lightForeground, } // Syntax highlighting colors theme.SyntaxCommentColor = lipgloss.AdaptiveColor{ Dark: darkComment, Light: lightComment, } theme.SyntaxKeywordColor = lipgloss.AdaptiveColor{ Dark: darkRed, Light: lightRed, } theme.SyntaxFunctionColor = lipgloss.AdaptiveColor{ Dark: darkGreen, Light: lightGreen, } theme.SyntaxVariableColor = lipgloss.AdaptiveColor{ Dark: darkForeground, Light: lightForeground, } theme.SyntaxStringColor = lipgloss.AdaptiveColor{ Dark: darkYellow, Light: lightYellow, } theme.SyntaxNumberColor = lipgloss.AdaptiveColor{ Dark: darkPurple, Light: lightPurple, } theme.SyntaxTypeColor = lipgloss.AdaptiveColor{ Dark: darkBlue, Light: lightBlue, } theme.SyntaxOperatorColor = lipgloss.AdaptiveColor{ Dark: darkCyan, Light: lightCyan, } theme.SyntaxPunctuationColor = lipgloss.AdaptiveColor{ Dark: darkForeground, Light: lightForeground, } return theme } func init() { // Register the Monokai Pro theme with the theme manager RegisterTheme("monokai", NewMonokaiProTheme()) } ================================================ FILE: internal/tui/theme/onedark.go ================================================ package theme import ( "github.com/charmbracelet/lipgloss" ) // OneDarkTheme implements the Theme interface with Atom's One Dark colors. // It provides both dark and light variants. type OneDarkTheme struct { BaseTheme } // NewOneDarkTheme creates a new instance of the One Dark theme. func NewOneDarkTheme() *OneDarkTheme { // One Dark color palette // Dark mode colors from Atom One Dark darkBackground := "#282c34" darkCurrentLine := "#2c313c" darkSelection := "#3e4451" darkForeground := "#abb2bf" darkComment := "#5c6370" darkRed := "#e06c75" darkOrange := "#d19a66" darkYellow := "#e5c07b" darkGreen := "#98c379" darkCyan := "#56b6c2" darkBlue := "#61afef" darkPurple := "#c678dd" darkBorder := "#3b4048" // Light mode colors from Atom One Light lightBackground := "#fafafa" lightCurrentLine := "#f0f0f0" lightSelection := "#e5e5e6" lightForeground := "#383a42" lightComment := "#a0a1a7" lightRed := "#e45649" lightOrange := "#da8548" lightYellow := "#c18401" lightGreen := "#50a14f" lightCyan := "#0184bc" lightBlue := "#4078f2" lightPurple := "#a626a4" lightBorder := "#d3d3d3" theme := &OneDarkTheme{} // Base colors theme.PrimaryColor = lipgloss.AdaptiveColor{ Dark: darkBlue, Light: lightBlue, } theme.SecondaryColor = lipgloss.AdaptiveColor{ Dark: darkPurple, Light: lightPurple, } theme.AccentColor = lipgloss.AdaptiveColor{ Dark: darkOrange, Light: lightOrange, } // Status colors theme.ErrorColor = lipgloss.AdaptiveColor{ Dark: darkRed, Light: lightRed, } theme.WarningColor = lipgloss.AdaptiveColor{ Dark: darkOrange, Light: lightOrange, } theme.SuccessColor = lipgloss.AdaptiveColor{ Dark: darkGreen, Light: lightGreen, } theme.InfoColor = lipgloss.AdaptiveColor{ Dark: darkBlue, Light: lightBlue, } // Text colors theme.TextColor = lipgloss.AdaptiveColor{ Dark: darkForeground, Light: lightForeground, } theme.TextMutedColor = lipgloss.AdaptiveColor{ Dark: darkComment, Light: lightComment, } theme.TextEmphasizedColor = lipgloss.AdaptiveColor{ Dark: darkYellow, Light: lightYellow, } // Background colors theme.BackgroundColor = lipgloss.AdaptiveColor{ Dark: darkBackground, Light: lightBackground, } theme.BackgroundSecondaryColor = lipgloss.AdaptiveColor{ Dark: darkCurrentLine, Light: lightCurrentLine, } theme.BackgroundDarkerColor = lipgloss.AdaptiveColor{ Dark: "#21252b", // Slightly darker than background Light: "#ffffff", // Slightly lighter than background } // Border colors theme.BorderNormalColor = lipgloss.AdaptiveColor{ Dark: darkBorder, Light: lightBorder, } theme.BorderFocusedColor = lipgloss.AdaptiveColor{ Dark: darkBlue, Light: lightBlue, } theme.BorderDimColor = lipgloss.AdaptiveColor{ Dark: darkSelection, Light: lightSelection, } // Diff view colors theme.DiffAddedColor = lipgloss.AdaptiveColor{ Dark: "#478247", Light: "#2E7D32", } theme.DiffRemovedColor = lipgloss.AdaptiveColor{ Dark: "#7C4444", Light: "#C62828", } theme.DiffContextColor = lipgloss.AdaptiveColor{ Dark: "#a0a0a0", Light: "#757575", } theme.DiffHunkHeaderColor = lipgloss.AdaptiveColor{ Dark: "#a0a0a0", Light: "#757575", } theme.DiffHighlightAddedColor = lipgloss.AdaptiveColor{ Dark: "#DAFADA", Light: "#A5D6A7", } theme.DiffHighlightRemovedColor = lipgloss.AdaptiveColor{ Dark: "#FADADD", Light: "#EF9A9A", } theme.DiffAddedBgColor = lipgloss.AdaptiveColor{ Dark: "#303A30", Light: "#E8F5E9", } theme.DiffRemovedBgColor = lipgloss.AdaptiveColor{ Dark: "#3A3030", Light: "#FFEBEE", } theme.DiffContextBgColor = lipgloss.AdaptiveColor{ Dark: darkBackground, Light: lightBackground, } theme.DiffLineNumberColor = lipgloss.AdaptiveColor{ Dark: "#888888", Light: "#9E9E9E", } theme.DiffAddedLineNumberBgColor = lipgloss.AdaptiveColor{ Dark: "#293229", Light: "#C8E6C9", } theme.DiffRemovedLineNumberBgColor = lipgloss.AdaptiveColor{ Dark: "#332929", Light: "#FFCDD2", } // Markdown colors theme.MarkdownTextColor = lipgloss.AdaptiveColor{ Dark: darkForeground, Light: lightForeground, } theme.MarkdownHeadingColor = lipgloss.AdaptiveColor{ Dark: darkPurple, Light: lightPurple, } theme.MarkdownLinkColor = lipgloss.AdaptiveColor{ Dark: darkBlue, Light: lightBlue, } theme.MarkdownLinkTextColor = lipgloss.AdaptiveColor{ Dark: darkCyan, Light: lightCyan, } theme.MarkdownCodeColor = lipgloss.AdaptiveColor{ Dark: darkGreen, Light: lightGreen, } theme.MarkdownBlockQuoteColor = lipgloss.AdaptiveColor{ Dark: darkYellow, Light: lightYellow, } theme.MarkdownEmphColor = lipgloss.AdaptiveColor{ Dark: darkYellow, Light: lightYellow, } theme.MarkdownStrongColor = lipgloss.AdaptiveColor{ Dark: darkOrange, Light: lightOrange, } theme.MarkdownHorizontalRuleColor = lipgloss.AdaptiveColor{ Dark: darkComment, Light: lightComment, } theme.MarkdownListItemColor = lipgloss.AdaptiveColor{ Dark: darkBlue, Light: lightBlue, } theme.MarkdownListEnumerationColor = lipgloss.AdaptiveColor{ Dark: darkCyan, Light: lightCyan, } theme.MarkdownImageColor = lipgloss.AdaptiveColor{ Dark: darkBlue, Light: lightBlue, } theme.MarkdownImageTextColor = lipgloss.AdaptiveColor{ Dark: darkCyan, Light: lightCyan, } theme.MarkdownCodeBlockColor = lipgloss.AdaptiveColor{ Dark: darkForeground, Light: lightForeground, } // Syntax highlighting colors theme.SyntaxCommentColor = lipgloss.AdaptiveColor{ Dark: darkComment, Light: lightComment, } theme.SyntaxKeywordColor = lipgloss.AdaptiveColor{ Dark: darkPurple, Light: lightPurple, } theme.SyntaxFunctionColor = lipgloss.AdaptiveColor{ Dark: darkBlue, Light: lightBlue, } theme.SyntaxVariableColor = lipgloss.AdaptiveColor{ Dark: darkRed, Light: lightRed, } theme.SyntaxStringColor = lipgloss.AdaptiveColor{ Dark: darkGreen, Light: lightGreen, } theme.SyntaxNumberColor = lipgloss.AdaptiveColor{ Dark: darkOrange, Light: lightOrange, } theme.SyntaxTypeColor = lipgloss.AdaptiveColor{ Dark: darkYellow, Light: lightYellow, } theme.SyntaxOperatorColor = lipgloss.AdaptiveColor{ Dark: darkCyan, Light: lightCyan, } theme.SyntaxPunctuationColor = lipgloss.AdaptiveColor{ Dark: darkForeground, Light: lightForeground, } return theme } func init() { // Register the One Dark theme with the theme manager RegisterTheme("onedark", NewOneDarkTheme()) } ================================================ FILE: internal/tui/theme/opencode.go ================================================ package theme import ( "github.com/charmbracelet/lipgloss" ) // OpenCodeTheme implements the Theme interface with OpenCode brand colors. // It provides both dark and light variants. type OpenCodeTheme struct { BaseTheme } // NewOpenCodeTheme creates a new instance of the OpenCode theme. func NewOpenCodeTheme() *OpenCodeTheme { // OpenCode color palette // Dark mode colors darkBackground := "#212121" darkCurrentLine := "#252525" darkSelection := "#303030" darkForeground := "#e0e0e0" darkComment := "#6a6a6a" darkPrimary := "#fab283" // Primary orange/gold darkSecondary := "#5c9cf5" // Secondary blue darkAccent := "#9d7cd8" // Accent purple darkRed := "#e06c75" // Error red darkOrange := "#f5a742" // Warning orange darkGreen := "#7fd88f" // Success green darkCyan := "#56b6c2" // Info cyan darkYellow := "#e5c07b" // Emphasized text darkBorder := "#4b4c5c" // Border color // Light mode colors lightBackground := "#f8f8f8" lightCurrentLine := "#f0f0f0" lightSelection := "#e5e5e6" lightForeground := "#2a2a2a" lightComment := "#8a8a8a" lightPrimary := "#3b7dd8" // Primary blue lightSecondary := "#7b5bb6" // Secondary purple lightAccent := "#d68c27" // Accent orange/gold lightRed := "#d1383d" // Error red lightOrange := "#d68c27" // Warning orange lightGreen := "#3d9a57" // Success green lightCyan := "#318795" // Info cyan lightYellow := "#b0851f" // Emphasized text lightBorder := "#d3d3d3" // Border color theme := &OpenCodeTheme{} // Base colors theme.PrimaryColor = lipgloss.AdaptiveColor{ Dark: darkPrimary, Light: lightPrimary, } theme.SecondaryColor = lipgloss.AdaptiveColor{ Dark: darkSecondary, Light: lightSecondary, } theme.AccentColor = lipgloss.AdaptiveColor{ Dark: darkAccent, Light: lightAccent, } // Status colors theme.ErrorColor = lipgloss.AdaptiveColor{ Dark: darkRed, Light: lightRed, } theme.WarningColor = lipgloss.AdaptiveColor{ Dark: darkOrange, Light: lightOrange, } theme.SuccessColor = lipgloss.AdaptiveColor{ Dark: darkGreen, Light: lightGreen, } theme.InfoColor = lipgloss.AdaptiveColor{ Dark: darkCyan, Light: lightCyan, } // Text colors theme.TextColor = lipgloss.AdaptiveColor{ Dark: darkForeground, Light: lightForeground, } theme.TextMutedColor = lipgloss.AdaptiveColor{ Dark: darkComment, Light: lightComment, } theme.TextEmphasizedColor = lipgloss.AdaptiveColor{ Dark: darkYellow, Light: lightYellow, } // Background colors theme.BackgroundColor = lipgloss.AdaptiveColor{ Dark: darkBackground, Light: lightBackground, } theme.BackgroundSecondaryColor = lipgloss.AdaptiveColor{ Dark: darkCurrentLine, Light: lightCurrentLine, } theme.BackgroundDarkerColor = lipgloss.AdaptiveColor{ Dark: "#121212", // Slightly darker than background Light: "#ffffff", // Slightly lighter than background } // Border colors theme.BorderNormalColor = lipgloss.AdaptiveColor{ Dark: darkBorder, Light: lightBorder, } theme.BorderFocusedColor = lipgloss.AdaptiveColor{ Dark: darkPrimary, Light: lightPrimary, } theme.BorderDimColor = lipgloss.AdaptiveColor{ Dark: darkSelection, Light: lightSelection, } // Diff view colors theme.DiffAddedColor = lipgloss.AdaptiveColor{ Dark: "#478247", Light: "#2E7D32", } theme.DiffRemovedColor = lipgloss.AdaptiveColor{ Dark: "#7C4444", Light: "#C62828", } theme.DiffContextColor = lipgloss.AdaptiveColor{ Dark: "#a0a0a0", Light: "#757575", } theme.DiffHunkHeaderColor = lipgloss.AdaptiveColor{ Dark: "#a0a0a0", Light: "#757575", } theme.DiffHighlightAddedColor = lipgloss.AdaptiveColor{ Dark: "#DAFADA", Light: "#A5D6A7", } theme.DiffHighlightRemovedColor = lipgloss.AdaptiveColor{ Dark: "#FADADD", Light: "#EF9A9A", } theme.DiffAddedBgColor = lipgloss.AdaptiveColor{ Dark: "#303A30", Light: "#E8F5E9", } theme.DiffRemovedBgColor = lipgloss.AdaptiveColor{ Dark: "#3A3030", Light: "#FFEBEE", } theme.DiffContextBgColor = lipgloss.AdaptiveColor{ Dark: darkBackground, Light: lightBackground, } theme.DiffLineNumberColor = lipgloss.AdaptiveColor{ Dark: "#888888", Light: "#9E9E9E", } theme.DiffAddedLineNumberBgColor = lipgloss.AdaptiveColor{ Dark: "#293229", Light: "#C8E6C9", } theme.DiffRemovedLineNumberBgColor = lipgloss.AdaptiveColor{ Dark: "#332929", Light: "#FFCDD2", } // Markdown colors theme.MarkdownTextColor = lipgloss.AdaptiveColor{ Dark: darkForeground, Light: lightForeground, } theme.MarkdownHeadingColor = lipgloss.AdaptiveColor{ Dark: darkSecondary, Light: lightSecondary, } theme.MarkdownLinkColor = lipgloss.AdaptiveColor{ Dark: darkPrimary, Light: lightPrimary, } theme.MarkdownLinkTextColor = lipgloss.AdaptiveColor{ Dark: darkCyan, Light: lightCyan, } theme.MarkdownCodeColor = lipgloss.AdaptiveColor{ Dark: darkGreen, Light: lightGreen, } theme.MarkdownBlockQuoteColor = lipgloss.AdaptiveColor{ Dark: darkYellow, Light: lightYellow, } theme.MarkdownEmphColor = lipgloss.AdaptiveColor{ Dark: darkYellow, Light: lightYellow, } theme.MarkdownStrongColor = lipgloss.AdaptiveColor{ Dark: darkAccent, Light: lightAccent, } theme.MarkdownHorizontalRuleColor = lipgloss.AdaptiveColor{ Dark: darkComment, Light: lightComment, } theme.MarkdownListItemColor = lipgloss.AdaptiveColor{ Dark: darkPrimary, Light: lightPrimary, } theme.MarkdownListEnumerationColor = lipgloss.AdaptiveColor{ Dark: darkCyan, Light: lightCyan, } theme.MarkdownImageColor = lipgloss.AdaptiveColor{ Dark: darkPrimary, Light: lightPrimary, } theme.MarkdownImageTextColor = lipgloss.AdaptiveColor{ Dark: darkCyan, Light: lightCyan, } theme.MarkdownCodeBlockColor = lipgloss.AdaptiveColor{ Dark: darkForeground, Light: lightForeground, } // Syntax highlighting colors theme.SyntaxCommentColor = lipgloss.AdaptiveColor{ Dark: darkComment, Light: lightComment, } theme.SyntaxKeywordColor = lipgloss.AdaptiveColor{ Dark: darkSecondary, Light: lightSecondary, } theme.SyntaxFunctionColor = lipgloss.AdaptiveColor{ Dark: darkPrimary, Light: lightPrimary, } theme.SyntaxVariableColor = lipgloss.AdaptiveColor{ Dark: darkRed, Light: lightRed, } theme.SyntaxStringColor = lipgloss.AdaptiveColor{ Dark: darkGreen, Light: lightGreen, } theme.SyntaxNumberColor = lipgloss.AdaptiveColor{ Dark: darkAccent, Light: lightAccent, } theme.SyntaxTypeColor = lipgloss.AdaptiveColor{ Dark: darkYellow, Light: lightYellow, } theme.SyntaxOperatorColor = lipgloss.AdaptiveColor{ Dark: darkCyan, Light: lightCyan, } theme.SyntaxPunctuationColor = lipgloss.AdaptiveColor{ Dark: darkForeground, Light: lightForeground, } return theme } func init() { // Register the OpenCode theme with the theme manager RegisterTheme("opencode", NewOpenCodeTheme()) } ================================================ FILE: internal/tui/theme/theme.go ================================================ package theme import ( "github.com/charmbracelet/lipgloss" ) // Theme defines the interface for all UI themes in the application. // All colors must be defined as lipgloss.AdaptiveColor to support // both light and dark terminal backgrounds. type Theme interface { // Base colors Primary() lipgloss.AdaptiveColor Secondary() lipgloss.AdaptiveColor Accent() lipgloss.AdaptiveColor // Status colors Error() lipgloss.AdaptiveColor Warning() lipgloss.AdaptiveColor Success() lipgloss.AdaptiveColor Info() lipgloss.AdaptiveColor // Text colors Text() lipgloss.AdaptiveColor TextMuted() lipgloss.AdaptiveColor TextEmphasized() lipgloss.AdaptiveColor // Background colors Background() lipgloss.AdaptiveColor BackgroundSecondary() lipgloss.AdaptiveColor BackgroundDarker() lipgloss.AdaptiveColor // Border colors BorderNormal() lipgloss.AdaptiveColor BorderFocused() lipgloss.AdaptiveColor BorderDim() lipgloss.AdaptiveColor // Diff view colors DiffAdded() lipgloss.AdaptiveColor DiffRemoved() lipgloss.AdaptiveColor DiffContext() lipgloss.AdaptiveColor DiffHunkHeader() lipgloss.AdaptiveColor DiffHighlightAdded() lipgloss.AdaptiveColor DiffHighlightRemoved() lipgloss.AdaptiveColor DiffAddedBg() lipgloss.AdaptiveColor DiffRemovedBg() lipgloss.AdaptiveColor DiffContextBg() lipgloss.AdaptiveColor DiffLineNumber() lipgloss.AdaptiveColor DiffAddedLineNumberBg() lipgloss.AdaptiveColor DiffRemovedLineNumberBg() lipgloss.AdaptiveColor // Markdown colors MarkdownText() lipgloss.AdaptiveColor MarkdownHeading() lipgloss.AdaptiveColor MarkdownLink() lipgloss.AdaptiveColor MarkdownLinkText() lipgloss.AdaptiveColor MarkdownCode() lipgloss.AdaptiveColor MarkdownBlockQuote() lipgloss.AdaptiveColor MarkdownEmph() lipgloss.AdaptiveColor MarkdownStrong() lipgloss.AdaptiveColor MarkdownHorizontalRule() lipgloss.AdaptiveColor MarkdownListItem() lipgloss.AdaptiveColor MarkdownListEnumeration() lipgloss.AdaptiveColor MarkdownImage() lipgloss.AdaptiveColor MarkdownImageText() lipgloss.AdaptiveColor MarkdownCodeBlock() lipgloss.AdaptiveColor // Syntax highlighting colors SyntaxComment() lipgloss.AdaptiveColor SyntaxKeyword() lipgloss.AdaptiveColor SyntaxFunction() lipgloss.AdaptiveColor SyntaxVariable() lipgloss.AdaptiveColor SyntaxString() lipgloss.AdaptiveColor SyntaxNumber() lipgloss.AdaptiveColor SyntaxType() lipgloss.AdaptiveColor SyntaxOperator() lipgloss.AdaptiveColor SyntaxPunctuation() lipgloss.AdaptiveColor } // BaseTheme provides a default implementation of the Theme interface // that can be embedded in concrete theme implementations. type BaseTheme struct { // Base colors PrimaryColor lipgloss.AdaptiveColor SecondaryColor lipgloss.AdaptiveColor AccentColor lipgloss.AdaptiveColor // Status colors ErrorColor lipgloss.AdaptiveColor WarningColor lipgloss.AdaptiveColor SuccessColor lipgloss.AdaptiveColor InfoColor lipgloss.AdaptiveColor // Text colors TextColor lipgloss.AdaptiveColor TextMutedColor lipgloss.AdaptiveColor TextEmphasizedColor lipgloss.AdaptiveColor // Background colors BackgroundColor lipgloss.AdaptiveColor BackgroundSecondaryColor lipgloss.AdaptiveColor BackgroundDarkerColor lipgloss.AdaptiveColor // Border colors BorderNormalColor lipgloss.AdaptiveColor BorderFocusedColor lipgloss.AdaptiveColor BorderDimColor lipgloss.AdaptiveColor // Diff view colors DiffAddedColor lipgloss.AdaptiveColor DiffRemovedColor lipgloss.AdaptiveColor DiffContextColor lipgloss.AdaptiveColor DiffHunkHeaderColor lipgloss.AdaptiveColor DiffHighlightAddedColor lipgloss.AdaptiveColor DiffHighlightRemovedColor lipgloss.AdaptiveColor DiffAddedBgColor lipgloss.AdaptiveColor DiffRemovedBgColor lipgloss.AdaptiveColor DiffContextBgColor lipgloss.AdaptiveColor DiffLineNumberColor lipgloss.AdaptiveColor DiffAddedLineNumberBgColor lipgloss.AdaptiveColor DiffRemovedLineNumberBgColor lipgloss.AdaptiveColor // Markdown colors MarkdownTextColor lipgloss.AdaptiveColor MarkdownHeadingColor lipgloss.AdaptiveColor MarkdownLinkColor lipgloss.AdaptiveColor MarkdownLinkTextColor lipgloss.AdaptiveColor MarkdownCodeColor lipgloss.AdaptiveColor MarkdownBlockQuoteColor lipgloss.AdaptiveColor MarkdownEmphColor lipgloss.AdaptiveColor MarkdownStrongColor lipgloss.AdaptiveColor MarkdownHorizontalRuleColor lipgloss.AdaptiveColor MarkdownListItemColor lipgloss.AdaptiveColor MarkdownListEnumerationColor lipgloss.AdaptiveColor MarkdownImageColor lipgloss.AdaptiveColor MarkdownImageTextColor lipgloss.AdaptiveColor MarkdownCodeBlockColor lipgloss.AdaptiveColor // Syntax highlighting colors SyntaxCommentColor lipgloss.AdaptiveColor SyntaxKeywordColor lipgloss.AdaptiveColor SyntaxFunctionColor lipgloss.AdaptiveColor SyntaxVariableColor lipgloss.AdaptiveColor SyntaxStringColor lipgloss.AdaptiveColor SyntaxNumberColor lipgloss.AdaptiveColor SyntaxTypeColor lipgloss.AdaptiveColor SyntaxOperatorColor lipgloss.AdaptiveColor SyntaxPunctuationColor lipgloss.AdaptiveColor } // Implement the Theme interface for BaseTheme func (t *BaseTheme) Primary() lipgloss.AdaptiveColor { return t.PrimaryColor } func (t *BaseTheme) Secondary() lipgloss.AdaptiveColor { return t.SecondaryColor } func (t *BaseTheme) Accent() lipgloss.AdaptiveColor { return t.AccentColor } func (t *BaseTheme) Error() lipgloss.AdaptiveColor { return t.ErrorColor } func (t *BaseTheme) Warning() lipgloss.AdaptiveColor { return t.WarningColor } func (t *BaseTheme) Success() lipgloss.AdaptiveColor { return t.SuccessColor } func (t *BaseTheme) Info() lipgloss.AdaptiveColor { return t.InfoColor } func (t *BaseTheme) Text() lipgloss.AdaptiveColor { return t.TextColor } func (t *BaseTheme) TextMuted() lipgloss.AdaptiveColor { return t.TextMutedColor } func (t *BaseTheme) TextEmphasized() lipgloss.AdaptiveColor { return t.TextEmphasizedColor } func (t *BaseTheme) Background() lipgloss.AdaptiveColor { return t.BackgroundColor } func (t *BaseTheme) BackgroundSecondary() lipgloss.AdaptiveColor { return t.BackgroundSecondaryColor } func (t *BaseTheme) BackgroundDarker() lipgloss.AdaptiveColor { return t.BackgroundDarkerColor } func (t *BaseTheme) BorderNormal() lipgloss.AdaptiveColor { return t.BorderNormalColor } func (t *BaseTheme) BorderFocused() lipgloss.AdaptiveColor { return t.BorderFocusedColor } func (t *BaseTheme) BorderDim() lipgloss.AdaptiveColor { return t.BorderDimColor } func (t *BaseTheme) DiffAdded() lipgloss.AdaptiveColor { return t.DiffAddedColor } func (t *BaseTheme) DiffRemoved() lipgloss.AdaptiveColor { return t.DiffRemovedColor } func (t *BaseTheme) DiffContext() lipgloss.AdaptiveColor { return t.DiffContextColor } func (t *BaseTheme) DiffHunkHeader() lipgloss.AdaptiveColor { return t.DiffHunkHeaderColor } func (t *BaseTheme) DiffHighlightAdded() lipgloss.AdaptiveColor { return t.DiffHighlightAddedColor } func (t *BaseTheme) DiffHighlightRemoved() lipgloss.AdaptiveColor { return t.DiffHighlightRemovedColor } func (t *BaseTheme) DiffAddedBg() lipgloss.AdaptiveColor { return t.DiffAddedBgColor } func (t *BaseTheme) DiffRemovedBg() lipgloss.AdaptiveColor { return t.DiffRemovedBgColor } func (t *BaseTheme) DiffContextBg() lipgloss.AdaptiveColor { return t.DiffContextBgColor } func (t *BaseTheme) DiffLineNumber() lipgloss.AdaptiveColor { return t.DiffLineNumberColor } func (t *BaseTheme) DiffAddedLineNumberBg() lipgloss.AdaptiveColor { return t.DiffAddedLineNumberBgColor } func (t *BaseTheme) DiffRemovedLineNumberBg() lipgloss.AdaptiveColor { return t.DiffRemovedLineNumberBgColor } func (t *BaseTheme) MarkdownText() lipgloss.AdaptiveColor { return t.MarkdownTextColor } func (t *BaseTheme) MarkdownHeading() lipgloss.AdaptiveColor { return t.MarkdownHeadingColor } func (t *BaseTheme) MarkdownLink() lipgloss.AdaptiveColor { return t.MarkdownLinkColor } func (t *BaseTheme) MarkdownLinkText() lipgloss.AdaptiveColor { return t.MarkdownLinkTextColor } func (t *BaseTheme) MarkdownCode() lipgloss.AdaptiveColor { return t.MarkdownCodeColor } func (t *BaseTheme) MarkdownBlockQuote() lipgloss.AdaptiveColor { return t.MarkdownBlockQuoteColor } func (t *BaseTheme) MarkdownEmph() lipgloss.AdaptiveColor { return t.MarkdownEmphColor } func (t *BaseTheme) MarkdownStrong() lipgloss.AdaptiveColor { return t.MarkdownStrongColor } func (t *BaseTheme) MarkdownHorizontalRule() lipgloss.AdaptiveColor { return t.MarkdownHorizontalRuleColor } func (t *BaseTheme) MarkdownListItem() lipgloss.AdaptiveColor { return t.MarkdownListItemColor } func (t *BaseTheme) MarkdownListEnumeration() lipgloss.AdaptiveColor { return t.MarkdownListEnumerationColor } func (t *BaseTheme) MarkdownImage() lipgloss.AdaptiveColor { return t.MarkdownImageColor } func (t *BaseTheme) MarkdownImageText() lipgloss.AdaptiveColor { return t.MarkdownImageTextColor } func (t *BaseTheme) MarkdownCodeBlock() lipgloss.AdaptiveColor { return t.MarkdownCodeBlockColor } func (t *BaseTheme) SyntaxComment() lipgloss.AdaptiveColor { return t.SyntaxCommentColor } func (t *BaseTheme) SyntaxKeyword() lipgloss.AdaptiveColor { return t.SyntaxKeywordColor } func (t *BaseTheme) SyntaxFunction() lipgloss.AdaptiveColor { return t.SyntaxFunctionColor } func (t *BaseTheme) SyntaxVariable() lipgloss.AdaptiveColor { return t.SyntaxVariableColor } func (t *BaseTheme) SyntaxString() lipgloss.AdaptiveColor { return t.SyntaxStringColor } func (t *BaseTheme) SyntaxNumber() lipgloss.AdaptiveColor { return t.SyntaxNumberColor } func (t *BaseTheme) SyntaxType() lipgloss.AdaptiveColor { return t.SyntaxTypeColor } func (t *BaseTheme) SyntaxOperator() lipgloss.AdaptiveColor { return t.SyntaxOperatorColor } func (t *BaseTheme) SyntaxPunctuation() lipgloss.AdaptiveColor { return t.SyntaxPunctuationColor } ================================================ FILE: internal/tui/theme/theme_test.go ================================================ package theme import ( "testing" ) func TestThemeRegistration(t *testing.T) { // Get list of available themes availableThemes := AvailableThemes() // Check if "catppuccin" theme is registered catppuccinFound := false for _, themeName := range availableThemes { if themeName == "catppuccin" { catppuccinFound = true break } } if !catppuccinFound { t.Errorf("Catppuccin theme is not registered") } // Check if "gruvbox" theme is registered gruvboxFound := false for _, themeName := range availableThemes { if themeName == "gruvbox" { gruvboxFound = true break } } if !gruvboxFound { t.Errorf("Gruvbox theme is not registered") } // Check if "monokai" theme is registered monokaiFound := false for _, themeName := range availableThemes { if themeName == "monokai" { monokaiFound = true break } } if !monokaiFound { t.Errorf("Monokai theme is not registered") } // Try to get the themes and make sure they're not nil catppuccin := GetTheme("catppuccin") if catppuccin == nil { t.Errorf("Catppuccin theme is nil") } gruvbox := GetTheme("gruvbox") if gruvbox == nil { t.Errorf("Gruvbox theme is nil") } monokai := GetTheme("monokai") if monokai == nil { t.Errorf("Monokai theme is nil") } // Test switching theme originalTheme := CurrentThemeName() err := SetTheme("gruvbox") if err != nil { t.Errorf("Failed to set theme to gruvbox: %v", err) } if CurrentThemeName() != "gruvbox" { t.Errorf("Theme not properly switched to gruvbox") } err = SetTheme("monokai") if err != nil { t.Errorf("Failed to set theme to monokai: %v", err) } if CurrentThemeName() != "monokai" { t.Errorf("Theme not properly switched to monokai") } // Switch back to original theme _ = SetTheme(originalTheme) } ================================================ FILE: internal/tui/theme/tokyonight.go ================================================ package theme import ( "github.com/charmbracelet/lipgloss" ) // TokyoNightTheme implements the Theme interface with Tokyo Night colors. // It provides both dark and light variants. type TokyoNightTheme struct { BaseTheme } // NewTokyoNightTheme creates a new instance of the Tokyo Night theme. func NewTokyoNightTheme() *TokyoNightTheme { // Tokyo Night color palette // Dark mode colors darkBackground := "#222436" darkCurrentLine := "#1e2030" darkSelection := "#2f334d" darkForeground := "#c8d3f5" darkComment := "#636da6" darkRed := "#ff757f" darkOrange := "#ff966c" darkYellow := "#ffc777" darkGreen := "#c3e88d" darkCyan := "#86e1fc" darkBlue := "#82aaff" darkPurple := "#c099ff" darkBorder := "#3b4261" // Light mode colors (Tokyo Night Day) lightBackground := "#e1e2e7" lightCurrentLine := "#d5d6db" lightSelection := "#c8c9ce" lightForeground := "#3760bf" lightComment := "#848cb5" lightRed := "#f52a65" lightOrange := "#b15c00" lightYellow := "#8c6c3e" lightGreen := "#587539" lightCyan := "#007197" lightBlue := "#2e7de9" lightPurple := "#9854f1" lightBorder := "#a8aecb" theme := &TokyoNightTheme{} // Base colors theme.PrimaryColor = lipgloss.AdaptiveColor{ Dark: darkBlue, Light: lightBlue, } theme.SecondaryColor = lipgloss.AdaptiveColor{ Dark: darkPurple, Light: lightPurple, } theme.AccentColor = lipgloss.AdaptiveColor{ Dark: darkOrange, Light: lightOrange, } // Status colors theme.ErrorColor = lipgloss.AdaptiveColor{ Dark: darkRed, Light: lightRed, } theme.WarningColor = lipgloss.AdaptiveColor{ Dark: darkOrange, Light: lightOrange, } theme.SuccessColor = lipgloss.AdaptiveColor{ Dark: darkGreen, Light: lightGreen, } theme.InfoColor = lipgloss.AdaptiveColor{ Dark: darkBlue, Light: lightBlue, } // Text colors theme.TextColor = lipgloss.AdaptiveColor{ Dark: darkForeground, Light: lightForeground, } theme.TextMutedColor = lipgloss.AdaptiveColor{ Dark: darkComment, Light: lightComment, } theme.TextEmphasizedColor = lipgloss.AdaptiveColor{ Dark: darkYellow, Light: lightYellow, } // Background colors theme.BackgroundColor = lipgloss.AdaptiveColor{ Dark: darkBackground, Light: lightBackground, } theme.BackgroundSecondaryColor = lipgloss.AdaptiveColor{ Dark: darkCurrentLine, Light: lightCurrentLine, } theme.BackgroundDarkerColor = lipgloss.AdaptiveColor{ Dark: "#191B29", // Darker background from palette Light: "#f0f0f5", // Slightly lighter than background } // Border colors theme.BorderNormalColor = lipgloss.AdaptiveColor{ Dark: darkBorder, Light: lightBorder, } theme.BorderFocusedColor = lipgloss.AdaptiveColor{ Dark: darkBlue, Light: lightBlue, } theme.BorderDimColor = lipgloss.AdaptiveColor{ Dark: darkSelection, Light: lightSelection, } // Diff view colors theme.DiffAddedColor = lipgloss.AdaptiveColor{ Dark: "#4fd6be", // teal from palette Light: "#1e725c", } theme.DiffRemovedColor = lipgloss.AdaptiveColor{ Dark: "#c53b53", // red1 from palette Light: "#c53b53", } theme.DiffContextColor = lipgloss.AdaptiveColor{ Dark: "#828bb8", // fg_dark from palette Light: "#7086b5", } theme.DiffHunkHeaderColor = lipgloss.AdaptiveColor{ Dark: "#828bb8", // fg_dark from palette Light: "#7086b5", } theme.DiffHighlightAddedColor = lipgloss.AdaptiveColor{ Dark: "#b8db87", // git.add from palette Light: "#4db380", } theme.DiffHighlightRemovedColor = lipgloss.AdaptiveColor{ Dark: "#e26a75", // git.delete from palette Light: "#f52a65", } theme.DiffAddedBgColor = lipgloss.AdaptiveColor{ Dark: "#20303b", Light: "#d5e5d5", } theme.DiffRemovedBgColor = lipgloss.AdaptiveColor{ Dark: "#37222c", Light: "#f7d8db", } theme.DiffContextBgColor = lipgloss.AdaptiveColor{ Dark: darkBackground, Light: lightBackground, } theme.DiffLineNumberColor = lipgloss.AdaptiveColor{ Dark: "#545c7e", // dark3 from palette Light: "#848cb5", } theme.DiffAddedLineNumberBgColor = lipgloss.AdaptiveColor{ Dark: "#1b2b34", Light: "#c5d5c5", } theme.DiffRemovedLineNumberBgColor = lipgloss.AdaptiveColor{ Dark: "#2d1f26", Light: "#e7c8cb", } // Markdown colors theme.MarkdownTextColor = lipgloss.AdaptiveColor{ Dark: darkForeground, Light: lightForeground, } theme.MarkdownHeadingColor = lipgloss.AdaptiveColor{ Dark: darkPurple, Light: lightPurple, } theme.MarkdownLinkColor = lipgloss.AdaptiveColor{ Dark: darkBlue, Light: lightBlue, } theme.MarkdownLinkTextColor = lipgloss.AdaptiveColor{ Dark: darkCyan, Light: lightCyan, } theme.MarkdownCodeColor = lipgloss.AdaptiveColor{ Dark: darkGreen, Light: lightGreen, } theme.MarkdownBlockQuoteColor = lipgloss.AdaptiveColor{ Dark: darkYellow, Light: lightYellow, } theme.MarkdownEmphColor = lipgloss.AdaptiveColor{ Dark: darkYellow, Light: lightYellow, } theme.MarkdownStrongColor = lipgloss.AdaptiveColor{ Dark: darkOrange, Light: lightOrange, } theme.MarkdownHorizontalRuleColor = lipgloss.AdaptiveColor{ Dark: darkComment, Light: lightComment, } theme.MarkdownListItemColor = lipgloss.AdaptiveColor{ Dark: darkBlue, Light: lightBlue, } theme.MarkdownListEnumerationColor = lipgloss.AdaptiveColor{ Dark: darkCyan, Light: lightCyan, } theme.MarkdownImageColor = lipgloss.AdaptiveColor{ Dark: darkBlue, Light: lightBlue, } theme.MarkdownImageTextColor = lipgloss.AdaptiveColor{ Dark: darkCyan, Light: lightCyan, } theme.MarkdownCodeBlockColor = lipgloss.AdaptiveColor{ Dark: darkForeground, Light: lightForeground, } // Syntax highlighting colors theme.SyntaxCommentColor = lipgloss.AdaptiveColor{ Dark: darkComment, Light: lightComment, } theme.SyntaxKeywordColor = lipgloss.AdaptiveColor{ Dark: darkPurple, Light: lightPurple, } theme.SyntaxFunctionColor = lipgloss.AdaptiveColor{ Dark: darkBlue, Light: lightBlue, } theme.SyntaxVariableColor = lipgloss.AdaptiveColor{ Dark: darkRed, Light: lightRed, } theme.SyntaxStringColor = lipgloss.AdaptiveColor{ Dark: darkGreen, Light: lightGreen, } theme.SyntaxNumberColor = lipgloss.AdaptiveColor{ Dark: darkOrange, Light: lightOrange, } theme.SyntaxTypeColor = lipgloss.AdaptiveColor{ Dark: darkYellow, Light: lightYellow, } theme.SyntaxOperatorColor = lipgloss.AdaptiveColor{ Dark: darkCyan, Light: lightCyan, } theme.SyntaxPunctuationColor = lipgloss.AdaptiveColor{ Dark: darkForeground, Light: lightForeground, } return theme } func init() { // Register the Tokyo Night theme with the theme manager RegisterTheme("tokyonight", NewTokyoNightTheme()) } ================================================ FILE: internal/tui/theme/tron.go ================================================ package theme import ( "github.com/charmbracelet/lipgloss" ) // TronTheme implements the Theme interface with Tron-inspired colors. // It provides both dark and light variants, though Tron is primarily a dark theme. type TronTheme struct { BaseTheme } // NewTronTheme creates a new instance of the Tron theme. func NewTronTheme() *TronTheme { // Tron color palette // Inspired by the Tron movie's neon aesthetic darkBackground := "#0c141f" darkCurrentLine := "#1a2633" darkSelection := "#1a2633" darkForeground := "#caf0ff" darkComment := "#4d6b87" darkCyan := "#00d9ff" darkBlue := "#007fff" darkOrange := "#ff9000" darkPink := "#ff00a0" darkPurple := "#b73fff" darkRed := "#ff3333" darkYellow := "#ffcc00" darkGreen := "#00ff8f" darkBorder := "#1a2633" // Light mode approximation lightBackground := "#f0f8ff" lightCurrentLine := "#e0f0ff" lightSelection := "#d0e8ff" lightForeground := "#0c141f" lightComment := "#4d6b87" lightCyan := "#0097b3" lightBlue := "#0066cc" lightOrange := "#cc7300" lightPink := "#cc0080" lightPurple := "#9932cc" lightRed := "#cc2929" lightYellow := "#cc9900" lightGreen := "#00cc72" lightBorder := "#d0e8ff" theme := &TronTheme{} // Base colors theme.PrimaryColor = lipgloss.AdaptiveColor{ Dark: darkCyan, Light: lightCyan, } theme.SecondaryColor = lipgloss.AdaptiveColor{ Dark: darkBlue, Light: lightBlue, } theme.AccentColor = lipgloss.AdaptiveColor{ Dark: darkOrange, Light: lightOrange, } // Status colors theme.ErrorColor = lipgloss.AdaptiveColor{ Dark: darkRed, Light: lightRed, } theme.WarningColor = lipgloss.AdaptiveColor{ Dark: darkOrange, Light: lightOrange, } theme.SuccessColor = lipgloss.AdaptiveColor{ Dark: darkGreen, Light: lightGreen, } theme.InfoColor = lipgloss.AdaptiveColor{ Dark: darkCyan, Light: lightCyan, } // Text colors theme.TextColor = lipgloss.AdaptiveColor{ Dark: darkForeground, Light: lightForeground, } theme.TextMutedColor = lipgloss.AdaptiveColor{ Dark: darkComment, Light: lightComment, } theme.TextEmphasizedColor = lipgloss.AdaptiveColor{ Dark: darkYellow, Light: lightYellow, } // Background colors theme.BackgroundColor = lipgloss.AdaptiveColor{ Dark: darkBackground, Light: lightBackground, } theme.BackgroundSecondaryColor = lipgloss.AdaptiveColor{ Dark: darkCurrentLine, Light: lightCurrentLine, } theme.BackgroundDarkerColor = lipgloss.AdaptiveColor{ Dark: "#070d14", // Slightly darker than background Light: "#ffffff", // Slightly lighter than background } // Border colors theme.BorderNormalColor = lipgloss.AdaptiveColor{ Dark: darkBorder, Light: lightBorder, } theme.BorderFocusedColor = lipgloss.AdaptiveColor{ Dark: darkCyan, Light: lightCyan, } theme.BorderDimColor = lipgloss.AdaptiveColor{ Dark: darkSelection, Light: lightSelection, } // Diff view colors theme.DiffAddedColor = lipgloss.AdaptiveColor{ Dark: darkGreen, Light: lightGreen, } theme.DiffRemovedColor = lipgloss.AdaptiveColor{ Dark: darkRed, Light: lightRed, } theme.DiffContextColor = lipgloss.AdaptiveColor{ Dark: darkComment, Light: lightComment, } theme.DiffHunkHeaderColor = lipgloss.AdaptiveColor{ Dark: darkBlue, Light: lightBlue, } theme.DiffHighlightAddedColor = lipgloss.AdaptiveColor{ Dark: "#00ff8f", Light: "#a5d6a7", } theme.DiffHighlightRemovedColor = lipgloss.AdaptiveColor{ Dark: "#ff3333", Light: "#ef9a9a", } theme.DiffAddedBgColor = lipgloss.AdaptiveColor{ Dark: "#0a2a1a", Light: "#e8f5e9", } theme.DiffRemovedBgColor = lipgloss.AdaptiveColor{ Dark: "#2a0a0a", Light: "#ffebee", } theme.DiffContextBgColor = lipgloss.AdaptiveColor{ Dark: darkBackground, Light: lightBackground, } theme.DiffLineNumberColor = lipgloss.AdaptiveColor{ Dark: darkComment, Light: lightComment, } theme.DiffAddedLineNumberBgColor = lipgloss.AdaptiveColor{ Dark: "#082015", Light: "#c8e6c9", } theme.DiffRemovedLineNumberBgColor = lipgloss.AdaptiveColor{ Dark: "#200808", Light: "#ffcdd2", } // Markdown colors theme.MarkdownTextColor = lipgloss.AdaptiveColor{ Dark: darkForeground, Light: lightForeground, } theme.MarkdownHeadingColor = lipgloss.AdaptiveColor{ Dark: darkCyan, Light: lightCyan, } theme.MarkdownLinkColor = lipgloss.AdaptiveColor{ Dark: darkBlue, Light: lightBlue, } theme.MarkdownLinkTextColor = lipgloss.AdaptiveColor{ Dark: darkCyan, Light: lightCyan, } theme.MarkdownCodeColor = lipgloss.AdaptiveColor{ Dark: darkGreen, Light: lightGreen, } theme.MarkdownBlockQuoteColor = lipgloss.AdaptiveColor{ Dark: darkYellow, Light: lightYellow, } theme.MarkdownEmphColor = lipgloss.AdaptiveColor{ Dark: darkYellow, Light: lightYellow, } theme.MarkdownStrongColor = lipgloss.AdaptiveColor{ Dark: darkOrange, Light: lightOrange, } theme.MarkdownHorizontalRuleColor = lipgloss.AdaptiveColor{ Dark: darkComment, Light: lightComment, } theme.MarkdownListItemColor = lipgloss.AdaptiveColor{ Dark: darkBlue, Light: lightBlue, } theme.MarkdownListEnumerationColor = lipgloss.AdaptiveColor{ Dark: darkCyan, Light: lightCyan, } theme.MarkdownImageColor = lipgloss.AdaptiveColor{ Dark: darkBlue, Light: lightBlue, } theme.MarkdownImageTextColor = lipgloss.AdaptiveColor{ Dark: darkCyan, Light: lightCyan, } theme.MarkdownCodeBlockColor = lipgloss.AdaptiveColor{ Dark: darkForeground, Light: lightForeground, } // Syntax highlighting colors theme.SyntaxCommentColor = lipgloss.AdaptiveColor{ Dark: darkComment, Light: lightComment, } theme.SyntaxKeywordColor = lipgloss.AdaptiveColor{ Dark: darkCyan, Light: lightCyan, } theme.SyntaxFunctionColor = lipgloss.AdaptiveColor{ Dark: darkGreen, Light: lightGreen, } theme.SyntaxVariableColor = lipgloss.AdaptiveColor{ Dark: darkOrange, Light: lightOrange, } theme.SyntaxStringColor = lipgloss.AdaptiveColor{ Dark: darkYellow, Light: lightYellow, } theme.SyntaxNumberColor = lipgloss.AdaptiveColor{ Dark: darkBlue, Light: lightBlue, } theme.SyntaxTypeColor = lipgloss.AdaptiveColor{ Dark: darkPurple, Light: lightPurple, } theme.SyntaxOperatorColor = lipgloss.AdaptiveColor{ Dark: darkPink, Light: lightPink, } theme.SyntaxPunctuationColor = lipgloss.AdaptiveColor{ Dark: darkForeground, Light: lightForeground, } return theme } func init() { // Register the Tron theme with the theme manager RegisterTheme("tron", NewTronTheme()) } ================================================ FILE: internal/tui/tui.go ================================================ package tui import ( "context" "fmt" "strings" "github.com/charmbracelet/bubbles/key" tea "github.com/charmbracelet/bubbletea" "github.com/charmbracelet/lipgloss" "github.com/opencode-ai/opencode/internal/app" "github.com/opencode-ai/opencode/internal/config" "github.com/opencode-ai/opencode/internal/llm/agent" "github.com/opencode-ai/opencode/internal/logging" "github.com/opencode-ai/opencode/internal/permission" "github.com/opencode-ai/opencode/internal/pubsub" "github.com/opencode-ai/opencode/internal/session" "github.com/opencode-ai/opencode/internal/tui/components/chat" "github.com/opencode-ai/opencode/internal/tui/components/core" "github.com/opencode-ai/opencode/internal/tui/components/dialog" "github.com/opencode-ai/opencode/internal/tui/layout" "github.com/opencode-ai/opencode/internal/tui/page" "github.com/opencode-ai/opencode/internal/tui/theme" "github.com/opencode-ai/opencode/internal/tui/util" ) type keyMap struct { Logs key.Binding Quit key.Binding Help key.Binding SwitchSession key.Binding Commands key.Binding Filepicker key.Binding Models key.Binding SwitchTheme key.Binding } type startCompactSessionMsg struct{} const ( quitKey = "q" ) var keys = keyMap{ Logs: key.NewBinding( key.WithKeys("ctrl+l"), key.WithHelp("ctrl+l", "logs"), ), Quit: key.NewBinding( key.WithKeys("ctrl+c"), key.WithHelp("ctrl+c", "quit"), ), Help: key.NewBinding( key.WithKeys("ctrl+_", "ctrl+h"), key.WithHelp("ctrl+?", "toggle help"), ), SwitchSession: key.NewBinding( key.WithKeys("ctrl+s"), key.WithHelp("ctrl+s", "switch session"), ), Commands: key.NewBinding( key.WithKeys("ctrl+k"), key.WithHelp("ctrl+k", "commands"), ), Filepicker: key.NewBinding( key.WithKeys("ctrl+f"), key.WithHelp("ctrl+f", "select files to upload"), ), Models: key.NewBinding( key.WithKeys("ctrl+o"), key.WithHelp("ctrl+o", "model selection"), ), SwitchTheme: key.NewBinding( key.WithKeys("ctrl+t"), key.WithHelp("ctrl+t", "switch theme"), ), } var helpEsc = key.NewBinding( key.WithKeys("?"), key.WithHelp("?", "toggle help"), ) var returnKey = key.NewBinding( key.WithKeys("esc"), key.WithHelp("esc", "close"), ) var logsKeyReturnKey = key.NewBinding( key.WithKeys("esc", "backspace", quitKey), key.WithHelp("esc/q", "go back"), ) type appModel struct { width, height int currentPage page.PageID previousPage page.PageID pages map[page.PageID]tea.Model loadedPages map[page.PageID]bool status core.StatusCmp app *app.App selectedSession session.Session showPermissions bool permissions dialog.PermissionDialogCmp showHelp bool help dialog.HelpCmp showQuit bool quit dialog.QuitDialog showSessionDialog bool sessionDialog dialog.SessionDialog showCommandDialog bool commandDialog dialog.CommandDialog commands []dialog.Command showModelDialog bool modelDialog dialog.ModelDialog showInitDialog bool initDialog dialog.InitDialogCmp showFilepicker bool filepicker dialog.FilepickerCmp showThemeDialog bool themeDialog dialog.ThemeDialog showMultiArgumentsDialog bool multiArgumentsDialog dialog.MultiArgumentsDialogCmp isCompacting bool compactingMessage string } func (a appModel) Init() tea.Cmd { var cmds []tea.Cmd cmd := a.pages[a.currentPage].Init() a.loadedPages[a.currentPage] = true cmds = append(cmds, cmd) cmd = a.status.Init() cmds = append(cmds, cmd) cmd = a.quit.Init() cmds = append(cmds, cmd) cmd = a.help.Init() cmds = append(cmds, cmd) cmd = a.sessionDialog.Init() cmds = append(cmds, cmd) cmd = a.commandDialog.Init() cmds = append(cmds, cmd) cmd = a.modelDialog.Init() cmds = append(cmds, cmd) cmd = a.initDialog.Init() cmds = append(cmds, cmd) cmd = a.filepicker.Init() cmds = append(cmds, cmd) cmd = a.themeDialog.Init() cmds = append(cmds, cmd) // Check if we should show the init dialog cmds = append(cmds, func() tea.Msg { shouldShow, err := config.ShouldShowInitDialog() if err != nil { return util.InfoMsg{ Type: util.InfoTypeError, Msg: "Failed to check init status: " + err.Error(), } } return dialog.ShowInitDialogMsg{Show: shouldShow} }) return tea.Batch(cmds...) } func (a appModel) Update(msg tea.Msg) (tea.Model, tea.Cmd) { var cmds []tea.Cmd var cmd tea.Cmd switch msg := msg.(type) { case tea.WindowSizeMsg: msg.Height -= 1 // Make space for the status bar a.width, a.height = msg.Width, msg.Height s, _ := a.status.Update(msg) a.status = s.(core.StatusCmp) a.pages[a.currentPage], cmd = a.pages[a.currentPage].Update(msg) cmds = append(cmds, cmd) prm, permCmd := a.permissions.Update(msg) a.permissions = prm.(dialog.PermissionDialogCmp) cmds = append(cmds, permCmd) help, helpCmd := a.help.Update(msg) a.help = help.(dialog.HelpCmp) cmds = append(cmds, helpCmd) session, sessionCmd := a.sessionDialog.Update(msg) a.sessionDialog = session.(dialog.SessionDialog) cmds = append(cmds, sessionCmd) command, commandCmd := a.commandDialog.Update(msg) a.commandDialog = command.(dialog.CommandDialog) cmds = append(cmds, commandCmd) filepicker, filepickerCmd := a.filepicker.Update(msg) a.filepicker = filepicker.(dialog.FilepickerCmp) cmds = append(cmds, filepickerCmd) a.initDialog.SetSize(msg.Width, msg.Height) if a.showMultiArgumentsDialog { a.multiArgumentsDialog.SetSize(msg.Width, msg.Height) args, argsCmd := a.multiArgumentsDialog.Update(msg) a.multiArgumentsDialog = args.(dialog.MultiArgumentsDialogCmp) cmds = append(cmds, argsCmd, a.multiArgumentsDialog.Init()) } return a, tea.Batch(cmds...) // Status case util.InfoMsg: s, cmd := a.status.Update(msg) a.status = s.(core.StatusCmp) cmds = append(cmds, cmd) return a, tea.Batch(cmds...) case pubsub.Event[logging.LogMessage]: if msg.Payload.Persist { switch msg.Payload.Level { case "error": s, cmd := a.status.Update(util.InfoMsg{ Type: util.InfoTypeError, Msg: msg.Payload.Message, TTL: msg.Payload.PersistTime, }) a.status = s.(core.StatusCmp) cmds = append(cmds, cmd) case "info": s, cmd := a.status.Update(util.InfoMsg{ Type: util.InfoTypeInfo, Msg: msg.Payload.Message, TTL: msg.Payload.PersistTime, }) a.status = s.(core.StatusCmp) cmds = append(cmds, cmd) case "warn": s, cmd := a.status.Update(util.InfoMsg{ Type: util.InfoTypeWarn, Msg: msg.Payload.Message, TTL: msg.Payload.PersistTime, }) a.status = s.(core.StatusCmp) cmds = append(cmds, cmd) default: s, cmd := a.status.Update(util.InfoMsg{ Type: util.InfoTypeInfo, Msg: msg.Payload.Message, TTL: msg.Payload.PersistTime, }) a.status = s.(core.StatusCmp) cmds = append(cmds, cmd) } } case util.ClearStatusMsg: s, _ := a.status.Update(msg) a.status = s.(core.StatusCmp) // Permission case pubsub.Event[permission.PermissionRequest]: a.showPermissions = true return a, a.permissions.SetPermissions(msg.Payload) case dialog.PermissionResponseMsg: var cmd tea.Cmd switch msg.Action { case dialog.PermissionAllow: a.app.Permissions.Grant(msg.Permission) case dialog.PermissionAllowForSession: a.app.Permissions.GrantPersistant(msg.Permission) case dialog.PermissionDeny: a.app.Permissions.Deny(msg.Permission) } a.showPermissions = false return a, cmd case page.PageChangeMsg: return a, a.moveToPage(msg.ID) case dialog.CloseQuitMsg: a.showQuit = false return a, nil case dialog.CloseSessionDialogMsg: a.showSessionDialog = false return a, nil case dialog.CloseCommandDialogMsg: a.showCommandDialog = false return a, nil case startCompactSessionMsg: // Start compacting the current session a.isCompacting = true a.compactingMessage = "Starting summarization..." if a.selectedSession.ID == "" { a.isCompacting = false return a, util.ReportWarn("No active session to summarize") } // Start the summarization process return a, func() tea.Msg { ctx := context.Background() a.app.CoderAgent.Summarize(ctx, a.selectedSession.ID) return nil } case pubsub.Event[agent.AgentEvent]: payload := msg.Payload if payload.Error != nil { a.isCompacting = false return a, util.ReportError(payload.Error) } a.compactingMessage = payload.Progress if payload.Done && payload.Type == agent.AgentEventTypeSummarize { a.isCompacting = false return a, util.ReportInfo("Session summarization complete") } else if payload.Done && payload.Type == agent.AgentEventTypeResponse && a.selectedSession.ID != "" { model := a.app.CoderAgent.Model() contextWindow := model.ContextWindow tokens := a.selectedSession.CompletionTokens + a.selectedSession.PromptTokens if (tokens >= int64(float64(contextWindow)*0.95)) && config.Get().AutoCompact { return a, util.CmdHandler(startCompactSessionMsg{}) } } // Continue listening for events return a, nil case dialog.CloseThemeDialogMsg: a.showThemeDialog = false return a, nil case dialog.ThemeChangedMsg: a.pages[a.currentPage], cmd = a.pages[a.currentPage].Update(msg) a.showThemeDialog = false return a, tea.Batch(cmd, util.ReportInfo("Theme changed to: "+msg.ThemeName)) case dialog.CloseModelDialogMsg: a.showModelDialog = false return a, nil case dialog.ModelSelectedMsg: a.showModelDialog = false model, err := a.app.CoderAgent.Update(config.AgentCoder, msg.Model.ID) if err != nil { return a, util.ReportError(err) } return a, util.ReportInfo(fmt.Sprintf("Model changed to %s", model.Name)) case dialog.ShowInitDialogMsg: a.showInitDialog = msg.Show return a, nil case dialog.CloseInitDialogMsg: a.showInitDialog = false if msg.Initialize { // Run the initialization command for _, cmd := range a.commands { if cmd.ID == "init" { // Mark the project as initialized if err := config.MarkProjectInitialized(); err != nil { return a, util.ReportError(err) } return a, cmd.Handler(cmd) } } } else { // Mark the project as initialized without running the command if err := config.MarkProjectInitialized(); err != nil { return a, util.ReportError(err) } } return a, nil case chat.SessionSelectedMsg: a.selectedSession = msg a.sessionDialog.SetSelectedSession(msg.ID) case pubsub.Event[session.Session]: if msg.Type == pubsub.UpdatedEvent && msg.Payload.ID == a.selectedSession.ID { a.selectedSession = msg.Payload } case dialog.SessionSelectedMsg: a.showSessionDialog = false if a.currentPage == page.ChatPage { return a, util.CmdHandler(chat.SessionSelectedMsg(msg.Session)) } return a, nil case dialog.CommandSelectedMsg: a.showCommandDialog = false // Execute the command handler if available if msg.Command.Handler != nil { return a, msg.Command.Handler(msg.Command) } return a, util.ReportInfo("Command selected: " + msg.Command.Title) case dialog.ShowMultiArgumentsDialogMsg: // Show multi-arguments dialog a.multiArgumentsDialog = dialog.NewMultiArgumentsDialogCmp(msg.CommandID, msg.Content, msg.ArgNames) a.showMultiArgumentsDialog = true return a, a.multiArgumentsDialog.Init() case dialog.CloseMultiArgumentsDialogMsg: // Close multi-arguments dialog a.showMultiArgumentsDialog = false // If submitted, replace all named arguments and run the command if msg.Submit { content := msg.Content // Replace each named argument with its value for name, value := range msg.Args { placeholder := "$" + name content = strings.ReplaceAll(content, placeholder, value) } // Execute the command with arguments return a, util.CmdHandler(dialog.CommandRunCustomMsg{ Content: content, Args: msg.Args, }) } return a, nil case tea.KeyMsg: // If multi-arguments dialog is open, let it handle the key press first if a.showMultiArgumentsDialog { args, cmd := a.multiArgumentsDialog.Update(msg) a.multiArgumentsDialog = args.(dialog.MultiArgumentsDialogCmp) return a, cmd } switch { case key.Matches(msg, keys.Quit): a.showQuit = !a.showQuit if a.showHelp { a.showHelp = false } if a.showSessionDialog { a.showSessionDialog = false } if a.showCommandDialog { a.showCommandDialog = false } if a.showFilepicker { a.showFilepicker = false a.filepicker.ToggleFilepicker(a.showFilepicker) } if a.showModelDialog { a.showModelDialog = false } if a.showMultiArgumentsDialog { a.showMultiArgumentsDialog = false } return a, nil case key.Matches(msg, keys.SwitchSession): if a.currentPage == page.ChatPage && !a.showQuit && !a.showPermissions && !a.showCommandDialog { // Load sessions and show the dialog sessions, err := a.app.Sessions.List(context.Background()) if err != nil { return a, util.ReportError(err) } if len(sessions) == 0 { return a, util.ReportWarn("No sessions available") } a.sessionDialog.SetSessions(sessions) a.showSessionDialog = true return a, nil } return a, nil case key.Matches(msg, keys.Commands): if a.currentPage == page.ChatPage && !a.showQuit && !a.showPermissions && !a.showSessionDialog && !a.showThemeDialog && !a.showFilepicker { // Show commands dialog if len(a.commands) == 0 { return a, util.ReportWarn("No commands available") } a.commandDialog.SetCommands(a.commands) a.showCommandDialog = true return a, nil } return a, nil case key.Matches(msg, keys.Models): if a.showModelDialog { a.showModelDialog = false return a, nil } if a.currentPage == page.ChatPage && !a.showQuit && !a.showPermissions && !a.showSessionDialog && !a.showCommandDialog { a.showModelDialog = true return a, nil } return a, nil case key.Matches(msg, keys.SwitchTheme): if !a.showQuit && !a.showPermissions && !a.showSessionDialog && !a.showCommandDialog { // Show theme switcher dialog a.showThemeDialog = true // Theme list is dynamically loaded by the dialog component return a, a.themeDialog.Init() } return a, nil case key.Matches(msg, returnKey) || key.Matches(msg): if msg.String() == quitKey { if a.currentPage == page.LogsPage { return a, a.moveToPage(page.ChatPage) } } else if !a.filepicker.IsCWDFocused() { if a.showQuit { a.showQuit = !a.showQuit return a, nil } if a.showHelp { a.showHelp = !a.showHelp return a, nil } if a.showInitDialog { a.showInitDialog = false // Mark the project as initialized without running the command if err := config.MarkProjectInitialized(); err != nil { return a, util.ReportError(err) } return a, nil } if a.showFilepicker { a.showFilepicker = false a.filepicker.ToggleFilepicker(a.showFilepicker) return a, nil } if a.currentPage == page.LogsPage { return a, a.moveToPage(page.ChatPage) } } case key.Matches(msg, keys.Logs): return a, a.moveToPage(page.LogsPage) case key.Matches(msg, keys.Help): if a.showQuit { return a, nil } a.showHelp = !a.showHelp return a, nil case key.Matches(msg, helpEsc): if a.app.CoderAgent.IsBusy() { if a.showQuit { return a, nil } a.showHelp = !a.showHelp return a, nil } case key.Matches(msg, keys.Filepicker): a.showFilepicker = !a.showFilepicker a.filepicker.ToggleFilepicker(a.showFilepicker) return a, nil } default: f, filepickerCmd := a.filepicker.Update(msg) a.filepicker = f.(dialog.FilepickerCmp) cmds = append(cmds, filepickerCmd) } if a.showFilepicker { f, filepickerCmd := a.filepicker.Update(msg) a.filepicker = f.(dialog.FilepickerCmp) cmds = append(cmds, filepickerCmd) // Only block key messages send all other messages down if _, ok := msg.(tea.KeyMsg); ok { return a, tea.Batch(cmds...) } } if a.showQuit { q, quitCmd := a.quit.Update(msg) a.quit = q.(dialog.QuitDialog) cmds = append(cmds, quitCmd) // Only block key messages send all other messages down if _, ok := msg.(tea.KeyMsg); ok { return a, tea.Batch(cmds...) } } if a.showPermissions { d, permissionsCmd := a.permissions.Update(msg) a.permissions = d.(dialog.PermissionDialogCmp) cmds = append(cmds, permissionsCmd) // Only block key messages send all other messages down if _, ok := msg.(tea.KeyMsg); ok { return a, tea.Batch(cmds...) } } if a.showSessionDialog { d, sessionCmd := a.sessionDialog.Update(msg) a.sessionDialog = d.(dialog.SessionDialog) cmds = append(cmds, sessionCmd) // Only block key messages send all other messages down if _, ok := msg.(tea.KeyMsg); ok { return a, tea.Batch(cmds...) } } if a.showCommandDialog { d, commandCmd := a.commandDialog.Update(msg) a.commandDialog = d.(dialog.CommandDialog) cmds = append(cmds, commandCmd) // Only block key messages send all other messages down if _, ok := msg.(tea.KeyMsg); ok { return a, tea.Batch(cmds...) } } if a.showModelDialog { d, modelCmd := a.modelDialog.Update(msg) a.modelDialog = d.(dialog.ModelDialog) cmds = append(cmds, modelCmd) // Only block key messages send all other messages down if _, ok := msg.(tea.KeyMsg); ok { return a, tea.Batch(cmds...) } } if a.showInitDialog { d, initCmd := a.initDialog.Update(msg) a.initDialog = d.(dialog.InitDialogCmp) cmds = append(cmds, initCmd) // Only block key messages send all other messages down if _, ok := msg.(tea.KeyMsg); ok { return a, tea.Batch(cmds...) } } if a.showThemeDialog { d, themeCmd := a.themeDialog.Update(msg) a.themeDialog = d.(dialog.ThemeDialog) cmds = append(cmds, themeCmd) // Only block key messages send all other messages down if _, ok := msg.(tea.KeyMsg); ok { return a, tea.Batch(cmds...) } } s, _ := a.status.Update(msg) a.status = s.(core.StatusCmp) a.pages[a.currentPage], cmd = a.pages[a.currentPage].Update(msg) cmds = append(cmds, cmd) return a, tea.Batch(cmds...) } // RegisterCommand adds a command to the command dialog func (a *appModel) RegisterCommand(cmd dialog.Command) { a.commands = append(a.commands, cmd) } func (a *appModel) findCommand(id string) (dialog.Command, bool) { for _, cmd := range a.commands { if cmd.ID == id { return cmd, true } } return dialog.Command{}, false } func (a *appModel) moveToPage(pageID page.PageID) tea.Cmd { if a.app.CoderAgent.IsBusy() { // For now we don't move to any page if the agent is busy return util.ReportWarn("Agent is busy, please wait...") } var cmds []tea.Cmd if _, ok := a.loadedPages[pageID]; !ok { cmd := a.pages[pageID].Init() cmds = append(cmds, cmd) a.loadedPages[pageID] = true } a.previousPage = a.currentPage a.currentPage = pageID if sizable, ok := a.pages[a.currentPage].(layout.Sizeable); ok { cmd := sizable.SetSize(a.width, a.height) cmds = append(cmds, cmd) } return tea.Batch(cmds...) } func (a appModel) View() string { components := []string{ a.pages[a.currentPage].View(), } components = append(components, a.status.View()) appView := lipgloss.JoinVertical(lipgloss.Top, components...) if a.showPermissions { overlay := a.permissions.View() row := lipgloss.Height(appView) / 2 row -= lipgloss.Height(overlay) / 2 col := lipgloss.Width(appView) / 2 col -= lipgloss.Width(overlay) / 2 appView = layout.PlaceOverlay( col, row, overlay, appView, true, ) } if a.showFilepicker { overlay := a.filepicker.View() row := lipgloss.Height(appView) / 2 row -= lipgloss.Height(overlay) / 2 col := lipgloss.Width(appView) / 2 col -= lipgloss.Width(overlay) / 2 appView = layout.PlaceOverlay( col, row, overlay, appView, true, ) } // Show compacting status overlay if a.isCompacting { t := theme.CurrentTheme() style := lipgloss.NewStyle(). Border(lipgloss.RoundedBorder()). BorderForeground(t.BorderFocused()). BorderBackground(t.Background()). Padding(1, 2). Background(t.Background()). Foreground(t.Text()) overlay := style.Render("Summarizing\n" + a.compactingMessage) row := lipgloss.Height(appView) / 2 row -= lipgloss.Height(overlay) / 2 col := lipgloss.Width(appView) / 2 col -= lipgloss.Width(overlay) / 2 appView = layout.PlaceOverlay( col, row, overlay, appView, true, ) } if a.showHelp { bindings := layout.KeyMapToSlice(keys) if p, ok := a.pages[a.currentPage].(layout.Bindings); ok { bindings = append(bindings, p.BindingKeys()...) } if a.showPermissions { bindings = append(bindings, a.permissions.BindingKeys()...) } if a.currentPage == page.LogsPage { bindings = append(bindings, logsKeyReturnKey) } if !a.app.CoderAgent.IsBusy() { bindings = append(bindings, helpEsc) } a.help.SetBindings(bindings) overlay := a.help.View() row := lipgloss.Height(appView) / 2 row -= lipgloss.Height(overlay) / 2 col := lipgloss.Width(appView) / 2 col -= lipgloss.Width(overlay) / 2 appView = layout.PlaceOverlay( col, row, overlay, appView, true, ) } if a.showQuit { overlay := a.quit.View() row := lipgloss.Height(appView) / 2 row -= lipgloss.Height(overlay) / 2 col := lipgloss.Width(appView) / 2 col -= lipgloss.Width(overlay) / 2 appView = layout.PlaceOverlay( col, row, overlay, appView, true, ) } if a.showSessionDialog { overlay := a.sessionDialog.View() row := lipgloss.Height(appView) / 2 row -= lipgloss.Height(overlay) / 2 col := lipgloss.Width(appView) / 2 col -= lipgloss.Width(overlay) / 2 appView = layout.PlaceOverlay( col, row, overlay, appView, true, ) } if a.showModelDialog { overlay := a.modelDialog.View() row := lipgloss.Height(appView) / 2 row -= lipgloss.Height(overlay) / 2 col := lipgloss.Width(appView) / 2 col -= lipgloss.Width(overlay) / 2 appView = layout.PlaceOverlay( col, row, overlay, appView, true, ) } if a.showCommandDialog { overlay := a.commandDialog.View() row := lipgloss.Height(appView) / 2 row -= lipgloss.Height(overlay) / 2 col := lipgloss.Width(appView) / 2 col -= lipgloss.Width(overlay) / 2 appView = layout.PlaceOverlay( col, row, overlay, appView, true, ) } if a.showInitDialog { overlay := a.initDialog.View() appView = layout.PlaceOverlay( a.width/2-lipgloss.Width(overlay)/2, a.height/2-lipgloss.Height(overlay)/2, overlay, appView, true, ) } if a.showThemeDialog { overlay := a.themeDialog.View() row := lipgloss.Height(appView) / 2 row -= lipgloss.Height(overlay) / 2 col := lipgloss.Width(appView) / 2 col -= lipgloss.Width(overlay) / 2 appView = layout.PlaceOverlay( col, row, overlay, appView, true, ) } if a.showMultiArgumentsDialog { overlay := a.multiArgumentsDialog.View() row := lipgloss.Height(appView) / 2 row -= lipgloss.Height(overlay) / 2 col := lipgloss.Width(appView) / 2 col -= lipgloss.Width(overlay) / 2 appView = layout.PlaceOverlay( col, row, overlay, appView, true, ) } return appView } func New(app *app.App) tea.Model { startPage := page.ChatPage model := &appModel{ currentPage: startPage, loadedPages: make(map[page.PageID]bool), status: core.NewStatusCmp(app.LSPClients), help: dialog.NewHelpCmp(), quit: dialog.NewQuitCmp(), sessionDialog: dialog.NewSessionDialogCmp(), commandDialog: dialog.NewCommandDialogCmp(), modelDialog: dialog.NewModelDialogCmp(), permissions: dialog.NewPermissionDialogCmp(), initDialog: dialog.NewInitDialogCmp(), themeDialog: dialog.NewThemeDialogCmp(), app: app, commands: []dialog.Command{}, pages: map[page.PageID]tea.Model{ page.ChatPage: page.NewChatPage(app), page.LogsPage: page.NewLogsPage(), }, filepicker: dialog.NewFilepickerCmp(app), } model.RegisterCommand(dialog.Command{ ID: "init", Title: "Initialize Project", Description: "Create/Update the OpenCode.md memory file", Handler: func(cmd dialog.Command) tea.Cmd { prompt := `Please analyze this codebase and create a OpenCode.md file containing: 1. Build/lint/test commands - especially for running a single test 2. Code style guidelines including imports, formatting, types, naming conventions, error handling, etc. The file you create will be given to agentic coding agents (such as yourself) that operate in this repository. Make it about 20 lines long. If there's already a opencode.md, improve it. If there are Cursor rules (in .cursor/rules/ or .cursorrules) or Copilot rules (in .github/copilot-instructions.md), make sure to include them.` return tea.Batch( util.CmdHandler(chat.SendMsg{ Text: prompt, }), ) }, }) model.RegisterCommand(dialog.Command{ ID: "compact", Title: "Compact Session", Description: "Summarize the current session and create a new one with the summary", Handler: func(cmd dialog.Command) tea.Cmd { return func() tea.Msg { return startCompactSessionMsg{} } }, }) // Load custom commands customCommands, err := dialog.LoadCustomCommands() if err != nil { logging.Warn("Failed to load custom commands", "error", err) } else { for _, cmd := range customCommands { model.RegisterCommand(cmd) } } return model } ================================================ FILE: internal/tui/util/util.go ================================================ package util import ( "time" tea "github.com/charmbracelet/bubbletea" ) func CmdHandler(msg tea.Msg) tea.Cmd { return func() tea.Msg { return msg } } func ReportError(err error) tea.Cmd { return CmdHandler(InfoMsg{ Type: InfoTypeError, Msg: err.Error(), }) } type InfoType int const ( InfoTypeInfo InfoType = iota InfoTypeWarn InfoTypeError ) func ReportInfo(info string) tea.Cmd { return CmdHandler(InfoMsg{ Type: InfoTypeInfo, Msg: info, }) } func ReportWarn(warn string) tea.Cmd { return CmdHandler(InfoMsg{ Type: InfoTypeWarn, Msg: warn, }) } type ( InfoMsg struct { Type InfoType Msg string TTL time.Duration } ClearStatusMsg struct{} ) func Clamp(v, low, high int) int { if high < low { low, high = high, low } return min(high, max(low, v)) } ================================================ FILE: internal/version/version.go ================================================ package version import "runtime/debug" // Build-time parameters set via -ldflags var Version = "unknown" // A user may install pug using `go install github.com/opencode-ai/opencode@latest`. // without -ldflags, in which case the version above is unset. As a workaround // we use the embedded build version that *is* set when using `go install` (and // is only set for `go install` and not for `go build`). func init() { info, ok := debug.ReadBuildInfo() if !ok { // < go v1.18 return } mainVersion := info.Main.Version if mainVersion == "" || mainVersion == "(devel)" { // bin not built using `go install` return } // bin built using `go install` Version = mainVersion } ================================================ FILE: main.go ================================================ package main import ( "github.com/opencode-ai/opencode/cmd" "github.com/opencode-ai/opencode/internal/logging" ) func main() { defer logging.RecoverPanic("main", func() { logging.ErrorPersist("Application terminated due to unhandled panic") }) cmd.Execute() } ================================================ FILE: opencode-schema.json ================================================ { "$schema": "http://json-schema.org/draft-07/schema#", "definitions": { "agent": { "description": "Agent configuration", "properties": { "maxTokens": { "description": "Maximum tokens for the agent", "minimum": 1, "type": "integer" }, "model": { "description": "Model ID for the agent", "enum": [ "gpt-4.1", "llama-3.3-70b-versatile", "azure.gpt-4.1", "openrouter.gpt-4o", "openrouter.o1-mini", "openrouter.claude-3-haiku", "claude-3-opus", "gpt-4o", "gpt-4o-mini", "o1", "meta-llama/llama-4-maverick-17b-128e-instruct", "azure.o3-mini", "openrouter.gpt-4o-mini", "openrouter.o1", "claude-3.5-haiku", "o4-mini", "azure.gpt-4.1-mini", "openrouter.o3", "grok-3-beta", "o3-mini", "qwen-qwq", "azure.o1", "openrouter.gemini-2.5-flash", "openrouter.gemini-2.5", "o1-mini", "azure.gpt-4o", "openrouter.gpt-4.1-mini", "openrouter.claude-3.5-sonnet", "openrouter.o3-mini", "gpt-4.1-mini", "gpt-4.5-preview", "gpt-4.1-nano", "deepseek-r1-distill-llama-70b", "azure.gpt-4o-mini", "openrouter.gpt-4.1", "bedrock.claude-3.7-sonnet", "claude-3-haiku", "o3", "gemini-2.0-flash-lite", "azure.o3", "azure.gpt-4.5-preview", "openrouter.claude-3-opus", "grok-3-mini-fast-beta", "claude-4-sonnet", "azure.o4-mini", "grok-3-fast-beta", "claude-3.5-sonnet", "azure.o1-mini", "openrouter.claude-3.7-sonnet", "openrouter.gpt-4.5-preview", "grok-3-mini-beta", "claude-3.7-sonnet", "gemini-2.0-flash", "openrouter.deepseek-r1-free", "vertexai.gemini-2.5-flash", "vertexai.gemini-2.5", "o1-pro", "gemini-2.5", "meta-llama/llama-4-scout-17b-16e-instruct", "azure.gpt-4.1-nano", "openrouter.gpt-4.1-nano", "gemini-2.5-flash", "openrouter.o4-mini", "openrouter.claude-3.5-haiku", "claude-4-opus", "openrouter.o1-pro", "copilot.gpt-4o", "copilot.gpt-4o-mini", "copilot.gpt-4.1", "copilot.claude-3.5-sonnet", "copilot.claude-3.7-sonnet", "copilot.claude-sonnet-4", "copilot.o1", "copilot.o3-mini", "copilot.o4-mini", "copilot.gemini-2.0-flash", "copilot.gemini-2.5-pro" ], "type": "string" }, "reasoningEffort": { "description": "Reasoning effort for models that support it (OpenAI, Anthropic)", "enum": [ "low", "medium", "high" ], "type": "string" } }, "required": [ "model" ], "type": "object" } }, "description": "Configuration schema for the OpenCode application", "properties": { "agents": { "additionalProperties": { "description": "Agent configuration", "properties": { "maxTokens": { "description": "Maximum tokens for the agent", "minimum": 1, "type": "integer" }, "model": { "description": "Model ID for the agent", "enum": [ "gpt-4.1", "llama-3.3-70b-versatile", "azure.gpt-4.1", "openrouter.gpt-4o", "openrouter.o1-mini", "openrouter.claude-3-haiku", "claude-3-opus", "gpt-4o", "gpt-4o-mini", "o1", "meta-llama/llama-4-maverick-17b-128e-instruct", "azure.o3-mini", "openrouter.gpt-4o-mini", "openrouter.o1", "claude-3.5-haiku", "o4-mini", "azure.gpt-4.1-mini", "openrouter.o3", "grok-3-beta", "o3-mini", "qwen-qwq", "azure.o1", "openrouter.gemini-2.5-flash", "openrouter.gemini-2.5", "o1-mini", "azure.gpt-4o", "openrouter.gpt-4.1-mini", "openrouter.claude-3.5-sonnet", "openrouter.o3-mini", "gpt-4.1-mini", "gpt-4.5-preview", "gpt-4.1-nano", "deepseek-r1-distill-llama-70b", "azure.gpt-4o-mini", "openrouter.gpt-4.1", "bedrock.claude-3.7-sonnet", "claude-3-haiku", "o3", "gemini-2.0-flash-lite", "azure.o3", "azure.gpt-4.5-preview", "openrouter.claude-3-opus", "grok-3-mini-fast-beta", "claude-4-sonnet", "azure.o4-mini", "grok-3-fast-beta", "claude-3.5-sonnet", "azure.o1-mini", "openrouter.claude-3.7-sonnet", "openrouter.gpt-4.5-preview", "grok-3-mini-beta", "claude-3.7-sonnet", "gemini-2.0-flash", "openrouter.deepseek-r1-free", "vertexai.gemini-2.5-flash", "vertexai.gemini-2.5", "o1-pro", "gemini-2.5", "meta-llama/llama-4-scout-17b-16e-instruct", "azure.gpt-4.1-nano", "openrouter.gpt-4.1-nano", "gemini-2.5-flash", "openrouter.o4-mini", "openrouter.claude-3.5-haiku", "claude-4-opus", "openrouter.o1-pro", "copilot.gpt-4o", "copilot.gpt-4o-mini", "copilot.gpt-4.1", "copilot.claude-3.5-sonnet", "copilot.claude-3.7-sonnet", "copilot.claude-sonnet-4", "copilot.o1", "copilot.o3-mini", "copilot.o4-mini", "copilot.gemini-2.0-flash", "copilot.gemini-2.5-pro" ], "type": "string" }, "reasoningEffort": { "description": "Reasoning effort for models that support it (OpenAI, Anthropic)", "enum": [ "low", "medium", "high" ], "type": "string" } }, "required": [ "model" ], "type": "object" }, "description": "Agent configurations", "properties": { "coder": { "$ref": "#/definitions/agent" }, "task": { "$ref": "#/definitions/agent" }, "title": { "$ref": "#/definitions/agent" } }, "type": "object" }, "contextPaths": { "default": [ ".github/copilot-instructions.md", ".cursorrules", ".cursor/rules/", "CLAUDE.md", "CLAUDE.local.md", "opencode.md", "opencode.local.md", "OpenCode.md", "OpenCode.local.md", "OPENCODE.md", "OPENCODE.local.md" ], "description": "Context paths for the application", "items": { "type": "string" }, "type": "array" }, "data": { "description": "Storage configuration", "properties": { "directory": { "default": ".opencode", "description": "Directory where application data is stored", "type": "string" } }, "required": [ "directory" ], "type": "object" }, "debug": { "default": false, "description": "Enable debug mode", "type": "boolean" }, "debugLSP": { "default": false, "description": "Enable LSP debug mode", "type": "boolean" }, "lsp": { "additionalProperties": { "description": "LSP configuration for a language", "properties": { "args": { "description": "Command arguments for the LSP server", "items": { "type": "string" }, "type": "array" }, "command": { "description": "Command to execute for the LSP server", "type": "string" }, "disabled": { "default": false, "description": "Whether the LSP is disabled", "type": "boolean" }, "options": { "description": "Additional options for the LSP server", "type": "object" } }, "required": [ "command" ], "type": "object" }, "description": "Language Server Protocol configurations", "type": "object" }, "mcpServers": { "additionalProperties": { "description": "MCP server configuration", "properties": { "args": { "description": "Command arguments for the MCP server", "items": { "type": "string" }, "type": "array" }, "command": { "description": "Command to execute for the MCP server", "type": "string" }, "env": { "description": "Environment variables for the MCP server", "items": { "type": "string" }, "type": "array" }, "headers": { "additionalProperties": { "type": "string" }, "description": "HTTP headers for SSE type MCP servers", "type": "object" }, "type": { "default": "stdio", "description": "Type of MCP server", "enum": [ "stdio", "sse" ], "type": "string" }, "url": { "description": "URL for SSE type MCP servers", "type": "string" } }, "required": [ "command" ], "type": "object" }, "description": "Model Control Protocol server configurations", "type": "object" }, "providers": { "additionalProperties": { "description": "Provider configuration", "properties": { "apiKey": { "description": "API key for the provider", "type": "string" }, "disabled": { "default": false, "description": "Whether the provider is disabled", "type": "boolean" }, "provider": { "description": "Provider type", "enum": [ "anthropic", "openai", "gemini", "groq", "openrouter", "bedrock", "azure", "vertexai", "copilot" ], "type": "string" } }, "type": "object" }, "description": "LLM provider configurations", "type": "object" }, "tui": { "description": "Terminal User Interface configuration", "properties": { "theme": { "default": "opencode", "description": "TUI theme name", "enum": [ "opencode", "catppuccin", "dracula", "flexoki", "gruvbox", "monokai", "onedark", "tokyonight", "tron" ], "type": "string" } }, "type": "object" }, "wd": { "description": "Working directory for the application", "type": "string" } }, "title": "OpenCode Configuration", "type": "object" } ================================================ FILE: scripts/check_hidden_chars.sh ================================================ #!/bin/bash # Script to check for hidden/invisible characters in Go files # This helps detect potential prompt injection attempts echo "Checking Go files for hidden characters..." # Find all Go files in the repository go_files=$(find . -name "*.go" -type f) # Counter for files with hidden characters files_with_hidden=0 for file in $go_files; do # Check for specific Unicode hidden characters that could be used for prompt injection # This excludes normal whitespace like tabs and newlines # Looking for: # - Zero-width spaces (U+200B) # - Zero-width non-joiners (U+200C) # - Zero-width joiners (U+200D) # - Left-to-right/right-to-left marks (U+200E, U+200F) # - Bidirectional overrides (U+202A-U+202E) # - Byte order mark (U+FEFF) if hexdump -C "$file" | grep -E 'e2 80 8b|e2 80 8c|e2 80 8d|e2 80 8e|e2 80 8f|e2 80 aa|e2 80 ab|e2 80 ac|e2 80 ad|e2 80 ae|ef bb bf' > /dev/null 2>&1; then echo "Hidden characters found in: $file" # Show the file with potential issues echo " Hexdump showing suspicious characters:" hexdump -C "$file" | grep -E 'e2 80 8b|e2 80 8c|e2 80 8d|e2 80 8e|e2 80 8f|e2 80 aa|e2 80 ab|e2 80 ac|e2 80 ad|e2 80 ae|ef bb bf' | head -10 files_with_hidden=$((files_with_hidden + 1)) fi done if [ $files_with_hidden -eq 0 ]; then echo "No hidden characters found in any Go files." else echo "Found hidden characters in $files_with_hidden Go file(s)." fi exit $files_with_hidden # Exit with number of affected files as status code ================================================ FILE: scripts/release ================================================ #!/usr/bin/env bash # Parse command line arguments minor=false while [ "$#" -gt 0 ]; do case "$1" in --minor) minor=true; shift 1;; *) echo "Unknown parameter: $1"; exit 1;; esac done git fetch --force --tags # Get the latest Git tag latest_tag=$(git tag --sort=committerdate | grep -E '[0-9]' | tail -1) # If there is no tag, exit the script if [ -z "$latest_tag" ]; then echo "No tags found" exit 1 fi echo "Latest tag: $latest_tag" # Split the tag into major, minor, and patch numbers IFS='.' read -ra VERSION <<< "$latest_tag" if [ "$minor" = true ]; then # Increment the minor version and reset patch to 0 minor_number=${VERSION[1]} let "minor_number++" new_version="${VERSION[0]}.$minor_number.0" else # Increment the patch version patch_number=${VERSION[2]} let "patch_number++" new_version="${VERSION[0]}.${VERSION[1]}.$patch_number" fi echo "New version: $new_version" git tag $new_version git push --tags ================================================ FILE: scripts/snapshot ================================================ #!/usr/bin/env bash set -e goreleaser build --clean --snapshot --skip validate ================================================ FILE: sqlc.yaml ================================================ version: "2" sql: - engine: "sqlite" schema: "internal/db/migrations" queries: "internal/db/sql" gen: go: package: "db" out: "internal/db" emit_json_tags: true emit_prepared_queries: true emit_interface: true emit_exact_table_names: false emit_empty_slices: true