{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "source": [
    "import torch\n",
    "import torch.nn as nn\n",
    "from torchdiffeq import odeint\n",
    "import sys; sys.path.append(3*'../')\n",
    "from src import *\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "device = torch.device('cpu') # for second gpu"
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "source": [
    "class ControlledPendulum(nn.Module):\n",
    "    \"\"\"\n",
    "    Inverted pendulum with torsional spring\n",
    "    \"\"\"\n",
    "    def __init__(self, u, m=1., k=.5, l=1., qr=0., β=.01, g=9.81):\n",
    "        super().__init__()\n",
    "        self.u = u # controller (nn.Module)\n",
    "        self.nfe = 0 # number of function evaluations\n",
    "        self.cur_f = None # current function evaluation\n",
    "        self.cur_u = None # current controller evaluation \n",
    "        self.m, self.k, self.l, self.qr, self.β, self.g = m, k, l, qr, β, g # physics\n",
    "        \n",
    "    def forward(self, t, x):\n",
    "        self.nfe += 1\n",
    "        q, p = x[..., :1], x[..., 1:]\n",
    "        self.cur_u = self.u(t, x)\n",
    "        dq = p/self.m\n",
    "        dp = -self.k*(q - self.qr) - self.m*self.g*self.l*torch.sin(q) \\\n",
    "            -self.β*p/self.m + self.cur_u\n",
    "        self.cur_f = torch.cat([dq, dp], -1)\n",
    "        return self.cur_f\n",
    "\n",
    "ControlledSystem = ControlledPendulum\n",
    "def R(X, u, Δt, method, hypereuler=None):\n",
    "    system = ControlledSystem(RandConstController((1,1), -1, 1))\n",
    "    system.u.u0 = u\n",
    "    t_span = torch.tensor([0, Δt]).to(device)\n",
    "    with torch.no_grad():\n",
    "        x_fine = odeint(system, X, t_span, method='dopri5')[-1]\n",
    "        if hypereuler:\n",
    "            xfu = torch.cat([X, system(0, X), u], -1)\n",
    "            x_coarse =  X + Δt*system(0, X) + (Δt**2)*hypereuler(xfu)\n",
    "        else:\n",
    "            x_coarse = odeint(system, X, t_span, method=method)[-1]\n",
    "        return torch.norm(x_fine - x_coarse, p=2, dim=-1)"
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "markdown",
   "source": [
    "## Compare multiple $\\Delta t$"
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "source": [
    "def residual_loss(g, X, u, Δt, method, order=1):\n",
    "    system = ControlledSystem(RandConstController((1,1), -1, 1))\n",
    "    system.u.u0 = u    \n",
    "    t_span = torch.tensor([0, Δt]).to(device)\n",
    "    x_fine = odeint(system, X, t_span, method='dopri5')[-1]\n",
    "    x_coarse = odeint(system, X, t_span, method=method)[-1]\n",
    "    xfu = torch.cat([X, system(0, X), u], -1)\n",
    "    return torch.norm(x_fine - x_coarse - (Δt**(order+1))*g(xfu), p=2, dim=-1)/(Δt**(order+1))"
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "source": [
    "from tqdm import tqdm\n",
    "\n",
    "Δt_test = [0.01, 0.03, 0.1, 0.3, 1.0]\n",
    "hs_list = []\n",
    "\n",
    "x_min_train, x_max_train = -5, 5\n",
    "u_lim = 10\n",
    "bs_hs = 1000\n",
    "n_grid = 100\n",
    "megalosses = []"
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "source": [
    "\n",
    "# Train HyperEuler on multiple delta t\n",
    "for Δt in Δt_test:\n",
    "    print('Training HyperEuler with Δt = ',Δt, '...')\n",
    "    hdim = 32\n",
    "    hs = nn.Sequential(nn.Linear(5, 32), nn.Softplus(), nn.Linear(32, 32), nn.Tanh(), nn.Linear(32, 2)).to(device)\n",
    "    losses = []\n",
    "    opt_hs = torch.optim.Adam(hs.parameters(), lr=1e-3)\n",
    "    '''Hypersolver training loop with given Δt_test'''\n",
    "    for epoch in tqdm(range(100000)):\n",
    "        opt_hs.zero_grad()\n",
    "        # Randomize state and controller values\n",
    "        x = torch.Tensor(bs_hs, 2).uniform_(x_min_train, x_max_train).to(device)\n",
    "        #_grid = x.repeat(n_grid, 1, 1).detach()\n",
    "        u_rand = torch.Tensor(bs_hs, 1).uniform_(-u_lim, u_lim).to(device)\n",
    "        loss = residual_loss(hs, x[:, None, :], u_rand[:, None, :],  Δt, 'euler').mean()\n",
    "        # Optimization step\n",
    "        loss.backward()\n",
    "        opt_hs.step()\n",
    "        losses.append(loss.detach().cpu().item())\n",
    "        # print(f'Step: {epoch}, Residual loss: {loss:.3f}')\n",
    "    print(f'Residual loss: {loss:.3f}')\n",
    "    megalosses.append(losses)\n",
    "    torch.save(hs, f'saved_models/hs{Δt}.pt')\n",
    "    hs_list.append(hs)"
   ],
   "outputs": [
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "  0%|          | 21/100000 [00:00<08:19, 200.30it/s]"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "Training HyperEuler with Δt =  0.01 ...\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "100%|██████████| 100000/100000 [07:09<00:00, 232.67it/s]\n",
      "  0%|          | 18/100000 [00:00<09:29, 175.49it/s]"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "Residual loss: 0.050\n",
      "Training HyperEuler with Δt =  0.03 ...\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "100%|██████████| 100000/100000 [08:51<00:00, 188.01it/s]\n",
      "  0%|          | 16/100000 [00:00<10:42, 155.50it/s]"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "Residual loss: 0.078\n",
      "Training HyperEuler with Δt =  0.1 ...\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "100%|██████████| 100000/100000 [11:15<00:00, 148.02it/s]\n",
      "  0%|          | 10/100000 [00:00<18:03, 92.28it/s]"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "Residual loss: 0.068\n",
      "Training HyperEuler with Δt =  0.3 ...\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "100%|██████████| 100000/100000 [18:29<00:00, 90.16it/s]\n",
      "  0%|          | 4/100000 [00:00<48:21, 34.46it/s]"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "Residual loss: 0.052\n",
      "Training HyperEuler with Δt =  1.0 ...\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "100%|██████████| 100000/100000 [50:50<00:00, 32.79it/s]"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "Residual loss: 0.088\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "\n"
     ]
    }
   ],
   "metadata": {}
  },
  {
   "cell_type": "markdown",
   "source": [
    "## Comparison plot"
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "source": [
    "hs_list = []\n",
    "for t in Δt_test:\n",
    "    hs_list.append(torch.load(f'saved_models/hs{t}.pt'))"
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "source": [
    "import matplotlib.pyplot as plt\n",
    "\n",
    "n_x, n_u = 10000, 100\n",
    "x = torch.Tensor(n_x, 2).uniform_(-2, 2).to(device)\n",
    "qp = torch.linspace(x_min_train, x_max_train, 100) ; Q, P = torch.meshgrid(qp, qp) ; x = torch.cat([Q.reshape(-1, 1), P.reshape(-1, 1)], -1).to(device)\n",
    "u = torch.linspace(-u_lim, u_lim, n_u)[:,None].to(device)\n",
    "\n",
    "X = x.repeat(n_u, 1, 1)\n",
    "U = u.repeat(n_x, 1, 1).permute(1, 0, 2)\n",
    "X = X.cpu()\n",
    "U = U.cpu()\n",
    "res_eu, res_he, res_mp, res_rk, res_dp = [], [], [], [], []\n",
    "\n",
    "for Δt, i in zip(Δt_test, range(len(Δt_test))):\n",
    "    res_eu.append(R(X, U, Δt, 'euler').cpu()) \n",
    "    res_he.append(R(X, U, Δt, 'euler', hypereuler=hs_list[i].cpu()).cpu())\n",
    "    res_mp.append(R(X, U, Δt, 'midpoint').cpu())\n",
    "    res_rk.append(R(X, U, Δt, 'rk4').cpu())\n",
    "    res_dp.append(R(X, U, Δt, 'dopri5').cpu())\n",
    "u = u.cpu().squeeze()\n"
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "source": [
    "plt.rcParams.update({\n",
    "    \"text.usetex\": True,\n",
    "    \"font.family\": \"serif\",\n",
    "    \"font.serif\": [\"Palatino\"],\n",
    "})"
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 42,
   "source": [
    "\n",
    "\n",
    "alpha = .1\n",
    "\n",
    "fig, ax = plt.subplots(1, len(hs_list), figsize=(10, 2))\n",
    "# plt.tight_layout()\n",
    "\n",
    "for Δt, i in zip(Δt_test, range(len(Δt_test))):\n",
    "        # HyperEuler\n",
    "    ax[i].plot(u, res_he[i].mean(1), label='HyperEuler', color='tab:orange')\n",
    "    # ax[i].fill_between(u, res_he[i].mean(1) - res_he[i].std(1),\n",
    "    #                 res_he[i].mean(1) + res_he[i].std(1), alpha=alpha, color='tab:orange')\n",
    "    ax[i].plot(u, res_eu[i].mean(1), label='Euler', color='tab:red')\n",
    "    # ax[i].fill_between(u, res_eu[i].mean(1) - res_eu[i].std(1),\n",
    "    #                 res_eu[i].mean(1) + res_eu[i].std(1), alpha=alpha, color='tab:red')\n",
    "\n",
    "    ax[i].plot(u, res_mp[i].mean(1), label='Midpoint', color='tab:green')\n",
    "    # ax[i].fill_between(u, res_mp[i].mean(1) - res_mp[i].std(1),\n",
    "    #                 res_mp[i].mean(1) + res_mp[i].std(1), alpha=alpha, color='tab:green')\n",
    "    ax[i].plot(u, res_rk[i].mean(1), label='RK4', color='tab:purple')\n",
    "    # ax[i].fill_between(u, res_rk[i].mean(1) - res_rk[i].std(1),\n",
    "    #                 res_rk[i].mean(1) + res_rk[i].std(1), alpha=alpha, color='tab:purple')\n",
    "#     ax[i].plot(u, res_dp[i].mean(1), label='Dopri5', color='purple')\n",
    "#     ax[i].fill_between(u, res_dp[i].mean(1) - res_dp[i].std(1),\n",
    "#                     res_dp[i].mean(1) + res_dp[i].std(1), alpha=alpha, color='purple')\n",
    "    \n",
    "    ax[i].set_title(r'$\\epsilon = {}~[s]$'.format(Δt))\n",
    "    ax[i].set_xlabel(r'$u~[N]$')\n",
    "#     ax[i].legend()\n",
    "    ax[i].set_yscale('log')\n",
    "    ax[i].label_outer()\n",
    "    ax[i].set_ylim([1e-8, 1e2])\n",
    "    ax[i].set_yticks([1e-8, 1e-5, 1e-2, 1e01])\n",
    "#     ax[i].yaxis.set_major_locator(plt.MaxNLocator(5))\n",
    "    \n",
    "ax[0].set_ylabel(r'Mean Residual $R$')\n",
    "plt.legend(loc='lower center', ncol=1, bbox_to_anchor=(1.53, 0.15), fancybox=True)\n",
    "# plt.suptitle('Solver residuals with different step sizes',fontsize=14, weight='semibold', y=1.04)\n",
    "\n",
    "# ## Saving\n",
    "import tikzplotlib\n",
    "fig.savefig('residuals_pendulum_multi_epsilon.pdf',  bbox_inches = 'tight')\n",
    "tikzplotlib.save(\"residuals_pendulum_multi_epsilon.tex\")"
   ],
   "outputs": [
    {
     "output_type": "display_data",
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 720x144 with 5 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     }
    }
   ],
   "metadata": {}
  }
 ],
 "metadata": {
  "kernelspec": {
   "name": "python3",
   "display_name": "Python 3.8.5 64-bit ('base': conda)"
  },
  "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.5"
  },
  "interpreter": {
   "hash": "d77f8d9122331bf0c813f643ab906d6086736a1197fa074182ffff0b1ac62f18"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}