"""
This script generates data for the 50 nodes network
"""



import numpy as np
import networkx as nx
import matplotlib.pyplot as plt
import time
from itertools import combinations
import random
import math




# Plot directed graph
# Input:  (1) adjacency_matrix: Adjacency matrix of directed graph
# Output: (1) Plot of the directed graph
def plot_directed_graph(adjacency_matrix):
    nNodes=adjacency_matrix.shape[0]
    G=nx.DiGraph()
    G.add_nodes_from(['V'+str(ind) for ind in range(0,nNodes)])
    for ind1 in range(nNodes):
        for ind2 in range(nNodes):
            if adjacency_matrix[ind1,ind2]:
                G.add_edge('V'+str(ind2),'V'+str(ind1))
    # explicitly set positions
    if nNodes == 6:
        pos = {'V'+str(0): (1, 0), 'V'+str(1): (0.5, 0.866),'V'+str(2): (-0.5, 0.866), 
               'V'+str(3): (-1, 0), 'V'+str(4): (-0.5, -0.866), 'V'+str(5): (0.5, -0.866)}
    elif nNodes == 10:
        pos = {'V'+str(0): (1.0000, 0.0000), 'V'+str(1): (0.8090, 0.5878),'V'+str(2): (0.3090, 0.9511), 
               'V'+str(3): (-0.3090, 0.9511), 'V'+str(4): (-0.8090, 0.5878), 'V'+str(5): (-1.0000, 0.0000),
               'V'+str(6): (-0.8090, -0.5878), 'V'+str(7): (-0.3090, -0.9511), 'V'+str(8): (0.3090, -0.9511),
               'V'+str(9): (0.8090, -0.5878)}
    else:
        # pos = {f'V{i}': (math.cos(2 * math.pi * i / 20), math.sin(2 * math.pi * i / 20)) for i in range(20)}
        pos = nx.spring_layout(G)
        
    options = {
    "font_size": 12,
    "node_size": 1000,
    "node_color": "white",
    "edgecolors": "black",
    "linewidths": 2,
    "width": 2,
    }
    
    plt.figure(figsize=(10, 8))
    nx.draw(G, pos, with_labels=True, node_color='skyblue', node_size=800, arrowsize=20, font_size=10)
    
    # nx.draw_networkx(G, pos, **options)
    # Set margins for the axes so that nodes aren't clipped
    # ax = plt.gca()
    # ax.margins(0.20)
    plt.axis("off")
    plt.show(block=False)
    plt.pause(0.001)



def generate_graph(num_nodes,max_in_degree):
    # random_dag = generate_random_dag(num_nodes, avg_degree)
    # generates a random dag with maximum in degree equal to max_in_degree
    G = nx.DiGraph()
    nodes = list(range(num_nodes))
    G.add_nodes_from(node for node in nodes)
    for node in nodes:
        # in_degree = max_in_degree 
        in_degree = random.randint(1, max_in_degree)
        if node - in_degree < 0:
            in_degree = node  # Avoid in-degrees exceeding node indices
        in_neighbors = random.sample(nodes[:node], in_degree)
        G.add_edges_from((neighbor, node) for neighbor in in_neighbors)
    Adj = nx.adjacency_matrix(G).toarray().T
    print("Number of nodes:", G.number_of_nodes())
    print("Number of edges:", G.number_of_edges())
    # print("Adjacency matrix:", Adj.shape," \n", Adj)
    return G,Adj

    

# Find moral graph and skeleton for a given graph given its adjacency matrix
# Input:  (1) adjacency_matrix: Adjacency matrix of directed graph
# Output: (1) MG: Adjacency matrix of the estimated moral graph
#         (2) Skeleton: Adjacency matrix of the estimated skeleton
def find_moral_graph_and_skeleton_from_adjacency_matrix(Adj):
    nNodes=Adj.shape[0]
    Skeleton=(Adj+Adj.T)
    MG=np.zeros([nNodes,nNodes],dtype=bool)
    MG=np.copy(Skeleton)
    for ind1 in range(nNodes):
        for ind2 in range(nNodes):
            for ind3 in range(nNodes):
                if (Adj[ind1,ind2]==1) and ((Adj[ind1,ind3]==1)) and ind2!=ind3:
                    MG[ind2,ind3]=1
                    MG[ind3,ind2]=1
    return MG,Skeleton



