from __future__ import annotations

from abc import ABC, abstractmethod
from typing import Any, Dict, List, Optional


class BasePlanGenerator(ABC):
    """
    Base interface for all plan generators.

    A generator is used by calling `start(...)` once and then repeatedly
    calling `next_query(...)` for each turn. Subclasses can implement
    different preparation flows (e.g., precomputing a full plan vs.
    generating per turn from an internal strategy), while callers share
    the same interface.
    """

    def __init__(self) -> None:
        self.harmful_behavior: Optional[str] = None
        self.max_turns: Optional[int] = None
        self._plan_ready: bool = False

    def start(self, harmful_behavior: str, max_turns: Optional[int] = None, **kwargs) -> "BasePlanGenerator":
        """
        Prepare internal state for a new harmful behavior.
        """
        self.save_info = []
        self.harmful_behavior = harmful_behavior
        self.max_turns = max_turns
        self._prepare_plan(harmful_behavior, max_turns=max_turns, **kwargs)
        self._plan_ready = True
        return self

    def next_query(self, cur_turn: int, conversation_history: Optional[str] = None, **kwargs) -> str:
        """
        Generate or fetch the query for the given turn.
        """
        if not self._plan_ready:
            raise RuntimeError("Call start() before requesting the next query.")
        return self._next_query(cur_turn, conversation_history=conversation_history, **kwargs)

    def reset(self) -> None:
        self._plan_ready = False
        self.harmful_behavior = None
        self.max_turns = None

    @staticmethod
    def format_history(history: List[Dict[str, str]]) -> str:
        """
        Convert a list of role/content dicts into a simple text transcript.
        """
        return "\n".join(f"{msg['role']}: {msg['content']}" for msg in history)

    @abstractmethod
    def _prepare_plan(self, harmful_behavior: str, max_turns: Optional[int] = None, **kwargs) -> Any:
        """
        Subclasses set up whatever state they need to generate turns.
        """

    @abstractmethod
    def _next_query(
        self, cur_turn: int, conversation_history: Optional[str] = None, **kwargs
    ) -> str:
        """
        Subclasses return the query for the requested turn.
        """