import numpy as np 
import random
from csl.graph.SHD import SHD
from csl.utils.DAG2CPDAG import dag2cpdag
from csl.utils.TXT2GeneralGraph import dagadjmat2generalgraph
from csl.search.ConstraintBased.PC import pc
from csl.search.ScoreBased.GES import ges
from csl.utils import cit
import math

from itertools import combinations
import random
from causallearn.utils.cit import CIT
import matplotlib.pyplot as plt
from collections import Counter
############### linear gaussian ################
##########x-y ###########
print('cause###########')
for i in range(10):
    num_sample = 100000
    mu = 3
    mu1 = 2
    mus = 3
    sigma_x = random.uniform(1,2)
    sigma_y = random.uniform(1,3)
    sigma_l = random.uniform(1,3)
    sigma_s = random.uniform(3,6)
    sigma_n = random.uniform(0,2)
    num_sample = 100000
    a = random.uniform(2,4)
    startn = 1
    endn = 3
    noise_x = np.random.uniform(startn,endn,num_sample)
    noise_y = np.random.uniform(startn,endn,num_sample)
    print(a)
    l1 = random.uniform(1,3)
    l2 = random.uniform(1,2)
    s1 = random.uniform(1,3)
    s2 = random.uniform(2,5)

    Ex = np.random.normal(mu,sigma_x,num_sample)
    Ey = np.random.normal(mu1,sigma_y,num_sample)
    El = np.random.normal(mu,sigma_l,num_sample)
    Es = np.random.normal(mus,sigma_s,num_sample)
    Esy = np.random.normal(mu,sigma_s,num_sample)
# ####################case 1 (cause)#####################
    x = Ex
    c = np.zeros(100000)
    y = a * x + Ey
    o_data =  np.stack((x,y,c), axis = 1)

    per_cx = np.ones(100000)
    Ex_per = np.random.normal(mu,sigma_x,num_sample)
    do_x = Ex_per + noise_x
    Ey_per = np.random.normal(mu1,sigma_y,num_sample)
    per_y = a * do_x + Ey_per
    pertub_data_x = np.stack((do_x, per_y, per_cx), axis=1)
    data_per_x = np.concatenate((o_data,pertub_data_x), axis = 0)
    # print(data_per_x.shape)
    np.save('./linear_gaussian/case1_do_a(x).npy', data_per_x)
    # print(pertub_data_x)

    kci_obj = CIT(data_per_x, "fisherz")
    pValue = kci_obj(1,2,set([]))
    print(1,2,())
    print(pValue)
    if pValue >0.05:
        continue
    pValue = kci_obj(1,2,set([0]))
    print(1,2,(0))
    print(pValue)
    if pValue < 0.05:
        continue


    per_cy = np.ones(100000)
    Ex_per = np.random.normal(mu,sigma_x,num_sample)
    Ey_per = np.random.normal(mu1,sigma_y,num_sample)
    per_x = Ex_per
    do_y = a * per_x + Ey_per + noise_y
    perturb_data_y = np.stack((per_x,do_y,per_cy), axis = 1)
    data_per_y = np.concatenate((o_data,perturb_data_y), axis = 0)
    np.save('./linear_gaussian/case1_do_a(y).npy', data_per_y)

    kci_obj = CIT(data_per_y, "fisherz") # construct a CIT instance with data and method name

    pValue = kci_obj(0,2,set([]))
    print(0,2,())
    print(pValue)
    if pValue < 0.05:
        continue
    pValue = kci_obj(0,2,set([1]))
    print(0,2,(1))
    print(pValue)
    if pValue < 0.05:
        break

#####################case 2 (latent)########################
print('latent ############')
for i in range(10):
    mu = 1
    mu1 = 2
    mus = 3
    sigma_x = random.uniform(1,2)
    sigma_y = random.uniform(1,3)
    sigma_l = random.uniform(1,3)
    sigma_s = random.uniform(3,6)
    sigma_n = random.uniform(0,2)
    num_sample = 100000
    a = random.uniform(2,4)
    # noise_x = np.random.normal(2,sigma_n, num_sample)
    # noise_y = np.random.normal(2,sigma_n, num_sample)
    print(a)
    l1 = random.uniform(1,3)
    l2 = random.uniform(1,2)
    s1 = random.uniform(1,3)
    s2 = random.uniform(2,5)

    Ex = np.random.normal(mu,sigma_x,num_sample)
    Ey = np.random.normal(mu1,sigma_y,num_sample)
    El = np.random.normal(mu,sigma_l,num_sample)
    Es = np.random.normal(mus,sigma_s,num_sample)
    Esy = np.random.normal(mu,sigma_s,num_sample)

    startn = 1
    endn = 3
    noise_x = np.random.uniform(startn,endn,num_sample)
    noise_y = np.random.uniform(startn,endn,num_sample)

    L = El
    c = np.zeros(100000)
    x = Ex + l1 * L
    y = Ey + l2 * L
    o_data = np.stack((x,y,c),axis=1)

    per_cx = np.ones(100000)
    El_per_x = np.random.normal(mu,sigma_l,num_sample)
    Ex_per = np.random.normal(mu,sigma_x,num_sample)
    Ey_per = np.random.normal(mu1,sigma_y,num_sample)
    do_x = Ex_per + l1 * El_per_x + noise_x

    per_y = Ey_per + l2 * El_per_x
    pertub_data_x = np.stack((do_x,per_y, per_cx), axis=1)
    data_per_x = np.concatenate((o_data,pertub_data_x), axis = 0)
    np.save('./linear_gaussian/case2_do_a(x).npy', data_per_x)

    kci_obj = CIT(data_per_x, "fisherz")
    pValue = kci_obj(1,2,set([]))
    print(1,2,())
    print(pValue)
    if pValue < 0.05:
        continue
    pValue = kci_obj(1,2,set([0]))
    print(1,2,(0))
    print(pValue)
    if pValue > 0.05:
        continue

    per_cy = np.ones(100000)
    Ex_per = np.random.normal(mu,sigma_x,num_sample)
    Ey_per = np.random.normal(mu1,sigma_y,num_sample)
    El_per_y = np.random.normal(mu,sigma_l,num_sample)
    do_y = Ey_per + l2 * El_per_y + noise_y
    per_x = Ex_per + l1 * El_per_y
    perturb_data_y = np.stack((per_x,do_y,per_cy), axis = 1)
    data_per_y = np.concatenate((o_data,perturb_data_y), axis = 0)
    np.save('./linear_gaussian/case2_do_a(y).npy', data_per_y)

    kci_obj = CIT(data_per_y, "fisherz")
    pValue = kci_obj(0,2,set([]))
    print(0,2,())
    print(pValue)
    if pValue < 0.05:
        continue
    pValue = kci_obj(0,2,set([1]))
    print(0,2,(1))
    print(pValue)
    if pValue < 0.05:
        break

#####################case 3 (selection)########################
print('selection ############')
for i in range(10):
    mu = 2
    mu1 = 3
    mus = 2
    sigma_x = random.uniform(0,2)
    sigma_y = random.uniform(0,2)
    sigma_l = random.uniform(0,2)
    sigma_s = random.uniform(1,2)
    sigma_n = random.uniform(0,2)
    num_sample = 100000
    a = random.uniform(3,6)
    # noise_x = np.random.uniform(2,4, num_sample)
    # noise_y = np.random.uniform(2,4, num_sample)
    # noise_x = np.random.normal(2,sigma_n, num_sample)
    # noise_y = np.random.normal(2,sigma_n, num_sample)
    startn = 1
    endn = 3
    noise_x = np.random.uniform(startn,endn,num_sample)
    noise_y = np.random.uniform(startn,endn,num_sample)
    print(a)
    l1 = random.uniform(1,3)
    l2 = random.uniform(1,2)
    s1 = random.uniform(0,1)
    s2 = random.uniform(0,2)

    Ex = np.random.normal(mu,sigma_x,num_sample)
    Ey = np.random.normal(mu1,sigma_y,num_sample)
    El = np.random.normal(mu,sigma_l,num_sample)
    Es = np.random.normal(mus,sigma_s,num_sample)

    c = np.zeros(100000)
    x = Ex
    y = Ey
    mask = (s1 * x + s2 * y + Es > 4) & (s1 * x + s2 * y + Es < 7)
    x = x[mask]
    y = y[mask]
    c = c[mask]
    o_data = np.stack((x,y,c),axis=1)

    per_cx = np.ones(100000)
    Es_per_x = np.random.normal(mu,sigma_s,num_sample)
    Ex_per = np.random.normal(mu,sigma_x,num_sample)
    Ey_per = np.random.normal(mu1,sigma_y,num_sample)
    do_x = Ex_per + noise_x
    per_y = Ey_per
    mask_per_x = (s1 * do_x + s2 * per_y + Es_per_x > 4) & (s1 * do_x + s2 * per_y + Es_per_x < 7)
    do_x = do_x[mask_per_x]
    per_y = per_y[mask_per_x]
    per_cx = per_cx[mask_per_x]
    pertub_data_x = np.stack((do_x,per_y, per_cx), axis=1)
    print(pertub_data_x.shape)
    data_per_x = np.concatenate((o_data,pertub_data_x), axis = 0)
    np.save('./linear_gaussian/case3_do_a(x).npy', data_per_x)
    print(data_per_x.shape)

    kci_obj = CIT(data_per_x, "fisherz")
    pValue = kci_obj(1,2,set([]))
    print(1,2,())
    print(pValue)
    if pValue > 0.05:
        continue
    pValue = kci_obj(1,2,set([0]))
    print(1,2,(0))
    print(pValue)
    if pValue < 0.05:
        continue

    per_cy = np.ones(100000)
    Es_per_y = np.random.normal(mu,sigma_s,num_sample)
    Ex_per = np.random.normal(mu,sigma_x,num_sample)
    Ey_per = np.random.normal(mu1,sigma_y,num_sample)
    do_y = Ey_per + noise_y
    per_x = Ex_per
    mask_per_y = (s1 * per_x + s2 * do_y + Es_per_y > 4) & (s1 * per_x + s2 * do_y + Es_per_y < 7)
    # print(s1 * per_x + s2 * do_y + Es_per_y)
    per_x = per_x[mask_per_y]
    do_y = do_y[mask_per_y]
    per_cy = per_cy[mask_per_y]
    pertub_data_y = np.stack((per_x,do_y, per_cy), axis=1)
    print(pertub_data_y.shape)
    data_per_y = np.concatenate((o_data,pertub_data_y), axis = 0)
    np.save('./linear_gaussian/case3_do_a(y).npy', data_per_y)
    print(data_per_y.shape)

    kci_obj = CIT(data_per_y, "fisherz")
    pValue = kci_obj(0,2,set([]))
    print(0,2,())
    print(pValue)
    if pValue > 0.05:
        continue
    pValue = kci_obj(0,2,set([1]))
    print(0,2,(1))
    print(pValue)
    if pValue > 0.05:
        break

#####################case 4 (casue with selection)########################
print('casue with selection #############')
for i in range(10):
    mu = 2
    mu1 = 3
    mus = 2
    sigma_x = random.uniform(0,2)
    sigma_y = random.uniform(0,2)
    sigma_l = random.uniform(0,2)
    sigma_s = random.uniform(1,2)
    sigma_n = random.uniform(0,2)
    num_sample = 100000
    a = random.uniform(3,6)
    # noise_x = np.random.uniform(2,4, num_sample)
    # noise_y = np.random.uniform(2,4, num_sample)
    # noise_x = np.random.normal(2,sigma_n, num_sample)
    # noise_y = np.random.normal(2,sigma_n, num_sample)
    startn = 1
    endn = 3
    noise_x = np.random.uniform(startn,endn,num_sample)
    noise_y = np.random.uniform(startn,endn,num_sample)
    print(a)
    l1 = random.uniform(1,3)
    l2 = random.uniform(1,2)
    s1 = random.uniform(0,1)
    s2 = random.uniform(0,2)

    Ex = np.random.normal(mu,sigma_x,num_sample)
    Ey = np.random.normal(mu1,sigma_y,num_sample)
    El = np.random.normal(mu,sigma_l,num_sample)
    Es = np.random.normal(mus,sigma_s,num_sample)

    c = np.zeros(100000)
    x = Ex 
    y = Ey + a * Ex
    mask = (s1 * x + s2 * y + Es > 5) & (s1 * x + s2 * y + Es < 20)
    print(s1 * x + s2 * y + Es)
    x = x[mask]
    y = y[mask]
    c = c[mask]
    o_data = np.stack((x,y,c),axis=1)

    per_cx = np.ones(100000)
    Es_per_x = np.random.normal(mu,sigma_s,num_sample)
    Ex_per = np.random.normal(mu,sigma_x,num_sample)
    Ey_per = np.random.normal(mu1,sigma_y,num_sample)
    do_x = Ex_per + noise_x
    per_y = Ey_per + do_x * a
    mask_per_x = (s1 * do_x + s2 * per_y + Es_per_x > 5) & (s1 * do_x + s2 * per_y + Es_per_x < 20)
    do_x = do_x[mask_per_x]
    per_y = per_y[mask_per_x]
    per_cx = per_cx[mask_per_x]
    pertub_data_x = np.stack((do_x,per_y, per_cx), axis=1)
    data_per_x = np.concatenate((o_data,pertub_data_x), axis = 0)
    np.save('./linear_gaussian/case4_do_a(x).npy', data_per_x)
    print(pertub_data_x.shape)
    print(data_per_x.shape)

    kci_obj = CIT(data_per_x, "fisherz")
    pValue = kci_obj(1,2,set([]))
    print(1,2,())
    print(pValue)
    if pValue > 0.05:
        continue
    pValue = kci_obj(1,2,set([0]))
    print(1,2,(0))
    print(pValue)
    if pValue < 0.05:
        continue


    per_cy = np.ones(100000)
    Es_per_y = np.random.normal(mu,sigma_s,num_sample)
    Ey_per = np.random.normal(mu1,sigma_y,num_sample)
    Ex_per = np.random.normal(mu,sigma_x,num_sample)
    per_x = Ex_per
    do_y = Ey_per + noise_y + a * per_x
    mask_per_y = (s1 * per_x + s2 * do_y + Es_per_y > 5) & (s1 * per_x + s2 * do_y + Es_per_y < 20)
    per_x = per_x[mask_per_y]
    do_y = do_y[mask_per_y]
    per_cy = per_cy[mask_per_y]
    pertub_data_y = np.stack((per_x,do_y, per_cy), axis=1)
    data_per_y = np.concatenate((o_data,pertub_data_y), axis = 0)
    np.save('./linear_gaussian/case4_do_a(y).npy', data_per_y)
    print(pertub_data_y.shape)
    print(data_per_y.shape)

    kci_obj = CIT(data_per_y, "fisherz")
    pValue = kci_obj(0,2,set([]))
    print(0,2,())
    print(pValue)
    if pValue > 0.05:
        continue
    pValue = kci_obj(0,2,set([1]))
    print(0,2,(1))
    print(pValue)
    if pValue < 0.05:
        break

