Pular para o conteúdo principal

Discord (Bot API)

Status: pronto para DMs e canais de servidor via gateway oficial do Discord.

Pareamento

As DMs do Discord usam o modo de pareamento por padrão.

Comandos de barra

Comportamento nativo de comandos e catálogo de comandos.

Solução de problemas de canal

Diagnósticos e fluxo de reparo entre canais.

Configuração rápida

Você precisará criar um novo aplicativo com um bot, adicionar o bot ao seu servidor e pareá-lo com o OpenClaw. Recomendamos adicionar seu bot ao seu próprio servidor privado. Se você ainda não tiver um, crie um primeiro (escolha Create My Own > For me and my friends).
1

Criar um aplicativo e bot do Discord

Vá para o Discord Developer Portal e clique em New Application. Dê a ele um nome como “OpenClaw”.Clique em Bot na barra lateral. Defina o Username como o nome que você dá ao seu agente OpenClaw.
2

Habilitar intents privilegiadas

Ainda na página Bot, role para baixo até Privileged Gateway Intents e habilite:
  • Message Content Intent (obrigatória)
  • Server Members Intent (recomendada; obrigatória para allowlists de função e correspondência de nome para ID)
  • Presence Intent (opcional; necessária apenas para atualizações de presença)
3

Copiar o token do seu bot

Role de volta para cima na página Bot e clique em Reset Token.
Apesar do nome, isso gera seu primeiro token — nada está sendo “redefinido”.
Copie o token e salve-o em algum lugar. Este é o seu Bot Token e você precisará dele em breve.
4

Gerar uma URL de convite e adicionar o bot ao seu servidor

Clique em OAuth2 na barra lateral. Você vai gerar uma URL de convite com as permissões corretas para adicionar o bot ao seu servidor.Role para baixo até OAuth2 URL Generator e habilite:
  • bot
  • applications.commands
Uma seção Bot Permissions aparecerá abaixo. Habilite:
  • View Channels
  • Send Messages
  • Read Message History
  • Embed Links
  • Attach Files
  • Add Reactions (opcional)
Copie a URL gerada na parte inferior, cole-a no navegador, selecione seu servidor e clique em Continue para conectar. Agora você deve ver seu bot no servidor do Discord.
5

Habilitar o Modo de Desenvolvedor e coletar seus IDs

De volta ao app do Discord, você precisa habilitar o Modo de Desenvolvedor para poder copiar IDs internos.
  1. Clique em User Settings (ícone de engrenagem ao lado do seu avatar) → Advanced → ative Developer Mode
  2. Clique com o botão direito no ícone do seu servidor na barra lateral → Copy Server ID
  3. Clique com o botão direito no seu próprio avatarCopy User ID
Salve seu Server ID e User ID junto com seu Bot Token — você enviará os três ao OpenClaw na próxima etapa.
6

Permitir DMs de membros do servidor

Para que o pareamento funcione, o Discord precisa permitir que seu bot envie DMs para você. Clique com o botão direito no ícone do seu servidorPrivacy Settings → ative Direct Messages.Isso permite que membros do servidor (incluindo bots) enviem DMs para você. Mantenha isso ativado se quiser usar DMs do Discord com o OpenClaw. Se você pretende usar apenas canais do servidor, pode desativar as DMs após o pareamento.
7

Definir o token do seu bot com segurança (não o envie no chat)

O token do seu bot do Discord é um segredo (como uma senha). Defina-o na máquina que executa o OpenClaw antes de enviar mensagens ao seu agente.
export DISCORD_BOT_TOKEN="YOUR_BOT_TOKEN"
openclaw config set channels.discord.token --ref-provider default --ref-source env --ref-id DISCORD_BOT_TOKEN --dry-run
openclaw config set channels.discord.token --ref-provider default --ref-source env --ref-id DISCORD_BOT_TOKEN
openclaw config set channels.discord.enabled true --strict-json
openclaw gateway
Se o OpenClaw já estiver em execução como um serviço em segundo plano, reinicie-o pelo app OpenClaw para Mac ou parando e reiniciando o processo openclaw gateway run.
8

Configurar o OpenClaw e parear

Converse com seu agente OpenClaw em qualquer canal existente (por exemplo, Telegram) e diga a ele. Se o Discord for seu primeiro canal, use a aba CLI / config em vez disso.
“Já configurei meu token do bot do Discord na config. Conclua a configuração do Discord com o User ID <user_id> e o Server ID <server_id>.”
9

Aprovar o primeiro pareamento por DM

