AgentGuard security monitoring plugin for Hermes Agent
Project description
agentguard-hermes
AgentGuard security monitoring plugin for Hermes Agent.
Forwards Hermes session, LLM, and tool-call events to an AgentGuard Timeplus stream for real-time observability, and synchronously blocks risky tool calls based on rule decisions from the AgentGuard server.
How it works
Two paths run depending on the hook type:
Async observation path (most hooks)
For every hook except pre_tool_call, the plugin POSTs the normalized event to Timeplus via urllib and the callback returns immediately — Hermes never waits.
Synchronous hold-and-wait path (pre_tool_call only)
When AGENTGUARD_HOLDS_ENABLED is true (the default), pre_tool_call:
- POSTs the event to AgentGuard's
/api/holdsendpoint synchronously viaurllib.request.urlopen(timeout=600). The endpoint ingests the event itself, so the plugin does NOT also call its async ingest path for this hook. - AgentGuard's backend ingests the event, then waits up to ~500 ms for any rule's materialized view to detect a match (also checks for any pre-existing open threats from earlier in the session).
- Backend resolves to one of three outcomes based on the matched rule's
block_policy:allow(no rule fired, or rule policy islog_only) → returned immediatelyblock(rule policy isauto_block) → returned immediately with reasonhold(rule policy ishold) → opens a hold record, long-polls for human Approve/Deny in the AgentGuard UI (up to 5 minutes), returns the human's decision
- Plugin returns
{"action": "block", "message": "..."}from the callback onblock(Hermes short-circuits the tool call), orNoneonallow(Hermes runs the tool normally).
The agent's hot path is paused for the duration of the hold — no LLM tokens are burned waiting for human review.
If AGENTGUARD_HOLDS_ENABLED is false, pre_tool_call falls back to the async observation path and never blocks. Useful for toggling blocking on/off during testing without uninstalling the plugin.
Hermes timeout note: unlike Claude Code (600s hook cap) and OpenClaw (configurable approval timeout), Hermes provides no hook timeout and no abort signal. The plugin self-imposes
urllib.request.urlopen(timeout=600)so a stuck/api/holdsrequest can't hang Hermes forever. The backend's hold-timeout ceiling is 540 s, leaving 60 s of margin under the plugin timeout.
Installation
pip install agentguard-hermes-plugin
agentguard-hermes install
This copies the plugin into ~/.hermes/plugins/agentguard/ where Hermes auto-loads it on startup.
Configuration
Set environment variables before starting Hermes:
Async event ingest (Hermes → Timeplus)
| Variable | Default | Description |
|---|---|---|
AGENTGUARD_TIMEPLUS_URL |
http://localhost:3218 |
Timeplus Enterprise HTTP endpoint |
AGENTGUARD_USERNAME |
proton |
Timeplus username |
AGENTGUARD_PASSWORD |
(empty) | Timeplus password |
AGENTGUARD_STREAM |
agentguard_hook_events |
Target Timeplus stream |
Identity
| Variable | Default | Description |
|---|---|---|
AGENTGUARD_AGENT_ID |
hostname | Identifier for this agent instance |
AGENTGUARD_DEPLOYMENT_ID |
local |
Deployment environment tag |
AGENTGUARD_DEPLOYMENT_NAME |
Local Dev |
Human-readable deployment name |
Synchronous tool blocking (pre_tool_call → AgentGuard /api/holds)
| Variable | Default | Description |
|---|---|---|
AGENTGUARD_URL |
http://localhost:8080 |
AgentGuard server base URL — where pre_tool_call POSTs its hold request |
AGENTGUARD_HOLD_FAIL_POLICY |
deny |
What to return when /api/holds is unreachable: deny blocks the tool call, allow lets it proceed |
AGENTGUARD_HOLDS_ENABLED |
true |
When false, pre_tool_call reverts to async observation-only mode (no blocking, event still ingested) |
Server-side prerequisite — Timeplus service credentials
The plugin's synchronous /api/holds call hits the AgentGuard server with no user session cookie (server-to-server). The AgentGuard server in turn needs Timeplus credentials to ingest the event and query rules. You must set TIMEPLUS_USER and TIMEPLUS_PASSWORD env vars on the AgentGuard process — these are the canonical service credentials for plugin endpoints.
Without these, /api/holds returns:
HTTP 503
{"error":"service credentials not configured: set TIMEPLUS_USER and TIMEPLUS_PASSWORD env vars on the AgentGuard server"}
For the standard docker-compose.yaml setup these are pre-set to proton / timeplus@t+. Edit them if your wizard credentials differ, then docker compose up -d --build agentguard.
Example
# Async ingest
export AGENTGUARD_TIMEPLUS_URL=http://timeplus.example.com:3218
export AGENTGUARD_USERNAME=proton
export AGENTGUARD_PASSWORD=secret
# Identity
export AGENTGUARD_DEPLOYMENT_ID=production
export AGENTGUARD_DEPLOYMENT_NAME="Production Hermes"
# Sync tool blocking
export AGENTGUARD_URL=http://agentguard.example.com:8080
export AGENTGUARD_HOLD_FAIL_POLICY=deny
# AGENTGUARD_HOLDS_ENABLED defaults to true
hermes
What gets captured
Every Hermes hook event is forwarded to the agentguard_hook_events stream:
| Hermes hook | Sent as | Description |
|---|---|---|
on_session_start |
on_session_start |
New session begins |
on_session_end |
conversation_end |
Single chat turn completed |
on_session_finalize |
on_session_end |
Session fully torn down (CLI exit, /reset) — also fires POST /api/holds/_abandon-session to clean up any pending hold |
on_session_reset |
on_session_reset |
Session rotated via /new |
pre_llm_call |
pre_llm_call |
Before each LLM turn |
post_llm_call |
post_llm_call |
After each LLM turn |
pre_api_request |
pre_api_request |
Before each raw API call (token metrics) |
post_api_request |
post_api_request |
After each raw API call |
pre_tool_call |
pre_tool_call |
Before each tool execution — synchronous when AGENTGUARD_HOLDS_ENABLED=true: ingest happens via /api/holds (not the async path), and the callback returns {"action": "block", "message": ...} on deny |
post_tool_call |
post_tool_call |
After each tool execution |
conversation_history is stripped from all events before ingestion — it is unbounded and contains no information not already available from the individual turn events.
Verifying the synchronous hold flow
To exercise blocking end-to-end:
- Open the AgentGuard UI at
http://localhost:8080, log in, and install a rule that matches Hermes tool calls (e.g. Privilege Guard for shell-like tools). - Set its
block_policytoholdon the rule detail page (/rules/:id→ Block Policy panel). - In Hermes, ask the agent to run a matching tool call.
- In the AgentGuard UI you should see within ~1 s:
- A toast notification bottom-right: "Hold pending:
<tool>on<agent>" - A pending-holds badge on the Threats sidebar entry
- The threat detail page shows an Approve / Deny banner with the rule message and a truncated
args_summary
- A toast notification bottom-right: "Hold pending:
- Click Approve or Deny → Hermes resumes (or rejects) the tool call within ~1 s. On deny, Hermes feeds the deny message back to the LLM as a tool error.
- Audit trail: every hold lifecycle row is in the
agentguard_holdsstream:curl -s -X POST http://localhost:3218/proton/v1/query \ -u proton:'timeplus@t+' -H 'Content-Type: application/json' \ -d '{"query":"SELECT hold_id, tool_name, status, decided_by, decided_at FROM table(mv_holds_current) ORDER BY created_at DESC LIMIT 10"}'
If the plugin reports AgentGuard unreachable — backend returned 503, the AgentGuard server is missing service credentials — see Server-side prerequisite.
To temporarily disable hold blocking without changing rules or uninstalling:
export AGENTGUARD_HOLDS_ENABLED=false
hermes # restart so the env var takes effect
Custom install path
agentguard-hermes install --hermes-dir /path/to/hermes/data
Manual installation (Makefile / Docker)
If you run Hermes via the provided Docker Compose setup in agents/hermes/:
make configure # copies plugin into .hermes/plugins/agentguard/
make cli # start Hermes CLI with AgentGuard env vars pre-set
make start # start Hermes gateway + web dashboard
Publishing to PyPI
pip install hatch
cd agents/hermes/agentguard-plugin
hatch build # produces dist/agentguard_hermes_plugin-*.whl
hatch publish
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file agentguard_hermes_plugin-0.2.0.tar.gz.
File metadata
- Download URL: agentguard_hermes_plugin-0.2.0.tar.gz
- Upload date:
- Size: 10.0 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.12.9
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
1cc7f2ec462317fb054a30d08b84f846d5e1220dfd0e8afc0ec8ca018b7938eb
|
|
| MD5 |
e270e7de360ec9d7d2d099e61531f1a0
|
|
| BLAKE2b-256 |
70168ac309739bceadc1b275e5965083127c0eba7d194ff9f40dc48d1da1dd52
|
File details
Details for the file agentguard_hermes_plugin-0.2.0-py3-none-any.whl.
File metadata
- Download URL: agentguard_hermes_plugin-0.2.0-py3-none-any.whl
- Upload date:
- Size: 10.7 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.12.9
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
68a7d1baa4128cda4883adf3dad64024ffae7b2910eb5cbe4ea3c58727e17d79
|
|
| MD5 |
981dbe8bf399ea7ad2513614ea883ce7
|
|
| BLAKE2b-256 |
54e62eef7f05bbe50505a546fafb8bf020dcec3cab9a1022c469224d7d7f42f3
|