import argparse
from copy import deepcopy
import torch
import torch.utils.data
from md17.dataset import MD17Dataset
from models.model import *
import os
from torch import nn, optim
import json

import random
import numpy as np
# import matplotlib.pyplot as plt
import time

from models.model_t import EqMotion

parser = argparse.ArgumentParser(description='NS_EGNN')
parser.add_argument('--batch_size', type=int, default=100,
                    help='input batch size for training')
parser.add_argument('--epochs', type=int, default=500,
                    help='number of epochs to train')
parser.add_argument('--no_cuda', action='store_true', default=False,
                    help='enables CUDA training')
parser.add_argument('--seed', type=int, default=1, metavar='S',
                    help='random seed (default: 1)')
parser.add_argument('--test_interval', type=int, default=5, metavar='N',
                    help='how many epochs to wait before logging test')
parser.add_argument('--outf', type=str, default='md17_result', metavar='N',
                    help='folder to output the json log file')
parser.add_argument('--nf', type=int, default=16, metavar='N',
                    help='hidden dim')

parser.add_argument('--attention', type=int, default=0, metavar='N',
                    help='attention in the ae model')
parser.add_argument('--max_training_samples', type=int, default=500, metavar='N',
                    help='maximum amount of training samples')
parser.add_argument('--weight_decay', type=float, default=1e-12, metavar='N',
                    help='weight decay')
parser.add_argument('--data_dir', type=str, default='md17',
                    help='Data directory.')
parser.add_argument('--hop_1', type=int, default=5,
                    help='Window length of STFT.')
parser.add_argument('--hop_2', type=int, default=10,
                    help='Hop length of STFT.')
parser.add_argument('--hop_3', type=int, default=20,
                    help='Length of STFT.')
parser.add_argument('--mol', type=str, default='aspirin',
                    help='Name of the molecule: aspirin, benzene, ethanol, malonaldehyde, naphthalene,salicylic, toluene, uracil ')
parser.add_argument('--n_layers', type=int, default=2, metavar='N',
                    help='number of layers for the autoencoder')
parser.add_argument('--degree', type=int, default=2, metavar='N',
                    help='degree of the TFN and SE3')
parser.add_argument('--div', type=float, default=1, metavar='N',
                    help='timing experiment')
parser.add_argument('--exp_name', type=str, default='exp_10000', metavar='N', help='experiment_name')
parser.add_argument('--num_past', type=int, default=100,
                    help='Number of length of whole past time series.')
parser.add_argument('--num_pred', type=int, default=10,
                    help='Number of length of whole future time series.')
parser.add_argument('--time_point', type=int, default=5,
                    help='Time point of past time series (egnn):1,5,10.')
parser.add_argument('--delta_frame', type=int, default=5,
                    help='Number of frames delta.')
parser.add_argument('--model', type=str, default='estag', metavar='N',
                    help='available models: baseline, egnn, estag')
parser.add_argument('--lr', type=float, default=5e-3, metavar='N',
                    help='learning rate')
parser.add_argument('--fft', type=eval, default=False,
                    help='Use FFT ')
parser.add_argument('--eat', type=eval, default=True,
                    help='Use EAT')
parser.add_argument('--with_mask', action='store_true', default=False,
                    help='mask the future frame if use eat')
parser.add_argument('--window_length', type=int, default=10,
                    help='Window length of STFT.')
parser.add_argument('--hop_length', type=int, default=5,
                    help='Hop length of STFT.')
parser.add_argument('--n_fft', type=int, default=10,
                    help='Length of STFT.')

parser.add_argument('--final', type=eval, default=True, help='To evaluate the trajectory or final position')

parser.add_argument('--tempo', type=eval, default=True, help='Use temporal pooling')
parser.add_argument('--save_m', type=eval, default=True, help='whether to save model')
parser.add_argument('--sp_filter', type=eval, default=True, help='whether to use spectral filter on frequency')
parser.add_argument('--verbose', type=eval, default=True, help='whether to output')
parser.add_argument('--gpu', type=int, default=0,
                    help='Idx of gpu used for training')

