import scipy.io as scio
import torchvision
import torch
from torch import nn 
from torch.utils.data import Dataset,DataLoader,TensorDataset
from torchvision import datasets, transforms
import time
import numpy as np 
import pandas as pd
import random
import math
from torch.nn import functional as F
import csv

import matplotlib
import matplotlib.pyplot as plt
import matplotlib.gridspec as gridspec
import sys, random, time
from mpl_toolkits.axes_grid1.inset_locator import zoomed_inset_axes
from mpl_toolkits.axes_grid1.inset_locator import mark_inset
from mpl_toolkits.axes_grid1.inset_locator import inset_axes
matplotlib.rcParams['pdf.fonttype'] = 42
matplotlib.rcParams['ps.fonttype'] = 42
device = torch.device("cuda:0" if torch.cuda.is_available() else "cpu")

filename="../../ref_traj/"+'A'+"_reftraj.mat"
center_list_test=np.array([[-0.57415846,-1.167481 ],[-1.33820054,1.55999908],[-0.8732721 ,-1.38164796],[-0.0 ,-0]])

t_data=[]
y_data=[]
sigma_data=[]
file_data=scio.loadmat(filename)['refTraj'][0]
for data in file_data:
    t_data.append([data[0][0][0]-1.0])
    y_data.append([data[1][0][0],data[1][1][0],data[1][2][0],data[1][3][0]])
    sigma_data.append(data[2]+0.001*np.identity(4))

t_data=np.array(t_data)
y_data=np.array(y_data)
sigma_data=np.array(sigma_data)

batch_size_K = 400
meta_lambda=0.03
n_epochs = 100

redius=2.0
less=False
weight=500.0
softplus_para=200.0

class Model(torch.nn.Module):
    def __init__(self):
        super(Model, self).__init__()
        self.params = [
                    torch.Tensor(128, 8).uniform_(-1./math.sqrt(8), 1./math.sqrt(8)).requires_grad_(),
                    torch.Tensor(128).zero_().requires_grad_(),

                    torch.Tensor(128, 128).uniform_(-1./math.sqrt(128), 1./math.sqrt(128)).requires_grad_(),
                    torch.Tensor(128).zero_().requires_grad_(),

                    torch.Tensor(128, 128).uniform_(-1./math.sqrt(128), 1./math.sqrt(128)).requires_grad_(),
                    torch.Tensor(128).zero_().requires_grad_(),

                    torch.Tensor(2, 128).uniform_(-1./math.sqrt(128), 1./math.sqrt(128)).requires_grad_(),
                    torch.Tensor(2).zero_().requires_grad_(),
                ]
        self.lambada= 0.5

    def dense(self, x, params):
        y = F.linear(x, params[0], params[1])
        y = F.relu(y)

        y = F.linear(y, params[2], params[3])
        y = F.relu(y)

        y = F.linear(y, params[4], params[5])
        y = F.relu(y)

        y = F.linear(y, params[6], params[7])

        return y

    def input_process(self, x):
        x2=torch.pow(x, 2)
        x3=torch.pow(x, 3)
        x4=torch.pow(x, 4)
        x_sin=torch.sin(x*3.14)
        x_cos=torch.cos(x*3.14)
        x_sin_2=torch.sin(2*x*3.14)
        x_cos_2=torch.cos(2*x*3.14)
        return torch.cat((x,x2,x3,x4,x_sin,x_cos,x_sin_2,x_cos_2), 1)

    def forward(self, x, params):
        v = torch.ones(x.shape,dtype=torch.float) 
        position=self.dense(self.input_process(x), params)*10.0
        position1,position2=position.split([1,1],dim=1)
        vel1=torch.autograd.grad(position1,x,v,retain_graph=True, create_graph=True)[0]
        vel2=torch.autograd.grad(position2,x,v,retain_graph=True, create_graph=True)[0]
        return torch.cat((position1,position2,vel1,vel2), 1)
    
    def forward1(self, x, params):
        position=self.dense(self.input_process(x), params)*10.0
        return position

def my_mse_loss(outputs, Q, Sigma):
    a=outputs - Q 
    a=torch.reshape(a,(-1,4,1))
    b=torch.reshape(a,(-1,1,4))
    return torch.mean(torch.matmul(torch.matmul(b,torch.inverse(Sigma)),a))

def constraint_voilations(outputs, center=[0.0,0.0], less=less, redius=redius, weight=weight):
    position,vel=outputs.split([2,2],dim=1)
    center_tensor=torch.tensor(center, dtype= torch.float)
    constraint_voilations=0.0
    if less:
        constraint_voilations= (F.softplus((torch.norm(position-center_tensor,dim=1)- redius),softplus_para)-0.001)*weight
    else:
        constraint_voilations= (F.softplus((-torch.norm(position-center_tensor,dim=1)+ redius),softplus_para)-0.001)*weight
    return torch.mean(constraint_voilations)

