mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-06-24 10:52:21 +00:00
Remove the dashboard --insecure auth-bypass, add an MCP persistence guard + IOC blocklist, and raise the API-server key entropy floor. Driven by the June 2026 hermes-0day campaign (r/hermesagent, live 854.media instance): scanners find exposed Hermes dashboards/API servers, drive the root agent to plant a 'command: bash' MCP entry that appends an attacker SSH key to authorized_keys, which cron + startup then re-execute every tick. - dashboard: --insecure no longer disables the auth gate. should_require_auth returns True for every non-loopback bind; a public bind ALWAYS requires an auth provider (bundled password provider or OAuth). --insecure kept as a warned no-op for backward compat. Fail-closed error now points at the password provider, not at --insecure. - mcp_security: validate_mcp_server_entry now also rejects shell payloads that write to OS persistence surfaces (authorized_keys/.ssh/pam.d/sudoers/cron/ rc files) and hard-rejects a hermes-0day IOC blocklist (attacker SSH key + source IPs) anywhere in command/args/env. Runs at save AND spawn time. - api_server: raise network-bind API_SERVER_KEY entropy floor 8->16 chars; warn when a network-accessible API server runs an unsandboxed local backend.
176 lines
7.4 KiB
Python
176 lines
7.4 KiB
Python
"""Tests for gateway weak credential rejection at startup.
|
|
|
|
Ported from openclaw/openclaw#64586: rejects known-weak placeholder
|
|
tokens at gateway startup instead of letting them silently fail
|
|
against platform APIs.
|
|
"""
|
|
|
|
import logging
|
|
|
|
import pytest
|
|
|
|
from gateway.config import PlatformConfig, Platform, _validate_gateway_config
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
# Helper: create a minimal GatewayConfig with one enabled platform
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
|
def _make_gateway_config(platform, token, enabled=True, **extra_kwargs):
|
|
"""Create a minimal GatewayConfig-like object for validation testing."""
|
|
from gateway.config import GatewayConfig
|
|
|
|
config = GatewayConfig(platforms={})
|
|
pconfig = PlatformConfig(enabled=enabled, token=token, **extra_kwargs)
|
|
config.platforms[platform] = pconfig
|
|
return config
|
|
|
|
|
|
def _validate_and_return(config):
|
|
"""Call _validate_gateway_config and return the config (mutated in place)."""
|
|
_validate_gateway_config(config)
|
|
return config
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
# Unit tests: platform token placeholder rejection
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
|
class TestPlatformTokenPlaceholderGuard:
|
|
"""Verify that _validate_gateway_config disables platforms with placeholder tokens."""
|
|
|
|
def test_rejects_triple_asterisk(self, caplog):
|
|
"""'***' is the .env.example placeholder — should be rejected."""
|
|
config = _make_gateway_config(Platform.TELEGRAM, "***")
|
|
with caplog.at_level(logging.ERROR):
|
|
_validate_and_return(config)
|
|
assert config.platforms[Platform.TELEGRAM].enabled is False
|
|
assert "placeholder" in caplog.text.lower()
|
|
|
|
def test_rejects_changeme(self, caplog):
|
|
config = _make_gateway_config(Platform.DISCORD, "changeme")
|
|
with caplog.at_level(logging.ERROR):
|
|
_validate_and_return(config)
|
|
assert config.platforms[Platform.DISCORD].enabled is False
|
|
|
|
def test_rejects_your_api_key(self, caplog):
|
|
config = _make_gateway_config(Platform.SLACK, "your_api_key")
|
|
with caplog.at_level(logging.ERROR):
|
|
_validate_and_return(config)
|
|
assert config.platforms[Platform.SLACK].enabled is False
|
|
|
|
def test_rejects_placeholder(self, caplog):
|
|
config = _make_gateway_config(Platform.MATRIX, "placeholder")
|
|
with caplog.at_level(logging.ERROR):
|
|
_validate_and_return(config)
|
|
assert config.platforms[Platform.MATRIX].enabled is False
|
|
|
|
def test_accepts_real_token(self, caplog):
|
|
"""A real-looking bot token should pass validation."""
|
|
config = _make_gateway_config(
|
|
Platform.TELEGRAM, "7123456789:AAHdqTcvCH1vGWJxfSeOfSAs0K5PALDsaw"
|
|
)
|
|
with caplog.at_level(logging.ERROR):
|
|
_validate_and_return(config)
|
|
assert config.platforms[Platform.TELEGRAM].enabled is True
|
|
assert "placeholder" not in caplog.text.lower()
|
|
|
|
def test_accepts_empty_token_without_error(self, caplog):
|
|
"""Empty tokens get a warning (existing behavior), not a placeholder error."""
|
|
config = _make_gateway_config(Platform.TELEGRAM, "")
|
|
with caplog.at_level(logging.WARNING):
|
|
_validate_and_return(config)
|
|
# Empty token doesn't trigger placeholder rejection — enabled stays True
|
|
# (the existing empty-token warning is separate)
|
|
assert config.platforms[Platform.TELEGRAM].enabled is True
|
|
|
|
def test_disabled_platform_not_checked(self, caplog):
|
|
"""Disabled platforms should not be validated."""
|
|
config = _make_gateway_config(Platform.TELEGRAM, "***", enabled=False)
|
|
with caplog.at_level(logging.ERROR):
|
|
_validate_and_return(config)
|
|
assert "placeholder" not in caplog.text.lower()
|
|
|
|
def test_rejects_whitespace_padded_placeholder(self, caplog):
|
|
"""Whitespace-padded placeholders should still be caught."""
|
|
config = _make_gateway_config(Platform.TELEGRAM, " *** ")
|
|
with caplog.at_level(logging.ERROR):
|
|
_validate_and_return(config)
|
|
assert config.platforms[Platform.TELEGRAM].enabled is False
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
# Integration test: API server placeholder key on network-accessible host
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
|
class TestAPIServerPlaceholderKeyGuard:
|
|
"""Verify that the API server rejects placeholder keys on network hosts."""
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_refuses_wildcard_with_placeholder_key(self):
|
|
from gateway.platforms.api_server import APIServerAdapter
|
|
|
|
adapter = APIServerAdapter(
|
|
PlatformConfig(enabled=True, extra={"host": "0.0.0.0", "key": "changeme"})
|
|
)
|
|
result = await adapter.connect()
|
|
assert result is False
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_refuses_wildcard_with_asterisk_key(self):
|
|
from gateway.platforms.api_server import APIServerAdapter
|
|
|
|
adapter = APIServerAdapter(
|
|
PlatformConfig(enabled=True, extra={"host": "0.0.0.0", "key": "***"})
|
|
)
|
|
result = await adapter.connect()
|
|
assert result is False
|
|
|
|
def test_allows_loopback_with_placeholder_key(self):
|
|
"""Loopback with a placeholder key is fine — not network-exposed."""
|
|
from gateway.platforms.api_server import APIServerAdapter
|
|
from gateway.platforms.base import is_network_accessible
|
|
|
|
adapter = APIServerAdapter(
|
|
PlatformConfig(enabled=True, extra={"host": "127.0.0.1", "key": "changeme"})
|
|
)
|
|
# On loopback the placeholder guard doesn't fire
|
|
assert is_network_accessible(adapter._host) is False
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_refuses_wildcard_with_short_random_key(self):
|
|
"""A short but non-placeholder key is brute-forceable on a public bind.
|
|
|
|
June 2026 hermes-0day hardening raised the network-bind entropy floor
|
|
from 8 to 16 chars. A 12-char random key (which passed the old guard)
|
|
must now be refused — the API server dispatches terminal-capable agent
|
|
work, so a guessable key is RCE.
|
|
"""
|
|
from gateway.platforms.api_server import APIServerAdapter
|
|
|
|
adapter = APIServerAdapter(
|
|
PlatformConfig(enabled=True, extra={"host": "0.0.0.0", "key": "a1b2c3d4e5f6"})
|
|
)
|
|
result = await adapter.connect()
|
|
assert result is False
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_allows_wildcard_with_strong_key(self):
|
|
"""A 32-char random key clears the entropy floor (connect proceeds past
|
|
the credential guard). We don't assert full startup success here — the
|
|
port/runner setup is environment-dependent — only that the weak-key
|
|
guard does not reject it."""
|
|
from gateway.platforms.api_server import APIServerAdapter
|
|
from hermes_cli.auth import has_usable_secret
|
|
|
|
strong = "0123456789abcdef0123456789abcdef"
|
|
assert has_usable_secret(strong, min_length=16) is True
|
|
adapter = APIServerAdapter(
|
|
PlatformConfig(enabled=True, extra={"host": "0.0.0.0", "key": strong})
|
|
)
|
|
# The credential guard itself accepts the key (start may still fail on
|
|
# later env-specific steps, which is out of scope for this guard test).
|
|
assert adapter._api_key == strong
|