{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Solution Error vs. Advection Coefficient for Frozen PINNs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Imports\n",
    "import sys\n",
    "sys.path.append('../../')\n",
    "sys.path.append('../../src')\n",
    "from swimpde import Domain\n",
    "from swimpde import BoundaryCompliantAnsatz\n",
    "from swimpde import AdvectionSolver\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "from sklearn.metrics import mean_squared_error\n",
    "np.random.seed(2)\n",
    "import time"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Problem setup"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "# create equally spaced points in a square\n",
    "n_points_1d = 400 \n",
    "x_lim = [0, 2 * np.pi]\n",
    "\n",
    "# coordinates of boundary points (excluding corners)\n",
    "left = x_lim[0]\n",
    "right = x_lim[1]\n",
    "boundary_points = np.row_stack([left, right])\n",
    "\n",
    "# initial condition\n",
    "def u0(x):\n",
    "    return np.sin(x)\n",
    "\n",
    "# forcing\n",
    "def forcing(x, t):\n",
    "    return np.zeros(x.shape[0])\n",
    "\n",
    "# boundary condition\n",
    "boundary_condition = \"periodic\" #strict\n",
    "\n",
    "# Analytical solution\n",
    "def analytical_sol(x, t, beta):\n",
    "    return np.sin(x - beta * t)\n",
    "\n",
    "# Test data\n",
    "t_eval = np.linspace(0, 1, 100) # time domain\n",
    "x_space_test = np.linspace(x_lim[0], x_lim[-1], 256).reshape((-1, 1)) # space domain\n",
    "xx_test, yy_test = np.meshgrid(x_space_test, t_eval)\n",
    "X_test = np.hstack((xx_test.reshape(-1, 1), yy_test.reshape(-1, 1))) # Uniform test points\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Fit and evaluate Frozen-PINN-swim and Frozen-PINN-elm for good hyper-parameters "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[0.033990939458211265, 0.04529658953348795, 6.21734451910065e-10, 2.461633726310224e-10, 4.376065445597293e-10, 1.3672746306216698e-10, 8.80987655586032e-10, 3.4880951486459583e-10, 6.200813909803206e-10, 1.937406022254471e-10]\n",
      "[0.03229506810506185, 0.03488659858703613, 6.246828525005253e-10, 2.44683541092217e-10, 4.37025971764743e-10, 1.3635571380227039e-10, 8.85165376375862e-10, 3.4671257243080615e-10, 6.192586481839443e-10, 1.9321381441102464e-10]\n",
      "[0.030913035074869793, 0.03629692395528158, 6.753044727772935e-10, 2.312712513669025e-10, 5.136094600325858e-10, 1.840734858032241e-10, 9.568829633881106e-10, 3.2770332387154185e-10, 7.277667510758903e-10, 2.6082573072880554e-10]\n",
      "[0.034858147303263344, 0.03917463620503744, 1.7918849421330857e-09, 2.4852942607655333e-10, 2.6204805645128592e-09, 1.150038270825521e-09, 2.5363530896855235e-09, 3.5178507441256137e-10, 3.7092024269990283e-09, 1.6278406346741118e-09]\n",
      "[0.03962484995524088, 0.042928059895833336, 4.1672278781393916e-09, 1.2611127485416254e-09, 1.0116718540867953e-08, 5.811514377023974e-09, 5.893949708534749e-09, 1.7836641849341508e-09, 1.4308656027205203e-08, 8.219558533933434e-09]\n",
      "[0.0704043706258138, 0.07095702489217122, 1.3570838933654503e-08, 6.021030111918092e-09, 3.726342966214891e-08, 2.2532723725824278e-08, 1.9191795541211574e-08, 8.514903125763686e-09, 5.2697709164226046e-08, 3.186563696222232e-08]\n",
      "[0.12911136945088705, 0.1297945181528727, 3.309170176566287e-08, 1.557435869774587e-08, 9.217532127597422e-08, 5.615762387462385e-08, 4.679916250706412e-08, 2.2025671233246e-08, 1.3035678461272848e-07, 7.941960145567812e-08]\n",
      "[0.47474320729573566, 0.5159637928009033, 1.6405340598040107e-07, 7.889599339343763e-08, 4.5885777797745067e-07, 2.805691257611658e-07, 2.320094328691476e-07, 1.1157716948007939e-07, 6.489309514784711e-07, 3.967895903131525e-07]\n",
      "[0.9059840043385824, 0.9287545680999756, 3.277987994544717e-07, 1.580397347340128e-07, 9.173362710494195e-07, 5.609677796828165e-07, 4.635856186665709e-07, 2.2350584664280908e-07, 1.2973320934898532e-06, 7.933421221464559e-07]\n",
      "[4.3903718789418535, 4.322274128595988, 1.6380509576538274e-06, 7.910830505188286e-07, 4.585177254582324e-06, 2.805020598589407e-06, 2.3165266578501117e-06, 1.1187472322137204e-06, 6.4843436595045445e-06, 3.96685155738864e-06]\n",
      "[7.934646129608154, 8.270491361618042, 3.2755422353345884e-06, 1.5820717270282667e-06, 9.169010681133713e-06, 5.609276810731465e-06, 4.6324489608621445e-06, 2.2374513901307847e-06, 1.2967310738282409e-05, 7.932942598863125e-06]\n"
     ]
    }
   ],
   "source": [
    "conv_coeffs = [1e-3, 1e-2, 1e-1, 1, 10, 40, 100, 500, 1000, 5000, 10000] # Values of convection coefficient\n",
    "n_c = 400 # Number of data points for convection coefficient 100\n",
    "seeds = [1, 2, 3]\n",
    "experiments = []\n",
    "\n",
    "n_IBF = 380\n",
    "n_OBF = 14\n",
    "\n",
    "# Parameter sampler for ELM: Sample weights from a normal distribution and biases uniformly from [-4, 4]\n",
    "def sample_parameters_randomly(x, _, rng):\n",
    "    weights = rng.normal(loc=0, scale=1, size=(x.shape[1], n_IBF))\n",
    "    biases = rng.uniform(low=-4, high=4, size=(1, n_IBF)) # low=-np.pi, high=np.pi,  2 * np.pi\n",
    "    idx0 = None\n",
    "    idx1 = None\n",
    "    return weights, biases, idx0, idx1\n",
    "\n",
    "# Periodic basis functions\n",
    "def periodic_outer_basis_sine(x_space, n_outer_basis=None, initial_condition=None):\n",
    "    return np.column_stack([\n",
    "        sine_cos(x_space * (i+1))\n",
    "        for i in range(n_outer_basis//2)\n",
    "        for sine_cos in [np.sin, np.cos]\n",
    "    ])\n",
    "\n",
    "for conv_coeff in conv_coeffs:\n",
    "    # Ground truth for the selected convection coefficient\n",
    "    u_true_test = analytical_sol(x_space_test, t_eval, conv_coeff).T\n",
    "\n",
    "    # Loop over different seeds\n",
    "    rmse_swim = np.ones((len(seeds), ))\n",
    "    rmse_elm = np.ones((len(seeds), ))\n",
    "    rel_err_swim = np.ones((len(seeds), ))\n",
    "    rel_err_elm = np.ones((len(seeds), ))\n",
    "    time_swim = np.ones((len(seeds), ))\n",
    "    time_elm = np.ones((len(seeds), ))\n",
    "    j = 0\n",
    "    info = []\n",
    "    for seed in seeds:\n",
    "        ansatz_swim = BoundaryCompliantAnsatz(\n",
    "                n_outer_basis=n_OBF,\n",
    "                n_inner_basis=n_IBF,\n",
    "                activation=\"tanh\",\n",
    "                random_state=seed,\n",
    "                regularization_scale=1e-10,\n",
    "                target_gen=periodic_outer_basis_sine,\n",
    "            )\n",
    "        ansatz_elm = BoundaryCompliantAnsatz(\n",
    "                n_outer_basis=n_OBF,\n",
    "                n_inner_basis=n_IBF,\n",
    "                activation=\"tanh\",\n",
    "                random_state=seed,\n",
    "                regularization_scale=1e-10,\n",
    "                target_gen=periodic_outer_basis_sine,\n",
    "                parameter_sampler = sample_parameters_randomly\n",
    "        )\n",
    "        \n",
    "        # Collocation points in space\n",
    "        x_space = np.linspace(x_lim[0], x_lim[1], n_c).reshape((-1, 1))\n",
    "\n",
    "        # interior points\n",
    "        x_space_inner = x_space[1:-1]\n",
    "        interior_points = x_space_inner\n",
    "\n",
    "        # Domain\n",
    "        domain = Domain(\n",
    "            interior_points=interior_points,\n",
    "            boundary_points=boundary_points,\n",
    "        )\n",
    "        \n",
    "        adv_solver_swim = AdvectionSolver(\n",
    "            domain=domain, \n",
    "            ansatz=ansatz_swim,\n",
    "            u0=u0,\n",
    "            boundary_condition=boundary_condition,\n",
    "            c=conv_coeff,\n",
    "            forcing=forcing,\n",
    "            regularization_scale=1e-10,\n",
    "        )\n",
    "        adv_solver_elm = AdvectionSolver(\n",
    "            domain=domain, \n",
    "            ansatz=ansatz_elm,\n",
    "            u0=u0,\n",
    "            boundary_condition=boundary_condition,\n",
    "            c=conv_coeff,\n",
    "            forcing=forcing,\n",
    "            regularization_scale=1e-10,\n",
    "        )\n",
    "        \n",
    "        # Compute weights and biases of the SWIM network\n",
    "        time_blocks = 1\n",
    "        t_swim_start = time.time()\n",
    "        sol_swim, solver_status_swim = adv_solver_swim.fit(t_span=[0, np.max(t_eval)], rtol = 1e-10, atol = 1e-10, svd_cutoff=1e-12);\n",
    "        t_swim_stop = time.time()\n",
    "        time_swim[j] = t_swim_stop - t_swim_start\n",
    "        \n",
    "        t_elm_start = time.time()\n",
    "        sol_elm, solver_status_elm = adv_solver_elm.fit(t_span=[0, np.max(t_eval)], rtol = 1e-10, atol = 1e-10, svd_cutoff=1e-12);\n",
    "        t_elm_stop = time.time()\n",
    "        time_elm[j] = t_elm_stop - t_elm_start\n",
    "        \n",
    "        # Evaluate on test data\n",
    "        u_elm_test = adv_solver_elm.evaluate(x_eval=x_space_test, t_eval = t_eval, solver_status=solver_status_elm).T\n",
    "        u_swim_test = adv_solver_swim.evaluate(x_eval=x_space_test, t_eval = t_eval, solver_status=solver_status_swim).T #, solver_status=solver_status\n",
    "                    \n",
    "        # Compute metrics\n",
    "        rmse_elm[j] = np.sqrt(mean_squared_error(u_true_test, u_elm_test))\n",
    "        rmse_swim[j] = np.sqrt(mean_squared_error(u_true_test, u_swim_test))  # mean squared error\n",
    "        rel_err_elm[j] = rmse_elm[j]/np.sqrt(mean_squared_error(u_true_test, np.zeros_like(u_true_test)))\n",
    "        rel_err_swim[j] = rmse_swim[j]/np.sqrt(mean_squared_error(u_true_test, np.zeros_like(u_true_test)))\n",
    "        j += 1\n",
    "\n",
    "    info.append(np.mean(time_elm))\n",
    "    info.append(np.mean(time_swim))\n",
    "    info.append(np.mean(rmse_elm))\n",
    "    info.append(np.std(rmse_elm))\n",
    "    info.append(np.mean(rmse_swim))\n",
    "    info.append(np.std(rmse_swim))\n",
    "    info.append(np.mean(rel_err_elm))\n",
    "    info.append(np.std(rel_err_elm))\n",
    "    info.append(np.mean(rel_err_swim))\n",
    "    info.append(np.std(rel_err_swim))\n",
    "\n",
    "    print(info)\n",
    "    experiments.append(info) \n",
    "\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Error Vs advection coefficient for Frozen-PINN-swim and Frozen-PINN-elm"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(11, 10)\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAYYAAAEiCAYAAAD9DXUdAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjguMywgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/H5lhTAAAACXBIWXMAAA9hAAAPYQGoP6dpAABLz0lEQVR4nO3deVxU9frA8c+AMCAoiKiAIphLaYW7pKjZtXJpsUXKfUMtI9O83sxMzaW8pWn2k9xStMwlNevaLbsuZKKGG7jhLigKbrGJwLDM+f0xMjqByhkGZoDn/XqNzpw5851njuN55pzv+T5fjaIoCkIIIcRtdtYOQAghhG2RxCCEEMKEJAYhhBAmJDEIIYQwIYlBCCGECUkMQgghTEhiEEIIYUISgxBCCBNVrB2ArejSpQt//vknVaoYNkmnTp349ddfi/VavV5PYmIi1apVQ6PRlGaYQgihmqIo3Lx5Ex8fH+zsHnw8IInhLl9//TUDBgxQ/brExER8fX1LISIhhLCchIQE6tWr98D1JDFYQLVq1QDDRq9evbqVoxFCCFPp6en4+voa91UPorGlWkkZGRnMnj2bqKgo9u3bR0pKCuHh4QwZMqTQujqdjilTpvDtt9+SkpJCQEAAM2fO5JlnnjHrvbt06cLx48cBaNGiBZ9//jkBAQHFem16ejpubm6kpaVJYhBC2By1+yib6ny+ceMG06dP58SJEzRv3vy+6w4ZMoS5c+fSv39/5s+fj729PT179iQyMtKs9/7ss8+Ii4vj4sWLPPPMM/To0YObN2+a1ZYQQpRrig3Jzs5WkpKSFEVRlP379yuAEh4eXmi9qKgoBVBmz55tXJaVlaU0bNhQad++vcm6QUFBClDkbdKkSfeM5eGHH1b+97//FSvutLQ0BVDS0tKKtb4QQpQltfsom+pj0Gq1eHl5PXC9DRs2YG9vz8iRI43LnJycCAkJ4YMPPiAhIcHYGWzuEYSdnR2K7ZxlE0KIMmNTiaG4oqOjadKkSaFzZe3atQMgJiZG1VVCqamp7N+/n86dO6PRaAgLCyM5OZnAwECLxp2fn09ubq5F2xSiNDg4OGBvb2/tMISVlMvEkJSUhLe3d6HlBcsSExNVtZebm8vEiRM5deoUDg4OtGjRgl9++QU3N7ci19fpdOh0OuPj9PT0+7avKApXrlwhNTVVVVxCWJO7uzteXl4yNqcSKpeJISsrC61WW2i5k5OT8Xk1atWqxYEDB4q9/qxZs5g2bVqx1y9ICrVr16Zq1aryH03YNEVRyMzM5Nq1awBF/ggTVqIoELMaAl4De4dSe5tymRicnZ1NfrEXyM7ONj5fmiZOnMi4ceOMjwuuES5Kfn6+MSnUrFmzVOMSwlIK/g9du3aN2rVry2klW7Hrc9gxA45thAEboZR+ZJbLxODt7c3ly5cLLU9KSgLAx8enVN9fq9Wi1WoJCwsjLCyM/Pz8e65b0KdQtWrVUo1JCEsr+M7m5uZKYrAFxzcZkgLAI8+VWlIAGxvHUFwtWrTg9OnThc7tR0VFGZ8vC6GhocTGxrJ///4Hriunj0R5I99ZG3L5IGx603D/ibegbUipvl25TAy9e/cmPz+fJUuWGJfpdDrCw8MJDAyUukVCiIojNQHW9IW8bGjcDZ6dWepvaXOnkhYsWEBqaqrxyqLNmzdz6dIlAEaPHo2bmxuBgYEEBwczceJErl27RqNGjVi5ciXx8fEsW7aszGItzqkkIczVpUsXWrRowRdffGHtUIS16G7Cmj6QcRVqPwq9l4FdGZzWK83Rdubw8/O750jluLg443pZWVnK+PHjFS8vL0Wr1Spt27ZVtmzZYpWY7zeqMCsrS4mNjVWysrKsEFnJDB48uMh/hzNnzlg7tGKLi4szid3Dw0N55plnlEOHDhnXefLJJ5UxY8aYPAaUNWvWmLQ1b948xc/Pz/g4PDxcAZRu3bqZrJeSkqIASkRERIli/3tcZa08f3crhPw8RfnuNUWZWl1RPmukKCkXzW5K7chnmzuVFB8fj6IoRd78/f2N6zk5OTF79mySkpLIzs5m3759dOvWrUxjDQsLo1mzZrRt27ZM37csde/enaSkJJNbgwYNTNbJycmxUnTFt23bNpKSkvjtt9/IyMigR48e9x1X4uTkxIcffvjAAYlVqlRh27ZtREREWDhiUeltnQKnt0AVJ+i7BtzL7hS5zSWG8kRN53N5VVCm5O5b165defvttxk7diyenp7GhLxz507atWuHVqvF29ub999/n7y8PMCQ8DUaTaFbly5djO8VGRlJp06dcHZ2xtfXl3feeYdbt24Zn/f39+eTTz5h2LBhVKtWjfr165v0M91PzZo18fLyok2bNsyZM4erV68aL1YoSt++fUlNTWXp0qX3bdfFxYVhw4bx/vvvFyuOux07dowePXrg6upKnTp1GDhwIDdu3Ljn+v7+/sycOZNBgwbh6uqKn58f//nPf7h+/Tq9evXC1dWVgIAAVWNyhI06EA57Fxjuv7QQ6rUp07eXxGAFiqKQmZNnlZtiofpPK1euxNHRkd27d7No0SIuX75Mz549adu2LYcPH2bhwoUsW7aMmTMNHWW+vr4mRx3R0dHUrFmTzp07A3Du3Dm6d+/Oq6++ypEjR1i3bh2RkZG8/fbbJu/7+eef06ZNG6Kjo3nrrbcYNWoUp06dUhV7wTX69zvSqV69OpMmTWL69OkmyakoH330EUePHmXDhg3FjiE1NZV//OMftGzZkgMHDrBlyxauXr3Ka6+9dt/XzZs3j6CgIKKjo3nuuecYOHAggwYNYsCAARw6dIiGDRsyaNAgqfNVnp2LgP/+03D/qQ/hsVfKPASb63wuT8ztfM7KzafZlN9KKar7i53ejaqOxf9n//nnn3F1dTU+7tGjBwCNGzfms88+My6fNGkSvr6+LFiwAI1GwyOPPEJiYiITJkxgypQp2NvbGwskZmdn89JLL9G+fXs++ugjwDCavH///owdO9bY/pdffsmTTz7JwoULjaPae/bsyVtvvQXAhAkTmDdvHhERETz88MPF+jypqanMmDEDV1dXY22te3nrrbeYP38+c+fOZfLkyfdcz8fHhzFjxjBp0iReeumlYsWxYMECWrZsySeffGJctnz5cnx9fTl9+jRNmjQp8nU9e/bkjTfeAGDKlCksXLiQtm3bEhwcDBi2Sfv27bl69WqxClIKG3P9NHw/GJR8CHgdOo+3ShhyxFACleFU0lNPPUVMTIzx9uWXXwLQunVrk/VOnDhB+/btTa59DwoKIiMjw3hVWYFhw4Zx8+ZNVq9ebZx/9vDhw6xYsQJXV1fjrVu3buj1euLi4oyvvXvyJI1Gg5eXl7F0Q8FpGVdXVx599FGT9+zQoQOurq7UqFGDw4cPs27dOurUqXPfz67Vapk+fTpz5sy57ykeMOyQr1+/zvLlyws9V1Rchw8fJiIiwuTzPvLII4Dh6Ole7v78BfE//vjjhZYVbBNRjtz6C1YHgy4NfJ+AF/+vVAex3Y8cMViBs4M9sdPLtqP87vdWw8XFhUaNGhW53BwzZ87kt99+Y9++fSbTDGZkZPDGG2/wzjvvFHpN/fr1jfcdHEzrw2g0GvR6PWCYs7ugTtbf11u3bh3NmjWjZs2auLu7FzveAQMGMGfOHGbOnGly8cPfubu7M3HiRKZNm8bzzz9v8lxRcWVkZPDCCy/w6aefFmrrfrWJ7v5cBUm4qGUF20SUE3k6WDcAUuLB3Q/6fAdVCteDKyuSGKxAo9GoOp1THjRt2pSNGzeiKIpx57R7926qVatmnHx848aNTJ8+nV9//ZWGDRuavL5Vq1bExsYWmYSKq27duvd8ztfXt9B7FoednR2zZs3ilVdeYdSoUfddd/To0Xz55ZfMnz//gXG1atWKjRs34u/vT5UqFeu7IFRSFNg8Bi7uAW116Pc9uHhaNSQ5lVQCleFy1eJ66623SEhIYPTo0Zw8eZKffvqJqVOnMm7cOOzs7Dh27BiDBg1iwoQJPProo1y5coUrV66QnJwMGE7F7Nmzh7fffpuYmBjOnDnDTz/9VKjz2Rqee+45AgMDWbx48X3Xc3JyYtq0acbTbfcTGhpKcnIyffv2Zf/+/Zw7d47ffvuNoUOHyoDJyiZyLhxeAxp7CA6H2o9YOyJJDCVRGfoYiqtu3br88ssv7Nu3j+bNm/Pmm28SEhLChx9+CMCBAwfIzMxk5syZeHt7G2+vvGK44iIgIICdO3dy+vRpOnXqRMuWLZkyZUqpF0Qsrk8//dRYvfd+Bg8ezEMPPfTA9Xx8fNi9ezf5+fk8++yzPP7444wdOxZ3d3djv4uoBI7/CNunG+73+BQaPW3VcApoFLmurcTS09Nxc3MjLS2t0Kxy2dnZxMXF0aBBA+OVNUKUB/LdLWWXD0L4c5CXBe3egJ6fPfg1ZrrfPqoo8tNECCHKWtql24XxsqDRM9Dtkwe/pgxJYigB6WMQQqimy4DVBYXxmkHv5WBvWxcgSGIoAeljEEKoos+HjcPh6lFwqQX91oHTg0/tlDVJDEIIUVa2ToHTv4K9FvqsAff6D37N36Rl3r+woyVIYhBCiLJwcMVdhfG+Al/1p6APJ6TS8dMdfL8/wbKx/Y0kBiGEKG3nf79TGK/LB/B4b9VNJKZmMfybA9zU5fHrsaRSLZQoiUEIIUrT9dPw/SDQ58HjwfDke6qbuKXLI2TlAa7f1PGIVzX+r1+rUp2TWxJDCchVSUKI+8pMhtWvQXYa1GsHLy5QXRgvX68wZm00J5LS8XTV8vXgNrhqS/cqJkkMJSBXJQlb0KVLF2O5cmFDjIXx4gydzH1Wg4P6gYL//vUE205cw7GKHUsHtaZejaqlEKwpSQzinoYMGVLkrGtnz561dmjF9veZ42rWrMmzzz5LdHS0cZ2/71i7dOmCRqNh7dq1Jm198cUXJhVWV6xYgUajoXv37ibrpaamotFo+P3330vjIxXyww8/MGPGjDJ5L1FMigKbx8KF3XcK47nWUt3Mmn0XWbrLUHb+8+DmtKxfw8KBFk0Sg7gvmfPZ9ud89vDwMClhLmxA5Dw4vBo0drcL4zVV3cTuszeY/OMxAMY904QXmpdd3TBJDOK+ZM7n0pnzOSUlhf79+1OrVi2cnZ1p3Lgx4eHhAPTu3dukquzYsWPRaDScPHkSMCRiFxcXtm3bBhQ+4pG5oa0s9j+wfZrhfo/PzCqMd/ZaBqNWHSRPr/BSCx9G/8P8cvTmkMQgzCJzPt9hzpzPkydPJjY2ll9//ZUTJ06wcOFCPD0NNfiffPJJk9NQO3fuxNPT07hs//795Obm0qFDh3u2L3NDW8nlQ/DDSMP9dm9AuxGqm0i5lUPIyv2kZ+fR2q8G/341oFSvQCqSIkosLS1NAZS0tLRCz2VlZSmxsbFKVlbWnYV6vaLoMqxz0+uL/bkGDx6s2NvbKy4uLsZb7969lSeffFJp2bKlyboffPCB8vDDDyv6u9oPCwtTXF1dlfz8/ELbJDAwUHn++eeNz4WEhCgjR440WW/Xrl2KnZ2dcdv5+fkpAwYMuGsz6pXatWsrCxcuvOdniIuLUwAlOjpaURRFSUlJUV5++WXF1dVVuXLliqIoivLkk08qY8aMMb6m4HF2drbi5+enTJ8+XVEURZk3b57i5+dnXC88PFxxc3NTFEVR3n//faVJkyZKbm6ukpKSogBKRETEPeN64YUXlKFDhxb53JEjRxSNRqNcu3ZNSU5OVhwdHZUZM2Yor7/+uqIoijJz5kylQ4cOheIt8PftlJSUpADK5MmTjcv27t2rAEpSUtI9YyzyuyvuLfWSosxuoihTqyvKt68qSl6u6iayc/OU4EV7FL8JPytB/96uXL+ZbZHQ7rePKoptVW6qLHIz4RMrzTPwQSI4Fn9azqeeeoqFCxcaH7u4uNC3b1/Vcz7fPT1nwZzPW7duNZnz+ciRI3z33XfG9RRFMc753LSp4Rztg+Z83rVrFwB+fn4cP37cuG6HDh2ws7Pj1q1bPPTQQ6rmfB49evQDZ2+bMGECixcvZvny5bz22msmzxUV16hRo3j11Vc5dOgQzz77LC+99JLxCOCxxx7Dw8ODnTt34ujoSMuWLXn++ecJCwsDDEcQd5+CK4qauaG9vLzu25YoBl0GrHkdMq5AraZmFcZTFIUPfjjGvrhkqmmrsHxIWzxdrTO9pySGEggLCyMsLKxCz7glcz6XzpzPPXr04MKFC/zyyy9s3bqVrl27Ehoaypw5c9BoNHTu3Jnff/8drVZLly5dCAgIQKfTcezYMfbs2cP48ePvG7fMDV2G9Pnwwwi4UrLCeAt3nmPjoUvY22lY0L8VTepY74ICSQwlEBoaSmhoqHESjGJzqGr45W4NDqVzDbTM+axuzmeAWrVqMXjwYAYPHkynTp3417/+xZw5cwBDP8PSpUvRarV8/PHH2NnZ0blzZ2bPno1OpyMoKEj1ZxGlZNtUOPXL7cJ4q6GGn+omthxL4rMthr6yj15oxpNN1F/aaknS+WwNGo3hdI41bqXUiSVzPqub83nKlCn89NNPnD17luPHj/Pzzz8bT5eB4Uqj2NhYjh8/TseOHY3LvvvuO9q0aWP2EZuwsIMrYc//Ge6/9BX4tlPdxJFLqYxdFwPAkA7+DGzvb7n4zCSJQViEzPlsUNw5nx0dHZk4cSIBAQF07twZe3t7kwF1jz/+OO7u7rRo0QJXV1fAkBjy8/Mf2L8gysj5nfDfcYb7XSaaVRgvKS2L4SsPkJ2rp8vDtfjwOfXjHUqDzPlsATLns6iI5Lt7HzfOwNddDTWQHusNr36t+mj8li6P4EV7iU1Kp0kdVzaO6kA1J4cHv9AMMudzCXz22Wf4+vpSrVo1WrZsyc2bN60dkhDC1vy9MF6vMDML48UQm5SOp6sjywa3LbWkYA7pfL4tLCyMLVu2sHv3bnx9fTl69CiOjo7WDksIYUvycmDdQEg+D271oc93ZhXG+3TLSbaduIpjFTsWD2yDr0fpF8ZTQxIDkJ+fz8cff8yuXbuMl0befR24EEKgKPDzu3AhEhyrGS5Lda2tupm1+y6y5I/zAMwJbk5rv7IpjKeGTZ1KysjIYOrUqXTv3h0PDw80Gg0rVqwocl2dTseECRPw8fHB2dmZwMBAtm7datb7Xrp0iczMTDZs2ECdOnV4+OGHH1gjRwhRyez+AmJW3S6MtwLqNFPdxJ6zN/jwdmG8sU835sUyLIynhk0lhhs3bjB9+nROnDhB8+bN77vukCFDmDt3Lv3792f+/PnY29vTs2dPIiMjVb/v5cuXSUtL4/Tp08THx7N+/Xo++OAD42hVIUQlF/sf2PaR4X73T6Gx+sJ4565n8Obtwni9Wvgwpmtjy8ZoQaoTQ25uLl27duXMmTMWD8bb25ukpCQuXLjA7Nmz77nevn37WLt2LbNmzWL27NmMHDmSHTt24Ofnx3vvmU6b17FjxyKremo0GuOllAVF1aZMmYKzszMBAQH06dOHX375xWKfTS7+EuWNfGdvS4y+Uxiv7QgIHKm6iZRbOYSsMBTGa1XfnU+tURhPBdV9DA4ODhw5cqQ0YjGWeH6QDRs2YG9vz8iRd/6BnJycCAkJ4YMPPiAhIQFfX1+AYh1BNGnSBEdHR5N/KEv9oxWUIcjMzDQmICHKg8zMTKBweZFKJT0R1vSFvCxo2BW6/1t1Ezl5et5cdZD4vzKpV8OZJYPa4ORgXwrBWo5Znc8DBgxg2bJl/Pvf6jeSJURHR9OkSZNC1+O2a2cYdRgTE2NMDMXh4uJC7969+fjjj/nyyy85f/4869atU1VG+V7s7e1xd3c3FnqrWrWqTf9SEEJRFDIzM7l27Rru7u7Y29v2TqzU6DJg9etwMwlqPWKYcMeMwniTNh0lKi4ZVysXxlPDrMSQl5fH8uXL2bZtG61bty40PH/u3LkWCe5ekpKS8Pb2LrS8YFliovo6RGFhYYSEhODp6YmnpyczZsygU6dORa6r0+nQ6XTGx+np6fdtu+AoqCA5CFEeuLu7V97Kq3q94fTRlSNQ1fN2YTwV9dBuW7TzPOsPXsJOAwv6tbRqYTw1zEoMx44do1WrVgCcPn3a5Lmy+DWclZWFVls46xaMziyoZKmGu7s7GzduLNa6s2bNYtq0acVuW6PR4O3tTe3atR84VaQQtsDBwaHyHinA7cJ4/72rMJ6/6ia2HEvi0y2GWfc+evFRujys/tJWazErMVhzflswdBbf/Yu9QEEtm9I+lz9x4kTGjRtnfJyenl6sU1f29vaV+z+bEOXBwZWw53YhxF5hUD9QdRNHL6UZC+MNbu/HIBsojKdGuRzg5u3tzeXLlwstT0pKAij1wmtarRatVlsp5mMQolKJ++NOYbwnJ0BAsOomktKyCFm5n+xcPU82qcXk59WPd7A2sxNDamoqy5Yt48SJEwA0a9aMkJAQdfMSmKlFixZERESQnp5u0gFdMLl7ixYtSj0GKMF8DEII23PjrKHchT4PHnvVUDFVpVu6PEJWHODaTR1N6rjyf/1aUsXepoaLFYtZER84cICGDRsyb948kpOTSU5OZt68eTRs2JBDhw5ZOsZCevfuTX5+PkuWLDEu0+l0hIeHExgYqOqKpJIICwujWbNmtG3btkzeTwhRSjKTYXUwZKdCvbbQ6yuzCuONXWdaGK+6DRXGU8OsstudOnWiUaNGLF26lCpVDAcdeXl5DB8+nPPnz/PHH3+YHdCCBQtITU0lMTGRhQsX8sorr9CyZUvAMENWwS/z1157jU2bNvHuu+/SqFEjVq5cyb59+9i+fTudO3c2+/3NobakrRDChuTlwLcvG2ogudWHEdvNqoH0yS8nWPLHeRyr2LFmxBM2VQNJ7T7KrMTg7OxMdHQ0jzzyiMny2NhY2rRpYxwYYw5/f38uXLhQ5HNxcXHGeXezs7OZPHkyq1atIiUlhYCAAGbMmEG3bt3Mfm9zSWIQopxSFPjpbUMNJMdqEPI/s2ogrd13kfd/OArA/D4t6NXi3tPMWoPafZRZfQzVq1fn4sWLhRJDQkKCyQTv5oiPjy/Wek5OTsyePfu+pTNKm3Q+C1HOlUJhPFtLCuYwq4/h9ddfJyQkhHXr1pGQkEBCQgJr165l+PDh9O3b19Ix2qzQ0FBiY2PZv3+/tUMRQqhlgcJ45+8qjPdic9sujKeGWUcMc+bMQaPRMGjQIPLy8gDDgJhRo0ZZrUyGEEIUm4UK4w27qzDeZ71tuzCeGqoTQ25uLj169GDRokXMmjWLc+fOAdCwYUOqVrWtWYhKm5xKEqIcSrsMq/sYCuM1errEhfHqujuzeKDtF8ZTw6zO51q1arFnzx4aN64Yh00lJZ3PQpQTugwI7w5XjkKtphDym+oaSIqi8N6GI6w/eAlXbRU2jurAw162XQNJ7T7KrD6GguqqQghRbujzbxfGO3q7MN5aixTGs/WkYI5yWV3VVsipJCHKkbsL4/VdU+LCeFNfKF+F8dQw61TSU089de8GNRp27NhRoqDKGzmVJISNO7gSNr9juP/qMni8t+omjl5KI3jxHrJz9Qxq78f0Xo9ZOMjSUybjGKxdXVUIIYrt/O93FcZ736ykcCUtm+Hf3CmMN6UcFsZTw6bmfBZCCIu6fhq+H3S7MF5v6PK+6iZu6fIIWbmfq+nluzCeGqo/XWnO+SyEEBZz6y9Y/Rpkp0G9doa5FVSOM9DfLox3PDGdmi7luzCeGnJVUglIdVUhbFSeDtYNgJQ4cK9vmIXNwUl1M59uOcnW2Ks4VrFjyaDW+HpUjrFaclVSCch8DELYIEWBzWPg4h7QVod+34NrLdXNrN13kcV/nAdgdu8AWvt5WDpSm1Uu53wWQoh72vU5HF4DGnsIDofaTVU3sefcncJ4Y7pWjMJ4ashVSUKIiuP4Jtgxw3C/x6eGkhcqnb+ewahVh8jTK7zQ3IexT1e+Cg8Vu2tdCFF5XDoIm9403A8cBe1GqG6ioDBeWlYuLeu7M7sCFcZTw+zEsGvXLgYMGED79u25fPkyAN9++y2RkZEWC04IIYolNQHW9IG8bGjcDbp9rLqJvxfGW1LBCuOpYVZi2LhxI926dTPO5KbT6QBIS0vjk08+sWiAtkyuShLCBuhuwurX4dY1qPMY9F4Gdup26IqiMGnTUaLiknHVVmH5kLbUqqYtpYBtn1mJYebMmSxatIilS5fi4HDnmt6goCAOHTpkseBsnUzUI4SV6fNhQwhcOw4utaHvWtCqL2pXGQrjqWFWYjh16hSdO3cutNzNzY3U1NSSxiSEEMXz2yQ48xtUcTIkBXdf1U1UlsJ4apiVGLy8vDh79myh5ZGRkTz00EMlDkoIIR5o/9cQtdBw/+XFUK+16iaOXkpj7LoYAAa392NwB3/LxVeOmZUYRowYwZgxY4iKikKj0ZCYmMh3333H+PHjGTVqlKVjFEIIU2e3wS/vGe7/YzI8+pLqJpLSsghZeacw3uQKXhhPDbPGMbz//vvo9Xq6du1KZmYmnTt3RqvVMn78eEaPHm3pGIUQ4o5rJ2D9UFDyoXk/6PRP1U3c0uURsuIA125WnsJ4apg1H0OBnJwczp49S0ZGBs2aNcPV1dWSsZUbMh+DEGUk4zp8/Q9IvQj1O8CgH6GKuquH8vUKb3x7kG0nruLp6simt4IqfA2kMpmPoYCjoyPNmsnhlxCiDORmw9p+hqRQowG8vkp1UgBDYbxtJwyF8RYPbFPhk4I55NipBGQcgxBlRFHgp1C4tM8wT3P/9eBSU3Uza/ddZMntwnhzgpvT2q+GpSOtEEp0KkkYyKkkIUpZxCzY+W+wqwIDfoCHnlTdxO6zNxi8fB95eoV3n27CmEpUA0ntPkqOGIQQtu3IekNSAHhurllJ4dz1DEatOkieXqFXCx/e6drIwkFWLJIYhBC262IU/PSW4X6H0dB6sOomkm8XxkvPzqO1Xw0+fbVyFsZTo8SJISsry1hE727Hjx8vadNlytXV1eRmZ2fH559/bu2whKi8UuINnc35OfDwc/D0NNVN6PLyefPbg1z4KxNfD2cWD2xdaQvjqVGixLBhwwYaN27Mc889R0BAAFFRUcbnBg4cWOLgylJGRobxdvr0aezs7HjllVesHZYQlVN2mqEwXuYN8AqAV5eaVRhv4saj7ItPppq2CssHt8XTtfIWxlOjRIlh5syZHDx4kJiYGMLDwwkJCWH16tWA4R+lvFq9ejXt27enQYMG1g5FiMonPw/WD4HrJ6GaN/RbB44uD3zZ34VFnOWH6MvY22n4akArGtep3IXx1ChRYsjNzaVOnToAtG7dmj/++IPFixczffp0s87hZWRkMHXqVLp3746HhwcajYYVK1YUua5Op2PChAn4+Pjg7OxMYGAgW7duLcnHMfr2228ZNGiQRdoSQqigKPDre3BuBzhUNRTGq+6jupnNhxOZ8z/DtMMzej1Gp8bq53yuzEqUGGrXrs2RI0eMjz08PNi6dSsnTpwwWV5cN27cYPr06Zw4cYLmzZvfd90hQ4Ywd+5c+vfvz/z587G3t6dnz54lnijoyJEjnD59muDg4BK1I4QwQ9QiOLAM0MCrX4NPC9VNHLqYwj/XHwZgeMcG9Ausb9kYKwOlBBISEpSkpKQin4uMjFTdXnZ2trG9/fv3K4ASHh5eaL2oqCgFUGbPnm1clpWVpTRs2FBp3769ybpBQUEKUORt0qRJhdoeP368EhwcrCrutLQ0BVDS0tJUvU4IcZdTWxTlI3dFmVpdUSLnm9XExb9uKa1n/E/xm/CzErJiv5KXr7dwkOWT2n1UiUpi1KtXr9Cy+Ph4/P39CQoKUt2eVqvFy8vrgett2LABe3t7Ro4caVzm5ORESEgIH3zwAQkJCfj6GuqyqzmC0Ov1rF69mkWLFqmOXQhRAleOwYZhoOih1SDDpakqpWfnErJyPzcycmjmXZ35fVpgbyeXpZrD4uMY3nvvPUs3WUh0dDRNmjQpNIKvXbt2AMTExJjV7vbt28nNzaVHjx4lDVEIUVw3rxiuQMrJgAadDYPYVPZR5uXrCf3uEKevZlCnupZlQ9rgoi3R795KzeJbTimDq5GSkpLw9vYutLxgWWJiolntfvvtt/Tp04cqVe6/WXQ6nXGeazAMNxdCmCEnE9b0hfRLULMxvPYN2Ds8+HV3URSFjzYfZ9eZGzg72LNscFu83ZxLKeDKQXViGDZsGF26dOGZZ54pcudcFiMKs7Ky0GoLX4/s5ORkfN4c33zzTbHWmzVrFtOmqR9sI4S4i14PP74JiYfAuYbhslRn9UXtwnfHs+rPi2g0ML9PCx6r61YKwVYuqhPDzZs36d69O1u2bOHSpUvY2dnRunVr42Q9ZcHZ2dnkF3uB7Oxs4/OlaeLEiYwbN874OD093dinIYQoph0zIPYnsHOA17+Dmg1VN7H9xFVm/DcWgA96NOXZRx/cRykeTHViCAwM5PPPP8fFxYUuXboQGBhIdHQ08+fPJyMjo0xKYXh7exdZhiMpKQkAHx/11z2rodVq0Wq1hIWFERYWRn5+fqm+nxAVTvQqiJxruP/il+Cv/mKV2MR0Rq+JRlGgb7v6DO8kA1ItRXViGD9+PGA4z75r1y7mzJmDXq/Hx8eHPn36cOLECYsH+XctWrQgIiKC9PR0kw7ogpIcLVq0KPUYAEJDQwkNDTWWtBVCFMP5nbB5jOF+539Bi36qm7iank3Iyv1k5uTTsZEn03s9KoXxLMjsq5K0Wi1PP/00kyZNYvLkyfTq1Ys9e/bg6elpyfiK1Lt3b/Lz81myZIlxmU6nIzw8nMDAQDmtI4Stun4avh8I+jx47FV4apLqJjJz8hi+8gBJadk0rOVCWP9WOMh8zRZlsauSPD096dOnD3369ClROwsWLCA1NdV4ZdHmzZu5dOkSAKNHj8bNzY3AwECCg4OZOHEi165do1GjRqxcuZL4+HiWLVtW4s9SXHIqSQgVbt2A73obCuT5BkKvr1RflqrXK7y7Loajl9PwcHEkfEg73JzVXcUkHszmZnDz9/fnwoULRT4XFxeHv78/YOhonjx5MqtWrSIlJYWAgABmzJhBt27dyjBaA5nBTYgHyM2Gb16EhCio4Q/Dt4OL+rMLs349weKd53G0t2P1iEDa+HtYPtYKSO0+yuYSQ3ly9xHD6dOnJTEIURS9Hn4YDsc2GuZrDtkGtZqobmbd/otM2HgUMFyW2qtFXUtHWmFJYrACOWIQ4j52zIQ/ZpdovuY9Z28w6PZ8zWOfbszYp9UnlspM5nwWQtiOmNWGpADwwnyzksLZaxm8edd8zWO6NrZwkOLvJDGUQFhYGM2aNaNt27bWDkUI2xO3C/7zjuF+x3HQcoDqJpJv5RCyUuZrLmtyKskC5FSSEH9z4wx8/TRkp0Kzl6B3ONip+x2qy8tnwNdR7I9PwdfDmR/fCqKmTM1pFjmVJISwrlt/wXfBhqRQry28vEh1UlAUhfc3HmV/fArVnKoQPqStJIUyJIlBCGE5eTpY1x9S4sC9PvRZAw7qa5ct2HGWTbfna17YvzWNast8zWXJ7MSwa9cuBgwYQPv27Y11i7799tsST61ZnkgfgxB3URT4KRQu7gWtG/RbD67q51r+z+FEPt9qmK955kuP0bFx6VdTEKbMSgwbN26kW7duODs7Ex0dbax0mpaWxieffGLRAG1ZaGgosbGx7N+/39qhCGF9v/8bjq43XJb62kqo/YjqJg5eSGH87fmaR3RqQN92Ml+zNZiVGGbOnMmiRYtYunQpDg53hqMHBQVx6NAhiwUnhCgnDq+Dnf823H9uLjR8SnUTCcmZjPzmADl5ep5pVof3ezS1cJCiuMxKDKdOnaJz586Flru5uZGamlrSmIQQ5Un8bvjP24b7QWOg9WDVTaRn5zJsxX7+upXDoz4yX7O1mZUYvLy8OHv2bKHlkZGRPPTQQyUOqryQPgZR6f11ztDZnJ8DTV+Erh+pbqJgvuYz127P1zy4LVUdZb5mazIrMYwYMYIxY8YQFRWFRqMhMTGR7777jvHjxzNq1ChLx2izpI9BVGoZ1w3VUrNSoG5reHmxWZelTv2P6XzNXm5OpRSwKC6z0vL777+PXq+na9euZGZmGqf1HD9+PKNHj7Z0jEIIW6PLgNXBkHz+zmWpjlVVN7N8dzzfRRnma/6yb0uZr9lGlGjkc05ODmfPniUjI4NmzZrh6upqydjKDRn5LCqVvBxY8zqc2wFVa8Kw/4FnI9XNbIu9yohvD6Ao8OFzTRneqfKchi5ravdRJTqR5+joSLNmzUrShBCiPNHrDR3N53aAQ1Xo971ZSeHIpVTjfM39AusT0lHma7YlZvUxDB8+nN9//93CoQghbN72j+DIOtDYw2vfQL02qptISM5k2IoDZOXm82STWkx/UeZrtjVmJYbr16/TvXt3fH19+de//sXhw4ctHVe5IFcliUpl71ewe77hfq8F0PgZ1U2kZeUydMV+bmToaOpdnbD+ragi8zXbHLP7GFJSUli/fj2rV69m165dPPLII/Tv359+/foZp9+sLKSPQVR4RzfAxhDD/a5TodM41U3k5OkZvHwfe8//hVd1J34MDZIrkMqIVWZwu3TpEmvWrGH58uWcOXOGvLy8kjZZrkhiEBXa+d9hVW/Q50K7N6DHp6Dy1I+iKPzz+8P8EH0ZV20V1r/Znqbe8n+lrJR52e3c3FwOHDhAVFQU8fHx1KlTp6RNCiFsRdJhWDvAkBSavQTdZ6lOCgDztp3hh9vVUr/q30qSgo0zOzFEREQwYsQI6tSpw5AhQ6hevTo///wzly5dsmR8QghrSYk3HCnk3AT/TrcHsNmrbub7Awl8uf0MAJ+8/Bidm6ivuCrKllmXq9atW5fk5GS6d+/OkiVLeOGFF9BqZRINISqMWzfg21fg1jWo8xj0+Q4c1PcHRJ65wQc/HAXg7aca8XpbqZZaHpiVGD766COCg4Nxd3e3cDhCCKvTZRhmYEs+B271of8GcFI/IvnklXRGrTpInl6hVwsf/vlsk1IIVpQGsxLDiBEjLB2HEMIW5OfC+iGQeAicPWDgD1DdW3UzV9OzGRa+n5u6PNo18OCz3gEyVqEcKXZiGDduHDNmzMDFxYVx4+5/qdrcuXNLHFh5EBYWRlhYGPn5+dYORYiSUxT4zztwditUcb49qrmx6mZu6fIYtmI/iWnZPFTLhSUDW6Otor5vQlhPsRNDdHQ0ubm5xvv3Upl+FYSGhhIaGmq8FEyIcm37NDi82jCqOXgF+KofuJmXr+ft1Yc4npiOp6sjK4e2w72qo+VjFaWq2IkhIiLCeH/lypXUq1cPu7+V2FUUhYSEBMtFJ4QoG38ugsh5hvsvzIeHu6tuoqCEdsSp6zg52PH14Lb4eqivuCqsz6zLVRs0aMCNGzcKLU9OTqZBAymGJUS5cuwH2PK+4f4/PoRWA81qZskf540ltOf3aUkLX3fLxSjKlFmJ4V6DpTMyMnBykiHuQpQbcX/ApjcABdqOgE7jzWrm5yOJzPr1JACTn2tGt0e9LBikKGuqrkoq6HTWaDRMmTKFqlXvHCbm5+cTFRVFixYtLBpgWYmJiSE0NJSjR4/i6enJBx98wPDhw60dlhCl58pRWHvXtJxmlLoAOBCfzLjvDYU0hwb5M0xKaJd7qhJDQaezoigcPXoUR8c7nUqOjo40b96c8ePN+8VhbQMHDiQ4OJhdu3YRExPDk08+SVBQEE2bNrV2aEJYXsoFWPUq6NLBLwheWWrWqOa4G7cY/s0BcvL0PNusDh8+J/OzVASqEkNBB/TQoUOZP39+hSoYFx8fT9++fbGzs6NVq1Y0bdqUkydPSmIQFc+tv2DVK5BxFWo3gz6rzRrVfCklk4HLokjNzKV5PTfm92mJvV3luSqxIjOrjyE8PJzq1asTGxvLli1b+M9//mNyM1dGRgZTp06le/fueHh4oNFoWLFiRZHr6nQ6JkyYgI+PD87OzgQGBrJ161az33v06NGsWrWKvLw89u3bx8WLF3niiSfMbk8Im5RzC1a/Bn+dher1YMBGcHZX3czl1Cz6Lv2TSylZ+NesyteD2+LsKGMVKgqzRj7HxcXx0ksvcfToUTQajbEzumAMg7kDvm7cuMH06dOpX78+zZs3v+8scUOGDGHDhg2MHTuWxo0bs2LFCnr27ElERAQdO3ZU/d49evRg0KBBfPzxxwAsW7YMb2/1Iz6FsFkFo5ovHwDnGrdHNfuobiYxNYu+S/4kIdmQFNaObE+talIrrUJRzPD8888rvXr1Uq5fv664uroqsbGxyq5du5R27dopf/zxhzlNKoqiKNnZ2UpSUpKiKIqyf/9+BVDCw8MLrRcVFaUAyuzZs43LsrKylIYNGyrt27c3WTcoKEgBirxNmjRJURRF+euvv5Rq1aop69evV/Ly8pTDhw8r3t7eysGDB4sVd1pamgIoaWlpZn5yIUqZXq8om95SlKnVFWVGHUW5GGVWM4mpmUrnz3YofhN+Vjp/tkNJTM20cKCiNKjdR5l1Kmnv3r1Mnz4dT09P7OzssLOzo2PHjsyaNYt33nnH7CSl1Wrx8nrwZW4bNmzA3t6ekSNHGpc5OTkREhLC3r17TQbZRUZGoihKkbeZM2cCcO7cOVxcXOjduzf29vYEBATQoUMHdu7cafZnEcKm7JgJMatAYwfB4eDbTnUTV9Ky6bvkTy78lUl9j6qsGfEE3m7OpRCssDazEkN+fj7VqlUDwNPTk8TERAD8/Pw4deqU5aK7h+joaJo0aVKo87tdO8OXPSYmRlV7TZo0ITMzk59++glFUYiNjWXXrl08/vjjlgpZCOuJWgy75hjuP/8FPNxDdRNX0rLpu/RP4v/KxNfDmTUjn8DHXZJCRWVWH8Njjz3G4cOHadCgAYGBgXz22Wc4OjqyZMkSHnroIUvHWEhSUlKR5/8LlhUkquJyc3Pj+++/Z8KECQwYMAAPDw/GjRvH008/XeT6Op0OnU5nfJyenq7q/YQoE9lp8Mt7cGSt4fFTk6D1YNXNXE3Ppt/SP4m7cYt6NZxZM+IJ6kpSqNDMSgwffvght27dAmD69Ok8//zzdOrUiZo1a7Ju3TqLBliUrKysIicGKhh1nZWVpbrNbt260a1bt2KtO2vWLKZNm6b6PYQoM3F/wKZRkH7JcPqo83vQ+V+qm7mWbjhSOH/jFnXdDUmhXg2pf1TRmZUY7t6BNmrUiJMnT5KcnEyNGjXKpLqqs7OzyS/2AtnZ2cbnS9PEiRNNSo+np6fj6+tbqu8pRLHkZsP26fBnmOFxjQaGKTnrB6pu6trN20nhuiEprB35hBTFqyTMSgxF8fDwsFRTD+Tt7c3ly5cLLU9KSgLAx0f9JXhqaLVatFqtzMcgbEvSYfjhDbh+wvC49RB49mPQuqpu6vpNHf2WRnHu+i183JxYM0KSQmWiaqKe4irtiXpatGhBREQE6enpJh3QUVFRxufLgszHIGyCPh92fwERs0CfCy614cX/M6t0NhQkhT85ey0Dbzcn1ox8gvo1JSlUJqom6imOsjiV1Lt3b+bMmcOSJUuMtZl0Oh3h4eEEBgaW2WkdOWIQVpccB5vehIQ/DY8fed4wn4KLp1nN3cjQ0f/rPzlzLQOv6oYjBb+aLhYMWJQHGkW5Rw1tK1mwYAGpqakkJiaycOFCXnnlFVq2bAkYylYU/DJ/7bXX2LRpE++++y6NGjVi5cqV7Nu3j+3bt9O5c+cyjbngiCEtLa1C1Y8SNkxR4NA3sGUi5N4Cx2rQ8zNo3tesCqkAf2UYTh+dunqTOtW1rBvZHn9PSQoVgdp9lM0lBn9/fy5cuFDkc3Fxcfj7+wOGjubJkyezatUqUlJSCAgIYMaMGcW+ssiSJDGIMpVxzTA38+lfDY/9guClhVDDz+wmk2/l0G/pn5y8cpPa1bSse6M9DSQpVBhllhh27drF4sWLOXfuHBs2bKBu3bp8++23NGjQwKxaReXR3aeSTp8+LYlBlL4TP8PmMZB5A+wd4R+ToX2oWSWzC6TcyqHvXUlhzcgnaFhLfYe1sF1qE4NZI583btxIt27dcHZ2Jjo62njpaFpaGp988ok5TZZLoaGhxMbGsn//fmuHIiq67HT4KRTW9TckhTqPwYgICHqnxEmh39dRnLxyk1rVtKweIUlBmJkYZs6cyaJFi1i6dCkODg7G5UFBQRw6dMhiwQkhgAt7YFEQRK8CNBA0BkbsAK/HStRsamYO/b+O4kRSOp6uWtaMeIJGtSUpCDPHMZw6darIDl43NzdSU1NLGlO5IVcliVKVp4OIj2H3l4AC7vXhpUXgH1TipguSQmxSOp6ujqwdGShJQRiZdcTg5eXF2bNnCy2PjIwsk1pJtkJOJYlSc/U4LP0H7J4PKNBiALy52yJJIS0zl4HL9nE8MZ2aLo63jxSqlTxmUWGYdcQwYsQIxowZw/Lly9FoNCQmJrJ3717Gjx/P5MmTLR2jEJWHPh/2hsGOGZCfA1VrwgtfQtPnLdJ8WlYuA5dHcfRyGjVdHFk94gka15GkIEyZlRjef/999Ho9Xbt2JTMzk86dO6PVahk/fjyjR4+2dIxCVA6pFw2F7y5EGh436W4Ywexa2yLNp2XlMmhZFEcupeHh4sh3IwJ52EuSgiisROMYcnJyOHv2LBkZGTRr1gxXV1eysrJKvYidrZDLVYVFKAocXmMokZ1zExxcoPssaDXI7MFqf5eebTh9dDghlRpVHVg94gmaest3tbKw2gA3nU5HWFgYn332GVeuXLFEk+WGDHATZrv1F/w8Bk5sNjz2DYSXF4GH5frqbt5OCjG3k8J3w5+gmY98TyuTUh3HoNPpmDhxIm3atKFDhw78+OOPAISHh9OgQQPmzZvHu+++a1bgQlQ6p3+Dr54wJAW7KtB1Cgz91WJJITdfz+bDiQQv2ktMQiruVR1YNTxQkoJ4IFV9DFOmTGHx4sU8/fTT7Nmzh+DgYIYOHcqff/7J3LlzCQ4Oxt7e/ME2QlQKugz43yQ4uMLwuNYj8MoS8G5ukeZTbuWwet9Fvt17gSvphjlK3Ks6sCokkEd9pAqweDBViWH9+vV88803vPjiixw7doyAgADy8vI4fPhwmVRVtTUyjkGolrAPfhgJKXGGx0+EGo4UHJxK3PTJK+ms2B3PpujL6PL0AHi6OtI/0I8BT/hRq1rhWQ+FKIqqPgZHR0fi4uKoW7cuYJgpbd++fTz++OOlFmB5IH0M4oHycmDnpxA5FxQ9VK8HL30FDz1Zombz9Qo7Tl4jfHcce879ZVz+WN3qDO3QgOebe6OtIkfxlZ3afZSqI4b8/HwcHR3vvLhKFVxdZbSkEPd17SRsGmmYYQ0g4HXo8Rk4u5vdZHp2LusPXGLlnnguJmcCYKeB7o95MTSoAW38ymaaXVExqUoMiqIwZMgQtFrDIWl2djZvvvkmLi6m5Xl/+OEHy0UoRHml18O+xbB1KuTrwLkGPD8PHn3Z7Cbjbtxi5Z541h9I4FaO4RSmm7MDfdr5Mqi9P3XdK8el4qJ0qUoMgwcPNnk8YMAAiwYjRIWRdgl+fAvidhoeN+wKvcKgurfqphRFIfLsDcJ3xxNx6hoFJ38b13ZlSJA/L7esS1VHi03fLoS6xBAeHl5acQhRMSgKHN0A//0n6NKgijN0mwltQlQPVsvKyeeH6Eus2B3PmWsZxuX/eKQ2Q4P86djIU04XiVIhPzNKQK5KEiYyk+G/4+D4JsPjuq3h5SXg2UhVM5dTs/hmbzxr9yWQlpULgIujPcFtfBncwV9mVhOlzuam9iyP5KokQfxu2BgCN5NAYw9PToBO/wT74v32UhSFAxdSCN8dx5ZjV9Df/l9Z36Mqgzv4E9ymHtWdHO7fiBD3UKpXJQkhinDoG/j5XdDnQc3G8Mpiw9FCMejy8tl8OIkVe+I4djnduDyoUU2GdmjAU4/Uxt5OTheJsiWJQQhz6fNh6xTYu8Dw+NFXDB3MjlUf+NJrN7NZ9edFVkdd4EZGDgDaKna80qouQzo0kKqnwqokMQhhjux02DgczvxmeNxlouH00QM6g49cSiV8dzw/H0kkN99wvsjbzYmB7f3o27Y+NVwc7/t6IcqCJAYh1Eq5AGv6wLVYqOJkGMH82Kv3XD0vX8+W41cI3x3PwQspxuWt/WowNMifbo964WBv1mSKQpQKSQxCqHHxT1jbHzJvgKsX9F19z/6ElFs5rNlvKGaXlGYoZudgr+H5AB+GdPCnua97GQYuRPFJYhCiuGLWwOZ3DFNuegVA37XgVrfQaqev3iR8dxyboi+TnXunmF2/QD8GBNandvWSF8wTojRJYigBGcdQSej1sGM6RM4zPG76Ary8GBxd7lpFIeLUNZbvjmP32TvF7B71qc7QoAa8IMXsRDki4xgsQMYxVGC6DNj0Bpz82fC403h4ahLYGfoE9HqFLcevMH/bGU5dvQkYitl1e9RQzK6tvxSzE9Yn4xiEsJS0S7C6D1w9CvZa6LUAAl4DDAnht+NXmL/9DCevGBJCNW0V+gXWZ2B7P+rVePAlq0LYKkkMQhTl0gFY0xduXQOXWtBnNfi2u2dCGNaxAcM6NsDNWUYni/JPEsNtx48fZ9SoUcTExFCvXj2++uorunTpYu2whDUc3WCojJqvg9qPQr+16Kv78r9jSXyxrYiEENQAt6qSEETFIYkByM3NpVevXrzzzjtEREQQERFB7969OXXqFDVr1rR2eKKs6PWw89+GmdYAmvRA//IS/nfuFl9s22WSEIZ2bECIJARRQUliAE6dOkVKSgrvvPMOAE8//TQtW7Zk06ZNDB8+3MrRiTKRkwk/joLYHwFQ2r/Dbz5vMn/JYU4kGWoYuWqrMCzIn5COD0lCEBWaTQ23zMjIYOrUqXTv3h0PDw80Gg0rVqwocl2dTseECRPw8fHB2dmZwMBAtm7davZ7//3iLEVROH78uNntiXIkPRHCe0Dsjyh2Dhxt8wk9Tz7Lm9/FcCIpHVdtFd75RyMiJzzFuGcflqQgKjybSgw3btxg+vTpnDhxgubNm9933SFDhjB37lz69+/P/Pnzsbe3p2fPnkRGRqp+34cffhh3d3fmzp1Lbm4uW7ZsYefOndy6dcvcjyLKi8RoWPoPSIohx9Gd96rO4IVIf2NCGH1XQnCvKnWMROVgU4nB29ubpKQkLly4wOzZs++53r59+1i7di2zZs1i9uzZjBw5kh07duDn58d7771nsm7Hjh3RaDRF3j788EMAHBwc+PHHH/npp5/w8vJi3rx5vP7669SrV69UP6+wsuM/oizvATeTuGBXn643p7L+Rn2ThPBPSQiiErKpPgatVouXl9cD19uwYQP29vaMHDnSuMzJyYmQkBA++OADEhIS8PX1BSj2EURAQAA7d+40Pu7QoYPMaV1RKQrKH7PRRHyMBojIb87o7NEojtV4O6gBIR0bSJVTUanZVGIorujoaJo0aVJoBF+7du0AiImJMSaG4jpy5AhNmjRBr9cTFhaGXq+ne/fuFotZ2AYlN4sr347A++JmAJbl9WC+3UCGPNVIEoIQt5XLxJCUlIS3t3eh5QXLEhMTVbcZHh5OeHg4er2eZ555hh9//PGe6+p0OnQ6nfFxenr6PdcVtkFRFP44dJw6vwzjkfxT5Cr2fKwMxaXTCHZ2fEgSghB3KZeJISsrC61WW2i5k5OT8Xm15s2bx7x584q17qxZs5g2bZrq9xBlT1EUtp+4xk+/beH91GnU1fxFmuLCr80+453nX8NDEoIQhZTLxODs7Gzyi71Adna28fnSNHHiRMaNG2d8nJ6ervrUlShdBQnhi+2n8U7awRcOYbhodCQ71cduwPf0qdfU2iEKYbPKZWLw9vbm8uXLhZYnJSUB4OPjU6rvr9Vq0Wq1UnbbBimKwo6T1/hi2xmOXk7lDfufmeCwFjuNQq7fk3j0WQnONawdphA2rVwmhhYtWhAREUF6erpJB3RUVJTx+bIQGhpKaGiosaStsB5FMcyH8MW2Mxy5lIYjuczTLudlze0rzdoOx6H7v8FeBqcJ8SA2NY6huHr37k1+fj5LliwxLtPpdISHhxMYGCindSoRwxHCVXqF7WbYigMcuZRGXcdbbK8115AUNPbQcw4897kkBSGKyeaOGBYsWEBqaqrxyqLNmzdz6dIlAEaPHo2bmxuBgYEEBwczceJErl27RqNGjVi5ciXx8fEsW7aszGKVU0nWoygKv5+6zhfbTnP4UhoAzg72jGuRx7ALM7BPvwhaNwgOh0ZdrRytEOWLzc3g5u/vz4ULF4p8Li4uDn9/f8DQ0Tx58mRWrVpFSkoKAQEBzJgxg27dupVhtAYyg1vZuVdCGNTBj9C656n+8xuQcxNqNIB+30OtJlaOWAjrU7uPsrnEUJ7cfcRw+vRpSQx3URSF3HyFrNx8snPzyczJJysn3/Rxbj7Zt//OvOu5gvWK+js9O5ektNtXnznYM6i9HyM6NcDz2DL434eg6MGvI7z+LVT1sPJWEMI2SGKwAnOOGDJz8jh6KQ0FKPgXUDDeKbiHotxZfme9O9VgFeMfhvWM69xj3Tv/2nfaLOq98vUKulx9kTvtzJzb94vaed/e2Wfm5pOvL52vljEhdH4ITycN/DIeDq00PNlqEPT8HKrI+AQhCsicz+XExeRMXl/yp7XDKBP2dhqqOtjj5GhPVUd7nB3scXIw/F3V0bDc+e7HDvY4Fyy76znn2881rOViKGyXmQyrBkH8LtDYwbMz4Ym3QKOx9kcWolyTxGAl2iqGHRxgqPZ6e3nBPk2DxmT/prn94O71ilpXc+cFpuve1U5R78VdrzesrzHslP+2Yy5qZ333TrtqEY8d7Evh4rfrp2HN65B8HhyrQe9l0KTs+5eEqIgkMZRASa5KauDpwvZ/drF8UJXBuR3w/RDQpYF7fei7Duo0s3ZUQlQY0sdgAXJVUinJz4VbNyDzxu2//4JrsRD5BSj5UL89vL4KXDytHakQNk36GITtysu5ayd/+++/7/hvXb+zLDvt3m017wcvfAFVChdTFEKUjCSGEqj0A9zydHft2K/Drb/u2uFfv72jL3j+L8OpH7U0dlC1JrjUuv23JzT8B7QcKJ3MQpQSOZVkAWadStLdhAt77vHkPXZ4Re4I77Xuvd64iCeKaleff2fHfuv6nZ373b/0c27e603uTWNv2LlX9QSXmrf/rnV7Wc27nru9zMkd7Mpl5RYhbIacSiovUhNg9WvWjqLk7Krc3pF73vllf/eO36WW6fOyoxfC5klisBYHJ/BpVcQT9ziAK/LATs2691j/XqtqNH/7Bf/3X/e3lzm5yykdISoYSQzW4vEQjIywdhRCCFGIHNOXQFhYGM2aNaNt27bWDkUIISxGOp8tQMYxCCFsmdp9lBwxCCGEMCGJQQghhAlJDEIIIUxIYigB6XwWQlRE0vlsAdL5LISwZTLy2QoKcmt6erqVIxFCiMIK9k3FPQ6QxGABN28aagb5+vpaORIhhLi3mzdv4ubm9sD15FSSBej1ehITE6lWrZpxprXyLD09HV9fXxISEuTUWDHJNjOPbDf1zNlmiqJw8+ZNfHx8sCtGrTI5YrAAOzs76tWrZ+0wLK569eryn1Ul2Wbmke2mntptVpwjhQJyVZIQQggTkhiEEEKYkMQgCtFqtUydOhWtVqbNLC7ZZuaR7aZeWWwz6XwWQghhQo4YhBBCmJDEICxGp9MxbNgw6tevT/Xq1XniiSfYu3evtcOyGdevX+e5557DxcWFhx9+mO3bt1s7JJsn36mS2bt3L3Z2dsycOVPV6yQxCIvJy8vD39+fyMhIUlNTGTt2LC+88AIZGRnWDs0mhIaG4uXlxfXr15k9ezavvfYaycnJ1g7Lpsl3ynx6vZ53333XrFpu0scgSpWPjw+bN2+mdevW1g7FqjIyMvDw8OD8+fPGMS9dunRh8ODBDB061MrRlS/ynSqeRYsWceLECdLS0mjUqBEffvhhsV8rRwwVUEZGBlOnTqV79+54eHig0WhYsWJFkevqdDomTJiAj48Pzs7OBAYGsnXrVovEcebMGZKTk2nUqJFF2itLlt6GZ86cwdXV1WQg5OOPP87x48dL82OUudL+7pXn79S9lMY2++uvv/jiiy+YNm2aWTFJYqiAbty4wfTp0zlx4gTNmze/77pDhgxh7ty59O/fn/nz52Nvb0/Pnj2JjIwsUQxZWVkMGDCAiRMnqhpxaSssvQ0zMjIKjVKtXr16hTslUprfvfL+nbqX0thmkyZNYuzYsbi7u5sXlCIqnOzsbCUpKUlRFEXZv3+/Aijh4eGF1ouKilIAZfbs2cZlWVlZSsOGDZX27dubrBsUFKQARd4mTZpksm5OTo7y3HPPKf369VP0er3lP2AZsPQ2PHTokFKjRg2T17799tvKP//5z9L5AFZSGt89RakY36l7KY3vWqtWrZS8vDxFURRl8ODByowZM1TFJEcMFZBWq8XLy+uB623YsAF7e3tGjhxpXObk5ERISAh79+4lISHBuDwyMhJFUYq83X3Fg16vZ+DAgWg0GlauXFluiwpaehs2btyYjIwMLl++bFzv2LFjPProo5YP3opK47tXUb5T92LpbbZz505OnTpF3bp18fLyYt26dXz66aeq+rIkMVRi0dHRNGnSpNApjnbt2gEQExOjus033niDpKQk1q9fT5UqFb9GY3G3oaurK7169WLq1KlkZWXx888/c+TIEXr16lXWIdsENd+9yvadupfibrORI0dy9uxZYmJiiImJ4cUXXyQ0NJR58+YV+70q71YWJCUl4e3tXWh5wbLExERV7V24cIGvv/4aJycnPD09jct//fVXOnXqVLJgbZSabfjVV18xePBgatasSb169Vi3bh0eHh5lFqstKe52q4zfqXsp7jarWrUqVatWNT7v7OyMq6urqv4GSQyVWFZWVpH1VpycnIzPq+Hn51fsGaIqCjXbsFatWvzyyy9lFpstK+52q4zfqXsx9//rva5wuh85lVSJOTs7o9PpCi3Pzs42Pi/uT7aheWS7qVeW20wSQyXm7e1NUlJSoeUFy3x8fMo6pHJHtqF5ZLupV5bbTBJDJdaiRQtOnz5tnCi8QFRUlPF5cX+yDc0j2029stxmkhgqsd69e5Ofn8+SJUuMy3Q6HeHh4QQGBuLr62vF6MoH2Ybmke2mXlluM+l8rqAWLFhAamqq8UqFzZs3c+nSJQBGjx6Nm5sbgYGBBAcHM3HiRK5du0ajRo1YuXIl8fHxLFu2zJrh2wTZhuaR7aaezW0zVcPhRLnh5+d3z5HKcXFxxvWysrKU8ePHK15eXopWq1Xatm2rbNmyxXqB2xDZhuaR7aaerW0zqa4qhBDChPQxCCGEMCGJQQghhAlJDEIIIUxIYhBCCGFCEoMQQggTkhiEEEKYkMQghBDChCQGIYQQJiQxCCGEMCGJQQghhAlJDEIIIUxIYhBCCGFCEoMQNmjFihU0a9aMqlWr0rRpU/773/9aOyRRiUhiEMLGbNy4kbfffpvJkydz7NgxunXrxptvvmntsEQlImW3hbAxQUFBPP3000ybNg2ArVu3EhwcTGpqqnUDE5WGHDEIYUNu3rzJn3/+Sc+ePY3LfvvtN1q2bGnFqERlI1N7CmFDDh8+jJ2dHc2bNyczM5PVq1fz5ZdfsmnTJmuHJioRSQxC2JCYmBgeeeQRDh48SMeOHQF45ZVX6NGjh5UjE5WJnEoSwobExMTQqlUrHn/8caKiopg7dy5btmxh+vTp1g5NVCJyxCCEDYmJiWHgwIFUr16ddu3a0a5dO06dOkVUVJS1QxOViBwxCGEj8vLyOH78OE2bNjVZfvjwYeNpJSHKghwxCGEjTp48SXZ2NtOnT6dWrVpUrVqVhQsXEh8fT0hIiLXDE5WIJAYhbERMTAze3t44OzvTqVMnXFxc6NixIxEREXh5eVk7PFGJSGIQwkbExMQQGBgol6YKq5M+BiFsRExMDAEBAdYOQwhJDELYisOHD0tiEDZBaiUJIYQwIUcMQgghTEhiEEIIYUISgxBCCBOSGIQQQpiQxCCEEMKEJAYhhBAmJDEIIYQwIYlBCCGECUkMQgghTEhiEEIIYUISgxBCCBP/DzkNbExSVU+AAAAAAElFTkSuQmCC",
      "text/plain": [
       "<Figure size 400x300 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "res = np.vstack(experiments)\n",
    "fontsize = 12\n",
    "\n",
    "# Extract errors\n",
    "conv_coeffs = np.reshape(np.asarray(conv_coeffs), (-1, 1))\n",
    "rmse_elm = np.reshape(res[:, -4], (-1, 1))\n",
    "rmse_swim = np.reshape(res[:, -2], (-1, 1))\n",
    "rel_l2_elm = np.reshape(res[:, -8], (-1, 1))\n",
    "rel_l2_swim = np.reshape(res[:, -6], (-1, 1))\n",
    "\n",
    "# Store the errors for different values of beta\n",
    "rmse = np.hstack((conv_coeffs, rmse_elm, rmse_swim))\n",
    "rel_l2 = np.hstack((conv_coeffs, rel_l2_elm, rel_l2_swim))\n",
    "print(np.shape(res))\n",
    "\n",
    "# Store rel l2 error values for convergence plots\n",
    "with open('adv_ode_swim_elm_rmse.npy', 'wb') as f:\n",
    "    np.save(f, rmse)\n",
    "\n",
    "with open('adv_ode_swim_elm_rel_l2.npy', 'wb') as f:\n",
    "    np.save(f, rel_l2)\n",
    "\n",
    "# Visualiza errors vs convection coefficient\n",
    "fig, ax = plt.subplots(1, 1, figsize=(4, 3))\n",
    "ax.loglog(conv_coeffs, res[:, -8], label='Frozen-PINN-elm')\n",
    "ax.loglog(conv_coeffs, res[:, -6], label='Frozen-PINN-swim')\n",
    "plt.xlabel(r'$\\beta$')\n",
    "plt.ylabel(r'Relative  $\\mathbb{L}_{2}$ error')\n",
    "plt.tick_params(axis='both', labelsize=fontsize)\n",
    "ax.legend()\n",
    "fig.tight_layout()\n",
    "plt.savefig('advection_ode_swim_elm.pdf', bbox_inches='tight')\n"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "test2",
   "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.12.2"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
