"""
CLI Task - CLI mode dedicated task

New architecture design: adapt to data separation and new task model.
Remove outdated Evaluator design, inline evaluation logic into submit_result tool.
"""

from typing import Dict, Any, Optional, TYPE_CHECKING, Literal
from pathlib import Path

# import task base class
from ape.tasks.base import BaseTask, BaseTaskData, BaseTaskConfig, register_task, BaseTaskResult, EvaluationResult

if TYPE_CHECKING:
    from ape.scaffolds.config import BaseScaffoldConfig
    from ape.tasks.models import WorkspaceInfo


class CLITaskData(BaseTaskData):
    """CLI task data model - completely separate verify and retrieve configurations

    Design principles:
    1. verify_* parameters → Target Workspace (used for code verification)
       - if provided: get Git repository through get_workspace, create WorkspaceInfo with Git information
       - if not provided: create local WorkspaceInfo using local_workspace_path (commit/repo/default_target=None)

    2. retrieve_* parameters → Reference Workspaces (used for semantic retrieval)
       - create reference workspace for LeanRetrieve tool usage
       - completely independent of target workspace

    3. Three scenarios:
       Scenario A: only local path (--workspace /path)
         - Target workspace: local path, no Git information
         - Reference workspaces: empty
         - LeanRetrieve: returns "Workspace 'target' not found" (naturally fails)

       Scenario B: only verify configuration (--verify_commit_hash abc123)
         - Target workspace: Git repository at version abc123
         - Reference workspaces: empty
         - LeanRetrieve: can search on target workspace

       Scenario C: verify + retrieve configuration
         - Target workspace: verify specified version
         - Reference workspaces: [retrieve specified version]
         - LeanRetrieve: can search on reference workspace, also on target workspace

    4. Same design as Batch mode:
       - Batch mode: task.data.target_workspace → target workspace
       - CLI mode: task.data.verify_*/local_workspace_path → target workspace
       - all passed consistently through task.data
    """

    task_type: Literal["cli_interactive"] = Field(
        default="cli_interactive",
        description="Task type identifier"
    )

    # ===== Verify configuration: control Target Workspace retrieval =====
    # if provided, get Git repository through get_workspace as target workspace
    verify_commit_hash: Optional[str] = None
    verify_repo_url: Optional[str] = None
    verify_default_target: Optional[str] = None

    # ===== Local workspace path =====
    # mutually exclusive with verify_*, used to specify local path as target workspace
    local_workspace_path: Optional[Path] = None

    # ===== Retrieve configuration: create Reference Workspaces for retrieval tool usage =====
    # if provided, create reference workspace for LeanRetrieve tool retrieval
    retrieve_commit_hash: Optional[str] = None
    retrieve_repo_url: Optional[str] = None
    retrieve_default_target: Optional[str] = None

    # ===== Reference workspaces list (future extension: support multiple references) =====
    reference_workspaces: Optional[list] = None

class CLITaskConfig(BaseTaskConfig):
    pass

class CLITask(BaseTask):
    """
    CLI-mode-only task - minimalist interactive experience.

    Features:
    1. Interactive dialog mode, user directly input messages
    2. No automatic evaluation and submit_result tool
    3. Use CLI-specific system prompt
    4. Never automatically terminate, user controls session end
    """
    
    # class variable configuration
    task_type = "cli_interactive"
    data_class = CLITaskData
    task_config_class = CLITaskConfig
    
    def __init__(self, data: CLITaskData, config: 'BaseScaffoldConfig'):
        """Initialize CLI task"""
        super().__init__(data, config)

    def _is_lean_task(self) -> bool:
        """
        Check if it is a Lean task (requires Lean workspace)

        New logic: CLI task does not use this method to determine, because:
        - CLI task's workspace is explicitly specified through verify_*/local_workspace_path
        - does not depend on the implicit judgment of _is_lean_task()

        To maintain compatibility with the base class interface, always return False
        """
        return False

    async def _setup_target_workspace(self) -> Optional['WorkspaceInfo']:
        """
        Override base class method, implement CLI-specific target workspace logic

        CLI mode:
        - Priority 1: verify configuration (Git repository)
        - Priority 2: local_workspace_path (local path, main.py ensures non-None)
        """
        from ape.tasks.models import WorkspaceInfo

        # Priority 1: verify configuration (Git repository)
        if self.data.verify_commit_hash:
            target_workspace_path = await self._get_lean_repo_workspace(
                self.data.verify_commit_hash,
                repo_url=self.data.verify_repo_url
            )
            self.logger.info(f"Target workspace from verify config: {target_workspace_path}")

            return WorkspaceInfo(
                name="target",
                path=target_workspace_path,
                commit_hash=self.data.verify_commit_hash,
                repo_url=self.data.verify_repo_url,
                default_target=self.data.verify_default_target,
                read_only_path_patterns=[]  # CLI mode writable
            )

        # Priority 2: local workspace path (main.py ensures non-None, default is cwd)
        elif self.data.local_workspace_path:
            target_workspace_path = self.data.local_workspace_path.resolve()
            if not target_workspace_path.exists():
                raise RuntimeError(f"Local workspace does not exist: {self.data.local_workspace_path}")
            self.logger.info(f"Target workspace from local path: {target_workspace_path}")

            return WorkspaceInfo(
                name="target",
                path=target_workspace_path,
                commit_hash=None,  # local workspace has no Git information
                repo_url=None,
                default_target=None,
                read_only_path_patterns=[]  # CLI mode writable
            )

        return None

    async def _setup_reference_workspaces(self) -> Optional[list['WorkspaceInfo']]:
        """
        Override base class method, implement CLI-specific reference workspaces logic

        CLI mode:
        - Create single reference workspace from retrieve_* parameters
        """
        from ape.tasks.models import WorkspaceInfo
        from ape.utils.file_ops import normalize_repo_url

        if self.data.retrieve_commit_hash:
            ref_workspace_path = await self._get_lean_repo_workspace(
                self.data.retrieve_commit_hash,
                repo_url=self.data.retrieve_repo_url
            )
            self.logger.info(f"Reference workspace from retrieve config: {ref_workspace_path}")

            # Extract repo name from repo_url as workspace name
            ref_workspace_name = normalize_repo_url(self.data.retrieve_repo_url)

            ref_ws_info = WorkspaceInfo(
                name=ref_workspace_name,  # use repo name (e.g. 'mathlib4')
                path=ref_workspace_path,
                commit_hash=self.data.retrieve_commit_hash,
                repo_url=self.data.retrieve_repo_url,
                default_target=self.data.retrieve_default_target,
                read_only_path_patterns=["**/*"]  # Reference workspace is completely read-only
            )
            return [ref_ws_info]

        return None

    @classmethod
    def create_data_from_dict(cls, data: Dict[str, Any]) -> CLITaskData:
        """Create task data instance from dictionary (for API call etc.)"""
        return CLITaskData(
            task_id=data.get("task_id", "cli-interactive"),
            verify_commit_hash=data.get("verify_commit_hash"),
            verify_repo_url=data.get("verify_repo_url"),
            verify_default_target=data.get("verify_default_target"),
            local_workspace_path=Path(data["local_workspace_path"]) if data.get("local_workspace_path") else None,
            retrieve_commit_hash=data.get("retrieve_commit_hash"),
            retrieve_repo_url=data.get("retrieve_repo_url"),
            retrieve_default_target=data.get("retrieve_default_target")
        )
    
    async def create_user_prompt(self) -> str:
        """
        CLI mode does not require specific user prompt, return empty string
        because user directly inputs messages in CLI mode
        """
        return ""

    def should_terminate(self, evaluation_result: EvaluationResult = None) -> bool:
        """CLI mode never automatically terminates, user decides when to end"""
        return False

    def create_result(self, success: bool, score: float = 0.0, error: Optional[str] = None) -> BaseTaskResult:
        """
        Create CLI task result (minimal version)

        CLI mode usually does not create formal BaseTaskResult,
        but to provide minimal implementation for compatibility with scaffold interface.
        """
        return BaseTaskResult(
            task_id=self.data.task_id,
            task_type=self.task_type,
            global_index=self.data.global_index,
            success=success,
            score=score,
            error=error,
            metadata={"mode": "cli_interactive"}
        )


# Register CLI task type
register_task("cli_interactive", CLITask)
