{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "05ac303a",
   "metadata": {},
   "outputs": [],
   "source": [
    "import pandas as pd\n",
    "import numpy as np\n",
    "from matplotlib import pyplot as plt\n",
    "import seaborn as sb\n",
    "import scipy.stats\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "26d0c49c",
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_paths_rf(dset, fair, meth):\n",
    "    rf_test = './{}_{}/rf_10-9_10_{}/'.format(dset, fair, meth)\n",
    "    rf_train = './{}_{}/train_rf_10-9_10_{}/'.format(dset, fair, meth)\n",
    "    rf_test_b = './{}_{}/baseline_rf_13-9_25_perc/'.format(dset, fair)\n",
    "    rf_train_b = './{}_{}/baseline_train_rf_13-9_25_perc/'.format(dset, fair)\n",
    "    lam_test = './{}_{}/rf_10-9_10_{}/'.format(dset, fair, 'lambda')\n",
    "    lam_train = './{}_{}/train_rf_10-9_10_{}/'.format(dset, fair, 'lambda')\n",
    "    return rf_test, rf_train, rf_test_b, rf_train_b, lam_test, lam_train"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "5be5a7f2",
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_paths_dt(dset, fair, meth):\n",
    "    test = './{}_{}/dt_9-9_10_{}/'.format(dset, fair, meth)\n",
    "    train = './{}_{}/train_dt_9-9_10_{}/'.format(dset, fair, meth)\n",
    "    test_b = './{}_{}/baseline_dt_13-9_10_perc/'.format(dset, fair)\n",
    "    train_b = './{}_{}/baseline_train_dt_13-9_10_perc/'.format(dset, fair)\n",
    "    lam_test = './{}_{}/dt_9-9_10_{}/'.format(dset, fair, 'lambda')\n",
    "    lam_train = './{}_{}/train_dt_9-9_10_{}/'.format(dset, fair, 'lambda')\n",
    "    return test, train, test_b, train_b, lam_test, lam_train"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "id": "6a07b0bc",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/var/folders/1y/3mt7fkvd4s1bm0m2bp4fqyh00000gn/T/ipykernel_2478/3646224204.py:1: MatplotlibDeprecationWarning: Support for setting an rcParam that expects a str value to a non-str value is deprecated since 3.5 and support will be removed two minor releases later.\n",
      "  plt.rcParams['text.latex.preamble']=[r\"\\usepackage{lmodern}\"]\n"
     ]
    }
   ],
   "source": [
    "plt.rcParams['text.latex.preamble']=[r\"\\usepackage{lmodern}\"]\n",
    "params = {'axes.labelsize': 20,\n",
    "          'axes.titlesize': 20,\n",
    "          'xtick.labelsize': 16,\n",
    "          'ytick.labelsize': 16,\n",
    "          'legend.title_fontsize': 16,\n",
    "          'legend.fontsize': 16,\n",
    "          'legend.handlelength': 1,\n",
    "          'figure.autolayout': 1,\n",
    "          'font.size': 20,\n",
    "          \"figure.figsize\": (6,4),\n",
    "           'text.usetex' : True,\n",
    "            'font.family' : 'lmodern'}\n",
    "plt.rcParams.update(params)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 66,
   "id": "71732db2",
   "metadata": {},
   "outputs": [],
   "source": [
    "def plot(train, test, trainb, testb, train_l, test_l, x, dset, save_path):\n",
    "    fig = plt.figure()\n",
    "    plt.figure().clear()\n",
    "    plt.close()\n",
    "    plt.cla()\n",
    "    plt.clf()\n",
    "    frames = []\n",
    "    y = 'accuracy'\n",
    "\n",
    "    for data in [train, test]:\n",
    "        # multiple y per x value\n",
    "        plot_df = data[0][[x, y, 'threshold', 'seed']].replace([np.inf, -np.inf], np.nan)\n",
    "        plot_df.columns = plot_df.columns[:2].tolist() + ['threshold', 'seed']\n",
    "        plot_df.sort_values(by=['seed', x], inplace=True)\n",
    "        ## INTERPOLATE DATA FOR PLOTTING\n",
    "        all_x_values = plot_df[x].unique()\n",
    "        seeds = plot_df['seed'].unique()\n",
    "        for seed in seeds:\n",
    "            per_seed_df = plot_df.loc[plot_df['seed'] == seed]\n",
    "            xp = per_seed_df[x].values\n",
    "            fp = per_seed_df[y].values\n",
    "\n",
    "            x_values = sorted(list(set(all_x_values).difference(set(np.unique(xp)))))\n",
    "            interpolated_df = {x: x_values, y: np.interp(x_values, xp, fp),\n",
    "                               'seed': [seed] * len(x_values), 'threshold': [np.nan] * len(x_values)}\n",
    "            plot_df = plot_df.append(pd.DataFrame.from_dict(interpolated_df), ignore_index=True)\n",
    "        plot_df['Fairness type'] = [x] * len(plot_df)\n",
    "        plot_df.sort_values(by=['seed', x], inplace=True)\n",
    "        plot_df = plot_df.rename(columns={x: 'Unfairness'})\n",
    "        frames.append(plot_df)\n",
    "\n",
    "\n",
    "    # sb.lineplot(data=dp, x='dp_diff', y='accuracy')\n",
    "    # df = pd.concat(frames, ignore_index=True)\n",
    "    trainb = train_b[0]\n",
    "    meany, lowy, uppery = mean_confidence_interval(trainb[y])\n",
    "    lowy = abs(meany-lowy)\n",
    "    uppery = abs(meany-uppery)\n",
    "    meanx, lowx, upperx = mean_confidence_interval(trainb[x])\n",
    "    lowx = abs(meanx-lowx)\n",
    "    upperx = abs(meanx-upperx)\n",
    "    plt.errorbar(meanx, meany, xerr=[[lowx], [upperx]], yerr=[[lowy], [uppery]], marker=\"*\", \n",
    "                 color='darkblue', label=\"Original (Train)\", zorder=20, capsize=10, linewidth=3)\n",
    "    testb = test_b[0]\n",
    "    meanyt, lowyt, upperyt = mean_confidence_interval(testb[y])\n",
    "    lowyt = abs(meanyt-lowyt)\n",
    "    upperyt = abs(meanyt-upperyt)\n",
    "    meanxt, lowxt, upperxt = mean_confidence_interval(testb[x])\n",
    "    lowxt = abs(meanxt-lowxt)\n",
    "    upperxt = abs(meanxt-upperxt)\n",
    "    plt.errorbar(meanxt, meanyt, xerr=[[lowxt], [upperxt]], yerr=[[lowyt], [upperyt]], marker=\"*\", \n",
    "                 color='darkorange', label=\"Original (Test)\", capsize=10, zorder=20, linewidth=3)\n",
    "    x_max = max([meanx+upperx, meanxt+upperxt])\n",
    "    \n",
    "    # lmbda=1 baseline\n",
    "    trainl = train_l[0]\n",
    "    trainl = trainl[trainl['threshold']==1.0]\n",
    "    testl = test_l[0]\n",
    "    testl = testl[testl['threshold']==1.0]\n",
    "    print(testl)\n",
    "    meanyl, lowyl, upperyl = mean_confidence_interval(trainl[y])\n",
    "    lowyl = abs(meanyl-lowyl)\n",
    "    upperyl = abs(meanyl-upperyl)\n",
    "    meanxl, lowxl, upperxl = mean_confidence_interval(trainl[x])\n",
    "    lowxl = abs(meanxl-lowxl)\n",
    "    upperxl = abs(meanxl-upperxl)\n",
    "    plotline, cap, barlinecols = plt.errorbar(meanxl, meanyl, xerr=[[lowxl], [upperxl]], \n",
    "                                              yerr=[[lowyl], [upperyl]], marker=\"o\", \n",
    "                 color='red', label=\"Baseline (Train)\", zorder=20, capsize=10, linewidth=3)\n",
    "    plotline.set_solid_capstyle('round')\n",
    "    cap[0].set_solid_capstyle('round')\n",
    "    cap[1].set_solid_capstyle('round')\n",
    "\n",
    "    \n",
    "    meanytl, lowytl, upperytl = mean_confidence_interval(testl[y])\n",
    "    lowytl = abs(meanytl-lowytl)\n",
    "    upperytl = abs(meanytl-upperytl)\n",
    "    meanxtl, lowxtl, upperxtl = mean_confidence_interval(testl[x])\n",
    "    lowxtl = abs(meanxtl-lowxtl)\n",
    "    upperxtl = abs(meanxtl-upperxtl)\n",
    "    plotline, cap, barlinecols = plt.errorbar(meanxtl, meanytl, xerr=[[lowxtl], [upperxtl]], \n",
    "                                              yerr=[[lowytl], [upperytl]], marker=\"o\", \n",
    "                 color='purple', label=\"Baseline (Test)\", capsize=10, zorder=20, linewidth=3)\n",
    "    plotline.set_solid_capstyle('round')\n",
    "    cap[0].set_solid_capstyle('round')\n",
    "    cap[1].set_solid_capstyle('round')    \n",
    "    x_maxl = max([meanxl+upperxl, meanxtl+upperxtl])\n",
    "    x_max = max([x_max, x_maxl])\n",
    "\n",
    "    new_x='Unfairness'\n",
    "    y='accuracy'\n",
    "    print(frames[0].columns)\n",
    "    sb.lineplot(data=frames[0][[new_x ,y , 'seed', 'Fairness type']].sort_values(by=['seed', new_x]), \n",
    "                x=new_x, y=y, estimator='mean', ci=95, color='b', label=\"PROFITT (Train)\")\n",
    "    sb.lineplot(data=frames[1][[new_x ,y , 'seed', 'Fairness type']].sort_values(by=['seed', new_x]), \n",
    "                x=new_x, y=y, estimator='mean', color='orange', ci=95, label=\"PROFITT (Test)\")\n",
    "\n",
    "    #plt.legend(ncol=6, loc='upper left')\n",
    "    #plt.ylim((.6, .9))\n",
    "    plt.xlim(right=x_max+.001)\n",
    "    \n",
    "    #plt.title(dset)\n",
    "    if dset == 'COMPAS':\n",
    "        plt.ylim((.4, .8))\n",
    "    elif dset=='adult':\n",
    "        plt.ylim((.7, .9))\n",
    "    elif dset=='default_credit':\n",
    "        plt.ylim((.8, 1))\n",
    "    else:\n",
    "        plt.ylim((.8, 1))\n",
    "    plt.ylabel(y.capitalize())\n",
    "    xlab = \"\"\n",
    "    if x==\"dp_diff\":\n",
    "        xlab = \"Demographic Parity Gap\"\n",
    "    if x==\"eodds_diff\":\n",
    "        xlab = \"Equalized Odds Gap\"\n",
    "    plt.xlabel(xlab)\n",
    "    \n",
    "    #plt.gca().set_axis_off()\n",
    "    ax = plt.gca()\n",
    "    ax.get_legend().remove()\n",
    "    plt.savefig('{}acc_v_unf_{}.pdf'.format(save_path, x), dpi=1000, facecolor='white', bbox_inches='tight')\n",
    "    \n",
    "    \n",
    "    \n",
    "    "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 67,
   "id": "544034ab",
   "metadata": {},
   "outputs": [],
   "source": [
    "def export_legend(legend, filename=\"legend.png\"):\n",
    "    fig  = legend.figure\n",
    "    fig.canvas.draw()\n",
    "    bbox  = legend.get_window_extent().transformed(fig.dpi_scale_trans.inverted())\n",
    "    fig.savefig(filename, dpi=\"figure\", bbox_inches=bbox)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 69,
   "id": "90658736",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/var/folders/1y/3mt7fkvd4s1bm0m2bp4fqyh00000gn/T/ipykernel_2478/2058678957.py:26: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n",
      "  plot_df = plot_df.append(pd.DataFrame.from_dict(interpolated_df), ignore_index=True)\n",
      "/var/folders/1y/3mt7fkvd4s1bm0m2bp4fqyh00000gn/T/ipykernel_2478/2058678957.py:26: FutureWarning: The frame.append method is deprecated and will be removed from pandas in a future version. Use pandas.concat instead.\n",
      "  plot_df = plot_df.append(pd.DataFrame.from_dict(interpolated_df), ignore_index=True)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "     Unnamed: 0  support  accuracy  precision    recall  \\\n",
      "500         500      498  0.877510   0.886555  0.983683   \n",
      "501         501      498  0.871486   0.876310  0.988180   \n",
      "502         502      498  0.889558   0.908696  0.969838   \n",
      "503         503      498  0.905622   0.917927  0.979263   \n",
      "504         504      498  0.877510   0.900442  0.962175   \n",
      "505         505      498  0.883534   0.886555  0.990610   \n",
      "506         506      498  0.917671   0.924078  0.986111   \n",
      "507         507      498  0.881526   0.897380  0.971631   \n",
      "508         508      498  0.885542   0.891667  0.988453   \n",
      "509         509      498  0.879518   0.887712  0.983568   \n",
      "\n",
      "                 conf_matrix  log_loss  selection_rate_neg  mean_prediction  \\\n",
      "500  [[ 15  54]\\n [  7 422]]  4.230665            0.044177         0.044177   \n",
      "501  [[ 16  59]\\n [  5 418]]  4.438726            0.042169         0.042169   \n",
      "502  [[ 25  42]\\n [ 13 418]]  3.814544            0.076305         0.076305   \n",
      "503  [[ 26  38]\\n [  9 425]]  3.259698            0.070281         0.070281   \n",
      "504  [[ 30  45]\\n [ 16 407]]  4.230679            0.092369         0.092369   \n",
      "505  [[ 18  54]\\n [  4 422]]  4.022595            0.044177         0.044177   \n",
      "506  [[ 31  35]\\n [  6 426]]  2.843564            0.074297         0.074297   \n",
      "507  [[ 28  47]\\n [ 12 411]]  4.091963            0.080321         0.080321   \n",
      "508  [[ 13  52]\\n [  5 428]]  3.953241            0.036145         0.036145   \n",
      "509  [[ 19  53]\\n [  7 419]]  4.161310            0.052209         0.052209   \n",
      "\n",
      "          tpr       tnr       fnr       fpr  dp_ratio   dp_diff  eodds_ratio  \\\n",
      "500  0.983683  0.217391  0.016317  0.782609  0.197869  0.120320     0.435000   \n",
      "501  0.988180  0.213333  0.011820  0.786667  0.053759  0.240570     0.066667   \n",
      "502  0.969838  0.373134  0.030162  0.626866  0.146465  0.264890     0.213992   \n",
      "503  0.979263  0.406250  0.020737  0.593750  0.087758  0.368779     0.306220   \n",
      "504  0.962175  0.400000  0.037825  0.600000  0.153871  0.311732     0.226368   \n",
      "505  0.990610  0.250000  0.009390  0.750000  0.116592  0.186875     0.049261   \n",
      "506  0.986111  0.469697  0.013889  0.530303  0.140119  0.275811     0.061425   \n",
      "507  0.971631  0.373333  0.028369  0.626667  0.091324  0.363470     0.209360   \n",
      "508  0.988453  0.200000  0.011547  0.800000  0.195098  0.102442     0.214112   \n",
      "509  0.983568  0.263889  0.016432  0.736111  0.138951  0.194954     0.342432   \n",
      "\n",
      "     eodds_diff  seed  threshold  selection_rate_pos  \n",
      "500    0.073854     0        1.0            0.955823  \n",
      "501    0.228838     1        1.0            0.957831  \n",
      "502    0.183036     2        1.0            0.923695  \n",
      "503    0.338221     3        1.0            0.929719  \n",
      "504    0.192308     4        1.0            0.907631  \n",
      "505    0.095074     5        1.0            0.955823  \n",
      "506    0.112629     6        1.0            0.925703  \n",
      "507    0.324128     7        1.0            0.919679  \n",
      "508    0.036932     8        1.0            0.963855  \n",
      "509    0.176190     9        1.0            0.947791  \n",
      "Index(['Unfairness', 'accuracy', 'threshold', 'seed', 'Fairness type'], dtype='object')\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAZMAAAEDCAYAAADweukiAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjUuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8qNh9FAAAACXBIWXMAAAsTAAALEwEAmpwYAAA/QElEQVR4nO29e5gcV33m/57u6bmPpmdGd+tijWzJVyxLI2MbE3uxBARCCEa2YSFhk9gjWPjtkvxAinfJWk5CzBgnJCybIBGSZYkDtoQDCWHBkjGObWxsaXwH3zS2ZFl3zbSkuU93n/3jPaWqqeme6duoZ6T38zz9dFfVqVOnTlV/33PO91yMtRZCCCFEMUTKnQAhhBDTH4mJEEKIopGYCCGEKBqJiRBCiKKRmAghhCgaiYkQQoiiqSjXhY0xcQDtAFqstRtzCL8BQBeAZgCw1m7J57gQQojJoyw1E2PMGgBrACwFEM8hfAeALmvtNicSS40x63I9LoQQYnIx5Ry06EQgbq1dP0G4HmttU2B7JYAOa+3aXI4LIYSYXKa8z8QJQ5gEWLOZ8LgQQojJZ8qLCegD6Q7t687juBBCiElmOohJPNsB58Sf6LgQQohJpmy9ufIgAddDK0BzHsfHYIxpB3uSoa6ubtUFF1xQXAqFEOIsYNeuXUettbMyHZsOYtKNsbWPOABYaxPGmHGPZ4rQ9fjaAgBtbW12586dpUqrEEKcsRhj9mQ7NuWbuay1nWDtI0gzgB25HBdCCDH5TEkxMca0hsaJ3BfaXgtgcx7HhRBCTCLlGrS40o1YXwdgjTFmQ6iL7zoAp8aeuHEorcaYNc7fsdtauy3X40IIISaXsg5anArIZyKEELlhjNllrW3LdGxKNnMJIYSYXkhMhBBCFI3ERAghRNFITIQQQhSNxEQIIUTRSEyEEEIUjcRECCFE0UhMhBBCFI3ERAghRNFITIQQQhSNxEQIIUTRSEyEEEIUjcRECCFE0UhMhBBCFI3ERAghRNFITIQQQhSNxEQIIUTRSEyEEEIUjcRECCFE0UhMhBBCFI3ERAghRNFITIQQQhSNxEQIIUTRSEyEEEIUjcRECCFE0UhMhBBCFI3ERAghRNFITIQQQhSNxEQIIUTRSEyEEEIUjcRECCFE0UhMhBBCFI3ERAghRNFITIQQQhSNxEQIIUTRSEyEEEIUjcRECCFE0VSU8+LGmA0AugA0A4C1dssE4TsA7AawFMDuYHhjTDuAVQC2ul03Auiw1nZNQtKFEEIEKFvNxAlDl7V2mxOFpcaYdeOE3w5gu7V2i7V2I4BVxpg1oWA3AdgOoAPAZgmJEEKcHsrZzNVurd0W2L4XwPpMAY0xrQDWWGt3BHZvB7AxGM5a22StNdbaVdbazpKnWAghREbKIibGmJUZdicAhGsaHpnCd40TXgghxGmkXD6TZgDdoX3h7SCdAGCMiVtrE4E4Ru1zfpNu5OiDEUIIURrKJSbxbAdCggEAsNZ2GWM6AbTCCQvG1lZ2Akh4fhJjzFZjTHeoKU0IIcQkUC4xScDVHgKEt8NcD+A2Y0wbWPvoAgBPeDL4SJ4CcBuAMWLiajDtALBo0aL8Ui6EEGIM5XLAd2Ns7SQO+OIQxlqbsNZudL25toGCdEpAMvTs6kJmXwtcHG3W2rZZs2YVkn4hhBAByiImrhaRCO1uBrBjbGjienQFuRHA5sCx7caYeCiMugYLIcRpoJxdg+8LjStZCycOAAUidHyX1wvMiUab52B3fpKNoVrNzeB4EyGEEJNM2UbAW2vXG2M2uOapVnBEe9C/sQ4UGG/frQBanc9kqbV2VSjKbW5EPQC0wA1wnMRbEEII4TDW2nKnoay0tbXZnTt3ljsZQggx5THG7LLWtmU6pokehRBCFI3ERAghRNFITIQQQhSNxEQIIUTRSEyEEEIUjcRECCFE0eQtJsaYFZOQDiGEENOYQmom3yh5KoQQQkxrChGTpcaYzxljbjDGzCh5ioQQQkw7CplO5Xpr7dMAYIy53s2T1WOt/WlJUyaEEGLakLeYeELifj8IAMaYRmPMDeDMv/dZa0+ULolCCCGmOqWa6PFGAJ8EJ2xcaox5Clz1ULUVIYQ4CyikN9cN7nuFMebrxphuUEzutNY2W2tvs9beD04Z/2FjzLmlTbIQQkw/Nm16rNxJKIqJ0l+IA/7vjDGvAngQwGsAllhr32Ot/V4wkLX2uNuXcbVDIYQ4m7jjjsfLnYSimCj9hTRzdQP4pOcvGQ9jzC0uvBBCiDOYQmomHbkIiePdGLs8rxBCiDOMvMXEWvsNY8wMz3fiYYy5PDw63lp7k5zwQghx5lPodCprAdxljHmXt991GTaabkUIIc4+CmnmarPWfs9ae1641uEEpbU0SRNCCDFdKERM5FAXQogSc2KCod7WAsePA52dY48991wvVq78Ljo7+wAA6fTo4z09PN8jneanv3/0/mx0n7L6JmuYQnpzLZ3g+GoA9xcQrxBCTFkOHqRRvvDC3ML39gL19TTWb73FfYODwNGjwIIFo8MODADPPw+0tQFVVcDQEEUjEgGiUX4PDjKuoSFfAPr6gMpK4L/9t8fx9NP7sGnTz/Ev/7IWnZ3AxRcDNTUM99prwMKFwKxZwDPPAMkk40yngYoKoKEBOP98XnPFCsbf2wvE4zz/0CEvpXW12e63EDHZYYz5CYAvWWsf8nY6X0mH+wghxKSRTgPHjtFY1tdnDpNMAi+9BFxyyej93d1AbS1QXT16v7UsqdfUAPv30+ivWOEff/llwJjsYpJMAgcOUHQuvhh48UXg7W8HRkZ8Mampubug+82Vf/3XZ2HMs5N4hWg025GC5uYyxnwZwDeMMUsChxIA2tV7SwiRiaNHaYqamgqPwzPyR4+yBD9nDnDeeZnDdndTcABg716eW1EBJBIUoCuvHB3+2WeBw4eBc8+lKIyMcL+1wM6dFIm6Or/G0ddHQYpGGecvfwn8wz/46entZdNVJOLHNf2JxbIdKWhuLmvtDgDnGWNaAVwOoAtAl7X2eGEJFEKcqXjt86+9RsOaSgGXXjq2RtHfz6acpibWAAA2/TQ0APPn02i/+irw05/SiD/7LPCHfwi88gqNdlMT0Oq6/zzxBPDIIwx35ZXAT37CuDs7gX/7N+DP/gy46CJg3z4KzLJlFJs33uBn/XrghhuA2bPZrPTAA8AXvgBccw2bft7xDmDXLp5XX89ay8aNwKOP8voLFgDXX0/xWbfOb26a/gwOZjtS1ESP1touUEi8cSbNABrd3FxCTHus9Q3b6WZkBDh5kgapqor7Xn6ZJd9IJLd09fTQGLa0jFemLA3pNNNbX+83hnR309g//zzQ3g7cdx/TE40CV1wx+vxnnqHvYMUKprWigjWEvXvZZv+5z/nG2mPvXhrxoSE2XR09CpxzDvCDHwAdHcC8eay9tLePPm/9eqbh6aeBD32IIvPRjzLtvb0Mc//9wO/9HvAbv+Gf9+ijTNMPfwj8+MeMe9YsYPv20Wnbtw/41reAF17wm7jI5wrI2ccAlHIqlqsAvKOA8+7GeG+Rsbm48rOdPHpxrGYAcQA3W2tvKzjS00xbW5vduXNnuZMhTjOpFA1XJMK27pERfgYHub+/n5+BARqdxkaWSGfMYCmzuprGc3iY8VVXMzzAeA4epBDFYv6nspKiUFnJT6a/ZU8PsGcPr7d/P5tPolE6T2fPpvFbtowG6uqrmV5jeB/WMk1eKfjgQRq42loeb2gALriA192/3xeqWbN4Tn8/DeX8+X7aa2uZ5v5+/vbur7Nz9D5P3PbuBZYs4XUGBoCvfQ34yld4rVTKv8+vfQ34rd/ivr17mYYdO5jmdesoCgDwzW9SXJqbgVtvHf+Zvve9FINt2/J/H0rFZZexxpSZu9HR8Tls3Dj2yLXXAg8/7G9fcgnwB38A/P7vA297G/CXf0lBTSSAT3969Lmf/jTDzJwJfPjD46fv+uuBxx8H7rmH79ojj/D7+98fG7apiccWLQLe9z7g61+/G8Bdr1t7OOPwj4JqJsaYBwBcD+A4gEb3HQewG8D6QuIU4nTy2mtsMqlw/wBjKAyHDtGg7d9Pg33iBLB4MdvRFy4E5s4d7YIMlsWMoQF9800a9TlzaASjURrNcLktGqUxrqmheCQSbDo5eJAl+q1b2RX0/PNpbAYHGeaqqyhqFRUsAdfW+rWUWIyG/M03aXxfeIHisGoVcN11PH9oiELwla+wFP/rv04BGBxkzaeri9dvbQU++EGGeeMNholEePyBB3h/fX3An/85xSIaZfNPezvz8b776EPIxLZtFIm//3v+XraMfgkvH738//GPge98x8/zG24APvEJ5tOf/AnzIJnksR//ePxn/r3vAf/+73yOf/3XzKMgxvjPaNEiily+XHVVZjF5//vZvLZoEd+nPXuYT5ddxlpjNOqLyS23ADfdxGd8zz3M58FBpnv5cubrTTf5cf/ar/H9mjmTNbIjRyjOH/wgj1dVMe6uLhZ6PvUpFiz6+ihWR44An/wkn9WaNcyjVav4TLq7mS+NjcDXvw74uT2WvGsmxpg7Aez0Zgk2xlwfWCTrcnDVxTfyirSMqGZy9vHaa+yRs3o127b37qVhOXhwdP/8eJx/on37/FJ1VRWN7Ny5NKwDA2x7f+klNuccPjz6WsbQWMydy8+iRTQIy5ezCSaZ5CeRAL77XRoK71pLl/JP/eSTNOZBli6lczmRYDqqqylKa9fy2JYtbK658UbeW2enL0yvvpp7Xn3zm2zS6e4GPvIR3t/f/E2wq+hYYjF2cX3ctcxccAHzB2Cee0KxahVFYTzq6mj0ABrkT3yC+eOVmisr+XvvXuCLX+S9eV1rg3z4w8Dv/A7zuqGB3088wRL5c88BP/oR7/HjH2f+/fVfM/z8+cCGDSxo7NvHY7/92+ytdfgw35+nn+Z5dXVsMnvveykqr7zCprmFCykIa9fejUce+Rz6+/0a54wZfq3x9dcp/ldeyUJIUxNF/9VX+b5eeinfzyee4LN9+WXmwfvfz0LPggXc3rePwjI46NeYT5zgtRYs4H0eOsT3srGR+Vtby/cplaIQHTjAZ/j668zL6mrgmmvuBvAnL1l7ImN/tkJqJl2h6eZP9ehyPb1WFBCnEKeN+nqWxL79bRqSxYv5R33/+2nsvU9DA8MPD9OYv/IK/9ivvsqSZTpNw/nNb/KPf9VVrMGcey7/nIcP+11FDx7k+Q895ItFXZ1f2t+7lwbmgx+kAMRiTFcqRcO3dy/w85+zWeL4cWD3bsbx7nfTWA0MsGT/3e9y/6xZFJRFi/xxCvfdx7b9IFu3cl9PD0uob3877+P11/1mFg+v5ByN0hm9YAHzMci2bcC99zJewK/97N/P73SaTW//839SSC6+mL6KBQuYb1/4wmjfw6WXUiQfeogivmgR86mx0RfQt95iE+B3vsN019ayFH3FFRRwT0DmzKH4zJnDz4wZwLvexfsGKLQPPMAmtpkz6TPp6aFhr6zk80inWcMAmO+dnXyul17KfJkzh2nt7mZhpLeX6ayr4zkXX0wRqKmh6HljPa64wi/cVFRQPKqrKQL9/bw/Y3iNVasoOpdfznubN4/50drKuDwfWW8vayNVVYw3FmM69+xhHlVWMl3GsED0wgt81+fP5/fs2UxzZSWfHxkIybRPITWTG4IOdmPM5621X852fKqjmsnZyeHDNGazZo12ZCeTNMxDQ34tJdiE5BmBID093O81mXkji73wkcA8E4ODFIJXXqFR2buXPoiaGjp7V6ygAfH6zDQ08HrHjrEJJhrl5557ePyjH/X3AYw3maST3jMIw8OM01oanYceAv74j2noW1uZzhkzfB+O1xT2m7/J7Zkzgc9+Fvjf/5vG+ZZbaBQHBnivx4/7XXCbmhh+YICGrKmJhswriR87xnM/+UmWsH/wA8aRSvH6Dz9M/8DMmayFrF5No33//Wy3f+c7GdesWcCvfsX0HzrE++7u5v2lUhSA7m5+ewb5uuv8QXgeJ04ADz7IfF+yBPjFL5jGOXPYBHX4MI21tXxfvObLgwdp7Ht7/XeloYFx7N/Pa9fXMx/mzuXxiy++G9Z+DsPDFKz588e+l57ABsfADA/ToAdJpXgNa5nHQf/bsWMUk/37Ka7GMD+XLRt7vXTaf69efpnvzoUX8p1uaRkd1pi7AXx+l7W2bWxMhdVMjBtfsgHAZviDGNsB9ICTQE4bMZn2bNrEz3RgCqXV+/P19tKYez6NqioaqpYWGoOaGv6Z+/r8cQtHjozu5WUtw1RXc19FBQ3QwIDfdAAw7hkzaCwuvnhsmqxl3C0tdKjW1/vCNTJCI3ryJD8f+xjT1NBAAzo0xDS0tvJ7cJAlV2+kdTLpT8WxahV9FXPnsiYyPMwaWipF47JsGc/bvp1Gc/lyNpV99au+cT58mPmUTvNal17K+Ovr+Z1K8XdLC2s6e/bQgNbWMv4vftGvEVx8sd9J4W1vY7PY/Pm8TnMzS83eSO9YjNcF/MGDixYxPfE4f3vP0hjG/dJLFFevphmkuprneYZz8WI+v5Ureb5n8I2hU7ymhnG3tDA9ySR7qqXT/rgVb1xJLMbfF1ww+pqVlZmFBOCzCRMWEoDG38uHMN69zJ9P4fPSlIng/uXL/XwLC0kuFNSbyxhzPbhU75estW8YYzYA+BIAC+BG1UxOI0Gv4VRniqW1u5slt3icn7q6saOiM+HVXlIpGp6+PjZnXXKJXzvx8EZV9/WxBP3WW35J0OspZS0Ncm8vS+EXXTTeOGM/7b/4BR3eMwJ9KtNpxjM8zNJomJERXu/JJ2lIrr7aT+fICM/38uDAAY7paGigcWlooCjE4zS2dXU8r7ubxzzjby1Ltr/6Fe8xaCCTSeZRVxdLzi0tY0eUP/kk0xKNMj1XX83rRqNjpyHJxuAgxWrhwvHDeQMSV63KbnAnYmiI74LXsy0bxrBmMl2ZqGZSVNfgsRczjdNt4KLE5DQyndI6SQwN0dgfP86SeiLBbGlupqH02sZz4dCh/MIHeeIJlnhXjrOo9ltvURA8P5DX4cBr/58ITziykU5nNuDevFHexIbNzbldb6qzadNj2LSpkPEdU4NNmx7DHXdcUzoxcWuYxKdT7WM8JCankemU1tNEOs1sOd0DI7u7/fEz2RgYYBPR5ZefvnSJMpDsA/r3AzPOn3CUrjEmq5gUUrH7JLRmiRAlIdeR7KWmuXl8IQHoH5CQlBibBkZ6i49n8AjjGo++PcDIBPPaA8BwDzCSYPjE86E49vJ4DhQiJtuttVmnvgyuviiEEGc9wdr40NGxBnsi0iPAsZ1uioMU0L8P6HuDNYrxrjlwEOjtGi0o/W/x/HBYm6ZAheMcPMz94fvIQEFiYoy5YZzxJDcWEKcQQkw90lkHfI9l4AANf3KA26lhYPAo0B1qRh85wXC5Xn+4B0gNASdeBnqeBk6+6s4fp0o78BbPTQ1TUADg5G5gYD9FJjkAdLsRo9EaXmPoKGBTo+OxaV4nNeiHz0JB65nAzcNljAlLlQF7dH0ql4hcL7AuFx+stVsmCN8BTtmyFMDucPh84zsr2bQJuOOO0sV3++1TpruvEHlj04DJUqYeOQGceAWoX0IjHL90/LhOdrlSvwFqFwID+4CaBTToHqlBINkLJPuBykYaaBMDGpYCFYGeDX1vAtVzWAMZOcHzhnuY3uHjQKyB7aPdu4Bm110uPcLj0SogPQykh3hvxnUNHDoGwHKfTbKWkx4B+vcw/kgFEAkNooIFhruBkeNAapyaEAqrmXQBWGKtjVhro8EPaMS/N8H5AE4JQ5e1dpsz+kuNMevGCb8dbGLbYq3dCGCVMWZNofGdtWza5Kq1GT7/+I9j+zfW1nJ/tnMkJCKMtfmV6CciPZJ7ST5fujvZlJPxukmW1I+/BIz00eAOHPLffY+TrwFD3c7gn+A5IwnXL3wvWL52pAYZb/IkjXlqhIb6hJtvxqaBo79gc1TPM9xO9dP422EgeZxxpAZ5Tjrl/DAngJ5ngeMvuH7eJykmSZfu1JCLI+3XNrzzjvwcePVvKHrJk8DRJ9ms1vMM833wIM8fSoyblYX05rrcWvt0occD4XqstU2B7ZUAOqy1azOEbQVrIiawbx2A9V74fOILckb05ppOqDfX1MNrM49MMLglE+nU2PP63qQzt3kVUFGChTyO7WJpO37JxGFz5eiTrHH0dgE184G6hTS6x19kutMpGvnjz9P4RyqB2Aw6z2MNLPkbA9QuYhyVTUDvG0B6EKiaSUHxmoxMBTDrHTTuva9TBCrjQLQ2UEtpYl5FqlkLirhRtdFaYPAAr29iwEgPUD2PtYXqOX7vKxMFUgM4VSs68ZLzf6SBqlm8x8H9AKJMTzRGgYhWAT92nbNW/hVrJnULGE+skWkb6XG/+2AWryvdCPgchKJpguOeoQ+TALAmw34AyBS+ywtfQHxCCI++PcDQEaBldX7n2TRw7EmgbhFQew73DXUDJ1+hIUs8B8x8e+Hpsmka1vQwEM0wDDzZz/b+4FQEuRauUn1sQvJ6RKWGWUtJDVAM+98Cauex5mCH2TQ0cpJG3BgKw1A3xSUS4/2mByk0Q0dp3L2mpFgj0NMJJAeBWD3DDyWA6hibnkyENZlIJZBM8LyUS5dNUcxMBYBhoG4xawu9u4GKRjoWhrqBmnlO4KL0l4wk+GxqzsEpAUmNABhhc5pXQ0kHfCSdn+X3NVspnMk+vzZjIkD9ueNmad5iksNEjh0AJnormwF0h/aFt4N0umvHrbWJQBwwxsQLiE+Is4eRk2BJ0y1tOHgM6HudBiJS5QzhCMcaVNSwlJwxnl727KlqBipqaXyHjrB0XTULgAWOPcXt9BAwUje+T8IjPULn8ozlfokcYDwnd7s2/0AcR58EZixz/oxzgWo3I2FPJ0vgVbP92lI6BdgRGtOeZ4DGC1n6T7kmoFicxxPPuS6yJ3h86CjjsCMUgdQI7yPZD1SmeTw1TENe1USjC8v8TB1x92zp3xg4ADRfzusNH3e1wAqX704ATAWFLJ10DnwLRKuBYWfQ00ker55DH0fnHwKLPgIs+TjTaIf9+CpqKYgv/SXz4JqtrN1Ea5gHTStYe0oNZHkeSb4DJ1/j/dg0UDXx/CqFOOB/CuAYxnYlaAWd44kc4ohnOxASDABc0dEY0+mu0el2B2sjecV3xlGqpqN77uEiC94qTwB9Jlu2cDKoYpluTXJnAskB4OjjLOHWLaax7d4JVFQDiAD2OI1o1Swg8SxLrfXLaCCHumnQa+bSiPY8TYNVUQdUNLCU7vkzep52xvYkDVr9UmDwEHsO1WaZiMqj/02KRkUDS9g2RcdytIpNTTE3qVY6Sf/B0BGgt5LXGTnp0uPmdhlO0AnevAro3eN8F+7eh465eF3zkuewBhhXyq00lnZO8l1fAy66IWR0LfMiNczf6UHmychJ1tBMDNj3feCcDzDuX3YAva8B7/w+43n5q8Cxx4E1j/I+6hazZmiTvP5Iwr/XSAVrIMk+iuexncDBB4Cuf2CYk68yndY6f0efq3GkgGc+H0hyCqioB47/EnjuC9w38xPABR/ILChpVxOrngO8+T2Kc6yeNaxxKKg3l7X2pkwH3JxdudQIEnA1iwATTZpwPYDbjDFt7hpdAGCtTRhj8orPGNMOTkyJRYsW5ZDcswRPMD7+cX9fqYREnF5smrWEijoAhmLR9ybb7GP1NC4eJkqjFKmiETvxIg3H4EHAVHLgWnqI7fzVs7idGuQ+CwBuTEN6iAbIRGkIK+P0QaT6WQOozjBZWP9+9oJK9bF5bOAADdzAfqY52e9qT/2sCcFSYNJJik3/mwwfv8Q5vk9SUHqec0JU79eORhL8RGtZeh9OAE/8J2D++4AFH6K4VLhustWzgef/D3DyR8A13wNiTqzSQ7z3xgtYy0ie5HbtAqbn2FPAnu+w5rLsMxQSABg+xmdxzC3y8tYPmMfPfQFY8rvAvLUubTXAC38CXHo77/mZDQxf2TR28GBVM2uL0Wqmq6KWYjwQ6lDQuxuonOnHBQDPfwtY/j5gIIO5jsTYRPbm/cDBHRO+aqdeowIc8OPOv5XLFPTOx7Er5FAfs2+CONaADvZVxcR3Rjjg83mG5ewarOlUJo/UIJuvaua4bpwDQOJFGtLqmX5TU6Zmp9QgS8czltMgD3nz3bsmsKoWGiyPk6+BvYGSrhkkSRHxahAe1tKQ1cyjoa6ew9pH3UJXI7DAwZ+yOSnWSLGD9WtCI730lZx8BaiYQYOd7GUNINlHwamoZ/qaV9MAJvv9gXcVta5paC7Q0EoxqprJNEViwBv/xJI3AFz9HV6ryollw3nAt64E5gFY/lk2f53/Kb7Dv+wAGpYBCz/Ee9z/IyD+NuCRD2V/Ppf/BTDrauCBq7IcvxuYcx3j3usWg7nmn4FHx4nT48LP+2Nczv0o8+HFPx8dpnouCwceB8B7yxPzMZTUAV/0RI7W2k5XmwjSDI5hyYgxptVa2xXYdSM4BX5B8Z21TKFp4EWR2DSN68AhGkibAvrjbNpK9dOQVrWMFo9M/otoNYUEwCnfRPUsho01jm2ejLp52KvPYVt/JIMZGTpK41Y1ExjqoeAMHeWx9LBzPBsgeQKod7MzVdSyZhCtZhg7AkTq6R8A2PafGqJhr6j30zx4CBg67ESikulJJ4GRYQoYkuz+a6LuXizwSGix9Pol/n3ULhg90PDlv+L3rGt4P56xP/BjYN6vAy/+2dj7D+M1Z5kK518J8XSG2YRzERIA+NWX/d977wXil40NExSSSaKgNeCz4Zzzua5ncp8xZp21dpvbXgsnDi6uVgArA8d3uSWCO53Tvc1auz7X+MrJzzb9DNdtuq7cyciLn236GR6+4+GSxXft7dfiupLFdnrI6blZi1MDwTzSSTcuwquFed+GHxNx4SPOyWv9kvopg+dIDdFYVjb7TTbHf8lwQ930W0RibPIwUTa9pAbYu8prd88HY3yHtrcdxuu5lYnuXcBLX/HHTbz7Cd7zwAEaVJukUU0NMA+qZvKejjwCNK30m98ilczDYG2nbjEF6adrgMUf5e+FHwKq5wPdT7PJqvFS4OI/cl1t65gn6WHgxGvAoQeAOdcDdeeOTXekgrWag9uBt/6Nzvwwv/i9sfc6wahwnATQC2BriYa81QPIsC7LKBLP5h/vwg+zS7EnnJl414OgtyEzhfTmcm99RjoB3JpLPNba9caYDa65yhtHsi0QZB0oCN6+WwG0Op/JUmvtqjzjKxsP3/HwtBOT6zZdN2Ga7zB+c9nt9vaJI821dS054HqoOEMN6/+2QQNtx4axaQBpt0yi+1gb+LaBgVtp34gH97m4Hr7jYVz3qYh/HIH4T13XYSrYLJQaxJgpKRhgdHiPiGvvDgpOpJICYQwNnDcAsGYOu3oOHWNJs2beWOd2ZSM/wGghSQ36TVWDR2jQ45eMbr6aCM/ZXuEGto6cAH55F53DCz7EdD+5fvQ5yV76Trw0pZNA/2ss/aeHKR4/uYLH5r8fWP5fnH8gQbEJNguZiF8r2PMdfh98ELj2h/RpABwXUrPAjW+x7M01dIyC/Orf8rP8s2PvrfcN4IU/LcwQj0cDxhr/1t8F9tznjyg/EDg2UdPTO74LvPVD4I1/LF0aAfqRZl3ji8nSW4Hd3/CPv+epCTvQFOqAf3cB543BWnvXBMfuCmxPKAzjxSemEd276NiEM+Q5YwJfXi3A+PtHbZtAf0QTCh+IK1oZiNeMDh/8c3mD1KLxibvCnjrHsqRu6kePlbApnBK62AyKgnWO54GDNEI180ZPv5GN4R5g538BTvyKPaxmLAMO7KBYR6uBhetY4+l5DrjkCxSdmvm+EKVdl9djTwHP3kbxW3EnMOMC4KfvYTwHfgzs+0Hm5pWfrgHO/W32rpp9DeNtdEsPjpz0hQQA9v8bPzMuZHrD/Pzjfo3nFGmgZvboXQ9eC1z1LeDxT3C7ZsHoMTSewZz3XqYdAB7No+bQvCpzjWTJ77BDwKGHQvs/Abz+LX975juY96/+LbDoRuDAVv+YiTFPw8x/HzDvfWy2W/aZicXkqm+zc8OzG3O7p8rG0c1v568HDj3oz+s1cnJSxCTH1AlRIOkRNuvk20QzGQTHPYyHieLUHEi5YgyNx5h9Ge7bGApLbAZyHpw3cBB49evsQrrkPwGJZ1iSP+f9LIUefHC0UfrZ+/hd0QCs/l803C/eidGC3gc82e4G5QVqYH17+PF45/10AnfvBN74Nj+L/yONaPMqNnONBNyvl30RePa/83cmIQF8Ibn6n4B9/+zXUvb/XwpDEE9IAM6RNXTu2PgWfxRYektuQjLnehpXALhwA4AI8NhNOJU3LVcAy/4/Cn3Dcjf+I00xP/pzP56FH2YhoMk1rMx7N4CAmLzjn4BHQ3PlLvuvwPz38DxvoObKv2SHgrConNfOnnORGH1KjRezR13VLHZFDrPsM3yWLVcyvVd+y5/b66LbgFe+CjS3sTfaBFPeFzQC3hjTCOD6YK8tY8zlPGyfyTdOUQS359C8NFWYTmmdyoSFJNkL7PshcOTf2YRT1Qx0/R9gzz/xeOvvAss+zd9BIZp9LcVp772h+E4CT/zuaLFY9hlg0c0A0sCOa0cfa14NdD/F321fA2a6xdBX/RWw/Ro/nJeerr8P3EsMWPOQE1AnJu/aDvzsN1zzXwZq5gPL/4A9yd74R+C5P84+0NJj8DDTeeEfAo99FDhvPe891Z85/MyrgQNOBNY+RgP72t/RbwPQ1/Xep9hc9au76G/wZt2d/U5+Jwd4njc+Y+5a1l48H9s77h1roKvnslYxcJBjRc75IPMzNURB6N9HP1ntItZSBg/RzwOwlnleuxtw6Zpc572XYnLpHfRbvfinofu8isITqeJ3epi96E520Z914Qbnq+oGkKn51qeQrsErwFl7OwC0W2t/Gjg27QTldHQNvsPckZtPYZqRt88kVw4+xHb5UtVMbNoNUEsGvpOh7RG3z/++44IEbn+x2neonwofDJviwDVv3QevZuGNtfBGN58a4TxEf0HduexBVN/KP22+99O3h11gD/wE2PcvLBFHKlnyDy+ItPbR7L4Rm/abmlreDpz7H4Fd/5WO7FlXAwceYGl68c2+CD24hmMi3nEvDWX1TI5GbzhvbBPfL7/si9Xij7IL7bO3+cfn/wbHVMC6gYNDbHJ5ZB2nO3n7N93AuVkcJX/+J+lnidVTDB77GEfz58LijwDn/2d/tH/9YtYcjj3F+3nlaxTM5lVs1tnxpxNGOa2ZBz7Dymb6tWyKtRdTwXco1uCmy0+yU0a0BmbOO0vXNRjsRfV3yDA7sKu13ADgmQLiFYJ/8le+yt/evEQ2ZOjHE4RMgpHRIZ4Lm4Bn/mjiYCbGP1v1bLZ6pAcC10769xGtpsFPPDd6EaLKZo5baDifXXkr44C3hsTAARrLWKM/JblXEgUoVnPfDZz7EZ7z+O9w/yX/g81Fc9eO72Q3EXa53f8jYOnv05Cs+iqbbSIVwHkZ+tO8cytrOFWBccEzlvE7PeJqFBH+vuCzjDc9TNGMVABzrgV+cQt7pS38Ld5fpJrGy3PUN63gvZoofS6wHOcxctyVut1Ss62fAJ7fxHMuvYPNOl6z1Xt+wefn+TBaruIYksYLXOm9knna0kaxWvLbTLc35fqZzsqv8N2NNQDNK/meHn0cgJu+pqKGx6MxoGpuZl9OgELERHNeCTx/z/Njti/92ATrPeRC8qQbyFbhejRl+Y5WAxVZjkUq/PEPJpb5+1SYTOe733iDvWfGS0e4K28uWEtj1vc6BwCefJUD8/Z8d+wf1kTZpJE86a9H4XH53TSe1bP8fZfczunEZ10DLPjN3NJTv8RvBgNYI8mE54wPNyn17aMTPFLpFnIaYA0jUsl2/sqm0XkUqWQ6Dz3Ebr0myoGIJ1/xm3CWfYY1pRkXcRp3b2CliQJN53G6l57n6J8AKD4Ny+hTWPAhYN57uD/+Nl9Mauf583LFGoDGixhfT5ppjjb6i0Q1LMst76YztYtYwzNRv7t6tNbvkn3qPa/gQFKvF18WChGTpRMcX43cxpmIacrz9zyPf23/11H7vO2iBWXGcuDXvl/aZq6CeYNNN6XGGApA9SzWADyspf/DmzYjWs1aSsTNmJt28zclnmGJuuH8sXEv+AA/QbwFmaJ1LG2O9LLGlGlAYph0kkZm5AS76wL+7LHWsuZkk24gYQNrDdEa/k71snYRdd10q2cD/Qf4bBuWUsT697NmUDOfPaFGjrMHkTFugam32KzlGbXKJrbhmwgNYSrOnltel+lIjE71mjlsFpv9Ls4bdsH/z3MqmylKtQv8MSy15wDHf8V8H+5hTznPN3Lz952BHWK+1S4EWlZxzZHUIB3TsRluJP4sGt0TL/N78AhrmrULnaClmS8znVhHosAPbgBe+2c/v5f+JnD15ylosQY+48HD7MYdqeQ9WEv/2KnZhMHnG5tBoayeDTS9jb8TL/L5zDifv6PV9BN9+1qmLVIZGLAKN8j0CHvrVdTxuUZrM8/aHKKgrsHGmJ8A+JK19lQfOOdL6XAfESLoXzgTGekfwf0fvx/3f1zliIIxhgYkPC2JR6SCI8JnXZP5eLJ/bOkx7Qx9wwVcz2LwsPM5tHAm2Yra0fN0hel7A4ChUYtfxlrCwCGgcgadwTULmO6ho/QztLydRr6yidc1vwQG3gRiTVw4qqrJLUH7KsW0spFCYoybTj7CeaSSbrr3+NvojI41uOneK/0ZgRsvAQYfYu0MoNhVz3YDPY/6648svdUZ+2YKaHoYaLrcv0ebcnOTVQCVLa7JzRur42ZRHjpMIaqopmBW1Lo5zmpo0Gcsp3H2Ztytb2Vc0WonAClXA1syev2XD94P/EVA0H/rB8DhR2jAI9W8h9qFo0XfGNbIvPFYI8d9sY1U+T64aBVFpX8vp6Opmsn09r/l7i1G/1BQKGrmUkS8d7Aynv3dCFFob64vA/iGMWZJ4FACIYe8EGcc2QZAJvu5PyNuTIv3h/fO8WZzjYaXSs2T9Ah72xgXtzfV/NBRprNhOdCwhDUKb/CiMcDwEnZbjdZxu3+fX9qtbHSTO86kwWq8iOmsjAPJn1M4mlayBB+tBBLPswWuZr5vLGMznC8lxQ4HsUZeu38fm/gGDlKkPEPZfDkNfWUTw5wYYpoHD9PwWesvUQswPdFq4PjLrF1EKjiWpnc3aykV9W7iw7coMDXnuMkq545etKt6rt9UaKqYfq+bd0Udm9BOvEo/TYUbE9SymnmdGgASL7DGAbhxNBfyt+f/AfzaVd2CiZ9n/BJeD8huzGvm+s8+NeivPVJzTqhJscKfrqbxAtdJxN1bJDa2xhGtzm8ga4CC2hGstTsAnOemPLkcXC736YJSIMRUZvAIEBy9biLgYkMRf2xJpBJoXOQ7zj2hODVaPsVeX8leGp9IpWuyiLJpZ/g4/9S5NDvZtJuq/IQ/5YgxbiqSKuDI44Hp1F0aa50BM2a0Ea1sZKm3dzfoME8Gur5W0ZBGa11buhO8SAUw+9fG1oKyrY8ea2SNJhao/dSfy+z0plb3qKgD4AZiRlwNAKBxrz/Ple5D+eNNLZ8eBqrmuE4KeygQsQYK4VAVgJQbABobO3aoooa1ospm+m6Cx+sW857jF7oZjAPHTMSflXkiGpa6ecJyoLIJaLost3i9+8lWmw1T1eI/t6YVuZ2TIwWJiTFmBoA1bpxJl9u3AgCmU7fg08mZ1DXY85mM9PvO4lhtDB/Y8oHSOOEPP+ZGwIfxRqJHA07DCiDiDHykgmG8rrkm4vZFXWk56sJ6c2MZF4cZuw8Gd8TuAuZeP2pfyddkqV9CYeh7g/6HCYnQQMbq2ZkAETZleAaiZi7jaWlj6b6yZfz2bm/p2vQQjWnTZbkN1JzAGesnN8qmuTCVM5i22Izs8XsCFqyNhKmZD1S+TmNdu9AXvtoFrJUk+1yTV8P4NcDmtszPNliTyHbPp96hDPx8E/B4Hk3cfzHB+3XV7cDVm3KPL8wpAUTuA3JzJG8xCY4zMcYkvGYta+0zxpjLjTErJChnNp5gBP0jJRMSgIbw1FTpnsGfJGM+ESX+w43BGDef1mV0eqaH3Yy5XnNayjU7BHrbDB7hXFSZSqMN5/Ocmnks2U9UYq2o4+JH6UEa1Mm+X49Yoz+tSsZ01WaeniVM1Dn8vRoUQH+I13kjWssaY/xt48dTzHsVvzR7vl29qTjjP43IcRKhUbRZa79nrT0v7B9xTV2tpUmamMqEhaNkQgLQQFTUOEPhphQfr/R3phB1xr+qhQMBq2dTFCqbKDixBpb0a+dmF4lYg1+Sr2rOTRxmXkEDHBvHEV9qItGJR63n0IMIgFuGtiUQd6CMbAx9EMHmvVJTrM/rDEHjTIQ408h1oslg+KCzeLoxkVDkMiGmKJpCaia5jDMRQojpzVVnjp9zDJNwb4WIyQ5jzE+MMf8huNMYs8KNP9me5TwhhJg+nMm+jkm4N40zEUIIUTQaZyKEEKJoipr8yFrbBTfOxMMYc6619o1i4j3TuPb2a8udhLzJdw34iaaLufb2a6fd0sXT8bkJUS7yXs9kwgiNudNae9vEIacGp2M9EyGEOBMwxmRdz6QQB3y2i5xrjPkSgA2lilMIIcT0oKhmLjetSjuAmwGsBJDjkmdCCCHOJAqqmRhjbnHdgHsAfBLAgwDOs9aeB+DLJUyfEEKIaUDONRO3HO/NANaB3YC/AeC4tfamYDhrbQ7rnAohhDiTGLdm4gYi/q0x5hiAbeC0re+21rY40XjqdCRSCCHE1GaimslHwNrI5iw9tErbFUwIIcS0ZNyaibX2j6y1zeAUKrcYY25wTnchhBDiFDn5TKy1D3q/jTHXu2lUXgfQkv0sIYQQZwuFzM31IAAYYxoBtDrH/KlFsowxt1hr/660yRRCCDGVKXicibX2ONijC8aYRmPMrQBWAbgVgMRECCHOIooatOgREJZvGGMyDrUXQghx5lKy6VQCbJyEOIUQQkxhSi4mQWe9EEKIs4PJqJkIIYQ4y5CYCCGEKBqJiRBCiKKRmAghhCgaiYkQQoiikZgIIYQompIMWiwUY8wGAF0AmgHAWrslh/AJtxm31t4VONYOjsDf6nbdCKDDWttV4mQLIYQIUTYxMcZ0AHjKWrvN2zbGrPO2M4TfEBKPleF9AG4ClxHuBHCrhEQIIU4P5Wzmag8Jx70A1o8T/ubghrW2E8Dq0L4ma62x1q5yx4UQQpwGyiImxpiVGXYnAKwZ57RuY4zXhOU1a91b4qQJIYQogHI1czUD6A7tC2+HWQ9glzGmB8CdALrCTWJOYLqRow9GCCFEaShXM1c82wFjTMZjzv9xJ4CdADoQauJy+3dYa7c5EVlrjFmX5RrtxpidxpidR44cKSD5QgghgpRLTBJwtYcA4e1RGGM2g2KxFsBaAO3BZi9rbWfI4f4UgEzr1sNau8Va22atbZs1a1Yh6RdCCBGgXGLSjbG1kzgAWGsT4cDOx5LwnOrW2h0AliDgYzHGhP0tXQAy+WaEEEKUmLKIiROFRGh3M4AdWU5pBnAsFEfCC2+MaQWwPUMTmboGCyHEaaCcXYPvC/k01gLY7G0YY1q9464msjZ4shOOLne8C8DGUK3mZtC3IoQQYpIp26BFa+16Y8wG1zzVCmB3qHfWOlBAvH3r3UDH3YE4gqs6bnMj5AGgBcB29eYSQojTg7HWljsNZaWtrc3u3Lmz3MkQQogpjzFml7W2LdMxTfQohBCiaCQmQgghikZiIoQQomgkJkIIIYpGYiKEEKJoJCZCCCGKRmIihBCiaCQmQgghikZiIoQQomgkJkIIIYpGYiKEEKJoJCZCCCGKRmIihBCiaCQmQgghikZiIoQQomgkJkIIIYpGYiKEEKJoJCZCCCGKRmIihBCiaCQmQgghikZiIoQQomgkJkIIIYpGYiKEEKJoJCZCCCGKRmIihBCiaCQmQgghikZiIoQQomgkJkIIIYpGYiKEEKJoJCZCCCGKRmIihBCiaCQmQgghikZiIoQQomgkJkIIIYpGYiKEEKJoJCZCCCGKRmIihBCiaCrKeXFjzAYAXQCaAcBauyWH8Am3GbfW3lVMfEIIIUpD2WomxpgOAF3W2m3O6C81xqwbJ/wGa+1d1totLvwOJx4FxSeEEKJ0lLOZq91auy2wfS+A9eOEvzm4Ya3tBLC6iPiEEEKUiLKIiTFmZYbdCQBrxjmt2xizNRBHOygYhcYnhBCiRJTLZ9IMoDu0L7wdZj2AXcaYHgB3wjVpFRGfEEKIElEuMYlnO2CMiVtrE+H91touY8ydANYC6ABwFwBPTPKKz9Vq2t1mrzHm5TzSXm5mAjha7kScQSg/S4fysrRMxfxcnO1AucQkAdfjKkB4exTGmM0ANltr7zLGrAGw1RjTaq29Md/4nIN+Wvb0MsbstNa2lTsdZwrKz9KhvCwt0y0/yyUm3Rhbm4gDQKZaifOJJJzTHdbaHcaYJQBeLyQ+IYQQpaUsDngnConQ7mYAO7Kc0gzgWCiOhBe+gPiEEEKUkHJ2Db4vNA5kLYDN3oYxptU7bq3d4Y4jcDwODlDMKb4ziGnZPDeFUX6WDuVlaZlW+WmsteW7OAcddgJoBUaPWHfH1lpr17rtVrBH124vTHiE+3jxCSGEmDzKKiZCCCHODMo6N5cYSz7zi7mmPq+L82oA21UbG00B+XmT21zqwm+c5CROG4qZ+84Ys9laqxkpAuT5brYDWAXAG7h9I4AOa21XtnNOO9ZafabIBxw/sy7bdqbwoe3d4LQyZb+XqfApID83gxOIetu7AGwo931MhU++eZnh3O3lvoep9Cng3WwH0APAuvdyZbnvIfzRFPRTi5znF3Ol6NbQ7s0AVJL2yXe+tjaMnoKnC6PnfzubKWjuuyxTHYkC8tNa22StNdbaVdYNk5hKSEymCAXOL7bGdUwIhg8LzFlJIfnp/qTBP/hKANtLnLRpR5Fz37VBeTiKM3UuQflMpg55zS9mOc6mKbR7LTS2xqOo+dpce/YOKx8UUGBeuq7694GCInwKzc92F25KrtckMZk6xLMdyDZfWTgMWLK5vqSpmr7Esx0YLz9DTvjdmcKchcSzHciWly4fE9bahDFm8lI2PYlnOzDOu7kTzM8uF26rMaY7VJMuK2rmmjokkOd8ZSG+AeDGqdiWWiYSKCA/rbUJywXY7gKwNrjswVlMAvnn5U2Wg43FWBLIMz+ttZ12dM+tpwDcVuJ0FYXEZOpQ8Pxirklms/68o8h3/rd4cOVOx3YAWq2zsLn09C5mJ+//upvcNkgX6NObMqiZa4pgre00xiRCuyecX8y1S3d6QmKMWSNRKSg/2wB0GGO2TCTeZxsF5GUz2DnE214NoNWJ9TY7lcZGlIF889N1stlujGkKvZtTKh9VM5la5DxfmdteA76EO13JuhVTrLRSZvKd/21j6M+6Flw3R+SZl9bau7wPWMNLuO0pZQDLSD752YWx7+bN4NiUKYOmU5li5DpfmXNw9mSIYpvlGi8CBc3/5v3BWwAcc8ZQIL+8DOxvB0drt4ErpKrm5yjy3dw91XpzSUyEEEIUjZq5hBBCFI3ERAghRNFITIQQQhSNxEQIIUTRSEyEEEIUjcRECCFE0WgEvAAAGGM6wH7srQC2gPMHAZzmoRlaxXFc3BQiHQDWWGvHndnQGzNQqjEsOTy7e0s5IWCp05/ntVfCn5PKGwD5lLV2mzvWrBkgyoPGmYhTuAFmHdba8NT23iCq1RoQOT7GGJuDmLSDI5qXlvC64z277eCUOyVZOC1b+o0xrZM5wt0byAdgffA6TtzWwF/KVmJSBtTMJcJkXFfBlUK7jDGbMx0XueNmJS6ZkATItibGRgAbSrXq4Tjpn7RJMYMjwsOC5bZ3YJovLjXdkZiInHEl2/bQ6o5iihNYlmCyje3aiYPkj3vfOjDOsrZOUNQMW0YkJiJfdmCKTTAnxsfN4wZM4iyzblLCyRKrDrCZbqL0a3ngMiIHvMiXToSaMwLLiQKcbnyztbYr6JQGlxj2jM1a0KG/LTBzqjfZ3SinrpsZuRX+cqVd4TZxF4e3uFCwqcebibXDxbHWpWEtAuu/BNLQDGBp0LcQuIc2AEsC97AazvEbzqDAqpdjwrlS9mZkcNS7a60BjX4zgJ0lWuzsJjDfTqU1x3sek2cubaPS7+Ja7X57BY3d1totnn/FXefWQD6sBPCgi+/WCe5zDXKodbj3KR7cV+pnK8bBWquPPrDWAkA7aAQmCmMD2xsAbAiF2Q0gHti2YO8fhPatyXBea2B7HehQReh67aEwmwPbm4PH3b44OMPyukAc3u81Ga65PcP5me5he4ZrZbzXDPloQ9trMlx3XTAf8312Lt3tLp2toWvlcs8Z8yzXewpdb8x7FX5vxrk3m2vYDNct2bPVZ4L8LncC9Jk6n3zFxPsjZggzyqC7P+vKUJjdGQzx9pDB6slkTIP7AewKnZPNcI1JQyCtm8NhM52fYd9KF298vOu4ew3vsxnCZBLXdeHrjvdc3Lf3WRc0poXcc6Y8yyePQs8snAe53luhYlLSZ6vP+B81c4l8icNve18DIJFhSdFmAKtC+zK1d+8MbSfcuV7zVrfNvPZFl7v2tsB5E2IzN6V0YOwSqgljTDzLtUfF51YTbMPoVfLy8k24pq9WO7ZL6/oM+8YjYXMbC5TzPWfJs0K4E3SgrwdOPd9c7y0BruExhtA6Hx7eyqOT8WxFFiQmIl9Wg34TgIa/O4PBG/Pnm+jPm4FWjC8SXo+ye8H2fO+aKxFYsW4irPPtuK6nCVAI4nmksyuQFi/ORB7nA1lWx8xTSHKmBPdcCFsAvA6/R9aEBj1A1m6/lk75u5z/YxcCAjwZz1ZkR2Ii8mUNgOvdb89RPBlM9Mf3Sv87ANzsHL0AxjrxxyPTCoEZ1ucej1YU30vqtC5lW4J7zuUaK0GnfwKgwBpjdrrndB9yrE06NgLYPd6gyEBNIjiYcSo827MGiYnIGWcIdnhNH9baHcaYjCOfjTEri2wi2QmgOdwk4XrrrIRfE1ljixvZ3QEgPAAv7q7dioBBDOMMZqLYGoQzhIlMeVaCfMxEwfecB60Z0r0ZnAql2+bRU8rVMDa68zOOZQn34nKU/dmeTWiciQiTsaYRWMv71tChWxEad+LawxOhffF8EuH+5Bvhz8PkcRs4lYcXf0sGn80osl07sL87sG8l/FpRc0jIwm3zHRibH3nfqyNTPpZkxHoozrj7Oe49F3gPXeMNaHUC0ooCarOutrndGLM9yzXa4Te/TtqzFeNQ7h4A+kyND/jn2QX2YOkAu4J2uM9WjNObBmz68iYbPNUdEzQcm12cW8EaRdyFtWDvrTWB6/cgc++sDS7udoztwhl35wU/p+Jw19zqrueNjwiev87tX+fCevM8dYTSYQPH1sGf1BH53Gso3GaM7gkWzMd1yNKTKkP+bwg8u80TnTfRPY+XZxOkf2UwjnHes3gu95XlfC9tWwPXanfpimNsV+Cin60+uX000aOYtrgS6kZr7foM+zejhJP+5TKBo5gYY8w6O8UGA+rZlgY1c4npzHqwhDoKS//NRrBZTkwRXNNToszJEJOExERMZ7Yju2DcjAxCUwgF+g8EAGPM1kBPuzWlqimWCj3b0qFmLjGtCc1n5dGKQK+zEsR/G9iWvgVsOlN30RxxnSPibjOXyRpPG3q2pUViIoQQomjUzCWEEKJoJCZCCCGKRmIihBCiaCQmQgghikZiIoQQomgkJkIIIYrm/wHaQ4veh3WFTAAAAABJRU5ErkJggg==\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# PLOTS ACC V FAIRNESS\n",
    "# test and train lines\n",
    "#date = '12-8_10'\n",
    "title = \"\"\n",
    "for dset in [\"CrimeCommunity\"]:#, \"COMPAS\", \"default_credit\", \"adult\"]:\n",
    "    for fair in [\"DP\"]:#, \"DP\"]:\n",
    "        for method in [\"thresh\"]:\n",
    "            #for x in ['dp_diff', 'eodds_diff']:\n",
    "            if fair=='DP':\n",
    "                x='dp_diff'\n",
    "            else:\n",
    "                x='eodds_diff'\n",
    "            test, train, test_b, train_b, lam_test, lam_train = get_paths_rf(dset, fair, method)\n",
    "            \n",
    "#                 train = '../results/{}_{}/train_{}_{}/'.format(dset, fair, date, method)\n",
    "#                 test = '../results/{}_{}/{}_{}/'.format(dset, fair, date, method)\n",
    "#                 train_b = '../results/{}_{}/baseline_train_{}_perc/'.format(dset, fair, date)\n",
    "#                 test_b = '../results/{}_{}/baseline_{}_perc/'.format(dset, fair, date)\n",
    "            #train = read_frames('../results/COMPAS_EODDS/12-8_10_thresh/', 0)\n",
    "\n",
    "            train = read_frames(train, 0)\n",
    "            test = read_frames(test, 0)\n",
    "            train_b = read_frames(train_b, 0)\n",
    "            test_b = read_frames(test_b, 0)\n",
    "            test_l = read_frames(lam_test, 0)\n",
    "            train_l = read_frames(lam_train, 0)\n",
    "            #save_to = \"../results/{}_{}/{}_{}/\".format(dset, fair, date, method)\n",
    "            save_to = '/Users/sierrawyllie/Desktop/double_baselines/rf_plots/{}_{}_{}_'.format(dset, fair, method)\n",
    "            #save_to = '/Users/sierrawyllie/Desktop/'.format(dset, fair, method)\n",
    "\n",
    "            plot(train, test, train_b, test_b, train_l, test_l, x, dset, save_to)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "858a309a",
   "metadata": {},
   "outputs": [],
   "source": [
    "def read_frames(dir_path, good_label):\n",
    "    performances = pd.read_csv('{}overall_performance_seedwise.csv'.format(dir_path))\n",
    "    # differences = pd.read_csv('{}group_differences_M-m_seedwise.csv'.format(dir_path))\n",
    "    group_perfs = pd.read_csv('{}group_performances_seedwise.csv'.format(dir_path))\n",
    "\n",
    "    # good label was passed to the script that formed these frames.\n",
    "    if good_label == 0:\n",
    "        # then selection rate is wrt label 0, we need wrt label 1\n",
    "        performances['selection_rate_pos'] = 1 - performances['selection_rate']\n",
    "        group_perfs['selection_rate_pos'] = 1 - group_perfs['selection_rate']\n",
    "        # we also gotta get precision and recall for the 1 label (JKJKJKJK)\n",
    "        # performances['recall'] = 1 - performances['recall']\n",
    "        # performances['precision'] = 1 - performances['precision']\n",
    "        # group_perfs['recall'] = 1 - group_perfs['recall']\n",
    "        # group_perfs['precision'] = 1 - group_perfs['precision']\n",
    "        performances = performances.rename(columns={'selection_rate': 'selection_rate_neg'})\n",
    "        group_perfs = group_perfs.rename(columns={'selection_rate': 'selection_rate_neg'})\n",
    "        # we also lowkey oopsied the pos/neg labels here :(\n",
    "        oops_map = {'tnr': 'tpr', 'tpr': 'tnr', 'fpr': 'fnr', 'fnr': 'fpr'}\n",
    "        performances = performances.rename(columns=oops_map)\n",
    "        group_perfs = group_perfs.rename(columns=oops_map)\n",
    "    else:\n",
    "        performances['selection_rate_neg'] = 1 - performances['selection_rate']\n",
    "        group_perfs['selection_rate_neg'] = 1 - group_perfs['selection_rate']\n",
    "        performances = performances.rename(columns={'selection_rate': 'selection_rate_pos'})\n",
    "        group_perfs = group_perfs.rename(columns={'selection_rate': 'selection_rate_pos'})\n",
    "    return performances, group_perfs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "7f1090f4",
   "metadata": {},
   "outputs": [],
   "source": [
    "def mean_confidence_interval(data, confidence=0.95):\n",
    "    a = 1.0 * np.array(data)\n",
    "    n = len(a)\n",
    "    m, se = np.mean(a), scipy.stats.sem(a)\n",
    "    h = se * scipy.stats.t.ppf((1 + confidence) / 2., n-1)\n",
    "    return m, m-h, m+h"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "79956edf",
   "metadata": {},
   "outputs": [],
   "source": [
    "def plot_performances(train, test, base_train, base_test, save_dir, xlabel):\n",
    "    metrics = list(train.columns)\n",
    "    print(metrics)\n",
    "    unwanted = ['Unnamed: 0', 'support', 'seed', 'threshold', 'sensitive_feature_0', 'conf_matrix']\n",
    "    for elem in unwanted:\n",
    "        if elem in metrics:\n",
    "            metrics.remove(elem)\n",
    "\n",
    "    seeds = train['seed'].unique()\n",
    "    cm_subsection = np.linspace(0, 1, len(seeds))\n",
    "    colors = [plt.cm.tab10(x) for x in cm_subsection]\n",
    "\n",
    "    for y in ['dp_diff', 'eodds_diff', 'accuracy']:  # metrics\n",
    "        sb.lineplot(x=\"threshold\", y=y, data=train, color='b', label=\"Train\")\n",
    "        sb.lineplot(x=\"threshold\", y=y, data=test, color='orange', label=\"Test\")\n",
    "        if base_train is not None:\n",
    "            mean, low, upper = mean_confidence_interval(base_train[y])\n",
    "            plt.axhline(y=mean, color='b', linestyle='--')\n",
    "            x = sorted(train['threshold'])\n",
    "            plt.fill_between(x=x, y1=low, y2=upper, alpha=.1)\n",
    "            \n",
    "            mean, low, upper = mean_confidence_interval(base_test[y])\n",
    "            plt.axhline(y=mean, color='orange', linestyle='--')\n",
    "            x = sorted(test['threshold'])\n",
    "            plt.fill_between(x=x, y1=low, y2=upper, alpha=.1)\n",
    "        xlab = xlabel\n",
    "        if xlabel == \"thresh\":\n",
    "            xlab = \"Threshold\"\n",
    "        elif xlabel == \"lambda\":\n",
    "            xlab = \"Lambda\"\n",
    "        plt.xlabel(xlab)\n",
    "        if y == 'dp_diff':\n",
    "            plt.ylabel('Demographic Parity Gap')\n",
    "        elif y == \"eodds_diff\":\n",
    "            plt.ylabel(\"Equalized Odds Gap\")\n",
    "        else:\n",
    "            plt.ylabel(y.capitalize())\n",
    "        # plt.legend(title='Set:')\n",
    "        #plt.xlim((0, .05))\n",
    "        plt.savefig('{}{}.png'.format(save_dir, y), dpi=300)\n",
    "        print(y)\n",
    "        plt.clf()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "01982364",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "CPoF",
   "language": "python",
   "name": "cpof"
  },
  "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.10"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
