std_Gauss = Normal(0,1)

#note: we have 10 variants of this file, applying different discretization tricks depending on epsilon and delta
#if you need implementation help, please reach out to the authors, who are willing to share all functions

# Variant for small delta but moderate/large epsilon:
# - Epsilon-aware φ-grid cap (smaller max_steps when epsilon is large).
# - Epsilon-aware tail multiplier (shorter tails when weights concentrate near k=0).
# - Filters negligible mixture components (very small weights) to speed integrand at large epsilon.
struct MGCacheV6
    epsilon::Float64
    Delta::Float64
    K::Int
    weights::Vector{Float64}
    weights_norm::Vector{Float64}
    mu::Vector{Float64}
    s_w::Float64
    inv_n::Float64
    exp_eps::Float64
end

const _mg6_cache = Base.RefValue{Union{Nothing,MGCacheV6}}(nothing)

function refresh_mg_cache_v6!()
    ks = collect(-K:K)
    w = exp.(-abs.(ks) .* epsilon)
    s_w = sum(w)
    w_norm = w ./ s_w
    # Drop negligible weights; keep at least one component.
    cutoff = 1e-15
    keep_idx = findall(w_norm .>= cutoff)
    if isempty(keep_idx)
        keep_idx = [argmax(w_norm)]
    end
    ks = ks[keep_idx]
    w = w[keep_idx]
    s_w = sum(w)
    w_norm = w ./ s_w
    _mg6_cache[] = MGCacheV6(float(epsilon), float(Delta), Int(K), w, w_norm, ks .* Delta, s_w, 1/(sqrt(2*pi)*s_w), exp(epsilon))
end

function get_mg_cache_v6()
    c = _mg6_cache[]
    if c === nothing || c.epsilon != epsilon || c.Delta != Delta || c.K != K
        refresh_mg_cache_v6!()
        c = _mg6_cache[]
    end
    return c::MGCacheV6
end

function _epsilon_scaled_max_steps(base::Int, eps::Float64)
    # More epsilon → more concentration → fewer φ steps needed
    scaled = round(Int, base / (1 + eps))
    return clamp(scaled, 4000, base)
end

function _epsilon_scaled_tail(base::Float64, eps::Float64)
    # More epsilon → tighter mass near k=0 → shorter tails OK
    return max(6.0, base / (1 + 0.5*eps))
end

"Multi Gaussian distribution CDF"
function mG_cdf(x, sigma)
    c = get_mg_cache_v6()
    num = 0.0
    @inbounds @simd for j in eachindex(c.weights)
        num += c.weights[j] * cdf(std_Gauss, (x - c.mu[j]) / sigma)
    end
    return num / c.s_w
end

"sample from Multi Gaussian distribution"
function sample_mG(sigma; number = 1)
    c = get_mg_cache_v6()
    probabilities = Weights(c.weights_norm)
    k_idx = sample(1:length(c.mu), probabilities)
    return rand(std_Gauss, number) .* sigma .+ c.mu[k_idx]
end

"Multi Gaussian distribution PDF"
function mG_pdf(x, sigma)
    c = get_mg_cache_v6()
    c_norm = sqrt(2*pi) * sigma * c.s_w
    terms = 0.0
    inv2σ2 = 1/(2*sigma^2)
    @inbounds @simd for j in eachindex(c.weights)
        terms += c.weights[j] * exp(- (x - c.mu[j])^2 * inv2σ2)
    end
    return terms / c_norm
end

"Multi Gaussian distribution expected absolute value"
function mG_amplitude(sigma)
    c = get_mg_cache_v6()
    c_norm = sqrt(2*pi) * sigma * c.s_w
    term = 0.0
    @inbounds @simd for j in eachindex(c.weights)
        kΔ = c.mu[j]
        term += c.weights[j] * ((2*sigma^2*exp(- (kΔ^2)/(2*sigma^2))) + (kΔ*sqrt(2*pi)*sigma*(1 - 2*cdf(std_Gauss, -kΔ/sigma))))
    end
    return term / c_norm
end

"multi Gaussian distribution expected squared value"
function mG_power(sigma)
    c = get_mg_cache_v6()
    num = 0.0
    @inbounds @simd for j in eachindex(c.weights)
        kΔ = c.mu[j]
        num += c.weights[j] * (sigma^2 + kΔ^2)
    end
    return num / c.s_w
end

function integrand_v6(x, varphi, sigma, c)
    inv2σ2 = 1/(2*sigma^2)
    num1   = 0.0
    num2   = 0.0
    @inbounds @simd for j in eachindex(c.weights)
        w  = c.weights[j]
        μ  = c.mu[j]
        z1 = exp(-(x       - μ)^2 * inv2σ2)
        z2 = exp(-(x + varphi   - μ)^2 * inv2σ2)
        num1 += w*z1
        num2 += w*z2
    end
    d = c.exp_eps*num1 - num2
    return d < 0 ? d * (c.inv_n/sigma) : 0.0
end

# Tailored for small delta, large epsilon; epsilon-aware caps/tails.
function privacy_shortfall(sigma; eta_override = nothing, base_max_steps = 20000, base_tail_mult = 12.0, quad_atol = nothing, quad_rtol = 1e-6)
    c = get_mg_cache_v6()
    η = isnothing(eta_override) ? eta : eta_override
    beta_raw = sqrt(2*pi)*sigma*delta*η
    ideal_steps = ceil(Int, Delta / max(beta_raw, eps()))
    max_steps = _epsilon_scaled_max_steps(base_max_steps, epsilon)
    steps = clamp(ideal_steps, 400, max_steps)
    beta = Delta / steps
    min_S = Inf
    atol = isnothing(quad_atol) ? max(delta * 1e-3, 1e-12) : quad_atol
    tail_mult = _epsilon_scaled_tail(base_tail_mult, epsilon)
    for i in steps:-1:0
        varphi = i * beta
        L = tail_mult*sigma + c.K*Delta + varphi
        S = try
            quadgk(x -> integrand_v6(x, varphi, sigma, c), -L, L; atol = atol, rtol = quad_rtol)[1] + ((1 - η)*delta)
        catch
            return -1.0
        end
        if isnan(S) || S < 0
            return -1.0
        end
        min_S = min(min_S, S)
    end
    return min_S
end

"Main function with epsilon-aware caps/tails."
function mG_sigma(; eta_override = nothing, base_max_steps = 20000, base_tail_mult = 12.0, quad_atol = nothing, quad_rtol = 1e-6)
    refresh_mg_cache_v6!()
    η = isnothing(eta_override) ? eta : eta_override
    shortfall = σ -> privacy_shortfall(σ; eta_override = η, base_max_steps = base_max_steps, base_tail_mult = base_tail_mult, quad_atol = quad_atol, quad_rtol = quad_rtol)

    lower = 1.0
    while shortfall(lower) > 0
        lower /= 2
    end

    r = sqrt(2*log(1.25/((1 - η)*delta)))*Delta/epsilon
    attempts = 0
    while shortfall(r) < 0 && attempts < 30
        r *= 2
        attempts += 1
    end
    if shortfall(r) < 0
        return r
    end

    sigma = find_zero(shortfall, (lower, r), method = Brent(), xatol = 1e-6, atol = 1e-6)
    return sigma
end
