Plugin pro Hermes Agent que SUBSTITUI o WebhookAdapter built-in pra
suportar session_chat_id estável derivado de campo no payload.
Por que existe
--------------
O WebhookAdapter built-in monta a chave de sessão como:
session_chat_id = f"webhook:{route}:{delivery_id}"
delivery_id é único por POST → cada msg cria sessão nova no Hermes. OK pra
webhooks one-shot, ERRADO pra integração de chat onde múltiplas mensagens
da mesma conversa precisam compartilhar memória de sessão.
Como funciona
-------------
Quando o caller (Captain) inclui `conversation_id` ou `hermes_session_id`
no payload, o plugin reescreve chat_id pra:
session_chat_id = f"webhook:{route}:session:{conversation_id}"
Mesma conversation_id em múltiplas POSTs → mesma sessão Hermes →
contexto e memória preservados. Sem o campo, fallback ao comportamento
default (session nova por POST). 100% backward-compatible.
Implementação
-------------
- kind: platform — registra com name="webhook" pra substituir built-in
(Hermes prioriza platform_registry sobre código built-in em
gateway/run.py:_create_adapter)
- Herda WebhookAdapter — só override `handle_message` (rewrite chat_id)
e `connect` (recupera gateway_runner via _gateway_runner_ref pq o
plugin path não seta isso explicitamente)
- Outros adapters (HMAC, rate limit, idempotency, parsing, deliver
dispatch) — herdados sem cópia
Validado end-to-end na VPS (profile valentina):
- POST com conversation_id=99999 (msg 1) → session:99999 criada
- POST com conversation_id=99999 (msg 2) → MESMA session reutilizada
- Hermes responde via Codex em ~10s (2 turnos cumulativos)
- http_callback faz POST de volta no Captain (HTTP 200)
- Logs mostram: [captain-webhook] Stable session: ... -> session:99999
Combinado com captain-http-callback, completa o ciclo Captain ↔ Hermes:
Captain manda webhook com conversation_id → Hermes processa em sessão
estável → http_callback POSTa resposta de volta → Captain envia ao
WhatsApp.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
|
||
|---|---|---|
| .. | ||
| __init__.py | ||
| adapter.py | ||
| plugin.yaml | ||
| README.md | ||
captain-webhook
Hermes Agent platform plugin que substitui o WebhookAdapter built-in pra suportar session_chat_id estável derivado de campo no payload.
Por que existe
O webhook adapter built-in do Hermes monta a chave de sessão como:
session_chat_id = f"webhook:{route_name}:{delivery_id}"
delivery_id é único por POST → cada mensagem cria sessão nova no Hermes. Isso funciona pra webhooks one-shot (alertas, GitHub events), mas é errado pra integração de chat onde múltiplas mensagens da mesma conversa precisam compartilhar memória de sessão.
Esse plugin permite que o caller (ex: Captain) inclua um identificador estável no payload — conversation_id (preferido) ou hermes_session_id — e o adapter reescreve a chave pra:
session_chat_id = f"webhook:{route_name}:session:{conversation_id}"
Mesmo conversation_id em múltiplas POSTs → mesma sessão Hermes → memória da conversa preservada.
Como funciona
CaptainWebhookAdapter herda de WebhookAdapter built-in e faz uma única override: o método handle_message(). Ele:
- Recebe o
eventjá montado pelo built-in - Lê
event.raw_message(o payload JSON do webhook) - Se houver
hermes_session_idouconversation_id, monta novochat_id - Mirror o
_delivery_infopra nova chave (pra osend()posterior achar config) - Modifica
event.source.chat_id - Chama
super().handle_message(event)
Toda outra lógica (HMAC, rate limit, idempotency, parsing JSON, signature validation, deliver dispatch) é herdada sem cópia.
Como o Hermes substitui o built-in
gateway/run.py:
# Plugin-registered platforms (checked first)
if platform_registry.is_registered(platform.value):
adapter = platform_registry.create_adapter(platform.value, config)
if adapter is not None:
return adapter
# Fall through to built-in adapters below
Se este plugin se registrar com name="webhook" (mesmo nome do built-in), is_registered("webhook") retorna True e o CaptainWebhookAdapter é usado em vez do built-in WebhookAdapter.
Instalação no profile do Hermes
# Copia plugin pro profile
cp -r hermes-plugins/captain-webhook /root/.hermes/profiles/<profile>/plugins/
# Ativa
HERMES_HOME=/root/.hermes/profiles/<profile> hermes plugins enable captain-webhook
# Reinicia gateway pra carregar
pkill -f "HERMES_HOME=/root/.hermes/profiles/<profile>"
HERMES_HOME=/root/.hermes/profiles/<profile> nohup hermes gateway run --replace > /var/log/hermes-<profile>.log 2>&1 &
Verifica:
HERMES_HOME=/root/.hermes/profiles/<profile> hermes plugins list | grep captain-webhook
Backward compatibility
Quando o payload NÃO traz hermes_session_id nem conversation_id, o adapter não modifica nada — comportamento idêntico ao built-in. Webhooks one-shot continuam funcionando normalmente.
Limitações
- O plugin estende a versão do
WebhookAdapterinstalada no Hermes. Quando o Hermes for atualizado, é prudente revisar se a interface base mudou (signature dohandle_message, formato dochat_id, etc). - Não modifica o ciclo de idempotency: cada POST ainda precisa de
delivery_idúnico (auto-gerado pelo Hermes ou via headerX-Request-ID). - Não persiste sessions entre restarts do Hermes — isso é responsabilidade do session store do próprio Hermes (SQLite por profile).