def create_A_B_coeffs_10_nodes(Adj,filter=None):
    # Create A and B matrix for IIR transfer functions
    # A is coeffs of interaction with the past of self, 
    # B interaction with the other nodes 
    # returns A,B
    nNodes=Adj.shape[0]
    A=np.array([[1, .6,.3],[1, .4,.2],[1, .7,.4],[1, .5,.3],[1, .6,.3],[1, .7,.3],[1, .65,.1],[1, .76,.3],[1, .8,.3],[1, .4,.2] ])
    
    B_scale1=(np.random.uniform(-1,1,size=[nNodes,nNodes,3]))
    B_scale1=2*(B_scale1>0)-1
    # generates a matrix with each entry rad*unif(delta,1) , where rad is the Rademacher rv {+1,-1} with prob 1/2 each
    B=np.random.uniform(0.2,0.4,size=[nNodes,nNodes,3])*B_scale1    
    Adj=np.repeat(Adj[:, :, np.newaxis], 3, axis=2)
    B=B*Adj
    B[:,:,1]=0
    B[:,:,2]=0

    return A,B



def create_A_B_coeffs_6_nodes(Adj,filter=None):
    # Create A and B matrix for IIR transfer functions
    # A is coeffs of interaction with the past of self, 
    # B interaction with the other nodes 
    # returns A,B
    nNodes=Adj.shape[0]
    
    A = np.array([[1.739381261594696859e-02,	1.043628756956818150e-02,	5.218143784784090751e-03],
                  [3.415208172002098808e-01,	1.366083268800839523e-01,	6.830416344004197615e-02],
                  [8.289102927104288199e-01,	5.802372048973001295e-01,	3.315641170841715502e-01],
                  [8.846579905691620560e-02,	4.423289952845810280e-02,	2.653973971707486099e-02],
                  [8.562817629518102436e-01,	5.137690577710861684e-01,	2.568845288855430842e-01],
                  [1.630296867565628194e-01,	1.141207807295939597e-01,	4.890890602696884581e-02]])

    B=np.random.uniform(0.2,0.4,size=(nNodes,nNodes,3))
    Adj=np.repeat(Adj[:, :, np.newaxis], 3, axis=2)
    B=B*Adj
    B[:,:,1:3]=0

    return A,B


def create_A_B_coeffs_20_nodes(Adj,filter=None):
    # Create A and B matrix for IIR transfer functions
    # A is coeffs of interaction with the past of self, 
    # B interaction with the other nodes 
    # returns A,B
    nNodes=Adj.shape[0]
    
    A = np.array([[1.739381261594696859e-02,	1.043628756956818150e-02,	5.218143784784090751e-03],
                  [3.415208172002098808e-01,	1.366083268800839523e-01,	6.830416344004197615e-02],
                  [8.289102927104288199e-01,	5.802372048973001295e-01,	3.315641170841715502e-01],
                  [8.846579905691620560e-02,	4.423289952845810280e-02,	2.653973971707486099e-02],
                  [8.562817629518102436e-01,	5.137690577710861684e-01,	2.568845288855430842e-01],
                  [1.630296867565628194e-01,	1.141207807295939597e-01,	4.890890602696884581e-02],
                  [1.739381261594696859e-02,	1.043628756956818150e-02,	5.218143784784090751e-03],
                  [3.415208172002098808e-01,	1.366083268800839523e-01,	6.830416344004197615e-02],
                  [8.289102927104288199e-01,	5.802372048973001295e-01,	3.315641170841715502e-01],
                  [8.846579905691620560e-02,	4.423289952845810280e-02,	2.653973971707486099e-02],
                  [8.562817629518102436e-01,	5.137690577710861684e-01,	2.568845288855430842e-01],
                  [1.630296867565628194e-01,	1.141207807295939597e-01,	4.890890602696884581e-02],
                  [1.739381261594696859e-02,	1.043628756956818150e-02,	5.218143784784090751e-03],
                  [3.415208172002098808e-01,	1.366083268800839523e-01,	6.830416344004197615e-02],
                  [8.289102927104288199e-01,	5.802372048973001295e-01,	3.315641170841715502e-01],
                  [8.846579905691620560e-02,	4.423289952845810280e-02,	2.653973971707486099e-02],
                  [8.562817629518102436e-01,	5.137690577710861684e-01,	2.568845288855430842e-01],
                  [1.630296867565628194e-01,	1.141207807295939597e-01,	4.890890602696884581e-02],
                  [1.739381261594696859e-02,	1.043628756956818150e-02,	5.218143784784090751e-03],
                  [3.415208172002098808e-01,	1.366083268800839523e-01,	6.830416344004197615e-02]])

    B=np.random.uniform(0.2,0.4,size=(nNodes,nNodes,3))
    Adj=np.repeat(Adj[:, :, np.newaxis], 3, axis=2)
    B=B*Adj
    B[:,:,1:3]=0

    return A,B



