import json
import random
import string
from typing import Dict, List, Any, Tuple
import os

class ProfileGenerator:
    def __init__(self, profile_complexity_depth=3, profile_complexity_width=500, k1=1, k2=1, k3=1, attribute_dict=None):
        """
        Initialize ProfileGenerator with hierarchical profile parameters.
        
        Args:
            profile_complexity_depth (int): Number of depth levels (layers)
            profile_complexity_width (int): Number of profiles to generate at each depth level
            k1 (int): Reserved for future layer stepping functionality (currently unused)
            k2 (int): Step size for profile index references (default: 1 = consecutive profiles)
            k3 (int): Number of additional references beyond the first one (default: 1 = 2 total references)
            attribute_dict (list): List of dictionaries defining attribute types for each layer
        """
        self.profile_complexity_depth = profile_complexity_depth
        self.profile_complexity_width = profile_complexity_width
        self.k1 = k1
        self.k2 = k2
        self.k3 = k3
        
        # 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": "reference_2",
                    "4": "reference_3",
                    "5": "lookup"
                },
                {  # Layer 3
                    "1": "condition",
                    "2": "condition", 
                    "3": "condition", 
                    "4": "reference_3",
                    "5": "lookup"
                }
            ]
        else:
            self.attribute_dict = attribute_dict
    
    def generate_random_string(self, length=2):
        """Generate random lowercase string of specified length"""
        return ''.join(random.choice(string.ascii_lowercase) for _ in range(length))
    
    def generate_hierarchical_profiles(self):
        """
        Generate profiles based on depth and width parameters with dynamic attribute types.
        
        Returns:
            list: List of tuples, one for each layer. Each tuple contains:
                - profiles_dict: Dictionary containing profiles for that layer
                - layer_i_keys: List of profile keys for that layer
                - attribute_lookup_values: Dictionary of lookup values organized by attribute name for that layer
            
        Attribute Types (based on attribute_dict):
            - condition: Random numbers 1-100
            - lookup: Random generated strings (2-character lowercase)
            - reference_i: List of primary keys accessing profiles at layer i
        """
        # Initialize list to store data for each layer
        layers_data = []
        
        for j in range(1, self.profile_complexity_depth + 1):  # depth levels
            # Initialize data structures for this layer
            layer_profiles = {}
            layer_keys = []
            layer_lookup_tables = {}  # Dictionary to store lookup values by attribute name
            
            # Get attribute configuration for this layer
            if j <= len(self.attribute_dict):
                layer_attr_config = self.attribute_dict[j - 1]
            else:
                # Use the last layer's configuration if we exceed the defined layers
                layer_attr_config = self.attribute_dict[-1]
            
            for i in range(1, self.profile_complexity_width + 1):  # profiles within each depth
                profile_key = f"profile_{j}_{i}"
                
                # Collect this layer's profile keys
                layer_keys.append(profile_key)
                
                # Generate attributes based on configuration
                profile_attributes = {}
                
                for attr_num, attr_type in layer_attr_config.items():
                    attr_name = f"profile_{j}_attribute_{attr_num}"
                    
                    if attr_type == "condition":
                        # Random numbers from 1-100
                        profile_attributes[attr_name] = random.randint(1, 100)
                    
                    elif attr_type == "lookup":
                        # Random 2-character lowercase string
                        lookup_value = self.generate_random_string(2)
                        profile_attributes[attr_name] = lookup_value
                        
                        # Collect ALL lookup values organized by attribute name
                        if attr_name not in layer_lookup_tables:
                            layer_lookup_tables[attr_name] = []
                        layer_lookup_tables[attr_name].append(lookup_value)
                    
                    elif attr_type.startswith("reference_"):
                        # Extract target layer number from reference_i
                        target_layer = int(attr_type.split("_")[1])
                        reference_list = []
                        
                        # Generate list of primary keys to target layer
                        # Start from next profile (i+1) and use k2 for stepping in profile index
                        for step in range(self.k3 + 1):  # k3+1 total keys
                            if target_layer <= self.profile_complexity_depth:
                                # Reference to specified layer, starting from next profile with k2 stepping
                                profile_index = i + 1 + (step * self.k2)  # Use k2 for profile index stepping
                                if profile_index <= self.profile_complexity_width:
                                    reference_list.append(f"profile_{target_layer}_{profile_index}")
                                else:
                                    # Wrap around to beginning profiles in target layer
                                    wrapped_index = ((profile_index - 1) % self.profile_complexity_width) + 1
                                    reference_list.append(f"profile_{target_layer}_{wrapped_index}")
                            else:
                                # If target layer doesn't exist, reference layer 1, starting from next profile with k2 stepping
                                profile_index = i + 1 + (step * self.k2)  # Use k2 for profile index stepping
                                if profile_index <= self.profile_complexity_width:
                                    reference_list.append(f"profile_1_{profile_index}")
                                else:
                                    # Wrap around to beginning profiles in layer 1
                                    wrapped_index = ((profile_index - 1) % self.profile_complexity_width) + 1
                                    reference_list.append(f"profile_1_{wrapped_index}")
                        
                        profile_attributes[attr_name] = reference_list
                    
                    else:
                        # Default case - treat as condition
                        profile_attributes[attr_name] = random.randint(1, 100)
                
                layer_profiles[profile_key] = profile_attributes
            
            # Add this layer's data as a tuple to the result list
            layers_data.append((layer_profiles, layer_keys, layer_lookup_tables))
        
        return layers_data
    
    def save_profile_files(self, output_dir: str = None):
        """Generate and save separate profile files for each layer"""
        # Use default relative path if none provided
        if output_dir is None:
            output_dir = "./Generated_data/Profiles"
            
        # Create output directory if it doesn't exist
        os.makedirs(output_dir, exist_ok=True)
        
        layers_data = self.generate_hierarchical_profiles()
        
        # Save each layer's profiles to separate files
        for layer_idx, (layer_profiles, layer_keys, layer_lookup_tables) in enumerate(layers_data, 1):
            layer_path = os.path.join(output_dir, f"profiles_{layer_idx}.json")
            with open(layer_path, 'w', encoding='utf-8') as f:
                json.dump(layer_profiles, f, indent=2, ensure_ascii=False)
            
            print(f"Layer {layer_idx}: {len(layer_profiles)} profiles saved to {layer_path}")
        
        print(f"Total profiles generated: {sum(len(layer_data[0]) for layer_data in layers_data)}")
        
        return layers_data

