{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {
    "collapsed": true
   },
   "outputs": [],
   "source": [
    "from src.model import *\n",
    "from scipy.stats import multivariate_t\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "from matplotlib.lines import Line2D\n",
    "plt.style.use('seaborn-whitegrid')\n",
    "\n",
    "# PLOT PREAMBLE: LIKE LATEX\n",
    "LINEWIDTH = 6.00117\n",
    "\n",
    "SCRIPT_SIZE = 8\n",
    "SMALL = 10\n",
    "NORMAL_SIZE = 10.95\n",
    "LARGE = 12\n",
    "Huge = 24.88\n",
    "\n",
    "#Direct input\n",
    "plt.rcParams['text.latex.preamble']=r\"\\usepackage{lmodern}\"\n",
    "#Options\n",
    "params = {'text.usetex' : True,\n",
    "          'font.size' : NORMAL_SIZE,\n",
    "          'axes.labelsize' : NORMAL_SIZE,\n",
    "          'xtick.labelsize' : NORMAL_SIZE,\n",
    "          'ytick.labelsize' : NORMAL_SIZE,\n",
    "          'legend.fontsize' : NORMAL_SIZE,\n",
    "          'figure.titlesize' : LARGE,\n",
    "          'font.family' : 'serif',\n",
    "          'font.serif': ['Computer Modern Roman'],\n",
    "          'lines.linewidth': 1.5\n",
    "          }\n",
    "plt.rcParams.update(params)\n",
    "\n",
    "# For the warning when hiding yticklabels\n",
    "import warnings\n",
    "warnings.filterwarnings(\"ignore\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "outputs": [],
   "source": [
    "# Fixed hyperparameters of statistical model\n",
    "d, l = 300, 350\n",
    "sigma_sq_stat = 1.5\n",
    "sigma_sq = 1\n",
    "gen_conf_strength = .53\n",
    "r_sq_stat = 5\n",
    "eta = get_max_eta(gen_conf_strength, r_sq_stat)\n",
    "conf_strength, eta = get_params_from_generalized_conf(conf_eta=gen_conf_strength, r_sq_stat=r_sq_stat, eta=eta)\n",
    "\n",
    "M, alpha, beta, sigma_sq = generate_causal_params_for_fixed_statistical(d=d, l=l, r_sq_stat=r_sq_stat,\n",
    "                                                                                sigma_sq_stat=sigma_sq_stat,\n",
    "                                                                                sigma_sq=sigma_sq,\n",
    "                                                                                conf_strength=conf_strength,\n",
    "                                                                                eta=eta)\n",
    "ridge = RidgeLimit(alpha=alpha, beta=beta, sigma_sq=sigma_sq, M=M)"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  },
  {
   "cell_type": "markdown",
   "source": [
    "## Student-t distribution (unbounded 4-th moment, df=3)"
   ],
   "metadata": {
    "collapsed": false
   }
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "outputs": [],
   "source": [
    "df = 3\n",
    "def sample_multivariate_t(n_samples):\n",
    "    return multivariate_t.rvs(loc=np.zeros(ridge.l), shape=(df - 2) / df  * np.eye(ridge.l), df=df, size=n_samples)\n",
    "ridge.sample_z = sample_multivariate_t"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "outputs": [],
   "source": [
    "gams = np.linspace(0.1, 5, 80)\n",
    "lam_small, lam_large = 0, .5\n",
    "res_dict_small = ridge.compute_risks(lam=lam_small, gams=gams, include_finite=True)\n",
    "res_dict_large = ridge.compute_risks(lam=lam_large, gams=gams, include_finite=True)\n",
    "res_oracle = ridge.compute_oracle(gams, include_finite=True)"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "outputs": [
    {
     "data": {
      "text/plain": "<Figure size 300.058x180.035 with 1 Axes>",
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAR8AAADWCAYAAADox4aFAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjUuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8qNh9FAAAACXBIWXMAAA9hAAAPYQGoP6dpAABAXUlEQVR4nO2de1wU1/n/P+A1KgsqXhJYNFTxsqjxW03K0hiVqgtJY8Aka5PaYBRN2ygmwV5SoVZt0gpNxV5ewppi4jcNayK5GVlzqVa/uyRN2qiwGPPKD+MOpBrBsIs3FHh+f0xm3IVddnZ29obn/XrNC3b2XJ45O/PMc55zznOiiIjAYDAYQSY61AIwGIwbE6Z8GAxGSGDKh8FghASmfBgMRkhgyofBYIQEpnwYDEZIYMqHwWCEBKZ8GAxGSGDKh8FghASmfBgMRkiIeOVjtVphMBhgNBphNBphsVhgNBoVK99kMmHSpEmwWq2KlakkFovFb9ksFgu+973vwWQyKSSVfOS2t6/5iouL4XA4fKpDaCcl769wJ5DXGvHKp7CwEHl5edDr9dDr9bBarbDZbOL3DofDrwbU6XTQarV+yylVDl/k5TgOVqsVGo3GL9m0Wq0i16gEctvbl3xWqxU7d+70mq77b6HVarFo0SKfZYsU3N17Wq02YAooopUPx3E9zuXl5SEuLq7XNKFAqhy+yFtcXIy8vDy5IrkQExOjSDmRAMdxUKvVktJ1x/ne6mu4u161Wg2bzeazlSiFiFY+arUaHMf10Mx6vV78v6ysLNhiuUWqHFLTORwOSQ8QwxWTyQSdTicpbbjcO8HC0/VmZWWhurpa8fr6K15ikCkoKEBRURFKSkqQlpaGpUuXiua3xWJBY2MjzGYzAN6E5DgOJSUlyMzMRF5eHkwmEwoLC1FaWirms1qt2L9/P6ZNmwYAsNvtLnUKfha1Wo3a2lqsX78eFotFLFej0YDjOJjNZmzfvt2tHO4Uh9R0AFBdXS3KJ+Tt7bp6k8+5DJVKBYfDgdraWqxevRoqlUpsE4vFIip8nU4HtVqtSL1y2ltKvu44HA7xerzR22/hcDhgsVjcXkv3Mtxd+/r1613K2LRpk0tbeGtnqWX5cp8uXbrU4/VqNBqUlJS4vNQVgfoANpuNKisrac2aNZSSkkLV1dXid1u3bqXKykqX9OXl5VReXi5+XrNmDZnNZiIistvtlJGR4ZI+IyOD6urqxLqys7PF7yorK8WyysvLac2aNeJ3ubm5Yj53crjDl3RC2VKuS4p8znnNZjPl5uaK1yz8L5CdnU12u93veuW2t7d87nBu14yMDFF+T3i6dzxdizvcpXduK2c5pLSzL2X5ep/2du85l6UUEd3tEvqharUaer0e27dvR2lpKcrLy2WXWV1djalTp7qcc7Y+KisrERsbC4vFAovFAgCora0Vv3e2RmJiYry+jeXS1tYm+S3uTG/yOZcnWC0OhwOVlZU92iQxMdEnU9xTvXLb21u+7lgsFmRmZkqWtzd8/Y27p3eWU6VSifmltLMvZYXDfdobEd3t4jgOdrvdZZRDq9V6VD6+9Pd7Y+rUqS51+mqOmkwmGI1G8QdPTEx0a7r3Jm9MTExAnIDhiLv2ljMC4/wQcxyHsrIyZGVloaSkxK/fwlfkvDTklOXPfark9Xoioi0fACgpKXH53P0NFxcXJz6knrR7fX29mEar1aKtrc3le+dRgKysLNTU1PSo0xvd5aioqEBVVRWqqqpcbnYp8gJAUlKS15Ex5+uSgnNai8UCrVYLlUrl9prr6+s9WhK+1Cu3vb3lc1ePMB1DeAiXLl0KjUbj92/hK57axtd29rUsb/dpb9cbGxvba145RLTlA/Da3GAwiG8Ah8PhMvys0+lQXFwMo9EovgX0ej1KSkrEH2Pq1KkwGo3QaDRQq9UoKCiA0WgUTVqVSgWj0YiCggJoNBoUFBSguLhYNF21Wi2sVqv4ZhUc2/X19aisrIRarXYrhzukpsvMzERZWZnL26m363I4HL3Kl5SUBLVaDavVCo7jUFtbi9LSUgAQr9lgMIjOy9LSUrHN/alXbnurVKpe83myCJznshgMBuj1eo/zpLr/Ft5+4+7dPk/phTYVnMQGg8HlWt21s9yyfLlPH374Ybz00ks97j2r1RoQKyiKiAWQj1TWrl3rcaSFwVCKoqKiXhW6XCK+23Ujs379+htqqj8j+HAcB41Go7jiAZjyiWgEUz9c150xIh+LxaL8/J5vCGm3i+M4VFZWin3SQHvXGQxG+BAyhzPHccjPz0dVVRUAYPny5bDb7QHTsgwGI7wImeWzfPly5OXluSxpUKvVAelbMhiM8CMkysfhcGD27Nn46KOPxLkZnoY7Ozo6YLfbMWjQIERHMxcVgxFudHV1ob29HbGxsejfX3pnKiTdrrq6OgDXJ4txHIfly5ejoqKiR1q73Y4vvvgiyBIyGAxfGT9+PEaOHCk5fUiUjzCLUqVSQaVSiZP7iouLxRXLAoMGDQLAT3sfMmRI0GV1JisrC+fOncOLL76IKVOmeE1/9epVnDhxAlOmTMHAgQMl10NEmD9/Pi5cuIA9e/bg1ltv9Uds2O12PPPMM3jyyScxZswYnD17Fs899xyefvpptzNXu7q68Pnnn2PChAkRZ21GquyRKjcAXLp0CY2NjeKzKpWQKB9hJqjzLEq1Wu12oaLwQwwZMiSkAa+ICB9++CGuXbuGsWPHSpKlsbER77//vs9vBAC4du0aTp48iaamJkyfPl2u2AD4dWB//etfPX7uTmdnJwBg2LBh6Nevn191B5tIlT1S5XbGV6UZEhXrHBdFgOO4gKwfUQqHw4Fr164BAEaNGhXw+saPHw8ArMvJ6LOERPmoVCro9XqXhW51dXVhPcx+7tw5AMDQoUNx0003Bbw+oavFlA+jrxKyzuWmTZtQW1sLg8GA4uJi6PX6sJlk2NLSgiVLlogjcRzHYeXKlQCCY/UAzPJh9H1Cuqq9u3M5XFi1ahWqqqpw9OhR7N69G8uWLUNDQwOA4Cmf5ORkAMBnn30WlPoYjGAT8SE1AsG2bdtw9OhRNDQ0ID09HQAQHx+P5uZmn5TPzTffjHvuuQc333yzzzIIS07q6+tx7do1DBgwwOcyBDo6OnD16lXJ6QXn56VLlyLO+Rmpsoer3AMHDvRp7o4vMOXjBrVajd27d4uKBwDuv/9+7NixI2iWz7hx4xATE4O2tjZ89tlnsvbmIiLYbDY0NzfLkiGSra5IlT0c5Y6Pj0dSUhKioqIULZcpHzdwHIdly5a5nHv55ZcB+Nbtamlpwf/93/9BrVZj9OjRPskQHR2NadOmwWKx4NixY7KUj6B4EhISMGzYsIibP8IILV1dXbhw4QKampoA8C9EJWHKxw3r1q1DQ0MDkpOT/fL5XL16Fa2trT51eZyZMWMGLBYLjh8/joceesinvB0dHaLiGTt2rKz6GYxhw4YBAJqampCQkKBoF4wpHzcIAei3bdsGtVqNQ4cOYfbs2Th79mzQul0AxMmFx48f9zmvoPCEm4fBkItwD129epUpn0AzcuRI7N27V/wsBO0KlfI5duyY7DJYV4vhL4G6h9idKRFhkmEwlY8w4vXll1/KdhozGOEKUz4SkaN84uLicNtttyEuLk5WnTExMeJ8Hzldr76A1WpFcXExJk2ahLVr13pMl5OTg9mzZ8NgMNww+5lFOqzbJYFLly7h0qVLAPhhR6ncdNNNSExM9Gs5xowZM9DQ0IDjx49j/vz5ssuJVDQaDTQaDdra2mA0Gt3ut261WpGYmIjExESXbZOKi4vBcRzb4SNMYZaPBASrZ+DAgT6trL948SK++OILXLx4UXbdSvh9+gKCEnIX+cBut7vdKjk9PR1ZWVnBEI8hA2b5SODLL78EAIwdO9aniVYOhwN1dXW44447ZIeH9WfEyx1EJFpxwWbIkCF+TVQTtkmWugC5t00XGaGHKR8JnDp1CgD8DuolhxkzZgDguxYdHR1+DXUSEb773e9K2t45EKSnp+PIkSOyFVBmZiaKiorAcZxo6QjRMM1ms0taq9WKwsJCce91i8WCkpISpKamiguYhf3IPSkpKXkcDgfKysowbdo02O12xMbGimmF/GlpaUhKSoLJZEJBQQHsdrvHcoUya2trkZSUFNaRHvyFdbskEErlc+utt2LYsGFob29XZOq90lPkg4lKpYJWq4XJZBLPeXIuazQarFq1Svys1WqxatUq1NXVITU1FVqtFjqdDiUlJR7rk5InNzcXq1evhk6ng16vR21trSifsD/8gQMHxD3ihWtwV25RUZGovFavXo2ioiJ/myysYZaPBEKpfIRlFjU1NTh+/DimTp0qu6yoqCgcOXIkYrtdwPV94fPy8uBwONz6enojNjZW7AKr1WpJI2Oe8hiNRpfvAD7Ubn5+vkt4GEHG7iFjupcr7A4K4IbYxYUpHwnIVT6DBg3CqFGjfI5t253p06ejpqYGx44dw9KlS/0qKyoqCkOHDvWrjFCi0+mQn58Pq9UKu93us1/H04DB8uXLxfhNAFBRUSEqDU95bDZbj+9UKpVLOQA8KsjueX1VpJEOUz4SkKt8RowYgTvuuAMjRozwq/6ZM2cCAP71r3/5VU5fQa/XY//+/eIkTCVwt3OKN5KSklBfX+9yjuO4G8JqUQLm8/FCR0cHbDYbgOvRBaXS1dWFa9euoauryy8Z7rzzTgBATU2N7EWqkY7wGwC89bNz505ZK/3b2toUy6PX68FxnEvXTXAqyynXbrf7LFskw5SPF5qamtDZ2YmBAwfilltu8Snv2bNnceDAAZw9e9YvGaZMmYJRo0bh8uXL+Oijj/wqK9KwWq1Yu3Yt9uzZg+LiYgC8I3fRokViN8VgMODAgQOoqakRZzhbrVYYjUbU19fDZDKJn+vq6mA0GsFxnDgJUSjXXd3e8lRUVKCsrAxGoxEGgwEajUYcobJYLDAajbBYLDAYDF7LdTgcovxC+UVFRX13xjaFORcvXqSPP/6YHA5HSOo/ePAgAaCJEyf6nJfjONq4cSNxHOe3HEuWLCEA9Nvf/lZSeqHdLl686HfdjBsbb/eSw+GQda8xy8cLoRzpcuauu+4CAPzzn/8MqRwMhlIw5eOFcFM+ZrNZ3D+MwYhkmPLxQrgon9TUVIwYMQIXL17Ef/7zn5DKwmAoAVM+XhCUj68jXQAwevRoLFiwwOf4ze6Ijo4WR71Y14vRF2DKxwv+WD79+vXDoEGDFNsKZe7cuQCY8mH0DZjy6YX29nZxRbsc5XP+/Hl89NFHOH/+vCLyCH6fI0eOoKOjQ5EyGYxQwZRPL5w+fRoAvz+7L0HEBNrb23H27Fm0t7crIs/06dMRGxuLtrY2HD16VJEyGYxQIUv5NDY2uj3/zjvv+CVMuOHc5QqH1eD9+vUT/T6HDh0KrTAMhp/IUj6//vWve5zjOK7X8ASRSLiMdDmzYMECAMBbb70VYkkYDP+QpXzMZrOLlbNz504sWbJEMaHCBX9GugLFfffdB4D3+/i7bCPSEJYpGI1GGI1GcUlCpGC1WpGTk9NrIHx/yl6+fDmWL1+ueNmBQpbyqaioABFhz549WLJkCerq6vD++++77HXVF/DX8omJicHUqVN9ivvsjaSkJMyePRtEhDfeeEOxcsMdg8EAk8mEvLw8MTDX6tWrkZubC6vVKqu87hQXFwdEMQh0D3DmK73Jp9FoXILnRwKylE9aWhoWLVqE2NhYaLVabNu2DTExMaipqVFavpDir/IZNmwYkpOTFd81NCcnBwBQVVWlaLnhitVqRXl5eY/V4iqVCgUFBcjPz/epPIfD0SPmDhD+Aee9yRcbGxtEafxHUjwfdz4egI8J/MEHH4j/v/POO1i4cKFy0oWYL774AoB85XP58mV8+eWXmDRpkqIKKCcnB7/85S/x/vvvo7W11bd9wYiAEEUyxJAhgAzHfWFhITIzM93GydFqteA4zqfA8p58k+EecD7c5fMVScqntrbWYwS91NRU8f++tC+43W4XdwmVq3xaW1vxn//8BzNnzlS0bVJSUqDRaGC1WvHWW29h2bJl0jISAd/9LhCiAPJITweOHPFZAVmtVmRmZnr8Xq1Ww2w2Q6/Xi0HbExMTkZ6eLuYXgr5bLBZwHAe73Q6DwQC1Wg2dTudTwHnAc5B3ITg/x3GwWq0oKCjwKbiYp6DzAFzkE2QoKytDUlISYmNje1hzQugOjUYDh8MhXr9OpxPDtlZWVmLatGniMx7UaIpSlr5bLBZJS+SlpvOFUIXUOHLkCAGgxMRE2WUoGVKjO4WFhQSA7rvvPrffuw2D0NVFlJ5OxKuh4B/p6bwMPmCz2SglJYUqKys9psnNzaXs7Gzxc2VlJeXm5rqkycjIoLq6OvH7wsLCHuVUV1fTmjVrXD5nZ2eT3W4nIiKz2exSjt1up5SUFJcyUlJSyGazeaynex3uqKyspIyMDDG9UF73vNnZ2aIs7q571qxZYt7y8vIesmRkZIjXZrfbxTq7E6iQGpIsn7S0tF6/r6mpgVqt9poukhAm8d12220hlcMTS5YswebNm2EymXDx4kVpcZmjonjLI4K6XcKb2DmSYXc4jusRWL/7G3zRokUoKyvzefdSX4O8C7GfhVG4uro6n+oT8BR0XsBiscDhcLhEc+x+zQ6HQ/QDdY8tbTQaoVKpxGsQ/nfelijQyIrh/OijjyI9PR0rVqzAihUrkJaWBrPZjHHjxuGBBx7wuTyr1QqLxRJW3nphh1Bh36xwY/r06UhOTkZDQwPefvttPPjgg9IyRkUBERZAXqPR9IiV7AzHcV79PXFxcbIGRHwN8u5wOLB27Vq/40t7q8dqtXrtzjlvsmg2m10c9oIyd96GaNWqVUF1Wssa7dLr9VixYgXq6+vBcRxWrlzpc9/WmcLCQrejD6FECctnwIABUKlUGDBggDJCOREVFSU+cDt37lS8/HCioKBAfNN3R3iDe3txcRyHxMREt/mVQvAbbd++HXl5eS5WidLzkaRs+6PRaKDT6WCxWLBlyxYXeZKSkgDwlpXzEczg97KUjyCgxWJx0dByliAYjcaw6651dHSI5rI/lk98fDzmzJkja12YFPLy8hAVFYV3330X/+///b+A1BEOaLVarFy5Ehs2bHA5z3EcDAYDSktLe+Tp3t2prq7G+vXrAfBdqdbWVkl1+xLk3WKxuAzACMrB09C+PwjdMedyBUe6gNVqFR3q3ZWKXq/vIZfJZArqpE1Z3S5hUpfRaBR/0FdeecVnk81qtSI1NRUOh8PrrgJdXV3o7OyUI67PfPrpp7hy5QqGDh2K8ePHy65XyBcouZOSkrBw4UIcOHAAO3bswO9+97sedfcV1q9fD4vFguLiYvGtbbPZUFVV5fZtnZiYKHYpLBYLdu3a5eJH2b9/vzjaBVwfGeI4DiaTCWq12iXIu1ardQnyrtfrUVZWBoAP8l5QUCDuWCpsJqjRaJCYmAij0Yi8vLwegePddRWFoPNCPYJF110+nU6HqqoqcatmAGLgfCGfWq3G7NmzxbLVajU2b94sWkBVVVUoKSmBRqNBbGws1Gp1r5ZPZ2en2/tK7u4sUUREcjLu2bNHdDLv3LkTra2tGD58OFasWCG5DOEHMBgM4DgOmzZt6pHm0qVLOHHihBwRZWMymbBhwwZMnz4df/vb32SXY7fbYTabkZ6eHrC+9KFDh1BQUIC4uDjs378fAwcOdPl+ypQpGDJkSEDqDleMRiOsVqvb++lGwWKxwGw2i8YBwFtGy5cvx3vvvedTWVKfQV/vNdmbBjo7OFeuXAnAt1XtvkwKA4AJEyYoMlempaUFjz32GJ577jlx9OLJJ5/Ejh07MHLkSFE2gB/l88fn09TUhCNHjmDChAlISEjwW3Z3pKam4o9//COamprQ0NAgzse6dOmSInu7RypSu1V9FavV2sPp7c2y8UZKSopb5XLhwgV8/vnnPpcnWfmsW7cOWVlZWLhwIdatW+c2TX19vaQZzlar1efZmtHR0YpEBPzxj3+M1157DceOHcPu3buxbNkyNDQ0ICoqSlybdvz4cQD8TqH+1BkdHa2o7O7o168f8vLysHHjRpSXl+Phhx8Wz9+IOHdbnCcF3mjk5eWJe4mpVCo4HA60tra69Y9JpV+/fm7vK+E+9xXJyicxMVEcdrTZbFi9enWPNFKdVXa7XZwJCvDOQAAufdxAsW3bNhw9ehQNDQ3iDNjk5GRs27ZNTBPuw+zdWblyJTZv3ozDhw+jrq7Oxel5o6HVam+YNW/e8KVnEQokKx/nOQJbtmzpMakLkL7RvVardbF8BI97MOb5qNVq7N69W1Q8ALB7925R9rNnz+LMmTOIiopSdC/wQJKQkID77rsPe/fuxbPPPouXXnop1CIxGF6RZS9FRUW5jWboTiF5w2AwwGKx9NhSNlBwHNdjLdSyZctEBShYPRMnTpQ2a7gX4uPjcddddwVsqN2ZX/3qVwCAl19+GZ9++mnA62Mw/EWW8ikuLu51xqkv5OXl4b333sN7770XFMtn3bp1aGhoQHJyMsxmszhLWPBjCcpHiWUVAwYMQExMTEAmGXZn5syZuO+++0BE2Lx5s3he7jAogyEQqHtIlvLR6XRuHcvPP/+83wIFmvLycuTk5ODQoUPQarU4dOgQcnJyUF5eDuD6zGYl/D2tra04duxY0EZeioqKAPDWjxD8/sKFC0Gpm9F3Ee6h7tM4/EXWULvNZsPChQuhVqvFKev0TWwfX+b5hIKRI0e6RFxUq9Uun5W0fC5fvgyO43D58mW/y5LCzJkzsXjxYrzxxhvYsmULnnnmGTQ1NQHgw53IHZVg3Jh0dXXhwoULaGpqQnx8PPr3lz0zxy2ySrNYLOLcHmfkruANF1pbW8XJVDNnzgyxNPL49a9/jTfeeAMvv/wyfvaznyE+Pl5UQAyGHOLj48VZ5UoiS/msX7/e7XqsoAYiCgBHjhxBV1cXUlJScPPNN4daHFnMnDkT999/P1599VU8/vjjOHz4MBISEnD16lXJZXR2duKzzz5DSkqK4vOFWltbsWXLFhQUFGDs2LE4c+YMSkpKsGHDBt8iMnogkLIHknCVe+DAgYpbPCI+Rf8JAcEMJvbEE08QAFq9erUi5QUymFhv2Gw2GjJkCAGgF154wef8HR0d9PHHH1NHR4fisuXk5BAASk5OJrPZTMnJyQSAcnJyiIioubmZcnJyxCBYNpuNcnJyqLm5OeSyB5JIlZtIfjAx5gRw4uDBgwCu74nuL0OHDsW3vvUtv4fsfUWtVovO5/Xr14fVUoNt27aJI4zp6eniyKMwyXPVqlWoqqrC3LlzYbFYMHfuXFRVVfm16wMjPGHK5xvOnz8vOpuVUj4qlQpTpkwJaowUgSeeeAKTJk3CV199JSqicECY5OmM8yRPb8qJ0XdgyucbDh8+DCLC5MmTMXbsWEXKbG9vR3Nzs2J7tfvCwIED8Ze//AUA8Je//AVHjhwJugzu8DbJ05tyYvQdmPL5BqHLNW/ePMXKPH/+PD744AOcP39esTJ9ISMjA7m5uejq6sIPf/jDsOh+eZvk6U05MfoOiiqfV155RcnigsqhQ4cAKNflChe2b9+Ob33rW7DZbHjsscdA8sI3KYa3SZ7elJMUWltb8cADD4gKi+M4LFmyBC0tLYG4JIZMJI2hSfnh6ZtJhnICyIea5uZmMYxGX1M+MTExeOmll5Ceng6j0YjMzEw88sgjIZPH2yRPQQlt27YNarUahw4dwrp168TzUvjtb3+LgwcP9gibAqDPbekdyUhSPp5CaHTHWyjUcOXw4cMA+IWxo0ePDrE0ynPHHXfgN7/5DTZs2ICf/OQnmD59ethOovSmnKTw1FNP4fTp072GTWGEHknKx1MIje5EqlMwEP4egA++NHjw4LCYNPaLX/wChw8fxjvvvIN7770XH330kWKO9XBj7Nix2LVrF+bMmSOeY07r8EOSz8eb4mlra8OJEydkhdQINUSEd999F4DyXa7Ro0fje9/7XlhYU/369YPRaMSkSZPQ2NiIxYsXB23NWbA5c+YMcnNzXc4xp3X4oYjDOSYmBmq1OiJWtXenvr4eJ0+exMCBA7FgwYJQixNQ4uLisG/fPgwfPhz/+te/8KMf/ajP7XIBAH/4wx/8dlozAo+sRRsXLlzA1q1bxS10BCLRrBVG6BYuXKj4DhNfffUV3nvvPdxyyy1hs1ZswoQJqKqqwsKFC/Hqq6/i0UcfRUVFRZ9a8f6rX/0KcXFxKC0tle20ZgQeWcpn69atSE1NFffc0mg0cDgcEdntevXVVwEgIKN0nZ2duHLlSthZF3PnzkVlZSUefPBBvPjiixg8eDB27Ngha9PHcCQuLg6vvPKK6Gtz57RuaWnBqlWrxFE1juNEBSXsYsIILLJed6mpqXjwwQeRmZmJqKgopKWlYdGiRUHfX8tfTpw4AavVigEDBuDee+8NtThBJScnB7t370ZUVBTKy8vx4x//OOyUZCBha8hCjyzlo1ar0dTUhJiYGHF+DND7VrLhiGD1LFiwQJFwDpHGD37wA/ztb39DVFQUysrKoNfrceXKlVCLFRTYGrLQI7ujn5GRgZqaGmRmZuL222/HihUrYDablZQt4Aj+nvvvvz/EkoSO3NxcVFZWYuDAgdi7dy90Ol1YLMMINGwNWRigRDwPq9VKBoMhIDF3AhXP59NPPyUA1L9/f2ppaVG0bIGLFy+SyWTyOc5JKPjHP/5BMTExBIAmTZpEr7zySkTGlpEaF8dms4mxhIQjOTlZjCMUbFg8H5nY7XbodDpxU8FIQHBAZmRkYMSIEQGpY9CgQYiPj8egQYMCUr6SzJs3D4cPH0ZiYiJOnjyJRx55BK+//nqoxQoYSqwhY/iHLOXz6KOPinN6VqxYgfr6elRWVkbMwtKuri688MILAAIzyiXgcDhw4sQJyTu5hprbbrsN//73vzFnzhxcunQJ999/P5566qmQhAQJNN4WuAaTlpYWPPDAAzhz5gyAG2ghrBwzy2QyERHf3VqwYEGP80oSiG7XgQMHCADFxMQENDxrqMKo+svly5fpBz/4gdgdmT59OtXW1oZaLElEYvdFCC2bkJBAhw8f7hFaNtwJardLiMxnsVhcHHSRMk9ECLL1yCOPRFRXMVgMGDAATz31FF577TWMGjUKx48fx6xZs/Dss8/i2rVroRavzyGMvDU1NWHOnDkBG3lraWnBkiVLwibUiCzlY7VaUVNTA6PRKG5GHyldrtOnT2Pfvn0AgJ/85Cchlia8+f73v4/a2lpkZWWhvb0dTz/9NL797W/jgw8+CLVofQq1Wo1du3a5nAvEyFu4zW2SpXxWrlwJjuOwadMmLFy4EDt37sTp06fDYuGeN+2+Y8cOdHV1Yf78+ZgyZUooRY0IxowZg3379mH37t2Ij49HbW0ttFotcnNz8eWXX4ZavJCilCXBcVyvC2GVqifs5jYp2fcrKSlRsjgi8t3n09vWLJcvX6b4+HgCQHv37lVc1u40NzfT888/L3nbl3DBk9/k3LlzlJubK/qChgwZQr/5zW+Csq2RVILp8/G2DZCv5Xjy+ShVDxGR2Wx2mV5gNpt9LqM7cn0+spTPggULehy33347TZ48WU5xveKr8ult/saLL75IACgxMZGuXbumuKzdiUTnJ5F3uT/88ENKS0sT2zc+Pp6Ki4vDYj5TMNtcqblCzc3NlJ2dTfv27aOOjo4ee5UpVU+g5jYFVfksX76crFYr2Ww28TCZTGEz2uVOu3d0dNCUKVMIAG3ZskVxOd1x+fJlOnjwIF2+fDko9SmFlAe4q6uLKisraeLEiWI7jx49mp555hn6+uuvgydsN4Kt8JWyJLzJrUQ9SlpQzgRV+XjSlBaLRU5xvaKU5bNt2zYCQMOHD6fW1lbF5XRHpA61+/IAX7t2jSoqKmj8+PFie8fExNBTTz1FDQ0NQZDWlUi0fIh6l1tJC8uf3WA9EVTl44kDBw4oWRwRKefzGTp0KAGgZ599VnEZPXEjKB+Bq1ev0u7du0mj0YgPSFRUFN1777104MAB6uzsDKDE14lEnw9R73IHymJRCrnKR1Y8H3dT0Nva2hATE4OFCxfKKVIx3O1+sHjxYnzyyScYPXo01qxZE1L5+ioDBgzAD3/4Qzz00EOorq7G9u3b8c477+DNN9/Em2++iXHjxmHFihV45JFHkJSUFGpxFUGJnTbCqZ6gI0fTZWdniz4e4QhEl4vI/xnOV65cIbVaTQDoj3/8o7LCeeFGsnzcceLECXr88ccpLi7Opctw1113kcFgCMiCXiUtn0B1U9wRqYMTREHudgVK0bjDX+Xzhz/8QRzGDLbjt7GxkTZt2kSNjY1BrddflH4QLl26RP/7v/9L8+bNo6ioKFEJ9e/fn3Q6HT3//PN07tw5RepSUvZgdneY8vGj8vr6ep/y2Gw22rp1K5WXl9OaNWuourrabTp/lI/NZhN9PTt37vQ5v79E6g0VSLltNhv97ne/o2nTprlYQ9HR0XTnnXdScXEx1dfXU1dXl6zylZRdCUevVOspUu8VojBwOLe1tfn0gG/dulX8326306xZs9wOH/qjfBYvXkwAKD09PWgOT2ci9YYKltyffvopbd68mW677TaXBxwAjRs3jh577DHau3cvnT9/XnKZSsvu7xC3VOspUu8VoiArn7a2NiosLKScnByXIz8/X1J+m81GKSkpLm+QNWvW0Jo1a3qklat8XnvtNdG0r6ur8ymvUpw5c4aee+45OnPmTEjql0soHoTTp0/Tn/70J1q4cCENGjSoh1U0e/ZsWr9+Pb399tu9TpUIN8tHahmRonzcWXI//vGPg6d8CgsLyWg0ktFoJIPBQBaLhUwmk08/irPlQ8Q7sZVSPg6HgxISEggAPf3005LzKc2N7nCWy4ULF+itt96iNWvW0OTJk3tYRdHR0TRjxgz66U9/Si+99BI1NDSI3bRw9PlIsZ5C3eZScdcmkyZNCp7yMRqNRMQ/5M5dLbnzfARLyJ2FIiif1tZW6ujo8Hpcu3aNHn74YbGB2traJOULxHH69GnauHEjnT59OmQyyDna29vp448/pvb29pDL0tHRQV988QW98MIL9Oijj/awIoRj9OjRdPfdd1NhYSFt27aNbDab3/WePXuWsrOz6dSpU9TR0UGnTp2i7OxsOnv2rOQyTp065dbyEcoMpzaXcr3urmf+/PmylE8UEZGvw/M1NTVISkpCQkIC8vPzUVpaCoAPqyEnMmBOTg5WrVoFnU7X47tLly75tCXPm2++iU2bNqFfv37YsWMHZs6c6bM8SmG323HkyBHceeedim9IeCPT3NyMo0eP4ujRo6itrcXJkyfR0dHRI118fDwmT56MiRMnIiUlBRMnToRarRb38woG69evx8GDB5GQkIBNmzahqKgITU1NmDdvHoqLi4MmhxSkynrs2DGsWLFC/Pz3v/8dKSkpmDJlCoYMGSK5PtnKZ/ny5aioqIDdbkdRURGmTZuGmJgYn5fnFxcXIysrCxqNxu33gvKZMGEChg0b1mtZVqsV3/nOd3D58mVs3rwZv/zlL32SRWmamppQUVGB5cuXIyEhIaSy+EJnZydqa2sxbdq0oD6ocrly5Qo++eQTfPzxx/j4449hNptx+vRpuLu1Bw8ejClTpkCj0WDKlCmYOnUqpkyZgvHjx6N/f1lzbnulpaUFjz32GJ577jlxc8Inn3wSO3bscNmcMBzanOM4ZGRkoKGhQTyXnJyM999/X4wt5C7N/PnzsXXrVp+VjyKjXXV1dbJ2r6isrHTxE7kbbpfq82lra6OpU6cSAFqwYEFIRrfcyfTmm29SW1tbqEXxiY6OyPA/uEOQvbW1lcxmM/35z3+mvLw8uv322+mmm25y22UDQAMGDKCpU6dSdnY2/fznP6fnn3+eDh8+TF9++aXsYX85coe6zb35p0Lu81ECs9ksKh+bzUZ1dXVUXl7eI52z8vE0Z+LMmTN09913EwAaO3Zs2IwuhcsN5SuRKjdR77J3dHTQ559/Tq+99hpt3ryZHnroIZoxYwYNHjzYo1IC+LhFqamptHjxYnriiSdo+/bt9NZbb1Ftba1isYz8bXMlZmNLGZkL+mhXYWEhTZ48mW6//XbKz893eZMbjUbas2ePTyM6drudUlJSehyVlZU90jorH0+jD7feeisBoMGDBwd19rU3Wltbac+ePUFbRa8UfVX5eKKzs5NOnTpF1dXVVFpaSj/96U9pwYIFdOutt1J0dHSvignfREqYOXMm3XvvvfT444/T73//e/r73/9Ohw8fpoaGBrpy5UpA5HZGiZE5uWXInecjyefT1taGsrIyFBQUeExz4MABaDQaJCYmSu3xSULw+aSkpKC1tRVz58516W+OGDEC58+fR1RUFF599VXk5OQoWr8/NDY24vnnn8eKFSsUb5dA0tnZiaNHj+K2226LCJ+PM0rLfvXqVXzxxRdoaGgQj1OnTomH1N1dR40ahYSEBNxyyy245ZZbcPPNN7sco0aNwpkzZ3DHHXfIkpvjuB7PRnJyMg4dOiQ5FnRLSwtWrVolLmDlOE5cwOrsn+pOW1sbPvvsM599PpI8bCaTqVfFAwCLFi2SPdolFWGL2/T0dPHc+fPnAQDPPfdcWCkeRt9g4MCBSElJQUpKitvvHQ4HbDabGMPcZrPBZrOhsbFRPNrb23Hu3DmcO3cOR48e7bW+2NhYjBkzBmPGjMHo0aMxevRojBkzBqNGjXI54uPjMXLkSNFJ7u7Z8DUI/ciRI8XNNIUynT8rjSTlI8E48imdXDiOw7Jly3qcX716NdtpkhESVCoVUlNTkZqa6vZ7IkJLSwuamprQ1NSE//73v+Jf4Thz5gzOnDmDq1evwm63w26347PPPpNUf1xcHOLj4xETE4OTJ0+6fLd48WJs3LgREydOxIgRIzB8+HAMHz4csbGxYWHRSlI+bW1tkgqTmk4uwha3I0eOdInc/9VXXwW0XgZDLlFRUYiPj0d8fDxmzJjhMV1HRweOHDmC0aNHo7m5GWfPnsW5c+dw9uxZfPXVVzh37pz499y5c6LF39ra6rHr19zcjMcff9ytTCqVCsOHD0dcXJz4NzY2FnFxceL/zodKpXL5O3jwYL/36ZOkfE6fPo0LFy70Os+mra0Np0+f9ksYb5SXl4OIcPjwYQDA448/jqamJhgMhoDWK5dBgwZhzJgxEbFXOyO0CAph8uTJkqySjo4OnD9/Hi0tLWhubsYXX3yBbdu2YdGiRejs7ITNZsPhw4cxfvx4XLhwAefPn8fXX3+NixcvgohEC0su/fv3h0qlQkxMDKZNm4aNGzf6XoaUREuXLhVnMrtTQBcuXMC6deu8+oX8ZeTIkaiqqsK+ffvQ1taGpUuXhvUuqSNGjMDs2bMxYsSIUIvC6GP0799f9AkBwJ133unWJdGdq1evorW1FV9//bXLX+Gw2+0u/zscDlFRORwOOBwOEJGo/M6fP4/BgwfLuwYpiaZOnYqFCxdi1qxZyMzMRGpqKlQqFRwOB2pra3HgwAEUFBQEbRO+e+65Jyj1+EtnZyfa29vR2dkZFn1sBmPgwIEuSstXurq6cPHiRdjtdrS1tYmHHCTPJ9fr9UhNTUVhYSGqq6vF81OnTsXevXsxdepUWQL0Zb766iu8++67SEpKiqihdgbDE9HR0YiJiUFMTIx4Thhq9xWfFrNoNBpUVVWJFToLwGAwGL4ga692AEzxMBgMv5CtfBgMBsMfmPJhMBghgSmfADJmzBgsWrQIY8aMCbUoDEbYwZRPAImOjsaAAQMQHc2amcHoDnsqAsj58+fx4YcfilPhGQzGdZjyCSDCaub29vZQi8IIZ1paEP3AAxhw5gz/meOAJUsAYf1iSwv/mePcf+9DPX6X46aM6J//3Dc5voEpH0Z4IeUBCdBD5HM9SimFVasQ9dprmLR6NWCxAHPnAlVVwKpV4veoquLPu/teqizeypFZRtQ//uHb9Qr4FHosBPi7V3so6VP7djU3E+XkEAkhNW02/rNzmE4pabyRk0MEECUnE5nN/F+APy8hTUdHB33y3nvUlZ3duxx+1iO5DCnYbNQl5BWO5GRX+Xv7Xqos3srxVkZXF9GpU0S33upSxoW5cyMrhrNUmPIJPm6VjxIPK5F3BSXlQeslTUdHB52fN8//B1FKGillnD1LtHgxUV0d0VdfEf3rX0SLFhHV1BAdP070738T1dRQx/btruVs2UL0978T7dpFZDAQFRS4fr9iBdGGDUQ//znRk08S5eYSqVSuaYYOJdLpiO65h/+bkUE0c6ZrmgkTiKZMIZo4kUitJurXz/X7qCiiQYOI+vd3Pe90XIy0APJSiWTlY7fb6dVXXyW73S6/ECWsCR+tlo6ODjq2bx9vPSikFESkKCiz2bUMd/uje0gjyt7t7UyJiUQHDhB98AHRwYNE1dVEzz7rmmbtWqLf/Y5o40aiX/yC6IkniLKzXdOkpxMtWEA0Zw7RHXfwD63z98OHE8XGEt10U88HuY8eTPmEIYoEYvf2sEpRLD5aLR2HD9OVhARFlYKIOwWVmEi0bx/RoUNEFRVEo0e7fh8XR5SXR/TYY0Q/+hFRVhb/cDun6d+faPRo6oqNpc4BA0L+QPZ6REXxVsro0UQJCbyFAlBn//7UlZLCWxoA0ahRvMUydiz/eehQ3mqKibn+ez7xBNHPfkb0+ON8OznXM3Ik0W9/y1tOu3YRzZ7Nnx8zhmjr1uvl3nkn0ZEjRK+/zsvjXIZazf+GjY1EZ84Q3X03f378eKJ//IPo1luZ8glH2tra6PXXX/dv3y5/++lSyvCQpstbFyMhgejll4mqqvibe+NG/s3vnOamm4huv51oxgw+/6hR1x+uYD/wo0bxD83kybx1AhANHkz0ne8QDRnCf05K4pVcfj5RSgp/bvhwojVr+IcZIJo1i8ho5PMBRDffTLRjx/UHd8EC3jfS1ER07hzRO+/0rpCbm6krO5uO7dvHv6i6v0CUesF4K0dmGRcCuXVOKIlk5ePV5yO1S9WbNSFFsbgr45//5P0RViv//969vP/AWfnceSfR3LlE06f3tDSUPKKj+Qd40iS+rvh4ovnziR54gD+Sk/k3+8aN/Bt761aib3+b6M9/JnrzTd43ctddRAcPUofVSl+npV1/Ox8+LO9BlJJGShkSfx+/reRgdc/dIHfrHKZ8AohX5aOU1dJdsZSV8dbIX/9K9OtfEy1bJpr2ih1DhvBWw6RJRGlpfPdg8WL+gV+1imjTJqKiIqL/+R+iv/2NyGTi5czI4POPG8f7XgTfjK8jRB6QPNoVLCSOiEXyXmlM+YQbzc3U9NBD15WP3LeicPOOG8d3bcaM4T9PmkT0yCN8f90fH8eIEbzTdMQI/vOwYdT54IN0TRg5mT2bVxwffkj02Wf8iM3Vq361i99v6F4Iu4dY4vWGndw+wJSPkihhkufk0Jc338wrn7ff9jwHpLvVsmUL3634yU94x+qkSb0Oc7r1bQhKZfFiotWriZ56ive5/PWvvLXx7rt82c7bSnsb7YoQIvUhjlS5ieQrH58iGd4wCLM4jx4Fdu8Gli0DhJ0ghU3UvKXZtg39770XiRyHwX/6Ez9LdPx4Pt+ePcDJk3zet992rXvDBs9yjRoFqNX8kZDAH7GxQGUlsHEj8D//A7S1AU88AZSXA73sMonvfc/188iR16+tsxPXxo5F1yuvsNjTjMARIGWoGIpbPko5CXtLc+UK0SefUOeGDe6tkt4crwA/9Pz000Tl5fxIycmTRJcuKXP9Eojkt3Ckyh6pchMxy0c6UqwatZr/zmnrWezezZ8XcJdmwgQgKwv49FOgo6Pnwjki3lKZMgWYNAlITATefZe3WtLTAbsdWLfOu9XCYPQBbjzls20br3gaGq4rjuRk/rwAx/FKyZlly4ADB4DmZuCDD4BDh3p2md555/r/AwagPSoKx2fMwIRZszD89deB//4XmD+fV34CW7Zc/1+luq4AGYw+Tt9b1e5tZa5gsTjT3apZt45XTuPGAc88w1srDQ28tZKWxvtU3ngD6Ojg099yCzB0KP9/WhpgswFffomW++/H/rvvxsWnnwY+/BDIyQHCdHdVBiPY9D3l4y1sgCerhuOAs2d5521MDH+cPg08/TTfHQKAri7e6fv97/Pn09OB48eBpibgxAleubz1Fq/I4uPR9fvfX69DreatGtadYjAA9MVul7dulWDVJCcDO3Zc9/loNPxIUXdSU4HvfpcvS6sFbr0VcLdFs6BcGAyGJPqe8unNWUwErFnDD3PHxPDOYaHrJCie224D5s0D7rqLVzrMUmEwAkJkKZ+WFr77tG0br0w4rufokLtuVU4Or1D++U/e6etMcjKwYAE/72XuXCA+XjFxR40ahXnz5mHUqFGKlclg9BUiS/lIGSYXulUjRwITJ/IjU4IvBwCGDOFHnHQ6YNEifng8QPTv3x9Dhw5F//6R1cwMRjCIrKfCkz/nj38EamuB118HPv+cP9/Scn2Ea+hQ4OGH+VGvOXOAwYODIu7XX3+NTz75BOPGjUO8ghYVg9EXCJnycTgcMBqNUKvV4DgOarUaOp2u90zu/Dm33853qQQLCOAdwlotcO+9/MjU5MnuncQB5sqVK2hqasKVK1eCXjeDEe6ETPnk5+ejoKAAGo0GALB27Vqo1Wrxs1vc+XOE7tSgQcDChcDixcA99wBsl1AGI6wJyTwfh8OBuro6F0Uzbdo0GI3G3jMK/hxhsaOwE+gdd/Azj998E1ixgikeBiMCCInlY7FYepxTqVRuz3d1dQEALl26BJSWInrMGHQ9+CD/5ZgxiP7Tn9D19NP8MLq7eTohpL29HSqVCu3t7WgLM9l6Q2jzCxcuRNxWz5Eqe6TKDXzzbOL6NUglJMrHbrcjNjbW5VxsbCzswkxiJ4TdPhsbG/kTK1Zc//LSJf7z2bP8EYbMmTMHra2taG1tDbUoPvO54LyPQCJV9kiVG+Cf1WHDhklOH/ajXbGxsRg/fjwGDRoUcW8EBuNGoKurC+3t7T0MCm+ERPm4s3LcWUMAP1dmJJtlzGCENb5YPAIhMSW0Wi0cDofLOYfDAa1WGwpxGAxGCAiJ8lGpVNBqtbBareI5i8UCvV4fCnEYDEYIiCIiCkXFDocDZWVlmDZtmvRJhgwGo88QMuUjBVmzoMMAjuNgMBig0WgizprjOA6VlZWIi4tDbW0tsrKyIqbNTSYTVCoVzGYz0tPTI67tAcBqtcJisSAvLy/UonjFYDDAYrGI94fJZEJFRYXk/GE92iVrFnSIsVgscDgc4DgurOX0RGVlJdavXw+AV/4ZGRliNzmcKSoqwqZNm6BWq5GZmYnZs2cjNTU14n6DwsJCpKamhloMydjtdhQVFUGr1WLTpk0+5Q3bsWvZs6BDjFarhU6nQ0xMTKhF8RmO47Bz505w34SgValUSEtLQ6WwhCWMsdvt4iRVlUoFAKirqwulSD5jNBqRlpYWajF8oqqqCidPnkRFRQXUzqGIJRC2lo8vs6AZyqBWq7Fy5UqXm6ixsRGJiYkhlEoaVU5B+a1WK1QqFTIzM0MokW9YrVakpqbC4XBE1Gx4gH9W1Wq1z8onbC0fX2ZBM5RD6HIBvCVktVqxevXqEEokHcFHWFZWhvfff1+0gCKB7lZ+pGAymZCamgqO47B27Vqf8oat8mGEnvz8fJSWlkbMQ6FSqaDX65GVlYX8/Pwec8nCFaPRGJHO8by8POh0OtEn2NjY6JNbJGyVjy+zoBnKU1xcjM2bN0fESFd3dDod6urqsKG3rafDBKvVGvbOfE+YTCaXz4mJiTCbzZLzh63yYbOgQ4fRaMTSpUtFi6f7TRZuWK1WzJ49W3SUA0Bqaur1xchhjN1uh8lkgsFggMFgQHV1Nerq6mAI8/3dOI5za13GxcVJLiNsHc7Os6CFh8BisaCgoCDEkkkj0pyGAs4OfY7jxGkD4YxKpYJarXaxiuvq6rBK2KstjNFqtS4vVKGtw32ej1qtRkFBgYtfraamBrt27ZJcRthPMoy0WdAWiwVWqxXl5eXinBOdTufzSEAocDgcmD17do/zmzZtCnufhMlkAsdxUKlUsFqtUKvVYf8Ad8dgMIg+E71eH/byO0/stFqt0Ol0PvVMwlr5MBiMvkvY+nwYDEbfhikfBoMREpjyYTAYIYEpHwaDERKY8mEwGCGBKR8GgxESmPJhMBghIWxnOAcLYVKgMFPTZrNh9erVEbUimsGIRG5oy8dgMMBkMiEvLw96vR56vR6rV69Gbm6uS3B7Ru8osQ6puLjY55AMveFOJqXrUJJIk1cJbljLR1gC8f7777ucV6lUKCgoQH5+Pt57770QSRc5KLX2Kz09XbEQGJ5kUrIOJYk0eZXihlU+hYWFyMzMdNu90mq14DguYuOsBJOSkhJFylEyWoEnmcI1IkKkyasUN6zysVqtvYbZVKvVMJvN0Ov1MJlMKCkpgUqlQmlpKdRqNYqLi7Fnzx5s3rwZGo0GlZWVmDZtGmpra7F06VKo1WpYLBaUlJQgLS0NSUlJMJlMKCgogN1uR0lJCRITE5Geni7K47wwT1hdLkQTdF5B7KlcjUYjKV9qaqrLjgPC/w6HA7W1tUhKSnJRusKOFu6uj+M42O12GAwGl4W/7vJwHOdWboB/GSQmJmL79u2wWq0oLCyEXq9HbGws9u/fj8bGRjFUqrdrdCeTUKZQh3C9wsJlIVaUTqfrtZ08KQR/fhN/5fWEIJPD4UBBQYGY1jlSREihGxCbzUYpKSlUWVnpMU1ubi5lZ2eLn6urqyk3N9elDLPZTEREGRkZZLfbiYjIbrdTRkaGmK6yslL8XF1dTTabTTzvXJ5QTl1dHRERpaSkuKQtLCx0SeupXG/5qqurKTs7W5TXbDa71Gu32yklJaWHXL1dX/c6esvjSe7q6mpas2aNKJNw3m6306xZs8TPUtvGnUzOdRCRSzsQEW3dupWqq6s9tpPz/eAOub+JEvJ2x2az0Zo1a8T7NDc3V8zrKU+wuSEdzkJ4C5vN5jENx3EugdOF6HhC39xisUCr1cJoNEKlUolvMuF/5z68UF/30Brdw2wsWrQIZWVlACDuBiD0+d3txOCuXCn5YmNjRXmFPdGEN2H3bqiU6+uOtzye2sNZPuF8fn4+Vq1a5ZJOyjV6w2g0urQDAGRlZbl0gbq3kxT/i9zfRAl5nbFYLNi+fTvUajW0Wi1KS0tFKyssrB7cwN0ujUaD+vp6j99zHNfD35OZmSmOjgkICsw52t+qVatcAltJjeUTFxeHmpoaALyJvXbtWkybNs1jenflSsnXfVuf3uSTcn2+5vHWHsLDYTQaYbfbe8S1kXKN3rDZbD3aobtSlbP9kdzfxBtS5HWm+72rUqlgt9vF7n04cMMqn4KCAixfvhwOh8Pj2777Ta/X65Gfnw+NRiP2/ZOSklBTU6PIDypYW0J//6OPPgLAv8Wqq6sBwK28AnLz9YYv1yc46JVoE8E/JPh5hLJ9vUZPgwZJSUk9Xj5CMDIlCSd5HQ5HWAW1uyG7XQA/krBy5coeQcaFrY5LS0t75NFoNFCpVDCZTOKPqNfrewyVmkwmSSZ6d/O7uroa69evh8Vicdm1UijL27C21HzdQ7z2th2Rt+uLjY1Fa2urT3mkIOxW272LLOUa3cnk7rqEMLHOMjqH6VUiFG4w5fVGa2trWI2g3bCWDwDxQS8uLkZSUhIA/iavqqry+EYRRmCcqaqqQklJCTQajeivEDY4NBqNcDgcMBgMPSypxMREsWtisViwa9cuqNVq6PV61NbWiv18jUaDxMREGI1G5OXleSzXWz6r1Qqj0Yi6ujoYjUZotVoUFxeL5ej1etHnVFRUJI7KeLo+gPdt7N+/Xxyp6a1NhHq7yy3IJYTlBK7HMjYajbBaraKi9naNnmTqXodOp0NFRQXKysqQlJQEh8MBjUYjWlfu2onjOBQXF7vsbSYg9zdRQl6p+BLcPRiwMKohQniofN3fmsGQiyfFGSpu2G5XOODN1GYwlMJisYhzysIFpnxCgGCi19fXh/2eWIy+AcdxYeXvAVi3i8FghAhm+TAYjJDAlA+DwQgJTPkwGIyQwJQPg8EICUz5MBiMkMCUD4PBCAlM+TAYjJDAlA+DwQgJ/x94CUiyn0TUTQAAAABJRU5ErkJggg==\n"
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "colors = ['black', 'red']\n",
    "marker_size = 15\n",
    "res_dicts = [res_dict_small, res_dict_large]\n",
    "save_path = f'../fig/multivariate_t_{df}.pdf'\n",
    "steps_finite_samples = 3\n",
    "\n",
    "fig, ax = plt.subplots(figsize=(.5*LINEWIDTH, .3*LINEWIDTH))\n",
    "\n",
    "# Orientation lines\n",
    "# ax.axhline(y=ridge.r_sq, linestyle='--', color='gray', linewidth=1) # base risk of 0 classifier\n",
    "ax.axvline(x=1, linestyle='--', color='gray', linewidth=1) # base risk of 0 classifier) # interpolation threshold\n",
    "\n",
    "# Min-norm interpolator\n",
    "ax.plot(gams, res_dict_small['risk_caus'], color=colors[0], linestyle='-')\n",
    "ax.scatter(gams[::steps_finite_samples], res_dict_small['finite_loss_caus'][::steps_finite_samples], color=colors[0], marker='x', s=marker_size)\n",
    "\n",
    "# Optimal ridge regularization\n",
    "ax.plot(gams, res_oracle['risk_optim_caus'], color=colors[1], linestyle='-')\n",
    "ax.scatter(gams[::steps_finite_samples], res_oracle['finite_loss_optim_caus'][::steps_finite_samples], color=colors[1], marker='x', s=marker_size)\n",
    "\n",
    "# Legend and axes\n",
    "ax.set_xlabel(r'Overparameterization ratio $\\gamma$')\n",
    "ax.set_ylabel(r'Causal risk')\n",
    "ax.set_xlim([0, max(gams)])\n",
    "ax.set_ylim([0, 6])\n",
    "\n",
    "ax.set_title(f'Student-t (unbounded 4-th moment)', fontsize=10)\n",
    "\n",
    "handles = [Line2D([], [], label='Min-norm', color='k', linestyle='-'),\n",
    "           Line2D([], [], label='Optimal ridge', color='red', linestyle='-')]\n",
    "\n",
    "plt.legend(handles=handles, framealpha=1, frameon=True, ncol=1)\n",
    "\n",
    "if save_path is not None:\n",
    "    plt.savefig(fname=save_path, bbox_inches='tight', dpi=300)\n",
    "\n",
    "plt.show()"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  },
  {
   "cell_type": "markdown",
   "source": [
    "## Student-t distribution (bounded 4-th moment, df=10)"
   ],
   "metadata": {
    "collapsed": false
   }
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "outputs": [],
   "source": [
    "df = 10\n",
    "def sample_multivariate_t(n_samples):\n",
    "    return multivariate_t.rvs(loc=np.zeros(ridge.l), shape=(df - 2) / df  * np.eye(ridge.l), df=df, size=n_samples)\n",
    "ridge.sample_z = sample_multivariate_t"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "outputs": [],
   "source": [
    "gams = np.linspace(0.1, 5, 80)\n",
    "lam_small, lam_large = 0, .5\n",
    "res_dict_small = ridge.compute_risks(lam=lam_small, gams=gams, include_finite=True)\n",
    "res_dict_large = ridge.compute_risks(lam=lam_large, gams=gams, include_finite=True)\n",
    "res_oracle = ridge.compute_oracle(gams, include_finite=True)"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "outputs": [
    {
     "data": {
      "text/plain": "<Figure size 300.058x180.035 with 1 Axes>",
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAR8AAADWCAYAAADox4aFAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjUuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8qNh9FAAAACXBIWXMAAA9hAAAPYQGoP6dpAAA/HUlEQVR4nO2df3wU1dX/P+E3gWwCBESSDRBpEDagVFGzURQjkPC0SkJ1aSs1IAF/QbCG1lqTWrD1kaQWaJ9qsviEr5SaRUnVUrJYrDG4G1toRZMJ4stG2Ul4BAJmNxCIhJzvH+MMu9nd7Mxk9hfc9+u1r2Rn771z5teZc88999wYIiIwGAxGiBkQbgEYDMaVCVM+DAYjLDDlw2AwwgJTPgwGIyww5cNgMMICUz4MBiMsMOXDYDDCAlM+DAYjLDDlw2AwwgJTPgwGIyww5fMNHMfBbDbDYrHAYrHAbrfDYrFo1r7VasXUqVPBcZxmbWqJ3W73kM1qtUrnwWq1hlGyS5SWlsLlcimqY7fbcdddd2l6LaOBaDhepny+obi4GAUFBTCZTDCZTOA4Dg6HQ/rd5XL164JmZ2fDaDT2W065ciiRl+d5cBwHg8EAQHhg7XY7TCYTjEYjCgsLFT30a9asgdlsViWLPziOw9atWwOW670vo9GIBQsW9GvfkY6v82s0GiNeATHlA+Hh601BQQESEhL6LBMO5MqhRN7S0lIUFBRI3202m4ei3LdvH3Q6nez2Vq1ahezsbFWy+IPneej1elnleuN+HS9HfB2zXq+Hw+FQbCmGEqZ8IFwonue93hQmk0n6v7y8PNRi+USuHHLLuVyugA91fHy8rLZEDAaDR5v9PXdWq9VDmfVFpFynUOLvmBcuXIiampoQSyOfQeEWIFIoKipCSUkJysrKkJGRgSVLlkhvf7vdjpaWFthsNgCCScvzPMrKypCTk4OCggJYrVYUFxdj8+bNUj2O47Bnzx7MmDEDAOB0Oj32KfpZ9Ho9GhoasG7dOtjtdqldg8EAnudhs9mwZcsWn3L4UhxyywFATU2NJJ9Yt6mpCTzPw+l0wuVyoaKiAtu2bYPBYOhTPvGYi4uLpe6rP1l8HbsvXC6XbKurr+N2uVyw2+1e8vprx9cxrlu3zqOd9evXS3U4joPdbpdeZNnZ2dJxKm1LyX2xZMkSv8dsMBhQVlbm8RKNKIgh4XA4qKqqilavXk1paWlUU1Mj/bZx40aqqqryKF9RUUEVFRXS99WrV5PNZiMiIqfTSVlZWR7ls7KyqLGxUdpXbm6u9FtVVZXUVkVFBa1evVr6LT8/X6rnSw5fKCkntu2+zf3Y3fcfSD7xd/d995alr2PvjXu9rKwscjqdAY/H13XqS15f+KrjLqO7LA6Hg/Lz8z3q5+bmSr8rbUvpfdHXtXZvK9Jglg8uvV31er30xrZaraioqJBt7vempqYG06dP99jmbn1UVVUhPj4edrtd2tbQ0CD9726NxMXFeVlNWtHR0aHInyPSH/kCHbuI3W5HTk6OYtl8oUbe3nXcr59Op4PT6YROp0NVVZXXtU5OTkZNTY1kdShpKxLui1DAlA8gdTHcnaxGoxEVFRU+yyvxQfTF9OnTPfap1DwWh8PFGzA5Odlnd6IveePi4kLqlBSH7eUeu7vPgud5lJeXY+HChSgrK+vXcatBjZJW01Z/7gutjzmYMOXzDWVlZaiurpa+937rJiQkSA+pv7dNU1OTVMZoNHrFx7iPSixcuBDFxcUev9vt9oDD8b3lqKyslFXOHykpKeB5XhpmF+nvG9VdofWWRe6x9/5eUlKCJUuWQK/X9/u41eBPSfs6nqamJjz77LOatBXovujrmJUOFoQSpny+wWQywWw2S28kl8vlMfycnZ2N0tJSWCwW6UYwmUwoKyuTTOTp06fDYrFIoz1FRUWwWCySia3T6WCxWFBUVASDwYCioiKUlpZKprTRaATHcdLbXnRsNzU1oaqqCnq93qccvpBbLicnB+Xl5dLb0m63o76+Hk1NTUhPTwfP82hsbJTk5nm+T/lcLhdqamoQHx8vOV17yyKem97H7g/3OBaz2QyTyeSlLP0dd6Dz6csR768OAMnpzvM8zGazx7U0m82Sk3jz5s3Q6XSq21JyX/zwhz/Ejh07vK41x3ERbQXFELEE8lc6a9as6XP0hxGdlJSUoKioSNPuopawOB8G1q1bF/HRsAxliF3pSFU8AFM+DEDqfkTqvDOGcsTpMZFMWLtdPM+jqqpK6ttGcv+UwWBoS9gczjzPo7CwUBphWrZsGZxOZ8RrawaDoQ1hs3yWLVuGgoICj6kIer0+ovuoDAZDO8KifFwuF2bPno0DBw5IsS/+hk67u7vhdDoxdOhQDBjAXFQMRqTR09ODrq4uxMfHY9Ag+Z2psHS7GhsbAVwKnuJ5HsuWLfMZOOZ0OvHFF1+EWEIGg6GUSZMmYcyYMbLLh0X5iNGYOp0OOp1OCsorLS31mt08dOhQAEIIfWxsbMhldWfhwoU4efIkXnnlFUybNi1g+a+//hqHDx/GtGnTMGTIENn7ISLceeedOHPmDHbu3InJkyf3R2zF9PT04LPPPsOUKVOiztqMVtmjVW4A6OzsREtLi/SsyiUsykeMKnWPxtTr9T5zj4gXIjY2FnFxcaER0AdEhH/84x+4cOECxo8fL0uWlpYWvPPOO4rfCABw4cIFHDlyBK2trZg5c6ZasVVx8eJFAMDIkSMxcODAkO67v0Sr7NEqtztKlWZYVKx7jhURnucjeh6Ky+XChQsXAABjx44N+v4mTZoEAKzLybhsCYvy0el0UqIpkcbGxogeZj958iQAYMSIERg+fHjQ9yd2tZjyYVyuhC3OZ/369SgtLQXP82hvb4fJZIroIENR+YTC6gGY5cO4/AnrrHZ/qTMjkVArn9TUVADAp59+GpL9MRihhqXUkIka5XP11VfjO9/5Dq6++mrF+xOnnDQ1NeHChQsYPHiw4jZEuru78fXXX8suLzo/Ozs7o875Ga2yR6rcQ4YMURS7owSmfGQSastn4sSJiIuLQ0dHBz799FO/QZh9QURwOBxoa2tTJUM0W13RKnskyp2YmIiUlBTExMRo2i5TPjJRo3xOnTqF999/H3q9HuPGjVO0vwEDBmDGjBmw2+346KOPVCkfUfEkJSVh5MiRURc/wggvPT09OHPmDFpbWwEIL0QtYcpHJqL1oET5fP3112hvb1fU5XHnuuuug91ux8cff4wf/OAHiup2d3dLimf8+PGq9s9gjBw5EgDQ2tqKpKQkTbtg7FUok1B3uwBIwYUff/yx4rqiwhNvHgZDLeI9pPYl6g+mfGQSTuXz0UcfqW6DdbUY/SVY9xC7M2USDuUjjngdO3ZMtdOYwYhUmPKRiRrlk5CQgOuvvx4JCQmq9hkXFyfF+6jpel0OcByH0tJSTJ06FWvWrPFbLi8vD7Nnz4bZbA7pOmQM9TCHsww6OzvR2dkJQBh2lMvw4cORnJzcr+kY1113HZqbm/Hxxx/jzjvvVN1OtGIwGGAwGNDR0QGLxeJz7XaO45CcnIzk5GSP5Y7ECHq2MkdkwiwfGYhWz5AhQxTNrD979iy++OILnD17VvW+p0yZAgDSPDie57F48WKcOnVKdZvRiKiEfGU+cDqdPtffyszMxMKFC0MhHkMFzPKRwbFjxwAA48ePVxRo5XK50NjYiJtvvll1eti6ujoAwJtvvgm73Y6lS5eiubkZALBr1y7F7RGRZMWFmtjY2H4FqplMJlgsFtkTkAOt/soIL0z5yODzzz8HgJAn9QKErsOcOXPw9ddfIzMzE4Aw72vTpk2K2yIi3HrrrR7ZBEJJZmYm9u/fr1oB5eTkoKSkBDzPS5aOmA3TZrN5lOU4DsXFxdI67na7HWVlZUhPT5cmMIvrmvtTUnLquFwulJeXY8aMGXA6ndJKre71MzIykJKSAqvViqKiIjidTr/tim02NDQgJSUlojM99BfW7ZJBOJVPZmaml89o+/btPrsZctA6RD6U6HQ6GI1GWK1WaZs/57LBYMDKlSul70ajEStXrkRjYyPS09NhNBqRnZ2NsrIyv/uTUyc/Px+rVq1CdnY2TCYTGhoaJPmMRiNMJhP27t0Lk8kEk8kkHYOvdktKSiTltWrVKpSUlPT3lEU0zPKRQTiVT2trK3p6ejy2LV26FLW1tYoVUExMDPbv3x+13S5A6HqVlZWhoKAALpdL8TmIj4+XusDi2vJq61gsFo/fACHVbmFhoUd6GFHG3iljercrrjIK4IpYxYUpHxmoVT5Dhw7F2LFjFee2dWft2rXo6uoCANx///2w2+1obm7G2rVrVfl8YmJiMGLECNXyhJvs7GwUFhaC4zg4nU7Ffh1/AwbLli2TVlIBgMrKSklp+KvjcDi8ftPpdB7tAPCrIHvXVWvNRitM+chArfIZPXo0br75ZowePVr1visqKtDc3IxDhw7h2LFjqK2txdq1a1FRUaG6zWjHZDJhz549UhCmFvhaOSUQKSkpaGpq8tjG8/wVYbVoAfP5BKC7uxsOhwPApeyCcunp6cGFCxe8uk1KGDNmDHbs2AEAqK+vx1VXXYVdu3YpTkgf7YjXABCsn61bt6qa6d/R0aFZHZPJBJ7nPbpuolNZTbtOp1OxbNEMUz4BaG1txcWLFzFkyBBMmDBBUd3jx49j7969OH78eL9kmDZtGsaOHYtz587hwIED/Wor2uA4DmvWrMHOnTtRWloKQHDkLliwQOqmmM1m7N27F/X19VKEM8dxsFgsaGpqgtVqlb43NjbCYrGA53kpCFFs19e+A9WprKxEeXk5LBYLzGYzDAaDNEJlt9thsVhgt9thNpsDtutyuST5xfZLSkou34htinDOnj1LBw8eJJfLFZb9v/vuuwSAvvWtbymuy/M8PfPMM8TzfL/lWLx4MQGgX/3qV7LKi+ft7Nmz/d4348om0L3kcrlU3WvM8glAOEe63Ln99tsBAO+9915Y5WAwtIIpnwBEmvKx2WzS+mEMRjTDlE8AIkX5pKenY/To0Th79iz+/e9/h1UWBkMLmPIJgKh8lI50AcC4ceMwb948xfmbfTFgwADcdtttAFjXi3F5wJRPAPpj+QwcOBBDhw7VbCmUO+64AwBTPozLA6Z8+qCrq0ua0a5G+Zw+fRoHDhzA6dOnNZFH9Pvs378f3d3dmrTJYIQLpnz64OjRowCE9dmVJBET6erqwvHjx6XpEf1l5syZiI+PR0dHBw4dOqRJmwxGuFClfFpaWnxuf/vtt/slTKTh3uWKhNngAwcOlPw+tbW14RWGwegnqpTPL37xC69tPM/3mZ4gGomUkS535s2bBwD4y1/+EmZJGIz+oUr52Gw2Dytn69atWLx4sWZCRQr9GekKFosWLQIg+H36O20j2hCnKVgsFlgsFmlKQrTAcRzy8vL6TITfn7aXLVuGZcuWad52sFClfCorK0FE2LlzJxYvXozGxka88847qlI8RDL9tXzi4uIwffp0RXmfA5GSkoLZs2eDiPDmm29q1m6kYzabYbVaUVBQICXmWrVqFfLz88FxnKr2elNaWhoUxSDSO8GZUvqSz2AweCTPjwZUKZ+MjAwsWLAA8fHxMBqN2LRpE+Li4lBfX6+1fGGlv8pn5MiRSE1N1XzV0Ly8PABAdXU1Tp06hcWLF0s5ZMQE8+3t7ZruM5xwHIeKigqv2eI6nQ5FRUUoLCxU1J7L5fLKuQNEfsL5QPLFx8eHUJr+Iyufjy8fDyDkBP7ggw+k/99++23Mnz9fO+nCzBdffAFAvfI5d+4cjh07hqlTp2qqgPLy8vCzn/0M77zzDvLz87F7924cOnQI27dvlxLMX3XVVXjwwQe9KxMBYcpkiNhYQIXjvri4GDk5OT7z5BiNRvA8ryixvD/fZKQnnI90+ZQiS/k0NDRgyZIlPn9LT0+X/r+c1gV3Op3SKqFqlU97ezv+/e9/Y9asWZqem7S0NBgMBnAch6ysLDQ1NaG5udkjwXxRURG++uorz4pEwK23AmFKII/MTGD/fsUKiOM45OTk+P1dr9fDZrPBZDJJSduTk5Ol88FxnJT03W63g+d5OJ1OmM1m6PV6ZGdnK0o4D/hP8u6+xBHHcSgqKlKUXMxf0nkAHvKJMpSXlyMlJQXx8fFe1pyYusNgMMDlcknHn52dLaVtraqqwowZM6RnPKTZFOVMfbfb7bKmyMstp4RwpdTYv38/AaDk5GTVbWiZUqM3xcXFBIAWLVpENpuNAEgfm83mOw1CTw9RZiaRoIZC/8nMFGRQgMPhoLS0NKqqqvJbJj8/n3Jzc6XvVVVVlJ+f71EmKyuLGhsbpd+Li4u92qmpqaHVq1d7fM/NzSWn00lERDabzaMdp9NJaWlpHm2kpaWRw+Hwu5/e+/BFVVUVZWVlSeXF9nrXzc3NlWTxddw33nijVLeiosJLlqysLOnYnE6ntM/eBCulhizLJyMjo8/f6+vrodfrA5aLJsQgvuuvvz6scvhj8eLF2LBhA2pqarwCDpcuXYq//e1v3pViYgTLI4q6XeKb2D2TYW94nsf06dN91hNZsGABysvLFa9eqjTJu5j7WRyFa2xsVLQ/EX9J50XsdjtcLpdHNsfex+xyuSQ/UO/c0haLBTqdTjoG8X/3ZYmCjSqH8/Lly/Hyyy8DAB588EFwHIeqqiq89tprqoTgOM7n6EM4+eijjwAIyxVHIjNnzkRqaiq6urrwxRdfIDU1FTabDampqWhubvYfcxUTA4wYEZ6PykBNg8HglSvZHZ7nA+ZzTkhI8Bsc2xdKk7y7XC6sWbNGWtpZLYH2w3FcwO6cuMiiy+WCzWbzcNiLytxqtUqflStXhtRprSqBvMlkwoIFC9DU1ASe5yVFtHfvXlVCFBcXe/iOIgEtLJ/BgwdDp9Nh8ODB2gjlRkxMDEwmE5577jmMGzdOWkpHTDD/9NNP4//+7/803284KCoqwrJly3yu0y6+wQMNM/M8j+TkZK/tShzVgRD9RmKqW7vdLi3v7Ev2/iBn2R+DwSD5eZ599lmP/aekpKC+vt6vZRUKVFk+4kHY7XYPDa1mCoLFYom47trx48elnDnXXXed6vXRExMTMWfOHFXzwuRQUFCAmJgYnDhxAl9//TUA4abctWsXEhISgrLPcGA0GrFixQo8/fTTHtt5nofZbMbmzZu96vTu7tTU1GDdunUAhK6U3FAEJUne7Xa7x0tUVA7+hvb7g6g03NsVHekiHMdJDvXeis9kMnnJZbVaQxq0qcryEYO6LBaLdEFfe+01xSYbx3FIT0+Hy+UKuKpAT08PLl68qEZcxfzwhz9ET08PYmJicOzYMSxfvhzNzc0gIkVdS1HeYMmdkpKC+fPnY+/evXjppZfw3//93177vlxYt24d7HY7SktLkZKSAkDoOlRXV/u0KJKTk6WVQ+12O7Zt2+bhR9mzZ4802gVcGhnieR5WqxV6vd4jybvRaPRI8m4ymVBeXg5ASPJeVFQkrVgqLiZoMBiQnJwMi8WCgoICr8TxviwuMem8uB/RoustX3Z2Nqqrq6WlmgFIifPFenq9HrNnz5ba1uv12LBhg+Qnqq6uRllZGQwGA+Lj46HX6/u0zi5evOjzvlK7OksMEZGaijt37pSczFu3bkV7eztGjRrlO7bED+IFMJvN4Hke69ev9yrT2dmJw4cPqxFRNWLovjtJSUkoLy/H+PHjZbfjdDphs9mQmZkZtL50bW0tioqKkJCQgD179mDIkCEev0+bNg2xsbFB2XekYrFYwHGcz/vpSsFut8Nms0nGASBYRsuWLcO+ffsUtSX3GVR6r6leNPC+++6T/l+xYgUAZbPalfa1p0yZErI4IovF4rXt1VdfVRzk1draiv3792PKlClISkrSSjwP0tPT8dvf/hatra1obm6W4rE6Ozvx6aefBmWf0cDlFOGtBo7jvJzwgSybQKSlpflULmfOnMFnn32muD3Zymft2rVYuHAh5s+fj7Vr1/os09TUJCvCmeM4xQ/ygAEDNMsIGIh//vOfXtvy8/MVr48+YMAA6W+wZB84cCAKCgrwzDPPoKKiAj/84Q+l7Vci7t0W96DAK42CggJpLTGdTgeXy4X29naf/jG5DBw40Od9Jd7nSpGtfJKTk6VhR4fDgVWrVnmVkeuscjqdUiQoAGlEwL2PG07EKSMTJkzAa6+9Jk1ZULs+erBZsWIFNmzYgLq6OjQ2NkbcyGEoMRqNqK6uDrcYEYFWo3jBQrbycY8RePbZZ72CugD5C90bjUYPy0f0uEeC4nHPPPj3v/8dU6dOjfj10ZOSkrBo0SLs2rULzz33nLS8MoMRyaiyl2JiYnwGbPlSSIEwm82w2+1eS8qGCzG4MC0tDVOnTgVwafha6froiYmJuP3224M21O7Oz3/+cwCCb+qTTz4J+v4YjP6iSvmUlpb2GXGqhIKCAuzbtw/79u2LCMtHVD5aTKsYPHgw4uLighJk2JtZs2Zh0aJFICJs2LBB2q52GJTBEAnWPaRK+WRnZ/t0LIuRztGMGNmsxbSK9vZ2fPTRRyEbeSkpKQEgWD9i8vszZ86EZN+MyxfxHuodxtFfVA21OxwOzJ8/H3q9XgpZp29y+yiJ84lEtLR8zp07B57nce7cuX63JYdZs2bhnnvuwZtvvolnn30Wv/71r9Ha2gpASHeidlSCcWXS09ODM2fOoLW1FYmJiRg0SHVkjk9UtWa326XYHnfUzuCNFNrb26VgqlmzZoVZGnX84he/wJtvvolXX30VP/nJT5CYmCgpIAZDDYmJiVJUuZaoUj7r1q3zOR8rpImIgsD+/fvR09ODtLQ0XH311eEWRxWzZs3C9773Pbz++ut47LHHUFdXh6SkJBw4cMAjuXhlZaXfruXFixfx6aefIi0tLerihaJV9kiVe8iQIZpbPCKqWvU3ETTSJogq5d133wUAzJ07N8yS9I8XXngBe/bswfvvv4/t27dj7ty5uP/++9Hc3CyVuf/++/0GTYrzd2JjYyPqQZBDtMoerXL3B+YEcENUPuKa6P1lxIgRuOaaazBixAhN2pOLXq+XnM/r1q3DI488gubmZq+cP/4i1RmMUMCUzzecPn1acjZrpXx0Oh2mTZumaR4XuTz++OOYOnUqTpw4gauvvhp5eXmora2F0WhEbW0t8vLyIjZoknFlwJTPN9TV1YGIcO211yqaud4XXV1daGtr02ytdiUMGTIE//M//wNACIFYu3at1MVSGzTJYGgJUz7fEAx/z+nTp/HBBx/g9OnTmrWphKysLOTn56Onpwf333//FT/TmxFZaKp81OZwjgRqa2sBaNflihS2bNmCa665Bg6HAw899BD8pW/qvfjgl19+iXvvvVdx9kYGQy6yRrvkOCbFIMN77723vzKFnLa2Nnz88ccALj/lExcXhx07diAzMxMWiwU5OTl44IEHvMqtXLkS1dXVOHToELZt24ZVq1ahtbUVMTExETmTnxH9yFI+/lJo9CZQKtRIpa6uDoAwMXbcuHFhlkZ7br75Zvzyl7/E008/jUceeQQzZ870CqLctGkTDh06hObmZsyZMweAsPjgpk2bwiAx40pAlvLxl0KjN9EaZBis+J6BAwdi2LBhERG38eSTT6Kurg5vv/027r77bhw4cMDDsa7X67F9+3ZplU8AHnmPGQytkeXzCaR4Ojo6cPjwYVUpNcINEUkL7Gnd5Ro3bhzuuuuuiLCmBg4cCIvFgqlTp6KlpQX33HOPx5wznuexdOlSjzr5+fmar7rAYIho4nCOi4uDXq+PylntTU1NOHLkCIYMGYJ58+aFW5ygkpCQgN27d2PUqFH45z//iR/96EdSZO3atWulQERxSgYLRGQEE1XTK86cOYONGzdKS+iIRKOJLo7QzZ8/X/MVJk6cOIF9+/ZhwoQJETNXbMqUKaiursb8+fPx+uuvY/ny5aisrJQCDjdt2oQJEyagvLwcL7/8MgtEZAQNVcpn48aNSE9Pl9bcMhgMcLlcUdntev311wEgKKN0Fy9exPnz5yNuDa077rgDVVVVuO+++/DKK69g2LBheOmll6RRrYsXL2L8+PF47bXXIsJfxbg8UdXtSk9Px3333YecnBzExMQgIyMDCxYsCPn6Wmpwj2c5fPgwOI5DTEwMbr311nCLFlLy8vKwfft2xMTEoKKiAg8//HDEKUnG5Y0q5aPX69Ha2oq4uDgpPgboeynZSEGMZ7njjjvwwgsvABCczu6Lq10pfP/738f//u//IiYmBuXl5TCZTDh//ny4xWJcIahO1JGVlYXKykrk5OTgpptuwowZMxAXFxfxQYbu8SxiiomxY8desfEs+fn5iI2NxdKlS7Fr1y60tbWxoEJGSFBl+WRkZOCTTz5BRkYGsrOzsW3bNmRkZHgkLo9UxHgWd1555ZWgOMtHjx6NW265BaNHj9a8bS257777YLVaERcXh/feew+ZmZn4/PPPFbfTe4oGz/NYvHgxm6LB8IkmQ+1OpxPZ2dnSooKRjK94lkcffTQo8SxDhw5FYmIihg4dqnnbWjN37lzU1dUhOTkZR44cwQMPPIA33nhDURvuXVq73Y477rgD1dXVWLlyZXCEZkQ1qpTP8uXLpZieBx98EE1NTaiqqoqKiaViPIu4nE1iYmLQ4llcLhcOHz4seyXXcHP99dfjX//6F+bMmYPOzk5873vfwxNPPCE7JcimTZukRGWZmZlS3NCV2qVlBIBUYLVaiYiI4ziaN2+e13YtOXv2LB08eJBcLpcm7bW1tVFmZiYBoLi4OGpqaqK8vDxqa2vTpH13eJ6nZ555hnie17ztYHLu3Dn6/ve/TwAIAM2cOZMaGhpk1bXZbFI9AGSz2YIsrSfd3d108OBB6u7uDul++0u0yk1E5HK56ODBg3T27FlF9VRZPmJmPrvd7uEriYmJ6a8uDDpjxoyRkmg98MADmDZtGkus1YvBgwfjiSeewJ///GeMHTsWH3/8MW688UY899xzuHDhgt96vrq0S5cu1bxLy3xLlweqlA/Hcaivr4fFYpEWo4+GLhcAHD16FLt37wYAPPLII2GWJrL57ne/i4aGBixcuBBdXV146qmncMMNN+CDDz7wWd59iobaXNFyFAvzLV0mqDW1LBYL2e12IiIym81UWlpKW7duVducX7Tudj355JMEgO68805N2uuLaO129e4C9PT00Pbt2ykxMZEAUExMDD3wwAPU2trqUa+trY3y8vLI4XAQEZHD4VDcpc3LyyMAlJqaSjabjVJTUwkA5eXlSWUcDoe0XfykpqaSw+GI2u5LtMpNpL7bpVr5+KKsrEzL5ohIW+Vz7tw56QHatWuXBtL1TVtbG7388stB8ScFE38PwsmTJyk/P1964GNjY+mXv/ylZi8Gor4Vizv+fEvR+hBHq9xEIVY+8+bN8/rcdNNNdO2116pprk+0VD6vvPIKAaDk5GS6cOGCBtL1TbTeUL7kdrdq/vGPf9ANN9wgPfiJiYlUWlqq+ObzRyCnNbN8IouQKp9ly5YRx3HkcDikj9VqjejRru7ubpo2bRoBoGeffVYj6frm3Llz9O6779K5c+dCsj+t8PUg9O4OTZ48mQDQyJEjJQUwbtw4+vWvf01fffWV6n3LsXz66ppF60McrXIThVj59DaBRUQfkJZopXy2b99OAGjUqFHU3t6ukXR9c7n4fIj8K4Xm5maqrKykSZMmSdvj4uLoiSeeoObmZq+2A/mF5Ph8+mojWh/iaJWbKEJ8Pnv37tWyOSJSrnx83ZiLFi2S3tTPPfec5jL643JSPkR9d4e+/vpr2r59OxkMBun3mJgYuvvuu2nv3r108eJFIgqsXPrrtI7Whzha5SYKsfIpLCz0+ixfvpwKCwvVNNcnSpWPv5tb7BacOXNGcxn9cTkpH7mO4IsXL9Lu3btp/vz5HmUnTpxI69evp/r6elntaCm7L7QYmdMSpnxkkpubK/l4xE8wulxEypWPr4dk4MCBBIB++9vfBkVGf1xOykdOd6g3hw8fpscee4wSEhI8rsesWbOCFgUt9yFWczzBhCkfmQRL0fhCjc+nd/cAACUlJYXc8dvS0kLr16+nlpaWkO63vwQa7SJSZil0dnbSH//4R5o7dy7FxMR4XZvExEQ6dOhQ0GT3hVxLLhS0tbVRbm4u7d69m7q7u8NuhSklrD4fl8tFTU1Niuo4HA7auHEjVVRU0OrVq6mmpsZnOS0sHwC0ceNGRfJpQbS+zYIpd3Z2NgGgIUOGeF2j2267jUpLS6mpqYl6enpUta9E9nDPQxMRrbCkpCSqq6sLuxWmlLA7nDs6OhRFOLsrA6fTSTfeeKPPi98fn89tt90m3Vi5ubmyZdMKpny8cbegPvnkEyoqKqL4+HgvRTRx4kR66KGHaNeuXXT69GnNZY8kyyeSZFFDSJVPR0cHFRcXU15ensdHrsPZ4XBQWlqax8ldvXo1rV692qus2tEus9lMAGjQoEF01113hcWE/fLLL+mFF16gL7/8MuT77g/hUJpHjx6l3/3udzR//nwaOnSox4M4YMAAmj17Nq1bt47++te/9hkqEa0+n7q6uoiwwtQQUuVTXFxMFouFLBYLmc1mstvtZLVaFWnq3t2g3NxcTZQPkXAykpKSCAA99dRTsutpzeXkcA4lZ86cob/85S+0evVquvbaa72sogEDBtB1111Hjz76KO3YsYOam5ulblo0jnZdqZZPDBERFLJz507cd9996OjowM6dO/Hggw8CAN5++23Mnz9faXPgeR533XUXqqurYTAYPH7r7OzE4cOHMWXKFIwcOTJgW0SE/Px87NixA6mpqfjoo48wfPhwxTJpQWtrKyorK7Fs2TIkJSWFRQY1XLx4EQ0NDZgxY0ZELJ3T0tKC9957D++99x5qa2ul3NvujBs3DrNnz8a3v/1tDBs2DPv378eLL74IvV4Pnufx4x//GC+99FJEpk6599578ec//xlJSUnYsWMHli9fjubmZuTm5kZFtogzZ87gs88+w7Rp0xAbGyu7nqoE8uLqFUlJSZqsXlFYWIjNmzd7KR53PvvsM1ltvfXWW9ixYwcGDhyIn/3sZzhy5IgqmbRAPB+fffYZTp48GTY51NLQ0BBuESQMBgMMBgMeeeQRtLW14dChQzh06BAaGhpw5MgRnDhxAn/961/x17/+VapzzTXXID09Hc3NzThz5gyOHTuGLVu2RIRCdefRRx9Fe3s7nnjiCYwcORJbtmzBb37zGzz66KM4dOhQuMULGqosn/r6eixbtgyVlZVwOp0oKSmRVq9QmjKztLQUCxcu9Kt4lFg+HMfhlltuwblz57Bhwwb87Gc/UySL1jDLJzScP38eH374IQ4ePIiDBw/ivffeQ0tLi8+yw4YNw7Rp02AwGDBt2jRMnz4d06ZNw6RJkzBokOrFXPpNX+f81KlTeOihh/DCCy9EpCWn1vLRZLSrsbGRzGaz4vlXVVVVHv1aX8Ptcn0+HR0dNH36dAJA8+bNk8L5w0lHRwe99dZb1NHREW5RFBFun09/EGW3Wq0ePpTp06fT8OHDvfxH4mfw4ME0ffp0ys3NpZ/+9Kf08ssvU11dHR07dkz1sL8auX2d80hzjvdGrc9HE1UvmsRKsNvt0v88z8PlcqlOt9nd3Y0lS5agqakJ48ePx/bt2zFggCYLc/SL4cOHY8KECWHzOV2pfPnll1izZo3HtvPnz6OpqUmyMBobG6UVa48cOSL93tTU5NVebGwsUlNTcc011yA1NRWTJ0/G5MmTMWnSJEycOLHPVVtOnTqFlStXYtOmTZLVsnbtWlRUVMi2WtzXmsvMzASAyyMxvxwNVVxcTNdeey3ddNNNVFhY6PEmt1gstHPnTkUjOk6nk9LS0rw+VVVVXmUDWT49PT308MMPEwAaNmxYSKOvA9He3k47d+4M2Sx6rYh2y2fu3LmKLIWLFy/S559/TjU1NbR582Z69NFHad68eTRx4kS/lpL7Z9SoUTRr1iy6++676bHHHqPnn3+e/vSnP1FdXR0tWLBAliyBznmkBET6IqhD7S6Xi0pLS/ssY7VagzKkHEj5bNy4UZpBHYrshEpgQ+2hp7u7m/bt20e5ubn9HkYXuzvJycn0m9/8hsaMGUMAaMKECTRr1iyvOWtyP3FxcVRYWEh/+MMf6M9//jN98MEH9J///IdsNpvPcy5nKF5O6ECwwguCqnx27twpqzG55ZTQl/L5/e9/L12MUE8alQNTPqFHS9kDPfRtbW303e9+l/72t7/R7t276Ve/+hWlpaXR9773Pbr99tvpmmuu8QqYDPSJj4+ntLQ0uu2222jx4sX08MMPS0nwxo0bR7/73e8oOTmZANCiRYskWeX4hYLlOwqq8rFYLLIak1tOCf6Uz5YtW6QLtm7dOs33qwVM+YQerWXvq7sj52E+evSoV/dt1KhRdP/999M999xDN910E6WkpPic6ybnk5CQQFOmTKFZs2ZRbGysx2+JiYn0+9//nvbu3UsHDhyguro6L1l6K1M11tPDDz8cPIdzR0eHnGKyy6lFdN5df/31KCkpAQCkpaXhJz/5SVD3y7gy8bcOWW1tLfR6vSxH8OOPP46jR48iNTUV27dvx9KlS9Hc3IzOzk6P5ai7u7uxf/9+jBs3Dm1tbTh+/DhOnjyJ48eP48SJEzh58qT09+TJkzh9+jQAoL29He3t7T7lb2trw2OPPdbnMQ4cOBAPPPAAEhIS8K9//QsOhwPvvvsuli5dildffVXa7wsvvID4+Hg8/vjj2LNnDz788EP88Y9/xNKlSzF48GAp0FgRcjRUcXFxwOFil8tFxcXFijSfHNwtH/FNM2DAAEnrI4KGHHtz4sQJevHFF+nEiRPhFkURzPIRkGPZBHIEy/WzKJX7woULdPz4cWpqaqK6ujoym83SyiziJzY2lm655RaaOXMmJScne1lGWn3S09OD1+3iOI6WL1/uVwF1dHTQ8uXLFafVkIO78om2OTDR+hBHq9xE2soeSHFoeT/2V24lPp+JEyfSn/70J5owYQIBoNmzZ9OLL75Izz33HN1///0exzNz5kxKT08nvV5P8fHxPvMxTZ06NbgTS6uqqmjq1Km0du1a2rp1K+3cuZO2bt1KhYWFdO211wZlwUAib59PJA859qarq4tsNht1dXWFWxRFMOUjDy0duP2VW4vRLjnK9IsvvvBYLAAA3XrrrcGf1d7Y2Ei5ubk0depU6ZObm0scxynaqRKi2fJhDufQE0rZtRy6joRzrnbELOiWT2+0XKWyL3z5fCI1zLw3TPmEnmiVPRLkjsjRLl/0FVIeLCoqKgBAClWvra2VQtUZDEb/GDNmDHbt2iV91+v1Ht/9lXn++efx6aefKt5f+KbxqkDOyWEwGNFB+GdfMhiMKxKmfILIVVddhQULFuCqq64KtygMRsTBlE8QGTBgAAYPHhwR6T0YjEiDPRVB5PTp0/jHP/4hhcIzGIxLMOUTRLq6unDy5El0dXWFWxQGQxtOnQIWLwbExH88jwE//amqppjyYUQfPh4ALF4sbI805Mh66hQG3HsvBn/5pe8yWh1voHbk7GflSqC6GrjjDsBuB+64AzF//7syOUS0C1EKDmrW7YoUWJChCtraiPLyiMSodYdD+O4eNZyXRwQQpaYS2WzCX4AoL0++7HL2I6dMIPqQtXeZ80lJ1F1X511GThv9PG+yfu/pIfr8c6LJk4Xt33zO3HFHeJdLDhZM+YSesCofOQ+aw3Fpu/hJTSVyOOTLrkAp+C0j54H//HOiSZM8ZdXrid59l+jjj4n+9S+iN96gnquv9iwzbhzR735HtG0b0fPPEyUmev4+ahTRmjVEP/0p0Y9/THTNNcL2uDiinByikSOF7xMmEH3nO0TZ2USZmUTDhnm2M2QI0ZQpRN/6liDXwIGev8fEEA0dSjRokOd2t8/ZUE+vCBXRrHycTie9/vrr5HQ6wy2KIoKqfAI9sH0oFg9sNs8y30ww7u7upg/37aOeRYuIPvmE6ORJog8+IJo3j+i994T/332X6JVXiK66yrONMWOInnyS6JlnhL8rVhDpdJ5lYmOJbruNaM4cQQEARIMHEyUlXXpwBw8mGj7c+0G+TD9M+UQgkTBfRzZuSqG7u5s+2r2benJzLykFrbopojUxeTLR3r1EKSnC97lziWpriXbvJlq/3vMGz88X3u4PPUT0ox8RLVwoPNzuZQYNIho3jnri46knJibsD2Sfn5gYQamNGycorZQU6hk82LPMsGFERqNgsdx1l6D03H+PixPOy+OPE/3kJ0RPPUW0fLlnmbVriV58kchsFiyoLVuEfbqXGT+eyGIh2r+f6I03BHncf9frBcXe0kL05ZdE//VfwvZJk4j+/neiyZOZ8olEOjo66I033ujful1a+B3k4NbF6K6ro/PiTdjbHzB5MlFNjXBTAkS33kpUXS3c3DNnCtvi44nuuYdoxAjh+5gxRNddJ1gwY8YID184HvixY4WH5tpriQwGoTvhXmbECKJ77xWUXGEh0cMPEyUkeJZJTCT67W+FB/aNN4h+8xvP3195RbC4Pv+cqLX10sM6ebLwgAfL5yPHYuyvz4fI5/14RuXEUqZ8gogmPh85N0QgfCmwRYuIDh8m4jihO1Je7uVX6ImNJcrIEJRKUlLwlMbIkUTJyURTp1562IcPJ7rzzku+i299S+gObdwofG64gej3vyd66y2iP/2J6Pbbid59l7o5jj7avZu633rLcx+98z5p4fOR88DLtBh7cnMFubu7vcto4UyW047KF11QE8iHk8ta+ci52IFucPc2OjuJ3n9feBD/3/8j+sMfiH7xi0ujE8OGCd2cAQO0URpjxxJNny4oqOxsovvuI7r7bs8yTz5J9NprRFar8FC8/fYlqynQ8fg7J30gdRk1Ugp9ltHixeAmd7+66KGykH3AlE8EElD5yL1533/f80EqKiL6+c+JHnjgUh9erUIZPVqwKm64wcuv0DNmjGBdWK2ChdGX0iDSxvTvJ93d3XR67tyg7kNCwwc+qvyDvWDKJ9Joa6PWH/zgkvKRa9WMGyc4Dx95RHCsfutbyrs7gwYRffvbgt9l1Sqi4mLBYeteZvduogsXLski1+fTn27KN+clmG9oabQrNzcsVoBamPKJQKJW+eTl0ckxY6ji4YfpeE3NpQfxzjsFZ+X69URLlwqOT7lKJSFBiMsQH/Bf/Upw9G7a1Ld/Q6FvIqijXUEmWh/iaJWbiCmf0NPXg3b+PFFNDfWMHau8GzR0qGCxPPUUUUUF0euvCzEqn37qvR/xu8ZdnWh+EKJV9miVm4gpn9AjPtATJwrxE6NHC991uj6jQWnUKKJbbhH8NdOnC9uSkgRHrBrfRBC6OtH8IESr7NEqNxFTPtri72FtbSWqrxfiPO65p28lo9PR+ZEj6Z+zZ1O7GCWbkkJ09Gjg/SjppgShqxPND0K0yh6tchOpVz5sVrsvxJm7t90GPPccMGOG8F2vBzIygMcfB958E+ju9q73l78ADgeQlYXTcXHY81//hY5XXwVSU4Xtjz9+qfyYMcCuXUK7gPB31y5hu1y0aIPBCANXnvLpK23A8eNAVRUwfDgweDBw9Cjw1FOA0ymU7ekBxo4FvvtdYN06YPx4z7b37QOuu05QAGYzaO5cYfvMmUBtLZCXB7CVNhgMAFei8nHPR/K3vwGzZwvfJ08WlMn3vw/s2AFcuOBZr6QE+M9/BAX11lvC/19+KVg0Npvwt7kZWLtWKD9mDHqef/5SfWaRMBgeXH7Kpy/LhggoKABGjxYUxfz5gjIBgI4O4e/11wMrVgC9k77/8Y+CNRQTI3yvqBAsmdpawGhklg2DoZDLT/n0zrQ2Z47w/YYbgKQkICcH6J1T+Z57gNdeA06eBD78UPj9+HH/Vg0gy9cyduxYzJ07F2PHjg36YTMY0UZULRooi02bgIMHBWWRmXlp+9Gjwt/hwwXrpbPz0m8NDcDNNwOJicJ30XrZtElQKrW1guJRaNUMGjQII0aMwKBBl99pZjD6S3RZPv66VG1tggLZsAFYtEgYVXJnwgRgzRpg716hq9XZ2W+rRg5fffUVPvzwQ3z11VeqD5nBuFwJ2yvZ5XLBYrFAr9eD53no9XpkZ2f3XUnsUh06BGzbBixZAhw7JiiVs2f91xs6FCgqEpTIDTcAAwf226qRw/nz59Ha2orz589r3jaDEe2EzfIpLCyE0WhEdnY2CgoKsGfPHnAc13elTZsuWSpz5giKBxAUz9ChwhD4t78tbHO3bD7/3GMUisXFMBjhJyzKx+VyobGxEQaDQdo2Y8YMWCyWvivq9cD27Z7b5s8HXn9d6Hq99Rbw9ttsFIrBiALC0u2y2+1e23Q6nc/tPT09AIDOzk7g+HEMKClBzNSp0u908SJ6rrtOGEbv6ACGDBG6ZIDwPSHB83sI6erqgk6nQ1dXFzpCvO/+IJ7zM2fORN1Sz9Eqe7TKDXzzbOLSMcglLMrH6XQiPj7eY1t8fDycYiSxG+Jqny0tLcIG98C9Sw1eikKOMObMmYP29na0t7eHWxTFfPbZZ+EWQTXRKnu0yg0Iz+rIkSNll4/4MeD4+HhMmjQJQ4cOjbo3AoNxJdDT04Ouri4vgyIQYVE+vqwcX9YQIMTKjGHOYAYjolFi8YiExZQwGo1wuVwe21wuF4xGYzjEYTAYYSAsyken08FoNHoMrdvtdphMpnCIw2AwwkAMEVE4duxyuVBeXo4ZM2bIDzJkMBiXDWFTPnJQFQUdAfA8D7PZDIPBEHXWHM/zqKqqQkJCAhoaGrBw4cKoOedWqxU6nQ42mw2ZmZlRd+4BgOM42O12FBQUhFuUgJjNZtjtdun+sFqtqKyslF0/oke7CgsLUVRUJAUjrlmzBnq93iM4MdKw2+1wuVzgeT6i5fRHVVUV1q1bB0BQ/llZWVI3OZIpKSnB+vXrodfrkZOTg9mzZyM9PT3qrkFxcTHS09PDLYZsnE4nSkpKYDQasX79ekV1I3bsWnUUdJgRp4zExcWFWxTF8DyPrVu3gv9m4q5Op0NGRgaqqqrCLFlgnE6nFKSq0+kAAI2NjeEUSTEWiwUZGRnhFkMR1dXVOHLkCCorK6EXpyzJJGItHyVR0Axt0Ov1WLFihcdN1NLSguTk5DBKJY/q6mrpf47joNPpkJOTE0aJlMFxHNLT0+FyuaIqGh4QnlW9Xq9Y+USs5aMkCpqhHWKXCxAsIY7jsGrVqjBKJB/RR1heXo533nlHsoCigd5WfrRgtVqRnp4OnuexZs0aRXUjVvkwwk9hYSE2b94cNQ+FTqeDyWTCwoULUVhY6BVLFqlYLJaodI4XFBQgOztb8gm2tLQocotErPJREgXN0J7S0lJs2LAhKka6epOdnY3GxkY8/fTT4RYlIBzHRbwz3x9Wq9Xje3JyMmw2m+z6Eat8WBR0+LBYLFiyZIlk8fS+ySINjuMwe/ZsyVEOAOnp6ZcmI0cwTqcTVqsVZrMZZrMZNTU1aGxshNlsDrdofcLzvE/rMiEhQXYbEetwdo+CFh8Cu92OoqKiMEsmj2hzGoq4O/R5npfCBiIZnU4HvV7vYRU3NjZi5cqVYZRKHkaj0eOFKp7rSI/z0ev1KCoq8vCr1dfXY5uYvkYGER9kGG1R0Ha7HRzHoaKiQoo5yc7OVjwSEA5cLhdmz57ttX39+vUR75OwWq3geR46nQ4cx0Gv10f8A9wbs9ks+UxMJlPEy+8e2MlxHLKzsxX1TCJa+TAYjMuXiPX5MBiMyxumfBgMRlhgyofBYIQFpnwYDEZYYMqHwWCEBaZ8GAxGWGDKh8FghIWIjXAOFWJQoBip6XA4sGrVqqiaEc1gRCNXtOVjNpthtVpRUFAAk8kEk8mEVatWIT8/P/C68QwJLeYhlZaWKk7J0Be+ZNJ6H1oSbfJqwRVr+YhTIN555x2P7TqdDkVFRSgsLMS+ffvCJF30oNXcr8zMTM1SYPiTSct9aEm0yasVV6zyKS4uRk5Ojs/uldFoBM/zUZtnJZSUlZVp0o6W2Qr8yRSpGRGiTV6tuGKVD8dxfabZ1Ov1sNlsMJlMsFqtKCsrg06nw+bNm6HX61FaWoqdO3diw4YNMBgMqKqqwowZM9DQ0IAlS5ZAr9fDbrejrKwMGRkZSElJgdVqRVFREZxOJ8rKypCcnIzMzExJHveJeeLscjGboPsMYn/tGgwGWfXS09M9VhwQ/3e5XGhoaEBKSoqH0hVXtPB1fDzPw+l0wmw2e0z89VWH53mfcgPCyyA5ORlbtmwBx3EoLi6GyWRCfHw89uzZg5aWFilVaqBj9CWT2Ka4D/F4xYnLYq6o7OzsPs+TP4XQn2vSX3n9IcrkcrlQVFQklXXPFBFW6ArE4XBQWloaVVVV+S2Tn59Pubm50veamhrKz8/3aMNmsxERUVZWFjmdTiIicjqdlJWVJZWrqqqSvtfU1JDD4ZC2u7cnttPY2EhERGlpaR5li4uLPcr6azdQvZqaGsrNzZXktdlsHvt1Op2UlpbmJVdfx9d7H33V8Sd3TU0NrV69WpJJ3O50OunGG2+Uvss9N75kct8HEXmcByKijRs3Uk1Njd/z5H4/+ELtNdFC3t44HA5avXq1dJ/m5+dLdf3VCTVXpMNZTG/hcDj8luF53iNxupgdT+yb2+12GI1GWCwW6HQ66U0m/u/ehxf31zu1Ru80GwsWLEB5eTkASKsBiH1+Xysx+GpXTr34+HhJXnFNNPFN2LsbKuf4ehOojr/z4S6fuL2wsBArV670KCfnGANhsVg8zgMALFy40KML1Ps8yfG/qL0mWsjrjt1ux5YtW6DX62E0GrF582bJyooIqwdXcLfLYDCgqanJ7+88z3v5e3JycqTRMRFRgbln+1u5cqVHYiu5uXwSEhJQX18PQDCx16xZgxkzZvgt76tdOfV6L+vTl3xyjk9pnUDnQ3w4LBYLnE6nV14bOccYCIfD4XUeeitVNcsfqb0mgZAjrzu9712dTgen0yl17yOBK1b5FBUVYdmyZXC5XH7f9r1vepPJhMLCQhgMBqnvn5KSgvr6ek0uqGhtif39AwcOABDeYjU1NQDgU14RtfX6QsnxiQ56Lc6J6B8S/Txi20qP0d+gQUpKitfLR0xGpiWRJK/L5YqopHZXZLcLEEYSVqxY4ZVkXFzqePPmzV51DAYDdDodrFardBFNJpPXUKnVapVlovc2v2tqarBu3TrY7XaPVSvFtgINa8ut1zvFa1/LEQU6vvj4eLS3tyuqIwdxtdreXWQ5x+hLJl/HJaaJdZfRPU2vFqlwQylvINrb2yNqBO2KtXwASA96aWkpUlJSAAg3eXV1td83ijgC4051dTXKyspgMBgkf4W4wKHFYoHL5YLZbPaypJKTk6Wuid1ux7Zt26DX62EymdDQ0CD18w0GA5KTk2GxWFBQUOC33UD1OI6DxWJBY2MjLBYLjEYjSktLpXZMJpPkcyopKZFGZfwdHyD4Nvbs2SON1PR1TsT99pZblEtMywlcymVssVjAcZykqAMdoz+Zeu8jOzsblZWVKC8vR0pKClwuFwwGg2Rd+TpPPM+jtLTUY20zEbXXRAt55aIkuXsoYGlUw4T4UCld35rBUIs/xRkurthuVyQQyNRmMLTCbrdLMWWRAlM+YUA00ZuamiJ+TSzG5QHP8xHl7wFYt4vBYIQJZvkwGIywwJQPg8EIC0z5MBiMsMCUD4PBCAtM+TAYjLDAlA+DwQgLTPkwGIywwJQPg8EIC/8f2c3tKafyFzsAAAAASUVORK5CYII=\n"
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "colors = ['black', 'red']\n",
    "marker_size = 15\n",
    "res_dicts = [res_dict_small, res_dict_large]\n",
    "save_path = f'../fig/multivariate_t_{df}.pdf'\n",
    "steps_finite_samples = 3\n",
    "\n",
    "fig, ax = plt.subplots(figsize=(.5*LINEWIDTH, .3*LINEWIDTH))\n",
    "\n",
    "# Orientation lines\n",
    "# ax.axhline(y=ridge.r_sq, linestyle='--', color='gray', linewidth=1) # base risk of 0 classifier\n",
    "ax.axvline(x=1, linestyle='--', color='gray', linewidth=1) # base risk of 0 classifier) # interpolation threshold\n",
    "\n",
    "# Min-norm interpolator\n",
    "ax.plot(gams, res_dict_small['risk_caus'], color=colors[0], linestyle='-')\n",
    "ax.scatter(gams[::steps_finite_samples], res_dict_small['finite_loss_caus'][::steps_finite_samples], color=colors[0], marker='x', s=marker_size)\n",
    "\n",
    "# Optimal ridge regularization\n",
    "ax.plot(gams, res_oracle['risk_optim_caus'], color=colors[1], linestyle='-')\n",
    "ax.scatter(gams[::steps_finite_samples], res_oracle['finite_loss_optim_caus'][::steps_finite_samples], color=colors[1], marker='x', s=marker_size)\n",
    "\n",
    "# Legend and axes\n",
    "ax.set_xlabel(r'Overparameterization ratio $\\gamma$')\n",
    "ax.set_ylabel(r'Causal risk')\n",
    "ax.set_xlim([0, max(gams)])\n",
    "ax.set_ylim([0, 6])\n",
    "\n",
    "ax.set_title(f'Student-t (finite 4-th moment)', fontsize=10)\n",
    "\n",
    "handles = [Line2D([], [], label='Min-norm', color='k', linestyle='-'),\n",
    "           Line2D([], [], label='Optimal ridge', color='red', linestyle='-')]\n",
    "\n",
    "plt.legend(handles=handles, framealpha=1, frameon=True, ncol=1)\n",
    "\n",
    "if save_path is not None:\n",
    "    plt.savefig(fname=save_path, bbox_inches='tight', dpi=300)\n",
    "\n",
    "plt.show()"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  },
  {
   "cell_type": "markdown",
   "source": [
    "## Mixture of Gaussians"
   ],
   "metadata": {
    "collapsed": false
   }
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "outputs": [],
   "source": [
    "def generate_gaussian_mixture(n_mixture, dim):\n",
    "    means = [n_mixture/(n_mixture - 1) * n_mixture/dim * np.random.standard_normal(size=dim) for _ in range(n_mixture)]\n",
    "    # means = [np.sqrt(dim) * np.random.standard_normal(size=dim) for _ in range(n_mixture)]\n",
    "    covs = [np.eye(dim) for _ in range(n_mixture)]\n",
    "    # covs = [np.sqrt(dim) * wishart.rvs(df=dim, scale=np.eye(dim)) for _ in range(n_mixture)]\n",
    "    # mixture_probs = np.random.dirichlet(np.ones(n_mixture))\n",
    "    mixture_probs = 1 / n_mixture * np.ones(n_mixture)\n",
    "    mean_total = np.zeros(dim)\n",
    "    cov_total = np.zeros((dim, dim))\n",
    "    for i in range(n_mixture):\n",
    "        mean_total += mixture_probs[i] * means[i]\n",
    "        cov_total += mixture_probs[i] * (covs[i] + means[i].reshape(-1, 1) @ means[i].reshape(1, -1))\n",
    "    mean_total = mean_total.flatten()\n",
    "    cov_total -= mean_total.reshape(-1, 1) @ mean_total.reshape(1, -1)\n",
    "\n",
    "    eigvals, V = np.linalg.eigh(cov_total)\n",
    "    normalization_matrix = V @ np.diag(eigvals ** (-1/2)) @ V.T\n",
    "\n",
    "    def sample_GMM(n_samples):\n",
    "        Z = []\n",
    "        for i in range(n_samples):\n",
    "            class_idx = np.random.choice(a=list(range(len(mixture_probs))), p=mixture_probs)\n",
    "            Z.append(np.random.multivariate_normal(mean=means[class_idx], cov=covs[class_idx]).flatten() - mean_total)\n",
    "        return np.array(Z) @ normalization_matrix\n",
    "\n",
    "    return sample_GMM\n",
    "ridge.sample_z = generate_gaussian_mixture(n_mixture=5, dim=ridge.l)"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "outputs": [],
   "source": [
    "gams = np.linspace(0.1, 5, 80)\n",
    "lam_small, lam_large = 0, .5\n",
    "res_dict_small = ridge.compute_risks(lam=lam_small, gams=gams, include_finite=True)\n",
    "# res_dict_large = ridge.compute_risks(lam=lam_large, gams=gams, include_finite=True)\n",
    "res_oracle = ridge.compute_oracle(gams, include_finite=True)"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "outputs": [
    {
     "data": {
      "text/plain": "<Figure size 300.058x180.035 with 1 Axes>",
      "image/png": "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pyNOQyE3Nxe7du3Crl27IsLzkcRHi2kVsbGxiIuL65FBhhLSooL79u2Tm14cx6GgoACA+m5QBkOip64hVeJjsVgCBpZffPHFbhukN9LIZi3iPQ0NDfjss896tOfFd2nnwsJCAGKObanOpqamHqubcXUgXUN9+/bVtFxVXe1utxuzZs0Cx3HykHX6LrdPtHfvaun5nD9/HoIg4Pz5890uSwmTJk3CvffeizfffBOrV6/G7373O9TV1QEQ052o7ZVgXJ20tbWhqakJdXV1SEhIQJ8+2i5wrKo0l8slj+3xRe0M3kihoaFBHkw1adIkna1Rx9NPP40333wTf//73/GrX/0KCQkJsgAxGGpISEiQR5VriSrxWb58ecD5WGFNRNQD7N27F21tbUhJScENN9ygtzmqmDRpEu677z689tpr+MUvfoHKyko8+eSTOHDgAJKSkrBq1SoUFBSgtrYWd999d8DA9KVLl3DkyBGkpKSEbWyVVkSr7ZFqd9++fTX3eCRUlRpsImikTRANlffffx8AMG3aNJ0t6R7PPfccduzYgQ8//BClpaX4wx/+gLvuugu7d+/G7t27AYgDEZ988smAI1al+TsDBgyIqBtBCdFqe7Ta3R1YEMAHSXykNdG7y7XXXoubbrqp2132oeIbfF6+fDni4uJQWlrqt09paWnUe6qM6IaJz3ecOXNGDjZrJT4GgwFjx44Na44UicceewxjxozBN998g8cee6xDnueHH3444nIoMa4umPh8R2VlJYgIt9xyC4YPH65JmS0tLaivr9dsrfZQ6Nu3L/7nf/4HALBlyxbU1NQgOTkZTqdTzoAYbI4egxEOmPh8R0/Ee86cOYP9+/fjzJkzmpUZCtOnT0d2djYAoH///njzzTdhNpuxZ88eZGVlYePGjbrYxWAAGovPq6++qmVxYWXPnj0AtGtyRQrr16/HTTfdhPPnz2P16tUgInAch23btmHIkCF6m8e4ilHU26XEPZcGGapJIK839fX1chqNK0184uLi8MorryA9PR12ux0ZGRl45JFH9DaLwVAmPsFSaLSnq1SokUplZSUAcWLssGHDdLZGe26//XY8++yzeOqpp/Dzn/8c48ePj9pBlIwrB0XiEyyFRnuiteu2p8b39O7dG9dcc01EjNt44oknUFlZiXfffRf33HMPPv74Y80C6wyGGhTFfLoSnsbG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\n"
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "colors = ['black', 'red']\n",
    "marker_size = 15\n",
    "res_dicts = [res_dict_small, res_dict_large]\n",
    "save_path = '../fig/GMM.pdf'\n",
    "steps_finite_samples = 3\n",
    "\n",
    "fig, ax = plt.subplots(figsize=(.5*LINEWIDTH, .3*LINEWIDTH))\n",
    "\n",
    "# Orientation lines\n",
    "# ax.axhline(y=ridge.r_sq, linestyle='--', color='gray', linewidth=1) # base risk of 0 classifier\n",
    "ax.axvline(x=1, linestyle='--', color='gray', linewidth=1) # base risk of 0 classifier) # interpolation threshold\n",
    "\n",
    "# Min-norm interpolator\n",
    "ax.plot(gams, res_dict_small['risk_caus'], color=colors[0], linestyle='-')\n",
    "ax.scatter(gams[::steps_finite_samples], res_dict_small['finite_loss_caus'][::steps_finite_samples], color=colors[0], marker='x', s=marker_size)\n",
    "\n",
    "# Optimal ridge regularization\n",
    "ax.plot(gams, res_oracle['risk_optim_caus'], color=colors[1], linestyle='-')\n",
    "ax.scatter(gams[::steps_finite_samples], res_oracle['finite_loss_optim_caus'][::steps_finite_samples], color=colors[1], marker='x', s=marker_size)\n",
    "\n",
    "# Legend and axes\n",
    "ax.set_xlabel(r'Overparameterization ratio $\\gamma$')\n",
    "ax.set_ylabel(r'Causal risk')\n",
    "ax.set_xlim([0, max(gams)])\n",
    "ax.set_ylim([0, 6])\n",
    "\n",
    "ax.set_title('GMM', fontsize=10)\n",
    "\n",
    "handles = [Line2D([], [], label='Min-norm', color='k', linestyle='-'),\n",
    "           Line2D([], [], label='Optimal ridge', color='red', linestyle='-')]\n",
    "\n",
    "plt.legend(handles=handles, framealpha=1, frameon=True, ncol=1)\n",
    "\n",
    "if save_path is not None:\n",
    "    plt.savefig(fname=save_path, bbox_inches='tight', dpi=300)\n",
    "\n",
    "plt.show()"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 2
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython2",
   "version": "2.7.6"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 0
}