Discord (Bot API)
Status: ready for DMs and guild channels via the official Discord gateway.
Pairing
Discord DMs default to pairing mode.
Slash commands
Native command behavior and command catalog.
Channel troubleshooting
Cross-channel diagnostics and repair flow.
Quick setup
You will need to create a new application with a bot, add the bot to your server, and pair it to Mayros. We recommend adding your bot to your own private server. If you don't have one yet, create one first (choose Create My Own > For me and my friends).
Create a Discord application and bot
Go to the Discord Developer Portal and click New Application. Name it something like "Mayros".
Click Bot on the sidebar. Set the Username to whatever you call your Mayros agent.
Enable privileged intents
Still on the Bot page, scroll down to Privileged Gateway Intents and enable:
- Message Content Intent (required)
- Server Members Intent (recommended; required for role allowlists and name-to-ID matching)
- Presence Intent (optional; only needed for presence updates)
Copy your bot token
Scroll back up on the Bot page and click Reset Token.
Despite the name, this generates your first token — nothing is being "reset."
Copy the token and save it somewhere. This is your Bot Token and you will need it shortly.
Generate an invite URL and add the bot to your server
Click OAuth2 on the sidebar. You'll generate an invite URL with the right permissions to add the bot to your server.
Scroll down to OAuth2 URL Generator and enable:
botapplications.commands
A Bot Permissions section will appear below. Enable:
- View Channels
- Send Messages
- Read Message History
- Embed Links
- Attach Files
- Add Reactions (optional)
Copy the generated URL at the bottom, paste it into your browser, select your server, and click Continue to connect. You should now see your bot in the Discord server.
Enable Developer Mode and collect your IDs
Back in the Discord app, you need to enable Developer Mode so you can copy internal IDs.
- Click User Settings (gear icon next to your avatar) → Advanced → toggle on Developer Mode
- Right-click your server icon in the sidebar → Copy Server ID
- Right-click your own avatar → Copy User ID
Save your Server ID and User ID alongside your Bot Token — you'll send all three to Mayros in the next step.
Allow DMs from server members
For pairing to work, Discord needs to allow your bot to DM you. Right-click your server icon → Privacy Settings → toggle on Direct Messages.
This lets server members (including bots) send you DMs. Keep this enabled if you want to use Discord DMs with Mayros. If you only plan to use guild channels, you can disable DMs after pairing.
Step 0: Set your bot token securely (do not send it in chat)
Your Discord bot token is a secret (like a password). Set it on the machine running Mayros before messaging your agent.
bashmayros config set channels.discord.token '"YOUR_BOT_TOKEN"' --json mayros config set channels.discord.enabled true --json mayros gateway
If Mayros is already running as a background service, use mayros gateway restart instead.
Configure Mayros and pair
Chat with your Mayros agent on any existing channel (e.g. Telegram) and tell it. If Discord is your first channel, use the CLI / config tab instead.
"I already set my Discord bot token in config. Please finish Discord setup with User ID
<user_id>and Server ID<server_id>."
Approve first DM pairing
Wait until the gateway is running, then DM your bot in Discord. It will respond with a pairing code.
Send the pairing code to your agent on your existing channel:
"Approve this Discord pairing code:
<CODE>"
Pairing codes expire after 1 hour.
You should now be able to chat with your agent in Discord via DM.
Token resolution is account-aware. Config token values win over env fallback. DISCORD_BOT_TOKEN is only used for the default account.
Recommended: Set up a guild workspace
Once DMs are working, you can set up your Discord server as a full workspace where each channel gets its own agent session with its own context. This is recommended for private servers where it's just you and your bot.
Add your server to the guild allowlist
This enables your agent to respond in any channel on your server, not just DMs.
"Add my Discord Server ID
<server_id>to the guild allowlist"
Allow responses without @mention
By default, your agent only responds in guild channels when @mentioned. For a private server, you probably want it to respond to every message.
"Allow my agent to respond on this server without having to be @mentioned"
Plan for memory in guild channels
By default, long-term memory (MEMORY.md) only loads in DM sessions. Guild channels do not auto-load MEMORY.md.
"When I ask questions in Discord channels, use memory_search or memory_get if you need long-term context from MEMORY.md."
Now create some channels on your Discord server and start chatting. Your agent can see the channel name, and each channel gets its own isolated session — so you can set up #coding, #home, #research, or whatever fits your workflow.
Runtime model
- Gateway owns the Discord connection.
- Reply routing is deterministic: Discord inbound replies back to Discord.
- By default (
session.dmScope=main), direct chats share the agent main session (agent:main:main). - Guild channels are isolated session keys (
agent:<agentId>:discord:channel:<channelId>). - Group DMs are ignored by default (
channels.discord.dm.groupEnabled=false). - Native slash commands run in isolated command sessions (
agent:<agentId>:discord:slash:<userId>), while still carryingCommandTargetSessionKeyto the routed conversation session.
Forum channels
Discord forum and media channels only accept thread posts. Mayros supports two ways to create them:
- Send a message to the forum parent (
channel:<forumId>) to auto-create a thread. The thread title uses the first non-empty line of your message. - Use
mayros message thread createto create a thread directly. Do not pass--message-idfor forum channels.
Example: send to forum parent to create a thread
bashmayros message send --channel discord --target channel:<forumId> \ --message "Topic title\nBody of the post"
Example: create a forum thread explicitly
bashmayros message thread create --channel discord --target channel:<forumId> \ --thread-name "Topic title" --message "Body of the post"
Forum parents do not accept Discord components. If you need components, send to the thread itself (channel:<threadId>).
Interactive components
Mayros supports Discord components v2 containers for agent messages. Use the message tool with a components payload. Interaction results are routed back to the agent as normal inbound messages and follow the existing Discord replyToMode settings.
Supported blocks:
text,section,separator,actions,media-gallery,file- Action rows allow up to 5 buttons or a single select menu
- Select types:
string,user,role,mentionable,channel
By default, components are single use. Set components.reusable=true to allow buttons, selects, and forms to be used multiple times until they expire.
To restrict who can click a button, set allowedUsers on that button (Discord user IDs, tags, or *). When configured, unmatched users receive an ephemeral denial.
The /model and /models slash commands open an interactive model picker with provider and model dropdowns plus a Submit step. The picker reply is ephemeral and only the invoking user can use it.
File attachments:
fileblocks must point to an attachment reference (attachment://<filename>)- Provide the attachment via
media/path/filePath(single file); usemedia-galleryfor multiple files - Use
filenameto override the upload name when it should match the attachment reference
Modal forms:
- Add
components.modalwith up to 5 fields - Field types:
text,checkbox,radio,select,role-select,user-select - Mayros adds a trigger button automatically
Example:
json5{ 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" }, ], }, ], }, }, }
Access control and routing
channels.discord.dmPolicy controls DM access (legacy: channels.discord.dm.policy):
pairing(default)allowlistopen(requireschannels.discord.allowFromto include"*"; legacy:channels.discord.dm.allowFrom)disabled
If DM policy is not open, unknown users are blocked (or prompted for pairing in pairing mode).
DM target format for delivery:
user:<id><@id>mention
Bare numeric IDs are ambiguous and rejected unless an explicit user/channel target kind is provided.
Role-based agent routing
Use bindings[].match.roles to route Discord guild members to different agents by role ID. Role-based bindings accept role IDs only and are evaluated after peer or parent-peer bindings and before guild-only bindings. If a binding also sets other match fields (for example peer + guildId + roles), all configured fields must match.
json5{ bindings: [ { agentId: "opus", match: { channel: "discord", guildId: "123456789012345678", roles: ["111111111111111111"], }, }, { agentId: "sonnet", match: { channel: "discord", guildId: "123456789012345678", }, }, ], }
Developer Portal setup
Native commands and command auth
commands.nativedefaults to"auto"and is enabled for Discord.- Per-channel override:
channels.discord.commands.native. commands.native=falseexplicitly clears previously registered Discord native commands.- Native command auth uses the same Discord allowlists/policies as normal message handling.
- Commands may still be visible in Discord UI for users who are not authorized; execution still enforces Mayros auth and returns "not authorized".
See Slash commands for command catalog and behavior.
Default slash command settings:
ephemeral: true
Feature details
Tools and action gates
Discord message actions include messaging, channel admin, moderation, presence, and metadata actions.
Core examples:
- messaging:
sendMessage,readMessages,editMessage,deleteMessage,threadReply - reactions:
react,reactions,emojiList - moderation:
timeout,kick,ban - presence:
setPresence
Action gates live under channels.discord.actions.*.
Default gate behavior:
| Action group | Default |
|---|---|
| reactions, messages, threads, pins, polls, search, memberInfo, roleInfo, channelInfo, channels, voiceStatus, events, stickers, emojiUploads, stickerUploads, permissions | enabled |
| roles | disabled |
| moderation | disabled |
| presence | disabled |
Components v2 UI
Mayros uses Discord components v2 for exec approvals and cross-context markers. Discord message actions can also accept components for custom UI (advanced; requires Carbon component instances), while legacy embeds remain available but are not recommended.
channels.discord.ui.components.accentColorsets the accent color used by Discord component containers (hex).- Set per account with
channels.discord.accounts.<id>.ui.components.accentColor. embedsare ignored when components v2 are present.
Example:
json5{ channels: { discord: { ui: { components: { accentColor: "#5865F2", }, }, }, }, }
Voice channels
Mayros can join Discord voice channels for realtime, continuous conversations. This is separate from voice message attachments.
Requirements:
- Enable native commands (
commands.nativeorchannels.discord.commands.native). - Configure
channels.discord.voice. - The bot needs Connect + Speak permissions in the target voice channel.
Use the Discord-only native command /vc join|leave|status to control sessions. The command uses the account default agent and follows the same allowlist and group policy rules as other Discord commands.
Auto-join example:
json5{ channels: { discord: { voice: { enabled: true, autoJoin: [ { guildId: "123456789012345678", channelId: "234567890123456789", }, ], tts: { provider: "openai", openai: { voice: "alloy" }, }, }, }, }, }
Notes:
voice.ttsoverridesmessages.ttsfor voice playback only.- Voice is enabled by default; set
channels.discord.voice.enabled=falseto disable it.
Voice messages
Discord voice messages show a waveform preview and require OGG/Opus audio plus metadata. Mayros generates the waveform automatically, but it needs ffmpeg and ffprobe available on the gateway host to inspect and convert audio files.
Requirements and constraints:
- Provide a local file path (URLs are rejected).
- Omit text content (Discord does not allow text + voice message in the same payload).
- Any audio format is accepted; Mayros converts to OGG/Opus when needed.
Example:
bashmessage(action="send", channel="discord", target="channel:123", path="/path/to/audio.mp3", asVoice=true)
Troubleshooting
Configuration reference pointers
Primary reference:
High-signal Discord fields:
- startup/auth:
enabled,token,accounts.*,allowBots - policy:
groupPolicy,dm.*,guilds.*,guilds.*.channels.* - command:
commands.native,commands.useAccessGroups,configWrites,slashCommand.* - reply/history:
replyToMode,historyLimit,dmHistoryLimit,dms.*.historyLimit - delivery:
textChunkLimit,chunkMode,maxLinesPerMessage - streaming:
streamMode,draftChunk,blockStreaming,blockStreamingCoalesce - media/retry:
mediaMaxMb,retry - actions:
actions.* - presence:
activity,status,activityType,activityUrl - UI:
ui.components.accentColor - features:
pluralkit,execApprovals,intents,agentComponents,heartbeat,responsePrefix
Safety and operations
- Treat bot tokens as secrets (
DISCORD_BOT_TOKENpreferred in supervised environments). - Grant least-privilege Discord permissions.
- If command deploy/state is stale, restart gateway and re-check with
mayros channels status --probe.