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>
52 lines
1.6 KiB
Python
52 lines
1.6 KiB
Python
from __future__ import annotations
|
|
|
|
import random
|
|
|
|
from mcp_common.stats import cointegration_test
|
|
|
|
|
|
def test_cointegrated_synthetic_pair():
|
|
"""Costruisco coppia cointegrata: B random walk, A = 2*B + noise stazionario."""
|
|
r = random.Random(1)
|
|
b = [100.0]
|
|
for _ in range(300):
|
|
b.append(b[-1] + r.gauss(0, 1))
|
|
a = [2 * b[i] + r.gauss(0, 0.5) for i in range(len(b))]
|
|
out = cointegration_test(a, b)
|
|
assert out["cointegrated"] is True
|
|
assert out["beta"] == pytest_approx(2.0, rel=0.05)
|
|
assert out["adf_t_stat"] is not None
|
|
assert out["adf_t_stat"] < -2.86
|
|
|
|
|
|
def test_not_cointegrated_independent_walks():
|
|
"""Due random walk indipendenti → spread non stazionario → no cointegration."""
|
|
r = random.Random(2)
|
|
a = [100.0]
|
|
b = [100.0]
|
|
for _ in range(300):
|
|
a.append(a[-1] + r.gauss(0, 1))
|
|
b.append(b[-1] + r.gauss(0, 1))
|
|
out = cointegration_test(a, b)
|
|
# Per due RW indipendenti, t-stat ADF è solitamente > -2.86 → non cointegrate
|
|
assert out["cointegrated"] is False or out["adf_t_stat"] > -3.0
|
|
|
|
|
|
def test_cointegration_short_series():
|
|
out = cointegration_test([1.0, 2.0], [3.0, 4.0])
|
|
assert out["cointegrated"] is None
|
|
assert out["beta"] is None
|
|
|
|
|
|
def test_cointegration_mismatched_length():
|
|
out = cointegration_test([1.0, 2.0, 3.0], [1.0, 2.0])
|
|
assert out["cointegrated"] is None
|
|
|
|
|
|
def pytest_approx(value, rel):
|
|
"""Tiny helper to avoid importing pytest just for approx."""
|
|
class _Approx:
|
|
def __eq__(self, other):
|
|
return abs(other - value) <= abs(value) * rel
|
|
return _Approx()
|