import numpy as np

from pe.dp import DP
from pe.logging import execution_logger

class Exponential(DP):
    """The Exponential mechanism for Differential Privacy (DP) histogram."""

    def set_epsilon_and_delta(self, num_iterations, epsilon, delta, noise_multiplier=None):
        """Set the epsilon and delta for the Exponential mechanism.

        :param num_iterations: The number of PE iterations
        :type num_iterations: int
        :param epsilon: The epsilon value of DP
        :type epsilon: float
        :param delta: The delta value of DP
        :type delta: float
        :raises ValueError: If delta is None
        """
        if delta is None:
            raise ValueError("Delta should not be None")
        self._delta = delta

        self._delta = 0.0
        self._epsilon = epsilon

        execution_logger.info(
            f"Exponential DP epsilon={self._epsilon}, delta={self._delta}, num_iterations={num_iterations}."
        )
    
    def add_noise(self, syn_data):
        # this function is not used in Exponential mechanism
        pass

    def get_probabilistic(self, utility_function_results, utility_delta):
        """Get the probabilistic result from the utility function results.

        :param utility_function_results: The utility function results
        :type utility_function_results: list
        :param utility_delta: The delta value of the utility function
        :type utility_delta: float
        :return: The probabilistic result
        :rtype: list
        :raises ValueError: If utility_function_results is empty or output is None
        """
        if not utility_function_results:
            raise ValueError("Utility function results should not be empty")

        utilities = np.array(utility_function_results)
        scores = np.exp(self._epsilon * utilities / (2 * utility_delta))
        selection_probability = scores / np.sum(scores)
        
        if selection_probability is None:
            raise ValueError("Failed to select a valid output")

        return selection_probability