import numpy as np
from abc import ABC, abstractmethod
from typing import Dict, Any, Optional


class BaseClusteringHandler(ABC):

    def __init__(self, n_clusters: int = 10, random_state: int = 42, **kwargs):
        self.n_clusters = n_clusters
        self.random_state = random_state
        self.kwargs = kwargs
        self.clusterer = None

    @abstractmethod
    def fit_predict(self, data: np.ndarray) -> np.ndarray:
        pass

    def get_cluster_centers(self, data: np.ndarray, labels: np.ndarray) -> np.ndarray:
        unique_labels = np.unique(labels)
        n_features = data.shape[1]
        centers = np.zeros((len(unique_labels), n_features))

        for i, label in enumerate(unique_labels):
            mask = labels == label
            if np.sum(mask) > 0:
                centers[i] = np.mean(data[mask], axis=0)

        return centers

    def cluster(self, data: np.ndarray) -> np.ndarray:
        actual_n_clusters = min(self.n_clusters, data.shape[0])
        if actual_n_clusters != self.n_clusters:
            original_n_clusters = self.n_clusters
            self.n_clusters = actual_n_clusters

        labels = self.fit_predict(data)

        if hasattr(self.clusterer, 'cluster_centers_'):
            cluster_centers = self.clusterer.cluster_centers_
        else:
            cluster_centers = self.get_cluster_centers(data, labels)

        if actual_n_clusters != self.n_clusters:
            self.n_clusters = original_n_clusters

        return cluster_centers

    def get_algorithm_name(self) -> str:
        return self.__class__.__name__.replace('Handler', '').lower()

    def get_info(self) -> Dict[str, Any]:
        return {
            'algorithm': self.get_algorithm_name(),
            'n_clusters': self.n_clusters,
            'random_state': self.random_state,
            'parameters': self.kwargs
        }
