from pydantic import BaseModel, model_validator
from typing_extensions import Self


class ExperimentPlan(BaseModel):
    """
    Represents the experiment plan with a title, objective, steps, and deliverables.

    Attributes:
        objective (str): The main goal or objective of the experiment
        steps (str): List of steps to be followed to implement the experiment
        deliverables (str): List of expected outcomes or deliverables from the experiment
    """

    objective: str
    steps: str
    deliverables: str


class Experiment(BaseModel):
    """
    Represents an experiment with a hypothesis and corresponding experiment plan.
    Attributes:
        hypothesis (str): A natural-language hypothesis representing an assertion about the world
        experiment_plan (ExperimentPlan): The structured experiment plan to verify the hypothesis
    """

    hypothesis: str
    experiment_plan: ExperimentPlan


class ExperimentList(BaseModel):
    """
    A collection of experiments.

    Attributes:
        experiments (list[Experiment]): List of Experiment objects
    """

    experiments: list[Experiment]


class ExperimentHypothesis(BaseModel):
    """
    Represents an experiment with an experiment plan and a hypothesis.
    Attributes:
        experiment_plan (ExperimentPlan): A structured experiment plan to verify a hypothesis
        hypothesis (str): A natural-language hypothesis representing an assertion about the world that can be
                          tested by the experiment
    """

    experiment_plan: ExperimentPlan
    hypothesis: str


class ExperimentHypothesisList(BaseModel):
    """
    A collection of experiment hypotheses.

    Attributes:
        experiments (list[ExperimentHypothesis]): List of ExperimentHypothesis objects
    """

    experiments: list[ExperimentHypothesis]


class ExperimentCode(BaseModel):
    """
    Contains the code implementation for an experiment.

    Attributes:
        code (str): The actual code to be executed for the experiment
    """

    code: str


class ExperimentReviewer(BaseModel):
    """
    Review of an experiment's execution and results.

    Attributes:
        hypothesis (str): The scientific hypothesis in natural language
        workflow (str): The summary of the full workflow starting from data loading
    """

    hypothesis: str
    workflow: str


class ExperimentReflexion(BaseModel):
    """
    Reflection of the attempted experiment based on its evaluation score

    Attributes:
        reflection (str): An explanation of why the attempted experiment is wrong
    """

    reflection: str


class ExperimentAnalyst(BaseModel):
    """
    Analysis of experiment results.

    Attributes:
        analysis (Optional[str]): Detailed analysis of the experiment outcomes
        success (bool): Whether the experiment was successful
    """

    analysis: str
    success: bool

    @model_validator(mode="after")
    def analysis_required_on_success(self) -> Self:
        if self.success and self.analysis is None:
            raise ValueError("analysis is required when success is True")
        return self


class ExperimentEvaluation(BaseModel):
    """
    Analysis of two hypotheses.

    Attributes:
        answer (str): one of the options from A) definitely similar B) somewhat similar C) uncertain D) somewhat different E) definitely different
    """

    answer: str
