import numpy as np
from typing import Dict, Any, List
from scipy.stats import truncnorm


class HumanActionRandomizer:
    """
    Models the inherent variability of a human agent's performance.
    It generates a consistent performance multiplier for a single simulation run.
    """

    def __init__(self, mean=1.0, std_dev=0.2, min_multiplier=0.5, max_multiplier=2.0,
                 pause_probability=0.1, min_pause_duration=5.0, max_pause_duration=20.0):
        if min_multiplier <= 0:
            raise ValueError("min_multiplier must be greater than 0.")
        
        self.mean = mean
        self.std_dev = std_dev
        self.min_val = min_multiplier
        self.max_val = max_multiplier
        self.pause_prob = pause_probability
        self.min_pause = min_pause_duration
        self.max_pause = max_pause_duration

        self.performance_multiplier = self._get_performance_multiplier()


    def _get_performance_multiplier(self):
        a = (self.min_val - self.mean) / self.std_dev
        b = (self.max_val - self.mean) / self.std_dev
        dist = truncnorm(a, b, loc=self.mean, scale=self.std_dev)
        return dist.rvs(1)[0]

    def get_randomized_time_for_task(self, eet):
        base_exec_time = eet * self.performance_multiplier
        return base_exec_time
    

class ScenarioManager:
    """
    Manages high-level, discrete scenario events like goal changes.
    """


    def __init__(self, plan: List, goal_change_probability: float = 0.5):
        self.plan = plan
        self.goal_change_prob = goal_change_probability
        self.goal_change_occurs = False
        self.trigger_step = -1
        self.new_goal_info = "A new sub-goal has been added: 'slice_tomato'"

    def setup_scenario(self):

        if self.plan and np.random.rand() < self.goal_change_prob:
            self.goal_change_occurs = True

            self.trigger_step = np.random.randint(0, len(self.plan))
            print(f"[HumanSim] GOAL CHANGE SCHEDULED at human's step {self.trigger_step}.")
        else:
            self.goal_change_occurs = False
            print("[HumanSim] No dynamic goal change in this scenario.")

    def check_for_event(self, current_step: int):

        if self.goal_change_occurs and current_step == self.trigger_step:
            return self.new_goal_info
        return None


def generate_ground_truth(human_schedule: Dict[str, Any],
                          randomizer: HumanActionRandomizer,
                          scenario_mode: str = "pause",
                          is_calibrated: bool = False,
                          schedule_start_time: float = 0.0,
                          generate_async_events: bool = True) -> List[Dict[str, Any]]:
    """
    Generates a realistic, unpredictable 'ground truth' event log for a human agent
    based on a planned schedule. This log includes performance variations and potential interrupts.
    """


    print(f"\n[HumanSim] Generating ground truth (Scenario: {scenario_mode}, Calibrated: {is_calibrated})...")
    if not human_schedule:
        return []

    sorted_tasks = sorted(human_schedule.items(), key=lambda item: item[1]['start'])
    
    event_log = []
    

    
    if generate_async_events:
        if scenario_mode == "goal_change" and sorted_tasks:
            trigger_step = 0
            print(f"[HumanSim] GOAL CHANGE event scheduled after human's task #{trigger_step}.")
        else:
            trigger_step = -1

        if scenario_mode == "pause":
            pause_probability_per_task = 0.5
        else:
            pause_probability_per_task = 0.0
    else:
        trigger_step = -1
        pause_probability_per_task = 0.0

    goal_change_triggered = False
    pause_triggered = False
    last_actual_finish_time = schedule_start_time
    

    for idx, (task_name, info) in enumerate(sorted_tasks):
        planned_start_time = info['start']
        planned_duration = info['end'] - planned_start_time
        

        actual_start_time = planned_start_time

        event_log.append({
            "event_time": actual_start_time, "event_type": "human_task_actual_start", "task_name": task_name
        })

        if is_calibrated:
            actual_base_duration = planned_duration
        else:
            actual_base_duration = randomizer.get_randomized_time_for_task(planned_duration)
        
        pause_duration = 0.0
        

        if not pause_triggered and np.random.rand() < pause_probability_per_task:
            pause_triggered = True
            pause_duration = np.random.uniform(5.0, 15.0)
            pause_start_offset = np.random.uniform(0.3, 0.7) * actual_base_duration
            pause_start_time = actual_start_time + pause_start_offset
            
            event_log.append({
                "event_time": pause_start_time, "event_type": "human_interrupt_pause",
                "task_in_progress": task_name, "duration": pause_duration
            })

        actual_end_time = actual_start_time + actual_base_duration + pause_duration
        
        event_log.append({
            "event_time": actual_end_time, "event_type": "human_task_actual_end",
            "task_name": task_name, "planned_start_time": planned_start_time,
            "actual_start_time": actual_start_time,
            "planned_end_time": info['end'], "actual_end_time": actual_end_time
        })
        
        last_actual_finish_time = actual_end_time


        if generate_async_events and scenario_mode == "goal_change" and not goal_change_triggered and idx == trigger_step:
            goal_change_triggered = True
            event_log.append({
                "event_time": actual_end_time,
                "event_type": "human_interrupt_goal_change",
                "task_name": task_name,
                "trigger_task": task_name,
                "details": "User wants to add 'slice_tomato' to the salad."
            })

    event_log.sort(key=lambda e: e['event_time'])
    print("[HumanSim] Ground truth generated.")
    return event_log