import numpy as np

def truncated_gumbel(alpha, truncation):
    gumbel = np.random.gumbel() + np.log(alpha)
    return -np.log(np.exp(-gumbel) + np.exp(-truncation))

# Calculating the posterior using using A* sampling idea
def topdown(alphas, k):
    topgumbel = np.random.gumbel() + np.log(sum(alphas))
    gumbels = []
    for i in range(len(alphas)):
        if i == k:
            gumbel = topgumbel
        else:
            gumbel = truncated_gumbel(alphas[i], topgumbel)
        gumbels.append(gumbel)
    return gumbels

# posterior of noise for i th event, using A*
def posterior_A_star(i, lambdas, lambda_max, indicators):
    p_1 = lambdas[i]/lambda_max
    p_0 = 1 - p_1
    if indicators[i] == True:
        gumbels = np.array(topdown(np.array([p_0, p_1]), 1))
        return gumbels - np.log(np.array([p_0, p_1]))
    else:
        gumbels = np.array(topdown(np.array([p_0, p_1]), 0))
        return gumbels - np.log(np.array([p_0, p_1]))