from __future__ import annotations

import argparse
import json
import sys
from typing import Any, Dict, Optional

from schemas.mcp_contract import make_response
from traditional.chebyshev import chebyshev_approximate
from traditional.minimax import minimax_approximate
from traditional.taylor import taylor_approximate


def _read_json(path: Optional[str], inline: Optional[str]) -> Dict[str, Any]:
    if path:
        with open(path, "r", encoding="utf-8") as handle:
            return json.load(handle)
    if inline:
        return json.loads(inline)
    raw = sys.stdin.read()
    if not raw.strip():
        raise ValueError("No JSON input provided.")
    return json.loads(raw)


def _emit(payload: Dict[str, Any], out: Optional[str]) -> None:
    text = json.dumps(payload, indent=2, ensure_ascii=True)
    if out:
        with open(out, "w", encoding="utf-8") as handle:
            handle.write(text)
            handle.write("\n")
    else:
        print(text)


def _run_method(method: str, payload: Dict[str, Any]) -> Dict[str, Any]:
    function = payload.get("function")
    interval = payload.get("interval")
    degree = int(payload.get("degree", 8))
    if not function or not interval:
        raise ValueError("function and interval are required fields.")

    if method == "chebyshev":
        sample_points = payload.get("sample_points")
        return chebyshev_approximate(function, interval, degree, sample_points=sample_points)
    if method == "minimax":
        max_iter = int(payload.get("max_iter", 8))
        grid_size = int(payload.get("grid_size", 4096))
        return minimax_approximate(function, interval, degree, max_iter=max_iter, grid_size=grid_size)
    if method == "taylor":
        center = payload.get("center")
        return taylor_approximate(function, interval, degree, center=center)
    raise ValueError(f"Unsupported method: {method}")


def main() -> int:
    parser = argparse.ArgumentParser(description="Traditional approximation tools")
    subparsers = parser.add_subparsers(dest="method", required=True)

    for name in ("chebyshev", "minimax", "taylor", "compare"):
        sub = subparsers.add_parser(name, help=f"Run {name} approximation")
        sub.add_argument("--input_path", type=str, default=None)
        sub.add_argument("--input", type=str, default=None)
        sub.add_argument("--out", type=str, default=None)

    args = parser.parse_args()
    payload = _read_json(args.input_path, args.input)

    if args.method == "compare":
        results = {
            "chebyshev": _run_method("chebyshev", payload),
            "minimax": _run_method("minimax", payload),
            "taylor": _run_method("taylor", payload),
        }
        response = make_response("ok", data={"method": "compare", "result": results})
        _emit(response, args.out)
        return 0

    result = _run_method(args.method, payload)
    response = make_response("ok", data={"method": args.method, "result": result})
    _emit(response, args.out)
    return 0


if __name__ == "__main__":
    raise SystemExit(main())