args = parser.parse_args()

###########################
# args.vel = False

args.cuda = not args.no_cuda and torch.cuda.is_available()

torch.cuda.set_device(args.gpu)
device = torch.device("cuda" if args.cuda else "cpu")

loss_mse = nn.MSELoss()

print(args)
try:
    os.makedirs(args.outf)
except OSError:
    pass

try:
    exp_path = args.outf + f"/exp_{args.model}"
    # exp_path = args.outf + "/" + args.exp_name
    os.makedirs(exp_path)
except OSError:
    pass


def main():
    # fix seed
    seed = args.seed
    # seed = random.randint(0, 1000)
    random.seed(seed)
    np.random.seed(seed)
    torch.manual_seed(seed)
    torch.cuda.manual_seed(seed)
    print(f"The seed is {seed}")

    dataset_train = MD17Dataset(partition='train', max_samples=args.max_training_samples, data_dir=args.data_dir,
                                molecule_type=args.mol, delta_frame=args.delta_frame, num_past=args.num_past,
                                num_pred=args.num_pred, final=args.final)
    loader_train = torch.utils.data.DataLoader(dataset_train, batch_size=args.batch_size, shuffle=True, drop_last=True,
                                               num_workers=0)

    dataset_val = MD17Dataset(partition='val', max_samples=2000, data_dir=args.data_dir,
                              molecule_type=args.mol, delta_frame=args.delta_frame, num_past=args.num_past,
                              num_pred=args.num_pred, final=args.final)
    loader_val = torch.utils.data.DataLoader(dataset_val, batch_size=args.batch_size, shuffle=False, drop_last=False,
                                             num_workers=0)

    dataset_test = MD17Dataset(partition='test', max_samples=2000, data_dir=args.data_dir,
                               molecule_type=args.mol, delta_frame=args.delta_frame, num_past=args.num_past,
                               num_pred=args.num_pred, final=args.final)
    loader_test = torch.utils.data.DataLoader(dataset_test, batch_size=args.batch_size, shuffle=False, drop_last=False,
                                              num_workers=0)

    in_edge_nf = dataset_train.edge_attr.shape[-1]
    nodes_att_dim = 0
    if args.fft and args.model == 'estag':
        in_edge_nf = args.num_past - 1
        nodes_att_dim = args.num_past - 1

    n_nodes = dataset_train.x_t.shape[-2]
    print('n_nodes', n_nodes)
    print(dataset_train.x_t.shape)
    print("in_edge_nf", in_edge_nf)

    if args.final:
        future_forecast = 1
        print('only predict the final position')
    else:
        future_forecast = args.num_pred
        print('predict the trajectory')

    if args.model == 'egnn':
        model = EGNN(num_past=args.num_past, num_future=future_forecast, in_node_nf=1, in_edge_nf=in_edge_nf,
                     hidden_nf=args.nf, device=device, n_layers=args.n_layers)
    elif args.model == 'estag':
        model = ESTAG(num_past=args.num_past, num_future=future_forecast, in_node_nf=1, in_edge_nf=in_edge_nf,
                      hidden_nf=args.nf, fft=args.fft, eat=args.eat, nodes_att_dim=nodes_att_dim,
                      device=device, n_layers=args.n_layers, n_nodes=n_nodes, with_mask=args.with_mask,
                      tempo=args.tempo)
    elif args.model == 'stft':
        model = STFT(num_past=args.num_past, num_future=future_forecast, in_node_nf=1, in_edge_nf=in_edge_nf,
                     hidden_nf=args.nf, fft=args.fft, eat=args.eat, nodes_att_dim=nodes_att_dim,
                     device=device, n_layers=args.n_layers, n_nodes=n_nodes, with_mask=args.with_mask, tempo=args.tempo,
                     hop_length=args.hop_length, window_length=args.window_length, n_fft=args.n_fft)
    elif args.model == 'nsegnn':
        model = NS_EGNN(num_past=args.num_past, num_future=future_forecast, in_node_nf=1, in_edge_nf=in_edge_nf,
                        hidden_nf=args.nf, fft=args.fft, eat=args.eat, nodes_att_dim=nodes_att_dim,
                        device=device, n_layers=args.n_layers, n_nodes=n_nodes, with_mask=args.with_mask,
                        tempo=args.tempo, hop1=args.hop_1, hop2=args.hop_2, hop3=args.hop_3)
    elif args.model == 'gnn':
        input_dim = 3 + 1
        model = GNN(num_past=args.num_past, num_future=future_forecast, input_dim=input_dim, in_edge_nf=in_edge_nf,
                    hidden_nf=args.nf, n_layers=args.n_layers, device=device, recurrent=True)
    elif args.model == 'stgcn':
        num_features = 3 + 1
        out_dim = 3
        model = STGCN(num_nodes=n_nodes, num_features=num_features, num_timesteps_input=args.num_past,
                      num_timesteps_output=future_forecast, out_dim=out_dim, device=device)
    elif args.model == 'baseline':  # past 1 --> future 1
        model = EGNN(num_past=1, num_future=1, in_node_nf=1, in_edge_nf=in_edge_nf, hidden_nf=args.nf, device=device,
                     n_layers=args.n_layers)
    elif args.model == 'stag_neq':  # None-Equivariant STAG
        model = STAG(num_nodes=n_nodes, num_features=4, num_timesteps_input=args.num_past, num_timesteps_output=1).to(
            device)
    elif args.model == 'se3_transformer' or args.model == 'tfn':
        from se3_dynamics.dynamics import OurDynamics as SE3_Transformer
        model = SE3_Transformer(num_past=args.num_past, num_future=future_forecast, n_particles=n_nodes, n_dimesnion=3,
                                nf=int(args.nf / args.degree), n_layers=args.n_layers, model=args.model,
                                num_degrees=args.degree, div=1, device=device)
    elif args.model == 'aglstan':
        num_features = 3 + 1
        out_dim = 3
        model = AGLSTAN(num_nodes=n_nodes, batch_size=args.batch_size, input_dim=num_features, output_dim=out_dim,
                        window=args.num_past, num_layers=args.n_layers, filter_size=32, embed_dim=args.nf, cheb_k=3,
                        num_future=future_forecast)
        model.to(device)
    elif args.model == 'eqmotion':
        model = EqMotion(in_node_nf=args.num_past, in_edge_nf=2, hidden_nf=args.nf, in_channel=args.num_past, \
                         hid_channel=32, out_channel=future_forecast, device=device, n_layers=args.n_layers,
                         recurrent=True)
    else:
        raise Exception("Wrong model specified")
    print(model)

    optimizer = optim.Adam(model.parameters(), lr=args.lr, weight_decay=args.weight_decay)
    # scheduler = get_linear_schedule_with_warmup(optimizer=optimizer, num_warmup_steps=int(0.2 * args.epochs), num_training_steps=args.epochs)

    results = {'epochs': [], 'test loss': [], 'val loss': [], 'train loss': []}
    best_val_loss = 1e8
    best_test_loss = 1e8
    best_epoch = 0
    best_train_loss = 1e8
    for epoch in range(args.epochs):
        start_time = time.time()
        train_loss = train(model, optimizer, epoch, loader_train)
        results['train loss'].append(train_loss)
        if epoch % args.test_interval == 0:
            val_loss = train(model, optimizer, epoch, loader_val, backprop=False)
            test_loss = train(model, optimizer, epoch, loader_test, backprop=False)
            results['epochs'].append(epoch)
            results['val loss'].append(val_loss)
            results['test loss'].append(test_loss)
            if val_loss < best_val_loss:
                best_val_loss = val_loss
                best_test_loss = test_loss
                best_train_loss = train_loss
                best_epoch = epoch

                if args.save_m:
                    torch.save(model, f'{exp_path}/saved_model.pth')
            if args.verbose:
                print("*** Best Val Loss: %.8f \t Best Test Loss: %.8f \t Best apoch %d"
                      % (best_val_loss, best_test_loss, best_epoch))

        # scheduler.step()
        json_object = json.dumps(results, indent=4)
        with open(f"{exp_path}/loss.json", "w") as outfile:
            outfile.write(json_object)

    end_time = time.time()
    print(f'************training time(s) of {args.model}: {(end_time - start_time) / args.epochs}')
    os.makedirs(exp_path, exist_ok=True)
    result_path = f"{exp_path}/result.txt"
    with open(result_path, "a") as f:
        f.write(
            f'************training time(s) of {args.model} on {args.mol}: {(end_time - start_time) / args.epochs}\n')
        f.write(f'The seed is {seed}\n')

    return best_train_loss, best_val_loss, best_test_loss, best_epoch


