import numpy as np


def gaussian_mechanism_rdp_variance(epsilon: float, alpha: float, sensitivity: float) -> float:
    """Computes the noise variance for the Gaussian mechanism under RDP.

    Args:
        epsilon (float): Must be positive.
        alpha (float): Must be > 1
        sensitivity (float): Must be positive.

    Returns:
        float: Required noise variance.
    """
    if alpha <= 1:
        raise ValueError(f"Alpha must be greater than 1. Got alpha={alpha}.")
    if epsilon <= 0:
        raise ValueError(f"Epsilon must be positive. Got epsilon={epsilon}.")
    if sensitivity <= 0:
        raise ValueError(f"Sensitivity must positive. Got sensitivity={sensitivity}.")

    return alpha * sensitivity**2 / (2 * epsilon)


def gaussian_mechanism_rdp_epsilon( 
        sigma: float, alpha: float, sensitivity: float
        ) -> float:
    """Computes the epsilon for the Gaussian mechanism under RDP.

    Args:
        sigma (float): Standard deviation of the Gaussian noise. Must be positive.
        alpha (float): Must be > 1
        sensitivity (float): Must be positive.

    Returns:
        float: The resulting epsilon.
    """
    if alpha <= 1:
        raise ValueError(f"Alpha must be greater than 1. Got alpha={alpha}.")
    if sigma <= 0:
        raise ValueError(f"Sigma must be positive. Got sigma={sigma}.")
    if sensitivity <= 0:
        raise ValueError(f"Sensitivity must positive. Got sensitivity={sensitivity}.")
    return alpha * sensitivity**2 / (2 * sigma**2)


def gaussian_mechanism_rdp(
        true_value: np.ndarray | float, epsilon: float, alpha: float, 
        sensitivity: float
        ) -> np.ndarray | float:
    """Releases a private value with the Gaussian mechanism with RDP guarantee.

    Args:
        true_value (np.ndarray | float): Private value to release.
        epsilon (float): Must be positive.
        alpha (float): Must be > 1
        sensitivity (float): Must be positive.

    Returns:
        np.ndarray | float: Released noisy value.
    """

    variance = gaussian_mechanism_rdp_variance(epsilon, alpha, sensitivity)

    noise = np.random.normal(0, np.sqrt(variance), size=np.shape(true_value))
    return true_value + noise


def laplace_mechanism_scale(epsilon: float, sensitivity: float) -> float:
    """Computes scale parameter for pure-DP Laplace mechanism.

    Args:
        epsilon (float): Must be positive.
        sensitivity (float): Must be positive.

    Returns:
        float: Required Laplace distribution scale parameter.
    """
    if epsilon <= 0:
        raise ValueError(f"Epsilon must be positive. Got epsilon={epsilon}.")
    if sensitivity <= 0:
        raise ValueError(f"Sensitivity must positive. Got sensitivity={sensitivity}.")

    return sensitivity / epsilon


def laplace_mechanism_variance(epsilon: float, sensitivity: float) -> float:
    """Computes noise variance for pure-DP Laplace mechanism.

    Args:
        epsilon (float): Must be positive.
        sensitivity (float): Must be positive.

    Returns:
        float: Required noise variance.
    """
    scale = laplace_mechanism_scale(epsilon, sensitivity)
    return 2 * scale**2


def laplace_mechanism(
        true_value: np.ndarray | float, epsilon: float, sensitivity: float
        ) -> np.ndarray | float:
    """Releases private value with the Laplace mechanism with pure-DP guarantee.

    Args:
        true_value (np.ndarray | float): Private value to release.
        epsilon (float): Must be positive.
        sensitivity (float): Must be positive.

    Returns:
        np.ndarray | float: Released noisy value.
    """

    scale = laplace_mechanism_scale(epsilon, sensitivity)

    noise = np.random.laplace(0, scale, size=np.shape(true_value))
    return true_value + noise