{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "from src.utils.training import get_objective, get_optimizer\n",
    "from src.utils.data import load_dataset\n",
    "from scipy.optimize import minimize\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "import torch\n",
    "from tqdm import tqdm\n",
    " \n",
    "from src.optim.SOREL import Sorel\n",
    "from src.optim.SOREL_batch import SorelBatch"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "# import folktables\n",
    "from folktables import ACSDataSource, ACSEmployment\n",
    "\n",
    "from sklearn.model_selection import train_test_split\n",
    "from sklearn.preprocessing import StandardScaler\n",
    "from sklearn.model_selection import KFold"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Load Dataset"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "# law school data\n",
    "dir = './data/lawschool/'\n",
    "features = np.load(os.path.join(dir,'regression_X.npy'))\n",
    "labels = np.load(os.path.join(dir,'regression_y.npy'))\n",
    "group = np.load(os.path.join(dir,'regression_group.npy'))\n",
    "\n",
    "# acs employment data\n",
    "# data_source = ACSDataSource(survey_year='2018', horizon='1-Year', survey='person')\n",
    "# acs_data = data_source.get_data(states=[\"CA\"], download=True)\n",
    "# acs_data = acs_data.sample(n=10000, random_state=1)\n",
    "# features, labels, group = ACSEmployment.df_to_numpy(acs_data)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_data_fold(features, label, group, train_index, test_index):\n",
    "    X_train, X_test = features[train_index], features[test_index]\n",
    "    y_train, y_test = label[train_index], label[test_index]\n",
    "    group_train, group_test = group[train_index], group[test_index]\n",
    "\n",
    "    y_mean = np.mean(y_train)\n",
    "    y_train = y_train - y_mean\n",
    "    y_test = y_test - y_mean\n",
    "\n",
    "    scaler = StandardScaler().fit(X_train)\n",
    "    X_train = scaler.transform(X_train)\n",
    "    X_test = scaler.transform(X_test)\n",
    "\n",
    "    X_train = torch.tensor(X_train)\n",
    "    X_test = torch.tensor(X_test)\n",
    "    y_train = torch.tensor(y_train)\n",
    "    y_test = torch.tensor(y_test)\n",
    "    return X_train, y_train, X_test, y_test, group_train, group_test"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "val_num = 15\n",
    "kf = KFold(n_splits=5, shuffle=True, random_state=val_num) \n",
    "train_index, test_index = next(kf.split(features))\n",
    "X_train, y_train, X_test, y_test, group_train, group_test = get_data_fold(features, labels, group, train_index, test_index)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Get Objective and Optimizer"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "X_train, y_train, X_test, y_test, group_train, group_test = get_data_fold(features, labels, group, train_index, test_index)\n",
    "model_cfg = {\n",
    "    \"objective\": \"cvar\", # Options: 'cvar', 'extremile', 'esrm'.\n",
    "    \"para_value\": 0.95,  # to apply alpha-cvar, rho-esrm or r-extremile,  set para_value = 1-alpha (cvar) or rho (esrm) or r (extremile).\n",
    "    \"l2_reg\": 1.0,\n",
    "    \"loss\": \"squared_error\",  # 'squared_error' for law school data, 'binary_cross_entropy' for acs employment data.\n",
    "    \"n_class\": None,\n",
    "    \"shift_cost\": 0,\n",
    "}\n",
    "autodiff = False # non-autodiff variants\n",
    "train_obj = get_objective(model_cfg, X_train, y_train, autodiff=autodiff)\n",
    "val_obj   = get_objective(model_cfg, X_test, y_test, autodiff=autodiff)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Training"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "L-BGFS sucsess! Minimum loss: 0.0248\n"
     ]
    }
   ],
   "source": [
    "# Get the optimal solution by L-BFGS\n",
    "train_obj_ = get_objective(model_cfg, X_train, y_train)\n",
    "\n",
    "# Define function and Jacobian oracles.\n",
    "def fun(w):\n",
    "    return train_obj_.get_batch_loss(torch.tensor(w, dtype=torch.float64)).item()\n",
    "\n",
    "def jac(w):\n",
    "    return (\n",
    "        train_obj_.get_batch_subgrad(\n",
    "            torch.tensor(w, dtype=torch.float64, requires_grad=True)\n",
    "        )\n",
    "        .detach()\n",
    "        .numpy()\n",
    "    )\n",
    "\n",
    "# Run optimizer.\n",
    "d = train_obj.d\n",
    "init = np.zeros((d,), dtype=np.float64)\n",
    "if model_cfg[\"n_class\"]:\n",
    "    init = np.zeros((model_cfg[\"n_class\"] * d,), dtype=np.float64)\n",
    "else:\n",
    "    init = np.zeros((d,), dtype=np.float64)\n",
    "output = minimize(fun, init, method=\"L-BFGS-B\", jac=jac)\n",
    "if output.success:\n",
    "    print(f\"L-BGFS sucsess! Minimum loss: {output.fun:0.4f}\")\n",
    "else:\n",
    "    raise Exception(output.message)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "lr = 3e-4\n",
    "lrd = 1e3\n",
    "seed = 1\n",
    "optimizer = SorelBatch(\n",
    "            train_obj,\n",
    "            lr=lr,\n",
    "            smooth_coef=0,\n",
    "            smoothing=\"l2\",\n",
    "            seed=seed,\n",
    "            length_epoch=train_obj.n,\n",
    "            lrdcon=lrd,\n",
    "            xlrcon=20,\n",
    "            batch_size = 64\n",
    "        )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "# You can get baseline optimizers by running the following code.\n",
    "# optim_cfg = {\n",
    "#         \"optimizer\": \"sgd\", # Options: 'sgd', 'lsvrg_batch', 'prospect_batch'\n",
    "#         \"lr\": lr,\n",
    "#         \"epoch_len\": train_obj.n, # Used as an update interval for LSVRG, and otherwise is simply a logging interval for other methods.\n",
    "#         \"shift_cost\": 0,\n",
    "#     }\n",
    "\n",
    "# optimizer = get_optimizer(optim_cfg, train_obj, seed)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [00:03<00:00, 27.94it/s]\n"
     ]
    }
   ],
   "source": [
    "n_epochs = 100\n",
    "\n",
    "train_losses = []\n",
    "epoch_len = optimizer.get_epoch_len()\n",
    "for epoch in tqdm(range(n_epochs)):\n",
    "\n",
    "    optimizer.start_epoch()\n",
    "    for _ in range(epoch_len):\n",
    "        optimizer.step()\n",
    "    optimizer.end_epoch()\n",
    "    train_losses.append(train_obj.get_batch_loss(optimizer.weights).item())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAcIAAAHACAYAAAAx0GhOAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjQuMywgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/MnkTPAAAACXBIWXMAAA9hAAAPYQGoP6dpAABMH0lEQVR4nO3deVwU9f8H8NeysMsNAgqioJgnoiAr4v31SAzNpK+VlhepFYqmkh1qltlBWab2FfVrplZaHpXm1yyjwxMPRPDCs0BQQASVU1hY5veHPzeHBeVYmD1ez8djH4/mM7Mzbybl5cx85vORCYIggIiIyExZSF0AERGRlBiERERk1hiERERk1hiERERk1hiERERk1hiERERk1hiERERk1hiERERk1iylLkDfKioqkJGRAQcHB8hkMqnLISIiiQiCgIKCAnh6esLCovrrPpMLwoyMDHh5eUldBhERGYj09HS0bNmy2vUmF4QODg4A7v7gjo6OEldDRERSyc/Ph5eXlzYXqmNyQXjvdqijoyODkIiIHvqYjJ1liIjIrDEIiYjIrDEIiYjIrJncM0Ii0iUIAsrLy6HRaKQuhUhv5HI5LC0t6/2qHIOQyMSp1WpkZmaiuLhY6lKI9M7W1hbNmzeHQqGo8z4YhEQmrKKiAikpKZDL5fD09IRCoeBAE2QSBEGAWq3GjRs3kJKSgnbt2j3wpfkHYRASmTC1Wo2Kigp4eXnB1tZW6nKI9MrGxgZWVla4cuUK1Go1rK2t67QfdpYhMgN1/ZcykaHTx59tg/zbsWvXLnTo0AHt2rXD2rVrpS6HiIhMmMEFYXl5OaKiovDHH3/gxIkT+Oijj3Dz5k2pyyIiIzdgwADMmjVL6jKwd+9eyGQy3L59W5LjL1y4EAEBAY1+nPDwcISFhTX4cevC4ILw2LFj6Ny5M1q0aAEHBwcMGzYMe/bskbosImokMpnsgZ/w8PA67feHH37Au+++W6/asrOz8dJLL8Hb2xtKpRIeHh4YOnQoDh8+XK/9NhSZTIYdO3aI2ubMmYPff/+90WtZvnw5NmzYoF02lH+YAA0QhPv378eIESPg6elZ5f8EAFi5ciV8fHxgbW0NlUqFAwcOaNdlZGSgRYsW2uWWLVvi2rVr+i6zRopKy5GdXyLJsYnMVWZmpvazbNkyODo6itqWL18u2r6srKxG+3VxcXno4MsPM2rUKJw8eRJffvklLl68iJ07d2LAgAFGddfK3t4erq6ujX5cJycnODs7N/pxa0LvQVhUVAR/f3+sWLGiyvVbtmzBrFmzMH/+fCQmJqJfv34IDQ1FWloagLtdYitrzO7eBy/lIGpLEoZ8ug9+C/cg+ufzjXZsooZWUSEgt7BUsk9Fhe7f78o8PDy0HycnJ8hkMu1ySUkJnJ2dsXXrVgwYMADW1tbYuHEjcnNz8eyzz6Jly5awtbVFly5d8O2334r2W/kKpHXr1vjggw8wadIkODg4wNvbG2vWrKm2rtu3b+PgwYP46KOPMHDgQLRq1Qo9evTA3LlzMXz4cABAamoqZDIZkpKSRN+TyWTYu3evaH+HDh2Cv78/rK2tERwcjNOnT4vWf//99+jcuTOUSiVat26NJUuWiNa3bt0a7777Lp577jnY29vD09MT//nPf0TrAeDJJ5+ETCbTLld3y/KDDz6Au7s7nJ2d8c4776C8vByvvvoqXFxc0LJlS6xbt050/Ndffx3t27eHra0t2rRpgwULFjzwHyX33xoNDw/Hvn37sHz5cu2VfkpKCtq2bYtPPvlE9L0zZ87AwsICf/31V7X7ri+9vz4RGhqK0NDQatd/+umnmDx5MqZMmQIAWLZsGfbs2YNVq1YhOjoaLVq0EF0BXr16FcHBwdXur7S0FKWlpdrl/Pz8etWfklOIHxL/Of6pq7frtT8iQ3KrWA3Ve79JdvyENx+Fq72y3vt5/fXXsWTJEqxfvx5KpRIlJSVQqVR4/fXX4ejoiJ9++gnjx49HmzZtHvj7Y8mSJXj33Xcxb948fPfdd5g6dSr69++Pjh076mxrb28Pe3t77NixAz179oRSWb+f49VXX8Xy5cvh4eGBefPm4YknnsDFixdhZWWFhIQEPPPMM1i4cCFGjx6NuLg4TJs2Da6urqJbwx9//DHmzZuHhQsXYs+ePZg9ezY6duyIIUOGID4+Hs2aNcP69evx2GOPQS6XV1vLH3/8gZYtW2L//v04dOgQJk+ejMOHD6N///44evQotmzZgoiICAwZMkQ736uDgwM2bNgAT09PnD59Gi+88AIcHBzw2muvPfRnX758OS5evAg/Pz8sWrQIANC0aVNMmjQJ69evx5w5c7Tbrlu3Dv369cMjjzxSxzP9cI36jFCtViMhIQEhISGi9pCQEMTFxQEAevTogTNnzuDatWsoKCjA7t27MXTo0Gr3GR0dDScnJ+2nvpPy+rVwEi3/nVOEwtLyeu2TiPRr1qxZ+Pe//w0fHx94enqiRYsWmDNnDgICAtCmTRvMmDEDQ4cOxbZt2x64n2HDhmHatGlo27YtXn/9dbi5uelcud1jaWmJDRs24Msvv4SzszP69OmDefPm4dSpU3X6Gd5++20MGTIEXbp0wZdffonr169j+/btAO5eMAwePBgLFixA+/btER4ejunTp+Pjjz8W7aNPnz5444030L59e8yYMQNPPfUUli5dCuBusACAs7MzPDw8tMtVcXFxwWeffYYOHTpg0qRJ6NChA4qLizFv3jy0a9cOc+fOhUKhwKFDh7TfefPNN9G7d2+0bt0aI0aMwCuvvIKtW7fW6Gd3cnKCQqGAra2t9mpfLpfj+eefx4ULF3Ds2DEAd297b9y4EZMmTar5ia2DRg3CnJwcaDQauLu7i9rd3d2RlZUF4O4ftiVLlmDgwIHo1q0bXn311Qfez547dy7y8vK0n/T09HrV2Km5I+QW/9yKFQQgOaN+V5lEpF/du3cXLWs0Grz//vvo2rUrXF1dYW9vj19//VX7yKU6Xbt21f73vVuw2dnZ1W4/atQoZGRkYOfOnRg6dCj27t2LwMBAUSeQmurVq5f2v11cXNChQwecO3cOAHDu3Dn06dNHtH2fPn1w6dIl0Xix9+/j3vK9fdRG586dRe/jubu7o0uXLtpluVwOV1dX0bn57rvv0LdvX3h4eMDe3h4LFix46Pl+mObNm2P48OHa27C7du1CSUkJnn766Xrt92Ek6TVa+ZmfIAiitnu3CC5fvowXX3zxgftSKpXaSXj1MRmvtZUc7ZrZi9pOX8ur1z6JSL/s7OxEy0uWLMHSpUvx2muv4Y8//kBSUhKGDh0KtVr9wP1YWVmJlmUyGSoqKh74HWtrawwZMgRvvfUW4uLiEB4ejrfffhvAPy9339/Xoaadee4d/973q/o9WZt91EZV5+FB5+bIkSMYM2YMQkNDsWvXLiQmJmL+/PkPPd81MWXKFGzevBl37tzB+vXrMXr06AYfFalRh1hzc3ODXC7XXv3dk52drXOVWFsxMTGIiYnRy+j6XVo44XxWgXb5DIOQTEQTWwUS3nxU0uM3hAMHDmDkyJEYN24cgLtjrF66dAmdOnVqkOPdz9fXV9s7/t7tx8zMTHTr1g0ARB1n7nfkyBF4e3sDAG7duoWLFy9qn036+vri4MGDou3j4uLQvn170bO+I0eO6Ozz/uebVlZWDTLjyKFDh9CqVSvMnz9f23blypVa7UOhUFRZ27Bhw2BnZ4dVq1bh559/xv79++td78M0ahAqFAqoVCrExsbiySef1LbHxsZi5MiR9dp3ZGQkIiMjkZ+fDycnp4d/4QG6tnTCtoSr2mV2mCFTYWEh00tnFUPTtm1bfP/994iLi0OTJk3w6aefIisrS69BmJubi6effhqTJk1C165d4eDggOPHj2Px4sXa3182Njbo2bMnPvzwQ7Ru3Ro5OTl48803q9zfokWL4OrqCnd3d8yfPx9ubm7aXpWvvPIKgoKC8O6772L06NE4fPgwVqxYgZUrV4r2cejQISxevBhhYWGIjY3Ftm3b8NNPP2nXt27dGr///jv69OkDpVKJJk2a6OVctG3bFmlpadi8eTOCgoLw008/aZ9v1lTr1q1x9OhRpKamwt7eHi4uLrCwsIBcLkd4eDjmzp2Ltm3b6tz+bQh6vzVaWFiIpKQk7b+CUlJSkJSUpL13HBUVhbVr12LdunU4d+4cZs+ejbS0NEREROi7lDpjhxki47JgwQIEBgZi6NChGDBgADw8PPQ+iom9vT2Cg4OxdOlS9O/fH35+fliwYAFeeOEF0eti69atQ1lZGbp3746ZM2fivffeq3J/H374IWbOnAmVSoXMzEzs3LlTO5VQYGAgtm7dis2bN8PPzw9vvfUWFi1apDOYwCuvvIKEhAR069YN7777LpYsWSLqXLhkyRLExsbCy8tLe4WqDyNHjsTs2bMxffp0BAQEIC4uDgsWLKjVPubMmQO5XA5fX180bdpU9Hxx8uTJUKvVDd5JRkvQsz///FMAoPOZOHGidpuYmBihVatWgkKhEAIDA4V9+/bV+7grVqwQOnXqJLRv314AIOTl5dV5X3fU5UKbuT8JrV7fpf0c/Tu33jUSNbY7d+4IycnJwp07d6QuhfSsVatWwtKlS6Uuo0EcPHhQsLS0FLKysh667YP+jOfl5dUoD/R+a3TAgAEPfag7bdo0TJs2Ta/H1eet0XsdZu5/Tnj6Wh56+LjUt0wiIqpGaWkp0tPTsWDBAjzzzDP17jtSUwY31qih6FLp9ig7zBARNaxvv/0WHTp0QF5eHhYvXtxoxzWZiXn12WsUALqwwwwRGbDU1FSpS9C78PDwOg+qXh8mc0UYGRmJ5ORkxMfH62V/la8I2WGGiMg0mUwQ6htHmCEiMg8MwmpwhBkyJQ/rwEZkrPTxZ5tB+ADsMEPG7t4wWcXFxRJXQtQw7v3ZrjwkXG2ws8wDsMMMGTu5XA5nZ2ftYMm2traNOr8nUUMRBAHFxcXIzs6Gs7PzA6eZehiTCUJ9vkd4T3UdZuyVJnPayAx4eHgAwANnVSAyVvemmaoP/kZ/gHsdZjT/P6v2vQ4zfLGejIlMJkPz5s3RrFmzWs2EQGTorKys6nUleA+D8AGqGmEmKf0Wg5CMklwu18svDSJTYzKdZWJiYuDr64ugoCC97rebt7No+bdzvL1ERGRKTCYI9f1C/T2DO4rHujueehO5haV6PQYREUnHZIKwofRt5wZbxT+3kyoE4Ldz1yWsiIiI9IlB+BDWVnL8q31TUduvZxmERESmgkFYA0M7i7vmHricw3FHiYhMBIOwBgZ2bAbL+8YdVZdXYN+FGxJWRERE+mIyQdhQvUYBwMnGCr0ecRW1/ZqcpffjEBFR4zOZIGyoXqP3hFS6PfrH+Wyoyysa5FhERNR4TCYIG1qIr/g1ioKSchz+O1eiaoiISF8YhDXk7miNAC9nUduvZ3l7lIjI2DEIa6Fy79E9Z6/z9igRkZFjENbC0M7i26M5haX44cTVarYmIiJjwCCshTZN7RHUuomoLWbvZZRpeFVIRGSsGIS1NGNQO9Fy+s07+DEpQ6JqiIiovkwmCBvyPcL79WvnBv9KnWZW/nlZO2chEREZF5MJwoZ+j/AemUyGmYPbitr+zinCrlO8KiQiMkYmE4SNaWCHZvBr4ShqW/HHZVTwqpCIyOgwCOtAJpNh+kDxs8JL2YX46XSmRBUREVFdMQjrKMTXHR09HERtH+w+hyLOSkFEZFQYhHVkYSHDzMHiq8LMvBIs++2iRBUREVFdMAjr4TE/D/Rt6yZqW3coFecy8yWqiIiIaotBWA8ymQyLRnaGQv7PadRUCJi//TQ7zhARGQkGYT21aWqPqQMeEbWdSLuNrcfTJaqIiIhqg0GoB1MHPILWrraitg92n8PVW8USVURERDVlMkHYWCPLVMXaSo5FI/1Ebfkl5Yj8JpGzUxARGTiZIAgm9TArPz8fTk5OyMvLg6Oj48O/oEczNyfqjDs6qY8P3hrh26h1EBFRzfPAZK4IDcF7YX46t0jXHUrBL2c4gS8RkaFiEOqRg7UVYsYGQmEpPq2vfncSV3KLJKqKiIgehEGoZ509nfB2pVuhBSXlmLQhHnnFZRJVRURE1WEQNoDnenhjZICnqO2vG0V4aeNxlJZrJKqKiIiqwiBsADKZDB882QUd3MVjkR75+ybe+P40TKx/EhGRUWMQNhA7pSXWPR+EZg5KUfv2xGtYGsvxSImIDAWDsAG1cLbBuvAg2CrkovbP/riMlXsvS1QVERHdj0HYwPxaOGHFc91gIRO3L/7lAsOQiMgAMAgbwaCO7jojzwAMQyIiQ8AgbCTjerbCW4/rjjCz+JcLWPHHJQkqIiIiwECD8Mknn0STJk3w1FNPSV2KXk3q61NlGH7y60Us/uU8e5MSEUnAIIPw5ZdfxldffSV1GQ2iujBcufcvvPO/ZM5jSETUyAwyCAcOHAgHB4eHb2ikJvX1wTtPdNZp3xCXirk/nIaGYUhE1GhqHYT79+/HiBEj4OnpCZlMhh07duhss3LlSvj4+MDa2hoqlQoHDhzQR60mZWLv1lg8qitklXqTbjmejte/P8UwJCJqJLUOwqKiIvj7+2PFihVVrt+yZQtmzZqF+fPnIzExEf369UNoaCjS0tK026hUKvj5+el8MjIyqtynqXomyAvLx3SDZaV3K75LuMowJCJqJJa1/UJoaChCQ0OrXf/pp59i8uTJmDJlCgBg2bJl2LNnD1atWoXo6GgAQEJCQh3L1VVaWorS0lLtcn5+vt723Rie8PeEjZUckZtOQK35ZxLf7xKuAgA+GtUV8sovIRIRkd7o9RmhWq1GQkICQkJCRO0hISGIi4vT56G0oqOj4eTkpP14eXk1yHEa0hBfd6waFwiFXPy/496VITvQEBE1HL0GYU5ODjQaDdzd3UXt7u7uyMqq+eS0Q4cOxdNPP43du3ejZcuWiI+Pr3bbuXPnIi8vT/tJT0+vc/1SGtzJHavHVx2Gb/zAMCQiaii1vjVaE7JKPUAEQdBpe5A9e/bUeFulUgmlUvnwDY3AoI53wzDia/Ft0q3Hr8Li/2e0sOBtUiIivdLrFaGbmxvkcrnO1V92drbOVaK+xcTEwNfXF0FBQQ16nIZ2Lwyt5OLA2xyfjjd/PMMrQyIiPdNrECoUCqhUKsTGxoraY2Nj0bt3b30eSkdkZCSSk5MfeBvVWAzq6I5VY1U6YfjN0TS887+zHIGGiEiPan1rtLCwEJcv/zNQdEpKCpKSkuDi4gJvb29ERUVh/Pjx6N69O3r16oU1a9YgLS0NERERei3c1D3q646Y5wIxbdMJlN93Ffjl4StQWFpg3rBOtbrdTEREVat1EB4/fhwDBw7ULkdFRQEAJk6ciA0bNmD06NHIzc3FokWLkJmZCT8/P+zevRutWrXSX9VViImJQUxMDDQaTYMepzGFdPbAiucCMf0bcRh+fiAF1lZyvBLSQcLqiIhMg0wwsfts+fn5cHJyQl5eHhwdHaUuRy92ncrAy98movLjwVeGtMeMwe2kKYqIyMDVNA8McqxREnu8qyeWPOOvMxzbktiL+Hz/39IURURkIkwmCE2l12h1nuzWEtFPdtFpf3/3OXx9OLXxCyIiMhG8NWpkvjqcird+PKvTvnhUVzwTZHyj6hARNRTeGjVRE3q1xrxhHXXaX//hFHaeNK9By4mI9IFBaIRe7P8IZj/aXtQmCEDUliT8fu66RFURERknkwlCU39GWNnLg9ti6oBHRG3lFQKmbjqBuL9yJKqKiMj48BmhERMEAQt3nsWXh6+I2u0UcmycEoxu3k0kqoyISHp8RmgGZDIZ3h7RGaMCW4rai9QahK+Px/ks45qbkYhICgxCI2dhIcNHo7og1M9D1J53pwzjvziG1JwiiSojIjIODEITYCm3wLIxAejfvqmo/UZBKcauPYrMvDsSVUZEZPhMJgjNrbNMZUpLOVaPC0T3VuLngtdu38G4tUeRW1gqUWVERIaNnWVMTH5JGZ5dcwRnM8TPB7u2dMI3L/SEvbJB5mImIjI47CxjphytrfDVpB54pKmdqP3U1TxEfJ0AdXmFRJURERkmBqEJcrVXYuOUYLRwthG1H7ycg6itSZzlnojoPgxCE9XcyQZfTe4BFzuFqH3XqUzOck9EdB8GoQl7pKk91ocHwVYhF7V/efgKlv9+SaKqiIgMi8kEobn3Gq2Ov5cz/jteBSu5eDLDZb9dwpdxqdIURURkQNhr1Ez872QGXt6ciMr/t5ePCcDIgBbSFEVE1IDYa5RERvh7YtFIP532V7aexJ/nsyWoiIjIMDAIzcj4nq3wyhDx9E13Z6xIQMKVWxJVRUQkLQahmZk+qC0m9fERtZWUVWDShnhcul4gUVVERNJhEJoZmUyGN4d3wr+7iZ8L5t0pw4R1x5Bxm+OSEpF5YRCaIQsLGT56qisGdBAP0p2ZV4IJ644h706ZRJURETU+kwlCvj5RO1ZyC6wcG4hu3s6i9svZhXj520RoOPoMEZkJvj5h5m4VqfHU6jj8dUM8b+GL/dtg3rBOElVFRFR/fH2CaqSJnQJfTQ6Gm714KLY1+//GDyeuSlQVEVHjYRASWjjbYNU43dFn3vjhNBLT+FoFEZk2BiEBAIJau+DdSi/cq8sr8MJXCbh6q1iiqoiIGh6DkLTG9PBGeO/WoracwlJM3nAc+SXsSUpEpolBSCJvDu+Evm3dRG0XrhcgctMJlGk4qS8RmR4GIYlYyi0QMzYQbZvZi9oPXMrBWz9yHkMiMj0MQtLhZGOF9eFBOj1Jvz2WhhV/XJaoKiKihsEgpCp5udji8wndobQU/xFZEnsRm45ekagqIiL9YxBStbp5N8Gy0QGQid+qwJs7zmD36UxpiiIi0jOTCUIOsdYwQrs0x8IRnUVtggDM2pyEuMs5ElVFRKQ/HGKNauTT2Iv47PdLojZbhRxfTw6GqlUTiaoiIqoeh1gjvZr9aDuMDfYWtRWrNQhffwxnruVJVBURUf0xCKlGZDIZFo30w/AuzUXtBSXlmLDuGCf1JSKjxSCkGpNbyLB0dIDOPIY3i9QYu/YoUnOKqvkmEZHhYhBSrSgsLbB6nAq92riK2rMLSjF27VFc4wz3RGRkGIRUa9ZWcqyd2B2BlSb1vXb7DsZ+fgTZ+SXSFEZEVAcMQqoTO6Ul1j/fA509xT2xUnOLMXbtUeQWlkpUGRFR7TAIqc6cbKzw9eRgtHcXj0t6KbsQY9cexY0ChiERGT4GIdWLi50CGycHw8fNTtR+PqsAo/97GBl8ZkhEBo5BSPXWzNEam6YEo4Wzjaj975wiPL36MK7ksjcpERkuBiHphaezDTa/2BNeLuIwvHb7Dp5afRinr/KleyIyTAYXhOnp6RgwYAB8fX3RtWtXbNu2TeqSqIa8XGyx7aXeOnMZ3igoxajVcdgany5RZURE1TO4sUYzMzNx/fp1BAQEIDs7G4GBgbhw4QLs7Owe/mVwrFFDkFtYignrjuFsRr7Oumd7eOHtEZ1hbSWXoDIiMidGO9Zo8+bNERAQAABo1qwZXFxccPPmTWmLolpxtVfimxd6ooePi866b4+l46nVcfj7RqEElRER6ap1EO7fvx8jRoyAp6cnZDIZduzYobPNypUr4ePjA2tra6hUKhw4cKBOxR0/fhwVFRXw8vKq0/dJOk42Vtg0JRiT+vjorDtzLR+P/+cgth5Ph4HdkCAiM1TrICwqKoK/vz9WrFhR5fotW7Zg1qxZmD9/PhITE9GvXz+EhoYiLS1Nu41KpYKfn5/OJyMjQ7tNbm4uJkyYgDVr1tThxyJDYCW3wFsjfPHZs91gU+lWaLFag9e+O4Xp3yYi706ZRBUSEdXzGaFMJsP27dsRFhambQsODkZgYCBWrVqlbevUqRPCwsIQHR1do/2WlpZiyJAheOGFFzB+/PiHblta+s+L2/n5+fDy8uIzQgNzIasA0zYl4K8buq9SeDpZ49PRAehZafxSIqL6kOQZoVqtRkJCAkJCQkTtISEhiIuLq9E+BEFAeHg4Bg0a9NAQBIDo6Gg4OTlpP7yNapg6eDhg14x+eK7SnIYAkJFXgmc/P4IPfz4PdXmFBNURkTnTaxDm5ORAo9HA3d1d1O7u7o6srKwa7ePQoUPYsmULduzYgYCAAAQEBOD06dPVbj937lzk5eVpP+np7KJvqGwUcnzwZBesHqeCs62VaJ0gAKv3/YV/rzqEFE7nRESNyLIhdiqTyUTLgiDotFWnb9++qKio+VWBUqmEUqmsVX0krcf8PODv5YRXtp5E3F+5onVnruXj8c8O4L0n/fBkt5YSVUhE5kSvV4Rubm6Qy+U6V3/Z2dk6V4n6FhMTA19fXwQFBTXocUg/mjvZYOPkYMwf1gkKufiPYZFag9lbTuKVrSdRVFouUYVEZC70GoQKhQIqlQqxsbGi9tjYWPTu3Vufh9IRGRmJ5ORkxMfHN+hxSH8sLGR4oX8b7IjsozMaDQB8f+IqnlhxEBevF0hQHRGZi1oHYWFhIZKSkpCUlAQASElJQVJSkvb1iKioKKxduxbr1q3DuXPnMHv2bKSlpSEiIkKvhZPp8PV0xM7pfTC6u25Hp79uFGHkikP44cRVCSojInNQ69cn9u7di4EDB+q0T5w4ERs2bABw94X6xYsXIzMzE35+fli6dCn69++vl4KrExMTg5iYGGg0Gly8eJGvTxipH5OuYf72Myis4pYoh2cjotqo6esTBjfWaH1xrFHjl5pThMhvTlQ5VmmAlzNWj1PBw8lagsqIyJgY7VijRK3d7PD91N5VvnOYlH4bI1YcRMIVjj9LRPphMkHIXqOmxdrq7juHy8cEwFYhvhV6o6AUY9YcwbfH0qr5NhFRzfHWKBm881n5ePGrBKTdLNZZ91ywN94e4QulJZ8bEpEYb42SyejocbdXab92bjrrvjmahmfXHEF2fokElRGRKWAQklFwtlVgfXgQXuzfRmfdibTbePw/fG5IRHVjMkHIZ4Smz1JugXnDOuGzZ7vB2kr8Rzf7/58bbjp6hXMcElGt8BkhGaXkjHy8+PVxXL11R2fd6O5eeGck3zckMnd8RkgmzdfTEf+b3rfK54ZbjqdjzJojuFFQWsU3iYjEGIRktJrYKbDh+R6I+NcjOuuS0m8jLOYQzmfpvpRPRHQ/BiEZNbmFDG+EdsSK57rBptKt0Gu372DUyjj8eT5bouqIyBiYTBCys4x5e7yrJ7ZH9kbLJjai9iK1BpO/jMfqfX+xEw0RVYmdZcik5BSW4sWvjuNE2m2ddY919sDHT3eFg7VV4xdGRI2OnWXILLnZK/HNCz3xhL+nzrpfzmYhLOYQLmdzfkMi+geDkEyOtZUcy8cE4JUh7SGTidfdm9/wj/PXpSmOiAwOg5BMkkwmw4zB7bA+PAhONuJboXefGx7H5/v/5nNDImIQkmkb0KEZds3oi86e4ucDggC8v/scXvvuFErLNRJVR0SGwGSCkL1GqTpeLrb4fmpvhAXoPjfclnAV49cew80itQSVEZEhYK9RMhuCIGDl3r/w8Z4LOuu8XWyxLrw72jZzkKAyImoI7DVKVIlMJkPkwLZYPU6l8/J92s1iPLkyDvsv3pCoOiKSCoOQzM5jfh74bmovNHeyFrUXlJQjfP0xxPx5GRUVJnWjhIgegEFIZqmzpxN+jOwD/5ZOovYKAfh4zwVM/jIet4v53JDIHDAIyWw1c7TGlpd6YXjX5jrr/rxwA8M/O4iT6bcbvzAialQMQjJr1lZy/GdMN7w6tAMsKr18f+32HTy1Og5fxqXyfUMiE2YyQcjXJ6iuLCzudqLZOCUYbvYK0boyjYC3d57FjG8TUVhaLlGFRNSQ+PoE0X2u55dgxjeJOJZ6U2ddGzc7rJmg4isWREaCr08Q1YG7ozW+eSEYL/2rjc66v3OKEBYTh9+SOU4pkSlhEBJVYim3wNzQTvh8Qnc4WluK1hWWluOFr48j5s/LfG5IZCIYhETVGOLrjp9e7ge/FrrjlH685wJe3pzEcUqJTACDkOgBvFxs8V1Eb4ysYpzS/53MQPi6eOSXlElQGRHpC4OQ6CGsreRYNjoAc0M76sxvePjvXDyz+jCu55dIUxwR1RuDkKgGZDIZXvrXI1gXHgQ7hXic0vNZBfj3yjik5BRJVB0R1QeDkKgWBnZohs0v9tJ53/Da7Tt4ds0RXMllGBIZGwYhUS11aemEH6b2gY+bnag9K78Ez645gvSbxRJVRkR1wSAkqgNvV1t8F9ELXVqIB+3OyCvBGIYhkVExmSDkEGvU2Fztlfh6cg909hS/XnHt9h08t/YIstmBhsgocIg1onq6VaTGc2uP4lxmvqi9o4cDtrzUC042VhJVRmTeOMQaUSNpYqfApinB6OghHoP0fFYBXvjyOErK+NI9kSFjEBLpgYudAl9PDkZrV1tR+7HUm5j+TSLKNRUSVUZED8MgJNKTpg5KfD05GE0dlKL2385dx9s7z3JsUiIDxSAk0iMvF1t8NakHHCoN1r3paBo2xKVKUxQRPRCDkEjPOjV3xBcTg6C0FP/1endXMvZeyJaoKiKqDoOQqAH08HHBkmf8RW0VAjDjm0Rcul4gUVVEVBUGIVEDebyrJ2Y92k7UVlBajklfxuNmkVqiqoioMgYhUQOaObgdHu/aXNSWfvMOpm1KQBl7khIZBAYhUQOSyWT45Gl/+LcUD8V25O+bWLjzrERVEdH9GIREDczaSo7PJ3SHu6P4tYpNR9Pw9eFUaYoiIi2DC8KCggIEBQUhICAAXbp0weeffy51SUT11szRGp9P6K7Tk3Th/5IRdzlHoqqICDDAsUY1Gg1KS0tha2uL4uJi+Pn5IT4+Hq6urjX6PscaJUP2Y9I1zNycJGpztVNg98x+cHe0lqYoIhNltGONyuVy2NreHaaqpKQEGo2GI3KQyRgZ0AKRAx8RteUWqTGDw7ARSabWQbh//36MGDECnp6ekMlk2LFjh842K1euhI+PD6ytraFSqXDgwIFaHeP27dvw9/dHy5Yt8dprr8HNza22ZRIZrFeGdMDgjs1EbcdSb+LT2IsSVURk3modhEVFRfD398eKFSuqXL9lyxbMmjUL8+fPR2JiIvr164fQ0FCkpaVpt1GpVPDz89P5ZGRkAACcnZ1x8uRJpKSk4JtvvsH169fr+OMRGR4LCxmWPOOPFs42ovaVe//Cnxx5hqjR1esZoUwmw/bt2xEWFqZtCw4ORmBgIFatWqVt69SpE8LCwhAdHV3rY0ydOhWDBg3C008/XeX60tJSlJaWapfz8/Ph5eXFZ4Rk8BLTbuGZ/x5Gmeafv4LOtlb46eV+OiFJRLUnyTNCtVqNhIQEhISEiNpDQkIQFxdXo31cv34d+fl3JzjNz8/H/v370aFDh2q3j46OhpOTk/bj5eVV9x+AqBF1826CuaGdRG23i8swbdMJlJZzDkOixqLXIMzJyYFGo4G7u7uo3d3dHVlZWTXax9WrV9G/f3/4+/ujb9++mD59Orp27Vrt9nPnzkVeXp72k56eXq+fgagxPd+nNR7r7CFqO5l+G+//dE6iiojMj+XDN6k9mUwmWhYEQaetOiqVCklJSTU+llKphFKpfPiGRAZIJpNh8dNdcT4rH6m5xdr2rw5fQaB3E4R1ayFhdUTmQa9XhG5ubpDL5TpXf9nZ2TpXifoWExMDX19fBAUFNehxiPTN0doKq8apYG0l/uv4xg+ncD4rX6KqiMyHXoNQoVBApVIhNjZW1B4bG4vevXvr81A6IiMjkZycjPj4+AY9DlFD6NTcEe+HdRG1lZRVIOLrBOTdKZOoKiLzUOsgLCwsRFJSkvb2ZUpKCpKSkrSvR0RFRWHt2rVYt24dzp07h9mzZyMtLQ0RERF6LZzI1IxStcTYYG9RW2puMWZtTkRFBQeVIGootX5GePz4cQwcOFC7HBUVBQCYOHEiNmzYgNGjRyM3NxeLFi1CZmYm/Pz8sHv3brRq1Up/VVchJiYGMTEx0GjY246M11sjfHHmWh5OXs3Ttv154QaW/XYRUSHV954morozuLFG64tjjZKxy7h9ByP+cxC5lSbvXT1Ohcf8PKr5FhFVZrRjjRKZO09nG8SMDYTcQtzT+pWtSThzLa+abxFRXZlMELLXKJmSnm1c8eZw8cv2RWoNnv38CBKu3JKoKiLTxFujRAZKEAS8su0kfjhxTdRuq5Dji4lB6PVIzaYmIzJXvDVKZORkMhk+eLILelcKvGK1BuHrj2HfxRsSVUZkWhiERAbM2kqOdeFBGNihqai9tLwCL3x1HPsZhkT1ZjJByGeEZKqsreT47/juCK3UY1T9/2F46HKORJURmQY+IyQyEuWaCszZdhI7kjJE7dZWFtjwfA/0bMNnhkT34zNCIhNjKbfAkmcCMDLAU9ReUlaBSRvikZjG3qREdcEgJDIicgsZljztj+Fdm4vai9UavPR1Aq7nl0hUGZHxYhASGRlLuQWWjQ7Qmccwu6AUERsTOKkvUS2ZTBCyswyZEyu5BT57thv6tBU/F0xMu423dpyFiT36J2pQ7CxDZMRuFanxRMxBpN+8I2p/d2RnjO/VWpqiiAwEO8sQmYEmdgqsGd8dNlZyUfs7/0vmuKRENcQgJDJynZo74pOn/UVt5RUC5mw7yeeFRDXAICQyAcO7NkfEvx4RtZ3PKsDy3y5JVBGR8WAQEpmIqCHt0am5+DnI6n1/8f1CoocwmSBkr1EydwpLCyx52h9W8n/mMawQgFe2nURJGW+RElWHvUaJTMx/fr+EJbEXRW1927rh09H+aOZgLVFVRI2PvUaJzNTUAY+ga0snUdvByzkYtvwAZ6sgqgKDkMjEWMrv3iK1thL/9c4pVGPCumN4d1cyCkrKJKqOyPAwCIlMUDt3B3w9ORjujkqddV8cTMHAT/Zia3w6KipM6skIUZ0wCIlMVFBrF/w8sz8Gd2ymsy6nUI3Xvj+FJ2IOYt/FGxySjcwaO8sQmThBELD+UCo+/Pk81JqKKrfp5u2MWY+2R/92bpDJZFVuQ2RszK6zDF+fIKqaTCbDpL4+2DO7Px7tpHt1CNwdrHviumMYs+YIMm7fqXIbIlPFK0IiM7P/4g0s2pWMy9mFVa53trXC0mcCMLCKW6pExsTsrgiJqGb6t2+Kn2f2w+KnusLbxVZn/e3iMjy/IR4f/XIe5dXcSiUyJbwiJDJjZZoKbE+8huW/XcK1Km6J9mrjipVjA9HETiFBdUT1wytCInooK7kFnunuhZ9n9dOZ8R4ADv+di7CVh3A5u0CC6ogaB4OQiOBobYVV4wLx9ghf0VilAHAltxhPxsThz/PZElVH1LAYhEQE4G7v0uf7+GDrS73QzEH8In5BaTme3xCPGd8mIv1msUQVEjUMBiERiXTzboKd0/vqjFcKAP87mYHBS/bhvV3JuF2slqA6Iv1jEBKRDg8na2x9qRdG+HvqrFNrKrD2YAr6Lf4TMX9eRrG6XIIKifSHQUhEVbK2kuOzMQF4e4QvHK0tddYXlJTj4z0X0H/xXnx95Ao0HLeUjBRfnyCih7pVpMaKPy/jq8OpKNNU/StD1aoJPn6qK9o0tW/k6oiqZnavT3CINaKG08ROgQWP++L3qAF4oorbpQCQcOUWQpcfwBcHUzirBRkVXhESUa2dzcjDJ3su4M8LVU/0G9S6CT552h+tXO0auTKif5jdFSERNZ7Onk5Y/3wPfPNCMFq76g7TFp96C48tO4CvD6fy6pAMHoOQiOqs9yNu2D2zH8J7t9ZZd6dMgwU/nsX4dUeRXVDS+MUR1RCDkIjqxVZhiYVPdMbmF3vCy8VGZ/2hy7l4evVhvohPBotBSER60bONK36Z2R/jenrrrLuSW4xn/nsYf92oeuonIikxCIlIb+yUlngvrAu+ntwDzZ2sResy80rwzOrDOJuRJ1F1RFVjEBKR3vVr1xQ/Tu+Djh4OovbcIjXGrDmCM9cYhmQ4GIRE1CCaOVhj84s94e/lLGovKCnH+C+O4nxWvjSFEVXCICSiBuNsq8CmKcHo2cZF1H6ruAzj1h7F5Ww+MyTpMQiJqEHZKy2x4fke6P2Iq6g9p1CNsWuP8JkhSY5BSEQNztpKjrUTuyOodRNR+/X8Ujz+n4OYvSWJr1eQZAw2CIuLi9GqVSvMmTNH6lKISA9sFZZYFx6EgErPDAUB2J54DYOW7EX07nMo11RIUyCZLYMNwvfffx/BwcFSl0FEeuRgbYUvJ/WoctLfMo2A/+7/GzO+TYS6nGFIjccgg/DSpUs4f/48hg0bJnUpRKRnTjZW2PpSL8wJaQ97pe48hz+fycLUjQkoKdNIUB2Zo1oH4f79+zFixAh4enpCJpNhx44dOtusXLkSPj4+sLa2hkqlwoEDB2p1jDlz5iA6Orq2pRGRkbC2kmP6oHbY/9pATO7rA4Vc/Kvo9/PZeOGr47ijZhhSw6t1EBYVFcHf3x8rVqyocv2WLVswa9YszJ8/H4mJiejXrx9CQ0ORlpam3UalUsHPz0/nk5GRgR9//BHt27dH+/bt6/5TEZFRcPn/eQ6/nNQDtgq5aN2BSzmYtimBs1dQg6vXfIQymQzbt29HWFiYti04OBiBgYFYtWqVtq1Tp04ICwur0VXe3LlzsXHjRsjlchQWFqKsrAyvvPIK3nrrrSq3Ly0tRWlpqXY5Pz8fXl5enI+QyMgkXLmJ8HXxKCgtF7W/80RnTKxidguih5FkPkK1Wo2EhASEhISI2kNCQhAXF1ejfURHRyM9PR2pqan45JNP8MILL1Qbgve2d3Jy0n68vLzq9TMQkTRUrVywcUowHK3Fzw2jfz6HvzlYNzUgvQZhTk4ONBoN3N3dRe3u7u7IysrS56G05s6di7y8PO0nPT29QY5DRA3P38sZK8eqRG0lZRWI2nqSr1VQg2mQXqMymUy0LAiCTltNhIeH45NPPnngNkqlEo6Ojvj666/Rs2dPDB48uNbHISLD0bedGyb0aiVqS0q/jf/u/1uiisjU6TUI3dzcIJfLda7+srOzda4S9S0yMhLJycmIj49v0OMQUcN7I7QjfNzsRG1LYy9yODZqEHoNQoVCAZVKhdjYWFF7bGwsevfurc9DEZEJs1VYYskz/rC470ZSeYWAd3Ymox79+4iqVOsgLCwsRFJSEpKSkgAAKSkpSEpK0r4eERUVhbVr12LdunU4d+4cZs+ejbS0NEREROi1cCIybYHeTTB1wCOitmOpN7HnbMP0NyDzpTusw0McP34cAwcO1C5HRUUBACZOnIgNGzZg9OjRyM3NxaJFi5CZmQk/Pz/s3r0brVq1qm6XehETE4OYmBhoNHwBl8hUzBjUDj8mZeDqrTvatuifz2NQR3coLA1yYCwyQvV6j9AQ1fS9ESIyDrtOZWD6N4mitjeHd8KUfm0kqoiMhSTvERIR6dvwLs0R6O0savvs90u4VaSWpiAyOSYThDExMfD19UVQUJDUpRCRHslkMrz5uK+oLb+kHEtiL7DjDOkFb40SkVGY8W0i/ncyQ9T2aKdm+ODfXeBqp8Thv3Lx0+kM3CoqQ6fmjgjp7I6OHg51eoeZTENN84BBSERGIf1mMQZ/uk9nrkJnWyvYWsmRkVei8x0vFxuM6OqJiAGPwNHaqrFKJQPBZ4REZFK8XGwxN7SjTvvt4rIqQxAA0m/ewcq9f2HkikO4kFXQ0CWSkTKZIOQzQiLT93wfH6wZr4KbvbJW30vJKcKTKw9h16mMh29MZoe3RonI6NwqUuOtnWd1nhn6NneEv5cz9l+8gWu371T53cl9ffDq0A6wtpJXuZ5MB58RMgiJTN7v565j58kMeDhZY6R/C/h63v07LwgCzmbkY9720zh1VXd80jZudvjoqa4Iau3S2CVTI2IQMgiJzF5JmQZv/XgGW49frXL9hF6t8EZoR9gqaj3IFhkBdpYhIrNnbSXHR6O64oMnu0Ah1/1199XhK5i0IR5lnOvQrJlMELKzDBFVRSaT4blgb+x6uS8CvJx11h/5+yaid59v/MLIYPDWKBGZDU2FgPWHUvDJrxdQUia+Clw+JgAjA1pIVBk1BN4aJSKqRG4hw5R+bbDlxV46t0rf+P40zmflS1QZSYlBSERmx9/LGYtGdha13SnTIOLrBOSXlElUFUmFQUhEZmlMD28828NL1JaaW4wVf1yWqCKSCoOQiMzWwic6w7+lk6jt68NXkFNYKlFFJAWTCUL2GiWi2lJayrF8TDfILf6ZoeJOmQafH/hbwqqosZlMEEZGRiI5ORnx8fFSl0JERqS1mx3CKvUW/frwFeTyqtBsmEwQEhHV1fRBbXHfRSGK1Rp8fiBFuoKoUTEIicjs+bjZIayb+Krwq8OpuFmklqgiakwMQiIiADMGtdO5KlzLZ4VmgUFIRIS7V4WVR5b5Mi4V2QVVT/pLpoNBSET0/yo/KyxSa/DJngvSFUSNwmSCkK9PEFF9PdLUHqMCW4ratiVcxekq5jQk08FBt4mI7pOdX4KBn+xFkVqjbeveqgm2RfSCTCZ7wDfJ0HDQbSKiOmjmaI3IQW1Fbcev3MKuU5kSVUQNjUFIRFTJpD4+8HKxEbV9+PN5lJRpqvkGGTMGIRFRJdZWcswf1knUdu32Haw7xJfsTRGDkIioCkM7e6BXG1dR24ZDqVCXV1TzDTJWDEIioirIZDLMq3RVmF1Qip/P8FmhqWEQEhFVo0tLJ/Ro7SJq2xCXKk0x1GAYhEREDxDep7VoOTHtNpLSb0tSCzUMBiER0QOE+LrD08la1LaBnWZMCoOQiOgBLOUWGN+rtajtp9OZyM7nGKSmwmSCkEOsEVFDGRPkBaXlP78uyzQCNh5Nk7Ai0ieTCULOUE9EDaWJnQJPVpqv8JujV1BazhfsTYHJBCERUUOq3Gkmp1CNX89el6YY0isGIRFRDXT0cESwj/hViq3H0yWqhvSJQUhEVENjeniJlg9ezsHVW8USVUP6wiAkIqqhUL/mcLC21C4LAvBdwlUJKyJ9YBASEdWQtZUcT/h7itq2Hb+KigqTmtbV7DAIiYhqYXSQ+Pbotdt3cOivHImqIX1gEBIR1UKXFk7o6OEgatsSz04zxoxBSERUCzKZTOeq8Nez13GrSC1RRVRfDEIioloKC2gBhfyfX59qTQV2JF2TsCKqDwYhEVEtNbFTIKSzu6ht58kMiaqh+jLIILS0tERAQAACAgIwZcoUqcshItIxStVStHzqah4KSsokqobqw/LhmzQ+Z2dnJCUlSV0GEVG1evq4QiG3gFpTAQDQVAg4lnITgzu5P+SbZGgM8oqQiMjQ2Sjk6ObtLGqL+ytXmmKoXmodhPv378eIESPg6ekJmUyGHTt26GyzcuVK+Pj4wNraGiqVCgcOHKjVMfLz86FSqdC3b1/s27evtiUSETWK3o+4iZYZhMap1kFYVFQEf39/rFixosr1W7ZswaxZszB//nwkJiaiX79+CA0NRVraP3N3qVQq+Pn56XwyMu4+bE5NTUVCQgJWr16NCRMmID8/v44/HhFRw+nd1lW0fC4zHzf5GoXRkQmCUOexgWQyGbZv346wsDBtW3BwMAIDA7Fq1SptW6dOnRAWFobo6OhaHyM0NBTvvvsuunfvXuX60tJSlJaWapfz8/Ph5eWFvLw8ODo61vp4REQ1pS6vgP87v+JO2T/zEq4cG4hhXZpLWBXdk5+fDycnp4fmgV6fEarVaiQkJCAkJETUHhISgri4uBrt49atW9pgu3r1KpKTk9GmTZtqt4+OjoaTk5P24+XlVe22RET6pLC0QI9KUzMduszh1oyNXoMwJycHGo0G7u7iXlPu7u7Iysqq0T7OnTuH7t27w9/fH48//jiWL18OFxeXarefO3cu8vLytJ/0dA51RESNp/cj4tujh/mc0Og0yOsTMplMtCwIgk5bdXr37o3Tp0/X+FhKpRJKpRIxMTGIiYmBRqN5+JeIiPSkcoeZv3OKkJl3B82dbCSqiGpLr1eEbm5ukMvlOld/2dnZOleJ+hYZGYnk5GTEx8c36HGIiO7n6+kIR2vxNQWvCo2LXoNQoVBApVIhNjZW1B4bG4vevXvr81BERAZBbiFDzzbi26N8jcK41PrWaGFhIS5fvqxdTklJQVJSElxcXODt7Y2oqCiMHz8e3bt3R69evbBmzRqkpaUhIiJCr4VXxlujRCSV3o+44tfk69rluMs5tXokRNKq9esTe/fuxcCBA3XaJ06ciA0bNgC4+0L94sWLkZmZCT8/PyxduhT9+/fXS8EPU9PuskRE+nLpegGGLN0vats7ZwBau9lJVBEBNc+Der1HaIgYhETU2ARBQND7vyOn8J93mpeNDkBYtxYSVkWSvEdIRGSOZDIZOnjYi9puFJRWszUZGpMJwpiYGPj6+iIoKEjqUojIDLnYKUXLuRxqzWiYTBDy9QkikpKrnUK0fLOIV4TGwmSCkIhISpWDMLeQV4TGgkFIRKQHLvaVgpC3Ro2GyQQhnxESkZRcdZ4R8taosTCZIOQzQiKSkmulK8KbvDVqNEwmCImIpFT5GWGRWoOSMo50ZQwYhEREelD51ijA54TGgkFIRKQHjjaWsLQQjy3K26PGwWSCkJ1liEhKMpkMLpVuj+aww4xRMJkgZGcZIpJa5SDkFaFxMJkgJCKSmpu9+DnhTT4jNAoMQiIiPeGtUePEICQi0hO+S2icTCYI2VmGiKSmM94ob40aBZMJQnaWISKpcSom42QyQUhEJDWdW6N8RmgUGIRERHrCqZiME4OQiEhPXCu9PlGs1uCOmuONGjoGIRGRnlR+fQLgdEzGgEFIRKQnjtaWsJJXGm+UHWYMHoOQiEhPqhpvlD1HDZ/JBCHfIyQiQ6DzCgU7zBg8kwlCvkdIRIbAja9QGB2TCUIiIkOgc2uUV4QGj0FIRKRHlWeq5zNCw8cgJCLSI93RZRiEho5BSESkR7q3RvmM0NAxCImI9IgzUBgfBiERkR5VvjXKzjKGj0FIRKRHlTvL3CnToFhdLlE1VBMMQiIiPXKxr2K8UV4VGjSTCUKOLENEhsBBaQmFXPyrlT1HDZvJBCFHliEiQ1DVeKMMQsNmMkFIRGQoKgdhDl+hMGgMQiIiPeNL9caFQUhEpGd8l9C4MAiJiPTM1Z5TMRkTBiERkZ7pdpbhM0JDxiAkItIz3ho1LgxCIiI9461R48IgJCLSM0drS9FyYSmHWDNkDEIiIj2zsJBJXQLVAoOQiIjMGoOQiIjMmkEGYUpKCgYOHAhfX1906dIFRUVFUpdEREQmyvLhmzS+8PBwvPfee+jXrx9u3rwJpVL58C8RERHVgcEF4dmzZ2FlZYV+/foBAFxcXCSuiIiITFmtb43u378fI0aMgKenJ2QyGXbs2KGzzcqVK+Hj4wNra2uoVCocOHCgxvu/dOkS7O3t8cQTTyAwMBAffPBBbUskIiKqsVpfERYVFcHf3x/PP/88Ro0apbN+y5YtmDVrFlauXIk+ffrgv//9L0JDQ5GcnAxvb28AgEqlQmmp7pBDv/76K8rKynDgwAEkJSWhWbNmeOyxxxAUFIQhQ4ZUWU9paaloX/n5+bX9kYiIyIzVOghDQ0MRGhpa7fpPP/0UkydPxpQpUwAAy5Ytw549e7Bq1SpER0cDABISEqr9fsuWLREUFAQvLy8AwLBhw5CUlFRtEEZHR+Odd96p7Y9BREQEQM+9RtVqNRISEhASEiJqDwkJQVxcXI32ERQUhOvXr+PWrVuoqKjA/v370alTp2q3nzt3LvLy8rSf9PT0ev0MRERkXvTaWSYnJwcajQbu7u6idnd3d2RlZdWsIEtLfPDBB+jfvz8EQUBISAgef/zxardXKpXsVUpERHXWIL1GZTLx8EKCIOi0PcjDbr9WJSYmBjExMdBoNLX6HhERmTe93hp1c3ODXC7XufrLzs7WuUrUt8jISCQnJyM+Pr5Bj0NERKZFr0GoUCigUqkQGxsrao+NjUXv3r31eSgiIiK9qPWt0cLCQly+fFm7nJKSgqSkJLi4uMDb2xtRUVEYP348unfvjl69emHNmjVIS0tDRESEXguvjLdGiYioLmodhMePH8fAgQO1y1FRUQCAiRMnYsOGDRg9ejRyc3OxaNEiZGZmws/PD7t370arVq30V3UVIiMjERkZifz8fDg5OTXosYiIyHTUOggHDBgAQRAeuM20adMwbdq0OhdVH/dq44v1RCSVwoJ8VJQWa5c1Mkv+TpLAvXP+sMwyuLFG66ugoAAAtC/kExEZAqcPpa7AfBUUFDzwTqFMeFhUGpmKigpkZGTAwcGhVq9sVJafnw8vLy+kp6fD0dFRjxUaP56b6vHcVI/n5sF4fqpX13MjCAIKCgrg6ekJC4vq+4aa3BWhhYUFWrZsqbf9OTo68g9lNXhuqsdzUz2emwfj+aleXc5NTfqMGOTEvERERI2FQUhERGaNQVgNpVKJt99+m+OYVoHnpno8N9XjuXkwnp/qNfS5MbnOMkRERLXBK0IiIjJrDEIiIjJrDEIiIjJrDEIiIjJrDMIqrFy5Ej4+PrC2toZKpcKBAwekLqnRRUdHIygoCA4ODmjWrBnCwsJw4cIF0TaCIGDhwoXw9PSEjY0NBgwYgLNnz0pUsXSio6Mhk8kwa9YsbZs5n5tr165h3LhxcHV1ha2tLQICApCQkKBdb87npry8HG+++SZ8fHxgY2ODNm3aYNGiRaioqNBuYy7nZ//+/RgxYgQ8PT0hk8mwY8cO0fqanIfS0lLMmDEDbm5usLOzwxNPPIGrV6/WvhiBRDZv3ixYWVkJn3/+uZCcnCzMnDlTsLOzE65cuSJ1aY1q6NChwvr164UzZ84ISUlJwvDhwwVvb2+hsLBQu82HH34oODg4CN9//71w+vRpYfTo0ULz5s2F/Px8CStvXMeOHRNat24tdO3aVZg5c6a23VzPzc2bN4VWrVoJ4eHhwtGjR4WUlBTht99+Ey5fvqzdxlzPjSAIwnvvvSe4uroKu3btElJSUoRt27YJ9vb2wrJly7TbmMv52b17tzB//nzh+++/FwAI27dvF62vyXmIiIgQWrRoIcTGxgonTpwQBg4cKPj7+wvl5eW1qoVBWEmPHj2EiIgIUVvHjh2FN954Q6KKDEN2drYAQNi3b58gCIJQUVEheHh4CB9++KF2m5KSEsHJyUlYvXq1VGU2qoKCAqFdu3ZCbGys8K9//UsbhOZ8bl5//XWhb9++1a4353MjCIIwfPhwYdKkSaK2f//738K4ceMEQTDf81M5CGtyHm7fvi1YWVkJmzdv1m5z7do1wcLCQvjll19qdXzeGr2PWq1GQkICQkJCRO0hISGIi4uTqCrDkJeXBwBwcXEBcHdC5qysLNG5UiqV+Ne//mU25yoyMhLDhw/Ho48+Kmo353Ozc+dOdO/eHU8//TSaNWuGbt264fPPP9euN+dzAwB9+/bF77//josXLwIATp48iYMHD2LYsGEAeH7uqcl5SEhIQFlZmWgbT09P+Pn51fpcmdyg2/WRk5MDjUYDd3d3Ubu7uzuysrIkqkp6giAgKioKffv2hZ+fHwBoz0dV5+rKlSuNXmNj27x5M06cOIH4+HiddeZ8bv7++2+sWrUKUVFRmDdvHo4dO4aXX34ZSqUSEyZMMOtzAwCvv/468vLy0LFjR8jlcmg0Grz//vt49tlnAZj3n5371eQ8ZGVlQaFQoEmTJjrb1Pb3NYOwCpWnbxIEoV5TOhm76dOn49SpUzh48KDOOnM8V+np6Zg5cyZ+/fVXWFtbV7udOZ6biooKdO/eHR988AEAoFu3bjh79ixWrVqFCRMmaLczx3MDAFu2bMHGjRvxzTffoHPnzkhKSsKsWbPg6emJiRMnarcz1/NTWV3OQ13OFW+N3sfNzQ1yuVznXxPZ2dk6/zIxFzNmzMDOnTvx559/iqa38vDwAACzPFcJCQnIzs6GSqWCpaUlLC0tsW/fPnz22WewtLTU/vzmeG6aN28OX19fUVunTp2QlpYGwLz/3ADAq6++ijfeeANjxoxBly5dMH78eMyePRvR0dEAeH7uqcl58PDwgFqtxq1bt6rdpqYYhPdRKBRQqVSIjY0VtcfGxqJ3794SVSUNQRAwffp0/PDDD/jjjz/g4+MjWu/j4wMPDw/RuVKr1di3b5/Jn6vBgwfj9OnTSEpK0n66d++OsWPHIikpCW3atDHbc9OnTx+d12wuXryIVq1aATDvPzcAUFxcrDNBrFwu174+Ye7n556anAeVSgUrKyvRNpmZmThz5kztz1WduviYsHuvT3zxxRdCcnKyMGvWLMHOzk5ITU2VurRGNXXqVMHJyUnYu3evkJmZqf0UFxdrt/nwww8FJycn4YcffhBOnz4tPPvssybZzbsm7u81Kgjme26OHTsmWFpaCu+//75w6dIlYdOmTYKtra2wceNG7Tbmem4EQRAmTpwotGjRQvv6xA8//CC4ubkJr732mnYbczk/BQUFQmJiopCYmCgAED799FMhMTFR+6paTc5DRESE0LJlS+G3334TTpw4IQwaNIivT+hLTEyM0KpVK0GhUAiBgYHaVwbMCYAqP+vXr9duU1FRIbz99tuCh4eHoFQqhf79+wunT5+WrmgJVQ5Ccz43//vf/wQ/Pz9BqVQKHTt2FNasWSNab87nJj8/X5g5c6bg7e0tWFtbC23atBHmz58vlJaWarcxl/Pz559/Vvk7ZuLEiYIg1Ow83LlzR5g+fbrg4uIi2NjYCI8//riQlpZW61o4DRMREZk1PiMkIiKzxiAkIiKzxiAkIiKzxiAkIiKzxiAkIiKzxiAkIiKzxiAkIiKzxiAkMnNVzQ5OZE4YhEQSCg8Ph0wm0/k89thjUpdGZDY4DRORxB577DGsX79e1KZUKiWqhsj88IqQSGJKpRIeHh6iz73JRmUyGVatWoXQ0FDY2NjAx8cH27ZtE33/9OnTGDRoEGxsbODq6ooXX3wRhYWFom3WrVuHzp07Q6lUonnz5pg+fbpofU5ODp588knY2tqiXbt22Llzp3bdrVu3MHbsWDRt2hQ2NjZo166dTnATGTMGIZGBW7BgAUaNGoWTJ09i3LhxePbZZ3Hu3DkAd6f1eeyxx9CkSRPEx8dj27Zt+O2330RBt2rVKkRGRuLFF1/E6dOnsXPnTrRt21Z0jHfeeQfPPPMMTp06hWHDhmHs2LG4efOm9vjJycn4+eefce7cOaxatQpubm6NdwKIGlr9xxAnorqaOHGiIJfLBTs7O9Fn0aJFgiDcnQUkIiJC9J3g4GBh6tSpgiAIwpo1a4QmTZoIhYWF2vU//fSTYGFhIWRlZQmCIAienp7C/Pnzq60BgPDmm29qlwsLCwWZTCb8/PPPgiAIwogRI4Tnn39ePz8wkQHiM0IiiQ0cOBCrVq0Stbm4uGj/u1evXqJ1vXr1QlJSEgDg3Llz8Pf3h52dnXZ9nz59UFFRgQsXLkAmkyEjIwODBw9+YA1du3bV/rednR0cHByQnZ0NAJg6dSpGjRqFEydOICQkBGFhYWY1SSyZPgYhkcTs7Ox0blU+jEwmAwAIgqD976q2sbGxqdH+rKysdL57b9b00NBQXLlyBT/99BN+++03DB48GJGRkfjkk09qVTORoeIzQiIDd+TIEZ3ljh07AgB8fX2RlJSEoqIi7fpDhw7BwsIC7du3h4ODA1q3bo3ff/+9XjU0bdoU4eHh2LhxI5YtW4Y1a9bUa39EhoRXhEQSKy0tRVZWlqjN0tJS2yFl27Zt6N69O/r27YtNmzbh2LFj+OKLLwAAY8eOxdtvv42JEydi4cKFuHHjBmbMmIHx48fD3d0dALBw4UJERESgWbNmCA0NRUFBAQ4dOoQZM2bUqL633noLKpUKnTt3RmlpKXbt2oVOnTrp8QwQSYtBSCSxX375Bc2bNxe1dejQAefPnwdwt0fn5s2bMW3aNHh4eGDTpk3w9fUFANja2mLPnj2YOXMmgoKCYGtri1GjRuHTTz/V7mvixIkoKSnB0qVLMWfOHLi5ueGpp56qcX0KhQJz585FamoqbGxs0K9fP2zevFkPPzmRYZAJgiBIXQQRVU0mk2H79u0ICwuTuhQik8VnhEREZNYYhEREZNb4jJDIgPHJBVHD4xUhERGZNQYhERGZNQYhERGZNQYhERGZNQYhERGZNQYhERGZNQYhERGZNQYhERGZNQYhERGZtf8DIdcOxd56Qv0AAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 500x500 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Display training suboptimality.\n",
    "\n",
    "fig, ax = plt.subplots(1, 1, figsize=(5, 5))\n",
    "\n",
    "t = np.arange(len(train_losses))\n",
    "\n",
    "minimum_loss = output.fun\n",
    "train_losses = np.array(train_losses)\n",
    "subopt = (train_losses - minimum_loss) / (train_losses[0] - minimum_loss)\n",
    "\n",
    "ax.plot(np.arange(len(subopt)), subopt, linewidth=3, label=r\"Train Suboptimality\")\n",
    "ax.set_xlabel(\"Epochs\")\n",
    "\n",
    "ax.legend()\n",
    "ax.set_yscale(\"log\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Compute the Fairness Metrics "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test SMD: 0.036091131113593446\n"
     ]
    }
   ],
   "source": [
    "# Test SMD for the law school regression task \n",
    "group_label1 = 0\n",
    "group_label2 = 1\n",
    "y_pred = torch.matmul(val_obj.X, optimizer.weights.clone().detach())\n",
    "y_pred_mean = torch.mean(y_pred)\n",
    "y_pred_g1 = torch.mean(y_pred[group_test==group_label1])\n",
    "y_pred_g2 = torch.mean(y_pred[group_test==group_label2])\n",
    "print(\"Test SMD:\", torch.abs(y_pred_g1-y_pred_mean).item() + torch.abs(y_pred_g2-y_pred_mean).item())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Test EO for the acs employment classification task\n",
    "# group_label1 = 1\n",
    "# group_label2 = 2\n",
    "# logits = torch.matmul(val_obj.X, optimizer.weights.clone().detach())\n",
    "# y = logits > 0\n",
    "# y = torch.where(y, torch.tensor(1), torch.tensor(0))\n",
    "# eo_1 = torch.mean(y[(val_obj.y == 1) & (group_test == group_label1)].float())\n",
    "# eo_2= torch.mean(y[(val_obj.y == 1) & (group_test == group_label2)].float())\n",
    "# print(\"Test Equal opportunity:\", (eo_1-eo_2).item())"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "lrm",
   "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.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
