{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Advection equation without separating space and time with SWIM and ELM (no ODE solver)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "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 BasicAnsatz\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",
    "import time\n",
    "from scipy.stats import qmc"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Problem setup"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(25600, 2)\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAASEAAAEiCAYAAAC2iWS1AAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjguMywgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/H5lhTAAAACXBIWXMAAA9hAAAPYQGoP6dpAABCVUlEQVR4nO3deVhUZfsH8O/MwMywCIKsIgKaooiCgiC4oMkrmmubZBZKRmn6SyPXSsUl0Uq0tzTSFM0lLTM1M01JLANFIRWVXFiEV9lRkEXAmfP7Y+DEMAszwyws9+e6uLzmcJ9znoHh9pznec79cBiGYUAIIQbCNXQDCCEdGyUhQohBURIihBgUJSFCiEFREiKEGBQlIUKIQVESIoQYFCUhQohBURIihBgUJSEt4nA4iIqKUinW1dUVM2fOVPsc2dnZ4HA42LVrl9r76srIkSMxcuTIZuM4HA7mzZun+wbpwK5du8DhcJCdnW3opjRL1d9Ha0FJqJGGD9rly5e1crzExERERUXh0aNHWjkeIbpUVVWFqKgoJCQk6PW8Rno9WztXXV0NI6N/f6SJiYlYtWoVZs6cic6dO0vF3rp1C1wu/R9AWo+qqiqsWrUKAPR6JUV/BVokFAqlkpAyAoEAxsbGOm4Raa2qqqoM3YRWg5JQM2bOnAlzc3Pcv38fU6ZMgbm5OWxtbbFw4UKIRCKp2MZ9QlFRUVi0aBEAwM3NDRwOR6pPoWmfUGlpKRYuXIj+/fvD3NwcFhYWGDduHK5evapRu1U9XkJCAjgcDr7//nt8/PHH6NatG4RCIUaPHo27d+/KHHfbtm3o2bMnTExM4Ofnhz///FPttu3btw/u7u4QCoXw8fHBH3/8IRPz999/Y9y4cbCwsIC5uTlGjx6NCxcuSMVERUWBw+HI7Cuv/8bV1RUTJkzA+fPn4efnB6FQiB49euDbb7+V2f/GjRt49tlnYWJigm7dumHt2rUQi8UycUePHsX48ePRtWtXCAQC9OzZE2vWrJH5XIwcORKenp5ISUnBiBEjYGpqig8++AAzZsyAjY0N6urqZI49ZswYuLu7K/wZNlDl91FbW4sVK1bAx8cHlpaWMDMzw/Dhw3H27Fk2Jjs7G7a2tgCAVatWsZ/Xhs/ztWvXMHPmTPTo0QNCoRAODg544403UFJS0mwbm0O3YyoQiUQICQmBv78/PvvsM5w5cwYbN25Ez549MWfOHLn7vPDCC7h9+za+++47bNq0CTY2NgDA/qKbyszMxJEjR/Dyyy/Dzc0NBQUF+PrrrxEUFISbN2+ia9euarVZ3eOtX78eXC4XCxcuRFlZGT755BNMnz4dFy9eZGN27NiBt99+G4GBgViwYAEyMzMxadIkWFtbw9nZWaV2nTt3DgcPHsS7774LgUCArVu3YuzYsUhOToanpycASRIYPnw4LCwssHjxYhgbG+Prr7/GyJEjce7cOfj7+6v1s2hw9+5dvPTSS5g1axZmzJiBnTt3YubMmfDx8UG/fv0AAPn5+Rg1ahSePn2KpUuXwszMDNu2bYOJiYnM8Xbt2gVzc3NERkbC3Nwcv//+O1asWIHy8nJ8+umnUrElJSUYN24cXnnlFbz22muwt7eHmZkZvv32W5w6dQoTJkxgY/Pz8/H7779j5cqVSt+Pqr+P8vJyfPPNN5g2bRoiIiLw+PFj7NixAyEhIUhOToa3tzdsbW3x1VdfYc6cOXj++efxwgsvAAAGDBgAADh9+jQyMzMRHh4OBwcH3LhxA9u2bcONGzdw4cIFuf8ZqIwhrLi4OAYAc+nSJXbbjBkzGADM6tWrpWIHDhzI+Pj4SG0DwKxcuZJ9/emnnzIAmKysLJlzubi4MDNmzGBfP3nyhBGJRFIxWVlZjEAgkDp3VlYWA4CJi4tT+l5UPd7Zs2cZAEzfvn2Zmpoadvvnn3/OAGDS0tIYhmGY2tpaxs7OjvH29paK27ZtGwOACQoKUtoehpH8fAAwly9fZrfdu3ePEQqFzPPPP89umzJlCsPn85mMjAx224MHD5hOnToxI0aMYLetXLmSkfcRbvg9Nv65u7i4MACYP/74g91WWFjICAQC5v3332e3LViwgAHAXLx4USrO0tJS5phVVVUy53777bcZU1NT5smTJ+y2oKAgBgATGxsrFSsSiZhu3boxoaGhUttjYmIYDofDZGZmyhy/gTq/j6dPn0rFMAzDPHz4kLG3t2feeOMNdltRUZHMZ1jZe/3uu+9kfqaaoNsxFc2ePVvq9fDhw5GZmam14wsEArajWiQSoaSkBObm5nB3d0dqaqrOjxceHg4+n8++Hj58OACw7/Hy5csoLCzE7NmzpeJmzpwJS0tLldsVEBAAHx8f9nX37t0xefJknDp1CiKRCCKRCL/99humTJmCHj16sHGOjo549dVXcf78eZSXl6t8vsY8PDzY9wVIrkrd3d2lfo8nTpzAkCFD4OfnJxU3ffp0meM1vjp6/PgxiouLMXz4cFRVVeGff/6RihUIBAgPD5faxuVyMX36dBw7dgyPHz9mt+/btw+BgYFwc3NT+F7U+X3weDw2RiwWo7S0FE+fPoWvr6/Kn63G7/XJkycoLi7GkCFDAECjz2djlIRUIBQKZW6jrKys8PDhQ62dQywWY9OmTejVqxcEAgFsbGxga2uLa9euoaysTOfH6969u9RrKysrAGDf47179wAAvXr1koozNjaWShbNabo/APTu3RtVVVUoKipCUVERqqqq5PaH9O3bF2KxGLm5uSqfr7Gm7xGQ/T3eu3dPbhvltefGjRt4/vnnYWlpCQsLC9ja2uK1114DAJmfsZOTk1SyaBAWFobq6mr89NNPACSjpikpKXj99deVvhd1fx+7d+/GgAEDIBQK0aVLF9ja2uKXX35R+bNVWlqK+fPnw97eHiYmJrC1tWWTpCafz8aoT0gFPB5P5+dYt24dli9fjjfeeANr1qyBtbU1uFwuFixYILdTVNvHU/QemVZc/VdRP0TTjuEG2nyPjx49QlBQECwsLLB69Wr07NkTQqEQqampWLJkiczPWF6fEiC5OvPx8cHevXsRFhaGvXv3gs/nY+rUqWq3SZG9e/di5syZmDJlChYtWgQ7OzvweDxER0cjIyNDpWNMnToViYmJWLRoEby9vWFubg6xWIyxY8dq9PlsjJKQDqnTWXfo0CGMGjUKO3bskNr+6NEjtlNbHdo+nouLCwDgzp07ePbZZ9ntdXV1yMrKgpeXl0rHuXPnjsy227dvw9TUlL3aNDU1xa1bt2Ti/vnnH3C5XLbTteFq7dGjR1LzsBquEjTh4uIit41N25OQkICSkhIcPnwYI0aMYLdnZWWpfc6wsDBERkYiLy8P+/fvx/jx49n3pqydgGq/j0OHDqFHjx44fPiw1Geyace3os/rw4cPER8fj1WrVmHFihXsdnk/J03Q7ZgOmZmZAYBKM6Z5PJ7M/8g//PAD7t+/r9G5tX08X19f2NraIjY2FrW1tez2Xbt2qTUjPCkpSaoPITc3F0ePHsWYMWPA4/HA4/EwZswYHD16VGqIvaCgAPv378ewYcNgYWEBAOjZsycASA3xV1ZWYvfu3Rq9RwB47rnncOHCBSQnJ7PbioqKsG/fPqm4hquqxj/j2tpabN26Ve1zTps2DRwOB/Pnz0dmZiZ7S6eMOr8PeW29ePEikpKSpOJMTU0ByH5e5e0PAJs3b262naqgKyEdauiA/fDDD/HKK6/A2NgYEydOZJNTYxMmTMDq1asRHh6OwMBApKWlYd++fWr1t+jyeMbGxli7di3efvttPPvsswgNDUVWVhbi4uLUOqanpydCQkKkhugBsDN1AWDt2rU4ffo0hg0bhnfeeQdGRkb4+uuvUVNTg08++YSNGzNmDLp3745Zs2Zh0aJF4PF42LlzJ2xtbZGTk6PR+1y8eDH27NmDsWPHYv78+ewQvYuLC65du8bGBQYGwsrKCjNmzMC7774LDoeDPXv2aHRrZ2tri7Fjx+KHH35A586dMX78+Gb3Uef3MWHCBBw+fBjPP/88xo8fj6ysLMTGxsLDwwMVFRVsnImJCTw8PHDw4EH07t0b1tbW8PT0hKenJ0aMGIFPPvkEdXV1cHJywm+//abRVZ9cLRpba2cUDdGbmZnJxMobHoac4c01a9YwTk5ODJfLlRrilTdE//777zOOjo6MiYkJM3ToUCYpKYkJCgqSGm5VZ4heleM1DNH/8MMPUvsrOs/WrVsZNzc3RiAQML6+vswff/whc0xFADBz585l9u7dy/Tq1YsRCATMwIEDmbNnz8rEpqamMiEhIYy5uTljamrKjBo1iklMTJSJS0lJYfz9/Rk+n890796diYmJUThEP378eJn95bX92rVrTFBQECMUChknJydmzZo1zI4dO2SO+ddffzFDhgxhTExMmK5duzKLFy9mTp06xQCQek9BQUFMv379lP5svv/+ewYA89ZbbymNa0qV34dYLGbWrVvHuLi4sD/z48ePMzNmzGBcXFykjpeYmMj4+PgwfD5f6vP8v//9j3n++eeZzp07M5aWlszLL7/MPHjwQOGQvjo4DNOKex4J6SCOHj2KKVOm4I8//pCaRtARUBIipBWYMGEC0tPTcffu3ZbNPm6DqE+IEAM6cOAArl27hl9++QWff/55h0tAAF0JEWJQHA4H5ubmCA0NRWxsrMpVGNqTjveOCWlF6BqA5gkRQgyMkhAhxKA63O2YWCzGgwcP0KlTpw7ZCUiIvjAMg8ePH6Nr165KSxl3uCT04MEDlQtwEUJaLjc3F926dVP4/Q6XhDp16gRA8oNpeAaJEKJ95eXlcHZ2Zv/mFDFoEvrjjz/w6aefIiUlBXl5efjpp58wZcoUpfskJCQgMjISN27cgLOzMz766CO11u9quAWzsLCgJESIHjTX7WHQjunKykp4eXlhy5YtKsVnZWVh/PjxGDVqFK5cuYIFCxbgzTffxKlTp3TcUkKIrhj0SmjcuHEYN26cyvGxsbFwc3PDxo0bAUgq7Z0/fx6bNm1CSEiIrppJCNGhNjVEn5SUhODgYKltISEhMnVRWkokZpCUUYKjV+4jKaMEIjFNKCMEACAWAVl/AmmHJP+K5VexVEeb6pjOz8+Hvb291DZ7e3uUl5ejurpabgnNmpoa1NTUsK+bK5J+8noeoo7dRH75E3abg4UQUZM8MNbTsYXv4F8ikUjuelOEaIuxsbF2SxPfPAacXAKUP/h3m0VXYOwGwGOSxodtU0lIE9HR0VIFs5Q5eT0Ps/fKrhyQX/4Es/emIva1QS1ORAzDID8/n9anJ3rRuXNnODg4tHxO3M1jwPdyiu+XP5Bsn7pH40TUppKQg4MDCgoKpLYVFBTAwsJCYSHxZcuWITIykn3dMGzYlEjMYOnhNKXnX3Y4Df/xcACPq/kvtCEB2dnZwdTUlCZMEp1gGAZVVVUoLCwEIFkySWNiEfDzfOUxP88H+owHuOpfebWpJBQQEIATJ05IbTt9+jQCAgIU7iMQCCAQCJo99oXMEjyqUn579LCqDhcySzD0GfULxQOSW7CGBNSlSxeNjkGIqhr+Yy4sLGRX2NBI9nmgulR5THWpJK5HkNqHN2jHdEVFBa5cuYIrV64AkAzBX7lyha0PvGzZMoSFhbHxs2fPRmZmJhYvXox//vkHW7duxffff4/33nuvxW1JylBtTW1V4+Rp6ANqKChOiK41fNZa1P+YJbu+fYvimjBoErp8+TIGDhyIgQMHAgAiIyMxcOBAdlmRvLw8qYLlbm5u+OWXX3D69Gl4eXlh48aN+Oabb7Q0PK/qCFjLR8roFozoi1Y+a6oeQsNTGfR2bOTIkUrrqezatUvuPn///bfW2xLQwwZfnm1+IbiAHprdihHSZrkMA/CpinHqa1PzhHRpSM8u6GxqrDSms6kxhvTsmH05I0eOxIIFC9Ta58iRI3jmmWfA4/HU2leTc+mTq6ur1tbcahPchgMmyhdjhIm1JE4DlITq8bgchPoqftIXAEJ9u7VoZKyjefvtt/HSSy8hNzcXa9asYZciJm0MlwcMlDM839jA1zQaGQMoCbFEYgbHruYpjTl2Na/VzJ5u7bO6KyoqUFhYiJCQEHTt2rXZJ6k7EpFI1OL12/VKLAKuH1Iec/1HjWdPUxKql5xViryyJ0pj8sqeIDmrmaFKPTh5PQ/DNvyOadsvYP6BK5i2/QKGbfgdJ68rT6LaVFNTg4ULF8LJyQlmZmbw9/dHQkICAEmlg4ak8+yzz4LD4WDkyJHYvXs3jh49Cg6HAw6Hw8bL8/TpU8ybNw+WlpawsbHB8uXLpfoPHz58iLCwMFhZWcHU1BTjxo2TWhs9KioK3t7eUsfcvHkzXF1d2dcNV2afffYZHB0d0aVLF8ydO1dqJKmwsBATJ06EiYkJ3NzcZJaDBoCYmBj0798fZmZmcHZ2xjvvvCO1sumuXbvQuXNnHDt2DB4eHhAIBDh//jyMjY2Rn58vdawFCxa0vnXH7iVKz5KWp/y+JE4DlITqFT5WnoDUjdOVk9fzMGdvqkzCzC97gjl7U/WWiObNm4ekpCR2yZqXX34ZY8eOxZ07dxAYGIhbt24BAH788Ufk5eXh2LFjmDp1KsaOHYu8vDzk5eUhMDBQ4fF3794NIyMjJCcn4/PPP0dMTAy++eYb9vszZ87E5cuXcezYMSQlJYFhGDz33HNqD0WfPXsWGRkZOHv2LHbv3o1du3ZJDYjMnDkTubm5OHv2LA4dOoStW7eyEwAbcLlc/Pe//8WNGzewe/du/P7771i8eLFUTFVVFTZs2IBvvvkGN27cgK+vL3r06IE9e/awMXV1ddi3bx/eeOMNtd6DzlUUNB+jTlwTbWqyoi7ZdRJqNU4XRGIGq36+KXeSAAPJCOmqn2+2eFZ3c3JychAXF4ecnBx07doVALBw4UKcPHkScXFxWLduHezs7AAA1tbWcHBwACCZPFdTU8O+VsbZ2RmbNm0Ch8OBu7s70tLSsGnTJkRERODOnTs4duwY/vrrLzaR7du3D87Ozjhy5Ahefvllld+LlZUVvvzyS/B4PPTp0wfjx49HfHw8IiIicPv2bfz6669ITk7G4MGDAQA7duxA3759pY7RuBPd1dUVa9euxezZs7F161Z2e11dHbZu3QovLy9226xZsxAXF4dFixYBAH7++Wc8efIEU6dOVbn9emFu33yMOnFN0JVQPR8XKzT3d8vlSOIMpblbRgb6uWVMS0uDSCRC7969YW5uzn6dO3cOGRnNT3NQxZAhQ6TmuAQEBODOnTsQiURIT0+HkZER/P392e936dIF7u7uSE9PV+s8/fr1k5pJ7OjoyF7pNJzHx8eH/X6fPn3QuXNnqWOcOXMGo0ePhpOTEzp16oTXX38dJSUlqKqqYmP4fD4GDBggtd/MmTNx9+5dXLhwAYDktm3q1KkwMzNT6z3onLM/wGkmVXB4kjgN0JVQvZR7D9Fc366YkcQFGGiYvrXcMlZUVIDH4yElJUXmUQBzc3OdnltVXC5XZg6avFs1Y2PpaRkcDketTuPs7GxMmDABc+bMwccffwxra2ucP38es2bNQm1tLTtj2cTERGbioJ2dHSZOnIi4uDi4ubnh119/VdpPZjC5FwGmmZ8JI5LEaTBMT0moXmv5A1emtdwyDhw4ECKRCIWFhWp1ovL5fIhEqo2gXLx4Uer1hQsX0KtXL/B4PPTt2xdPnz7FxYsX2duxkpIS3Lp1Cx4eHgAAW1tb5Ofng2EY9o+/4fEgVfXp0wdPnz5FSkoKezt269YtqQoIKSkpEIvF2LhxI7uixPfff6/yOd58801MmzYN3bp1Q8+ePTF06FC12qgXOu4TotuxejZmzT/kqk6cLvi5WcPRUqhwdjwHgKOlEH5u1jptR+/evTF9+nSEhYXh8OHDyMrKQnJyMqKjo/HLL78o3M/V1RXXrl3DrVu3UFxcrLQTOScnB5GRkbh16xa+++47fPHFF5g/X/Ikd69evTB58mRERETg/PnzuHr1Kl577TU4OTlh8uTJACQTHouKivDJJ58gIyMDW7Zswa+//qrW+3R3d8fYsWPx9ttv4+LFi0hJScGbb74pVbHhmWeeQV1dHb744gtkZmZiz549iI2NVfkcISEhsLCwwNq1axEeHq5W+/TGVMWnBFSNa4KSUAMdPx+jDTwuBysneshtRsPrlRM99DKhMi4uDmFhYXj//ffh7u6OKVOm4NKlS+jevbvCfSIiIuDu7g5fX1/Y2trir7/+UhgbFhaG6upq+Pn5Ye7cuZg/fz7eeustqfP7+PhgwoQJCAgIAMMwOHHiBHt71bdvX2zduhVbtmyBl5cXkpOTsXDhQo3eZ9euXREUFIQXXngBb731FtvpDgBeXl6IiYnBhg0b4OnpiX379iE6Olrl43O5XMycORMikUjqYe1WRdXnzzR8To3DdLDFsMvLy2FpaYmysjKp1TaOXrmP+QeuNLv/5694Y7K3k0bnfvLkCbKysuDm5gahUPNbppPX87Dq55tSndSOlkKsnKjd6o9EP2bNmoWioiIcO3ZM68fWymcu7RDw46zm417cAfR/iX2p6G+tKeoTqtda+ltUMdbTEf/xcEByVikKHz+BXSfJLRg9UtK2lJWVIS0tDfv379dJAtIaHQ/RUxKq5+dmjc6mxkoLm3U2NdZ5f4uqeFyOwUbpiHZMnjwZycnJmD17Nv7zn/8YujmKuQRKHmCtfqg4xsRaEqcBSkJqoOsMok2tcjheId19+qljul5yVqlK5V1bw7NjhOjVvUTVyrvSs2Mt0xbmCRFiEDRPSD/awjwhQgyC5gnpSRuYJ0SIQeh4nhAloXrFFTXNB6kRR0i7UVmk3bgmKAnVszFX8XZMxThC2g0zW+3GNUFJqIH+VvxpVzgcDo4cOaJyfEJCAjgcjlaWwW7tNatbe8F+lan6UIWGD19QEqpXXKni7ZiKcR1FXl4exo0bp9VjyivNSgyoqli7cU3QZMV6bemxDQCSouL3EiXDoub2ktmqGq52oIna2lrw+XyVqiQSzTAMA5FIBCMjA/+ZUmVF/WgLlRVZN48Bmz2B3RMkDxbuniB5fVN3zx+NHDkS8+bNw4IFC2BjY8Ouetv0diwxMRHe3t4QCoXw9fXFkSNHwOFwZGr5pKSkwNfXF6amplI1qXft2oVVq1bh6tWrbEF8eYtgNrZq1SrY2trCwsICs2fPRm1tLfu9mpoavPvuu7Czs4NQKMSwYcNw6dIl9vsNRegba2hzg4Yrsz179sDV1RWWlpZ45ZVX8PjxYzamsrISYWFhMDc3h6OjIzZu3CjTzj179sDX1xedOnWCg4MDXn31Val61Q23qr/++it8fHwgEAiwd+9ecLlcXL58WepYmzdvhouLi35W7dBxZUVKQvXUqaxoUDePAd+Hya5+UJ4n2a7DRLR7927w+Xz89ddfcmvmlJeXY+LEiejfvz9SU1OxZs0aLFmyRO6xPvzwQ2zcuBGXL1+GkZERW9w9NDQU77//Pvr168cWxA8NDVXYpvj4eKSnpyMhIQHfffcdDh8+jFWrVrHfX7x4MX788Ufs3r0bqampeOaZZxASEoLSUvVmvmdkZODIkSM4fvw4jh8/jnPnzmH9+vXs9xctWoRz587h6NGj+O2335CQkIDU1FSpY9TV1WHNmjW4evUqjhw5guzsbMycOVPmXEuXLsX69euRnp6OSZMmITg4GHFxcVIxcXFxmDlzJltITafUqayoAbodq9cmZkyLRcDJJZDfO15f6v7kUqDPeJ3cmvXq1QuffPKJwu/v378fHA4H27dvh1AohIeHB+7fv4+IiAiZ2I8//hhBQUEAJH9048ePx5MnT2BiYgJzc3MYGRmpdKvH5/Oxc+dOmJqaol+/fli9ejUWLVqENWvWoLq6Gl999RV27drF9ltt374dp0+fxo4dO9gC86oQi8XYtWsXu5TR66+/jvj4eHz88ceoqKjAjh07sHfvXowePRqAJGF36ya9mGbjVTR69OiB//73vxg8eDAqKiqkyuKuXr1a6oHWN998E7Nnz0ZMTAwEAgFSU1ORlpaGo0ePqtz+FmnvM6a3bNkCV1dXCIVC+Pv7Izk5WWn85s2b4e7uDhMTEzg7O+O9997DkyctTwxtok+o2fWfmBat/9ScxgXf5bl16xYGDBggVbfGz89Pbmzjou+OjpIaSE2X0lGFl5cXW8cZkBTEr6ioQG5uLjIyMlBXVydVMtXY2Bh+fn5qF8R3dXWVWsCxcUH8jIwM1NbWShXet7a2hru7u9QxUlJSMHHiRHTv3h2dOnVik3BOTo5UnK+vr9TrKVOmgMfj4aeffgIguYUcNWqU1BpqOtWe+4QOHjyIyMhIrFy5EqmpqfDy8kJISIjCD+P+/fuxdOlSrFy5Eunp6dixYwcOHjyIDz74oMVtaRN9Qjr+H6k52lwFonGB+Yb+F0OsSqqvgviVlZVsKdd9+/bh0qVLbFJp3IcFyP6c+Xw+wsLCEBcXh9raWuzfv1+/a5O15z6hmJgYREREIDw8HB4eHoiNjYWpqSl27twpNz4xMRFDhw7Fq6++CldXV4wZMwbTpk1r9upJFW2iT0jH/yO1VMP6YDU1/05jaNwJrCp1CuJfvXoV1dXV7OsLFy7A3Nwczs7O6NmzJ9uH1aCurg6XLl2SKoj/+PFjVFZWsjHqFsTv2bMnjI2NpYrzP3z4ELdv32Zf//PPPygpKcH69esxfPhw9OnTR60rvzfffBNnzpzB1q1b8fTpU7zwwgtqtbFFdNwnZLAkVFtbi5SUFAQHB//bGC4XwcHBSEpKkrtPYGAgUlJS2KSTmZmJEydO4Lnnnmtxe9pEn5BLIGDRFYofYOMAFk4aF5dqqVdffRVisRhvvfUW0tPTcerUKXz22WeSlqnxXJGrqyuysrJw5coVFBcXSyW1pmprazFr1izcvHkTJ06cwMqVKzFv3jxwuVyYmZlhzpw5WLRoEU6ePImbN28iIiICVVVVmDVLUq7U398fpqam+OCDD5CRkYH9+/c3OxrXlLm5OWbNmoVFixbh999/x/Xr12U6jbt37w4+n88WxD927BjWrFmj8jn69u2LIUOGYMmSJZg2bZpUsX2da699QsXFxRCJRLC3l/5f297eXmZ97gavvvoqVq9ejWHDhsHY2Bg9e/bEyJEjld6O1dTUoLy8XOpLnjbRJ8TlAWM31L9QUOp+7Hq9zhdqzMLCAj///DOuXLkCb29vfPjhh1ixYgUAqFXf+MUXX8TYsWMxatQo2Nra4rvvvlMYO3r0aPTq1QsjRoxAaGgoJk2ahKioKPb769evx4svvojXX38dgwYNwt27d3Hq1ClYWUluq62trbF3716cOHEC/fv3x3fffSe1v6o+/fRTDB8+HBMnTkRwcDCGDRsm1Ydma2uLXbt24YcffoCHhwfWr1/PJmhVNaxlpvdlonV8BW6wQvcPHjyAk5MTEhMTERAQwG5fvHgxzp07J7PuFCCZR/HKK69g7dq18Pf3x927dzF//nxERERg+fLlcs8TFRUlNWTboGnx7dqnYvRZ/qvSWzIuB/hnzTjwjTTL3doqdI+bxySjZI07qS2cJAnIY5Lmx9WBffv2ITw8HGVlZfr937sdWrNmDX744Qdcu3ZN5X208pl7Wgt8bK/8lozDAz7MB4z47KZWX+jexsYGPB4PBQXSl3AFBQUKh2aXL1+O119/HW+++SYAoH///qisrMRbb72FDz/8UO6ciWXLliEyMpJ9XV5eDmdnZ5m4trACK8tjkmQY3oAzphX59ttv0aNHDzg5OeHq1atYsmQJpk6dSgmoBSoqKpCdnY0vv/wSa9eu1X8DdLwCq8Fux/h8Pnx8fBAfH89uE4vFiI+Pl7oyaqyqqkom0TQsQ6zogk4gEMDCwkLqS5420SfUGJcn+YX3f0nybytIQACQn5+P1157DX379sV7772Hl19+Gdu2bTN0s9q0efPmwcfHByNHjtT/rRig8z4hg05WjIyMxIwZM+Dr6ws/Pz9s3rwZlZWV7EqUYWFhcHJyYheTmzhxImJiYjBw4ED2dmz58uWYOHGizJro6moTfUJtwOLFi7F48WJDN6Nd2bVrl9qd5VrVnpf8CQ0NRVFREVasWIH8/Hx4e3vj5MmTbGd1Tk6O1JXPRx99BA6Hg48++gj379+Hra0tJk6ciI8//rjFbWmYJ9Rcn1CreHaMEH1qmCfUXJ+QhvOEaAXWekkZJZi2/UKz+38XMUTjPiGtdUwToiKtfOay/pQ8JN2cGcel+oRU7Zg2+GMbrUWb6xMiRF/a6zyh1ob6hAhRoD0/O9aaNCwDrYxVK1oGmhC9cQmULPOsTAuWgaYkpIYO1XlGiBTdffopCdVTZRnoRx14GWh1i7ZnZ2fLrajY0uNqs1C+NjV9v6q209XVFZs3b9Z5+1rkXiJQ3cyD2y1YBpqKmtWjjmnlDh8+LFPOQhlnZ2fk5eXBxkayKmdCQgJGjRqFhw8fSpVTVfe4bUVgYCDy8vJgaWkJQDLXZ8GCBTJJ6dKlS1otkaIT7XmyYmvS1tYdE4lFSC1MRVFVEWxNbTHIbhB4Opw1bW2tXl8Yj8dTqTKiusdtK1RdBMDWVrO1uvSK1h3Tkza07tiZe2cQ8mMI3jj1Bpb8uQRvnHoDIT+G4My9Mzo7Z9PbJldXV6xbtw5vvPEGOnXqhO7du0s9ntH49iQ7OxujRo0CAFhZWYHD4bC1lZset7li8Kp49OgR3n77bdjb20MoFMLT0xPHjx9nv//jjz+iX79+EAgEcHV1lSlK39x7A4Dk5GQMHDiQLej/999/S32/8e1YQkIC+xBvQ/H+hif1m96O5eTkYPLkyTA3N4eFhQWmTp0q9XylKkX3tY7WHdOPtrLu2Jl7ZxCZEImCKulL38KqQkQmROo0ETW1ceNG9g/wnXfewZw5c9hVMxpzdnbGjz/+CEBSAjYvLw+ff/653GOqWgxeEbFYjHHjxuGvv/7C3r17cfPmTaxfv559rCclJQVTp07FK6+8grS0NERFRWH58uUyj0Uoe28VFRWYMGECPDw8kJKSgqioKCxcuFBhmwIDA7F582ZYWFiwxfvlxYvFYkyePBmlpaU4d+4cTp8+jczMTJlC/80V3dc6WndMP9rCPCGRWIT1yevByLkcY8CAAw42JG/AKOdROr01a/Dcc8/hnXfeAQAsWbIEmzZtwtmzZ2VqK/N4PPa2y87OTmaJncZULQavyJkzZ5CcnIz09HT07t2bPU6DmJgYjB49mi390rt3b9y8eROffvqpVLJT9t72798PsViMHTt2QCgUol+/fvjf//6HOXPmyG0Tn8+HpaUlOByO0lu0+Ph4pKWlISsri6308O2336Jfv364dOkSBg8eDEB50X2doHlC+tEWakynFqbKXAE1xoBBflU+UgtTFcZoU+Ni9Q1/YJoUq29M1WLwily5cgXdunVjE1BT6enpUoXvAWDo0KG4c+eOVElZZe8tPT1dpqC/osoP6khPT4ezs7NUqRkPDw907txZqjC/sqL7OtGea0y3Jm2hxnRRVZFW41qqpcXfm1KnGLwi2qpbpO33pk16b1t7rTHd2uSXqzb0rmqcLtiaqjb6oGqcPvH5kop7ygrYt7QYPCC5gvnf//4nVWS+sb59+0oVvgeAv/76C71791a5HEzfvn1x7do1qaWmLlxQ/vCzKsX7+/bti9zcXOTm5rLbbt68iUePHrGF+Q3icZ5245qgJFSvtEK1DmdV43RhkN0g2Jvag6Og0D0HHDiYOmCQ3SA9t6x5Li4u4HA4OH78OIqKilBRUSET09Ji8AAQFBSEESNG4MUXX8Tp06eRlZWFX3/9FSdPngQAvP/++4iPj8eaNWtw+/Zt7N69G19++aXSjuWmXn31VXA4HERERLAF9purF+3q6oqKigrEx8ejuLgYVVVVMjHBwcHo378/pk+fjtTUVCQnJyMsLAxBQUEya5HpVaWKV9aqxjVBSaietRm/+SA14nSBx+Vhqd9SAJBJRA2vl/gt0UuntLqcnJywatUqLF26FPb29pg3b55MjDaKwQOSIfjBgwdj2rRp8PDwwOLFi9mrkEGDBuH777/HgQMH4OnpiRUrVmD16tVqjcCZm5vj559/RlpaGgYOHIgPP/wQGzZsULpPYGAgZs+ejdDQUNja2spdyZbD4eDo0aOwsrLCiBEjEBwcjB49euDgwYNqvX+t0/E8IaonVK8t1RM6c+8M1ievl+qkdjB1wBK/JQh2CVayJ+lo2kI9IRqir9eWKisGuwRjlPMovc6YJh2YjisrUhKq16ZW24Dk1myww2BDN4N0BO11tY3Wpi2MjhFiEDQ6ph9tYXSMEIOg0TH9aAujY4QYBD1Frx8OlqrNtFU1TpnWMvOWtH9a+ax1ctRuXBPUMV3Pz80ajpZC5JUp7vNxtBS2qMY0n88Hl8vFgwcPYGtrCz6fDw6nmQfWCNEAwzCora1FUVERuFwuO2NdIy6BgEVXoPyB4hgLJ41rTFMSqsfjcjDJyxFf/5GlMGaSlyN4zT3lqgSXy4Wbmxvy8vLw4IGSXyghWmJqaoru3bvLLJ+uFi4P8HwJSPyv4hjPFzVeipySUD2RmMGxq8p7949dzcPisX1blIj4fD66d++Op0+fNvssESEtwePxYGRk1PKrbbEIuH5Iecz1H4HgKI0SESWheslZpUpvxQAgr+wJkrNKWzxPiMPhwNjYuF3WVibt0L1E5bdiAFB+XxJH84Q0R/OECFGgvc8T2rJlC1xdXSEUCuHv74/k5GSl8Y8ePcLcuXPh6OgIgUCA3r1748SJEy1uB80TIkQBHc8TMujt2MGDBxEZGYnY2Fj4+/tj8+bNCAkJwa1bt2BnZycTX1tbi//85z+ws7PDoUOH4OTkhHv37iktF6oqmidEiAI6nidk0CQUExODiIgIhIeHAwBiY2Pxyy+/YOfOnVi6dKlM/M6dO1FaWorExES2P8XV1VUrbdHnPCFC2hQdzxMy2O1YbW0tUlJSEBz8b+kJLpeL4OBgJCUlyd3n2LFjCAgIwNy5c2Fvbw9PT0+sW7dO6ShTTU0NysvLpb7kUWUt+s60Fj3piFwCAZNmqke0xbXoi4uLIRKJYG8vXaHf3t4e+fn5cvfJzMzEoUOHIBKJcOLECSxfvhwbN27E2rVrFZ4nOjoalpaW7FfjIuLqommFpOPS3aff4B3T6hCLxbCzs8O2bdvg4+OD0NBQfPjhh4iNjVW4z7Jly1BWVsZ+Na7f25gqa9E/7MBr0ZMO7F6iZK15ZdriWvQ2Njbg8XhSq0sCQEFBgcK1mRwdHWFsbCxVkLxv377Iz89HbW2t3KnpAoEAAkHzSzfTWvSEKKDjtegNdiXE5/Ph4+OD+Ph4dptYLEZ8fLzCNZyGDh2Ku3fvSj2Ud/v2bTg6Orbs2Ri0jcUPCTGI9rz4YWRkJLZv347du3cjPT0dc+bMQWVlJTtaFhYWhmXLlrHxc+bMQWlpKebPn4/bt2/jl19+wbp16zB37twWt6UtLH5IiEHoePFDgw7Rh4aGoqioCCtWrEB+fj68vb1x8uRJtrM6JydH6sE7Z2dnnDp1Cu+99x4GDBgAJycnzJ8/H0uWLGlxW9paeVdC9EbH5V0N/uzYvHnz5C7/AgAJCQky2wICAppdaE4T1CdEiALttU+otaE+IUIUaM99Qq0JTVYkRIH2OlmxLaLJiqTjosmKOkeTFQlRQMeTFSkJ1aOOaUIUoI5p/bAxb35WtTpxhLQbtOSPnjQzR0jtOELaC0bFD72qcU1QEqpXXKlaxURV4whpN6qKtRvXBCWhejZmKt6OqRhHSLthaqPduCYoCTVQdQSSxulJR6PqkkEaLi1ESahesYoF7FWNI6Td0HGhe7WTUE5ODhg5HVAMwyAnJ0ejRrQGNDpGiAKtbXTMzc0NRUWyGa+0tBRubm4aNaJVoNExQuRrbaNjDMPIXVa2oqICQmHbfbiTRscIUUDHo2Mql/KIjIwEIFnCePny5TA1NWW/JxKJcPHiRXh7e2vUiNaAbscIUaC1rDv2999/A5BcCaWlpUmVU+Xz+fDy8sLChQs1akSrQLdjhMin49sxlZPQ2bNnAQDh4eH4/PPPYWFhodEJWyu6HSNEgdZyO9YgLi5OoxO1dlTUjBAFqKiZfvi5WcPRUqhwLiIHgKOlkIqakY7HJRCw6ArFM3U5gIUTFTVrKR6Xg5UTPRR2+TAAVk70AK+5JTkIaW+4PGDsBijuEGWAseslcZocXuOGEUKIFlASqicSM1h6OE1pzNLDaRA1ty4QIe2NWAT8/K7ymJ/nS+I0QEmo3oWMkmbLuz6qqsOFjBI9tYiQViLrT6D6ofKY6lJJnAYoCdVLylRteFHVOELajXvntRvXBCUhFtXyIEQuHU/kpSRUT9WlnWkJaNLhqLq0swZLQAOUhFiDXa2brcnE4UjiCOlQugeg+TsATn2c+lpFEtqyZQtcXV0hFArh7++P5ORklfY7cOAAOBwOpkyZ0uI2pNx72OyjLwwjiSOkQ8m9iObvtZj6OPUZPAkdPHgQkZGRWLlyJVJTU+Hl5YWQkBAUFhYq3S87OxsLFy7E8OGaXQI2ReuOEaJAe193LCYmBhEREQgPD4eHhwdiY2NhamqKnTt3KtxHJBJh+vTpWLVqFXr06KGVdlChe0IUaM+F7mtra5GSkoLg4GB2G5fLRXBwMJKSkhTut3r1atjZ2WHWrFnNnqOmpgbl5eVSX3LR4Bgh8rXnQvfFxcUQiUSwt5d++tbe3h75+fly9zl//jx27NiB7du3q3SO6OhoWFpasl/Ozs7y20KF7gmRr7UVujekx48f4/XXX8f27dthY6Papd+yZctQVlbGfuXm5sqNo1IehCig41IeatcT0iYbGxvweDwUFEh3aBUUFMDBwUEmPiMjA9nZ2Zg4cSK7TSwWAwCMjIxw69Yt9OzZU2ofgUAAgaD5fhwfFytwOYCyR8O4HEkcIR2Ksz/A4QKMWHEMhyeJ04BBr4T4fD58fHwQHx/PbhOLxYiPj0dAgOycgz59+iAtLQ1XrlxhvyZNmoRRo0bhypUrCm+1VJFy76HSBARIEhQN0ZMOJ/ei8gQEAIxI4yF6g14JAZIC+jNmzICvry/8/PywefNmVFZWIjw8HAAQFhYGJycnREdHQygUwtPTU2r/zp07A4DMdnXRED0hCuh4iN7gSSg0NBRFRUVYsWIF8vPz4e3tjZMnT7Kd1Tk5OeBydX/BRn1ChCig4z4hDiNvOdV2rLy8HJaWligrK5Mq1i8SMxi24XfklSm+0nG0FOL8kmepuiLpWMQiYLMnUP5AcYyFE7AgTaq6oqK/taba1OiYLvG4HEzyclQaM8nLkRIQ6Xi4PMDzJeUxni9SedeWEokZHLuapzTm2NU8qqxIOh6xCLh+SHnM9R+psmJLJWeVKr0VA4C8sidIzirVU4sIaSXuJSq/FQOA8vuSOA1QEqpHo2OEKNDeH2BtLWh0jBAFaPFD/WiYMa0MzZgmHVLDjGll2uqM6daEZkwTooCOZ0xTEqqXX1at1ThC2o3HykeN1Y5rgpJQvdLKWq3GEdJuUCkP/bA2V61ioqpxhLQbZrbajWuCklA9BwvVRr1UjSOk3eik/EkCteOaoCRUz8/NGo6WyhOMo6UQfm605A/pYFwCAYuuymMsnCRxGqAkVI+eHSNEAXp2TD/o2TFCFKBnx/SDnh0jRAF6dkw/6NkxQhSgZ8f0w0bFoXdV4whpN2iIXk9U7eqhLiHS0ahafFXDIq2UhOoVV6q4+KGKcYS0G1XF2o1rgpJQPVqLnhAF2vNa9K0KrUVPiHzteS361oTWoidEAXqAVT+osiIhClBlRf3wc7NGZ1NjpTGdTY3p2THS8bgEAibNVBQ1saZnx/SBuoNIx6W7Tz8loXrJWaV4VFWnNOZhVR09tkE6nnuJQHUzn/vq0rb92MaWLVvg6uoKoVAIf39/JCcnK4zdvn07hg8fDisrK1hZWSE4OFhpvKryy1V7HEPVOELajfZe3vXgwYOIjIzEypUrkZqaCi8vL4SEhKCwsFBufEJCAqZNm4azZ88iKSkJzs7OGDNmDO7fv9+idpSqOOqlahwh7UZ7Hx2LiYlBREQEwsPD4eHhgdjYWJiammLnzp1y4/ft24d33nkH3t7e6NOnD7755huIxWLEx8e3qB3WZnytxhHSbrTnZ8dqa2uRkpKC4OBgdhuXy0VwcDCSkpJUOkZVVRXq6upgbd2yUSsHSxOtxhHSbrTn8q7FxcUQiUSwt5eeX2Bvb4/8/HyVjrFkyRJ07dpVKpE1VlNTg/LycqkveWiInhAFaIhesfXr1+PAgQP46aefIBTKn0QYHR0NS0tL9svZ2Vnj89EQPem42ukQvY2NDXg8HgoKpIshFRQUwMHBQem+n332GdavX4/ffvsNAwYMUBi3bNkylJWVsV+5ubly42iInhAF2vMQPZ/Ph4+Pj1SnckMnc0BAgML9PvnkE6xZswYnT56Er6+v0nMIBAJYWFhIfclDlRUJUUDHlRWNNNpLiyIjIzFjxgz4+vrCz88PmzdvRmVlJcLDwwEAYWFhcHJyQnR0NABgw4YNWLFiBfbv3w9XV1e278jc3Bzm5uYat4MqKxKigI5HxwyehEJDQ1FUVIQVK1YgPz8f3t7eOHnyJNtZnZOTAy733wu2r776CrW1tXjpJeklSFauXImoqCjNG0KVFQmRT8eVFQ2ehABg3rx5mDdvntzvJSQkSL3Ozs7WSRuosiIhClBlRf2gUh6EKEClPPSD5gkRogDNE2o9aJ4Q6bja6Tyh1oTmCRGiQHueJ9Sa0DwhQhSgFVj1gzqmCVGAOqb1w8fFCtxmbnu5HEkcIR2Ksz/AaSZVcHiSOA1QEqqXcu8hxM3MtRIzkjhCOpTciwAjVh7DiCRxGqAkVI/6hAhRgPqE9IOeHSNEgfZcWbFVoWfHCJFPx8+OURKqR8+OEaIAPTumH3Q7RogCdDumJ3Q7Roh8dDumH3Q7RogCdDumHzRjmhAFaMa0flApD0IUoFIerQeV8iAdF5Xy0Dkq5UGIAlTKQz/yy6q1GkdIu/E4T7txTVASqldaWavVOELajcoi7cY1QUmonrWKkxBVjSOk3aDJivrhYKHa0LuqcYS0G50ctRvXBCWheqoM0VvRED3piFwCJUPwytAQvX7QExuk49Ldp5+SUD1Vhugf0RA96YjuJQLVzVQUbcEQfatYBro1kB16F4NnmgGeaQY4xo8AiMExqsTGq0cwoNAZ5sbmKKwshIgR4VHNIwiMBOCAAxsTG3A5XJgameLWw1sQ8ASwEdqgoq4C+ZX5EPAE6GLSBfZm9qisq5SciRGjqq4K1aJqWAusUV1XjSpRFRgw4DAcmPHN8EznZ3Dn4R08qHwAIU+I3la9UfW0CiVPSmDCM4GpsSmKq4pRLapGF2EXOJg5oKKuAgDAMAwqaiuQV5kHPo8PGxMbOJg6oLyuHEVVRagV18Ldyh0VdRXs8UyMTVBSVYJqUTWsBFbgcXiwNbNFVV0VGIZBUXURGIZBjagGVkIrcDlc2JnaoaKuAoVVhagR1aBP5z6oeFqBouoi1DytgZXACuAAXA4X9ib2qHhaATEjRnF1McSMmD0Wj8ODvak9ymvL2fdkJZCcozEOOHA0d4QF3wKlNaVIL05HjbgGjqaOMOWboqS6BE+ePoGVwAocDkcy346B5OfaePIdg3/n4nEAiAGG828MwzCS/aVPLjlW4+8x0vs1xDTdT8yI8fDJQ5gYmcDGxAYWfAsUVBWAafwAaH07Hz55CIGRZDCEw+HAzNgMz1g+g7tld1FZWwlwABsTG5Q8KUEXYRdwOVyYGZshvSQdAq6AfV81ohpYC63hYO4AS2NL3H50m/382ZpIOpQbPqOOZo5wt3JHWW0ZbpbchKCyBLDtAgCo5XLgUVOHgCdP4PukBrzG703DIXoOw2j46KsWbdmyBZ9++iny8/Ph5eWFL774An5+fgrjf/jhByxfvhzZ2dno1asXNmzYgOeee06lc5WXl8PS0hJlZWWwsLBgt+/4MxNrfkkHABh1ug6Bw2Fwjapa9sYIaccsRSJEFZciuKr+P/CQdUDAXPb7iv7WmjL47djBgwcRGRmJlStXIjU1FV5eXggJCUFhYaHc+MTEREybNg2zZs3C33//jSlTpmDKlCm4fv16i9rRMPRu1Ok6hE57weFRAiJEmTIuF+/Z2eCMqYlkg4ZD9Aa/EvL398fgwYPx5ZdfAgDEYjGcnZ3xf//3f1i6dKlMfGhoKCorK3H8+HF225AhQ+Dt7Y3Y2Nhmz6coOydllGDa9kSYPbMeHKNyNL36JoTIwTCwF4lwKvcBeDOOA27D2W+1iSuh2tpapKSkIDg4mN3G5XIRHByMpKQkufskJSVJxQNASEiIwviamhqUl5dLfcnj52YNW9v74BpTAiJEZRwOCoyMkGrt1DaH6IuLiyESiWBvL12HxN7eHvn5+XL3yc/PVys+OjoalpaW7Jezs7PcOB6XgxcGK87WhBDFigZOA7i85gPlMHifkK4tW7YMZWVl7Fdubq7C2ODevfTYMkLaD9teYzXe16BD9DY2NuDxeCgokF40raCgAA4ODnL3cXBwUCteIBBAIFDtea9BdoNgZ2KHwmr5neKEEFn2JvYYZDdI4/0NeiXE5/Ph4+OD+Ph4dptYLEZ8fDwCAgLk7hMQECAVDwCnT59WGK8OHpeHZf7LWnwcQjqSpf5LwdPwVgxoBbdjkZGR2L59O3bv3o309HTMmTMHlZWVCA8PBwCEhYVh2bJ/E8P8+fNx8uRJbNy4Ef/88w+ioqJw+fJlzJs3TyvtCXYJxqaRm2DJt9TK8QhprzrzO2PTyE0IdgluPlgJg8+YDg0NRVFREVasWIH8/Hx4e3vj5MmTbOdzTk4OuNx/c2VgYCD279+Pjz76CB988AF69eqFI0eOwNPTU2ttCnYJxijnUbhccBkXH1zEg8oHEIvFeFjzEEIjIWxNbWnGNM2Y7jgzpuvPDzFQy9TCo4sHAroGwNfet0VXQOyPxNDzhPRN1bkLhJCWaRPzhAghhJIQIcSgDN4npG8Nd5+KZk4TQrSj4W+suR6fDpeEHj9+DAAKZ04TQrTr8ePHsLRUPNrc4TqmxWIxHjx4gE6dOsmOeDRSXl4OZ2dn5ObmdsgO7I78/um9a+e9MwyDx48fo2vXrlIj3E11uCshLpeLbt26qRxvYWHR4T6IjXXk90/vveXvXdkVUAPqmCaEGBQlIUKIQVESUkAgEGDlypUqP/za3nTk90/vXb/vvcN1TBNCWhe6EiKEGBQlIUKIQVESIoQYFCUhBbZs2QJXV1cIhUL4+/sjOTnZ0E3SuejoaAwePBidOnWCnZ0dpkyZglu3bhm6WQaxfv16cDgcLFiwwNBN0Zv79+/jtddeQ5cuXWBiYoL+/fvj8uXLOj8vJSE51F0Lrb04d+4c5s6diwsXLuD06dOoq6vDmDFjUFlZaeim6dWlS5fw9ddfY8CAAYZuit48fPgQQ4cOhbGxMX799VfcvHkTGzduhJWVle5PzhAZfn5+zNy5c9nXIpGI6dq1KxMdHW3AVulfYWEhA4A5d+6coZuiN48fP2Z69erFnD59mgkKCmLmz59v6CbpxZIlS5hhw4YZ5Nx0JdSEJmuhtVdlZWUAAGtrawO3RH/mzp2L8ePHy6xt194dO3YMvr6+ePnll2FnZ4eBAwdi+/btejk3JaEmNFkLrT0Si8VYsGABhg4dqtXSua3ZgQMHkJqaiujoaEM3Re8yMzPx1VdfoVevXjh16hTmzJmDd999F7t379b5uTvcA6xENXPnzsX169dx/vx5QzdFL3JzczF//nycPn0aQqHQ0M3RO7FYDF9fX6xbtw4AMHDgQFy/fh2xsbGYMWOGTs9NV0JNaLIWWnszb948HD9+HGfPnlWr4kBblpKSgsLCQgwaNAhGRkYwMjLCuXPn8N///hdGRkYQiUSGbqJOOTo6wsPDQ2pb3759kZOTo/NzUxJqQpO10NoLhmEwb948/PTTT/j999/h5uZm6CbpzejRo5GWloYrV66wX76+vpg+fTquXLkCHq/lq0q0ZkOHDpWZjnH79m24uLjo/uQG6Q5v5Q4cOMAIBAJm165dzM2bN5m33nqL6dy5M5Ofn2/opunUnDlzGEtLSyYhIYHJy8tjv6qqqgzdNIPoSKNjycnJjJGREfPxxx8zd+7cYfbt28eYmpoye/fu1fm5KQkp8MUXXzDdu3dn+Hw+4+fnx1y4cMHQTdI5SFbJkvmKi4szdNMMoiMlIYZhmJ9//pnx9PRkBAIB06dPH2bbtm16OS89RU8IMSjqEyKEGBQlIUKIQVESIoQYFCUhQohBURIihBgUJSFCiEFREiKEGBQlIUKIQVESIoQYFCUhQohBURIihBgUJSHSahUVFcHBwYEttAUAiYmJ4PP5UqVWSNtGD7CSVu3EiROYMmUKEhMT4e7uDm9vb0yePBkxMTGGbhrREkpCpNWbO3cuzpw5A19fX6SlpeHSpUsQCASGbhbREkpCpNWrrq6Gp6cncnNzkZKSgv79+xu6SUSLqE+ItHoZGRl48OABxGIxsrOzDd0comV0JURatdraWvj5+cHb2xvu7u7YvHkz0tLSYGdnZ+imES2hJERatUWLFuHQoUO4evUqzM3NERQUBEtLSxw/ftzQTSNaQrdjpNVKSEjA5s2bsWfPHlhYWIDL5WLPnj34888/8dVXXxm6eURL6EqIEGJQdCVECDEoSkKEEIOiJEQIMShKQoQQg6IkRAgxKEpChBCDoiRECDEoSkKEEIOiJEQIMShKQoQQg6IkRAgxKEpChBCD+n+gH6E1AvizzgAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 300x300 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# create equally spaced points in a square\n",
    "x_lim = [0, 2 * np.pi] # space domain\n",
    "t_lim = [0, 1] # time domain\n",
    "rng = np.random.default_rng(seed=123)\n",
    "\n",
    "# initial condition\n",
    "def u0(x):\n",
    "    return np.sin(x)\n",
    "\n",
    "# boundary condition: Loss term that tries to satisfy the solution and it's first derivative zero at the boundary\n",
    "boundary_condition = \"periodic strict\" \n",
    "\n",
    "# forcing\n",
    "def forcing(x, t):\n",
    "    return np.zeros(x.shape[0])\n",
    "\n",
    "# Boundary and initial data\n",
    "n_initial = 200 # No. of initial points where solution is known\n",
    "n_boundary = 200 # No. of boundary points where solution is known\n",
    "rng = np.random.default_rng(seed=123)\n",
    "n_edges = 4 # edges of the domain\n",
    "pps = int(n_boundary/n_edges) # points per surface \n",
    "\n",
    "# Initial data\n",
    "x_0 = qmc.LatinHypercube(d=1, seed=221).random(n=n_initial)\n",
    "x_0 = qmc.scale(x_0, 0, 2 * np.pi)\n",
    "X_init = np.hstack((x_0, np.zeros_like(x_0)))\n",
    "u_init = u0(x_0)\n",
    "data_init = np.column_stack((X_init, u_init.reshape(-1, 1)))\n",
    "\n",
    "# Boundary data\n",
    "y_0 = qmc.LatinHypercube(d=1, seed=3).random(n=pps)\n",
    "y_0 = qmc.scale(y_0, 0, 1)\n",
    "X_left = np.hstack((np.zeros_like(y_0), y_0))\n",
    "X_right = np.hstack((np.ones((np.shape(y_0))) * 2 * np.pi, y_0))\n",
    "\n",
    "# Visualiza initial and boundary data\n",
    "fig, ax = plt.subplots(1,1,figsize=(3, 3))\n",
    "ax.scatter(X_left[:, 0], X_left[:, 1], label='left boundary')\n",
    "ax.scatter(X_right[:, 0], X_right[:, 1], label='right boundary')\n",
    "ax.scatter(X_init[:, 0], X_init[:, 1], label='initial condition')\n",
    "ax.set_xlabel(\"x\")\n",
    "ax.set_ylabel(\"t\")\n",
    "ax.set_title(\"Initial and boundary data\")\n",
    "ax.legend()\n",
    "fig.tight_layout()\n",
    "\n",
    "# Test data : \n",
    "n_test_space = 256\n",
    "n_test_time = 100\n",
    "x_test = np.linspace(x_lim[0], x_lim[1], n_test_space)\n",
    "t_test = np.linspace(t_lim[0], t_lim[1], n_test_time)\n",
    "xx_test, yy_test = np.meshgrid(x_test, t_test)\n",
    "X_test = np.hstack((xx_test.reshape(-1, 1), yy_test.reshape(-1, 1))) # Uniform test points\n",
    "print(np.shape(X_test))\n",
    "\n",
    "# coordinates of boundary points (excluding corners)\n",
    "boundary_points = np.row_stack([X_left, X_right])\n",
    "\n",
    "# Analytical solution of the PDE\n",
    "def analytical_sol(x, t, c):\n",
    "    return np.sin(x - c * t)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(961, 2)\n",
      "[0.01, 1000, 500, 1e-10, 0.7666193644205729, 0.7719058195749918, 7.624150780823968e-05, 4.958232324972673e-05, 2.533663166931941e-07, 1.1182647115067283e-07, 0.00010803296854460549, 7.025734041722731e-05, 3.5901592332648247e-07, 1.584562791000999e-07]\n",
      "(961, 2)\n",
      "[0.1, 1000, 500, 1e-10, 1.1939496994018555, 1.0760585467020671, 7.20710394114307e-05, 2.86644068485884e-05, 1.1904987125384789e-07, 1.813983796960685e-08, 0.00010212215755486992, 4.061646808918038e-05, 1.686895292250245e-07, 2.5703519836543283e-08]\n",
      "(961, 2)\n",
      "[1, 1000, 500, 1e-10, 0.8708645502726237, 0.8127375443776449, 0.00022621065728965715, 0.0001380509307775816, 8.216252732364676e-06, 3.1343416111720854e-06, 0.0003201936050946522, 0.0001954064664413457, 1.162983041676931e-05, 4.436555519107532e-06]\n",
      "(7921, 2)\n",
      "[10.0, 8000, 4000, 1e-10, 72.8531699180603, 101.7482848962148, 0.00026008312715635357, 0.0001003575986889398, 0.005460748863210299, 0.0007137372016488022, 0.0003678505030980984, 0.00014194151528042226, 0.00772345072357073, 0.0010094795136345203]\n",
      "(7921, 2)\n",
      "[40, 8000, 4000, 1e-10, 74.22406665484111, 105.98912612597148, 3.7884228219570804, 0.21024570034196285, 6.8134694354427054, 0.2693349524411097, 5.357563860134977, 0.29732815444263766, 9.635565860783336, 0.38089180518773946]\n",
      "(7921, 2)\n",
      "[100, 8000, 4000, 1e-10, 74.08567921320598, 99.08116634686787, 19.150778389001815, 0.9634140082314742, 20.669859339977638, 0.1495935729168811, 27.083538837329854, 1.3624856482778591, 29.23186341699443, 0.21155919929792671]\n"
     ]
    }
   ],
   "source": [
    "# Hyper-parameter run\n",
    "conv_coeffs = [1e-2, 1e-1, 1, 1e1, 40, 100] # Values of convection coefficients\n",
    "n_s = 1000 # No. of collocation points\n",
    "n_s_2 = 8000 # No. of collocation points for beta = 40, 100\n",
    "ratio = 2 # data points/width\n",
    "reg = 1e-10 # regularization constants\n",
    "seeds = [1, 2, 3] # random seeds\n",
    "\n",
    "experiments = []\n",
    "for conv_coeff in conv_coeffs:\n",
    "    # Ground truth for the selected convection coefficient\n",
    "    u_true_test = analytical_sol(xx_test, yy_test, conv_coeff).reshape(-1, 1)\n",
    "\n",
    "    # Collocation points\n",
    "    if conv_coeff < 5:\n",
    "        n_c = n_s\n",
    "    else:\n",
    "        n_c = n_s_2\n",
    "\n",
    "    # Collocation points\n",
    "    n_train = int(np.sqrt(n_c))\n",
    "    sampler_2d_m = qmc.LatinHypercube(d=2, seed=3)\n",
    "    xy_space_train = sampler_2d_m.random(n=n_train)\n",
    "    xy_space_scaled_train = qmc.scale(xy_space_train, [0, 0], [2 * np.pi, 1])\n",
    "    xx_train, yy_train = np.meshgrid(xy_space_scaled_train[:, 0], xy_space_scaled_train[:, 1])\n",
    "    X_train = np.hstack((xx_train.reshape(-1, 1), yy_train.reshape(-1, 1))) # Uniform test points\n",
    "    print(np.shape(X_train))\n",
    "    \n",
    "    # interior points\n",
    "    interior_points = X_train\n",
    "    \n",
    "    # Domain\n",
    "    domain = Domain(\n",
    "        interior_points=interior_points,\n",
    "        boundary_points=boundary_points,\n",
    "    )\n",
    "    # Loop over hyper-params\n",
    "    info = []\n",
    "    info.append(conv_coeff)\n",
    "    info.append(n_c)\n",
    "    n_b = int(n_c//ratio)\n",
    "    info.append(n_b)\n",
    "    info.append(reg)\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",
    "    for s in seeds:\n",
    "        ansatz_elm = BasicAnsatz(\n",
    "            activation='tanh',\n",
    "            n_neurons=n_b,\n",
    "            random_state=s,\n",
    "            regularization_scale=reg,\n",
    "            parameter_sampler=\"random\"\n",
    "        )\n",
    "        ansatz_swim = BasicAnsatz(\n",
    "            activation='tanh',\n",
    "            n_neurons=n_b,\n",
    "            random_state=s,\n",
    "            regularization_scale=reg,\n",
    "            parameter_sampler=\"tanh\"\n",
    "        )\n",
    "        adv_solver_elm = AdvectionSolver(\n",
    "            domain=domain, \n",
    "            ansatz=ansatz_elm,\n",
    "            boundary_condition=boundary_condition,\n",
    "            c=conv_coeff,\n",
    "            forcing=forcing,\n",
    "            regularization_scale=reg,\n",
    "            u0=u0\n",
    "        )\n",
    "        adv_solver_swim = AdvectionSolver(\n",
    "            domain=domain, \n",
    "            ansatz=ansatz_swim,\n",
    "            boundary_condition=boundary_condition,\n",
    "            c=conv_coeff,\n",
    "            forcing=forcing,\n",
    "            regularization_scale=reg,\n",
    "            u0=u0\n",
    "        )\n",
    "        # ELM (No-ODE)\n",
    "        t_elm_start = time.time()\n",
    "        adv_solver_elm.fit_no_ode_periodic_bc(data_init=data_init, svd_cutoff = 1e-12); #, num_components=min(1000, int(n_b//2) - 1)\n",
    "        t_elm_stop = time.time()\n",
    "        time_elm[s-1] = t_elm_stop - t_elm_start\n",
    "\n",
    "        # SWIM (No-ODE)\n",
    "        t_swim_start = time.time()\n",
    "        adv_solver_swim.fit_no_ode_periodic_bc(data_init=data_init, svd_cutoff = 1e-12); #, num_components=min(1000, int(n_b//2) - 1)\n",
    "        t_swim_stop = time.time()\n",
    "        time_swim[s-1] = t_swim_stop - t_swim_start\n",
    "        \n",
    "        # Evaluate on test data\n",
    "        u_swim_test = adv_solver_swim.evaluate_no_ode(X_test).T\n",
    "        u_elm_test = adv_solver_elm.evaluate_no_ode(X_test).T\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": 18,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(6, 14)\n",
      "Information for beta = 40:\n",
      "time_swim =  105.98912612597148 time_elm =  74.22406665484111\n",
      "rmse elm =  3.7884228219570804 +- 0.21024570034196285\n",
      "rmse swim =  6.8134694354427054 +- 0.2693349524411097\n",
      "rel error elm =  5.357563860134977 +- 0.29732815444263766\n",
      "rel error swim =  9.635565860783336 +- 0.38089180518773946\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 400x300 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Convergence plots\n",
    "fontsize = 12\n",
    "res = np.vstack(experiments)\n",
    "\n",
    "# For beta = 40 \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])\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",
    "\n",
    "# Store rmse error values for convergence plots\n",
    "with open('adv_swim_elm_rmse.npy', 'wb') as f:\n",
    "    np.save(f, rmse)\n",
    "\n",
    "# Store rel l2 error values for convergence plots\n",
    "with open('adv_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[:, -4], label='rmse_elm')\n",
    "ax.loglog(conv_coeffs, res[:, -2], label='rmse_swim')\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_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
}
