from __future__ import annotations

import argparse
from dataclasses import dataclass
from typing import Protocol

import numpy as np
Array = np.ndarray

def str2bool(v):
    if isinstance(v, bool):
        return v
    if v.lower() in ('yes', 'true', 'y', '1'):
        return True
    elif v.lower() in ('no', 'false', 'n', '0'):
        return False
    else:
        raise argparse.ArgumentTypeError('Boolean value expected.')

# ---- Oracles that you will supply -----
class GradEstimatorX(Protocol):
    """Return a stochastic gradient estimate for x"""
    def __call__(self, x: Array, y: Array, *, samples: Array | None) -> Array: ...

class GradEstimatorY(Protocol):
    """Return a stochastic gradient estimate for y"""
    def __call__(self, x: Array, y: Array, *, samples: Array | None) -> tuple[Array, Array]: ...

class ConstraintFn(Protocol):
    """Return constraint value c(x, y)."""
    def __call__(self, x: Array, y: Array) -> Array: ...

class Projection(Protocol):
    """Projection onto feasible set."""
    def __call__(self, v: Array) -> Array: ...


# ---- Algorithm configuration ----------
@dataclass(slots=True)
class BasicAlgoConfig:
    """Basic configuration for algorithms solving the toy example."""
    max_iters: int = 1000
    log_every: int = 100
    seed: int | None = 42
