- HTTP: configurable server_header, response_code, fake_app presets (apache/nginx/wordpress/phpmyadmin/iis), extra_headers, custom_body, static files directory mount - SSH/Cowrie: configurable kernel_version, hardware_platform, ssh_banner, and users/passwords via COWRIE_USERDB_ENTRIES; switched to build mode so cowrie.cfg.j2 persona fields and userdb.txt generation work - SMTP: configurable banner and MTA hostname - MySQL: configurable version string in protocol greeting - Redis: configurable redis_version and os string in INFO response - BYOS: [custom-*] INI sections define bring-your-own Docker services - Stealth: rename all *_honeypot.py → server.py; replace HONEYPOT_NAME env var with NODE_NAME across all 22+ service templates and plugins; strip "honeypot" from all in-container file content - Config: DeckyConfig.service_config dict; INI [decky-N.svc] subsections; composer passes service_cfg to compose_fragment - 350 tests passing (100%) Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
244 lines
9.0 KiB
Python
244 lines
9.0 KiB
Python
"""
|
|
Tests for the composer — verifies BASE_IMAGE injection and distro heterogeneity.
|
|
"""
|
|
|
|
import pytest
|
|
from decnet.config import DeckyConfig, DecnetConfig
|
|
from decnet.composer import generate_compose
|
|
from decnet.distros import all_distros, DISTROS
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
# Helpers
|
|
# ---------------------------------------------------------------------------
|
|
|
|
APT_COMPATIBLE = {
|
|
"debian:bookworm-slim",
|
|
"ubuntu:22.04",
|
|
"ubuntu:20.04",
|
|
"kalilinux/kali-rolling",
|
|
}
|
|
|
|
BUILD_SERVICES = [
|
|
"ssh", "http", "rdp", "smb", "ftp", "smtp", "elasticsearch",
|
|
"pop3", "imap", "mysql", "mssql", "redis", "mongodb", "postgres",
|
|
"ldap", "vnc", "docker_api", "k8s", "sip",
|
|
"mqtt", "llmnr", "snmp", "tftp",
|
|
]
|
|
|
|
UPSTREAM_SERVICES = ["telnet", "conpot"]
|
|
|
|
|
|
def _make_config(services, distro="debian", base_image=None, build_base=None):
|
|
profile = DISTROS[distro]
|
|
decky = DeckyConfig(
|
|
name="decky-01",
|
|
ip="10.0.0.10",
|
|
services=services,
|
|
distro=distro,
|
|
base_image=base_image or profile.image,
|
|
build_base=build_base or profile.build_base,
|
|
hostname="test-host",
|
|
)
|
|
return DecnetConfig(
|
|
mode="unihost",
|
|
interface="eth0",
|
|
subnet="10.0.0.0/24",
|
|
gateway="10.0.0.1",
|
|
deckies=[decky],
|
|
)
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
# BASE_IMAGE injection — build services
|
|
# ---------------------------------------------------------------------------
|
|
|
|
@pytest.mark.parametrize("svc", BUILD_SERVICES)
|
|
def test_build_service_gets_base_image_arg(svc):
|
|
"""Every build service must have BASE_IMAGE injected in compose args."""
|
|
config = _make_config([svc], distro="debian")
|
|
compose = generate_compose(config)
|
|
key = f"decky-01-{svc}"
|
|
fragment = compose["services"][key]
|
|
assert "build" in fragment, f"{svc}: missing 'build' key"
|
|
assert "args" in fragment["build"], f"{svc}: build section missing 'args'"
|
|
assert "BASE_IMAGE" in fragment["build"]["args"], f"{svc}: BASE_IMAGE not in args"
|
|
|
|
|
|
@pytest.mark.parametrize("distro,expected_build_base", [
|
|
("debian", "debian:bookworm-slim"),
|
|
("ubuntu22", "ubuntu:22.04"),
|
|
("ubuntu20", "ubuntu:20.04"),
|
|
("kali", "kalilinux/kali-rolling"),
|
|
("rocky9", "debian:bookworm-slim"),
|
|
("alpine", "debian:bookworm-slim"),
|
|
])
|
|
def test_build_service_base_image_matches_distro(distro, expected_build_base):
|
|
"""BASE_IMAGE arg must match the distro's build_base."""
|
|
config = _make_config(["http"], distro=distro)
|
|
compose = generate_compose(config)
|
|
fragment = compose["services"]["decky-01-http"]
|
|
assert fragment["build"]["args"]["BASE_IMAGE"] == expected_build_base
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
# BASE_IMAGE NOT injected for upstream-image services
|
|
# ---------------------------------------------------------------------------
|
|
|
|
@pytest.mark.parametrize("svc", UPSTREAM_SERVICES)
|
|
def test_upstream_service_has_no_build_section(svc):
|
|
"""Upstream-image services must not receive a build section or BASE_IMAGE."""
|
|
config = _make_config([svc])
|
|
compose = generate_compose(config)
|
|
fragment = compose["services"][f"decky-01-{svc}"]
|
|
assert "build" not in fragment
|
|
assert "image" in fragment
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
# service_config propagation tests
|
|
# ---------------------------------------------------------------------------
|
|
|
|
def test_service_config_http_server_header():
|
|
"""service_config for http must inject SERVER_HEADER into compose env."""
|
|
from decnet.config import DeckyConfig, DecnetConfig
|
|
from decnet.distros import DISTROS
|
|
profile = DISTROS["debian"]
|
|
decky = DeckyConfig(
|
|
name="decky-01", ip="10.0.0.10",
|
|
services=["http"], distro="debian",
|
|
base_image=profile.image, build_base=profile.build_base,
|
|
hostname="test-host",
|
|
service_config={"http": {"server_header": "nginx/1.18.0"}},
|
|
)
|
|
config = DecnetConfig(
|
|
mode="unihost", interface="eth0",
|
|
subnet="10.0.0.0/24", gateway="10.0.0.1",
|
|
deckies=[decky],
|
|
)
|
|
compose = generate_compose(config)
|
|
env = compose["services"]["decky-01-http"]["environment"]
|
|
assert env.get("SERVER_HEADER") == "nginx/1.18.0"
|
|
|
|
|
|
def test_service_config_ssh_kernel_version():
|
|
"""service_config for ssh must inject COWRIE_HONEYPOT_KERNEL_VERSION."""
|
|
from decnet.config import DeckyConfig, DecnetConfig
|
|
from decnet.distros import DISTROS
|
|
profile = DISTROS["debian"]
|
|
decky = DeckyConfig(
|
|
name="decky-01", ip="10.0.0.10",
|
|
services=["ssh"], distro="debian",
|
|
base_image=profile.image, build_base=profile.build_base,
|
|
hostname="test-host",
|
|
service_config={"ssh": {"kernel_version": "5.15.0-76-generic"}},
|
|
)
|
|
config = DecnetConfig(
|
|
mode="unihost", interface="eth0",
|
|
subnet="10.0.0.0/24", gateway="10.0.0.1",
|
|
deckies=[decky],
|
|
)
|
|
compose = generate_compose(config)
|
|
env = compose["services"]["decky-01-ssh"]["environment"]
|
|
assert env.get("COWRIE_HONEYPOT_KERNEL_VERSION") == "5.15.0-76-generic"
|
|
|
|
|
|
def test_service_config_for_one_service_does_not_affect_another():
|
|
"""service_config for http must not bleed into ftp fragment."""
|
|
from decnet.config import DeckyConfig, DecnetConfig
|
|
from decnet.distros import DISTROS
|
|
profile = DISTROS["debian"]
|
|
decky = DeckyConfig(
|
|
name="decky-01", ip="10.0.0.10",
|
|
services=["http", "ftp"], distro="debian",
|
|
base_image=profile.image, build_base=profile.build_base,
|
|
hostname="test-host",
|
|
service_config={"http": {"server_header": "nginx/1.18.0"}},
|
|
)
|
|
config = DecnetConfig(
|
|
mode="unihost", interface="eth0",
|
|
subnet="10.0.0.0/24", gateway="10.0.0.1",
|
|
deckies=[decky],
|
|
)
|
|
compose = generate_compose(config)
|
|
ftp_env = compose["services"]["decky-01-ftp"]["environment"]
|
|
assert "SERVER_HEADER" not in ftp_env
|
|
|
|
|
|
def test_no_service_config_produces_no_extra_env():
|
|
"""A decky with no service_config must not have new persona env vars."""
|
|
config = _make_config(["http", "mysql"])
|
|
compose = generate_compose(config)
|
|
for svc in ("http", "mysql"):
|
|
env = compose["services"][f"decky-01-{svc}"]["environment"]
|
|
assert "SERVER_HEADER" not in env
|
|
assert "MYSQL_VERSION" not in env
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
# Base container uses distro image, not build_base
|
|
# ---------------------------------------------------------------------------
|
|
|
|
@pytest.mark.parametrize("distro", list(DISTROS.keys()))
|
|
def test_base_container_uses_full_distro_image(distro):
|
|
"""The IP-holder base container must use distro.image, not build_base."""
|
|
config = _make_config(["ssh"], distro=distro)
|
|
compose = generate_compose(config)
|
|
base = compose["services"]["decky-01"]
|
|
expected = DISTROS[distro].image
|
|
assert base["image"] == expected, (
|
|
f"distro={distro}: base container image '{base['image']}' != '{expected}'"
|
|
)
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
# Distro profile — build_base is always apt-compatible
|
|
# ---------------------------------------------------------------------------
|
|
|
|
def test_all_distros_have_build_base():
|
|
for slug, profile in all_distros().items():
|
|
assert profile.build_base, f"Distro '{slug}' has empty build_base"
|
|
|
|
|
|
def test_all_distro_build_bases_are_apt_compatible():
|
|
for slug, profile in all_distros().items():
|
|
assert profile.build_base in APT_COMPATIBLE, (
|
|
f"Distro '{slug}' build_base '{profile.build_base}' is not apt-compatible. "
|
|
f"Allowed: {APT_COMPATIBLE}"
|
|
)
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
# Heterogeneity — multiple deckies with different distros get different images
|
|
# ---------------------------------------------------------------------------
|
|
|
|
def test_multiple_deckies_different_build_bases():
|
|
"""A multi-decky deployment with ubuntu22 and debian must differ in BASE_IMAGE."""
|
|
deckies = [
|
|
DeckyConfig(
|
|
name="decky-01", ip="10.0.0.10",
|
|
services=["http"], distro="debian",
|
|
base_image="debian:bookworm-slim", build_base="debian:bookworm-slim",
|
|
hostname="host-01",
|
|
),
|
|
DeckyConfig(
|
|
name="decky-02", ip="10.0.0.11",
|
|
services=["http"], distro="ubuntu22",
|
|
base_image="ubuntu:22.04", build_base="ubuntu:22.04",
|
|
hostname="host-02",
|
|
),
|
|
]
|
|
config = DecnetConfig(
|
|
mode="unihost", interface="eth0",
|
|
subnet="10.0.0.0/24", gateway="10.0.0.1",
|
|
deckies=deckies,
|
|
)
|
|
compose = generate_compose(config)
|
|
|
|
base_img_01 = compose["services"]["decky-01-http"]["build"]["args"]["BASE_IMAGE"]
|
|
base_img_02 = compose["services"]["decky-02-http"]["build"]["args"]["BASE_IMAGE"]
|
|
|
|
assert base_img_01 == "debian:bookworm-slim"
|
|
assert base_img_02 == "ubuntu:22.04"
|
|
assert base_img_01 != base_img_02
|