if __name__ == "__main__":
    # Default attribute configuration (can be customized)
    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 a custom attribute configuration (uncomment to use)
    # custom_attribute_dict = [
    #     {  # Layer 1: Different configuration
    #         "1": "condition",
    #         "2": "lookup", 
    #         "3": "reference_2",
    #         "4": "reference_3"
    #     },
    #     {  # Layer 2
    #         "1": "condition",
    #         "2": "condition", 
    #         "3": "reference_1",
    #         "4": "lookup"
    #     }
    # ]
    
    # Generate profile files with configurable parameters
    generator = ProfileGenerator(
        profile_complexity_depth=3, 
        profile_complexity_width=500, 
        k1=1, 
        k2=1, 
        k3=1,
        attribute_dict=default_attribute_dict  # Change to custom_attribute_dict to use custom config
    )
    
    # Use default relative path (can be overridden by passing a custom path)
    layers_data = generator.save_profile_files()
    
    # Display summary
    print("\n=== Profile Files Generation Summary ===")
    print(f"Layers: {generator.profile_complexity_depth}")
    print(f"Profiles per layer: {generator.profile_complexity_width}")
    print(f"Parameters: k1={generator.k1}, k2={generator.k2}, k3={generator.k3}")
    print(f"Attribute configuration:")
    for i, layer_config in enumerate(generator.attribute_dict, 1):
        print(f"  Layer {i}: {layer_config}")
    
    # Display lookup tables for each layer
    print("\n=== Lookup Tables for Each Layer ===")
    for layer_idx, (layer_profiles, layer_keys, layer_lookup_tables) in enumerate(layers_data, 1):
        print(f"\nLayer {layer_idx} Lookup Tables:")
        if layer_lookup_tables:
            for attr_name, lookup_values in layer_lookup_tables.items():
                unique_values = list(set(lookup_values))  # Get unique values
                print(f"  {attr_name}: {len(unique_values)} unique values")
                print(f"    Values: {sorted(unique_values)}")
                print(f"    Total occurrences: {len(lookup_values)}")
        else:
            print("  No lookup attributes in this layer") 