# #####################case 5 (casue with latent)########################
print('casue with latent #############')
for i in range(10):
    mu = 1
    mu1 = 2
    mus = 3
    sigma_x = random.uniform(1,2)
    sigma_y = random.uniform(1,3)
    sigma_l = random.uniform(1,3)
    sigma_s = random.uniform(3,6)
    sigma_n = random.uniform(0,2)
    num_sample = 100000
    a = random.uniform(2,4)
    # noise_x = np.random.normal(2,sigma_n, num_sample)
    # noise_y = np.random.normal(2,sigma_n, num_sample)
    startn = 1
    endn = 3
    noise_x = np.random.uniform(startn,endn,num_sample)
    noise_y = np.random.uniform(startn,endn,num_sample)
    print(a)
    l1 = random.uniform(1,3)
    l2 = random.uniform(1,2)
    s1 = random.uniform(1,3)
    s2 = random.uniform(2,5)

    Ex = np.random.normal(mu,sigma_x,num_sample)
    Ey = np.random.normal(mu1,sigma_y,num_sample)
    El = np.random.normal(mu,sigma_l,num_sample)
    Es = np.random.normal(mus,sigma_s,num_sample)
    Esy = np.random.normal(mu,sigma_s,num_sample)

    L = El
    c = np.zeros(100000)
    x = Ex + l1 * L
    y = a * Ex + Ey + l2 * L
    o_data = np.stack((x,y,c),axis=1)

    per_cx = np.ones(100000)
    Ex_per = np.random.normal(mu,sigma_x,num_sample)
    Ey_per = np.random.normal(mu1,sigma_y,num_sample)
    El_per_x = np.random.normal(mu,sigma_l,num_sample)
    do_x = Ex_per + El_per_x * l1 + noise_x
    per_y = Ey_per + l2 * El_per_x + do_x * a
    pertub_data_x = np.stack((do_x,per_y, per_cx), axis=1)
    data_per_x = np.concatenate((o_data,pertub_data_x), axis = 0)
    np.save('./linear_gaussian/case5_do_a(x).npy', data_per_x)

    kci_obj = CIT(data_per_x, "fisherz")
    pValue = kci_obj(1,2,set([]))
    print(1,2,())
    print(pValue)
    if pValue > 0.05:
        continue
    pValue = kci_obj(1,2,set([0]))
    print(1,2,(0))
    print(pValue)
    if pValue > 0.05:
        continue

    per_cy = np.ones(100000)
    Ey_per = np.random.normal(mu1,sigma_y,num_sample)
    Ex_per = np.random.normal(mu,sigma_x,num_sample)
    El_per_y = np.random.normal(mu,sigma_l,num_sample)
    per_x = Ex_per + l1 * El_per_y
    do_y = Ey_per + l2 * El_per_y + noise_y + a * per_x
    perturb_data_y = np.stack((per_x,do_y,per_cy), axis = 1)
    data_per_y = np.concatenate((o_data,perturb_data_y), axis = 0)
    np.save('./linear_gaussian/case5_do_a(y).npy', data_per_y)

    kci_obj = CIT(data_per_y, "fisherz")
    pValue = kci_obj(0,2,set([]))
    print(0,2,())
    print(pValue)
    if pValue < 0.05:
        continue
    pValue = kci_obj(0,2,set([1]))
    print(0,2,(1))
    print(pValue)
    if pValue < 0.05:
        break


# #####################case 6 (selection with latent)########################
print('selection with latent  ############')
for i in range(10):
    mu = 2
    mu1 = 3
    mus = 2
    sigma_x = random.uniform(0,2)
    sigma_y = random.uniform(0,2)
    sigma_l = random.uniform(0,2)
    sigma_s = random.uniform(1,2)
    sigma_n = random.uniform(0,2)
    num_sample = 100000
    a = random.uniform(3,6)
    # noise_x = np.random.uniform(2,4, num_sample)
    # noise_y = np.random.uniform(2,4, num_sample)
    # noise_x = np.random.normal(2,sigma_n, num_sample)
    # noise_y = np.random.normal(2,sigma_n, num_sample)
    startn = 1
    endn = 3
    noise_x = np.random.uniform(startn,endn,num_sample)
    noise_y = np.random.uniform(startn,endn,num_sample)
    print(a)
    l1 = random.uniform(1,3)
    l2 = random.uniform(1,2)
    s1 = random.uniform(0,1)
    s2 = random.uniform(0,2)

    Ex = np.random.normal(mu,sigma_x,num_sample)
    Ey = np.random.normal(mu1,sigma_y,num_sample)
    El = np.random.normal(mu,sigma_l,num_sample)
    Es = np.random.normal(mus,sigma_s,num_sample)

    L = El
    c = np.zeros(100000)
    x = Ex + l1 * L
    y = Ey + l2 * L
    mask = (s1 * x + s2 * y + Es > 5) & (s1 * x + s2 * y + Es < 50)
    x = x[mask]
    y = y[mask]
    c = c[mask]
    o_data = np.stack((x,y,c),axis=1)

    per_cx = np.ones(100000)
    El_per_x = np.random.normal(mu,sigma_l,num_sample)
    Es_per_x = np.random.normal(mu,sigma_s,num_sample)
    Ex_per = np.random.normal(mu,sigma_x,num_sample)
    do_x = Ex_per + l1 * El_per_x + noise_x
    Ey_per = np.random.normal(mu1,sigma_y,num_sample)
    per_y = Ey_per + l2 * El_per_x
    mask_per_x = (s1 * do_x + s2 * per_y + Es_per_x > 5) & (s1 * do_x + s2 * per_y + Es_per_x < 50)
    do_x = do_x[mask_per_x]
    per_y = per_y[mask_per_x]
    per_cx = per_cx[mask_per_x]
    pertub_data_x = np.stack((do_x,per_y, per_cx), axis=1)
    data_per_x = np.concatenate((o_data,pertub_data_x), axis = 0)
    np.save('./linear_gaussian/case6_do_a(x).npy', data_per_x)
    print(pertub_data_x.shape)
    print(data_per_x.shape)

    kci_obj = CIT(data_per_x, "fisherz")
    pValue = kci_obj(1,2,set([]))
    print(1,2,())
    print(pValue)
    if np.isnan(pValue):
        continue
    if pValue > 0.05:
        continue
    pValue = kci_obj(1,2,set([0]))
    print(1,2,(0))
    print(pValue)
    if pValue > 0.05:
        continue

    per_cy = np.ones(100000)
    El_per_y = np.random.normal(mu,sigma_l,num_sample)
    Es_per_y = np.random.normal(mu,sigma_s,num_sample)
    Ey_per = np.random.normal(mu1,sigma_y,num_sample)
    do_y = Ey_per + l2 * El_per_y + noise_y
    Ex_per = np.random.normal(mu,sigma_x,num_sample)
    per_x = Ex_per + l1 * El_per_y
    mask_per_y = (s1 * per_x + s2 * do_y + Es_per_y > 5) & (s1 * per_x + s2 * do_y + Es_per_y < 50)
    per_x = per_x[mask_per_y]
    do_y = do_y[mask_per_y]
    per_cy = per_cy[mask_per_y]
    pertub_data_y = np.stack((per_x,do_y, per_cy), axis=1)
    data_per_y = np.concatenate((o_data,pertub_data_y), axis = 0)
    np.save('./linear_gaussian/case6_do_a(y).npy', data_per_y)
    print(pertub_data_y.shape)
    print(data_per_y.shape)

    kci_obj = CIT(data_per_y, "fisherz")
    pValue = kci_obj(0,2,set([]))
    print(0,2,())
    print(pValue)
    if np.isnan(pValue):
        continue
    if pValue > 0.05:
        continue
    pValue = kci_obj(0,2,set([1]))
    print(0,2,(1))
    print(pValue)
    if pValue < 0.05:
        break
# #####################case 7 (cause with selection and latent)########################
print('cause with selection and latent ################')
for i in range(10):
    mu = 2
    mu1 = 3
    mus = 2
    sigma_x = random.uniform(0,2)
    sigma_y = random.uniform(0,2)
    sigma_l = random.uniform(0,2)
    sigma_s = random.uniform(1,2)
    sigma_n = random.uniform(0,2)
    num_sample = 100000
    a = random.uniform(3,6)
    # noise_x = np.random.uniform(2,4, num_sample)
    # noise_y = np.random.uniform(2,4, num_sample)
    # noise_x = np.random.normal(2,sigma_n, num_sample)
    # noise_y = np.random.normal(2,sigma_n, num_sample)
    startn = 1
    endn = 3
    noise_x = np.random.uniform(startn,endn,num_sample)
    noise_y = np.random.uniform(startn,endn,num_sample)
    print(a)
    l1 = random.uniform(1,3)
    l2 = random.uniform(1,2)
    s1 = random.uniform(0,1)
    s2 = random.uniform(0,2)

    Ex = np.random.normal(mu,sigma_x,num_sample)
    Ey = np.random.normal(mu1,sigma_y,num_sample)
    El = np.random.normal(mu,sigma_l,num_sample)
    Es = np.random.normal(mus,sigma_s,num_sample)

    L = El
    c = np.zeros(100000)
    x = Ex + l1 * L
    y = Ey + l2 * L + a * x
    mask = (s1 * x + s2 * y + Es > 10) & (s1 * x + s2 * y + Es < 40)
    print(s1 * x + s2 * y + Es)
    x = x[mask]
    y = y[mask]
    c = c[mask]
    o_data = np.stack((x,y,c),axis=1)
    # print(o_data)

    per_cx = np.ones(100000)
    El_per_x = np.random.normal(mu,sigma_l,num_sample)
    Es_per_x = np.random.normal(mus,sigma_s,num_sample)
    Ex_per = np.random.normal(mu,sigma_x,num_sample)
    do_x = Ex_per + l1 * El_per_x + noise_x
    Ey_per = np.random.normal(mu1,sigma_y,num_sample)
    per_y = Ey_per + l2 * El_per_x + a * do_x
    mask_per_x = (s1 * do_x + s2 * per_y + Es_per_x > 10) & (s1 * do_x + s2 * per_y + Es_per_x < 40)
    do_x = do_x[mask_per_x]
    per_y = per_y[mask_per_x]
    per_cx = per_cx[mask_per_x]
    pertub_data_x = np.stack((do_x,per_y, per_cx), axis=1)
    data_per_x = np.concatenate((o_data,pertub_data_x), axis = 0)
    np.save('./linear_gaussian/case7_do_a(x).npy', data_per_x)
    print(pertub_data_x.shape)
    print(data_per_x.shape)
    kci_obj = CIT(data_per_x, "fisherz")
    pValue = kci_obj(1,2,set([]))
    print(1,2,())
    if np.isnan(pValue):
        continue
    print(pValue)
    if pValue > 0.05:
        continue
    pValue = kci_obj(1,2,set([0]))
    print(1,2,(0))
    print(pValue)
    if pValue > 0.05:
        continue


    per_cy = np.ones(100000)
    El_per_y = np.random.normal(mu,sigma_l,num_sample)
    Es_per_y = np.random.normal(mus,sigma_s,num_sample)
    Ey_per = np.random.normal(mu1,sigma_y,num_sample)
    Ex_per = np.random.normal(mu,sigma_x,num_sample)
    per_x = Ex_per + l1 * El_per_y
    do_y = Ey_per + l2 * El_per_y + noise_y + a * per_x
    mask_per_y = (s1 * per_x + s2 * do_y + Es_per_y > 10) & (s1 * per_x + s2 * do_y + Es_per_y < 40)
    per_x = per_x[mask_per_y]
    do_y = do_y[mask_per_y]
    per_cy = per_cy[mask_per_y]
    pertub_data_y = np.stack((per_x,do_y, per_cy), axis=1)
    data_per_y = np.concatenate((o_data,pertub_data_y), axis = 0)
    np.save('./linear_gaussian/case7_do_a(y).npy', data_per_y)
    print(pertub_data_y.shape)
    print(data_per_y.shape)

    kci_obj = CIT(data_per_y, "fisherz")
    pValue = kci_obj(0,2,set([]))
    if np.isnan(pValue):
        continue
    print(0,2,())
    print(pValue)
    if pValue > 0.05:
        continue
    pValue = kci_obj(0,2,set([1]))
    print(0,2,(1))
    print(pValue)
    if pValue < 0.05:
        break








############### linear non-gaussian ################

# startx = -1
# endx = 2
# starty = 3
# endy = 5
# startl = 7
# endl = 9
# starts = 2
# ends = 4
# num_sample = 4000
# a = random.uniform(5,15)
# print(a)
# l1 = random.uniform(4,6)
# l2 = random.uniform(4,6)
# s1 = random.uniform(-1,3)
# s2 = random.uniform(1,2)
# Ex = np.random.uniform(startx,endx,num_sample)
# Ey = np.random.uniform(starty,endy,num_sample)
# El = np.random.uniform(startl,endl,num_sample)
# Es = np.random.uniform(starts,ends,num_sample)



#####################linear non-gaussian #####################

