import numpy as np
import torch

def analyze_function_variance(func, bounds, num_samples=1000, dim=None):
    """
    Analyze the variance of a function within given bounds and optionally at a specific dimension.
    
    Args:
        func: The function to analyze
        bounds: List/tuple of [lower_bound, upper_bound]
        num_samples: Number of random samples to use for variance calculation
        dim: Optional dimension to analyze. If None, analyzes all dimensions.
    
    Returns:
        dict: Dictionary containing variance information
    """
    import torch
    import numpy as np
    
    # Convert bounds to tensor
    bounds = bounds.to("cpu")
    
    # Generate random samples within bounds
    samples = torch.rand(num_samples, dim) * (bounds[1] - bounds[0]) + bounds[0]
    
    # Evaluate function at all samples
    values = torch.tensor([func(sample) for sample in samples])
    
    # Calculate statistics
    variance = torch.var(values)
    
    return variance

def add_noise(func, noise_level=0.01, bounds=[0.0, 1.0], num_samples=1000, dim=4):
    """
    Add adaptive noise to a function based on its variance within bounds.
    
    Args:
        func: The function to add noise to
        noise_level: Base noise level (fraction of function variance)
        bounds: List/tuple of [lower_bound, upper_bound]
        num_samples: Number of samples to use for variance analysis
    
    Returns:
        function: A noisy version of the input function
    """
    def noisy_func(x, bounds=bounds):
        # Analyze function variance
        variance = analyze_function_variance(func, bounds, num_samples, dim)
        
        # Calculate noise scale based on function variance
        noise_scale = noise_level * torch.sqrt(variance).to("cuda")
        
        # Generate noise
        noise = torch.randn(1).to("cuda") * noise_scale
        
        # Apply noise to function output
        return func(x) + noise
    
    return noisy_func

def schwefel(x):
    return -(418.9829 * 4 - torch.sum(x * torch.sin(torch.sqrt(torch.abs(x))))) #-500 to 500 bounds

def eggholder(x):
    return -(-(x[1] + 47) * torch.sin(torch.sqrt(torch.abs(x[0] / 2 + (x[1] + 47)))) - x[0] * torch.sin(torch.sqrt(torch.abs(x[0] - (x[1] + 47)))))

def ackley(x):
    a = 20
    b = 0.2
    c = 2 * np.pi
    d = len(x)
    sum1 = torch.sum(x**2)
    sum2 = torch.sum(torch.cos(c * x))
    term1 = -a * torch.exp(-b * torch.sqrt(sum1 / d))
    term2 = -torch.exp(sum2 / d)
    return -(term1 + term2 + a + torch.exp(torch.tensor(1)))

def shekel(x, m=10):
    x = x.to("cuda")
    a = torch.tensor([
        [4.0, 4.0, 4.0, 4.0],
        [1.0, 1.0, 1.0, 1.0],
        [8.0, 8.0, 8.0, 8.0],
        [6.0, 6.0, 6.0, 6.0],
        [3.0, 7.0, 3.0, 7.0],
        [2.0, 9.0, 2.0, 9.0],
        [5.0, 5.0, 3.0, 3.0],
        [8.0, 1.0, 8.0, 1.0],
        [6.0, 2.0, 6.0, 2.0],
        [7.0, 3.6, 7.0, 3.6]
    ], device="cuda")
    c = torch.tensor([0.1, 0.2, 0.2, 0.4, 0.4, 0.6, 0.3, 0.7, 0.5, 0.5], device="cuda")
    sum = 0
    for i in range(m):
        sum += 1 / (torch.dot(x - a[i], x - a[i]) + c[i])
    return -(-sum)

def griewank(x):
    x = x.to("cuda")
    sum_term = torch.sum(x**2) / 4000
    prod_term = torch.prod(torch.cos(x / torch.sqrt(torch.arange(1, len(x) + 1, device="cuda"))))
    return -(sum_term - prod_term + 1)

def hartmann_3(x):
    x = x.to("cuda")
    alpha = torch.tensor([1.0, 1.2, 3.0, 3.2], device="cuda")
    A = torch.tensor([
        [3.0, 10, 30],
        [0.1, 10, 35],
        [3.0, 10, 30],
        [0.1, 10, 35]
    ], device="cuda")
    P = 10**-4 * torch.tensor([
        [3689, 1170, 2673],
        [4699, 4387, 7470],
        [1091, 8732, 5547],
        [381, 5743, 8828]
    ], device="cuda")
    sum = 0
    for i in range(4):
        sum += -alpha[i] * torch.exp(-torch.sum(A[i] * (x - P[i])**2))
    return -(sum)

def hartmann_6(x):
    x = x.to("cuda")
    alpha = torch.tensor([1.0, 1.2, 3.0, 3.2])
    A = torch.tensor([
        [10, 3, 17, 3.5, 1.7, 8],
        [0.05, 10, 17, 0.1, 8, 14],
        [3, 3.5, 1.7, 10, 17, 8],
        [17, 8, 0.05, 10, 0.1, 14]
    ], device="cuda")
    P = 10**-4 * torch.tensor([
        [1312, 1696, 5569, 124, 8283, 5886],
        [2329, 4135, 8307, 3736, 1004, 9991],
        [2348, 1451, 3522, 2883, 3047, 6650],
        [4047, 8828, 8732, 5743, 1091, 381]
    ], device="cuda")
    sum = 0
    for i in range(4):
        sum += alpha[i] * torch.exp(-torch.sum(A[i] * (x - P[i])**2))
    return -(-sum)

def powell(x):
    n = len(x)
    sum = 0
    for i in range(0, n, 4):
        term1 = (x[i] + 10 * x[i+1])**2
        term2 = 5 * (x[i+2] - x[i+3])**2
        term3 = (x[i+1] - 2 * x[i+2])**4
        term4 = 10 * (x[i] - x[i+3])**4
        sum += term1 + term2 + term3 + term4
    return -(sum)