import numpy as np

import numpy as np
from typing import Callable


def _m_normed(N: int, K: int, i: int, j: int) -> float:
    if i == j and i >= K - 1:
        return (
            K / (N - K + 1)
            * np.prod(np.arange(i - K + 2, i + 1) / np.arange(N - K + 2, N + 1))
        )
    elif j > i and j >= K - 1 and K >= 2:
        return (
            K / (N - K + 1)
            * (K - 1) / N
            * np.prod(np.arange(j - K + 2, j) / np.arange(N - K + 2, N))
        )
    return 0


def _m_diagonal(N: int, K: int) -> np.ndarray:
    return np.array([_m_normed(N, K, i, i) for i in range(N)])


def rho(g: np.ndarray, K: int) -> float:
    """See Equation (3)."""
    return (np.sort(g) * _m_diagonal(len(g), K)).sum()


def _delta(N: int, K: int, i: int) -> float:
    return _m_normed(N, K, i, i + 1) - _m_normed(N, K, i + 1, i + 1)


def _deltas(N: int, K: int) -> np.ndarray:
    return np.array([_delta(N - 1, K, i) for i in range(N - 2)])


def _sorted_apply(func: Callable) -> Callable:
    def inner(x: np.ndarray, *args, **kwargs) -> np.ndarray:
        i_sort = np.argsort(x)
        func_x = np.zeros_like(x)
        func_x[i_sort] = func(x[i_sort], *args, **kwargs)
        return func_x

    return inner


@_sorted_apply
def s(g: np.ndarray, K: int) -> np.ndarray:
    """See Equation (19)."""
    N = len(g)
    c = g * _m_diagonal(N, K)
    c[: N - 1] += g[1:] * _deltas(N + 1, K)
    return np.cumsum(c[::-1])[::-1]


@_sorted_apply
def _b(g: np.ndarray, K: int) -> np.ndarray:
    N = len(g)
    w = (_m_diagonal(N - 1, K) * np.arange(1, N)).astype(float)
    w[1:] += _deltas(N, K) * np.arange(1, N - 1)
    c1 = np.array([(w * g[1:]).sum()])
    c2 = (g[:-1] - g[1:]) * w
    return np.cumsum(np.concatenate((c1, c2)))


def sloo(g: np.ndarray, K: int) -> np.ndarray:
    """See Equation (29)."""
    return s(g, K) - _b(g, K) / (len(g) - 1)


def sloo_minus_one(g: np.ndarray, K: int) -> np.ndarray:
    """See Equation (33)."""
    return s(g, K) - _b(g, K - 1) * K / (K - 1) / len(g)