import numpy as np
import torch
import torch.nn as nn
from sklearn.preprocessing import QuantileTransformer
from tabularbert import TabularBERTTrainer
from tabularbert.utils.metrics import ClassificationError, RMSE
from tabularbert.utils.data import UniformDiscretize

import json
import os
import random

def seed_everything(seed=0):
    '''
    Sets the seed of the entire notebook so results are the same every time we run.
    This is for REPRODUCIBILITY.
    '''
    random.seed(seed)
    # Set a fixed value for the hash seed
    os.environ['PYTHONHASHSEED'] = str(seed)
    np.random.seed(seed)
    torch.manual_seed(seed)
    
    if torch.cuda.is_available():
        torch.cuda.manual_seed(seed)
        torch.cuda.manual_seed_all(seed)
        
# Load model parameters
with open('config.json', 'r') as f:
    config = json.load(f)

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

# Datasets are available at: https://github.com/jyansir/t2g-former
# Load and preprocess data
train_num_X = np.load("data/churn/X_num_train.npy")[:, :-1]
train_cat_X = np.load("data/churn/X_cat_train.npy")
valid_num_X = np.load("data/churn/X_num_val.npy")[:, :-1]
valid_cat_X = np.load("data/churn/X_cat_val.npy")
test_num_X = np.load("data/churn/X_num_test.npy")[:, :-1]
test_cat_X = np.load("data/churn/X_cat_test.npy")

train_labels = np.load("data/churn/y_train.npy")
valid_labels = np.load("data/churn/y_val.npy")
test_labels = np.load("data//churn/y_test.npy")

categories = [{k.item(): i for i, k in enumerate(np.unique(train_cat_X[:, j]))} for j in range(train_cat_X.shape[1])]
train_cat_XX = np.array([[categories[j][train_cat_X[i, j]] for i in range(train_cat_X.shape[0])] for j in range(train_cat_X.shape[1])]).T
valid_cat_XX = np.array([[categories[j][valid_cat_X[i, j]] for i in range(valid_cat_X.shape[0])] for j in range(valid_cat_X.shape[1])]).T
test_cat_XX = np.array([[categories[j][test_cat_X[i, j]] for i in range(test_cat_X.shape[0])] for j in range(test_cat_X.shape[1])]).T

scaler = QuantileTransformer(n_quantiles=max(min(train_num_X.shape[0] // 30, 1000), 10),
                             output_distribution='uniform',
                             subsample=None)

scaler.fit(train_num_X)
train_num_XX = scaler.transform(train_num_X)
valid_num_XX = scaler.transform(valid_num_X)
test_num_XX = scaler.transform(test_num_X)

train_XX = np.concatenate((train_num_XX, train_cat_XX), axis=1)
valid_XX = np.concatenate((valid_num_XX, valid_cat_XX), axis=1)
test_XX = np.concatenate((test_num_XX, test_cat_XX), axis=1)

encoding_info = {9: 'categorical'}

# Pretraining
seed_everything(0)
trainer = TabularBERTTrainer(x=train_XX,
                             num_bins=50,
                             encoding_info=encoding_info,
                             device=device)
trainer.setup_directories_and_logging(save_dir='./pretraining',
                                    phase='pretraining',
                                    project_name='CH data pretraining',
                                    use_wandb=False)
trainer.set_bert(embedding_dim=config['embedding_dim'], 
                 n_layers=config['n_layers'],
                 n_heads=config['n_heads'],
                 dropout=config['pretraining_dropout'],
                 mode=config['mode'])
trainer.set_optimizer(lr=2e-4, weight_decay=config['weight_decay'])
trainer.pretrain(lamb=config['lamb'],
                penalty='squaredL2',
                epochs=1000,
                batch_size=256,
                mask_token_prob=0.15,
                random_token_prob=0.1,
                unchanged_token_prob=0.1,
                num_workers=0)


# Fine-tuning
trainer = TabularBERTTrainer.from_pretrained(save_path = './pretraining/version0/model_checkpoint.pt',
                                                device = device)
trainer.setup_directories_and_logging(save_dir='./fine-tuning',
                                    phase='fine-tuning',
                                    project_name='CH data fine-tuning',
                                    use_wandb=False)
trainer.set_head(output_dim=2, 
                 activation='ReLU',
                 dropouts=config['fine_tuning_dropout'],
                 hidden_layers=[config['embedding_dim']] * config['head_n_hidden_layers'])
trainer.set_optimizer(lr=config['fine_tuning_lr'],
                     weight_decay=config['weight_decay'])
trainer.finetune(x=train_XX,
                y=train_labels,
                valid_x=valid_XX,
                valid_y=valid_labels,
                epochs=2000,
                penalty=config['fine_tuning_penalty'],
                batch_size=256,
                criterion=nn.CrossEntropyLoss(),
                metric=ClassificationError(),
                patience=100,
                num_workers=0)


# Evaluation
discretizer = UniformDiscretize(num_bins=50, encoding_info=encoding_info)
discretizer.fit(train_XX)
test_XXX = discretizer.discretize(test_XX)

model = TabularBERTTrainer.from_finetuned(f"./fine-tuning/version0/model_checkpoint.pt",
                                            device=device)
model.eval()

predictions = model(torch.tensor(test_XXX).to(device))
pred_class = predictions.argmax(dim=1)
accuracy = (test_labels == pred_class.cpu().numpy()).mean()