#####################case 1 (cause)#####################
# for i in range(10):
#     startx = 0
#     endx = 2
#     starty = 4
#     endy = 6
#     startl = 7
#     endl = 9
#     starts = 2
#     ends = 4
#     startn = 2
#     endn = 4
#     num_sample = 4000
#     a = random.uniform(5,15)
#     print(a)
#     l1 = random.uniform(4,6)
#     l2 = random.uniform(4,6)
#     s1 = random.uniform(-1,3)
#     s2 = random.uniform(1,2)
#     noise_x = np.random.uniform(startn,endn,num_sample)
#     noise_y = np.random.uniform(startn,endn,num_sample)
#     Ex = np.random.uniform(startx,endx,num_sample)
#     Ey = np.random.uniform(starty,endy,num_sample)
#     El = np.random.uniform(startl,endl,num_sample)
#     Es = np.random.uniform(starts,ends,num_sample)

#     x = Ex
#     c = np.zeros(4000)
#     y = a * x + Ey
#     o_data =  np.stack((x,y,c), axis = 1)


#     per_cx = np.ones(4000)
#     Ex_per = np.random.uniform(startx,endx,num_sample)
#     do_x = Ex_per + noise_x
#     Ey_per = np.random.uniform(starty,endy,num_sample)
#     per_y = Ey_per + a * do_x
#     pertub_data_x = np.stack((do_x,Ey_per, per_cx), axis=1)
#     data_per_x = np.concatenate((o_data,pertub_data_x), axis = 0)
#     print(data_per_x.shape)
#     np.save('./linear_non-gaussian/case1_do_a(x).npy', data_per_x)
#     # print(pertub_data_x)


#     per_cy = np.ones(4000)
#     Ey_per = np.random.uniform(starty,endy,num_sample)
#     Ex_per = np.random.uniform(startx,endx,num_sample)
#     per_x = Ex_per
#     do_y = Ey_per + a * per_x + noise_y
#     perturb_data_y = np.stack((per_x,do_y,per_cy), axis = 1)
#     data_per_y = np.concatenate((o_data,perturb_data_y), axis = 0)
#     np.save('./linear_non-gaussian/case1_do_a(y).npy', data_per_y)


#####################case 2 (latent)########################
# for i in range(10):
#     startx = 0
#     endx = 2
#     starty = 4
#     endy = 6
#     startl = 7
#     endl = 9
#     starts = 2
#     ends = 4
#     startn = 2
#     endn = 4
#     num_sample = 4000
#     a = random.uniform(5,15)
#     print(a)
#     l1 = random.uniform(1,3)
#     l2 = random.uniform(1,3)
#     s1 = random.uniform(-1,3)
#     s2 = random.uniform(1,2)
#     noise_x = np.random.uniform(startn,endn,num_sample)
#     noise_y = np.random.uniform(startn,endn,num_sample)
#     Ex = np.random.uniform(startx,endx,num_sample)
#     Ey = np.random.uniform(starty,endy,num_sample)
#     El = np.random.uniform(startl,endl,num_sample)
#     Es = np.random.uniform(starts,ends,num_sample)

#     L = El
#     c = np.zeros(4000)
#     x = Ex + l1 * L
#     y = Ey + l2 * L
#     o_data = np.stack((x,y,c),axis=1)

#     per_cx = np.ones(4000)
#     Ex_per = np.random.uniform(startx, endx,num_sample)
#     Ey_per = np.random.uniform(starty,endy,num_sample)
#     El_per_x = np.random.uniform(startl, endl,num_sample)
#     do_x = Ex_per + l1 * El_per_x + noise_x
#     per_y = Ey_per + l2 * El_per_x
#     pertub_data_x = np.stack((do_x,per_y, per_cx), axis=1)
#     data_per_x = np.concatenate((o_data,pertub_data_x), axis = 0)
#     np.save('./linear_non-gaussian/case2_do_a(x).npy', data_per_x)
#     kci_obj = CIT(data_per_x, "kci", est_width = 'median')
#     pValue = kci_obj(1,2,set([]))
#     print(1,2,())
#     print(pValue)
#     if pValue < 0.05:
#         continue
#     pValue = kci_obj(1,2,set([0]))
#     print(1,2,(0))
#     print(pValue)
#     if pValue > 0.05:
#         continue

#     per_cy = np.ones(4000)
#     Ey_per = np.random.uniform(starty,endy,num_sample)
#     Ex_per = np.random.uniform(startx, endx,num_sample)
#     El_per_y = np.random.uniform(startl, endl,num_sample)
#     per_x = Ex_per + l1 * El_per_y
#     do_y = Ey_per + l2 * El_per_y + noise_y
#     perturb_data_y = np.stack((per_x,do_y,per_cy), axis = 1)
#     data_per_y = np.concatenate((o_data,perturb_data_y), axis = 0)
#     np.save('./linear_non-gaussian/case2_do_a(y).npy', data_per_y)

#     kci_obj = CIT(data_per_y, "kci", est_width = 'median')
#     pValue = kci_obj(0,2,set([]))
#     print(0,2,())
#     print(pValue)
#     if pValue < 0.05:
#         continue
#     pValue = kci_obj(0,2,set([1]))
#     print(0,2,(1))
#     print(pValue)
#     if pValue < 0.05:
#         break
#####################case 3 (selection)########################
# for i in range(10):
#     startx = -1
#     endx = 2
#     starty = 3
#     endy = 5
#     startl = 7
#     endl = 9
#     starts = 2
#     ends = 4
#     startn = 2
#     endn = 4
#     num_sample = 4000
#     a = random.uniform(5,15)
#     print(a)
#     l1 = random.uniform(4,6)
#     l2 = random.uniform(4,6)
#     s1 = random.uniform(-1,3)
#     s2 = random.uniform(1,2)
#     Ex = np.random.uniform(startx,endx,num_sample)
#     Ey = np.random.uniform(starty,endy,num_sample)
#     El = np.random.uniform(startl,endl,num_sample)
#     Es = np.random.uniform(starts,ends,num_sample)
#     noise_x = np.random.uniform(startn,endn,num_sample)
#     noise_y = np.random.uniform(startn,endn,num_sample)


#     c = np.zeros(4000)
#     x = Ex
#     y = Ey
#     mask = (s1 * x + s2 * y + Es > 0) & (s1 * x + s2 * y + Es < 20)
#     print(s1 * x + s2 * y + Es)
#     x = x[mask]
#     y = y[mask]
#     c = c[mask]
#     o_data = np.stack((x,y,c),axis=1)

#     per_cx = np.ones(4000)
#     Es_per_x = np.random.uniform(starts, ends,num_sample)
#     Ex_per = np.random.uniform(startx, endx,num_sample)
#     do_x = Ex_per + noise_x
#     Ey_per = np.random.uniform(starty,endy,num_sample)
#     per_y = Ey_per
#     mask_per_x = (s1 * do_x + s2 * per_y + Es_per_x > 0) & (s1 * do_x + s2 * per_y + Es_per_x < 20)
#     print(s1 * do_x + s2 * per_y + Es_per_x)
#     do_x = do_x[mask_per_x]
#     per_y = per_y[mask_per_x]
#     per_cx = per_cx[mask_per_x]
#     pertub_data_x = np.stack((do_x,per_y, per_cx), axis=1)
#     print(pertub_data_x.shape)
#     data_per_x = np.concatenate((o_data,pertub_data_x), axis = 0)
#     np.save('./linear_non-gaussian/case3_do(x).npy', data_per_x)
#     print(data_per_x.shape)
#     kci_obj = CIT(data_per_x, "kci")
#     pValue = kci_obj(1,2,set([]))
#     print(1,2,())
#     print(pValue)
#     if pValue > 0.05:
#         continue
#     pValue = kci_obj(1,2,set([0]))
#     print(1,2,(0))
#     print(pValue)
#     if pValue < 0.05:
#         continue



#     per_cy = np.ones(4000)
#     Es_per_y = np.random.uniform(starts, ends,num_sample)
#     Ey_per = np.random.uniform(starty,endy,num_sample)
#     do_y = Ey_per + noise_y
#     Ex_per = np.random.uniform(startx, endx,num_sample)
#     per_x = Ex_per
#     mask_per_y = (s1 * per_x + s2 * do_y + Es_per_y > 0) & (s1 * per_x + s2 * do_y + Es_per_y < 20)
#     print(s1 * per_x + s2 * do_y + Es_per_y)
#     per_x = per_x[mask_per_y]
#     do_y = do_y[mask_per_y]
#     per_cy = per_cy[mask_per_y]
#     pertub_data_y = np.stack((per_x,do_y, per_cy), axis=1)
#     print(pertub_data_y.shape)
#     data_per_y = np.concatenate((o_data,pertub_data_y), axis = 0)
#     np.save('./linear_non-gaussian/case3_do(y).npy', data_per_y)
#     print(data_per_y.shape)

#     kci_obj = CIT(data_per_y, "kci")
#     pValue = kci_obj(0,2,set([]))
#     print(0,2,())
#     print(pValue)
#     if pValue > 0.05:
#         continue
#     pValue = kci_obj(0,2,set([1]))
#     print(0,2,(1))
#     print(pValue)
#     if pValue > 0.05:
#         break

#####################case 4 (casue with selection)########################
# for i in range(10):
#     startx = -1
#     endx = 2
#     starty = 3
#     endy = 5
#     startl = 7
#     endl = 9
#     starts = 2
#     ends = 4
#     num_sample = 4000
#     startn = 2
#     endn = 4
#     a = random.uniform(5,15)
#     print(a)
#     l1 = random.uniform(4,6)
#     l2 = random.uniform(4,6)
#     s1 = random.uniform(-1,3)
#     s2 = random.uniform(1,2)
#     Ex = np.random.uniform(startx,endx,num_sample)
#     Ey = np.random.uniform(starty,endy,num_sample)
#     El = np.random.uniform(startl,endl,num_sample)
#     Es = np.random.uniform(starts,ends,num_sample)
#     noise_x = np.random.uniform(startn,endn,num_sample)
#     noise_y = np.random.uniform(startn,endn,num_sample)

#     c = np.zeros(4000)
#     x = Ex 
#     y = Ey + a * Ex
#     mask = (s1 * x + s2 * y + Es > 0) & (s1 * x + s2 * y + Es < 30)
#     print(s1 * x + s2 * y + Es)
#     x = x[mask]
#     y = y[mask]
#     c = c[mask]
#     o_data = np.stack((x,y,c),axis=1)

#     per_cx = np.ones(4000)
#     Es_per_x = np.random.uniform(starts, ends,num_sample)
#     Ex_per = np.random.uniform(startx, endx,num_sample)
#     do_x = Ex_per + noise_x
#     Ey_per = np.random.uniform(starty,endy,num_sample)
#     per_y = Ey_per + do_x * a
#     mask_per_x = (s1 * do_x + s2 * per_y + Es_per_x > 0) & (s1 * do_x + s2 * per_y + Es_per_x < 30)
#     do_x = do_x[mask_per_x]
#     per_y = per_y[mask_per_x]
#     per_cx = per_cx[mask_per_x]
#     pertub_data_x = np.stack((do_x,per_y, per_cx), axis=1)
#     data_per_x = np.concatenate((o_data,pertub_data_x), axis = 0)
#     np.save('./linear_non-gaussian/case4_do(x).npy', data_per_x)
#     print(pertub_data_x.shape)
#     print(data_per_x.shape)

#     kci_obj = CIT(data_per_x, "kci")
#     pValue = kci_obj(1,2,set([]))
#     print(1,2,())
#     print(pValue)
#     if pValue > 0.05:
#         continue
#     pValue = kci_obj(1,2,set([0]))
#     print(1,2,(0))
#     print(pValue)
#     if pValue < 0.05:
#         continue


#     per_cy = np.ones(4000)
#     Es_per_y = np.random.uniform(starts, ends,num_sample)
#     Ey_per = np.random.uniform(starty,endy,num_sample)
#     do_y = Ey_per + noise_y
#     Ex_per = np.random.uniform(startx, endx,num_sample)
#     per_x = Ex_per
#     mask_per_y = (s1 * per_x + s2 * do_y + Es_per_y > 0) & (s1 * per_x + s2 * do_y + Es_per_y < 30)
#     per_x = per_x[mask_per_y]
#     do_y = do_y[mask_per_y]
#     per_cy = per_cy[mask_per_y]
#     pertub_data_y = np.stack((per_x,do_y, per_cy), axis=1)
#     data_per_y = np.concatenate((o_data,pertub_data_y), axis = 0)
#     np.save('./linear_non-gaussian/case4_do_a(y).npy', data_per_y)
#     print(pertub_data_y.shape)
#     print(data_per_y.shape)

#     kci_obj = CIT(data_per_y, "kci")
#     pValue = kci_obj(0,2,set([]))
#     print(0,2,())
#     print(pValue)
#     if pValue > 0.05:
#         continue
#     pValue = kci_obj(0,2,set([1]))
#     print(0,2,(1))
#     print(pValue)
#     if pValue < 0.05:
#         break

# #####################case 5 (casue with latent)########################
# for i in range(10):
#     startx = -1
#     endx = 2
#     starty = 3
#     endy = 5
#     startl = 7
#     endl = 9
#     starts = 2
#     ends = 4
#     num_sample = 4000
#     startn = 2
#     endn = 4
#     a = random.uniform(5,15)
#     print(a)
#     l1 = random.uniform(4,6)
#     l2 = random.uniform(4,6)
#     s1 = random.uniform(-1,3)
#     s2 = random.uniform(1,2)
#     Ex = np.random.uniform(startx,endx,num_sample)
#     Ey = np.random.uniform(starty,endy,num_sample)
#     El = np.random.uniform(startl,endl,num_sample)
#     Es = np.random.uniform(starts,ends,num_sample)
#     noise_x = np.random.uniform(startn,endn,num_sample)
#     noise_y = np.random.uniform(startn,endn,num_sample)

#     L = El
#     c = np.zeros(4000)
#     x = Ex + l1 * L
#     y = a * Ex + Ey + l2 * L
#     o_data = np.stack((x,y,c),axis=1)

#     per_cx = np.ones(4000)
#     Ex_per = np.random.uniform(startx, endx,num_sample)
#     Ey_per = np.random.uniform(starty,endy,num_sample)
#     El_per_x = np.random.uniform(startl, endl,num_sample)
#     do_x = Ex_per + noise_x + l1 * El_per_x
#     per_y = Ey_per + l2 * El_per_x + do_x * a
#     pertub_data_x = np.stack((do_x, per_y, per_cx), axis=1)
#     data_per_x = np.concatenate((o_data,pertub_data_x), axis = 0)
#     np.save('./linear_non-gaussian/case5_do_a(x).npy', data_per_x)

