import numpy as np
from typing import Optional, Union
from scipy.stats import norm, binom, beta
from scipy.optimize import brentq

def regression_function(x1: np.ndarray, x2: np.ndarray) -> np.ndarray:
    return np.sin(2 * x1) * 10 + 2 * (x2 - 2) ** 3

def invert_binomial_tail(alpha: float, n: int, P: float) -> float:
    k = int(np.round(alpha * n))
    def f(q: float) -> float:
        return binom.cdf(k, n, 1.0 - q) - P
    return float(brentq(f, 0.0, 1.0))

def radius_from_probabilities(p_x: np.ndarray, q: float, sigma: float) -> np.ndarray:
    return sigma * (norm.ppf(np.clip(p_x, 1e-9, 1 - 1e-9)) - norm.ppf(np.clip(q, 1e-9, 1 - 1e-9)))

def clopper_pearson_lower(k: int, n: int, delta: float) -> float:
    if k <= 0:
        return 0.0
    return float(beta.ppf(delta, k, n - k + 1))

def pA_lower_bound(x: np.ndarray,
                   reg_fn,
                   sigma: float,
                   epsilon_y: float,
                   n: int,
                   delta: float,
                   rng: Optional[np.random.Generator] = None) -> float:
    rng = np.random.default_rng() if rng is None else rng
    fx = float(reg_fn(x[0], x[1]))
    k = 0
    for _ in range(n):
        e = rng.normal(0.0, sigma, size=2)
        fxe = float(reg_fn(x[0] + e[0], x[1] + e[1]))
        if abs(fx - fxe) <= epsilon_y:
            k += 1
    return clopper_pearson_lower(k, n, delta)
