std_Gauss = Normal(0,1)

# Small utilities to avoid recomputing invariants for a given sigma
struct QGCache
    sigma::Float64
    inv_sigma::Float64
    inv_sigma2::Float64
    exp_eps::Float64
    cdf_D::Float64
    inv_c::Float64
end

function qg_cache(sigma::Real)
    σ = float(sigma)
    invσ = 1/σ
    invσ2 = invσ^2
    exp_eps = exp(epsilon)
    cdf_D = cdf(std_Gauss, Delta*invσ)
    inv_c = 1/(sqrt(2*pi)*σ*(exp_eps + 2*cdf_D))
    return QGCache(σ, invσ, invσ2, exp_eps, cdf_D, inv_c)
end

@inline function qG_pdf(x::Real, c::QGCache)
    absx = abs(x)
    term0 = c.exp_eps * exp(-x^2 * 0.5 * c.inv_sigma2)
    term1 = exp(- (absx - Delta)^2 * 0.5 * c.inv_sigma2)
    return (term0 + term1) * c.inv_c
end

"Quasi Gaussian distribution PDF"
qG_pdf(x, sigma) = qG_pdf(x, qg_cache(sigma))

"Quasi Gaussian distribution CDF"
function qG_cdf(x, sigma)
    c = qg_cache(sigma)
    denom = c.exp_eps + 2*c.cdf_D
    if x < 0
        num = c.exp_eps*cdf(std_Gauss, x*c.inv_sigma) + cdf(std_Gauss, (x + Delta)*c.inv_sigma)
        return num / denom
    else
        num = c.exp_eps*cdf(std_Gauss, x*c.inv_sigma) + cdf(std_Gauss, (x - Delta)*c.inv_sigma) + c.cdf_D - cdf(std_Gauss, -Delta*c.inv_sigma)
        return num / denom
    end
end

function sample_quasi(sigma)
    c = qg_cache(sigma)
    p0 = c.exp_eps / (c.exp_eps + 2*c.cdf_D)
    if rand() < p0
        return rand(std_Gauss)*c.sigma
    else
        p = rand()
        s = rand([-1, 1])
        x = Delta + c.sigma*quantile(std_Gauss, cdf(std_Gauss, -Delta*c.inv_sigma) + p*c.cdf_D)
        return s*x
    end
end

"Quasi Gaussian distribution expected absolute value"
function qG_amplitude(sigma)
    c = qg_cache(sigma)
    num = sqrt(2/pi)*c.sigma*(c.exp_eps + exp(- (Delta^2)*0.5*c.inv_sigma2)) + 2*Delta*c.cdf_D
    denom =  c.exp_eps + 2*c.cdf_D
    return num / denom
end

"Quasi Gaussian distribution expected squared value"
function qG_power(sigma)
    c = qg_cache(sigma)
    term = c.cdf_D*(c.sigma^2 + Delta^2) + ((c.sigma*Delta)/(sqrt(2*pi)))*exp(- (Delta^2)*0.5*c.inv_sigma2)
    num = c.exp_eps*c.sigma^2 + 2*term
    denom =  c.exp_eps + 2*c.cdf_D
    return num / denom
end

"The function used in determining sigma_1"
function hG(sigma)
    invσ = 1/sigma
    c1 = cdf(std_Gauss, - epsilon*sigma/Delta - Delta*invσ)
    c2 = cdf(std_Gauss, -epsilon*sigma/Delta + Delta*invσ)
    cD = cdf(std_Gauss, Delta*invσ)
    return exp(2*epsilon)*c1 - c2 + (exp(epsilon)+2*cD)*delta
end

"minimizer of the qG pdf"
function qG_minimizer(sigma)
    c = qg_cache(sigma)
    x_min = Delta
    root_finder = sqrt(max(Delta^2 - 4*c.sigma^2, 0))
    x2 = (Delta + root_finder)/2
    t = -c.exp_eps + ((Delta - x2)/x2)*exp((2*x2*Delta - Delta^2)*0.5*c.inv_sigma2)
    if Delta^2 > 4*c.sigma^2 && t > 0
        x_temp = Optim.minimizer(optimize(x -> qG_pdf(x,c), Delta/2, x2, GoldenSection()))
        if qG_pdf(x_temp,c) < qG_pdf(x_min,c)
            x_min = x_temp
        end
    end
    return x_min, qG_pdf(x_min,c)
end

"maximizer of the qG pdf"
function qG_maximizer(sigma)
    c = qg_cache(sigma)
    root_finder = sqrt(max(Delta^2 - 4*c.sigma^2, 0))
    x1 = (Delta - root_finder)/2
    if Delta^2 <= 4*c.sigma^2
        rrr = Optim.minimizer(optimize(x -> -qG_pdf(x,c), 0, Delta/2, GoldenSection()))
        return rrr, qG_pdf(rrr,c)
    else
        rrr = Optim.minimizer(optimize(x -> -qG_pdf(x,c), 0, x1, GoldenSection()))
        return rrr, qG_pdf(rrr,c)
    end
end

"max/min ratio of the qG pdf within [0, Delta]"
function qG_maxmin_ratio(sigma)
    max_val = qG_maximizer(sigma)[2]
    min_val = qG_minimizer(sigma)[2]
    return min_val == 0 ? Inf : (max_val/min_val) - exp(epsilon)
end

"tune optimal sigma for qG mechanism"
function qG_sigmas()
    sigma_1 = 0.0
    if exp(epsilon) + 2 < 1/delta
        r1 = sqrt(2*(epsilon - log(delta)))*Delta/epsilon
        sigma_1 = find_zero(hG, (1e-8, r1), Bisection())
    end

    r2 = sqrt(2*log(1.25/delta))*Delta/epsilon
    while qG_maxmin_ratio(r2) > 0
        r2 *= 2
    end
    sigma_2 = find_zero(qG_maxmin_ratio, (1e-8, r2), Bisection())

    return sigma_1, sigma_2
end
