import numpy as np
import random
from config import alphabet, alphabet_idx, Hs, alphas, num_reps, Ns

def sample_edge_noise(alphabet, max_H, alpha=1.0):
    edge_noise = {}

    def add_edge_noise(prefix, depth):
        if depth == max_H-1:
            return
        for ch in alphabet:
            key = (tuple(prefix), ch)
            edge_noise[key] = alpha if random.random() < 0.5 else 1.0 / alpha
            add_edge_noise(prefix + [ch], depth + 1)

    add_edge_noise([], 0)
    return edge_noise

def sample_tokenwise(estimated_value, edge_noise, H):
    steps=0
    while True:
        seq = []
        for _ in range(H):
            steps+=1
            scores = []
            for ch in alphabet:
                prefix = seq + [ch]
                f_val = estimated_value(prefix)

                noise_factor = edge_noise.get((tuple(seq), ch), 1.0)
                noisy_f_val = noise_factor * f_val

                scores.append(noisy_f_val)

            probs = np.array(scores, dtype=np.float64)
            total = probs.sum()

            if total == 0.0 or np.isnan(total):
                break

            probs /= total
            next_ch = np.random.choice(alphabet, p=probs)
            seq.append(next_ch)
        else:
            final_val = estimated_value(seq)
            if final_val > 0.0:
                return seq, steps
