SMS (Twilio)
SMS Setup (Twilio)
Section titled “SMS Setup (Twilio)”Hermes connects to SMS through the Twilio API. People text your Twilio phone number and get AI responses back — same conversational experience as Telegram or Discord, but over standard text messages.
:::info Shared Credentials
The SMS gateway shares credentials with the optional telephony skill. If you’ve already set up Twilio for voice calls or one-off SMS, the gateway works with the same TWILIO_ACCOUNT_SID, TWILIO_AUTH_TOKEN, and TWILIO_PHONE_NUMBER.
:::
Prerequisites
Section titled “Prerequisites”- Twilio account — Sign up at twilio.com (free trial available)
- A Twilio phone number with SMS capability
- A publicly accessible server — Twilio sends webhooks to your server when SMS arrives
- aiohttp —
pip install 'hermes-agent[sms]'
Step 1: Get Your Twilio Credentials
Section titled “Step 1: Get Your Twilio Credentials”- Go to the Twilio Console
- Copy your Account SID and Auth Token from the dashboard
- Go to Phone Numbers → Manage → Active Numbers — note your phone number in E.164 format (e.g.,
+15551234567)
Step 2: Configure Hermes
Section titled “Step 2: Configure Hermes”Interactive setup (recommended)
Section titled “Interactive setup (recommended)”hermes gateway setupSelect SMS (Twilio) from the platform list. The wizard will prompt for your credentials.
Manual setup
Section titled “Manual setup”Add to ~/.hermes/.env:
TWILIO_ACCOUNT_SID=ACxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxTWILIO_AUTH_TOKEN=your_auth_token_hereTWILIO_PHONE_NUMBER=+15551234567
# Security: restrict to specific phone numbers (recommended)SMS_ALLOWED_USERS=+15559876543,+15551112222
# Optional: set a home channel for cron job deliverySMS_HOME_CHANNEL=+15559876543Step 3: Configure Twilio Webhook
Section titled “Step 3: Configure Twilio Webhook”Twilio needs to know where to send incoming messages. In the Twilio Console:
- Go to Phone Numbers → Manage → Active Numbers
- Click your phone number
- Under Messaging → A MESSAGE COMES IN, set:
- Webhook:
https://your-server:8080/webhooks/twilio - HTTP Method:
POST
- Webhook:
:::tip Exposing Your Webhook If you’re running Hermes locally, use a tunnel to expose the webhook:
# Using cloudflaredcloudflared tunnel --url http://localhost:8080
# Using ngrokngrok http 8080Set the resulting public URL as your Twilio webhook. :::
Set SMS_WEBHOOK_URL to the same URL you configured in Twilio. This is required for Twilio signature validation — the adapter will refuse to start without it:
# Must match the webhook URL in your Twilio ConsoleSMS_WEBHOOK_URL=https://your-server:8080/webhooks/twilioThe webhook port defaults to 8080. Override with:
SMS_WEBHOOK_PORT=3000Step 4: Start the Gateway
Section titled “Step 4: Start the Gateway”hermes gatewayYou should see:
[sms] Twilio webhook server listening on 0.0.0.0:8080, from: +1555***4567If you see Refusing to start: SMS_WEBHOOK_URL is required, set SMS_WEBHOOK_URL to the public URL configured in your Twilio Console (see Step 3).
Text your Twilio number — Hermes will respond via SMS.
Environment Variables
Section titled “Environment Variables”| Variable | Required | Description |
|---|---|---|
TWILIO_ACCOUNT_SID | Yes | Twilio Account SID (starts with AC) |
TWILIO_AUTH_TOKEN | Yes | Twilio Auth Token (also used for webhook signature validation) |
TWILIO_PHONE_NUMBER | Yes | Your Twilio phone number (E.164 format) |
SMS_WEBHOOK_URL | Yes | Public URL for Twilio signature validation — must match the webhook URL in your Twilio Console |
SMS_WEBHOOK_PORT | No | Webhook listener port (default: 8080) |
SMS_WEBHOOK_HOST | No | Webhook bind address (default: 0.0.0.0) |
SMS_INSECURE_NO_SIGNATURE | No | Set to true to disable signature validation (local dev only — not for production) |
SMS_ALLOWED_USERS | No | Comma-separated E.164 phone numbers allowed to chat |
SMS_ALLOW_ALL_USERS | No | Set to true to allow anyone (not recommended) |
SMS_HOME_CHANNEL | No | Phone number for cron job / notification delivery |
SMS_HOME_CHANNEL_NAME | No | Display name for the home channel (default: Home) |
SMS-Specific Behavior
Section titled “SMS-Specific Behavior”- Plain text only — Markdown is automatically stripped since SMS renders it as literal characters
- 1600 character limit — Longer responses are split across multiple messages at natural boundaries (newlines, then spaces)
- Echo prevention — Messages from your own Twilio number are ignored to prevent loops
- Phone number redaction — Phone numbers are redacted in logs for privacy
Security
Section titled “Security”Webhook signature validation
Section titled “Webhook signature validation”Hermes validates that inbound webhooks genuinely originate from Twilio by verifying the X-Twilio-Signature header (HMAC-SHA1). This prevents attackers from injecting forged messages.
SMS_WEBHOOK_URL is required. Set it to the public URL configured in your Twilio Console. The adapter will refuse to start without it.
For local development without a public URL, you can disable validation:
# Local dev only — NOT for productionSMS_INSECURE_NO_SIGNATURE=trueUser allowlists
Section titled “User allowlists”The gateway denies all users by default. Configure an allowlist:
# Recommended: restrict to specific phone numbersSMS_ALLOWED_USERS=+15559876543,+15551112222
# Or allow all (NOT recommended for bots with terminal access)SMS_ALLOW_ALL_USERS=trueSMS has no built-in encryption. Don’t use SMS for sensitive operations unless you understand the security implications. For sensitive use cases, prefer Signal or Telegram.
Troubleshooting
Section titled “Troubleshooting”Messages not arriving
Section titled “Messages not arriving”- Check your Twilio webhook URL is correct and publicly accessible
- Verify
TWILIO_ACCOUNT_SIDandTWILIO_AUTH_TOKENare correct - Check the Twilio Console → Monitor → Logs → Messaging for delivery errors
- Ensure your phone number is in
SMS_ALLOWED_USERS(orSMS_ALLOW_ALL_USERS=true)
Replies not sending
Section titled “Replies not sending”- Check
TWILIO_PHONE_NUMBERis set correctly (E.164 format with+) - Verify your Twilio account has SMS-capable numbers
- Check Hermes gateway logs for Twilio API errors
Webhook port conflicts
Section titled “Webhook port conflicts”If port 8080 is already in use, change it:
SMS_WEBHOOK_PORT=3001Update the webhook URL in Twilio Console to match.