# llm_openai.py
import json
import os
import time
from pathlib import Path
from typing import Any, Dict, List, Optional, Tuple

from openai import OpenAI


# -----------------------------
# .env loading (simple + safe)
# -----------------------------
def load_env(env_path: Optional[str] = None) -> None:
    """
    Loads KEY=VALUE pairs from .env into os.environ (if not already set).
    Looks in:
      - explicit env_path (if provided)
      - same folder as this file
      - current working directory
    """
    candidates: List[Path] = []
    if env_path:
        candidates.append(Path(env_path).expanduser().resolve())

    here = Path(__file__).resolve().parent
    cwd = Path.cwd()
    candidates += [here / ".env", cwd / ".env"]

    for p in candidates:
        if p.exists() and p.is_file():
            for raw in p.read_text(encoding="utf-8").splitlines():
                line = raw.strip()
                if not line or line.startswith("#") or "=" not in line:
                    continue
                k, v = line.split("=", 1)
                k = k.strip()
                v = v.strip().strip('"').strip("'")
                if k and k not in os.environ:
                    os.environ[k] = v
            return


# -----------------------------
# Helpers
# -----------------------------
def _build_input(instructions: Optional[str], user_input: str) -> List[Dict[str, str]]:
    msgs: List[Dict[str, str]] = []
    if instructions and instructions.strip():
        msgs.append({"role": "system", "content": instructions})
    msgs.append({"role": "user", "content": user_input})
    return msgs


def _extract_output_text(resp: Any) -> str:
    """
    Extracts the first output_text content from Responses API response.
    """
    try:
        if hasattr(resp, "output") and resp.output:
            for item in resp.output:
                if getattr(item, "type", None) == "message":
                    contents = getattr(item, "content", None) or []
                    for c in contents:
                        if getattr(c, "type", None) == "output_text":
                            return getattr(c, "text", "") or ""
                        if getattr(c, "type", None) == "refusal":
                            # Safety refusal; surface it
                            return getattr(c, "refusal", "") or ""
        return str(resp)
    except Exception:
        return str(resp)


def _safe_json_parse(s: str) -> Dict[str, Any]:
    """
    Attempts strict parse; if it fails, tries extracting the first {...} block.
    """
    s = (s or "").strip()
    if not s:
        raise ValueError("Empty string; cannot parse JSON.")

    # Direct parse
    try:
        return json.loads(s)
    except Exception:
        pass

    # Try first JSON object in the text
    start = s.find("{")
    end = s.rfind("}")
    if start != -1 and end != -1 and end > start:
        return json.loads(s[start : end + 1])

    raise ValueError("Could not locate a JSON object in model output.")


def _ensure_mentions_json(instructions: str) -> str:
    """
    OpenAI JSON mode requires 'JSON' to appear somewhere in context.
    """
    instr = (instructions or "").strip()
    if "JSON" not in instr.upper():
        instr = (instr + "\n\nReturn JSON only.").strip()
    return instr


