Phase 4 hardening: status CLI, lock file, backup job, hash enforce, pooling, real bias

Sei interventi mirati sui rischi operativi rilevati nell'audit
post-Fase 4. 317 test pass, mypy strict pulito, ruff clean.

1. status CLI: legge SQLite reale e mostra kill_switch, posizioni
   aperte, environment, config_version, last_health_check, started_at.
   Sostituisce il placeholder "phase 0 skeleton".

2. Lock file single-instance: runtime/lockfile.py acquisisce
   data/.lockfile via fcntl.flock al boot di run_forever; un secondo
   container fallisce subito con LockError.

3. Backup orario nello scheduler: nuovo job APScheduler 0 * * * *
   chiama scripts.backup.backup_database + prune_backups.

4. config_hash enforce su start: il CLI start verifica l'integrità
   del file (enforce_hash=True). Mismatch → exit 1 prima di toccare
   stato. dry-run resta enforce_hash=False per debug.

5. Connection pooling MCP: RuntimeContext espone un httpx.AsyncClient
   long-lived condiviso da tutti i wrapper (limits 20/10
   connections/keepalive). aclose() chiamato in run_forever finale.

6. Bias direzionale reale: deribit.historical_close +
   deribit.adx_14 popolano TrendContext con spot a 30 giorni e
   ADX(14) effettivi. Sblocca bull_put e bear_call. Quando i dati
   storici mancano l'engine emette alert MEDIUM e cade su no_entry
   in modo deterministico.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
2026-04-28 00:15:28 +02:00
parent ca1e6379df
commit 411b747e93
11 changed files with 439 additions and 36 deletions
+29 -3
View File
@@ -27,15 +27,41 @@ def test_cli_help_lists_status_command() -> None:
assert "status" in result.output
def test_cli_status_runs(tmp_data_dir: Path) -> None:
def test_cli_status_when_state_missing(tmp_data_dir: Path) -> None:
runner = CliRunner()
result = runner.invoke(
cli_main,
["--log-dir", str(tmp_data_dir / "log"), "status"],
[
"--log-dir",
str(tmp_data_dir / "log"),
"status",
"--db",
str(tmp_data_dir / "missing.sqlite"),
],
)
assert result.exit_code == 0
assert "Cerbero Bite" in result.output
assert "phase: 0" in result.output
assert "never started" in result.output
def test_cli_status_after_kill_switch_arm(tmp_data_dir: Path) -> None:
runner = CliRunner()
db_path = tmp_data_dir / "state.sqlite"
audit_path = tmp_data_dir / "audit.log"
runner.invoke(
cli_main,
[
"--log-dir", str(tmp_data_dir / "log"),
"kill-switch", "arm",
"--reason", "smoke",
"--db", str(db_path),
"--audit", str(audit_path),
],
)
result = runner.invoke(cli_main, ["status", "--db", str(db_path)])
assert result.exit_code == 0
assert "ARMED" in result.output
assert "open positions: 0" in result.output
def test_cli_kill_switch_arm_persists_state(tmp_data_dir: Path) -> None: