from typing import List, Optional
import logger
import re
from agent.base import BaseAgent
from pydantic import model_validator
from schema import Message
from globals import execute_nb_code
from tool.execute_nb_code import ExecuteNbCode
import utils

REACT_SYS_PROMPT = """You are a Data Science Agent capable of solving data science tasks using Python code execution in a Jupyter Notebook environment.
You are participating in a Data Science competition (like Kaggle).

You function in a ReAct (Reasoning and Acting) loop:
1. **Thought**: Analyze the request and the current state. Decide what to do next.
2. **Action**: Write Python code to perform the task. The code will be executed in a stateful Jupyter Notebook environment.
3. **Observation**: You will see the output of your code execution.
4. Repeat steps 1-3 until the task is completed.

Guidelines:
- **Environment**: You are in a Jupyter Notebook. Variables are persistent across steps.
- **Data Persistence**: Save important intermediate results. The current working directory is the workspace root. Data is in `data/`.
- **Submission**: You MUST generate a submission file named `submission.csv` in `data/submission/` directory. Create the directory if it doesn't exist.
- **Output**: When you have completed the user's request, output the Final Answer.
- **Code Format**: Always wrap your code in a python code block: ```python ... ```.
- **Libraries**: Standard data science libraries (pandas, numpy, sklearn, matplotlib, etc.) are available.
- **Visualization**: If relevant, plot data.
- **Error Handling**: If code fails, analyze the error in the Observation and correct it in the next step.

Input Format:
User Request: [The user's request]
History: [Previous steps details]

Your Output Format:
Thought: [Your reasoning]
Action: 
```python
[Your code]
```
"""

class ReactAgent(BaseAgent):
    name: str = "ReactAgent"
    description: str = "A ReAct-based Data Science Agent."
    
    execute_code: ExecuteNbCode | None = None

    @model_validator(mode="after")
    def initialize_react(self) -> "ReactAgent":
        if self.execute_code is None:
            self.execute_code = execute_nb_code
        return self

    def act(self, requirement: str, run_id: str = "", complexity: str = "auto", extra_context: dict = None) -> dict:
        """
        Executes the ReAct loop.
        """
        self.requirement = requirement
        self.run_id = run_id
        
        history = ""
        max_steps = 30
        step = 0
        
        logger.info(f"【ReactAgent启动】Run ID: {run_id}")
        
        # Append submission sample if available
        req_with_context = requirement
        if extra_context and "submission_example" in extra_context:
            req_with_context += f"\n\nSubmission Example:\n{extra_context['submission_example']}"

        while step < max_steps:
            logger.info(f"--- Step {step + 1} ---")
            
            # Construct Prompt
            messages = [
                {"role": "system", "content": REACT_SYS_PROMPT},
                {"role": "user", "content": f"User Request: {req_with_context}\n\nHistory:\n{history}\n\nPlease provide your next Thought and Action."}
            ]
            
            # Call LLM
            response = self.llm.chat_completion(messages=messages)
             # Handle 2-tuple return if strict_mode is on (rare but possible in some implementations) 
            # or just access content directly. Assuming standard schema.
            if isinstance(response, tuple):
                 content = response[0]
            else:
                 content = response.choices[0].message.content
            
            logger.info(f"Agent Output:\n{content}")
            
            history += f"Step {step + 1}:\nAgent:\n{content}\n"
            
            # Parse - look for python code
            code_match = re.search(r"```python\s*(.*?)\s*```", content, re.DOTALL)
            
            observation = ""
            if code_match:
                code = code_match.group(1)
                result, success = self.execute_code.run(code)
                observation = f"Observation:\nOutput:\n{result}\nSuccess: {success}"
                logger.info(f"Execution Result:\n{result}")
            else:
                if "Final Answer" in content or "Mission Complete" in content:
                    logger.success("ReactAgent finished.")
                    break
                observation = "Observation: No code block found. If you have finished, say 'Final Answer'. Otherwise provide a python code block."
            
            history += f"{observation}\n\n"
            step += 1
            
        return {"status": "finished", "history": history}
