import numpy as np
from scipy import special
import scipy.optimize as optim

def delta(epsilon: float, sens_per_sigma: float) -> float:
    """Compute delta for given epsilon and sensitivity per noise standard deviation for the Gaussian mechanism.

    Args:
        epsilon (float)
        sens_per_sigma (float): Sensitivity per noise standard deviation.

    Returns:
        float: Delta
    """
    if sens_per_sigma <= 0:
        return 0
    mu = sens_per_sigma**2 / 2
    term1 = special.erfc((epsilon - mu) / np.sqrt(mu) / 2)
    term2 = np.exp(epsilon) * special.erfc((epsilon + mu) / np.sqrt(mu) / 2)
    return 0.5 * (term1 - term2)

def find_sens_per_sigma(epsilon: float, delta_bound: float, sigma_upper_bound: float=20) -> float:
    """Find the required sensitivity per noise standard deviation for (epsilon, delta)-DP with Gaussian mechanism.

    Args:
        epsilon (float)
        delta_bound (float)
        sigma_upper_bound (float, optional): Upper bound guess on sensitivity per sigma. Defaults to 20.

    Returns:
        float: The required sensitivity per noise standard deviation.
    """
    sens_per_sigma, res = optim.brentq(lambda sigma: delta(epsilon, sigma) - delta_bound, 0, sigma_upper_bound, full_output=True)
    if res.converged:
        return sens_per_sigma
    else:
        raise RuntimeError("Failed to find valid sensitivity per sigma.")

def sigma(epsilon: float, delta_bound: float, sensitivity: float, sigma_upper_bound: float=20) -> float:
    """Find the required noise standard deviation for the Gaussian mechanism with (epsilon, delta)-DP.

    Args:
        epsilon (float)
        delta_bound (float)
        sensitivity (float): Sensitivity of the Gaussian mechanism.
        sigma_upper_bound (float, optional): Guess for an upper bound on sensitivity / sigma. Defaults to 20.

    Returns:
        float: The required noise standard deviation.
    """
    return sensitivity / find_sens_per_sigma(epsilon, delta_bound, sigma_upper_bound)

def epsilon(mu: float, delta_bound: float, epsilon_upper_bound: float=100) -> float:
    """Convert Gaussian DP bound to (epsilon, delta)-DP bound.

    Args:
        mu (float): Gaussian DP parameter.
        delta_bound (float): Delta to convert to.
        epsilon_upper_bound (float, optional): Upper bound guess on epsilon. Defaults to 100.

    Returns:
        float: Epsilon.
    """
    if delta(0, mu) <= delta_bound: return 0
    eps, res = optim.brentq(lambda epsilon: delta(epsilon, mu) - delta_bound, 0, epsilon_upper_bound)
    if res.converged:
        return eps
    else:
        raise RuntimeError("Failed to find valid epsilon.")