from llms.llms import async_call_llm
from utils.wordCounter import count_words
from json_repair import repair_json
import json
import re
import asyncio
import logging
import copy

class GenerationAgent:

    @staticmethod
    async def async_generate(model, example, semaphore):
        # 深拷贝输入数据，避免修改原始数据
        example = copy.deepcopy(example)
        
        # 先标准化数据结构，确保格式正确
        if 'weekly_plan' in example:
            if not isinstance(example['weekly_plan'], list):
                example['weekly_plan'] = []
        
        if example['type'] == 'Week':
            example = await GenerationAgent.async_generate_week(model, example, semaphore)
        elif example["type"] == "Floor":
            example = await GenerationAgent.async_generate_floor(model, example, semaphore)
        elif example["type"] == "Menu Week":
            example = await GenerationAgent.async_generate_menu(model, example, semaphore)
        elif example["type"] == "Block":
            example = await GenerationAgent.async_generate_block(model, example, semaphore)
        return example

    @staticmethod
    async def async_generate_week(model, example, semaphore):
        # 确保 weekly_plan 是列表
        if 'weekly_plan' not in example or not example['weekly_plan']:
            example['weekly_plan'] = []
        elif not isinstance(example['weekly_plan'], list):
            if isinstance(example['weekly_plan'], str):
                example['weekly_plan'] = []
            elif isinstance(example['weekly_plan'], dict):
                example['weekly_plan'] = [example['weekly_plan']]
            else:
                example['weekly_plan'] = []

        # 创建一个新列表来保存处理后的周计划
        processed_weeks = []
        
        # 确保每个元素都包含必要字段
        for i, week in enumerate(example['weekly_plan']):
            # 如果元素不是字典，创建一个新字典
            if not isinstance(week, dict):
                processed_week = {
                    'week_id': f"Week {i+1}",
                    'events': str(week) if week else ""
                }
            else:
                processed_week = week.copy()
            
            # 确保必要字段存在且设置默认值
            if 'week_id' not in processed_week:
                processed_week['week_id'] = f"Week {i+1}"
            if 'events' not in processed_week:
                processed_week['events'] = ""
            
            processed_week['length_requirement'] = 200
            processed_weeks.append(processed_week)
        
        # 用处理后的列表替换原始列表
        example['weekly_plan'] = processed_weeks

        async def process_week(week):
            week_id = week.get('week_id', "Unknown Week")
            events = week.get('events', "")
            
            # 确保events是字符串
            if not isinstance(events, str):
                if isinstance(events, list):
                    events = ", ".join([str(event) for event in events])
                else:
                    events = str(events)
            
            # 增强的prompt，更注重用户需求的精确匹配
            prompt = f"""You are an expert writer with perfect attention to user requirements and special event placement.

TASK: Write a detailed 200-word weekly diary entry for {week_id}.

MANDATORY REQUIREMENTS TO VERIFY AND INCLUDE:
1. Specific events planned for this week: {events}
2. User requirements that must be satisfied: {example['prompt']}
3. Overall yearly context: {example.get('special_events', [])}

CRITICAL VERIFICATION STEPS:
1. CHECK: Does the user requirement specify ANY special events for {week_id} specifically?
2. CHECK: Are there birthdays, anniversaries, or special occasions that fall in {week_id}?
3. CHECK: Does the events list contain special requirements that must be prominently featured?
4. CHECK: Does this week have any periodic events (weekly, monthly, etc.) that should occur?

WRITING INSTRUCTIONS:
- If user requirements mention specific events for {week_id}, feature them PROMINENTLY
- If the events list contains special occasions, make them the FOCUS of the entry
- Ensure the diary entry PROVES that specified requirements are being met
- Use specific details that demonstrate compliance with user requirements
- Make special events clearly identifiable in the text

VERIFICATION BEFORE FINALIZING:
✓ Does this entry include ALL events specified for {week_id}?
✓ Does this entry satisfy user requirements for this specific week?
✓ Are special events from user requirements clearly featured and identifiable?
✓ Would an evaluator reading this entry be able to confirm that requirements are met?

Return ONLY valid JSON:
{{
    "week_id": "{week_id}",
    "check": "Detailed explanation of how this entry satisfies user requirements and includes all specified events with specific evidence",
    "diary_entry": "Exactly 200 words describing the week's activities, ensuring all user requirements and specified events are clearly included and prominently featured" 
}}"""

            logging.info(f"Generating initial diary entry for week {week_id}")

            # 最多尝试3次获取有效响应
            max_attempts = 3
            for attempt in range(max_attempts):
                try:
                    async with semaphore:
                        response = await async_call_llm(model, prompt)
                    
                    response = repair_json(response)
                    match = re.search(r"\{.*\}", response, re.DOTALL)
                    
                    if match:
                        try:
                            diary_entry = json.loads(match.group(0))
                            
                            if 'diary_entry' in diary_entry and len(diary_entry['diary_entry'].strip()) > 50:
                                week['diary_entry'] = str(diary_entry['diary_entry'])
                                week['check'] = str(diary_entry.get('check', ''))
                                week['length_requirement'] = 200
                                break
                        except json.JSONDecodeError:
                            continue
                            
                except Exception as e:
                    logging.error(f"Error generating diary for {week_id}, attempt {attempt+1}: {e}")

            # 确保diary_entry字段存在
            if 'diary_entry' not in week:
                week['diary_entry'] = f"Weekly diary for {week_id}: {events}. This week fulfills the user requirements through planned activities and events."
                week['check'] = "Fallback entry created to ensure requirements are addressed"

            # Refine the diary entry - 添加限制避免无限循环
            try:
                current_length = count_words(week['diary_entry'])
                target_length = 200
                
                if abs(current_length - target_length) > 30:  # 允许±30词的误差
                    refinement_prompt = f"""Refine this diary entry to be closer to 200 words while maintaining ALL content and ensuring user requirements are clearly met:

Current entry ({current_length} words):
{week['diary_entry']}

User requirements: {example['prompt']}
Specific events for {week_id}: {events}

Instructions:
- Adjust length to approximately 200 words
- Keep ALL special events and requirements clearly mentioned
- Ensure compliance with user requirements is obvious
- Make special events prominent and identifiable

Return only the refined diary entry text (no JSON, just the text):"""

                    async with semaphore:
                        refined_text = await async_call_llm(model, refinement_prompt)
                    
                    if refined_text and len(refined_text.strip()) > 50:
                        # Clean up the response to extract just the diary text
                        refined_text = refined_text.strip()
                        # Remove any JSON formatting if present
                        if refined_text.startswith('{') or refined_text.startswith('"'):
                            # Try to extract just the diary content
                            lines = refined_text.split('\n')
                            for line in lines:
                                if len(line.strip()) > 50 and not line.strip().startswith('{') and not line.strip().startswith('"'):
                                    refined_text = line.strip()
                                    break
                        
                        week['diary_entry'] = refined_text
                    
            except Exception as e:
                logging.error(f"Error refining diary for {week_id}: {e}")
            
            logging.info(f"Final diary entry word count for {week_id}: {count_words(week['diary_entry'])}")
            return week

        # Process all weeks concurrently
        tasks = [process_week(week) for week in example['weekly_plan']]
        
        try:
            processed_results = await asyncio.gather(*tasks)
            example['weekly_plan'] = processed_results
        except Exception as e:
            logging.error(f"Error during concurrent processing: {e}")
            # 确保至少有一些结果
            processed_results = []
            for week in example['weekly_plan']:
                if 'diary_entry' not in week:
                    week['diary_entry'] = f"Weekly diary for {week.get('week_id', 'Unknown')}: Basic activities and events to meet user requirements."
                processed_results.append(week)
            example['weekly_plan'] = processed_results
        
        example['final_text'] = GenerationAgent.get_final_week_text(example['weekly_plan'])
        return example

    @staticmethod
    def get_final_week_text(weekly_plan):
        """
        Get the final text from the generated text with respect to the plan.
        """
        text = ""

        for week in weekly_plan:
            if isinstance(week, dict):
                week_id = week.get('week_id', 'Unknown Week')
                diary_entry = week.get('diary_entry', '')
                text += f"{week_id}:\n{diary_entry}\n\n"
            else:
                text += f"{str(week)}\n\n"

        text += '*** finished ***'
        return text

    @staticmethod
    async def async_generate_floor(model, example, semaphore):
        # 确保 floor_plan 是列表
        if 'floor_plan' not in example or not example['floor_plan']:
            example['floor_plan'] = []
        elif not isinstance(example['floor_plan'], list):
            example['floor_plan'] = [example['floor_plan']] if example['floor_plan'] else []

        # 创建一个新列表来保存处理后的楼层计划
        processed_floors = []
        
        # 处理每个元素
        for i, floor in enumerate(example['floor_plan']):
            if not isinstance(floor, dict):
                processed_floor = {
                    'floor_id': f"Floor {i+1}",
                    'purpose': str(floor) if floor else ""
                }
            else:
                processed_floor = floor.copy()
            
            if 'floor_id' not in processed_floor:
                processed_floor['floor_id'] = f"Floor {i+1}"
            if 'purpose' not in processed_floor:
                processed_floor['purpose'] = ""
            
            processed_floors.append(processed_floor)
        
        # 用处理后的列表替换原始列表
        example['floor_plan'] = processed_floors

        async def process_floor(floor):
            floor_id = floor.get('floor_id', "Unknown Floor")
            purpose = floor.get('purpose', "")
            
            prompt = f"""You are an expert architect with perfect attention to user specifications and building requirements.

TASK: Write a detailed 150-word floor plan description for {floor_id}.

MANDATORY REQUIREMENTS TO VERIFY AND INCLUDE:
1. Floor purpose: {purpose}
2. User requirements: {example['prompt']}

CRITICAL VERIFICATION STEPS:
1. CHECK: Does the user requirement specify ANY facilities for {floor_id} specifically?
2. CHECK: Are there special building requirements that must be included on this floor?
3. CHECK: Does this floor need specific amenities or features as per user requirements?
4. CHECK: Are there accessibility or special use requirements?

DESIGN INSTRUCTIONS:
- If user requirements mention facilities for {floor_id}, feature them PROMINENTLY
- Include specific architectural details that demonstrate compliance
- Ensure the floor plan PROVES that specified requirements are being met
- Make special facilities clearly identifiable and well-integrated

VERIFICATION BEFORE FINALIZING:
✓ Does this plan include ALL facilities specified for {floor_id}?
✓ Does this plan satisfy user requirements for this specific floor?
✓ Are special requirements clearly featured and identifiable?
✓ Would an evaluator be able to confirm that floor requirements are met?

Return ONLY valid JSON:
{{
    "floor_id": "{floor_id}",
    "check": "Detailed explanation of how this plan meets user requirements and includes specified facilities",
    "plan": "Exactly 150 words describing the floor plan, ensuring all user requirements and specified purposes are clearly included"
}}"""

            try:
                async with semaphore:
                    response = await async_call_llm(model, prompt)
                
                response = repair_json(response)
                match = re.search(r"\{.*\}", response, re.DOTALL)
                
                if match:
                    floor_data = json.loads(match.group(0))
                    if 'plan' in floor_data:
                        floor['plan'] = str(floor_data['plan'])
                        floor['check'] = str(floor_data.get('check', ''))
                    else:
                        floor['plan'] = f"Floor plan for {floor_id}: {purpose}. Designed to meet user requirements with appropriate facilities."
                else:
                    floor['plan'] = f"Floor plan for {floor_id}: {purpose}. Designed to meet user requirements."
                    
            except Exception as e:
                logging.error(f"Error generating floor plan for {floor_id}: {e}")
                floor['plan'] = f"Floor plan for {floor_id}: {purpose}. Designed to meet user requirements."
            
            return floor

        # Process all floors concurrently
        tasks = [process_floor(floor) for floor in example['floor_plan']]
        
        try:
            processed_results = await asyncio.gather(*tasks)
            example['floor_plan'] = processed_results
        except Exception as e:
            logging.error(f"Error during floor processing: {e}")
            processed_results = []
            for floor in example['floor_plan']:
                if 'plan' not in floor:
                    floor['plan'] = f"Basic floor plan for {floor.get('floor_id', 'Unknown')}"
                processed_results.append(floor)
            example['floor_plan'] = processed_results
        
        example['final_text'] = GenerationAgent.get_final_floor_text(example['floor_plan'])

        return example

    @staticmethod
    def get_final_floor_text(floor_plan):
        """
        Get the final text from the generated text with respect to the plan.
        """
        text = ""

        for floor in floor_plan:
            if isinstance(floor, dict):
                floor_id = floor.get('floor_id', 'Unknown Floor')
                plan = floor.get('plan', '')
                text += f"{floor_id}:\n{plan}\n\n"
            else:
                text += f"{str(floor)}\n\n"

        text += '*** finished'
        return text
    
    @staticmethod
    async def async_generate_menu(model, example, semaphore):
        # 确保 weekly_plan 是列表
        if 'weekly_plan' not in example or not example['weekly_plan']:
            example['weekly_plan'] = []
        elif not isinstance(example['weekly_plan'], list):
            example['weekly_plan'] = [example['weekly_plan']] if example['weekly_plan'] else []

        # 创建一个新列表来保存处理后的菜单计划
        processed_weeks = []

        # 确保每个元素都包含必要字段
        for i, week in enumerate(example['weekly_plan']):
            if not isinstance(week, dict):
                processed_week = {
                    'week_id': f"Week {i+1}",
                    'dishes': [str(week)] if week else []
                }
            else:
                processed_week = week.copy()
            
            if 'week_id' not in processed_week:
                processed_week['week_id'] = f"Week {i+1}"
            if 'dishes' not in processed_week:
                processed_week['dishes'] = []
            
            processed_weeks.append(processed_week)

        # 用处理后的列表替换原始列表
        example['weekly_plan'] = processed_weeks

        async def process_menu(week):
            week_id = week.get('week_id', "Unknown Menu Week")
            dishes = week.get('dishes', [])
            
            # 确保dishes是字符串列表
            if isinstance(dishes, str):
                dishes_str = dishes
            elif isinstance(dishes, list):
                dishes_str = ", ".join([str(dish) for dish in dishes])
            else:
                dishes_str = str(dishes)
            
            prompt = f"""You are an expert chef with perfect attention to user specifications and dietary requirements.

TASK: Write a detailed 200-word menu plan for {week_id}.

MANDATORY REQUIREMENTS TO VERIFY AND INCLUDE:
1. Planned dishes: {dishes_str}
2. User requirements: {example['prompt']}

CRITICAL VERIFICATION STEPS:
1. CHECK: Does the user requirement specify ANY special dishes for {week_id} specifically?
2. CHECK: Are there dietary restrictions, cultural preferences, or special occasions?
3. CHECK: Does this week require specific cuisines or meal types?
4. CHECK: Are there seasonal or ingredient requirements to consider?

MENU PLANNING INSTRUCTIONS:
- If user requirements mention dishes for {week_id}, feature them PROMINENTLY
- Include specific culinary details that demonstrate compliance
- Ensure the menu PROVES that specified requirements are being met
- Make special dishes and dietary considerations clearly identifiable

VERIFICATION BEFORE FINALIZING:
✓ Does this menu include ALL dishes specified for {week_id}?
✓ Does this menu satisfy user dietary and cultural requirements?
✓ Are special dishes clearly featured and prominent?
✓ Would an evaluator be able to confirm that menu requirements are met?

Return ONLY valid JSON:
{{
    "week_id": "{week_id}",
    "check": "Detailed explanation of how this menu meets user requirements and includes specified dishes",
    "week_menu": "Exactly 200 words describing the weekly menu, ensuring all user requirements and specified dishes are clearly included"
}}"""

            try:
                async with semaphore:
                    response = await async_call_llm(model, prompt)
                
                response = repair_json(response)
                match = re.search(r"\{.*\}", response, re.DOTALL)
                
                if match:
                    menu_data = json.loads(match.group(0))
                    if 'week_menu' in menu_data:
                        week['week_menu'] = str(menu_data['week_menu'])
                        week['check'] = str(menu_data.get('check', ''))
                    else:
                        week['week_menu'] = f"Menu for {week_id}: {dishes_str}. Planned to meet user dietary requirements."
                else:
                    week['week_menu'] = f"Menu for {week_id}: {dishes_str}. Designed to satisfy user requirements."
                    
            except Exception as e:
                logging.error(f"Error generating menu for {week_id}: {e}")
                week['week_menu'] = f"Menu for {week_id}: {dishes_str}. Meets user requirements."
            
            return week

        # 并发处理所有菜单周
        tasks = [process_menu(week) for week in example['weekly_plan']]

        try:
            processed_results = await asyncio.gather(*tasks)
            example['weekly_plan'] = processed_results
        except Exception as e:
            logging.error(f"Error during menu processing: {e}")
            processed_results = []
            for week in example['weekly_plan']:
                if 'week_menu' not in week:
                    week['week_menu'] = f"Basic menu for {week.get('week_id', 'Unknown')}"
                processed_results.append(week)
            example['weekly_plan'] = processed_results

        example['final_text'] = GenerationAgent.get_final_menu_text(example['weekly_plan'])
        return example

    @staticmethod
    def get_final_menu_text(weekly_plan):
        """
        Get the final text from the generated text with respect to the plan.
        """
        text = ""

        for week in weekly_plan:
            if isinstance(week, dict):
                week_id = week.get('week_id', 'Unknown Week')
                menu = week.get('week_menu', '')
                text += f"{week_id}:\n{menu}\n\n"
            else:
                text += f"{str(week)}\n\n"

        text += '*** finished ***'
        return text

    @staticmethod
    async def async_generate_block(model, example, semaphore):
        # 确保 block_plan 是列表
        if 'block_plan' not in example or not example['block_plan']:
            example['block_plan'] = []
        elif not isinstance(example['block_plan'], list):
            example['block_plan'] = [example['block_plan']] if example['block_plan'] else []

        # 创建一个新列表来保存处理后的区块计划
        processed_blocks = []
        
        # 处理每个元素
        for i, block in enumerate(example['block_plan']):
            if not isinstance(block, dict):
                processed_block = {
                    'block_id': f"Block {i+1}",
                    'use': str(block) if block else ""
                }
            else:
                processed_block = block.copy()
            
            if 'block_id' not in processed_block:
                processed_block['block_id'] = f"Block {i+1}"
            if 'use' not in processed_block:
                processed_block['use'] = ""
            
            processed_blocks.append(processed_block)
        
        # 用处理后的列表替换原始列表
        example['block_plan'] = processed_blocks

        async def process_block(block):
            block_id = block.get('block_id', "Unknown Block")
            use = block.get('use', "")
            
            prompt = f"""You are an expert urban planner with perfect attention to user specifications and community needs.

TASK: Write a detailed 150-word city block plan for {block_id}.

MANDATORY REQUIREMENTS TO VERIFY AND INCLUDE:
1. Block use: {use}
2. User requirements: {example['prompt']}

CRITICAL VERIFICATION STEPS:
1. CHECK: Does the user requirement specify ANY facilities for {block_id} specifically?
2. CHECK: Are there community needs or zoning requirements mentioned?
3. CHECK: Does this block need specific infrastructure or amenities?
4. CHECK: Are there accessibility or public service requirements?

PLANNING INSTRUCTIONS:
- If user requirements mention facilities for {block_id}, feature them PROMINENTLY
- Include specific urban planning details that demonstrate compliance
- Ensure the block plan PROVES that specified requirements are being met
- Make special facilities and community features clearly identifiable

VERIFICATION BEFORE FINALIZING:
✓ Does this plan include ALL facilities specified for {block_id}?
✓ Does this plan satisfy user community and zoning requirements?
✓ Are special requirements clearly featured and integrated?
✓ Would an evaluator be able to confirm that block requirements are met?

Return ONLY valid JSON:
{{
    "block_id": "{block_id}",
    "check": "Detailed explanation of how this plan meets user requirements and includes specified facilities",
    "plan": "Exactly 150 words describing the block plan, ensuring all user requirements and specified uses are clearly included"
}}"""

            try:
                async with semaphore:
                    response = await async_call_llm(model, prompt)
                
                response = repair_json(response)
                match = re.search(r"\{.*\}", response, re.DOTALL)
                
                if match:
                    block_data = json.loads(match.group(0))
                    if 'plan' in block_data:
                        block['plan'] = str(block_data['plan'])
                        block['check'] = str(block_data.get('check', ''))
                    else:
                        block['plan'] = f"Block plan for {block_id}: {use}. Designed to meet user community requirements."
                else:
                    block['plan'] = f"Block plan for {block_id}: {use}. Planned to satisfy user requirements."
                    
            except Exception as e:
                logging.error(f"Error generating block plan for {block_id}: {e}")
                block['plan'] = f"Block plan for {block_id}: {use}. Meets user requirements."
            
            return block

        # Process all blocks concurrently
        tasks = [process_block(block) for block in example['block_plan']]
        
        try:
            processed_results = await asyncio.gather(*tasks)
            example['block_plan'] = processed_results
        except Exception as e:
            logging.error(f"Error during block processing: {e}")
            processed_results = []
            for block in example['block_plan']:
                if 'plan' not in block:
                    block['plan'] = f"Basic plan for {block.get('block_id', 'Unknown')}"
                processed_results.append(block)
            example['block_plan'] = processed_results

        example['final_text'] = GenerationAgent.get_final_block_text(example['block_plan'])

        return example

    @staticmethod
    def get_final_block_text(block_plan):
        """
        Get the final text from the generated text with respect to the plan.
        """
        text = ""

        for block in block_plan:
            if isinstance(block, dict):
                block_id = block.get('block_id', 'Unknown Block')
                plan = block.get('plan', '')
                text += f"{block_id}:\n{plan}\n\n"
            else:
                text += f"{str(block)}\n\n"

        text += '*** finished'
        return text