SignalSignal
État : intégration CLI externe. La Gateway communique avec CLIGatewaysignal-cliRPC via HTTP — soit le démon natif (JSON-RPC + SSE), soit le conteneur bbernhard/signal-cli-rest-api (REST + WebSocket).
Prérequis
Section intitulée « Prérequis »- OpenClaw installé sur votre serveur (le flux Linux ci-dessous a été testé sur Ubuntu 24).
- L’un des éléments suivants :
signal-clidisponible sur l’hôte (mode natif), ou- Conteneur
bbernhard/signal-cli-rest-apiDocker Docker (mode conteneur).
- Un numéro de téléphone qui peut recevoir un SMS de vérification (pour le chemin d’inscription par SMS).
- Accès navigateur pour le captcha Signal (Signal
signalcaptchas.org) lors de l’inscription.
Configuration rapide (débutant)
Section intitulée « Configuration rapide (débutant) »- Utilisez un numéro Signal distinct pour le bot (recommandé).
- Installez
signal-cli(Java requis si vous utilisez la version JVM). - Choisissez un chemin de configuration :
- Chemin A (lien QR) :
signal-cli link -n "OpenClaw"Signal et scannez avec Signal. - Chemin B (inscription SMS) : enregistrez un numéro dédié avec captcha + vérification SMS.
- Chemin A (lien QR) :
- Configurez OpenClaw et redémarrez la passerelle.
- Envoyez un premier DM et approuvez le jumelage (
openclaw pairing approve signal <CODE>).
Configuration minimale :
{ channels: { signal: { enabled: true, account: "+15551234567", cliPath: "signal-cli", dmPolicy: "pairing", allowFrom: ["+15557654321"], }, },}Référence des champs :
| Champ | Description |
|---|---|
account | Numéro de téléphone du bot au format E.164 (+15551234567) |
cliPath | Chemin vers signal-cli (signal-cli si sur PATH) |
dmPolicy | Politique d’accès DM (pairing recommandé) |
allowFrom | Numéros de téléphone ou valeurs uuid:<id> autorisés à envoyer des DM |
Ce que c’est
Section intitulée « Ce que c’est »- Canal Signal via Signal
signal-cli(pas de libsignal intégré). - Routage déterministe : les réponses retournent toujours à Signal.
- Les DMs partagent la session principale de l’agent ; les groupes sont isolés (
agent:<agentId>:signal:group:<groupId>).
Écritures de configuration
Section intitulée « Écritures de configuration »Par défaut, Signal est autorisé à écrire des mises à jour de configuration déclenchées par /config set|unset (nécessite commands.config: true).
Désactiver avec :
{ channels: { signal: { configWrites: false } },}Le modèle de numéro (important)
Section intitulée « Le modèle de numéro (important) »- La passerelle se connecte à un appareil Signal (le compte
signal-cli). - Si vous exécutez le bot sur votre compte personnel Signal, il ignorera vos propres messages (protection contre les boucles).
- Pour « J’envoie un message au bot et il répond », utilisez un numéro de bot distinct.
Chemin de configuration A : lier un compte Signal existant (QR)
Section intitulée « Chemin de configuration A : lier un compte Signal existant (QR) »- Installez
signal-cli(version JVM ou native). - Lier un compte de bot :
signal-cli link -n "OpenClaw"puis scannez le QR dans Signal.
- Configurez Signal et démarrez la passerelle.
Exemple :
{ channels: { signal: { enabled: true, account: "+15551234567", cliPath: "signal-cli", dmPolicy: "pairing", allowFrom: ["+15557654321"], }, },}Support multi-compte : utilisez channels.signal.accounts avec une configuration par compte et name en option. Voir gateway/configuration pour le modèle partagé.
Chemin de configuration B : enregistrer un numéro de bot dédié (SMS, Linux)
Section intitulée « Chemin de configuration B : enregistrer un numéro de bot dédié (SMS, Linux) »Utilisez cette méthode lorsque vous souhaitez un numéro de bot dédié au lieu de lier un compte d’application Signal existant.
- Obtenez un numéro qui peut recevoir des SMS (ou une vérification vocale pour les lignes fixes).
- Utilisez un numéro de bot dédié pour éviter les conflits de compte/session.
- Installez
signal-clisur l’hôte de la passerelle :
VERSION=$(curl -Ls -o /dev/null -w %{url_effective} https://github.com/AsamK/signal-cli/releases/latest | sed -e 's/^.*\/v//')curl -L -O "https://github.com/AsamK/signal-cli/releases/download/v${VERSION}/signal-cli-${VERSION}-Linux-native.tar.gz"sudo tar xf "signal-cli-${VERSION}-Linux-native.tar.gz" -C /optsudo ln -sf /opt/signal-cli /usr/local/bin/signal-cli --versionSi vous utilisez la version JVM (signal-cli-${VERSION}.tar.gz), installez d’abord JRE 25+.
Gardez signal-cli à jour ; les notes en amont indiquent que les anciennes versions peuvent cesser de fonctionner lorsque les API du serveur Signal changent.
- Enregistrez et vérifiez le numéro :
signal-cli -a +<BOT_PHONE_NUMBER> registerSi un captcha est requis :
- Ouvrez
https://signalcaptchas.org/registration/generate.html. - Complétez le captcha, copiez la cible du lien
signalcaptcha://...depuis « Ouvrir Signal ». - Exécutez depuis la même IP externe que la session du navigateur si possible.
- Exécutez l’enregistrement à nouveau immédiatement (les jetons de captcha expirent rapidement) :
signal-cli -a +<BOT_PHONE_NUMBER> register --captcha '<SIGNALCAPTCHA_URL>'signal-cli -a +<BOT_PHONE_NUMBER> verify <VERIFICATION_CODE>- Configurez OpenClaw, redémarrez la passerelle, vérifiez le canal :
# If you run the gateway as a user systemd service:systemctl --user restart openclaw-gateway.service
# Then verify:openclaw doctoropenclaw channels status --probe- Associez votre expéditeur DM :
- Envoyez n’importe quel message au numéro du bot.
- Approuvez le code sur le serveur :
openclaw pairing approve signal <PAIRING_CODE>. - Enregistrez le numéro du bot comme un contact sur votre téléphone pour éviter “Contact inconnu”.
Références en amont :
- README
signal-cli:https://github.com/AsamK/signal-cli - Flux de vérification (Captcha) :
https://github.com/AsamK/signal-cli/wiki/Registration-with-captcha - Flux de liaison :
https://github.com/AsamK/signal-cli/wiki/Linking-other-devices-(Provisioning)
Mode démon externe (httpUrl)
Section intitulée « Mode démon externe (httpUrl) »Si vous souhaitez gérer signal-cli vous-même (démarrages à froid JVM lents, initialisation de conteneur ou CPU partagés), exécutez le démon séparément et pointez OpenClaw vers celui-ci :
{ channels: { signal: { httpUrl: "http://127.0.0.1:8080", autoStart: false, }, },}Cela évite le démarrage automatique et l’attente de démarrage à l’intérieur de OpenClaw. Pour les démarrages lents lors du démarrage automatique, définissez channels.signal.startupTimeoutMs.
Mode conteneur (bbernhard/signal-cli-rest-api)
Section intitulée « Mode conteneur (bbernhard/signal-cli-rest-api) »Au lieu d’exécuter signal-cli en mode natif, vous pouvez utiliser le conteneur Docker bbernhard/signal-cli-rest-api. Cela encapsule signal-cli derrière une interface REST API et WebSocket.
Configuration requise :
- Le conteneur doit s’exécuter avec
MODE=json-rpcpour la réception de messages en temps réel. - Enregistrez ou liez votre compte Signal à l’intérieur du conteneur avant de connecter OpenClaw.
Exemple de service docker-compose.yml :
signal-cli: image: bbernhard/signal-cli-rest-api:latest environment: MODE: json-rpc ports: - "8080:8080" volumes: - signal-cli-data:/home/.local/share/signal-cliConfiguration OpenClaw :
{ channels: { signal: { enabled: true, account: "+15551234567", httpUrl: "http://signal-cli:8080", autoStart: false, apiMode: "container", // or "auto" to detect automatically }, },}Le champ apiMode contrôle le protocole utilisé par OpenClaw :
| Valeur | Comportement |
|---|---|
"auto" | (Par défaut) Sonde les deux transports ; le flux valide la réception WebSocket du conteneur |
"native" | Force le signal-cli natif (JSON-RPC sur /api/v1/rpc, SSE sur /api/v1/events) |
"container" | Forcer le conteneur bbernhard (REST sur /v2/send, WebSocket sur /v1/receive/{account}) |
Lorsque apiMode est "auto"OpenClaw, OpenClaw met en cache le mode détecté pendant 30 secondes pour éviter des sondages répétés. La réception par conteneur n’est sélectionnée pour le streaming qu’après que /v1/receive/{account} a effectué une mise à niveau vers WebSocket, ce qui nécessite MODE=json-rpc.
Le mode conteneur prend en charge les mêmes opérations de canal Signal que le mode natif là où le conteneur expose des API correspondantes : envois, réceptions, pièces jointes, indicateurs de frappe, accusés de réception/lecture, réactions, groupes et texte stylé. OpenClaw traduit ses appels RPC Signal natifs en charges utiles REST du conteneur, y compris les ID de groupe SignalOpenClawSignalRPCgroup.{base64(internal_id)} et text_mode: "styled" pour le texte formaté.
Notes opérationnelles :
- Utilisez
autoStart: falseOpenClaw avec le mode conteneur. OpenClaw ne doit pas lancer de démon natif lorsqueapiMode: "container"est sélectionné. - Utilisez
MODE=json-rpcpour la réception.MODE=normalpeut faire paraître/v1/aboutsain, mais/v1/receive/{account}OpenClaw n’effectue pas de mise à niveau WebSocket, donc OpenClaw ne sélectionnera pas le streaming de réception par conteneur en modeauto. - Définissez
apiMode: "container"lorsque vous savez quehttpUrlAPI pointe vers l’API REST de bbernhard. DéfinissezapiMode: "native"lorsque vous savez qu’il pointe vers le JSON-RPC/SSEsignal-cliRPC natif. Utilisez"auto"lorsque le déploiement peut varier. - Les téléchargements de pièces jointes de conteneur respectent les mêmes limites d’octets de média que le mode natif. Les réponses trop volumineuses sont rejetées avant d’être entièrement mises en tampon lorsque le serveur envoie
Content-Length, et pendant le streaming sinon.
Contrôle d’accès (DMs + groupes)
Section intitulée « Contrôle d’accès (DMs + groupes) »DMs :
- Par défaut :
channels.signal.dmPolicy = "pairing". - Les expéditeurs inconnus reçoivent un code d’appariement ; les messages sont ignorés jusqu’à approbation (les codes expirent après 1 heure).
- Approuver via :
openclaw pairing list signalopenclaw pairing approve signal <CODE>
- L’appariement est l’échange de jetons par défaut pour les DM Signal. Détails : Appariement
- Les expéditeurs UUID uniquement (issus de
sourceUuid) sont stockés sous forme deuuid:<id>danschannels.signal.allowFrom.
Groupes :
channels.signal.groupPolicy = open | allowlist | disabled.channels.signal.groupAllowFromcontrôle quels groupes ou expéditeurs peuvent déclencher des réponses de groupe lorsqueallowlistSignal est défini ; les entrées peuvent être des ID de groupe Signal (bruts,group:<id>, ousignal:group:<id>), des numéros de téléphone d’expéditeur, des valeursuuid:<id>, ou*.channels.signal.groups["<group-id>" | "*"]peut remplacer le comportement du groupe avecrequireMention,tools, ettoolsBySender.- Utilisez
channels.signal.accounts.<id>.groupspour des remplacements par compte dans les configurations multi-comptes. - L’ajout sur liste blanche d’un groupe Signal via Signal
groupAllowFromne désactive pas par lui-même le filtrage par mention. Une entréechannels.signal.groups["<group-id>"]spécifiquement configurée traite chaque message de groupe, sauf sirequireMention=trueest défini. - Note d’exécution : si
channels.signalest complètement manquant, l’exécution revient àgroupPolicy="allowlist"pour les vérifications de groupe (même sichannels.defaults.groupPolicyest défini).
Fonctionnement (comportement)
Section intitulée « Fonctionnement (comportement) »- Mode natif :
signal-clis’exécute en tant que démon ; la passerelle lit les événements via SSE. - Mode conteneur : la passerelle envoie via l’API REST et reçoit via WebSocket.
- Les messages entrants sont normalisés dans l’enveloppe de canal partagée.
- Les réponses sont toujours acheminées vers le même numéro ou groupe.
Médias + limites
Section intitulée « Médias + limites »- Le texte sortant est découpé en morceaux de
channels.signal.textChunkLimit(4000 par défaut). - Découpage optionnel par nouvelle ligne : définissez
channels.signal.chunkMode="newline"pour diviser sur les lignes vides (limites de paragraphe) avant le découpage par longueur. - Pièces jointes prises en charge (base64 récupéré depuis
signal-cli). - Les pièces jointes de notes vocales utilisent le nom de fichier
signal-clicomme solution de repli MIME lorsquecontentTypeest manquant, afin que la transcription audio puisse toujours classer les mémos vocaux AAC. - Limite média par défaut :
channels.signal.mediaMaxMb(par défaut 8). - Utilisez
channels.signal.ignoreAttachmentspour ignorer le téléchargement des médias. - Le contexte d’historique de groupe utilise
channels.signal.historyLimit(ouchannels.signal.accounts.*.historyLimit), avec repli surmessages.groupChat.historyLimit. Définissez0pour désactiver (par défaut 50).
Indicateurs de frappe + accusés de lecture
Section intitulée « Indicateurs de frappe + accusés de lecture »- Indicateurs de frappe : OpenClaw envoie des signaux de frappe via
signal-cli sendTypinget les actualise pendant qu’une réponse est en cours. - Accusés de lecture : lorsque
channels.signal.sendReadReceiptsest vrai, OpenClaw transfère les accusés de lecture pour les DMs autorisés. - Signal-cli n’expose pas les accusés de lecture pour les groupes.
Réactions (message tool)
Section intitulée « Réactions (message tool) »- Utilisez
message action=reactavecchannel=signal. - Cibles : E.164 ou UUID de l’expéditeur (utilisez
uuid:<id>depuis la sortie de l’appariement ; l’UUID seul fonctionne aussi). messageIdest l’horodatage Signal du message auquel vous réagissez.- Les réactions de groupe nécessitent
targetAuthoroutargetAuthorUuid.
Exemples :
message action=react channel=signal target=uuid:123e4567-e89b-12d3-a456-426614174000 messageId=1737630212345 emoji=🔥message action=react channel=signal target=+15551234567 messageId=1737630212345 emoji=🔥 remove=truemessage action=react channel=signal target=signal:group:<groupId> targetAuthor=uuid:<sender-uuid> messageId=1737630212345 emoji=✅Configuration :
channels.signal.actions.reactions: activer/désactiver les actions de réaction (par défaut vrai).channels.signal.reactionLevel:off | ack | minimal | extensive.off/ackdésactive les réactions de l’agent (l’outil de messagereactrenverra une erreur).minimal/extensiveactive les réactions de l’agent et définit le niveau de guidage.
- Remplacements par compte :
channels.signal.accounts.<id>.actions.reactions,channels.signal.accounts.<id>.reactionLevel.
Cibles de livraison (CLI/cron)
Section intitulée « Cibles de livraison (CLI/cron) »- DMs :
signal:+15551234567(ou E.164 brut). - DMs UUID :
uuid:<id>(ou UUID seul). - Groupes :
signal:group:<groupId>. - Noms d’utilisateur :
username:<name>Signal (si pris en charge par votre compte Signal).
Dépannage
Section intitulée « Dépannage »Exécutez d’abord cette échelle :
openclaw statusopenclaw gateway statusopenclaw logs --followopenclaw doctoropenclaw channels status --probeConfirmez ensuite l’état du jumelage DM si nécessaire :
openclaw pairing list signalPannes courantes :
- Démon accessible mais pas de réponses : vérifiez les paramètres de compte/démon (
httpUrl,account) et le mode de réception. - DMs ignorés : l’expéditeur est en attente d’approbation de jumelage.
- Messages de groupe ignorés : le filtrage des expéditeurs/mentions de groupe bloque la livraison.
- Erreurs de validation de configuration après modifications : exécutez
openclaw doctor --fix. - Signal absent des diagnostics : confirmez Signal
channels.signal.enabled: true.
Vérifications supplémentaires :
openclaw pairing list signalpgrep -af signal-cligrep -i "signal" "/tmp/openclaw/openclaw-$(date +%Y-%m-%d).log" | tail -20Pour le flux de triage : /channels/troubleshooting.
Notes de sécurité
Section intitulée « Notes de sécurité »signal-clistocke les clés de compte localement (généralement~/.local/share/signal-cli/data/).- Sauvegardez l’état du compte Signal avant la migration ou la reconstruction du serveur.
- Gardez
channels.signal.dmPolicy: "pairing"sauf si vous voulez explicitement un accès DM plus large. - La vérification par SMS n’est nécessaire que pour les flux d’enregistrement ou de récupération, mais la perte de contrôle du numéro/compte peut compliquer la réinscription.
Référence de configuration (Signal)
Section intitulée « Référence de configuration (Signal) »Configuration complète : Configuration
Options du fournisseur :
channels.signal.enabled: activer/désactiver le démarrage du channel.channels.signal.apiMode:auto | native | container(par défaut : auto). Voir Container mode.channels.signal.account: E.164 pour le compte bot.channels.signal.cliPath: chemin verssignal-cli.channels.signal.httpUrl: URL complète du démon (remplace l’hôte/le port).channels.signal.httpHost,channels.signal.httpPort: liaison du démon (par défaut 127.0.0.1:8080).channels.signal.autoStart: démarrage automatique du démon (par défaut true sihttpUrlnon défini).channels.signal.startupTimeoutMs: délai d’attente de démarrage en ms (max 120000).channels.signal.receiveMode:on-start | manual.channels.signal.ignoreAttachments: ignorer les téléchargements de pièces jointes.channels.signal.ignoreStories: ignorer les stories du démon.channels.signal.sendReadReceipts: transférer les accusés de réception.channels.signal.dmPolicy:pairing | allowlist | open | disabled(par défaut : jumelage).channels.signal.allowFrom: liste d’autorisation de DM (E.164 ouuuid:<id>).opennécessite"*". Signal n’a pas de noms d’utilisateur ; utilisez les identifiants téléphone/UUID.channels.signal.groupPolicy:open | allowlist | disabled(par défaut : liste d’autorisation).channels.signal.groupAllowFrom: liste d’autorisation de groupe ; accepte les ID de groupe Signal (bruts,group:<id>ousignal:group:<id>), les numéros E.164 de l’expéditeur, ou les valeursuuid:<id>.channels.signal.groups: remplacements par groupe, indexés par l’ID de groupe Signal (ou"*"). Champs pris en charge :requireMention,tools,toolsBySender.channels.signal.accounts.<id>.groups: version par compte dechannels.signal.groupspour les configurations multi-comptes.channels.signal.historyLimit: nombre maximum de messages de groupe à inclure en contexte (0 désactive).channels.signal.dmHistoryLimit: limite d’historique DM en tours d’utilisateur. Remplacements par utilisateur :channels.signal.dms["<phone_or_uuid>"].historyLimit.channels.signal.textChunkLimit: taille des blocs sortants (caractères).channels.signal.chunkMode:length(par défaut) ounewlinepour diviser sur les lignes vides (limites de paragraphe) avant le découpage par longueur.channels.signal.mediaMaxMb: limite de capacité média entrant/sortant (Mo).
Options globales connexes :
agents.list[].groupChat.mentionPatternsSignal (Signal ne prend pas en charge les mentions natives).messages.groupChat.mentionPatterns(secours global).messages.responsePrefix.
Connexes
Section intitulée « Connexes »- Aperçu des canaux — tous les canaux pris en charge
- Appairage — authentification DM et flux d’appairage
- Groupes — comportement du chat de groupe et filtrage des mentions
- Channel Routing — routage de session pour les messages
- Security — model d’accès et durcissement