# monitoring.py

import time
from functools import wraps
import inspect
# --- 1. Import the tabulate function ---



class TimeMonitor:
    """
    A decorator class to monitor the execution time of functions and methods.
    It tracks the number of calls, total time, and average time, and
    distinguishes methods by their class name.
    """

    def __init__(self):
        self.timings = {}

    def __call__(self, func):
        @wraps(func)
        def wrapper(*args, **kwargs):
            func_key = func.__name__
            if args and hasattr(args[0], '__class__') and not inspect.isfunction(args[0]):
                class_name = args[0].__class__.__name__
                func_key = f"{class_name}.{func.__name__}"

            if func_key not in self.timings:
                self.timings[func_key] = {'calls': 0, 'total_time': 0.0}

            start_time = time.perf_counter()
            result = func(*args, **kwargs)
            end_time = time.perf_counter()

            elapsed_time = end_time - start_time
            self.timings[func_key]['calls'] += 1
            self.timings[func_key]['total_time'] += elapsed_time

            return result

        return wrapper

    # --- 2. Modify the report method ---
    def report(self, time_to_process_pos: float = None, time_to_process_neg: float = None):
        from tabulate import tabulate
        """Prints a formatted report of all monitored functions using tabulate."""
        print("\n--- Function Execution Time Report ---")
        if not self.timings:
            print("No functions were monitored.")
            return

        # Prepare data for tabulate
        table_data = []

        # Sort by average time, descending
        sorted_functions = sorted(
            self.timings.items(),
            key=lambda item: (item[1]['total_time'] / item[1]['calls']) if item[1]['calls'] > 0 else 0,
            reverse=True
        )

        for func_name, data in sorted_functions:
            calls = data['calls']
            total_time_s = data['total_time']
            # Calculate average time in milliseconds
            avg_time_ms = (total_time_s / calls) * 1000 if calls > 0 else 0

            # Add a row to our table data
            table_data.append([
                func_name,
                f"{calls:,}",  # Add comma for thousands separator
                f"{total_time_s:.6f}",
                f"{avg_time_ms:.6f}"
            ])

        # Define the headers for our table
        headers = [
            "Function/Method Name",
            "Calls",
            "Total Time (s)",
            "Average Time (ms)"
        ]

        # Print the table
        #print(f'Time to Process Positive Instances: ({round(time_to_process_pos, 2)} sec)\n'
        #      f'Time to Process Negative Instances: ({round(time_to_process_neg, 2)} sec)\n'
        #      f'Total time to process instances: ({round(time_to_process_pos + time_to_process_neg, 2)} sec)\n)')
        print(tabulate(table_data, headers=headers, tablefmt="grid", numalign="right"))



# --- The Central Registry (no change needed here) ---
time_monitor = TimeMonitor()
