import json
import numpy as np


def generate_cmdp_json(filename, X, A, L, constraint_type):
    transitions = {}

    available_states = X[:]
    np.random.shuffle(available_states)
    state_to_layer = {}

    first = [available_states.pop()]
    last = [available_states.pop()]
    middle_layers = [[] for _ in range(L - 2)]

    for i in range(L - 2):
        x = available_states.pop()
        middle_layers[i].append(x)

    for x in available_states:
        idx = np.random.randint(0, L - 2)
        middle_layers[idx].append(x)

    layers = [first] + middle_layers + [last]
    x0 = layers[0][0]

    for layer_idx, layer in enumerate(layers):
        for state in layer:
            state_to_layer[state] = layer_idx

    det_path_states = []
    det_path_actions = {}
    det_actions = []

    if constraint_type == "adv":
        det_path_states = [layer[0] for layer in layers]
        det_path_actions = {s: np.random.choice(A) for s in det_path_states[:-1]}
        det_actions = [(int(s), int(a)) for s, a in det_path_actions.items()]
        print(f"{det_actions}")

    for k in range(len(layers) - 1):
        l_k = layers[k]
        l_k1 = layers[k + 1]
        for x in l_k:
            for a in A:
                if constraint_type == "adv" and x in det_path_states[:-1] and a == det_path_actions[x]:
                    chosen_next = det_path_states[k + 1]
                    probs = [1.0 if x_prime == chosen_next else 0.0 for x_prime in l_k1]
                else:
                    probs = np.random.dirichlet(np.ones(len(l_k1)))
                transitions[f"{x}_{a}"] = {str(x_prime): float(p) for x_prime, p in zip(l_k1, probs)}

    cmdp = {
        "X": X,
        "A": A,
        "L": L,
        "layers": layers,
        "state_to_layer": state_to_layer,
        "transitions": transitions,
        "x0": x0
    }

    if constraint_type == "adv":
        cmdp["det_actions"] = det_actions

    with open(filename, 'w') as f:
        json.dump(cmdp, f, indent=4)

