import numpy as np


class OGD:
    def __init__(self, A, eta):
        self.A = A
        self.eta = eta
        self.weights = np.ones(A) / A

    def predict(self):
        #eps = 0.5
        #return (1 - eps) * self.weights + eps / self.A
        return self.weights

    def update(self, gradient_vector):
        self.weights -= self.eta * np.array(gradient_vector)
        self.weights = self._project_to_simplex(self.weights)

    @staticmethod
    def _project_to_simplex(v):
        v = np.array(v)
        if np.sum(v) == 1 and np.all(v >= 0):
            return v
        u = np.sort(v)[::-1]
        cssv = np.cumsum(u)
        rho = np.nonzero(u * np.arange(1, len(u) + 1) > (cssv - 1))[0][-1]
        theta = (cssv[rho] - 1) / (rho + 1)
        w = np.maximum(v - theta, 0)
        return w