def create_A_B_coeffs_50_nodes(Adj,filter=None):
    # Create A and B matrix for IIR transfer functions
    # A is coeffs of interaction with the past of self, 
    # B interaction with the other nodes 
    # returns A,B
    nNodes=Adj.shape[0]
    
    A = np.array([[1.739381261594696859e-02,	1.043628756956818150e-02,	5.218143784784090751e-03],
                  [3.415208172002098808e-01,	1.366083268800839523e-01,	6.830416344004197615e-02],
                  [8.289102927104288199e-01,	5.802372048973001295e-01,	3.315641170841715502e-01],
                  [8.846579905691620560e-02,	4.423289952845810280e-02,	2.653973971707486099e-02],
                  [8.562817629518102436e-01,	5.137690577710861684e-01,	2.568845288855430842e-01],
                  [1.630296867565628194e-01,	1.141207807295939597e-01,	4.890890602696884581e-02],
                  [1.739381261594696859e-02,	1.043628756956818150e-02,	5.218143784784090751e-03],
                  [3.415208172002098808e-01,	1.366083268800839523e-01,	6.830416344004197615e-02],
                  [8.289102927104288199e-01,	5.802372048973001295e-01,	3.315641170841715502e-01],
                  [8.846579905691620560e-02,	4.423289952845810280e-02,	2.653973971707486099e-02],
                  [8.562817629518102436e-01,	5.137690577710861684e-01,	2.568845288855430842e-01],
                  [1.630296867565628194e-01,	1.141207807295939597e-01,	4.890890602696884581e-02],
                  [1.739381261594696859e-02,	1.043628756956818150e-02,	5.218143784784090751e-03],
                  [3.415208172002098808e-01,	1.366083268800839523e-01,	6.830416344004197615e-02],
                  [8.289102927104288199e-01,	5.802372048973001295e-01,	3.315641170841715502e-01],
                  [8.846579905691620560e-02,	4.423289952845810280e-02,	2.653973971707486099e-02],
                  [8.562817629518102436e-01,	5.137690577710861684e-01,	2.568845288855430842e-01],
                  [1.630296867565628194e-01,	1.141207807295939597e-01,	4.890890602696884581e-02],
                  [1.739381261594696859e-02,	1.043628756956818150e-02,	5.218143784784090751e-03],
                  [3.415208172002098808e-01,	1.366083268800839523e-01,	6.830416344004197615e-02],
                  [1.739381261594696859e-02,	1.043628756956818150e-02,	5.218143784784090751e-03],
                  [3.415208172002098808e-01,	1.366083268800839523e-01,	6.830416344004197615e-02],
                  [8.289102927104288199e-01,	5.802372048973001295e-01,	3.315641170841715502e-01],
                  [8.846579905691620560e-02,	4.423289952845810280e-02,	2.653973971707486099e-02],
                  [8.562817629518102436e-01,	5.137690577710861684e-01,	2.568845288855430842e-01],
                  [1.630296867565628194e-01,	1.141207807295939597e-01,	4.890890602696884581e-02],
                  [1.739381261594696859e-02,	1.043628756956818150e-02,	5.218143784784090751e-03],
                  [3.415208172002098808e-01,	1.366083268800839523e-01,	6.830416344004197615e-02],
                  [8.289102927104288199e-01,	5.802372048973001295e-01,	3.315641170841715502e-01],
                  [8.846579905691620560e-02,	4.423289952845810280e-02,	2.653973971707486099e-02],
                  [8.562817629518102436e-01,	5.137690577710861684e-01,	2.568845288855430842e-01],
                  [1.630296867565628194e-01,	1.141207807295939597e-01,	4.890890602696884581e-02],
                  [1.739381261594696859e-02,	1.043628756956818150e-02,	5.218143784784090751e-03],
                  [3.415208172002098808e-01,	1.366083268800839523e-01,	6.830416344004197615e-02],
                  [8.289102927104288199e-01,	5.802372048973001295e-01,	3.315641170841715502e-01],
                  [8.846579905691620560e-02,	4.423289952845810280e-02,	2.653973971707486099e-02],
                  [8.562817629518102436e-01,	5.137690577710861684e-01,	2.568845288855430842e-01],
                  [1.630296867565628194e-01,	1.141207807295939597e-01,	4.890890602696884581e-02],
                  [1.739381261594696859e-02,	1.043628756956818150e-02,	5.218143784784090751e-03],
                  [3.415208172002098808e-01,	1.366083268800839523e-01,	6.830416344004197615e-02],
                  [8.562817629518102436e-01,	5.137690577710861684e-01,	2.568845288855430842e-01],
                  [1.630296867565628194e-01,	1.141207807295939597e-01,	4.890890602696884581e-02],
                  [1.739381261594696859e-02,	1.043628756956818150e-02,	5.218143784784090751e-03],
                  [3.415208172002098808e-01,	1.366083268800839523e-01,	6.830416344004197615e-02],
                  [8.289102927104288199e-01,	5.802372048973001295e-01,	3.315641170841715502e-01],
                  [8.846579905691620560e-02,	4.423289952845810280e-02,	2.653973971707486099e-02],
                  [8.562817629518102436e-01,	5.137690577710861684e-01,	2.568845288855430842e-01],
                  [1.630296867565628194e-01,	1.141207807295939597e-01,	4.890890602696884581e-02],
                  [1.739381261594696859e-02,	1.043628756956818150e-02,	5.218143784784090751e-03],
                  [3.415208172002098808e-01,	1.366083268800839523e-01,	6.830416344004197615e-02]])

    B=np.random.uniform(0.2,0.4,size=(nNodes,nNodes,3))
    Adj=np.repeat(Adj[:, :, np.newaxis], 3, axis=2)
    B=B*Adj
    B[:,:,1:3]=0
    return A,B



