Set up and send emails via a user-provided Mailgun account (BYO email provider)
assistant skills install mailgun-setupSend emails through the user's own Mailgun account. The user provides their Mailgun API key and domain, and you send via their infrastructure.
Before starting, check whether Mailgun is already configured:
bun skills/mailgun-setup/scripts/check-config.ts
The script outputs JSON: { "configured": boolean, "hasApiKey": boolean, "hasWebhookKey": boolean, "details": string }.
configured is true — Mailgun is already set up. Offer to verify the connection or reconfigure.configured is false — continue to Step 1.Run the store script to securely collect the API key:
bun skills/mailgun-setup/scripts/store-api-key.ts
The script opens a secure credential prompt, stores the key, and exits. If it exits 0, the key is stored. Never ask for the key in chat.
Note: Mailgun uses HTTP Basic Auth with username api and the API key as the password. The credential proxy cannot construct Basic Auth headers automatically. Instead, use curl -u "api:$KEY" in bash commands — retrieve the key from the vault at runtime. See the sending examples below.
After storing the API key, automatically detect the user's domain — don't ask them for it. Retrieve the API key from the vault and call the Mailgun Domains API:
curl -s --user "api:$MAILGUN_API_KEY" \
https://api.mailgun.net/v4/domains?state=active
The response contains an items array of domain objects with name and state fields. Pick the first domain with "state": "active". If no active domains are found, try the EU endpoint (https://api.eu.mailgun.net/v4/domains?state=active). If still none, tell the user they need to verify a domain in their Mailgun dashboard first.
Use hi@<domain> as the default sender address. Remember the domain and region (US/EU) for future sends.
Confirm the API key works by checking the domain response returned in Step 2. A successful response (HTTP 200) with domain data confirms the connection.
If the user also wants to receive emails via Mailgun, run the webhook setup script:
bun skills/mailgun-setup/scripts/setup-webhook.ts --domain "<verified domain>" [--region eu]
This script:
If the script fails because no public base URL is configured (self-hosted only), load the public-ingress skill to walk the user through setting one up, then retry.
Summarize with the completed checklist:
"Setup complete!
✅ API key configured
✅ Domain detected: <domain> (region: US/EU)
✅ Connection verified
{webhook_line}
Default sender: hi@<domain>"
For {webhook_line}:
✅ Inbound route and webhook signing key configured⬜ Webhook — run setup again to enable inbound emailUse bash with curl to call the Mailgun API. Pass the API key via -u for Basic Auth:
curl -s --user "api:$MAILGUN_API_KEY" \
https://api.mailgun.net/v3/DOMAIN/messages \
-F from="Name <sender@example.com>" \
-F to="recipient@example.com" \
-F subject="Hello" \
-F text="Plain text body" \
-F html="<p>HTML body</p>"
Replace DOMAIN with the user's Mailgun sending domain.
| Parameter | Type | Required | Description |
|---|---|---|---|
from | string | ✅ | Sender address ("Name <email>" format) |
to | string | ✅ | Recipient(s), comma-separated for multiple |
subject | string | ✅ | Email subject |
text | string | Plain text body | |
html | string | HTML body | |
cc | string | CC recipients, comma-separated | |
bcc | string | BCC recipients, comma-separated | |
h:Reply-To | string | Reply-to address | |
h:In-Reply-To | string | Message-ID of parent (for threading) | |
h:References | string | Space-separated chain of ancestor Message-IDs |
To reply in a thread, include custom headers:
curl -s --user "api:$MAILGUN_API_KEY" \
https://api.mailgun.net/v3/DOMAIN/messages \
-F from="bot@example.com" \
-F to="user@example.com" \
-F subject="Re: Original subject" \
-F text="Reply body" \
-F "h:In-Reply-To=<original-message-id>" \
-F "h:References=<original-message-id>"
Success returns { "id": "<message-id>", "message": "Queued. Thank you." } with HTTP 200.
Errors return { "message": "error description" } with 4xx/5xx status.
Mailgun has US and EU regions:
https://api.mailgun.net/v3/DOMAIN/messageshttps://api.eu.mailgun.net/v3/DOMAIN/messagesAsk the user which region their account uses if sends fail with 401.
from address must be from the user's verified Mailgun domain.hi@<domain> — use this unless the user specifies otherwise.text for plain text, html for rich formatting. Provide both when possible.