from datetime import datetime from sqlalchemy import Boolean, Date, DateTime, Float, ForeignKey, Index, Integer, String, Text, UniqueConstraint from sqlalchemy.dialects.postgresql import JSONB from sqlalchemy.orm import Mapped, mapped_column, relationship from sqlalchemy.types import JSON from app.core.database import Base JsonType = JSON().with_variant(JSONB, "postgresql") class Asset(Base): __tablename__ = "assets" id: Mapped[int] = mapped_column(Integer, primary_key=True) ticker: Mapped[str] = mapped_column(String(32), unique=True, index=True) name: Mapped[str] = mapped_column(String(220)) category: Mapped[str] = mapped_column(String(80), index=True) sector: Mapped[str] = mapped_column(String(120), index=True) industry: Mapped[str] = mapped_column(String(160), default="") country: Mapped[str] = mapped_column(String(80), index=True) asset_type: Mapped[str] = mapped_column(String(24), index=True) currency: Mapped[str] = mapped_column(String(16), default="USD") exchange: Mapped[str] = mapped_column(String(40), default="") description: Mapped[str] = mapped_column(Text, default="") is_active: Mapped[bool] = mapped_column(Boolean, default=True, index=True) created_at: Mapped[datetime] = mapped_column(DateTime, default=datetime.utcnow) updated_at: Mapped[datetime] = mapped_column(DateTime, default=datetime.utcnow, onupdate=datetime.utcnow) prices = relationship("PriceHistory", back_populates="asset", cascade="all, delete-orphan") signals = relationship("SignalSnapshot", back_populates="asset", cascade="all, delete-orphan") class PriceHistory(Base): __tablename__ = "price_history" __table_args__ = ( UniqueConstraint("asset_id", "date", name="uq_price_asset_date"), Index("ix_price_asset_date", "asset_id", "date"), ) id: Mapped[int] = mapped_column(Integer, primary_key=True) asset_id: Mapped[int] = mapped_column(ForeignKey("assets.id", ondelete="CASCADE"), index=True) date: Mapped[datetime] = mapped_column(Date, index=True) open: Mapped[float | None] = mapped_column(Float) high: Mapped[float | None] = mapped_column(Float) low: Mapped[float | None] = mapped_column(Float) close: Mapped[float] = mapped_column(Float) volume: Mapped[float | None] = mapped_column(Float) provider: Mapped[str] = mapped_column(String(40), default="yfinance") created_at: Mapped[datetime] = mapped_column(DateTime, default=datetime.utcnow) asset = relationship("Asset", back_populates="prices") class NewsArticle(Base): __tablename__ = "news_articles" id: Mapped[int] = mapped_column(Integer, primary_key=True) source: Mapped[str] = mapped_column(String(160), index=True) source_url: Mapped[str] = mapped_column(Text, default="") published_at: Mapped[datetime | None] = mapped_column(DateTime, index=True) title: Mapped[str] = mapped_column(Text) summary: Mapped[str] = mapped_column(Text, default="") body: Mapped[str] = mapped_column(Text, default="") url: Mapped[str] = mapped_column(Text, unique=True) canonical_key: Mapped[str] = mapped_column(String(260), unique=True, index=True) quality_score: Mapped[float] = mapped_column(Float, default=0.0) theme_tags: Mapped[dict] = mapped_column(JsonType, default=dict) created_at: Mapped[datetime] = mapped_column(DateTime, default=datetime.utcnow) links = relationship("NewsAssetLink", back_populates="article", cascade="all, delete-orphan") sentiments = relationship("SentimentAnalysis", back_populates="article", cascade="all, delete-orphan") class NewsAssetLink(Base): __tablename__ = "news_asset_links" __table_args__ = (UniqueConstraint("article_id", "asset_id", name="uq_news_asset"),) id: Mapped[int] = mapped_column(Integer, primary_key=True) article_id: Mapped[int] = mapped_column(ForeignKey("news_articles.id", ondelete="CASCADE"), index=True) asset_id: Mapped[int] = mapped_column(ForeignKey("assets.id", ondelete="CASCADE"), index=True) relevance_score: Mapped[float] = mapped_column(Float, default=0.0) article = relationship("NewsArticle", back_populates="links") asset = relationship("Asset") class SentimentAnalysis(Base): __tablename__ = "sentiment_analysis" id: Mapped[int] = mapped_column(Integer, primary_key=True) article_id: Mapped[int | None] = mapped_column(ForeignKey("news_articles.id", ondelete="CASCADE"), index=True) asset_id: Mapped[int | None] = mapped_column(ForeignKey("assets.id", ondelete="CASCADE"), index=True) model_name: Mapped[str] = mapped_column(String(120), index=True) label: Mapped[str] = mapped_column(String(40), index=True) score: Mapped[float] = mapped_column(Float) confidence: Mapped[float] = mapped_column(Float, default=0.0) baseline_vader: Mapped[float | None] = mapped_column(Float) raw_payload: Mapped[dict] = mapped_column(JsonType, default=dict) created_at: Mapped[datetime] = mapped_column(DateTime, default=datetime.utcnow) article = relationship("NewsArticle", back_populates="sentiments") asset = relationship("Asset") class TechnicalIndicator(Base): __tablename__ = "technical_indicators" __table_args__ = (UniqueConstraint("asset_id", "date", name="uq_indicator_asset_date"),) id: Mapped[int] = mapped_column(Integer, primary_key=True) asset_id: Mapped[int] = mapped_column(ForeignKey("assets.id", ondelete="CASCADE"), index=True) date: Mapped[datetime] = mapped_column(Date, index=True) indicators: Mapped[dict] = mapped_column(JsonType, default=dict) created_at: Mapped[datetime] = mapped_column(DateTime, default=datetime.utcnow) asset = relationship("Asset") class SignalSnapshot(Base): __tablename__ = "signal_snapshots" id: Mapped[int] = mapped_column(Integer, primary_key=True) asset_id: Mapped[int] = mapped_column(ForeignKey("assets.id", ondelete="CASCADE"), index=True) ticker: Mapped[str] = mapped_column(String(32), index=True) classification: Mapped[str] = mapped_column(String(80), index=True) blum_score: Mapped[float] = mapped_column(Float, index=True) risk_level: Mapped[str] = mapped_column(String(40), index=True) time_horizon: Mapped[str] = mapped_column(String(80), default="Short/Medium term") score_breakdown: Mapped[dict] = mapped_column(JsonType, default=dict) technical_summary: Mapped[dict] = mapped_column(JsonType, default=dict) narrative_summary: Mapped[dict] = mapped_column(JsonType, default=dict) explanation: Mapped[str] = mapped_column(Text, default="") watch_points: Mapped[dict] = mapped_column(JsonType, default=dict) created_at: Mapped[datetime] = mapped_column(DateTime, default=datetime.utcnow, index=True) asset = relationship("Asset", back_populates="signals") class ThemeCluster(Base): __tablename__ = "theme_clusters" id: Mapped[int] = mapped_column(Integer, primary_key=True) label: Mapped[str] = mapped_column(String(120), index=True) keywords: Mapped[dict] = mapped_column(JsonType, default=dict) article_ids: Mapped[dict] = mapped_column(JsonType, default=dict) asset_tickers: Mapped[dict] = mapped_column(JsonType, default=dict) centroid: Mapped[dict] = mapped_column(JsonType, default=dict) sentiment_score: Mapped[float] = mapped_column(Float, default=0.0) created_at: Mapped[datetime] = mapped_column(DateTime, default=datetime.utcnow, index=True) class EmbeddingVector(Base): __tablename__ = "embedding_vectors" id: Mapped[int] = mapped_column(Integer, primary_key=True) article_id: Mapped[int | None] = mapped_column(ForeignKey("news_articles.id", ondelete="CASCADE"), index=True) asset_id: Mapped[int | None] = mapped_column(ForeignKey("assets.id", ondelete="CASCADE"), index=True) model_name: Mapped[str] = mapped_column(String(160), index=True) vector: Mapped[dict] = mapped_column(JsonType, default=dict) created_at: Mapped[datetime] = mapped_column(DateTime, default=datetime.utcnow) class AIInsight(Base): __tablename__ = "ai_insights" id: Mapped[int] = mapped_column(Integer, primary_key=True) asset_id: Mapped[int] = mapped_column(ForeignKey("assets.id", ondelete="CASCADE"), index=True) model_name: Mapped[str] = mapped_column(String(160), index=True) insight_type: Mapped[str] = mapped_column(String(80), default="asset_explanation") structured_output: Mapped[dict] = mapped_column(JsonType, default=dict) explanation: Mapped[str] = mapped_column(Text, default="") created_at: Mapped[datetime] = mapped_column(DateTime, default=datetime.utcnow, index=True) class ETFTrend(Base): __tablename__ = "etf_trends" id: Mapped[int] = mapped_column(Integer, primary_key=True) asset_id: Mapped[int] = mapped_column(ForeignKey("assets.id", ondelete="CASCADE"), index=True) ticker: Mapped[str] = mapped_column(String(32), index=True) category: Mapped[str] = mapped_column(String(120), index=True) momentum_score: Mapped[float] = mapped_column(Float, default=0.0) thematic_score: Mapped[float] = mapped_column(Float, default=0.0) confirmation_score: Mapped[float] = mapped_column(Float, default=0.0) details: Mapped[dict] = mapped_column(JsonType, default=dict) created_at: Mapped[datetime] = mapped_column(DateTime, default=datetime.utcnow, index=True) class BacktestResult(Base): __tablename__ = "backtest_results" id: Mapped[int] = mapped_column(Integer, primary_key=True) run_name: Mapped[str] = mapped_column(String(160), index=True) benchmark: Mapped[str] = mapped_column(String(32), default="SPY") parameters: Mapped[dict] = mapped_column(JsonType, default=dict) metrics: Mapped[dict] = mapped_column(JsonType, default=dict) results: Mapped[dict] = mapped_column(JsonType, default=dict) created_at: Mapped[datetime] = mapped_column(DateTime, default=datetime.utcnow, index=True)