# Generate data according to the VAR model
# Inputs: (1) nSamples: Total number of samples
#         (2) B: p x p x DELAY-1 across other variables
#         (3) A: Matrix with coefficients representing self dynamics
#         (4) noise_pow: Noise power (noise is i.i.d. Gausian)
#         (5) nlags: required lag on other variables
# Output: (1) Timeseries data for a generative graph entailed by A adn B arrays
def continuous_data_generation_arbitrary_lags(nSamples,B,A=None,noise_pow=-1,nlags=0):
    nNodes = B.shape[0]
    if noise_pow.all() == -1:
        noise_pow = np.ones(B.shape[0])
    print("Generating continous data.. nSamples: ",nSamples," for nNodes :",nNodes)
    x = np.zeros((nSamples,nNodes))
    for ind in range(nlags):
        x[ind,:] = np.random.randn(nNodes)*np.sqrt(noise_pow)
    for ind_Samp in range(nlags,nSamples):
        x[ind_Samp,:] = np.random.randn(nNodes)*np.sqrt(noise_pow) - A[:,0]*x[ind_Samp-1,:] - A[:,1]*x[ind_Samp-2,:] - A[:,2]*x[ind_Samp-3,:]
        x[ind_Samp,:] = x[ind_Samp,:] + np.dot(B[:,:,0],x[ind_Samp-nLags,:])
    return x



# Compute least squates solution: Project X on to Z, Z=a0x_0+a1x1+...am xm, where m is the number of columns of X
def Least_Squares_Solution(X,Z):
    X_star = X.conj().T
    XX=np.matmul(X_star, X)
    if np.isscalar(XX):
        return np.matmul(np.divide( X_star,XX), Z)
    else:
        return np.matmul(np.matmul(np.linalg.inv(np.matmul(X_star, X)), X_star), Z)



# Estimate the Wiener filters in time domain.
# Inputs: (1) x: The data set to be used for calculating projection
#         (2) i: The node on which to be projected
#         (3) L: The number of past and future samples to be used for calculating the projections
# Output: (1) w: the computed Wiener projection coefficients.
def Wiener_time(x,i,L):
    # ||xi-YB||^2
    m = (2*L+1)
    T = x.shape[0]-1
    xi = x[L:T-L+1,i]
    C = [ind for ind in range(x.shape[1])]
    C = np.delete(C,i)
    y = np.zeros([T-2*L+1,m*C.shape[0]])
    for ind in range(L,T-L+1,L):
        y[ind-L,:] = x[ind-L:ind+L+1,C].reshape(m*len(C))
    w1 = Least_Squares_Solution(y,xi)
    # w1 = np.linalg.lstsq(y,xi)[0]
    w = w1.reshape(m,len(C))
    return w



# Function for estimating the Wiener filters in time domain.
# Inputs:  (1) data: time-series data.
#          (2) i: First node.
#          (3) j: Second node.
#          (4) z: Conditioning set of nodes.
# Outputs: (1) W: The estimated Wiener filters
#          (2) W_sum: Sum of all the time domain Wiener coefficients
def compute_partial_Wiener_coeffs_time(data,i,j,z=[],nlags=0):
    c=np.append(j,np.int32(z))
    indexes = np.unique(c, return_index=True)[1]
    c = [c[index] for index in sorted(indexes)]
    X = data[:,np.append(i,c)] # Wiener_time computes projection of x_i on x_ibar. So need to pass x_{i,c}
    W = Wiener_time(X,0,nlags)
    W_sum = np.sum(abs(W),axis=0)
    W_time_avg = W_sum/W.shape[0]
    return W,W_time_avg





