605f95098c
Part 1 — long-only. The momentum edge is long top-momentum; the gate was
qualifying shorts on high-momentum names (fighting the trend), which showed as
the -0.13R Short(qual.) drag. While the gate is active, shorts no longer qualify
(backend qualification, backtest _momentum_qualifies, and the frontend mirror).
Part 2 — production wiring. Live setups now carry a real momentum rank, so the
dashboard, the Track Record's qualified stats, and outcome evaluation all gate on
the same value instead of deferring to floors:
- new momentum_service.compute_momentum_percentiles: 12-1 momentum per ticker,
ranked across the universe into a {symbol: percentile} map.
- the daily R:R scan ranks the universe up front and stores each setup's
percentile (new trade_setups.momentum_percentile column, migration 010).
- enhance_trade_setup mutates the same row, so the percentile is preserved;
_trade_setup_to_dict + TradeSetupResponse expose it to the API.
Until a fresh scan runs, pre-existing setups have a null percentile and the gate
falls back to floors for them (longs) / excludes them (shorts) — they fill in on
the next scan. 341 backend tests pass; frontend build clean.
Needs the alembic upgrade (migration 010) on deploy.
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
53 lines
1.4 KiB
Python
53 lines
1.4 KiB
Python
"""Pydantic schemas for trade setup endpoints."""
|
|
|
|
from __future__ import annotations
|
|
|
|
from datetime import date, datetime
|
|
|
|
from pydantic import BaseModel, Field
|
|
|
|
|
|
class TradeTargetResponse(BaseModel):
|
|
price: float
|
|
distance_from_entry: float
|
|
distance_atr_multiple: float
|
|
rr_ratio: float
|
|
probability: float
|
|
classification: str
|
|
sr_level_id: int
|
|
sr_strength: float
|
|
is_primary: bool = False
|
|
|
|
|
|
class RecommendationSummaryResponse(BaseModel):
|
|
action: str
|
|
reasoning: str | None
|
|
risk_level: str | None
|
|
composite_score: float
|
|
|
|
|
|
class TradeSetupResponse(BaseModel):
|
|
"""A single trade setup detected by the R:R scanner."""
|
|
|
|
id: int
|
|
symbol: str
|
|
direction: str
|
|
entry_price: float
|
|
stop_loss: float
|
|
target: float
|
|
rr_ratio: float
|
|
composite_score: float
|
|
detected_at: datetime
|
|
confidence_score: float | None = None
|
|
targets: list[TradeTargetResponse] = Field(default_factory=list)
|
|
conflict_flags: list[str] = Field(default_factory=list)
|
|
recommended_action: str | None = None
|
|
reasoning: str | None = None
|
|
risk_level: str | None = None
|
|
actual_outcome: str | None = None
|
|
outcome_date: date | None = None
|
|
evaluated_at: datetime | None = None
|
|
current_price: float | None = None
|
|
momentum_percentile: float | None = None
|
|
recommendation_summary: RecommendationSummaryResponse | None = None
|