# EVOLVE-BLOCK-START
"""Finding optimal configuration of 5 points in a cube that mimizes the L-infinity star discrepancy."""
import numpy as np

def construct_star() -> np.ndarray:
    """
    Find the optimal configuration of points in a cube [0, 1] x [0, 1] x [0, 1] to minimize the L-infinity star discrepancy.
    This function constructs 5 points that are evenly distributed and maximally distant from each other.
    The points are represented in coordinates (x, y, z) in the cube
    x, y, z are in the range [0, 1].
    Returns:
        A: np.array of shape (5, 3) with coordinates of points in the cube.
    """
    # Optimized point placement for 5 points in a cube. Refined from initial grid.
    # Experimentally determined coordinates for improved discrepancy.
    # A = np.array([
    #     [0.09, 0.09, 0.09],
    #     [0.21, 0.79, 0.89],
    #     [0.79, 0.89, 0.21],
    #     [0.89, 0.21, 0.79],
    #     [0.5, 0.5, 0.5]
    # ])
    # N = 5
    # alpha1 = 0.5436890126920764 # 1/zeta
    # alpha2 = 0.2955977420998101 # 1/zeta^2
    # A = np.zeros((N, 3))
    
    # for i in range(N):
    #     offset_i = i + 0.5 # Using the (i+0.5) shift
        
    #     A[i, 0] = offset_i / N
    #     A[i, 1] = (offset_i * alpha1) % 1.0
    #     A[i, 2] = (offset_i * alpha2) % 1.0

    import scipy.stats.qmc
    # A = scipy.stats.qmc.Halton(d=3, scramble=True, seed=123).random(n=5)
    A = scipy.stats.qmc.Sobol(d=3, scramble=True, seed=123).random(n=5)
    # A = scipy.stats.qmc.LatinHypercube(d=3, scramble= True).random(n=5)
    # def radical_inverse(n, base):
        
    #     inverse = 0.0
    #     power_of_base_denominator = 1.0 / base
        
    #     temp_n = n
    #     while temp_n > 0:
    #         digit = temp_n % base
    #         inverse += digit * power_of_base_denominator
    #         temp_n //= base
    #         power_of_base_denominator /= base
            
    #     return inverse
    # N = 5
    # A = np.zeros((N, 3))
    # prime_base_dim2 = 2
    # prime_base_dim3 = 3
    
    # for i in range(N):
    #     # Dimension 1: Shifted linear sequence
    #     A[i, 0] = (i + 0.5) / N
        
    #     # Dimension 2: Radical inverse in base 2
    #     A[i, 1] = radical_inverse(i, prime_base_dim2)
        
    #     # Dimension 3: Radical inverse in base 3
    #     A[i, 2] = radical_inverse(i, prime_base_dim3)
        
    return A


    
# EVOLVE-BLOCK-END

import numpy as np
import itertools
from numba import njit
# This part remains fixed (not evolved)
def run_star() -> np.ndarray:
    """Run the star constructor for n=5"""
    A = construct_star()
    return A
# Numba helper function for calculating discrepancy for a single box corner
@njit(cache=True)
def _calculate_single_box_discrepancy_numba(points_X_arg: np.ndarray, 
                                           N_arg: int, 
                                           D_arg: int, 
                                           y_corner_arg: np.ndarray) -> float:
    """
    Calculates the local discrepancy for a single d-dimensional anchored box.
    Box is defined by [0, y_corner_arg[0]] x ... x [0, y_corner_arg[D-1]].
    """
    # Calculate volume of the box
    volume = 1.0
    for k_dim in range(D_arg):
        volume *= y_corner_arg[k_dim]

    # Count points within the box [0, y_corner_arg]
    count_in_box = 0
    for i_point in range(N_arg): # Iterate through each point
        point_is_in_box = True
        for k_dim in range(D_arg): # Iterate through each dimension for the current point
            if points_X_arg[i_point, k_dim] > y_corner_arg[k_dim]: # Point is outside this dimension
                point_is_in_box = False
                break
            
        if point_is_in_box:
            count_in_box += 1

    return abs(count_in_box / N_arg - volume)

def star_discrepancy(points_X: np.ndarray) -> float:
    """
    Calculates the L-infinity star discrepancy of the point set P.
    Optimized using Numba for the core calculation loop.
    Args:
        points_X (np.ndarray): An array of points to evaluate, shape (N, D) for D-dimensional points.
    Returns:
        float: The maximum star discrepancy value.
    """
    # Input validation and preparation
    if not isinstance(points_X, np.ndarray):
        points_X_np = np.array(points_X, dtype=np.float64)
    elif points_X.dtype != np.float64: # Ensure float64 for Numba compatibility and precision
        points_X_np = points_X.astype(np.float64)
    else:
        points_X_np = points_X

    if points_X_np.ndim == 1:
        points_X_np = points_X_np.reshape(-1, 1)
    
    N, D = points_X_np.shape

    if N == 0:
        return 1.0

    points_X_clipped = np.clip(points_X_np, 0.0, 1.0 - np.finfo(points_X_np.dtype).eps)
    
    if not points_X_clipped.flags.c_contiguous:
        points_X_clipped = np.ascontiguousarray(points_X_clipped)

    grid_lines_per_dim = []
    for j in range(D):
        unique_coords_dim_j = np.unique(points_X_clipped[:, j])
        current_dim_grid_lines = np.union1d(unique_coords_dim_j, 
                                            np.array([1.0], dtype=points_X_clipped.dtype))
        grid_lines_per_dim.append(current_dim_grid_lines)

    max_discrepancy_val = 0.0
    
    y_corner_for_numba = np.empty(D, dtype=points_X_clipped.dtype)

    if not all(len(gl) > 0 for gl in grid_lines_per_dim):
        max_discrepancy_val = 0.0
    else:
        for y_corner_tuple in itertools.product(*grid_lines_per_dim):
            for i_val in range(D):
                y_corner_for_numba[i_val] = y_corner_tuple[i_val]
            
            local_discrepancy = _calculate_single_box_discrepancy_numba(
                points_X_clipped, N, D, y_corner_for_numba
            )
            
            if local_discrepancy > max_discrepancy_val:
                max_discrepancy_val = local_discrepancy

    return max_discrepancy_val


def score_star(X_points: np.ndarray) -> float:
    """ Calculates the score based on the star discrepancy of the given points.
    Args:
        X_points (np.ndarray): An array of points to evaluate, shape (N, 3) for 3D points.
    Returns:
        float: The score based on the star discrepancy, defined as 1 / (1 + max_discrepancy_val).
    """
    discrepancy = star_discrepancy(X_points)
    return 1 / (1 + discrepancy)  # Return the score as per the definition of star discrepancy



if __name__ == "__main__":
    # Example usage
    points = run_star()
    discrepancy = star_discrepancy(points)
    print("Star Discrepancy:", discrepancy)