#     kci_obj = CIT(data_per_x, "kci")
#     pValue = kci_obj(1,2,set([]))
#     print(1,2,())
#     print(pValue)
#     if pValue > 0.05:
#         continue
#     pValue = kci_obj(1,2,set([0]))
#     print(1,2,(0))
#     print(pValue)
#     if pValue > 0.05:
#         continue

#     per_cy = np.ones(4000)
#     Ey_per = np.random.uniform(starty,endy,num_sample)
#     Ex_per = np.random.uniform(startx, endx,num_sample)
#     El_per_y = np.random.uniform(startl, endl,num_sample)
#     per_x = Ex_per + l1 * El_per_y
#     do_y = Ey_per + noise_y + l2 * El_per_y + a * per_x
#     perturb_data_y = np.stack((per_x,do_y,per_cy), axis = 1)
#     data_per_y = np.concatenate((o_data,perturb_data_y), axis = 0)
#     np.save('./linear_non-gaussian/case5_do_a(y).npy', data_per_y)

#     kci_obj = CIT(data_per_y, "kci")
#     pValue = kci_obj(0,2,set([]))
#     print(0,2,())
#     print(pValue)
#     if pValue < 0.05:
#         continue
#     pValue = kci_obj(0,2,set([1]))
#     print(0,2,(1))
#     print(pValue)
#     if pValue < 0.05:
#         break

# #####################case 6 (selection with latent)########################
# for i in range(10):
#     startx = -1
#     endx = 2
#     starty = 3
#     endy = 5
#     startl = 7
#     endl = 9
#     starts = 2
#     ends = 4
#     num_sample = 4000
#     startn = 2
#     endn = 4
#     a = random.uniform(5,15)
#     print(a)
#     l1 = random.uniform(4,6)
#     l2 = random.uniform(4,6)
#     s1 = random.uniform(-1,3)
#     s2 = random.uniform(1,2)
#     Ex = np.random.uniform(startx,endx,num_sample)
#     Ey = np.random.uniform(starty,endy,num_sample)
#     El = np.random.uniform(startl,endl,num_sample)
#     Es = np.random.uniform(starts,ends,num_sample)
#     noise_x = np.random.uniform(startn,endn,num_sample)
#     noise_y = np.random.uniform(startn,endn,num_sample)
#     L = El
#     c = np.zeros(4000)
#     x = Ex + l1 * L
#     y = Ey + l2 * L
#     mask = (s1 * x + s2 * y + Es > 60) & (s1 * x + s2 * y + Es < 130)
#     print(s1 * x + s2 * y + Es)
#     x = x[mask]
#     y = y[mask]
#     c = c[mask]
#     o_data = np.stack((x,y,c),axis=1)

#     per_cx = np.ones(4000)
#     Ex_per = np.random.uniform(startx, endx,num_sample)
#     El_per_x = np.random.uniform(startl, endl,num_sample)
#     Es_per_x = np.random.uniform(starts, ends,num_sample)
#     do_x = Ex_per + noise_x + l1 * El_per_x
#     Ey_per = np.random.uniform(starty,endy,num_sample)
#     per_y = Ey_per + l2 * El_per_x
#     mask_per_x = (s1 * do_x + s2 * per_y + Es_per_x > 60) & (s1 * do_x + s2 * per_y + Es_per_x < 130)
#     do_x = do_x[mask_per_x]
#     per_y = per_y[mask_per_x]
#     per_cx = per_cx[mask_per_x]
#     pertub_data_x = np.stack((do_x,per_y, per_cx), axis=1)
#     data_per_x = np.concatenate((o_data,pertub_data_x), axis = 0)
#     np.save('./linear_non-gaussian/case6_do(x)_a.npy', data_per_x)
#     print(pertub_data_x.shape)
#     print(data_per_x.shape)

#     kci_obj = CIT(data_per_x, "kci")
#     pValue = kci_obj(1,2,set([]))
#     print(1,2,())
#     print(pValue)
#     if np.isnan(pValue):
#         continue
#     if pValue > 0.05:
#         continue
#     pValue = kci_obj(1,2,set([0]))
#     print(1,2,(0))
#     print(pValue)
#     if pValue > 0.05:
#         continue

#     per_cy = np.ones(4000)
#     Ey_per = np.random.uniform(starty,endy,num_sample)
#     El_per_y = np.random.uniform(startl, endl,num_sample)
#     Es_per_y = np.random.uniform(starts, ends,num_sample)
#     do_y = Ey_per + noise_y + l2 * El_per_y
#     Ex_per = np.random.uniform(startx, endx,num_sample)
#     per_x = Ex_per + l1 * El_per_y
#     mask_per_y = (s1 * per_x + s2 * do_y + Es_per_y > 60) & (s1 * per_x + s2 * do_y + Es_per_y < 130)
#     print(s1 * per_x + s2 * do_y + Es_per_y)
#     per_x = per_x[mask_per_y]
#     do_y = do_y[mask_per_y]
#     per_cy = per_cy[mask_per_y]
#     pertub_data_y = np.stack((per_x,do_y, per_cy), axis=1)
#     data_per_y = np.concatenate((o_data,pertub_data_y), axis = 0)
#     np.save('./linear_non-gaussian/case6_do(y)_a.npy', data_per_y)
#     print(pertub_data_y.shape)
#     print(data_per_y.shape)

#     kci_obj = CIT(data_per_y, "kci")
#     pValue = kci_obj(0,2,set([]))
#     print(0,2,())
#     print(pValue)
#     if np.isnan(pValue):
#         continue
#     if pValue > 0.05:
#         continue
#     pValue = kci_obj(0,2,set([1]))
#     print(0,2,(1))
#     print(pValue)
#     if pValue < 0.05:
#         break
# #####################case 7 (cause with selection and latent)########################
# for i in range(10):
#     startx = -1
#     endx = 2
#     starty = 1
#     endy = 3
#     startl = 1
#     endl = 3
#     starts = 2
#     ends = 4
#     num_sample = 4000
#     startn = 2
#     endn = 4
#     a = random.uniform(10,15)
#     print(a)
#     l1 = random.uniform(1,3)
#     l2 = random.uniform(1,3)
#     s1 = random.uniform(2,3)
#     s2 = random.uniform(1,2)
#     Ex = np.random.uniform(startx,endx,num_sample)
#     Ey = np.random.uniform(starty,endy,num_sample)
#     El = np.random.uniform(startl,endl,num_sample)
#     Es = np.random.uniform(starts,ends,num_sample)
#     noise_x = np.random.uniform(startn,endn,num_sample)
#     noise_y = np.random.uniform(startn,endn,num_sample)

#     L = El
#     c = np.zeros(4000)
#     x = Ex + l1 * L
#     y = Ey + l2 * L + a * x
#     mask = (s1 * x + s2 * y + Es > 5) & (s1 * x + s2 * y + Es < 100)
#     print(s1 * x + s2 * y + Es)
#     x = x[mask]
#     y = y[mask]
#     c = c[mask]
#     o_data = np.stack((x,y,c),axis=1)
#     # print(o_data)

#     per_cx = np.ones(4000)
#     El_per_x = np.random.uniform(startl, endl,num_sample)
#     Es_per_x = np.random.uniform(starts,ends,num_sample)
#     Ex_per = np.random.uniform(startx, endx,num_sample)
#     do_x = Ex_per + noise_x + l1 * El_per_x
#     Ey_per = np.random.uniform(starty,endy,num_sample)
#     per_y = Ey_per + l2 * El_per_x + a * do_x
#     mask_per_x = (s1 * do_x + s2 * per_y + Es_per_x > 5) & (s1 * do_x + s2 * per_y + Es_per_x < 100)
#     print(s1 * do_x + s2 * per_y + Es_per_x )
#     do_x = do_x[mask_per_x]
#     per_y = per_y[mask_per_x]
#     per_cx = per_cx[mask_per_x]
#     pertub_data_x = np.stack((do_x,per_y, per_cx), axis=1)
#     data_per_x = np.concatenate((o_data,pertub_data_x), axis = 0)
#     np.save('./linear_non-gaussian/case7_do_a(x).npy', data_per_x)
#     print(pertub_data_x.shape)
#     print(data_per_x.shape)

#     kci_obj = CIT(data_per_x, "kci")
#     pValue = kci_obj(1,2,set([]))
#     print(1,2,())
#     print(pValue)
#     if pValue > 0.05:
#         continue
#     pValue = kci_obj(1,2,set([0]))
#     print(1,2,(0))
#     print(pValue)
#     if pValue > 0.05:
#         continue

#     per_cy = np.ones(4000)
#     El_per_y = np.random.uniform(startl, endl,num_sample)
#     Es_per_y = np.random.uniform(starts,ends,num_sample)
#     Ey_per = np.random.uniform(starty,endy,num_sample)
#     Ex_per = np.random.uniform(startx, endx,num_sample)
#     per_x = Ex_per + l1 * El_per_y
#     do_y = Ey_per + l2 * El_per_x + a * per_x + noise_y
#     mask_per_y = (s1 * per_x + s2 * do_y + Es_per_y > 5) & (s1 * per_x + s2 * do_y + Es_per_y < 100)
#     print(s1 * per_x + s2 * do_y + Es_per_y)
#     per_x = per_x[mask_per_y]
#     do_y = do_y[mask_per_y]
#     per_cy = per_cy[mask_per_y]
#     pertub_data_y = np.stack((per_x,do_y, per_cy), axis=1)
#     data_per_y = np.concatenate((o_data,pertub_data_y), axis = 0)
#     np.save('./linear_non-gaussian/case7_do(y).npy', data_per_y)
#     print(pertub_data_y.shape)
#     print(data_per_y.shape)

#     kci_obj = CIT(data_per_y, "kci")
#     pValue = kci_obj(0,2,set([]))
#     print(0,2,())
#     print(pValue)
#     if pValue > 0.05:
#         continue
#     pValue = kci_obj(0,2,set([1]))
#     print(0,2,(1))
#     print(pValue)
#     if pValue < 0.05:
#         break








############## non-linear gaussian ################

function = ['order2', 'order3', 'x_2', 'x_2_sinx', 'x_3', 'x_3_sinx', 'x_2_tanh', 'x_3_tanh']

def polynomial(cof,r,x):
    f = 0
    for i in range(r+1):
        f = f + cof[i] * x**i

    return f

def power_f(cof,r,x):
    return cof[0] * x**r
def combine1(cof, r,x):
    return cof[0] * x**r + np.sin(x)
def combine2(cof, r,x):
    return cof[0] * x**r + np.tanh(x)

def choose_function(low=0):
    a1 = random.uniform(low,2)
    a2 = random.uniform(low,2)
    a3 = random.uniform(low,2)
    b = random.uniform(low,2)
    cof = [b, a1 ,a2 ,a3]
    f_index = random.randint(0,7)
    f = function[f_index]
    print(f)
    if f == 'order2':
        non_linear = polynomial
        degree = 2
    elif f == 'order3':
        non_linear = polynomial
        degree = 3
    elif f =='x_2':
        non_linear = power_f
        degree =2
    elif f=='x_2_sinx':
        non_linear= combine1
        degree =2
    elif f == 'x_3':
        non_linear = power_f
        degree =3
    elif f=='x_3_sinx':
        non_linear = combine1
        degree = 3
    elif f == 'x_2_tanh':
        non_linear = combine2
        degree = 2
    elif f=='x_3_tanh':
        non_linear=combine2
        degree=3
    else:
        print('function not in the list')

    return non_linear, degree, cof
#########x-y ###########

####################case 1 (cause)#####################
# print('cause###########')
# for i in range(10):
#     mu = 2
#     mu1 = 3
#     mus = 5
#     sigma_x = random.uniform(0,2)
#     sigma_y = random.uniform(1,3)
#     sigma_l = random.uniform(1,3)
#     sigma_s = random.uniform(3,6)
#     sigma_n = random.uniform(1,2)
#     num_sample = 4000

#     # possible_values = np.array([0,1])
#     # probabilities = np.array([0.5, 0.5])
#     # random_sample = np.random.choice(possible_values, size=num_sample, p=probabilities)
#     # Ex = np.random.choice(possible_values, size=num_sample, p=probabilities)
#     # Ey = np.random.choice(possible_values, size=num_sample, p=probabilities)


#     Ex = np.random.normal(mu,sigma_x,num_sample)
#     Ey = np.random.normal(mu1,sigma_y,num_sample)
#     El = np.random.normal(mu,sigma_l,num_sample)
#     Es = np.random.normal(mus,sigma_s,num_sample)
#     Esy = np.random.normal(mu,sigma_s,num_sample)
#     # noise_x = np.random.choice(possible_values, size=num_sample, p=probabilities)
  
#     # noise_x = np.random.normal(3,sigma_n, num_sample)
#     # noise_y = np.random.normal(3,sigma_n, num_sample)
#     startn = 1
#     endn = 3
#     noise_x = np.random.uniform(startn,endn,num_sample)
#     noise_y = np.random.uniform(startn,endn,num_sample)

#     non_linear,de, cof = choose_function()
#     # non_linear = polynomial
#     # de =2

#     x = Ex
#     c = np.zeros(num_sample)
#     # parameters = initialize_parameters(input_size, hidden_size, output_size)
#     # y = forward_propagation(x.reshape(-1,1), parameters).reshape(-1) + Ey
#     # mlp_c.fit(x.reshape(-1,1), np.random.rand(x.shape[0]))
#     # y = mlp_c.predict(x.reshape(-1,1)).reshape(-1) + Ey
#     y = Ey + non_linear(cof, de, x)
#     # noise = np.zeros(num_sample)
#     o_data = np.stack((x,y,c), axis = 1)

