Plugins
Schreiben Sie Ihre eigenen Plugins, um OpenCode zu erweitern.
Mit Plugins können Sie OpenCode erweitern, indem Sie verschiedene Ereignisse einbinden und das Verhalten anpassen. Sie können Plugins erstellen, um neue Funktionen hinzuzufügen, externe Dienste zu integrieren oder das Standardverhalten von OpenCode zu ändern.
Beispiele finden Sie im von der Community erstellten plugins.
Verwenden Sie ein Plugin
Es gibt zwei Möglichkeiten, Plugins zu laden.
Aus lokalen Dateien
Platzieren Sie JavaScript- oder TypeScript-Dateien im Plugin-Verzeichnis.
.opencode/plugins/– Plugins auf Projektebene~/.config/opencode/plugins/– Globale Plugins
Dateien in diesen Verzeichnissen werden beim Start automatisch geladen.
Von npm
Geben Sie npm-Pakete in Ihrer Konfigurationsdatei an.
{ "$schema": "https://opencode.ai/config.json", "plugin": ["opencode-helicone-session", "opencode-wakatime", "@my-org/custom-plugin"]}Es werden sowohl reguläre als auch bereichsbezogene NPM-Pakete unterstützt.
Durchsuchen Sie die verfügbaren Plugins im ecosystem.
Wie Plugins installiert werden
NPM-Plugins werden beim Start automatisch mit Bun installiert. Pakete und ihre Abhängigkeiten werden in ~/.cache/opencode/node_modules/ zwischengespeichert.
Lokale Plugins werden direkt aus dem Plugin-Verzeichnis geladen. Um externe Pakete zu verwenden, müssen Sie ein package.json in Ihrem Konfigurationsverzeichnis erstellen (siehe Dependencies) oder das Plugin auf npm und add it to your config veröffentlichen.
Auftrag laden
Plugins werden aus allen Quellen geladen und alle Hooks werden nacheinander ausgeführt. Die Ladereihenfolge lautet:
- Globale Konfiguration (
~/.config/opencode/opencode.json) - Projektkonfiguration (
opencode.json) - Globales Plugin-Verzeichnis (
~/.config/opencode/plugins/) - Projekt-Plugin-Verzeichnis (
.opencode/plugins/)
Doppelte npm-Pakete mit demselben Namen und derselben Version werden einmal geladen. Allerdings werden ein lokales Plugin und ein NPM-Plugin mit ähnlichen Namen beide separat geladen.
Erstellen Sie ein Plugin
A plugin is a JavaScript/TypeScript module that exports one or more plugin Funktionen. Jede Funktion empfängt ein Kontextobjekt und gibt ein Hooks-Objekt zurück.
Dependencies
Lokale Plugins und benutzerdefinierte Tools können externe npm-Pakete verwenden. Fügen Sie Ihrem Konfigurationsverzeichnis ein package.json mit den benötigten Abhängigkeiten hinzu.
{ "dependencies": { "shescape": "^2.1.0" }}OpenCode führt beim Start bun install aus, um diese zu installieren. Ihre Plugins und Tools können sie dann importieren.
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) } }, }}Grundstruktur
export const MyPlugin = async ({ project, client, $, directory, worktree }) => { console.log("Plugin initialized!")
return { // Hook implementations go here }}Die Plugin-Funktion erhält:
project: Die aktuellen Projektinformationen.directory: Das aktuelle Arbeitsverzeichnis.worktree: Der Git-Worktree-Pfad.client: Ein OpenCode-SDK-Client für die Interaktion mit AI.$: Buns shell API zum Ausführen von Befehlen.
TypeScript support
Für TypeScript-Plugins können Sie Typen aus dem Plugin-Paket importieren:
import type { Plugin } from "@opencode-ai/plugin"
export const MyPlugin: Plugin = async ({ project, client, $, directory, worktree }) => { return { // Type-safe hook implementations }}Events
Plugins können Ereignisse abonnieren, wie unten im Abschnitt „Beispiele“ gezeigt. Hier finden Sie eine Liste der verschiedenen verfügbaren Veranstaltungen.
Befehlsereignisse
command.executed
Dateiereignisse
file.editedfile.watcher.updated
Installationsereignisse
installation.updated
LSP Ereignisse
lsp.client.diagnosticslsp.updated
Nachrichtenereignisse
message.part.removedmessage.part.updatedmessage.removedmessage.updated
Berechtigungsereignisse
permission.askedpermission.replied
Serverereignisse
server.connected
Sitzungsereignisse
session.createdsession.compactedsession.deletedsession.diffsession.errorsession.idlesession.statussession.updated
Todo-Events
todo.updated
Shell-Events
shell.env
Tool-Ereignisse
tool.execute.aftertool.execute.before
TUI Ereignisse
tui.prompt.appendtui.command.executetui.toast.show
Examples
Hier sind einige Beispiele für Plugins, mit denen Sie OpenCode erweitern können.
Senden Sie Benachrichtigungen
Senden Sie Benachrichtigungen, wenn bestimmte Ereignisse eintreten:
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"'` } }, }}Wir verwenden osascript, um AppleScript unter macOS auszuführen. Hier verwenden wir es zum Versenden von Benachrichtigungen.
.env protection
Verhindern Sie, dass OpenCode .env-Dateien liest:
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") } }, }}Umgebungsvariablen einfügen
Fügen Sie Umgebungsvariablen in alle Shell-Ausführungen ein (AI-Tools und Benutzerterminals):
export const InjectEnvPlugin = async () => { return { "shell.env": async (input, output) => { output.env.MY_API_KEY = "secret" output.env.PROJECT_ROOT = input.cwd }, }}Benutzerdefinierte Werkzeuge
Plugins können OpenCode auch benutzerdefinierte Tools hinzufügen:
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})` }, }), }, }}Der tool-Helfer erstellt ein benutzerdefiniertes Tool, das OpenCode aufrufen kann. Es verwendet eine Zod-Schemafunktion und gibt eine Tooldefinition zurück mit:
description: Was das Tool machtargs: Zod-Schema für die Argumente des Toolsexecute: Funktion, die ausgeführt wird, wenn das Tool aufgerufen wird
Ihre benutzerdefinierten Tools stehen neben den integrierten Tools für OpenCode zur Verfügung.
Logging
Verwenden Sie client.app.log() anstelle von console.log für die strukturierte Protokollierung:
export const MyPlugin = async ({ client }) => { await client.app.log({ body: { service: "my-plugin", level: "info", message: "Plugin initialized", extra: { foo: "bar" }, }, })}Ebenen: debug, info, warn, error. Weitere Informationen finden Sie unter SDK documentation.
Verdichtungshaken
Passen Sie den Kontext an, der beim Komprimieren einer Sitzung einbezogen wird:
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`) }, }}Der Hook experimental.session.compacting wird ausgelöst, bevor der Hook LLM eine Fortsetzungszusammenfassung generiert. Verwenden Sie es, um domänenspezifischen Kontext einzufügen, der der standardmäßigen Komprimierungsaufforderung entgehen würde.
Sie können die Komprimierungsaufforderung auch vollständig ersetzen, indem Sie output.prompt festlegen:
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.` }, }}Wenn output.prompt festgelegt ist, ersetzt es die standardmäßige Komprimierungsaufforderung vollständig. Das Array output.context wird in diesem Fall ignoriert.