#!/usr/bin/env python3
"""
Simple script to fetch the top N upcoming Kalshi events using the public trial API
(unauthenticated). It mirrors but simplifies the prophet arena ingestion pipeline.

Example:
    $ python3 get_kalshi_events.py --limit 1
"""


import argparse
import os
from datetime import datetime, timedelta, timezone
from typing import Any, Dict, Iterable, List
import requests  # type: ignore
BASE_URL = os.getenv("KALSHI_BASE_URL", "https://api.elections.kalshi.com")
MIN_RUNWAY = timedelta(hours=6)


def kalshi_request(path: str, *, params: Dict[str, Any] | None = None) -> Dict[str, Any]:
    response = requests.get(
        f"{BASE_URL}{path}",
        params=params,
        timeout=30,
    )
    response.raise_for_status()
    return response.json()


def parse_close_time(value: Any) -> datetime | None:
    if not value:
        return None

    if isinstance(value, (int, float)):
        # Kalshi sometimes sends unix timestamps
        return datetime.fromtimestamp(float(value), tz=timezone.utc)

    if isinstance(value, str):
        text = value.strip()
        if not text:
            return None
        if text.endswith("Z"):
            text = text[:-1] + "+00:00"
        try:
            return datetime.fromisoformat(text)
        except ValueError:
            return None

    if isinstance(value, datetime):
        return value if value.tzinfo else value.replace(tzinfo=timezone.utc)

    return None


def collect_candidates(events: Iterable[Dict[str, Any]]) -> List[Dict[str, Any]]:
    now = datetime.now(timezone.utc)
    merged: Dict[str, Dict[str, Any]] = {}

    for event in events:
        ticker = event.get("event_ticker")
        if not ticker:
            continue

        nested_markets = event.get("markets") or []
        if not nested_markets:
            # fallback: treat event as a single market with aggregate fields
            nested_markets = [event]

        close_time = parse_close_time(event.get("close_time"))
        if not close_time:
            close_candidates = [
                parse_close_time(
                    market.get("expected_expiration_time") or market.get("close_time")
                )
                for market in nested_markets
            ]
            close_candidates = [candidate for candidate in close_candidates if candidate]
            close_time = min(close_candidates, default=None)

        if not close_time or close_time - now < MIN_RUNWAY:
            continue

        merged.setdefault(
            ticker,
            {
                "title": event.get("title") or ticker,
                "close_time": close_time,
                "open_interest": 0.0,
                "volume": 0.0,
                "markets": set(),
            },
        )

        entry = merged[ticker]
        entry["close_time"] = min(entry["close_time"], close_time)

        for market in nested_markets:
            entry["open_interest"] += float(market.get("open_interest") or 0)
            entry["volume"] += float(market.get("volume") or 0)
            subtitle = market.get("yes_sub_title") or market.get("title")
            if subtitle:
                entry["markets"].add(subtitle)

    summaries = []
    for ticker, data in merged.items():
        summaries.append({
            "event_ticker": ticker,
            "title": data["title"],
            "close_time": data["close_time"],
            "open_interest": data["open_interest"],
            "volume": data["volume"],
            "markets": sorted(data["markets"])[:5],
        })
    return summaries


def fetch_top_events(limit: int) -> List[Dict[str, Any]]:
    payload = kalshi_request(
        "/trade-api/v2/events",
        params={
            "status": "open",
            "limit": 200,
            "with_nested_markets": "true",
        },
    )
    events = payload.get("events") or []
    candidates = collect_candidates(events)
    candidates.sort(
        key=lambda e: (
            -e["open_interest"],
            -e["volume"],
            e["close_time"],
        )
    )
    return candidates[:limit]


def main() -> None:
    parser = argparse.ArgumentParser(description="Fetch top Kalshi events.")
    parser.add_argument("--limit", type=int, default=5, help="number of events to show")
    args = parser.parse_args()
    events = fetch_top_events(args.limit)
    for idx, event in enumerate(events, start=1):
        print(f"{idx}. {event['event_ticker']} — {event['title']}")
        print(
            f"   close: {event['close_time'].isoformat()} | "
            f"open interest: {event['open_interest']:.0f} | "
            f"volume: {event['volume']:.0f}"
        )
        for market in event["markets"]:
            print(f"   • {market}")


if __name__ == "__main__":
    main()

