Building plugins

Criando plugins de provedor

Este guia percorre a criação de um Plugin de provedor que adiciona um provedor de modelo (LLM) ao OpenClaw. Ao final, você terá um provedor com um catálogo de modelos, autenticação por chave de API e resolução dinâmica de modelos.

Passo a passo

  • Pacote e manifesto

    Etapa 1: Pacote e manifesto

    package.json
    {"name": "@myorg/openclaw-acme-ai","version": "1.0.0","type": "module","openclaw": {  "extensions": ["./index.ts"],  "providers": ["acme-ai"],  "compat": {    "pluginApi": ">=2026.3.24-beta.2",    "minGatewayVersion": "2026.3.24-beta.2"  },  "build": {    "openclawVersion": "2026.3.24-beta.2",    "pluginSdkVersion": "2026.3.24-beta.2"  }}}
    openclaw.plugin.json
    {"id": "acme-ai","name": "Acme AI","description": "Acme AI model provider","providers": ["acme-ai"],"modelSupport": {  "modelPrefixes": ["acme-"]},"providerAuthEnvVars": {  "acme-ai": ["ACME_AI_API_KEY"]},"providerAuthAliases": {  "acme-ai-coding": "acme-ai"},"providerAuthChoices": [  {    "provider": "acme-ai",    "method": "api-key",    "choiceId": "acme-ai-api-key",    "choiceLabel": "Acme AI API key",    "groupId": "acme-ai",    "groupLabel": "Acme AI",    "cliFlag": "--acme-ai-api-key",    "cliOption": "--acme-ai-api-key <key>",    "cliDescription": "Acme AI API key"  }],"configSchema": {  "type": "object",  "additionalProperties": false}}

    O manifesto declara providerAuthEnvVars para que o OpenClaw consiga detectar credenciais sem carregar o runtime do seu Plugin. Adicione providerAuthAliases quando uma variante de provedor deve reutilizar a autenticação do ID de outro provedor. modelSupport é opcional e permite que o OpenClaw carregue automaticamente seu Plugin de provedor a partir de IDs de modelo abreviados como acme-large antes que existam hooks de runtime. Se você publicar o provedor no ClawHub, esses campos openclaw.compat e openclaw.build serão obrigatórios em package.json.

  • Registre o provedor

    Um provedor de texto mínimo precisa de id, label, auth e catalog. catalog é o hook de runtime/configuração pertencente ao provedor; ele pode chamar APIs de fornecedores ao vivo e retorna entradas models.providers.

    index.ts
    import { definePluginEntry } from "openclaw/plugin-sdk/plugin-entry";import { createProviderApiKeyAuthMethod } from "openclaw/plugin-sdk/provider-auth"; export default definePluginEntry({  id: "acme-ai",  name: "Acme AI",  description: "Acme AI model provider",  register(api) {    api.registerProvider({      id: "acme-ai",      label: "Acme AI",      docsPath: "/providers/acme-ai",      envVars: ["ACME_AI_API_KEY"],       auth: [        createProviderApiKeyAuthMethod({          providerId: "acme-ai",          methodId: "api-key",          label: "Acme AI API key",          hint: "API key from your Acme AI dashboard",          optionKey: "acmeAiApiKey",          flagName: "--acme-ai-api-key",          envVar: "ACME_AI_API_KEY",          promptMessage: "Enter your Acme AI API key",          defaultModel: "acme-ai/acme-large",        }),      ],       catalog: {        order: "simple",        run: async (ctx) => {          const apiKey =            ctx.resolveProviderApiKey("acme-ai").apiKey;          if (!apiKey) return null;          return {            provider: {              baseUrl: "https://api.acme-ai.com/v1",              apiKey,              api: "openai-completions",              models: [                {                  id: "acme-large",                  name: "Acme Large",                  reasoning: true,                  input: ["text", "image"],                  cost: { input: 3, output: 15, cacheRead: 0.3, cacheWrite: 3.75 },                  contextWindow: 200000,                  maxTokens: 32768,                },                {                  id: "acme-small",                  name: "Acme Small",                  reasoning: false,                  input: ["text"],                  cost: { input: 1, output: 5, cacheRead: 0.1, cacheWrite: 1.25 },                  contextWindow: 128000,                  maxTokens: 8192,                },              ],            },          };        },      },    });     api.registerModelCatalogProvider({      provider: "acme-ai",      kinds: ["text"],      liveCatalog: async (ctx) => {        const apiKey = ctx.resolveProviderApiKey("acme-ai").apiKey;        if (!apiKey) return null;        return [          {            kind: "text",            provider: "acme-ai",            model: "acme-large",            label: "Acme Large",            source: "live",          },        ];      },    });  },});

    registerModelCatalogProvider é a superfície de catálogo de plano de controle mais recente para IU de lista/ajuda/seletor. Use-a para linhas de texto, geração de imagem, geração de vídeo e geração de música. Mantenha as chamadas aos endpoints do fornecedor e o mapeamento de respostas no Plugin; o OpenClaw controla o formato compartilhado das linhas, os rótulos de origem e a renderização de ajuda.

    Isso é um provedor funcional. Os usuários agora podem executar openclaw onboard --acme-ai-api-key <key> e selecionar acme-ai/acme-large como modelo.

    Se o provedor upstream usar tokens de controle diferentes dos do OpenClaw, adicione uma pequena transformação bidirecional de texto em vez de substituir o caminho de stream:

    typescript
    api.registerTextTransforms({  input: [    { from: /red basket/g, to: "blue basket" },    { from: /paper ticket/g, to: "digital ticket" },    { from: /left shelf/g, to: "right shelf" },  ],  output: [    { from: /blue basket/g, to: "red basket" },    { from: /digital ticket/g, to: "paper ticket" },    { from: /right shelf/g, to: "left shelf" },  ],});

    input reescreve o prompt de sistema final e o conteúdo das mensagens de texto antes do transporte. output reescreve deltas de texto do assistente e o texto final antes que o OpenClaw analise seus próprios marcadores de controle ou a entrega pelo canal.

    Para provedores incluídos que registram apenas um provedor de texto com autenticação por chave de API mais um único runtime respaldado por catálogo, prefira o helper mais restrito defineSingleProviderPluginEntry(...):

    typescript
    import { defineSingleProviderPluginEntry } from "openclaw/plugin-sdk/provider-entry"; export default defineSingleProviderPluginEntry({  id: "acme-ai",  name: "Acme AI",  description: "Acme AI model provider",  provider: {    label: "Acme AI",    docsPath: "/providers/acme-ai",    auth: [      {        methodId: "api-key",        label: "Acme AI API key",        hint: "API key from your Acme AI dashboard",        optionKey: "acmeAiApiKey",        flagName: "--acme-ai-api-key",        envVar: "ACME_AI_API_KEY",        promptMessage: "Enter your Acme AI API key",        defaultModel: "acme-ai/acme-large",      },    ],    catalog: {      buildProvider: () => ({        api: "openai-completions",        baseUrl: "https://api.acme-ai.com/v1",        models: [{ id: "acme-large", name: "Acme Large" }],      }),      buildStaticProvider: () => ({        api: "openai-completions",        baseUrl: "https://api.acme-ai.com/v1",        models: [{ id: "acme-large", name: "Acme Large" }],      }),    },  },});

    buildProvider é o caminho de catálogo ao vivo usado quando o OpenClaw consegue resolver a autenticação real do provedor. Ele pode executar descoberta específica do provedor. Use buildStaticProvider apenas para linhas offline que sejam seguras para exibir antes que a autenticação seja configurada; ele não deve exigir credenciais nem fazer solicitações de rede. A exibição models list --all do OpenClaw atualmente executa catálogos estáticos apenas para Plugins de provedor incluídos, com configuração vazia, env vazio e nenhum caminho de agente/workspace.

    Se seu fluxo de autenticação também precisar corrigir models.providers.*, aliases e o modelo padrão do agente durante o onboarding, use os helpers de preset de openclaw/plugin-sdk/provider-onboard. Os helpers mais restritos são createDefaultModelPresetAppliers(...), createDefaultModelsPresetAppliers(...) e createModelCatalogPresetAppliers(...).

    Quando o endpoint nativo de um provedor oferecer suporte a blocos de uso transmitidos por stream no transporte normal openai-completions, prefira os helpers de catálogo compartilhados em openclaw/plugin-sdk/provider-catalog-shared em vez de codificar verificações de IDs de provedor. supportsNativeStreamingUsageCompat(...) e applyProviderNativeStreamingUsageCompat(...) detectam suporte a partir do mapa de capacidades do endpoint, então endpoints nativos no estilo Moonshot/DashScope ainda optam por participar mesmo quando um Plugin está usando um ID de provedor personalizado.

  • Adicione resolução dinâmica de modelos

    Se seu provedor aceitar IDs de modelo arbitrários (como um proxy ou roteador), adicione resolveDynamicModel:

    typescript
    api.registerProvider({  // ... id, label, auth, catalog from above   resolveDynamicModel: (ctx) => ({    id: ctx.modelId,    name: ctx.modelId,    provider: "acme-ai",    api: "openai-completions",    baseUrl: "https://api.acme-ai.com/v1",    reasoning: false,    input: ["text"],    cost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0 },    contextWindow: 128000,    maxTokens: 8192,  }),});

    Se a resolução exigir uma chamada de rede, use prepareDynamicModel para aquecimento assíncrono

    • resolveDynamicModel executa novamente depois que ela é concluída.
  • Adicione hooks de runtime (conforme necessário)

    A maioria dos provedores precisa apenas de catalog + resolveDynamicModel. Adicione hooks incrementalmente conforme seu provedor precisar deles.

    Os builders de helpers compartilhados agora cobrem as famílias mais comuns de replay/compatibilidade de ferramentas, então os Plugins geralmente não precisam conectar manualmente cada hook, um por um:

    typescript
    import { buildProviderReplayFamilyHooks } from "openclaw/plugin-sdk/provider-model-shared";import { buildProviderStreamFamilyHooks } from "openclaw/plugin-sdk/provider-stream";import { buildProviderToolCompatFamilyHooks } from "openclaw/plugin-sdk/provider-tools"; const GOOGLE_FAMILY_HOOKS = {  ...buildProviderReplayFamilyHooks({ family: "google-gemini" }),  ...buildProviderStreamFamilyHooks("google-thinking"),  ...buildProviderToolCompatFamilyHooks("gemini"),}; api.registerProvider({  id: "acme-gemini-compatible",  // ...  ...GOOGLE_FAMILY_HOOKS,});

    Famílias de replay disponíveis hoje:

    Família O que ela conecta Exemplos incluídos
    openai-compatible Política compartilhada de repetição no estilo OpenAI para transportes compatíveis com OpenAI, incluindo sanitização de ids de chamadas de ferramenta, correções de ordenação com assistente primeiro e validação genérica de turnos do Gemini quando o transporte precisa disso moonshot, ollama, xai, zai
    anthropic-by-model Política de repetição ciente do Claude escolhida por modelId, para que transportes de mensagens da Anthropic recebam limpeza de blocos de pensamento específica do Claude somente quando o modelo resolvido for de fato um id do Claude amazon-bedrock, anthropic-vertex
    google-gemini Política de repetição nativa do Gemini mais sanitização de repetição de bootstrap e modo de saída de raciocínio marcada google, google-gemini-cli
    passthrough-gemini Sanitização de assinatura de pensamento do Gemini para modelos Gemini executados por transportes proxy compatíveis com OpenAI; não habilita validação de repetição nativa do Gemini nem reescritas de bootstrap openrouter, kilocode, opencode, opencode-go
    hybrid-anthropic-openai Política híbrida para provedores que misturam superfícies de modelo de mensagens da Anthropic e compatíveis com OpenAI em um Plugin; a remoção opcional de blocos de pensamento apenas do Claude permanece limitada ao lado Anthropic minimax

    Famílias de fluxo disponíveis hoje:

    Família O que ela conecta Exemplos incluídos
    google-thinking Normalização de payload de pensamento do Gemini no caminho de fluxo compartilhado google, google-gemini-cli
    kilocode-thinking Wrapper de raciocínio do Kilo no caminho de fluxo proxy compartilhado, com kilo/auto e ids de raciocínio proxy sem suporte ignorando pensamento injetado kilocode
    moonshot-thinking Mapeamento de payload de pensamento nativo binário do Moonshot a partir da configuração + nível /think moonshot
    minimax-fast-mode Reescrita de modelo de modo rápido MiniMax no caminho de fluxo compartilhado minimax, minimax-portal
    openai-responses-defaults Wrappers compartilhados nativos de Responses da OpenAI/Codex: cabeçalhos de atribuição, /fast/serviceTier, verbosidade de texto, pesquisa web nativa do Codex, formatação de payload de compatibilidade de raciocínio e gerenciamento de contexto de Responses openai, openai-codex
    openrouter-thinking Wrapper de raciocínio do OpenRouter para rotas proxy, com ignoros de modelo sem suporte/auto tratados centralmente openrouter
    tool-stream-default-on Wrapper tool_stream ativado por padrão para provedores como Z.AI que querem streaming de ferramenta salvo quando explicitamente desabilitado zai
    Interfaces SDK que alimentam os construtores de família

    Cada construtor de família é composto a partir de auxiliares públicos de nível mais baixo exportados pelo mesmo pacote, que você pode usar quando um provedor precisa sair do padrão comum:

    • openclaw/plugin-sdk/provider-model-shared - ProviderReplayFamily, buildProviderReplayFamilyHooks(...) e os construtores brutos de repetição (buildOpenAICompatibleReplayPolicy, buildAnthropicReplayPolicyForModel, buildGoogleGeminiReplayPolicy, buildHybridAnthropicOrOpenAIReplayPolicy). Também exporta auxiliares de repetição do Gemini (sanitizeGoogleGeminiReplayHistory, resolveTaggedReasoningOutputMode) e auxiliares de endpoint/modelo (resolveProviderEndpoint, normalizeProviderId, normalizeGooglePreviewModelId).
    • openclaw/plugin-sdk/provider-stream - ProviderStreamFamily, buildProviderStreamFamilyHooks(...), composeProviderStreamWrappers(...), mais os wrappers compartilhados da OpenAI/Codex (createOpenAIAttributionHeadersWrapper, createOpenAIFastModeWrapper, createOpenAIServiceTierWrapper, createOpenAIResponsesContextManagementWrapper, createCodexNativeWebSearchWrapper), wrapper compatível com OpenAI do DeepSeek V4 (createDeepSeekV4OpenAICompatibleThinkingWrapper), limpeza de preenchimento antecipado de pensamento de Anthropic Messages (createAnthropicThinkingPrefillPayloadWrapper) e wrappers compartilhados de proxy/provedor (createOpenRouterWrapper, createToolStreamWrapper, createMinimaxFastModeWrapper).
    • openclaw/plugin-sdk/provider-tools - ProviderToolCompatFamily, buildProviderToolCompatFamilyHooks("gemini") e auxiliares subjacentes de schema do Gemini (normalizeGeminiToolSchemas, inspectGeminiToolSchemas).

    Alguns auxiliares de fluxo permanecem locais ao provedor de propósito. @openclaw/anthropic-provider mantém wrapAnthropicProviderStream, resolveAnthropicBetas, resolveAnthropicFastMode, resolveAnthropicServiceTier e os construtores de wrapper Anthropic de nível mais baixo em sua própria interface pública api.ts / contract-api.ts porque eles codificam o tratamento beta de OAuth do Claude e o controle de context1m. O Plugin xAI, de modo semelhante, mantém a formatação nativa de Responses xAI em seu próprio wrapStreamFn (aliases de /fast, tool_stream padrão, limpeza de ferramenta estrita sem suporte, remoção de payload de raciocínio específica da xAI).

    O mesmo padrão de raiz de pacote também sustenta @openclaw/openai-provider (construtores de provedor, auxiliares de modelo padrão, construtores de provedor em tempo real) e @openclaw/openrouter-provider (construtor de provedor mais auxiliares de integração/configuração).

    Troca de token

    Para provedores que precisam de uma troca de token antes de cada chamada de inferência:

    typescript
    prepareRuntimeAuth: async (ctx) => {  const exchanged = await exchangeToken(ctx.apiKey);  return {    apiKey: exchanged.token,    baseUrl: exchanged.baseUrl,    expiresAt: exchanged.expiresAt,  };},

    Cabeçalhos personalizados

    Para provedores que precisam de cabeçalhos de requisição personalizados ou modificações no corpo:

    typescript
    // wrapStreamFn returns a StreamFn derived from ctx.streamFnwrapStreamFn: (ctx) => {  if (!ctx.streamFn) return undefined;  const inner = ctx.streamFn;  return async (params) => {    params.headers = {      ...params.headers,      "X-Acme-Version": "2",    };    return inner(params);  };},

    Identidade nativa do transporte

    Para provedores que precisam de cabeçalhos de requisição/sessão nativos ou metadados em transportes HTTP ou WebSocket genéricos:

    typescript
    resolveTransportTurnState: (ctx) => ({  headers: {    "x-request-id": ctx.turnId,  },  metadata: {    session_id: ctx.sessionId ?? "",    turn_id: ctx.turnId,  },}),resolveWebSocketSessionPolicy: (ctx) => ({  headers: {    "x-session-id": ctx.sessionId ?? "",  },  degradeCooldownMs: 60_000,}),

    Uso e cobrança

    Para provedores que expõem dados de uso/cobrança:

    typescript
    resolveUsageAuth: async (ctx) => {  const auth = await ctx.resolveOAuthToken();  return auth ? { token: auth.token } : null;},fetchUsageSnapshot: async (ctx) => {  return await fetchAcmeUsage(ctx.token, ctx.timeoutMs);},
    Todos os ganchos de provedor disponíveis

    OpenClaw chama os ganchos nesta ordem. A maioria dos provedores usa apenas 2-3: Campos de provedor somente para compatibilidade que o OpenClaw não chama mais, como ProviderPlugin.capabilities e suppressBuiltInModel, não estão listados aqui.

    # Gancho Quando usar
    1 catalog Catálogo de modelos ou padrões de URL base
    2 applyConfigDefaults Padrões globais pertencentes ao provedor durante a materialização da configuração
    3 normalizeModelId Limpeza de aliases legados/de pré-visualização de id de modelo antes da busca
    4 normalizeTransport Limpeza de api / baseUrl de família de provedor antes da montagem genérica do modelo
    5 normalizeConfig Normalizar configuração models.providers.<id>
    6 applyNativeStreamingUsageCompat Reescritas de compatibilidade de uso de streaming nativo para provedores de configuração
    7 resolveConfigApiKey Resolução de autenticação por marcador de env pertencente ao provedor
    8 resolveSyntheticAuth Autenticação sintética local/auto-hospedada ou apoiada por configuração
    9 shouldDeferSyntheticProfileAuth Rebaixar placeholders sintéticos de perfil armazenado atrás de autenticação por env/configuração
    10 resolveDynamicModel Aceitar ids arbitrários de modelos upstream
    11 prepareDynamicModel Busca assíncrona de metadados antes da resolução
    12 normalizeResolvedModel Reescritas de transporte antes do executor
    13 contributeResolvedModelCompat Flags de compatibilidade para modelos de fornecedor por trás de outro transporte compatível
    14 normalizeToolSchemas Limpeza de schema de ferramentas pertencente ao provedor antes do registro
    15 inspectToolSchemas Diagnósticos de schema de ferramentas pertencentes ao provedor
    16 resolveReasoningOutputMode Contrato de saída de raciocínio marcada vs nativa
    17 prepareExtraParams Parâmetros de requisição padrão
    18 createStreamFn Transporte StreamFn totalmente personalizado
    19 wrapStreamFn Wrappers personalizados de cabeçalhos/corpo no caminho de fluxo normal
    20 resolveTransportTurnState Cabeçalhos/metadados nativos por turno
    21 resolveWebSocketSessionPolicy Cabeçalhos/cool-down nativos de sessão WS
    22 formatApiKey Formato personalizado de token de runtime
    23 refreshOAuth Atualização OAuth personalizada
    24 buildAuthDoctorHint Orientação de reparo de autenticação
    25 matchesContextOverflowError Detecção de estouro pertencente ao provedor
    26 classifyFailoverReason Classificação de limite de taxa/sobrecarga pertencente ao provedor
    27 isCacheTtlEligible Controle de elegibilidade de TTL do cache de prompt
    28 buildMissingAuthMessage Dica personalizada de autenticação ausente
    29 augmentModelCatalog Linhas sintéticas de compatibilidade futura
    30 resolveThinkingProfile Conjunto de opções /think específico do modelo
    31 isBinaryThinking Compatibilidade de pensamento binário ligado/desligado
    32 supportsXHighThinking Compatibilidade com suporte de raciocínio xhigh
    33 resolveDefaultThinkingLevel Compatibilidade da política padrão de /think
    34 isModernModelRef Correspondência de modelo live/smoke
    35 prepareRuntimeAuth Troca de token antes da inferência
    36 resolveUsageAuth Análise personalizada de credenciais de uso
    37 fetchUsageSnapshot Endpoint de uso personalizado
    38 createEmbeddingProvider Adaptador de embeddings pertencente ao provedor para memória/pesquisa
    39 buildReplayPolicy Política personalizada de repetição/compaction de transcrição
    40 sanitizeReplayHistory Reescritas de repetição específicas do provedor após a limpeza genérica
    41 validateReplayTurns Validação estrita de turnos de repetição antes do executor incorporado
    42 onModelSelected Callback pós-seleção (por exemplo, telemetria)

    Observações sobre fallback de runtime:

    • normalizeConfig verifica primeiro o provedor correspondente, depois outros Plugins de provedor capazes de usar ganchos até que um realmente altere a configuração. Se nenhum gancho de provedor reescrever uma entrada de configuração da família Google compatível, o normalizador de configuração Google incluído ainda se aplica.
    • resolveConfigApiKey usa o gancho do provedor quando exposto. O caminho amazon-bedrock incluído também tem aqui um resolvedor integrado de marcador de env AWS, embora a autenticação de runtime do Bedrock em si ainda use a cadeia padrão do AWS SDK.
    • resolveSystemPromptContribution permite que um provedor injete orientação de prompt de sistema ciente de cache para uma família de modelos. Prefira-o a before_prompt_build quando o comportamento pertence a uma família de provedor/modelo e deve preservar a divisão estável/dinâmica do cache.

    Para descrições detalhadas e exemplos reais, consulte Internos: Ganchos de Runtime de Provedor.

  • Adicionar capacidades extras (opcional)

    Etapa 5: Adicionar capacidades extras

    Um Plugin de provedor pode registrar fala, transcrição em tempo real, voz em tempo real, compreensão de mídia, geração de imagens, geração de vídeo, busca de páginas web e pesquisa na web junto com inferência de texto. O OpenClaw classifica isso como um Plugin de capacidade híbrida - o padrão recomendado para Plugins de empresa (um Plugin por fornecedor). Veja Internos: propriedade de capacidade.

    Registre cada capacidade dentro de register(api) junto com sua chamada existente api.registerProvider(...). Escolha apenas as abas necessárias:

    Fala (TTS)

    typescript
    import {  assertOkOrThrowProviderError,  postJsonRequest,} from "openclaw/plugin-sdk/provider-http"; api.registerSpeechProvider({  id: "acme-ai",  label: "Acme Speech",  isConfigured: ({ config }) => Boolean(config.messages?.tts),  synthesize: async (req) => {    const { response, release } = await postJsonRequest({      url: "https://api.example.com/v1/speech",      headers: new Headers({ "Content-Type": "application/json" }),      body: { text: req.text },      timeoutMs: req.timeoutMs,      fetchFn: fetch,      auditContext: "acme speech",    });    try {      await assertOkOrThrowProviderError(response, "Acme Speech API error");      return {        audioBuffer: Buffer.from(await response.arrayBuffer()),        outputFormat: "mp3",        fileExtension: ".mp3",        voiceCompatible: false,      };    } finally {      await release();    }  },});

    Use assertOkOrThrowProviderError(...) para falhas HTTP do provedor, para que os Plugins compartilhem leituras limitadas do corpo de erro, análise de erros JSON e sufixos de ID de solicitação.

    Transcrição em tempo real

    Prefira createRealtimeTranscriptionWebSocketSession(...) - o helper compartilhado lida com captura de proxy, backoff de reconexão, liberação ao fechar, handshakes de pronto, enfileiramento de áudio e diagnósticos de eventos de fechamento. Seu Plugin apenas mapeia eventos upstream.

    typescript
    api.registerRealtimeTranscriptionProvider({  id: "acme-ai",  label: "Acme Realtime Transcription",  isConfigured: () => true,  createSession: (req) => {    const apiKey = String(req.providerConfig.apiKey ?? "");    return createRealtimeTranscriptionWebSocketSession({      providerId: "acme-ai",      callbacks: req,      url: "wss://api.example.com/v1/realtime-transcription",      headers: { Authorization: `Bearer ${apiKey}` },      onMessage: (event, transport) => {        if (event.type === "session.created") {          transport.sendJson({ type: "session.update" });          transport.markReady();          return;        }        if (event.type === "transcript.final") {          req.onTranscript?.(event.text);        }      },      sendAudio: (audio, transport) => {        transport.sendJson({          type: "audio.append",          audio: audio.toString("base64"),        });      },      onClose: (transport) => {        transport.sendJson({ type: "audio.end" });      },    });  },});

    Provedores de STT em lote que fazem POST de áudio multipart devem usar buildAudioTranscriptionFormData(...) de openclaw/plugin-sdk/provider-http. O helper normaliza nomes de arquivo de upload, incluindo uploads AAC que precisam de um nome de arquivo no estilo M4A para APIs de transcrição compatíveis.

    Voz em tempo real

    typescript
    api.registerRealtimeVoiceProvider({  id: "acme-ai",  label: "Acme Realtime Voice",  capabilities: {    transports: ["gateway-relay"],    inputAudioFormats: [{ encoding: "pcm16", sampleRateHz: 24000, channels: 1 }],    outputAudioFormats: [{ encoding: "pcm16", sampleRateHz: 24000, channels: 1 }],    supportsBargeIn: true,    supportsToolCalls: true,  },  isConfigured: ({ providerConfig }) => Boolean(providerConfig.apiKey),  createBridge: (req) => ({    // Set this only if the provider accepts multiple tool responses for    // one call, for example an immediate "working" response followed by    // the final result.    supportsToolResultContinuation: false,    connect: async () => {},    sendAudio: () => {},    setMediaTimestamp: () => {},    handleBargeIn: () => {},    submitToolResult: () => {},    acknowledgeMark: () => {},    close: () => {},    isConnected: () => true,  }),});

    Declare capabilities para que talk.catalog possa expor modos válidos, transportes, formatos de áudio e flags de recurso para clientes Talk de navegador e nativos. Implemente handleBargeIn quando um transporte puder detectar que uma pessoa está interrompendo a reprodução do assistente e o provedor oferecer suporte a truncar ou limpar a resposta de áudio ativa.

    Compreensão de mídia

    typescript
    api.registerMediaUnderstandingProvider({  id: "acme-ai",  capabilities: ["image", "audio"],  describeImage: async (req) => ({ text: "A photo of..." }),  transcribeAudio: async (req) => ({ text: "Transcript..." }),});

    Geração de imagens e vídeo

    Capacidades de vídeo usam um formato consciente de modo: generate, imageToVideo e videoToVideo. Campos agregados simples como maxInputImages / maxInputVideos / maxDurationSeconds não são suficientes para anunciar suporte a modo de transformação ou modos desativados de forma clara. A geração de música segue o mesmo padrão, com blocos generate / edit explícitos.

    typescript
    api.registerImageGenerationProvider({  id: "acme-ai",  label: "Acme Images",  generate: async (req) => ({ /* image result */ }),}); api.registerVideoGenerationProvider({  id: "acme-ai",  label: "Acme Video",  capabilities: {    generate: { maxVideos: 1, maxDurationSeconds: 10, supportsResolution: true },    imageToVideo: {      enabled: true,      maxVideos: 1,      maxInputImages: 1,      maxInputImagesByModel: { "acme/reference-to-video": 9 },      maxDurationSeconds: 5,    },    videoToVideo: { enabled: false },  },  generateVideo: async (req) => ({ videos: [] }),});

    Busca de páginas web e pesquisa

    typescript
    api.registerWebFetchProvider({  id: "acme-ai-fetch",  label: "Acme Fetch",  hint: "Fetch pages through Acme's rendering backend.",  envVars: ["ACME_FETCH_API_KEY"],  placeholder: "acme-...",  signupUrl: "https://acme.example.com/fetch",  credentialPath: "plugins.entries.acme.config.webFetch.apiKey",  getCredentialValue: (fetchConfig) => fetchConfig?.acme?.apiKey,  setCredentialValue: (fetchConfigTarget, value) => {    const acme = (fetchConfigTarget.acme ??= {});    acme.apiKey = value;  },  createTool: () => ({    description: "Fetch a page through Acme Fetch.",    parameters: {},    execute: async (args) => ({ content: [] }),  }),}); api.registerWebSearchProvider({  id: "acme-ai-search",  label: "Acme Search",  search: async (req) => ({ content: [] }),});
  • Testar

    Etapa 6: Testar

    src/provider.test.ts
    import { describe, it, expect } from "vitest";// Export your provider config object from index.ts or a dedicated fileimport { acmeProvider } from "./provider.js"; describe("acme-ai provider", () => {  it("resolves dynamic models", () => {    const model = acmeProvider.resolveDynamicModel!({      modelId: "acme-beta-v3",    } as any);    expect(model.id).toBe("acme-beta-v3");    expect(model.provider).toBe("acme-ai");  });   it("returns catalog when key is available", async () => {    const result = await acmeProvider.catalog!.run({      resolveProviderApiKey: () => ({ apiKey: "test-key" }),    } as any);    expect(result?.provider?.models).toHaveLength(2);  });   it("returns null catalog when no key", async () => {    const result = await acmeProvider.catalog!.run({      resolveProviderApiKey: () => ({ apiKey: undefined }),    } as any);    expect(result).toBeNull();  });});
  • Publicar no ClawHub

    Plugins de provedor são publicados da mesma forma que qualquer outro Plugin de código externo:

    bash
    clawhub package publish your-org/your-plugin --dry-runclawhub package publish your-org/your-plugin

    Não use o alias legado de publicação somente para skill aqui; pacotes de Plugin devem usar clawhub package publish.

    Estrutura de arquivos

    Code
    <bundled-plugin-root>/acme-ai/├── package.json              # openclaw.providers metadata├── openclaw.plugin.json      # Manifest with provider auth metadata├── index.ts                  # definePluginEntry + registerProvider└── src/    ├── provider.test.ts      # Tests    └── usage.ts              # Usage endpoint (optional)

    catalog.order controla quando seu catálogo é mesclado em relação aos provedores integrados:

    Ordem Quando Caso de uso
    simple Primeiro passo Provedores simples com chave de API
    profile Após simple Provedores condicionados a perfis de autenticação
    paired Após profile Sintetizar várias entradas relacionadas
    late Último passo Substituir provedores existentes (vence em colisão)

    Próximas etapas

    Relacionado

    Was this useful?