# Estimate essential graph from estimated skeleton and the d-separation sets
# Inputs:  (1) Top_est: The estimated skeleton
#          (2) C: A disctionary containing the d-separation sets
# Outputs: (1) Est_adj: An adjacency matrix of the estimated essential grapg
#          (2) colliders: The set of estimated colliders
def estimate_essential_graph(Top_est,C):
    num_nodes = Top_est.shape[0]
    colliders=list()
    Est_adj=np.copy(Top_est)
    # # Estimate MEG from skeleton and d-separating set, C
    for ind1 in range(num_nodes):
        for ind2 in range(num_nodes):   
            for ind3 in range(ind2+1,num_nodes):
                # check if ind3 is present in the d-separating set of (ind1,ind2) 
                # is present in skeleton when ind3--ind1 and ind3--ind2 
                # are present in the skeleton. Not present means ind3 is a collider
                # If not present, then ind3 is a collider
                if (Top_est[ind1,ind2]==1) and (Top_est[ind1,ind3]==1) and ind1!=ind3 and ind2!=ind1 and ind2!=ind3:
                    # print(ind1,ind2,ind3,C[ind1,ind2])
                    if C[(ind3,ind2)]!=None: #None means no d-separating set
                        if (ind1 not in C[(ind3,ind2)]):
                            # print(ind1, " is a collider")
                            colliders.append(ind1)
                            Est_adj[ind2,ind1]=0
                            Est_adj[ind3,ind1]=0
    colliders=list(set(colliders))
    return Est_adj,colliders



# Compute the FFT of the data
# Inputs:  (1) data: time-series data
#          (2) nfft: numer of FFT points
# Output:  (1) y: the FFT of the provided data samples
def compute_fft(data,nfft):
    nSamples,nNodes=data.shape
    nTrajectories=np.int32(nSamples/nfft) 
    y=np.zeros((nTrajectories,nNodes,nfft),dtype=complex)
    for ind in range(nTrajectories-1): # discard final few residual samples
        x=data[ind*(nfft):(ind+1)*nfft,:]
        X=np.fft.fft(x,axis=0) #/np.sqrt(nfft)
        y[ind,:,:]=np.transpose(X) #X is (nfft,nNodes)
    return y



# Compute the partial Wiener coefficients using FFT based approach
# Inputs:  (1) data: time-series data.
#          (2) i: First node.
#          (3) j: Second node.
#          (4) z: Conditioning set of nodes.
# Outputs: (1) W: The estimated Wiener filters
#          (2) W_mag: Maximum magnitude of the Wiener filters taken over the frequencies
def compute_partial_Wiener_coeffs_FFT(data,i,j,z=[]):
    c=np.append(j,np.int32(z))
    indexes = np.unique(c, return_index=True)[1]
    c=[c[index] for index in sorted(indexes)]
    n1=len(c)
    nFFT=data.shape[2]
    W=np.zeros([nFFT,n1],dtype='complex')
    W_mag=np.zeros([n1])
    for freq_index in range(nFFT):
        fft_data=data[:,:,freq_index]
        X=fft_data[:,i]
        X_bar=fft_data[:,c]
        W[freq_index,:]=Least_Squares_Solution(X_bar,X)
    W_mag=np.amax(abs(W),axis=0)
    return W,W_mag



# Algorithm for estimatin the d-separation sets using Wiener projections estimated using FFT based approach.
# Inputs:  (1) data: time-series data
#          (2) nFFT: numer of FFT points
#          (3) thresh: Thresold for comparing the Wiener filter extimates to check for d-separation
#          (4) Q: The maximum number of elements in the conditioning set
# Outputs: (1) C: A dictionary containing node pairs as key and the corresponding d-separation sets as the elements
def Run_PC_Wiener_test_FFT_h_inf(data,nFFT,thresh,Q):
    # print("Run_PC_Wiener_test_FFT_h_inf")
    nNodes = data.shape[1]
    C = {}
    data_FFT = compute_fft(data,nfft=nFFT)
    for ind1 in range(nNodes):
        for ind2 in range(nNodes):
            if ind1 > ind2:
                # continue
                flag = False
                # the complement set through which we condition
                D=[i for i in range(nNodes) if i != ind1 and i !=ind2]
                # iterate over the combinations of various size in the increasing order
                for ind_cond in range(Q+1):
                    if flag == True:
                        break
                    combination_set = [i for i in combinations(D,ind_cond)]
                    for c in combination_set:
                        # print("\n combination set: ",c)
                        W_i_jZ,W_mag_i_jZ = compute_partial_Wiener_coeffs_FFT(data_FFT,ind1,ind2,c)
                        # print("Projection coefficient of X"+str(ind1+1)+" on "+str(np.append(ind2,c)+1)+"=",W_mag_i_jZ) #%%
                        if W_mag_i_jZ[0] < thresh:
                            C[ind1,ind2] = c
                            flag=True
                            break
                            c=np.array([ind1])
                        if np.shape(c)[0]==min(Q,nNodes-2): #checking if |c|==q, which means |{c,q}|>q
                            C[ind1,ind2]=None
                            break

    return C


def Wiener_Proj(X,Z):
    # Project X on to Z, Z=a0x_0+a1x1+...am xm, 
    # where m is the number of columns of X
    X_star = X.conj().T
    XX=np.matmul(X_star, X)
    if np.isscalar(XX):
        return np.matmul(np.divide( X_star,XX), Z)
    else:
        return np.matmul(np.matmul(np.linalg.inv(np.matmul(X_star, X)), X_star), Z)



