import re
import logging
from typing import Dict, Any, List
from .base import BaseAgent
from .common_prompts import (
    TASK_CONTEXT, RTL_CODING_STANDARDS
)

# =============================================================================
# MERGE AGENT PROMPT COMPONENTS
# =============================================================================

MERGE_AGENT_BASE_PROMPT = """
<MERGE_AGENT_ROLE>
# Merge Agent - Verilog HDL Code Integration and Final Assembly Specialist

## **Your Role in the System**
You are the **Merge Agent** in this multi-agent collaborative system. As the final integration specialist, you combine subtask results from RTL agents into a complete, synthesizable Verilog module that meets all architectural and functional requirements.

## **Primary Mission**
Integrate code fragments from multiple RTL agent subtasks into a single, complete Verilog module, ensuring architectural coherence, signal connectivity, and functional correctness.

## **Key Responsibilities**
* **Code Fragment Integration**: Combine multiple code fragments following pseudocode structure
* **Signal Declaration Management**: Ensure all signals are properly declared with correct types and widths
* **Interface Completion**: Add missing signal connections and interface definitions
* **Syntax Validation**: Fix any syntax issues arising from integration
* **Architectural Verification**: Ensure final module meets all document specifications

## **Critical Integration Process**
Think through integration systematically:
1. **Fragment Analysis**: Review each code fragment's functionality and signal requirements
2. **Gap Identification**: Identify missing declarations, connections, and interface elements
3. **Integration Strategy**: Plan how fragments fit together following pseudocode architecture
4. **Signal Resolution**: Ensure all signals are properly declared and connected
5. **Syntax Validation**: Verify complete module has valid Verilog syntax
6. **Final Assembly**: Generate complete, synthesizable module code

**Important**: We extract only the **LAST** Verilog code block from your response
</MERGE_AGENT_ROLE>
"""

MERGE_INTEGRATION_GUIDELINES = """
<MERGE_INTEGRATION_GUIDELINES>
# Critical Merge Integration Guidelines

## **Core Integration Principles**

### **Signal Declaration Preservation (ABSOLUTE PRIORITY)**
* **STRICT IO SIGNAL PRESERVATION**: Absolutely preserve ALL IO signal declarations from subtask results - DO NOT lose any signals, DO NOT add new signals, except the document has mentioned.
* **Conditional Port Ordering**: Follow RTL rules - place `ifdef/`ifndef conditional ports BEFORE regular ports to prevent undefined last ports
* **Zero Tolerance for Signal Loss**: Every signal from any subtask MUST be preserved in the final module

### **Pseudocode-Driven Assembly**
* **Framework Compliance**: Assemble subtasks according to pseudocode structure as the primary organization principle
* **Syntax Problem Resolution**: After assembly, fix ALL syntax issues including missing signals and intermediate signal definitions
* **Structural Integrity**: Maintain the architectural integrity defined by the pseudocode framework

### **Parameter and Macro Handling**
* **Local Parameter Awareness**: Note that some subtasks may define local parameters or custom macros
* **Header Macro Priority**: Use ONLY macros from header files - do NOT add module header parameters unless explicitly required by documentation
* **Macro Reference Only**: Reference header macros with backtick prefix (`MACRO_NAME), never define new ones
* **Documentation Override**: Only add module parameters if documentation explicitly requires it

### **Document-First Compliance**
* **Primary Standard**: Use original documentation as the FIRST and ONLY standard for final results
* **Specification Adherence**: Ensure final module completely complies with all document specifications
* **Functional Verification**: Verify that all documented requirements are properly implemented

## **Integration Process**
1. **Preserve All Subtask IO**: Collect and preserve every IO signal declaration from all subtasks
2. **Order by RTL Rules**: Arrange signals following conditional compilation port ordering rules
3. **Assemble by Pseudocode**: Combine subtasks according to pseudocode block structure
4. **Resolve Syntax Issues**: Fix all syntax problems including missing signal declarations and connections
5. **Verify Against Document**: Ensure final result meets all documentation requirements
6. **Final Validation**: Confirm complete module functionality and correctness

</MERGE_INTEGRATION_GUIDELINES>
"""