#     per_cx = np.ones(num_sample)
#     # Ex_per = np.random.choice(possible_values, size=num_sample, p=probabilities)
#     Ex_per = np.random.uniform(mu,sigma_x,num_sample)
#     do_x = Ex_per + noise_x
#     # Ey_per = np.random.choice(possible_values, size=num_sample, p=probabilities)
#     Ey_per = np.random.normal(mu1,sigma_y,num_sample)
#     per_y = Ey_per + non_linear(cof, de, do_x)
#     pertub_data_x = np.stack((do_x,per_y, per_cx), axis=1)
#     data_per_x = np.concatenate((o_data,pertub_data_x), axis = 0)
#     # print(data_per_x.shape)
#     np.save('./non_linear_gaussian/case1_do_a(x).npy', data_per_x)

#     # condition = (data_per_x[:,0] == 0)
#     # data = data_per_x[condition]
#     # condition1 = (data[:,2] == 0)
#     # condition2 = (data[:,2] == 1)
    
#     # print(Counter(list(data[:,0].reshape(-1))))
#     # print(Counter(list(data[:,1].reshape(-1))))


#     # import pdb
#     # pdb.set_trace()
#     # plt.hist(data[condition1][:,1].reshape(-1))
#     # plt.show()
#     # plt.hist(data[condition2][:,1].reshape(-1))
#     # plt.show()
#     # # plt.hist(data[:,1].reshape(-1))
#     # plt.scatter(data[:,2].reshape(-1), data[:,1].reshape(-1))
#     # plt.show()



#     kci_obj = CIT(data_per_x, "kci")
#     pValue = kci_obj(1,2,set([]))
#     print(1,2,())
#     print(pValue)
#     if pValue >0.05:
#         continue
#     pValue = kci_obj(1,2,set([0]))
#     print(1,2,(0))
#     print(pValue)
#     if pValue < 0.05:
#         continue


#     per_cy = np.ones(num_sample)
#     Ex_per = np.random.normal(mu,sigma_x,num_sample)
#     per_x = Ex_per
#     Ey_per = np.random.normal(mu1,sigma_y,num_sample)
#     do_y =Ey_per + noise_y + non_linear(cof,de,per_x)
    
#     perturb_data_y = np.stack((per_x,do_y,per_cy), axis = 1)
#     data_per_y = np.concatenate((o_data,perturb_data_y), axis = 0)
#     np.save('./non_linear_gaussian/case1_do_a(y).npy', data_per_y)

#     kci_obj = CIT(data_per_y, "kci") # construct a CIT instance with data and method name

#     pValue = kci_obj(0,2,set([]))
#     print(0,2,())
#     print(pValue)
#     if pValue < 0.05:
#         continue
#     pValue = kci_obj(0,2,set([1]))
#     print(0,2,(1))
#     print(pValue)
#     if pValue < 0.05:
#         break


# ###################case 2 (latent)########################
# print('latent ############')
# for i in range(10):
#     mu = 4
#     mu1 = 10
#     mus = 3
#     sigma_x = random.uniform(0,2)
#     sigma_y = random.uniform(3,5)
#     sigma_l = random.uniform(1,3)
#     sigma_s = random.uniform(3,6)
#     num_sample = 4000

#     l1 = random.uniform(1,3)
#     l2 = random.uniform(1,2)
#     s1 = random.uniform(1,3)
#     s2 = random.uniform(2,5)

#     Ex = np.random.normal(mu,sigma_x,num_sample)
#     Ey = np.random.normal(mu1,sigma_y,num_sample)
#     El = np.random.normal(mu,sigma_l,num_sample)
#     Es = np.random.normal(mus,sigma_s,num_sample)
#     Esy = np.random.normal(mu,sigma_s,num_sample)
#     sigma_n = random.uniform(1,2)

#     # noise_x = np.random.normal(3,sigma_n, num_sample)
#     # noise_y = np.random.normal(3,sigma_n, num_sample)

#     startn = 1
#     endn = 3
#     noise_x = np.random.uniform(startn,endn,num_sample)
#     noise_y = np.random.uniform(startn,endn,num_sample)

#     L1, de1, cof1 = choose_function()
#     L2, de2, cof2 = choose_function()

#     L = El
#     c = np.zeros(4000)
#     x = Ex + L1(cof1, de1, L)
#     y = Ey + L2(cof2, de2, L)
#     o_data = np.stack((x,y,c),axis=1)

#     per_cx = np.ones(4000)
#     Ex_per = np.random.uniform(mu,sigma_x,num_sample)
#     Ey_per = np.random.normal(mu1,sigma_y,num_sample)
#     El_per_x = np.random.normal(mu,sigma_l,num_sample)
#     do_x = Ex_per + noise_x + L1(cof1, de1, El_per_x)
#     per_y = Ey_per + L2(cof2, de2, El_per_x)
#     pertub_data_x = np.stack((do_x,per_y, per_cx), axis=1)
#     data_per_x = np.concatenate((o_data,pertub_data_x), axis = 0)
#     np.save('./non_linear_gaussian/case2_do(x)_a.npy', data_per_x)
#     kci_obj = CIT(data_per_x, "kci")
#     pValue = kci_obj(1,2,set([]))
#     print(1,2,())
#     print(pValue)
#     if pValue < 0.05:
#         continue
#     pValue = kci_obj(1,2,set([0]))
#     print(1,2,(0))
#     print(pValue)
#     if pValue > 0.05:
#         continue

#     per_cy = np.ones(4000)
#     Ex_per = np.random.normal(mu,sigma_x,num_sample)
#     El_per_y = np.random.normal(mu,sigma_l,num_sample)
#     Ey_per = np.random.normal(mu1,sigma_y,num_sample)
#     do_y = Ey_per + noise_y + L2(cof2, de2, El_per_y)
#     per_x = Ex_per + L1(cof1, de1, El_per_y)
#     perturb_data_y = np.stack((per_x,do_y,per_cy), axis = 1)
#     data_per_y = np.concatenate((o_data,perturb_data_y), axis = 0)
#     np.save('./non_linear_gaussian/case2_do(y)_a.npy', data_per_y)

#     kci_obj = CIT(data_per_y, "kci")
#     pValue = kci_obj(0,2,set([]))
#     print(0,2,())
#     print(pValue)
#     if pValue < 0.05:
#         continue
#     pValue = kci_obj(0,2,set([1]))
#     print(0,2,(1))
#     print(pValue)
#     if pValue < 0.05:
#         break

# ###################case 3 (selection)########################
# for i in range(10):
#     mu = 2
#     mu1 = 4
#     mus = 3
#     sigma_x = random.uniform(0,2)
#     sigma_y = random.uniform(3,5)
#     sigma_l = random.uniform(1,3)
#     sigma_s = random.uniform(3,6)
#     num_sample = 4000

#     Ex = np.random.normal(mu,sigma_x,num_sample)
#     Ey = np.random.normal(mu1,sigma_y,num_sample)
#     El = np.random.normal(mu,sigma_l,num_sample)
#     Es = np.random.normal(mus,sigma_s,num_sample)
#     Esy = np.random.normal(mu,sigma_s,num_sample)
#     sigma_n = random.uniform(1,2)

#     # noise_x = np.random.normal(3,sigma_n, num_sample)
#     # noise_y = np.random.normal(3,sigma_n, num_sample)
#     startn = 1
#     endn = 3
#     noise_x = np.random.uniform(startn,endn,num_sample)
#     noise_y = np.random.uniform(startn,endn,num_sample)

#     S1, de1, cof1 = choose_function(0)
#     S2, de2, cof2 = choose_function(-1)

#     c = np.zeros(4000)
#     x = Ex
#     y = Ey
#     mask = (S1(cof1, de1,x) + S2(cof2, de2,y) + Es > 0) & (S1(cof1, de1,x) + S2(cof2, de2,y) + Es < 60)
#     print(S1(cof1, de1,x) + S2(cof2, de2,y) + Es)
#     x = x[mask]
#     y = y[mask]
#     c = c[mask]
#     o_data = np.stack((x,y,c),axis=1)

#     per_cx = np.ones(4000)
#     Es_per_x = np.random.normal(mu,sigma_s,num_sample)
#     Ex_per = np.random.normal(mu,sigma_x,num_sample)
#     do_x = Ex_per + noise_x
#     Ey_per = np.random.normal(mu1,sigma_y,num_sample)
#     per_y = Ey_per
#     mask_per_x = (S1(cof1, de1,do_x) + S2(cof2, de2,per_y) + Es_per_x > 0) & (S1(cof1, de1,do_x) + S2(cof2, de2,per_y) + Es_per_x < 60)
#     print(S1(cof1, de1,do_x) + S2(cof2, de2,per_y) + Es_per_x)
#     do_x = do_x[mask_per_x]
#     per_y = per_y[mask_per_x]
#     per_cx = per_cx[mask_per_x]
#     pertub_data_x = np.stack((do_x,per_y, per_cx), axis=1)
#     print(pertub_data_x.shape)
#     data_per_x = np.concatenate((o_data,pertub_data_x), axis = 0)
#     np.save('./non_linear_gaussian/case3_do_a(x).npy', data_per_x)
#     print(data_per_x.shape)
#     kci_obj = CIT(data_per_x, "kci")
#     pValue = kci_obj(1,2,set([]))
#     print(1,2,())
#     print(pValue)
#     if pValue > 0.05:
#         continue
#     pValue = kci_obj(1,2,set([0]))
#     print(1,2,(0))
#     print(pValue)
#     if pValue < 0.05:
#         continue

#     per_cy = np.ones(4000)
#     Es_per_y = np.random.normal(mu,sigma_s,num_sample)
#     Ey_per = np.random.normal(mu1,sigma_y,num_sample)
#     do_y = Ey_per + noise_y
#     Ex_per = np.random.normal(mu,sigma_x,num_sample)
#     per_x = Ex_per
#     mask_per_y = (S1(cof1, de1, per_x) + S2(cof2, de2, do_y) + Es_per_y > 0) & (S1(cof1, de1, per_x) + S2(cof2, de2, do_y) + Es_per_y < 60)
#     print(S1(cof1, de1, per_x) + S2(cof2, de2, do_y) + Es_per_y)
#     per_x = per_x[mask_per_y]
#     do_y = do_y[mask_per_y]
#     per_cy = per_cy[mask_per_y]
#     pertub_data_y = np.stack((per_x,do_y, per_cy), axis=1)
#     print(pertub_data_y.shape)
#     data_per_y = np.concatenate((o_data,pertub_data_y), axis = 0)
#     np.save('./non_linear_gaussian/case3_do(y)_a.npy', data_per_y)
#     print(data_per_y.shape)

#     kci_obj = CIT(data_per_y, "kci")
#     pValue = kci_obj(0,2,set([]))
#     print(0,2,())
#     print(pValue)
#     if pValue > 0.05:
#         continue
#     pValue = kci_obj(0,2,set([1]))
#     print(0,2,(1))
#     print(pValue)
#     if pValue > 0.05:
#         break

# ###################case 4 (casue with selection)########################
# for i in range(10):
#     mu = 1
#     mu1 = 2
#     mus = 3
#     sigma_x = random.uniform(0,2)
#     sigma_y = random.uniform(2,4)
#     sigma_l = random.uniform(1,3)
#     sigma_s = random.uniform(3,6)
#     num_sample = 4000

#     Ex = np.random.normal(mu,sigma_x,num_sample)
#     Ey = np.random.normal(mu1,sigma_y,num_sample)
#     El = np.random.normal(mu,sigma_l,num_sample)
#     Es = np.random.normal(mus,sigma_s,num_sample)
#     Esy = np.random.normal(mu,sigma_s,num_sample)

#     sigma_n = random.uniform(1,2)

#     # noise_x = np.random.normal(3,sigma_n, num_sample)
#     # noise_y = np.random.normal(3,sigma_n, num_sample)

#     startn = 2
#     endn = 4
#     noise_x = np.random.uniform(startn,endn,num_sample)
#     noise_y = np.random.uniform(startn,endn,num_sample)

#     S1, de1, cof1 = choose_function(-2)
#     S2, de2, cof2 = choose_function(-2)
#     non_linear,de, cof = choose_function(0)

#     c = np.zeros(4000)
#     x = Ex 
#     y = Ey + non_linear(cof, de, x)
#     mask = (S1(cof1, de1,x) + S2(cof2, de2,y) + Es > 5) & (S1(cof1, de1,x) + S2(cof2, de2,y) + Es < 100)
#     print(S1(cof1, de1,x) + S2(cof2, de2,y) + Es)
#     x = x[mask]
#     y = y[mask]
#     c = c[mask]
#     o_data = np.stack((x,y,c),axis=1)

#     per_cx = np.ones(4000)
#     Es_per_x = np.random.normal(mu,sigma_s,num_sample)
#     Ex_per = np.random.normal(mu,sigma_x,num_sample)
#     do_x = Ex_per + noise_x
#     Ey_per = np.random.normal(mu1,sigma_y,num_sample)
#     per_y = Ey_per + non_linear(cof,de,do_x)
#     mask_per_x = (S1(cof1, de1,do_x) + S2(cof2, de2,per_y) + Es_per_x > 5) & (S1(cof1, de1,do_x) + S2(cof2, de2,per_y) + Es_per_x < 100)
#     do_x = do_x[mask_per_x]
#     per_y = per_y[mask_per_x]
#     per_cx = per_cx[mask_per_x]
#     pertub_data_x = np.stack((do_x,per_y, per_cx), axis=1)
#     data_per_x = np.concatenate((o_data,pertub_data_x), axis = 0)
#     np.save('./non_linear_gaussian/case4_do_a(x).npy', data_per_x)
#     print(pertub_data_x.shape)
#     print(data_per_x.shape)

#     kci_obj = CIT(data_per_x, "kci")
#     pValue = kci_obj(1,2,set([]))
#     print(1,2,())
#     print(pValue)
#     if pValue > 0.05:
#         continue
#     pValue = kci_obj(1,2,set([0]))
#     print(1,2,(0))
#     print(pValue)
#     if pValue < 0.05:
#         continue