Aguarde até que o gateway esteja em execução e então envie uma DM para seu bot no Discord. Ele responderá com um código de pareamento.
Envie o código de pareamento ao seu agente no seu canal existente:
“Aprove este código de pareamento do Discord: <CODE>
Os códigos de pareamento expiram após 1 hora.Agora você deve conseguir conversar com seu agente no Discord via DM.
A resolução de token reconhece contas. Valores de token na config têm prioridade sobre o fallback de env. DISCORD_BOT_TOKEN é usado apenas para a conta padrão. Para chamadas avançadas de saída (ferramenta de mensagem/ações de canal), um token explícito por chamada é usado para aquela chamada. Isso se aplica a ações de envio e de leitura/sondagem (por exemplo, read/search/fetch/thread/pins/permissions). As configurações de política/retry da conta ainda vêm da conta selecionada no snapshot ativo do runtime.

Recomendado: configurar um workspace de servidor

Quando as DMs estiverem funcionando, você poderá configurar seu servidor do Discord como um workspace completo, em que cada canal recebe sua própria sessão de agente com seu próprio contexto. Isso é recomendado para servidores privados onde há apenas você e seu bot.
1

Adicionar seu servidor à allowlist de servidores

Isso permite que seu agente responda em qualquer canal do seu servidor, não apenas em DMs.
“Adicione meu Server ID do Discord <server_id> à allowlist de servidores”
2

Permitir respostas sem @mention

Por padrão, seu agente só responde em canais de servidor quando é mencionado com @. Em um servidor privado, você provavelmente vai querer que ele responda a todas as mensagens.
“Permita que meu agente responda neste servidor sem precisar ser mencionado com @”
3

Planejar memória em canais de servidor

Por padrão, a memória de longo prazo (MEMORY.md) só é carregada em sessões de DM. Canais de servidor não carregam MEMORY.md automaticamente.
“Quando eu fizer perguntas em canais do Discord, use memory_search ou memory_get se precisar de contexto de longo prazo do MEMORY.md.”
Agora crie alguns canais no seu servidor do Discord e comece a conversar. Seu agente pode ver o nome do canal, e cada canal recebe sua própria sessão isolada — assim, você pode configurar #coding, #home, #research ou o que melhor se encaixar no seu fluxo de trabalho.

Modelo de runtime

  • O gateway é responsável pela conexão com o Discord.
  • O roteamento de respostas é determinístico: respostas recebidas do Discord voltam para o Discord.
  • Por padrão (session.dmScope=main), chats diretos compartilham a sessão principal do agente (agent:main:main).
  • Canais de servidor usam chaves de sessão isoladas (agent:<agentId>:discord:channel:<channelId>).
  • DMs em grupo são ignoradas por padrão (channels.discord.dm.groupEnabled=false).
  • Comandos de barra nativos são executados em sessões de comando isoladas (agent:<agentId>:discord:slash:<userId>), enquanto ainda carregam CommandTargetSessionKey para a sessão de conversa roteada.

Canais de fórum

Canais de fórum e mídia do Discord aceitam apenas postagens em tópicos. O OpenClaw oferece suporte a duas formas de criá-los:
  • Envie uma mensagem para o pai do fórum (channel:<forumId>) para criar um tópico automaticamente. O título do tópico usa a primeira linha não vazia da sua mensagem.
  • Use openclaw message thread create para criar um tópico diretamente. Não passe --message-id para canais de fórum.
Exemplo: enviar ao pai do fórum para criar um tópico
openclaw message send --channel discord --target channel:<forumId> \
  --message "Topic title\nBody of the post"
Exemplo: criar explicitamente um tópico de fórum
openclaw message thread create --channel discord --target channel:<forumId> \
  --thread-name "Topic title" --message "Body of the post"
Pais de fórum não aceitam componentes do Discord. Se você precisar de componentes, envie para o próprio tópico (channel:<threadId>).

Componentes interativos

O OpenClaw oferece suporte a contêineres Discord components v2 para mensagens do agente. Use a ferramenta de mensagem com uma carga components. Os resultados de interação são roteados de volta ao agente como mensagens recebidas normais e seguem as configurações existentes de replyToMode do Discord. Blocos compatíveis:
  • text, section, separator, actions, media-gallery, file
  • Linhas de ação permitem até 5 botões ou um único menu de seleção
  • Tipos de seleção: string, user, role, mentionable, channel
