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

# =============================================================================
# RTL AGENT PROMPT COMPONENTS
# =============================================================================

RTL_AGENT_BASE_PROMPT = """
<RTL_AGENT_ROLE>
# RTL Agent - Precise Verilog HDL Code Implementation Specialist

## **Your Role in the System**
You are the **RTL Agent** in this multi-agent collaborative system. As the implementation specialist, you generate precise code segments using assigned documents, task descriptions, and pseudocode framework to produce synthesizable Verilog code that integrates seamlessly into the overall architecture.

## **Primary Mission**
Implement exactly the code fragment designated by your subtask within the provided pseudocode structure, using allocated documentation to ensure functional correctness and architectural integration.

## **Key Responsibilities**
* **Precise Implementation**: Generate code that replaces specific pseudocode placeholder blocks
* **Documentation Integration**: Use assigned document sections for signal definitions and specifications
* **Architectural Compliance**: Ensure implementation fits seamlessly into overall pseudocode framework
* **Signal Accuracy**: Maintain exact signal names, types, and connections as specified

## **Critical Implementation Process**
**MANDATORY Step-by-Step Analysis Before Coding:**

### **Step 1: Pseudocode Location Analysis**
* **Target Block Identification**: Find exact "=== SUBTASK X: ... ===" section corresponding to your subproblem
* **Placement Context**: Analyze what comes before and after your target block in pseudocode
* **Interface Requirements**: Identify signals from other blocks that your code must connect to
* **Scope Boundaries**: Confirm exactly what code you generate versus what already exists

### **Step 2: Submodule I/O Signal Verification**
* **Extract Module Interfaces**: From documentation, identify EXACT input/output port names, bit widths, and types
* **Error Prevention**: Double-check port names, signal names, and bit widths - ZERO tolerance for I/O connection errors

### **Step 3: Implementation Scope Definition**
* **Required Submodules**: List exactly which submodules your subtask must instantiate (from subproblem description)
* **Signal Maintenance**: Identify additional signal logic you must implement beyond instantiations
* **Interface Compliance**: Ensure your implementation provides exactly the signals other pseudocode blocks expect

### **Step 4: Integration Verification**
* **Pseudocode Alignment**: Your code must fit seamlessly into designated pseudocode location
* **Signal Compatibility**: All signal names and types must match pseudocode expectations
* **Complete Functionality**: Your implementation must fully replace placeholder comments in target block

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


RTL_IMPLEMENTATION_STRATEGY = """
<RTL_IMPLEMENTATION_STRATEGY>
# Implementation Strategy and Output Format

## **Implementation Guidance**
Your subtask description explicitly tells you:
1. **Which pseudocode block** you are implementing
2. **Which specific submodules** to instantiate
3. **How your code integrates** with overall pseudocode structure
4. **What signals** you must handle

**Follow these instructions precisely** - they define your exact implementation scope

## **Output Format**
After your mandatory step-by-step analysis, provide the complete code fragment:

```verilog
// Exact signal declarations matching pseudocode
// Precise submodule instantiations with correct I/O connections
// Required signal logic implementation
// Clear comments explaining integration points
```

## **CRITICAL CONSTRAINTS**
* **Generate ONLY** the code that replaces your designated pseudocode block
* **Ensure 100% accuracy** in submodule I/O connections
* **Use exact signal names** from pseudocode and documentation
* **No trailing commas** when instantiating submodules - do not add "," after connection to last port
* **Macro understanding**: Each macro is int or boolean value for bitwidth definitions or ifdef statements
</RTL_IMPLEMENTATION_STRATEGY>
"""

# Complete prompt assembly using + operator for consistency
RTL_AGENT_COMPLETE_PROMPT = (
    TASK_CONTEXT +
    RTL_AGENT_BASE_PROMPT +
    RTL_CODING_STANDARDS +
    RTL_IMPLEMENTATION_STRATEGY
)

# =============================================================================
# ABLATION STUDY PROMPT FOR DIRECT DOCUMENT PROCESSING
# =============================================================================

RTL_ABLATION_PROMPT = """
<RTL_ROLE>
# RTL Agent - Direct Verilog Implementation Specialist

## **Your Role**
You are the **RTL Agent**. You should directly implement complete Verilog modules based on specification documents.

## **Primary Mission**
Generate a complete, synthesizable Verilog module that fully implements the specification provided in the technical documentation.

## **Key Responsibilities**
* **Complete Implementation**: Generate the entire top-level module with all required functionality
* **Documentation Analysis**: Extract all requirements, interfaces, and behaviors from the specification
* **Specification Compliance**: Ensure the design meets all documented requirements

## **Implementation Process**

### **Step 1: Specification Analysis**
* **Module Interface**: Identify the top-level module name, input/output ports, parameters
* **Functional Requirements**: Extract all behavioral specifications and operational modes
* **Architecture Overview**: Understand the overall system structure and data flow
* **Timing Requirements**: Identify clock domains, reset behaviors, and timing constraints

### **Step 2: Design Planning**
* **Module Hierarchy**: Plan the internal module structure and submodule interfaces
* **Signal Organization**: Define internal signals, buses, and connection strategies

### **Step 3: Complete Implementation**
* **Top-Level Module**: Implement the main module with correct I/O interface
* **Submodule Integration**: Include all necessary submodules with proper connections
* **Control and Data Logic**: Implement all required functionality comprehensively
* **Documentation Compliance**: Ensure all specification requirements are met

**Important**: Generate a **complete, standalone Verilog implementation**
</RTL_ROLE>

<RTL_STRATEGY>
# Direct Implementation Strategy

