"""Miscellaneous utilities."""
import functools
import importlib
from typing import List, Dict, Optional
from datetime import datetime


def require_python_package(
    import_name: str, install_command: str | None = None, install_link: str | None = None
):
    """Check if a package is available and provide installation hints on import failure.

    Args:
        import_name: The top-level importable module name a package provides.
        install_command: Installation command.
        install_link: URL link to installation guide.

    Returns:
        Callable: A decorator function that wraps the target function with package availability check.

    Raises:
        ImportError: When the specified package is not available, with installation
            instructions included in the error message.

    Example:
        >>> @require_python_package(
        ...     import_name='faiss',
        ...     install_command='pip install faiss-cpu',
        ...     install_link='https://github.com/facebookresearch/faiss/blob/main/INSTALL.md'
        ... )
        ... def create_faiss_index():
        ...     from faiss import IndexFlatL2  # Actual import in function
        ...     return IndexFlatL2(128)
    """
    def decorator(func):
        @functools.wraps(func)
        def wrapper(*args, **kwargs):
            try:
                importlib.import_module(import_name)
            except ImportError:
                error_msg = f"Missing required module - '{import_name}'\n"
                error_msg += f"💡 Install command: {install_command}\n" if install_command else ""
                error_msg += f"💡 Install guide:   {install_link}\n" if install_link else ""
                
                raise ImportError(error_msg) from None
            return func(*args, **kwargs)
        
        return wrapper
    
    return decorator


def extract_timestamp_suffix(name: str) -> Optional[str]:
    """Extract timestamp suffix from collection name.
    
    Try to parse a valid timestamp suffix (YYYYMMDD_HHMMSS) from the end of collection_name.
    - If successful, return this suffix string, e.g., "20251206_104523"
    - If failed, return None
    
    Args:
        name: Collection name string
        
    Returns:
        Timestamp suffix string or None
    """
    parts = name.split("_")
    # Need at least two parts to be xxx_YYYYMMDD_HHMMSS
    if len(parts) < 3:
        return None
    
    date_part, time_part = parts[-2], parts[-1]
    
    if len(date_part) != 8 or len(time_part) != 6:
        return None
    if not (date_part.isdigit() and time_part.isdigit()):
        return None
    
    candidate = f"{date_part}_{time_part}"
    
    try:
        datetime.strptime(candidate, "%Y%m%d_%H%M%S")
    except ValueError:
        return None
    
    return candidate


def is_sublist(list1: List[str], list2: List[str]) -> bool:
    """Check if list1 is a sublist of list2 (as sets).
    
    Args:
        list1: First list
        list2: Second list
        
    Returns:
        True if list1 is a subset of list2
    """
    return set(list1).issubset(set(list2))


def export_plan_to_txt(raw_plan: Dict) -> str:
    """Export plan dictionary to formatted text string.
    
    Args:
        raw_plan: Plan dictionary with step keys and content dicts
        
    Returns:
        Formatted plan text
    """
    plan_steps = []
    for step, content in raw_plan.items():
        plan_steps.append(f"{step}. {content['action']}: {content['details']}")
    return "\n".join(plan_steps)



def calculate_optimized_degree(current_performance: Dict, parent_performance: Dict = None) -> Optional[float]:
    """Calculate optimization degree (performance improvement ratio)
    
    Args:
        current_performance: Current performance metrics
        parent_performance: Parent version (before optimization) performance metrics
        
    Returns:
        Optimization degree (parent_mean / current_mean), returns None if cannot calculate
    """
    if parent_performance is None:
        return None
    if current_performance is None:
        return None
    
    parent_mean = parent_performance.get('mean')
    if parent_mean is None:
        return None
    
    current_mean = current_performance.get('mean')
    if current_mean is None or current_mean == 0:
        return None
    
    return parent_mean / current_mean