import math
import numpy as np
from typing import List

def relu(array):
    """
    Apply the ReLU activation function element-wise to the input array.
    """
    return np.maximum(0, array)

def sign(array):
    """
    Apply the ReLU activation function element-wise to the input array.
    """
    return np.where(array > 0, 1.0, -1.0)

class BasePredictor:
    def eval(self, W, X) -> np.ndarray:
        """
        Return the n_examples x n_parameters array of the
        base predictor on every pair (example, parameter).
        """
        raise NotImplementedError
    
    def max_output(self) -> float:
        raise NotImplementedError

class Sign(BasePredictor):
    def eval(self, W, X) -> np.ndarray:
        W = np.array(W)
        return sign(X @ W.T)
    
    def max_output(self) -> float:
        return 1

class Relu(BasePredictor):
    def eval(self, W, X) -> np.ndarray:
        W = np.array(W)
        return relu(X @ W.T)
    
    def max_output(self) -> float:
        return math.inf

class StumpParam:
    def __init__(self, idx, value) -> None:
        self.idx = int(idx)
        self.value = value

    def __eq__(self, other: object) -> bool:
        return self.idx == other.idx and self.value == other.value


class Stump(BasePredictor):
    def eval(self, W: List[StumpParam], X) -> np.ndarray:
        idxs = [stump_param.idx for stump_param in W]
        values = [stump_param.value for stump_param in W]
        return sign(X[:, idxs] - values)
    
    def max_output(self) -> float:
        return 1