def adjust_learning_rate(optimizer, epoch, lr):
    for param_group in optimizer.param_groups:
        param_group['lr'] = lr 


for center in center_list_test:

    model = Model()
    model = model.to(device)

    """
    learning_rate=0.00006
    optimizer0 = torch.optim.SGD(model.params,lr=learning_rate,weight_decay=0.1)
    lr_lamabada=0.04
    """
    learning_rate0=0.003
    optimizer0 = torch.optim.Adam(model.params,lr=learning_rate0,weight_decay=0.1)
    lr_lamabada=0.04


    data_loader_train = torch.utils.data.DataLoader(TensorDataset(torch.tensor(t_data).float().requires_grad_(),torch.tensor(y_data).float(),torch.tensor(sigma_data).float()),shuffle = True, batch_size = batch_size_K)
    data_loader_test = torch.utils.data.DataLoader(TensorDataset(torch.tensor(t_data).float().requires_grad_(),torch.tensor(y_data).float(),torch.tensor(sigma_data).float()),shuffle = False, batch_size = 400)

    model.train() 
    optimizer=optimizer0
    (step_train, data_train_now) = list(enumerate(data_loader_train))[0]
    (step_test, data_test_now) = list(enumerate(data_loader_test))[0]

    for epoch in range(n_epochs):
        loss_train_sum = 0.0
        loss_test_sum = 0.0
        loss_test_constraint_sum=0.0

        (features, labels, sigmas)=data_train_now
        features = features.to(device)
        labels = labels.to(device)
        sigmas=sigmas.to(device)
        outputs = model(features, model.params)
        loss_train = my_mse_loss(outputs, labels,sigmas)
        
        (features_constraint, labels_constraint, sigmas_constraint)=data_test_now
        features_constraint = features_constraint.to(device)
        outputs1 = model(features_constraint, model.params)
        loss_train+=constraint_voilations(outputs1,center=center)*model.lambada
        
        optimizer.zero_grad()
        loss_train.backward(retain_graph=True)
        optimizer.step()

        outputs1 = model(features_constraint, model.params)
        gradient_lambada=constraint_voilations(outputs1,center=center).item()
        if gradient_lambada>0.5:
            gradient_lambada=0.5
        if gradient_lambada<0:
            gradient_lambada=-0.5
            
        model.lambada+=lr_lamabada*gradient_lambada
        if model.lambada<0: 
            model.lambada=0.0 
        
        print(model.lambada)

        loss_train_sum += loss_train.item()
        
        if (step_train+1) % 1 == 0:
            print(f'epoch = {epoch+1}, step = {step_train+1}, train loss = {loss_train_sum / 1:.6f}')
            loss_train_sum=0

        (features, labels, sigmas)=data_test_now
        features = features.to(device)
        labels = labels.to(device)
        sigmas=sigmas.to(device)
        outputs = model(features, model.params)
        loss_test = my_mse_loss(outputs, labels,sigmas)
        constraint_test=constraint_voilations(outputs,center=center)
        loss_test_sum += loss_test.item()
        loss_test_constraint_sum +=constraint_test.item() 

        if (step_test+1) % 1 == 0:
            print(f'epoch = {epoch+1}, step = {step_test+1}, test mse loss = {loss_test_sum / 1:.6f}, test constraint loss = {loss_test_constraint_sum / 1:.6f}')
            loss_test_sum=0

    outputs=[]
    inputss=[]
    labelss=[]
    with torch.no_grad():
        for i, (inputs, labels, sigma) in enumerate(data_loader_test):
            inputs = inputs.to(device)
            labels = labels.to(device)
            output = model.forward1(inputs, model.params)
            outputs = outputs+output.cpu().numpy().tolist()
            inputss=inputss+inputs.cpu().numpy().tolist()
            labelss=labelss+labels.cpu().numpy().tolist()
    outputs=np.array(outputs)
    inputss=np.array(inputss)
    labelss=np.array(labelss)
    print(inputss.shape)
    print(outputs.shape)
    x=inputss[:,0]
    y=outputs[:,0]
    z=outputs[:,1]
    y_label=labelss[:,0]
    z_label=labelss[:,1]
    #plt.scatter(x, y, 1, alpha=0.8)
    #plt.show()
    #plt.scatter(x, z, 1, alpha=0.8)
    #plt.show()
    theta = np.linspace(0, 2 * np.pi, 200)
    x111 = np.cos(theta)*redius+center[0]
    y111 = np.sin(theta)*redius+center[1]
    fig, ax = plt.subplots()
    plt.scatter(y, z, alpha=0.4,color='red',label="Imitation learning")
    plt.scatter(y_label, z_label, alpha=0.4,color='blue',label="Turth")
    ax.plot(x111, y111, color="darkred", linewidth=2)
    plt.xlabel('x',size=28)
    plt.ylabel("y",size=28)
    plt.show()

