import json
import random
import string
import re
import sys
import os
from typing import Dict, List, Any

# Dynamic path setup - will be configured when UserRequestGenerator is initialized
# These will be set by the _setup_dynamic_paths method

try:
    from all_tools import (
        Get_Profile_Layer_1, Search_Profile_Layer_1,
        Get_Profile_Layer_2, Search_Profile_Layer_2,
        Get_Profile_Layer_3, Search_Profile_Layer_3
    )
except ImportError:
    # Handle case where tools are not available
    print("Warning: Tools not available for query execution")

try:
    from exec import compute_all_tasks
except ImportError:
    # Handle case where exec.py is not available
    print("Warning: exec.py not available for ground truth computation")
    compute_all_tasks = None

class UserRequestGenerator:
    def __init__(self, attribute_dict: List[Dict[str, str]], layer_lookup_tables: Dict[int, List[str]] = None, 
                 task_layer_requirements: Dict[int, List[int]] = None, global_attributes: Dict[str, Any] = None,
                 base_output_dir: str = None):
        self.layers = len(attribute_dict)  # Dynamically determine number of layers
        self.max_profile_index = 500  # Updated to 1-500 range
        self.attribute_dict = attribute_dict
        self.global_attributes_count = 3  # Number of global attributes (1-3)
        
        # Store base output directory for dynamic path resolution
        self.base_output_dir = base_output_dir
        
        # Setup dynamic paths for tools and tasks
        self._setup_dynamic_paths()
        
        # Store custom global attributes or use defaults
        self.global_attributes = global_attributes or {
            "global_attribute_1": 42,
            "global_attribute_2": 37, 
            "global_attribute_3": 55
        }
        
        # Store lookup tables for each layer (dynamically initialize based on layer count)
        if layer_lookup_tables is None:
            self.layer_lookup_tables = {i: [] for i in range(1, self.layers + 1)}
        else:
            self.layer_lookup_tables = layer_lookup_tables
        
        # Define which layers each task type requires (now customizable)
        if task_layer_requirements is None:
            # Default task requirements based on available layers
            all_layers = list(range(1, self.layers + 1))
            self.task_layer_requirements = {
                1: all_layers,  # Task_Type_1: all layers
                2: [1],         # Task_Type_2: layer 1 only
                3: all_layers,  # Task_Type_3: all layers
                4: all_layers,  # Task_Type_4: all layers
                5: all_layers[:2] if len(all_layers) >= 2 else [1]  # Task_Type_5: first 2 layers
            }
        else:
            self.task_layer_requirements = task_layer_requirements
        
        # Set task_types dynamically based on task_layer_requirements
        self.task_types = max(self.task_layer_requirements.keys()) if self.task_layer_requirements else 5
    
    def _setup_dynamic_paths(self):
        """Setup dynamic paths for tools and tasks based on base_output_dir"""
        if self.base_output_dir:
            # Add the specific Tools and Task directories to the path
            tools_path = os.path.join(os.path.dirname(__file__), self.base_output_dir, 'Tools')
            task_path = os.path.join(os.path.dirname(__file__), self.base_output_dir, 'Task')
        else:
            # Fallback to default paths
            tools_path = os.path.join(os.path.dirname(__file__), 'Generated_data', 'Tools')
            task_path = os.path.join(os.path.dirname(__file__), 'Generated_data', 'Task')
        
        # Add paths to sys.path if they exist and aren't already added
        if os.path.exists(tools_path) and tools_path not in sys.path:
            sys.path.insert(0, tools_path)
        if os.path.exists(task_path) and task_path not in sys.path:
            sys.path.insert(0, task_path)
        
        # Store paths for later use
        self.tools_path = tools_path
        self.task_path = task_path
        
    def _get_lookup_string(self, layer: int) -> str:
        """Get a random lookup string from the appropriate layer's lookup table"""
        if layer in self.layer_lookup_tables and self.layer_lookup_tables[layer]:
            return random.choice(self.layer_lookup_tables[layer])
        else:
            # Fallback to random generation if no lookup table is provided
            return ''.join(random.choices(string.ascii_lowercase, k=2))
    
    def _generate_random_string(self) -> str:
        """Generate a random two-character lowercase string (kept for backwards compatibility)"""
        return ''.join(random.choices(string.ascii_lowercase, k=2))
    
    def _generate_condition(self, layer: int, comparison_type: str = "any") -> str:
        """Generate safe selection conditions that always guarantee one instance"""
        layer_attrs = list(self.attribute_dict[layer-1].keys())
        
        # Only use safe condition types that always guarantee a selection
        condition_types = [
            "extremum_based",
            "safe_mathematical_extremum",
            "safe_global_extremum"
        ]
        
        condition_type = random.choice(condition_types)
        
        if condition_type == "extremum_based":
            attr = random.choice(layer_attrs)
            extremum = random.choice(["smallest", "largest"])
            return f"attribute_{attr} has the {extremum} value"
            
        elif condition_type == "safe_mathematical_extremum":
            attrs = random.sample(layer_attrs, min(2, len(layer_attrs)))
            extremum = random.choice(["smallest", "largest"])
            attr_names = [f"attribute_{attr}" for attr in attrs]
            operations = [
                f"sum of {' and '.join(attr_names)} has the {extremum} value",
                f"average of {' and '.join(attr_names)} has the {extremum} value",
                f"product of {' and '.join(attr_names)} has the {extremum} value",
                f"result of attribute_{attrs[0]} minus attribute_{attrs[1] if len(attrs) > 1 else attrs[0]} has the {extremum} value"
            ]
            return random.choice(operations)
            
        elif condition_type == "safe_global_extremum":
            attr = random.choice(layer_attrs)
            global_attr = random.randint(1, self.global_attributes_count)
            extremum = random.choice(["smallest", "largest"])
            operations = [
                f"sum of attribute_{attr} and global_attribute_{global_attr} has the {extremum} value",
                f"result of attribute_{attr} minus global_attribute_{global_attr} has the {extremum} value",
                f"product of attribute_{attr} and global_attribute_{global_attr} has the {extremum} value"
            ]
            return random.choice(operations)
        
        # Fallback to simple extremum
        attr = random.choice(layer_attrs)
        extremum = random.choice(["smallest", "largest"])
        return f"attribute_{attr} has the {extremum} value"
    
    def _generate_selection_method(self, layer: int) -> str:
        """Generate safe direct selection for profiles that always guarantees one instance"""
        condition_attrs = [attr for attr, attr_type in self.attribute_dict[layer-1].items() 
                          if attr_type == 'condition']
        
        if not condition_attrs:
            # Fallback if no condition attributes available
            all_attrs = list(self.attribute_dict[layer-1].keys())
            attr = random.choice(all_attrs)
            extremum = random.choice(["largest", "smallest"])
            return f"with the {extremum} attribute_{attr}"
        
        # Generate safe selection methods that always work
        selection_methods = []
        
        # Single attribute extremum
        attr = random.choice(condition_attrs)
        extremum = random.choice(["largest", "smallest"])
        selection_methods.append(f"with the {extremum} attribute_{attr}")
        
        # Sum of multiple attributes extremum
        if len(condition_attrs) >= 2:
            attrs = random.sample(condition_attrs, min(2, len(condition_attrs)))
            extremum = random.choice(["largest", "smallest"])
            attr_names = [f"attribute_{attr}" for attr in attrs]
            selection_methods.append(f"with the {extremum} sum of {' and '.join(attr_names)}")
        
        # Product of attributes (if multiple available)
        if len(condition_attrs) >= 2:
            attrs = random.sample(condition_attrs, 2)
            extremum = random.choice(["largest", "smallest"])
            attr_names = [f"attribute_{attr}" for attr in attrs]
            selection_methods.append(f"with the {extremum} product of {' and '.join(attr_names)}")
        
        # Global attribute comparison for selection
        attr = random.choice(condition_attrs)
        global_attr = random.randint(1, self.global_attributes_count)
        extremum = random.choice(["largest", "smallest"])
        selection_methods.append(f"with the {extremum} result of attribute_{attr} minus global_attribute_{global_attr}")
        
        # Average of attributes
        if len(condition_attrs) >= 2:
            attrs = random.sample(condition_attrs, min(2, len(condition_attrs)))
            extremum = random.choice(["largest", "smallest"])
            attr_names = [f"attribute_{attr}" for attr in attrs]
            selection_methods.append(f"with the {extremum} average of {' and '.join(attr_names)}")
        
        selection = random.choice(selection_methods)
        
        return f"select the profile {selection}"
    
    def _get_valid_source_layers_for_adjacent_access(self, target_layer: int, processed_layers: List[int]) -> List[int]:
        """
        Determine which processed layers can serve as valid sources for adjacent access to the target layer.
        
        Args:
            target_layer: The layer we want to access
            processed_layers: List of layers that have been processed so far
            
        Returns:
            List of layer numbers that can serve as valid sources for adjacent access
        """
        valid_sources = []
        
        for source_layer in processed_layers:
            # Check if source_layer has a reference attribute that can actually point to target_layer
            source_attributes = self.attribute_dict[source_layer - 1]  # Convert to 0-indexed
            
            # Only check for reference types that actually exist in the attribute_dict
            has_valid_reference = False
            
            for attr_num, attr_type in source_attributes.items():
                # Check if this attribute type is a reference and can reach the target layer
                if attr_type.startswith("reference_"):
                    reference_number = attr_type.split("_")[1]
                    
                    # Determine what this reference can point to based on common patterns
                    if attr_type == "reference_1" and source_layer == target_layer:
                        # Same layer reference (layer 1 -> layer 1)
                        has_valid_reference = True
                        break
                    elif attr_type == "reference_2" and target_layer == 2:
                        # Can reference layer 2
                        has_valid_reference = True
                        break
                    elif attr_type == "reference_3" and target_layer == 3:
                        # Can reference layer 3
                        has_valid_reference = True
                        break
                    elif attr_type == "reference_4" and target_layer == 4:
                        # Can reference layer 4
                        has_valid_reference = True
                        break
                    elif attr_type == "reference_5" and target_layer == 5:
                        # Can reference layer 5
                        has_valid_reference = True
                        break
                    # Add more reference patterns as needed for higher layers
                    elif attr_type == f"reference_{target_layer}":
                        # Generic pattern for any layer number
                        has_valid_reference = True
                        break
            
            if has_valid_reference:
                valid_sources.append(source_layer)
        
        return valid_sources
    
    def _generate_layer_1_access(self) -> str:
        """Generate layer 1 access specification"""
        access_methods = []
        
        # Direct access (no disambiguation needed - accessing by specific index)
        access_methods.append(f"my profile_1_id is {random.randint(1, self.max_profile_index)}")
        
        # String lookup with safe selection (guaranteed to work)
        condition_attrs = [attr for attr, attr_type in self.attribute_dict[0].items() 
                          if attr_type == 'condition']
        
        if condition_attrs:
            attr = random.choice(condition_attrs)
            extremum = random.choice(["largest", "smallest"])
            access_methods.append(f"my profile_1_info is '{self._get_lookup_string(1)}'; then select the instance with the {extremum} attribute_{attr}")
        else:
            # Fallback if no condition attributes
            access_methods.append(f"my profile_1_info is '{self._get_lookup_string(1)}'")
        
        # Adjacent profile access with safe selection
        base_profile_index = random.randint(1, self.max_profile_index)
        base_lookup_string = self._get_lookup_string(1)
        
        # Choose between index-based or lookup-based access for the base profile
        if random.choice([True, False]):
            # Index-based access to current profile
            selection_method = self._generate_selection_method(1)
            access_methods.append(f"my profile_1_id is {base_profile_index}, do the task for my relatives {selection_method}")
        else:
            # Lookup-based access to current profile
            selection_method = self._generate_selection_method(1)
            access_methods.append(f"my profile_1_info is '{base_lookup_string}'; do the task for my relatives {selection_method}")
        
        return random.choice(access_methods)
    
    def _generate_higher_layer_access(self, layer: int) -> str:
        """Generate higher layer access specification for layers 2 and above"""
        access_methods = []
        
        # Validate layer number
        if layer < 2 or layer > len(self.attribute_dict):
            raise ValueError(f"Invalid layer {layer}. Must be between 2 and {len(self.attribute_dict)}")
        
        # String lookup with safe selection
        condition_attrs = [attr for attr, attr_type in self.attribute_dict[layer-1].items() 
                          if attr_type == 'condition']
        
        if condition_attrs:
            attr = random.choice(condition_attrs)
            extremum = random.choice(["largest", "smallest"])
            access_methods.append(f"The target profile_{layer}_info is '{self._get_lookup_string(layer)}', select the instance with the {extremum} attribute_{attr}")
        else:
            # Fallback if no condition attributes
            access_methods.append(f"The target profile_{layer}_info is '{self._get_lookup_string(layer)}'")
        
        # Safe selection from adjacent profiles
        access_methods.append(f"get my profile_{layer} by {self._generate_selection_method(layer)}")
        
        return random.choice(access_methods)
    
    def generate_user_request(self, task_type: int) -> Dict[str, Any]:
        """Generate a comprehensive user request with structured execution plan"""
        required_layers = self.task_layer_requirements[task_type]
        access_specifications = []
        execution_plan = []  # Structured execution instructions
        processed_layers = []  # Track which layers have been processed
        
        # Always start with layer 1 access
        layer_1_access, layer_1_instructions = self._generate_layer_1_access_with_plan()
        access_specifications.append(layer_1_access)
        execution_plan.extend(layer_1_instructions)
        processed_layers.append(1)
        
        # Process additional layers dynamically
        sorted_layers = sorted([layer for layer in required_layers if layer > 1])
        for layer in sorted_layers:
            layer_access, layer_instructions = self._generate_higher_layer_access_with_plan(layer, processed_layers)
            access_specifications.append(f"then {layer_access}")
            execution_plan.extend(layer_instructions)
            processed_layers.append(layer)
        
        # Combine all access specifications
        full_access_spec = ", ".join(access_specifications)
        user_request = f"{full_access_spec}. I want to perform task_type_{task_type}."
        
        return {
            "user_request": user_request,
            "execution_plan": execution_plan,
            "task_type": task_type
        }
    
    def _generate_layer_1_access_with_plan(self) -> tuple:
        """Generate layer 1 access specification with execution plan"""
        access_methods = []
        execution_plans = []
        
        # Direct access (no disambiguation needed - accessing by specific index)
        index = random.randint(1, self.max_profile_index)
        direct_access = f"my profile_1_id is {index}"
        direct_plan = [
            {
                "action": "get_profile",
                "layer": 1,
                "method": "index",
                "value": f"profile_1_{index}"
            }
        ]
        access_methods.append((direct_access, direct_plan))
        
        # String lookup with safe selection
        condition_attrs = [attr for attr, attr_type in self.attribute_dict[0].items() 
                          if attr_type == 'condition']
        
        if condition_attrs:
            attr = random.choice(condition_attrs)
            extremum = random.choice(["largest", "smallest"])
            lookup_string = self._get_lookup_string(1)
            
            string_access = f"my profile_1_info is '{lookup_string}'; then select the instance with the {extremum} attribute_{attr}"
            string_plan = [
                {
                    "action": "search_profile",
                    "layer": 1,
                    "lookup_string": lookup_string
                },
                {
                    "action": "select_profile",
                    "criteria": f"{extremum} attribute_{attr}",
                    "from": "search_results"
                }
            ]
            access_methods.append((string_access, string_plan))
        
        # Adjacent profile access with safe selection
        base_profile_index = random.randint(1, self.max_profile_index)
        base_lookup_string = self._get_lookup_string(1)
        selection_method = self._generate_selection_method(1)
        
        # Index-based access to current profile
        adjacent_access = f"my profile_1_id is {base_profile_index}, do the task for my relatives {selection_method}"
        adjacent_plan = [
            {
                "action": "get_profile",
                "layer": 1,
                "method": "index", 
                "value": f"profile_1_{base_profile_index}"
            },
            {
                "action": "access_adjacent",
                "layer": 1,
                "from_layer": 1
            },
            {
                "action": "select_profile",
                "criteria": selection_method.replace("select the profile ", ""),
                "from": "adjacent_results"
            }
        ]
        access_methods.append((adjacent_access, adjacent_plan))
        
        chosen_method, chosen_plan = random.choice(access_methods)
        return chosen_method, chosen_plan
    
    def _generate_higher_layer_access_with_plan(self, layer: int, processed_layers: List[int]) -> tuple:
        """Generate access specification with execution plan for layers 2 and above"""
        access_methods = []
        
        # Validate layer number
        if layer < 2 or layer > len(self.attribute_dict):
            raise ValueError(f"Invalid layer {layer}. Must be between 2 and {len(self.attribute_dict)}")
        
        # String lookup with safe selection - always available
        condition_attrs = [attr for attr, attr_type in self.attribute_dict[layer-1].items() 
                          if attr_type == 'condition']
        
        if condition_attrs:
            attr = random.choice(condition_attrs)
            extremum = random.choice(["largest", "smallest"])
            lookup_string = self._get_lookup_string(layer)
            
            string_access = f"The target profile_{layer}_info is '{lookup_string}', select the instance with the {extremum} attribute_{attr}"
            string_plan = [
                {
                    "action": "search_profile",
                    "layer": layer,
                    "lookup_string": lookup_string
                },
                {
                    "action": "select_profile", 
                    "criteria": f"{extremum} attribute_{attr}",
                    "from": "search_results"
                }
            ]
            access_methods.append((string_access, string_plan))
        
        # Adjacent access - only available if there are valid source layers
        valid_source_layers = self._get_valid_source_layers_for_adjacent_access(layer, processed_layers)
        
        if valid_source_layers:
            # Randomly choose one of the valid source layers
            chosen_source_layer = random.choice(valid_source_layers)
            selection_method = self._generate_selection_method(layer)
            adjacent_access = f"get my profile_{layer} from the obtained profile_{chosen_source_layer} by {selection_method}"
            adjacent_plan = [
                {
                    "action": "access_adjacent",
                    "layer": layer,
                    "from_layer": chosen_source_layer  # Access from a valid source layer
                },
                {
                    "action": "select_profile",
                    "criteria": selection_method.replace("select the profile ", ""),
                    "from": "adjacent_results"
                }
            ]
            access_methods.append((adjacent_access, adjacent_plan))
        
        # If no valid source layers exist, only use string lookup
        if not access_methods:
            # Fallback to simple lookup without selection if no condition attributes
            lookup_string = self._get_lookup_string(layer)
            fallback_access = f"The target profile_{layer}_info is '{lookup_string}'"
            fallback_plan = [
                {
                    "action": "search_profile",
                    "layer": layer,
                    "lookup_string": lookup_string
                }
            ]
            access_methods.append((fallback_access, fallback_plan))
        
        chosen_method, chosen_plan = random.choice(access_methods)
        return chosen_method, chosen_plan
    
    def generate_multiple_requests(self, num_requests: int) -> List[Dict[str, Any]]:
        """Generate multiple user requests with execution plans"""
        requests = []
        for _ in range(num_requests):
            task_type = random.randint(1, self.task_types)
            request_data = self.generate_user_request(task_type)
            requests.append(request_data)
        return requests
    
    def _transform_final_profiles_for_exec(self, final_profiles: Dict[int, Dict]) -> Dict[int, Dict]:
        """
        Transform final_profiles format from qa_generator to exec.py expected format.
        
        qa_generator format: {layer: {"profile_layer_attribute_x": value, ...}}
        exec.py format: {layer: {"Profile_layer_Attribute_x": value, ...}}
        """
        transformed = {}
        for layer, profile_data in final_profiles.items():
            if profile_data:
                transformed_profile = {}
                for key, value in profile_data.items():
                    # Transform key format: profile_1_attribute_2 -> Profile_1_Attribute_2
                    if key.startswith('profile_'):
                        parts = key.split('_')
                        if len(parts) >= 4:
                            # Capitalize 'Profile' and 'Attribute'
                            new_key = f"Profile_{parts[1]}_Attribute_{parts[3]}"
                            transformed_profile[new_key] = value
                        else:
                            transformed_profile[key] = value
                    else:
                        transformed_profile[key] = value
                transformed[layer] = transformed_profile
        return transformed
    
    def _extract_task_type_from_request(self, user_request: str) -> int:
        """Extract task type from user request string"""
        import re
        match = re.search(r'task_type_(\d+)', user_request)
        if match:
            return int(match.group(1))
        return 1  # Default fallback
    
    def _compute_ground_truth_action(self, final_profiles: Dict[int, Dict], task_type: int) -> Any:
        """Compute ground truth action using exec.py"""
        try:
            # Dynamic import of compute_all_tasks from the correct path
            compute_all_tasks_func = self._import_compute_all_tasks()
            if compute_all_tasks_func is None:
                return None
            
            # Transform final_profiles to exec.py format
            transformed_profiles = self._transform_final_profiles_for_exec(final_profiles)
            
            # Compute all tasks
            #import ipdb; ipdb.set_trace()
            all_results = compute_all_tasks_func(transformed_profiles)
            
            # Get the result for the specific task type
            task_arguments = all_results.get(task_type, None)
            
            if task_arguments is not None:
                # Format as finish_task_k([arguments])
                return f"finish_task_{task_type}({task_arguments})"
            else:
                return None
        except Exception as e:
            print(f"Error computing ground truth action: {e}")
            return None
    
    def _import_compute_all_tasks(self):
        """Dynamically import compute_all_tasks from the correct exec.py"""
        try:
            # Try to import from the dynamic path first
            if hasattr(self, 'task_path') and os.path.exists(os.path.join(self.task_path, 'exec.py')):
                import importlib.util
                spec = importlib.util.spec_from_file_location("exec", os.path.join(self.task_path, 'exec.py'))
                exec_module = importlib.util.module_from_spec(spec)
                spec.loader.exec_module(exec_module)
                return getattr(exec_module, 'compute_all_tasks', None)
            else:
                # Fallback to global import
                try:
                    from exec import compute_all_tasks
                    return compute_all_tasks
                except ImportError:
                    return None
        except Exception as e:
            print(f"Warning: Could not import compute_all_tasks: {e}")
            return None
    
    def save_requests_to_json(self, output_path: str, num_requests: int = 50, include_rollouts: bool = True, profiles_path: str = "Generated_data/Profiles/"):
        """Generate and save user requests to JSON file with optional query execution"""
        request_data_list = self.generate_multiple_requests(num_requests)
        
        # Initialize query executor if rollouts are requested
        executor = None
        if include_rollouts:
            try:
                executor = QueryExecutor(profiles_path=profiles_path, global_attributes=self.global_attributes, attribute_dict=self.attribute_dict, base_output_dir=self.base_output_dir)
                print("Query executor initialized successfully")
            except Exception as e:
                print(f"Warning: Could not initialize query executor: {e}")
                include_rollouts = False
        
        # Process requests and add rollouts if available
        processed_requests = []
        for i, request_data in enumerate(request_data_list):
            output_entry = {
                "user_request": request_data["user_request"],
                "execution_plan": request_data["execution_plan"]
            }
            
            if include_rollouts and executor:
                try:
                    execution_result = executor.execute_structured_query(request_data["execution_plan"])
                    output_entry["rollout"] = execution_result["rollout"]
                    output_entry["final_profiles"] = execution_result["final_profiles"]
                    
                    # Compute ground truth action using exec.py
                    task_type = request_data["task_type"]
                    ground_truth = self._compute_ground_truth_action(execution_result["final_profiles"], task_type)
                    output_entry["ground_truth_action"] = ground_truth
                    
                    print(f"Processed request {i+1}/{len(request_data_list)} with rollout and ground truth")
                except Exception as e:
                    output_entry["rollout"] = [
                        {
                            "step": 1,
                            "action": "Query execution",
                            "result": None,
                            "error": f"Execution failed: {str(e)}"
                        }
                    ]
                    output_entry["final_profiles"] = {}
                    output_entry["ground_truth_action"] = None
                    print(f"Failed to execute request {i+1}: {e}")
            else:
                print(f"Processed request {i+1}/{len(request_data_list)} without rollout")
            
            processed_requests.append(output_entry)
        
        # Simple output format - just the requests array
        output_data = processed_requests
        
        with open(output_path, 'w', encoding='utf-8') as f:
            json.dump(output_data, f, indent=2, ensure_ascii=False)
        
        print(f"Generated {len(request_data_list)} user requests and saved to: {output_path}")
        if include_rollouts:
            successful_rollouts = len([r for r in processed_requests if "rollout" in r and not any("error" in step for step in r.get("rollout", []))])
            print(f"Successfully generated {successful_rollouts}/{len(request_data_list)} rollouts")
        return output_data

