{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import time\n",
    "import torch\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as pl\n",
    "import torchdiffeq\n",
    "import torch.nn as n\n",
    "import FrEIA.framework as Ff\n",
    "import FrEIA.modules as Fm\n",
    "from mpl_toolkits import mplot3d\n",
    "\n",
    "\n",
    "##########################################################3\n",
    "# system dynamics\n",
    "\n",
    "xx=torch.tensor([0.,0.,0.])\n",
    "tt=torch.tensor([0.,5.])\n",
    "t_d=torch.linspace(0,2,800)\n",
    "devicestr=\"cpu\"\n",
    "device_str=\"cuda\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([0., 0., 0.], device='cuda:0')"
      ]
     },
     "execution_count": 2,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "xx.to(device_str)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 864x720 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "def test_fun(t,x):\n",
    "    sig=10.\n",
    "    rho=28.\n",
    "    beta=8/3\n",
    "    vel=torch.zeros((3,1))\n",
    "    vel[0]=sig*(x[1]-x[0])\n",
    "    vel[1]=x[0]*(rho-x[2])-x[1]\n",
    "    vel[2]=x[0]*x[1]-beta*x[2]\n",
    "    return(vel)\n",
    "\n",
    "tt_tors=torch.tensor([[.15,.15,.15,0.,0.,0.]])\n",
    "xy_d_list=[]\n",
    "for i in range(len(tt_tors)):\n",
    "    noise_c=torch.normal(torch.zeros((len(t_d),3)),0.01*torch.ones((len(t_d),3)))\n",
    "    xy_d=torchdiffeq.odeint(test_fun,tt_tors[i,:3][None].T,t_d).reshape((-1,3))#+noise_c #no noise for test\n",
    "    xy_d_list.append(xy_d.clone().detach())\n",
    "\n",
    "pl.figure(figsize=(12,10))\n",
    "for i in range(len(xy_d_list)):\n",
    "    xy_d=xy_d_list[i]\n",
    "    pl.plot(xy_d[:,0],xy_d[:,1],marker='o')\n",
    "\n",
    "xx=xy_d_list[0] "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "ours time mean:\n",
      "0.21159330559967202\n",
      "ours time std:\n",
      "0.005907203188608058\n",
      "torch.Size([800, 3])\n",
      "Ours Interpolation MAE: 0.2994\n"
     ]
    }
   ],
   "source": [
    "N_DIM = 6\n",
    "def subnet_fc(dims_in, dims_out):\n",
    "    return n.Sequential(n.Linear(dims_in, 1500), n.ReLU(),\n",
    "                         n.Linear(1500, dims_out))\n",
    "inn2 = Ff.SequenceINN(N_DIM)\n",
    "for k in range(5):\n",
    "    inn2.append(Fm.AllInOneBlock, subnet_constructor=subnet_fc)\n",
    "    \n",
    "f_x_base=n.Sequential(\n",
    "n.Linear(6, 30),\n",
    "n.Tanh(),\n",
    "n.Linear(30, 30),\n",
    "n.Tanh(),\n",
    "n.Linear(30, 30),\n",
    "n.Tanh(),\n",
    "n.Linear(30, 6)\n",
    ")\n",
    "def fx_base(t,x):\n",
    "    return(f_x_base(x))\n",
    "    \n",
    "def for_inn2(x):\n",
    "    return(inn2(x)[0])\n",
    "def rev_inn2(x):\n",
    "    return(inn2(x,rev=True)[0])\n",
    "def linear_val_ode2(init_v,t_d):\n",
    "    init_v_in=rev_inn2(init_v)\n",
    "    eval_lin=torchdiffeq.odeint(fx_base,init_v_in,t_d,\n",
    "                                method='dopri5')[:,0,:]#options={'step_size':0.01}\n",
    "    eval_out=for_inn2(eval_lin)\n",
    "    return(eval_out)\n",
    "    \n",
    "f_x_base.load_state_dict(torch.load('f_x_base_save_good_eod2.tar'))\n",
    "inn2.load_state_dict(torch.load('inn2_save_good_eod2.tar'))   \n",
    "f_x_base = f_x_base.to(device_str)\n",
    "inn2 = inn2.to(device_str)\n",
    "\n",
    "f_x_base = f_x_base.to(device_str)\n",
    "tt_tors = tt_tors.to(device_str)\n",
    "t_d = t_d.to(device_str)\n",
    "    \n",
    "time_list=[]\n",
    "for i in range(10):\n",
    "        tic = time.perf_counter()\n",
    "        tx=linear_val_ode2(tt_tors,t_d)\n",
    "        toc = time.perf_counter()\n",
    "        time_list.append(toc-tic)\n",
    "\n",
    "\n",
    "ours=np.array(time_list)\n",
    "print('ours time mean:')\n",
    "print(ours.mean())\n",
    "print('ours time std:')\n",
    "print(ours.std())\n",
    "\n",
    "\n",
    "sum_num=0\n",
    "    \n",
    "for i in range(0,len(tt_tors)):\n",
    "    xy_d=xy_d_list[i]\n",
    "    print(xy_d.shape)\n",
    "    error=torch.mean(torch.norm(tx[:,:3].to('cpu')-xy_d,dim=1))\n",
    "    sum_num+=error\n",
    "    \n",
    "print(f'Ours Interpolation MAE: {sum_num:.4f}')\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.8.10"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