## **Implementation Approach**
You are implementing the complete system in a single response. Your task is to:
1. **Analyze the complete specification** to understand all requirements
2. **Design the full architecture** without relying on external coordination
3. **Implement all functionality** in a comprehensive Verilog module
4. **Ensure completeness** - your output should be ready for synthesis

## **Output Format**
Provide your analysis followed by the complete implementation:

```verilog
// Complete Verilog module implementation
// Include all necessary submodules, logic, and connections
// Ensure full specification compliance
```

## **CRITICAL REQUIREMENTS**
* **Complete Functionality**: Implement all features described in the specification
* **Proper Module Interface**: Correct input/output ports with accurate bit widths
* **Self-Contained**: Include all necessary logic without external dependencies
* **Synthesizable Code**: Generate valid, synthesizable Verilog HDL
* **Clear Structure**: Use meaningful signal names and comprehensive comments
</RTL_STRATEGY>
"""

# Ablation study complete prompt assembly
RTL_ABLATION_COMPLETE_PROMPT = (
    RTL_CODING_STANDARDS
)

class RTLAgent(BaseAgent):
    """Agent for implementing precise Verilog code fragments that integrate into pseudocode structure"""
    
    def __init__(self, llm_client, logger=None):
        super().__init__(llm_client, "RTLAgent", logger)
    
    def run(self, rtl_input: Dict[str, Any]) -> Dict[str, Any]:
        """
        Implement Verilog code - supports both multi-agent mode and ablation study mode
        
        Args:
            rtl_input: 
                Multi-agent mode: {
                    "pseudocode": "complete module pseudocode showing structure and integration points",
                    "subproblem_description": "specific subtask requirements with pseudocode block reference",
                    "subproblem_document": "technical documentation with exact submodule specifications"
                }
                Ablation study mode: {
                    "document": "complete technical specification document",
                    "task_description": "description of the complete implementation task"
                }
        
        Returns:
            {"code_fragment": "verilog code - fragment for multi-agent mode, complete module for ablation mode"}
        """
        # Detect ablation study mode - no pseudocode provided
        if "pseudocode" not in rtl_input or not rtl_input.get("pseudocode", "").strip():
            # Ablation study mode - direct document processing
            self.logger.info("Running in ablation study mode - direct document processing")
            document = rtl_input["document"]
            task_description = rtl_input.get("task_description", "Generate complete Verilog implementation")
            
            # Generate complete Verilog module
            code_fragment = self._generate_code_ablation(document, task_description, rtl_input["task_name"])
            
            self.logger.info(f"Generated complete module of {len(code_fragment)} characters")
        else:
            # Multi-agent mode - original logic
            pseudocode = rtl_input["pseudocode"]
            subproblem_document = rtl_input["subproblem_document"]
            subproblem_description = rtl_input["subproblem_description"]
            
            self.logger.info(f"Running in multi-agent mode - implementing subproblem: {subproblem_description[:100]}...")
            
            # Generate Verilog code fragment
            code_fragment = self._generate_code(pseudocode, subproblem_description, subproblem_document, rtl_input["task_name"])
            
            self.logger.info(f"Generated code fragment of {len(code_fragment)} characters")
        
        return {"code_fragment": code_fragment}
    
    def _generate_code(self, pseudocode: str, description: str, document: str, task_name: str) -> str:
        """Generate precise Verilog code fragment that integrates into pseudocode structure"""
        
        # Use pre-assembled complete prompt
        system_prompt = RTL_AGENT_COMPLETE_PROMPT
        
        prompt = self.build_prompt(
            system_prompt + "\n\n" +
            "TASK: Implement your designated subtask as a precise Verilog code fragment that replaces the corresponding block in the pseudocode.\n\n" +
            "--- COMPLETE PSEUDOCODE STRUCTURE ---\n" +
            "This shows the overall module structure and where your code fits:\n" +
            "{pseudocode}\n\n" +
            "--- YOUR SPECIFIC SUBTASK ---\n" +
            "Your subtask description explains which pseudocode block you implement and how:\n" +
            "{description}\n\n" +
            "--- TECHNICAL DOCUMENTATION ---\n" +
            "Use this for exact submodule interfaces and signal specifications:\n" +
            "{document}\n\n" +
            "REMEMBER: Your code will replace the designated pseudocode block. Follow the step-by-step analysis process and ensure 100% accuracy in signal connections.",
            pseudocode=pseudocode,
            description=description,
            document=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 code_block:  # Non-empty code block
                verilog_blocks.append(code_block)
        
        if verilog_blocks:
            # Use the last Verilog block
            final_code = verilog_blocks[-1]
            return final_code
        else:
            raise ValueError(f"No Verilog code found in LLM response for subproblem.")
    
    def _generate_code_ablation(self, document: str, task_description: str, task_name: str) -> str:
        """Generate complete Verilog module for ablation study mode - direct document processing"""
        
        # Use ablation study prompt
        system_prompt = ''#RTL_ABLATION_COMPLETE_PROMPT
        
        prompt = self.build_prompt(
            system_prompt + "\n\n" +
            "TASK: Generate a complete Verilog module implementation based on the provided specification.\n\n" +
            "--- COMPLETE SPECIFICATION DOCUMENT ---\n" +
            "Analyze this complete specification to implement all required functionality:\n" +
            "{document}\n\n" +
            "## **Output Format**\nProvide your analysis followed by the complete implementation:\n\n```verilog\n// Complete Verilog module implementation\n// Include all necessary submodules, logic, and connections\n// Ensure full specification compliance\n```",
            document=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 code_block:  # Non-empty code block
                verilog_blocks.append(code_block)
        
        if verilog_blocks:
            # Use the last Verilog block
            final_code = verilog_blocks[-1]
            return final_code
        else:
            raise ValueError(f"No Verilog code found in LLM response for ablation study.") 