{
 "cells": [
  {
   "cell_type": "markdown",
   "source": [
    "# Training the Hypersolver - Plotting Results"
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "source": [
    "import os, sys\r\n",
    "import matplotlib.pyplot as plt\r\n",
    "import numpy as np\r\n",
    "import torch\r\n",
    "import torch.nn as nn\r\n",
    "import sys; sys.path.append(2*'../') # go 2 dirs back\r\n",
    "from src import *\r\n",
    "from math import pi\r\n",
    "import matplotlib\r\n"
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "source": [
    "device = 'cpu'"
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "source": [
    "ControlledSystem = QuadcopterGym\r\n"
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "markdown",
   "source": [
    "## Generate distribution of initial conditions"
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "source": [
    "from torch.distributions import Uniform\r\n",
    "\r\n",
    "low = -torch.ones(12).to(device)\r\n",
    "high = torch.ones(12).to(device)\r\n",
    "low[0:3]*=5\r\n",
    "high[0:3]*=5\r\n",
    "low[3:6]*=50\r\n",
    "high[3:6]*=50\r\n",
    "low[6:9]*=50\r\n",
    "high[6:9]*=50\r\n",
    "low[9:12]*=100\r\n",
    "high[9:12]*=100\r\n",
    "\r\n",
    "dist = Uniform(low, high)\r\n",
    "print(dist.sample_n(2))"
   ],
   "outputs": [
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "tensor([[  2.2692,   0.4009,  -1.4150, -12.4893,  -9.8831, -36.9169, -34.7797,\n",
      "         -15.0665,  21.2993,  78.6990,  23.4708, -45.4064],\n",
      "        [ -4.2104,   2.8153,  -2.3430, -28.1650,  13.6959, -21.1912, -24.7754,\n",
      "          39.9353, -39.5445, -51.5116,  72.7932, -40.0482]])\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/home/botu/anaconda3/lib/python3.8/site-packages/torch/distributions/distribution.py:161: UserWarning: sample_n will be deprecated. Use .sample((n,)) instead\n",
      "  warnings.warn('sample_n will be deprecated. Use .sample((n,)) instead', UserWarning)\n"
     ]
    }
   ],
   "metadata": {}
  },
  {
   "cell_type": "markdown",
   "source": [
    "## New output scaling\n",
    "Since position and angular positions derivatives $\\dot{x},~\\dot{\\psi}$ propagated via the augmented state without any calculation, then there is no need to add an additional term. Indeed, doing so would just degrade the performance unless it was always 0. So we put the output scaling to 0 for updates on positions and angular positions"
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "source": [
    "in_scal = torch.cat([1/high, 1/high])\n",
    "u_scal = 1/20000*torch.ones(4).to(device)\n",
    "in_scal = torch.cat([in_scal, u_scal])\n",
    "\n",
    "out_scal = high\n",
    "\n",
    "# Zero out positions contribution term\n",
    "out_scal[0:6] = 0 \n",
    "print('Input scaling:\\n', in_scal)\n",
    "print('Output scaling:\\n', out_scal)"
   ],
   "outputs": [
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "Input scaling:\n",
      " tensor([2.0000e-01, 2.0000e-01, 2.0000e-01, 2.0000e-02, 2.0000e-02, 2.0000e-02,\n",
      "        2.0000e-02, 2.0000e-02, 2.0000e-02, 1.0000e-02, 1.0000e-02, 1.0000e-02,\n",
      "        2.0000e-01, 2.0000e-01, 2.0000e-01, 2.0000e-02, 2.0000e-02, 2.0000e-02,\n",
      "        2.0000e-02, 2.0000e-02, 2.0000e-02, 1.0000e-02, 1.0000e-02, 1.0000e-02,\n",
      "        5.0000e-05, 5.0000e-05, 5.0000e-05, 5.0000e-05])\n",
      "Output scaling:\n",
      " tensor([  0.,   0.,   0.,   0.,   0.,   0.,  50.,  50.,  50., 100., 100., 100.])\n"
     ]
    }
   ],
   "metadata": {}
  },