from llms.llms import async_call_llm
from json_repair import repair_json
import json
import re
import asyncio
import logging

class PlanningAgent:
    @staticmethod
    async def async_create_hierarchy(model, example, semaphore):
        if example["type"] == "Week":
            example = await PlanningAgent.async_create_week_plan(model, example, semaphore)
        elif example["type"] == "Floor":
            example = await PlanningAgent.async_create_floor_plan(model, example, semaphore)
        elif example["type"] == "Menu Week":
            example = await PlanningAgent.async_create_menu_plan(model, example, semaphore)
        elif example["type"] == "Block":
            example = await PlanningAgent.async_create_block_plan(model, example, semaphore)

        return example

    @staticmethod
    async def async_create_week_plan(model, example, semaphore):
        plan_prompt = f"""
You are an expert planner with perfect attention to detail. Create a precise 52-week plan that EXACTLY matches ALL user requirements.

User requirements:
{example['prompt']}

CRITICAL ANALYSIS INSTRUCTIONS:
1. Read the user requirements word-by-word and identify EVERY specific event, date, and requirement
2. For EACH specific event, determine the EXACT week it must occur in (count from Week 1 = January 1st-7th)
3. Pay special attention to dates, months, and any temporal indicators like "Week X", specific dates, holidays
4. If an event mentions "Week X" or a specific date, calculate and place it in that EXACT week
5. For birthdays and special dates, calculate which week they fall in based on the calendar
6. Double-check that NO requirements are missed or misplaced

SPECIAL EVENT IDENTIFICATION:
- Look for phrases like "in week X", "on [date]", "during [month]", "every X weeks", "birthday on [date]"
- Calculate exact weeks for specific dates (e.g., May 13th = Week 19-20)
- Identify recurring events and their frequency
- Note any special occasions with specific timing requirements

IMPORTANT: Create AT LEAST 40 weeks of plans. If you cannot create 52 weeks, provide as many as possible (minimum 40).

Return your analysis and plan in this EXACT format:
{{
    "analysis": "Detailed word-by-word analysis of EVERY requirement in the user prompt, listing each specific event and its calculated exact week timing",
    "special_events": [
        {{
            "event_name": "EXACT event name from user requirements",
            "week_numb": "EXACT week specification calculated from requirements (e.g., 'Week 19' for May 13th)",
            "requirement_match": "How this exactly matches and fulfills the user requirement",
            "calculation": "How the week was calculated from the date/requirement"
        }}
    ],
    "weekly_plan": [
        {{
            "week_id": "Week 1 (January 1st - January 7th)",
            "events": "Events for this week - MUST include special events if they fall in this week according to calculations"
        }}
    ]
}}

REMEMBER: Each week covers 7 days. Week 1 = Jan 1-7, Week 2 = Jan 8-14, etc. Calculate precisely which week each event falls into."""

        print(plan_prompt)
        trial = 0
        max_trials = 5
        best_response = None
        
        while trial < max_trials:
            try:
                async with semaphore:
                    response = await async_call_llm(model, plan_prompt)

                response = repair_json(response)
                match = re.search(r"\{.*\}", response, re.DOTALL)
                
                if match:
                    try:
                        response_dict = json.loads(match.group(0))
                        
                        # 降低要求：接受至少40周的计划
                        if "weekly_plan" in response_dict and len(response_dict["weekly_plan"]) >= 40:
                            example["weekly_plan"] = response_dict["weekly_plan"]
                            example["special_events"] = response_dict.get("special_events", [])
                            example["analysis"] = response_dict.get("analysis", "")
                            best_response = response_dict
                            break
                        elif "weekly_plan" in response_dict and len(response_dict["weekly_plan"]) > 0:
                            # 保存最好的响应，以防没有更好的
                            if not best_response or len(response_dict["weekly_plan"]) > len(best_response.get("weekly_plan", [])):
                                best_response = response_dict
                            logging.warning(f"Partial plan received ({len(response_dict['weekly_plan'])} weeks), retrying... (attempt {trial+1})")
                        else:
                            logging.warning(f"Invalid plan structure received, retrying... (attempt {trial+1})")
                            
                    except json.JSONDecodeError as e:
                        logging.error(f"JSON decode error on attempt {trial+1}: {e}")
                        
            except Exception as e:
                logging.error(f"Error creating plan on attempt {trial+1}: {e}")
                
            trial += 1
            if trial < max_trials:
                await asyncio.sleep(1 * trial)  # 减少等待时间

        # 如果没有成功获得足够的周计划，使用最佳可用响应或创建基础计划
        if "weekly_plan" not in example or len(example.get("weekly_plan", [])) < 20:
            if best_response and len(best_response.get("weekly_plan", [])) > 0:
                example["weekly_plan"] = best_response["weekly_plan"]
                example["special_events"] = best_response.get("special_events", [])
                example["analysis"] = best_response.get("analysis", "")
                logging.info(f"Using best available response with {len(best_response['weekly_plan'])} weeks")
            else:
                # 创建基础的52周计划作为备用
                logging.warning("Creating fallback weekly plan")
                example["weekly_plan"] = []
                for i in range(1, 53):
                    week_data = {
                        "week_id": f"Week {i} (Week of year {i})",
                        "events": f"Weekly activities and events to fulfill user requirements: {example['prompt'][:100]}..."
                    }
                    example["weekly_plan"].append(week_data)
                example["special_events"] = []
                example["analysis"] = "Fallback analysis: Created basic weekly structure to ensure task completion"

        # 简化的修订提示，减少失败风险
        if len(example.get("weekly_plan", [])) >= 20:  # 只有当有足够的计划时才进行修订
            revise_prompt = f"""
Review and improve this weekly plan to better match user requirements:

User requirements: {example['prompt']}
Current plan weeks: {len(example['weekly_plan'])}

Make minor improvements to ensure special events are placed correctly and user requirements are met.
Keep the existing structure but enhance content quality.

Return the improved plan in the same JSON format with 'revised_weekly_plan' field."""

            trial = 0
            max_trials = 2  # 减少修订尝试次数
            
            while trial < max_trials:
                try:
                    async with semaphore:
                        response = await async_call_llm(model, revise_prompt)

                    response = repair_json(response)
                    match = re.search(r"\{.*\}", response, re.DOTALL)
                    
                    if match:
                        try:
                            response_dict = json.loads(match.group(0))
                            
                            if "revised_weekly_plan" in response_dict and len(response_dict["revised_weekly_plan"]) >= len(example["weekly_plan"]) * 0.8:
                                example["weekly_plan"] = response_dict["revised_weekly_plan"]
                                example["corrections_made"] = response_dict.get("corrections_made", [])
                                break
                            else:
                                logging.warning(f"Revision did not improve plan quality, keeping original")
                                break
                                
                        except json.JSONDecodeError:
                            logging.error(f"JSON decode error in revision step (attempt {trial+1})")
                            break  # 不再重试，保持原计划
                            
                except Exception as e:
                    logging.error(f"Error revising plan on attempt {trial+1}: {e}")
                    break  # 不再重试，保持原计划
                    
                trial += 1

        return example

    @staticmethod
    async def async_create_floor_plan(model, example, semaphore):
        plan_prompt = f"""
You are an expert architect and building planner. Create a precise floor plan that EXACTLY matches ALL user requirements.

User requirements:
{example['prompt']}

CRITICAL ANALYSIS INSTRUCTIONS:
1. Identify EVERY specific floor mentioned in user requirements
2. Note EXACT purposes, facilities, or special features for each floor
3. Pay attention to floor numbers and their designated uses
4. Identify any special requirements for specific floors
5. Ensure all mentioned facilities are included in the correct floors

Return your analysis and plan in this EXACT format:
{{
    "analysis": "Detailed analysis of every floor requirement in the user prompt",
    "special_floors": [
        {{
            "floor_number": "Floor number as specified",
            "purpose": "Exact purpose/use from requirements",
            "special_features": "Any special features mentioned"
        }}
    ],
    "floor_plan": [
        {{
            "floor_id": "Floor 1",
            "purpose": "Exact purpose matching user requirements"
        }}
    ]
}}"""

        trial = 0
        max_trials = 3
        
        while trial < max_trials:
            try:
                async with semaphore:
                    response = await async_call_llm(model, plan_prompt)

                response = repair_json(response)
                match = re.search(r"\{.*\}", response, re.DOTALL)
                
                if match:
                    try:
                        response_dict = json.loads(match.group(0))
                        
                        if "floor_plan" in response_dict:
                            example["floor_plan"] = response_dict["floor_plan"]
                            example["special_floors"] = response_dict.get("special_floors", [])
                            break
                            
                    except json.JSONDecodeError:
                        continue
                        
            except Exception as e:
                logging.error(f"Error creating floor plan: {e}")
                
            trial += 1

        # 确保有基础的楼层计划
        if "floor_plan" not in example or not example["floor_plan"]:
            example["floor_plan"] = [
                {"floor_id": "Floor 1", "purpose": "Ground floor facilities"},
                {"floor_id": "Floor 2", "purpose": "Second floor amenities"}
            ]
            example["special_floors"] = []

        return example
    
    @staticmethod
    async def async_create_menu_plan(model, example, semaphore):
        plan_prompt = f"""
You are an expert chef and menu planner. Create a precise weekly menu plan that EXACTLY matches ALL user requirements.

User requirements:
{example['prompt']}

CRITICAL ANALYSIS INSTRUCTIONS:
1. Identify EVERY specific dish, cuisine type, or dietary requirement mentioned
2. Note any special meals for specific weeks or occasions
3. Pay attention to seasonal requirements, special diets, or cultural preferences
4. Identify any recurring meal patterns or special events

Create AT LEAST 40 weeks of menu plans. If you cannot create 52 weeks, provide as many as possible (minimum 40).

Return your analysis and plan in this EXACT format:
{{
    "analysis": "Detailed analysis of every menu requirement in the user prompt",
    "special_requirements": [
        {{
            "requirement_type": "Type of special requirement",
            "details": "Specific details from user requirements",
            "week_specification": "If specified for particular weeks"
        }}
    ],
    "weekly_plan": [
        {{
            "week_id": "Week 1 (January 1st - January 7th)",
            "dishes": ["List of dishes matching user requirements for this week"]
        }}
    ]
}}"""

        trial = 0
        max_trials = 3
        best_response = None
        
        while trial < max_trials:
            try:
                async with semaphore:
                    response = await async_call_llm(model, plan_prompt)

                response = repair_json(response)
                match = re.search(r"\{.*\}", response, re.DOTALL)
                
                if match:
                    try:
                        response_dict = json.loads(match.group(0))
                        
                        if "weekly_plan" in response_dict and len(response_dict["weekly_plan"]) >= 40:
                            example["weekly_plan"] = response_dict["weekly_plan"]
                            example["special_requirements"] = response_dict.get("special_requirements", [])
                            break
                        elif "weekly_plan" in response_dict and len(response_dict["weekly_plan"]) > 0:
                            if not best_response or len(response_dict["weekly_plan"]) > len(best_response.get("weekly_plan", [])):
                                best_response = response_dict
                            
                    except json.JSONDecodeError:
                        continue
                        
            except Exception as e:
                logging.error(f"Error creating menu plan: {e}")
                
            trial += 1

        # 使用最佳响应或创建基础计划
        if "weekly_plan" not in example or len(example.get("weekly_plan", [])) < 20:
            if best_response and len(best_response.get("weekly_plan", [])) > 0:
                example["weekly_plan"] = best_response["weekly_plan"]
                example["special_requirements"] = best_response.get("special_requirements", [])
            else:
                # 创建基础菜单计划
                example["weekly_plan"] = []
                for i in range(1, 53):
                    example["weekly_plan"].append({
                        "week_id": f"Week {i}",
                        "dishes": [f"Healthy meals for week {i} following user dietary requirements"]
                    })
                example["special_requirements"] = []

        return example

    @staticmethod
    async def async_create_block_plan(model, example, semaphore):
        plan_prompt = f"""
You are an expert urban planner. Create a precise city block plan that EXACTLY matches ALL user requirements.

User requirements:
{example['prompt']}

CRITICAL ANALYSIS INSTRUCTIONS:
1. Identify EVERY specific building, facility, or land use mentioned
2. Note EXACT locations or block specifications
3. Pay attention to zoning requirements, special facilities, or community needs
4. Identify any special features for specific blocks

Return your analysis and plan in this EXACT format:
{{
    "analysis": "Detailed analysis of every block planning requirement in the user prompt",
    "special_blocks": [
        {{
            "block_number": "Block identifier as specified",
            "use": "Exact use/purpose from requirements",
            "special_features": "Any special features mentioned"
        }}
    ],
    "block_plan": [
        {{
            "block_id": "Block 1",
            "use": "Exact use matching user requirements"
        }}
    ]
}}"""

        trial = 0
        max_trials = 3
        
        while trial < max_trials:
            try:
                async with semaphore:
                    response = await async_call_llm(model, plan_prompt)

                response = repair_json(response)
                match = re.search(r"\{.*\}", response, re.DOTALL)
                
                if match:
                    try:
                        response_dict = json.loads(match.group(0))
                        
                        if "block_plan" in response_dict:
                            example["block_plan"] = response_dict["block_plan"]
                            example["special_blocks"] = response_dict.get("special_blocks", [])
                            break
                            
                    except json.JSONDecodeError:
                        continue
                        
            except Exception as e:
                logging.error(f"Error creating block plan: {e}")
                
            trial += 1

        # 确保有基础的区块计划
        if "block_plan" not in example or not example["block_plan"]:
            example["block_plan"] = [
                {"block_id": "Block 1", "use": "Residential area"},
                {"block_id": "Block 2", "use": "Commercial district"}
            ]
            example["special_blocks"] = []

        return example