Docker terminal sessions are secret-dark by default. This adds terminal.docker_forward_env as an explicit allowlist for env vars that may be forwarded into Docker containers. Values resolve from the current shell first, then fall back to ~/.hermes/.env. Only variables the user explicitly lists are forwarded — nothing is auto-exposed. Cherry-picked from PR #1449 by @teknium1, conflict-resolved onto current main. Fixes #1436 Supersedes #1439
87 lines
3.5 KiB
Python
87 lines
3.5 KiB
Python
"""Tests for _parse_env_var and _get_env_config env-var validation."""
|
|
|
|
import json
|
|
from unittest.mock import patch
|
|
|
|
import pytest
|
|
|
|
import sys
|
|
import tools.terminal_tool # noqa: F401 -- ensure module is loaded
|
|
_tt_mod = sys.modules["tools.terminal_tool"]
|
|
from tools.terminal_tool import _parse_env_var
|
|
|
|
|
|
class TestParseEnvVar:
|
|
"""Unit tests for _parse_env_var."""
|
|
|
|
# -- valid values work normally --
|
|
|
|
def test_valid_int(self):
|
|
with patch.dict("os.environ", {"TERMINAL_TIMEOUT": "300"}):
|
|
assert _parse_env_var("TERMINAL_TIMEOUT", "180") == 300
|
|
|
|
def test_valid_float(self):
|
|
with patch.dict("os.environ", {"TERMINAL_CONTAINER_CPU": "2.5"}):
|
|
assert _parse_env_var("TERMINAL_CONTAINER_CPU", "1", float, "number") == 2.5
|
|
|
|
def test_valid_json(self):
|
|
volumes = '["/host:/container"]'
|
|
with patch.dict("os.environ", {"TERMINAL_DOCKER_VOLUMES": volumes}):
|
|
result = _parse_env_var("TERMINAL_DOCKER_VOLUMES", "[]", json.loads, "valid JSON")
|
|
assert result == ["/host:/container"]
|
|
|
|
def test_get_env_config_parses_docker_forward_env_json(self):
|
|
with patch.dict("os.environ", {
|
|
"TERMINAL_ENV": "docker",
|
|
"TERMINAL_DOCKER_FORWARD_ENV": '["GITHUB_TOKEN", "NPM_TOKEN"]',
|
|
}, clear=False):
|
|
config = _tt_mod._get_env_config()
|
|
assert config["docker_forward_env"] == ["GITHUB_TOKEN", "NPM_TOKEN"]
|
|
|
|
def test_create_environment_passes_docker_forward_env(self):
|
|
fake_env = object()
|
|
with patch.object(_tt_mod, "_DockerEnvironment", return_value=fake_env) as mock_docker:
|
|
result = _tt_mod._create_environment(
|
|
"docker",
|
|
image="python:3.11",
|
|
cwd="/root",
|
|
timeout=180,
|
|
container_config={"docker_forward_env": ["GITHUB_TOKEN"]},
|
|
)
|
|
|
|
assert result is fake_env
|
|
assert mock_docker.call_args.kwargs["forward_env"] == ["GITHUB_TOKEN"]
|
|
|
|
def test_falls_back_to_default(self):
|
|
with patch.dict("os.environ", {}, clear=False):
|
|
# Remove the var if it exists, rely on default
|
|
import os
|
|
env = os.environ.copy()
|
|
env.pop("TERMINAL_TIMEOUT", None)
|
|
with patch.dict("os.environ", env, clear=True):
|
|
assert _parse_env_var("TERMINAL_TIMEOUT", "180") == 180
|
|
|
|
# -- invalid int raises ValueError with env var name --
|
|
|
|
def test_invalid_int_raises_with_var_name(self):
|
|
with patch.dict("os.environ", {"TERMINAL_TIMEOUT": "5m"}):
|
|
with pytest.raises(ValueError, match="TERMINAL_TIMEOUT"):
|
|
_parse_env_var("TERMINAL_TIMEOUT", "180")
|
|
|
|
def test_invalid_int_includes_bad_value(self):
|
|
with patch.dict("os.environ", {"TERMINAL_SSH_PORT": "ssh"}):
|
|
with pytest.raises(ValueError, match="ssh"):
|
|
_parse_env_var("TERMINAL_SSH_PORT", "22")
|
|
|
|
# -- invalid JSON raises ValueError with env var name --
|
|
|
|
def test_invalid_json_raises_with_var_name(self):
|
|
with patch.dict("os.environ", {"TERMINAL_DOCKER_VOLUMES": "/host:/container"}):
|
|
with pytest.raises(ValueError, match="TERMINAL_DOCKER_VOLUMES"):
|
|
_parse_env_var("TERMINAL_DOCKER_VOLUMES", "[]", json.loads, "valid JSON")
|
|
|
|
def test_invalid_json_includes_type_label(self):
|
|
with patch.dict("os.environ", {"TERMINAL_DOCKER_VOLUMES": "not json"}):
|
|
with pytest.raises(ValueError, match="valid JSON"):
|
|
_parse_env_var("TERMINAL_DOCKER_VOLUMES", "[]", json.loads, "valid JSON")
|