class QueryExecutor:
    def __init__(self, profiles_path="Generated_data/Profiles/", global_attributes=None, attribute_dict=None, base_output_dir=None):
        self.profiles_path = profiles_path
        self.base_output_dir = base_output_dir
        self.global_attributes = global_attributes or {
            "global_attribute_1": 42,
            "global_attribute_2": 37, 
            "global_attribute_3": 55
        }
        
        # Extract layer count from base_output_dir and setup dynamic paths
        self.layers = self._extract_layer_count()
        self._setup_dynamic_paths()
        self._import_tools()
        
        # Tool mapping - Dynamically create based on actual layer count
        self.tools = {}
        for layer in range(1, self.layers + 1):
            self.tools[f"get_layer_{layer}"] = self.imported_tools[f"Get_Profile_Layer_{layer}"]
            self.tools[f"search_layer_{layer}"] = self.imported_tools[f"Search_Profile_Layer_{layer}"]
        
        # Store attribute_dict for dynamic reference resolution
        # Use provided attribute_dict or default for backwards compatibility
        if attribute_dict is None:
            # Default 3-layer structure for backwards compatibility
            self.attribute_dict = [
                {  # Layer 1
                    "1": "condition",
                    "2": "condition", 
                    "3": "reference_1",
                    "4": "reference_2",
                    "5": "condition",
                    "6": "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 _extract_layer_count(self):
        """Extract the layer count from base_output_dir"""
        if not self.base_output_dir:
            # Default to 3 layers if no base_output_dir specified
            return 3
        
        # Parse the base_output_dir to find layer count
        # Expected format: Generated_data_layer_X_task_Y where X is the layer count
        import re
        match = re.search(r'layer_(\d+)', self.base_output_dir)
        if match:
            return int(match.group(1))
        else:
            # Fallback to 3 if pattern not found
            return 3
    
    def _setup_dynamic_paths(self):
        """Setup dynamic paths for tools based on base_output_dir"""
        if self.base_output_dir:
            # Add the specific Tools directory to the path
            tools_path = os.path.join(os.path.dirname(__file__), self.base_output_dir, 'Tools')
        else:
            # Fallback to default paths
            tools_path = os.path.join(os.path.dirname(__file__), 'Generated_data', 'Tools')
        
        # Add paths to sys.path if they exist and aren't already added
        if os.path.exists(tools_path) and tools_path not in sys.path:
            sys.path.insert(0, tools_path)
        
        # Store paths for later use
        self.tools_path = tools_path
    
    def _import_tools(self):
        """Import the proper tools from all_tools.py dynamically based on layer count"""
        self.imported_tools = {}
        try:
            # Dynamically import tools based on the actual layer count
            import importlib
            all_tools_module = importlib.import_module('all_tools')
            
            # Import tools for each layer
            for layer in range(1, self.layers + 1):
                get_tool_name = f"Get_Profile_Layer_{layer}"
                search_tool_name = f"Search_Profile_Layer_{layer}"
                
                # Get the tool classes from the module
                if hasattr(all_tools_module, get_tool_name):
                    self.imported_tools[get_tool_name] = getattr(all_tools_module, get_tool_name)
                else:
                    print(f"Warning: {get_tool_name} not found in all_tools.py")
                    self.imported_tools[get_tool_name] = self._create_get_tool(layer)
                    
                if hasattr(all_tools_module, search_tool_name):
                    self.imported_tools[search_tool_name] = getattr(all_tools_module, search_tool_name)
                else:
                    print(f"Warning: {search_tool_name} not found in all_tools.py")
                    self.imported_tools[search_tool_name] = self._create_search_tool(layer)
                    
        except ImportError as e:
            print(f"Warning: Could not import tools from all_tools.py: {e}")
            # Fallback to custom tools if import fails
            self.imported_tools = {}
            for layer in range(1, self.layers + 1):
                self.imported_tools[f"Get_Profile_Layer_{layer}"] = self._create_get_tool(layer)
                self.imported_tools[f"Search_Profile_Layer_{layer}"] = self._create_search_tool(layer)
    
    def _create_get_tool(self, layer):
        """Create a custom get tool that uses the correct path"""
        class CustomGetTool:
            def __init__(self, layer_num, profiles_path):
                self.layer = layer_num
                self.profiles_path = profiles_path
            
            def invoke(self, index_value):
                try:
                    import json
                    profile_file = f"{self.profiles_path}profiles_{self.layer}.json"
                    with open(profile_file, 'r') as file:
                        data = json.load(file)
                    
                    if index_value in data:
                        result = data[index_value]
                        return {
                            "status": f"Get_Profile_Layer_{self.layer} invocation is successful",
                            "result": result
                        }
                    return {
                        "status": f"Get_Profile_Layer_{self.layer} invocation failed - index_value not found",
                        "result": None
                    }
                except Exception as e:
                    return {
                        "status": f"Get_Profile_Layer_{self.layer} invocation failed - {str(e)}",
                        "result": None
                    }
        
        return CustomGetTool(layer, self.profiles_path)
    
    def _create_search_tool(self, layer):
        """Create a custom search tool that uses the correct path"""
        class CustomSearchTool:
            def __init__(self, layer_num, profiles_path):
                self.layer = layer_num
                self.profiles_path = profiles_path
            
            def invoke(self, key_value):
                try:
                    import json
                    profile_file = f"{self.profiles_path}profiles_{self.layer}.json"
                    with open(profile_file, 'r') as file:
                        data = json.load(file)
                    
                    # Search through all entries to find matches
                    matching_results = []
                    lookup_attr = f"profile_{self.layer}_attribute_{6 if self.layer <= 2 else 5}"
                    
                    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) == key_value:
                            matching_results.append(profile)
                    
                    return {
                        "status": f"Search_Profile_Layer_{self.layer} invocation is successful",
                        "result": matching_results
                    }
                except Exception as e:
                    return {
                        "status": f"Search_Profile_Layer_{self.layer} invocation failed - {str(e)}",
                        "result": []
                    }
        
        return CustomSearchTool(layer, self.profiles_path)
    
    def _parse_index_access(self, text):
        """Parse index access like 'by index 123' and return the index"""
        match = re.search(r'by index (\d+)', text)
        if match:
            return int(match.group(1))
        return None
    
    def _parse_string_lookup(self, text):
        """Parse string lookup like "with string 'ab'" and return the string"""
        match = re.search(r"with string '([^']+)'", text)
        if match:
            return match.group(1)
        return None
    
    def _find_extremum_in_profiles(self, profiles, condition):
        """Find the profile that satisfies the extremum condition with detailed logging"""
        if not profiles:
            return None, "No profiles available for selection"
            
        # Parse extremum condition (e.g., "largest attribute_1", "smallest sum of attribute_1 and attribute_2")
        if "largest" in condition:
            extremum = "largest"
        elif "smallest" in condition:
            extremum = "smallest"
        else:
            return profiles[0], "No extremum specified, selected first profile"
        
        selection_reasoning = []
        
        # Extract what we're comparing
        if "sum of" in condition:
            # Handle sum of attributes
            attr_matches = re.findall(r'attribute_(\w+)', condition)
            if len(attr_matches) >= 2:
                def calc_value(profile):
                    layer = self._get_layer_from_profile(profile)
                    key1 = f"profile_{layer}_attribute_{attr_matches[0]}"
                    key2 = f"profile_{layer}_attribute_{attr_matches[1]}"
                    val1 = profile.get(key1, 0)
                    val2 = profile.get(key2, 0)
                    return val1 + val2, f"sum({val1} + {val2}) = {val1 + val2}"
                criterion = f"sum of attribute_{attr_matches[0]} and attribute_{attr_matches[1]}"
            else:
                return profiles[0], "Invalid sum condition, selected first profile"
                
        elif "product of" in condition:
            # Handle product of attributes
            attr_matches = re.findall(r'attribute_(\w+)', condition)
            if len(attr_matches) >= 2:
                def calc_value(profile):
                    layer = self._get_layer_from_profile(profile)
                    key1 = f"profile_{layer}_attribute_{attr_matches[0]}"
                    key2 = f"profile_{layer}_attribute_{attr_matches[1]}"
                    val1 = profile.get(key1, 0)
                    val2 = profile.get(key2, 0)
                    return val1 * val2, f"product({val1} * {val2}) = {val1 * val2}"
                criterion = f"product of attribute_{attr_matches[0]} and attribute_{attr_matches[1]}"
            else:
                return profiles[0], "Invalid product condition, selected first profile"
                
        elif "average of" in condition:
            # Handle average of attributes
            attr_matches = re.findall(r'attribute_(\w+)', condition)
            if len(attr_matches) >= 2:
                def calc_value(profile):
                    layer = self._get_layer_from_profile(profile)
                    key1 = f"profile_{layer}_attribute_{attr_matches[0]}"
                    key2 = f"profile_{layer}_attribute_{attr_matches[1]}"
                    val1 = profile.get(key1, 0)
                    val2 = profile.get(key2, 0)
                    avg = (val1 + val2) / 2
                    return avg, f"average({val1} + {val2})/2 = {avg}"
                criterion = f"average of attribute_{attr_matches[0]} and attribute_{attr_matches[1]}"
            else:
                return profiles[0], "Invalid average condition, selected first profile"
                
        elif "result of" in condition and "minus" in condition:
            # Handle result of A minus B for attributes or attribute and global
            if "global_attribute_" in condition:
                attr_match = re.search(r'attribute_(\w+)', condition)
                global_match = re.search(r'global_attribute_(\d+)', condition)
                if attr_match and global_match:
                    def calc_value(profile):
                        layer = self._get_layer_from_profile(profile)
                        key = f"profile_{layer}_attribute_{attr_match.group(1)}"
                        profile_val = profile.get(key, 0)
                        global_val = self.global_attributes.get(f"global_attribute_{global_match.group(1)}", 0)
                        diff = profile_val - global_val
                        return diff, f"({profile_val} - {global_val}) = {diff}"
                    criterion = f"result of attribute_{attr_match.group(1)} minus global_attribute_{global_match.group(1)}"
                else:
                    return profiles[0], "Invalid global subtraction condition, selected first profile"
            else:
                attr_matches = re.findall(r'attribute_(\w+)', condition)
                if len(attr_matches) >= 2:
                    def calc_value(profile):
                        layer = self._get_layer_from_profile(profile)
                        key1 = f"profile_{layer}_attribute_{attr_matches[0]}"
                        key2 = f"profile_{layer}_attribute_{attr_matches[1]}"
                        val1 = profile.get(key1, 0)
                        val2 = profile.get(key2, 0)
                        diff = val1 - val2
                        return diff, f"({val1} - {val2}) = {diff}"
                    criterion = f"result of attribute_{attr_matches[0]} minus attribute_{attr_matches[1]}"
                else:
                    return profiles[0], "Invalid subtraction condition, selected first profile"
        elif "difference between" in condition:
            # Handle legacy "difference between" terminology - treat as subtraction for backward compatibility
            if "global_attribute_" in condition:
                attr_match = re.search(r'attribute_(\w+)', condition)
                global_match = re.search(r'global_attribute_(\d+)', condition)
                if attr_match and global_match:
                    def calc_value(profile):
                        layer = self._get_layer_from_profile(profile)
                        key = f"profile_{layer}_attribute_{attr_match.group(1)}"
                        profile_val = profile.get(key, 0)
                        global_val = self.global_attributes.get(f"global_attribute_{global_match.group(1)}", 0)
                        diff = profile_val - global_val
                        return diff, f"({profile_val} - {global_val}) = {diff}"
                    criterion = f"result of attribute_{attr_match.group(1)} minus global_attribute_{global_match.group(1)}"
                else:
                    return profiles[0], "Invalid global subtraction condition, selected first profile"
            else:
                attr_matches = re.findall(r'attribute_(\w+)', condition)
                if len(attr_matches) >= 2:
                    def calc_value(profile):
                        layer = self._get_layer_from_profile(profile)
                        key1 = f"profile_{layer}_attribute_{attr_matches[0]}"
                        key2 = f"profile_{layer}_attribute_{attr_matches[1]}"
                        val1 = profile.get(key1, 0)
                        val2 = profile.get(key2, 0)
                        diff = val1 - val2
                        return diff, f"({val1} - {val2}) = {diff}"
                    criterion = f"result of attribute_{attr_matches[0]} minus attribute_{attr_matches[1]}"
                else:
                    return profiles[0], "Invalid subtraction condition, selected first profile"
        else:
            # Handle single attribute
            attr_match = re.search(r'attribute_(\w+)', condition)
            if attr_match:
                def calc_value(profile):
                    layer = self._get_layer_from_profile(profile)
                    key = f"profile_{layer}_attribute_{attr_match.group(1)}"
                    val = profile.get(key, 0)
                    return val, f"attribute_{attr_match.group(1)} = {val}"
                criterion = f"attribute_{attr_match.group(1)}"
            else:
                return profiles[0], "No valid attribute found, selected first profile"
        
        # Calculate values for all profiles and build reasoning
        profile_values = []
        for i, profile in enumerate(profiles):
            value, explanation = calc_value(profile)
            profile_values.append((i, value, explanation))
            selection_reasoning.append(f"Profile {i+1}: {explanation}")
        
        # Find extremum
        if extremum == "largest":
            best_idx, best_value, best_explanation = max(profile_values, key=lambda x: x[1])
            reasoning_summary = f"Comparing {criterion}: {'; '.join(selection_reasoning)}. Profile {best_idx+1} has the largest value ({best_value}), so it was selected."
        else:  # smallest
            best_idx, best_value, best_explanation = min(profile_values, key=lambda x: x[1])
            reasoning_summary = f"Comparing {criterion}: {'; '.join(selection_reasoning)}. Profile {best_idx+1} has the smallest value ({best_value}), so it was selected."
        
        print(f"SELECTION: {reasoning_summary}")
        
        return profiles[best_idx], reasoning_summary
    
    def _get_layer_from_profile(self, profile):
        """Determine which layer a profile belongs to based on its keys"""
        # Look for any profile_X_attribute_1 pattern to determine the layer
        import re
        for key in profile.keys():
            match = re.match(r'profile_(\d+)_attribute_1', key)
            if match:
                return int(match.group(1))
        return 1  # Default fallback
    
    def execute_structured_query(self, execution_plan: List[Dict[str, Any]]) -> Dict[str, Any]:
        """Execute a structured query plan and return the rollout with results"""
        rollout = []
        current_profiles = {}
        current_search_results = {}  # Store search results for selection
        current_adjacent_results = {}  # Store adjacent results for selection
        step_counter = 1
        
        try:
            for instruction in execution_plan:
                action = instruction["action"]
                
                if action == "get_profile":
                    # Direct profile access by index
                    layer = instruction["layer"]
                    value = instruction["value"]
                    result = self.tools[f"get_layer_{layer}"].invoke(value)
                    
                    if result["status"].startswith(f"Get_Profile_Layer_{layer} invocation is successful"):
                        current_profiles[layer] = result["result"]
                        rollout.append({
                            "step": step_counter,
                            "action": f"Get_Profile_Layer_{layer}({value})",
                            "result": result["result"]
                        })
                    else:
                        rollout.append({
                            "step": step_counter,
                            "action": f"Get_Profile_Layer_{layer}({value})",
                            "result": None,
                            "error": result["status"]
                        })
                    step_counter += 1
                
                elif action == "search_profile":
                    # Search profiles by lookup string
                    layer = instruction["layer"]
                    lookup_string = instruction["lookup_string"]
                    result = self.tools[f"search_layer_{layer}"].invoke(lookup_string)
                    
                    if result["status"].startswith(f"Search_Profile_Layer_{layer} invocation is successful"):
                        matching_profiles = result["result"]
                        current_search_results[layer] = matching_profiles
                        rollout.append({
                            "step": step_counter,
                            "action": f"Search_Profile_Layer_{layer}('{lookup_string}')",
                            "result": matching_profiles,
                            "candidates_found": len(matching_profiles)
                        })
                    else:
                        rollout.append({
                            "step": step_counter,
                            "action": f"Search_Profile_Layer_{layer}('{lookup_string}')",
                            "result": None,
                            "error": result["status"]
                        })
                    step_counter += 1
                
                elif action == "access_adjacent":
                    # Access adjacent profiles
                    layer = instruction["layer"]
                    from_layer = instruction["from_layer"]
                    
                    if from_layer in current_profiles and current_profiles[from_layer]:
                        # Determine the reference attribute dynamically using attribute_dict
                        try:
                            ref_attr = self._get_reference_attribute(from_layer, layer)
                        except ValueError as e:
                            rollout.append({
                                "step": step_counter,
                                "action": f"Access adjacent layer {layer} profiles",
                                "result": None,
                                "error": str(e)
                            })
                            step_counter += 1
                            continue
                        
                        adjacent_refs = current_profiles[from_layer].get(ref_attr, [])
                        if adjacent_refs:
                            adjacent_profiles = []
                            for ref in adjacent_refs:
                                result = self.tools[f"get_layer_{layer}"].invoke(ref)
                                if result["status"].startswith(f"Get_Profile_Layer_{layer} invocation is successful"):
                                    adjacent_profiles.append(result["result"])
                            
                            current_adjacent_results[layer] = adjacent_profiles
                            rollout.append({
                                "step": step_counter,
                                "action": f"Access adjacent layer {layer} profiles",
                                "result": adjacent_profiles,
                                "adjacent_candidates": len(adjacent_profiles)
                            })
                        else:
                            rollout.append({
                                "step": step_counter,
                                "action": f"Access adjacent layer {layer} profiles",
                                "result": None,
                                "error": "No adjacent references found"
                            })
                    else:
                        rollout.append({
                            "step": step_counter,
                            "action": f"Access adjacent layer {layer} profiles",
                            "result": None,
                            "error": f"No current layer {from_layer} profile available"
                        })
                    step_counter += 1
                
                elif action == "select_profile":
                    # Select profile based on criteria
                    criteria = instruction["criteria"]
                    from_source = instruction["from"]
                    
                    if from_source == "search_results":
                        # Find the layer with current search results
                        source_profiles = None
                        source_layer = None
                        for layer, profiles in current_search_results.items():
                            if profiles:
                                source_profiles = profiles
                                source_layer = layer
                                break
                    elif from_source == "adjacent_results":
                        # Find the layer with current adjacent results
                        source_profiles = None
                        source_layer = None
                        for layer, profiles in current_adjacent_results.items():
                            if profiles:
                                source_profiles = profiles
                                source_layer = layer
                                break
                    
                    if source_profiles:
                        selected_profile, selection_reason = self._find_extremum_in_profiles(source_profiles, criteria)
                        if source_layer:
                            current_profiles[source_layer] = selected_profile
                        rollout.append({
                            "step": step_counter,
                            "action": f"Select profile with {criteria}",
                            "result": selected_profile,
                            "reasoning": selection_reason
                        })
                        
                        # Clear the source after selection
                        if from_source == "search_results" and source_layer:
                            current_search_results[source_layer] = []
                        elif from_source == "adjacent_results" and source_layer:
                            current_adjacent_results[source_layer] = []
                    else:
                        rollout.append({
                            "step": step_counter,
                            "action": f"Select profile with {criteria}",
                            "result": None,
                            "error": f"No profiles available from {from_source}"
                        })
                    step_counter += 1
                
        except Exception as e:
            rollout.append({
                "step": step_counter,
                "action": "Query execution",
                "result": None,
                "error": str(e)
            })
        
        return {
            "rollout": rollout,
            "final_profiles": current_profiles
        }

    def _get_reference_attribute(self, from_layer: int, target_layer: int) -> str:
        """
        Dynamically determine the reference attribute based on layer transition using attribute_dict.
        
        Args:
            from_layer: The layer we're transitioning from
            target_layer: The layer we're transitioning to
            
        Returns:
            The attribute name that contains references for this transition
        """
        # Find the attribute in the from_layer that can reference the target_layer
        layer_attributes = self.attribute_dict[from_layer - 1]  # Convert to 0-indexed
        
        for attr_num, attr_type in layer_attributes.items():
            # Check if this attribute can reference the target layer
            if attr_type.startswith("reference_"):
                if attr_type == "reference_1" and from_layer == target_layer:
                    # Same layer reference
                    return f"profile_{from_layer}_attribute_{attr_num}"
                elif attr_type == "reference_2" and target_layer == 2:
                    # Can reference layer 2
                    return f"profile_{from_layer}_attribute_{attr_num}"
                elif attr_type == "reference_3" and target_layer == 3:
                    # Can reference layer 3
                    return f"profile_{from_layer}_attribute_{attr_num}"
                elif attr_type == "reference_4" and target_layer == 4:
                    # Can reference layer 4
                    return f"profile_{from_layer}_attribute_{attr_num}"
                elif attr_type == "reference_5" and target_layer == 5:
                    # Can reference layer 5
                    return f"profile_{from_layer}_attribute_{attr_num}"
                elif attr_type == f"reference_{target_layer}":
                    # Generic pattern for any layer number
                    return f"profile_{from_layer}_attribute_{attr_num}"
        
        # If no matching attribute found, raise an error
        raise ValueError(f"No reference attribute found in layer {from_layer} that can access layer {target_layer}")

def create_qa_generator(attribute_dict: List[Dict[str, str]], 
                      layer_lookup_tables: Dict[int, List[str]] = None,
                      task_layer_requirements: Dict[int, List[int]] = None,
                      global_attributes: Dict[str, Any] = None,
                      base_output_dir: str = None) -> UserRequestGenerator:
    """
    Helper function to create a UserRequestGenerator with custom configurations.
    
    Args:
        attribute_dict: List of dictionaries defining attributes for each layer
        layer_lookup_tables: Lookup strings for each layer (optional)
        task_layer_requirements: Which layers each task type requires (optional)
        global_attributes: Custom global attribute values (optional)
    
    Returns:
        UserRequestGenerator: Configured generator instance
    
    Example:
        # Custom global attributes
        global_attrs = {
            "global_attribute_1": 100,
            "global_attribute_2": 200,
            "global_attribute_3": 300
        }
        
        # Create generator
        generator = create_qa_generator(
            attribute_dict=my_attribute_dict,
            global_attributes=global_attrs
        )
        
        # Generate questions and answers
        generator.save_requests_to_json("output.json", num_requests=100)
    """
    return UserRequestGenerator(
        attribute_dict=attribute_dict,
        layer_lookup_tables=layer_lookup_tables,
        task_layer_requirements=task_layer_requirements,
        global_attributes=global_attributes,
        base_output_dir=base_output_dir
    )

def generate_qa_with_custom_globals(attribute_dict: List[Dict[str, str]],
                                   global_attributes: Dict[str, Any],
                                   output_path: str,
                                   num_requests: int = 50,
                                   layer_lookup_tables: Dict[int, List[str]] = None,
                                   profiles_path: str = "Generated_data/Profiles/",
                                   base_output_dir: str = None) -> Dict[str, Any]:
    """
    Convenience function to generate Q&A with custom global attributes.
    
    Args:
        attribute_dict: Layer attribute definitions
        global_attributes: Custom global attribute values
        output_path: Where to save the generated Q&A
        num_requests: Number of requests to generate
        layer_lookup_tables: Lookup strings for each layer (optional)
        profiles_path: Path to profile data files
    
    Returns:
        Generated Q&A data
    """
    generator = create_qa_generator(
        attribute_dict=attribute_dict,
        layer_lookup_tables=layer_lookup_tables,
        global_attributes=global_attributes,
        base_output_dir=base_output_dir
    )
    
    return generator.save_requests_to_json(
        output_path=output_path,
        num_requests=num_requests,
        include_rollouts=True,
        profiles_path=profiles_path
    )

if __name__ == "__main__":
    # Define attribute_dict (this should be passed from policy_generator)
    # This example shows a 3-layer structure, but can be extended to any number of layers
    attribute_dict = [
        {  # Layer 1
            "1": "condition",
            "2": "condition", 
            "3": "reference_1",
            "4": "reference_2",
            "5": "condition",
            "6": "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"
        }
    ]
    
    layer_1_lookup_table = ['dt']
    layer_2_lookup_table = ['ul']
    layer_3_lookup_table = ['vr']
    
    # Create lookup tables dictionary (dynamically expandable for more layers)
    layer_lookup_tables = {
        1: layer_1_lookup_table,
        2: layer_2_lookup_table,
        3: layer_3_lookup_table
        # For more layers, add entries like:
        # 4: layer_4_lookup_table,
        # 5: layer_5_lookup_table,
    }
    
    # Define custom global attributes (users can modify these values)
    custom_global_attributes = {
        "global_attribute_1": 100,  # Changed from 42 for testing
        "global_attribute_2": 200,  # Changed from 37 for testing
        "global_attribute_3": 300   # Changed from 55 for testing
    }
    
    # Generate user requests
    generator = UserRequestGenerator(
        attribute_dict=attribute_dict, 
        layer_lookup_tables=layer_lookup_tables, 
        global_attributes=custom_global_attributes
    )
    
    # Generate and save requests to the specified path
    output_path = "Generated_data/Queries/qa.json"
    generator.save_requests_to_json(output_path, num_requests=30, include_rollouts=True)


class UserRequestGeneratorVariant(UserRequestGenerator):
    """
    Specialized UserRequestGenerator that can handle different exec file imports
    for policy and task variants.
    """
    
    def __init__(self, attribute_dict: List[Dict[str, str]], layer_lookup_tables: Dict[int, List[str]] = None, 
                 task_layer_requirements: Dict[int, List[int]] = None, global_attributes: Dict[str, Any] = None,
                 base_output_dir: str = None, variant_type: str = "policy"):
        """
        Initialize variant generator with specific variant type.
        
        Args:
            variant_type (str): Either "policy" or "task" to specify which variant exec to use
        """
        super().__init__(attribute_dict, layer_lookup_tables, task_layer_requirements, global_attributes, base_output_dir)
        self.variant_type = variant_type
    
    def _import_compute_all_tasks(self):
        """Dynamically import compute_all_tasks from the correct variant exec.py"""
        try:
            # Try to import from the dynamic path first
            if hasattr(self, 'task_path'):
                if self.variant_type == "policy":
                    exec_file = os.path.join(self.task_path, 'exec_overide_policy.py')
                    function_suffix = "_overide_policy"
                elif self.variant_type == "task":
                    exec_file = os.path.join(self.task_path, 'exec_overide_task.py')
                    function_suffix = "_overide_task"
                else:
                    # Fallback to original exec
                    exec_file = os.path.join(self.task_path, 'exec.py')
                    function_suffix = ""
                
                if os.path.exists(exec_file):
                    import importlib.util
                    spec = importlib.util.spec_from_file_location("exec_variant", exec_file)
                    exec_module = importlib.util.module_from_spec(spec)
                    spec.loader.exec_module(exec_module)
                    
                    # Get the function with the appropriate suffix
                    function_name = f'compute_all_tasks{function_suffix}'
                    compute_function = getattr(exec_module, function_name, None)
                    
                    if compute_function:
                        print(f"Successfully imported {function_name} from {exec_file}")
                        return compute_function
                    else:
                        print(f"Warning: Function {function_name} not found in {exec_file}")
                        return None
                else:
                    print(f"Warning: Variant exec file {exec_file} not found")
                    return None
            else:
                # Fallback to global import
                try:
                    if self.variant_type == "policy":
                        from exec_overide_policy import compute_all_tasks_overide_policy
                        return compute_all_tasks_overide_policy
                    elif self.variant_type == "task":
                        from exec_overide_task import compute_all_tasks_overide_task
                        return compute_all_tasks_overide_task
                    else:
                        from exec import compute_all_tasks
                        return compute_all_tasks
                except ImportError:
                    return None
        except Exception as e:
            print(f"Warning: Could not import variant compute_all_tasks: {e}")
            return None 