import numpy as np

def _pairwise_sq_dists(A, B):

    A2 = np.sum(A*A, axis=1)[:, None]
    B2 = np.sum(B*B, axis=1)[None, :]
    return A2 + B2 - 2 * (A @ B.T)

def _pairwise_dists(A, B):
    return np.sqrt(np.maximum(_pairwise_sq_dists(A, B), 0.0))

def rbf_kernel(A, B, lengthscale):
    D2 = _pairwise_sq_dists(A, B)
    return np.exp(-D2 / (2.0 * lengthscale**2))

def laplace_kernel(A, B, lengthscale):
    D = _pairwise_dists(A, B)
    return np.exp(-D / lengthscale)

def linear_kernel(A, B):
    return A @ B.T

def polynomial_kernel(A, B, degree=2, gamma=None, coef0=1.0):
    if gamma is None:  
        gamma = 1.0 / A.shape[1]
    return (gamma * (A @ B.T) + coef0) ** degree

def rational_quadratic_kernel(A, B, alpha=1.0, lengthscale=1.0):
    D2 = _pairwise_sq_dists(A, B)
    return (1.0 + 0.5 * D2 / (alpha * lengthscale**2)) ** (-alpha)

def matern_kernel(A, B, nu=1.5, lengthscale=1.0):
    D = _pairwise_dists(A, B)
    r = np.sqrt(2.0 * nu) * D / lengthscale
    if nu == 0.5:       
        K = np.exp(-r)
    elif nu == 1.5:   
        K = (1.0 + r) * np.exp(-r)
    elif nu == 2.5:     
        K = (1.0 + r + r*r/3.0) * np.exp(-r)
    else:
        raise ValueError("Supported nu in {0.5, 1.5, 2.5}")
    return K

def cauchy_kernel(A, B, gamma=1.0):
    D2 = _pairwise_sq_dists(A, B)
    return 1.0 / (1.0 + D2 / (gamma**2))

def cosine_kernel(A, B):

    An = A / (np.linalg.norm(A, axis=1, keepdims=True) + 1e-12)
    Bn = B / (np.linalg.norm(B, axis=1, keepdims=True) + 1e-12)
    return An @ Bn.T
