import numpy as np
from scipy.optimize import bisect
from scipy.special import erf
from scipy.stats import norm

def privacy_gaussian_mechanis_via_analytic_formula(sigma, epsilon, s=1.0):
    """
    Given the privacy paramter epsilon, the standard deviation sigma of the Gaussian mechanism, and the sensitivity s,
    compute the privacy parameter delta of the Gaussian mechanism via the analytic formula.
    """
    term1 = (1 - np.exp(epsilon)) / 2
    a = -epsilon * sigma / np.sqrt(2 * s**2)
    b = s / np.sqrt(8 * sigma**2)  
    term2 = 0.5 * (erf(a + b) - np.exp(epsilon) * erf(a - b))
    return term1 + term2

def compute_sigma_gaussian_mechanis_via_analytic_formula(epsilon, delta, s=1.0, min_sigma=1e-5, max_sigma=1e5):
    """
    Given the privacy paramter epsilon, the privacy parameter delta of the Gaussian mechanism, and the sensitivity s,
    compute the standard deviation sigma of the Gaussian mechanism via the analytic formula.
    """
    return bisect(lambda x: privacy_gaussian_mechanis_via_analytic_formula(x, epsilon, s) - delta, min_sigma, max_sigma)


def privacy_gaussian_mechanis_via_plrv(sigma, epsilon, s=1.0):
    """
    Given the privacy paramter epsilon, the standard deviation sigma of the Gaussian mechanism, and the sensitivity s,
    compute the privacy parameter delta of the Gaussian mechanism via the PLRV formula. 
    """
    assert s==1.0, "has not been implemented s!=1.0"
    term2 = (2*sigma**2*epsilon + 1)/(2*sigma)
    term1 = (2*sigma**2*epsilon - 1)/(2*sigma)

    return (1 - norm.cdf(term1)) - np.exp(epsilon)*(1 - norm.cdf(term2))

def compute_sigma_gaussian_mechanis_via_plrv(epsilon, delta, s=1.0, min_sigma=1e-5, max_sigma=1e5):
    """
    Given the privacy paramter epsilon, the privacy parameter delta of the Gaussian mechanism, and the sensitivity s,
    compute the standard deviation sigma of the Gaussian mechanism via the PLRV formula.
    """
    return bisect(lambda x: privacy_gaussian_mechanis_via_plrv(x, epsilon, s) - delta, min_sigma, max_sigma)