def train(model, optimizer, epoch, loader, backprop=True):
    if args.final:
        future_forecast = 1
    else:
        future_forecast = args.num_pred

    if backprop:
        model.train()
    else:
        model.eval()

    res = {'epoch': epoch, 'loss': 0, 'coord_reg': 0, 'counter': 0, 'loss_stick': 0, 'reg_loss': 0}

    for batch_idx, data in enumerate(loader):
        batch_size, _, n_nodes, _ = data[0].size()
        data = [d.to(device) for d in data]

        data = [
            d.view(-1, d.size(2)) if len(d.shape) == 3 else (d.permute(1, 0, 2, 3)).reshape(d.size(1), -1, d.size(3))
            for d in data]  # construct mini-batch graphs

        ### loc: (num_past, b*n_node, 3)
        ### edge_attr: (n_edge, 3)
        ### loc_end: (b*n_node, 3)
        ### chargs: (b*n_node, 1)
        loc, edge_attr, charges, loc_end = data

        ### edges: (2, n_edge)
        edges = loader.dataset.get_edges(batch_size, n_nodes)
        edges = [edges[0].to(device), edges[1].to(device)]

        optimizer.zero_grad()

        if args.model == 'egnn':
            loc_pred = model(charges, loc, edges, edge_attr)
        if args.model == 'ns_trans':
            nodes = torch.cat((charges.unsqueeze(0).repeat(loc.shape[0], 1, 1), loc), dim=-1)
            loc_pred = model(nodes, edges, edge_attr)
        elif args.model == 'estag':
            loc_pred = model(charges, loc, edges, edge_attr)
        elif args.model == 'etimesnet':
            loc_pred = model(charges, loc, edges, edge_attr)
        elif args.model == 'stft':
            loc_pred = model(charges, loc, edges, edge_attr)
        elif args.model == 'nsegnn':
            loc_pred = model(charges, loc, edges, edge_attr)
        elif args.model == 'segnn':
            loc_pred = model(charges, loc, edges, edge_attr)
        elif args.model == 'gnn':
            nodes = torch.cat((charges.unsqueeze(0).repeat(loc.shape[0], 1, 1), loc), dim=-1)
            loc_pred = model(nodes, edges, edge_attr)
        elif args.model == 'stgcn':
            feature = torch.cat((charges.unsqueeze(0).repeat(loc.shape[0], 1, 1), loc), dim=-1)

            node = feature.permute(1, 0, 2).reshape(batch_size, n_nodes, feature.shape[0], feature.shape[2])
            Adj = loader.dataset.A.to(device)
            loc_pred = loc[-1].unsqueeze(0).repeat(future_forecast, 1, 1) + model(Adj, node).reshape(future_forecast,
                                                                                                     -1, 3)
            # loc_pred = model(Adj, node).reshape(-1,3)
        elif args.model == 'baseline':
            loc_pred = model(charges, loc[args.time_point - 1].unsqueeze(0), edges, edge_attr)
        elif args.model == 'stag_neq':
            feature = torch.cat([charges.unsqueeze(0).repeat(args.num_past, 1, 1), loc], axis=2)
            # feature=loc
            node = feature.permute(1, 0, 2).reshape(batch_size, n_nodes, feature.shape[0], feature.shape[2])
            Adj = loader.dataset.A.to(device)
            loc_pred = loc[-1] + model(Adj, node).reshape(-1, 3)
        elif args.model == 'se3_transformer' or args.model == 'tfn':
            # only one batch edges
            edges = loader.dataset.get_edges(1, n_nodes)
            edges = [edges[0].to(device), edges[1].to(device)]
            vel = torch.zeros_like(loc).to(device)
            ### (num_past, b*n_node, 3)
            loc_pred = model(loc, vel, charges, edges)
        elif args.model == 'aglstan':
            feature = torch.cat((charges.unsqueeze(0).repeat(loc.shape[0], 1, 1), loc), dim=-1)

            node = feature.permute(1, 0, 2).reshape(batch_size, feature.shape[0], n_nodes, feature.shape[2])

            loc_pred = model(node)
            # loc_pred = loc[-1] + loc_pred.reshape(-1,3)
            loc_pred = loc_pred.permute(1, 0, 2, 3).reshape(future_forecast, -1, 3)
        elif args.model == 'eqmotion':
            nodes = charges.reshape(batch_size, n_nodes, charges.shape[-1]).repeat(1, 1, args.num_past)
            loc_pred, _ = model(nodes, loc.reshape(batch_size, n_nodes, args.num_past, 3), None)
            loc_pred = loc_pred.permute(2, 0, 1, 3)
            loc_pred = loc[-1].unsqueeze(0).repeat(future_forecast, 1, 1) + loc_pred.reshape(future_forecast, -1, 3)
        else:
            raise Exception("Wrong model")

        # print(loc_end.shape)
        # exit()
        loss = loss_mse(loc_pred, loc_end)

        clip_value = 0.1
        if backprop:
            loss.backward()
            for param in model.parameters():
                if param.grad is not None:
                    # 计算梯度的范数
                    grad_norm = param.grad.data.norm(2)  # L2范数
                    if grad_norm > clip_value:
                        # 裁剪梯度
                        param.grad.data = param.grad.data / grad_norm * clip_value
            optimizer.step()
        res['loss'] += loss.item() * batch_size
        res['counter'] += batch_size

    if not backprop:
        prefix = "==> "
    else:
        prefix = ""

    # for simplicity in log
    if args.verbose:
        print('%s epoch %d avg loss: %.5f reg loss: %.5f'
              % (prefix + loader.dataset.partition, epoch,
                 res['loss'] / res['counter'], res['reg_loss'] / res['counter']))

    return res['loss'] / res['counter']


