#!/usr/bin/env python3
"""
Tool Synthesis Script - Automatically generates tool classes for different layers
"""

import json
import os
from typing import Any, Dict

class ToolGenerator:
    def __init__(self, depth_complexity=3, number_of_tasks=5, output_path=None, attribute_dict=None, root_path=None):
        """
        Initialize ToolGenerator with configurable parameters.
        
        Args:
            depth_complexity (int): Number of layers to generate tools for (default: 3)
            number_of_tasks (int): Number of finish_task tools to generate (default: 5)
            output_path (str): Path to save the generated all_tools.py file
            attribute_dict (list): List of dictionaries defining attribute types for each layer
            root_path (str): Root directory path for absolute path construction
        """
        self.depth_complexity = depth_complexity
        self.number_of_tasks = number_of_tasks
        
        if output_path is None:
            self.output_path = './Generated_data/Tools/all_tools.py'
        else:
            self.output_path = output_path
        
        # Set root path for absolute path construction
        if root_path is None:
            self.root_path = os.getcwd()
        else:
            self.root_path = root_path
            
        # Default attribute configuration if none provided
        if attribute_dict is None:
            self.attribute_dict = [
                {  # Layer 1
                    "1": "condition",
                    "2": "condition", 
                    "3": "reference_1",
                    "4": "reference_2",
                    "5": "lookup"
                },
                {  # Layer 2
                    "1": "condition",
                    "2": "condition", 
                    "3": "condition",
                    "4": "reference_2",
                    "5": "reference_3",
                    "6": "lookup"
                },
                {  # Layer 3
                    "1": "condition",
                    "2": "condition", 
                    "3": "condition", 
                    "4": "reference_3",
                    "5": "lookup"
                }
            ]
        else:
            self.attribute_dict = attribute_dict

    def get_lookup_attribute_index(self, layer_num):
        """Get the attribute index that corresponds to 'lookup' for the given layer."""
        if layer_num <= len(self.attribute_dict):
            layer_config = self.attribute_dict[layer_num - 1]
        else:
            # Use the last layer's configuration if we exceed the defined layers
            layer_config = self.attribute_dict[-1]
        
        # Find the attribute number that has type 'lookup'
        for attr_num, attr_type in layer_config.items():
            if attr_type == "lookup":
                return attr_num
        
        # Default to "5" if no lookup attribute found
        return "5"

    def get_profiles_path(self, layer_num):
        """Construct the absolute path to the profiles JSON file for a given layer."""
        # Extract the generated_data directory name from output_path
        output_dir = os.path.dirname(self.output_path)
        # Go up one level from Tools directory to get the Generated_data_layer_k_task_j directory
        generated_data_dir = os.path.dirname(output_dir)
        
        # Construct absolute path
        profiles_path = os.path.join(self.root_path, generated_data_dir, "Profiles", f"profiles_{layer_num}.json")
        return profiles_path

    def generate_get_tool_class(self, layer_num):
        """Generate a Get tool class for a specific layer number."""
        
        profiles_path = self.get_profiles_path(layer_num)
        
        template = f'''class Get_Profile_Layer_{layer_num}(Tool):
    @staticmethod
    def invoke(index_value: str) -> Dict[str, Any]:
        try:
            # Always read data from the specified JSON file using absolute path
            with open('{profiles_path}', 'r') as file:
                data = json.load(file)
            
            if index_value in data:
                result = data[index_value]
                return {{
                    "status": "Get_Profile_Layer_{layer_num} invocation is successful",
                    "result": result
                }}
            return {{
                "status": "Get_Profile_Layer_{layer_num} invocation failed - index_value not found",
                "result": None
            }}
        except Exception as e:
            return {{
                "status": f"Get_Profile_Layer_{layer_num} invocation failed - {{str(e)}}",
                "result": None
            }}

    @staticmethod
    def get_info() -> Dict[str, Any]:
        return {{
            "description": "This tool majorly helps you to locate a profile at layer {layer_num} with user specified index_value."
        }}'''
        
        return template

    def generate_search_tool_class(self, layer_num):
        """Generate a Search tool class for a specific layer number."""
        
        # Get the lookup attribute index for this layer
        lookup_attr_index = self.get_lookup_attribute_index(layer_num)
        lookup_attr_name = f"profile_{layer_num}_attribute_{lookup_attr_index}"
        profiles_path = self.get_profiles_path(layer_num)
        
        template = f'''class Search_Profile_Layer_{layer_num}(Tool):
    @staticmethod
    def invoke(key_value: str) -> Dict[str, Any]:
        try:
            # Always read data from the specified JSON file using absolute path
            with open('{profiles_path}', 'r') as file:
                data = json.load(file)
            
            # Search through all entries to find matches
            matching_results = []
            for k, profile in data.items():
                # Check if the profile has the lookup attribute and if it matches the key_value
                if isinstance(profile, dict) and profile.get('{lookup_attr_name}') == key_value:
                    matching_results.append(profile)
            
            return {{
                "status": "Search_Profile_Layer_{layer_num} invocation is successful",
                "result": matching_results
            }}
        except Exception as e:
            return {{
                "status": f"Search_Profile_Layer_{layer_num} invocation failed - {{str(e)}}",
                "result": []
            }}

    @staticmethod
    def get_info() -> Dict[str, Any]:
        return {{
            "description": "This tool searches through all profiles at layer {layer_num} to find entries where the lookup attribute ({lookup_attr_name}) matches the specified key_value, returning a list of all matching profiles."
        }}'''
        
        return template

    def generate_finish_task_class(self, task_num):
        """Generate a finish_task class for a specific task number."""
        
        template = f'''class finish_task_{task_num}(Tool):
    @staticmethod
    def invoke(attributes: list) -> Dict[str, Any]:
        try:
            result_string = f'finishing_task_{task_num} with the list of arguments {{attributes}}'
            return {{
                "status": "finish_task_{task_num} invocation is successful",
                "result": result_string
            }}
        except Exception as e:
            return {{
                "status": f"finish_task_{task_num} invocation failed - {{str(e)}}",
                "result": None
            }}

    @staticmethod
    def get_info() -> Dict[str, Any]:
        return {{
            "description": "This tool finishes task {task_num} with a provided list of attributes and returns a formatted string containing those attributes."
        }}'''
        
        return template

    def generate_conflict_tool_class(self):
        """Generate a conflict tool class."""
        
        template = '''class conflict_tool(Tool):
    @staticmethod
    def invoke() -> Dict[str, Any]:
        try:
            result_string = "raising conflict, consulting human"
            return {
                "status": "conflict_tool invocation is successful",
                "result": result_string
            }
        except Exception as e:
            return {
                "status": f"conflict_tool invocation failed - {str(e)}",
                "result": None
            }

    @staticmethod
    def get_info() -> Dict[str, Any]:
        return {
            "description": "This tool raises a conflict and returns a message indicating human consultation is needed. It requires no parameters."
        }'''
        
        return template

    def generate_tools_py_content(self):
        """Generate the content for Tools.py file."""
        
        content = '''import abc
from typing import Any


class Tool(abc.ABC):
    @staticmethod
    def invoke(*args, **kwargs):
        raise NotImplementedError

    @staticmethod
    def get_info() -> dict[str, Any]:
        raise NotImplementedError
'''
        return content

    def generate_all_tools(self):
        """Generate all_tools.py with tool classes for the specified depth complexity and number of tasks."""
        
        # File header with imports
        file_content = '''import json
from typing import Any, Dict
from Tools import Tool

'''
        
        # Generate classes for each layer up to depth_complexity
        for layer in range(1, self.depth_complexity + 1):
            file_content += self.generate_get_tool_class(layer) + '\n\n'
            file_content += self.generate_search_tool_class(layer) + '\n\n'
        
        # Generate finish_task classes for each task up to number_of_tasks
        for task in range(1, self.number_of_tasks + 1):
            file_content += self.generate_finish_task_class(task) + '\n\n'
        
        # Generate conflict tool
        file_content += self.generate_conflict_tool_class() + '\n\n'
        
        # Write to all_tools.py
        try:
            with open(self.output_path, 'w') as f:
                f.write(file_content)
            print(f"Successfully generated {self.output_path}")
            
            # Generate Tools.py in the same directory
            tools_dir = os.path.dirname(self.output_path)
            tools_py_path = os.path.join(tools_dir, "Tools.py")
            tools_py_content = self.generate_tools_py_content()
            
            with open(tools_py_path, 'w') as f:
                f.write(tools_py_content)
            print(f"Successfully generated {tools_py_path}")
            
            print(f"Generated Get and Search tool classes for layers 1 to {self.depth_complexity}")
            print(f"Generated finish_task tool classes for tasks 1 to {self.number_of_tasks}")
            print("Generated conflict_tool class")
            return file_content
        except Exception as e:
            print(f"Error writing to file: {e}")
            return None

