from __future__ import annotations

from typing import Any, Dict, Tuple

import numpy as np

from schemas.approx_spec import FunctionSpec, PrecisionSpec, TargetSpec
from worker.spec_loader import build_fun_eval


def _build_callable(function: Dict[str, Any]):
    func_spec = FunctionSpec.model_validate(function)
    target = TargetSpec(name=func_spec.builtin or "expr", function=func_spec, vars=["x"])
    precision = PrecisionSpec()
    return build_fun_eval(target, precision)


def _horner_code(coeffs: np.ndarray) -> str:
    if coeffs.size == 0:
        return "0.0"
    coeffs = coeffs.astype(np.float64)
    code = f"{coeffs[-1]:.17g}"
    for c in coeffs[-2::-1]:
        code = f"({code}*x + {c:.17g})"
    return code


def chebyshev_approximate(
    function: Dict[str, Any],
    interval: Dict[str, float],
    degree: int,
    sample_points: int | None = None,
) -> Dict[str, Any]:
    start = float(interval["start"])
    end = float(interval["end"])
    if start >= end:
        raise ValueError("interval.start must be less than interval.end")

    fun_eval = _build_callable(function)
    n_samples = sample_points or max(256, degree * 64)
    x = np.linspace(start, end, n_samples, dtype=np.float64)
    y = np.array([fun_eval(float(val)) for val in x], dtype=np.float64)

    cheb = np.polynomial.Chebyshev.fit(x, y, degree, domain=[start, end])
    poly = cheb.convert(kind=np.polynomial.Polynomial)
    coeffs = np.array(poly.coef, dtype=np.float64)

    y_hat = poly(x)
    abs_err = np.abs(y_hat - y)
    metrics = {
        "max_error": float(np.max(abs_err)),
        "avg_error": float(np.mean(abs_err)),
        "p99_error": float(np.quantile(abs_err, 0.99)),
    }

    return {
        "method": "chebyshev",
        "coefficients": coeffs.tolist(),
        "polynomial_code": _horner_code(coeffs),
        "metrics": metrics,
        "evaluation_ops": int(degree),
    }
