4d9db750be
- pyproject.toml: ruff target-version py311 → py313 (auto-fix 42 lint warnings via UP rules); aggiunto consider_namespace_packages = true che risolve la collisione conftest tra servizi e permette di lanciare pytest sull'intera suite cross-servizio. - mcp_common.audit: nuovo helper audit_write_op() con logger dedicato mcp.audit. Wirato su tutti i write endpoint di deribit, bybit, alpaca e hyperliquid (place_order, place_combo_order, cancel_*, set_*, close_*, transfer_*, switch_*, amend_*) con principal + target + payload non-sensibile + result summarizzato. - mcp_common.app_factory: ExchangeAppSpec + run_exchange_main() centralizza il boilerplate dei __main__.py (configure_root_logging, fail_fast_if_missing, summarize, load creds, resolve_environment, load token store, uvicorn). I 4 __main__.py exchange ridotti da ~60 LOC ognuno a ~25 LOC dichiarativi. mcp_common.env_validation promosso da mcp_deribit (mantenuto re-export shim per back-compat test_env_validation). - 8 test nuovi (4 audit + 4 app_factory). Suite full: 450/450 verdi. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
117 lines
3.5 KiB
Python
117 lines
3.5 KiB
Python
from __future__ import annotations
|
|
|
|
import pytest
|
|
from mcp_common.environment import resolve_environment
|
|
|
|
|
|
def test_env_var_overrides_secret(monkeypatch):
|
|
monkeypatch.setenv("DERIBIT_TESTNET", "false")
|
|
creds = {"testnet": True, "base_url_live": "L", "base_url_testnet": "T"}
|
|
info = resolve_environment(
|
|
creds,
|
|
env_var="DERIBIT_TESTNET",
|
|
flag_key="testnet",
|
|
exchange="deribit",
|
|
)
|
|
assert info.environment == "mainnet"
|
|
assert info.source == "env"
|
|
assert info.env_value == "false"
|
|
assert info.base_url == "L"
|
|
|
|
|
|
def test_secret_used_when_env_missing(monkeypatch):
|
|
monkeypatch.delenv("DERIBIT_TESTNET", raising=False)
|
|
creds = {"testnet": True, "base_url_live": "L", "base_url_testnet": "T"}
|
|
info = resolve_environment(
|
|
creds,
|
|
env_var="DERIBIT_TESTNET",
|
|
flag_key="testnet",
|
|
exchange="deribit",
|
|
)
|
|
assert info.environment == "testnet"
|
|
assert info.source == "credentials"
|
|
assert info.env_value is None
|
|
assert info.base_url == "T"
|
|
|
|
|
|
def test_default_when_both_missing(monkeypatch):
|
|
monkeypatch.delenv("FOO_TESTNET", raising=False)
|
|
creds = {"base_url_live": "L", "base_url_testnet": "T"}
|
|
info = resolve_environment(
|
|
creds,
|
|
env_var="FOO_TESTNET",
|
|
flag_key="testnet",
|
|
exchange="foo",
|
|
)
|
|
assert info.environment == "testnet"
|
|
assert info.source == "default"
|
|
assert info.env_value is None
|
|
|
|
|
|
@pytest.mark.parametrize("raw,expected", [
|
|
("1", "testnet"),
|
|
("true", "testnet"),
|
|
("yes", "testnet"),
|
|
("on", "testnet"),
|
|
("TRUE", "testnet"),
|
|
("0", "mainnet"),
|
|
("false", "mainnet"),
|
|
("no", "mainnet"),
|
|
("off", "mainnet"),
|
|
("garbage", "mainnet"),
|
|
])
|
|
def test_env_value_truthy_parsing(monkeypatch, raw, expected):
|
|
monkeypatch.setenv("X_TESTNET", raw)
|
|
info = resolve_environment(
|
|
{"base_url_live": "L", "base_url_testnet": "T"},
|
|
env_var="X_TESTNET",
|
|
flag_key="testnet",
|
|
exchange="x",
|
|
)
|
|
assert info.environment == expected
|
|
|
|
|
|
def test_default_base_urls_applied_when_creds_missing(monkeypatch):
|
|
monkeypatch.delenv("X_TESTNET", raising=False)
|
|
creds: dict = {}
|
|
info = resolve_environment(
|
|
creds,
|
|
env_var="X_TESTNET",
|
|
flag_key="testnet",
|
|
exchange="x",
|
|
default_base_url_live="https://live.example",
|
|
default_base_url_testnet="https://test.example",
|
|
)
|
|
assert info.base_url == "https://test.example"
|
|
assert creds["base_url_live"] == "https://live.example"
|
|
assert creds["base_url_testnet"] == "https://test.example"
|
|
|
|
|
|
def test_creds_base_urls_override_defaults(monkeypatch):
|
|
monkeypatch.delenv("X_TESTNET", raising=False)
|
|
creds = {"base_url_live": "L", "base_url_testnet": "T"}
|
|
info = resolve_environment(
|
|
creds,
|
|
env_var="X_TESTNET",
|
|
flag_key="testnet",
|
|
exchange="x",
|
|
default_base_url_live="https://live.example",
|
|
default_base_url_testnet="https://test.example",
|
|
)
|
|
assert info.base_url == "T"
|
|
assert creds["base_url_live"] == "L"
|
|
|
|
|
|
def test_alpaca_paper_flag_key(monkeypatch):
|
|
"""Alpaca usa 'paper' invece di 'testnet' nel secret."""
|
|
monkeypatch.delenv("ALPACA_PAPER", raising=False)
|
|
creds = {"paper": False, "base_url_live": "L", "base_url_testnet": "T"}
|
|
info = resolve_environment(
|
|
creds,
|
|
env_var="ALPACA_PAPER",
|
|
flag_key="paper",
|
|
exchange="alpaca",
|
|
)
|
|
assert info.environment == "mainnet"
|
|
assert info.source == "credentials"
|