from __future__ import annotations
import numpy as np

def cosine(a, b) -> float:
    a = a.reshape(-1)
    b = b.reshape(-1)
    na = np.linalg.norm(a) + 1e-12
    nb = np.linalg.norm(b) + 1e-12
    return float((a @ b) / (na * nb))

def rel_error(a, b) -> float:
    num = np.linalg.norm(a - b)
    den = np.linalg.norm(b) + 1e-12
    return float(num / den)

def hv_residual(hvp, v, g, damping: float) -> float:
    r = hvp(v) + damping * v - g
    num = float(np.linalg.norm(r))
    den = float(np.linalg.norm(g) + 1e-12)
    return num / den

def cv_loss_delta(*args, **kwargs):  # pragma: no cover - placeholder
    raise NotImplementedError


# -----------------------------
# Correlation helpers
# -----------------------------

def _rankdata(x: np.ndarray) -> np.ndarray:
    """Return average ranks for array x (1-based ranks like SciPy).

    Ties receive the average of the ranks they span.
    """
    x = np.asarray(x)
    n = x.size
    sorter = np.argsort(x, kind="mergesort")
    inv = np.empty(n, dtype=int)
    inv[sorter] = np.arange(n)
    x_sorted = x[sorter]
    # Find tie groups
    diffs = np.r_[True, x_sorted[1:] != x_sorted[:-1]]
    group_starts = np.flatnonzero(diffs)
    group_ends = np.r_[group_starts[1:], n]
    ranks = np.empty(n, dtype=float)
    for s, e in zip(group_starts, group_ends):
        # ranks are 1-based
        avg_rank = 0.5 * (s + 1 + e)
        ranks[s:e] = avg_rank
    return ranks[inv]


def pearson_corr(a: np.ndarray, b: np.ndarray) -> float:
    a = np.asarray(a, dtype=float).reshape(-1)
    b = np.asarray(b, dtype=float).reshape(-1)
    if a.size == 0 or b.size == 0 or a.size != b.size:
        return float("nan")
    if np.allclose(a, a[0]) or np.allclose(b, b[0]):
        return float("nan")
    c = np.corrcoef(a, b)
    return float(c[0, 1])


def spearman_corr(a: np.ndarray, b: np.ndarray) -> float:
    a = np.asarray(a, dtype=float).reshape(-1)
    b = np.asarray(b, dtype=float).reshape(-1)
    if a.size == 0 or b.size == 0 or a.size != b.size:
        return float("nan")
    ra = _rankdata(a)
    rb = _rankdata(b)
    if np.allclose(ra, ra[0]) or np.allclose(rb, rb[0]):
        return float("nan")
    c = np.corrcoef(ra, rb)
    return float(c[0, 1])
