from __future__ import annotations

import json
import os
import tempfile
from typing import Any, Dict, Optional

from server.run_manager import RunManager, make_response
from tools.codegen_emit import emit_codegen
from tools.baseline_runner import run_baseline
from tools.candidate_loader import load_candidate_json
from tools.math_inspector import inspect_math
from python_src.spec_io import load_spec_dict, normalize_spec_dict, read_json_path
from verify.verifier import evaluate_candidate


def _read_json(path: Optional[str]) -> Optional[Dict[str, Any]]:
    return read_json_path(path)


class ToolClient:
    def call(self, name: str, payload: Dict[str, Any]) -> Dict[str, Any]:
        raise NotImplementedError


def build_internal_tool_client() -> ToolClient:
    return LocalToolClient()


class LocalToolClient(ToolClient):
    def __init__(self) -> None:
        self.manager = RunManager()

    def call(self, name: str, payload: Dict[str, Any]) -> Dict[str, Any]:
        if name == "anum.baseline.run":
            return run_baseline(payload)

        if name == "anum.spec.validate":
            try:
                spec = load_spec_dict(payload.get("spec"), payload.get("spec_path"))
            except ValueError:
                return make_response(
                    "error",
                    errors=[{"code": "missing_spec", "message": "spec is required", "details": {}}],
                )
            try:
                normalized = normalize_spec_dict(spec)
            except Exception as exc:
                return make_response(
                    "error",
                    errors=[{"code": "spec_invalid", "message": str(exc), "details": {}}],
                )
            return make_response("ok", data={"spec": normalized})

        if name == "anum.math.inspect":
            return inspect_math(payload)

        if name == "anum.strategy.suggest":
            from tools.strategy_suggest import suggest_strategy

            return suggest_strategy(payload)

        if name == "anum.run.submit":
            spec = payload.get("spec")
            if spec is None:
                return make_response(
                    "error",
                    errors=[{"code": "missing_spec", "message": "spec is required", "details": {}}],
                )
            return self.manager.submit(
                spec=spec,
                task_tag=payload.get("task_tag"),
                runner=payload.get("runner"),
                request_id=payload.get("request_id"),
            )

        if name == "anum.run.poll":
            task_tag = payload.get("task_tag")
            if not task_tag:
                return make_response(
                    "error",
                    errors=[{"code": "missing_task_tag", "message": "task_tag is required", "details": {}}],
                )
            return self.manager.poll(task_tag)

        if name == "anum.run.stop":
            task_tag = payload.get("task_tag")
            if not task_tag:
                return make_response(
                    "error",
                    errors=[{"code": "missing_task_tag", "message": "task_tag is required", "details": {}}],
                )
            return self.manager.stop(task_tag)

        if name == "anum.run.result":
            task_tag = payload.get("task_tag")
            if not task_tag:
                return make_response(
                    "error",
                    errors=[{"code": "missing_task_tag", "message": "task_tag is required", "details": {}}],
                )
            return self.manager.result(task_tag)

        if name == "anum.artifacts.get":
            artifact_id = payload.get("artifact_id")
            if not artifact_id:
                return make_response(
                    "error",
                    errors=[{"code": "missing_artifact_id", "message": "artifact_id is required", "details": {}}],
                )
            return self.manager.artifact_get(artifact_id, fmt=payload.get("format"))

        if name == "anum.artifacts.list":
            artifact_id = payload.get("artifact_id")
            if not artifact_id:
                return make_response(
                    "error",
                    errors=[{"code": "missing_artifact_id", "message": "artifact_id is required", "details": {}}],
                )
            return self.manager.artifact_list(artifact_id)

        if name == "anum.run.list":
            return self.manager.list(status_filter=payload.get("status"))

        if name == "anum.verify.evaluate":
            spec_path = payload.get("spec_path")
            spec = payload.get("spec")
            candidate = payload.get("candidate") or payload.get("candidate_path")
            if not spec_path and spec is None:
                return make_response(
                    "error",
                    errors=[
                        {
                            "code": "missing_inputs",
                            "message": "spec_path (or spec) and candidate are required",
                            "details": {},
                        }
                    ],
                )
            if candidate is None:
                try:
                    candidate = load_candidate_json(
                        candidate_artifact_id=payload.get("candidate_artifact_id"),
                        task_tag=payload.get("task_tag"),
                        manager=self.manager,
                    )
                except Exception:
                    return make_response(
                        "error",
                        errors=[
                            {
                                "code": "missing_inputs",
                                "message": "candidate, candidate_path, candidate_artifact_id, or task_tag is required",
                                "details": {},
                            }
                        ],
                    )
            temp_path = None
            if spec is not None and not spec_path:
                try:
                    with tempfile.NamedTemporaryFile(
                        mode="w", encoding="utf-8", suffix="_spec.json", delete=False
                    ) as handle:
                        json.dump(spec, handle, ensure_ascii=False, indent=2)
                        temp_path = handle.name
                    spec_path = temp_path
                except Exception as exc:
                    return make_response(
                        "error",
                        errors=[{"code": "spec_write_failed", "message": str(exc), "details": {}}],
                    )
            try:
                result = evaluate_candidate(
                    spec_path=spec_path,
                    candidate=candidate,
                    piece_id=payload.get("piece_id"),
                    level=int(payload.get("level", 1)),
                    max_points=payload.get("max_points"),
                )
            except Exception as exc:
                return make_response(
                    "error",
                    errors=[{"code": "verify_failed", "message": str(exc), "details": {}}],
                )
            finally:
                if temp_path:
                    try:
                        os.remove(temp_path)
                    except OSError:
                        pass
            return make_response("ok", data=result)

        if name == "anum.codegen.emit":
            return emit_codegen(
                candidate_path=payload.get("candidate_path"),
                candidate=payload.get("candidate"),
                candidate_artifact_id=payload.get("candidate_artifact_id"),
                task_tag=payload.get("task_tag"),
            )

        return make_response(
            "error",
            errors=[{"code": "tool_unavailable", "message": f"Unknown tool: {name}", "details": {}}],
        )
