############################################
# To facilitate checking the fulfillment of the double-blind principle,
# we only placed file that contain core algorithms in the supplementary material.
#
# All source code required for conducting and analyzing the experiments 
# will be made publicly available upon publication of the paper.
#
# Functions regarding the specific implementation of the algorithm are commented
# with references to corresponding sections.
############################################

import random
from utils import compute_jsd, HSIC

def estimate_hsic_distribution(model_h, S, T):
    """
    Algorithm 1: estimate_hsic_distribution # Section 3.1
    :param model_h: model function h_l(x) -> representation
    :param S: sample set, list or numpy array
    :param T: permutation times
    :return: H (list of HSIC values)
    """
    S_perm = random.sample(S, len(S))
    mid = len(S_perm) // 2
    S1, S2 = S_perm[:mid], S_perm[mid:]

    H = []
    for _ in range(T):
        S2 = random.sample(S2, len(S2))
        V = HSIC(model_h(S1), model_h(S2))
        H.append(V)
    return H


def is_in_training(model_h, T, S_tar, S_in, S_out):
    """
    Algorithm 2: is_in_training # Section 3.2
    :return: True if S_tar closer to in-training HSIC distribution
    """
    H_tar = estimate_hsic_distribution(model_h, S_tar, T)
    H_in = estimate_hsic_distribution(model_h, S_in, T)
    H_out = estimate_hsic_distribution(model_h, S_out, T)

    D_tar_in = compute_jsd(H_tar, H_in)
    D_tar_out = compute_jsd(H_tar, H_out)

    return D_tar_in < D_tar_out


def unlearn_eval(h_un, T, D_f, S_in, S_out, n, m):
    """
    Algorithm 3: unlearn_eval
    :return: OOT rate
    """
    target_list = []
    for _ in range(m):
        S_tar = random.sample(D_f, n)
        target_list.append(S_tar)

    OOT_Count = 0
    for S_tar in target_list:
        if not is_in_training(h_un, T, S_tar, S_in, S_out):
            OOT_Count += 1

    return OOT_Count / m

