import numpy as np
from collections import Counter
from scipy.stats import entropy, binom
from itertools import product

def uniform_bc_table(H):
    keys = [''.join(t) for t in product('bc', repeat=H)]
    probs = np.full(2**H, 1 / 2**H)
    return keys, probs

def kl_divergence(samples, H):
    filtered = ["".join(seq) for seq in samples]
    counter = Counter(filtered)
    total = sum(counter.values())
    if total == 0:
        return float('inf')
    keys, true = uniform_bc_table(H)
    empirical = np.array([counter.get(k, 0) / total for k in keys])
    return entropy(empirical, true, base=2)

def binomial_reference(H, p_b=1/2):
    return np.array([binom.pmf(k, H, p_b) for k in range(H+1)])

def kl_bcount(samples, H, true_binomial_probs):
    bcounts = [seq.count('b') for seq in samples]
    counter = Counter(bcounts)
    total = sum(counter.values())
    if total == 0:
        return float('inf')
    empirical = np.array([counter.get(k, 0) / total for k in range(H+1)])
    return entropy(empirical, true_binomial_probs, base=2) 