{
 "cells": [
  {
   "cell_type": "markdown",
   "source": [
    "# Hypersolver training with different architectures"
   ],
   "metadata": {}
  },
  {
   "cell_type": "markdown",
   "source": [
    "Here we perform the training with different architectures. In particular, we consider Multi-layers Perceptrons with different activations:\n",
    "1. Tanh\n",
    "2. ReLU\n",
    "3. Snake\n",
    "And also the architecture of\n",
    "4. SIREN\n",
    "\n",
    "And save the results for an analysis on generalization"
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 63,
   "source": [
    "import torch \n",
    "import torch.nn as nn \n",
    "from warnings import warn\n",
    "import matplotlib.pyplot as plt\n",
    "import sys; sys.path.append(2*'../')\n",
    "from src import *\n",
    "\n",
    "device = torch.device('cuda:1' if(torch.cuda.is_available()) else 'cpu')\n",
    "print('Device:', device)"
   ],
   "outputs": [
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "Device: cuda:1\n"
     ]
    }
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 64,
   "source": [
    "device = 'cpu' # override"
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 65,
   "source": [
    "# Reinitialize\n",
    "bs = 1000\n",
    "\n",
    "U = RandConstController([bs, 1], -3, 3)\n",
    "system = ControlledPendulum(U)"
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "markdown",
   "source": [
    "## Residuals computation"
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 66,
   "source": [
    "from torchdiffeq import odeint as odeint\n",
    "\n",
    "def compute_residual(system, t, Δt, x, u, hs, model='SIREN', method='dopri5' ):\n",
    "    \"\"\"\n",
    "    x: u_batch, batch, dimr\n",
    "    u: u_batch, batch, dim\n",
    "    \"\"\"\n",
    "    t_span = torch.tensor([t, t + Δt]).to(x)\n",
    "    system.u.u0 = u\n",
    "    f = system._dynamics(t, x)\n",
    "\n",
    "    if x.shape == u.shape or u.shape[1] == 16:  # plots\n",
    "        xfu = torch.cat([x, f, u], -1)\n",
    "    else:\n",
    "        xfu = torch.cat([x, f, u.repeat(1, x.shape[1], 1)], -1)\n",
    "    \n",
    "    xfu = torch.cat([x, f, u], -1)\n",
    "    if model == 'SIREN':\n",
    "        g, _ = hs(xfu) # unpack for SIREN coord (unused)\n",
    "    elif model == 'euler':\n",
    "        g = 0\n",
    "    else:\n",
    "        g = hs(xfu)\n",
    "    zd = odeint(system._dynamics, x, t_span, method=method)[-1]\n",
    "    z  = system(x, t_span).squeeze(0)[-1]\n",
    "  \n",
    "    R = (zd - z)/(Δt**2)\n",
    "    L = torch.norm(R - g, dim=-1, p=2)\n",
    "    return L                  "
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 67,
   "source": [
    "# Common settings\n",
    "\n",
    "from math import pi\n",
    "x_min_train, x_max_train = -2*pi, 2*pi\n",
    "x_min_plot, x_max_plot = -15, 15 # we plot unexplored regions\n",
    "t = 0\n",
    "Δt = 0.1\n",
    "n_grid = 100 ; u_b = 10\n",
    "u_grid = torch.linspace(-u_b, u_b, n_grid).unsqueeze(1)\n",
    "\n",
    "\n",
    "EPOCHS = 200000\n",
    "lr = 1e-4"
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "markdown",
   "source": [
    "## Training the system with `SIREN-HyperEuler`"
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 68,
   "source": [
    "hs_siren = Siren(in_features=5, out_features=2, hidden_features=64, hidden_layers=2, outermost_linear=True)\n",
    "\n",
    "pytorch_total_params = sum(p.numel() for p in hs_siren.parameters())\n",
    "print(pytorch_total_params)\n",
    "# for name, param in hs.named_parameters():\n",
    "#     if param.requires_grad:\n",
    "#         print(name, param.data)"
   ],
   "outputs": [
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "8834\n"
     ]
    }
   ],
   "metadata": {}
  },
  {
   "cell_type": "markdown",
   "source": [
    "## Training loop SIREN-HyperEuler"
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 69,
   "source": [
    "from tqdm import trange\n",
    "\n",
    "opt = torch.optim.Adam(hs_siren.parameters(), lr=lr)\n",
    "\n",
    "losses = []\n",
    "\n",
    "bs = 1000\n",
    "count = 0\n",
    "\n",
    "with trange(0, EPOCHS, desc=\"Epochs\") as epochs:\n",
    "    for epoch in epochs:\n",
    "        opt.zero_grad()\n",
    "        \n",
    "        x = torch.Tensor(bs, 2).uniform_(x_min_train, x_max_train)\n",
    "        \n",
    "        # Choose random indices\",\n",
    "        index = torch.randint(u_grid.shape[0], (bs,))\n",
    "        u_rand = u_grid[index]\n",
    "\n",
    "        # compute again residual for hypersolver training\n",
    "        loss = compute_residual(system, t, Δt, x[:, None, :], u_rand[:, None, :], hs_siren).mean()\n",
    "\n",
    "        # optimization step\n",
    "        loss.backward()\n",
    "        opt.step()\n",
    "        \n",
    "        losses.append(loss.detach().cpu())\n",
    "        # print(f\"e:{epoch}, loss:%.3f\" % loss, end=\"\\r\")\n",
    "        epochs.set_postfix(loss=(loss.detach().cpu().item()))\n"
   ],
   "outputs": [
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "Epochs: 100%|██████████| 200000/200000 [27:47<00:00, 119.93it/s, loss=3.78]\n"
     ]
    }
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 70,
   "source": [
    "torch.save(hs_siren, 'saved_models/hs_siren.pt')"
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 71,
   "source": [
    "import numpy as np\n",
    "\n",
    "dummmy = np.linspace(0,EPOCHS,EPOCHS)\n",
    "fig, ax = plt.subplots(figsize=(12,6))\n",
    "ax.set_xlabel('Epoch'); ax.set_ylabel('Loss')\n",
    "ax.plot( np.asarray(losses))\n",
    "ax.set_title('Hypersolver loss')"
   ],
   "outputs": [
    {
     "output_type": "execute_result",
     "data": {
      "text/plain": [
       "Text(0.5, 1.0, 'Hypersolver loss')"
      ]
     },
     "metadata": {},
     "execution_count": 71
    },
    {
     "output_type": "display_data",
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 864x432 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     }
    }
   ],
   "metadata": {}
  },
  {
   "cell_type": "markdown",
   "source": [
    "## Snake"
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 72,
   "source": [
    "class SnakeHyperSolver(nn.Module):\n",
    "    def __init__(self, in_dim, out_dim, hidden_dim=32):\n",
    "        super().__init__()\n",
    "\n",
    "        # Initialize layers\n",
    "        self.fc1 = nn.Linear(in_dim, hidden_dim)\n",
    "        self.fc2 = nn.Linear(hidden_dim, hidden_dim)\n",
    "        self.fc3 = nn.Linear(hidden_dim, out_dim)\n",
    "\n",
    "        # Initialize activations\n",
    "        self.a1 = Snake(hidden_dim)\n",
    "        self.a2 = Snake(hidden_dim)\n",
    "\n",
    "    \n",
    "    def forward(self, x):\n",
    "        x = self.a1(self.fc1(x))\n",
    "        x = self.a2(self.fc2(x))\n",
    "        x = self.fc3(x)\n",
    "        return x\n",
    "\n",
    "hs_snake = SnakeHyperSolver(5, 2, hidden_dim=64)\n",
    "\n",
    "pytorch_total_params = sum(p.numel() for p in hs_snake.parameters())\n",
    "print(pytorch_total_params)"
   ],
   "outputs": [
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "4802\n"
     ]
    }
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 73,
   "source": [
    "opt = torch.optim.Adam(hs_snake.parameters(), lr=lr)\n",
    "\n",
    "losses = []\n",
    "\n",
    "bs = 1000\n",
    "count = 0\n",
    "\n",
    "with trange(0, EPOCHS, desc=\"Epochs\") as epochs:\n",
    "    for epoch in epochs:\n",
    "        opt.zero_grad()\n",
    "        \n",
    "        x = torch.Tensor(bs, 2).uniform_(x_min_train, x_max_train)\n",
    "        \n",
    "        # Choose random indices\",\n",
    "        index = torch.randint(u_grid.shape[0], (bs,))\n",
    "        u_rand = u_grid[index]\n",
    "\n",
    "        # compute again residual for hypersolver training\n",
    "        loss = compute_residual(system, t, Δt, x[:, None, :], u_rand[:, None, :], hs_snake, model='snake').mean()\n",
    "\n",
    "        # optimization step\n",
    "        loss.backward()\n",
    "        opt.step()\n",
    "        \n",
    "        losses.append(loss.detach().cpu())\n",
    "        # print(f\"e:{epoch}, loss:%.3f\" % loss, end=\"\\r\")\n",
    "        epochs.set_postfix(loss=(loss.detach().cpu().item()))"
   ],
   "outputs": [
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "Epochs: 100%|██████████| 200000/200000 [28:17<00:00, 117.85it/s, loss=0.135]\n"
     ]
    }
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 74,
   "source": [
    "import numpy as np\n",
    "\n",
    "dummmy = np.linspace(0,EPOCHS,EPOCHS)\n",
    "fig, ax = plt.subplots(figsize=(12,6))\n",
    "ax.set_xlabel('Epoch'); ax.set_ylabel('Loss')\n",
    "ax.plot( np.asarray(losses))\n",
    "ax.set_title('Hypersolver loss')"
   ],
   "outputs": [
    {
     "output_type": "execute_result",
     "data": {
      "text/plain": [
       "Text(0.5, 1.0, 'Hypersolver loss')"
      ]
     },
     "metadata": {},
     "execution_count": 74
    },
    {
     "output_type": "display_data",
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 864x432 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     }
    }
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 75,
   "source": [
    "torch.save(hs_snake, 'saved_models/hs_snake.pt')"
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "markdown",
   "source": [
    "## ReLU"
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 76,
   "source": [
    "class ReLUHyperSolver(nn.Module):\n",
    "    def __init__(self, in_dim, out_dim, hidden_dim=32):\n",
    "        super().__init__()\n",
    "\n",
    "        # Initialize layers\n",
    "        self.fc1 = nn.Linear(in_dim, hidden_dim)\n",
    "        self.fc2 = nn.Linear(hidden_dim, hidden_dim)\n",
    "        self.fc3 = nn.Linear(hidden_dim, out_dim)\n",
    "\n",
    "        # Initialize activations\n",
    "        self.a1 = nn.ReLU()\n",
    "        self.a2 = nn.ReLU()\n",
    "\n",
    "    \n",
    "    def forward(self, x):\n",
    "        x = self.a1(self.fc1(x))\n",
    "        x = self.a2(self.fc2(x))\n",
    "        x = self.fc3(x)\n",
    "        return x\n",
    "\n",
    "hs_relu = ReLUHyperSolver(5, 2, hidden_dim=64)\n",
    "\n",
    "pytorch_total_params = sum(p.numel() for p in hs_relu.parameters())\n",
    "print(pytorch_total_params)"
   ],
   "outputs": [
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "4674\n"
     ]
    }
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 77,
   "source": [
    "opt = torch.optim.Adam(hs_relu.parameters(), lr=lr)\n",
    "\n",
    "losses = []\n",
    "\n",
    "bs = 1000\n",
    "count = 0\n",
    "\n",
    "with trange(0, EPOCHS, desc=\"Epochs\") as epochs:\n",
    "    for epoch in epochs:\n",
    "        opt.zero_grad()\n",
    "        \n",
    "        x = torch.Tensor(bs, 2).uniform_(x_min_train, x_max_train)\n",
    "        \n",
    "        # Choose random indices\",\n",
    "        index = torch.randint(u_grid.shape[0], (bs,))\n",
    "        u_rand = u_grid[index]\n",
    "\n",
    "        # compute again residual for hypersolver training\n",
    "        loss = compute_residual(system, t, Δt, x[:, None, :], u_rand[:, None, :], hs_relu, model='relu').mean()\n",
    "\n",
    "        # optimization step\n",
    "        loss.backward()\n",
    "        opt.step()\n",
    "        \n",
    "        losses.append(loss.detach().cpu())\n",
    "        # print(f\"e:{epoch}, loss:%.3f\" % loss, end=\"\\r\")\n",
    "        epochs.set_postfix(loss=(loss.detach().cpu().item()))"
   ],
   "outputs": [
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "Epochs: 100%|██████████| 200000/200000 [25:19<00:00, 131.61it/s, loss=0.0862]\n"
     ]
    }
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 78,
   "source": [
    "import numpy as np\n",
    "\n",
    "dummmy = np.linspace(0,EPOCHS,EPOCHS)\n",
    "fig, ax = plt.subplots(figsize=(12,6))\n",
    "ax.set_xlabel('Epoch'); ax.set_ylabel('Loss')\n",
    "ax.plot( np.asarray(losses))\n",
    "ax.set_title('Hypersolver loss')"
   ],
   "outputs": [
    {
     "output_type": "execute_result",
     "data": {
      "text/plain": [
       "Text(0.5, 1.0, 'Hypersolver loss')"
      ]
     },
     "metadata": {},
     "execution_count": 78
    },
    {
     "output_type": "display_data",
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 864x432 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     }
    }
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 79,
   "source": [
    "torch.save(hs_relu, 'saved_models/hs_relu.pt')"
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "markdown",
   "source": [
    "## Tanh"
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 80,
   "source": [
    "class TanhHyperSolver(nn.Module):\n",
    "    def __init__(self, in_dim, out_dim, hidden_dim=32):\n",
    "        super().__init__()\n",
    "\n",
    "        # Initialize layers\n",
    "        self.fc1 = nn.Linear(in_dim, hidden_dim)\n",
    "        self.fc2 = nn.Linear(hidden_dim, hidden_dim)\n",
    "        self.fc3 = nn.Linear(hidden_dim, out_dim)\n",
    "\n",
    "        # Initialize activations\n",
    "        self.a1 = nn.Tanh()\n",
    "        self.a2 = nn.Tanh()\n",
    "\n",
    "    \n",
    "    def forward(self, x):\n",
    "        x = self.a1(self.fc1(x))\n",
    "        x = self.a2(self.fc2(x))\n",
    "        x = self.fc3(x)\n",
    "        return x\n",
    "\n",
    "hs_tanh = TanhHyperSolver(5, 2, hidden_dim=64)\n",
    "\n",
    "pytorch_total_params = sum(p.numel() for p in hs_tanh.parameters())\n",
    "print(pytorch_total_params)"
   ],
   "outputs": [
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "4674\n"
     ]
    }
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 81,
   "source": [
    "opt = torch.optim.Adam(hs_tanh.parameters(), lr=lr)\n",
    "\n",
    "losses = []\n",
    "\n",
    "bs = 1000\n",
    "count = 0\n",
    "\n",
    "with trange(0, EPOCHS, desc=\"Epochs\") as epochs:\n",
    "    for epoch in epochs:\n",
    "        opt.zero_grad()\n",
    "        \n",
    "        x = torch.Tensor(bs, 2).uniform_(x_min_train, x_max_train)\n",
    "        \n",
    "        # Choose random indices\",\n",
    "        index = torch.randint(u_grid.shape[0], (bs,))\n",
    "        u_rand = u_grid[index]\n",
    "\n",
    "        # compute again residual for hypersolver training\n",
    "        loss = compute_residual(system, t, Δt, x[:, None, :], u_rand[:, None, :], hs_tanh, model='tanh').mean()\n",
    "\n",
    "        # optimization step\n",
    "        loss.backward()\n",
    "        opt.step()\n",
    "        \n",
    "        losses.append(loss.detach().cpu())\n",
    "        # print(f\"e:{epoch}, loss:%.3f\" % loss, end=\"\\r\")\n",
    "        epochs.set_postfix(loss=(loss.detach().cpu().item()))"
   ],
   "outputs": [
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "Epochs: 100%|██████████| 200000/200000 [25:13<00:00, 132.13it/s, loss=0.026]\n"
     ]
    }
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 82,
   "source": [
    "import numpy as np\n",
    "\n",
    "dummmy = np.linspace(0,EPOCHS,EPOCHS)\n",
    "fig, ax = plt.subplots(figsize=(12,6))\n",
    "ax.set_xlabel('Epoch'); ax.set_ylabel('Loss')\n",
    "ax.plot( np.asarray(losses))\n",
    "ax.set_title('Hypersolver loss')"
   ],
   "outputs": [
    {
     "output_type": "execute_result",
     "data": {
      "text/plain": [
       "Text(0.5, 1.0, 'Hypersolver loss')"
      ]
     },
     "metadata": {},
     "execution_count": 82
    },
    {
     "output_type": "display_data",
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 864x432 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     }
    }
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 83,
   "source": [
    "torch.save(hs_tanh, 'saved_models/hs_tanh.pt')"
   ],
   "outputs": [],
   "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
}