{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "GRdoE3U-SybK"
   },
   "source": [
    "## Compare convergence rates 5d for Online Sinkhorn\n",
    "\n",
    "This notebook compares the theoretical convergence rates with the old paper.\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 1680,
     "status": "ok",
     "timestamp": 1694679065047,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "Zes9WhQ1rcDp",
    "outputId": "6ac2b15a-9015-4148-c0a2-3dbf6430470e"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Drive already mounted at /content/drive; to attempt to forcibly remount, call drive.mount(\"/content/drive\", force_remount=True).\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "['closed_forms.py',\n",
       " 'helper.py',\n",
       " 'gq.py',\n",
       " 'recombination.py',\n",
       " 'recombination2.py',\n",
       " 'README.md',\n",
       " 'compression.py',\n",
       " 'algorithms.py',\n",
       " 'OS_theoretical_cts_5d.pkl',\n",
       " 'example_gmm_5d.pkl',\n",
       " 'example_gmm_2d.pkl',\n",
       " 'example_cts_1d.ipynb',\n",
       " 'example_cts_1d.pkl',\n",
       " 'example_gmm_5d.ipynb',\n",
       " 'example_gmm_2d.ipynb',\n",
       " '__pycache__',\n",
       " 'OS_theoretical_cts_1d_test.pkl',\n",
       " 'OS_theoretical_cts_1d.pkl',\n",
       " 'OS_theortical_cts_1d.ipynb',\n",
       " 'OS_theortical_cts_2d.ipynb',\n",
       " 'OS_theoretical_cts_2d.pkl',\n",
       " 'OS_theortical_cts_5d.ipynb']"
      ]
     },
     "execution_count": 4,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "from google.colab import drive\n",
    "drive.mount(\"/content/drive\")\n",
    "import os\n",
    "path=\"/content/drive/My Drive/Colab Notebooks/COT-gcopy\"\n",
    "os.chdir(path)\n",
    "os.listdir(path)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 5863,
     "status": "ok",
     "timestamp": 1694679070901,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "FQYGMCvpsMZN",
    "outputId": "fc348a18-9617-472a-f017-b2af58698046"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Collecting celer\n",
      "  Downloading celer-0.7.3-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl (4.5 MB)\n",
      "\u001b[2K     \u001b[90m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m4.5/4.5 MB\u001b[0m \u001b[31m14.8 MB/s\u001b[0m eta \u001b[36m0:00:00\u001b[0m\n",
      "\u001b[?25hRequirement already satisfied: seaborn>=0.7 in /usr/local/lib/python3.10/dist-packages (from celer) (0.12.2)\n",
      "Requirement already satisfied: matplotlib>=2.0.0 in /usr/local/lib/python3.10/dist-packages (from celer) (3.7.1)\n",
      "Collecting libsvmdata>=0.3 (from celer)\n",
      "  Downloading libsvmdata-0.4.1-py3-none-any.whl (7.0 kB)\n",
      "Requirement already satisfied: scikit-learn>=1.0 in /usr/local/lib/python3.10/dist-packages (from celer) (1.2.2)\n",
      "Requirement already satisfied: xarray in /usr/local/lib/python3.10/dist-packages (from celer) (2023.7.0)\n",
      "Collecting download (from celer)\n",
      "  Downloading download-0.3.5-py3-none-any.whl (8.8 kB)\n",
      "Requirement already satisfied: tqdm in /usr/local/lib/python3.10/dist-packages (from celer) (4.66.1)\n",
      "Requirement already satisfied: numpy>=1.12 in /usr/local/lib/python3.10/dist-packages (from libsvmdata>=0.3->celer) (1.23.5)\n",
      "Requirement already satisfied: scipy in /usr/local/lib/python3.10/dist-packages (from libsvmdata>=0.3->celer) (1.11.2)\n",
      "Requirement already satisfied: contourpy>=1.0.1 in /usr/local/lib/python3.10/dist-packages (from matplotlib>=2.0.0->celer) (1.1.0)\n",
      "Requirement already satisfied: cycler>=0.10 in /usr/local/lib/python3.10/dist-packages (from matplotlib>=2.0.0->celer) (0.11.0)\n",
      "Requirement already satisfied: fonttools>=4.22.0 in /usr/local/lib/python3.10/dist-packages (from matplotlib>=2.0.0->celer) (4.42.1)\n",
      "Requirement already satisfied: kiwisolver>=1.0.1 in /usr/local/lib/python3.10/dist-packages (from matplotlib>=2.0.0->celer) (1.4.5)\n",
      "Requirement already satisfied: packaging>=20.0 in /usr/local/lib/python3.10/dist-packages (from matplotlib>=2.0.0->celer) (23.1)\n",
      "Requirement already satisfied: pillow>=6.2.0 in /usr/local/lib/python3.10/dist-packages (from matplotlib>=2.0.0->celer) (9.4.0)\n",
      "Requirement already satisfied: pyparsing>=2.3.1 in /usr/local/lib/python3.10/dist-packages (from matplotlib>=2.0.0->celer) (3.1.1)\n",
      "Requirement already satisfied: python-dateutil>=2.7 in /usr/local/lib/python3.10/dist-packages (from matplotlib>=2.0.0->celer) (2.8.2)\n",
      "Requirement already satisfied: joblib>=1.1.1 in /usr/local/lib/python3.10/dist-packages (from scikit-learn>=1.0->celer) (1.3.2)\n",
      "Requirement already satisfied: threadpoolctl>=2.0.0 in /usr/local/lib/python3.10/dist-packages (from scikit-learn>=1.0->celer) (3.2.0)\n",
      "Requirement already satisfied: pandas>=0.25 in /usr/local/lib/python3.10/dist-packages (from seaborn>=0.7->celer) (1.5.3)\n",
      "Requirement already satisfied: six in /usr/local/lib/python3.10/dist-packages (from download->celer) (1.16.0)\n",
      "Requirement already satisfied: requests in /usr/local/lib/python3.10/dist-packages (from download->celer) (2.31.0)\n",
      "Requirement already satisfied: pytz>=2020.1 in /usr/local/lib/python3.10/dist-packages (from pandas>=0.25->seaborn>=0.7->celer) (2023.3.post1)\n",
      "Requirement already satisfied: charset-normalizer<4,>=2 in /usr/local/lib/python3.10/dist-packages (from requests->download->celer) (3.2.0)\n",
      "Requirement already satisfied: idna<4,>=2.5 in /usr/local/lib/python3.10/dist-packages (from requests->download->celer) (3.4)\n",
      "Requirement already satisfied: urllib3<3,>=1.21.1 in /usr/local/lib/python3.10/dist-packages (from requests->download->celer) (2.0.4)\n",
      "Requirement already satisfied: certifi>=2017.4.17 in /usr/local/lib/python3.10/dist-packages (from requests->download->celer) (2023.7.22)\n",
      "Installing collected packages: download, libsvmdata, celer\n",
      "Successfully installed celer-0.7.3 download-0.3.5 libsvmdata-0.4.1\n"
     ]
    }
   ],
   "source": [
    "!pip install celer"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 14344,
     "status": "ok",
     "timestamp": 1694679085236,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "ybc4BzbN62xN",
    "outputId": "e6065efa-92f1-4680-c9b7-90b8c77e9bd8"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Collecting pykeops\n",
      "  Downloading pykeops-2.1.2.tar.gz (88 kB)\n",
      "\u001b[2K     \u001b[90m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m88.9/88.9 kB\u001b[0m \u001b[31m1.7 MB/s\u001b[0m eta \u001b[36m0:00:00\u001b[0m\n",
      "\u001b[?25h  Preparing metadata (setup.py) ... \u001b[?25l\u001b[?25hdone\n",
      "Requirement already satisfied: numpy in /usr/local/lib/python3.10/dist-packages (from pykeops) (1.23.5)\n",
      "Collecting pybind11 (from pykeops)\n",
      "  Downloading pybind11-2.11.1-py3-none-any.whl (227 kB)\n",
      "\u001b[2K     \u001b[90m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m227.7/227.7 kB\u001b[0m \u001b[31m6.2 MB/s\u001b[0m eta \u001b[36m0:00:00\u001b[0m\n",
      "\u001b[?25hCollecting keopscore==2.1.2 (from pykeops)\n",
      "  Downloading keopscore-2.1.2.tar.gz (84 kB)\n",
      "\u001b[2K     \u001b[90m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m84.5/84.5 kB\u001b[0m \u001b[31m7.6 MB/s\u001b[0m eta \u001b[36m0:00:00\u001b[0m\n",
      "\u001b[?25h  Preparing metadata (setup.py) ... \u001b[?25l\u001b[?25hdone\n",
      "Building wheels for collected packages: pykeops, keopscore\n",
      "  Building wheel for pykeops (setup.py) ... \u001b[?25l\u001b[?25hdone\n",
      "  Created wheel for pykeops: filename=pykeops-2.1.2-py3-none-any.whl size=114071 sha256=03b4df4925650f888b47cd8a34d53782face7eab16959b73a7a5ad86f8943911\n",
      "  Stored in directory: /root/.cache/pip/wheels/93/91/9e/279e56403818cf05d868c2d90a13bde97572bcd11673d6e8ef\n",
      "  Building wheel for keopscore (setup.py) ... \u001b[?25l\u001b[?25hdone\n",
      "  Created wheel for keopscore: filename=keopscore-2.1.2-py3-none-any.whl size=146448 sha256=b63ec1de80ab9b1693d8db39632bb58850a6b84001926ce599ef484ad2529a6a\n",
      "  Stored in directory: /root/.cache/pip/wheels/63/ac/b7/75fb4d24be97d9a930905eddf822cb765ca204b83df7aeaaa9\n",
      "Successfully built pykeops keopscore\n",
      "Installing collected packages: pybind11, keopscore, pykeops\n",
      "Successfully installed keopscore-2.1.2 pybind11-2.11.1 pykeops-2.1.2\n"
     ]
    }
   ],
   "source": [
    "import locale\n",
    "locale.getpreferredencoding = lambda: \"UTF-8\"\n",
    "!pip install pykeops"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {
    "executionInfo": {
     "elapsed": 15,
     "status": "ok",
     "timestamp": 1694679085236,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "tfRHebKAdox5"
   },
   "outputs": [],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 27318,
     "status": "ok",
     "timestamp": 1694679112540,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "6f4hsauPrjGD",
    "outputId": "1e8604cd-151e-48d9-a6e2-a5642df365c1"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[KeOps] Compiling cuda jit compiler engine ... OK\n",
      "[pyKeOps] Compiling nvrtc binder for python ... OK\n",
      "Import recombination2\n",
      "2\n",
      "Importing algorithms.py\n"
     ]
    }
   ],
   "source": [
    "import math\n",
    "import timeit\n",
    "import random\n",
    "import pickle\n",
    "\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as pl\n",
    "# exact solution of Gaussian OT\n",
    "from closed_forms import closed_form\n",
    "# for Sinkhorn, Online Sinkhorn, Compressed\n",
    "import algorithms"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 95,
   "metadata": {
    "executionInfo": {
     "elapsed": 187,
     "status": "ok",
     "timestamp": 1694679822831,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "eutcKoFrrjN5"
   },
   "outputs": [],
   "source": [
    "# generate data and parameters\n",
    "size = 3000 # problem dimension (for Sinkhorn)\n",
    "d = 5 # sample dimension\n",
    "\n",
    "##\n",
    "if d == 1:\n",
    "    mean1, cov1 = 1., 2.\n",
    "    mean2, cov2 = 3., 2.5\n",
    "else:\n",
    "    mean1 = np.random.rand(d)\n",
    "    mean2 = 5*np.random.rand(d)\n",
    "    cov1 = 0.3*algorithms.random_cov_matrix(d)\n",
    "    cov2 = 0.3*algorithms.random_cov_matrix(d)\n",
    "## objective function\n",
    "def loss(f,g):\n",
    "  deg=20\n",
    "  return algorithms.GaussHermiteIntegrate(mean1,cov1,f,mean2,cov2,g,deg)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 96,
   "metadata": {
    "executionInfo": {
     "elapsed": 4,
     "status": "ok",
     "timestamp": 1694679823022,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "Cam41gGGR206"
   },
   "outputs": [],
   "source": [
    "# test distribution\n",
    "def samp1(n, d):\n",
    "    # generate samples from test distribution 2\n",
    "    if d == 1:\n",
    "      x = np.random.normal(mean1,cov1,n)\n",
    "    else:\n",
    "      x = np.random.multivariate_normal(mean1, cov1, n)\n",
    "    return x\n",
    "# test distribution\n",
    "def samp2(n, d):\n",
    "    # generate samples from test distribution 2\n",
    "    if d == 1:\n",
    "      x = np.random.normal(mean2,cov2,n)\n",
    "    else:\n",
    "      x = np.random.multivariate_normal(mean2, cov2, n)\n",
    "    return x\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 97,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 247,
     "status": "ok",
     "timestamp": 1694679823265,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "vL9MUt27399P",
    "outputId": "2aee680a-2360-408a-8aff-961ffc6e212a"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Exact form\n",
      "A mean= [0.87770428 0.41940588 0.93174186 0.66011752 0.1461924 ] covariance= [[0.1864228  0.35132251 0.27147366 0.33949693 0.33145486]\n",
      " [0.35132251 0.7270259  0.56218423 0.69022258 0.63015611]\n",
      " [0.27147366 0.56218423 0.50402204 0.56594546 0.41044567]\n",
      " [0.33949693 0.69022258 0.56594546 0.68453456 0.57056452]\n",
      " [0.33145486 0.63015611 0.41044567 0.57056452 0.6841664 ]]\n",
      "B mean= [4.13232811 3.83830784 2.76850228 3.37476041 4.75089067] covariance= [[0.31237876 0.35326139 0.15913971 0.37397995 0.11915217]\n",
      " [0.35326139 0.53969084 0.25215102 0.39058501 0.13333708]\n",
      " [0.15913971 0.25215102 0.20133576 0.29355989 0.10932277]\n",
      " [0.37397995 0.39058501 0.29355989 0.67909787 0.24593157]\n",
      " [0.11915217 0.13333708 0.10932277 0.24593157 0.09735967]]\n",
      "epsilon is 0.4\n",
      "a= 1.5 ; b= -0.55\n"
     ]
    }
   ],
   "source": [
    "size_var=10\n",
    "size_obj=3000\n",
    "params = {'b': -0.55,# b<a-1 with b=0 corresponding to Sinkhorn\n",
    "          'eta_decay_const':35, # eta=(1+(t/eta_decay_const)^b)\n",
    "          'init_batch':1000,\n",
    "          'batch_const':1., # batch_size=init_batch+batch_const*pow(i, 2*a))\n",
    "          'a': 1.5, # n_t=t^{2a} for a>1+b (number of samples upto iteration t)\n",
    "          'epsilon': 0.4, # regularisation parametr\n",
    "          \"dim\": d,\n",
    "          'maxits': 25,\n",
    "          'no_initial_sinkhorn_its':30,\n",
    "          'compress':True,\n",
    "          'method':\"Fourier\",\n",
    "          'fourier_solver':\"nnls\",\n",
    "          'min_compress':1000,\n",
    "          'get_samples1':samp1,\n",
    "          'get_samples2':samp2,\n",
    "          'zeta':0.9, # compression error rate\n",
    "          'compression_skip':3,# number of steps between compression\n",
    "           #  m = int(10+ (((t/params['compression_const'])**a)/etat)**(1/zeta))\n",
    "          'compression_const':0.3, #\n",
    "          't_sampling':'qmc_gau',\n",
    "#          't_sampling':'qmc_uniform',\n",
    "#          't_sampling':'uniform',\n",
    "          'test_samples1':samp1(size,d),# Sinkhorn\n",
    "          'test_samples2':samp2(size,d),\n",
    "          'test_samples1_obj':algorithms.get_qmc_samples(mean1,cov1,size_obj ),\n",
    "          'test_samples2_obj':algorithms.get_qmc_samples(mean2,cov2,size_obj ),\n",
    "          'test_samples1_var':algorithms.get_qmc_samples(mean1,cov1,size_var ),\n",
    "          'test_samples2_var':algorithms.get_qmc_samples(mean2,cov2,size_var )\n",
    "          }\n",
    "#\n",
    "exact=algorithms.get_gaussian_potentials(cov1,cov2,mean1,mean2,params['epsilon'])\n",
    "params['exact_f']=lambda tt: exact[0](tt)\n",
    "params['exact_g']=lambda tt: exact[1](tt)\n",
    "\n",
    "#\n",
    "if (params['a']-params['b']<1):\n",
    "    print('Params violate a-b>1 condition')\n",
    "    exit()\n",
    "print('epsilon is', params['epsilon'])\n",
    "print('a=', params['a'], '; b=', params['b'])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 98,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 7,
     "status": "ok",
     "timestamp": 1694679823266,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "KW2va0Ju4Qll",
    "outputId": "8c6b07a4-2680-4337-af7e-0acce84e9c7d"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Objective (theoretical)  55.75761363001998\n"
     ]
    }
   ],
   "source": [
    "# Theoretical results for sqeuclidean cost\n",
    "sigma=(params[\"epsilon\"]/2)**0.5\n",
    "if d == 1:\n",
    "  theoretical = closed_form(np.array([[cov1]]), np.array([[cov2]]), sigma, mean1, mean2)\n",
    "else:\n",
    "  theoretical = closed_form(cov1, cov2, sigma, mean1, mean2)\n",
    "#theoretical-=2*params['epsilon']\n",
    "print(\"Objective (theoretical) \",exact[2])\n",
    "#print(\"Objective (Gauss-Hermite quadrature)\",loss(params['exact_f'],params['exact_g']))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 99,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 15923,
     "status": "ok",
     "timestamp": 1694679839184,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "2L38LtCe4hQT",
    "outputId": "88160251-1f8c-4caa-ce34-cf81fee53813"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "running online sinkhorn...\n",
      "Online Sinkhorn with Compression= False\n",
      "Running 30 Sinkhorn iterations to start-up.\n",
      "step is 0 obj is 55.70428995960059 err is 0.49192989990876645 ; eta= 1.0 OS step time 0.02\n",
      "step is 1 obj is 55.72175611730602 err is 0.9281598155266231 ; eta= 0.9846254318831709 OS step time 0.03\n",
      "step is 2 obj is 55.73333555020507 err is 0.5027021566408365 ; eta= 0.9698989209359247 OS step time 0.04\n",
      "step is 3 obj is 55.73654839227082 err is 0.6687128892601741 ; eta= 0.9557767158560582 OS step time 0.05\n",
      "step is 4 obj is 55.741989434882335 err is 0.39538956509064427 ; eta= 0.9422190714177984 OS step time 0.07\n",
      "step is 5 obj is 55.74390573174276 err is 0.3309304284444323 ; eta= 0.9291897901388063 OS step time 0.08\n",
      "step is 6 obj is 55.75131055212441 err is 0.48124349086966056 ; eta= 0.9166558263420831 OS step time 0.09\n",
      "step is 7 obj is 55.75208677971145 err is 0.1542163916357886 ; eta= 0.904586942828011 OS step time 0.11\n",
      "step is 8 obj is 55.755206817628334 err is 0.23618081198640084 ; eta= 0.8929554120991513 OS step time 0.13\n",
      "step is 9 obj is 55.75664293324378 err is 0.4649599426698199 ; eta= 0.8817357554703594 OS step time 0.16\n",
      "step is 10 obj is 55.75748751049716 err is 0.17537383450903254 ; eta= 0.8709045145212391 OS step time 0.19\n",
      "step is 11 obj is 55.760325351877384 err is 0.40612363103841354 ; eta= 0.8604400502623589 OS step time 0.22\n",
      "step is 12 obj is 55.761270606465246 err is 0.09952109706678414 ; eta= 0.850322366133873 OS step time 0.26\n",
      "step is 13 obj is 55.762942544477525 err is 0.23322129857629714 ; eta= 0.8405329515686639 OS step time 0.31\n",
      "step is 14 obj is 55.76335669070548 err is 0.11520061787263991 ; eta= 0.8310546433580427 OS step time 0.36\n",
      "step is 15 obj is 55.76349517204142 err is 0.12701863553691695 ; eta= 0.8218715024770431 OS step time 0.43\n",
      "step is 16 obj is 55.76440631895464 err is 0.23092734426827022 ; eta= 0.8129687043747789 OS step time 0.50\n",
      "step is 17 obj is 55.7648769218342 err is 0.08860363217220524 ; eta= 0.8043324410262209 OS step time 0.58\n",
      "step is 18 obj is 55.7653780317759 err is 0.12269881889311485 ; eta= 0.7959498332855063 OS step time 0.67\n",
      "step is 19 obj is 55.76543132301184 err is 0.07631019008028872 ; eta= 0.7878088522858808 OS step time 0.90\n",
      "step is 20 obj is 55.7657742195041 err is 0.07042189017322364 ; eta= 0.7798982488043699 OS step time 1.04\n",
      "step is 21 obj is 55.76592717502219 err is 0.1013044662162752 ; eta= 0.7722074896557402 OS step time 1.44\n",
      "step is 22 obj is 55.76560487872886 err is 0.07084248346745281 ; eta= 0.7647267003047225 OS step time 1.66\n",
      "step is 23 obj is 55.766026255860595 err is 0.10663665931119581 ; eta= 0.7574466129914648 OS step time 1.90\n",
      "step is 24 obj is 55.766277843507545 err is 0.04500394591849144 ; eta= 0.7503585197557545 OS step time 2.18\n",
      "Time usage for OS is 15.783992772999909\n",
      "Objective (theoretical)  55.75761363001998\n"
     ]
    }
   ],
   "source": [
    "print(\"running online sinkhorn...\")\n",
    "params[\"compress\"] = False\n",
    "start = timeit.default_timer()\n",
    "(ft_os, gt_os, x_os, y_os, obj_os, err_os) = algorithms.online_sinkhorn(params)\n",
    "run_time = timeit.default_timer()-start\n",
    "data_os={\"rt\":run_time,\"obj\":obj_os,\"err\":err_os}\n",
    "print(\"Time usage for OS is\", run_time)\n",
    "print(\"Objective (theoretical) \",exact[2])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 100,
   "metadata": {
    "executionInfo": {
     "elapsed": 7,
     "status": "ok",
     "timestamp": 1694679839185,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "t4y--YKor6ga"
   },
   "outputs": [],
   "source": [
    "# plotting the objective values\n",
    "a = params['a']\n",
    "b = params['b']\n",
    "maxits = params['maxits']\n",
    "batch = [2*math.ceil(pow(i, 2*a)) for i in range(1, maxits+2)]\n",
    "N = np.cumsum(batch)[:len(obj_os)]\n",
    "ref_value=theoretical"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 101,
   "metadata": {
    "executionInfo": {
     "elapsed": 6,
     "status": "ok",
     "timestamp": 1694679839185,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "EWaY4L1Br6ga"
   },
   "outputs": [],
   "source": [
    "save_data={\"N\":N,\"ref_value\":ref_value,\n",
    "            \"os\":data_os,\n",
    "           \"a\":a,\"b\":b,\"d\":d,\"theoretical\":theoretical\n",
    "           }\n",
    "if True:\n",
    "  filehandler = open('OS_theoretical_cts_5d.pkl', 'wb')\n",
    "  pickle.dump(save_data, filehandler)\n",
    "  filehandler.close()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {
    "executionInfo": {
     "elapsed": 6,
     "status": "ok",
     "timestamp": 1694679839186,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "A5iwINx0r6ga"
   },
   "outputs": [],
   "source": [
    "import pickle\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as pl\n",
    "filehandler = open('OS_theoretical_cts_5d.pkl','rb')\n",
    "in_data= pickle.load(filehandler)\n",
    "\n",
    "N=in_data[\"N\"]\n",
    "ref_value=in_data[\"ref_value\"]\n",
    "obj_os=in_data[\"os\"][\"obj\"]\n",
    "err_os=in_data[\"os\"][\"err\"]\n",
    "a=in_data[\"a\"]\n",
    "b=in_data[\"b\"]\n",
    "d=in_data[\"d\"]\n",
    "theoretical=in_data[\"theoretical\"]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 496
    },
    "executionInfo": {
     "elapsed": 896,
     "status": "ok",
     "timestamp": 1694679937369,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "k2DHeq2N-CqH",
    "outputId": "b3284da4-4499-43b3-8bce-eab5039a7723"
   },
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAm8AAAHOCAYAAADOnu5eAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/bCgiHAAAACXBIWXMAAA9hAAAPYQGoP6dpAAClP0lEQVR4nOzdd1xV9f8H8Ne57L2XDHGhIDhQUFQU3CtHWd/co0xLU7OppaZlaWVlhZV7tX9uzS0oblwoIirKVvbecO/5/UFcQe45dy94Px8PHw8453PO+Vyvct98xvvNsCzLghBCCCGE6AWBtjtACCGEEEJkR8EbIYQQQogeoeCNEEIIIUSPUPBGCCGEEKJHKHgjhBBCCNEjFLwRQgghhOgRCt4IIYQQQvSIobY7QFRPJBLhyZMnsLKyAsMw2u4OIYQQQmTAsixKSkrQqlUrCATc42sUvDVDT548gaenp7a7QQghhBAFpKWlwcPDg/M8BW/NkJWVFYC6N9/a2lrLvSGEEEKILIqLi+Hp6Sn+HOdCwVszVD9Vam1tTcEbIYQQomekLXmiDQuEEEIIIXqEgjdCCCGEED1CwRshhBBCiB6h4I0QQgghRI9Q8EYIIYQQokcoeCOEEEII0SMUvBFCCCGE6BEK3gghhBBC9AgFb4QQQggheoSCN0IIIYQQPULBGyGEEEKIHqHgjchFJBIhPz8ftbW12u4KIYQQ0iJR8EbkcvHiRbi6usLJyQkff/wxYmNjIRKJtN0tQgghpMVgWJZltd0JolrFxcWwsbFBUVERrK2tVXrvsWPH4uDBg42O2dvbY8CAAQgPD0dYWBg6d+4MgYB+LyCEEELkIevnNwVvzZC6grdHjx7Bx8dH6kibo6OjOJgLDw+Hr68vGIZRWT8IIYSQ5oiCtxYoIiICEREREAqFePDggUqDN6FQiNDQUFy6dEnua52dnREWFoawsDCEh4ejY8eOFMwRQgghz6HgrQVTx8hbeno6QkJCkJ6ervS9XF1dxcHcwIED0aFDBxX0kBBCCNFvFLy1YOqaNo2NjcWsWbNw48YNld1zxIgR+Pfff1V2P0IIIURfyfr5TavKicy6du2K69ev4+TJk2jVqhVcXV0xf/58BAYGKjwNGh4eruJeEkIIIc2bobY7QPTP4MGDcffuXTx58gR+fn4AgIKCAkRHRyMyMhJRUVGIjY2FLIO6YWFhUtt8+OGHyMzMFK+Z8/b2VvIVEEIIIfqLpk2bIXWmCpFVfn4+zp07h8jISOzfvx+pqalN2lhbWyMvLw+Ghty/Q7AsC29v70bXe3t7iwO5sLAweHl5qeU1EEIIIZpEa95aMF0I3uplZGTA398fhYWF8PDwQN++fXH37l3ExcVh9OjROHToEO/1jx8/Rrt27XjbtG3btlEw5+HhocqXQAghhGiErJ/fNG1K1IZlWcyaNQuFhYUA6nas7tmzBytWrMDx48dRVlYm9R5RUVFS2zx+/BiPHz/G1q1bAQDt27cXB3JhYWFo1aqVMi+DEEII0Sk08tYM6crI288//4y33npL4rmjR49i+PDhUu8xdepU7N69W6l++Pj4NArmXF1dlbofIYQQog6025RoVVlZGT755BOJ5yZPnixT4AYArVu3VjoP3IMHD/Drr79i4sSJcHNzw4kTJ5S6HyGEEKJNNPLWDOnKyNvdu3cxbdq0Rnnh3N3dcefOHdjZ2cl1r/T0dJw9e1a8m/XRo0cK9YlhGOTm5sLe3l6h6wkhhBB1oZE3onWdO3fG5cuXsWzZMhgYGAAAtm7dKnfgBgAeHh6YPHkyNm/ejMTERKSkpGDnzp2YOXMm2rRpI/N9unbtKjVwKy0txcKFC7Fv3z7k5eXJ3VdCCCFEnWjkrRnSlZG3hq5evYrTp09jyZIlarl/cnIyoqKiEBUVhcjISImpSQDgnXfewbfffst7r2PHjmHEiBEA6kbqunTpIt7N2r9/f4WCT0IIIUQaShXSguli8CaPnJwcXL16FaNGjVLoepZlkZycLJ5ijYyMFNdkPXDgAMaMGcN7/UcffYS1a9dKPMcwDLp169YomLOxsVGon4QQQkhDFLy1YPocvLEsiwkTJmDv3r147bXX8O233yr9GliWxePHjxEVFYUJEyZIDbZ69eqFq1evynRvgUCA7t27i3ezhoaG6t3fOSGEEN1AwVsLps/B2+7duzF16lTx997e3tixYwf69++vkecXFxfD3t4eQqFQoesFAgF69OghDub69esHKysrFfeSEEJIc0TBWwumr8FbWloaAgICUFRU1Og4wzDYtm0bpk+frvY+3Lp1C8OGDUN2drZK7mdgYICRI0fi4MGDKrkfIYSQ5ot2mxK9IhKJMGvWrCaBGwA4ODjInBdOWd26dUNmZibi4+MRERGBCRMmwNHRUeH7CYVCmJmZqbCHhBBCWjoqj0V0AsMwmDhxIi5fvozS0tJG53755Re4uLhotC++vr7w9fXFW2+9BZZlcffuXfHmh7Nnz8qVQiQ8PFxqm8uXL6OsrAx9+vShYI8QQggvmjZthvR12hQAkpKSMGPGDJw7dw5AXXmsnTt3arlXjYlEIsTFxTUK5goKCjjb37t3D506deK954svvoh9+/bB2NgYvXr1Eq+ZCwkJgampqapfAiGEEB1Ea95aMH0O3oC64Oi7777DL7/8gpiYGNja2mq7S7xEIhFu374tDubOnTuHwsJCAICrqyuePHkChmF4r3dyckJ+fn6TcyYmJujdu7c4mOvduzdMTEzU9VIIIYRoEQVvLZi+B2/1qqurYWxsLFPbgoICnUmeKxQKERsbi6ioKIhEIrz33nu87WNjY9GtWzeZ7m1qaoqQkBBxMBccHEzBHCGENBOyfn7Tmjeis2QN3A4cOIDp06fjp59+wuTJk3lHuTTBwMAAgYGBCAwMlKl9ZGSkzPeurKxEZGSk+BozMzP06dNHHMwFBQXJ/PdGCCFEP9FuU6LXsrOzMXv2bBQVFWHq1Kl4+eWXkZOTo+1uyeXixYsKX1tRUYHTp0/jk08+Qb9+/WBnZ4ehQ4fi7NmzKuwhIYQQXULBG9FbLMti7ty5jYK1PXv2ICAgAP/++68Weyaf3377DZcuXcKXX36JoUOHwtzcXOF7lZeX4+TJk6itrVVhDwkhhOgSCt6I3tq1axf27dvX5HhWVpbKkuxqgpGREXr37o2PPvoIx48fR0FBAS5cuIDPP/8cgwcPljt1iJGREUJCQqS2i4uLU7iSBCGEEO2hNW9Eb3HVHx07dqxGqjGoi7GxMfr06YM+ffrg448/RlVVFWJiYhAZGYmoqChcvHgRlZWVnNf37t1b6uhdTk4OAgICYG1tjf79+yMsLAzh4eHo2rUrDAwMVP2SCCGEqBDtNm2GmstuU1ns2bMHc+bMESfNdXJyQlxcHJydnbXcM/WprKzElStXxKlJLl26hOrqavH5ZcuWYdWqVbz3+Oeff/DKK680OW5ra4v+/fuLN0B06dIFAgEN0BNCiCZQqpAWrCUFbwCQmZmJN954A4cOHcLevXsxfvx4bXdJoyoqKnD58mVxMPf555+jf//+vNfMmzcPGzZskHpvOzs7DBgwQBzM+fv7UzBHCCFqQsFbC9bSgjegbvPCuXPnMGDAAG13RS/4+fnh3r17cl/n4OAgDubCw8Ph5+en9dQshBDSXFDw1oK1xOBNHkVFRZg1axa++OILdOzYUdvd0bisrCy4urqq5F5OTk4ICwtDWFgY5s6dS6NyhBCiBFk/v+knLWlxFi5ciL1796J79+748ccfIRKJtN0ljbKxscGxY8fw4YcfolevXkptUMjJycE///yDdevWUeBGCCEaQiNvzRCNvHHbt28fXnzxxUbHBg8ejK1bt8LT01NLvdKukpISnD9/Xrxm7vr163IHtK+99ho2b97M26a6uhpGRkY0zUoIIRxo5I2Q52RnZ2POnDlNjp86dQoLFizQQo90g5WVFUaMGIG1a9fi6tWrKCgowOHDh/Huu++iR48eMo2ohYWFSW3zzTffwNPTE1OmTMHmzZvx6NEj0O+OhBAiPxp5a4Zo5E2yt99+Gz/99FOT49bW1rhz5w68vLy00CvdV1hYiOjoaHGeuVu3bjUJutLS0uDh4cF7nyFDhuDUqVONjnl6eopzzIWFhaFNmzYq7z8hhOgL2rDQglHwJll5eTmWLFmCH374odHxHTt2YNq0aVrqlf7Jz89vFMxVVFTg/v37vNdUV1fD1tYWFRUVvO1at27dKJhr3bq1KrtOCCE6jYK3FoyCN36nTp3CzJkzkZ6ejvHjx2PPnj20DksJlZWVMDU15W1z4cIF9OvXT+57t2nTplEw11LXJRJCWgZZP7+pPBZpcQYPHow7d+7gk08+wYoVKyhwU5K0wA0AIiMjFbp3UlISkpKSsG3bNgBAu3btxMHcoEGDVJbyhBBC9AltWCAtkq2tLX766Sc4OTnJ1H7fvn289UQJv9DQULz55pvw9fVV6j6PHj3Cli1bMGXKFOzYsUNFvSOEEP1CI2+ESHHy5Em8+OKL8PPzw65duxAYGKjtLumdAQMGiKtfZGZm4uzZs+I1c9LWy3EJDw9XZRcJIURv0Jq3ZojWvKlOQUEBAgICkJGRAQAwNDTEihUr8NFHH8HQkH73UYUnT54gKipKnGcuMTFR6jVWVlbIz8/nfQ9YlsXQoUPRoUMH8Zo5WUdaCSFEG2jDQgtGwZvqTJ06Fbt3725yPDw8HKdPn6b1cmqQnp7eKJh7/PhxkzYjR47EkSNHeO+TlJSEtm3bNjrWuXNncSA3YMAAODo6qrTvhBCiDNqwQIiS9uzZIzFwA4CXXnqJAjc18fDwwJQpUzBlyhQAQGpqaqNgLjk5WaYp06ioqCbH7t69i7t374rz/XXp0kW8AaJ///6wt7dX6WshhBB1oJG3ZohG3lQjPj4eU6dOxY0bNxodHzJkCI4dO0a1PLUkOTkZFhYWUqdAp02bhl27dsl8X4Zh0LVrV3EwFxoaCjs7O2W7SwghMqNp0xaMgjfVqampweeff47Vq1dDKBTCxsYGcXFxUqsJEO1iWRatW7dGWlqawvdgGAbdu3dvFMzZ2NiosJeEENIYBW8tGAVvqnf16lVMnToVy5YtE0/nEd1VXl6O2bNnIzIyEk+fPlXJPQ0NDZGXl0f/pwghakPBWwtGwZt6VFVVwdjYWKa1bkVFRTRKowNYlsWDBw/E6+WioqKQlZWl0L26du2KW7duqbaDhBDSgKyf37RohxAZmZiYyBS4lZWVoWfPnnjttddQXFysgZ4RLgzDoGPHjpgzZw7+/PNPPH36FPfu3cOGDRvwyiuvwNnZWeZ7ybJJ4sqVK1i6dClOnDiBsrIyZbpOCCGcVDLy9vTpU1y8eBHp6enIyclBXl4ezMzM4OTkBCcnJwQEBKBHjx6UF4tHTEwMVqxYgUuXLqG6uhqdO3fGokWLMGnSJLnvRSNv2jVv3jxs2LABAODt7Y3t27eLE9QS3cKyLO7duycelYuKikJubq7Etvv378fYsWN577dkyRKsWbMGQN00a3BwsDg1SZ8+fWBubq7y10AIaT7UOm3KsixOnTqFv/76C1FRUUhKSpJ6jZmZGXr16oVRo0Zh4sSJcHNzk/exzVZUVBSGDRsGY2NjvPrqq7CxscHevXuRlJSE1atXY+nSpXLdj4I37Tl+/DiGDx/e6BjDMHjnnXfwxRdfwMTEREs9I7IQiUSIj49HZGQkIiMjcfbsWeTn54NhGOTl5UndfRoSEoLLly9LPGdkZIRevXqJg7mQkBCYmZmp42UQQvSUWoK3srIy/PLLL4iIiEBKSgqAukBOHgzDwMDAAGPHjsXixYsREhIi1/XNTW1tLTp16oT09HRcunQJ3bt3BwCUlJQgJCQE9+/fR3x8PDp06CDzPSl4046CggL4+/vjyZMnTc6FhITg3LlzNPqsZ0QiEe7cuYPbt29j6tSpvG1LSkpgZ2cHoVAo072NjY3Ru3dvcTDXu3dvmJqaqqLbhBA9pdLgrba2Fj/++CO+/PJL5OXliQO2Nm3aoFevXggODkaPHj3g7OwMe3t72NnZoaKiAvn5+SgoKMCDBw8QExODq1evIiYmRlzgm2EYDBs2DF999RX8/f1V9NL1y4kTJzBs2DDMnDkTW7dubXTur7/+wquvvoolS5bgiy++kPmeFLxpR1RUFMaMGYOSkpJGx83NzXHr1i25AnCif44dO4YRI0YofL2JiQlCQkLEwVyvXr1opJaQFkbmz29WBh07dmQFAgHLMAzr4eHBvvvuu+yNGzdkubSJkpISdseOHeywYcNYQ0NDlmEY1tDQkN21a5dC91NGVlYWe+jQIXbZsmXs8OHDWQcHBxYAC4CdPn26XPdKSUlh3333XbZTp06subk5a2dnxwYFBbFff/01W1ZWxnndkiVLWADsH3/80eRcfn4+C4Dt06ePXH0pKipiAbBFRUVyXUeUl5SUxA4YMED87wgAu2HDBm13i2jA1q1bWSsrq0bvvTJ/zMzM2JUrV2r7ZRFCNEjWz2+ZgjeGYdjOnTuzv//+OysUClXSQZatC3jeeOMN1sTERCs/pPh+cMoTvB0+fJi1sbHhvFfHjh3ZR48eSbx2woQJLAD22rVrEs87OjqyTk5Ocr0uCt60SygUst9++y1rYmLCDhs2jBWJRNruEtGQmpoa9urVq+zatWvZESNGsJaWlkoFcBEREdp+SYQQDZL181umadN//vkHEyZMUFstx4yMDKSmpmp8/VvD1+Pp6QlfX1+cOHECADB9+nRs375d6j1iY2PRp08flJeXw9LSEkuWLEF4eDgqKirw559/YtOmTQCATp06ISYmBpaWlo2uHzp0KE6ePImHDx+iffv2Te7frl07pKeno6qqSubXRdOmuiE+Ph42NjZwd3fXdleIltTU1ODGjRvi3aznz5+XK4VIfHw8fH19edvs27cPrVq1oh39hDQDKi1M//LLL6usY5K4u7tr5QNu+fLlCAoKQlBQEFxcXJCcnIw2bdrIdY9FixahvLwchoaGOHHiRKMAdODAgejQoQM++OADJCQk4Ntvv8Xy5ctV/TKIjvLz85O57erVq+Hl5YUpU6ZQwftmpH6Haa9evfDRRx+hpqYGMTEx4qTBFy5cQEVFhcRrXVxc0KlTJ977i0QizJ49G3l5ebCyskK/fv3Ea+a6d+9OwRwhzZVGxgH1RFJSklzTplevXhW3nzNnjsQ2QqGQ9fX1ZQGwdnZ2bHV1daPzNG1KoqOjWYZhWADsiy++yGZnZ2u7S0RDqqqq2OjoaHbVqlXswIEDWVNTU/HPlP/9739Sr4+NjeWccrW2tmZHjRrFfvPNN+y1a9fY2tpaDbwiQogyZP38pgoLSti/f7/465kzZ0psIxAIMG3aNAB1qSSioqIana/fgfjw4cMm1xYUFCA3N5d2KTZjpaWlmD59ungH9969e+Hv74+DBw9quWdEE4yNjdGvXz8sW7YMp0+fRmFhIc6dO4eVK1di8uTJUq9//udJQ8XFxThy5Ajee+899OzZEw4ODhgzZgy+++473Lx5EyKRSIWvhBCiSRS8KSE6OhoAYGFhgR49enC2a5hd//z58xLP1a+1a6j+GGXnb77ee+89PH78uNGx7OxsvPzyy0hPT9dSr4i2mJiYIDQ0FMuXL8cLL7wgtX1kZKTM9y4qKsKhQ4ewePFiBAYGwtHREePGjcP69esRGxtLwRwhekSmBRGzZs1S+YMZhsGWLVtUfl9NunfvHgCgffv2vGtLGq5bqb+m3qBBg9C2bVv8/vvvWLBgAbp16wagLuHnZ599BkNDQ8yYMUPlfSfaFxsbi19//VXiuVWrVsHDw0PDPSL6hGVZzmoOsigoKMCBAwdw4MABAIC9vT0GDBiAH3/8kTbZEKLjZAretm/frtJF1CzL6n3wVllZKa6BKO1D1s7ODhYWFigrK0NaWlqjc4aGhti8eTOGDRuG0NBQTJw4EdbW1uLyWJ9//jl8fHx4719VVdVoNyoVQ9cPXbt2xZ49ezBnzpxG9TT79u2L9957T4s9I/qAYRg8evQIFy9eFG+AuHr1KmpraxW6X35+Pg4fPozdu3eruKeEEFWTKXjz8vKiHXDPaZhF//n0H5LUB2+lpaVNzoWHh+P8+fNYsWIF/v77b3Fh+s8++0ymdS9ffvklVq5cKd8LIDrhxRdfRN++fTF79mwcOnQIFhYW2LFjBwwMDLTdNaIHzM3NMXjwYAwePBhAXQnDCxcuiIO5mJgYmct1AUCvXr1gbm4u8Vz9uszS0lJkZGSgY8eO9LlAiJbIFLwlJyeruRv6p77EF1C36Fia+jI3XGkBgoODcfToUYX6smTJEixevFj8fXFxMTw9PRW6F9E8FxcXHDhwQDzC3a5dO213iegpCwsLDB06FEOHDgVQ90tmw2Du2rVrvGvbwsLCOM/VB2onTpzAhAkT4OrqirCwMISFhSE8PBwdOnSgYI4QDaEkQApqWEC6urpaavv6aU0zMzOV98XExEQjNRB/OP0QvdrYo1dbB7U/q6VhGIZzx7IkcXFxMDQ0lJoHjLRsVlZWGD58OIYPHw6g7he78+fPi5MG37hxo1EwFx4eLvWe9ZskMjMz8eeff+LPP/8EALRq1UocyIWFhaFdu3YUzBGiJhS8KcjKykr8taSp0OfVZ1WXZYpVF+29kY5vTz6AoYDBp2M6Y0rv1truUotVWVmJiRMnIjExEV999RXmzZsHgYA2jhPprK2tMXLkSIwcORIAUFhYKA7mzp8/j969e0u9B1d6kidPnuD333/H77//DqBuLXDDYK5NmzYUzBGiIhS8KcjU1BSOjo7Izc2VmtKhoKBAHLzp43Tm7fRCfLT3DgCgVsTik/1xiH9ajE9f6AxjQwoaNG358uWIi4sDACxYsAAHDhzAtm3b9PLfFtEuW1tbjB49GqNHj5apfXZ2Nu7evStT2/T0dOzevVu8AcLLy6tRMOft7a1otwlp8eiTVwn1NQcTExN5d3glJCQ0uUZfZJdU4o2d11Fd23idzO9XUjF582XklMhec5UoLzo6Gt98802jY6dPn4a/vz9u3LihpV6RluLs2bMKX5uamoqdO3di5syZaNOmDdq0aYNZs2aJU5UQQmSnkuCturoa27Ztw9ixY+Ht7Q1LS0sYGBjw/mkONff69esHoG5K9Pr165ztGv7A69u3r9r7pSpVtUK8ufsGMosrJZ6PSS7AmJ/O4056kYZ71jKVlJQ0qsbQUNu2beHv76+FXpGWZPDgwfi///s/zJ8/X+l/b8nJydi2bRtVEyFEAUoHbw8ePEC3bt3w+uuv49ChQ0hNTUV5eTlYlpX6R9+NGzdO/PW2bdskthGJRNi5cyeAuikKWRYE64o91zNwPaWAt83TokpM+OUiDtzK0FCvWi5DQ0OJWfeNjY2xa9cumXY9E6IMOzs7vPTSS/jxxx9x584dZGVl4e+//8Zbb72l8KwC3w7Xeg3zWBJClAzeysrKMGLECCQkJIBhGIwbNw6zZ88GULd7btmyZZg/f754ESzDMOjTpw9WrFiB5cuXK997LQsODkZoaCgAYMuWLbh06VKTNuvWrRNXVVi4cCGMjIw02kdlTAz2xLtD+BMEA0BVrQgL/7yFL/69B6FI/4NyXWVmZob169fj1KlTjRJDr169mkbdiFY4Ozvj5ZdfRkREBOLj48U7UOfOnYuOHTvKdA9ZgrcxY8agY8eOmDt3Lv78809kZmYq2XNC9BvDKjEEtm7dOrz//vswMDDA8ePHMXDgQNy9excBAQFgGKZRcshbt25hypQpSEhIwPfff4/58+er5AUo4/z580hMTBR/n5ubi/fffx9A3fTm66+/3qi9pDJVN2/eRN++fVFRUQFLS0ssXboU4eHhqKiowJ9//omNGzcCAHx8fHDt2rVGu1TVpbi4GDY2NigqKoK1tbXS9zsZn4V3/rqF0irpmdv7+zjhx1e7w8Zcf4JUfVRYWIiFCxciKSkJkZGRlNSX6KSnT58iKipKnGfu4cOHjc63a9eu0c9gSaqrq2FnZ4fy8vJGxzt16iTe/BAWFgZnZ2eV958QTZP181up4C0sLAzR0dF49dVX8dtvvwEAZ/AGADk5OejatStyc3Nx6dIl3mLumjBjxgzs2LFD5vZcf1WHDh3ClClTOMtS+fj44MiRI2jfvr1C/ZRVREQEIiIiIBQK8eDBA5UFbwDwMKsEs3deQ3JeudS23g7m2DStJzq4qD9QbekqKipkyh1YW1uLmpoateQZJERWGRkZ4kAuKioK4eHh2LRpE+81Fy5cEK8v5tO5c2fxbtYBAwbA0dFRVd0mRGNkDd6UmjaNj48HAIwfP17i+eeDHScnJyxevBi1tbX46aeflHm0TnnhhRdw+/ZtvPPOO/Dx8YG5uTlsbW3Rs2dPrF27Fjdv3lR74AYA8+bNQ3x8PGJiYlR+7w4uVjgwrx/6+zhJbZucV47xGy7iZHyWyvtBGpM1GFuzZg169OjBu7GGEHVzd3fH5MmTsXnzZiQmJiIiIkLqNVx55Z539+5dREREYMKECXByckKXLl2wYMEC7Nu3D3l5eUr2nBDdotTIm7GxMYRCIS5fvoygoCAAdWkzfHx8wDAMCgsLm0wTXrp0CX379oW3tzceP36sXO+JRKqeNm1IKGLx1bEE/HpOtvdu8RAfzA9vD4GAknNqy40bN9CrVy/U1tbC0NAQy5Ytw5IlS/Rq/SVpuQYPHozTp08rdQ+GYdClSxfxFOugQYM0soSFEHlpZOStvoBxw6zZtra24q9TU1ObXFPflhac6icDAYMlI32x/tVuMJEhQe+3Jx9g3u83UCbDejmiepWVlZg6dao4D2FtbS1WrFiBvn374v79+1ruHSHSzZo1C1OnTm20SUdeLMsiNjYW69evx/jx42VONEyIrlIqeGvTpg2AurIo9RwdHWFvbw+gbq3C8+qnbSitgX4b280d/ze3D9xsTKW2PRqXiZd+voi0fOnr5YhqffLJJ+LlDQ3duHEDRUWUn4/ovkmTJmHnzp1ITU1FYmIiNm3ahMmTJ6NVq1YK3c/S0lLr660JUZZSwVvPnj0BANeuXWt0fNCgQWBZFl9//XWjtQbJyclYu3YtGIZBt27dlHk00QEBHjY4OL8fgrztpLZNyCzBCz+dx4XEXA30jAB1u/TOnTsn8dzSpUsRHBys4R4RojiGYdCuXTu8/vrr2L17N9LT0/HgwQP8+uuvmDhxIlxdXWW6T2hoqNQlA8nJyQgODsYHH3yAo0ePoqSkRBUvgRCVUWrN299//41XX30VXbp0wa1bt8THL1y4gNDQUDAMA1tbWwwcOBDl5eU4f/48SkpKwDAMdu3ahUmTJqniNZDnqHPNmyTVtSJ8eugufr/SdJr8eQYCBp+M8sWMPt5UpFoDampqsHr1anz++efi3d+BgYG4dOkSjX6TZoVlWTx48EC8kzUyMhLZ2dlN2q1duxYffPAB7722b9+OmTNnir83MDBAz549xbtZ+/btC0tLS5W/BkI0kiqkvLwcI0eOhFAoxPbt29GuXTvxuU8//RSrVq2qe8h/H9L1j5o1axY2b96s6GOJFJoO3urtvpyCTw/eRa0MiXon9PDA5+P8YWpE+ck0ISYmBlOnTkVycjKuX7+Ozp07a7tLhKgVy7JISEgQB3NRUVHIycnB1atXxRvsuEhLI2VoaIigoCBxnrm+ffuK14ATogyNBG/SnD59Gps3b8bdu3dRW1uLDh06YNq0aXjppZfU9UgC7QVvAHA1KR9v7r6OvLJqqW27edri16k94GItfd0cUV55eTmuXLmiVyXaCFEVlmURHx+Pjh078tbWZlkW3t7eEjfccTEyMkJwcLA4mOvTpw/lVCQK0YngjWiHNoM3AMgorMCcXdcQlyE5aXFDzlYm+GVqDwR6SV83RzSnvhLKV199pZV/Q4RoS1JSEtq2bavUPYyNjdG7d2/xNGvv3r1hakq/pBLpNJIqhOiWiIgI+Pn5SZ0SUDd3WzP8M6cPxnSVvhssu6QKr/56GX9fS9NAz4gsampqMHXqVPz666/o0qULzp49q+0uEaIx5eXlGDNmDGxsbBS+R/1moVWrViE8PFymChGEyING3pohbY+81WNZFhvPPcaaYwmQ5V/ZjD7e+HiUL4wM6HcKbVq1ahVWrFgh/p5hGLzzzjtYvXo1jR6QFkMoFCI2Nla8+eHcuXOcJRClWbBgAdavX6/iHpLmSGvTpsXFxSgpKWlS11QSLy8vVT6a/EdXgrd6Ufez8fYfN1FSKT1Rb0hbB0RMDoS9Be2E1IZr164hJCREnNS3odWrV2Pp0qVa6BUh2icUCnHz5k1xMBcdHS1zCpF9+/Zh3LhxvG22b9+OjIwMhIeHIygoiCqgtFAaDd5OnjyJDRs2IDo6GgUFBTJdwzCMxA8IojxdC94A4HFOKWbvvIZHOWVS23rYmWHTtJ7wddONvrcULMsiJCQEV65caXKuU6dOuHHjBi3CJuQ/tbW1uHHjRqNgrqys6c83hmGQm5srTl7PpU+fPrh06RKAuupF/fr1E2+A6NGjBwVzLYTGgrcFCxaIiwvLcyuGYWQanSPy08XgDQBKKmvwzl+3cOpe09xLzzMzMsC6V7piZICbBnpG6iUnJ2PGjBmN1rkZGBjg8uXL4qTchJCmampqcP36dXEwd/78eZSXl6Nbt264efMm77WlpaWws7PjHNCwtLREv379xBsgAgMDeXfMEv2lkeDt999/x5QpUwAApqamGDduHHr06AF7e3sIBNLXLU2fPl3RRxMeuhq8AYBIxOK7Uw/w45lEmdq/PbA93hnsQ4XtNUgkEmH9+vVYsmQJqqqqsGLFCnz66afa7hYheqW6uhrXrl1DWVkZhgwZwtv2+PHjGD58uMz3trKyQmhoqDiY6969OwwMKGdmc6CR4G3AgAGIjo6Gp6cnzpw50yhJL9EeXQ7e6v175yne/TsWFTXSR18H+zrju/91g5UpTRtoUnx8PNatW4dffvmFpmwIUaMlS5ZgzZo1Cl9vbW2N/v37i6dZu3btSsGcntJIqpDbt2+DYRisWLGCAjcil5EBbtjzZh942ElfQ3XqXjbGb7iIpFzp6+WI6vj5+WHLli0yBW4ikQj//vuvXEsnCCF1SkpKYGJiovD1xcXFOHz4MN5991306NEDjo6OOHz4sAp7SHSNUsFbTU0NAKB79+4q6QxpWfxaWePg/H4IaesgtW1idinG/nQeUfelr5cjmrd+/XqMGjUKEyZMQE5Ojra7Q4he+emnn1BQUIAzZ85g+fLlCA0NVWq0u7CwEG3atFFhD4muUWra1N/fH/fu3UNUVBRCQ0NV2S+iBH2YNm2oRijC6iP3sP1istS2Agb4cHgnvNG/LRW21xHx8fEIDAxEVVUVAMDZ2RmbNm3CmDFjtNwzQvRXeXk5Ll26JN4AcfXqVfGAiTTOzs7IzMzk/RkpEomwceNGhIaGws/Pj36e6giNTJu++OKLAOpqmBKiKCMDAT4d0xlfTegCYykJekUs8OXRBCz66xYqZVgvR9SrvhpDfeAGANnZ2Rg7diw+/vhjLfaMEP1mbm6OQYMG4bPPPsP58+dRUFCAkydPYunSpQgJCeHdbRoWFiY1GLt79y7efPNN+Pv7w8XFBa+88go2bNiAe/fu0fIHPaDUyFtRURG6deuGgoICXL58GZ06dVJl34iC9G3kraEbqQWYu+s6skuqpLb1d7fGr1N7wt2Wco9py4oVK7Bq1aomxxmGQWRkJAYMGKCFXhHS/JWWluLChQvikblr166J029t2LABb775Ju/1P/74IxYsWCDxnIuLC8LCwsS7WX18fGhkTkM0luft/v37GDNmDHJzc/HZZ59h4sSJsLOjIuPaEBERgYiICAiFQjx48EAvgzcAyCquxJxd13ErrVBqW0dLY2yY3APBbfgTYJLGMosqEf0wB5U1QvRq6wAfFyuF7rNv3z688cYbyM3NbXR88eLFWLdunSq6SgiRQUlJCc6fP4+oqCi88cYbUjcRvvTSS9i7d69M93ZzcxMHcmFhYWjfvj0Fc2qikeCtbdu2AOrm5rOzs8EwDBiGgaOjI8zNzXmvZRgGjx49UvTRhIc+j7zVq6wR4uN9cdhzI11qW0MBg5VjO2Nyr9Ya6Jl+yyiswE9nHuKfa+moFdX91xcwwPvDOuHNMMV2jGdlZeGNN97AwYMHAdTtUr1+/TrVQSVER4lEIjg5OSE/P1+h693d3RsFc23b0hpkVdFI8CZLIl7OB1OFBbVpDsEbUFexY9uFZKz+9x6EIun/TCf38sKKFzrD2JAK2z8vq7gSEZGJ+PNqGqqFIoltVo3tjGkh3grdn2VZbN++He+99x5OnDiBHj16KNFbQog63b9/H76+vipb2+bp6YkhQ4Zg8+bNFMQpSSPB28yZMxW9FACwbds2pa4nkjWX4K3ehcRczPv9BgrLpe+0CvK2w4bJPeBkpXjOpOYkt7QKP0c9wu7LKaiqlRy01TMQMNg6IwgDfJwUfl5ZWRksLCxkalteXi51hJ4Qoh75+fmIjo5GZGQkoqKiEBsbq9T9+vbti/Pnz6uody2XRgvTE93S3II3AEjNK8cbu64hIbNEals3G1NsnNoTAR42GuiZbiooq8av5x5jx8VkmapY1LMyMcTet/qgg4Jr4GSVlJSE3r174+OPP8b8+fOVGsUnhCgvLy8P586dEwdzd+7ckev6jz/+GJ9//jlvm9TUVAgEAnh4eCjT1WaNgrcWrDkGbwBQVlWL9/6JxdG4TKltTQwF+GpCF4zt5q6BnumOoooabDmfhK3nk1BaJbnItTSe9mbY/1ZfOFiqZ/RSKBQiPDwc0dHRAICBAwdi27Zt8PLyUsvzCCHyy8nJaRTM3b17l7f9qVOnMGjQIN42CxYswI8//oh27dqJ18uFh4ejVatWquy6XqPgrQVrrsEbUFfY/qfIRHx78oFM7ef0b4sPhneCQTMvbF9aVYvtF5Kw8dxjFFcqFrQ1FORth92v94KJoerrI65btw7vvfdeo2PW1tb48ccfMW3aNJU/jxCivOzsbERFRYlTkyQkJIjPGRsbo6CgQOoyiICAAMTFxTU57uPjIw7kBgwYADc3N5X3X19oLXjLyspCXFyceBeLvb29OAkg0YzmHLzVO3E3E+/8dQtl1dKnBAf4OOGHV7vDxrz5FVevqBZi56Vk/HL2EQpkWBMoj5cCPfDNy11UugA5Li4OPXr0QHV1dZNzr7/+OjZt2qSyZxFC1CczM1MczFVUVGDHjh287XNycuDs7CzTvTt16tQomGtJ8YNGgzeWZbFx40b89NNPiI+Pl9jGz88Pb7/9NmbPnk27UdSsJQRvAPAgqwRv7LyG5LxyqW29HcyxeXpPtHdW71ouTamsEeL3K6nYEPUIuaXSExo/b7CvC0yMBDhy+ylvuw+Gd8RbYe0V7WYT33//Pd55550mx729vXH79m1YWTWP94cQ0tj//d//4eWXX1boWj8/v0bBnJOT4puqdJ3GgreCggK88MILuHTpEgBwbj2uD9j69OmDQ4cOwdbWVpnHEh4tJXgDgKLyGsz/4waiH+ZKbWtpYojv/9cNg/3097e46loR/rqWhogzicgsrpT7+gE+Tlg8xAddPW1RXSvC1C1XcCWJP9fTL1MCMdxfddMYp0+fxsyZM5GWlgag7mfD2bNnqT4yIc3Y/PnzERERoZJ7+fv7IywsDNOnT0fPnj1Vck9doZHgjWVZDBgwQLw92MHBAa+88gp69eoFV1dXsCyLrKwsXL16FX///Tdyc3PBMAz69euHs2fPKvpYIkVLCt4AQChi8dWxBPx67rHUtgwDLB7sg/kD9StDeI1QhL030vHD6URkFFbIfX2fdg5YPMQHPb0bV6IoKKvGuA0XkMIzemlmZIB/5obA3111u3eLioqwcOFC7NixA++//z6++uorld2bEKJ70tPTcebMGURGRiIyMhIpKSlK33PHjh3Nbp2sRoK33377DVOnTgXDMJg0aRI2bNjAOe1RWlqKefPmYdeuXWAYBrt378bEiRMVfTTh0dKCt3r7b2bgwz23peYzA4CRAa74ekJXWJhwF3fWBUIRiwO3MrD+9EPeAItLz9Z2WDzUB33aOXK2ScwuxfgNF1DCs9HBxdoEB+b1g6uNaqsmHD16FOHh4VSNgZAWJjk5WbyTNTIyUjwSL4+UlBSpu9QrKipgZqY/9a81EryNGjUKR48eRVhYGM6cOSPTNeHh4Th79ixGjBiBI0eOKPpowqOlBm8AcCe9CG/suoanRdKnFDu5WmHTtJ7wtNe9RLEiEYt/457iu5MP8CinTO7ru3rYYPHQjujfwVGmEcbzD3MxfdtV3koWAe42+HtOCMyMVb8DVRqWZbFo0SJMmzaNqjcQ0sywLIukpKRGwVxGRgbvNW3btpVaYrOmpgaOjo5o3769eM1caGgobGx0NweoRoI3Nzc3ZGdn459//sGLL74o0zV79+7FhAkT4OrqiidPnij6aMKjJQdvAJBTUoU3d1/HtZQCqW3tzI0QMSkQfdpzj0xpEsuyOBGfhe9OPpApIfHz/NyssXiIDwb5Oss9Lbz7cgo+2d90G39DI/xdETEpEAINp17ZtGkT3njjDRgaGmLZsmVYsmQJjIya3+5hQkjdz8FHjx41CuaePm28uWrWrFnYsmUL730uXbqEPn36NDomEAgQGBjYKJjTpY1SGgneTExMUFtbi2vXrqF79+4yXXPz5k306NEDxsbGqKyUf8E14RYREYGIiAgIhUI8ePCgxQZvQN3C/hUH7+KPq6lS2xoIGHwyyhcz+nhrbR0cy7KIup+Db08+wJ2MIrmv93GxxOIhPhjq56pUYLXy0F1su5DM22ZeeDu8P6yTws+Q1+PHj9GlSxeUlT0bgQwKCsKuXbvQsWNHjfWDEKIdLMviwYMH4kAuKioK33zzDaZMmcJ73RdffIGPP/6Yt42BgQF69OghThrcr18/WFpaqrL7ctFI8Obs7Iy8vDwcPHgQo0aNkumaI0eO4IUXXoCjoyOys7MVfTTh0dJH3hrafTkFnx68i1oZCtu/3MMDn4/3V0tiWi4sy+JCYh7WnbyPm6mFcl/f1tECCwd3wOgurVSSiFgoYvHajhhE3c/hbfftK13xYqD6S9wIhUKEhYVJrJnYtm1b3L9/H4aGur1ukRCiWizLQigUSv2/P3ToUJw8eVKuexsaGqJnz57iYK5v374y12tWBY0EbwMHDsTZs2cxfvx4/N///Z9M17z00kvYt2+fXOvkiHwoeGvsyuM8vPXbDeSVNU0M+7xunrb4dWoPuFirfwH9lcd5WHfyAa5KSdUhiZe9ORYM6oBx3VrB0EC1dUFLKmvw0s8X8SCrlLONsYEAv83uhaDndq+q2tatW/Haa69JPHfkyBGMHDlSrc8nhOin6upq2NnZobxc/o1eDRkaGiI4OFgczIWFhan1F0ZZP7+V+qk/YcIEsCyLffv24dNPP+XM8QbURcqffvop9u3bB4ZhFE7WR4i8erV1wMG3+6FzK+mB7K20Qrzw43ncTJW+Xk5RN1ILMGXzFfxv42W5A7dWNqb48sUAnH53ACb08FB54AYAVqZG2DI9CA4WxpxtqoUizNl1HakK7ICVx9SpU7F8+XIYGDQeDX3jjTcocCOEcKqtrcWaNWswfvx42Nsr/ktmbW0tLl68iNWrV2PcuHG8cY4mKTXyVlNTg65duyIhIQEMw8DPzw8zZsxAr1694OLiAoZhkJmZiStXrmDHjh24e/cuWJaFr68vYmNjabpDTWjkTbKKaiE+2HMbh2Klb5QxNhBg9Xh/vNzTU2XPv5NehG9P3keklClJSZytTDB/YHv8L8hTY9O611PyMXHjFVQLuVOvdHC2xJ63+sDaVL2bB2JiYjB16lTcv38fbdu2RWxsrFbXpRBC9IdIJMKdO3fEa+bOnj2LwsJCue8zfPhwHD16VPUdbEBjFRZSUlIwcOBAJCUlSV3szbIs2rZtizNnzkjNzUIUR8EbN5Zl8eu5x1h7LAGy/Muf2dcbH4/0VWqE697TYnx38gFOxGfJfa2jpTHmDmiHKb1bw9RI8yk69t/MwKK/bvG2GeDjhC3Te6plFLCh8vJyLFmyBC+//DL69eun1mcRQpovoVCI27dvi4O5c+fOoahI+kaxNWvW4MMPP1Rr3zRa27SsrAyffvoptmzZwhnN2tra4vXXX8fy5cvpN2Y1o+BNusj72Vjwx03exLT1+rRzQMSkQNjxTCNKkphdgu9OPZRaP1QSW3MjzOnfDtP7tIa5sXZHqNeduI8fzyTytpnRxxufjumsoR7J5vLly/Dz86P/A4QQXkKhELdu3WoUzJWUNE3VdOXKFQQHB6u1LxoN3upVV1fj+vXriIuLQ35+3Voee3t7+Pv7i9ODEPWj4E02j3NKMXvnNZmS4Hram2HTtJ7o5Cr97zM5twzrTz/EgVsZkGGTayNWpoaYHdoWM/t6w0rNU5GyEolYvP3HTRy5wx+Efja2M6aGeGumU1I8efIE/v7+sLa2xvbt2xEWFqbtLhFC9ERtbS1u3rwpDuaio6MB1NVyV/dyL60Eb0Q3UPAmu+LKGrzz5y2cTpCetsbc2ADrXu6KEQGSi7Sn5ZfjxzMPsedGBm+lAkksjA0ws28bzA5tCxtz3QjaGqqoFuJ/Gy/hdjr31IKBgMH2mUEI7eCkwZ41xbIsRo4ciWPHjomPvfPOO1i9erVelckhhOiG2tpaPHr0SCN5JSl4a8EoeJOPSMTi25MP8FMk/9RgvQUD22PRYB9xMtzMokr8FPkQf8WkoUYo338nUyMBpod4Y86AdrCXc1pW07KLKzE24gJv6TErU0Pse6sP2jtrL2P5r7/+irlz5zY57uvri4sXL8LW1lbznSKEEBlQ8NaCUfCmmCO3n+K9f2JRUSOU2nawrws+HuWLnZeS8duVVFTXcu/IlMTYUIDJvbzwZlg7OFvpT1H2uIwivPzLJd6/Iy97c+yf11crwWhiYiK6du0qMbfTq6++ij/++EPjfSKEEFmpNHjbuXOn+Otp06ZJPK6IhvciqkPBm+LinxRj9s5ryCisUMv9jQwY/C/IE/PC28PNRj+n8I7fzcTc3dd5d+sGe9tj1+vBGq1WAQCpqamYPn06oqKiGh13c3NDXFycUvmeCCFE3VQavAkEAjAMA4ZhUFtb2+S4Ip6/F1EdCt6Uk19WjXm/3cClx3kqu6eBgMGEQA/MH9genvbmKruvtvxy9hHWHE3gbTOhhwe+ntBF4/ViRSIRfvjhB3z00UeoqqoCABw9ehTDhw/XaD8IIUReKg/egLqASygUNjmuiOfvRVSHgjfl1QhFWH3kHrZfTFbqPgIGGNfNHQsGdYC3o+bq46kby7L44P9u45/r6bztPhrRCXMHtNNQrxqLj4/HtGnTEBQUhJ9//lkrfSCEEHmoNHhLSUkRf926dWuJxxXR8F5EdSh4U52/Y9Lwyf443ioDXEZ3ccOiwR20unhfnaprRZiy5QpviS+GAX6Z0gPDOrtqsGfP1NTUoLa2VqZdppWVlTAxMdH4SCEhhNSjDQstGAVvqnU9pQBzd19HTkmVTO2H+rngnSE+8HVr/n/3+WXVGL/hAlJ4apyaGRngn7kh8He30WDP5MOyLF555RUIhUL8+uuvcHLSXLqTwvJqXEjMQ35ZFYLbOKCja/MM9gkh0lHw1gJFREQgIiICQqEQDx48oOBNhTKLKjFn93XEphVytgnv6ITFQzoiwEN3gxR1SMwuxfgNF3irVbham+LA/L5wsdbNnbW///47Jk+eDABwdnbGpk2bMGbMGLU/90ZqAd7cfR1Zxc9+MXh7YHu8O1T9+aQIIbpHI8HbrFmzwDAMPv/8c7i5SU5c+rycnBx8+OGHYBgGW7ZsUfTRhAeNvKlHZY0Qy/bHNVnn1a+9I94Z4oMere201DPti36YgxnbYniTE3fxsMFfb4TAzFjzNVr5pKenIyAgoElpv5kzZ+L7779X2/+hRzmlGB9xAcUSgt4/3+iN3m0d1PJcQoju0kjwVr/b9M6dO/Dz85PpmkePHqFDhw60YUGNKHhTr2vJ+Yi6nwMAGOzngm6ettrtkI7YdTkFy/bH8bYZGeCKnyYGihMcaxvLshg2bBhOnjzZ5JyDgwPi4uLg6qr69XqF5dUYF3EByRzTzaO7uOGnSYEqf66uqhGKcPpeNgAWfm428HLQ/x3ZhChC1s9v7Va8JkQP9fS2R09vyhf2vKm9W+NRdinvDt1/72TiO6cHOjMtmJSUhFu3bkk898svv6glcKsRivDm7hucgRsAXE3KB8uyLWLzRFxGEebuvo70grrcisaGAnw4vBNe69dGyz0jRHcpnutDQZWVdaV1TExMNP1oQoiafTLKFwN8+Bf7/3gmEftu8qcY0ZS2bdsiLi4OY8eObXR88uTJmDBhgsqfx7Islh+Ik5pDMLukCqn53MFdc1EjFOG9f2LFgRtQt4v5s8PxuJ9ZosWeEaLbNB68XbhwAQDg4uKi6UcTQtTM0ECAHyd1h4+LJW+7D//vDq4lc6cY0SRnZ2fs27cP27Ztg5WVFdzd3fHjjz+q5VlbLyTjj6tpMrW9wpOCpbmIup+DBI4gbfvFJA33hhD9Ide06apVqyQe37BhA5ydnXmvraqqwqNHj3Dw4EEwDIO+ffvK82hCiJ6wNjXClulBGBtxAfll1RLbVAtFmLPrOvbP66sTFScYhsGMGTMQHh6Op0+fws5O9ZtPIhOysfpIvMztrybl45Wenirvhy45GveU89zN1ELNdYQQPSPXhoXny2HVXyrPugyWZWFqaopLly6ha9eucnSVyIo2LBBdcC05H5M2XeFNcOzjYok9b/aBlamRBnumGocPH8bjx48xf/58qdVm7meW4KWfL6K0SvaSgK0dzHH2/XBlu6mzqmtF6Pn5SYm7bYG66iR3Vw7Xud3JhKiTrJ/fck+bsiwr/lNf77ThMa4/JiYm8Pb2xuTJkylwI6QF6Oltj7UTAnjbPMgqxdt/3EStAhUstCknJwevvfYaFi5ciCFDhiA1NZWzbW5pFV7bESNX4AYAKXnlyCyqVLarOuvy4zzOwA0ARCwQ/7RIgz0iRH/IFbyJRKJGf+pH3uLi4pqce/5PeXk5Hj16hF27dlHgRkgLMb67B+aHt+dtE3U/B6v/vaehHimPZVnMmTMH2dnZAIAzZ84gICAAO3fuxPMTGVW1Qszddb3Rgnx5XNWRdYHqcDQuU2qb2+kUvBEiiVIbFry8vODl5QVjY2NV9YcQ0swsHuKDkQH8KTe2XUjG7svK1UrWlN27d2Pfvn2NjhUXF2P69OmN8sWxLIsle+/gWkoB7/28eNb8xTTTTQtCEYuT8RS8EaIopYK35ORkJCUloX17/t+sCSEtl0DAYN3L3RAgpbbpioN3cf5hroZ6pZji4mIsWLBA4rkXXngBQ4YMEX//y9nH2Hsjg/d+E4O9sHRkJ87zV5tp8BaTnI/cUsmbWRq6nV6o/s4Qooc0niqEENLymBkbYPP0nnDlqW0qFLF487frSMwu1WDP5GNtbY09e/bA07PxLlBHR0ds2rRJvHnr+N1MfHU8gfdeIW0dsGpsZwTxJHy+n1WCAo4du/rsmAxTpgDwOLcMJZU1au4NIfqHgjdCiEa4WJti8/SeMDPi3j1YUlmL13bE6HTAMnDgQNy5cwfTp08XH9u4caM4d2VcRhEW/XkLfPv4vR3M8fOUQBgZCOBgaYL2ztx58aRNu+obkYjF8buyBW8sC9x9UqzmHhGif1RWHisyMhL79+9HbGwscnNzUVFR0WTxbkMMw+DRo0eqejwhRA/4u9vgu/91w9zd1znbpOSVY87u69j9Wi8YG+rm75c2NjbYvn07xo4di3PnzmH8+PEAgOziSszeeQ0VNdx1m61NDbFlRhBszZ+tFQ7ytucccbyalIchfs0nqXlseiGeyrGL9nZ6IXq3dVBjj565k16E6yn5sDE3QpiPM+wsaD030U1KB2/Z2dl49dVXcfbsWQDgDNjqU4o0/J4Q0vIM93fFh8M7Ye0x7mnFq0n5+HjfHXw1oYtO/6wYP368OHCrrBFi9q7rnIGJqKoMosKn2LB0Mto5NR5p69XGHn9clZxupLmte5N1yrSeJjYtsCyLtcfu45ezzwYU7MyNsGNWMLp42Kr9+YTIS6lfa2tqajBixAicPXsWLMuiW7duGDVqFIC64Gzq1KkYPXo0WrVqJc4L16NHD0yfPh3Tpk1TyQsghOifuQPa4qVAD942/1xPx8ZzjzXUI+WwLIv3/olFbFohZ5v8U5vwdNe7OP17BGpqGq/jCm7Dve4t7kkxyuTMEaerWJbFMRmnTOvdyVB/8HYlKb9R4AYABeU1mP/7Td4ZJEK0Rangbfv27bh58yYAYNu2bbh+/TrWrFkjPr9jxw4cPHgQ6enp2LdvH9zc3BAfH4/Ro0dj27ZtyvWcEKK3GIbBFy/6I5hnsT4ArDmWgG0XklBdq9tJfNeffojDt7lLPZU/uISyuFMQCYX49NNP0bdvXyQkPBt5bGVrBndbM4nXCkUsbqQ2j3Vv956WICWvXK5rUvLKUVSu3k0Lv1+RPOqZml9O6UqITlIqeNuzZw8AYPjw4Y0W70oyduxYnD17FsbGxpgxYwYePnyozKMJIXrOxNAAv0ztwZvnjGWBlYfiMeDrSOy4mIxKnrVk2nIo9gm+P8X980xYVojiUxsaHYuJiUH37t0b5YvrxTP61lymTo/x1DLlczujULUdeQ5fcPwoR3d3P5OWS6ngLTY2FgzDYMqUKRLPPz/c3K5dOyxcuBBlZWVYv369Mo8mEkRERMDPzw9BQUHa7gohMrG3MMbWGT1hZcq//PZpUSVWHLyLfmsjsfHcI52ZRryVVoj3/onlPM+yLCojf0ZVSdPgwNjYGD169BB/zzd12myCN54pU2MD7o8jdY5+5ZRU8VbAyFCwOgYh6qRU8JafX/cDpU2bNuJjDastlJc3HR4fNGgQADTKRE5UY968eYiPj0dMTIy2u0KIzNo7WyFiUiAMBNI3JuSWVuGLfxPQd+0Z/Hj6IYoqtJcD7ElhBWbvvIYqnildOzMDTB7ZHwYGTdOj/PDDD/Dy8hJ/H8QTvN1MK0RVre6NOsrjUU4pHmRxj2JN7u3Fee6OGoO3WzzrFAHgSREFb0T3KBW81QdqDQM2a2tr8dcZGU2zi5uamnKeI4S0TP19nPDpC34yty8sr8G6kw/Qb80ZrDtxH/kazgtXXl2L13dcQ05JFWcbIwMGv04LxvdffYFLly6hY8eO4nPjxo1rsmmrraMFHC0lp6aorhXp/dorvl2mHV2sMLabO+d5dW5auJXGv55Q0bq0hKiT0rVNASArK0t8zMXFBVZWVgCAK1euNLnm7t27AChVCCGksakh3ngzrJ1c15RU1eLHM4not/YMvvj3HrJLZM8fpiiRiMWiP28h/il/8tjV4wLQ67/8ZEFBQbh58yYWLlwIFxcX/Prrr01+BjIM06ynTo/yrHcb7u+KTq5WMOQYfc0orEBuKXegrIybqYW8558UUvBGdI9SwVtgYCAAiHec1uvfvz9YlsX69etRVfXsP1xRURG++uorMAwDPz/Zf8smhLQMHw7vhIhJgWjtwL2JQZLyaiE2nnuMfmsjseJAnFo/cL85cR8n4rN427zRvy1eCWpcQsvMzAzff/897t27B2dnZ4nXPV8qixU+mxbW5+AtLb8ccRncwe5wf1eYGhmgk5sVZxt1TJ0KRazUEc2MQv6E84Rog1LB26BBg8CyLI4cOdLo+Ny5cwHUBXUBAQF4//33MW/ePAQEBIi3x1OeN0KIJKO6uOH04gH4/n/deMtGSVJdK8KOSykY8HUkPtpzGyl5ZSrt257r6dgQxV8ZZrCvCz4czl1s3s7OjvNcw5G3iuRbyNg0F5UptwEA11MKIBTpZxDBVw7L28EcnVzrgrYAd1vOduqYNk7MLkWplM0vlTUijU/LEyKNUsHbuHHj4OXlhfT09EalrkaNGoVZs2aBZVkkJibi22+/xS+//IL09HQAwNChQ/Hmm28q13NCSLNlaCDAuO7uOLGoP36eHAg/N2vpFzVQI2TxZ0waBq47i8V/3UJidonSfbqWnI8le+/wtunkaoXvX+0m0+YLyddbw8rUEKLKUuT9+z2ERVnI+nMp8k9vQnFpGe5JmarVVUd51rsN93cTTyF38bDhbHdHDelCbsqYP+9Jofqn4wmRh1LBm62tLZKTk5GSkoJ27RqvVdm8eTM2bdqEXr16wcLCAiYmJggICMDXX3+NQ4cOQSDQzZqFhBDdIRAwGBHghiML+mHrjJ7o5mkr1/VCEYu9NzMw5LtzmPfbDcQrWOQ8Lb8cc3ZdR7WQe2epo6UxNk/vCUsTxasOGggY9Gxth/zTGyEsyRUfL7l2AE+3L8RfR6MUvre2ZBVX4noKd5A0wt9V/HWAO3fwpo6RN2k7TetlFMqXWJgQdVNZYXpJXnvtNbz22mvqfAQhpAVgGAYDO7kgvKMzLj7Kww+nH+KKHGvAWBY4cucpjtx5isG+zpg/sIPMgWBJZQ1e33ENeTxTZ8aGAvw6tSc87ORbqyeJUdo1lMWdaXK8Nj8dZ85G47PXxyr9DE06wTNl2srGtNFoW0dXKxgbCiRW1MguqUJmUSVcbUxV1jdpmxXqZdDIG9ExNPxFCNEbDMOgb3tH/DUnBH/PCUF/Hye573HqXjbGRVzA1C1XcOVxHm9boYjFwj9v4X4W/7TrVy91QY/W3GvZ5JF09YTE46atu6K8/RC9WzzPN2U6zN+10a5bIwMB7xT57fRClfWrtKoWD2ScTqdEvUTXUPBGCNFLwW3ssXNWMA7M64shfi5yXx/9MBf/23gZr/xyCdEPcyQGRV/8ew9nErJ57/P2wPYY1507R5m8Du39PzgNeQOM4bOcb4yJBRxGLkJBRa1elWvKL6vmHSEd4e/W5Bj/ujfVTZ3eTiuErHEwTZsSXUPBGyFEr3X1tMWmaT1xdGEoRndxg7wpJK8m52PqlqsYt+EiTsVniYO4P66mYsv5JN5rRwa44p3BPop2XSJTY0MMmjATbtPXw9i1PQDAfvAcGFrXjTLKM12sbSfjMzl3yDpamkgcrdTUurebMq53A2jDAtE9Mq15GzhwIIC6KYvTp083Oa6I5+9FCCHK8HWzxk+TArEouxQbohJx4NYTuVJrxKYV4vWd1+DrZo3RXdzw3ckHvO0D3G2w7uVuECi4s5RPcBt7XHrsCdcp36D8/gWY+/YXn7ualI/JvVqr/JnqwFdVYWhnF4m7crvyrEW8k1EElmVVkuRd1vVuQF2uN0J0iUzBW1RUFICmVRGioqLAMIxcazDq21OFBUKIOrR3tsS3r3TDokE++PnsI/zf9TTUCGX/GXXvabHUlBwu1ibYNK0nzIyb1ixVhV7/5XtjDAxh4Teg0bmrSfkSf4aWlZXho48+wrJlyziTAGtScWUNzifmcp5vuMu0oXZOljAzMkBFTdNarvll1UgvqICnvXIbQ1iWlVoW6/nnllfXwtxYrXv8CJGZTP8S+/fvLzHY4jpOCCHa5uVgji9fDMCCQe3x69nH+ONqKm8ReVmZGgmweVqQSnc9Pq+7lx0MBQxqJYwcPi2qlBjAfPjhh4iIiMDff/+NjRs3YuxY7e5KPXMvmzNotjEzQu//Soc9z0DAwN/dGjHJkoOrOxlFSgdv6QUVyC2VL/Huk8JKuZNGE6Iuco28yXqcEEJ0hZuNGT4d0xnzwttj8/nH2HUpBeXVTUd1ZPXtK90QwLOoXhXMjA0Q4GHDObV3NSm/UQBz4sQJREREAACys7Mxbtw4zJgxA99//z1sbNTbVy58tUwH+7rAyIB7yXWAuy1n8HY7vQgjA5pudJCHPOvd6mUUVlDwRnQGbVgghLQITlYmWDLCFxc+HIgFA9vDylT+KbB3h/goHTjIiq9IfUzys00LBQUFmDVrVpM227dvx+zZs9XSN2nKq2tx9kEO53muKdN66q60cEuO9W71qEA90SVKBW+rVq3CqlWrcPz4cVX1hxBC1MrOwhiLh3bEhY8G4v1hHWFnbiTTdWO7tcL8ge3V3Ltngr25g7eGReo//vhjZGRkNGljbm6O1atXq6Vv0py9n4PKGslT1BbGBujXwZH3er7g7XZ6EURK1ni9Kcd6t3qU643oEqWCt08//RQrV65EVVWVqvpDCCEaYW1qhHnh7XHho4H4ZJQvnKxMONt297LF2pe6aHSNb8/W9pxpTx7nliG7pC59xaeffipxfdvXX3+NDh06qLOLnPgS8w70dYGpEf9GD28HC1hxlBkrqaxFSr7iedeqaoW4q0CZNNpxSnSJUsGbg0PdglMvLy+VdIYQQjTN3NgQr4e2RfQH4fhsbGe0em4jQnAbe2ya1lNqwKFqNuZG6OTKXW3g2n9rwpydnbFv3z5s374dVlZWAIAhQ4bgzTff1Eg/n1dVK+RNbDy8M/+UKVBX09afN99boSJdAwDce1oisfyWNBS8EV2iVPDWvn3dFEJmJvdvWYQQog9MjQwwNcQbUe+H47fXe2HZaD/snBWM31/vBUdL7lE5dQr25i651XDqlGEYTJ8+HXfu3MG4ceOwdetWrWUCuJCYi9KqWonnTAwFCOsoW0kz3nVvSiTrvZXKPWXq784dLNO0KdElSgVv//vf/8CyLP7++29V9YcQQrTK2FCAvu0d8Vq/Nujv4wRDnl2R6hbcRnI6DUBypYXWrVtj37598PDwkOn+Fy9ehEikfPqUho7e4f5lfoCPEyw4pkOf18XDlvPcbSXKZPHtNB3e2VVi4mAAyCyulCvpMyHqpNRPpbfeegtdu3bFzp07sWPHDlX1iRBCCICgNtwjbwmZxSiqqFH43hcvXkRoaCgGDx6M1NRUhe/TUI1QhJP3sjjPjwiQPmVaj2/kLS6jSOFAiq+yQk9ve7haS87fJxSxyCqmMllENygVvGVmZmLz5s3w9/fHrFmzMHToUGzfvh03btxAUlISUlNTef8QQgjh5mxlijaOFhLPsSxwPUWxOqelpaWYNm0aRCIRIiMjERAQgB07dshVLUeSK4/zUVguOaA0MmAwsJOLzPfysDODLcdO4PJqIR7nlMrdv7zSKqRybHYQMHUlz1rZcidfpnVvRFcoVevD29tbvK6CZVmcPn1a5nqlDMOgtlbyughCCCF1gr3tkZRbJvHclaR8uQKieh988AEePXok/r64uBgzZszA0aNH8ccffyi8Xu7YXe7EvH3aOcLGTLa0LEDdZ0SAuw2iH0ousXU7vQgdXKzk6t8tninTjq7WsDAxhLutGWIgeV0c5XojukLpxRwsy4p/W6v/WtY/hBBC+PEm65Ww7k2a48eP4+eff5Z4rlu3bgoHbiIRi+N3eaZMpSTmlYQ/Wa/86974grdunrYAAHc7M8426bRpgegIpUbetm3bpqp+EBWIiIhAREQEhELFS/8QQnQLX/B2O70IFdVCmBnLnsbE3NwcXl5eTZau9OnTB++//77C/byeWoCcEsk5PwUMMMRP/hFCvk0LsQqkC+Fb79bdq+5ZrWy5gzcaeSO6Qqngbfr06arqB1GBefPmYd68eSguLtZaPUNCiGp52JnBzcYUT4uaLpavFbG4mVqAPu35KxY0FBoaitu3b2PRokXYvn07gLqAbseOHTAwUDyXHd8u0+A29nBQIN0K38hb/JNi1AhFvDVSGxKJWMTyjLwF/he8ufMEb7TmjegKqm1KCCE6jGEY3tG3q8nyT53a2Nhg27Zt2LdvH5ycnPDNN9+I83YqgmVZHL/LHbyN8FesHqyrtSlnjr2qWhEeZsm+aeFRTilKOPLPWZkaoq1jXdF53uCNpk2JjqDgjRBCdFyQjHVO5TVu3DgkJCRg7ty5MrWvqZG8k/RORhHvqNQwGaoqSMIwjMqK1PPld+vmaQvBf/nd+Na8PSmsoPXaRCdQ8EYIITquF8/I243UAoXKPdWzt7eXaZNCVVUVevfujZUrVzYJ4vhqmXb3soWrDXf6DWmkFamXFd96t/rNCkBduTQ7jhQlZdVCpXLrEaIqSq15a6ikpASnTp1CbGwscnNzUVHB/xsKwzDYsmWLqh5PCCHNVntnS9hbGCO/rLrJucoaEeKeFCHQizuhryqsWLECN27cwI0bN3D48GHs2rULnTp1AsuyOMYTvCmyy7Qh1QVv3GWx6jcr1Gtla4YCjnx16QUVsDU3lvm5hKiD0sGbSCTCZ599hnXr1qGsTHIuouexLEvBGyGEyIhhGAR523Gm4rialK/W4O38+fP46quvxN9fu3YN3bt3x5o1azD05RmceegAxde71eMrUJ+QWYyqWiFMDPk3WpRV1eJBVgnn+a7P7Wp1tzXD3SfFEts+Kazg7RMhmqD0tOmMGTOwatUqlJaWQiAQwMnJSTzi5uHhAQsLi0Z53RwdHdG6dWt4eXkp+2hCCGkx1LXuTZrS0lJMnz69yUxKZWUltm7diiOxaZzXdm5lDU97c6We72xlCjeOadcaIYv7mdxBWb3b6UXgqqbV2sG8yU5YvnQhtOOU6AKlgrfjx49j9+7dAOqCuOzsbJw6dUp8PiUlBcXFxUhISMDChQshEAhgZ2eHo0ePIikpSbmeE0JIC9KLp0h9THK+2oqmR0VFISUlpclxY2Nj7Nq1C6fucweOwxXcqPA8ZadO+ZLzdm+w3q2eh5RNC4Rom1LBW32S3s6dO2Pr1q2ws7OTuPDVx8cH3333Hfbt24dHjx5h5MiRKCqSPzs2IYS0VL5uVrA0kbzSpaSyVqYRKEWMHj0aly5dQqdOnRod/+yzz2DVqh0SeJ4rTyF6PnzJem/LkKyXb71bNwnBG+V6I7pOqeDt8uXLYBgG8+bNk6n96NGjMX36dKSkpOCHH35Q5tGEENKiGBoIENiae13b1aQ8tT07KCgIN27cwKJFiwAA/fr1w7vvvsu7UaG9syXaO8tXe5RLAM8aM2kjbyzL8qYJ6S5hrSDvtCnleiM6QKngLTs7G0DdyFq9hhm6q6qalkqZMGECWJbFvn37lHk0IYS0OHwpQ2KSuUeXVMHMzAzfffcdTp8+La7GcCyOuxC9qqZMAf7g7WF2KSqquUsCPimq5CzbZWwogK+bdZPjfLneMgqbVrogRNNUkufN3v7ZDxQrq2e/adUHdw05OzsDAJKTk1XxaEIIaTH4Ki1cScrXSALZgQMHom3btsgorEAsz6jXcH9X/PLLL4iMjFT6mXYWxvC0lxxQCUUs4p9K3hkKALd48rv5t7KGsWHTj0EHC2OYSDgOALmlVaisofrRRLuUCt5cXOoKDefn5zc6ZmxclwPn9u3bTa6pL4ZcWUm/vRBCiDy6eNhIDDaAuqCCL2WHqh3nmTL1tDdDddYjvP322xg4cCDeeecdVFQoN92o6Lo3/vxukqehGYbhXfcmqc4sIZqkVPAWEBAAAIiPjxcfMzQ0RPfu3QE829DQ0K+//goAaN26tTKPJoSQFsfE0EDiAvt6MQrUOVUU33q3wT72mD59Ompr62qJfv/99wgMDMS1a9cUfl4XnqnTOzwjgNLKYnHhnTqldW9Ey5QK3sLCwsCybKP0IAAwZcoU8bq2adOm4ciRI/jnn3/wwgsv4Pjx42AYBmPHjlWq44QQ0hLxrXu7osZ8bw1ll1QiJoX7WY+ObUVcXFyjYwkJCejXrx8yM7mDPj4BfOlCMiQHb9W1IsRxnAOaVlZoqJUN37q3cs5zhGiCUsHb+PHjAQDHjh1DVtazzN9z5sxBYGAgWJbFb7/9hjFjxuDVV1/Fv//+CwDw8vLChx9+qMyjCSGkRdJWst6GTtzNAtfyOsuSVOze+JPEc++//z5cXRXbyMBX1eBRTilKq2qbHK+rwCC57qujpQnv1ChtWiC6TKngrU2bNnj8+DHi4uJgbf1sx46hoSFOnjyJyZMnw9DQsFGFhVGjRiE6Ohp2duqtw0cIIc1RYGs7GAgkF5JPL6jQSBLZ43e5R89eHNwX3377LUxNG1dF6N69O5YtW6bwM61NjdDWyULiOZYF7koYYeNNzutlKzEvaT3eXG80bUq0TOndpt7e3mjXrh3MzBr/Q7ezs8OuXbuQm5uLa9eu4fLly8jJycGhQ4fg4eGh7GMJIaRFsjQxhH+rpukt6ql73VtheTUuPeLOKTeiizsWLVqEGzduoGfPngAAExMT7Nq1S7yZTVF8694k5Xu7ybPTlG+9G8Cf601VAXJReQ3+uZaGtccScPj2E9QIJY8SEvI8pQvTS2NlZYXAwEB1P4YQQlqMIG97zjQdV5LyMbabu9qefeDWE9RylOKytzBGkHfdrIqvry8uXryIL774AjY2NujcubPSzw7wsMX+W08knpO07k3ayBsfvhJZqqiykJJXhulbryI579n6ueA29tg0rSdszIyUvr8iKqqF2H05BXFPiuBlb47x3d3R1slSK30h/JQK3goKCmj6kxBCNCy4jT02n5dcH1qd696KKmqw/vRDzvND/VxgaPBsQsfIyAgrVqyQ+f5JSUmwsLAQ5wN9Hl+N0zvPpQspKKvmTJ0iYPhTjwCAi7UpGAYS1/Y9LaqASMRCwDF9LYs1RxMaBW5A3Xu37UISFg324bhKfQrLq/HqxsuNyp1tv5CMbTOD0JNnnSXRDqWmTd3c3DB+/Hj8888/EqspEEIIUT2+TQuJ2aXIK1XPz+OfzjxEflk15/nh/opXVaitrcWrr74Kf39/HDhwQGKbzq2swRUvJeeVo6i8Rvz9LZ7cbz4u3HVi6xkbCuBiZSrxXI2QRY4Sf8cllTWc6wb/uZau8H2Vse1CcpM6tSVVtVj97z2t9IfwUyp4q66uxsGDB/Hqq6/C2dkZs2bNwqlTpzSS5ZsQQloqOwtj+LhwT2epo1RWUm4Ztl9M5jzv7WCO0A5OCt9/zZo1uHr1KnJycjBu3DjMnDkTRUWNp0LNjQ3Rgade6p0GU6d8692kTZnW49txmq7EpoWEzBJwzDwjo7ACuWoKvvkcuJUh8fjN1EJkFdPuWl2jVPA2d+5cODg4gGVZlJSUYMeOHRg2bBjc3d3x7rvv4vr166rqJyGEkAb4SmWpY+p09ZF7qBFy/2K+dKQv5y5YaW7cuIGVK1c2OrZ9+3Z06dKlURJ4QFq+t0Lx13yVFaRtVqjHW6BeiXVv93jKeQH8SYfVoaCsuskUbkMpPOeIdigVvG3YsAFPnz7F4cOHMWnSJJibm4NlWWRmZuL7779HcHAwOnXqhM8//xyPHj1SVZ8JIaTFC27jwHlO1TtOzz/Mxal7WZzn+7RzwBA/F4XuXVlZialTp4qrMTRkZmYGb2/vRsf4173VBT0iEYtY3s0Ksq3V5ksXosyOU2nBWyzPlK868E0xA0B6AQVvukbpVCEGBgYYOXIkdu/ejaysLPz2228YOXIkDAwMwLIsHjx4gBUrVsDHxwchISGIiIhATk6OKvpOCCEtVjDPure7T4pQUlnDeV4etUIRPjscz3lewADLRvvx5kzjU1FRgQ4dOjQ5bmBggJ07d8Lc3LzR8QAZ0oU8zi1DcWXTYBAArEwM0V7GHZTutpLXvAHK5XqLf1rCe15S2hN1usUzxQwAafmU107XKB28NWRubo6JEyfi8OHDePr0KSIiItCnTx8AAMuyuHLlChYsWAB3d3eMHDlSlY8mhJAWxdXGFF725hLPiVjgeopq1r39GZOG+1ncwcarwV7wdePOOyeNnZ0d9u3bh+3btzdK9v7xxx8jODi4SXtfN2sYckzPZhRWIK+0ijdFSBdPG5l3ifKteVN05E0oYnE/k3/k7XZ6oUbXjvP9fQE08qaLVBq8NeTg4IA333wT58+fx+PHj7F69Wp07twZLMuitrYWx48fV9ejCSGkReBb96aKqdOiihp8e/IB53krE0MsHqJ8WguGYTB9+nTcuXMHAwcORGBgID755BOJbU2NDNDRlXvTwu2MIt71bt09ZU9v5W4rOTgGFF/zlpxXhsoa/mS8uaXVeFKkmU0CLMtKnaZNo+BN56gteGuodevWePHFFzFu3DjY2tpq4pGEENLs8U2dqmLTgrTUIG8Pag9HSxOln1PPy8sLJ0+exLFjx2BkxJ2otn7dG8uKwLKNA6E76UW8I0myblYAgFZqmDaVtt6t3m0po2GqkpxXjsJy/il2ZXbWEvVQa/D29OlTfPvtt+jZsyf8/PzwxRdfiLd+P1/3jhBCiHz4Rt5i04pQWSNU+N7SUoO0djDH9D7eCt+fi0AggJMTf8qRAHdbAEDJ9UPI+vMT1BZni89dScprkq+soW4ypgkBACtTI1ibSs4HV1JVi2IF1hXKGrxxVdBQtVtp0qfXnxZVopZKd+kUlQdvxcXF2Lp1KwYNGgQvLy+8//77uHHjBliWBcMwGDhwILZu3YrMTO7CxoQQQqRr7WAOZyvJI1/VQhHvjktpvviXPzXIxyN9YWJooPD9ldHFwwY1uWkoPLsDVam38WTLfJTeOQ2WZXEhMQ9CjiRqnvZmco8UutvxTJ0qMCJ1T8pmhXp3GqQ9USdpmxWAunV6TzU0jUtko5LaptXV1Th8+DB+++03/Pvvv6iurhtmr19w2b17d0yePBkTJ06Em5ubKh5JCCEtHsMwCG5jj8O3n0o8fzUpH73acqcU4XIhMRcn49WTGkQV2tibIu/fb8HW/vdZU12OvH+/Q/nDS3AYNh8GFrYSr5NnvVs9d1tTztGyJ4UVcm/WkHnaNL1I6RJcspC2WaFeekEFPDk2yBDNUyp4O3PmDH777Tfs3bsXxcV1/yDrAzZvb29MmjQJU6ZMQadOnZTvKSGEkCZ4gzcFNi3UCkVYdUh9qUFU4Zuv1qDqadMaqxUPL6Omxxju4E2OKdN6fLne5N20UFheLfMIVkllLZLzytRaGL6yRoh4GYPJtIJyhED+XwSIeigVvA0ePBgMw4gDNgcHB7zyyiuYPHmyOEUIIYQQ9eFb93Y9pQC1QlGjYvHS/HWNPzXI/4KUSw2irMrKSuzcuVPiOaueY2HaugvntfJsVqjHW2VBzmlTWadM691OL1Jr8Bb/tJh3arwh2rSgW5Re82Zqaor//e9/OHjwYJPcboQQQtTLx9kKNmaSd2aWVwtx94lsIysAUFxZg3Un+FODvDtU+dQgyjA1NcWNGzcwYPTLjY4b2nvAtv80zuuMDQTwayV/0MmX603ekTdZp0zrqbvSgizr3eql51O6EF2iVPC2fft2ZGVl4Y8//sDo0aNhaKiSJXREQREREfDz80NQUJC2u0II0RCBgEGQN/daLnlShvx4WrOpQRRlY2ODXzdtgdP4jyEwtwEYARxHvwuBEXff/FpZK7TBQpXTpvIGb+qutCDrejeARt50jVLB27Rp02Bpqb4hXSKfefPmIT4+HjExMdruCiFEg3iL1Mu47k1bqUEU1c7JAg6d+6HVrAg4vvA+TNyalthqSJH1boBq65vek1JZ4Xl3nxSpNUWHfMEbjbzpEo0k6SWEEKI+0orUizhSZzQkLTXIUi2mBpHE0ECAzq2sYWBhCwvfUKntu3naIjk5GZ9//jlqamTPz+ZoaQJjjjWD2SVVqK6VLbiqFYrwIKtU5ucCQGWNCA+z5btGVnmlVUiVYyr0aXGlzK+VqJ9MwVt6erq6+4EnT56o/RmEENIcdW5lDTMjyYFVYXmN1ABAWmqQkLYOGKrF1CBcAjy4i9Q/r5uHDWbMmIFly5ahT58+SEhIkOk6gYCBG0elBZYFnhbJNvr2OLdMoeDntprWvcm7nk6e10rUT6bgrUOHDpg3b55agri//voL/v7+2Lx5s8rvTQghLYGRgQA9WvOse+OZOhWKWHx2mD81yPIXtJsahEtXD1uZ2jlaGmPPzo04e/YsAODatWvo3r071q9fD5FIekClinVv8q53q6euSgvybFaol5ZPwZuukCl4E4lE+OWXX9C+fXtMmzYNx48fl+kfPJfU1FSsXbsWnTp1wqRJkxAfHw8TE+0vgiWEEH3Fu+6NZ9PCnzGpvOWktJ0ahI+sI2+tBflYunRpo2OVlZVYtGgR1q1bJ/V63uBNxoX8fPnUOrlacZ5T18jbTQWqb9C6N90h0/bQu3fvYvHixeIqCr/99hucnZ0xduxY9O7dG0FBQfDz4/7NLDc3FzExMbh69SpOnz6NixcvgmVZsCwLJycnrFy5Em+88YZKXxghhLQkQbxF6vPEJQobkpYaxFIHUoPwaeNgAUsTQ5RW1XK2YVkWN3Z/iaqqqibnWrdujTlz5kh9Dm+uN5lH3vgCZE+s5EiMnPC0BJU1QphyTIsrQiRi5dqsUC+NgjedIVPw1r59exw8eBCXL1/GZ599hmPHjiErKwubNm3Cpk2bAADGxsZwcHCAnZ0d7OzsUFFRgfz8fBQUFIiL0QPPKjC4urpi4cKFmD9/PiwsLNTw0gghpOXo7mULIwNG4qaDrOIqpOVXwMuhcXmjn84k8qcGGagbqUG4CAQM/N2tcfkx98giwzD4ZPU3+Gbp243WuTEMgx07dsDaWvqoIl+uN1l3nPJNm4Z2cEQrG1M8kVB9oVbE4t7TYnT3kr+0F5fHuWUoqeQOeLlQuhDdIddu0969e+PIkSNISEjABx98AG9vb/EIWlVVFZ48eYK7d+/iwoULuH79OpKSklBYWChuY2xsjJEjR+KPP/5ASkoKPvzwQwrcCCFEBUyNDHjXgF1Jymv0fXJuGbZdSOJs39rBHDP6equod+rTRcq6N4YBXhkxADdu3MCiRYvEx9955x0MGDBApmcou+Ytt7QKOSVNR/4AwMRQAG8HC97Xoep8b3yjblYm3GM6FLzpDoWy6nbo0AFr1qzBmjVrkJqaiujoaFy8eBHp6enIyclBfn4+TE1N4eTkBCcnJwQEBCA0NBTBwcEwNjZW9WsghBACIKiNPa6lFEg8dzUpHy/39BR/v1rPUoNw6SJl3ZuPsxWsTI0AGOG7777DmDFj8NVXX2H16tUyP4M/15v0WqV8o24dXa1gaCBAF08bHLubKbGNqist3EqT/G8EAIZ2dsWeG5I3J6ZRlQWdoXRJBC8vL0yePBmTJ09WRX8IIYQoKLiNPX6OeiTxXEyDHaf6mhpEki7utrznn69nGh4ejrCwMJnuzbIsbt26Bb8A7nqpGYUVEIlYCATcu3H5grf6zQp8o6Z3NDjyNrSzCw7FPkG1hOTA2SVVKl9/RxRDSXoJIaSZ6NHaDlwxRHJeObKKK6WmBmEYYNlo3UwNIomnvRlnbVdAcmUFhmFken1btmxBYGAgPnzvXThITvWG6loRcsskT4nW49usUL+T19+dewQxMaeUd1OGPCprhEjg6U+gl51K67kS9aDgjRBCmglrUyPetB5Xk/LxV0wab2qQV4M8FSrgri0Mw6Drc6NrDQXy5L/j8/jxY7zzzjsAgPXr1+PBL/NQ9fShxLbSpk75Rt7q3y8bMyO0dZS8BpxlgbgM1Yy+xWUUoZaj4oa7rRmcrEzgwRO80bo33UDBGyGENCN8+d7OJGRj3Yn7nOfrUoN0VEe31OrVIE+JxwO9bNHBWf7620KhEDNmzEBp6bPKFKVZKcjc9S6Krx1o0p4v11tVrRCJPBUufF2fBcp86/dUle+Nb8q023+jlB525pxtaN2bbqDgjRBCmpFePMHbvpsZyNPj1CBchnV2xZTeXo2Oudua4YsXA+Sa/q3PjPDdd98hOjpaQgMRDO1aNTnMly4kMbuUd6TLxvzZlC/fjlNVVVrgS87b/b8RTBp5031Kb1gghBCiO3ryJOvl42WvH6lBJDEQMPh8XAAm9PDE9ZQCuNmYom97R961cJIwDAOWZWFmZgZTU1NUVjaeDrXsOgzm7YKaXMe3DoxvfZmvW+PKCl09NTDyxlMWq35zh6c998gbVVnQDTTyRgghzYijpQnaOcmfP1NfUoPw6eZpi9f6tcHIADe5A7d6DMNg3rx5uHHjBnr27Ck+7urhBbvw1yRewzcaJct6t3p+bjYw4NhxkpZfwZtQWRY5JVWcgaahgBFvmuAbeUujkTedQMEbIYQ0M8FtHORq37utPYZ11o/UIJri6+uLixcvYsWKFTA2NsZXP/wKgYnkESm+adN7mbIHb2bGBrxr9O4ouWmBb71bJzcrcQoQvuAtg0bedAIFb4QQ0szwrXt7HsMAy0d31pvUIJpkZGSETz/9FI8fP8YLwwZytms4miUSPcuPxrKsTGlCGuLL93ZbgXqkDfEl522YD8/J0gQmhpLDg9zSapRXqyZtCVEcBW+EENLMBMkRvOlbahBtcHd3h7WpISyMDcT1uRsqqqhBaVUtnjx5gs6dO2P//v0A6pLack11mhsboLWEtWVdeNa9KbtpgXenqeezlCoMw0gZfaOpU22j4I0QQpoZd1sz3pJO9SxNDLF4iP6lBtEGgUDAm0Ijo6Acr7/+OhISEjB+/HjMmDEDVx9ILjMF1JXFklSVgXfkTYlNCyIRi9tp3MHf85Uo+DYtpNHUqdYpFbwNHDgQAwcOxLZt21TVH0IIISogy9Tp/IHt4WSlf6lBtKWVrSnn9PLGjRtx9OhR8fc7duzA9FH9UZESK7E9VzLljq5WMOaYsswuqUJmkfRaqpI8yilFCUeVBitTwyYJgildiG5TKniLjo7G2bNn4e3traLuEEIIUQVpU6de9uaYqaepQbTl+bJR9XnhagqeImLNiibtC7KfovTWMXHbhriCNyMDAfx4qmQoWqSeL79bN0/bJqOAnjyjjBS8aZ9SwZuzszMAwNbWVhV9IYQQoiJ8lRaA5pEaRNPcbRsHNPWjcGVxZ1BV0XQq0djaAfZD35R4L19XK4nHAaCrGiot8K93s21yjKos6DalgreuXbsCAB48eKCSzhBCCFGNto4W6MQRIFBqEMW0sm1anZ5hGNj0m4TBc1fC2rrxiJndsLdhYGYtblePZVl04hld46u0cFvBTQuyJOdtiKZNdZtSwdvrr78OlmXxyy+/qKo/hBBCVIBhGHw0ohOMDBpPh7lYm+DrCV0pNYgCuAIahmFg02Uwbt++jfDwcADAhCkzYNq2p8T2XvbmsDThLnDEV+P0TkaRxB2vfCqqhbifxZ2yRFLwRhsWdJtSwduLL76IKVOm4OzZs5g1axbKyspU1S9CCCFKCuvojL1v9sXoLm7o5mmLWX3b4J85fXg/mAm3Vjw7eDMKK9C6dWucOnUKGzduxOjXP+BsKy01S1snS1gYS57SLiyvQaqc05Z3Moog5Kiv6mlvBgcJ9WztzI1gztOHksoaufpAVEup2qY7d+7EoEGDcPv2bezYsQMHDhzACy+8gC5dusDOzg4GBvzrKaZNm6bM4wkhhEgR4GGDnyYFarsbzYKzlSkMBYzEQvNZxZWoEYpgZCDA7Nmz8enBu5z3abhZ4csvv8SkSZPQunVr8TGD/0pVXUnKl3h9bHoRWjvIXgKNPzmvncTjDMPA086cc8QuvaACvm6KlSAjylMqeJsxY0ajofeCggLs2rVLpmsZhqHgjRBCiN4wEDBwszVFWn7TNV8iFsgsqhSPasbLUNP0zz//xNKlS/Hll1/ihx9+wPTp08WfqV09bTmDt9tphRjTtZXM/ZZ3s0I9DzszKcEbJXfWFqWT9NZvla6fg2/4vbQ/hBBCiD5pZcM/dQrUl8XiDt783KyRkZGBt956CwBQUlKCmTNnYvz48cjOzgbAv+5N3k0L8m5WqMc3vZ5O6960SqmRt6SkJFX1gxBCCNF57nZmAMdHX32B+ozCCpRUciTENTGEu60pRk56DQUFjaczDxw4gPv37yMuLo630kLck7o1bAYSKjQ8L6u4Ek84EvsaGTDozLP+jm/HqaTRR6I5SgVvDefoCSGEkObOg2/Twn8pNPiK0Xdys8KOHTtw/PhxiedXrlwJAwMDeNiZwc7cCAXlTTcGlFcLkZhdio48ueLq3eQZdfN1s4apEffadP50ITTypk1U25QQQgiREd+O0ydFdcFbgpT1bi+99BJmzpzZ5NzEiRPxyiuvAKhbF86f761Qpv4qut4NkJKol3K9aRUFb4QQQoiMni+R1VB98tp7mfzBm7W1NbZu3Yr9+/fDyckJANCqVSv89NNPjdryV1qQbd0b/05TW95r+Utk0cibNik1bdqQUCjE/v37cerUKcTFxSE/v26XjL29Pfz9/TF48GCMGzdOavoQQgghRFdJy/UG8E+bNtyhOXbsWISEhGDu3LmYM2cO7O0blzQLUHLkTShicYcnyJMWvNmYG8HK1FDi+r2SyloUldfAxpzShWiDSoK3Y8eO4Y033kBGRob4WP1uUoZhcPHiRWzcuBEeHh7YuHEjhg0bporHEkIIIRrlzjdtWliBsqpaJOdJTljPMEBHl8br1JydnbF3716J7SWNvNXkpcHQ1g33npagulYEY0PuCbSH2SUoqxZKPGdjZoQ2jtJzxXnYmXPunE0rKIeNOffoIFEfpadNd+3ahdGjRyMjI0OcAqR169bo3bs3evXqJd7UwLIs0tLSMGrUKPz2229Kd5wQQgjRNFMjAzhaGks8V1kjwuXHeeDKhNXGwQJmHFULJHG2NoWr9bN6qsLyImT+sQSZu99DaVYyEnimZwH+FCFdPW1lKpHmSTVOdZJSwVtKSgreeOMNiEQimJub4/PPP0dmZiYeP36Mixcv4tKlS3j8+DEyMzOxevVqWFpaQiQSYfbs2UhNTVXVayCEEEI0hm/q9NS9LM5ziiS1rc/3xrIs8o9HQFRWiOrMRDzdvhBrv14HkUjEea0ymxXq8W1aoHVv2qNU8LZ+/XpUVVXB0tIS0dHRWLp0KZydnZu0c3JywpIlSxAdHQ1LS0tUVVVh/fr1yjyaEEII0Qq+qdNT97I5z/m6SU/t8byu/wVZZfFRKH9w8dkJYQ1+/+FzDBo0CCUlTdfYpReUI/phLud9u8scvNHImy5SKng7ceIEGIbB+++/j27duklt37VrV7z33ntgWZYzxw0hhBCiy/hG3nJKqjjPKTryVluci/yTv0g8b2lpCUtLy0bHDsY+wYj10eINFJJ0lTF446uykJZPI2/aolTwVj/1OXjwYJmvGTJkSKNrCSGEEH3CN/LGR6Hgzd0WoqoyGFraNzknMLPG+oifxWvXSqtqsfjvW1jwx03OCg8A0NrBHPYWktftPU/ZkbcbqQWYue0qhn9/DjO3XeUtG0Zkp9RuU6GwbheLPOk/6tvyzdMTQgghuoov1xsXGzMjuNmYSm/4/HXmRvDp5AtD2+9ReG4nSq4dBFC3I8J+2DzkiczRFsDN1AIs+usWUvKkj4b17+Ak8/N5S2QVlINlWc6ND7fTC/HyL5cgFNX1NyGzBNEPc/F/b/aRec0dkUypkTd3d3cAwMWLF6W0fKa+batWrZR5NCGEEKIVioy8+bpZybS7U5IuHrYQGJnAftBsuExcDQNrJ1j4hcGiY1/cTC3AT2ceYsIvl2QK3GzMjPBavzYyP9vK1Ai2HLncyquFEst3AXUbLJbsvSMO3OrVilisO3Ff5ucTyZQK3sLDw8GyLNasWYMnT55IbZ+eno41a9aAYRgMHDhQmUcTQgghWqFY8Cb/lGm9Lg3yvZl6dUGrWRGwH/omAODr4/fxzYkHTYKkeiz7bJartYM5fnu9F7xlyO/WkCI1ThMyS3D3ieQp0guJuSit4p7WJdIpFby9/fbbEAgEyMnJQa9evfDPP/+Ip1IbEgqF+PvvvxESEoLs7GwIBALMnz9fmUcTQgghWmFrbgQznoLukigXvNk2+l5gYg6BSV0AViPkSCr3n7yjPyL/1EaMDXDEkQWh8HeXP6kuX5mstHzJ697238qQeBwARGzdNC9RnFJr3vz9/fHZZ5/h448/xpMnT/Dqq6/C1tYW3bt3h4uLCxiGQWZmJm7evInCwkJx1YXPPvsM/v7+KnkBhBBCiCYxDAN3OzMkZpfKfI2fEsGbv7s1BExd0COP8oeXUXbnJADgxOcPMMVnJ4KCguR+vrwjbyIRi0O3+GfjriUXIFSOtXekMaXLYy1ZsgQ2Njb44IMPUF5ejoKCAkRGRjZqUx+0mZub4+uvv8abb76p7GMJIYQQrXG3lT14MxAwaO9sKb0hB3NjQ3RwtsL9LO6aqc8TlhWi+MSzQvcJCQkICQnBxx9/jE8++QRGRrLXJOVNFyIheLuWUoAnRZW897yeQiNvylC6PBYAvPXWW0hOTsbatWsxZMgQuLq6wtjYGMbGxnB1dcWQIUOwdu1aJCcnU+BGCCFE7/HlenteOycLmMo5zfq8LhLqnHIRMID9rR2oKi1sdFwoFOK7777D06dP5Xq2vOlC+KZM691MLUCtkLJOKEolhekBwNHREe+//z7ef/99Vd2yRdm9ezeio6Nx/fp13LlzB9XV1di2bRtmzJih7a4RQgh5Dl9A8zxl1rvV6+Jpi3+up0tt52VvjneCLDDp+2iJ59evXw8vLy+5ns2/5q3xyFt1rQj/3pEeHJZVC5GQWaLQGjyi5MibQCCAoaEhvvrqK1X1p8X65JNPsHHjRqSkpMDNzU3b3SGEEMJDnh2nnVyVD966PbdpQZKXAj3w78JQjA8Pxs2bN9GzZ89G58eMGaPQgABfXrv0ggrx0igAiH6Yg0KO9CHPu0GbFhSmVPBmbGwMlmURGhqqqv60WJs3b0ZycjJycnIwd+5cbXeHEEIID3mmTRWpafo8f3dr+LtLDgKtTA3x48TuWPdKV1ia1E2oderUCRcvXsTKlSthaGgIJycnbNq0SaFcc+bGhnDgqMhQVStCbmm1+Pv9UjYqNHQtmYI3RSkVvNUn2pWnwgKRbPDgwWjdurW2u0EIIUQG8lRZUGanaT2GYbBqrD8cLU0aHQ/2tsfRhaF4oWvTxPdGRkZYvnw5Ll26hN27d8PZ2Vnh53vIsGmhrKoWJ+MzZb4nbVpQnFLBW//+/QEAN27cUElnFJGdnY3Dhw9j+fLlGDFiBBwdHcEwDBiGkXt4ODU1Fe+99x58fX1hYWEBe3t7BAcH45tvvkF5ORXgJYQQUsfFygQGAumjWA4WxnCyMpHaThaBXnY4tigUq8f7Y8GgDvhnbgj+mtMbHjxr0gCgZ8+eGDp0qEzPOHv2LPbt29fkuCybFk7EZ6KyRvZNCBmFFXhaJL0+qiZV1ghxMPYJ1p96iMiEbFTVNs1dqwuU2rDw9ttv4/fff8c333yDSZMmwdpa+d8u5OXi4qKS+xw5cgSTJ09GUVGR+Fh5eTliYmIQExODzZs3499//0Xbtm1V8jxCCCH6y9BAAFdrU2QU8gcfvm7WCpfFksTR0gSTe6lnlqaoqAhTp05FWloapk2bhh9++AE2NnUbCmTZtHBAjinTeteSC/BCV/krVqjD06IKzNp+DfeePqsM0auNPbbMCBJPR+sKpUbeevTogR9//BEpKSkYMGCAXDVO1cHT01Pm3y4aio2NxSuvvIKioiJYWlpi9erVuHjxIk6fPo3Zs2cDAO7fv49Ro0ahtFT2pIyEEEKaL1k2LahivZumLFy4EGlpaQCAnTt3IiAgAKdPnwYgfeQtr7QK0Q9z5X6mLk2drjvxoFHgBgBXkvIREZmopR5xUyqUnDVrFgCgY8eOiI2NRWhoKDw9PdGlSxfY2dnxroVjGAZbtmxR5vEAgOXLlyMoKAhBQUFwcXFBcnIy2rSRveguACxatAjl5eUwNDTEiRMnEBISIj43cOBAdOjQAR988AESEhLw7bffYvny5U3u4ejoiLy8PJmfGRkZibCwMLn6SQghRHe0sjWV2kYVaUI0Yd++fdixY0ejY2lpaRg8eDCioqLg6ebHeW16QTmO3HnKWV+Vz7WUfLmvUYeKaiEOcOSnO3rnKT4c3knDPeKnVPC2fft28XAwwzBgWRapqaniyJ0Ly7IqC95Wrlyp1PUxMTGIiooCALz22muNArd67777LrZt24Z79+7h+++/x5IlS5pkp544cSJKSmTPfu3q6qpUvwkhhGiXLJsW9CF4KygowJw5cySeGzhwIEJDQ5GUx73uO72ggnfKdLCvM07dy5Z47t7TEpRV1cJCy9OSt9MLOevEpuaXQyhiZVrjqClK/W15eXmpdC5fG/bv3y/+eubMmRLbCAQCTJs2DUuWLEFBQQGioqIwZMiQRm1+/PFHdXaTEEKIjpGWLsTIgEE7J8XLYmmKra0tvvnmG7z99tsoLn42bWhtbY1t27ZBIBDwThGn5pcjKbeM8/zs0LZIyCyRWI1BKGIRm1aIPu0dlXsRAEoqa/BTZCL2XM8AwGJoZ1e8O8QHDpbSN4zcSC3kPCdigbzSKjhbSx9p1RSlgrfk5GQVdUN7oqPrslBbWFigR48enO0GDBgg/vr8+fNNgjdCCCEti7Q1b+2drWBsqJIqlGrFMAymTZuGsLAwzJw5E2fOnAFQNyhRX43B1MgAzlYmyC6panI933RpKxtTBHnbo2drO4nBG1BXC1UVwdvCP2/hTMKzEb7fr6QiLqMI+9/qC4GUUTNpCYOzS3QreNP9f1Vqdu/ePQBA+/btYWjIHct26vRsvrv+GkIIIS2XtBJZ+rRZAaibTTt58iTWr1+PSZMmYerUqY3Oy1MSrN4L3VpBIGDQw9ues801FWxauPQor1HgVu92ehGOSCnXxbIsbkoJ3rKKK5Xqn6opNfImEAjAMAy+/PJLfPDBB6rqk8ZUVlYiN7dud4yHhwdvWzs7O1hYWKCsrEzqmj5FbN68GefPnwcA3LlzR3ysfj3euHHjMG7cOJU/lxBCiGKkTZuqIjmvpgkEAixYsEC8Nr0hT3vzJtOLoppKiMqLYWgjOQHwuG7uAICere04n3kzpUDpNWXH73InBz4RnyUxiXG9tPyKRlUiJMkqbjriqE1KBW/GxsaoqanR2/JYDTcYWFpKX5dQH7ypI13I+fPnm+z0uXDhAi5cuAAA8Pb25gzeqqqqUFX17B9WwzULhBBC1MPc2BB25kYo4KjlqQ+bFbhIWs8uaeSt8OwOlN45BfvBc2DhP6jRdT4ulujkavXf11awMjFESVVtk3uUVNXiQVaJUn9f0Q9zOM9dTMyFSMRyTp3KUmNV10beWnR5rMrKZ2+GsbHkum0NmZjULXqsqFB9Rujt27eDZVnOP59++inntV9++SVsbGzEfzw9PVXeP0IIIU21d5b8i7+A0e/gTZLnE/VWJN9CyfVDYKsrkPfv98jZtxrCskLx+bHd3MXBnIGAQTcvW857KzN1mlFYgUc53Bsm8sqqkZDJnQ1CluAtu6QZBW+6UB5LGaamzxYfVlfzD5kCEI9umZnpRjboekuWLEFRUZH4jzqmdQkhhDQ1hmM6bri/K+w5irnrq4ZluESVpcj79/tG5yseXsaTLW+hIiUWQNO/m56tude9XU9WPN/beZ5Rt3oXErkTCMs28qZb06ZKBW9vv/02DAwM8M033+jlVJ2V1bPFpLJMhZaV1UX2skyxapKJiQmsra0b/SGEEKJ+rwZ74cVA90bH/Nys8fEo7qS2+qrhtGn+6Y0QljQNiNiaKhhaOqBHazt4PlfMvqc397q36zIEUFzOyVDZ4TxH8FZRLcS9p9JztDaraVNdK48lL1NTUzg61m1PTk9P521bUFAgDt5oWpIQQggAGBkIsO7lrjg0vx8+fcEP22cGYe9bfWQqnaVvWtmagWEAViQEGMnhg23YDBg5eGBct6Yjkt08bTk3JaTlVyBbgQBJKGJ5R9XqXU3Kl1hk/nZ6oUyVIXRt5E3vy2Mpy9fXF9HR0UhMTERtbS1nupCEhIRG1xBCCCFA3edZgIcNAjxstN0VtTI2FMDV2hRPiyrhOHIRzDv0Rt6xnyAqLwQAmLbuBqvAUTAQMBgZ4NbkegsTQ/i6WSEuQ/JM3bWUAonX8YnLKEIhx4aRhipqhLiZWojebR0aHedLzttQXlkVaoQiGBnoRoY1vS+Ppax+/fohOjoaZWVluH79Onr16iWx3dmzZ8Vf9+3bV1PdI4QQQnSGp505nhbVjZCZd+gNE3df5B3/CZUpt+EwciEYRoDQDo6cVQ16eNlxB2/J8gdvXNOhklxIzJUQvMk2XcuyQG5pFdxsdGNEtcWXxxo3bhy+/PJLAMC2bdskBm8ikQg7d+4EUFdGJDw8XKN9JIQQQnSBh50ZriY/+97A3AZO45aitigLhtZOAJ7ldpOkh7c9tl9Mlhg7XFegSP25B9I3K9S7kJiLd4d2FH8vS3LehrKLm0nw1hzKYwUHByM0NBTR0dHYsmULpk+f3qQ4/bp168RVFRYuXNikKD0hhBDSEng8twkBqJt5M7J1BQCYGRlgiJ8L5/VM9kNk7noPDiMXwtjRq9G5u0+KUVEthJmxbOnHSqtqZR45A4DY9CIUV9bA2rTuM1yW5LwN6dKmBaWCN11w/vx5JCYmir+vr5gAAImJidi+fXuj9jNmzGhyj/Xr16Nv376oqKjA0KFDsXTpUoSHh6OiogJ//vknNm7cCADw8fHBu+++q5bXQQghhOg6aSWyhvi5wMJEcmhRXl6Od+fNRvXTh3i6fSHsBkyHVc8xYP7b/FArYnErrRAh7RwkXv+8K4/zUCOUvtmgnlDE4srjfHFwKU/gBwBZEuq6aoveB2+bN29uUpmgXsMKBfUkBW/du3fHX3/9hSlTpqC4uBhLly5t0sbHxwdHjhxplF5E10RERCAiIgJCYdMdNYQQQoiypAVvYyXsMq33wQcf4OHDh3XfCGtQcGYzyhOvwHHkIhjaPAuoZA3eomVIEfK8C4m5CgdviuyGVRfd2DahA1544QXcvn0b77zzDnx8fGBubg5bW1v07NkTa9euxc2bN9G+fXttd5PXvHnzEB8fj5iYGG13hRBCSDPk7WDBec7W3AihHZwknjtx4gQiIiKaHK9KvYOye+fE31+TI1nvORmS8z6v4QYHuUfe9DF4W7x4MRYvXozs7GyJ54VCIVJTU5Gamsp7n8ePHyMwMBA9evSQr6ccpJWVev4Pn9atW+Pbb7/F/fv3UVZWhoKCAsTExOCDDz6AuXnTeX5CCCGkJWllawY/jrJfL/fwgLGh5LDim2++kXjcuFVHWAe/KP7+ekoBRDLkXUsvKMdjnpJYXBKzS5FZVIny6lqZkvM2pEu53mQO3r7//nusX7++0ZqyhhISEuDt7Y22bdvy3qeiogK3bt3CrVu35OooIYQQQrRv2Wg/mBk13lTQxtECCwZ14LzmwIEDeOeddxrtMmWMTOA4ajEYwbN7FVfWIjFHesWj8zxTpn5u1ujkyr3E6UJiLm6nF8mUnLchXRp5U/maN2mjW4QQQgjRXyHtHHBgfl/8HZOGrJIqdHK1wsy+3jA35g4pzMzM8O2332LMmDGYMWMGUlJSYBc2C0b2TdOKXEsugI8L//ryaJ78bqE+jhAKWc5i9BcSc9HeRf4yl9m0YYEQQggh+srHxQqfjJa/fmtYWBhu376Nqe9/iVu2khPeX0vJx6ReXhLPAdJLYvXv4IRqoQibzydJPH8+MRfFlbXydRxAflk1qmqFMDGULZWJOtGGBUIIIYRojLW1Nd59dzFnkv/rKc82ElRVVWHNmjWoqKgQH+MriWVqJECP1nYI9raHkYHk+2eXVPEm9+UovwoAyNGR0TcK3gghhBCiUd29bDmDpJS8cnGQtHLlSixZsgSBgYHiTArRPLtMe7VxgKmRASxMDNHdy46zXbVQJPG4qZEAXTxsOa/TlU0LFLwRQgghRKOsTI3Q0VXyrlWgbvTt4sWLWLt2LYC6TZEhISFYsWIFohIyOa8L7eAo/rpfe0fOdly6uNvCnSeXXU6JbmxaoOCNEEIIIRrXszX3yNjFhDRMmzYNItGzETKhUIhVq1bh+M8rOa9rmGeub3vZkv021L21LVysTDnP08gbUbmIiAj4+fkhKChI210hhBBCePXgCd5+++krPHr0qMlxQ0MjWAWNk3iNs5UJfBrsIu3iYQtLjlJdXAK97OBibcJ5XlfShVDw1oxQhQVCCCH6gi94q+04GD169mxyfMCkeTB2biPxmtAOTo02QRgZCNC7rb1cferuZQsXa90feZM7VciGDRvg7Ozc5HjDygurVq3ivJ6rQgMhhBBCWg4POzO4WJtIDIgYOw8s+P5PpJz5HatWrUJtbS369OkDYefRQL7k0a/+Pk3XuPVt74hT92SLOzzszOBsZQpnnpG3bB1Z8yZ38Pbzzz9znquPeFeu5J6PJoQQQghhGAY9W9vjyJ2nEs8vP5SAGX1eRPT54Vjw9jys+eEXTP2HuwRnXwkbFOTZtBD43+5U/pE33Qje5Jo2laeGqKL1RQkhhBDSMvBNnQLA9ovJ+Op6NfafiEKG0IazXedW1nC0rBsxS0xMFB9v72wJZyvukbSGAr1sAUgL3vRs2jQyMlKd/SCEEEJICxPW0QmrDvO3uZFaiBd+vAA3W+6gqn6X6a1btxAcHIyJEyfihx9+gI2NDfq1d8TemxlS+xL4XyBpaWIIC2MDlFULm7QpqqhBZY0QpkbarbIgc/A2YMAAdfaDEEIIIS1MWydL/K+nJ/66lsbbLq+sGnll1Zzn+3dwRFVVFaZOnYqamhrs3LkTkZGR2LZtG/q27yg1eDM1EsDX7VneORdrUzzOLZPYNru4Cl4O5rz3UzfabUoIIYQQrfl8vD+mh7RW+HpTIwF6eNth+fLliIuLEx9PS0vD4MGDcXzLWqlLtrq428LI4FlIxLdpIUsHNi1Q8EYIIYQQrTEyEGDlWH+sf7UbzBSYjuzVxgExly/h66+/lnjeyc4aHVyseO/RvbVto++deRP1UvBGCCGEEIKx3dxxYH5ftHWykOu60A6OyMnJgZVV0wCta9euWLFihdRdp4HP1UHlS9SbrQObFih4I4QQQohO8HGxwsH5/TAywFXma/r7OGH8+PG4c+cOBg4cKD5ubGyMXbt2wdjYWGIakYaaBm88I280bUoIIYQQ8oyliSEiJgVi2Wg/GAoY3rYu1ibo4FxXEsvLywsnT57EDz/8AFNTU3z++ecICAgAAPRqaw8Djnt52ZvD6bl0Is48wRuNvBGVotqmhBBCmgOGYfBavzb4443evHnahnV2bVQSSyAQ4O2338bdu3exePFi8XFrUyO82N1d4j0mBns2OebC80xa80ZUimqbEkIIaU6CvO1xZEEoQto6NDlnYWyA+QPbS7yubdu2MDBovPnhg+GdxIl46w33c8Sez9/Etm3bGu1I1fUqC3KXxyKEEEII0RQnKxPsei0Yf1xNxY5LKSirqkUnVyssf6Ez765QSff5a04IrjzOR1JuKTq72+DfnRH49fQpnD59Cvv378fGjRvh4uLCX99UB6ZNGZbqVTU7xcXFsLGxQVFREaytraVfQAghhLQwN2/eRHBwMGpra8XHHB0dsXHjRowfPx4Bnx5HSWWtxGvvrhwGCxPVj3/J+vlN06aEEEIIaVEqKysxderURoEbAOTm5uLDDz9EdXU179Rpdol2R98oeCOEEEJIi3Lw4EHcvXu3yXGBQCBOL8KX603b694oeCOEEEJIi/LKK6/gwIEDcHZ2bnR86dKl6NWrFwDARYerLFDwRgghhJAWZ8yYMYiLi8P48eMBAN27d8eyZcvE5/lyveVoedqUdpsSQgghpEVycnLCnj17sGvXLgQGBsLY2Fh8TpenTSl4I4QQQkiLxTAMpk2b1uQ4VxqSsnvRuOdQCYzyU3fXONG0KSGEEELIcySNvNXkpSHv3+/wx9JJ2LVrlxZ6VYdG3gghhBBCnvN8qhBWJETukW/B1lYDAgP4+dHIGyGEEEKIzni+WH3Rpb9R/fQhAMAhdCICAwO10S0AFLwRQgghhDRhamQAW3MjAEB11mMUXfwTAGDs1gHmQRNQWiW5+oImUPDWjERERMDPzw9BQUHa7gohhBCi9+pzvRnau8OqxwtgjEzgOGoxGANDZGmxxinVNm2GqLYpIYQQorypW64g+mGu+HthWQEMLOwAAL+/3gt92juq9HlU25QQQgghRAnPb1qoD9wAIKtEe7neKHgjhBBCCJGAP1Gv9qZNKXgjhBBCCJHg+ZG3hrIpeCOEEEII0S1cVRYAmjYlhBBCCNE5fNOm2Vqsb0rBGyGEEEKIBHzTprTmjRBCCCFExzha8m1YqIS2sq1R8EYIIYQQIoGxoQAOFsYSz1XVilBcoZ0qCxS8EUIIIYRwcOabOtXSpgUK3gghhBBCOPDneqPgjRBCCCFEp7jwpQvR0qYFCt4IIYQQQjjQyBshhBBCiB7hW/OWU0Ijb4QQQgghOoU/1xuNvBElRUREwM/PD0FBQdruCiGEENIs0LQpUat58+YhPj4eMTEx2u4KIYQQ0izoYpUFCt4IIYQQQjg4WBhDwEg+l12inSoLFLwRQgghhHAwNBBwlsmqEbIoKK/RcI8oeCOEEEII4aVrmxYoeCOEEEII4eFspVubFih4I4QQQgjhwZfrLVsLmxYoeCOEEEII4aFr6UIoeCOEEEII4cG75q2EgjdCCCGEEJ3CN/JG06aEEEIIITrG2Ypv5I2CN0IIIYQQncI3bZqthTVvhhp/IiGEEEKIHnGwMEagly2crEzgYm0KF2tTOP/3tasNd2CnLhS8EUIIIYTwEAgY7H2rr7a7IUbTpoQQQggheoSCN0IIIYQQPULBGyGEEEKIHqHgjRBCCCFEj1DwRgghhBCiRyh4a0YiIiLg5+eHoKAgbXeFEEIIIWrCsCzLarsTRLWKi4thY2ODoqIiWFtba7s7hBBCCJGBrJ/fNPJGCCGEEKJHKHgjhBBCCNEjFLwRQgghhOgRCt4IIYQQQvQI1TZthur3oBQXF2u5J4QQQgiRVf3ntrS9pBS8NUMlJSUAAE9PTy33hBBCCCHyKikpgY2NDed5ShXSDIlEIjx58gRWVlZgGEbtzwsKCkJMTIzW7yfrdbK0k9aG67ysx4uLi+Hp6Ym0tDStpnPRhfdOnmuUfe/kPSfpWHN879T9f07Wtup873TlfQPovZP3eEt671iWRUlJCVq1agWBgHtlG428NUMCgQAeHh4ae56BgYFK/0Mpej9Zr5OlnbQ2XOflPW5tba3VH0a68N7Jc42y75285/jaN6f3Tt3/52Rtq4n3TtvvG0DvnaLHW8p7xzfiVo82LBClzZs3TyfuJ+t1srST1obrvLzHtU0X3jt5rlH2vZP3nK6+b4Bq+6bu/3OytqX3TnP30oX3Tt9+XgK68d4BNG1KiFZQFQz9Re+dfqL3TX/Re9cUjbwRogUmJiZYsWIFTExMtN0VIid67/QTvW/6i967pmjkjRBCCCFEj9DIGyGEEEKIHqHgjRA9kpGRge+//x5Dhw6Fl5cXjI2N4erqipdeeglXrlzRdvcIh8LCQixYsAAhISFwdXWFiYkJ3N3dMXDgQOzZs0dqQk6iW7766iswDAOGYXD58mVtd4fw8Pb2Fr9Xz/+ZO3eutrunMJo2JUSPfPTRR1i7di3atWuHAQMGwNnZGQ8fPsT+/fvBsiz++OMPvPLKK9ruJnlOYmIiunXrht69e6N9+/awt7dHdnY2Dh06hOzsbMyePRsbN27UdjeJDO7du4fu3bvD0NAQZWVluHTpEnr37q3tbhEO3t7eKCwsxKJFi5qc69mzJ0aPHq35TqkABW+E6JG9e/fCyckJoaGhjY5HR0dj0KBBsLKywpMnT2hhr44RCoVgWRaGho1Ta5aUlKB3796Ij49HXFwcOnfurKUeElkIhUKEhISAYRj4+Phg9+7dFLzpOG9vbwBAcnKyVvuhajRtSogeefHFF5sEbgAQGhqK8PBw5Ofn486dO1roGeFjYGDQJHADACsrKwwbNgxA3egc0W1r165FbGwstm7dCgMDA213h7RgFLwRIqPs7GwcPnwYy5cvx4gRI+Do6CheOzFjxgy57pWamor33nsPvr6+sLCwgL29PYKDg/HNN9+gvLxcof4ZGRkBgMQgoSXT5fetsrISZ86cAcMw8PPzk/v65k6X3ru4uDisXLkSn3zyCY2QykCX3ruqqirs2LEDX3zxBX7++WfExsYq+Kp0CEsIkQkAzj/Tp0+X+T6HDx9mbWxsOO/VsWNH9tGjR3L1LSUlhTUxMWFdXV3Z2tpaOV9Z86ZL71tBQQG7YsUKdtmyZeycOXNYT09PFgC7YsUK5V5kM6Ur711NTQ3bo0cPtmvXrmx1dTXLsiw7ffp0FgB76dIlZV9ms6Qr713r1q0lXjd8+HA2JydHBa9UOyh4I0RGDf/je3p6skOHDpX7h9GtW7dYc3NzFgBraWnJrl69mr148SJ7+vRpdvbs2eL7derUiS0pKZHpntXV1Wz//v1ZAOzOnTuVeIXNky69b0lJSY36Y2RkxH799desSCRS0attXnTlvVu5ciVraGjIXr9+XXyMgjd+uvTeRUVFsTk5OWxxcTF7+fJldsSIESwANiQkRG//71HwRoiMli9fzh46dIjNzMxkWbbxB7GsP4zCwsJYAKyhoSF78eLFJue/+uor8T1Xrlwp9X5CoZCdMmUKC4CdPXu2XK+npdDF9622tpZNSkpiv/zyS9bY2JgdP348W1NTI9fragl04b27desWa2RkxH700UeNjlPwxk8X3jsuQqGQ7devHwuAPXz4sMzX6RIK3ghRkLw/jK5evSpuP2fOHIlthEIh6+vrywJg7ezsxFM0kohEInbWrFksAHbKlCmsUChU9KW0KNp+355X/wG0YcMGma9pqbTx3nXt2pXt1KkTW1lZ2eg4BW/y0bX/d1u2bGEBsEuWLJH5Gl1CGxYI0ZD9+/eLv545c6bENgKBANOmTQMAFBQUICoqSmI7kUiE1157DVu3bsXEiROxfft2CAT031kdVPm+STJ06FAAkOsaIhtVvHexsbFISEiAqalpowSvO3bsAABx6pCGzyLKU/f/O0dHRwBQeIOYttG2NEI0JDo6GgBgYWGBHj16cLYbMGCA+Ovz589jyJAhjc6L/r+9O4+J4nzjAP5d5L4PgYIHYJF4QFGLoCIBammtCmpXirG2gJrGpGq1eFRtEa+mitZYa1KtwkZLW4nBelQbNQqGCgHUoohSUDxBFEQrpyw7vz/8MRHZXdAuSwe+n4RkmHnfd57dJ+KTd+adUakwZ84cJCcnIyoqCnv37uVjCzqRrvKmSVlZGQCuEu4Musjd7Nmz1fY5c+YMiouLERERAUdHR/F5YqQbnf3vruWNNFLNG/9aEOnJlStXAACenp5a/6MeNGhQmz4tWmbcFAoFIiMj8dNPP7Fw62S6yNtff/0FDw8P2NjYtNr/8OFDrFixAgDw3nvv6Spk+j9d5G7Xrl1q+8TExKC4uBjLly/nQ3o7gS5yV1hYCFdXV9ja2rban5mZiW+//RYmJiZ4//33dRe0HrF4I9KDhoYGVFZWAgD69u2rta2dnR0sLCxQW1uL27dvtzq2Zs0aKBQKWFpawsvLC+vWrWvTf8qUKRg2bJjOYu/JdJU3hUKBXbt2ITQ0FG5ubrCwsMDNmzfx+++/o6amBnK5HDNmzOi0z9ET6Sp3pH+6yl1qaio2btyIcePGwd3dHSYmJigoKMDx48dhYGCAH374Af379++0z9GZWLwR6cGTJ0/EbUtLy3bbt/wxqqmpabW/5RUvNTU1WL9+vdq+7u7uLN50RFd5mzZtGh4/fozs7GycOXMGdXV1sLe3x9ixY/Hxxx9j+vTpkMlkOo+/J9NV7kj/dJW70NBQXLlyBefPn0dGRgYaGhrg7OyMqKgoLFq0CP7+/jqPXV9YvBHpQUNDg7htbGzcbvuWd5PW19e32q9QKKBQKHQaG2mmq7yNHTsWY8eO1W1wpJWucqcJ/y12Hl3lLjg4uNU9cd0Jl6cR6YGpqam4/fTp03bbNzY2AgDMzMw6LSZqH/MmXcyddDF37WPxRqQHVlZW4nZHLsvU1tYC6NglA+o8zJt0MXfSxdy1j8UbkR6YmpqKzxW6c+eO1rbV1dXiH6N+/fp1emykGfMmXcyddDF37WPxRqQngwcPBgCUlJRAqVRqbHf16tU2fajrMG/SxdxJF3OnHYs3Ij1puWG9trYW586d09guIyND3A4MDOz0uEg75k26mDvpYu60Y/FGpCdTpkwRt5OTk9W2UalU2LNnDwDA1tYWoaGh+giNtGDepIu5ky7mTjsWb0R64u/vj6CgIADA7t27kZWV1abN5s2bxaeEf/bZZzAyMtJrjNQW8yZdzJ10MXfayQRBELo6CCIpyMzMRElJifh7ZWUllixZAuDZdP2cOXNatY+JiWkzxoULFxAYGIj6+npYWlpixYoVCA0NRX19PX799Vfs3LkTAODl5YW8vLxWq67o1TBv0sXcSRdz18kEIuqQ6OhoAUCHfzQ5dOiQYG1trbGfl5eXUFxcrMdP1r0xb9LF3EkXc9e5eNmUSM/Cw8Nx8eJFLFq0CF5eXjA3N4etrS38/PywYcMGXLhwAZ6enl0dJr2AeZMu5k66mDv1eNmUiIiISEI480ZEREQkISzeiIiIiCSExRsRERGRhLB4IyIiIpIQFm9EREREEsLijYiIiEhCWLwRERERSQiLNyIiIiIJYfFGREREJCEs3oiIiIgkhMUbERERkYSweCMiIiKSEBZvREQEAJDJZJDJZEhISOjqUIhIC8OuDoCIqDPU1tYiJSUFBw8eRH5+PiorK2FoaAgnJyc4OzvD19cXISEhCA4OhouLS1eHS0TUYSzeiKjbycnJQVRUFG7cuNFqf2NjI0pLS1FaWors7Gzs2LEDzs7OuHfvXtcESkT0Cli8EVG3UlJSgrCwMPzzzz8AgIiICEybNg1eXl4wNjZGZWUl8vPzceLECZw+fbqLoyUienks3oioW1m5cqVYuCUlJSE2NrZNm7CwMCxevBgPHjxAamqqvkMkIvpXuGCBiLqN5uZmHDlyBADg5+entnB7nqOjIz799FN9hEZEpDMs3oio23jw4AHq6uoAAJ6enq88ztOnT3H48GHMmzcPI0eOhJ2dHYyMjODg4ICAgAAkJCSgsrJS6xju7u6QyWSIiYkBAJw/fx4ffvgh+vXrBzMzM3h6euLzzz9vM87Zs2cRGRmJ/v37w9TUFK+//jqWLVuGJ0+eaDxXSEgIZDIZQkJCAABFRUX45JNP4OHhAVNTU7i4uCAyMhJZWVmv/J08r6ioCAsWLMDQoUNhY2MDMzMzDBgwALGxsTh//rzWvg0NDfjuu+8QEhKC3r17w8jICPb29hg0aBAmTJiALVu2tLlXkYheIBARdRNVVVUCAAGA4Ovr+8rjREdHi+No+nFwcBAyMzM1juHm5iYAEKKjo4U9e/YIxsbGasfx8vISysvLBUEQhMTEREEmk6ltN2LECOHJkydqzxUcHCwAEIKDg4WjR48KFhYWascwMDAQNm/erDHmlnarVq3S2GbNmjWCoaGhxu9FJpMJ8fHxavuWlZUJQ4YMafe7jYuL03h+IhIEzrwRUbdhb28PNzc3AEB+fj42bNgAlUr10uMolUoMGDAAcXFx2LdvH7KyspCbm4v9+/dj7ty5MDY2RlVVFaZOnYr79+9rHSs/Px9z5syBp6cnkpKSkJubi1OnTmHmzJkAgL///huLFy/GgQMHsGTJEgQEBCAlJQV5eXn4448/MGHCBADPZu7WrVun9VxlZWWYMWMGDA0N8fXXX+Ps2bM4e/Ys1q9fD2tra6hUKsTFxSEtLe2lvxMAiI+PR3x8PJRKJcaMGYNdu3YhKysLeXl5SElJwejRoyEIAtasWYNt27a16T9//nwUFhYCAGbOnIm0tDRkZ2cjNzcXR44cwerVqzF8+PBXio2oR+nq6pGISJc2bdrUahbHzc1NmDdvnpCSkiKUlJR0aIySkhJBpVJpPH7x4kXB0tJSACB8+eWXatu0zLwBEMaMGSPU1ta2aRMZGSkAEHr16iXY29sLcrlcUCqVrdoolUph1KhR4mxfU1NTm3FaZt4ACDY2NkJhYWGbNgUFBYK1tbUAQHB1dRUaGxvbtIGWmbecnBzBwMBA62dubm4WZs6cKQAQrKyshOrqavFYfX29YGRk1KGZtaqqKq3HiXo6Fm9E1K00NzcLs2bN0nhJztnZWYiKihIOHTqktUBrz8KFCwUAgre3t9rjzxdv6oopQRCE06dPi23Mzc01Fi1JSUliu/z8/DbHny/eNm3apDHmDRs2iO1SU1PbHNdWvMnlcgGA8Oabb2r93qqrqwUTExMBgPDjjz+K++/evSuOf/DgQY39iah9vGxKRN2KgYEBdu/ejWPHjiEsLAwGBq3/zFVUVGDfvn2IiIiAv78/rl271u6Y1dXVuHbtGi5fvoyCggIUFBTA1tYWAFBYWIimpiaNfX19fTF48GC1x9544w1xOywsDPb29hrHaHH9+nWN55LJZIiOjtZ4PDY2FjKZDABw8uRJje1e1NTUhGPHjgEApk2bJo6hjq2tLXx8fACg1QIJBwcHGBsbAwD27t0LpVLZ4fMTUWss3oioWxo/fjyOHz+OyspKHD58GKtWrcKkSZNgY2MjtsnLy0NQUBDKy8vb9L906RJmzZoFFxcX2Nvbw9PTE97e3vDx8YGPj4/4/k+VSoXq6mqNcXh5eWk81lIAvkw7batOPTw80Lt3b43HHR0d4e7uDgAoKCjQ2O5FhYWF4ire5cuXi+9A1fSTl5cHAK3eXGFiYoKoqCgAwP79++Hp6YmlS5fi6NGjePz4cYdjISIWb0TUzdnZ2WHSpElISEjA4cOHUVFRgaSkJNjZ2QEAysvL8dVXX7Xqs3v3bowYMQLJyckdenVWfX29xmPm5uYajz0/K9jRds3NzRrbOTk5aTzWwtnZGQDw8OHDdtu2aG9RhiYtBV+L77//HuHh4QCAmzdvIjExERMnToSDgwP8/f2xadMm8QHLRKQZ37BARD2KiYkJYmNj4erqivHjxwMA0tLSsHPnThgYGODq1auYO3culEolnJycsGTJErz11ltwd3eHlZUVjIyMADx7e8Ps2bMBAIIgdNnneZ62y5ktXiXW5wvGxMRE8Xtrj4WFRavfra2tcejQIeTk5CA1NRWnT59Gfn4+mpubkZubi9zcXCQmJuK3337D6NGjXzpOop6CxRsR9Ujvvvsu+vXrh9u3b6O6uhpVVVVwdHSEQqGAUqlEr169kJ6ervF+NW2XSrtKRUVFu21aZtE03V+njoODg7jd1NQEb2/vlw/uOf7+/vD39wfw7DJweno6kpOTceDAAdy/fx9yuRzXrl2DmZnZvzoPUXfFy6ZE1GO5urqK2y2XJi9fvgxA+0IDAOJ9Xf8lpaWlqKqq0nj8wYMH4tsLXqYAGzp0qLjY4Pjx4/8qxhdZWVkhPDwcaWlpWLBgAYBnl7IzMzN1eh6i7oTFGxH1SHV1deIDY62trcWZqJZVkC/er/W8e/fu4eDBg50f5EsSBAF79uzReFyhUIiXTd9+++0Oj2tubo5x48YBANLT05GTk/PvAtWg5RwA2n39GFFPxuKNiLqNmpoaBAQE4MiRI1rfrKBSqTB//nxx5WZERIR4v9jAgQMBPHvzQXZ2dpu+dXV1mDFjhtZFCl1p7dq1KCoqarP/ypUrWL9+PQDAxcUFkydPfqlxV65cKX5H06dP1/qIlebmZvz888+4c+eOuO/69evIyMjQeo7nZ/U8PDxeKj6inoT3vBFRt5KTk4Pw8HD06dMHU6ZMwejRo+Hm5gYrKys8evQIFy5cQFJSEi5dugQAsLGxwdq1a8X+H330EbZt2waVSoUJEyZg6dKlGDNmDExNTXHu3Dls2bIFxcXFCAwMxJ9//tlVH1OtgQMH4v79+xg1ahSWLVsmvqg+PT0d33zzjfhIjm3btomXQTsqMDAQ8fHxWL16NUpLSzFs2DDMnj0b77zzDlxcXNDY2IgbN24gKysL+/fvR1lZGS5duoS+ffsCAG7duoXQ0FAMGTIEU6dOhZ+fH/r06QMAuH37Nvbt24fU1FQAwPDhwxEQEKCjb4Wo+2HxRkTdhqGhIV577TXcu3cPd+/exfbt27F9+3aN7QcOHIhffvlFfPYZAIwcORKrV6/GqlWrUF1djeXLl7fpFxcXB29v7/9c8ebq6ootW7bggw8+UBu3gYEBNm7cCLlc/krjJyQkwNbWFl988QVqamqwdetWbN26VW1bY2NjmJqattlfWFgoXq5WZ/DgwUhLS+vQylminorFGxF1G6amprh79y6ys7Nx8uRJZGdno6ioCBUVFWhoaICFhQVcXV3h6+uLyZMnQy6Xq52Bio+Ph5+fH7Zu3Yrc3FzU1tbCyckJ/v7+mDt3LsLCwqBQKPT/ATtg4sSJyMvLQ2JiIk6dOoXy8nLY2toiKCgIcXFx//oRHAsXLkRkZCR27NiBEydOoKSkBI8ePYKJiQn69OkDHx8fhIWFQS6Xt3pgcFBQELKysnDixAmkp6fj1q1bYl7s7e3h6+sLuVyOmJiYl54VJOppZMJ/5QFFRET0SkJCQpCRkYHg4GCkp6d3dThE1Mm4YIGIiIhIQli8EREREUkIizciIiIiCWHxRkRERCQhLN6IiIiIJISrTYmIiIgkhDNvRERERBLC4o2IiIhIQli8EREREUkIizciIiIiCWHxRkRERCQhLN6IiIiIJITFGxEREZGEsHgjIiIikpD/AZnGTUGbE9SIAAAAAElFTkSuQmCC",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "dimension: 5 ; Convergence rate: theory=-0.38; old theory=-0.14; OS=  -0.41 \n"
     ]
    }
   ],
   "source": [
    "# plotting the error at every step\n",
    "pl.loglog(N[1:], err_os[1:], label='OS',linewidth=5)\n",
    "pl.loglog(N[1:],6*N[1:]**(-a/(2*a+1)),'k:',label='Theoretical',linewidth=4)\n",
    "pl.loglog(N[1:],3*N[1:]**(b/(2*a+1)),'k--',label='Old Theoretical',linewidth=4)\n",
    "#pl.legend(loc=\"lower left\", borderaxespad=0.,fontsize=\"18\")\n",
    "pl.xlabel('Samples',fontsize=20)\n",
    "pl.ylabel('Error (variational)',fontsize=20)\n",
    "pl.xticks(fontsize=20)\n",
    "pl.yticks(fontsize=20)\n",
    "pl.show()\n",
    "i=10\n",
    "z_os = np.polyfit(np.log(N[i:]), np.log(err_os[i:]), 1)\n",
    "print('dimension:',d,'; Convergence rate: theory={:2.2}; old theory={:2.2}; OS=  {:2.2} '.format(-a/(2*a+1),b/(2*a+1),z_os[0]))"
   ]
  }
 ],
 "metadata": {
  "accelerator": "GPU",
  "colab": {
   "provenance": []
  },
  "kernelspec": {
   "display_name": "Python 3",
   "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.8.3"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 1
}