#     per_cy = np.ones(4000)
#     Es_per_y = np.random.normal(mu,sigma_s,num_sample)
#     Ey_per = np.random.normal(mu1,sigma_y,num_sample)
#     Ex_per = np.random.normal(mu,sigma_x,num_sample)
#     per_x = Ex_per
#     do_y = Ey_per + noise_y + non_linear(cof,de,per_x)
#     mask_per_y = (S1(cof1, de1, per_x) + S2(cof2, de2,do_y) + Es_per_y > 5) & (S1(cof1, de1,per_x) + S2(cof2, de2,do_y) + Es_per_y < 100)
#     per_x = per_x[mask_per_y]
#     do_y = do_y[mask_per_y]
#     per_cy = per_cy[mask_per_y]
#     pertub_data_y = np.stack((per_x,do_y, per_cy), axis=1)
#     data_per_y = np.concatenate((o_data,pertub_data_y), axis = 0)
#     np.save('./non_linear_gaussian/case4_do_a(y).npy', data_per_y)
#     print(pertub_data_y.shape)
#     print(data_per_y.shape)


#     kci_obj = CIT(data_per_y, "kci")
#     pValue = kci_obj(0,2,set([]))
#     print(0,2,())
#     print(pValue)
#     if pValue > 0.05:
#         continue
#     pValue = kci_obj(0,2,set([1]))
#     print(0,2,(1))
#     print(pValue)
#     if pValue < 0.05:
#         break

# #####################case 5 (casue with latent)########################
# for i in range(10):
#     mu = 1
#     mu1 = 2
#     mus = 3
#     sigma_x = random.uniform(0,2)
#     sigma_y = random.uniform(2,4)
#     sigma_l = random.uniform(1,3)
#     sigma_s = random.uniform(3,6)
#     num_sample = 4000

#     Ex = np.random.normal(mu,sigma_x,num_sample)
#     Ey = np.random.normal(mu1,sigma_y,num_sample)
#     El = np.random.normal(mu,sigma_l,num_sample)
#     Es = np.random.normal(mus,sigma_s,num_sample)

#     sigma_n = random.uniform(1,2)

#     # noise_x = np.random.normal(3,sigma_n, num_sample)
#     # noise_y = np.random.normal(3,sigma_n, num_sample)
#     startn = 2
#     endn = 4
#     noise_x = np.random.uniform(startn,endn,num_sample)
#     noise_y = np.random.uniform(startn,endn,num_sample)

#     L1, de1, cof1= choose_function()
#     L2, de2, cof2 = choose_function()
#     non_linear,de, cof_c = choose_function()


#     L = El
#     c = np.zeros(4000)
#     x = Ex + L1(cof1,de1,El)
#     y = Ey + L2(cof2,de2,El) + non_linear(cof_c, de, x)
#     o_data = np.stack((x,y,c),axis=1)

#     per_cx = np.ones(4000)
#     Ex_per = np.random.normal(mu,sigma_x,num_sample)
#     Ey_per = np.random.normal(mu1,sigma_y,num_sample)
#     El_per_x = np.random.normal(mu,sigma_l,num_sample)
#     do_x = Ex_per + noise_x + L1(cof1,de1,El_per_x)
#     per_y = Ey_per + non_linear(cof_c,de, do_x) + L2(cof2,de2,El_per_x)
#     pertub_data_x = np.stack((do_x, per_y, per_cx), axis=1)
#     data_per_x = np.concatenate((o_data,pertub_data_x), axis = 0)
#     np.save('./non_linear_gaussian/case5_do_a(x).npy', data_per_x)

#     kci_obj = CIT(data_per_x, "kci")
#     pValue = kci_obj(1,2,set([]))
#     print(1,2,())
#     print(pValue)
#     if pValue > 0.05:
#         continue
#     pValue = kci_obj(1,2,set([0]))
#     print(1,2,(0))
#     print(pValue)
#     if pValue > 0.05:
#         continue

#     per_cy = np.ones(4000)
#     Ey_per = np.random.normal(mu1,sigma_y,num_sample)
#     Ex_per = np.random.normal(mu,sigma_x,num_sample)
#     El_per_y = np.random.normal(mu,sigma_l,num_sample)
#     per_x = Ex_per + L1(cof1,de1,El_per_y)
#     do_y = Ey_per + noise_y + L2(cof2,de2,El_per_y) + non_linear(cof_c,de, per_x)
#     perturb_data_y = np.stack((per_x,do_y,per_cy), axis = 1)
#     data_per_y = np.concatenate((o_data,perturb_data_y), axis = 0)
#     np.save('./non_linear_gaussian/case5_do_a(y).npy', data_per_y)

#     kci_obj = CIT(data_per_y, "kci")
#     pValue = kci_obj(0,2,set([]))
#     print(0,2,())
#     print(pValue)
#     if pValue < 0.05:
#         continue
#     pValue = kci_obj(0,2,set([1]))
#     print(0,2,(1))
#     print(pValue)
#     if pValue < 0.05:
#         break

# ####################case 6 (selection with latent)########################
# for i in range(10):
#     mu = 1
#     mu1 = 2
#     mus = 3
#     sigma_x = random.uniform(0,2)
#     sigma_y = random.uniform(2,4)
#     sigma_l = random.uniform(1,3)
#     sigma_s = random.uniform(3,6)
#     num_sample = 4000

#     Ex = np.random.normal(mu,sigma_x,num_sample)
#     Ey = np.random.normal(mu1,sigma_y,num_sample)
#     El = np.random.normal(mu,sigma_l,num_sample)
#     Es = np.random.normal(mus,sigma_s,num_sample)

#     sigma_n = random.uniform(1,2)

#     # noise_x = np.random.normal(3,sigma_n, num_sample)
#     # noise_y = np.random.normal(3,sigma_n, num_sample)

#     startn = 2
#     endn = 4
#     noise_x = np.random.uniform(startn,endn,num_sample)
#     noise_y = np.random.uniform(startn,endn,num_sample)

#     L1, de1, cof_l1 = choose_function()
#     L2, de2, cof_l2 = choose_function()

#     S1, ds1, cof_s1 = choose_function()
#     S2, ds2, cof_s2 = choose_function(-1)

#     L = El
#     c = np.zeros(4000)
#     x = Ex + L1(cof_l1, de1, L)
#     y = Ey + L2(cof_l2, de2, L)
#     mask = (S1(cof_s1, ds1,x) + S2(cof_s2, ds2,y) + Es > 5) & (S1(cof_s1, ds1,x) + S2(cof_s2, ds2,y) + Es < 50)
#     x = x[mask]
#     y = y[mask]
#     c = c[mask]
#     o_data = np.stack((x,y,c),axis=1)

#     per_cx = np.ones(4000)
#     El_per_x = np.random.normal(mu,sigma_l,num_sample)
#     Es_per_x = np.random.normal(mu,sigma_s,num_sample)
#     Ex_per = np.random.normal(mu,sigma_x,num_sample)
#     do_x = Ex_per + noise_x + L1(cof_l1,de1, El_per_x)
#     Ey_per = np.random.normal(mu1,sigma_y,num_sample)
#     per_y = Ey_per + L2(cof_l2, de2, El_per_x)
#     mask_per_x = (S1(cof_s1, ds1,do_x) + S2(cof_s2, ds2,per_y) + Es_per_x > 5) & (S1(cof_s1, ds1,do_x) + S2(cof_s2, ds2,per_y) + Es_per_x < 50)
#     do_x = do_x[mask_per_x]
#     per_y = per_y[mask_per_x]
#     per_cx = per_cx[mask_per_x]
#     pertub_data_x = np.stack((do_x,per_y, per_cx), axis=1)
#     data_per_x = np.concatenate((o_data,pertub_data_x), axis = 0)
#     np.save('./non_linear_gaussian/case6_do_a(x).npy', data_per_x)
#     kci_obj = CIT(data_per_x, "kci", est_width='median')
#     pValue = kci_obj(1,2,set([]))
#     print(1,2,())
#     print(pValue)
#     if np.isnan(pValue):
#         continue
#     if pValue > 0.05:
#         continue
#     pValue = kci_obj(1,2,set([0]))
#     print(1,2,(0))
#     print(pValue)
#     if pValue > 0.05:
#         continue

#     per_cy = np.ones(4000)
#     El_per_y = np.random.normal(mu,sigma_l,num_sample)
#     Es_per_y = np.random.normal(mu,sigma_s,num_sample)
#     Ey_per = np.random.normal(mu1,sigma_y,num_sample)
#     do_y = Ey_per + noise_y + L2(cof_l2, de2, El_per_y)
#     Ex_per = np.random.normal(mu,sigma_x,num_sample)
#     per_x = Ex_per + L1(cof_l1, de1, El_per_y)
#     mask_per_y = (S1(cof_s1, ds1,per_x) + S2(cof_s2, ds2, do_y) + Es_per_y > 5) & (S1(cof_s1, ds1,per_x) + S2(cof_s2, ds2, do_y) + Es_per_y < 50)
#     per_x = per_x[mask_per_y]
#     do_y = do_y[mask_per_y]
#     per_cy = per_cy[mask_per_y]
#     pertub_data_y = np.stack((per_x,do_y, per_cy), axis=1)
#     data_per_y = np.concatenate((o_data,pertub_data_y), axis = 0)
#     np.save('./non_linear_gaussian/case6_do(y)_a.npy', data_per_y)

#     kci_obj = CIT(data_per_y, "kci", est_width='median')
#     pValue = kci_obj(0,2,set([]))
#     print(0,2,())
#     print(pValue)
#     if np.isnan(pValue):
#         continue
#     if pValue > 0.05:
#         continue
#     pValue = kci_obj(0,2,set([1]))
#     print(0,2,(1))
#     print(pValue)
#     if pValue < 0.05:
#         break

# ####################case 7 (cause with selection and latent)########################
# for i in range(10):
#     mu = 0
#     mu1 = 3
#     mus = 3
#     sigma_x = random.uniform(0,1)
#     sigma_y = random.uniform(1,3)
#     sigma_l = random.uniform(1,2)
#     sigma_s = random.uniform(2,3)
#     num_sample = 4000

#     Ex = np.random.normal(mu,sigma_x,num_sample)
#     Ey = np.random.normal(mu1,sigma_y,num_sample)
#     El = np.random.normal(mu,sigma_l,num_sample)
#     Es = np.random.normal(mus,sigma_s,num_sample)

#     sigma_n = random.uniform(1,2)
#     # noise_x = np.random.normal(3,sigma_n, num_sample)
#     # noise_y = np.random.normal(3,sigma_n, num_sample)

#     startn = 2
#     endn = 4
#     noise_x = np.random.uniform(startn,endn,num_sample)
#     noise_y = np.random.uniform(startn,endn,num_sample)

#     L1, de1, cof_l1 = choose_function()
#     L2, de2, cof_l2 = choose_function()

#     S1, ds1, cof_s1 = choose_function()
#     S2, ds2, cof_s2 = choose_function(-2)
#     non_linear,de, cof_c = choose_function()

#     L = El
#     c = np.zeros(4000)
#     x = Ex + L1(cof_l1,de1,L)
#     y = Ey + L2(cof_l2,de2,L) + non_linear(cof_c,de,x)
#     mask = (S1(cof_s1, ds1, x) + S2(cof_s2, ds2, y) + Es > 0) & (S1(cof_s1, ds1, x) + S2(cof_s2, ds2, y) + Es < 100)
#     print(S1(cof_s1, ds1, x) + S2(cof_s2, ds2, y) + Es)
#     x = x[mask]
#     y = y[mask]
#     c = c[mask]
#     o_data = np.stack((x,y,c),axis=1)
#     # print(o_data)

#     per_cx = np.ones(4000)
#     El_per_x = np.random.normal(mu,sigma_l,num_sample)
#     Es_per_x = np.random.normal(mus,sigma_s,num_sample)
#     Ex_per = np.random.normal(mu,sigma_x,num_sample)
#     do_x = Ex_per + noise_x + L1(cof_l1,de1, El_per_x)
#     Ey_per = np.random.normal(mu1,sigma_y,num_sample)
#     per_y = Ey_per + L2(cof_l2,de2, El_per_x) + non_linear(cof_c,de,do_x)
#     mask_per_x = (S1(cof_s1, ds1,do_x) + S2(cof_s2, ds2,per_y) + Es_per_x > 0) & (S1(cof_s1, ds1,do_x) + S2(cof_s2, ds2,per_y) + Es_per_x < 100)
#     print(S1(cof_s1, ds1,do_x) + S2(cof_s2, ds2,per_y) + Es_per_x)
#     do_x = do_x[mask_per_x]
#     per_y = per_y[mask_per_x]
#     per_cx = per_cx[mask_per_x]
#     pertub_data_x = np.stack((do_x,per_y, per_cx), axis=1)
#     data_per_x = np.concatenate((o_data,pertub_data_x), axis = 0)
#     np.save('./non_linear_gaussian/case7_do(x)_a.npy', data_per_x)
#     print(pertub_data_x.shape)
#     print(data_per_x.shape)

#     kci_obj = CIT(data_per_x, "kci")
#     pValue = kci_obj(1,2,set([]))
#     print(1,2,())
#     if np.isnan(pValue):
#         continue
#     print(pValue)
#     if pValue > 0.05:
#         continue
#     pValue = kci_obj(1,2,set([0]))
#     print(1,2,(0))
#     print(pValue)
#     if pValue > 0.05:
#         continue

#     per_cy = np.ones(4000)
#     El_per_y = np.random.normal(mu,sigma_l,num_sample)
#     Es_per_y = np.random.normal(mus,sigma_s,num_sample)
#     Ey_per = np.random.normal(mu1,sigma_y,num_sample)
#     Ex_per = np.random.normal(mu,sigma_x,num_sample)
#     per_x = Ex_per + L1(cof_l1,de1,El_per_y)
#     do_y = Ey_per + noise_y + L2(cof_l2,de2, El_per_y) + non_linear(cof_c,de,per_x)

#     mask_per_y = (S1(cof_s1, ds1,per_x) + S2(cof_s2, ds2, do_y) + Es_per_y > 0) & (S1(cof_s1, ds1,per_x) + S2(cof_s2, ds2, do_y) + Es_per_y < 100)
#     print(S1(cof_s1, ds1,per_x) + S2(cof_s2, ds2, do_y) + Es_per_y)
#     per_x = per_x[mask_per_y]
#     do_y = do_y[mask_per_y]
#     per_cy = per_cy[mask_per_y]
#     pertub_data_y = np.stack((per_x,do_y, per_cy), axis=1)
#     data_per_y = np.concatenate((o_data,pertub_data_y), axis = 0)
#     np.save('./non_linear_gaussian/case7_do_a(y).npy', data_per_y)
#     print(pertub_data_y.shape)
#     print(data_per_y.shape)

