from __future__ import annotations

from typing import Dict, Any
from difflib import SequenceMatcher
import numpy as np


def calculate_code_similarity(code1: str, code2: str) -> float:
    return SequenceMatcher(None, code1, code2).ratio()


def calculate_behavioral_similarity_solver(utility_vec1: np.ndarray, utility_vec2: np.ndarray) -> float:
    if len(utility_vec1) == 0 or len(utility_vec2) == 0:
        return 0.0

    dot_product = np.dot(utility_vec1, utility_vec2)
    norm1 = np.linalg.norm(utility_vec1)
    norm2 = np.linalg.norm(utility_vec2)
    
    if norm1 == 0 or norm2 == 0:
        return 0.0

    similarity = dot_product / (norm1 * norm2)
    return (similarity + 1) / 2


def calculate_behavioral_similarity_generator(utility_vec1: np.ndarray, utility_vec2: np.ndarray) -> float:
  
    if len(utility_vec1) == 0 or len(utility_vec2) == 0:
        return 0.0
    
    dot_product = np.dot(utility_vec1, utility_vec2)
    norm1 = np.linalg.norm(utility_vec1)
    norm2 = np.linalg.norm(utility_vec2)
    
    if norm1 == 0 or norm2 == 0:
        return 0.0
    
    similarity = dot_product / (norm1 * norm2)
    return (similarity + 1) / 2


def calculate_combined_similarity(
    code_sim: float, 
    behavior_sim: float, 
    weight_code: float = 0.5, 
    weight_behavior: float = 0.5
) -> float:
   
    total_weight = weight_code + weight_behavior
    if total_weight == 0:
        return 0.0
    return (weight_code * code_sim + weight_behavior * behavior_sim) / total_weight


def config_to_string(config: Dict[str, Any]) -> str:
    return str(sorted(config.items()))

