import json
import os
import random
from typing import Dict, Any, List


class ManifestParser:
    def __init__(self, manifest_path: str):
        self.manifest_path = manifest_path
        self.manifest = self._load_manifest()

    def _load_manifest(self) -> Dict[str, Any]:
        with open(self.manifest_path, "r") as f:
            return json.load(f)

    @property
    def type(self) -> str:
        return self.manifest.get("type", "blender")

    @property
    def name(self) -> str:
        return self.manifest.get("name", "")

    @property
    def scene_file(self) -> str:
        scene_file = self.manifest.get("scene", "")
        return os.path.join(os.path.dirname(self.manifest_path), scene_file)

    @property
    def script_file(self) -> str:
        script_file = self.manifest.get("script", "")
        return os.path.join(os.path.dirname(self.manifest_path), script_file)

    @property
    def render_frame(self) -> int:
        render_info = self.manifest.get("render", {})
        return render_info.get("frame", 0)

    @property
    def input_variables(self) -> Dict[str, Dict[str, Any]]:
        return self.manifest.get("input", {})

    @property
    def tasks(self) -> List[Dict[str, Any]]:
        if "tasks" in self.manifest:
            return self.manifest.get("tasks", [])
        elif "task" in self.manifest:
            return [self.manifest.get("task", {})]
        return []

    def _resolve_value(
        self,
        value: Any,
        variables: Dict[str, Any],
    ) -> Any:
        if isinstance(value, str) and value.startswith("$"):
            var_name = value[1:]
            return variables.get(var_name)

        return value

    def _sample_value(
        self,
        definition: Dict[str, Any],
        variables: Dict[str, Any],
    ) -> Any:
        var_type = definition.get("type", "int")

        if var_type == "int":
            min_val = self._resolve_value(definition.get("min", 0), variables)
            max_val = self._resolve_value(definition.get("max", 10), variables)
            return random.randint(min_val, max_val)
        elif var_type == "enum":
            values = definition.get("values", [])
            if values:
                return random.choice(values)
        elif var_type == "shuffle":
            array = definition.get("array", [])
            if array:
                shuffled = array.copy()
                random.shuffle(shuffled)
                return shuffled

        return None

    def sample_input_variables(self) -> Dict[str, Any]:
        result = {}

        for var_name, var_def in self.input_variables.items():
            result[var_name] = self._sample_value(var_def, result)

        return result

    def sample_answer_variants(
        self,
        correct_answer: Any,
        variables: Dict[str, Any] = {},
        task_index: int = 0,
    ) -> List[Any]:
        variants = [correct_answer]
        answer_def = self.get_answer_definition(task_index)
        answer_variants = answer_def.get("variants", None)
        num_variants = answer_def.get("num_variants", 4)

        if not answer_variants:
            return variants

        attempts = 0
        max_attempts = 50

        while len(variants) < num_variants and attempts < max_attempts:
            attempts += 1
            value = self._sample_value(answer_variants, variables)

            if value not in variants:
                variants.append(value)

        random.shuffle(variants)

        return variants

    def get_correct_answer(self, variables: Dict[str, Any], task_index: int = 0) -> Any:
        tasks = self.tasks
        if task_index >= len(tasks):
            return None

        answer_def = tasks[task_index].get("answer", {})
        correct = answer_def.get("correct", "")

        return self._resolve_value(correct, variables)

    def get_task_prompt(
        self, task_index: int = 0, variables: Dict[str, Any] = {}
    ) -> str:
        tasks = self.tasks
        if task_index >= len(tasks):
            return ""

        prompt_template = tasks[task_index].get("prompt", "")

        for var_name, var_value in variables.items():
            placeholder = f"${var_name}"
            if placeholder in prompt_template and var_value is not None:
                prompt_template = prompt_template.replace(placeholder, str(var_value))

        return prompt_template

    def get_answer_definition(self, task_index: int = 0) -> Dict[str, Any]:
        tasks = self.tasks
        if task_index >= len(tasks):
            return {}

        return tasks[task_index].get("answer", {})
