import numpy as np
from scipy.special import betaln, digamma, polygamma
from scipy.optimize import minimize
import json


def constrained_mle(n_list, k_list, alpha_init, beta_init):
    low_bar, high_bar = 0.01, 1
    def loss(params):
        a, b = params
        if a < low_bar or a > high_bar or b < low_bar or b > high_bar:
            return np.inf  # frontier penalty
        neg_log_likelihood = -np.sum([betaln(a + k, b + n - k) - betaln(a, b) for n, k in zip(n_list, k_list)])
        return neg_log_likelihood #+ 0.5 * (a+b)

    result = minimize(
        loss,
        x0=[alpha_init, beta_init],
        bounds=[(low_bar, high_bar), (low_bar, high_bar)],
        method='L-BFGS-B'  # optimization algorithm supporting frontier
    )
    return result.x