#     kci_obj = CIT(data_per_y, "kci")
#     pValue = kci_obj(0,2,set([]))
#     if np.isnan(pValue):
#         continue
#     print(0,2,())
#     print(pValue)
#     if pValue > 0.05:
#         continue
#     pValue = kci_obj(0,2,set([1]))
#     print(0,2,(1))
#     print(pValue)
#     if pValue < 0.05:
#         break





############## non-linear non-gaussian ################
#########x-y ###########
####################case 1 (cause)#####################
# for i in range(10):
#     startx = 1
#     endx = 3
#     starty = 3
#     endy = 6
#     startl = 7
#     endl = 9
#     starts = 2
#     ends = 4
#     startn = 2
#     endn = 4
#     num_sample = 4000

#     Ex = np.random.uniform(startx,endx,num_sample)
#     Ey = np.random.uniform(starty,endy,num_sample)
#     El = np.random.uniform(startl,endl,num_sample)
#     Es = np.random.uniform(starts,ends,num_sample)
#     noise_x = np.random.uniform(startn,endn,num_sample)
#     noise_y = np.random.uniform(startn,endn,num_sample)
#     non_linear,de, cof = choose_function()

#     x = Ex
#     c = np.zeros(4000)
#     y = non_linear(cof, de, x) + Ey
#     o_data =  np.stack((x,y,c), axis = 1)

#     per_cx = np.ones(num_sample)
#     Ex_per = np.random.uniform(startx,endx,num_sample)
#     do_x = Ex_per + noise_x
#     Ey_per = np.random.uniform(starty,endy,num_sample)
#     per_y = Ey_per + non_linear(cof, de, do_x)
#     pertub_data_x = np.stack((do_x,per_y, per_cx), axis=1)
#     data_per_x = np.concatenate((o_data,pertub_data_x), axis = 0)
#     np.save('./non_linear_nongaussian/case1_do_a(x).npy', data_per_x)
#     # print(pertub_data_x)
#     kci_obj = CIT(data_per_x, "kci")
#     pValue = kci_obj(1,2,set([]))
#     print(1,2,())
#     print(pValue)
#     if pValue >0.05:
#         continue
#     pValue = kci_obj(1,2,set([0]))
#     print(1,2,(0))
#     print(pValue)
#     if pValue < 0.05:
#         continue

#     per_cy = np.ones(num_sample)
#     Ey_per = np.random.uniform(starty,endy,num_sample)
#     Ex_per = np.random.uniform(startx,endx,num_sample)
#     per_x = Ex_per
#     do_y = Ey_per + noise_y + non_linear(cof, de, per_x)
#     perturb_data_y = np.stack((per_x,do_y,per_cy), axis = 1)
#     data_per_y = np.concatenate((o_data,perturb_data_y), axis = 0)
#     np.save('./non_linear_nongaussian/case1_do_a(y).npy', data_per_y)

#     kci_obj = CIT(data_per_y, "kci") # construct a CIT instance with data and method name

#     pValue = kci_obj(0,2,set([]))
#     print(0,2,())
#     print(pValue)
#     if pValue < 0.05:
#         continue
#     pValue = kci_obj(0,2,set([1]))
#     print(0,2,(1))
#     print(pValue)
#     if pValue < 0.05:
#         break

####################case 2 (latent)########################
# for i in range(10):

#     startx = 1
#     endx = 3
#     starty = 3
#     endy = 6
#     startl = 7
#     endl = 9
#     starts = 2
#     ends = 4
#     startn = 2
#     endn = 4
#     num_sample = 4000

#     Ex = np.random.uniform(startx,endx,num_sample)
#     Ey = np.random.uniform(starty,endy,num_sample)
#     El = np.random.uniform(startl,endl,num_sample)
#     Es = np.random.uniform(starts,ends,num_sample)
#     noise_x = np.random.uniform(startn,endn,num_sample)
#     noise_y = np.random.uniform(startn,endn,num_sample)
    
#     L1, de1, cof1 = choose_function()
#     L2, de2, cof2 = choose_function()

#     L = El
#     c = np.zeros(4000)
#     x = Ex + L1(cof1,de1,L)
#     y = Ey + L2(cof2,de2,L)
#     o_data = np.stack((x,y,c),axis=1)

#     per_cx = np.ones(4000)
#     Ex_per = np.random.uniform(startx,endx,num_sample)
#     El_per_x = np.random.uniform(startl,endl,num_sample)
#     do_x = Ex_per + noise_x + L1(cof1,de1,El_per_x)
#     Ey_per = np.random.uniform(starty,endy,num_sample)
#     per_y = Ey_per + L2(cof2,de2,El_per_x)
#     pertub_data_x = np.stack((do_x,per_y, per_cx), axis=1)
#     data_per_x = np.concatenate((o_data,pertub_data_x), axis = 0)
#     np.save('./non_linear_nongaussian/case2_do_a(x).npy', data_per_x)

#     kci_obj = CIT(data_per_x, "kci")
#     pValue = kci_obj(1,2,set([]))
#     print(1,2,())
#     print(pValue)
#     if pValue < 0.05:
#         continue
#     pValue = kci_obj(1,2,set([0]))
#     print(1,2,(0))
#     print(pValue)
#     if pValue > 0.05:
#         continue

#     per_cy = np.ones(4000)
#     Ey_per = np.random.uniform(starty,endy,num_sample)
#     Ex_per = np.random.uniform(startx,endx,num_sample)
#     El_per_y = np.random.uniform(startl,endl,num_sample)
#     do_y = Ey_per + noise_y + L2(cof2,de2,El_per_y)
#     per_x = Ex_per + L1(cof1,de1,El_per_y)
#     perturb_data_y = np.stack((per_x,do_y,per_cy), axis = 1)
#     data_per_y = np.concatenate((o_data,perturb_data_y), axis = 0)
#     np.save('./non_linear_nongaussian/case2_do_a(y).npy', data_per_y)

#     kci_obj = CIT(data_per_y, "kci")
#     pValue = kci_obj(0,2,set([]))
#     print(0,2,())
#     print(pValue)
#     if pValue < 0.05:
#         continue
#     pValue = kci_obj(0,2,set([1]))
#     print(0,2,(1))
#     print(pValue)
#     if pValue < 0.05:
#         break

####################case 3 (selection)########################
# for i in range(10):
#     startx = 1
#     endx = 3
#     starty = 3
#     endy = 6
#     startl = 7
#     endl = 9
#     starts = 2
#     ends = 4
#     startn = 2
#     endn = 4
#     num_sample = 4000

#     Ex = np.random.uniform(startx,endx,num_sample)
#     Ey = np.random.uniform(starty,endy,num_sample)
#     El = np.random.uniform(startl,endl,num_sample)
#     Es = np.random.uniform(starts,ends,num_sample)
#     noise_x = np.random.uniform(startn,endn,num_sample)
#     noise_y = np.random.uniform(startn,endn,num_sample)

#     S1, de1, cof1 = choose_function(0)
#     S2, de2, cof2 = choose_function(-1)

#     c = np.zeros(4000)
#     x = Ex
#     y = Ey
#     mask = (S1(cof1, de1,x) + S2(cof2, de2,y)+ Es > 5) & (S1(cof1, de1,x) + S2(cof2, de2,y) + Es < 80)
#     print(S1(cof1, de1,x) + S2(cof2, de2,y) + Es)
#     x = x[mask]
#     y = y[mask]
#     c = c[mask]
#     o_data = np.stack((x,y,c),axis=1)

#     per_cx = np.ones(4000)
#     Ex_per = np.random.uniform(startx,endx,num_sample)
#     Es_per_x = np.random.uniform(starts,ends,num_sample)
#     do_x = Ex_per + noise_x
#     Ey_per = np.random.uniform(starty,endy,num_sample)
#     per_y = Ey_per
#     mask_per_x = (S1(cof1, de1,do_x) + S2(cof2, de2,per_y) + Es_per_x > 5) & (S1(cof1, de1,do_x) + S2(cof2, de2,per_y) + Es_per_x < 80)
#     print(S1(cof1, de1,do_x) + S2(cof2, de2,per_y) + Es_per_x)
#     do_x = do_x[mask_per_x]
#     per_y = per_y[mask_per_x]
#     per_cx = per_cx[mask_per_x]
#     pertub_data_x = np.stack((do_x,per_y, per_cx), axis=1)
#     print(pertub_data_x.shape)
#     data_per_x = np.concatenate((o_data,pertub_data_x), axis = 0)
#     np.save('./non_linear_nongaussian/case3_do_a(x)_a.npy', data_per_x)

#     print(data_per_x.shape)
#     kci_obj = CIT(data_per_x, "kci")
#     pValue = kci_obj(1,2,set([]))
#     print(1,2,())
#     print(pValue)
#     if pValue > 0.05:
#         continue
#     pValue = kci_obj(1,2,set([0]))
#     print(1,2,(0))
#     print(pValue)
#     if pValue < 0.05:
#         continue


#     per_cy = np.ones(4000)
#     Ey_per = np.random.uniform(starty,endy,num_sample)
#     Es_per_y = np.random.uniform(starts,ends,num_sample)
#     do_y = Ey_per + noise_y
#     Ex_per = np.random.uniform(startx,endx,num_sample)
#     per_x = Ex_per
#     mask_per_y = (S1(cof1, de1, per_x) + S2(cof2, de2, do_y) + Es_per_y > 5) & (S1(cof1, de1, per_x) + S2(cof2, de2, do_y) + Es_per_y < 80)
#     print(S1(cof1, de1, per_x) + S2(cof2, de2, do_y) + Es_per_y)
#     per_x = per_x[mask_per_y]
#     do_y = do_y[mask_per_y]
#     per_cy = per_cy[mask_per_y]
#     pertub_data_y = np.stack((per_x,do_y, per_cy), axis=1)
#     print(pertub_data_y.shape)
#     data_per_y = np.concatenate((o_data,pertub_data_y), axis = 0)
#     np.save('./non_linear_nongaussian/case3_do(y).npy', data_per_y)
#     print(data_per_y.shape)

#     kci_obj = CIT(data_per_y, "kci")
#     pValue = kci_obj(0,2,set([]))
#     print(0,2,())
#     print(pValue)
#     if pValue > 0.05:
#         continue
#     pValue = kci_obj(0,2,set([1]))
#     print(0,2,(1))
#     print(pValue)
#     if pValue > 0.05:
#         break

####################case 4 (casue with selection)########################
# for i in range(10):
#     startx = 1
#     endx = 2
#     starty = 3
#     endy = 6
#     startl = 7
#     endl = 9
#     starts = 2
#     ends = 4
#     startn = 2
#     endn = 4
#     num_sample = 4000

#     Ex = np.random.uniform(startx,endx,num_sample)
#     Ey = np.random.uniform(starty,endy,num_sample)
#     El = np.random.uniform(startl,endl,num_sample)
#     Es = np.random.uniform(starts,ends,num_sample)
#     noise_x = np.random.uniform(startn,endn,num_sample)
#     noise_y = np.random.uniform(startn,endn,num_sample)

#     S1, de1, cof1 = choose_function()
#     S2, de2, cof2 = choose_function(-2)
#     non_linear, de, cof = choose_function()

#     c = np.zeros(4000)
#     x = Ex 
#     y = Ey + non_linear(cof, de, x)
#     mask = (S1(cof1, de1,x) + S2(cof2, de2,y) + Es > 5) & (S1(cof1, de1,x) + S2(cof2, de2,y) + Es < 70)
#     print(S1(cof1, de1,x) + S2(cof2, de2,y) + Es)
#     x = x[mask]
#     y = y[mask]
#     c = c[mask]
#     o_data = np.stack((x,y,c),axis=1)

#     per_cx = np.ones(4000)
#     Ex_per = np.random.uniform(startx,endx,num_sample)
#     Es_per_x = np.random.uniform(starts,ends,num_sample)
#     do_x = Ex_per + noise_x
#     Ey_per = np.random.uniform(starty,endy,num_sample)
#     per_y = Ey_per + non_linear(cof, de, do_x)
#     mask_per_x = (S1(cof1, de1, do_x) + S2(cof2, de2, per_y) + Es_per_x > 5) & (S1(cof1, de1, do_x) + S2(cof2, de2, per_y) + Es_per_x < 70)
#     print(S1(cof1, de1, do_x) + S2(cof2, de2, per_y) + Es_per_x)
#     do_x = do_x[mask_per_x]
#     per_y = per_y[mask_per_x]
#     per_cx = per_cx[mask_per_x]
#     pertub_data_x = np.stack((do_x,per_y, per_cx), axis=1)
#     data_per_x = np.concatenate((o_data,pertub_data_x), axis = 0)
#     np.save('./non_linear_nongaussian/case4_do_a(x).npy', data_per_x)
#     print(pertub_data_x.shape)
#     print(data_per_x.shape)

#     kci_obj = CIT(data_per_x, "kci")
#     pValue = kci_obj(1,2,set([]))
#     print(1,2,())
#     print(pValue)
#     if np.isnan(pValue):
#         continue
#     if pValue > 0.05:
#         continue
#     pValue = kci_obj(1,2,set([0]))
#     print(1,2,(0))
#     print(pValue)
#     if pValue < 0.05:
#         continue

