Install
openclaw skills install signal-messenger-standaloneFull Signal messenger integration for OpenClaw agents. Send/receive text and voice messages via signal-cli with role-based permissions (owner/trusted/untrusted), automatic voice transcription (Whisper), voice replies (TTS), conversation history, new contact triage, typing indicators, read receipts, and instant wake-on-message via OpenClaw hooks API. Use when setting up Signal messaging, handling Signal contacts, sending/receiving Signal messages, or managing Signal contact permissions.
openclaw skills install signal-messenger-standaloneComplete Signal messenger integration with security-first contact management.
/hooks/wake API on new messages for immediate response (no waiting for next heartbeat)tts tool): Text-to-speech for voice replies# Download latest release
SIGNAL_CLI_VERSION="0.13.12"
curl -L "https://github.com/AsamK/signal-cli/releases/download/v${SIGNAL_CLI_VERSION}/signal-cli-${SIGNAL_CLI_VERSION}-Linux.tar.gz" | tar xz
sudo mv signal-cli-${SIGNAL_CLI_VERSION}/bin/signal-cli /usr/local/bin/
sudo mv signal-cli-${SIGNAL_CLI_VERSION}/lib /usr/local/lib/signal-cli
# Or install to user directory
mv signal-cli-${SIGNAL_CLI_VERSION} ~/.local/share/signal-cli-install
ln -s ~/.local/share/signal-cli-install/bin/signal-cli ~/.local/bin/signal-cli
Requires Java 21+: sudo apt install openjdk-21-jre-headless
# Register with SMS verification
signal-cli -a +YOUR_NUMBER register
# Enter the verification code
signal-cli -a +YOUR_NUMBER verify CODE
# Set your profile name
signal-cli -a +YOUR_NUMBER updateProfile --given-name "YourName" --family-name "Bot"
Edit scripts/signal-poll.sh and scripts/signal-send.sh:
SIGNAL_NUMBER to your registered numberSIGNAL_CLI to your signal-cli binary pathSTATE_DIR to your preferred state directory (default: ~/.signal-state)ALLOWLIST and CONTACTS in signal-poll.sh# Poll every minute
crontab -e
# Add: * * * * * /path/to/scripts/signal-poll.sh
Add to your openclaw.json config:
{
"hooks": {
"wake": {
"enabled": true,
"token": "your-secret-token"
}
}
}
Then set the same token in signal-poll.sh (WAKE_TOKEN variable) and the OpenClaw URL (WAKE_URL).
signal-cli ←→ signal-poll.sh (cron every 1min)
├── Parses text + attachments
├── Logs to conversations/<sender>.log
├── Writes pending_wakes file
└── POSTs to OpenClaw /hooks/wake API
Agent (heartbeat/wake)
├── Reads pending_wakes
├── Reads conversation history for context
├── Transcribes voice messages (Whisper)
├── Generates reply (text or voice)
└── Sends via signal-send.sh
# Text message
scripts/signal-send.sh +1234567890 "Hello!"
# With attachment
signal-cli -a +YOUR_NUMBER send +RECIPIENT -m "Check this out" -a /path/to/file
# Voice message (generate TTS then send as attachment)
# 1. Generate audio (use your TTS engine)
# 2. Convert to m4a: ffmpeg -i voice.wav -c:a aac -b:a 64k voice.m4a
# 3. Send: signal-cli -a +YOUR_NUMBER send +RECIPIENT -m "" -a voice.m4a
The poll script handles receiving automatically. In your HEARTBEAT.md, add:
### Signal Messages (check first!)
cat /path/to/.signal-state/pending_wakes 2>/dev/null
When processing a wake:
pending_wakes for new message summaryconversations/<sender>.log for full contextsignal-send.sh or voice attachment> /path/to/.signal-state/pending_wakes# Convert to WAV for Whisper
ffmpeg -i /path/to/attachment.m4a -ar 16000 -ac 1 -c:a pcm_s16le /tmp/audio.wav -y
# Transcribe (whisper.cpp server example)
curl -s http://127.0.0.1:8080/inference -F "file=@/tmp/audio.wav" -F "language=en"
# Or use OpenAI Whisper, faster-whisper, etc.
# 1. Generate speech (example with a local TTS server)
curl -X POST http://127.0.0.1:5002/tts \
-H "Content-Type: application/json" \
-d '{"text": "Your message here", "language": "en"}' \
-o /tmp/voice.wav
# 2. Convert to m4a for Signal
ffmpeg -i /tmp/voice.wav -c:a aac -b:a 64k /tmp/voice.m4a -y
# 3. Send as attachment
signal-cli -a +YOUR_NUMBER send +RECIPIENT -m "" -a /tmp/voice.m4a
When someone new messages you, signal-cli may show them as a UUID (no phone number). The poll script handles both phone numbers and UUIDs.
# Accept a message request
signal-cli -a +YOUR_NUMBER sendMessageRequestResponse --type accept UUID_OR_NUMBER
# List all known identities
signal-cli -a +YOUR_NUMBER listIdentities
Unknown senders are logged to .signal-state/triage.log and flagged in pending_wakes with ⚠️ NEW CONTACT. The agent should:
ALLOWLIST in signal-poll.shThe send script automatically shows a typing indicator before sending. The poll script sends read receipts and "viewed" receipts for voice messages.
scripts/signal-poll.sh — Cron-based message receiver with wake integrationscripts/signal-send.sh — Send wrapper with typing indicator and conversation loggingreferences/troubleshooting.md — Common issues and fixes