if __name__ == "__main__":
    # Default configuration (can be customized)
    default_depth_complexity = 3
    default_number_of_tasks = 5

    default_attribute_dict = [
        {  # Layer 1
            "1": "condition",
            "2": "condition", 
            "3": "reference_1",
            "4": "reference_2",
            "5": "lookup"
        },
        {  # Layer 2
            "1": "condition",
            "2": "condition", 
            "3": "condition",
            "4": "reference_2",
            "5": "reference_3",
            "6": "lookup"
        },
        {  # Layer 3
            "1": "condition",
            "2": "condition", 
            "3": "condition", 
            "4": "reference_3",
            "5": "lookup"
        }
    ]
    
    # Example of custom configuration (uncomment to use)
    # custom_depth_complexity = 4
    # custom_number_of_tasks = 10
    # custom_output_path = "/path/to/custom/all_tools.py"
    # custom_attribute_dict = [
    #     {"1": "condition", "2": "lookup", "3": "reference_2"},
    #     {"1": "condition", "2": "condition", "3": "reference_1", "4": "lookup"}
    # ]
    
    # Generate tool files with configurable parameters
    generator = ToolGenerator(
        depth_complexity=default_depth_complexity,
        number_of_tasks=default_number_of_tasks,
        attribute_dict=default_attribute_dict
        # output_path=custom_output_path  # Uncomment to use custom path
    )
    
    generator.generate_all_tools()
    
    # Display summary
    print("\n=== Tool Files Generation Summary ===")
    print(f"Depth complexity: {generator.depth_complexity}")
    print(f"Number of tasks: {generator.number_of_tasks}")
    print(f"Output path: {generator.output_path}")
    print(f"Attribute configuration:")
    for i, layer_config in enumerate(generator.attribute_dict, 1):
        lookup_attr = generator.get_lookup_attribute_index(i)
        print(f"  Layer {i}: {layer_config} (lookup: attribute_{lookup_attr})")