Por padrão, componentes são de uso único. Defina components.reusable=true para permitir que botões, seleções e formulários sejam usados várias vezes até expirarem. Para restringir quem pode clicar em um botão, defina allowedUsers nesse botão (IDs de usuário do Discord, tags ou *). Quando configurado, usuários sem correspondência recebem uma negação efêmera. Os comandos de barra /model e /models abrem um seletor de modelo interativo com menus suspensos de provider e modelo, além de uma etapa Submit. A resposta do seletor é efêmera e apenas o usuário que a invocou pode usá-la. Anexos de arquivo:
  • Blocos file devem apontar para uma referência de anexo (attachment://<filename>)
  • Forneça o anexo via media/path/filePath (arquivo único); use media-gallery para vários arquivos
  • Use filename para substituir o nome do upload quando ele precisar corresponder à referência de anexo
Formulários modais:
  • Adicione components.modal com até 5 campos
  • Tipos de campo: text, checkbox, radio, select, role-select, user-select
  • O OpenClaw adiciona automaticamente um botão de gatilho
Exemplo:
{
  channel: "discord",
  action: "send",
  to: "channel:123456789012345678",
  message: "Optional fallback text",
  components: {
    reusable: true,
    text: "Choose a path",
    blocks: [
      {
        type: "actions",
        buttons: [
          {
            label: "Approve",
            style: "success",
            allowedUsers: ["123456789012345678"],
          },
          { label: "Decline", style: "danger" },
        ],
      },
      {
        type: "actions",
        select: {
          type: "string",
          placeholder: "Pick an option",
          options: [
            { label: "Option A", value: "a" },
            { label: "Option B", value: "b" },
          ],
        },
      },
    ],
    modal: {
      title: "Details",
      triggerLabel: "Open form",
      fields: [
        { type: "text", label: "Requester" },
        {
          type: "select",
          label: "Priority",
          options: [
            { label: "Low", value: "low" },
            { label: "High", value: "high" },
          ],
        },
      ],
    },
  },
}

Controle de acesso e roteamento

channels.discord.dmPolicy controla o acesso por DM (legado: channels.discord.dm.policy):
  • pairing (padrão)
  • allowlist
  • open (exige que channels.discord.allowFrom inclua "*"; legado: channels.discord.dm.allowFrom)
  • disabled
Se a política de DM não for open, usuários desconhecidos serão bloqueados (ou convidados a parear no modo pairing).Precedência com múltiplas contas:
  • channels.discord.accounts.default.allowFrom se aplica apenas à conta default.
  • Contas nomeadas herdam channels.discord.allowFrom quando o próprio allowFrom não está definido.
  • Contas nomeadas não herdam channels.discord.accounts.default.allowFrom.
Formato do alvo de DM para entrega:
  • user:<id>
  • menção <@id>
IDs numéricos sem qualificador são ambíguos e rejeitados, a menos que um tipo explícito de alvo user/channel seja fornecido.

Roteamento de agente baseado em função

Use bindings[].match.roles para rotear membros de servidores do Discord para agentes diferentes por ID de função. Bindings baseados em função aceitam apenas IDs de função e são avaliados após bindings de peer ou parent-peer e antes de bindings somente de servidor. Se um binding também definir outros campos de correspondência (por exemplo peer + guildId + roles), todos os campos configurados devem corresponder.
{
  bindings: [
    {
      agentId: "opus",
      match: {
        channel: "discord",
        guildId: "123456789012345678",
        roles: ["111111111111111111"],
      },
    },
    {
      agentId: "sonnet",
      match: {
        channel: "discord",
        guildId: "123456789012345678",
      },
    },
  ],
}

Configuração do Developer Portal

  1. Discord Developer Portal -> Applications -> New Application
  2. Bot -> Add Bot
  3. Copie o token do bot
Em Bot -> Privileged Gateway Intents, habilite:
  • Message Content Intent
  • Server Members Intent (recomendado)
Presence intent é opcional e necessária apenas se você quiser receber atualizações de presença. Definir a presença do bot (setPresence) não exige habilitar atualizações de presença para membros.
Gerador de URL OAuth:
  • escopos: bot, applications.commands
Permissões típicas de linha de base:
  • View Channels
  • Send Messages
  • Read Message History
  • Embed Links
  • Attach Files
  • Add Reactions (opcional)
Evite Administrator, a menos que seja explicitamente necessário.
Habilite o Discord Developer Mode e então copie:
  • ID do servidor
  • ID do canal
  • ID do usuário
Prefira IDs numéricos na config do OpenClaw para auditorias e sondagens confiáveis.

Comandos nativos e autenticação de comandos

  • commands.native usa "auto" por padrão e está habilitado para Discord.
  • Substituição por canal: channels.discord.commands.native.
  • commands.native=false remove explicitamente comandos nativos do Discord registrados anteriormente.
  • A autenticação de comandos nativos usa as mesmas allowlists/políticas do Discord que o tratamento normal de mensagens.
  • Os comandos ainda podem ficar visíveis na interface do Discord para usuários não autorizados; a execução ainda aplica a autenticação do OpenClaw e retorna “not authorized”.
Consulte Comandos de barra para catálogo e comportamento de comandos. Configurações padrão dos comandos de barra:
  • ephemeral: true

Detalhes do recurso

O Discord oferece suporte a tags de resposta na saída do agente:
  • [[reply_to_current]]
  • [[reply_to:<id>]]
Controlado por channels.discord.replyToMode:
  • off (padrão)
  • first
  • all
  • batched
Observação: off desativa o encadeamento implícito de respostas. Tags explícitas [[reply_to_*]] ainda são respeitadas. first sempre anexa a referência implícita de resposta nativa à primeira mensagem enviada pelo Discord no turno. batched só anexa a referência implícita de resposta nativa do Discord quando o turno de entrada foi um lote com debounce de várias mensagens. Isso é útil quando você quer respostas nativas principalmente para chats ambíguos e intensos, não para todo turno de mensagem única.IDs de mensagem são expostos no contexto/histórico para que agentes possam direcionar mensagens específicas.
O OpenClaw pode transmitir respostas em rascunho enviando uma mensagem temporária e editando-a à medida que o texto chega.
  • channels.discord.streaming controla a transmissão de prévia (off | partial | block | progress, padrão: off).
  • O padrão continua sendo off porque edições de prévia no Discord podem atingir limites de taxa rapidamente, especialmente quando vários bots ou gateways compartilham a mesma conta ou tráfego do servidor.
  • progress é aceito para consistência entre canais e mapeia para partial no Discord.
  • channels.discord.streamMode é um alias legado e é migrado automaticamente.
  • partial edita uma única mensagem de prévia à medida que os tokens chegam.
  • block emite blocos do tamanho de rascunho (use draftChunk para ajustar tamanho e pontos de quebra).
Exemplo:
{
  channels: {
    discord: {
      streaming: "partial",
    },
  },
}
Padrões de fragmentação do modo block (limitados por channels.discord.textChunkLimit):
{
  channels: {
    discord: {
      streaming: "block",
      draftChunk: {
        minChars: 200,
        maxChars: 800,
        breakPreference: "paragraph",
      },
    },
  },
}
A transmissão de prévia é somente texto; respostas com mídia recorrem à entrega normal.Observação: a transmissão de prévia é separada da transmissão em blocos. Quando a transmissão em blocos é explicitamente habilitada para o Discord, o OpenClaw ignora a stream de prévia para evitar transmissão dupla.
Contexto de histórico do servidor:
  • channels.discord.historyLimit padrão 20
  • fallback: messages.groupChat.historyLimit
  • 0 desativa
Controles de histórico de DM:
  • channels.discord.dmHistoryLimit
  • channels.discord.dms["<user_id>"].historyLimit
Comportamento de tópicos:
  • tópicos do Discord são roteados como sessões de canal
  • metadados do tópico pai podem ser usados para vínculo com a sessão pai
  • a config do tópico herda a config do canal pai, a menos que exista uma entrada específica para o tópico
Os tópicos de canal são injetados como contexto não confiável (não como prompt do sistema). O contexto de resposta e de mensagem citada atualmente permanece como recebido. Allowlists do Discord controlam principalmente quem pode acionar o agente, e não um limite completo de redação de contexto suplementar.
O Discord pode vincular um tópico a um alvo de sessão para que mensagens de acompanhamento nesse tópico continuem sendo roteadas para a mesma sessão (incluindo sessões de subagente).Comandos:
  • /focus <target> vincula o tópico atual/novo a um alvo de subagente/sessão
  • /unfocus remove o vínculo do tópico atual
  • /agents mostra execuções ativas e o estado do vínculo
  • /session idle <duration|off> inspeciona/atualiza o desfoco automático por inatividade para vínculos focados
  • /session max-age <duration|off> inspeciona/atualiza a idade máxima rígida para vínculos focados
Config:
{
  session: {
    threadBindings: {
      enabled: true,
      idleHours: 24,
      maxAgeHours: 0,
    },
  },
  channels: {
    discord: {
      threadBindings: {
        enabled: true,
        idleHours: 24,
        maxAgeHours: 0,
        spawnSubagentSessions: false, // opt-in
      },
    },
  },
}
Observações:
  • session.threadBindings.* define padrões globais.
  • channels.discord.threadBindings.* substitui o comportamento do Discord.
  • spawnSubagentSessions deve ser true para criar/vincular automaticamente tópicos para sessions_spawn({ thread: true }).
  • spawnAcpSessions deve ser true para criar/vincular automaticamente tópicos para ACP (/acp spawn ... --thread ... ou sessions_spawn({ runtime: "acp", thread: true })).
  • Se os vínculos de tópico estiverem desabilitados para uma conta, /focus e operações relacionadas de vínculo de tópico ficarão indisponíveis.
Consulte Sub-agents, ACP Agents e Referência de configuração.
Para workspaces ACP estáveis e “sempre ativos”, configure vínculos ACP tipados no nível superior direcionados a conversas do Discord.Caminho de config:
  • bindings[] com type: "acp" e match.channel: "discord"
Exemplo:
{
  agents: {
    list: [
      {
        id: "codex",
        runtime: {
          type: "acp",
          acp: {
            agent: "codex",
            backend: "acpx",
            mode: "persistent",
            cwd: "/workspace/openclaw",
          },
        },
      },
    ],
  },
  bindings: [
    {
      type: "acp",
      agentId: "codex",
      match: {
        channel: "discord",
        accountId: "default",
        peer: { kind: "channel", id: "222222222222222222" },
      },
      acp: { label: "codex-main" },
    },
  ],
  channels: {
    discord: {
      guilds: {
        "111111111111111111": {
          channels: {
            "222222222222222222": {
              requireMention: false,
            },
          },
        },
      },
    },
  },
}
Observações:
  • /acp spawn codex --bind here vincula o canal ou tópico atual do Discord no local e mantém mensagens futuras roteadas para a mesma sessão ACP.
  • Isso ainda pode significar “iniciar uma nova sessão ACP do Codex”, mas não cria um novo tópico do Discord por si só. O canal existente continua sendo a superfície de chat.
  • O Codex ainda pode ser executado em seu próprio cwd ou workspace de backend em disco. Esse workspace é estado de runtime, não um tópico do Discord.
  • Mensagens do tópico podem herdar o vínculo ACP do canal pai.
  • Em um canal ou tópico vinculado, /new e /reset redefinem a mesma sessão ACP no local.
  • Vínculos temporários de tópico ainda funcionam e podem substituir a resolução do alvo enquanto ativos.
  • spawnAcpSessions só é necessário quando o OpenClaw precisa criar/vincular um tópico filho via --thread auto|here. Não é necessário para /acp spawn ... --bind here no canal atual.
Consulte ACP Agents para detalhes sobre o comportamento de vínculo.
Modo de notificação de reação por servidor:
  • off
  • own (padrão)
  • all
  • allowlist (usa guilds.<id>.users)
Eventos de reação são transformados em eventos do sistema e anexados à sessão do Discord roteada.
ackReaction envia um emoji de confirmação enquanto o OpenClaw está processando uma mensagem recebida.Ordem de resolução:
  • channels.discord.accounts.<accountId>.ackReaction
  • channels.discord.ackReaction
  • messages.ackReaction
  • fallback para emoji de identidade do agente (agents.list[].identity.emoji, senão ”👀”)
Observações:
  • O Discord aceita emoji unicode ou nomes de emoji personalizados.
  • Use "" para desativar a reação para um canal ou conta.
Gravações de config iniciadas pelo canal são habilitadas por padrão.Isso afeta os fluxos /config set|unset (quando os recursos de comando estão habilitados).Desativar:
{
  channels: {
    discord: {
      configWrites: false,
    },
  },
}
Roteie o tráfego WebSocket do gateway do Discord e as consultas REST de inicialização (ID do aplicativo + resolução de allowlist) por um proxy HTTP(S) com channels.discord.proxy.
{
  channels: {
    discord: {
      proxy: "http://proxy.example:8080",
    },
  },
}
Substituição por conta:
{
  channels: {
    discord: {
      accounts: {
        primary: {
          proxy: "http://proxy.example:8080",
        },
      },
    },
  },
}
Habilite a resolução do PluralKit para mapear mensagens com proxy para a identidade do membro do sistema:
{
  channels: {
    discord: {
      pluralkit: {
        enabled: true,
        token: "pk_live_...", // optional; needed for private systems
      },
    },
  },
}
Observações:
  • allowlists podem usar pk:<memberId>
  • nomes de exibição de membro são correspondidos por nome/slug apenas quando channels.discord.dangerouslyAllowNameMatching: true
  • buscas usam o ID da mensagem original e são restritas por janela de tempo
  • se a busca falhar, mensagens com proxy são tratadas como mensagens de bot e descartadas, a menos que allowBots=true
Atualizações de presença são aplicadas quando você define um campo de status ou atividade, ou quando habilita presença automática.Exemplo apenas de status:
{
  channels: {
    discord: {
      status: "idle",
    },
  },
}
Exemplo de atividade (status personalizado é o tipo de atividade padrão):
{
  channels: {
    discord: {
      activity: "Focus time",
      activityType: 4,
    },
  },
}
Exemplo de streaming:
{
  channels: {
    discord: {
      activity: "Live coding",
      activityType: 1,
      activityUrl: "https://twitch.tv/openclaw",
    },
  },
}
Mapa de tipo de atividade:
  • 0: Playing
  • 1: Streaming (exige activityUrl)
  • 2: Listening
  • 3: Watching
  • 4: Custom (usa o texto da atividade como estado do status; emoji é opcional)
  • 5: Competing
Exemplo de presença automática (sinal de integridade do runtime):
{
  channels: {
    discord: {
      autoPresence: {
        enabled: true,
        intervalMs: 30000,
        minUpdateIntervalMs: 15000,
        exhaustedText: "token exhausted",
      },
    },
  },
}
A presença automática mapeia a disponibilidade do runtime para o status do Discord: saudável => online, degradado ou desconhecido => idle, esgotado ou indisponível => dnd. Substituições opcionais de texto:
  • autoPresence.healthyText
  • autoPresence.degradedText
  • autoPresence.exhaustedText (oferece suporte ao placeholder {reason})
O Discord oferece suporte ao tratamento de aprovações com botões em DMs e pode opcionalmente publicar prompts de aprovação no canal de origem.Caminho de config:
  • channels.discord.execApprovals.enabled
  • channels.discord.execApprovals.approvers (opcional; usa fallback para commands.ownerAllowFrom quando possível)
  • channels.discord.execApprovals.target (dm | channel | both, padrão: dm)
  • agentFilter, sessionFilter, cleanupAfterResolve
O Discord habilita automaticamente aprovações exec nativas quando enabled não está definido ou é "auto" e pelo menos um aprovador pode ser resolvido, seja a partir de execApprovals.approvers ou de commands.ownerAllowFrom. O Discord não infere aprovadores exec a partir de allowFrom do canal, dm.allowFrom legado ou defaultTo de mensagem direta. Defina enabled: false para desabilitar explicitamente o Discord como cliente nativo de aprovação.Quando target é channel ou both, o prompt de aprovação fica visível no canal. Apenas aprovadores resolvidos podem usar os botões; outros usuários recebem uma negação efêmera. Prompts de aprovação incluem o texto do comando, então habilite a entrega no canal apenas em canais confiáveis. Se o ID do canal não puder ser derivado da chave de sessão, o OpenClaw recorre à entrega por DM.O Discord também renderiza os botões de aprovação compartilhados usados por outros canais de chat. O adaptador nativo do Discord adiciona principalmente roteamento de DM para aprovadores e fanout para canal. Quando esses botões estão presentes, eles são a UX principal de aprovação; o OpenClaw só deve incluir um comando manual /approve quando o resultado da ferramenta disser que aprovações via chat não estão disponíveis ou que a aprovação manual é o único caminho.A autenticação do gateway para esse manipulador usa o mesmo contrato compartilhado de resolução de credenciais que outros clientes do gateway:
  • autenticação local com prioridade para env (OPENCLAW_GATEWAY_TOKEN / OPENCLAW_GATEWAY_PASSWORD e então gateway.auth.*)
  • no modo local, gateway.remote.* pode ser usado como fallback somente quando gateway.auth.* não estiver definido; SecretRefs locais configurados, mas não resolvidos, falham fechados
  • suporte a modo remoto via gateway.remote.* quando aplicável
  • substituições de URL são seguras para override: substituições de CLI não reutilizam credenciais implícitas, e substituições de env usam apenas credenciais de env
Comportamento de resolução de aprovação:
  • IDs prefixados com plugin: são resolvidos por plugin.approval.resolve.
  • Outros IDs são resolvidos por exec.approval.resolve.
  • O Discord não faz aqui um salto extra de fallback de exec para plugin; o prefixo do ID decide qual método do gateway ele chama.
Aprovações exec expiram após 30 minutos por padrão. Se aprovações falharem com IDs de aprovação desconhecidos, verifique a resolução de aprovador, a habilitação do recurso e se o tipo de ID de aprovação entregue corresponde à solicitação pendente.Documentação relacionada: Aprovações exec

Ferramentas e gates de ação

As ações de mensagem do Discord incluem mensagens, administração de canal, moderação, presença e ações de metadados. Exemplos principais:
  • mensagens: sendMessage, readMessages, editMessage, deleteMessage, threadReply
  • reações: react, reactions, emojiList
  • moderação: timeout, kick, ban
  • presença: setPresence
A ação event-create aceita um parâmetro opcional image (URL ou caminho de arquivo local) para definir a imagem de capa do evento agendado. Os gates de ação ficam em channels.discord.actions.*. Comportamento padrão dos gates:
Grupo de açãoPadrão
reactions, messages, threads, pins, polls, search, memberInfo, roleInfo, channelInfo, channels, voiceStatus, events, stickers, emojiUploads, stickerUploads, permissionshabilitado
rolesdesabilitado
moderationdesabilitado
presencedesabilitado

UI de Components v2

O OpenClaw usa Discord components v2 para aprovações exec e marcadores de contexto cruzado. Ações de mensagem do Discord também podem aceitar components para UI personalizada (avançado; requer construir uma carga de componente via ferramenta do Discord), enquanto embeds legados continuam disponíveis, mas não são recomendados.
  • channels.discord.ui.components.accentColor define a cor de destaque usada por contêineres de componentes do Discord (hex).
  • Defina por conta com channels.discord.accounts.<id>.ui.components.accentColor.
  • embeds são ignorados quando components v2 estão presentes.
Exemplo:
{
  channels: {
    discord: {
      ui: {
        components: {
          accentColor: "#5865F2",
        },
      },
    },
  },
}

Canais de voz

O OpenClaw pode entrar em canais de voz do Discord para conversas contínuas em tempo real. Isso é separado de anexos de mensagens de voz. Requisitos:
  • Habilite comandos nativos (commands.native ou channels.discord.commands.native).
  • Configure channels.discord.voice.
  • O bot precisa de permissões Connect + Speak no canal de voz alvo.
Use o comando nativo exclusivo do Discord /vc join|leave|status para controlar sessões. O comando usa o agente padrão da conta e segue as mesmas regras de allowlist e group policy que outros comandos do Discord. Exemplo de entrada automática:
{
  channels: {
    discord: {
      voice: {
        enabled: true,
        autoJoin: [
          {
            guildId: "123456789012345678",
            channelId: "234567890123456789",
          },
        ],
        daveEncryption: true,
        decryptionFailureTolerance: 24,
        tts: {
          provider: "openai",
          openai: { voice: "alloy" },
        },
      },
    },
  },
}
Observações:
  • voice.tts substitui messages.tts apenas para reprodução por voz.
  • Turnos de transcrição de voz derivam o status de owner de allowFrom do Discord (ou dm.allowFrom); falantes que não são owners não podem acessar ferramentas exclusivas de owner (por exemplo gateway e cron).
  • Voz é habilitada por padrão; defina channels.discord.voice.enabled=false para desabilitá-la.
  • voice.daveEncryption e voice.decryptionFailureTolerance são repassados para as opções de entrada de @discordjs/voice.
  • Os padrões de @discordjs/voice são daveEncryption=true e decryptionFailureTolerance=24 se não estiverem definidos.
  • O OpenClaw também monitora falhas de descriptografia na recepção e faz recuperação automática saindo e entrando novamente no canal de voz após falhas repetidas em uma janela curta.
  • Se os logs de recepção mostrarem repetidamente DecryptionFailed(UnencryptedWhenPassthroughDisabled), isso pode ser o bug de recepção upstream de @discordjs/voice rastreado em discord.js #11419.

Mensagens de voz

Mensagens de voz do Discord exibem uma prévia de forma de onda e exigem áudio OGG/Opus mais metadados. O OpenClaw gera a forma de onda automaticamente, mas precisa de ffmpeg e ffprobe disponíveis no host do gateway para inspecionar e converter arquivos de áudio. Requisitos e restrições:
  • Forneça um caminho de arquivo local (URLs são rejeitadas).
  • Omita conteúdo de texto (o Discord não permite texto + mensagem de voz na mesma carga).
  • Qualquer formato de áudio é aceito; o OpenClaw converte para OGG/Opus quando necessário.
Exemplo:
message(action="send", channel="discord", target="channel:123", path="/path/to/audio.mp3", asVoice=true)

Solução de problemas

  • habilite Message Content Intent
  • habilite Server Members Intent quando você depender de resolução de usuário/membro
  • reinicie o gateway após alterar as intents
  • verifique groupPolicy
  • verifique a allowlist de servidor em channels.discord.guilds
  • se existir o mapa guild channels, apenas os canais listados serão permitidos
  • verifique o comportamento de requireMention e os padrões de menção
Verificações úteis:
openclaw doctor
openclaw channels status --probe
openclaw logs --follow
Causas comuns:
  • groupPolicy="allowlist" sem allowlist correspondente de servidor/canal
  • requireMention configurado no lugar errado (deve ficar em channels.discord.guilds ou na entrada do canal)
  • remetente bloqueado pela allowlist users do servidor/canal
Logs típicos:
  • Listener DiscordMessageListener timed out after 30000ms for event MESSAGE_CREATE
  • Slow listener detected ...
  • discord inbound worker timed out after ...
Ajuste de orçamento do listener:
  • conta única: channels.discord.eventQueue.listenerTimeout
  • múltiplas contas: channels.discord.accounts.<accountId>.eventQueue.listenerTimeout
Ajuste de timeout da execução do worker:
  • conta única: channels.discord.inboundWorker.runTimeoutMs
  • múltiplas contas: channels.discord.accounts.<accountId>.inboundWorker.runTimeoutMs
  • padrão: 1800000 (30 minutos); defina 0 para desabilitar
Linha de base recomendada:
{
  channels: {
    discord: {
      accounts: {
        default: {
          eventQueue: {
            listenerTimeout: 120000,
          },
          inboundWorker: {
            runTimeoutMs: 1800000,
          },
        },
      },
    },
  },
}
Use eventQueue.listenerTimeout para configuração lenta do listener e inboundWorker.runTimeoutMs apenas se você quiser uma válvula de segurança separada para turnos de agente enfileirados.
Verificações de permissão em channels status --probe funcionam apenas para IDs numéricos de canal.Se você usar chaves slug, a correspondência em runtime ainda pode funcionar, mas a sondagem não consegue verificar permissões completamente.
  • DM desabilitada: channels.discord.dm.enabled=false
  • política de DM desabilitada: channels.discord.dmPolicy="disabled" (legado: channels.discord.dm.policy)
  • aguardando aprovação de pareamento no modo pairing
Por padrão, mensagens criadas por bot são ignoradas.Se você definir channels.discord.allowBots=true, use regras estritas de menção e allowlist para evitar comportamento de loop. Prefira channels.discord.allowBots="mentions" para aceitar apenas mensagens de bot que mencionem o bot.
  • mantenha o OpenClaw atualizado (openclaw update) para que a lógica de recuperação de recepção de voz do Discord esteja presente
  • confirme channels.discord.voice.daveEncryption=true (padrão)
  • comece com channels.discord.voice.decryptionFailureTolerance=24 (padrão upstream) e ajuste somente se necessário
  • monitore logs para:
    • discord voice: DAVE decrypt failures detected
    • discord voice: repeated decrypt failures; attempting rejoin
  • se as falhas continuarem após a reentrada automática, colete logs e compare com discord.js #11419

Ponteiros para a referência de configuração

Referência principal: Campos do Discord de alto sinal:
  • inicialização/autenticação: enabled, token, accounts.*, allowBots
  • política: groupPolicy, dm.*, guilds.*, guilds.*.channels.*
  • comando: commands.native, commands.useAccessGroups, configWrites, slashCommand.*
  • fila de eventos: eventQueue.listenerTimeout (orçamento do listener), eventQueue.maxQueueSize, eventQueue.maxConcurrency
  • worker de entrada: inboundWorker.runTimeoutMs
  • resposta/histórico: replyToMode, historyLimit, dmHistoryLimit, dms.*.historyLimit
  • entrega: textChunkLimit, chunkMode, maxLinesPerMessage
  • streaming: streaming (alias legado: streamMode), draftChunk, blockStreaming, blockStreamingCoalesce
  • mídia/retry: mediaMaxMb, retry
    • mediaMaxMb limita uploads de saída do Discord (padrão: 100MB)
  • ações: actions.*
  • presença: activity, status, activityType, activityUrl
  • UI: ui.components.accentColor
  • recursos: threadBindings, bindings[] de nível superior (type: "acp"), pluralkit, execApprovals, intents, agentComponents, heartbeat, responsePrefix

Segurança e operações

  • Trate tokens de bot como segredos (DISCORD_BOT_TOKEN preferido em ambientes supervisionados).
  • Conceda permissões do Discord com privilégio mínimo.
  • Se o estado/implantação de comandos estiver desatualizado, reinicie o gateway e verifique novamente com openclaw channels status --probe.

Relacionado