

import numpy as np
import torch

import data as dat
import models as mod
from torch.linalg import norm

from utils import smoothing_metric, compute_randomwalk

torch.autograd.set_detect_anomaly(True)

device = 'cuda:0' if torch.cuda.is_available() else 'cpu'
print(device)


#%% data
data_name = 'Cora' # 'Cora', 'Pubmed', 'Citeseer', 'synthetic'
dense = True # for synthetic data

num_layers = 40 if dense else 100
data, num_classes, num_features = dat.load_data(data_name,
                                                p_intra = (0.05 if dense else 0.01),
                                                p_inter = (0.01 if dense else 0.005))
n = data.x.shape[0]

    
# GNN
GNNB = mod.GNNBiSto(num_node_features=num_features, num_classes=num_classes,
                    rec_intermediate_grad=True, num_layers=num_layers, num_units=32,
                    activation='leaky_relu', is_id=True,
                    std=0.00)

out = GNNB(data)


#%% lambda_f

Xs = GNNB.Xs
dirichx = []
for _ in range(num_layers):
    x = Xs[_].detach().cpu()
    dirichx.append(smoothing_metric(x, torch.ones(n)))

convslope = np.log(np.array(dirichx))
lambdaf = np.exp(np.polyfit(np.arange(len(convslope)), convslope, deg=1)[0])

#%% lambda_P

P, pi = compute_randomwalk(data)

eig = torch.linalg.eigvals(P)
lamb = torch.abs(eig[1])


#%% contracting P

J = torch.eye(n) - torch.ones((n,n))/n
contr = norm(J@P@J, ord=2)

print(f'density: {data.edge_index.shape[1]/(n*(n-1)):.3}, lambdaf: {lambdaf:.3}, ',
      f'lambda: {lamb:.3}, contract: {contr:.3}')