def compute_Wiener_coeffs(nNodes,data):
    # Returns W and W_mag
    # W: nNodes X nSamples X nNodes
    # W_mag: returns h-infinity norm over frequencies
    nSamples=data.shape[2]
    W=np.zeros([nNodes,nSamples,nNodes],dtype='complex')
    W_mag=np.zeros([nNodes,nNodes])
    for proj_ind in range(nNodes):
        W1=np.zeros([nSamples,nNodes],dtype='complex')
        for freq_index in range(nSamples):
            fft_data=data[:,:,freq_index]
            Z=fft_data[:,proj_ind]
            X_bar=np.delete(fft_data,proj_ind,1)
            W1[freq_index,:]=np.insert(Wiener_Proj(X_bar,Z),proj_ind,0)
        W[proj_ind,:,:]=W1
        W_mag[proj_ind,:]=np.amax(abs(W1),axis=0)
    return W,W_mag



def compute_partial_Wiener_coeffs_avg(data,i,j,z=[],freq_choice=[]):
    # Returns W and W_mag
    # W: nNodes X n_FFT X nNodes
    # W_mag: returns h-infinity norm over frequencies
    c=np.append(j,np.int32(z))
    # c=np.unique(c)
    indexes = np.unique(c, return_index=True)[1]
    c=[c[index] for index in sorted(indexes)]
    n1=len(c)
    n_FFT=data.shape[2]
    W=np.zeros([n_FFT,n1],dtype='complex')
    # W_mag=np.zeros([n1])
    W_mag_avg = np.zeros([n1],dtype=float)
    for freq_index in freq_choice:
        fft_data=data[:,:,freq_index]
        # print(fft_data.shape)
        # print(freq_index)
        X=fft_data[:,i]
        X_bar=fft_data[:,c]
        # print(Z.shape,X_bar.shape)
        # print(X_bar[:3,:])
        W[freq_index,:]=Wiener_Proj(X_bar,X)
        W_mag_avg = W_mag_avg + np.abs(W[freq_index,:])
    W_mag_avg = np.divide(W_mag_avg,len(freq_choice))
    # W_mag=np.amax(abs(W),axis=0)
    return W,W_mag_avg



def estimate_essential_graph_new(Top_est,C):
    nNodes = Top_est.shape[0]
    #% Collider Identification
    collider_graph = np.copy(Top_est)
    collider_set = []#np.ones((nNodes,1))*(-1)
    for i in range(nNodes):
        for j in range(nNodes):
            for k in range(nNodes):
                if i>j and j!=k and i!=k and Top_est[i,k]==1 and Top_est[j,k]==1 and Top_est[i,j]!=1 and (C[(i,j)]!=None) and (k not in C[(i,j)]):
                    collider_graph[i,k] = 0
                    collider_graph[j,k] = 0
                    collider_set.append(k)


    # % Orienting other edges if possible
    final_graph = np.copy(collider_graph)
    for i in range(nNodes):
        for j in range(nNodes):
            if i>j and collider_graph[i,j]==1 and collider_graph[j,i]==1:
                if i in collider_set:
                    final_graph[i,j] = 0
                if j in collider_set:
                    final_graph[j,i] = 0

    
    return final_graph,collider_graph




# Run theFFT based Wiener PC algorithm and calculate the nestork estimation errors
# Inputs:  (1) data: Time-series data
#          (2) nSamplles_list: The samples list to be used at which the errors are to be calculated
#          (3) nFFT: numer of FFT points
#          (4) thresh: Thresold for checking d-separation condition
#          (5) q: Maximum size of conditioning set
#          (6) true_ess_graph: Adjacency matrix of the true essential graph
# Outputs: (1) error_freq_hinf_PC: Array of network estimation errors at different sample sizes
#          (2) computation_freq: The Computation times at different sample sizes
def compute_PC_error_FFT_h_inf(data,nSamples_list,nFFT,true_ess_graph,thresh_freq=0.1,q=3):
    num_nodes=data.shape[1]
    error_freq_hinf_PC = np.zeros(len(nSamples_list))
    computation_freq = np.zeros(len(nSamples_list))
    for ind_samples in range(len(nSamples_list)):
        start_PC_freq = time.time()
        if nSamples_list[ind_samples] > 10*nFFT:
            C_FFT = Run_PC_Wiener_test_FFT_h_inf(data[:nSamples_list[ind_samples]],nFFT,thresh=thresh_freq,Q=q)
        else:
            continue
        Top_est_half = np.zeros([num_nodes,num_nodes])
        for c in C_FFT:
            if C_FFT[c] == None: Top_est_half[c]=1
        Top_est_FFT = (Top_est_half + Top_est_half.T)
        est_essential_graph_fft,est_colliders = estimate_essential_graph_new(Top_est_FFT,C_FFT)
        error_freq_hinf_PC[ind_samples] = sum(sum(est_essential_graph_fft!=true_ess_graph))/sum(sum(true_ess_graph))
        computation_freq[ind_samples] = time.time()-start_PC_freq
    return error_freq_hinf_PC, computation_freq




def find_skeleton_from_ess(Adj):
    nNodes=Adj.shape[0]
    Skeleton=np.zeros([nNodes,nNodes],dtype=bool)
    for ind1 in range(nNodes):
        for ind2 in range(nNodes):
            if ind1 != ind2:
                if (Adj[ind1,ind2]==1) or ((Adj[ind2,ind1]==1)):
                    Skeleton[ind1,ind2]=1
                    Skeleton[ind2,ind1]=1
    return Skeleton




def find_true_Essential_graph(Skeleton, Adj):
    collider_graph = np.copy(Skeleton)
    collider_nodes = []
    for ind1 in range(num_nodes): # ind1 is the collider node index
        for ind2 in range(num_nodes):    
        #ind2 and ind3 are the indices for the parent of colliders
            for ind3 in range(num_nodes):
                if (Adj[ind1,ind2] == 1) and ((Adj[ind1,ind3] == 1)) and Skeleton[ind2,ind3] !=1 and ind2 != ind3 and ind1 != ind3 and ind1 != ind2:
                    # print( ind1," is a collider")
                    collider_graph[ind2,ind1]=0
                    collider_graph[ind3,ind1]=0
                    collider_nodes.append(ind1)                       
    # % Orienting other edges if possible
    true_ess_graph = np.copy(collider_graph)
    for i in range(num_nodes):
        for j in range(num_nodes):
            if i>j and collider_graph[i,j]==1 and collider_graph[j,i]==1:
                if i in collider_nodes:
                    true_ess_graph[i,j] = 0
                if j in collider_nodes:
                    true_ess_graph[j,i] = 0
                    
    return true_ess_graph



def compute_undir_graph_degrees(Adj):
    nNodes = Adj.shape[0]
    node_degs = []
    for node in range(nNodes):
        node_degs.append(sum(Adj[:,node]))
    max_deg = max(node_degs)
    min_deg = min(node_degs)
    avg_deg = sum(node_degs)/nNodes
    
    return min_deg, max_deg, avg_deg


def plot_custom_graph(adj_matrix):
    # Create a directed graph from the adjacency matrix
    G = nx.from_numpy_array(adj_matrix, create_using=nx.DiGraph)

    # Rename nodes to 'V1', 'V2', ..., 'V20'
    mapping = {i: f'V{i+1}' for i in range(len(adj_matrix))}
    G = nx.relabel_nodes(G, mapping)

    # Use a layout that avoids overlapping nodes and messy edges
    pos = nx.spring_layout(G)  # seed for consistent layout
    # pos = nx.kamada_kawai_layout(G)

    # Draw nodes, edges, and labels
    plt.figure(figsize=(10, 8))
    nx.draw(G, pos, with_labels=True, node_color='skyblue', node_size=800, arrowsize=20, font_size=10)
    nx.draw_networkx_edge_labels(G, pos, edge_labels={(u, v): '' for u, v in G.edges()}, font_color='gray')
    plt.title("Graph from Custom Adjacency Matrix")
    plt.axis('off')
    plt.tight_layout()
    plt.show()
    
    
def is_cyclic_directed(adj_matrix):
    n = len(adj_matrix)
    visited = [False] * n
    rec_stack = [False] * n

    def dfs(v):
        visited[v] = True
        rec_stack[v] = True

        for u in range(n):
            if adj_matrix[v][u]:  # There's an edge from v to u
                if not visited[u]:
                    if dfs(u):
                        return True
                elif rec_stack[u]:
                    return True

        rec_stack[v] = False
        return False

    for node in range(n):
        if not visited[node]:
            if dfs(node):
                return True
    return False


def is_weakly_connected(adj_matrix):
    n = len(adj_matrix)

    # Step 1: Convert the directed graph into an undirected graph
    undirected = [[0] * n for _ in range(n)]
    for i in range(n):
        for j in range(n):
            if adj_matrix[i][j] or adj_matrix[j][i]:
                undirected[i][j] = undirected[j][i] = 1

    # Step 2: Perform DFS on the undirected version
    visited = [False] * n

    def dfs(v):
        visited[v] = True
        for u in range(n):
            if undirected[v][u] and not visited[u]:
                dfs(u)

    # Start DFS from the first node
    dfs(0)

    # Step 3: Check if all nodes were visited
    return all(visited)


# Main function ##################################################################################################################
##################################################################################################################################
if __name__ == '__main__':
    start_sim_time=time.time()
    print("Start time: ",start_sim_time)
    nNetworks = 1
    nFFT = 32
    nSamples_list = [640, 3200, 6400, 32000, 128*10**3, 32*10**4]
    num_nodes = 50
    max_in_degree = 2
    max_out_degree = 1
    noise_pow1 = 1*np.ones(num_nodes)
    avg_deg_lim = [2.1,3.5]
    max_deg_lim = 9
    min_deg_lim = 1
    random.seed(1)
    np.random.seed(1)
    minLags = 3
    maxLags = 4

    error_freq_hinf_Networks  = np.zeros([nNetworks, len(nSamples_list), (maxLags - minLags)])
    error_freq_wiener_phase_networks = np.zeros([nNetworks, len(nSamples_list), (maxLags - minLags)])
    error_time_Networks = np.zeros([nNetworks, len(nSamples_list), (maxLags - minLags)])
    
    computation_freq_Networks = np.zeros([nNetworks, len(nSamples_list), (maxLags - minLags)])
    computation_wiener_phase_networks = np.zeros([nNetworks, len(nSamples_list), (maxLags - minLags)])
    computation_time_Networks = np.zeros([nNetworks, len(nSamples_list), (maxLags - minLags)])
    
    thresh_freq = np.zeros([nNetworks, len(nSamples_list), (maxLags - minLags)])
    thresh_win_phase = np.zeros([nNetworks, len(nSamples_list), (maxLags - minLags)])
    thresh_time = np.zeros([nNetworks, len(nSamples_list), (maxLags - minLags)])


    for nLags in range(minLags,maxLags):
        print("nLags: ",nLags,"$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$")
        for ind_network in range(nNetworks):
            print("ind_network: ",ind_network,"-------------------------------------------------------")
            ########### ########### ########### ########### ########### ########### ########### 
            # Network and Data Generation
            ########### ########### ########### ########### ########### ########### ###########
            #  Random network generation
            Adj = np.array([[0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 1, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
                            ])
            Adj[20,19] = 1
            Adj[21,20] = 1
            Adj[22,21] = 1
            Adj[23,22] = 1
            Adj[24,23] = 1
            Adj[25,24] = 1
            Adj[27,26] = 1
            Adj[28,27] = 1
            Adj[33,28] = 1
            Adj[34,33] = 1
            Adj[29,24] = 1
            Adj[30,29] = 1
            Adj[31,29] = 1
            Adj[32,31] = 1
            Adj[34,32] = 1
            Adj[35,34] = 1
            Adj[36,35] = 1
            Adj[37,36] = 1
            Adj[38,37] = 1
            Adj[39,38] = 1
            Adj[40,39] = 1
            Adj[41,40] = 1
            Adj[42,40] = 1
            Adj[43,42] = 1
            Adj[45,39] = 1
            Adj[45,44] = 1
            Adj[49,44] = 1
            Adj[46,45] = 1
            Adj[47,46] = 1
            Adj[48,47] = 1
            Adj[49,48] = 1
            
            
            np.save("Adj_mat_50_nodes.npy",Adj)
            MG,Skeleton = find_moral_graph_and_skeleton_from_adjacency_matrix(Adj)
            A1,B1 = create_A_B_coeffs_50_nodes(Adj)
            
            true_ess_graph = find_true_Essential_graph(Skeleton, Adj)
            plot_directed_graph(Adj)
            min_degree, max_degree, avg_degree = compute_undir_graph_degrees(Skeleton)
            print(f"Max Degree = {max_degree}")
            print(f"Min Degree = {min_degree}")
            print(f"Avg Degree = {avg_degree}")
            # plot_directed_graph(true_ess_graph)
            ########### ########### Generate data ############################################  
            start_data=time.time()
            print("Data generation started at ", start_data)
            data=continuous_data_generation_arbitrary_lags(nSamples_list[-1],B=B1,A=A1, noise_pow=noise_pow1,nlags=nLags)
            print("Time taken to generate data = ", time.time()-start_data)
            
            np.savetxt("dataSet_50Nodes.txt", data)
            
            
            plt.plot(np.arange(0,data.shape[0]), data[:,0])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,1])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,2])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,3])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,4])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,5])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,6])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,7])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,8])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,9])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,10])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,11])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,12])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,13])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,14])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,15])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,16])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,17])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,18])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,19])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,20])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,21])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,22])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,23])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,24])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,25])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,26])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,27])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,28])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,29])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,30])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,31])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,32])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,33])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,34])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,35])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,36])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,37])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,38])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,39])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,40])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,41])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,42])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,43])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,44])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,45])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,46])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,47])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,48])
            plt.show()
            plt.plot(np.arange(0,data.shape[0]), data[:,49])
            plt.show()
    
    
  