
  {
   "cell_type": "markdown",
   "source": [
    "### Scaling\n",
    "We could apply a residual loss with scaling so to give the same \"importance\" to all of the parameters"
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "source": [
    "Δt = 0.02\n",
    "\n",
    "x_min_train, x_max_train = -1000, 1000\n",
    "\n",
    "bs_hs = 1024\n",
    "n_grid = 100\n",
    "\n"
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "markdown",
   "source": [
    "## Hypersolver"
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "source": [
    "\n",
    "hdim = 64\n",
    "\n",
    "class Hypersolver(nn.Module):\n",
    "    def __init__(self):\n",
    "        super().__init__()\n",
    "        layers = [nn.Linear(28, hdim),\n",
    "            nn.Softplus(),\n",
    "        #     Snake(hdim),\n",
    "            nn.Linear(hdim, hdim),\n",
    "            nn.Softplus(),\n",
    "        #     Snake(hdim),\n",
    "            nn.Linear(hdim, 12)]\n",
    "        self.layers = nn.Sequential(*layers)\n",
    "        \n",
    "    def forward(self, t, x):\n",
    "        x = x*in_scal\n",
    "        x = self.layers(x)\n",
    "        x = x*out_scal\n",
    "        return x\n",
    "    \n",
    "hs = torch.load('saved_models/hypersolver_0.02_new_quadcopter.pt')"
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "markdown",
   "source": [
    "## Residuals"
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "source": [
    "def calculate_residuals(X, u, Δt, method, hypereuler=None):\n",
    "    t_span = torch.tensor([0, Δt]).to(device)\n",
    "    with torch.no_grad():\n",
    "        x_fine = odeint(ControlledSystem(u)._dynamics, X, t_span, method='dopri5')[-1]\n",
    "        if hypereuler:\n",
    "            xfu = torch.cat([X, ControlledSystem(u)._dynamics(0, X), u.u0], -1)\n",
    "            x_coarse =  X + Δt*ControlledSystem(u)._dynamics(0, X) + (Δt**2)*hypereuler(0, xfu)\n",
    "        else:\n",
    "            x_coarse = odeint(ControlledSystem(u)._dynamics, X, t_span, method=method)[-1]\n",
    "    return torch.norm(((x_fine - x_coarse)), p=2, dim=-1)"
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "source": [
    "u_rand = RandConstController((3, 3), 1, 1)\n",
    "drone = ControlledSystem(u_rand)\n",
    "max_controller = drone.MAX_RPM\n",
    "\n",
    "bs = 1000\n",
    "x = dist.sample((bs,)).to(device)\n",
    "u_rand.u0 = torch.Tensor(bs, 4)[None].uniform_(0, max_controller).to(device)\n",
    "\n",
    "res_hypereuler = calculate_residuals(x[None], u_rand, Δt, 'euler', hypereuler=hs)\n",
    "res_euler = calculate_residuals(x[None], u_rand, Δt, 'euler')\n",
    "res_midpoint = calculate_residuals(x[None], u_rand, Δt, 'midpoint')\n",
    "res_rk4 = calculate_residuals(x[None], u_rand, Δt, 'rk4')"
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "source": [
    "res_he = res_hypereuler.squeeze(0).cpu().numpy()\n",
    "res_eu = res_euler.squeeze(0).cpu().numpy()\n",
    "res_mp = res_midpoint.squeeze(0).cpu().numpy()\n",
    "res_rk = res_rk4.squeeze(0).cpu().numpy()"
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "source": [
    "plt.rcParams.update({\n",
    "    \"text.usetex\": True,\n",
    "    \"font.family\": \"serif\",\n",
    "    \"font.serif\": [\"Palatino\"],\n",
    "})"
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 36,
   "source": [
    "colors = [ 'tab:orange', 'tab:red', 'tab:green', 'tab:purple']\n",
    "labels = [ 'HyperEuler','Euler', 'Midpoint', 'RK4']\n",
    "\n",
    "\n",
    "fig, ax = plt.subplots(1,1, figsize=(4,4))\n",
    "\n",
    "bp = ax.boxplot([res_he, res_eu, res_mp, res_rk],\n",
    "                patch_artist=True, labels=labels)\n",
    "for b, m, c in zip(bp['boxes'], bp['medians'], colors):\n",
    "    b.set_facecolor(c)\n",
    "    b.set_alpha(0.6)\n",
    "    m.set(color='black', linestyle='-.')\n",
    "ax.set_yscale('log')\n",
    "# ax.set_ylim(0, 0.02)\n",
    "# ax.set_title('Quadcopter residuals distribution')\n",
    "ax.set_ylabel(r'Mean Residual $R$')\n",
    "\n",
    "# ## Saving\n",
    "import tikzplotlib\n",
    "fig.savefig('media/quadcopter_residuals.pdf',  bbox_inches = 'tight')\n",
    "tikzplotlib.save(\"media/quacopter_residuals.tex\")"
   ],
   "outputs": [
    {
     "output_type": "display_data",
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 288x288 with 1 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": 4
}