Building plugins
Tool-Plugins
Tool-Plugins fügen OpenClaw agentenaufrufbare Tools hinzu, ohne einen Kanal, Model-Provider, Hook, Dienst oder ein Setup-Backend hinzuzufügen. Verwenden Sie defineToolPlugin, wenn das Plugin eine feste Liste von Tools besitzt und OpenClaw die Manifest-Metadaten erzeugen soll, die diese Tools auffindbar halten, ohne Laufzeitcode zu laden.
Der empfohlene Ablauf ist:
- Erstellen Sie ein Paketgerüst mit
openclaw plugins init. - Schreiben Sie Tools mit
defineToolPlugin. - Bauen Sie JavaScript.
- Erzeugen Sie
openclaw.plugin.json- undpackage.json-Metadaten mitopenclaw plugins build. - Validieren Sie die erzeugten Metadaten vor der Veröffentlichung oder Installation.
Für Provider-, Kanal-, Hook-, Dienst- oder Plugins mit gemischten Fähigkeiten beginnen Sie stattdessen mit Plugins erstellen, Kanal-Plugins oder Provider-Plugins.
Anforderungen
- Node >= 22.
- TypeScript-ESM-Paketausgabe.
typeboxfür Konfigurations- und Tool-Parameterschemas.openclaw >=2026.5.17, die erste OpenClaw-Version, dieopenclaw/plugin-sdk/tool-pluginexportiert.- Ein Paket-Root, der
dist/,openclaw.plugin.jsonundpackage.jsonausliefern kann.
Das erzeugte Plugin importiert typebox zur Laufzeit. Behalten Sie typebox daher in
dependencies, nicht nur in devDependencies.
Schnellstart
Erstellen Sie ein neues Plugin-Paket:
openclaw plugins init stock-quotes --name "Stock Quotes"cd stock-quotesnpm installnpm run plugin:buildnpm run plugin:validatenpm testDas Gerüst erstellt:
src/index.ts: einendefineToolPlugin-Einstieg mit einemecho-Tool.src/index.test.ts: einen kleinen Metadatentest.tsconfig.json: NodeNext-TypeScript-Ausgabe nachdist/.package.json: Skripte, Laufzeitabhängigkeiten undopenclaw.extensions: ["./dist/index.js"].openclaw.plugin.json: erzeugte Manifest-Metadaten für das anfängliche Tool.
Erwartete Validierungsausgabe:
Plugin stock-quotes is valid.Ein Tool schreiben
defineToolPlugin übernimmt die Plugin-Identität, ein optionales Konfigurationsschema und eine
statische Liste von Tools. Parameter- und Konfigurationstypen werden aus TypeBox-
Schemas abgeleitet.
export default defineToolPlugin({ id: "stock-quotes", name: "Stock Quotes", description: "Fetch stock quote snapshots.", configSchema: Type.Object({ apiKey: Type.Optional(Type.String({ description: "Quote API key." })), baseUrl: Type.Optional(Type.String({ description: "Quote API base URL." })), }), tools: (tool) => [ tool({ name: "stock_quote", label: "Stock Quote", description: "Fetch a stock quote snapshot.", parameters: Type.Object({ symbol: Type.String({ description: "Ticker symbol, for example OPEN." }), }), async execute({ symbol }, config, context) { context.signal?.throwIfAborted(); return { symbol: symbol.toUpperCase(), configured: Boolean(config.apiKey), baseUrl: config.baseUrl ?? "https://api.example.com", }; }, }), ],});Tool-Namen sind die stabile API. Wählen Sie Namen, die eindeutig, kleingeschrieben und spezifisch genug sind, um Kollisionen mit Core-Tools oder anderen Plugins zu vermeiden.
Optionale und Factory-Tools
Setzen Sie optional: true, wenn Benutzer das Tool ausdrücklich auf die Allowlist setzen sollen, bevor es
an ein Modell gesendet wird:
tool({ name: "workflow_run", description: "Run an external workflow.", parameters: Type.Object({ goal: Type.String() }), optional: true, execute: ({ goal }) => ({ queued: true, goal }),});openclaw plugins build schreibt den passenden toolMetadata.<tool>.optional-
Manifest-Eintrag, damit OpenClaw das Tool erkennen kann, ohne den Plugin-
Laufzeitcode zu laden.
Verwenden Sie factory, wenn ein Tool den Laufzeit-Tool-Kontext benötigt, bevor es
erstellt werden kann. Die Factory hält die Metadaten statisch, während das Tool für einen
bestimmten Lauf aussteigen, den Sandbox-Zustand prüfen oder Laufzeithelfer binden kann.
tool({ name: "local_workflow", description: "Run a local workflow outside sandboxed sessions.", parameters: Type.Object({ goal: Type.String() }), optional: true, factory({ api, toolContext }) { if (toolContext.sandboxed) { return null; } return createLocalWorkflowTool(api); },});Factories sind weiterhin für feste Tool-Namen gedacht. Verwenden Sie definePluginEntry direkt, wenn
das Plugin Tool-Namen dynamisch berechnet oder Tools mit Hooks,
Diensten, Providern, Befehlen oder anderen Laufzeitoberflächen kombiniert.
Rückgabewerte
defineToolPlugin umschließt einfache Rückgabewerte in das OpenClaw-Tool-Ergebnisformat:
- Geben Sie einen String zurück, wenn das Modell genau diesen Text sehen soll.
- Geben Sie einen JSON-kompatiblen Wert zurück, wenn das Modell formatiertes JSON sehen soll
und OpenClaw den ursprünglichen Wert in
detailsbehalten soll.
tool({ name: "echo_text", description: "Echo input text.", parameters: Type.Object({ input: Type.String(), }), execute: ({ input }) => input,});tool({ name: "echo_json", description: "Echo input as structured JSON.", parameters: Type.Object({ input: Type.String(), }), execute: ({ input }) => ({ input, length: input.length }),});Verwenden Sie ein Factory-Tool, wenn Sie ein benutzerdefiniertes AgentToolResult zurückgeben oder eine
vorhandene api.registerTool-Implementierung wiederverwenden müssen. Verwenden Sie definePluginEntry statt
defineToolPlugin, wenn Sie vollständig dynamische Tools oder gemischte Plugin-
Fähigkeiten benötigen.
Konfiguration
configSchema ist optional. Wenn Sie es weglassen, verwendet OpenClaw ein strikt leeres Objekt-
Schema, und das erzeugte Manifest enthält weiterhin configSchema.
export default defineToolPlugin({ id: "no-config-tools", name: "No Config Tools", description: "Adds tools that do not need configuration.", tools: () => [],});Wenn Sie configSchema einschließen, wird das zweite execute-Argument aus dem
Schema typisiert:
const configSchema = Type.Object({ apiKey: Type.String(),}); export default defineToolPlugin({ id: "configured-tools", name: "Configured Tools", description: "Adds configured tools.", configSchema, tools: (tool) => [ tool({ name: "configured_ping", description: "Check whether configuration is available.", parameters: Type.Object({}), execute: (_params, config) => ({ hasKey: config.apiKey.length > 0 }), }), ],});OpenClaw liest die Plugin-Konfiguration aus dem Plugin-Eintrag in der Gateway-Konfiguration. Hinterlegen Sie keine Secrets fest im Quellcode oder in Dokumentationsbeispielen. Verwenden Sie Konfiguration, Umgebungs- variablen oder SecretRefs gemäß dem Sicherheitsmodell des Plugins.
Erzeugte Metadaten
OpenClaw erkennt installierte Plugins aus kalten Metadaten. Es muss das
Plugin-Manifest lesen können, bevor Plugin-Laufzeitcode importiert wird. defineToolPlugin
stellt daher statische Metadaten bereit, und openclaw plugins build schreibt diese
Metadaten in das Paket.
Führen Sie den Generator aus, nachdem Sie Plugin-ID, Name, Beschreibung, Konfigurationsschema, Aktivierung oder Tool-Namen geändert haben:
npm run buildopenclaw plugins build --entry ./dist/index.jsFür ein Plugin mit einem Tool sieht das erzeugte Manifest so aus:
{ "id": "stock-quotes", "name": "Stock Quotes", "description": "Fetch stock quote snapshots.", "version": "0.1.0", "configSchema": { "type": "object", "additionalProperties": false, "properties": {} }, "activation": { "onStartup": true }, "contracts": { "tools": ["stock_quote"] }}contracts.tools ist der wichtige Discovery-Vertrag. Er teilt OpenClaw mit, welches
Plugin jedes Tool besitzt, ohne jede installierte Plugin-Laufzeit zu laden. Wenn das
Manifest veraltet ist, fehlt das Tool möglicherweise in der Discovery, oder das falsche Plugin
wird für einen Registrierungsfehler verantwortlich gemacht.
Paketmetadaten
Für den einfachen Tool-Plugin-Workflow richtet openclaw plugins build
package.json am ausgewählten einzelnen Laufzeiteinstieg aus:
{ "type": "module", "files": ["dist", "openclaw.plugin.json", "README.md"], "dependencies": { "typebox": "^1.1.38" }, "peerDependencies": { "openclaw": ">=2026.5.17" }, "openclaw": { "extensions": ["./dist/index.js"] }}Verwenden Sie gebautes JavaScript wie ./dist/index.js für installierte Pakete. Quell-
Einstiege sind in der Workspace-Entwicklung nützlich, veröffentlichte Pakete sollten jedoch nicht
vom Laden der TypeScript-Laufzeit abhängen.
In CI validieren
Verwenden Sie plugins build --check, damit CI fehlschlägt, wenn erzeugte Metadaten veraltet sind, ohne
Dateien neu zu schreiben:
npm run buildopenclaw plugins build --entry ./dist/index.js --checkopenclaw plugins validate --entry ./dist/index.jsnpm testplugins validate prüft, dass:
openclaw.plugin.jsonexistiert und den normalen Manifest-Loader besteht.- Der aktuelle Einstieg
defineToolPlugin-Metadaten exportiert. - Erzeugte Manifest-Felder mit den Einstiegsmetadaten übereinstimmen.
contracts.toolsmit den deklarierten Tool-Namen übereinstimmt.package.jsonopenclaw.extensionsauf den ausgewählten Laufzeiteinstieg zeigt.
Lokal installieren und prüfen
Installieren Sie den Paketpfad aus einem separaten OpenClaw-Checkout oder mit installierter CLI:
openclaw plugins install ./stock-quotesopenclaw plugins inspect stock-quotes --runtimeFür einen Paket-Smoke-Test packen Sie zuerst und installieren dann den Tarball:
npm packopenclaw plugins install npm-pack:./openclaw-plugin-stock-quotes-0.1.0.tgzopenclaw plugins inspect stock-quotes --runtime --jsonStarten oder starten Sie nach der Installation den Gateway neu und bitten Sie den Agenten, das Tool zu verwenden. Wenn Sie die Sichtbarkeit von Tools debuggen, prüfen Sie die Plugin-Laufzeit und den effektiven Tool-Katalog, bevor Sie den Code ändern.
Veröffentlichen
Veröffentlichen Sie über ClawHub, wenn das Paket bereit ist:
clawhub package publish your-org/stock-quotes --dry-runclawhub package publish your-org/stock-quotesInstallieren Sie mit einem expliziten ClawHub-Locator:
openclaw plugins install clawhub:your-org/stock-quotesBloße npm-Paketspezifikationen bleiben während der Launch-Umstellung unterstützt, aber ClawHub ist die bevorzugte Discovery- und Distributionsoberfläche für OpenClaw-Plugins.
Fehlerbehebung
plugin entry not found: ./dist/index.js
Die ausgewählte Einstiegsdatei existiert nicht. Führen Sie npm run build aus und anschließend erneut
openclaw plugins build --entry ./dist/index.js oder
openclaw plugins validate --entry ./dist/index.js.
plugin entry does not expose defineToolPlugin metadata
Der Einstieg hat keinen von defineToolPlugin erzeugten Wert exportiert. Prüfen Sie, dass der
Standardexport des Moduls das Ergebnis von defineToolPlugin(...) ist, oder übergeben Sie den richtigen
Einstieg mit --entry.
openclaw.plugin.json generated metadata is stale
Das Manifest stimmt nicht mehr mit den Einstiegsmetadaten überein. Führen Sie aus:
npm run buildopenclaw plugins build --entry ./dist/index.jsCommitten Sie sowohl die Änderungen an openclaw.plugin.json als auch an package.json.
package.json openclaw.extensions must include ./dist/index.js
Die Paketmetadaten zeigen auf einen anderen Laufzeiteinstieg. Führen Sie
openclaw plugins build --entry ./dist/index.js aus, damit der Generator die
Paketmetadaten an den Einstieg anpasst, den Sie ausliefern möchten.
Cannot find package 'typebox'
Das gebaute Plugin importiert typebox zur Laufzeit. Behalten Sie typebox in
dependencies, installieren Sie die Paketabhängigkeiten erneut, bauen Sie neu und führen Sie die Validierung erneut aus.
Tool erscheint nach der Installation nicht
Prüfen Sie Folgendes der Reihe nach:
openclaw plugins inspect <plugin-id> --runtimeopenclaw plugins validate --root <plugin-root> --entry ./dist/index.jsopenclaw.plugin.jsonenthältcontracts.toolsmit den erwarteten Tool-Namen.package.jsonenthältopenclaw.extensions: ["./dist/index.js"].- Der Gateway wurde nach der Installation des Plugins neu gestartet oder neu geladen.