"""Utility functions for checking heuristic modules."""

import types
from typing import Optional


def has_score_function(heuristic_module: types.ModuleType) -> bool:
    """
    Check if heuristic_module has a score function.
    
    Args:
        heuristic_module: compiled module object
        
    Returns:
        True if module has score function and is callable, otherwise False
    """
    # method 1: use hasattr to check if attribute exists
    if not hasattr(heuristic_module, 'score'):
        return False
    
    # method 2: check if callable
    score_func = getattr(heuristic_module, 'score', None)
    if not callable(score_func):
        return False
    
    return True


def check_module_attributes(heuristic_module: types.ModuleType) -> dict:
    """
    Check all attributes of the module, for debugging.
    
    Args:
        heuristic_module: compiled module object
        
    Returns:
        dictionary containing check results
    """
    result = {
        'has_score': hasattr(heuristic_module, 'score'),
        'score_callable': False,
        'all_attributes': [],
        'public_attributes': []
    }
    
    if result['has_score']:
        score_func = getattr(heuristic_module, 'score', None)
        result['score_callable'] = callable(score_func)
    
    result['all_attributes'] = dir(heuristic_module)
    
    result['public_attributes'] = [attr for attr in dir(heuristic_module) 
                                   if not attr.startswith('_')]
    
    return result


def validate_score_function(heuristic_module: types.ModuleType, 
                            item: float = 1.0, 
                            bins=None) -> tuple[bool, Optional[str]]:
    """
    Validate if score function can be called normally.
    
    Args:
        heuristic_module: compiled module object
        item: test item value
        bins: test bins array (if None, create a simple array)
        
    Returns:
        (is_valid, error_message)
        is_valid: True if function can be called normally
        error_message: if there is an error, contains error information; otherwise None
    """
    import numpy as np
    
    # check if there is a score function
    if not hasattr(heuristic_module, 'score'):
        return False, "Module does not have 'score' attribute"
    
    score_func = getattr(heuristic_module, 'score', None)
    if not callable(score_func):
        return False, "'score' exists but is not callable"
    
    try:
        if bins is None:
            bins = np.array([10.0, 20.0, 30.0])
        
        result = score_func(item, bins)
        
        if result is None:
            return False, "score function returned None"
        
        if not isinstance(result, np.ndarray):
            return False, f"score function returned {type(result)}, expected np.ndarray"
        
        if len(result) != len(bins):
            return False, f"score function returned {len(result)} values, expected {len(bins)}"
        
        return True, None
        
    except Exception as e:
        return False, f"Error calling score function: {e}"