if __name__ == "__main__":
    best_train_loss, best_val_loss, best_test_loss, best_epoch = main()
    print("best_train = %.8f" % best_train_loss)
    print("best_val = %.8f" % best_val_loss)
    print("best_test = %.8f" % best_test_loss)
    print("best_epoch = %d" % best_epoch)
    # result_path = "md_ablation_result" + f"/{args.mol}"
    # # exp_path = args.outf + "/" + args.exp_name
    # os.makedirs(result_path, exist_ok=True)
    # exp_path = f"{result_path}/result_fft.txt"
    # with open(exp_path, "a") as f:
    #     f.write(f'FFT 100\n')
    #     f.write(f'The seed is {args.seed}\n')
    #     f.write(f"best_train = {best_train_loss:.6f}\n")
    #     f.write(f"best_val = {best_val_loss:.6f}\n")
    #     f.write(f"best_test = {best_test_loss:.6f}\n")
    #     f.write(f"best_epoch = {best_epoch}\n")
    #     f.write("-" * 40 + "\n")  # 添加分隔线方便查看
    #
    # with open(f"{exp_path}/loss.json") as f:
    #     loss=json.load(f)
    # #plt.plot(loss['train loss'],label='Train')
    # plt.plot(loss['epochs'],[np.mean(loss['train loss'][i*5:(i+1)*5]) for i in range(len(loss['train loss'])//5)],label='Train')
    # plt.plot(loss['epochs'],loss['test loss'],label='Test')
    # plt.legend()
    # plt.title("Loss")
    # plt.savefig(f'{exp_path}/loss.png')
