{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Solving advection equation using ELM-ODE and SWIM-ODE for different values of flow velocities\n",
    "In this script, we see how the SWIM-ODE and ELM-ODE performs for different values of flow velocities."
   ]
  },
  {
   "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 SWIM-ODE and ELM-ODE for good hyper-parameters "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[0.033757527669270836, 0.043172359466552734, 6.217344519192376e-10, 2.4616337261142887e-10, 4.376065445360843e-10, 1.3672746304091418e-10, 8.809876555990293e-10, 3.488095148368321e-10, 6.200813909468158e-10, 1.9374060219533226e-10]\n",
      "[0.03362154960632324, 0.03935082753499349, 6.246828525178684e-10, 2.446835408742689e-10, 4.37025971573426e-10, 1.363557135024605e-10, 8.85165376400437e-10, 3.467125721219772e-10, 6.192586479128512e-10, 1.9321381398619897e-10]\n",
      "[0.03520782788594564, 0.040271759033203125, 6.75304471080849e-10, 2.312712505288523e-10, 5.136094542118958e-10, 1.84073482031596e-10, 9.568829609843076e-10, 3.2770332268405387e-10, 7.277667428281751e-10, 2.6082572538453967e-10]\n",
      "[0.033083438873291016, 0.04036959012349447, 1.7918848947462148e-09, 2.4852940211331203e-10, 2.6204806124823794e-09, 1.1500385541036766e-09, 2.5363530226109957e-09, 3.517850404933965e-10, 3.7092024948982767e-09, 1.6278410356448487e-09]\n",
      "[0.040961901346842446, 0.043559869130452476, 4.16722760872261e-09, 1.2611125708197152e-09, 1.0116718670261828e-08, 5.811514371521282e-09, 5.893949327483123e-09, 1.7836639335718467e-09, 1.430865621021439e-08, 8.219558526150662e-09]\n",
      "[0.0680088996887207, 0.06788341204325359, 1.3570838794486394e-08, 6.0210295403999214e-09, 3.7263429547958334e-08, 2.2532724633715202e-08, 1.91917953444009e-08, 8.514902317526264e-09, 5.2697709002738454e-08, 3.1865638246155986e-08]\n",
      "[0.12776931126912436, 0.12627657254536948, 3.309170044130356e-08, 1.5574359335397154e-08, 9.217531990270805e-08, 5.6157623520084134e-08, 4.6799160634120045e-08, 2.2025672135029356e-08, 1.3035678267061905e-07, 7.941960095427864e-08]\n",
      "[0.4315965970357259, 0.43122514088948566, 1.6405340060529916e-07, 7.889599538799764e-08, 4.5885777790778104e-07, 2.8056912624094756e-07, 2.320094252675112e-07, 1.1157717230084826e-07, 6.489309513799421e-07, 3.9678959099167477e-07]\n",
      "[0.7785285313924154, 0.7866193453470866, 3.2779879191092715e-07, 1.5803973707641966e-07, 9.17336270770009e-07, 5.609677824161414e-07, 4.6358560799820115e-07, 2.2350584995553053e-07, 1.2973320930947003e-06, 7.933421260120288e-07]\n",
      "[3.6184581915537515, 3.6159478028615317, 1.638050938842924e-06, 7.910830658404983e-07, 4.58517723554586e-06, 2.80502061379283e-06, 2.3165266312477897e-06, 1.1187472538815787e-06, 6.484343632583235e-06, 3.966851578889276e-06]\n",
      "[7.821764230728149, 7.412562131881714, 3.2755421906797827e-06, 1.582071755320782e-06, 9.169010640324028e-06, 5.609276825476654e-06, 4.632448897708904e-06, 2.23745143014359e-06, 1.2967310680567147e-05, 7.932942619716568e-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",
    "        "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(11, 10)\n",
      "Information for beta = 40:\n",
      "time_swim =  3.6159478028615317 time_elm =  3.6184581915537515\n",
      "rmse elm =  1.638050938842924e-06 +- 7.910830658404983e-07\n",
      "rmse swim =  4.58517723554586e-06 +- 2.80502061379283e-06\n",
      "rel error elm =  2.3165266312477897e-06 +- 1.1187472538815787e-06\n",
      "rel error swim =  6.484343632583235e-06 +- 3.966851578889276e-06\n"
     ]
    }
   ],
   "source": [
    "res = np.vstack(experiments)\n",
    "print(np.shape(res))\n",
    "\n",
    "# For beta = 40 (Values in the table in the paper)\n",
    "print('Information for beta = 40:')\n",
    "print('time_swim = ', res[-2, -9], 'time_elm = ', res[-2, -10])\n",
    "print('rmse elm = ', res[-2, -8], '+-', res[-2, -7])\n",
    "print('rmse swim = ', res[-2, -6], '+-', res[-2, -5])\n",
    "print('rel error elm = ', res[-2, -4], '+-', res[-2, -3])\n",
    "print('rel error swim = ', res[-2, -2], '+-', res[-2, -1])"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Error Vs flow velocity for SWIM-ODE and ELM-ODE"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(11, 10)\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAYYAAAEiCAYAAAD9DXUdAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjguMywgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/H5lhTAAAACXBIWXMAAA9hAAAPYQGoP6dpAABOm0lEQVR4nO3deVwU9f/A8deywIJyiaiAgpiIaXnghbdmeZZH5n1fWf7QUrNMy/ympaalWdphGt53ZllmWR6pmaLibeIB4oGiIiICC+zO74+VTQKVWXc538/HYx/Ozs5+5r3jMu+d+VwaRVEUhBBCiHvs8jsAIYQQBYskBiGEEFlIYhBCCJGFJAYhhBBZSGIQQgiRhSQGIYQQWUhiEEIIkYUkBiGEEFnY53cA+ally5b8/fff2NubDkOzZs345ZdfcvVeo9HIlStXcHV1RaPR2DJMIYR4LIqicOfOHXx9fbGze/T1QLFODAALFy6kX79+qt935coV/Pz8bBCREELYxsWLF6lQocIjtyv2icFSrq6ugOlAu7m55XM0QgjxYImJifj5+ZnPW4+iye+xkpKSkpg1axb79u1j//793Lp1i7CwMAYNGpRtW71ez3vvvceyZcu4desWNWvW5IMPPqB169YW7btly5acOHECgNq1a/PJJ59Qs2bNXL03MTERd3d3bt++LYlBCFGgqT1f5Xvl840bN5gyZQqnTp2iVq1aD9120KBBzJ49m759+zJ37ly0Wi0dOnRg9+7dFu175syZREVFERMTQ+vWrWnfvj137tyxqCwhhCgylHyWmpqqxMbGKoqiKOHh4QqghIWFZdtu3759CqDMmjXLvC4lJUWpXLmy0qhRoyzbNmnSRAFyfLzzzjsPjKVq1arKb7/9lqu4b9++rQDK7du3c7W9EELkF7Xnq3yvY9DpdHh7ez9yu/Xr16PVahk+fLh5nZOTE0OHDmXixIlcvHjRXBls6RWEnZ0dioxCLoQo5vI9MeRWREQEQUFB2e6PNWjQAIDDhw+raiWUkJBAeHg4zZs3R6PRMH/+fOLj4wkJCbFq3AaDgfT0dKuWKQo/BwcHtFptfochRI4KTWKIjY3Fx8cn2/rMdVeuXFFVXnp6OhMmTOD06dM4ODhQu3ZtNm/ejLu7e47b6/V69Hq9+XliYuJDy1cUhatXr5KQkKAqLlF8eHh44O3tLf1gRIFTaBJDSkoKOp0u23onJyfz62qUKVOGAwcO5Hr76dOn8/777+d6+8ykULZsWUqUKCF//MJMURSSk5OJi4sDyPEHjxA5UhSIPQK+tW26m0KTGJydnbP8Ys+Umppqft2WJkyYwNixY83PM9sF58RgMJiTQunSpW0alyicMr+vcXFxlC1bVm4ridzZPRv+mAptP4RGoTbbTaFJDD4+Ply+fDnb+tjYWAB8fX1tun+dTodOp2P+/PnMnz8fg8HwwG0z6xRKlChh05hE4Zb5/UhPT5fEIB7t5I/wxxTTsn32uyfWlO/9GHKrdu3aREZGZru3v2/fPvPreSE0NJSTJ08SHh7+yG3l9pF4GPl+iFy7chi+f8W03OAVqD/MprsrNImhW7duGAwGFixYYF6n1+sJCwsjJCRExi0SQhRNibGwqhekJ0PlZ6HtNJvvskDcSpo3bx4JCQnmlkWbNm3i0qVLAIwaNQp3d3dCQkLo3r07EyZMIC4ujsDAQJYsWUJ0dDSLFi3Ks1hzcyupuGnZsiW1a9fm008/ze9QVBk0aBAJCQls3Lgxv0MRImdpyaakcCcWyjwJ3cNAa/vTdoFIDB9//DEXLlwwP9+wYQMbNmwAoF+/fuYmpEuXLmXSpElZxkr66aefaN68eZ7FGhoaSmhoqHnsESGEsAmjETa+CrGHoURp6L0anPLmnFMgEkN0dHSutnNycmLWrFnMmjXLtgE9RHG7YkhLS8PR0THf9m8wGNBoNNnGkM/vuISwuR3T4OQPYOcAPZeDZ6U823WhqWMoKNRUPhdGLVu2ZOTIkYwePRovLy/atm3L8ePHad++PS4uLpQrV47+/ftz48YNi8rX6/WMGzeO8uXLU7JkSUJCQtixY4f59cWLF+Ph4cGPP/5I9erV0el0xMTEEBAQwNSpUxkwYABubm5ZhkZ5kIsXL9KjRw88PDzw9PSkc+fOD/0R0rJlS0aNGsXo0aMpVaoU5cqV45tvvuHu3bsMHjwYV1dXAgMDcz2ZkxAWO7oW/rz3A7jTZ1CxcZ7uXhJDHlEUheS0jDx/WDL205IlS3B0dGTPnj3MmDGDVq1aERwczIEDB9iyZQvXrl2jR48eFh2HkSNHsnfvXlavXs3Ro0fp3r077dq148yZM+ZtkpOT+eijj1i4cCEnTpygbNmygOmWY61atYiIiGDSpEkP3U96ejpt27bF1dWVXbt2sWfPHlxcXGjXrh1paWkP/exeXl7s37+fUaNGMWLECLp3707jxo05dOgQbdq0oX///iQnJ1v0+YV4pIv74YeRpuUmo6F2nzwPoUDcSioOUtINVH/v1zzf78kpbSnhqO6/uUqVKsycOROADz74gODgYKZN+7clxLfffoufnx+RkZEEBQXlutyYmBjCwsKIiYkx9zsZN24cW7ZsISwszLyP9PR0vvjii2zDsLdq1Yo33ngjV/tas2YNRqORhQsXmpuFhoWF4eHhwY4dO2jTpk2O76tVqxbvvvsuYOrUOGPGDLy8vHj55ZcBeO+99/jyyy85evQoDRs2zPVnFyJXEmJgdR8w6KHq8/Ds5HwJQxKDSsWhjqFu3brm5SNHjrB9+3ZcXFyybXfu3DlVieHYsWMYDIZs79Hr9Vl6iDs6OuY4YVK9evVyva8jR45w9uzZbDNWpaamcu7cuQe+7/79arVaSpcuTY0aNczrypUrB2AezkIIq9HfgZU94e51KFcDui6AXMzPbAuSGFSytFWSs4OWk1Pa2jCyB+9XrZIlS5qXk5KS6NixIx999FG27dSO8ZOUlIRWq+XgwYPZevren3icnZ1z7Px1f1y52VfdunVZsWJFttfKlCnzwPc5ODhkea7RaLKsy4zLaDTmOhYhHslogPVDIe4kuJSDPqtBl/3HWF6RxJBHNBqN6ls6BUGdOnX47rvvCAgIwN7+8eIPDg7GYDAQFxdHs2bNrBRhzurUqcOaNWsoW7asTL0qCr6t78GZX8HeCXqtAvcK+RqOVD6rNH/+fKpXr079+vXzO5Q8ERoaSnx8PL179yY8PJxz587x66+/MnjwYNW304KCgujbty8DBgxgw4YNREVFsX//fqZPn87PP/9s1bj79u2Ll5cXnTt3ZteuXURFRbFjxw5ee+01c+dJIQqEg0tg7zzTcpcvoELdh2+fByQxqFTUm6v+l6+vL3v27MFgMNCmTRtq1KjB6NGj8fDwyNa3IDfCwsIYMGAAb7zxBlWrVqVLly6Eh4fj7+9v1bhLlCjBn3/+ib+/P127dqVatWoMHTqU1NRUuYIQBUfULvj53qjNLSfA0y/lbzz3aBSZy9IimXUMt2/fznaiSU1NJSoqikqVKpnnixDiv+R7UszdPAfftILUBFNCeGkR2GhgxYedr3IiVwxCCJHXUm7Byh6mpFC+HnSeb7OkYAlJDMJqdu3ahYuLywMf1jRt2rQH7qd9+/ZW3ZcQVmVIh7UD4eZZcKsAvVaCg20nGlOr8DWTyWfFoR+DperVq8fhw4fzZF+vvvrqA3tf23o2PyEspijwy3iI2gkOJU3NUl3L5XdU2Ugdg4WkjkE8LvmeFEP7voZf3gI00GsFPPl8nuxW6hiEEKIgOvM7bHnbtPzc/yxKCslpGSzcdR6D0ba/5+VWkhBC2FrcP7B+MChGqN0PmryuugijUWHsmiNsOXGVyGt3mNmt1qPfZCG5YhBCCFu6e8PUAkmfCP6N4YU5FrVA+vi302w5cRVHrR0969t2KmNJDEIIYSsZeljTDxIuQKkA04Q79uonmPru4CW+2GEa/HFmt5rUrehp5UCzksSgUnEbEkMIYSFFgU2jIWYv6Nyg9xooWfqRb/uv8Oh4Jmw4BsDIZwLpElzeyoFmJ4lBpeI2JEZutGzZktGjR+d3GDYzaNAgunTpkt9hiMJmz6dwZCVo7KB7GJR9UnURF+OTeWXZQdIMRto/7c3Y1rkf5v5xSOWzEI8wd+5ci2bCE8XYqZ/g9/dNy+0+gsDnVBdxJzWdoUvCib+bxtPl3fikRy3s7PKmd7QkBvFQaWlpODqqvydqLQaDAY1Gk23AvryMS828G0IQewQ2vAwoUH8YhDx6fvL/yjAYGbUqgshrSZRz07FwQP08HbZfbiWJLFq2bMnIkSMZPXo0Xl5etG3bluPHj9O+fXtcXFwoV64c/fv358aNGxaVr9frGTduHOXLl6dkyZKEhISwY8cO8+uLFy/Gw8ODH3/8kerVq6PT6YiJiSEgIICpU6cyYMAA3NzcGD784X9saWlpjBw5Eh8fH5ycnKhYsSLTp08HTNOJvvDCC+ZtP/30UzQaDVu2bDGvCwwMZOHChUD2W0ktW7Zk1KhRjB49mlKlSlGuXDm++eYb7t69y+DBg3F1dSUwMJBffvnFomMkCrE7V2FVb0hPhieeMV0tWODDzafYcfo6Tg52LBxQH2/3vO0AKYkhrygKpN3N+4cFt0CWLFmCo6Mje/bsYcaMGbRq1Yrg4GAOHDjAli1buHbt2gOHo3iUkSNHsnfvXlavXs3Ro0fp3r077dq148yZM+ZtkpOT+eijj1i4cCEnTpygbNmyAHz88cfUqlWLiIgIJk2a9ND9fPbZZ/z444+sXbuW06dPs2LFCgICAgBo0aIFu3fvNg9rsnPnTry8vMwJ6vLly5w7d46WLVs+9Bh5eXmxf/9+Ro0axYgRI+jevTuNGzfm0KFDtGnThv79+5OcnGzRcRKFUHqKab7mxMvgFQTdF4NW/a/8FfsuELYnGoDZPWpTo0LeX7HKraS8kp4M03zzfr8Tr4Bj7qfEBKhSpQozZ84E4IMPPiA4OJhp06aZX//222/x8/MjMjJS1ZzPMTExhIWFERMTg6+v6ViMGzeOLVu2EBYWZt5Heno6X3zxBbVqZe3A06pVK954441c76tKlSo0bdoUjUZDxYoVza81a9aMO3fuEBERQd26dfnzzz9588032bhxIwA7duygfPnyBAYGPrD8WrVq8e677wIwYcIEZsyYgZeXFy+//DIA7733Hl9++SVHjx6lYcOGuTtAovBSFNj4f3D5IDiXgt6rwdlDdTF7zt7gvR9OADCuTRAdaqibPtdaJDGIbOrW/XcGqSNHjrB9+/YcR0c9d+6cqsRw7NgxDAZDtvfo9XpKl/63GZ+joyM1a9bM9v569erlel+DBg2idevWVK1alXbt2vHCCy/Qpk0bADw8PKhVqxY7duzA0dERR0dHhg8fzuTJk0lKSmLnzp20aNHioeXfH59Wq6V06dLUqFHDvK5cOdPAaHFxcbmOWRRiO2bAiQ1g52Dqq1C6suoizl1PYsTygxiMCi8Glyf0mQf/MLE1SQwqWTy6qkMJ06/3vOZQQvVbSpb89wojKSmJjh078tFH2e+V+vio+zWTlJSEVqvl4MGDaLXaLK/dn3icnZ3R5NAz9P64HqVOnTpERUXxyy+/8Pvvv9OjRw+ee+451q9fD5jqCXbs2IFOp6NFixZ4enpSrVo1du/ezc6dOx95ZeLg4JDluUajybIuM36j0ZjrmEUhdWw97JxhWn5hDgQ0VV1EQnIaw5YcIDE1g7oVSzG9a40c/wbyiiQGlUJDQwkNDTWPVphrGo3qWzoFQZ06dfjuu+8ICAjA3v7xvi7BwcEYDAbi4uJo1qyZlSJ8MDc3N3r27EnPnj3p1q0b7dq1Iz4+Hk9PT1q0aMG3336Lvb097dq1A0zJYtWqVURGRj60fkEIs0sHTLeQABqPgjr9VReRbjAyYvkhom7cpbyHM1/3r4uTg/bRb7QhqXwWDxUaGkp8fDy9e/cmPDycc+fO8euvvzJ48GDVV01BQUH07duXAQMGsGHDBqKioti/fz/Tp0/n559/tmrcs2fPZtWqVfzzzz9ERkaybt06vL298fDwAKB58+bcuXOHn376yZwEWrZsyYoVK/Dx8VF1i0wUUwkXTS2QDHoIagfPva+6CEVReO+H4+w9f5OSjloWDaqHl4vOBsGqI4lBPJSvry979uzBYDDQpk0batSowejRo/Hw8MjWtyA3wsLCGDBgAG+88QZVq1alS5cuhIeH4+/vb9W4XV1dmTlzJvXq1aN+/fpER0ezefNmc8ylSpWiRo0alClThiefNPVIbd68OUaj8ZH1C0KgTzIlhbtxUO5peGkh2Kn/lb9odxSr9l9Eo4HPegfzpPej50rICzJRj4Vkoh7xuOR7UkgZDaaB8U5vhpJl4OVt4KH+h822f64xdMkBFAXefb4aw5o9YYNgTWSiHpVmzpyJn58frq6uBAcHc+fOnfwOSQhRkP3xvikpaHXQa5VFSeGfq4mMWhmBokCv+n4MbVrJBoFarlgnhvnz57Nlyxb27NlDYmKiuWOXsMyuXbtwcXF54MOapk2b9sD9tG/f3qr7EsIsYgXsmWta7jwf/NSPsnwjSc/QxQe4m2ag4ROeTOn8dL62QMpJsW2VZDAY+PDDD9m1a5f5/nZObedF7tWrV4/Dhw/nyb5effXVB/a+dnZ2zpMYRDETvQc23Zt5rflbULO76iJS0w0MX3qAywkpBJQuwVf96uJoX/B+n+d7RElJSUyePJl27drh6emJRqNh8eLFOW6r1+sZP348vr6+ODs7ExISwtatWy3a76VLl0hOTmb9+vWUK1eOqlWr8s033zzGJxHOzs4EBgY+8GFNnp6eD9xP+fK2H69eFDPx5031CsZ0qN4FWk5QXYSiKLz93VEOxSTg5mTPokH18ShRMO9Q5HtiuHHjBlOmTOHUqVPZhkD4r0GDBjF79mz69u3L3Llz0Wq1dOjQgd27d6ve7+XLl7l9+zaRkZFER0ezbt06Jk6cyK5duyz9KEKIoij1NqzsBSnx4BsMXb4EC1rkzd9+lo2Hr6C10/BF37pULmPd26vWpOrTGQwG/vzzTxISEqwWgI+PD7GxsVy4cIFZs2Y9cLv9+/ezevVqpk+fzqxZsxg+fDjbtm2jYsWKvPXWW1m2zRwfJ6dH5vg2mbcb3nvvPZydnalZsya9evVi8+bNVvts0utVPIx8PwoBQwasGwQ3ToOrr2kMJEf1owlsPhbLx79FAvB+p6doWsXLyoFal6o6Bq1WS5s2bTh16pS5o9Dj0ul0eHt7P3K79evXo9Vqswy37OTkxNChQ5k4cSIXL17Ez880QXZuriCCgoJwdHTMUuljrQogR0dH7OzsuHLlCmXKlMm2H1G8KYpCWloa169fx87OTho8FGS/ToBz20xDy/RZDa6PPlf919FLCYxdexiAQY0D6New4sPfUACornx++umnOX/+PJUq5W3zqoiICIKCgrK1wW3QoAEAhw8fNieG3ChZsiTdunXjww8/5LPPPuP8+fOsWbPGPJbO47Czs6NSpUrExsZy5Uo+jI8kCoUSJUrg7+9vUUdBkQf2fwP7F5iWuy4An4ff6s7J1dupvLz0AKnpRloEleHd56tZOUjbUJ0YPvjgA8aNG8fUqVOpW7dutoHNctN5whKxsbE5DtqWuc6SE/D8+fMZOnQoXl5eeHl5MXXq1AeO4aPX69Hr9ebniYmJDy3b0dERf39/MjIy1A+4J4o8rVaLvb29XEkWVOe2wS/jTcvPToZqHVUXkZyWwbCl4VxL1FOlrAuf9wnGXls4fgSoTgwdOnQAoFOnTlm+1IqioNFobHYSTElJQafLPoZIZo/RlJQU1WV6eHjw3Xff5Wrb6dOn8/776sZCyRxx878jcQohCrDrkbB2ECgGqNUbmo5RXYTRqDB2zRGOX07Es6Qj3w6qj5tT4TkPqE4M27dvt0Ucj+Ts7JzlF3um1NRU8+u2NGHCBMaOHWt+npiYqOrWlRCiEEiOh5U9QH8b/BtBx7mmkZFV+mTrabacuIqj1o6v+9fFz1N9hXV+Up0Y8muAMR8fHy5fvpxtfWxsLIB5RjBb0el06HQ6y+djEEIUbBlppr4Kt6LAo6Jpwh179SOdbjh0ifnbzwEwvWsN6gd4WjtSm7Oo53NCQgKLFi3i1KlTADz11FMMGTJE3fwEKtWuXZvt27eTmJiYpR5j37595tfzgsXzMQghCi5FgZ/HwIU9oHODPmugpPompQei43n7u2MAjGhZmZfqVrB2pHlCdU3IgQMHqFy5MnPmzCE+Pp74+Hhmz55N5cqVOXTokC1iBKBbt24YDAYWLFhgXqfX6wkLCyMkJCTPbuvMnz+f6tWrU7+++jFShBAF1F+fQ8Ry0NhBtzAoq7710MX4ZF5ZdpA0g5G2T5XjzTZVbRBo3lA97HazZs0IDAzkm2++Mc/olZGRwbBhwzh//jx//vmn6iDmzZtHQkICV65c4csvv6Rr164EBwcDMGrUKPMv8x49evD9998zZswYAgMDWbJkCfv37+ePP/6gefPmqvf7ONQOYyuEKKD+2Qyr+wAKtPsIGr6quog7qem89OVfRF5L4ilfN9a92ogSjgVnKDq15yvVicHZ2ZmIiAjz5CaZTp48Sb169UhOTlYXMRAQEMCFCxdyfC0qKoqAgADAVNE8adIkli9fzq1bt6hZsyZTp06lbdu2qvf5uCQxCFEEXD0Gi9pC+l2oNwSen626stlgVBi2JJztp69T1lXHDyOb4ONesAZyVHu+Up3S3NzciImJyZYYLl68iKurq9riAIiOjs7Vdk5OTsyaNeuhQ2fYmlQ+C1FE3LlmGgMp/S5UagHtZ1rUAmna5lNsP30dnb0d3wyoV+CSgiVU1zH07NmToUOHsmbNGi5evMjFixdZvXo1w4YNo3fv3raIsUAJDQ3l5MmThIeH53coQghLpaeYbh8lXoLSgdBjCWjV9zNYtT+GRbujAJjdoza1/DysHGj+UH3F8PHHH6PRaBgwYAAZGRkAODg4MGLECGbMmGH1AIUQwqoUBX4YCZcPgJMH9FkLzqVUF/PXuRtM2ngcgLGtg3i+ZvaRGQorVXUMBoOBPXv2UKNGDXQ6HefOmdrqVq5cmRIlClcHDkvdfyspMjJS6hiEKGx2fAQ7poGdPfTfCJVyHgbnYc5fT+LFL/7idko6nWv78mnP2gV6eBObVz47OTlx6tSpPB9Er6CRymchCqHjG2D9YNNyx8+g7kDVRdxOTufFL/Zw/sZdgv09WPVyQ5wctFYO1LrUnq9U1zFkjq4qhBCFyqWDsHGEabnRSIuSQrrByIgVBzl/4y7lPZxZ0L9egU8KllCdGDJHV/3pp5+IjY0lMTExy0MIIQqc25dgdW/ISIWgdtB6iuoiFEVh8o8n+OvcTUo6alk4sB5lXNUPmVEYFJrRVQsKaa4qRCGTdhdW9YKka1D2KXhpIdip/5Uftiealfti0Ghgbq9gqvkU3VvIqusYdu7c+dDX82uQvbwmdQxCFAJGo2lgvNM/Q8ky8PI28PBXXcz203EMXRyOUYF3OlTj5eZP2CBY27FpB7f09HSmTJnCV199RZUqVSwOUggh8sQf/zMlBa0Oeq20KCmcvnqHUSsjMCrQs54fw5oV/YY3quoYHBwcOHr0qK1iEUII64lYDnvmmpY7zwe/BqqLuJGkZ+iScJL0GYRU8mRql6cLdLNUa1Fd+dyvXz8WLVpki1gKBRldVYhCIHo3bBptWm7+FtTsrroIfYaBV5cd5NKtFAJKl+CrfnVxtC8cU3M+LtV1DKNGjWLp0qVUqVIlxzmfZ8+ebdUACyqpYxCigIo/D9+0gpRb8NSL8NK3YKfuhK4oCm+sPcKGiMu4Otnz/f81IbCsi40Ctj2bD6J3/Phx6tSpA0BkZGSW14rDJZYQogBLSYCVPU1JwbcOdPlSdVIA+GLHOTZEXEZrp+HLvnULdVKwRKGZ81kIIR7KkAHrBsGNSHArD71XgYP6kU63HI9l1q+nAfhfp6doWkX9TG6FncU3zM6ePcuvv/5KSkoKYLr0EkKIfKEo8MtbcH47OJSE3qvB1Vt1Mccv32bMmiMADGocQP+GFa0daaGgOjHcvHmTZ599lqCgIDp06EBsbCwAQ4cO5Y033rB6gEII8Uj7F8CBRYAGXvoGfGqqLuJaYipDl4STkm6geVAZ3n1e/fSeRYXqxDBmzBgcHByIiYnJMqJqz5492bJli1WDK4ikVZIQBcyZ32HL26bl1u/Dk8+rLiIlzcCwJQe4lqinSlkX5vUJxl5bPFog5UR1HcNvv/3Gr7/+SoUKFbKsr1KlygOn5yxKQkNDCQ0NNdfyCyHyUdwpU72CYoTgftD4NdVFGI0KY9ce5tjl23iWdGTRwPq4OamftKcoUZ0S7969m+PcC/Hx8eh0RXNAKSFEAXT3BqzsAWl3oGJTeH6ORVNzzt4ayS/Hr+Kg1fBVv7r4ly4ec8s8jOrE0KxZM5YuXWp+rtFoMBqNzJw5k2eeecaqwQkhRI4y9LC6LyTEQKlK0HMZ2DuqLub7iEvM234WgOlda9Kgkqe1Iy2UVN9KmjlzJs8++ywHDhwgLS2Nt956ixMnThAfH8+ePXtsEaMQQvxLUeDH1+Di36BzN03NWUL9Cf3ghXjGrz8GwIiWlelWt8Ij3lF8WDRRT2RkJE2bNqVz587cvXuXrl27EhERQeXKlW0RoxBC/GvXJ3B0NWi00GMJlAlSXcTF+GSGLz1ImsFI26fK8WabqjYItPBSfcUA4O7uzjvvvGPtWIQQ4uFO/gDbppqWO8yCyupvX99JTWfoknBu3k3jKV835vSsjZ2djNpwv+LbHksIUbhcPgQbXjEth7wK9YeqLsJgVHhtVQSR15Io66pj4cB6lHC06PdxkSaJQSXpxyBEPki8Aqv7QEYKBLaGNh9aVMyHP59i++nr6Ozt+GZAPXzc1Q+ZURyoHl1VmMjoqkLkkbS78G07uHoUylSDob+Bk/q/uZX7Ypj4vamyeX6fOjxf08fakRZYas9XcsUghCi4jEbYMNyUFEp4QZ/VFiWFPWdv8N4PxwF4o3VQsUoKlpDEIIQouLZNgX9+Aq0j9FoBpQJUF3H+ehIjlh8kw6jQpbYvI1sFWj/OIkZVYti8eTPDhg3jrbfe4p9//sny2q1bt2jVqpVVgxNCFGOHV8LuOablTvPAv6HqIm7dTWPI4nASUzMI9vdgxks1Zd6YXMh1Yli5ciWdOnXi6tWr7N27l+DgYFasWGF+PS0tjZ07d9okSFtxcXHJ8rCzs+OTTz7J77CEEBf+MnViA2g2Dmr1VF1EWoaRESsOEn0zmfIezizoXw8nB62VAy2act1Oa9asWcyePZvXXjP9Z61du5YhQ4aQmprK0KHqm40VBElJSeblK1eu4O/vT9euXfMxIiEE8VGm4S6M6VC9Mzyjvs+Uoii8u/EYf5+Px0Vnz7eD6lPGVcZyy61cJ4YzZ87QsWNH8/MePXpQpkwZOnXqRHp6Oi+++KJNAswrK1eupFGjRlSqVCm/QxGi+Eq9fW9qznjwDYYuX1k0Nec3u86z9sAl7DTwee9gqnq72iDYoivXR9zNzY1r165lWffMM8/w008/8eabb/L5559bFEBSUhKTJ0+mXbt2eHp6otFoWLx4cY7b6vV6xo8fj6+vL87OzoSEhLB161aL9vtfy5YtY8CAAVYpSwhhAfPUnKfB1Rd6rQJH9SOd/nbiKtN/MdWBTnqhOs88WdbKgRZ9uU4MDRo04Jdffsm2vkWLFmzatIlPP/3UogBu3LjBlClTOHXqFLVq1XrotoMGDWL27Nn07duXuXPnotVq6dChA7t377Zo35mOHj1KZGQk3bt3f6xyhBCP4dcJcG4bOJQwNUt1U9+k9Pjl27y++jCKAv0a+jOocYD14ywGcp0YxowZg5OTU46vtWzZkk2bNln0i9vHx4fY2FguXLjArFmzHrjd/v37Wb16NdOnT2fWrFkMHz6cbdu2UbFiRd56660s2zZt2hSNRpPj4913381W9rJly+jYsSMeHh6q4xdCWMH+b0zTc6KBrt+Az8N/JObkWmIqw5YcICXdQLMqXkzu+JS0QLJQrusYWrRoQYsWLR74+jPPPJNlPoYZM2bw6quvPvJkq9Pp8PZ+9KTd69evR6vVMnz4cPM6Jycnhg4dysSJE7l48SJ+fn4Aqq4gjEYjK1eu5Kuvvsr1e4QQVnT2d/hlvGn5uclQ7QXVRaSkGXh56QGuJqZSuUxJ5vWpg0MxnprzcdnsyE2bNo34+HirlRcREUFQUFC27twNGjQA4PDhwxaV+8cff5Cenk779u0fN0QhhFpx/8C6waAYoHZfaDJadRFGo8Ib6w5z9NJtSpVw4NtB9XF3Lt5Tcz4umw0raO0hmGJjY/HxyX7PMXPdlStXLCp32bJl9OrVC3v7hx8KvV6PXq83P09MTLRof0KIezKn5tQngn9jeOFTi6fm3HzsKo5aOxYMqEfF0iWtH2sxU2jGm01JSclxTunMeo+UlBSLyr1/mtKHmT59Ou+//75F+xBC/EeGHtb0g4QLpmEuei63aGrODYfun5qzBvUDZGpOayg0N+GcnZ2z/GLPlJqaan7dliZMmMDt27fNj4sXL9p0f0IUWYoCm16HmL3/Ts1ZsrTqYsKj43n7O9NoqaHPVOYlmZrTagrNFYOPjw+XL1/Otj42NhYAX19fm+5fp9Oh0+mYP38+8+fPx2Aw2HR/QhRZuz6BI6tMU3N2D4My6qfVjLmZzCvLTFNztn/amzday9Sc1lRorhhq165NZGRktnv7+/btM7+eF0JDQzl58iTh4eF5sj8hipTjG7JOzRn4rOoibqekM2RJOPF306hR3p3ZPWRqTmuzWWJo1qyZVW/vdOvWDYPBwIIFC8zr9Ho9YWFhhISEmJuqCiEKqEsHYOMI03LDUIum5swwGBm58hBn45LwdnNi4cB6ODvKwHjWlqtbSWpa4GQ2J928eXOu3zNv3jwSEhLMLYs2bdrEpUuXABg1ahTu7u6EhITQvXt3JkyYQFxcHIGBgSxZsoTo6GgWLVqU6309LrmVJIQFEmJgVS/ISIWg9tBmquoiFEXhf5tOsOvMDZwdtCwcWI9ybjl3uhWPJ1dTe9rZ2T2yB6GiKGg0GotOmAEBAVy4cCHH16KioggICABMFc2TJk1i+fLl3Lp1i5o1azJ16lTatm2rep+PS6b2FCKXUhPh27YQdxLK1YAhW0DnorqYxXui+N+mk2g08FW/urR96tEdY4WJ2vNVrhKDmnkWHtY7uii4/4ohMjJSEoMQD2PIgFU9Tb2bXbzh5T/AXX3roe2n4xi6OByjAhPaP8krLSrbINiiyyaJQWQnVwxC5MLmN01jINk7w5BfTENpq3T66h1e+vIvkvQZ9KhXgY9kFjbV1J6vCk2rJCFEIbPv638HxnvpG4uSwo0kPUMWh5OkzyCkkicfdKkhSSEPSGJQaf78+VSvXp369evndyhCFFyRv8GWt03Lz/0PqnV86OY5SU03MHzpAS4npBBQugRf9auLo72csvKC3EqykNxKEuIBrh43VTanJUFwP+g0T/UYSIqiMHrNYX44fAU3J3u+D21C5TLqK6yFidxKEkLknzvXTFNzpiVBQDN4fo5FA+N9vu0sPxy+gr2dhi/71ZWkkMcsSgwZGRn8/vvvfP3119y5cwcwjW6alJRk1eCEEIVIegqs7g2Jl6B0IPRcZtHAeJuOXGH21kgApnZ5miaBXtaOVDyC6rGSLly4QLt27YiJiUGv19O6dWtcXV356KOP0Ov1RX7CG+ngJkQOjEb4/lW4fBCcS5kGxnMupbqYiJhbjFt3BIBhTSvRu4G/tSMVuaD6iuH111+nXr163Lp1K8uQFy+++CJ//PGHVYMriGSsJCFysP0DOLkR7Byg5woorb6fweWEFF5eehB9hpFnnyzLhA7VrB+nyBXVVwy7du3ir7/+wtEx6yViQEBAjqOfCiGKuMMrTSOmAnT6HAKaqC4iSZ/B0MXh3EjS86S3K3N7B6OVgfHyjeorBqPRmONtlEuXLuHq6mqVoIQQhUT0bvjxNdNys3FQu7fqIgxGhddXRfDP1TuUcdXx7aD6uOgKzYwARZLqxNCmTRs+/fRT83ONRkNSUhKTJ0+mQ4cO1oxNCFGQ3TxnmoXNmA7Vu8Az71hUzPTNp/jjnzh09nZ8M6Aevh62nXRLPJrqtPzJJ5/Qtm1bqlevTmpqKn369OHMmTN4eXmxatUqW8RYoEjlsxDA3Zuwojuk3ILy9eDFr8BOfSPHlftiWLg7CoBPetSitp+HlQMVlrCog1tGRgarV6/m6NGjJCUlUadOHfr27Wvz6TULEungJoqttGRY0hEuHwB3fxj2O7iWU13MnrM3GPjtfjKMCm+0DmLUs1VsEKwA9ecr1VcMqampODk50a9fP4sCFEIUYoYMWD/YlBScS0G/7yxKCueuJzFi+UEyjApdavsyslWgDYIVllJ97Ve2bFkGDhzI1q1bMRqNtohJCFEQKQr8PBYit4C9E/ReA2WCVBdzM0nP4LBwElMzqFuxFDNktNQCR3ViWLJkCcnJyXTu3Jny5cszevRoDhw4YIvYhBAFyc6P4NAS0NjBS4vAP0R1EanpBl5eeoCY+GT8PJ35un9dnBxkas6CRnViePHFF1m3bh3Xrl1j2rRpnDx5koYNGxIUFMSUKVNsEWOBIqOrimLp4BLYMd203OFjqPaC6iKMRoU31h7hUEwC7s4OhA1qgJeLzsqBCmuwyuiqJ0+epG/fvhw9erTYtNaRymdRbJzeAqv7gGKA5m9Cq3ctKmbGL//w1c5zOGg1LBsaQsMnSls5UPEgeTa6ampqKmvXrqVLly7UqVOH+Ph43nzzTUuLE0IURJcOwLpBpqRQu5/FfRVW7Y/hq53nAJjZraYkhQJOdaukX3/9lZUrV7Jx40bs7e3p1q0bv/32G82bN7dFfEKI/HLjrKmvQkYKBLaGjp9aNIT2zsjrvLvxOACjn6vCi8Hq53wWeUt1YnjxxRd54YUXWLp0KR06dMDBwcEWcQkh8tOda7C8K6TEm6bk7L4YtOr/1k/FJhK64hAGo0LXOuV5XfoqFAqqE8O1a9dkTCQhijL9HVjZHRIuQKlK0Gcd6NRPlHMtMdU8X3PDJzyZ0VWapRYWuUoMiYmJ5goLRVFITEx84LZSEStEIZaRBmsHQOwRKOEF/TeASxnVxdzVZzBkcTixt1OpXKYkX/erJ/M1FyK5SgylSpUiNjaWsmXL4uHhkWPWVxQFjUZTbFolCVHkKAr8OArObQOHktB3HXg+oboYg1HhtVURnLiSSOmSjoQNaoB7CbnlXJjkKjFs27YNT09PALZv327TgIQQ+eSP9+HoatBooccSKF/HomKm/nTy39FSB9bDv3QJKwcqbC1XiaFFixbm5UqVKuHn55ftqkFRFC5evGjd6AogGV1VFEn7FsDuOablTp9DldYWFfPt7igW/xUNwJyetanjr356T5H/VHdw02q15ttK97t58yZly5YtNidM6eAmioyTP8DagYBi6rzW3LL+SL+duMoryw+iKDCh/ZO80kL99J7CNmzewS2zLuG/kpKScHJyUlucECI/XfgLvnsZUKDeENMsbBY4eimB11cfRlGgT4g/w5urr5sQBUeum6uOHTsWMM3YNmnSJEqU+Pe+ocFgYN++fdSuXdvqAQohbCTuFKzqBQY9PPmCaQwkC5qTXrqVzJDFB0hJN9AiqAxTOj0lzVILuVwnhoiICMB0xXDs2DEcHR3Nrzk6OlKrVi3GjbPs14YQIo/dvgzLX4LU2+AXAi8tBDv1o5wmpqYzZHE4N5L0POntyrw+wdhrpVlqYZfrxJDZGmnw4MHMnTu3SNxXP3z4MKGhoRw7dgwvLy8mTpzIsGHD8jssIWwrJQFWdIPEy+AVBL1Xg4P62RfTDUb+b/khIq8lUc5NR9jg+rg6SbPUokB1ag8LCysSSQGgf//+tG3bloSEBNavX8+YMWM4depUfoclhO2kp8LqvhB3Ely8TTOwlfBUXYyiKEzccIzdZ29QwlHLooH18XEvPlP7FnWqh8QAOHDgAGvXriUmJoa0tLQsr23YsMEqgeWF6OhoevfujZ2dHXXq1KFatWr8888/VKtWLb9DE8L6jEb4/hW4sBscXaHfevDwV12MoihM/vEE6w5ewk4D8/oE83R5dxsELPKL6iuG1atX07hxY06dOsX3339Peno6J06cYNu2bbi7q/9yJCUlMXnyZNq1a4enpycajYbFixfnuK1er2f8+PH4+vri7OxMSEgIW7duVb3PTKNGjWL58uVkZGSwf/9+YmJiaNiwocXlCVFgKQr8OgFObgQ7B+i1ArxrWFCMwvubTrJ07wU0GpjZrRatnlQ/57Mo2FQnhmnTpjFnzhw2bdqEo6Mjc+fO5Z9//qFHjx74+6v/9XHjxg2mTJnCqVOnqFWr1kO3HTRoELNnz6Zv377MnTsXrVZLhw4d2L17t+r9ArRv356lS5fi5ORE48aN+eijj/Dx8bGoLCEKtL8+g31fmZZf/AqeaPHw7XOgKAof/HzK3IHto6416VZXhtAukhSVSpQooURFRSmKoiienp7K0aNHFUVRlJMnTyre3t5qi1NSU1OV2NhYRVEUJTw8XAGUsLCwbNvt27dPAZRZs2aZ16WkpCiVK1dWGjVqlGXbJk2aKECOj3feeUdRFEW5efOm4urqqqxbt07JyMhQjhw5ovj4+CgHDx7MVdy3b99WAOX27duqP7MQeerIGkWZ7GZ67PnMoiKMRqMy7eeTSsXxPykVx/+krNx3wcpBCltSe75SfcVQqlQp7ty5A0D58uU5ftw0AUdCQgLJycmqE5NOp8Pb2/uR261fvx6tVsvw4cPN65ycnBg6dCh79+7NMhzH7t27URQlx8cHH3wAwLlz5yhZsiTdunVDq9VSs2ZNGjduzM6dO1V/BiEKrHPbYeP/mZYbhkLjUaqLUBSFWb+e5us/zwPwQZen6d1A/d0BUXioTgzNmzc339fv3r07r7/+Oi+//DK9e/fm2WeftXqAmSIiIggKCsrWIqpBgwaAqempGkFBQSQnJ/PDDz+gKAonT55k165d1Kih/r6rEAXS+R2wpj8Y0+GprtDmA4uKmbM1ki92mKblfL/TU/RrWNGKQYqCSHWrpHnz5pGamgrAO++8g4ODA3/99RcvvfQS775r2SThuREbG5vj/f/MdVeuXFFVnru7O2vXrmX8+PH069cPT09Pxo4dy3PPPZfj9nq9Hr1eb37+sDkphMhX6anwxxT4e77peUAzU72CnfqOZ3N/P8Nn284C8N4L1RnYOMCKgYqCSnViyBx+G8DOzo63337bqgE9SEpKCjqdLtv6zPGZUlJSVJfZtm1b2rZtm6ttp0+fzvvvv696H0LkqavHYMNwUz8FgLqDoe2HYJ/9b+dR5m07w5zfIwF49/lqDGlayZqRigIs1zO45ZatOr85Oztn+cWeKfPqxdnZtp1rJkyYYB4vCkzHxM/Pz6b7FCLXjAbYOw+2fQCGNChZBjrPh6Dc/fD5ry92nOXj30xJ4e32TzKsmQyKV5zkKjE8aNa2+yk2nsHNx8eHy5cvZ1sfGxsLgK+vr032m0mn06HT6WQ+BlHwJMTA9yNMHdcAqj4PnT6Dkl4WFbfgz3PM3HIagDfbVuVVGT672MlVYigIs7bVrl2b7du3Z5l/GmDfvn3m1/NCaGgooaGh5vHNhcg3igJH18DmN0GfaJqOs/0MCO5v0SipAAt3nWfa5n8AGPNcEKHPBFozYlFIqJ7BLb9069aNjz/+mAULFphHcdXr9YSFhRESEpJnt3XkikEUCMnx8NMYU09mgAoNoOvXFs3RnGnxnig++Nk0Vthrz1bh9eeqWCFQURhZNFbSrl27+Prrrzl//jzr1q2jfPnyLFu2jEqVKtG0aVPV5c2bN4+EhARzy6JNmzZx6dIlwDRshbu7OyEhIXTv3p0JEyYQFxdHYGAgS5YsITo6mkWLFlnyMSwiVwwi353bDhtHwJ1YsLOHFm9D0zGgtejPGYBle6P53yZThXXoM5UZI0mheFPbg279+vWKs7OzMmzYMEWn0ynnzp1TFEVRPv/8c6V9+/Zqi1MURVEqVqz4wJ7Kmb2sFcXU03ncuHGKt7e3otPplPr16ytbtmyxaJ+PS3o+izyXlqwom8f/24v5s7qKcil3PfUfZsXfF8w9mqdtPqkYjUYrBCsKErXnK9VzPgcHBzNmzBgGDBiAq6srR44c4YknniAiIoL27dtz9epVa+euAuX+W0mRkZEy57PIG7FHTM1Qr5vu/1N/GLSeCo4lHv6+R1gTHsP4744B8HKzSkzsUE1mXyuC1M75rPra8/Tp0zRv3jzbend3dxISEtQWV+jIrSSRp4wG2DMXtk8z9WB2KWdqhlql9WMXve7ARd7eYEoKg5sESFIQZqoTg7e3N2fPniUgICDL+t27d/PEE9LWWQiruXUBvn8VYv4yPX/yBej4GZQs/dhFbzh0ibe+O4qiwMBGFXnvheqSFISZ6j7yL7/8Mq+//jr79u1Do9Fw5coVVqxYwbhx4xgxYoQtYixQ5s+fT/Xq1alfv35+hyKKKkWBwyvhyyampODoAp2/gJ7LrZIUfjh8mXHrjqAo0K+hP//r9JQkBZGF6joGRVGYNm0a06dPN4+mqtPpGDduHFOnTrVJkAWR2nt2QuRKcjxseh1O/Wh67tfQNM6Rp3WGo9h05Aqvr47AqEDvBn582KUGdnaSFIo6tecr1YkhU1paGmfPniUpKYnq1avj4uJCSkqKzYemKCgkMQirO/s7bAyFpKumZqjPTIQmo8FOa5XiNx+LZdSqCAxGhR71KjCja01JCsWE2vOV+uEW73F0dKR69eo0aNAABwcHZs+eTaVKRX+QLbmVJKwuLdnUe3n5S6ak4FUVhv0Bzd6wWlLYcvwqr91LCi/VkaQgHi7XiUGv1zNhwgTq1atH48aN2bhxIwBhYWFUqlSJOXPmMGbMGFvFWWCEhoZy8uRJwsPD8zsUURRciYAFLWD/AtPzBq/AKzvBt7bVdrH15DVGrjxEhlGhS21fZnaTpCAeLtetkt577z2+/vprnnvuOf766y+6d+/O4MGD+fvvv5k9ezbdu3dHq7XOrxshijyjAXbPhh0zwJgBLt7QZT4E5jwfiKW2/XON/1txkAyjQsdavnzcvRZaSQriEXKdGNatW8fSpUvp1KkTx48fp2bNmmRkZHDkyBFp0SCEGvFR8P0rcNE0ACTVO8MLn0IJz4e+TY3Y2yks23uBhbuiSDcoPF/Dhzk9amGvtfjusShGcp0YLl26RN26dQF4+umn0el0jBkzRpKCELmlKBCxHLa8DWlJ4OgKHWZBrV4Wj4b6XxExt/h2TzS/HIslw2hqV9L+aW8+7VVbkoLItVwnBoPBgKOj479vtLfHxcXFJkEVZDK6qrDI3RumZqj//GR67t/Y1Ay11OPPn5xuMLLl+FW+3RNFREyCeX2DSp4MaRJAm+reUqcgVMl1c1U7Ozvat29vnl5z06ZNtGrVipIlS2bZbsOGDdaPsgCS5qoi1yJ/gx9C4W4c2DlAq3eg8WuP3eLo1t00VoXHsGzvBWJvm2YydNTa0bGWL4ObBPB0eRmyRZjYbKykgQMHZnner18/9dEJUZyk3YXfJsGBe8PCl3kSui4An1qPVeyZa3cI+yuaDYcukZpuBMDLxZG+IRXp29Cfsq5Ojxu5KOZynRjCwsJsGYcQRcvlg6bRUG+eNT1v+H/w7GRwsOykbTQq7DxznbA90fwZed28vrqPG0OaVqJjLR909tIqUFiH5TN7CCGyM2T82wxVMYCrL3T5Aio/Y1FxyWkZfHfoMmF7ojh//S5gqqduXa0cQ5pWIqSSpzQAEVYniUEIa7l5ztQM9dK9zo9PvQjPz7aoGerlhBSW/hXNqv0xJKZmAOCis6dnfT8GNgrAv/TjzcMgxMNIYlBJWiWJbBQFDi2BLRMh/S7o3OH5j6FGd1XNUBVF4VDMLb7dHc2WE1cx3GtuWrF0CQY1DqBb3Qq4OjnY6lMIYWbxIHrFnbRKEoBpNNQfQuH0ZtPzgGbQ5Uvw8Mt1EWkZRjYfiyVsTxRHLt02r29cuTRDmlTimSfLSm9l8VhsPoObEOKeG2dhZQ+IPwdaR2g1CRqNBLvcdSSLv5vGyn0XWLr3AnF39AA42tvRpbYvg5tUopqP/OAQ+UMSgxCWOL8T1g6A1ARw94deK8CnZq7eevrqHcL2RPF9xGX0GabmpmVcdQxoWJE+If6UdtHZMHAhHk0SgxBqHVwMP79hGvyuQgPotRJcyjz0LUajwvbTcXy7J4o9Z2+a19co786QpgE8X8MXR3sZskIUDJIYhMgtowG2vgd755me1+gOneY9tG9Ckj6D7w5eImxPFNE3TTMe2mmg3dPeDGlSiboVS0lzU1HgSGIQIjf0d+C7YRC5xfT8mXeg+ZsPbHV0MT6ZJX9Fsyb8Inf0puamrk729G7gz4BGFalQSpqbioJLEoMQj5JwEVb1gmvHwd7J1GHt6ZeybaYoCvuj4gnbE81vJ69yr7UpT3iVZHCTALrWqUBJnfzJiYJPvqUqST+GYubSAVjV2zQAXsmy0HsVVKiXZRN9hoGfjsTy7Z4oTlxJNK9vVsWLIU0q0SKojIxuKgoV6cdgIenHUAwc/w42/h9kpEK5p6H36iz9E1LSDCz/+wJf/3meG0mm5qY6ezu61qnA4CYBBJVzza/IhchC+jEI8bgUBXbOhB3TTM+D2sFLC0FnOtGnphtYsS+GL3ecMycEbzcn+jeqSJ8G/pQq6figkoUoFCQxCHG/9FT4cSQcW2d63mgktJ4CdlpS0w2s2m9KCJkd0iqUcmZUq0C61qmAg8yQJooISQxCZEqKg9V94dJ+sLOH5z+BuoPQZxhYsy+a+dvPci3RlBDKezgzslUgL9WpIP0PRJFTrBPDiRMnGDFiBIcPH6ZChQp88cUXtGzZMr/DEvnh2klY2RNux4CTO/RYht6/KWv/vsAX28+aZ0jzcXdiZKtAutf1k4QgiqximxjS09Pp3Lkzr732Gtu3b2f79u1069aN06dPU7p06fwOT+SlyN9g/RBIuwOelUnruZr10U7MW7ODK/cSgrebE6HPVKZHfT+ZEEcUecU2MZw+fZpbt27x2muvAfDcc88RHBzM999/z7Bhw/I5OpEnFAX2fQW/TgTFiLFiUzZWmcEnYZe4nJACQFlXHaHPBNKzvh9ODpIQRPGQ79fCSUlJTJ48mXbt2uHpaZqNavHixTluq9frGT9+PL6+vjg7OxMSEsLWrVst3vd/W+oqisKJEycsLk8UIoZ0+HksbHkbFCPn/brybNxrjP0phssJKZRx1fHeC9X5861nGNg4QJKCKFbyPTHcuHGDKVOmcOrUKWrVevgk6YMGDWL27Nn07duXuXPnotVq6dChA7t371a936pVq+Lh4cHs2bNJT09ny5Yt7Ny5k7t371r6UURhkZIAK7rBgW9R0DDfcRCtzrxE1K0MvFwceff5avz55jMMaVpJEoIolvI9Mfj4+BAbG8uFCxeYNWvWA7fbv38/q1evZvr06cyaNYvhw4ezbds2KlasyFtvvZVl26ZNm6LRaHJ8vPvuuwA4ODiwceNGfvjhB7y9vZkzZw49e/akQoUKNv28Ip/Fn0dZ1BrO7yAFJ15OG8usxDaULqljYocn+fOtZxjW7AmcHSUhiOIr3+sYdDod3t7ej9xu/fr1aLVahg8fbl7n5OTE0KFDmThxIhcvXsTPz9QrNbdXEDVr1mTnzp3m540bN6Zfv34qP4EoLAxRu8lY1RddWgJXFE+GpY0j1rkK45tXZkCjijKOkRD3FJq/hIiICIKCgrJ1527QoAEAhw8fNieG3Dp69ChBQUEYjUbmz5+P0WikXbt2VotZFAxGo8KRn77g6UPvoSODw8YnGKd9mxfb1GVg4wBcJCEIkUWh+YuIjY3Fx8cn2/rMdVeuXFFdZlhYGGFhYRiNRlq3bs3GjRsfuK1er0ev15ufJyYmPnBbUTAYjQpbjl8h8edJ9NJ/B8CvNORc04/5vvmTuDo55HOEQhRMhSYxpKSkoNNln/LQycnJ/Lpac+bMYc6cObnadvr06bz//vuq9yHyntGo8NvJq3y59Rgj4mfSSxsOwD6/YTTqM422zjJ1phAPU2gSg7Ozc5Zf7JlSU1PNr9vShAkTGDt2rPl5YmKi6ltXwrYURWHryWt8+vsZbsZGs8jxY57WRpOhcSDt+c8Iqdcnv0MUolAoNInBx8eHy5cvZ1sfGxsLgK+vr033r9Pp0Ol0Mh9DAaQoCn+ciuPTPyI5fjmRpzXn+VH3CeU0tzCW8MK+1wrs/Rvmd5hCFBqFJjHUrl2b7du3k5iYmKUCet++febX80JoaCihoaHm8c1F/lEUhR2nrzPn90iOXroNQCfHA3xiPx8Hox7KVMOuzxooVTGfIxWicMn3fgy51a1bNwwGAwsWLDCv0+v1hIWFERISkme3debPn0/16tWpX79+nuxPZGdKCHG8+MVfDF4cztFLt3F2sGNx4C4+s5ttSgqBz8HQ3yQpCGGBAnHFMG/ePBISEswtizZt2sSlS5cAGDVqFO7u7oSEhNC9e3cmTJhAXFwcgYGBLFmyhOjoaBYtWpRnscoVQ/5RFIXdZ28wZ2skh2ISAHBysGNwiC+vJ8/D6eRa04Yhr0KbD0FbIL7eQhQ6BWJqz4CAAC5cuJDja1FRUQQEBACmiuZJkyaxfPlybt26Rc2aNZk6dSpt27bNw2hNZGpP2zEYFVLSDSSnZZCaZiQl3cClW8l8ueMcBy7cAkxTaPZvWJFX63vg9fMQiNkLGi20/wgavJzPn0CIgkXt+apAJIbC5P7K58jIyGKXGIz3Ttop6QZS0h78b3K6gdTM55mv3XuenGYgNf0/y2n3EkG6kTSD8YH7d7S3o2+IPyNaVKas/gKs7AG3okHnBt0XQ+CzeXYshCgsJDHkEUuuGM7G3WHV/osoCiiYDvv9Rz/zv0K5b/392ynm7TK3um875WHb3nvNvJx1X/e/R//fk/5/Tvj6jAeftK1No4ESDlqcHbWUcLSn1ZNlGdGyMuXcnODsH7BuMOhvQ6kA6L0Gyj6ZZ7EJUZioPV/JTdg8dPFWCot2R+V3GFbjfO+kneXf/6570PqH/FvCUYuTgxadvR0ajSb7jvd/A7+MB8UA/o2g5wooKZMrCWEtkhhUepx+DP6eJXi1RWXA9Gs485RnWtZkW8+9k6Lm30U0aO5bvm/9fSfQB5WXUxn3v1cD6BzsKHHvBO3kkHmitsfZ0Q6nzOV7J207uxxO2rZkyDBNqrP/a9PzWr2h41ywl57MQliT3EqykFQ+57HU26bpN8/+bnr+7GRoOgZyuqIQQmQht5JE4ZShh7s34O51SL5xb/nGveXrcGEvxJ8De2fougCqd8rviIUosiQxCNvIPNFnntjv3rxv+f6T/r1H2p1Hl+nqA71XgW+w7eMXohiTxKBSsR0rKSMt64k9+eZ9J/nr957ft6y3YFhyO3soURpKlvn335JeUMILXMrCk8+bngshbErqGCxkUR1D/Hk4viHn1x54rzyH9Wq2feD2Oa1TTPMhJ98w/cI339a5aWoWqpZG+++JvaTXfctlTK2ISpbJ+pqTh9QZCGEDUsdQkN08B9um5ncUltNo7/slX/q+k7zXfxLAvV/8Th5gV2iG4xJC3COJIS+5+UJw/xxeeMBFW46rH7Ttgy78VG7v5Jb1V/z9v+rlRC9EsSCJQaXHqmMo9xR0nmf9oIQQwoqkjsFC0o9BCFFYqD1fyX0BIYQQWUhiEEIIkYUkBiGEEFlIYhBCCJGFJAaVZM5nIURRJ62SLCStkoQQhYX0fM4jmfk0MdGCMYGEECIPZZ6ncnsdIInBQnfumEYD9fPzy+dIhBAid+7cuYO7u/sjt5NbSRYyGo1cuXIFV1fXnKefLIQSExPx8/Pj4sWLcntMBTlulpHjpp6lx0xRFO7cuYOvry92uRjWRq4YLGRnZ0eFChXyOwybcHNzkz9UC8hxs4wcN/UsOWa5uVLIJK2ShBBCZCGJQQghRBaSGISZTqdj8uTJ6HS6/A6lUJHjZhk5burl1TGTymchhBBZyBWDEEKILCQxiMei1+sZMmQI/v7+uLm50bBhQ/bu3ZvfYRU4169f5/nnn6dkyZJUrVqVP/74I79DKvDku/V49u7di52dHR988IHq90piEI8lIyODgIAAdu/eTUJCAqNHj6Zjx44kJSXld2gFSmhoKN7e3ly/fp1Zs2bRo0cP4uPj8zusAk2+W5YzGo2MGTPG4jHdpI5BWJ2vry+bNm2ibt26+R1KgZCUlISnpyfnz583931p2bIlAwcOZPDgwfkcXeEi363c+eqrrzh16hS3b98mMDCQd999V9X75YqhiEhKSmLy5Mm0a9cOT09PNBoNixcvznFbvV7P+PHj8fX1xdnZmZCQELZu3WqVOM6cOUN8fDyBgYFWKS8/WPtYnjlzBhcXlywdImvUqMGJEyds+THynK2/g0Xhu/VftjhmN2/e5NNPP+X999+3OC5JDEXEjRs3mDJlCqdOnaJWrVoP3XbQoEHMnj2bvn37MnfuXLRaLR06dGD37t2PFUNKSgr9+vVjwoQJqnpZFjTWPpZJSUnZeqm6ubkVuVsitvwOFpXv1n/Z4pi98847jB49Gg8PD8sDU0SRkJqaqsTGxiqKoijh4eEKoISFhWXbbt++fQqgzJo1y7wuJSVFqVy5stKoUaMs2zZp0kQBcny88847WbZNS0tTnn/+eaVPnz6K0Wi0/gfMQ9Y+locOHVJKlSqV5b0jR45U3njjDdt8gHxii++gohSt79Z/2eK7VqdOHSUjI0NRFEUZOHCgMnXqVNVxyRVDEaHT6fD29n7kduvXr0er1TJ8+HDzOicnJ4YOHcrevXu5ePGief3u3btRFCXHx/0tHYxGI/3790ej0bBkyZJCP6igtY9llSpVSEpK4vLly+btjh8/zlNPPWX94PORLb6DRe279V/WPmY7d+7k9OnTlC9fHm9vb9asWcNHH32kui5LEkMxExERQVBQULZbGw0aNADg8OHDqst85ZVXiI2NZd26ddjbF59xGXN7LF1cXOjcuTOTJ08mJSWFn376iaNHj9K5c+e8DrlAUPMdLK7frf/K7TEbPnw4Z8+e5fDhwxw+fJhOnToRGhrKnDlzVO2v+B7pYio2NhYfH59s6zPXXblyRVV5Fy5cYOHChTg5OeHl5WVe/8svv9CsWbPHC7aAU3Msv/jiCwYOHEjp0qWpUKECa9aswdPTM89iLUhye9yK83frv3J7zEqUKEGJEiXMrzs7O+Pi4qK6vkESQzGTkpKS4zgrTk5O5tfVqFixYq5nhSpq1BzLMmXKsHnz5jyLrSDL7XErzt+t/7L07/ZBLZweRW4lFTPOzs7o9fps61NTU82vi9yRY2kZOW7q5fUxk8RQzPj4+BAbG5ttfeY6X1/fvA6p0JJjaRk5burl9TGTxFDM1K5dm8jISPPk4Jn27dtnfl3kjhxLy8hxUy+vj5kkhmKmW7duGAwGFixYYF6n1+sJCwsjJCQEPz+/fIyucJFjaRk5burl9TGTyuciZN68eSQkJJhbKGzatIlLly4BMGrUKNzd3QkJCaF79+5MmDCBuLg4AgMDWbJkCdHR0SxatCg/wy9Q5FhaRo6begXymKnuEicKrIoVKz6wp3JUVJR5u5SUFGXcuHGKt7e3otPplPr16ytbtmzJv8ALIDmWlpHjpl5BPGYyuqoQQogspI5BCCFEFpIYhBBCZCGJQQghRBaSGIQQQmQhiUEIIUQWkhiEEEJkIYlBCCFEFpIYhBBCZCGJQQghRBaSGIQQQmQhiUEIG2nZsiWjR4/O7zCEUE0SgxAF1OLFi1XP1SuENUhiEEIIkYUkBiFsKCMjg5EjR+Lu7o6XlxeTJk0yT3Cv1+sZN24c5cuXp2TJkoSEhLBjxw4AduzYweDBg7l9+zYajQaNRsP//vc/AJYtW0a9evVwdXXF29ubPn36EBcXl0+fUBRFkhiEsKElS5Zgb2/P/v37mTt3LrNnz2bhwoUAjBw5kr1797J69WqOHj1K9+7dadeuHWfOnKFx48Z8+umnuLm5ERsbS2xsLOPGjQMgPT2dqVOncuTIETZu3Eh0dDSDBg3Kx08pihqZj0EIG2nZsiVxcXGcOHECjUYDwNtvv82PP/7Ili1beOKJJ4iJickykftzzz1HgwYNmDZtGosXL2b06NEkJCQ8dD8HDhygfv363LlzBxcXF1t+JFFMyBWDEDbUsGFDc1IAaNSoEWfOnOHYsWMYDAaCgoJwcXExP3bu3Mm5c+ceWubBgwfp2LEj/v7+uLq60qJFCwBiYmJs+llE8SFzPguRD5KSktBqtRw8eBCtVpvltYf96r979y5t27albdu2rFixgjJlyhATE0Pbtm1JS0uzddiimJDEIIQN7du3L8vzv//+mypVqhAcHIzBYCAuLo5mzZrl+F5HR0cMBkOWdf/88w83b95kxowZ+Pn5AaZbSUJYk9xKEsKGYmJiGDt2LKdPn2bVqlV8/vnnvP766wQFBdG3b18GDBjAhg0biIqKYv/+/UyfPp2ff/4ZgICAAJKSkvjjjz+4ceMGycnJ+Pv74+joyOeff8758+f58ccfmTp1aj5/SlHkKEIIm2jRooXyf//3f8qrr76quLm5KaVKlVImTpyoGI1GRVEUJS0tTXnvvfeUgIAAxcHBQfHx8VFefPFF5ejRo+YyXn31VaV06dIKoEyePFlRFEVZuXKlEhAQoOh0OqVRo0bKjz/+qABKREREPnxKURRJqyQhhBBZyK0kIYQQWUhiEEIIkYUkBiGEEFlIYhBCCJGFJAYhhBBZSGIQQgiRhSQGIYQQWUhiEEIIkYUkBiGEEFlIYhBCCJGFJAYhhBBZSGIQQgiRxf8D32LPDGNVgEQAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 400x300 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "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='rel_err_elm')\n",
    "ax.loglog(conv_coeffs, res[:, -6], label='rel_err_swim')\n",
    "plt.xlabel('beta')\n",
    "plt.ylabel('Relative 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": "base",
   "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
}
