import numpy as np
from abc import ABC, abstractmethod

class AbstractVSA(ABC):
    def __init__(self, dimension):
        self.d = dimension

    @abstractmethod
    def generate_vector(self):
        """Creates a random atomic vector in the VSA space."""
        pass

    @abstractmethod
    def bind(self, u, v):
        """
        The Binding Operation (Multiplication / Convolution).
        Associates two vectors: u * v
        """
        pass

    @abstractmethod
    def unbind(self, u, z):
        """
        The Unbinding Operation (Inverse).
        Given bound vector z = u * v, returns approximation of v.
        approx_v = u_inverse * z
        """
        pass

    @abstractmethod
    def bundle(self, vectors):
        """
        The Superposition Operation (Addition).
        Combines multiple vectors into one: a + b + c
        """
        pass

    @abstractmethod
    def similarity(self, u, v):
        """Returns the similarity score (e.g., Cosine Similarity) between u and v."""
        pass

    def cleanup(self, noisy_vector, codebook):
        """
        Standard cleanup memory.
        Finds the vector in 'codebook' closest to 'noisy_vector'.
        Returns: (best_vector, best_label, score)
        """
        best_score = -1.0
        best_item = None
        best_label = None

        for label, vec in codebook.items():
            score = self.similarity(noisy_vector, vec)
            if score > best_score:
                best_score = score
                best_item = vec
                best_label = label

        return best_item, best_label, best_score