# -----------------------------
# Client
# -----------------------------
class OpenAIClient:
    def __init__(self, api_key: Optional[str] = None) -> None:
        load_env()
        key = api_key or os.getenv("OPENAI_API_KEY")
        if not key:
            raise RuntimeError(
                "OPENAI_API_KEY is not set. Create a .env next to config.py and set OPENAI_API_KEY=..."
            )
        self.client = OpenAI(api_key=key)

    def _responses_create_bulletproof(self, kwargs: Dict[str, Any]) -> Any:
        """
        Handles common param differences:
          - max_output_tokens vs max_completion_tokens
          - temperature not supported (rare)
        """
        try:
            return self.client.responses.create(**kwargs)
        except Exception as e:
            msg = str(e)

            # Token param fallback
            if "max_output_tokens" in kwargs and ("max_completion_tokens" in msg or "Unknown parameter" in msg):
                kwargs2 = dict(kwargs)
                kwargs2["max_completion_tokens"] = kwargs2.pop("max_output_tokens")
                try:
                    return self.client.responses.create(**kwargs2)
                except Exception as e2:
                    msg2 = str(e2)
                    # Temperature fallback on the second attempt
                    if "temperature" in kwargs2 and ("temperature" in msg2 or "Unknown parameter" in msg2):
                        kwargs3 = dict(kwargs2)
                        kwargs3.pop("temperature", None)
                        return self.client.responses.create(**kwargs3)
                    raise

            # Temperature fallback
            if "temperature" in kwargs and ("temperature" in msg or "Unknown parameter" in msg):
                kwargs2 = dict(kwargs)
                kwargs2.pop("temperature", None)
                return self.client.responses.create(**kwargs2)

            raise

    def text(
        self,
        model: str,
        instructions: Optional[str],
        user_input: str,
        max_tokens: int = 800,
        temperature: float = 0.2,
        retries: int = 2,
    ) -> str:
        last_err: Optional[Exception] = None
        for attempt in range(retries + 1):
            try:
                kwargs: Dict[str, Any] = {
                    "model": model,
                    "input": _build_input(instructions, user_input),
                    "max_output_tokens": int(max_tokens),
                    "temperature": float(temperature),
                }
                resp = self._responses_create_bulletproof(kwargs)
                return _extract_output_text(resp)
            except Exception as e:
                last_err = e
                time.sleep(0.4 * (attempt + 1))
        raise RuntimeError(f"OpenAI text failed: {last_err}")

    def _repair_json(
        self,
        model: str,
        bad_json_text: str,
        max_tokens: int = 1200,
        retries: int = 1,
    ) -> Dict[str, Any]:
        """
        Ask the model to FIX the JSON. Run at temperature 0 and JSON mode.
        """
        repair_instructions = _ensure_mentions_json(
            "You are a JSON repair tool. "
            "You will be given text that is intended to be a single JSON object but may be invalid. "
            "Return a corrected, valid JSON object ONLY. No markdown, no commentary."
        )
        repair_prompt = (
            "Fix the following so that it is valid JSON representing the same object.\n\n"
            "BEGIN\n"
            f"{bad_json_text}\n"
            "END\n"
        )

        last_err: Optional[Exception] = None
        for attempt in range(retries + 1):
            try:
                kwargs: Dict[str, Any] = {
                    "model": model,
                    "input": _build_input(repair_instructions, repair_prompt),
                    "max_output_tokens": int(max_tokens),
                    "temperature": 0.0,
                    "text": {"format": {"type": "json_object"}},
                }
                resp = self._responses_create_bulletproof(kwargs)
                txt = _extract_output_text(resp)
                return _safe_json_parse(txt)
            except Exception as e:
                last_err = e
                time.sleep(0.4 * (attempt + 1))

        raise RuntimeError(f"OpenAI JSON repair failed: {last_err}")

    def json_object(
        self,
        model: str,
        instructions: Optional[str],
        user_input: str,
        max_tokens: int = 1200,
        temperature: float = 0.2,
        retries: int = 2,
    ) -> Dict[str, Any]:
        """
        Returns a Python dict.
        Strategy:
          1) Request JSON mode (valid JSON).
          2) Parse. If parse fails, run a repair pass.
          3) Retry original call a couple of times if needed.
        """
        instr = _ensure_mentions_json(instructions or "")

        last_err: Optional[Exception] = None
        last_txt: str = ""

        for attempt in range(retries + 1):
            try:
                kwargs: Dict[str, Any] = {
                    "model": model,
                    "input": _build_input(instr, user_input),
                    "max_output_tokens": int(max_tokens),
                    "temperature": float(temperature),
                    "text": {"format": {"type": "json_object"}},
                }

                resp = self._responses_create_bulletproof(kwargs)
                txt = _extract_output_text(resp)
                last_txt = txt

                try:
                    return _safe_json_parse(txt)
                except Exception:
                    # Repair pass
                    return self._repair_json(model=model, bad_json_text=txt)

            except Exception as e:
                last_err = e
                time.sleep(0.4 * (attempt + 1))

        # Final attempt: if we have any last_txt, try repair one more time
        if last_txt.strip():
            try:
                return self._repair_json(model=model, bad_json_text=last_txt, retries=2)
            except Exception as e:
                last_err = e

        raise RuntimeError(f"OpenAI json_object failed: {last_err}")


# Convenience instance creator used elsewhere
def get_openai_client() -> OpenAIClient:
    return OpenAIClient()
