Plugins
Escreva seus próprios plugins para estender o opencode.
Plugins permitem que você estenda o opencode conectando-se a vários eventos e personalizando o comportamento. Você pode criar plugins para adicionar novos recursos, integrar-se a serviços externos ou modificar o comportamento padrão do opencode.
Para exemplos, confira os plugins criados pela comunidade.
Usar um plugin
Existem duas maneiras de carregar plugins.
De arquivos locais
Coloque arquivos JavaScript ou TypeScript no diretório de plugins.
.opencode/plugins/- Plugins em nível de projeto~/.config/opencode/plugins/- Plugins globais
Os arquivos nesses diretórios são carregados automaticamente na inicialização.
Do npm
Especifique pacotes npm no seu arquivo de configuração.
{ "$schema": "https://opencode.ai/config.json", "plugin": ["opencode-helicone-session", "opencode-wakatime", "@my-org/custom-plugin"]}Pacotes npm regulares e escopados são suportados.
Navegue pelos plugins disponíveis no ecossistema.
Como os plugins são instalados
Plugins npm são instalados automaticamente usando Bun na inicialização. Pacotes e suas dependências são armazenados em cache em ~/.cache/opencode/node_modules/.
Plugins locais são carregados diretamente do diretório de plugins. Para usar pacotes externos, você deve criar um package.json dentro do seu diretório de configuração (veja Dependências), ou publicar o plugin no npm e adicioná-lo à sua configuração.
Ordem de carregamento
Os plugins são carregados de todas as fontes e todos os hooks são executados em sequência. A ordem de carregamento é:
- Configuração global (
~/.config/opencode/opencode.json) - Configuração do projeto (
opencode.json) - Diretório de plugins global (
~/.config/opencode/plugins/) - Diretório de plugins do projeto (
.opencode/plugins/)
Pacotes npm duplicados com o mesmo nome e versão são carregados uma vez. No entanto, um plugin local e um plugin npm com nomes semelhantes são carregados separadamente.
Criar um plugin
Um plugin é um módulo JavaScript/TypeScript que exporta uma ou mais funções de plugin. Cada função recebe um objeto de contexto e retorna um objeto de hooks.
Dependências
Plugins locais e ferramentas personalizadas podem usar pacotes npm externos. Adicione um package.json ao seu diretório de configuração com as dependências necessárias.
{ "dependencies": { "shescape": "^2.1.0" }}O opencode executa bun install na inicialização para instalar esses pacotes. Seus plugins e ferramentas podem então importá-los.
import { escape } from "shescape"
export const MyPlugin = async (ctx) => { return { "tool.execute.before": async (input, output) => { if (input.tool === "bash") { output.args.command = escape(output.args.command) } }, }}Estrutura básica
export const MyPlugin = async ({ project, client, $, directory, worktree }) => { console.log("Plugin initialized!")
return { // Hook implementations go here }}A função do plugin recebe:
project: As informações do projeto atual.directory: O diretório de trabalho atual.worktree: O caminho do worktree do git.client: Um cliente SDK do opencode para interagir com a AI.$: A API shell do Bun para executar comandos.
Suporte a TypeScript
Para plugins TypeScript, você pode importar tipos do pacote de plugin:
import type { Plugin } from "@opencode-ai/plugin"
export const MyPlugin: Plugin = async ({ project, client, $, directory, worktree }) => { return { // Type-safe hook implementations }}Eventos
Plugins podem se inscrever em eventos como visto abaixo na seção Exemplos. Aqui está uma lista dos diferentes eventos disponíveis.
Eventos de Comando
command.executed
Eventos de Arquivo
file.editedfile.watcher.updated
Eventos de Instalação
installation.updated
Eventos LSP
lsp.client.diagnosticslsp.updated
Eventos de Mensagem
message.part.removedmessage.part.updatedmessage.removedmessage.updated
Eventos de Permissão
permission.askedpermission.replied
Eventos de Servidor
server.connected
Eventos de Sessão
session.createdsession.compactedsession.deletedsession.diffsession.errorsession.idlesession.statussession.updated
Eventos de Todo
todo.updated
Eventos de Shell
shell.env
Eventos de Ferramenta
tool.execute.aftertool.execute.before
Eventos TUI
tui.prompt.appendtui.command.executetui.toast.show
Exemplos
Aqui estão alguns exemplos de plugins que você pode usar para estender o opencode.
Enviar notificações
Envie notificações quando certos eventos ocorrerem:
export const NotificationPlugin = async ({ project, client, $, directory, worktree }) => { return { event: async ({ event }) => { // Send notification on session completion if (event.type === "session.idle") { await $`osascript -e 'display notification "Session completed!" with title "opencode"'` } }, }}Estamos usando osascript para executar AppleScript no macOS. Aqui estamos usando para enviar notificações.
Proteção .env
Impeça o opencode de ler arquivos .env:
export const EnvProtection = async ({ project, client, $, directory, worktree }) => { return { "tool.execute.before": async (input, output) => { if (input.tool === "read" && output.args.filePath.includes(".env")) { throw new Error("Do not read .env files") } }, }}Injetar variáveis de ambiente
Injete variáveis de ambiente em todas as execuções de shell (ferramentas de AI e terminais de usuário):
export const InjectEnvPlugin = async () => { return { "shell.env": async (input, output) => { output.env.MY_API_KEY = "secret" output.env.PROJECT_ROOT = input.cwd }, }}Ferramentas personalizadas
Plugins também podem adicionar ferramentas personalizadas ao opencode:
import { type Plugin, tool } from "@opencode-ai/plugin"
export const CustomToolsPlugin: Plugin = async (ctx) => { return { tool: { mytool: tool({ description: "This is a custom tool", args: { foo: tool.schema.string(), }, async execute(args, context) { const { directory, worktree } = context return `Hello ${args.foo} from ${directory} (worktree: ${worktree})` }, }), }, }}O helper tool cria uma ferramenta personalizada que o opencode pode chamar. Ele aceita uma função de esquema Zod e retorna uma definição de ferramenta com:
description: O que a ferramenta fazargs: Esquema Zod para os argumentos da ferramentaexecute: Função que é executada quando a ferramenta é chamada
Suas ferramentas personalizadas estarão disponíveis para o opencode junto com as ferramentas integradas.
Registro
Use client.app.log() em vez de console.log para registro estruturado:
export const MyPlugin = async ({ client }) => { await client.app.log({ body: { service: "my-plugin", level: "info", message: "Plugin initialized", extra: { foo: "bar" }, }, })}Níveis: debug, info, warn, error. Veja a documentação do SDK para detalhes.
Hooks de compactação
Personalize o contexto incluído quando uma sessão é compactada:
import type { Plugin } from "@opencode-ai/plugin"
export const CompactionPlugin: Plugin = async (ctx) => { return { "experimental.session.compacting": async (input, output) => { // Inject additional context into the compaction prompt output.context.push(`## Custom Context
Include any state that should persist across compaction:- Current task status- Important decisions made- Files being actively worked on`) }, }}O hook experimental.session.compacting é acionado antes que o LLM gere um resumo de continuação. Use-o para injetar contexto específico de domínio que o prompt de compactação padrão poderia perder.
Você também pode substituir o prompt de compactação completamente definindo output.prompt:
import type { Plugin } from "@opencode-ai/plugin"
export const CustomCompactionPlugin: Plugin = async (ctx) => { return { "experimental.session.compacting": async (input, output) => { // Replace the entire compaction prompt output.prompt = `You are generating a continuation prompt for a multi-agent swarm session.
Summarize:1. The current task and its status2. Which files are being modified and by whom3. Any blockers or dependencies between agents4. The next steps to complete the work
Format as a structured prompt that a new agent can use to resume work.` }, }}Quando output.prompt é definido, ele substitui completamente o prompt de compactação padrão. O array output.context é ignorado neste caso.