MERGE_OUTPUT_FORMAT = """
<MERGE_OUTPUT_FORMAT>
# Output Format

## **Final Code Structure**
After your analysis and verification, provide only the complete, syntactically correct Verilog module:

```verilog
// Complete, syntactically correct Verilog module
// All subproblem functionality integrated and preserved
// All signals properly declared from subproblems and document
// All document-specified IO signals included
// All syntax errors fixed
// All macros verified and properly used
```
</MERGE_OUTPUT_FORMAT>
"""

# Complete prompt assembly using + operator for consistency
MERGE_AGENT_COMPLETE_PROMPT = (
    TASK_CONTEXT +
    MERGE_AGENT_BASE_PROMPT +
    MERGE_INTEGRATION_GUIDELINES +
    RTL_CODING_STANDARDS +
    MERGE_OUTPUT_FORMAT
)

class MergeAgent(BaseAgent):
    """Agent for merging subproblem implementations into complete module"""
    
    def __init__(self, llm_client, logger=None):
        super().__init__(llm_client, "MergeAgent", logger)
    
    def run(self, merge_input: Dict[str, Any]) -> Dict[str, Any]:
        """
        Merge subproblem implementations into complete module
        
        Args:
            merge_input: {
                "subproblems": [...],
                "subproblem_code_fragments": [...],
                "merge_document": "relevant document sections"
            }
        
        Returns:
            {
                "final_code": "complete verilog module",
                "comment": "merge assessment and confidence"
            }
        """
        subproblems = merge_input["subproblems"]
        subproblem_code_fragments = merge_input["subproblem_code_fragments"]
        merge_document = merge_input["merge_document"]
        
        self.logger.info(f"Merging {len(subproblem_code_fragments)} subproblem code fragments")
        
        # Generate final integrated code
        final_code = self._generate_merged_code(
            subproblems, subproblem_code_fragments, merge_document, merge_input["task_name"]
        )
        
        self.logger.info(f"Generated merged code of {len(final_code)} characters")
        
        return {"final_code": final_code}
    
    def _generate_merged_code(self, subproblems: List[Dict[str, Any]], 
                            subproblem_code_fragments: List[str], 
                            merge_document: str,
                            task_name: str) -> str:
        """Generate merged Verilog code using LLM"""
        
        # Prepare subproblem descriptions and code fragments for context
        subproblem_info = ""
        for i, (subproblem, code_fragment) in enumerate(zip(subproblems, subproblem_code_fragments)):
            subproblem_info += f"// Subproblem {i+1}: {subproblem.get('description', '')}\n"
            subproblem_info += f"// Code Fragment:\n{code_fragment}\n\n"
        
        # Use pre-assembled complete prompt
        system_prompt = MERGE_AGENT_COMPLETE_PROMPT
        
        prompt = self.build_prompt(
            system_prompt + "\n\n" +
            "TASK: Intelligently integrate subproblem code fragments into complete, syntactically correct Verilog module.\n\n" +
            "CRITICAL REQUIREMENTS:\n" +
            "- PRESERVE all functional code from subproblem fragments - do NOT rewrite or discard\n" +
            "- INTEGRATE subproblem code following structured pseudocode organization\n" +
            "- FIX syntax errors, missing declarations, and connection issues during integration\n" +
            "- ENSURE every signal from subproblems and document is properly declared\n" +
            "- VERIFY all document-specified IO signals are included in final module\n" +
            "- COMPLETE any missing signal connections between subproblem modules\n" +
            "- Only use macros explicitly mentioned in documentation\n\n" +
            "--- AUTHORITATIVE DOCUMENT SECTIONS (for verification and completion) ---\n{merge_document}\n\n" +
            "--- SUBPROBLEM CODE FRAGMENTS (to be integrated, NOT rewritten) ---\n{subproblem_info}\n\n" +
            "Remember: Your primary task is INTEGRATION with SYNTAX REPAIR, not rewriting. Follow the quality assurance checklist before outputting code.",
            subproblem_info=subproblem_info,
            merge_document=merge_document,
            task_name=task_name
        )
        
        response = self.llm_complete(prompt)
        
        # Extract the last Verilog code block from response
        verilog_blocks = []
        for match in re.finditer(r'```verilog\s*(.*?)\s*```', response, re.DOTALL):
            code_block = match.group(1).strip()
            if "module" in code_block and "endmodule" in code_block:
                verilog_blocks.append(code_block)
        
        if verilog_blocks:
            # Use the last complete Verilog block
            final_code = verilog_blocks[-1]
            return final_code
        else:
            raise ValueError(f"No complete Verilog code found in response.")