import numpy as np
import math
from typing import Dict
import multiprocessing as mp
from functools import partial


def get_target_depth(X: np.ndarray, target: float) -> float:
    """
    Compute the depth of a target value in a 1D array X.

    Args:
        X (np.ndarray): A 1D array of numerical values.
        target (float): The target value whose depth is to be computed.

    Returns:
        float: The depth of the target value.

    Raises:
        AssertionError: If the target is not in X or if X is not 1-dimensional.
    """
    assert target in X, "target not in X"
    assert len(X.shape) == 1, "X must be 1D"

    if len(set(X)) == 1:
        return 0.0
    if len(set(X)) == 2:
        return 1.0

    depth = 0.0
    t = np.where(X == target)[0][0]

    # Compute depth for the left part
    for i in range(1, t + 1):
        depth += (X[i] - X[i - 1]) / (X[t] - X[i - 1])

    # Compute depth for the right part
    for i in range(t + 1, len(X)):
        depth += (X[i] - X[i - 1]) / (X[i] - X[t])

    return depth


def _compute_depth_for_point(t: int, sorted_X: np.ndarray) -> float:
    """
    Helper function to compute depth for a single point in the sorted array.
    
    Args:
        t (int): Index of the point in the sorted array.
        sorted_X (np.ndarray): The sorted array.
        
    Returns:
        float: The depth of the point at index t.
    """
    depth = 0.0
    
    # Compute depth for the left part
    for i in range(1, t + 1):
        depth += (sorted_X[i] - sorted_X[i - 1]) / (sorted_X[t] - sorted_X[i - 1])
    
    # Compute depth for the right part
    for i in range(t + 1, len(sorted_X)):
        depth += (sorted_X[i] - sorted_X[i - 1]) / (sorted_X[i] - sorted_X[t])
    
    return depth


def get_depths(X: np.ndarray) -> np.ndarray:
    """
    Compute the depths of all elements in a 1D array X using multiprocessing.

    Args:
        X (np.ndarray): A 1D array of numerical values.

    Returns:
        np.ndarray: An array of depths corresponding to the elements in X.
    """
    if len(np.unique(X)) < len(X):
        raise ValueError("X must have unique values")
    
    sorted_X = np.sort(X)
    
    # Create a mapping from values to their indices in the sorted array
    value_to_idx = {val: i for i, val in enumerate(sorted_X)}
    
    # Use multiprocessing to compute depths in parallel
    with mp.Pool() as pool:
        depths_sorted = pool.map(partial(_compute_depth_for_point, sorted_X=sorted_X), 
                                range(len(sorted_X)))
    
    # Map each value in the original array to its depth
    return np.array([depths_sorted[value_to_idx[x]] for x in X])


def get_avg_depth(X: np.ndarray) -> np.ndarray:
    """
    Compute the average depth of each row across all columns in a 2D array X.

    Args:
        X (np.ndarray): A 2D array of shape (n_samples, n_features).

    Returns:
        np.ndarray: A 1D array containing the average depth of each row.
    """
    n_samples, n_features = X.shape
    all_depths = np.zeros((n_features, n_samples))
    
    for i in range(n_features):
        all_depths[i] = get_depths(X[:, i])
    
    return np.mean(all_depths, axis=0)


def main() -> None:
    """
    Main function to demonstrate depth calculations on a generated dataset.
    """
    X = np.array(
        [
            (math.cos(2 * math.pi * theta), math.sin(2 * math.pi * theta))
            for theta in np.random.random(200)
        ]
        + [(0, 0)]
    )

    print(f"avg_depth = {list(get_avg_depth(X))}")


if __name__ == "__main__":
    main()
