{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 44,
   "source": [
    "import torch\n",
    "import torch.nn as nn\n",
    "from torchdiffeq import odeint\n",
    "import sys; sys.path.append(2*'../')\n",
    "from src import *\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "# device = torch.device('cuda:1') # for second gpu\n",
    "device = 'cpu' # feel free to change :)\n",
    "\n",
    "\n",
    "plt.rcParams.update({\n",
    "    \"text.usetex\": True,\n",
    "    \"font.family\": \"serif\",\n",
    "    \"font.serif\": [\"Palatino\"],\n",
    "})"
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 45,
   "source": [
    "m, k = 1, 0.5\n",
    "\n",
    "class ControlledSystem(nn.Module):\n",
    "    \"\"\"Spring-mass system\"\"\"\n",
    "    def __init__(self, u):\n",
    "        super().__init__()\n",
    "        self.u = u\n",
    "        \n",
    "    def forward(self, t, x):\n",
    "        q, p = x[...,:1], x[...,1:]\n",
    "        dq = p/m\n",
    "        dp = -k*q + self.u\n",
    "        return torch.cat([dq, dp], -1)\n",
    "    \n",
    "def R(X, u, Δt, method, hypereuler=None, base_solver='dopri5'):\n",
    "    t_span = torch.tensor([0, Δt]).to(device)\n",
    "    with torch.no_grad():\n",
    "        if base_solver == 'analytical':\n",
    "             x_fine = forward_linear_system(X, u, Δt)\n",
    "        else:\n",
    "             x_fine = odeint(ControlledSystem(u), X, t_span, method='dopri5')[-1]\n",
    "        if hypereuler:\n",
    "            xfu = torch.cat([X, ControlledSystem(u)(0, X), u], -1)\n",
    "            x_coarse =  X + Δt*ControlledSystem(u)(0, X) + (Δt**2)*hypereuler(xfu)\n",
    "        else:\n",
    "            x_coarse = odeint(ControlledSystem(u), X, t_span, method=method)[-1]\n",
    "        R = (x_fine - x_coarse)/(Δt**2)\n",
    "        L = torch.norm(R, dim=-1, p=2)\n",
    "        return L"
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "markdown",
   "source": [
    "## [Supplementary] Calculating the analytical residuals for linear systems\n",
    "\n",
    "$\\dot x_t = A x_t + B u_t ~\\Rightarrow x_{k+1} = e^{A\\Delta t}x_k + (e^{A\\Delta t} - I)A^{-1}B u_k~$\n",
    "\n",
    "$x_{k+1} = x_k + \\Delta t (Ax_k + Bu_k)$\n",
    "\n",
    "$R_{x_k}(u_k) = e^{A\\Delta t}x_k + (e^{A\\Delta t} - I)A^{-1}B u_k - x_k - \\Delta t (Ax_k + Bu_k)$"
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 46,
   "source": [
    "device = 'cpu'"
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 47,
   "source": [
    "# System dynamics\n",
    "A = torch.tensor([[0, 1/m],\n",
    "                  [-k,  0]]).to(device)\n",
    "B = torch.tensor([[0],\n",
    "                  [1.]]).to(device)\n",
    "\n",
    "def forward_linear_system(x, u, Δt):\n",
    "    '''Solve linear system analytically for one step\n",
    "    Works with batched input\n",
    "    '''\n",
    "    eAΔt = torch.matrix_exp(A*Δt)\n",
    "    I = torch.eye(A.shape[0]).to(device)\n",
    "    Ainv = torch.inverse(A)\n",
    "    # Non pensar male...\n",
    "    x_anal = torch.einsum('kl, bjl-> bjk', eAΔt, x) \\\n",
    "           + torch.einsum('kl, bjl -> bjk', torch.mm(torch.mm((eAΔt - I), Ainv), B), u)\n",
    "    return x_anal"
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "markdown",
   "source": [
    "## Training with single $\\Delta t$"
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 48,
   "source": [
    "Δt = .03\n",
    "\n",
    "x_min_train, x_max_train = -20, 20\n",
    "u_lim = 100\n",
    "bs_hs = 1000\n",
    "n_grid = 100"
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 49,
   "source": [
    "def residual_loss(g, X, u, Δt, method, base_solver='dopri5', order=1):\n",
    "    t_span = torch.tensor([0, Δt]).to(device)\n",
    "    if base_solver == 'analytical':\n",
    "        x_fine = forward_linear_system(X, u, Δt)\n",
    "    else:\n",
    "        x_fine = odeint(ControlledSystem(u), X, t_span, method='dopri5')[-1]\n",
    "    x_coarse = odeint(ControlledSystem(u), X, t_span, method=method)[-1]\n",
    "    xfu = torch.cat([X, ControlledSystem(u)(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": "markdown",
   "source": [
    "## Training with Stochastic Exploration"
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 50,
   "source": [
    "hdim = 32\n",
    "hs_stochastic = nn.Sequential(\n",
    "    nn.Linear(5, hdim),\n",
    "    nn.Softplus(),\n",
    "    nn.Linear(hdim, hdim),\n",
    "    nn.Tanh(),\n",
    "    nn.Linear(hdim, 2)).to(device)\n",
    "\n",
    "from tqdm import trange\n",
    "\n",
    "EPOCHS = 10000\n",
    "losses = []\n",
    "opt_hs = torch.optim.Adam(hs_stochastic.parameters(), lr=1e-3)\n",
    "with trange(0, EPOCHS, desc=\"Epochs\") as epochs:\n",
    "    for epoch in epochs:\n",
    "        '''Hypersolver training loop with given Δt_test'''\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_stochastic, x[:, None, :], u_rand[:, None, :],  Δt, 'euler').mean()\n",
    "        # Optimization step\n",
    "        loss.backward()\n",
    "        opt_hs.step()\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%|██████████| 10000/10000 [00:56<00:00, 176.29it/s, loss=0.227]\n"
     ]
    }
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 51,
   "source": [
    "fig, ax = plt.subplots(1,1)\n",
    "ax.plot(losses); ax.set_yscale('log')\n"
   ],
   "outputs": [
    {
     "output_type": "display_data",
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     }
    }
   ],
   "metadata": {}
  },
  {
   "cell_type": "markdown",
   "source": [
    "## Training with Active Error Minimization"
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 52,
   "source": [
    "hdim = 32\n",
    "hs_active = nn.Sequential(\n",
    "    nn.Linear(5, hdim),\n",
    "    nn.Softplus(),\n",
    "    nn.Linear(hdim, hdim),\n",
    "    nn.Tanh(),\n",
    "    nn.Linear(hdim, 2)).to(device)\n",
    "losses = []\n",
    "\n",
    "opt_hs = torch.optim.Adam(hs_active.parameters(), lr=1e-3)\n",
    "with trange(0, EPOCHS, desc=\"Epochs\") as epochs:\n",
    "    for epoch in epochs:\n",
    "        '''Hypersolver training loop with given Δt_test'''\n",
    "        opt_hs.zero_grad()\n",
    "        \n",
    "        # Randomize state and controller values\n",
    "        x = torch.Tensor(bs_hs, 2).uniform_(x_min_train, x_max_train).to(device)\n",
    "\n",
    "        # Find the argmax\n",
    "        x_grid = x.repeat(n_grid, 1, 1).detach()\n",
    "        controller_grid = torch.Tensor(n_grid, 1, 1).uniform_(-u_lim, u_lim).to(device)\n",
    "        controller_grid = controller_grid.repeat(1, bs_hs, 1)\n",
    "        r = residual_loss(hs_active, x_grid, controller_grid,  Δt, method='euler')\n",
    "        max_r = torch.max(r, 0)\n",
    "        u_argmax = controller_grid[:,0,:][max_r.indices].to(device)\n",
    "        # Loss is based on the argmax of controllers we have found\n",
    "        loss = residual_loss(hs_active, x[:, None, :], u_argmax[:, None, :],  Δt, 'euler').mean()\n",
    "\n",
    "        # Optimization step\n",
    "        loss.backward()\n",
    "        opt_hs.step()\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%|██████████| 10000/10000 [10:31<00:00, 15.85it/s, loss=0.586]\n"
     ]
    }
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 53,
   "source": [
    "fig, ax = plt.subplots(1,1)\n",
    "ax.plot(losses); ax.set_yscale('log')"
   ],
   "outputs": [
    {
     "output_type": "display_data",
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     }
    }
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 54,
   "source": [
    "torch.save(hs_stochastic, 'saved_models/hs_stochastic.pt')\n",
    "torch.save(hs_active, 'saved_models/hs_active.pt')"
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "markdown",
   "source": [
    "## Plotting"
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 55,
   "source": [
    "import matplotlib.pyplot as plt\n",
    "\n",
    "n_x, n_u = 10000, 100\n",
    "x = torch.Tensor(n_x, 2).uniform_(x_min_train, x_max_train).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",
    "\n",
    "res_euler = R(X, U, Δt, 'euler').cpu()\n",
    "res_hypereuler_stochastic = R(X, U, Δt, 'euler', hypereuler=hs_stochastic.cpu()).cpu()\n",
    "res_hypereuler_active = R(X, U, Δt, 'euler', hypereuler=hs_active.cpu()).cpu()\n",
    "# res_hypereuler_max = R(X, U, Δt, 'euler', hypereuler=hs_max.cpu()).cpu()\n",
    "res_midpoint = R(X, U, Δt, 'midpoint').cpu()\n",
    "res_rk = R(X, U, Δt, 'rk4').cpu()\n",
    "# res_dopri = R(X, U, Δt, 'dopri5').cpu()\n",
    "\n",
    "u = u.cpu().squeeze()\n",
    "\n",
    "alpha = .1\n",
    "\n",
    "fig, ax = plt.subplots(1, 1, figsize=(6, 6))\n",
    "ax.plot(u, res_euler.mean(1), label='Euler', color='r')\n",
    "ax.fill_between(u, res_euler.mean(1) - res_euler.std(1),\n",
    "                res_euler.mean(1) + res_euler.std(1), alpha=alpha, color='red')\n",
    "ax.plot(u, res_midpoint.mean(1), label='Midpoint', color='g')\n",
    "ax.fill_between(u, res_midpoint.mean(1) - res_midpoint.std(1),\n",
    "                res_midpoint.mean(1) + res_midpoint.std(1), alpha=alpha, color='green')\n",
    "ax.plot(u, res_rk.mean(1), label='RK4', color='orange')\n",
    "ax.fill_between(u, res_rk.mean(1) - res_rk.std(1),\n",
    "                res_rk.mean(1) + res_rk.std(1), alpha=alpha, color='orange')\n",
    "# ax.plot(u, res_dopri.mean(1), label='Dopri5', color='purple')\n",
    "# ax.fill_between(u, res_dopri.mean(1) - res_dopri.std(1),\n",
    "#                 res_dopri.mean(1) + res_dopri.std(1), alpha=alpha, color='purple')\n",
    "\n",
    "# HyperEuler\n",
    "ax.plot(u, res_hypereuler_stochastic.mean(1), label='HyperEuler Stochastic', color='k')\n",
    "ax.fill_between(u, res_hypereuler_stochastic.mean(1) - res_hypereuler_stochastic.std(1),\n",
    "                res_hypereuler_stochastic.mean(1) + res_hypereuler_stochastic.std(1), alpha=alpha, color='black')\n",
    "ax.plot(u, res_hypereuler_active.mean(1), label='HyperEuler Active', color='purple')\n",
    "ax.fill_between(u, res_hypereuler_active.mean(1) - res_hypereuler_active.std(1),\n",
    "                res_hypereuler_active.mean(1) + res_hypereuler_active.std(1), alpha=alpha, color='black')\n",
    "\n",
    "\n",
    "ax.set_xlabel(r'Controller value $[N]$')\n",
    "ax.set_ylabel(r'Residual $\\mathcal{R}$')\n",
    "ax.legend()\n",
    "ax.set_yscale('log')\n",
    "\n",
    "# plt.savefig('images/residuals_linear.pdf')\n",
    "\n",
    "# import tikzplotlib\n",
    "# tikzplotlib.save(\"images/residuals_linear.tex\")"
   ],
   "outputs": [
    {
     "output_type": "display_data",
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 432x432 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     }
    }
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 56,
   "source": [
    "def mae(yhat, y):\n",
    "    '''Calculate MAE: Mean Average Error\n",
    "    '''\n",
    "    with torch.no_grad():\n",
    "        err = torch.abs(yhat - y).mean(dim=2)\n",
    "    return err.mean(1).numpy(), err.std(1).numpy()\n",
    "\n",
    "def mape(yhat, y):\n",
    "    '''Calculate MAPE: Mean Average Percentage Error\n",
    "    '''    \n",
    "    with torch.no_grad():\n",
    "        err = torch.abs((yhat - y)/yhat).mean(dim=2)\n",
    "    return err.mean(1).numpy(), err.std(1).numpy() "
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "markdown",
   "source": [
    "## Propagate the system forward"
   ],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 72,
   "source": [
    "tf = 0.3\n",
    "t_span = torch.linspace(0, tf, int(tf/Δt) + 1)\n",
    "bs = 10000\n",
    "x0 = torch.Tensor(bs, 2).uniform_(x_min_train/5, x_max_train/5)\n",
    "\n",
    "# We propagate the system forward one step at a time\n",
    "def random_forward_propagation(x0, t_span, hypersolver=None, bang_bang=False, method='midpoint'):\n",
    "    sys = ControlledSystem(1)\n",
    "    x, x_nom = [x0[None]], [x0[None]]\n",
    "    xt, xt_nom = x0[None], x0[None]\n",
    "    bs = x0.shape[0]\n",
    "    for t in range(len(t_span)-1):\n",
    "        if bang_bang: u = 2*u_lim*torch.randint(2, (1, bs, 1), dtype=torch.float) - u_lim\n",
    "        # if bang_bang: u = torch.Tensor([100], dtype=torch.float).repead()\n",
    "        else: u = torch.Tensor(1, bs, 1).uniform_(-u_lim, u_lim)\n",
    "        sys.u = (u.float())\n",
    "        diff_span = torch.linspace(t, t+Δt, 2)\n",
    "        if hypersolver:\n",
    "            f = sys(t, xt)\n",
    "            xfu = torch.cat([xt, f, u], -1)\n",
    "            # Hypersolver\n",
    "            xt = xt + Δt*f + Δt**2*hypersolver(xfu)\n",
    "        else:\n",
    "            xt = odeint(sys, xt, diff_span, method=method)[-1]\n",
    "        # Nominal\n",
    "        xt_nom = odeint(sys, xt_nom, diff_span, method='dopri5')[-1]\n",
    "        # You can also change to the analytical version\n",
    "        # xt_nom = forward_linear_system(xt_nom, u, Δt) \n",
    "        x.append(xt); x_nom.append(xt_nom)\n",
    "    return torch.cat(x), torch.cat(x_nom)\n",
    "\n",
    "he_traj, nom_traj = random_forward_propagation(x0, t_span, hs_stochastic)\n",
    "he_traj_argmax, nom_traj_argmax = random_forward_propagation(x0, t_span, hs_active)\n",
    "mp_traj, nom_traj_mp = random_forward_propagation(x0, t_span, method='midpoint')\n",
    "he_traj_bb, nom_traj_bb = random_forward_propagation(x0, t_span, hs_stochastic, bang_bang=True)\n",
    "he_traj_argmax_bb, nom_traj_argmax_bb = random_forward_propagation(x0, t_span, hs_active, bang_bang=True)\n",
    "mp_traj_bb, nom_traj_mp_bb = random_forward_propagation(x0, t_span, method='midpoint', bang_bang=True)"
   ],
   "outputs": [],
   "metadata": {}
  },
  {
   "cell_type": "code",
   "execution_count": 73,
   "source": [
    "# Plotting\n",
    "# stochastic, active hypersolvers and midpoint\n",
    "# random\n",
    "err_he, std_he = mae(nom_traj, he_traj)\n",
    "err_he_argmax, std_he_argmax = mae(nom_traj_argmax, he_traj_argmax)\n",
    "err_mp, std_mp = mae(mp_traj, nom_traj_mp)\n",
    "\n",
    "# bang-bang\n",
    "err_he_bb, std_he_bb = mae(nom_traj_bb, he_traj_bb)\n",
    "err_he_argmax_bb, std_he_argmax_bb = mae(nom_traj_argmax_bb, he_traj_argmax_bb)\n",
    "err_mp_bb, std_mp_bb = mae(mp_traj_bb, nom_traj_mp_bb)\n",
    "\n",
    "\n",
    "fig, ax = plt.subplots(2, 1, figsize=(3, 3))\n",
    "alpha = .1\n",
    "\n",
    "ax[0].plot(t_span, err_he, label='Stochastic exploration', color='tab:orange')\n",
    "ax[0].fill_between(t_span, err_he - std_he, err_he + std_he, alpha=alpha, color='tab:orange')\n",
    "ax[0].plot(t_span, err_he_argmax, label='Active residual minimization', color='grey')\n",
    "ax[0].fill_between(t_span, err_he_argmax - std_he_argmax, err_he_argmax + std_he_argmax, alpha=alpha,color='grey')\n",
    "ax[0].plot(t_span, err_mp, label='Midpoint', color='tab:green')\n",
    "ax[0].fill_between(t_span, err_mp - std_mp, err_mp + std_mp, alpha=alpha,color='tab:green')\n",
    "\n",
    "# ax[0].set_title(r'$Random~control~input$')\n",
    "ax[0].set_xlabel(r'$Time~[s]$')\n",
    "ax[0].set_ylabel(r'$MAE$')\n",
    "\n",
    "ax[1].plot(t_span, err_he_bb, label='Stochastic Exploration', color='tab:orange')\n",
    "ax[1].fill_between(t_span, err_he_bb - std_he_bb, err_he_bb + std_he_bb, alpha=alpha, color='tab:orange')\n",
    "ax[1].plot(t_span, err_he_argmax_bb, label='Active Error Minimization',color='grey')\n",
    "ax[1].fill_between(t_span, err_he_argmax_bb - std_he_argmax_bb, err_he_argmax_bb + std_he_argmax_bb, alpha=alpha, color='grey')\n",
    "ax[1].plot(t_span, err_mp_bb, label='Midpoint', color='tab:green')\n",
    "ax[1].fill_between(t_span, err_mp_bb - std_mp_bb, err_mp_bb + std_mp_bb, alpha=alpha,color='tab:green')\n",
    "ax[1].legend(loc='lower center', ncol=3, bbox_to_anchor=(0.5, -0.4))\n",
    "# ax[1].set_title(r'$Bang-bang~controller$')\n",
    "ax[1].set_xlabel(r'$Time~[s]$')\n",
    "ax[1].set_ylabel(r'$MAE$')\n",
    "# ax[0].set_yscale('log')\n",
    "# ax[1].set_yscale('log')\n",
    "\n",
    "for ax in ax:\n",
    "    ax.label_outer()\n",
    "    ax.yaxis.set_major_locator(plt.MaxNLocator(4))    \n",
    "    # ax.ticklabel_format(style='sci', axis='y', scilimits=(0,0))\n",
    "    ax.set_ylim(bottom=0)\n",
    "# plt.suptitle('Error propagation with different hypersolver pre-training',fontsize=14, weight='semibold', y=.97)\n",
    "ax.legend(loc='upper center', bbox_to_anchor=(0.5, -0.4),\n",
    "          fancybox=True, shadow=False, ncol=1)\n",
    "\n",
    "plt.savefig('media/residuals_stochastic_active.pdf', bbox_inches='tight')\n",
    "import tikzplotlib\n",
    "tikzplotlib.save(\"media/residuals_stochastic_active.tex\")"
   ],
   "outputs": [
    {
     "output_type": "display_data",
     "data": {
      "image/png": "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9K5du1Lf+c53zu7YsWNW17Fz586ruu6A2uufybU3CnrqZir2/auVXOyLz2ub3ZsLbP+vg5MdTqfT5ksvvdSybdu2Anhf4P3798ez2ay1Z8+e5K5du1KPP/748t7e3vLBgwebvvSlLw1t27atsGfPnmR/f3/k7bffjv/lX/7lAHjW7s0334zv2LEjvXnz5krtvn/0j/7R6LZt2wp9fX3h119/vWl4eNgEuOeeewq159q7d29iz549S4Mvfe15fvaznx2b7D727dvXsn379tE9e/YkX3nlldYXX3xxAGD//v3xP/qjP1oK8Nhjjw0999xzHQ8++GBuIqGOv89nnnkm8fd///etO3fuHOrv74985zvfOTv+fmo/q87OTrf22se399JLL7UcOHCg6bHHHhsa/zktRLRYFxh/8id/cub3f//3V6xcuXILwB/+4R+e/djHPpZra2tzPv/5z6d37969pLe3t7xr165UX19f7p577tn89a9//UxnZ6e7a9eus/v3748DDA4ORjZs2FAGePTRR1f/7Gc/O7Z///74hg0byslk0v3sZz/bMzg4+O6f/umfdr3//e8v+O2Fu7u73eBc4P1YfP3rX18OnlDHn2c8f/qnf9rV29tb7u/vj2zfvn10165dqYMHDzalUinzP//n/5z88pe/nPra1752/rOf/WzPhg0byrt37z736U9/ujeRSDjbt28fDdqZ6D5zudwvhRBL77nnnsI999xTmOh+Nm7cWA6uP5lMusG1T9Tea6+91vcXf/EXXeM/p3r++14NWqxTMYUFrBebN2+uBFaor68vfM8992zeuXPnL1tbW91kMukeOHCg6ZFHHkkHrwV44YUXEk8//fQpgG3bthVSqZS5cuXK8ubNmyubN2+uPPHEE6uDY/v27WsJvpwATzzxxNCjjz66+utf//ry3/u93zu/a9euVHCu8dc2/jwTXf8TTzwxtHnz5kpfX1842Pftb3/7zIc+9KENzz//fH8ymXQ3btxYDq4P4BOf+MTI8ePHI0BVrBPdJ8CKFSvKtecefz/JZNKd6Ponaq+7u9ud6HNaqOg+6wIjGFAC78u0ZcuWfO3xNWvWlN94442qVVu5cmV5zZo1lWeffbb6vosXL044GBUM+NR++UdGRqyf/exnxwYHB9994YUXEhO9r+bcl51nKjZv3lwJrO/3vve9xM6dO4cm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",
      "text/plain": [
       "<Figure size 216x216 with 2 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
}