#     per_cy = np.ones(4000)
#     Es_per_y = np.random.uniform(starts,ends,num_sample)
#     Ey_per = np.random.uniform(starty,endy,num_sample)
#     Ex_per = np.random.uniform(startx,endx,num_sample)
#     per_x = Ex_per
#     do_y = Ey_per + noise_y + non_linear(cof,de,per_x)
#     mask_per_y = (S1(cof1, de1,per_x) + S2(cof2, de2,do_y) + Es_per_y > 5) & (S1(cof1, de1,per_x) + S2(cof2, de2,do_y) + Es_per_y < 70)
#     print(S1(cof1, de1,per_x) + S2(cof2, de2,do_y) + Es_per_y)
#     per_x = per_x[mask_per_y]
#     do_y = do_y[mask_per_y]
#     per_cy = per_cy[mask_per_y]
#     pertub_data_y = np.stack((per_x,do_y, per_cy), axis=1)
#     data_per_y = np.concatenate((o_data,pertub_data_y), axis = 0)
#     np.save('./non_linear_nongaussian/case4_do_a(y).npy', data_per_y)
#     print(pertub_data_y.shape)
#     print(data_per_y.shape)

#     kci_obj = CIT(data_per_y, "kci")
#     pValue = kci_obj(0,2,set([]))
#     print(0,2,())
#     print(pValue)
#     if pValue > 0.05:
#         continue
#     pValue = kci_obj(0,2,set([1]))
#     print(0,2,(1))
#     print(pValue)
#     if pValue < 0.05:
#         break

# #####################case 5 (casue with latent)########################
# for i in range(10):
#     startx = 1
#     endx = 3
#     starty = 3
#     endy = 6
#     startl = 7
#     endl = 9
#     starts = 2
#     ends = 4
#     startn = 2
#     endn = 4
#     num_sample = 4000

#     Ex = np.random.uniform(startx,endx,num_sample)
#     Ey = np.random.uniform(starty,endy,num_sample)
#     El = np.random.uniform(startl,endl,num_sample)
#     Es = np.random.uniform(starts,ends,num_sample)

#     noise_x = np.random.uniform(startn,endn,num_sample)
#     noise_y = np.random.uniform(startn,endn,num_sample)

#     L1, de1, cof1= choose_function()
#     L2, de2, cof2 = choose_function()
#     non_linear,de, cof_c = choose_function()

#     L = El
#     c = np.zeros(4000)
#     x = Ex + L1(cof1,de1,L)
#     y = Ey + non_linear(cof_c,de,x) + L2(cof2,de2,L)
#     o_data = np.stack((x,y,c),axis=1)

#     per_cx = np.ones(4000)
#     El_per_x = np.random.uniform(startl,endl,num_sample)
#     Ex_per = np.random.uniform(startx,endx,num_sample)
#     do_x = Ex_per + noise_x + L1(cof1,de1,El_per_x)
#     Ey_per = np.random.uniform(starty,endy,num_sample)
#     per_y = Ey_per + non_linear(cof_c,de,do_x) + L2(cof2,de2,El_per_x)
#     pertub_data_x = np.stack((do_x, per_y, per_cx), axis=1)
#     data_per_x = np.concatenate((o_data,pertub_data_x), axis = 0)
#     np.save('./non_linear_nongaussian/case5_do(x)_a.npy', data_per_x)

#     kci_obj = CIT(data_per_x, "kci")
#     pValue = kci_obj(1,2,set([]))
#     print(1,2,())
#     print(pValue)
#     if pValue > 0.05:
#         continue
#     pValue = kci_obj(1,2,set([0]))
#     print(1,2,(0))
#     print(pValue)
#     if pValue > 0.05:
#         continue

#     per_cy = np.ones(4000)
#     Ey_per = np.random.uniform(starty,endy,num_sample)
#     Ex_per = np.random.uniform(startx,endx,num_sample)
#     El_per_y = np.random.uniform(startl,endl,num_sample)
#     per_x = Ex_per + L1(cof1,de1,El_per_y)
#     do_y = Ey_per + L2(cof2,de2,El_per_y) + noise_y + non_linear(cof_c,de,per_x)
#     perturb_data_y = np.stack((per_x,do_y,per_cy), axis = 1)
#     data_per_y = np.concatenate((o_data,perturb_data_y), axis = 0)
#     np.save('./non_linear_nongaussian/case5_do(y)_a.npy', data_per_y)

#     kci_obj = CIT(data_per_y, "kci")
#     pValue = kci_obj(0,2,set([]))
#     print(0,2,())
#     print(pValue)
#     if pValue < 0.05:
#         continue
#     pValue = kci_obj(0,2,set([1]))
#     print(0,2,(1))
#     print(pValue)
#     if pValue < 0.05:
#         break

#####################case 6 (selection with latent)########################
# for i in range(10):
#     startx = 1
#     endx = 3
#     starty = 2
#     endy = 4
#     startl = 2
#     endl = 4
#     starts = 1
#     ends = 3
#     startn = 2
#     endn = 4
#     num_sample = 4000


#     Ex = np.random.uniform(startx,endx,num_sample)
#     Ey = np.random.uniform(starty,endy,num_sample)
#     El = np.random.uniform(startl,endl,num_sample)
#     Es = np.random.uniform(starts,ends,num_sample)
#     noise_x = np.random.uniform(startn,endn,num_sample)
#     noise_y = np.random.uniform(startn,endn,num_sample)

#     L1, de1, cof_l1 = choose_function()
#     L2, de2, cof_l2 = choose_function()

#     S1, ds1, cof_s1 = choose_function()
#     S2, ds2, cof_s2 = choose_function(-2)

#     L = El
#     c = np.zeros(4000)
#     x = Ex + L1(cof_l1,de1,L)
#     y = Ey + L2(cof_l2,de2,L)
#     mask = (S1(cof_s1, ds1,x) + S2(cof_s2, ds2,y) + Es > 0) & (S1(cof_s1, ds1,x) + S2(cof_s2, ds2,y) + Es < 100)
#     print(S1(cof_s1, ds1,x) + S2(cof_s2, ds2,y) + Es)
#     x = x[mask]
#     y = y[mask]
#     c = c[mask]
#     o_data = np.stack((x,y,c),axis=1)

#     per_cx = np.ones(4000)
#     El_per_x = np.random.uniform(startl,endl,num_sample)
#     Es_per_x = np.random.uniform(starts,ends,num_sample)
#     Ex_per = np.random.uniform(startx,endx,num_sample)
#     do_x = Ex_per + L1(cof_l1, de1,El_per_x) + noise_x
#     Ey_per = np.random.uniform(starty,endy,num_sample)
#     per_y = Ey_per + L2(cof_l2,de2,El_per_x)
#     mask_per_x = (S1(cof_s1, ds1,do_x) + S2(cof_s2, ds2,per_y) + Es_per_x > 0) & (S1(cof_s1, ds1,do_x) + S2(cof_s2, ds2,per_y) + Es_per_x < 100)
#     print(S1(cof_s1, ds1,do_x) + S2(cof_s2, ds2,per_y) + Es_per_x)
#     do_x = do_x[mask_per_x]
#     per_y = per_y[mask_per_x]
#     per_cx = per_cx[mask_per_x]
#     pertub_data_x = np.stack((do_x,per_y, per_cx), axis=1)
#     data_per_x = np.concatenate((o_data,pertub_data_x), axis = 0)
#     np.save('./non_linear_nongaussian/case6_do_a(x).npy', data_per_x)
#     print(pertub_data_x.shape)
#     print(data_per_x.shape)
#     kci_obj = CIT(data_per_x, "kci")
#     pValue = kci_obj(1,2,set([]))
#     print(1,2,())
#     print(pValue)
#     if np.isnan(pValue):
#         continue
#     if pValue > 0.05:
#         continue
#     pValue = kci_obj(1,2,set([0]))
#     print(1,2,(0))
#     print(pValue)
#     if pValue > 0.05:
#         continue

#     per_cy = np.ones(4000)
#     El_per_y = np.random.uniform(startl,endl,num_sample)
#     Es_per_y = np.random.uniform(starts,ends,num_sample)
#     Ex_per = np.random.uniform(startx,endx,num_sample)
#     Ey_per = np.random.uniform(starty,endy,num_sample)
#     per_x = Ex_per + L1(cof_l1,de1,El_per_y)
#     do_y = Ey_per + noise_y + L2(cof_l2,de2, El_per_y)
#     mask_per_y = (S1(cof_s1, ds1,per_x) + S2(cof_s2, ds2, do_y) + Es_per_y > 0) & (S1(cof_s1, ds1,per_x) + S2(cof_s2, ds2, do_y) + Es_per_y < 100)
#     print(S1(cof_s1, ds1,per_x) + S2(cof_s2, ds2, do_y) + Es_per_y)
#     per_x = per_x[mask_per_y]
#     do_y = do_y[mask_per_y]
#     per_cy = per_cy[mask_per_y]
#     pertub_data_y = np.stack((per_x,do_y, per_cy), axis=1)
#     data_per_y = np.concatenate((o_data,pertub_data_y), axis = 0)
#     np.save('./non_linear_nongaussian/case6_do(y)_a.npy', data_per_y)
#     print(pertub_data_y.shape)
#     print(data_per_y.shape)

#     kci_obj = CIT(data_per_y, "kci")
#     pValue = kci_obj(0,2,set([]))
#     print(0,2,())
#     print(pValue)
#     if np.isnan(pValue):
#         continue
#     if pValue > 0.05:
#         continue
#     pValue = kci_obj(0,2,set([1]))
#     print(0,2,(1))
#     print(pValue)
#     if pValue < 0.05:
#         break

#####################case 7 (cause with selection and latent)########################
# for i in range(10):
#     startx = 1
#     endx = 2
#     starty = 2
#     endy = 3
#     startl = 1
#     endl = 2
#     starts = 2
#     ends = 4
#     startn = 2
#     endn = 4
#     num_sample = 4000

#     Ex = np.random.uniform(startx,endx,num_sample)
#     Ey = np.random.uniform(starty,endy,num_sample)
#     El = np.random.uniform(startl,endl,num_sample)
#     Es = np.random.uniform(starts,ends,num_sample)

#     noise_x = np.random.uniform(startn,endn,num_sample)
#     noise_y = np.random.uniform(startn,endn,num_sample)

#     L1, de1, cof_l1 = choose_function()
#     L2, de2, cof_l2 = choose_function()

#     S1, ds1, cof_s1 = choose_function()
#     S2, ds2, cof_s2 = choose_function(-1)
#     non_linear,de, cof_c = choose_function()

#     L = El
#     c = np.zeros(4000)
#     x = Ex + L1(cof_l1, de1, L)
#     y = Ey + L2(cof_l2, de2, L) + non_linear(cof_c, de, x)
#     mask = (S1(cof_s1, ds1, x) + S2(cof_s2, ds2, y) + Es > 0) & (S1(cof_s1, ds1, x) + S2(cof_s2, ds2, y) + Es < 100)
#     print(S1(cof_s1, ds1, x) + S2(cof_s2, ds2, y) + Es)
#     x = x[mask]
#     y = y[mask]
#     c = c[mask]
#     o_data = np.stack((x,y,c),axis=1)
#     # print(o_data)

#     per_cx = np.ones(4000)
#     El_per_x = np.random.uniform(startl,endl,num_sample)
#     Es_per_x = np.random.uniform(starts,ends,num_sample)
#     Ex_per = np.random.uniform(startx,endx,num_sample)
#     do_x = Ex_per + L1(cof_l1,de1, El_per_x) + noise_x
#     Ey_per = np.random.uniform(starty,endy,num_sample)
#     per_y = Ey_per + L2(cof_l2, de2, El_per_x) + non_linear(cof_c, de, do_x)
#     mask_per_x = (S1(cof_s1, ds1,do_x) + S2(cof_s2, ds2,per_y) + Es_per_x > 0) & (S1(cof_s1, ds1,do_x) + S2(cof_s2, ds2,per_y) + Es_per_x < 100)
#     print(S1(cof_s1, ds1,do_x) + S2(cof_s2, ds2,per_y) + Es_per_x)
#     do_x = do_x[mask_per_x]
#     per_y = per_y[mask_per_x]
#     per_cx = per_cx[mask_per_x]
#     pertub_data_x = np.stack((do_x,per_y, per_cx), axis=1)
#     data_per_x = np.concatenate((o_data,pertub_data_x), axis = 0)
#     np.save('./non_linear_nongaussian/case7_do_a(x).npy', data_per_x)
#     print(pertub_data_x.shape)
#     print(data_per_x.shape)

#     kci_obj = CIT(data_per_x, "kci")
#     pValue = kci_obj(1,2,set([]))
#     print(1,2,())
#     if np.isnan(pValue):
#         continue
#     print(pValue)
#     if pValue > 0.05:
#         continue
#     pValue = kci_obj(1,2,set([0]))
#     print(1,2,(0))
#     print(pValue)
#     if pValue > 0.05:
#         continue

#     per_cy = np.ones(4000)
#     Ey_per = np.random.uniform(starty,endy,num_sample)
#     El_per_y = np.random.uniform(startl,endl,num_sample)
#     Es_per_y = np.random.uniform(starts,ends,num_sample)
#     Ex_per = np.random.uniform(startx,endx,num_sample)
#     per_x = Ex_per + L1(cof_l1, de1, El_per_y)
#     do_y = Ey_per + L2(cof_l2,de2,El_per_y) + non_linear(cof_c,de,per_x) + noise_y
#     mask_per_y = (S1(cof_s1, ds1,per_x) + S2(cof_s2, ds2, do_y) + Es_per_y > 0) & (S1(cof_s1, ds1,per_x) + S2(cof_s2, ds2, do_y) + Es_per_y < 100)
#     print(S1(cof_s1, ds1,per_x) + S2(cof_s2, ds2, do_y) + Es_per_y)
#     per_x = per_x[mask_per_y]
#     do_y = do_y[mask_per_y]
#     per_cy = per_cy[mask_per_y]
#     pertub_data_y = np.stack((per_x,do_y, per_cy), axis=1)
#     data_per_y = np.concatenate((o_data,pertub_data_y), axis = 0)
#     np.save('./non_linear_nongaussian/case7_do(y)_a.npy', data_per_y)
#     print(pertub_data_y.shape)
#     print(data_per_y.shape)

#     kci_obj = CIT(data_per_y, "kci")
#     pValue = kci_obj(0,2,set([]))
#     if np.isnan(pValue):
#         continue
#     print(0,2,())
#     print(pValue)
#     if pValue > 0.05:
#         continue
#     pValue = kci_obj(0,2,set([1]))
#     print(0,2,(1))
#     print(pValue)
#     if pValue < 0.05:
#         break