import os
import asyncio
import json
import requests
from typing import Dict, Literal
from pydantic import BaseModel
from autogen_agentchat.agents import AssistantAgent
from autogen_agentchat.messages import TextMessage
from autogen_core import CancellationToken
from autogen_ext.models.openai import OpenAIChatCompletionClient
import re

# Load API keys from environment variables
OPENAI_API_KEY = os.getenv("OPENAI_API_KEY")
if not OPENAI_API_KEY:
    raise ValueError("Missing OPENAI_API_KEY. Set it as an environment variable.")
PAGODA_API_KEY = os.getenv("PAGODA_API_KEY")
if not PAGODA_API_KEY:
    raise ValueError("Missing PAGODA_API_KEY. Set it as an environment variable.")

# Define the expected response format as a Pydantic model
class AgentResponse(BaseModel):
    response: Literal["Accept", "Reject"]
    reasoning: str

# The ultimatum game2x2 simulation class
class Responder:
    def __init__(self, amount: int, offer: int, model: str, temperature: float, strategy=False, max_retries: int = 3):
        self.debug = False
        self.amount = amount
        self.offer = offer
        self.model = model
        self.temperature = temperature
        self.strategy = strategy
        self.max_retries = max_retries

        is_openai_model = model.startswith("gpt")
        is_pagoda_model = ":" in model

        base_url = (
            "https://api.openai.com/v1" if is_openai_model else
            "https://ollama-ui.pagoda.liris.cnrs.fr/ollama/api/generate" if is_pagoda_model else
            "http://localhost:11434/v1"
        )
        self.base_url = base_url

        if self.debug:
            print(f"Using model: {model}")
            print(f"Using base URL: {base_url}")

        key = OPENAI_API_KEY if is_openai_model else PAGODA_API_KEY

        model_info = {
            "temperature": self.temperature,
            "function_calling": True,
            "parallel_tool_calls": False,
            "family": "unknown",
            "json_output": True,
            "vision": False
        }

        self.model_client = OpenAIChatCompletionClient(
            model=self.model,
            base_url=base_url,
            api_key=key,
            model_info=model_info,
            response_format=AgentResponse
        )


    async def run(self) -> Dict:
        """Runs the model if strategy is False, otherwise uses a classical method."""
        if self.strategy:
            return self.apply_strategy()

        instruction = f"""
        You participate in an economics experiment. In this experiment each of you will
        be paired with a different person. You will not be told who these people are
        either during or after the experiment, and they will not be told who you are
        either during or after the experiment.

        The experiment is conducted as follows: A sum of ${self.amount} has been provisionally
        allocated to each pair and the person A can propose how much of this each person
        is to receive. A proposal form will then be sent to the person B.

        The person B will then be given a chance to accept or reject the proposal. If
        the person B accepts the proposal, then the amount of money will be divided as
        specified in the proposal. If the person B rejects the proposal, then both
        people in the pair receive zero. A response form will be sent to the person A.
        
        You are person B. Person A's offer is ${self.offer}.
        
         Return your response in JSON format with two keys:  
        - `response`: Your response (`"Accept"` or `"Reject"`).  
        - `reasoning`: A brief explanation of how you made your prediction.
        """
        if self.debug:
            print(instruction)

        is_pagoda_model = ":" in self.model
        if is_pagoda_model:
            return await self.run_pagoda(instruction)

        for attempt in range(self.max_retries):
            agent = AssistantAgent(
                name="Proposer",
                model_client=self.model_client,
                system_message="You are a helpful assistant."
            )

            response = await agent.on_messages(
                [TextMessage(content=instruction, source="user")],
                cancellation_token=CancellationToken(),
            )

            try:
                # Correct: get the content from the chat message
                raw_text = response.chat_message.content

                # Debug: show the raw content
                print(f"Raw content (Attempt {attempt + 1}): {raw_text}")

                # Try to load JSON directly
                try:
                    response_json = json.loads(raw_text)
                except json.JSONDecodeError:
                    # If it's wrapped in ```json ... ```, extract it
                    match = re.search(r'```json\s*(.*?)\s*```', raw_text, re.DOTALL)
                    if match:
                        response_json = json.loads(match.group(1))
                    else:
                        print(f"Could not parse JSON from response (Attempt {attempt + 1})")
                        continue

                agent_response = AgentResponse(**response_json)
                return agent_response

            except Exception as e:
                print(f"Error in OpenAI response handling (Attempt {attempt + 1}): {e}")

        raise ValueError("Model failed to provide a valid response after multiple attempts.")

    async def run_pagoda(self, instruction) -> Dict:
        """Runs the Pagoda model using a direct request."""
        url = self.base_url

        headers = {
            "Authorization": f"Bearer {PAGODA_API_KEY}",
            "Content-Type": "application/json"
        }

        payload = {
            "model": self.model,
            "temperature": self.temperature,
            "prompt": instruction,
            "stream": False,
            "response_format": {
                "type": "json_schema",
                "json_schema": {
                    "name": "AgentResponse",
                    "strict": True,
                    "schema": {
                        "title": "AgentResponse",
                        "type": "object",
                        "properties": {
                            "response": {
                                "title": "My Response",
                                "type": "string",
                            },
                            "reasoning": {
                                "title": "Reasoning",
                                "type": "string"
                            }
                        },
                        "required": ["response" "reasoning"],
                        "additionalProperties": False
                    }
                }
            }
        }

        for attempt in range(self.max_retries):
            try:
                response = requests.post(url, headers=headers, json=payload)
                response.raise_for_status()

                # Get the JSON response
                response_data = response.json()

                # Debug: print the raw response to check if fields are missing or named differently
                if self.debug:
                    print(f"Raw response (Attempt {attempt+1}): {response_data}")

                # The response field should be parsed correctly if it's already valid JSON
                response_json = response_data.get('response', '')

                # If the response is a string containing JSON, we need to extract and parse it
                if isinstance(response_json, str):
                    # Try to parse the response as JSON
                    try:
                        response_dict = json.loads(response_json)
                    except json.JSONDecodeError:
                        # If the response is not valid JSON, apply regex to extract the JSON portion
                        match = re.search(r"```json(.*?)```", response_json, re.DOTALL)
                        if match:
                            response_dict = json.loads(match.group(1))
                        else:
                            print(f"Invalid response format detected (Attempt {attempt + 1}): {response_json}")
                            continue
                elif isinstance(response_json, dict):
                    # If response_json is already a dictionary, just use it
                    response_dict = response_json
                else:
                    print(f"Unexpected format in 'response' field (Attempt {attempt + 1}): {response_json}")
                    continue

                # Validate the response structure
                agent_response = AgentResponse(**response_dict)
                return agent_response.model_dump()

            except Exception as e:
                print(f"Error in Pagoda request (Attempt {attempt + 1}): {e}")

        raise ValueError("Pagoda model failed to provide a valid response after multiple attempts.")

    def apply_strategy(self) -> Dict:
        """Generates a response based on predefined strategies."""
        if self.model == "gpt-4.5-preview-2025-02-27":
            # Threshold below which the offer is considered insulting or unfair
            fairness_threshold = 0.2  # 20% of the total amount
            offer_ratio = self.offer / self.amount
            if offer_ratio >= fairness_threshold:
                response = "Accept"
                reasoning = (
                    f"The offer of ${self.offer} is {offer_ratio:.0%} of the total amount (${self.amount}), "
                    f"which is deemed fair or acceptable. Accepting ensures a positive payout."
                )
            else:
                response = "Reject"
                reasoning = (
                    f"The offer of ${self.offer} is only {offer_ratio:.0%} of the total amount (${self.amount}), "
                    f"which is considered unfair. Rejecting signals a preference for fairness over gain."
                )
            return {
                "response": response,
                "reasoning": reasoning
            }
        if self.model in ["mistral_small"]:
            offer_percentage = self.offer / self.amount * 100
            if offer_percentage >= 50:
                # Accept the offer because it's fair or generous.
                response = "Accept"
                reasoning = f"The offer is {offer_percentage}% of the total amount, which is considered reasonable."
            elif offer_percentage < 25:
                # Reject the offer because it's too low.
                response = "Reject"
                reasoning = f"The offer is only {offer_percentage}% of the total amount, which is not acceptable."
            else:
                # Randomly decide for offers between 25% and 49%
                if random.random() < 0.5:  # You can adjust this probability as needed.
                    response = "Accept"
                    reasoning = f"The offer is {offer_percentage}% of the total amount, which could be acceptable."
                else:
                    response = "Reject"
                    reasoning = f"The offer is {offer_percentage}% of the total amount, which might not be enough."
            return {"response": response, "reasoning": reasoning}
        if self.model in ["llama3", "deepseek-r1", "qwen3", "mixtral:8x7b"]:
            if self.offer >= (self.amount / 2):
                # Accept offer
                return {"response": "Accept",
                        "reasoning": f"The offered amount is more than half of the total, so I accept."}
            else:
                # Reject offer
                return {"response": "Reject",
                        "reasoning": f"The offered amount is less than half of the total, so I reject."}
        if self.model in ["llama3.3:latest"]:
            if self.offer / self.amount >= 0.3:
                response = "Accept"
                reasoning = f"The offer of ${self.offer} is greater than or equal to 30% of the total amount (${self.amount}), so I accept."
            else:
                response = "Reject"
                reasoning = f"The offer of ${self.offer} is less than 30% of the total amount (${self.amount}), so I reject."
            return {"response": response, "reasoning": reasoning}
        return None

        # Run the async function and return the response
if __name__ == "__main__":
    agent = Responder(amount=100, offer=40, model="qwen3", temperature=0.7, strategy=False)
    # "gpt-4.5-preview-2025-02-27" "llama3", "mistral-small", "deepseek-r1", "qwen3", "mixtral:8x7b", "llama3.3:latest", "deepseek-r1:7b"
    response_json = asyncio.run(agent.run())
    print(response_json)