from __future__ import annotations from pathlib import Path import yaml ROOT = Path(__file__).resolve().parents[1] def _compose(path: str) -> dict: return yaml.safe_load((ROOT / path).read_text(encoding="utf-8")) def test_prod_compose_uses_registry_image_not_local_build(): prod = _compose("docker-compose.prod.yml") service = prod["services"]["matrix-bot"] assert "image" in service assert "build" not in service assert service["image"].startswith("${SURFACES_BOT_IMAGE:?") def test_fullstack_compose_keeps_local_dev_build_with_agent_api_context(): fullstack = _compose("docker-compose.fullstack.yml") service = fullstack["services"]["matrix-bot"] assert service["build"]["target"] == "development" assert service["build"]["additional_contexts"]["agent_api"] == "./external/platform-agent_api" assert service["extends"]["file"] == "docker-compose.prod.yml" def test_dockerfile_production_build_does_not_require_local_external_tree(): dockerfile = (ROOT / "Dockerfile").read_text(encoding="utf-8") assert "/app/external/platform-agent_api" not in dockerfile assert "external/platform-agent_api" not in dockerfile assert "git+https://git.lambda.coredump.ru/platform/agent_api.git" in dockerfile assert "python -m pip install --no-cache-dir --ignore-requires-python" in dockerfile assert "uv pip install --system --ignore-requires-python" not in dockerfile def test_dockerfile_installs_agent_api_after_final_uv_sync(): dockerfile = (ROOT / "Dockerfile").read_text(encoding="utf-8") development = dockerfile.split("FROM base AS development", maxsplit=1)[1].split( "FROM base AS production", maxsplit=1 )[0] production = dockerfile.split("FROM base AS production", maxsplit=1)[1] assert development.index("RUN uv sync --no-dev --frozen") < development.index( "pip install --no-cache-dir --ignore-requires-python -e /agent_api/" ) assert production.index("RUN uv sync --no-dev --frozen") < production.index( "git+https://git.lambda.coredump.ru/platform/agent_api.git" ) def test_dockerignore_excludes_local_only_and_runtime_artifacts(): dockerignore = (ROOT / ".dockerignore").read_text(encoding="utf-8") assert "external/" in dockerignore assert ".planning/" in dockerignore assert "config/matrix-agents.yaml" in dockerignore assert ".env" in dockerignore def test_agent_registry_example_documents_multi_agent_volume_contract(): registry = yaml.safe_load( (ROOT / "config" / "matrix-agents.example.yaml").read_text(encoding="utf-8") ) agents = registry["agents"] assert len(agents) >= 3 assert len({agent["id"] for agent in agents}) == len(agents) assert len({agent["workspace_path"] for agent in agents}) == len(agents) for index, agent in enumerate(agents): assert agent["base_url"].endswith(f"/agent_{index}/") assert agent["workspace_path"] == f"/agents/{index}" def test_smoke_compose_models_deploy_like_proxy_and_surface_checker(): smoke = _compose("docker-compose.smoke.yml") assert set(smoke["services"]) >= {"surface-smoke", "agent-proxy", "agent-0", "agent-1"} assert "tools.check_matrix_agents" in smoke["services"]["surface-smoke"]["command"] assert smoke["services"]["agent-proxy"]["ports"] == ["${SMOKE_PROXY_PORT:-7000}:7000"] def test_smoke_timeout_override_routes_one_agent_to_no_status_stub(): smoke_timeout = _compose("docker-compose.smoke.timeout.yml") assert set(smoke_timeout["services"]) >= {"agent-proxy", "agent-no-status"} def test_smoke_registry_targets_local_proxy_routes(): registry = yaml.safe_load( (ROOT / "config" / "matrix-agents.smoke.yaml").read_text(encoding="utf-8") ) assert [agent["base_url"] for agent in registry["agents"]] == [ "http://agent-proxy:7000/agent_0/", "http://agent-proxy:7000/agent_1/", ]