import numpy as np


def _prepare_distribution(ground_truth: dict, predicted: dict) -> tuple:
    assert set(ground_truth.keys()) == set(predicted.keys()), (
        "The keys of the ground truth and predicted distributions must match."
    )
    gt_array = []
    pr_array = []
    for key in ground_truth:
        gt_array.append(ground_truth[key].to_float())
        pr_array.append(predicted[key])
    return np.array(gt_array), np.array(pr_array)


def l1_distance(ground_truth: dict, predicted: dict) -> float:
    gt_array, pr_array = _prepare_distribution(ground_truth, predicted)
    return np.sum(np.abs(gt_array - pr_array))


def chebyshev_distance(ground_truth: dict, predicted: dict) -> float:
    gt_array, pr_array = _prepare_distribution(ground_truth, predicted)
    return np.max(np.abs(gt_array - pr_array))


def kl_divergence(gt_array, pr_array: dict) -> float:
    epsilon: float = 1e-5
    pr_array = np.clip(pr_array, epsilon, 1 - epsilon)
    gt_array = np.clip(gt_array, epsilon, 1 - epsilon)
    return np.sum(gt_array * np.log(gt_array / pr_array))


def symmetric_kl_divergence(ground_truth, predicted) -> float:
    gt_array, pr_array = _prepare_distribution(ground_truth, predicted)

    return kl_divergence(gt_array, pr_array) + kl_divergence(pr_array, gt_array)
