"""
Smooth Noise Distribution for Dynamic Pricing

Implements the noise distribution from Fan-Guo-Yu 2022:
    f_m(z) ∝ (1/4 - z²)^(m/2) for |z| ≤ 1/2

Uses Beta distribution for efficient sampling.
"""

import numpy as np
from scipy.special import beta as beta_function
from scipy.integrate import quad


class SmoothNoise:
    """
    Smooth noise distribution with m-th order smoothness.
    
    The density is:
        f_m(z) = C_m * (1/4 - z²)^(m/2) * I{|z| ≤ 1/2}
    
    where C_m = 1/B(m/2+1, m/2+1) is the normalization constant.
    
    Sampling: U ~ Beta(m/2+1, m/2+1), then z = U - 1/2
    """
    
    def __init__(self, m):
        """
        Parameters:
        -----------
        m : int
            Smoothness order (2, 4, or 6)
        """
        if m not in [2, 4, 6]:
            raise ValueError("m must be 2, 4, or 6")
        
        self.m = m
        self.alpha = m / 2 + 1  # Beta distribution parameter
        self.delta_z = 0.5  # Support is [-0.5, 0.5]
        
        # Normalization constant
        self.C_m = 1.0 / beta_function(self.alpha, self.alpha)
    
    def pdf(self, z):
        """
        Probability density function.
        
        Parameters:
        -----------
        z : float or np.ndarray
            Point(s) to evaluate
        
        Returns:
        --------
        float or np.ndarray
            Density value(s)
        """
        z = np.asarray(z)
        result = np.zeros_like(z, dtype=float)
        
        # Only non-zero for |z| ≤ 1/2
        mask = np.abs(z) <= self.delta_z
        if np.any(mask):
            z_valid = z[mask]
            result[mask] = self.C_m * ((0.25 - z_valid**2) ** (self.m / 2))
        
        return result if z.shape else float(result)
    
    def cdf(self, z):
        """
        Cumulative distribution function (numerical integration).
        
        Parameters:
        -----------
        z : float or np.ndarray
            Point(s) to evaluate
        
        Returns:
        --------
        float or np.ndarray
            CDF value(s)
        """
        z = np.asarray(z)
        
        if z.shape:
            return np.array([self._cdf_scalar(zi) for zi in z])
        else:
            return self._cdf_scalar(float(z))
    
    def _cdf_scalar(self, z):
        """Helper for scalar CDF computation."""
        if z <= -self.delta_z:
            return 0.0
        elif z >= self.delta_z:
            return 1.0
        else:
            result, _ = quad(self.pdf, -self.delta_z, z)
            return result
    
    def sample(self, size=1):
        """
        Sample from the distribution.
        
        Uses the Beta distribution trick:
            U ~ Beta(m/2+1, m/2+1)
            z = U - 1/2
        
        Parameters:
        -----------
        size : int or tuple
            Number of samples
        
        Returns:
        --------
        np.ndarray
            Samples from the distribution
        """
        U = np.random.beta(self.alpha, self.alpha, size=size)
        return U - 0.5
    
    def mean(self):
        """Expected value (should be 0 by symmetry)."""
        return 0.0
    
    def variance(self):
        """
        Variance of the distribution.
        
        For Beta(α, α), Var(U) = α/(2(2α+1))
        For z = U - 1/2, Var(z) = Var(U)
        """
        alpha = self.alpha
        return alpha / (4 * alpha * (2 * alpha + 1))


def test_noise_distribution():
    """Test the noise distribution implementation."""
    print("="*60)
    print("Testing Smooth Noise Distribution")
    print("="*60)
    
    for m in [2, 4, 6]:
        print(f"\n--- m = {m} ---")
        noise = SmoothNoise(m)
        
        # Test sampling
        samples = noise.sample(size=10000)
        
        print(f"Samples shape: {samples.shape}")
        print(f"Sample mean: {samples.mean():.6f} (expected ~0)")
        print(f"Sample std: {samples.std():.6f}")
        print(f"Theoretical std: {np.sqrt(noise.variance()):.6f}")
        print(f"Sample range: [{samples.min():.4f}, {samples.max():.4f}]")
        print(f"Expected range: [-0.5, 0.5]")
        
        # Test PDF
        z_test = np.array([-0.6, -0.5, -0.25, 0.0, 0.25, 0.5, 0.6])
        pdf_vals = noise.pdf(z_test)
        print(f"\nPDF values:")
        for z, p in zip(z_test, pdf_vals):
            print(f"  pdf({z:5.2f}) = {p:.6f}")
        
        # Test CDF
        cdf_vals = noise.cdf(z_test)
        print(f"\nCDF values:")
        for z, F in zip(z_test, cdf_vals):
            print(f"  cdf({z:5.2f}) = {F:.6f}")
        
        # Verify CDF(0.5) ≈ 1
        assert abs(noise.cdf(0.5) - 1.0) < 0.01, "CDF at upper bound should be ~1"
        
        # Verify mean close to 0
        assert abs(samples.mean()) < 0.01, f"Sample mean should be ~0, got {samples.mean()}"
        
        print(f"\n✓ All tests passed for m={m}")


if __name__ == "__main__":
    test_noise_distribution()
