"""
Poster QA Experiments Configuration
Clean architecture: AgentFormat × QuestionFormat (aligned with Web)
"""

from enum import Enum
from dataclasses import dataclass
from typing import Optional, List
from pathlib import Path


class BaselineMode(Enum):
    """Baseline modes for poster generation."""
    NO_USER = "no_user"


class AgentFormat(Enum):
    """Agent format - defines how questions are selected."""
    NAIVE_AGENT = "Naive_Agent"
    MPQC = "MPQC"
    RAG = "RAG"


class QuestionFormat(Enum):
    """Question format - defines how a question is phrased."""
    FREE_ASK = "Free_Ask"
    FIXED_BINARY = "Fixed_Binary"
    FIXED_MULTICHOICE = "Fixed_MultiChoice"
    FIXED_OPENTEXT = "Fixed_OpenText"
    FLEXIBLE = "Flexible"
    ADAPTIVE = "Adaptive"


@dataclass
class PosterInteractionConfig:
    """Configuration for poster QA interaction style."""
    id: str
    agent_format: Optional[AgentFormat] = None
    question_format: Optional[QuestionFormat] = None
    baseline_mode: Optional[BaselineMode] = None
    max_qa_cycles: int = 2
    max_questions_per_batch: int = 5
    model_version: str = "gemini25"  # Global model version for plan creation/updates
    question_agent_model_version: Optional[str] = None  # Model version for question agent (fallback to model_version)
    answer_agent_model_version: Optional[str] = None  # Model version for answer agent (fallback to model_version)
    verbose: bool = True

    def __post_init__(self):
        """Validate configuration."""
        if self.agent_format is not None and self.question_format is not None:
            if self.agent_format == AgentFormat.RAG:
                if self.question_format != QuestionFormat.FLEXIBLE:
                    raise ValueError(f"RAG agent only supports Flexible format, got {self.question_format}")
            if self.agent_format == AgentFormat.MPQC:
                if self.question_format != QuestionFormat.ADAPTIVE:
                    raise ValueError(f"MPQC agent only supports Adaptive format, got {self.question_format}")
        elif self.baseline_mode is None:
            raise ValueError("Either (agent_format + question_format) or baseline_mode must be set")
    
    @property
    def is_qa_mode(self) -> bool:
        return self.agent_format is not None and self.question_format is not None
    
    @property
    def is_adaptive_format(self) -> bool:
        return self.question_format == QuestionFormat.ADAPTIVE if self.question_format else False
    
    @property
    def is_flexible_format(self) -> bool:
        return self.question_format == QuestionFormat.FLEXIBLE if self.question_format else False
    
    @property
    def is_mpqc(self) -> bool:
        return self.agent_format == AgentFormat.MPQC if self.agent_format else False

    @property
    def is_rag(self) -> bool:
        return self.agent_format == AgentFormat.RAG if self.agent_format else False
    
    @property
    def is_free_ask_format(self) -> bool:
        return self.question_format == QuestionFormat.FREE_ASK if self.question_format else False
    
    @property
    def is_fixed_format(self) -> bool:
        if not self.question_format:
            return False
        return self.question_format in [QuestionFormat.FIXED_BINARY, QuestionFormat.FIXED_MULTICHOICE, QuestionFormat.FIXED_OPENTEXT]
    
    @property
    def effective_question_agent_model_version(self) -> str:
        """Get effective question agent model version (fallback to global model_version)"""
        return self.question_agent_model_version or self.model_version
    
    @property
    def effective_answer_agent_model_version(self) -> str:
        """Get effective answer agent model version (fallback to global model_version)"""
        return self.answer_agent_model_version or self.model_version


def generate_all_configs() -> List[PosterInteractionConfig]:
    """Generate the 8 required configurations only."""
    return [
        # Baseline: no user interaction
        PosterInteractionConfig(
            id="no_user",
            baseline_mode=BaselineMode.NO_USER,
            verbose=True,
        ),
        # Naive_Agent variants (5 formats)
        PosterInteractionConfig(
            id="Naive_Agent_Free_Ask",
            agent_format=AgentFormat.NAIVE_AGENT,
            question_format=QuestionFormat.FREE_ASK,
            verbose=True,
        ),
        PosterInteractionConfig(
            id="Naive_Agent_Fixed_Binary",
            agent_format=AgentFormat.NAIVE_AGENT,
            question_format=QuestionFormat.FIXED_BINARY,
            verbose=True,
        ),
        PosterInteractionConfig(
            id="Naive_Agent_Fixed_MultiChoice",
            agent_format=AgentFormat.NAIVE_AGENT,
            question_format=QuestionFormat.FIXED_MULTICHOICE,
            verbose=True,
        ),
        PosterInteractionConfig(
            id="Naive_Agent_Fixed_OpenText",
            agent_format=AgentFormat.NAIVE_AGENT,
            question_format=QuestionFormat.FIXED_OPENTEXT,
            verbose=True,
        ),
        PosterInteractionConfig(
            id="Naive_Agent_Flexible",
            agent_format=AgentFormat.NAIVE_AGENT,
            question_format=QuestionFormat.FLEXIBLE,
            verbose=True,
        ),
        # MPQC: Adaptive only
        PosterInteractionConfig(
            id="MPQC_Adaptive",
            agent_format=AgentFormat.MPQC,
            question_format=QuestionFormat.ADAPTIVE,
            verbose=True,
        ),
        # RAG: Flexible only
        PosterInteractionConfig(
            id="RAG_Flexible",
            agent_format=AgentFormat.RAG,
            question_format=QuestionFormat.FLEXIBLE,
            verbose=True,
        ),
    ]


def get_config_by_id(config_id: str) -> PosterInteractionConfig:
    """Get config by ID."""
    configs = generate_all_configs()
    for config in configs:
        if config.id == config_id:
            return config
    raise ValueError(f"Config not found: {config_id}")


@dataclass
class PosterSample:
    """Poster sample data"""
    index: str
    brand_name: str
    prompt_file: Path
    logo_file: Optional[Path] = None
    answer_image_path: Path = None
    
    def __post_init__(self):
        if isinstance(self.prompt_file, str):
            self.prompt_file = Path(self.prompt_file)
        if self.logo_file is not None and isinstance(self.logo_file, str):
            self.logo_file = Path(self.logo_file)
        if isinstance(self.answer_image_path, str):
            self.answer_image_path = Path(self.answer_image_path)


@dataclass
class PosterGenerationResult:
    """Poster generation result"""
    success: bool
    generated_image_path: Optional[Path] = None
    ad_text: Optional[str] = None
    conversation_history: Optional[list] = None
    design_plan: Optional[dict] = None
    token_stats: Optional[dict] = None
    judge_evaluation: Optional[dict] = None
    error: Optional[str] = None
    questions_asked: Optional[int] = None
    
    def to_dict(self):
        return {
            "success": self.success,
            "generated_image_path": str(self.generated_image_path) if self.generated_image_path else None,
            "ad_text": self.ad_text,
            "conversation_history": self.conversation_history,
            "design_plan": self.design_plan,
            "token_stats": self.token_stats,
            "judge_evaluation": self.judge_evaluation,
            "error": self.error,
            "questions_asked": self.questions_asked,
        }
