import json
from dataclasses import dataclass, field
from pathlib import Path
from typing import Any, Protocol

from rubric import RubricTree
from ..verify.base import BaseVerifier


class Grader(Protocol):
    def __init__(self, task_id: str):
        self.task_id = task_id

    def evaluate(self, include_reason: bool = False, **context: Any) -> float:
        raise NotImplementedError

    def save_as_file(self, file_path: str) -> None:
        """Save grading results to a file."""
        raise NotImplementedError


class VerifierGrader(Grader):
    """
    Grader implementation that uses BaseVerifier for evaluation.
    """
    
    def __init__(self, task_id: str, verifier: BaseVerifier):
        self.task_id = task_id
        self.verifier = verifier
    
    def evaluate(self, include_reason: bool = False, **context: Any) -> float | tuple[float, str]:
        """
        Evaluate using the verifier.
        
        Args:
            include_reason: Whether to return reasoning along with score
            **context: Must include 'original_file_path' and 'modified_file_path'
            
        Returns:
            Score (float) or tuple of (score, reason) if include_reason is True
        """
        if 'original_file_path' not in context or 'modified_file_path' not in context:
            raise ValueError("Context must include 'original_file_path' and 'modified_file_path'")
        
        # Extract verifier-specific parameters from context
        original_file_path = context.pop('original_file_path')
        modified_file_path = context.pop('modified_file_path')
        compute_strategy = context.pop('compute_strategy', 'default')
        non_critical_weight = context.pop('non_critical_weight', 0.3)
        conversion_mode = context.pop('conversion_mode', 'online')  
        use_cached_original_images = context.pop('use_cached_original_images', True)
        result = self.verifier.verify(
            original_file_path=original_file_path,
            modified_file_path=modified_file_path,
            include_reason=include_reason,
            compute_strategy=compute_strategy,
            non_critical_weight=non_critical_weight,
            conversion_mode=conversion_mode,
            use_cached_original_images=use_cached_original_images,
        )
        
        if include_reason:
            return result  # tuple[float, str]
        else:
            return result[0]  # just the score

    def save_as_file(self, file_path: Path) -> None:
        """Save grading results to a file."""
        self.verifier.rubric_tree.save_to_file(file_path)

@dataclass
class Task:
    task_id: str
    goal: str
    grader: Grader | None = None
    tags: list[str] = field(default_factory=list)
    file_path: str | None = None
    misc: dict = field(default_factory=dict)

    def load_from_file(self, file_path: str):
        """Load task from a JSON file."""
        with open(file_path, 'r', encoding='utf-8') as f:
            data = json.load(f)
        
        self.task_id = data.get('task_id', self.task_id)
        self.goal = data.get('goal', self.goal)
        self.tags = data.get('tags', [])
        self.file_path = data.get('file_path')
        self.misc = data.get('misc', {})

    def save_to_file(self, file_path: str):
        """Save task to a JSON file."""
        data = {
            'task_id': self.task_id,
            'goal': self.goal,
            'tags': self.tags,
            'file_path': self.file_path,
            'misc': self.misc
        }
        
        with open(file_path, 'w', encoding='utf-8') as f:
            json.dump(data, f, indent=2)


def create_verifier_grader(task_id: str, rubric_data: dict, verifier_class: type[BaseVerifier]) -> VerifierGrader:
    """
    Create a VerifierGrader from rubric data.
    
    Args:
        task_id: The task identifier
        rubric_data: Dictionary containing either 'rubric' object or 'rubric_path'
        verifier_class: The specific verifier class to instantiate
        
    Returns:
        VerifierGrader instance
    """
    if 'rubric_path' in rubric_data and rubric_data['rubric_path']:
        # Load from file path using RubricTree's built-in functionality
        rubric_path = rubric_data['rubric_path']
        verifier = verifier_class(rubric_path)
    elif 'rubric' in rubric_data and rubric_data['rubric']:
        # Create RubricTree from rubric object and save to temporary file
        import tempfile
        import os
        
        # Create RubricTree from the rubric data
        rubric_tree = RubricTree.from_dict(rubric_data['rubric'])
        
        # Save to temporary file using RubricTree's save functionality
        with tempfile.NamedTemporaryFile(mode='w', suffix='.json', delete=False) as f:
            temp_path = f.name
        
        try:
            rubric_tree.save_to_file(temp_path)
            verifier = verifier_class(temp_path)
        finally:
            # Clean up temporary file
            os.unlink(temp_path)
    else:
        raise ValueError("rubric_data must contain either 'rubric' or 'rubric_path'")
    
    return VerifierGrader(task_id, verifier)


def load_task_registry_from_directory(directory_path: str, verifier_class: type[BaseVerifier]) -> dict[str, Task]:
    """
    Load task registry from a directory containing JSON configuration.
    
    Expected JSON format:
    {
        "task-id": {
            "goal": "full task description",
            "rubric": <rubric_object> | null,
            "rubric_path": "path/to/rubric.json" | null,
            "tags": ["tag1", "tag2"],  # optional
            "file_path": "path/to/file",  # optional
            "misc": {}  # optional
        }
    }
    
    Args:
        directory_path: Path to directory containing the JSON file
        verifier_class: The verifier class to use for creating graders
        
    Returns:
        Dictionary mapping task_id to Task objects
    """
    directory = Path(directory_path)
    
    # Look for JSON files in the directory
    json_files = list(directory.glob("*.json"))
    if not json_files:
        raise FileNotFoundError(f"No JSON files found in {directory_path}")
    
    # Use the first JSON file found, or look for a specific name like "tasks.json"
    config_file = None
    for json_file in json_files:
        if json_file.name == "tasks.json":
            config_file = json_file
            break
    
    if config_file is None:
        config_file = json_files[0]  # Use first JSON file found
    
    with open(config_file, 'r', encoding='utf-8') as f:
        config_data = json.load(f)
    
    registry = {}
    
    for task_id, task_data in config_data.items():
        if 'goal' not in task_data:
            raise ValueError(f"Task {task_id} missing required 'goal' field")
        
        # Create the grader
        grader = create_verifier_grader(task_id, task_data, verifier_class)
        
        # Create the task
        task = Task(
            task_id=task_id,
            goal=task_data['goal'],
            grader=grader,
            tags=task_data.get('tags', []),
            file_path=task_data.get('file_path'),
            misc=task_data.get('misc', {})
        )
        
        registry[task_id] = task
    
    return registry


TASK_REGISTRY: dict[str, Task] = {}


def register_task(task_id: str, task: Task):
    TASK_REGISTRY[task_id] = task


def get_task(task_id: str) -> Task:
    return TASK_REGISTRY[task_id]


def get_all_tasks() -> list[Task]:
    return list(TASK_REGISTRY.values())
