{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### External Imports"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import pandas as pd\n",
    "import matplotlib.pyplot as plt\n",
    "import seaborn as sns\n",
    "import matplotlib as mpl\n",
    "from pathlib import Path\n",
    "import time\n",
    "\n",
    "from sklearn.cluster import KMeans \n",
    "from sklearn.cluster import SpectralClustering\n",
    "from sklearn.cluster import AgglomerativeClustering"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "import sys\n",
    "\n",
    "sys.path.append('../../src')\n",
    "\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Imports from src"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "from base_exp_gen import generate_experiment as GEN_EXP\n",
    "from clustering_algos import agglomerative_clustering, kmeans_clustering, box_clustering\n",
    "from evaluations import computeATT_per_cluster, predict_cf, calculate_ite, get_homogeneity\n",
    "\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Plotting Parameters"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "fonts = {'family': 'cmr10','weight': 'normal',\n",
    "            'size': 20}    \n",
    "div = 1\n",
    "\n",
    "l1 = 22/div\n",
    "l2 = 20/div\n",
    "l3 = 13/(div-0.2)\n",
    "\n",
    "mpl.rcParams['xtick.labelsize'] = l2\n",
    "mpl.rcParams['ytick.labelsize'] = l2\n",
    "mpl.rcParams['axes.labelsize'] = l1\n",
    "mpl.rcParams['font.serif'] = 'Times New Roman'\n",
    "mpl.rcParams['font.weight'] = 'normal'\n",
    "mpl.rcParams['font.size'] = 20\n",
    "mpl.rcParams['legend.fontsize'] = l3"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "np.random.seed(seed = 0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "pd.set_option(\"display.max_columns\", None)\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Convergence Experiment set up"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>X-parts</th>\n",
       "      <th>Clusters</th>\n",
       "      <th>N</th>\n",
       "      <th>iters</th>\n",
       "      <th>mean-hom</th>\n",
       "      <th>std-hom</th>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>box</th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>2</td>\n",
       "      <td>4</td>\n",
       "      <td>32</td>\n",
       "      <td>50000</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>4</td>\n",
       "      <td>16</td>\n",
       "      <td>512</td>\n",
       "      <td>6250</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>5</td>\n",
       "      <td>25</td>\n",
       "      <td>1250</td>\n",
       "      <td>2000</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>10</td>\n",
       "      <td>100</td>\n",
       "      <td>20000</td>\n",
       "      <td>250</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>12</td>\n",
       "      <td>144</td>\n",
       "      <td>41472</td>\n",
       "      <td>86</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>5</th>\n",
       "      <td>20</td>\n",
       "      <td>400</td>\n",
       "      <td>320000</td>\n",
       "      <td>15</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>6</th>\n",
       "      <td>25</td>\n",
       "      <td>625</td>\n",
       "      <td>781250</td>\n",
       "      <td>4</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>7</th>\n",
       "      <td>33</td>\n",
       "      <td>1089</td>\n",
       "      <td>2371842</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "     X-parts  Clusters        N  iters  mean-hom  std-hom\n",
       "box                                                      \n",
       "0          2         4       32  50000         0        0\n",
       "1          4        16      512   6250         0        0\n",
       "2          5        25     1250   2000         0        0\n",
       "3         10       100    20000    250         0        0\n",
       "4         12       144    41472     86         0        0\n",
       "5         20       400   320000     15         0        0\n",
       "6         25       625   781250      4         0        0\n",
       "7         33      1089  2371842      1         0        0"
      ]
     },
     "execution_count": 8,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "x_partition = [[i,i**2,2*i**4, 50000] for i in [2,4,5,10,12,20,25,33]]\n",
    "def its(x):\n",
    "    for i, (x1,x2,x3,x4) in enumerate(x[1:]):\n",
    "        x[i+1][3]=(int(x[i][3]/(2*np.sqrt(x3/x[i][2]))))\n",
    "    return x\n",
    "\n",
    "a = its(x_partition)\n",
    "exps = pd.DataFrame(a, columns = ['X-parts','Clusters', 'N', 'iters'])\n",
    "name = 'box'\n",
    "exps.index.name = name\n",
    "exps['mean-hom'] = 0\n",
    "exps['std-hom'] = 0\n",
    "exps"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Run Experiment"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "running: 4 32\n",
      "time: 1900.429239988327\n",
      "running: 16 512\n",
      "time: 404.80516171455383\n",
      "running: 25 1250\n",
      "time: 169.69006299972534\n",
      "running: 100 20000\n",
      "time: 66.88768672943115\n",
      "running: 144 41472\n",
      "time: 32.267313957214355\n",
      "running: 400 320000\n",
      "time: 17.009103775024414\n",
      "running: 625 781250\n",
      "time: 8.183529376983643\n",
      "running: 1089 2371842\n",
      "time: 5.2496867179870605\n",
      "CPU times: user 43min 21s, sys: 5.55 s, total: 43min 27s\n",
      "Wall time: 43min 24s\n"
     ]
    }
   ],
   "source": [
    "%%time\n",
    "col_select = ['x0', 'x1']\n",
    "gen = []\n",
    "for i,row in exps.iterrows():\n",
    "    c, n, iters = row.Clusters, row.N, row.iters\n",
    "    homc = []\n",
    "    print('running:',c,n)\n",
    "    ss = time.time()\n",
    "    for j in range(iters):\n",
    "        ###gneerate data\n",
    "        N,D = n, 2\n",
    "        x = int((N)**(1/4))\n",
    "        clusters = x**2\n",
    "       # print(\"Number of Clusters:{}, Number of Points:{}\".format(clusters, N))\n",
    "        kw_generate =  {'N':N, 'D':D,'f_gen_name': 'uniform_gen','low':0, 'high': 1}\n",
    "        centers = [[0.5,0.5],[0.5,0.5]]\n",
    "        r_small = [0, 0.325735]\n",
    "        r_big = [0.325735,0.46065886]\n",
    "        eligibilities = [2,1]\n",
    "        kw_cluster =  {'f_class_name':'circle_class4', 'centers': centers,\n",
    "                       'eligibilities':eligibilities,'r_small': r_small, 'r_big':r_big}\n",
    "        kw_treatment = {'f_treat_name' :'uniform_treat','choices':[0,1], 'probabilities':[0.5,0.5]}\n",
    "\n",
    "        std = 5\n",
    "        stats = np.array([[1, std], [0, std], [0, std], [1, std],[1, std],\n",
    "                          [2, std]])\n",
    "        kw_outcome = {'f_outcome_name': 'outcome1','treatment':'Treatment', 'cls':'C', 'stats':stats}\n",
    "\n",
    "        data = GEN_EXP(kw_generate, kw_cluster, kw_treatment, kw_outcome).dat\n",
    "        data = calculate_ite(data.copy(), treatment = 'Treatment',\n",
    "                             counterfactual = 'Ycf', outcome = 'Y', ite_name = 'ITE')\n",
    "        ### cluster data\n",
    "        data2,_ = box_clustering(data, clusters=c,col_select = col_select, cluster_name = 'A')\n",
    "        ### calc hom\n",
    "        res = computeATT_per_cluster(data2.copy(), hom_name = 'HOM',weight_names = 'W', \n",
    "                                     cluster_name  = \"A\", att = False)\n",
    "        hom = (res['HOM']*res['W']).sum(axis = 0)\n",
    "        homc.append(hom)\n",
    "        \n",
    "    ###get stats\n",
    "    ee = time.time() - ss\n",
    "    print('time:',ee)\n",
    "    homcnp = np.array(homc)\n",
    "    gen.append(homcnp)\n",
    "    mean = np.mean(homcnp)\n",
    "    std = np.std(homcnp)\n",
    "    exps.loc[i,'mean-hom'] = mean\n",
    "    exps.loc[i,'std-hom'] = std\n",
    "    \n",
    "    "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "\n",
    "### Convergence Statistics"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>X-parts</th>\n",
       "      <th>Clusters</th>\n",
       "      <th>N</th>\n",
       "      <th>iters</th>\n",
       "      <th>mean-hom</th>\n",
       "      <th>std-hom</th>\n",
       "      <th>std-err</th>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>box</th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>2</td>\n",
       "      <td>4</td>\n",
       "      <td>32</td>\n",
       "      <td>50000</td>\n",
       "      <td>0.508666</td>\n",
       "      <td>0.052547</td>\n",
       "      <td>0.009289</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>4</td>\n",
       "      <td>16</td>\n",
       "      <td>512</td>\n",
       "      <td>6250</td>\n",
       "      <td>0.741927</td>\n",
       "      <td>0.019040</td>\n",
       "      <td>0.000841</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>5</td>\n",
       "      <td>25</td>\n",
       "      <td>1250</td>\n",
       "      <td>2000</td>\n",
       "      <td>0.761064</td>\n",
       "      <td>0.010822</td>\n",
       "      <td>0.000306</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>10</td>\n",
       "      <td>100</td>\n",
       "      <td>20000</td>\n",
       "      <td>250</td>\n",
       "      <td>0.882046</td>\n",
       "      <td>0.002272</td>\n",
       "      <td>0.000016</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>12</td>\n",
       "      <td>144</td>\n",
       "      <td>41472</td>\n",
       "      <td>86</td>\n",
       "      <td>0.893937</td>\n",
       "      <td>0.001355</td>\n",
       "      <td>0.000007</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "     X-parts  Clusters      N  iters  mean-hom   std-hom   std-err\n",
       "box                                                               \n",
       "0          2         4     32  50000  0.508666  0.052547  0.009289\n",
       "1          4        16    512   6250  0.741927  0.019040  0.000841\n",
       "2          5        25   1250   2000  0.761064  0.010822  0.000306\n",
       "3         10       100  20000    250  0.882046  0.002272  0.000016\n",
       "4         12       144  41472     86  0.893937  0.001355  0.000007"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "exps['std-err'] = exps['std-hom']/np.sqrt(exps['N'])\n",
    "exps.head()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Appendix C Cluster Homogeneity"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAZwAAAEkCAYAAAAb2IchAAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAALEgAACxIB0t1+/AAAADh0RVh0U29mdHdhcmUAbWF0cGxvdGxpYiB2ZXJzaW9uMy4xLjEsIGh0dHA6Ly9tYXRwbG90bGliLm9yZy8QZhcZAAAgAElEQVR4nO3dd3xUZfb48c8BwkpUEJAiJQkgRelFiJSVLorYQNQFFAgESDRYWEVj30Vd11XK6i7ospbNT0T8SlFBpQQBYQG7LL0EEAUE6aEkOb8/7k0cQibJhGlJzvv1mtdl7nPnznkyIWee+5QrqooxxhgTaGVCHYAxxpjSwRKOMcaYoLCEY4wxJigs4RhjjAkKSzjGGGOCwhKOMcaYoLCEY4wxJijCIuGIyAARmSIiy0TkiIioiPyniOeqIyLTRWSPiJwSkR0iMlFEKvs7bmOMMYVXLtQBuB4DWgLHgN1Ak6KcREQaAF8A1YE5wAagPTAW6CMinVT1gF8iNsYY45OwaOEA9wONgIrAmPM4z6s4ySZJVW9W1fGq2h14GWgMTDjvSI0xxhSJhNvSNiLSFVgCpKjqYB9eVx/YCuwAGqhqlkfZxcBPgADVVfW4P2M2xhhTsHBp4fhDd3f7qWeyAVDVo8AKIBKIDXZgxhhjSlbCaexuN3kp3+xuGwUhFmOMMbmEy6ABf6jkbg97Kc/ef4m3E4hIPBAPUKFChbZ169b1X3RBkpWVRZkyJel7ROGV1rqX1npD6a17ONd706ZNv6hqtbzKSlLCKYi4W6+dVqo6DZgG0K5dO127dm0w4vKr1NRUunbtGuowQqK01r201htKb93Dud4ikuatLDxTZNFkt2AqeSmvmOs4Y4wxQVSSEs5Gd+utj6ahu/XWx2OMMSaASlLCWeJue4vIWfVyh0V3AtKBVcEOzBhjTDFMOCISISJN3FUFcqjqVuBTIAZIzPWyp4ELgbdsDo4xxoRGWAwaEJGbgZvdpzXd7dUi8ob7719UdZz779rAeiANJ7l4SsBZ2mayiPRwj+sAdMO5lJYciPiNMcYULCwSDtAKuDvXvvruA5zkMo4CqOpWEWkHPAP0Aa7HWWFgMvC0qh70W8TGGGN8EhYJR1WfAp4q5LE7+G2Ic17lu4Bh/ojLGGOM/xS7PhxjjDHFkyUcY4wxQWEJxxhjTFBYwjHGGBMUlnCMMcYAkJKSQkxMDGXKlCEmJoaUlBS/nj8sRqkZY4wJrZSUFOLj4zlx4gQAaWlpxMfHAzBo0CC/vIe1cIwxphQ7ceIE69ev54EHHshJNp5lycn+my9vLRxjjCnBjh49SlpaGjt27Dhnu2PHDvbv35/v63fu3Om3WCzhGGNMmElJSSE5OZmdO3cSFRXFhAkTvF7WOnTo0FkJxDOpbN++nV9//fWs4yMiIqhWrRrVq1endevWVK9enWrVqvHvf/+bQ4cOnXP+qKgov9XLEo4xxoSRvPpSRowYwZo1a4iKimLHjh18+eWXHDt2jB07dnDkyJGzXl++fHlq1KhB9erV6dChAzVq1MhJMA0aNODyyy+nWrVqVK1alfLly+e8rm3btme9L0BkZCQTJkzwW90s4RhjTJjIyMjgwQcfPKcv5eTJk0yaNAmAChUqUL16dapXr06XLl1y/l29enUaNmxIgwYNuPTSS6lcuTIRERGFfu/sFlRhW1ZFYQnHGGNC6PDhwyxYsIB58+bx8ccfn3MJzNP8+fOJiYlh9+7ddOvWjbJly/o1lkGDBvk1weRmCccYY4Js+/btzJs3j7lz57J06VIyMjKoWLEiV111FWvWrDnnMhlAdHQ0ffr0AeDnn3/2e7IJBks4xhgTYFlZWaxevTonyfzwww8A1K1bl5tuuomrrrqKHj160Lx5c2bNmhXwvpRQsYRjjDEBcPz4cRYuXMi8efP48MMP2bt3L2XKlKFp06bExcVx1VVX0bNnT+rVq0eZMr9NiQxGX0qoWMIxxhgf5Ddkec+ePXz44YfMnTuXRYsWcfLkSSIjI2nbti2DBw+mU6dOdO7cmWrVquX7HoHuSwkVSzjGGFNI3oYsf/DBB6SlpbF27VoAatSoQa9evWjfvj3dunWjdevWREZGhjL0sGAJxxhjCik5OTnPIcvvv/8+jRs3ZsiQIbRv354ePXrQqFGjYtmxH0iWcIwxpgBHjhxh/vz5pKWl5VkuIixevJhatWoFObLixRKOMcbk4ccff2Tu3LnMmTOHxYsXc+bMGUQEVT3n2KioKEs2hWAJxxhjAFXlf//7H7Nnz2bOnDmsWbMGgMsuu4wbbriBDh06cPr0aZ5//vkSOWQ5GCzhGGNKrczMTFauXJmTZLZs2QJAo0aNGDJkCB06dKBnz540bNgwZ+hy/fr1S+SQ5WCwhGOMKVXS09P57LPPmDNnDvPmzWP//v2UK1eOFi1aMGbMGGJjY+nRowe1a9fO8/UldchyMFjCMcaUeL/88gsfffQRs2fP5pNPPiE9PZ0LL7yQdu3a0aFDB7p06ULHjh2pUqVKqEMt0SzhGGNKpG3btjFnzhxmz57N8uXLycrK4tJLL6Vbt27ExsbSvXt3mx8TZJZwjDElgqry1Vdf5fTHfP/994Cz6OVtt91Ghw4d6NWrF02aNKFcOfvTFwr2UzfGFFtnzpzJ6Y+ZM2cOu3fvpkyZMlx55ZXExcXldPrHxMQgIqEOt9QLq4QjInWAZ4A+QFXgJ2A28LSqer9JxLnnuQW4F2gDXABsB94BXlDVk/6O2xgTPEeOHGHBggXMnj2buXPncvz4ccqXL0+bNm3o378/HTt25JprrqFGjRqhDtXkEjYJR0QaAF8A1YE5wAagPTAW6CMinVT1QCHO8yfgMeAY8D5wAOgMPA30FpFeqpoemFoYYwJhz549Z03CPH36NBUrViQ2NpbY2Fi6du1K+/btqVixYqhDNfkIm4QDvIqTbJJUdUr2ThF5CbgfmACMzu8EItIaSAYOAW1VdZu7X4DJwD3Aw8BTAYjfGOMnqsr69etzOv1Xr14NQM2aNbn++utzhi4fPXqUbt26hThaU1hhkXBEpD7QG9gBvJKr+EkgHhgiIg+q6vF8TnULIMDr2ckGQFVVRB4FEoExIvInVc30Zx2MMecnMzOTVatW5XT6b968GYCGDRsyePDgnCTTqFGjnEmYqampIYzY+CosEg7Q3d1+qqpZngWqelREVuAkpFhgUT7nqelut+UucM/zC04rqjnwzXlHbYw5L+np6SxcuJA5c+Ywd+7cnEmYzZs3Z/To0TlJpk6dOqEO1fhBuCScxu52k5fyzTgJpxH5J5xf3G293AUicjFwqfu0CZZwjAmY/G5SduDAgbMmYZ44cSLnJmVDhw6lc+fOdO7c2SZhlkCS18qnQQ9CZBowEhipqq/nUT4BeBR4VFWfy+c8HYEVwK9AG1Xd4VE2CUhynyao6j/yeH08zuU7atSo0XbGjBlFrlOoHDt2jIsuuijUYYREaa17uNV74cKFvPjii5w6dSpnX/ny5bnmmmvYv38/3333HVlZWVStWpX27dvTvn17WrVqRcWKFc+61XJhhFvdgyWc692tW7cvVbVdXmXh0sIpSPYA+nyzo6p+ISJTgVHAdyLyPnAQ6ARcBawDmgJ59t+o6jRgGkC7du20a9eufgk+mFJTUymOcftDaa17uNV76NChZyUbgNOnT/PZZ58RHR3NgAEDzpqEGRERUeT3Cre6B0txrXe4JJzD7raSl/KKuY7zSlVHi8hqnJbKQHf3l8C1QBxOwtlX9FCNMbllZGTwzTffkJqa6vUmZQCLFy+mXr16NgmzlAqXhLPR3TbyUt7Q3Xrr4zmLqk4HpufeLyLZl+vW+BSdMeYsGRkZfP3116SmprJ06VKWLVvGkSNHAChbtiyZmedeRIiOjqZ+/frBDtWEkXBJOEvcbW8RKeM5Us3t7O8EpAOrivoGItIbiAaWquqP5xOsMSVNfp38cHaCSU1NZdmyZRw9ehSAOnXq0LFjR5o2bUqbNm3Yt28fycnJdpMyc46wSDiqulVEPsUZiZYITPEofhq4EJjqOQdHRJq4r93geS4RqaiqR3Lta4DTN5MJjA9IJYwpplJSUoiPj89JEGlpacTHx7N161YuuOCCnBaMZ4Lp1KkTzZo1o02bNlx99dVERUWd1eFfrVo1u0mZOUdYJBxXAs7SNpNFpAewHugAdMO5lJac6/j17jb3xeB/iUg0Tr/Nr8DlQD8gAhihqkVuJRlTEuVujQCcOHGCJ598EnASTOfOnXMSTGxs7DkJJje7SZnJi18TjojchNvxr6pv+fJat5XTjt8W77weZ/HOyTiLdx4s5Kk+5LcBAxfjDBB4H2fhzu98icmY0mDnzp1ey2bMmEFsbCx169b1eciyMbn5u4XzPE7HvwI+JRwAVd0FDCvksXkOc1HVN4E3fX1vY0qjDRs2EBERwenTp88pi46O5vbbbw9BVKak8vdXlj3ATmCXn89rjPGjzMxMXnzxRVq1aoWInHNDMuvkN4Hg14Sjqj1UtZ6qnrO0jDEmPGzYsIHOnTvzxz/+kZYtWzJ16lRef/11oqOjERGio6OZNm2a9cEYvwunQQPGmADKzMzk5Zdf5rHHHiMiIoL777+fMWPG0LChM83t7rvvDnGEpqSzhGNMKbBx40aGDRvGypUrad++PQkJCdx5552UL18+1KGZUsQSjjElWGZmJhMnTuSxxx6jXLly3HfffSQkJOS0aowJpkInHBFZXMhDT+PcJuBL4B1V/bkogRljzs/GjRsZPnw4X3zxBVdddRVjxoxh0KBB1qoxIeNLC6eru1XOnWxJHmV3AhNE5B53bTNjTBDkbtWMHTuWxMREa9WYkPMl4XQDbgTuB/4LvAOkAVlADE6CiQUm4rRuugNDgakisl5VV/otamNMnjZt2sSwYcOsVWPCki8JJxO4F0hS1b/nUT5FRBKASUB3VY0TkeXAv4CxgCUcYwIkMzOTSZMmkZycnNOqSUhIoFEjbwuwGxN8viScx4F1XpINAKr6qoiMBB4DrlXVf4vIYzirPRtjAmDXrl1cc801rFixgnbt2pGQkGCtGhOWfEk4VwEfF+K4dTjroGX7H9DLl6CMMQXLzMxk8uTJjB8/noiICJKSkkhMTLRWjQlbviSc8kBUIY6LwlmZOVs6cMrLscaYIti8eTPDhg1jxYoV1ldjig1fEs53QEcR6aWqn+V1gIj0xLl85nkLgLrA/qKHaIzJlt2qefTRRylbtixJSUlcd9119OnTJ9ShGVMgX9ZS+5t7/DwRmSoiXUWknojEiMg1IvJPYJ577EsAIlIJaI0zqs0Ycx42b97MNddcwwMPPEDTpk2ZPHkyL7zwAhdccEGoQzOmUArdwlHV990BAM8AI9yHJ8EZIv2kqr7v7qsO/JXC9f0YY/KQlZWV06opU6YM9957L/fcc4/11Zhix6elbVT1WRFZANwD/B6o7RbtAT4HXlHVtR7Hb8YZ3WaMKYItW7YwbNgwli9fTtu2bXNGoP3ud78LdWjG+MzntdRU9StgeABiMca4srKymDJlCo888khOqyYxMZHGjRuHOjRjiswW7zQmzHi2atq0aUNCQgKDBw+2Vo0p9izhGBMmPFs1IsI999zDPffcY60aU2L4lHBEpDKQgLOuWi3A2/AYVdUG5xmbMaXGli1bGD58OMuWLaN169YkJiZaq8aUOL7cnuByYClQE++rRWfT8wnKmNIiKyuLv//974wfPx4RITExkXvuuYcmTZqEOjRj/M6XFs7fgMuAZcDLwGbgWCCCMqY02Lp1K8OHD+fzzz+3Vo0pFXy9H84OoJeqng5INMaUAllZWbzyyiuMHz8egISEBO69915r1ZgSz5eEo8BqSzbGFF3uVk32CDRbLcCUBr4knG9w+m+MMT6yVo0xviWcF4HZItJRVb8IVEDGlDRbt24lLi6OpUuX0rp1a8aMGcOQIUOsVWNKnUIv3qmqH+LcXvojEfmTiHR2F+6MyutRlGBEpI6ITBeRPSJySkR2iMhEdzi2L+fpLCJz3NefFJGdIvKxiNiSuiZosufVtGjRgjVr1jBmzBhSUlIYOXKkJRtTKvk68fNrYC/wqPvwRn09t4g0AL7AWfBzDrABaI9ze+o+ItJJVQ8U4jxjgFeB48AHwG6gDnArcJ2IPKaqE3yJzRhfbdu2jeHDh7N06VJatWpFQkKCtWpMqefLPJyuwAKcG7EBHMC/w6JfxUk2Sao6xeN9X8JpWU0ARhcQYwTwHHASaKuqGz3KnsVJmMki8qKq2k3hjN9lZWXx6quv8vDDDwMwevRokpKSuOKKK0IcmTGh50sr5E84yeYF4HlVPeSvIESkPtAbZ9j1K7mKnwTigSEi8qCqHs/nVFWASsB3nskGQFXXi8gmoDlwEXYXUuNn27ZtIy4ujtTUVFq1asWYMWO46667rFVjjMuXG7C1Ar5U1fH+TDau7u72U1XN8ixQ1aPACiASiC3gPPtw7i7aSEQaehaISCOgIfBNYS7NGVNY2SPQWrRowerVqxk9ejT/7//9P+Lj4y3ZGOPBlxZOOs7qAoGQvTrhJi/lm3FaQI2ARd5OoqoqIonAf4AvReQDnHv11AZuAdYBd/graGO2b9/O8OHDSU1NpWXLliQkJFirxhgvfEk4y4CmAYqjkrs97KU8e/8lBZ1IVd8TkT3AO8BdHkV7gX8D27y9VkTicS7fUaNGDVJTUwt6u7Bz7NixYhm3PwSz7llZWcyZM4dp06ZRpkwZEhISuOGGG6hQoQKrVq0KSgzZ7DNPDXUYQVds662qhXoAV+IMEhhb2Nf4cO5pOCPbRngpf9YtH1+Icw3GaY2lAE2ACu42xT3HzMLE1LZtWy2OlixZEuoQQiZYdd+2bZt269ZNAW3ZsqVOnTpV09PTg/LeebHPvPQJ53oDa9XL31VfWjjtcFoIL4nIAOATnCHHWXkdrKpv+XDu7BZMJS/lFXMdlye3n2Y68B0wRH/rD9ogIkNwLt3dJiJdVTXVh/iMISsri3/+85889NBDqCqjRo0iKSmJK6+8MtShGVMs+JJw3sBpIQjQCehYwPG+JJzsEWWNvJRnDwDw1seTrTcQASzVcwcfZInI50Bb95HqQ3ymlNu+fTtxcXEsWbKEFi1akJiYyJAhQ6hQoUKoQzOm2PAl4bxF4O5zs8Td9haRMp7JQkQuxklw6UBBF8ez13Wv5qU8e78tQGoKJSsri6lTp/LHP/6RrKws4uPjSUpKomnTQHVnGlNyFTrhqOrQQAWhqltF5FOcFkoiMMWj+GngQmCqeszBEZEm7ms3eBy7zN0OcCd3fudxfCtgAE7SXByQipgSZceOHQwfPjynVZM9As1aNcYUja9L2wRSAs7SNpNFpAewHuiAczvrTUByruPXu9ucu4+q6moR+TcwDFjjDotOA2KAm3Emrk5U1XUBrIcp5nK3akaOHMnYsWOtVWPMeQqbhOO2ctoBzwB9gOuBn4DJwNOqerCQp4oDPgeGAtcCFwNHgOXAa6o6w8+hmxJkx44dxMXFsXjxYpo3b05CQgJ33323tWqM8QOvCUdEfn8+J1bVz4vwml04rZPCHCte9ivOAIc3fH1/U3qpak6rJjMzkxEjRjB27FiaNWsW6tCMKTHya+GkUvRBAj6vFm1MqKSlpREXF8eiRYtyWjV33XUXkZGRoQ7NmBIlv6SwE+8JJxpn+f9f/B6RMUGiqkybNo1x48bltGqSkpJo3rx5qEMzpkTymnBUNcZbmYhkAe+p6vBABGVMoHm2apo1a0ZiYqK1aowJMLvsZUoVz1ZNRkYGcXFxjB071lo1xgSBJRxTaqSlpTFixAgWLlxIs2bNckagWavGmOCwhGNKrJSUFJKTk9m5cyeVK1fm+PHjiAjDhw9n7NixtGjRItQhGlOqWMIxJVJKSgrx8fGcOHECgIMHDyIi3H333UyZMsVaNcaEgC93/DSmWPjll19ISkrKSTbZVJUlS5ZYsjEmRCzhmBJh7969zJ07l549e1KzZk0OHsx7YYqdO3cGOTJjTLb8VhqIKuC1F+V3jKra/2wTUHv27OH//u//eO+991i2bBmqSs2aNbn++utZtmwZhw4dOuc1UVEF/VobYwIlvz6c7fmUKdDffXgrt/4h43e7du3i/fff57333mPlypWoKrVq1aJfv35069aNPn36UK9ePWbNmnVWHw5AZGQkEyZMCGH0xpRu+SWFPNcqK6Tzea0xZ9m+fTvvv/8+M2fOZM2aNQDUqVOHG2+8kU6dOnHTTTcRFRXFqlWraNKkCQCDBg0CyBmlFhUVxYQJE3L2G2OCL7+VBqx/x4TM5s2bmTVrFu+99x5ff/014FwOu/nmm+ncuTP9+vWjbt26+a7iPGjQIEswxoQRu+xlQsZznkxUVBSJiYmcPHmSd999l3XrnFsWxcTEcMstt9ClSxf69u1bYJIxxoQvSzgmJHLPk0lLS+Ohhx4CoH79+vTv358uXbpw3XXXUadOHRvKbEwJYAnHhERycvI582QAKleuzMcff0ydOnW48MILQxCZMSZQLOGYkPA2H+bQoUM0btw4yNEYY4LBBgaYoDt27Bhly5bNs8zmyRhTclnCMUGlqsTFxZGRkUG5cmc3sG2ejDElmyUcE1STJ09m5syZ9O3blxdffJHo6GhEhOjoaKZNm2bDmI0pwawPxwTNihUrGDduHE2bNuXhhx+mS5cujB07NtRhGWOCxFo4Jih+/vln+vfvT+XKlbnvvvu4+uqrQx2SMSbILOGYgMvIyGDgwIH8+uuvjBw5koEDB57Tf2OMKfl8SjgiEiMiU0Vki4icEJFML4+MQAVsip+HH36YZcuWceuttxIfH0/FihVDHZIxJgQK/TVTRJoCy4GKFLw4py3eaQB4//33eemll+jQoQNJSUlER0eHOiRjTIj40sKZAFQC5gMdgEqqWsbbIyDRmmJl48aN3H333dStW5exY8dy1VVXhTokY0wI+XIh/ffADuAWVT0TmHBMSXHs2DFuuukmAEaNGkXfvn2t38aYUs6XlsjvgDWBTDYiUkdEpovIHhE5JSI7RGSiiFQu5Ou7iogW4lE3UHUwzuTOYcOGsWnTJv7whz8wePBg67cxxvjUwtmEc0ktIESkAfAFUB2YA2wA2gNjgT4i0klVDxRwmh3A017KmgO3AutUdZdfgjZ5mjhxIrNmzaJHjx4MHz7c+m2MMYBvCec14EURiVHVHQGI5VWcZJOkqlOyd4rIS8D9OH1Io/M7gRvXU3mVicg77j+n+SFW48Xy5cv54x//SJMmTRgzZgzt2rULdUjGmDBR6EtqqvoqMBNYKCLXiYjfBgaISH2gN04L5ZVcxU8Cx4EhIlKk9epFpCpwC5AOvF30SE1+9u7dS//+/bnkkksYNWoUvXr1sn4bY0wOX4ZFb3P/GQN8CGSIyE9AVh6Hq6o28CGO7u72U1U963yqelREVuAkpFhgkQ/nzTYUpw/qLVX9tQivNwXIyMjg1ltv5ddffyUpKYlbbrnF+m2MMWfx5etnjMe/BYgAvK0lrz7GkX0DlE1eyjfjJJxGFC3hjHC3U4vwWlMI48aN44svvuD2229nwIAB1m9jjDmHLwmnXsCi+G0wwmEv5dn7L/H1xCJyDdAEZ7DAFwUcGw/EA9SoUYPU1FRf3y7kjh07FvS4ly5dyqRJk+jZsyd33HEHJ0+eDMnPLhR1Dweltd5QeuteXOtd6ISjqmmBDKQA2SsX+NpyAjeBUIjWjapOwx1U0K5dO+3atWsR3i60UlNTCWbcGzdu5IUXXiAqKopbb72V7t27h+xSWrDrHi5Ka72h9Na9uNY7XFYEyG7BeBt2XTHXcYUiIlWA/thggYA4fvw4N9xwAyLC8OHDuf76663fxhjjVbgknI3utpGX8obu1lsfjzd34wwWmKmqh4oSmMmbqjJkyBC2bt3KXXfdxbXXXmv9NsaYfPk8ZlVEbgMG4CQHbwt5+jpKbYm77S0iZTxHqonIxUAnnFbKKh/DHelube6Nn/3tb3/jgw8+4IYbbqBXr14238YYUyBfhkWXAWYBN+F9NWh1y3zqa1HVrSLyKc5ItERgikfx08CFwFRVPe4RTxP3tRu8xNsFuAL4oaDBAsY3n3/+OePHj6d58+b069ePHj162HwbY0yBfLmkNhq4GfgWJzH8H05iaQz0BbJn8j8L1C9CLAnAPmCyiMwWkedEZDHOKgObgORcx693H95kDxaw1o0fZU/urFy5MoMHD+baa6+1fhtjTKH4knCGACeB61R1IXAUQFU3q+p8VR2EM9/lEbz3xXilqluBdsAbOLc/eBBoAEwGri7EOmo53MU+B2CDBfwqIyODfv36cfjwYUaMGMHvf/9767cxxhSaL9dBrgBWqupe97kCiIioqgKo6r9F5H7gj8BnvgbjLqo5rJDHer3Jm7uaQAVf39/kLykpiTVr1jBs2DDat29v/TbGGJ/4enuCvR7PT7rb3EOZvwfank9QJvykpKTwj3/8g65du9KxY0frtzHG+MyXhPMTUMPj+c/utkmu42riLHtjSoh169YxcuRI6tWrx4033kivXr2s38YY4zNfEs5G4HKP5ytxRqQ9JCICOSPDrsH3+TImTB09epQbbriBMmXKEBcXx9VXX239NsaYIvEl4SwA6opI9o3pF+PcJO0mYI+IfAksxElC//BrlCYkVJXbb7+dtLQ0Ro0axZVXXmn9NsaYIvPlInwK8AtwBEBVM0XkJuB9oBnO5bYs4BVV/Ze/AzXB9+c//5n58+dz2223ccUVV1i/jTHmvPiyeOcvOEnHc99moIWINAaqAJvd40wx98knn/D000/TunVrOnfubP02xpjz5pevq6q6seCjTHGxc+dO7rzzTqpUqcKgQYNo37699dsYY85bkRfvFJHLReRqEfF5kqcJX6dPn+bGG2/k2LFjJCYmUr9+feu3Mcb4hU8JR0TKicgTIrIXZ9TacmC8R/lQEflCRJr5OU4TJKNGjeLbb78lLi6O2rVrW7+NMcZvCp1wRKQc8DHwJM6dN9dz7iKea4FYnHvQmGLmn//8J2+88QY9e/akadOm1m9jjPErX1o49wA9gUVAjKqe04pR1R+AHeCdHisAAB7USURBVDiLe5piZM2aNdx///3Ur1+ffv360a5dO+u3Mcb4lS/XSoYAB4CBBdzMbDu/3TDNFAMHDhygf//+lCtXjjFjxlC3bl3rtzHG+J0vLZzGwH8LcefMvUC1oodkgikjI4PbbruN3bt3c++993LJJZdYv40xJiB8STiKM7GzIDX5bWFPE+YeeeQRlixZwu23306tWrWs38YYEzC+JJztQEv3zp95EpEKQAvyvzGaCROzZs3i5Zdfpm3btnTu3Nn6bYwxAeVLwpkL1AHG5XPMw0BlYM75BGUCb/369YwcOZIqVaowdOhQatWqZf02xpiA8iXhvIRzS4LnROT/icit7v5LReQ6EZkOPA7sBF71c5zGj44cOcJtt93G8ePHeeCBB7jgggus38YYE3C+rKV2UET64LRe7gBux+nX6es+BNgF9FPVowGI1fhBRkYGcXFxrFu3jsTERC666CLrtzHGBIVPX2lV9XsRuRLnNtDXAfWBsjiJZj4wTVWP+z1K4xeqyosvvsisWbPo3bs3TZs2pXXr1tZvY4wJCp+voajqSZz73dg9b4qZTz/9lGeeeYYGDRowYMAALr30Uuu3McYETZEX7zTFy/bt24mLi6Ns2bLcd999iIj12xhjgsoSTilw9OhRBg8ezJ49e7j//vsBrN/GGBN0Xr/eisji8zivqmqP83i98ZMzZ84wbtw4vvjiC+644w4uu+wy67cxxoREftdTuuKMQsu9InRhaJGiMX6lqvzrX/9i+vTptGnThl69elG5cmXrtzHGhERhLuCvBt7GmYNjipHly5fz6KOPUrlyZRISEsjKyrJ+G2NMyOT3l+cd4GagPdAGWAC8AcxV1YzAh2bOx+7duxk5ciTHjh3jueeeIz09nX79+lm/jTEmZLwOGlDVQTgLcY7GubHaDcB7wE8iMlFEWvk7GBGpIyLTRWSPiJwSkR3ue1Uuwrmai8hbIrLLPdc+EVkqInf5O+5wc+TIEcaMGcPGjRsZNWoUkZGRtk6aMSbk8h2lpqpHVXWaqnYEmgAvAKeAJOBLEflaRJJE5NLzDUREGgBf4kwqXQ28DGwDxgIrRaSqD+caCnyN00JbBvwNmIXTH3X9+cYazlSV5557jg8//JCePXvStm1batasaf02xpiQ82Vpm03AIyKSDFwLDAVuxEkMfxWRmao65DxieRWoDiSp6pTsnSLyEnA/MAGntZUvEYkFXgd+APqo6s+5yiPOI8awpqps2bKFyZMnU79+fe6++25Onjxp/TbGmLDg8zwcVc1S1fmqejvO6tEfARE4SahIRKQ+zm2pdwCv5Cp+EjgODBGRCwtxuhdwltsZnDvZuPGfKWqc4W716tU8++yzlClThoceeohDhw7ZfBtjTNgo0tdeEWmM08IZAlzm7j6fe+B0d7efqupZN3lT1aMisgInIcUCi/KJqw7QBafPaZ2IdAPa4gzT/gZYkvv8JcWuXbsYO3YsaWlpPP7442RlZVm/jTEmrBQ64YhIReBOnETTHqc/5ADwd+ANVf36POJo7G43eSnfjJNwGpFPwgGu8jh+Mc5cIk/fi8itqrqliHGGpcOHD5OcnMx///tfbr/9durVq0elSpWs38YYE1byTTgiIkAvnCRzE1AByAQ+5rch0v64RFXJ3R72Up69/5ICzlPd3Q4EfgFuxUlQ1XAuzQ0BPhKR5qp6uujhho9Tp07xyiuv8M4779C6dWsGDBjAoUOHrN/GGBN28lvaZgJwF1ALpzXzP5wk87aq7g1KdB7huNuCVjAo67Edoaofus+PiMjdwBVAO6A/zjyjs99EJB6IB6hRowapqannGXbg7d69m5dffpnKlStz3333UbZsWaKiovjqq69CHVrQHTt2rFh8Zv5WWusNpbfuxbXe+X0FfgTnD/xanETzX3d/bRGpXdCJVdWXv3jZLZhKXsor5jrOm1/d7SmcVphnPCoic3ASTnvySDiqOg2YBtCuXTvt2rVrgYGH0po1a5gwYQKHDh3iL3/5C8ePH6dq1ar07t071KGFRGpqKuH+mQVCaa03lN66F9d6F+aaSzv34Qst5LmzbXS3jbyUN3S33vp4cp/nqJfBAdkJqYIPsYWltLQ0nnjiCTZs2EBiYiJVqlShUqVKXHhhYQbyGWNM8OWXFHYSvEU4l7jb3iJSxjNZiMjFQCcgHVhVwHm+w+m7uVREauRx6a+Zu91x/iGHzq+//sqLL77IggUL6NGjB126dOHo0aP06NGjVF5KM8YUD14TjqrGBCsIVd0qIp/ijERLBKZ4FD8NXAhM9bx9tYg0cV+7weM8GSIyFUgGXhCRYdnJS0Sa4wx+yMBZdaBYOnnyJG+++SbTp0+nfv36jBw5kv3799s6acaYsBdOw5gSgC+AySLSA2deTwegG86ltORcx2fP+8l9+4RngR44Ax6ai0gqzii1/sAFwIPFdVh0VlYWn3zyCS+99BIiwvjx4zl48KDNtzHGFAthc8dPVd2K01f0Bk6ieRBoAEwGrlbVA4U8zwmchPM0EInTYroRJ5ldr6ov+T34IFm7di3PP/88u3bt4oEHHiAiIsLWSTPGFBvh1MJBVXfhLN5ZmGO93hjOTTpPuY8SYdu2bbz88susWrWKgQMH0qxZM44cOWLzbYwxxUbYtHCMdwcOHOD1119n1qxZtG7dmoEDB7Jv3z5bJ80YU6xYwglz6enpzJw5k9dee43KlSvz4IMPsm/fPuu3McYUO3YtJoxlZmby2Wef8corr+RM7jxz5oz12xhjiiVr4YSx1atXM3XqVNatW0d8fDx16tQhIyPD+m2MMcWSJZwwtXnzZt566y3mz59P9+7d6dmzp/XbGGOKNUs4YWj//v3MmDGDt99+m5iYGMaMGcPPP/9s/TbGmGLNEk6YOX78OHPmzGH69OkAjB8/nmPHjlm/jTGm2LOOgDCSkZHBwoULefPNN9mxYwePP/44l1xyic23McYHR44cYd++fZw5U2LvJk+lSpVYv/58brLsu4iICKpXr35el/TtL1gYWbVqFe+99x7Lly9n4MCBtGnTht27d9s6acYU0pEjR9i7dy+1a9emQoUKOPeQLHmOHj3KxRdfHLT3U1XS09P58ccfAYr898guqYWJDRs2MHfuXGbOnEmrVq248847+emnn6zfxhgf7Nu3j9q1axMZGVlik00oiAiRkZHUrl2bffv2Ffk8lnDCwN69e5k3bx5vvvkmlSpVYty4cRw6dMj6bYzx0ZkzZ6hQodjf7ipsVahQ4bwuVVrCCbFjx47x4YcfkpKSwsGDBxk/fjwRERE238aYIrKWTeCc78/WEk4InTlzhs8++4y5c+fy7bffMnLkSBo0aGDzbYwxJZIlnBBRVVauXMmiRYuYN28e3bp1o0+fPtZvY4wpsSzhhMj//vc/UlNTefPNN4mOjiYhIYGDBw9av40x5ixvvPEGIpLzKFOmDA0bNqRfv36sXbs21OH5xDoIQuCnn37is88+46233kJVGT9+PFlZWdZvY4zxKiUlhfr165OVlcXGjRuZNGkSPXr04NtvvyUmJibU4RWKtXCC7MiRI8yfP5+5c+eydetW7rvvPmrUqGH9NsaYfLVo0YLY2Fg6duzIgAEDeOedd3L+nhQXlnCC6MyZM3z66aesWLGCJUuWMGDAADp06GD9NsaEqZSUFGJiYihTpgwxMTGkpKSEOqQc2V9OPYcpf/XVV/Tp04eKFSsSGRlJx44dWbBgQU55eno6zZo1o2XLlpw8eTJn/9dff80FF1zAgw8+GNCYLeEEiaqybNkyvv76a95++21atmzJoEGDOHDggPXbGBOGUlJSiI+PJy0tDVUlLS2N+Pj4kCWdzMxMMjIyOH36NFu3bmXs2LFceOGF3HTTTQB8//33dO7cmT179vDaa68xY8YMIiMj6du3L/PmzQOceTTvvvsumzdv5oEHHgCcqRm33347zZs35/nnnw9oHayzIEi+//571q5dy+uvv54zufP06dPWb2NMgN1333188803Pr9u1apVnDp16qx9J06cIC4ujtdee83n87Vq1YqJEyf6/DrP13uqXLkyH3zwQc6VkT/96U+ULVuWJUuWULVqVQCuv/56rrjiCh555BH69esHQNOmTZk4cSKjRo2iR48ezJ49m7179zJ//nwiIiKKHF9h2F+5IPjxxx9ZunQp7777LgcPHuS5557joosusnXSjAljuZNNQfsDbcaMGTRo0ABVZdu2bbz++uvccsstLFiwgM6dO5OamkqfPn1ykg1AuXLl+MMf/sAzzzzD/v37qVatGgDx8fEsWrSIQYMGcerUKd555x0aNGgQ8DpYwgmww4cPM3/+fJYtW8ZXX33F6NGjady4Mbt377Z+G2OCoKitipiYGNLS0s7ZHx0dTWpq6nlG5bumTZvSrFkzAJo0acKNN95I3bp1GTduHKtWrcqZVpHbZZddBsDBgwdzEg7AXXfdxcyZM4mKimLA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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "p = './convergence_plot/'\n",
    "Path('./convergence_plot/').mkdir(parents = True, exist_ok = True)\n",
    "exps.to_csv(p+name+'.csv')\n",
    "\n",
    "\n",
    "y = 'mean-hom'\n",
    "x = 'N'\n",
    "z = 'std-err'\n",
    "box = exps\n",
    "fig, ax = plt.subplots()\n",
    "line1 = ax.plot(box[x],box[y], 'k-o')\n",
    "ax.fill_between(box[x], box[y]-box[z], box[y]+box[z], color = 'k', alpha = 0.3)\n",
    "ax.set_yticks(np.arange(0.6,1.05,0.1))\n",
    "ax.set_xscale('log')\n",
    "ax.set_xticks([ 100, 1000, 10000, 100000, 1000000])\n",
    "ax.legend(['Box'], loc = 'lower right')\n",
    "ax.set_xlabel('Number of Points')\n",
    "ax.set_ylabel('Mean Hmg.')\n",
    "ax.grid(visible = True)\n",
    "plt.minorticks_off()\n",
    "fig.savefig('{}/CONVERGENCE.pdf'.format(p), bbox_inches='tight',pad_inches = 0)\n",
    "\n",
    "fig.show()"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python [conda env:jorje]",
   "language": "python",
   "name": "conda-env-jorje-py"
  },
  "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.7.2"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
