import logging
import numpy as np
import gc
import time
from OAEGN_analysis.spherical_symmetric_sample_generator import SphericalGaussianSampleGenerator, SphericalUniformSampleGenerator, SphericalExponentialSampleGenerator, SphericalGammaSampleGenerator
from OAEGN_analysis.privacy_estimator import _spherical_symmetric_DP_estimator


def generate_demo_configuration(noise_budget = 100, dim = 5, dof = 1, claimed_epsilon = 0.0, error_metric = 'l2'):
    query_1 = np.zeros(int(dim))    
    query_2 = np.zeros(int(dim))
    query_2[0] = 1  # Set the difference to 1

    dataset_settings = {
        'database_0': query_1.tolist(),
        'database_1': query_2.tolist(),
        'claimed_epsilon': claimed_epsilon,
    }

    mechanism_settings = {
        'sigma': np.sqrt(noise_budget/dof),
        'k': dof,
        'noise_budget': noise_budget,
    }

    kwargs = {
        'dataset_settings': dataset_settings, 
        'mechanism_settings': mechanism_settings,
        'error_metric': error_metric
    }
    return kwargs


def demo_elliptical_gaussian_DP_estimator(epsilon = 0.1, noise_budget = 10, dim = 5, dof = 1, sample_size = 2**16, error_metric = 'l2'):
    kwargs = generate_demo_configuration(noise_budget = noise_budget, dim = dim, dof = dof, claimed_epsilon = epsilon, error_metric = error_metric)

    return _spherical_symmetric_DP_estimator(sample_generator_cls=SphericalGaussianSampleGenerator, kwargs=kwargs, sample_size=sample_size)

def demo_spherical_gaussian_std_estimator(noise_budget = 100, dim = 5, dof = 1, sample_size = 2**16, error_metric = 'l2'):
    return demo_elliptical_gaussian_DP_estimator(epsilon = 0, noise_budget = noise_budget, dim = dim, dof = dof, sample_size = sample_size, error_metric = error_metric)

def spherical_exponential_std_estimator(noise_budget = 100, dim = 5, sample_size = 2**16, error_metric = 'l2'):
    kwargs = generate_demo_configuration(noise_budget = noise_budget, dim = dim, dof = 1, claimed_epsilon = 0, error_metric = error_metric)
    return _spherical_symmetric_DP_estimator(sample_generator_cls=SphericalExponentialSampleGenerator, kwargs=kwargs, sample_size=sample_size)

def demo_spherical_uniform_DP_estimator(epsilon = 0.1, noise_budget = 10, dim = 5, sample_size = 2**16, error_metric = 'l2'):
    kwargs = generate_demo_configuration(noise_budget = noise_budget, dim = dim, dof = 1, claimed_epsilon = epsilon, error_metric = error_metric)

    return _spherical_symmetric_DP_estimator(sample_generator_cls=SphericalUniformSampleGenerator, kwargs=kwargs, sample_size=sample_size)

def demo_spherical_gamma_std_estimator(noise_budget = 10, dim = 5, shape = 1, sample_size = 2**16, error_metric = 'l2'):
    kwargs = generate_demo_configuration(noise_budget = noise_budget, dim = dim, dof = shape, claimed_epsilon = 0, error_metric = error_metric)
    return _spherical_symmetric_DP_estimator(sample_generator_cls=SphericalGammaSampleGenerator, kwargs=kwargs, sample_size=sample_size)
    