{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# FairPredictor XGBoost Examples\n",
    "This file contains demo code for an extended version of the example in Readme.md (additionally handling more fairness over multiple groups),  and enforcing a range of fairness definition on COMPAS.\n",
    "\n",
    "It is a modified version of [quickstart_autogluon.ipynb](quickstart_autogluon.ipynb)\n",
    "\n",
    "FairPredictor is a postprocessing approach for enforcing fairness, with support for a wide range of performance metrics and fairness criteria, and support for inferred attributes, i.e. it does not require access to protected attributes at test time. \n",
    "Under the hood, FairPredictor works by adjusting the decision boundary for each group individually. Where groups are not available, it makes use of inferred group membership to adjust decision boundaries.\n",
    "\n",
    "The key idea underlying this toolkit is that for a wide range of use cases, the most suitable classifier should do more than maximize some form of accuracy.\n",
    "We offer a general toolkit that allows different measures to be optimized and additional constraints to be imposed by tuning the behavior of a binary predictor on validation data.\n",
    "\n",
    "For example, classifiers can be tuned to maximize performance for a wide range of metrics such as:\n",
    "\n",
    "* Accuracy\n",
    "* Balanced Accuracy\n",
    "* F1 score\n",
    "* MCC\n",
    "* Custom utility functions\n",
    "\n",
    "While also approximately satisfying a wide range of group constraints such as:\n",
    "\n",
    "* Demographic Parity (The idea that positive decisions should occur at the same rates for all protected groups, for example for men at the same rate as for women)\n",
    "* Equal Opportunity (The recall should be the same for all protected groups)\n",
    "* Minimum recall constraints (The recall should be above a particular level for all groups)\n",
    "* Minimum Precision constraints (The precision should be above a particular level for all groups)\n",
    "* Custom Fairness Metrics\n",
    "\n",
    "The full set of constraints and objectives can be seen in Readme.md "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/opt/miniconda3/envs/ag/lib/python3.10/site-packages/tqdm/auto.py:21: TqdmWarning: IProgress not found. Please update jupyter and ipywidgets. See https://ipywidgets.readthedocs.io/en/stable/user_install.html\n",
      "  from .autonotebook import tqdm as notebook_tqdm\n"
     ]
    }
   ],
   "source": [
    "# Load and train a baseline classifier\n",
    "\n",
    "import dataset_loader\n",
    "from anonfair import FairPredictor\n",
    "from anonfair import group_metrics as gm\n",
    "import xgboost\n",
    "import pandas as pd\n",
    "import numpy as np\n",
    "\n",
    "\n",
    "train_data, val_data, test_data = dataset_loader.adult('sex')\n",
    "predictor=xgboost.XGBClassifier().fit(X=train_data['data'],y=train_data['target'])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "((24421, 14), (12210, 14), (12211, 14))"
      ]
     },
     "execution_count": 2,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "train_data['data'].shape,val_data['data'].shape,test_data['data'].shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Groups passed twice to fairpredictor both as part of the dataset and as an argument. The argument will be used.\n"
     ]
    }
   ],
   "source": [
    "# Modify predictor to enforce fairness over the val_data with respect to groups given by the column 'sex'\n",
    "fpredictor = FairPredictor(predictor,train_data, 'sex')\n",
    "# Maximize accuracy while enforcing that the demographic parity (the difference in positive decision rates between men and women is at most 0.02)\n",
    "fpredictor.fit(gm.accuracy,gm.demographic_parity,0.02)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([0, 0, 1, ..., 0, 0, 0])"
      ]
     },
     "execution_count": 4,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# Evaluate on test data\n",
    "fpredictor.predict(test_data)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>original</th>\n",
       "      <th>updated</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>Accuracy</th>\n",
       "      <td>0.870527</td>\n",
       "      <td>0.853984</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Balanced Accuracy</th>\n",
       "      <td>0.801481</td>\n",
       "      <td>0.771373</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>F1 score</th>\n",
       "      <td>0.712074</td>\n",
       "      <td>0.667661</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>MCC</th>\n",
       "      <td>0.631174</td>\n",
       "      <td>0.578837</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Precision</th>\n",
       "      <td>0.760996</td>\n",
       "      <td>0.733115</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Recall</th>\n",
       "      <td>0.669062</td>\n",
       "      <td>0.612936</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>ROC AUC</th>\n",
       "      <td>0.925576</td>\n",
       "      <td>0.826407</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "                   original   updated\n",
       "Accuracy           0.870527  0.853984\n",
       "Balanced Accuracy  0.801481  0.771373\n",
       "F1 score           0.712074  0.667661\n",
       "MCC                0.631174  0.578837\n",
       "Precision          0.760996  0.733115\n",
       "Recall             0.669062  0.612936\n",
       "ROC AUC            0.925576  0.826407"
      ]
     },
     "execution_count": 5,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# Evaluate a range of performance measures, and compare against original classifier on test data\n",
    "fpredictor.evaluate(test_data, verbose=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>original</th>\n",
       "      <th>updated</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>Statistical Parity</th>\n",
       "      <td>0.188330</td>\n",
       "      <td>0.021014</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Predictive Parity</th>\n",
       "      <td>0.024491</td>\n",
       "      <td>0.343701</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Equal Opportunity</th>\n",
       "      <td>0.113892</td>\n",
       "      <td>0.272117</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in False Negative Rate</th>\n",
       "      <td>0.113892</td>\n",
       "      <td>0.272117</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Equalized Odds</th>\n",
       "      <td>0.090796</td>\n",
       "      <td>0.164818</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Conditional Use Accuracy</th>\n",
       "      <td>0.052279</td>\n",
       "      <td>0.243413</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in Accuracy</th>\n",
       "      <td>0.090947</td>\n",
       "      <td>0.052501</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Treatment Equality</th>\n",
       "      <td>0.203908</td>\n",
       "      <td>5.251126</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "                                                 original   updated\n",
       "Statistical Parity                               0.188330  0.021014\n",
       "Predictive Parity                                0.024491  0.343701\n",
       "Equal Opportunity                                0.113892  0.272117\n",
       "Average Group Difference in False Negative Rate  0.113892  0.272117\n",
       "Equalized Odds                                   0.090796  0.164818\n",
       "Conditional Use Accuracy                         0.052279  0.243413\n",
       "Average Group Difference in Accuracy             0.090947  0.052501\n",
       "Treatment Equality                               0.203908  5.251126"
      ]
     },
     "execution_count": 6,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# Evaluate against a range of standard fairness definitions and compare against original classifier on test data\n",
    "fpredictor.evaluate_fairness(test_data, verbose=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th>Accuracy</th>\n",
       "      <th>Balanced Accuracy</th>\n",
       "      <th>F1 score</th>\n",
       "      <th>MCC</th>\n",
       "      <th>Precision</th>\n",
       "      <th>Recall</th>\n",
       "      <th>ROC AUC</th>\n",
       "      <th>Number of Datapoints</th>\n",
       "      <th>Positive Count</th>\n",
       "      <th>Negative Count</th>\n",
       "      <th>Positive Label Rate</th>\n",
       "      <th>Positive Prediction Rate</th>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th></th>\n",
       "      <th>Groups</th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th rowspan=\"4\" valign=\"top\">original</th>\n",
       "      <th>Overall</th>\n",
       "      <td>0.870527</td>\n",
       "      <td>0.801481</td>\n",
       "      <td>0.712074</td>\n",
       "      <td>0.631174</td>\n",
       "      <td>0.760996</td>\n",
       "      <td>0.669062</td>\n",
       "      <td>0.925576</td>\n",
       "      <td>12211.0</td>\n",
       "      <td>2922.0</td>\n",
       "      <td>9289.0</td>\n",
       "      <td>0.239292</td>\n",
       "      <td>0.210384</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>0.931324</td>\n",
       "      <td>0.773859</td>\n",
       "      <td>0.645408</td>\n",
       "      <td>0.614207</td>\n",
       "      <td>0.739766</td>\n",
       "      <td>0.572398</td>\n",
       "      <td>0.940112</td>\n",
       "      <td>4048.0</td>\n",
       "      <td>442.0</td>\n",
       "      <td>3606.0</td>\n",
       "      <td>0.109190</td>\n",
       "      <td>0.084486</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>0.840377</td>\n",
       "      <td>0.796955</td>\n",
       "      <td>0.723178</td>\n",
       "      <td>0.613236</td>\n",
       "      <td>0.764257</td>\n",
       "      <td>0.686290</td>\n",
       "      <td>0.908275</td>\n",
       "      <td>8163.0</td>\n",
       "      <td>2480.0</td>\n",
       "      <td>5683.0</td>\n",
       "      <td>0.303810</td>\n",
       "      <td>0.272816</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Maximum difference</th>\n",
       "      <td>0.090947</td>\n",
       "      <td>0.023096</td>\n",
       "      <td>0.077770</td>\n",
       "      <td>0.000970</td>\n",
       "      <td>0.024491</td>\n",
       "      <td>0.113892</td>\n",
       "      <td>0.031837</td>\n",
       "      <td>4115.0</td>\n",
       "      <td>2038.0</td>\n",
       "      <td>2077.0</td>\n",
       "      <td>0.194620</td>\n",
       "      <td>0.188330</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th rowspan=\"4\" valign=\"top\">updated</th>\n",
       "      <th>Overall</th>\n",
       "      <td>0.853984</td>\n",
       "      <td>0.771373</td>\n",
       "      <td>0.667661</td>\n",
       "      <td>0.578837</td>\n",
       "      <td>0.733115</td>\n",
       "      <td>0.612936</td>\n",
       "      <td>0.826407</td>\n",
       "      <td>12211.0</td>\n",
       "      <td>2922.0</td>\n",
       "      <td>9289.0</td>\n",
       "      <td>0.239292</td>\n",
       "      <td>0.200066</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>0.889081</td>\n",
       "      <td>0.869256</td>\n",
       "      <td>0.624268</td>\n",
       "      <td>0.591911</td>\n",
       "      <td>0.495352</td>\n",
       "      <td>0.843891</td>\n",
       "      <td>0.940112</td>\n",
       "      <td>4048.0</td>\n",
       "      <td>442.0</td>\n",
       "      <td>3606.0</td>\n",
       "      <td>0.109190</td>\n",
       "      <td>0.186018</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>0.836580</td>\n",
       "      <td>0.761956</td>\n",
       "      <td>0.680096</td>\n",
       "      <td>0.594671</td>\n",
       "      <td>0.839053</td>\n",
       "      <td>0.571774</td>\n",
       "      <td>0.908275</td>\n",
       "      <td>8163.0</td>\n",
       "      <td>2480.0</td>\n",
       "      <td>5683.0</td>\n",
       "      <td>0.303810</td>\n",
       "      <td>0.207032</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Maximum difference</th>\n",
       "      <td>0.052501</td>\n",
       "      <td>0.107300</td>\n",
       "      <td>0.055828</td>\n",
       "      <td>0.002761</td>\n",
       "      <td>0.343701</td>\n",
       "      <td>0.272117</td>\n",
       "      <td>0.031837</td>\n",
       "      <td>4115.0</td>\n",
       "      <td>2038.0</td>\n",
       "      <td>2077.0</td>\n",
       "      <td>0.194620</td>\n",
       "      <td>0.021014</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "                             Accuracy  Balanced Accuracy  F1 score       MCC  \\\n",
       "         Groups                                                                \n",
       "original Overall             0.870527           0.801481  0.712074  0.631174   \n",
       "         0                   0.931324           0.773859  0.645408  0.614207   \n",
       "         1                   0.840377           0.796955  0.723178  0.613236   \n",
       "         Maximum difference  0.090947           0.023096  0.077770  0.000970   \n",
       "updated  Overall             0.853984           0.771373  0.667661  0.578837   \n",
       "         0                   0.889081           0.869256  0.624268  0.591911   \n",
       "         1                   0.836580           0.761956  0.680096  0.594671   \n",
       "         Maximum difference  0.052501           0.107300  0.055828  0.002761   \n",
       "\n",
       "                             Precision    Recall   ROC AUC  \\\n",
       "         Groups                                              \n",
       "original Overall              0.760996  0.669062  0.925576   \n",
       "         0                    0.739766  0.572398  0.940112   \n",
       "         1                    0.764257  0.686290  0.908275   \n",
       "         Maximum difference   0.024491  0.113892  0.031837   \n",
       "updated  Overall              0.733115  0.612936  0.826407   \n",
       "         0                    0.495352  0.843891  0.940112   \n",
       "         1                    0.839053  0.571774  0.908275   \n",
       "         Maximum difference   0.343701  0.272117  0.031837   \n",
       "\n",
       "                             Number of Datapoints  Positive Count  \\\n",
       "         Groups                                                     \n",
       "original Overall                          12211.0          2922.0   \n",
       "         0                                 4048.0           442.0   \n",
       "         1                                 8163.0          2480.0   \n",
       "         Maximum difference                4115.0          2038.0   \n",
       "updated  Overall                          12211.0          2922.0   \n",
       "         0                                 4048.0           442.0   \n",
       "         1                                 8163.0          2480.0   \n",
       "         Maximum difference                4115.0          2038.0   \n",
       "\n",
       "                             Negative Count  Positive Label Rate  \\\n",
       "         Groups                                                    \n",
       "original Overall                     9289.0             0.239292   \n",
       "         0                           3606.0             0.109190   \n",
       "         1                           5683.0             0.303810   \n",
       "         Maximum difference          2077.0             0.194620   \n",
       "updated  Overall                     9289.0             0.239292   \n",
       "         0                           3606.0             0.109190   \n",
       "         1                           5683.0             0.303810   \n",
       "         Maximum difference          2077.0             0.194620   \n",
       "\n",
       "                             Positive Prediction Rate  \n",
       "         Groups                                        \n",
       "original Overall                             0.210384  \n",
       "         0                                   0.084486  \n",
       "         1                                   0.272816  \n",
       "         Maximum difference                  0.188330  \n",
       "updated  Overall                             0.200066  \n",
       "         0                                   0.186018  \n",
       "         1                                   0.207032  \n",
       "         Maximum difference                  0.021014  "
      ]
     },
     "execution_count": 7,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# Evaluate a range of performance measures per group, and compare against original classifier on test data\n",
    "fpredictor.evaluate_groups(test_data, verbose=True, return_original=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "train_data, val_data, test_data = dataset_loader.adult('sex')\n",
    "predictor=xgboost.XGBClassifier().fit(X=train_data['data'],y=train_data['target'])\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Groups passed twice to fairpredictor both as part of the dataset and as an argument. The argument will be used.\n"
     ]
    }
   ],
   "source": [
    "fpredictor = FairPredictor(predictor,val_data,'race') \n",
    "# Maximize accuracy while enforcing that the demographic parity (the difference in positive decision rates between men and women is at most 0.02)\n",
    "fpredictor.fit(gm.accuracy, gm.demographic_parity, .02)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>original</th>\n",
       "      <th>updated</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>Statistical Parity</th>\n",
       "      <td>0.109018</td>\n",
       "      <td>0.076311</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Predictive Parity</th>\n",
       "      <td>0.033117</td>\n",
       "      <td>0.186202</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Equal Opportunity</th>\n",
       "      <td>0.159850</td>\n",
       "      <td>0.219112</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in False Negative Rate</th>\n",
       "      <td>0.159850</td>\n",
       "      <td>0.219112</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Equalized Odds</th>\n",
       "      <td>0.100719</td>\n",
       "      <td>0.131864</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Conditional Use Accuracy</th>\n",
       "      <td>0.036888</td>\n",
       "      <td>0.130660</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in Accuracy</th>\n",
       "      <td>0.053225</td>\n",
       "      <td>0.037582</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Treatment Equality</th>\n",
       "      <td>0.259746</td>\n",
       "      <td>6.710528</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "                                                 original   updated\n",
       "Statistical Parity                               0.109018  0.076311\n",
       "Predictive Parity                                0.033117  0.186202\n",
       "Equal Opportunity                                0.159850  0.219112\n",
       "Average Group Difference in False Negative Rate  0.159850  0.219112\n",
       "Equalized Odds                                   0.100719  0.131864\n",
       "Conditional Use Accuracy                         0.036888  0.130660\n",
       "Average Group Difference in Accuracy             0.053225  0.037582\n",
       "Treatment Equality                               0.259746  6.710528"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# Unlike the previous case, we find that demographic parity is still high on test data, although it is improved.\n",
    "fpredictor.evaluate_fairness(test_data)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>original</th>\n",
       "      <th>updated</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>Statistical Parity</th>\n",
       "      <td>0.101929</td>\n",
       "      <td>0.019384</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Predictive Parity</th>\n",
       "      <td>0.128801</td>\n",
       "      <td>0.151661</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Equal Opportunity</th>\n",
       "      <td>0.160757</td>\n",
       "      <td>0.180761</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in False Negative Rate</th>\n",
       "      <td>0.160757</td>\n",
       "      <td>0.180761</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Equalized Odds</th>\n",
       "      <td>0.104002</td>\n",
       "      <td>0.104552</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Conditional Use Accuracy</th>\n",
       "      <td>0.092449</td>\n",
       "      <td>0.115558</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in Accuracy</th>\n",
       "      <td>0.064770</td>\n",
       "      <td>0.046175</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Treatment Equality</th>\n",
       "      <td>0.302790</td>\n",
       "      <td>3.496003</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "                                                 original   updated\n",
       "Statistical Parity                               0.101929  0.019384\n",
       "Predictive Parity                                0.128801  0.151661\n",
       "Equal Opportunity                                0.160757  0.180761\n",
       "Average Group Difference in False Negative Rate  0.160757  0.180761\n",
       "Equalized Odds                                   0.104002  0.104552\n",
       "Conditional Use Accuracy                         0.092449  0.115558\n",
       "Average Group Difference in Accuracy             0.064770  0.046175\n",
       "Treatment Equality                               0.302790  3.496003"
      ]
     },
     "execution_count": 11,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "fpredictor.evaluate_fairness()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAkAAAAHFCAYAAAAaD0bAAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjguNCwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8fJSN1AAAACXBIWXMAAA9hAAAPYQGoP6dpAABb/klEQVR4nO3deVxUVeMG8GdAYAABBQUGQ0AtgVBTUENFc99y6+0VM7dCTdPErZTXBVxxa/HVxCS3ck3LNUJR08wlFcVUyAVRXAYXVMCFbTi/P3yZnwPDMsAwwH2+n898as6ce+45F2Kezj33XpkQQoCIiIhIQowM3QEiIiKi8sYARERERJLDAERERESSwwBEREREksMARERERJLDAERERESSwwBEREREksMARERERJLDAERERESSwwBERACAdevWQSaTaX1Nnjy5XPvy/PlzhISE4PDhwwX288aNG3rvx+HDhws8Ju+//77e919S5XmMiCqraobuABFVLGvXroW7u7tGmZOTU7n24fnz55g1axYA4J133tH4rGfPnjhx4gQUCkW59Wf+/Plo3769RpmdnV257Z+Iyh4DEBFp8PLygo+PT7HqZmVlQSaToVq18vtTUrt2bdSuXbvM2nv+/DksLCwKrfP666/j7bffLrN9EpHh8RQYERVL7umgH3/8EZMmTUKdOnVgZmaGa9euAQDWrFmDJk2aQC6Xw9bWFv369UNcXJxGG8OGDUP16tVx7do19OjRA9WrV4ezszMmTZqEjIwMAMCNGzfUAWfWrFnqU07Dhg0DUPDpnQMHDqBjx46wtraGhYUFWrdujYMHD2rUCQkJgUwmw9mzZ/H++++jZs2aqF+/fqmPzZ9//omOHTvCysoKFhYWaNWqFX799Vet+85L23hcXV3x7rvvIjIyEs2aNYO5uTnc3d2xZs2afNufPHkSrVu3hlwuh5OTE4KCgpCVlVXqMRFVdQxARKRBpVIhOztb4/WqoKAgJCYmYuXKldizZw/s7e0RGhqKgIAAvPnmm/jll1+wdOlS/P333/D19cXVq1c1ts/KykLv3r3RsWNH7Nq1Cx9//DG+/vprLFy4EACgUCgQGRkJAAgICMCJEydw4sQJzJgxo8A+b9iwAV26dIG1tTXWr1+Pn376Cba2tujatWu+EAQA7733Hho0aIBt27Zh5cqVRR6TnJycAo/JkSNH0KFDB6SkpGD16tXYvHkzrKys0KtXL2zdurXItgty/vx5TJo0CRMmTMCuXbvQuHFjBAQE4I8//lDXiY2NRceOHfHkyROsW7cOK1euxLlz5zB37twS75dIMgQRkRBi7dq1AoDWV1ZWlvj9998FANG2bVuN7R4/fizMzc1Fjx49NMoTExOFmZmZGDhwoLps6NChAoD46aefNOr26NFDNGzYUP3+wYMHAoAIDg4usJ8JCQlCCCGePXsmbG1tRa9evTTqqVQq0aRJE9GiRQt1WXBwsAAgZs6cWaxjkjtmba+rV68KIYR4++23hb29vUhLS1Nvl52dLby8vMRrr70mcnJyNPZd1HiEEMLFxUXI5XJx8+ZNddmLFy+Era2t+OSTT9Rl/v7+wtzcXCQlJWns293dPV+bRKSJM0BEpOGHH37A6dOnNV6vrvH517/+pVH/xIkTePHihfoUVS5nZ2d06NAh3wyMTCZDr169NMoaN26Mmzdvlqi/x48fx6NHjzB06FCNGZqcnBx069YNp0+fxrNnzzS2yTuGoixcuDDfMXF2dsazZ8/w119/4f3330f16tXV9Y2NjTF48GDcvn0bly9fLtG43nrrLdStW1f9Xi6X44033tA4Tr///js6duwIBwcHjX37+/uXaJ9EUsJF0ESkwcPDo9BF0HmvvkpOTtZaDry8eiwqKkqjzMLCAnK5XKPMzMwM6enpJervvXv3AKDQy9IfPXoES0tL9XtdryCrV6+e1mPy4MEDCCEKHDvw/8dHV9quMjMzM8OLFy/U75OTk+Ho6JivnrYyItLEAEREOsm7kDf3i1qpVOare/fuXdSqVUuv/cltf9myZQVeqfXqDAmQfwwlVbNmTRgZGRU49lf7lxv6MjIyYGZmpq738OHDEu/fzs4OSUlJ+cq1lRGRJp4CI6JS8fX1hbm5OTZs2KBRfvv2bRw6dAgdO3bUuc3cgPDqbEdBWrdujRo1aiA2NhY+Pj5aX6ampjr3oTgsLS3RsmVL/PLLLxp9zcnJwYYNG/Daa6/hjTfeAPDyyi4A+PvvvzXa2LNnT4n33759exw8eFA9Cwa8XMRemsXXRFLBGSAiKpUaNWpgxowZ+M9//oMhQ4bggw8+QHJyMmbNmgW5XI7g4GCd27SysoKLiwt27dqFjh07wtbWFrVq1VKHiFdVr14dy5Ytw9ChQ/Ho0SO8//77sLe3x4MHD3D+/Hk8ePAAYWFhZTBS7UJDQ9G5c2e0b98ekydPhqmpKVasWIGLFy9i8+bN6tmmHj16wNbWFgEBAZg9ezaqVauGdevW4datWyXe9/Tp07F792506NABM2fOhIWFBb799tt8a56IKD/OABFRqQUFBeH777/H+fPn0bdvX4wdOxZvvvkmjh8/jtdff71Eba5evRoWFhbo3bs3mjdvjpCQkALrDho0CL///juePn2KTz75BJ06dUJgYCDOnj1bohkoXbRr1w6HDh2CpaUlhg0bhgEDBiAlJQW7d+/WWIxsbW2NyMhIWFlZYdCgQRg1ahS8vLwwbdq0Eu/by8sLBw4cgLW1NYYOHYqRI0eicePGhd4ygIhekgkhhKE7QURERFSeOANEREREksMARERERJLDAERERESSwwBEREREksMARERERJLDAERERESSwxshapGTk4O7d+/CysqqzG6ZT0RERPolhEBaWhqcnJxgZFT4HA8DkBZ3796Fs7OzobtBREREJXDr1i289tprhdZhANLCysoKwMsDaG1tbeDeEBERUXGkpqbC2dlZ/T1eGAYgLXJPe1lbWzMAERERVTLFWb7CRdBEREQkOQxAREREJDkMQERERCQ5DEBEREQkOQxAREREJDkMQERERCQ5DEBEREQkOQxAREREJDkMQERERCQ5vBM0EVEJqHIETiU8wv20dNhbydHCzRbGRmXz8GR9tk1VC39XSo4BiIgqHF3+qGurC6DI7UvzxRF5UYlZe2KhTElXlyls5Aju5YluXooSjlr/bVPVwt+V0pEJIYShO1HRpKamwsbGBikpKXwWGFVJr37517I0A2TAw6cZ5RYgCqPLH3VtdWtYmAAAnjzPKnD70nxxRF5UYvSGs8j7hzN35GGDmpX4y0efbVPVwt8V7XT5/mYA0oIBiHRVVBgoaVgoTcgoaFttX/6v0neAKIwuf9QLqqvNq9sDKPEXhypHoM3CQwUeOxkARxs5/pzSQecwqM+2qWrh70rBdPn+5ikwolJQ5QgsP3QNa48l4MkL7YGhpGGhtLMU2rbt3USBVX8kFBoaXg0+uZJS0jF6w9lCA8SrdUoSglQ5ArP2xGrtm8DLP+qz9sSis6cj8L9/L+7/veVuH7L7EgBZsfah7YvjVMKjAr90cttQpqTjVMIj+Na3K2bv9N82VS38XSkbDEAk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XcfZ2Rn79+/HhAkT0LhxY9SpUweBgYGYMmVKuY+vOHSZzZEBcLR5eUk8ERERlZxB7wNUUZXnfYBOxCfjg/CTxaorAxA2qBm6eSn02iciIqLKqFLcB4heauFmC4WNHEXd4lBhI2f4ISIiKiMMQAZmbCRDcC9PACgwBE3o9Dr+nNKB4YeIiKiMMABVAN28FAgb1AyONpqLmxU2cqwc1AyBnd7gYzCIiIjKkEEXQdP/6+alQGdPRz7/i4iIqBwwAFUgxkYy3uCQiIioHPAUGBEREUkOAxARERFJDgMQERERSQ4DEBEREUkOAxARERFJDgMQERERSQ4DEBEREUkOAxARERFJDgMQERERSQ4DEBEREUkOAxARERFJDgMQERERSQ4DEBEREUkOAxARERFJDgMQERERSQ4DEBEREUkOAxARERFJDgMQERERSQ4DEBEREUkOAxARERFJDgMQERERSQ4DEBEREUkOAxARERFJDgMQERERSQ4DEBEREUkOAxARERFJDgMQERERSQ4DEBEREUkOAxARERFJDgMQERERSQ4DEBEREUkOAxARERFJDgMQERERSQ4DEBEREUkOAxARERFJDgMQERERSQ4DEBEREUkOAxARERFJDgMQERERSQ4DEBEREUkOAxARERFJDgMQERERSQ4DEBEREUkOAxARERFJDgMQERERSY7BA9CKFSvg5uYGuVwOb29vHD16tND6GzduRJMmTWBhYQGFQoGPPvoIycnJWutu2bIFMpkMffv21UPPiYiIqLIyaADaunUrxo8fj2nTpuHcuXPw8/ND9+7dkZiYqLX+n3/+iSFDhiAgIACXLl3Ctm3bcPr0aQwfPjxf3Zs3b2Ly5Mnw8/PT9zCIiIiokjFoAPrqq68QEBCA4cOHw8PDA9988w2cnZ0RFhamtf7Jkyfh6uqKcePGwc3NDW3atMEnn3yCM2fOaNRTqVT48MMPMWvWLNSrV688hkJERESViMECUGZmJqKjo9GlSxeN8i5duuD48eNat2nVqhVu376NiIgICCFw7949bN++HT179tSoN3v2bNSuXRsBAQF66z8RERFVXtUMteOHDx9CpVLBwcFBo9zBwQFJSUlat2nVqhU2btwIf39/pKenIzs7G71798ayZcvUdY4dO4bVq1cjJiam2H3JyMhARkaG+n1qaqpugyEiIqJKxeCLoGUymcZ7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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "#To debug this, we can look at the Pareto Frontier of solutions considered, on the validation set\n",
    "fpredictor.plot_frontier() "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAkAAAAHFCAYAAAAaD0bAAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjguNCwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8fJSN1AAAACXBIWXMAAA9hAAAPYQGoP6dpAABcEklEQVR4nO3dd1gU1+I+8HdBYOkKCiwGATURCFZQgorGhi223Nxg7Ama6NWILYlcC1ixpXg1YiS2xBpNrCEa1GiMJSqWqBArBsuiESOgSFvO7w+/7M9ll7K4y4Lzfp5nn9w9e2bmnCPPnfc5c2ZGJoQQICIiIpIQM1M3gIiIiKiyMQARERGR5DAAERERkeQwABEREZHkMAARERGR5DAAERERkeQwABEREZHkMAARERGR5DAAERERkeQwABERAGDNmjWQyWQ6P5MmTarUtmRnZyM6OhoHDx4ssZ03btwwejsOHjxY4pi89dZbRj9+RVXmGBFVVzVM3QAiqlpWr14NHx8fjTJ3d/dKbUN2djZmzJgBAHj99dc1fuvZsyeOHTsGhUJRae2ZO3cuOnTooFHm7OxcaccnIsNjACIiDf7+/ggMDCxX3fz8fMhkMtSoUXn/V1KnTh3UqVPHYPvLzs6GjY1NqXVefvllvPbaawY7JhGZHi+BEVG5FF0O+vbbbzFx4kTUrVsXVlZWuHr1KgBg1apVaNq0KeRyOZycnNCvXz8kJydr7GPYsGGws7PD1atX0aNHD9jZ2cHDwwMTJ05Ebm4uAODGjRvqgDNjxgz1Jadhw4YBKPnyzr59+9CpUyc4ODjAxsYGbdq0wf79+zXqREdHQyaT4fTp03jrrbdQq1YtNGjQ4LnH5rfffkOnTp1gb28PGxsbtG7dGj/++KPOYxenqz9eXl544403sGfPHrRo0QLW1tbw8fHBqlWrtLY/fvw42rRpA7lcDnd3d0RGRiI/P/+5+0T0omMAIiINKpUKBQUFGp9nRUZGIjU1FcuXL8euXbvg4uKCmJgYhIeH49VXX8UPP/yAxYsX448//kBwcDCuXLmisX1+fj569+6NTp06YceOHXjvvffw+eefY/78+QAAhUKBPXv2AADCw8Nx7NgxHDt2DNOmTSuxzevWrUNoaCgcHBywdu1afPfdd3ByckLXrl21QhAAvPnmm2jYsCG2bNmC5cuXlzkmhYWFJY7JoUOH0LFjR2RkZGDlypXYuHEj7O3t0atXL2zevLnMfZfk3LlzmDhxIsaPH48dO3agSZMmCA8Px6+//qquk5SUhE6dOuHhw4dYs2YNli9fjjNnzmD27NkVPi6RZAgiIiHE6tWrBQCdn/z8fPHLL78IAKJdu3Ya2/3zzz/C2tpa9OjRQ6M8NTVVWFlZiQEDBqjLhg4dKgCI7777TqNujx49RKNGjdTf//77bwFAREVFldjOlJQUIYQQjx8/Fk5OTqJXr14a9VQqlWjatKlo1aqVuiwqKkoAENOnTy/XmBT1WdfnypUrQgghXnvtNeHi4iKysrLU2xUUFAh/f3/x0ksvicLCQo1jl9UfIYTw9PQUcrlc/PXXX+qyJ0+eCCcnJ/HBBx+oy8LCwoS1tbVIS0vTOLaPj4/WPolIE2eAiEjDN998g5MnT2p8nl3j869//Uuj/rFjx/DkyRP1JaoiHh4e6Nixo9YMjEwmQ69evTTKmjRpgr/++qtC7T169CgePHiAoUOHaszQFBYWolu3bjh58iQeP36ssU3xPpRl/vz5WmPi4eGBx48f4/fff8dbb70FOzs7dX1zc3MMHjwYt27dwqVLlyrUr2bNmqFevXrq73K5HK+88orGOP3yyy/o1KkTXF1dNY4dFhZWoWMSSQkXQRORBl9f31IXQRe/+yo9PV1nOfD07rGEhASNMhsbG8jlco0yKysr5OTkVKi9d+/eBYBSb0t/8OABbG1t1d/1vYOsfv36Osfk77//hhCixL4D/3989KXrLjMrKys8efJE/T09PR1ubm5a9XSVEZEmBiAi0kvxhbxFJ2qlUqlV986dO6hdu7ZR21O0/yVLlpR4p9azMySAdh8qqlatWjAzMyux78+2ryj05ebmwsrKSl3v/v37FT6+s7Mz0tLStMp1lRGRJl4CI6LnEhwcDGtra6xbt06j/NatWzhw4AA6deqk9z6LAsKzsx0ladOmDWrWrImkpCQEBgbq/FhaWurdhvKwtbVFUFAQfvjhB422FhYWYt26dXjppZfwyiuvAHh6ZxcA/PHHHxr72LVrV4WP36FDB+zfv189CwY8XcT+PIuviaSCM0BE9Fxq1qyJadOm4b///S+GDBmCd955B+np6ZgxYwbkcjmioqL03qe9vT08PT2xY8cOdOrUCU5OTqhdu7Y6RDzLzs4OS5YswdChQ/HgwQO89dZbcHFxwd9//41z587h77//RmxsrAF6qltMTAy6dOmCDh06YNKkSbC0tMSyZctw4cIFbNy4UT3b1KNHDzg5OSE8PBwzZ85EjRo1sGbNGty8ebPCx546dSp27tyJjh07Yvr06bCxscGXX36pteaJiLRxBoiInltkZCS+/vprnDt3Dn379sWYMWPw6quv4ujRo3j55ZcrtM+VK1fCxsYGvXv3RsuWLREdHV1i3UGDBuGXX37Bo0eP8MEHH6Bz586IiIjA6dOnKzQDpY/27dvjwIEDsLW1xbBhw9C/f39kZGRg586dGouRHRwcsGfPHtjb22PQoEEYOXIk/P39MWXKlAof29/fH/v27YODgwOGDh2K999/H02aNCn1kQFE9JRMCCFM3QgiIiKiysQZICIiIpIcBiAiIiKSHAYgIiIikhwGICIiIpIcBiAiIiKSHAYgIiIikhw+CFGHwsJC3LlzB/b29gZ7ZD4REREZlxACWVlZcHd3h5lZ6XM8DEA63LlzBx4eHqZuBhEREVXAzZs38dJLL5VahwFIB3t7ewBPB9DBwcHErSEiIqLyyMzMhIeHh/o8XhoGIB2KLns5ODgwABEREVUz5Vm+wkXQREREJDkMQERERCQ5DEBEREQkOQxAREREJDkMQERERCQ5DEBEREQkOQxAREREJDkMQERERCQ5DEBEREQkOQxAREREVLnynwC/Lnr6XxNhACIiIqLKlbQTODALSN5lsiYwABEREVHlurhN878mwJehEtELT1UocCLlAe5l5cDFXo5W3k4wNyv7ZYlSJZXxkko/q4RH94BjXwKFBU+/X9v/9L9X9wF7pzz932Y1gOAxgF2dSmkSAxARVVt5BYX49tgN3EjPBiDQ7KWacK9lo3Ei23NBiRm7kqDMyFFvp3CUI6qXH7r5K16Yk+CTPBXmxifhRno2vJxt8N8efrC2NNeoU56+ljVeLwqp9LPKyEoDjscCqlxAZgbg//7uClXA8WWAKATMrQD/f1VaAJIJIUSlHKkayczMhKOjIzIyMuDg4GDq5hCRDjHxSYg7nIJCHf8PVnQiA4BR606jeJWiU/777byx85yy2p8ER3xzEglJ97TKu/i5IG5ISwDlO+HvuaAsdbxiB7WoVuNSEqn0s8q5lwx8NwRIv/o08BSRmQHODYG3vwFcfJ/rEPqcvxmAdGAAIqo8Jc1KlDZbEROfhK9+TSlz3zVtLPAwO1+v9lS3k2BJ4adIFz8X/KvFS2We8Lv4uaHt/AMaAal4XTdHOX77pGO1nCEroioUkuhnlfXkH2BBfe0A9PF1wLrWc+9en/M3L4ERkVE8z+WW3k0VJc7MdPRxRdzhssMPAL3DDwAIPD0JztiVhC5+bmWGMVN6kqcqNfwAQELSPZy/laEVfgDNvtrLLUoMBUV1lRk5OJHyAMENnJ+n2SZ1IuWBJPpZZf11VDP8AE+//3UM8OlRqU1hACIig3ueyy3KjBydsztpGTkYte40/tWirs7LXob07Ekw40lelV0rMjc+qVz10jJzS/ytqK/HrqWXa1/3skoOD9VBedtf3ftZZSXvfPrfesFAl1nAz1OBm8efljMAEVF1VlKwKQowRZdbZuxK0jkrUZKi2Yr482mGa2wZ9iWlYdWRG6X2xZQh6Onib0Mp37+Gi71cq6z4DFmAZy0k/vVPlZsxA3S3/3nqkZ4UzYE6vkDrDwEzc+DdeODoEqBG5Y83AxARGYyqUJQYbPS53FISASA7X/WcrSy/bWdvl9mXostkpuDlbIPDVwyzr+D6tfH96dtIy8jR2eeitTGtvJ00ynXN9pnJoDFLV1VmzACglbcTFI5yvftJBvLaSM3vZuZA23EmaQofhEhEBlPe9RXlvdxSkueNG2VtLwPgbGuJB49LXkP07GUyU/lvD7/n3ocMTwPKaw2c1XfOFR+fou9Fvx+7lo6Vv5/AtPg9GL11F+7mXoOZ/Lb6A6vbkFncV29fNGO254Lyudv7vMzNZOXqZ1WZsSLj4QwQERlM+ddNPN8inp5NFNj9R8VPpm7/t9B6xf+tNXq2NUWnvT7N3LHqyI0y92XKtSKHLt+DZQ0z5BUUll1Zh+LBxtHaEu+28cL2s3fw4HGeup7bM48VaDv/ANKyb8Gu4SIAgI13yft/dHUSRH7tKjNjVqSbvwKxg1pozVy5VaGZKjI+BiAiMpjyrpso63JLSYouTyzu3xx1a8pLfA6QLtN6+qK2vZXGmpTm9WqVeBJ0tLYsVwAy1VqRktZa6aN4sHl2HOzl5nirxUsIfVWBVt5OSEhKUx/PTF7youpnycxzIf5vEq2q3V3VzV+BLn5uVfLuPqocDEBEZDDlXV9RdLll1LrTkKF880HFL09E9vDDxFAfrD16A58lXMKTfN2zIEXHHNbGW+vkVtpJUFUoquxakdLWWpVlTIcGeNnVXt3XZ4PNs7JyVFh99C9Y1jBDK2+nCh+vuKp0d5W5maxKhDEyDa4BIiKD0Wd9RdFlCDdHzRkUhaMcH7TzhqJYuZujXOuuK8saZhjRrj4+D2sGWTmOWVKbgxs4o0+zughu4KyuV5XXipS11qo0bRrWUfcVQJnB5qtfU7Bk/5UKH6843l1FVQVngIjIoPRZX1HaDMzH3XzLfXnCWGs6qupakYrOoiiKzViVN0h9/dv1Ch2vuFo2Fry7iqoMBiAiMjh91leUdBlC38sTxlrTURXXilR0FqV/y3oa7U7LeFKu7R7lGubRA3zvElUlDEBEZBSmWF9hrGNWtbUiRWut9L0s5VXbRuP7s3d6laWmtQUynuQ/V4h5mJ1fZRZBE3ENEBFRNWNuJsO0nvo/A+jG/cca353srMq97bttnt7v/rzzXlVpETRJG2eAiIiqoVq2lnpv8/m+K7j5IBttXq4DNwc5XOzLF4CcbC0wpmNDNHKzw4xdSUjLLt92QqVdj4ugqapgACIiqoYqOpOy9fRtbD19GwDg5mCFmjYWeJhd8hOvAWB2H38ATx+U+HHXRnjw2BvCwhd21ir4KhyQrMxE+qNcfH34OjJyCgA8DT8iv7Z6H3zFBFU1DEBERNWQIWZSSntLfJEP2nnDzEym9aDEovd7NXVRoKnL07IGjo0wat1pALqfrs1XTFBVwjVARETVUNFCaEO8F62mjQXcHDQDlbOtJZYNaIHm9Wph1LrTWguudb3fq6RnO+l6hhORqcmEELwzsZjMzEw4OjoiIyMDDg4Opm4OEZFORa/DAJ7/FvP14UEwM5Np3OoPaL8i41lFl7V++6SjxsyOqlBUqccGkHToc/7mJTAiomqqpAc1VsT9x7no06yuRtmxa+ml7rek93tVtccGEOnCAEREVI0Vf1DjjfvZ+GLfZb1nhHStKSrvQuuqcms7Z55IHwxARETVXPEZl0ZudojemYS0zLKDSWl3Z5V3oXVVuLV9zwWl1kyYwsSvLKGqjYugiYheMN38FTgyuSPGd365XPVLujurrIXWMmi/X8wUitZClWehNlERBiAioheQuZkMEZ1fwfJBLaBw1D1Doyjj7ixzMxmiej194nTxEFRVbm1XFYoS32hfVDZjVxJUhbzfhzTxEhgR0Qvs2TVCaRlP8OBxHpzsrODmUL41MiUttHarIpeXynqjfUkLtYkYgIiIXnDPe1dW8YXWVWmBcXVbqE1Vh8kvgS1btgze3t6Qy+UICAjA4cOHS62/fv16NG3aFDY2NlAoFHj33XeRnp6uUefhw4cYPXo0FAoF5HI5fH19ER8fb8xuEBG90IpCVJ9mdRHcwLlKhB+gei3UpqrFpAFo8+bNGDduHKZMmYIzZ84gJCQE3bt3R2pqqs76v/32G4YMGYLw8HBcvHgRW7ZswcmTJzF8+HB1nby8PHTp0gU3btzA1q1bcenSJcTFxaFu3bo690lERNVXdVmoTVWPSZ8EHRQUhBYtWiA2NlZd5uvri759+yImJkar/qJFixAbG4tr166py5YsWYIFCxbg5s2bAIDly5dj4cKF+PPPP2FhYVGhdvFJ0ERE1UdJT8QuCkV8DYd06HP+NtkMUF5eHhITExEaGqpRHhoaiqNHj+rcpnXr1rh16xbi4+MhhMDdu3exdetW9OzZU11n586dCA4OxujRo+Hq6gp/f3/MnTsXKpWqxLbk5uYiMzNT40NERNUD30FGFWGyRdD379+HSqWCq6urRrmrqyvS0tJ0btO6dWusX78eYWFhyMnJQUFBAXr37o0lS5ao61y/fh0HDhzAwIEDER8fjytXrmD06NEoKCjA9OnTde43JiYGM2bMMFzniIioUlXlhdpUNZl8EbRMpvnHKYTQKiuSlJSEsWPHYvr06UhMTMSePXuQkpKCkSNHqusUFhbCxcUFK1asQEBAAPr3748pU6ZoXGYrLjIyEhkZGepP0eU0IiKqPqrqQm2qmkw2A1S7dm2Ym5trzfbcu3dPa1aoSExMDNq0aYOPPvoIANCkSRPY2toiJCQEs2fPhkKhgEKhgIWFBczNzdXb+fr6Ii0tDXl5ebC0tNTar5WVFaysrAzYOyIiIqrKTDYDZGlpiYCAACQkJGiUJyQkoHXr1jq3yc7OhpmZZpOLgk7RWu42bdrg6tWrKCwsVNe5fPkyFAqFzvBDRERE0mPSS2ATJkzA119/jVWrViE5ORnjx49Hamqq+pJWZGQkhgwZoq7fq1cv/PDDD4iNjcX169dx5MgRjB07Fq1atYK7uzsAYNSoUUhPT0dERAQuX76MH3/8EXPnzsXo0aNN0kciIiKqekz6JOiwsDCkp6dj5syZUCqV8Pf3R3x8PDw9PQEASqVS45lAw4YNQ1ZWFpYuXYqJEyeiZs2a6NixI+bPn6+u4+HhgZ9//hnjx49HkyZNULduXUREROCTTz6p9P4RERFR1WTS5wBVVXwOEBmaqlDw7hQiIiPT5/zNd4ERGdmeC0qtF0kqqsiLJImIpMrkt8ETvciKnlBb/G3VaRk5GLXuNPZcUJqoZURE0sYARGQkqkKBGbuSoOsac1HZjF1JUBXyKjQRUWVjACIykhMpD7Rmfp4lACgzcnAi5UHlNYqIiAAwABEZzb2sksNPReoREZHhMAARGYmLvbzsSnrUIyIiw2EAIjKSVt5OUDjKUdLN7jI8vRuslbdTZTaLiIjAAERkNOZmMkT18gMArRBU9D2qlx+fB0REZAIMQERG1M1fgdhBLeDmqHmZy81RjthBLfgcICIiE+GDEImMrJu/Al383PgkaCKiKoQBiKgSmJvJENzA2dTNICKi/8NLYERERCQ5DEBEREQkOQxAREREJDkMQERERCQ5DEBEREQkOQxAREREJDkMQERERCQ5DEBEREQkOQxAREREJDkMQERERCQ5DEBEREQkOQxAREREJDkMQERERCQ5DEBEREQkOQxAREREJDkMQERERCQ5DEBEREQkOQxAREREJDkMQERERCQ5DEBEREQkOQxAREREJDkMQERERCQ5DEBEREQkOQxAREREJDkMQERERCQ5DEBEREQkOQxAREREJDkMQERERCQ5DEBEREQkOQxAREREJDkMQERERCQ5DEBEREQkOQxAREREJDkMQERERCQ5DEBEREQkOQxAREREJDkMQERERCQ5DEBEREQkOQxAREREJDkmD0DLli2Dt7c35HI5AgICcPjw4VLrr1+/Hk2bNoWNjQ0UCgXeffddpKen66y7adMmyGQy9O3b1wgtJyIiourKpAFo8+bNGDduHKZMmYIzZ84gJCQE3bt3R2pqqs76v/32G4YMGYLw8HBcvHgRW7ZswcmTJzF8+HCtun/99RcmTZqEkJAQY3eDiIiIqhmTBqDPPvsM4eHhGD58OHx9ffHFF1/Aw8MDsbGxOusfP34cXl5eGDt2LLy9vdG2bVt88MEHOHXqlEY9lUqFgQMHYsaMGahfv35ldIWIiIiqEZMFoLy8PCQmJiI0NFSjPDQ0FEePHtW5TevWrXHr1i3Ex8dDCIG7d+9i69at6Nmzp0a9mTNnok6dOggPDzda+4mIiKj6qmGqA9+/fx8qlQqurq4a5a6urkhLS9O5TevWrbF+/XqEhYUhJycHBQUF6N27N5YsWaKuc+TIEaxcuRJnz54td1tyc3ORm5ur/p6ZmalfZ4iIiKhaMfkiaJlMpvFdCKFVViQpKQljx47F9OnTkZiYiD179iAlJQUjR44EAGRlZWHQoEGIi4tD7dq1y92GmJgYODo6qj8eHh4V7xARERFVeTIhhDDFgfPy8mBjY4MtW7agX79+6vKIiAicPXsWhw4d0tpm8ODByMnJwZYtW9Rlv/32G0JCQnDnzh3cvXsXzZs3h7m5ufr3wsJCAICZmRkuXbqEBg0aaO1X1wyQh4cHMjIy4ODgYJD+EhERkXFlZmbC0dGxXOdvk10Cs7S0REBAABISEjQCUEJCAvr06aNzm+zsbNSoodnkorAjhICPjw/Onz+v8fvUqVORlZWFxYsXlzizY2VlBSsrq+fpDhEREVUjJgtAADBhwgQMHjwYgYGBCA4OxooVK5Camqq+pBUZGYnbt2/jm2++AQD06tULI0aMQGxsLLp27QqlUolx48ahVatWcHd3BwD4+/trHKNmzZo6y4mIiEi6TBqAwsLCkJ6ejpkzZ0KpVMLf3x/x8fHw9PQEACiVSo1nAg0bNgxZWVlYunQpJk6ciJo1a6Jjx46YP3++qbpARERE1ZDJ1gBVZfpcQyQiIqKqQZ/zt8nvAiMiIiKqbAxAREREJDkMQERERCQ5DEBEREQkOQxAREREJDkMQERERCQ5DEBEREQkOQxAREREJDkMQERERCQ5DEBEREQkOQxAREREJDkMQERERCQ5DEBEREQkOQxAREREJDkMQERERCQ5DEBEREQkOQxAREREJDkMQERERCQ5DEBEREQkOQxAREREJDkMQERERCQ5DEBEREQkOQxAREREJDkMQERERCQ5DEBEREQkOQxAREREJDkMQERERCQ5DEBEREQkOQxAREREJDkMQERERCQ5DEBEREQkOQxAREREJDkMQERERCQ5DEBEREQkOQxAREREJDkMQERERCQ5DEBEREQkOQxAREREJDkMQERERCQ5DEBEREQkOXoHIC8vL8ycOROpqanGaA8RERGR0ekdgCZOnIgdO3agfv366NKlCzZt2oTc3FxjtI2IiIjIKPQOQB9++CESExORmJgIPz8/jB07FgqFAmPGjMHp06eN0UYiIiIig5IJIcTz7CA/Px/Lli3DJ598gvz8fPj7+yMiIgLvvvsuZDKZodpZqTIzM+Ho6IiMjAw4ODiYujlERERUDvqcv2tU9CD5+fnYtm0bVq9ejYSEBLz22msIDw/HnTt3MGXKFOzbtw8bNmyo6O6JiIiIjEbvAHT69GmsXr0aGzduhLm5OQYPHozPP/8cPj4+6jqhoaFo166dQRtKREREZCh6B6CWLVuiS5cuiI2NRd++fWFhYaFVx8/PD/379zdIA4mIiIgMTe8AdP36dXh6epZax9bWFqtXr65wo4iIiIiMSe+7wO7du4fff/9dq/z333/HqVOnDNIoIiIiImPSOwCNHj0aN2/e1Cq/ffs2Ro8ebZBGERERERmT3gEoKSkJLVq00Cpv3rw5kpKSDNIoIiIiImPSOwBZWVnh7t27WuVKpRI1auh/V/2yZcvg7e0NuVyOgIAAHD58uNT669evR9OmTWFjYwOFQoF3330X6enp6t/j4uIQEhKCWrVqoVatWujcuTNOnDihd7uIiIjoxaV3AOrSpQsiIyORkZGhLnv48CH++9//okuXLnrta/PmzRg3bhymTJmCM2fOICQkBN27dy/xPWO//fYbhgwZgvDwcFy8eBFbtmzByZMnMXz4cHWdgwcP4p133sEvv/yCY8eOoV69eggNDcXt27f17SoRERG9oPR+EvTt27fRrl07pKeno3nz5gCAs2fPwtXVFQkJCfDw8Cj3voKCgtCiRQvExsaqy3x9fdG3b1/ExMRo1V+0aBFiY2Nx7do1ddmSJUuwYMECneuSAEClUqFWrVpYunQphgwZUq528UnQRERE1Y8+52+9Z4Dq1q2LP/74AwsWLICfnx8CAgKwePFinD9/Xq/wk5eXh8TERISGhmqUh4aG4ujRozq3ad26NW7duoX4+HgIIXD37l1s3boVPXv2LPE42dnZyM/Ph5OTU4l1cnNzkZmZqfEhIiKiF1eFXoVha2uL999//7kOfP/+fahUKri6umqUu7q6Ii0tTec2rVu3xvr16xEWFoacnBwUFBSgd+/eWLJkSYnHmTx5MurWrYvOnTuXWCcmJgYzZsyoWEeIiIio2qnwu8CSkpKQmpqKvLw8jfLevXvrtZ/iL0wVQpT4EtWkpCSMHTsW06dPR9euXaFUKvHRRx9h5MiRWLlypVb9BQsWYOPGjTh48CDkcnmJbYiMjMSECRPU3zMzM/WazSIiIqLqpUJPgu7Xrx/Onz8PmUyGoiVERaFFpVKVaz+1a9eGubm51mzPvXv3tGaFisTExKBNmzb46KOPAABNmjSBra0tQkJCMHv2bCgUCnXdRYsWYe7cudi3bx+aNGlSalusrKxgZWVVrnYTERFR9af3GqCIiAh4e3vj7t27sLGxwcWLF/Hrr78iMDAQBw8eLPd+LC0tERAQgISEBI3yhIQEtG7dWuc22dnZMDPTbLK5uTkA4Nm13AsXLsSsWbOwZ88eBAYGlrtNREREJA16zwAdO3YMBw4cQJ06dWBmZgYzMzO0bdsWMTExGDt2LM6cOVPufU2YMAGDBw9GYGAggoODsWLFCqSmpmLkyJEAnl6aun37Nr755hsAQK9evTBixAjExsaqL4GNGzcOrVq1gru7O4Cnl72mTZuGDRs2wMvLSz3DZGdnBzs7O327S0RERC8gvQOQSqVSB4natWvjzp07aNSoETw9PXHp0iW99hUWFob09HTMnDkTSqUS/v7+iI+PV79sValUajwTaNiwYcjKysLSpUsxceJE1KxZEx07dsT8+fPVdZYtW4a8vDy89dZbGseKiopCdHS0vt0lIiKiF5DezwEKCQnBxIkT0bdvXwwYMAD//PMPpk6dihUrViAxMREXLlwwVlsrDZ8DREREVP3oc/7WewZo6tSpePz4MQBg9uzZeOONNxASEgJnZ2ds3ry5Yi0mIiIiqkR6zwDp8uDBA9SqVavE29erG84AERERVT9GexJ0QUEBatSooXWZy8nJ6YUJP0RERPTi0ysA1ahRA56enuV+1g8RERFRVaT3c4CmTp2KyMhIPHjwwBjtISIiIjI6vRdB/+9//8PVq1fh7u4OT09P2Nraavx++vRpgzWOiIiIyBj0DkB9+/Y1QjOIiIiIKo9B7gJ70fAuMCIiourHaHeBEREREb0I9L4EZmZmVuot77xDjIiIiKo6vQPQtm3bNL7n5+fjzJkzWLt2LWbMmGGwhhEREREZi8HWAG3YsAGbN2/Gjh07DLE7k+IaICIiourHJGuAgoKCsG/fPkPtjoiIiMhoDBKAnjx5giVLluCll14yxO6IiIiIjErvNUDFX3oqhEBWVhZsbGywbt06gzaOiIiIyBj0DkCff/65RgAyMzNDnTp1EBQUhFq1ahm0cURERETGoHcAGjZsmBGaQURERFR59F4DtHr1amzZskWrfMuWLVi7dq1BGkVERERkTHoHoHnz5qF27dpa5S4uLpg7d65BGkVERERkTHpfAvvrr7/g7e2tVe7p6YnU1FSDNIqIKoeqUOBEygPcy8qBi70crbydYG5W8pPeiYheFHoHIBcXF/zxxx/w8vLSKD937hycnZ0N1S4iMrI9F5SYsSsJyowcdZnCUY6oXn7o5q8wYcuIiIxP70tg/fv3x9ixY/HLL79ApVJBpVLhwIEDiIiIQP/+/Y3RRiIysD0XlBi17rRG+AGAtIwcjFp3GnsuKE3UMiKiyqH3DNDs2bPx119/oVOnTqhR4+nmhYWFGDJkCNcAEVUDqkKBGbuSoOsdOAKADMCMXUno4ufGy2FE9MLSOwBZWlpi8+bNmD17Ns6ePQtra2s0btwYnp6exmgfERnYiZQHWjM/zxIAlBk5OJHyAMENeFmbiF5MegegIi+//DJefvllQ7aFiCrBvaySw09F6hERVUd6rwF66623MG/ePK3yhQsX4t///rdBGkVExuNiLzdoPSKi6kjvAHTo0CH07NlTq7xbt2749ddfDdIoIjKeVt5OUDjKUdLqHhme3g3WytupMptFRFSp9A5Ajx49gqWlpVa5hYUFMjMzDdIoIjIeczMZonr5AYBWCCr6HtXLjwugieiFpncA8vf3x+bNm7XKN23aBD8/P4M0ioiMq5u/ArGDWsDNUfMyl5ujHLGDWvA5QET0wtN7EfS0adPwr3/9C9euXUPHjh0BAPv378eGDRuwdetWgzeQiIyjm78CXfzc+CRoIpIkvQNQ7969sX37dsydOxdbt26FtbU1mjZtigMHDsDBwcEYbSQiIzE3k/FWdyKSJJkQQtfz0Mrt4cOHWL9+PVauXIlz585BpVIZqm0mk5mZCUdHR2RkZDDUERERVRP6nL/1XgNU5MCBAxg0aBDc3d2xdOlS9OjRA6dOnaro7oiIiIgqjV6XwG7duoU1a9Zg1apVePz4Md5++23k5+fj+++/5wJoIiIiqjbKPQPUo0cP+Pn5ISkpCUuWLMGdO3ewZMkSY7aNiIiIyCjKPQP0888/Y+zYsRg1ahRfgUFERETVWrlngA4fPoysrCwEBgYiKCgIS5cuxd9//23MthEREREZRbkDUHBwMOLi4qBUKvHBBx9g06ZNqFu3LgoLC5GQkICsrCxjtpOIiIjIYJ7rNvhLly5h5cqV+Pbbb/Hw4UN06dIFO3fuNGT7TIK3wRMREVU/lXIbPAA0atQICxYswK1bt7Bx48bn2RURERFRpXnuByG+iDgDREREVP1U2gwQERERUXXEAERERESSwwBEREREksMARERERJLDAERERESSwwBEREREksMARERERJLDAERERESSwwBEREREksMARERERJJj8gC0bNkyeHt7Qy6XIyAgAIcPHy61/vr169G0aVPY2NhAoVDg3XffRXp6ukad77//Hn5+frCysoKfnx+2bdtmzC4QERFRNWPSALR582aMGzcOU6ZMwZkzZxASEoLu3bsjNTVVZ/3ffvsNQ4YMQXh4OC5evIgtW7bg5MmTGD58uLrOsWPHEBYWhsGDB+PcuXMYPHgw3n77bfz++++V1S0iIiKq4kz6MtSgoCC0aNECsbGx6jJfX1/07dsXMTExWvUXLVqE2NhYXLt2TV22ZMkSLFiwADdv3gQAhIWFITMzEz/99JO6Trdu3VCrVq1yv7GeL0MlIiKqfqrFy1Dz8vKQmJiI0NBQjfLQ0FAcPXpU5zatW7fGrVu3EB8fDyEE7t69i61bt6Jnz57qOseOHdPaZ9euXUvcJwDk5uYiMzNT40NEREQvLpMFoPv370OlUsHV1VWj3NXVFWlpaTq3ad26NdavX4+wsDBYWlrCzc0NNWvWxJIlS9R10tLS9NonAMTExMDR0VH98fDweI6eERERUVVn8kXQMplM47sQQqusSFJSEsaOHYvp06cjMTERe/bsQUpKCkaOHFnhfQJAZGQkMjIy1J+iy2lERET0YqphqgPXrl0b5ubmWjMz9+7d05rBKRITE4M2bdrgo48+AgA0adIEtra2CAkJwezZs6FQKODm5qbXPgHAysoKVlZWz9kjIiIiqi5MNgNkaWmJgIAAJCQkaJQnJCSgdevWOrfJzs6GmZlmk83NzQE8neUBgODgYK19/vzzzyXuk4iIiKTHZDNAADBhwgQMHjwYgYGBCA4OxooVK5Camqq+pBUZGYnbt2/jm2++AQD06tULI0aMQGxsLLp27QqlUolx48ahVatWcHd3BwBERESgXbt2mD9/Pvr06YMdO3Zg3759+O2330zWTyIiIqpaTBqAwsLCkJ6ejpkzZ0KpVMLf3x/x8fHw9PQEACiVSo1nAg0bNgxZWVlYunQpJk6ciJo1a6Jjx46YP3++uk7r1q2xadMmTJ06FdOmTUODBg2wefNmBAUFVXr/iIiIqGoy6XOAqio+B4iIiKj6qRbPASIiIiIyFQYgIiIikhwGICIiIpIcBiAiIiKSHAYgIiIikhwGICIiIpIcBiAiIiKSHAYgIiIikhwGICIiIpIcBiAiIiKSHAYgIiIikhwGICIiIpIcBiAiIiKSHAYgIiIikhwGICIiIpIcBiAiIiKSHAYgIiIikhwGICIiIpIcBiAiIiKSHAYgIiIikhwGICIiIpIcBiAiIiKSHAYgIiIikhwGICIiIpIcBiAiIiKSHAYgIiIikhwGICIiIpIcBiAiIiKSHAYgIiIikhwGICIiIpIcBiAiIiKSHAYgIiIikhwGICIiIpIcBiAiIiKSHAYgIiIikhwGICIiIpIcBiAiIiKSHAYgIiIikhwGICIiIpIcBiAiIiKSHAYgIiIikhwGICIiIpIcBiAiIiKSHAYgIiIikhwGICIiIpIcBiAiIiKSHAYgIiIikhwGICIiIpIcBiAiIiKSHAYgIiIikhyTB6Bly5bB29sbcrkcAQEBOHz4cIl1hw0bBplMpvV59dVXNep98cUXaNSoEaytreHh4YHx48cjJyfH2F0hIiKiasKkAWjz5s0YN24cpkyZgjNnziAkJATdu3dHamqqzvqLFy+GUqlUf27evAknJyf8+9//VtdZv349Jk+ejKioKCQnJ2PlypXYvHkzIiMjK6tbREREVMXJhBDCVAcPCgpCixYtEBsbqy7z9fVF3759ERMTU+b227dvx5tvvomUlBR4enoCAMaMGYPk5GTs379fXW/ixIk4ceJEqbNLz8rMzISjoyMyMjLg4OCgZ6+IiIjIFPQ5f5tsBigvLw+JiYkIDQ3VKA8NDcXRo0fLtY+VK1eic+fO6vADAG3btkViYiJOnDgBALh+/Tri4+PRs2fPEveTm5uLzMxMjQ8RERG9uGqY6sD379+HSqWCq6urRrmrqyvS0tLK3F6pVOKnn37Chg0bNMr79++Pv//+G23btoUQAgUFBRg1ahQmT55c4r5iYmIwY8aMinWEiIiIqh2TBaAiMplM47sQQqtMlzVr1qBmzZro27evRvnBgwcxZ84cLFu2DEFBQbh69SoiIiKgUCgwbdo0nfuKjIzEhAkT1N8zMzPh4eGhf2eIiKoRlUqF/Px8UzeDSC+WlpYwM3v+C1gmC0C1a9eGubm51mzPvXv3tGaFihNCYNWqVRg8eDAsLS01fps2bRoGDx6M4cOHAwAaN26Mx48f4/3338eUKVN0DpqVlRWsrKyes0dERNWDEAJpaWl4+PChqZtCpDczMzN4e3trnf/1ZbIAZGlpiYCAACQkJKBfv37q8oSEBPTp06fUbQ8dOoSrV68iPDxc67fs7GytkGNubg4hBEy43puIqMooCj8uLi6wsbEp16w7UVVQWFiIO3fuQKlUol69es/1t2vSS2ATJkzA4MGDERgYiODgYKxYsQKpqakYOXIkgKeXpm7fvo1vvvlGY7uVK1ciKCgI/v7+Wvvs1asXPvvsMzRv3lx9CWzatGno3bs3zM3NK6VfRERVlUqlUocfZ2dnUzeHSG916tTBnTt3UFBQAAsLiwrvx6QBKCwsDOnp6Zg5cyaUSiX8/f0RHx+vvqtLqVRqPRMoIyMD33//PRYvXqxzn1OnToVMJsPUqVNx+/Zt1KlTB7169cKcOXOM3h8ioqquaM2PjY2NiVtCVDFFl75UKtVzBSCTPgeoquJzgIjoRZWTk4OUlBT1E/iJqpvS/oarxXOAiIiIiEyFAYiIiEgP0dHRaNasmambQc+JAYiIiKqFkl6IffXqVaMdUyaTYfv27RplkyZN0njdElVPJn8QIhERVU+qQoETKQ9wLysHLvZytPJ2grmZcW+p79atG1avXq1RVqdOHY3veXl5z/2MmNLY2dnBzs7uufaRn5//XAt46flxBoiIiPS254ISbecfwDtxxxGx6SzeiTuOtvMPYM8FpVGPa2VlBTc3N41Pp06dMGbMGEyYMAG1a9dGly5dADx9ZlyrVq1gZWUFhUKByZMno6CgQL2v119/HWPHjsXHH38MJycnuLm5ITo6Wv27l5cXAKBfv36QyWTq77ouga1evRq+vr6Qy+Xw8fHBsmXL1L/duHEDMpkM3333HV5//XXI5XKsW7fOKOND5ccAREREetlzQYlR605DmZGjUZ6WkYNR604bPQTpsnbtWtSoUQNHjhzBV199hdu3b6NHjx5o2bIlzp07h9jYWKxcuRKzZ8/W2s7W1ha///47FixYgJkzZyIhIQEAcPLkSQBPw41SqVR/Ly4uLg5TpkzBnDlzkJycjLlz52LatGlYu3atRr1PPvkEY8eORXJyMrp27WqEUSB98BIYERGVm6pQYMauJOh6fooAIAMwY1cSuvi5GeVy2O7duzUuP3Xv3h0A0LBhQyxYsEBdPmXKFHh4eGDp0qWQyWTw8fHBnTt38Mknn2D69OnqNwY0adIEUVFRAICXX34ZS5cuxf79+9GlSxf1pbWaNWvCzc2txDbNmjULn376Kd58800AgLe3N5KSkvDVV19h6NCh6nrjxo1T1yHTYwAiIqJyO5HyQGvm51kCgDIjBydSHiC4geGfNN2hQwfExsaqv9va2uKdd95BYGCgRr3k5GQEBwdrvCqhTZs2ePToEW7duoV69eoBeBqAnqVQKHDv3r1yt+fvv//GzZs3ER4ejhEjRqjLCwoK4OjoqFG3eBvJtBiAiIio3O5llRx+KlJPX7a2tmjYsKHO8mcJIbTeE1X03N9ny4svRJbJZCgsLCx3e4rqxsXFISgoSOO34q9fKt5GMi0GICIiKjcX+/I9Pbq89YzFz88P33//vUYQOnr0KOzt7VG3bt1y78fCwgIqlarE311dXVG3bl1cv34dAwcOfO52U+XhImgiIiq3Vt5OUDjKUdLqHhkAhePTW+JN6T//+Q9u3ryJDz/8EH/++Sd27NiBqKgoTJgwQb3+pzy8vLywf/9+pKWl4Z9//tFZJzo6GjExMVi8eDEuX76M8+fPY/Xq1fjss88M1R0yAgYgIiIqN3MzGaJ6+QGAVggq+h7Vy8/ozwMqS926dREfH48TJ06gadOmGDlyJMLDwzF16lS99vPpp58iISEBHh4eaN68uc46w4cPx9dff401a9agcePGaN++PdasWQNvb29DdIWMhC9D1YEvQyWiF5WhXoa654ISM3YlaSyIVjjKEdXLD938FYZoKpFOhnoZKtcAERGR3rr5K9DFz63SnwRNZCgMQEREVCHmZjKj3OpOVBm4BoiIiIgkhwGIiIiIJIcBiIiIiCSHAYiIiIgkhwGIiIiIJIcBiIiIiCSHAYiIiCThxo0bkMlkOHv2bLm3WbNmDWrWrGnydhibTCbD9u3bAVTN9hkDAxAREVUbN2/eRHh4ONzd3WFpaQlPT09EREQgPT29zG09PDygVCrh7+9f7uOFhYXh8uXLz9PkakffcRo2bBj69u1r3EYZAQMQERFVC9evX0dgYCAuX76MjRs34urVq1i+fDn279+P4OBgPHjwoMRt8/LyYG5uDjc3N9SoUf5nAFtbW8PFxcUQzTe6/Px8g+ynIuNkCHl5eZV6PAYgIiKquPwnwK+Lnv7XyEaPHg1LS0v8/PPPaN++PerVq4fu3btj3759uH37NqZMmaKu6+XlhdmzZ2PYsGFwdHTEiBEjdF7a2blzJ15++WVYW1ujQ4cOWLt2LWQyGR4+fAhA+xJYdHQ0mjVrhm+//RZeXl5wdHRE//79kZWVpa6zZ88etG3bFjVr1oSzszPeeOMNXLt2Ta++enl5YdasWRgwYADs7Ozg7u6OJUuWaNSRyWRYvnw5+vTpA1tbW8yePRsAsGvXLgQEBEAul6N+/fqYMWMGCgoK1NtduXIF7dq1g1wuh5+fHxISEjT2q2ucLl68iJ49e8LBwQH29vYICQnBtWvXEB0djbVr12LHjh2QyWSQyWQ4ePAgAOD8+fPo2LEjrK2t4ezsjPfffx+PHj1S77No5igmJgbu7u545ZVX9Bqj58UAREREFZe0EzgwC0jeZdTDPHjwAHv37sV//vMfWFtba/zm5uaGgQMHYvPmzXj2/d4LFy6Ev78/EhMTMW3aNK193rhxA2+99Rb69u2Ls2fP4oMPPtAIUSW5du0atm/fjt27d2P37t04dOgQ5s2bp/798ePHmDBhAk6ePIn9+/fDzMwM/fr1Q2FhoV59XrhwIZo0aYLTp08jMjIS48eP1worUVFR6NOnD86fP4/33nsPe/fuxaBBgzB27FgkJSXhq6++wpo1azBnzhwAQGFhId58802Ym5vj+PHjWL58OT755JNS23H79m11YDpw4AASExPx3nvvoaCgAJMmTcLbb7+Nbt26QalUQqlUonXr1sjOzka3bt1Qq1YtnDx5Elu2bMG+ffswZswYjX3v378fycnJSEhIwO7du/Uan+cmSEtGRoYAIDIyMkzdFCIig3ry5IlISkoST548McwO14cJEeUgxIb+htlfCY4fPy4AiG3btun8/bPPPhMAxN27d4UQQnh6eoq+fftq1ElJSREAxJkzZ4QQQnzyySfC399fo86UKVMEAPHPP/8IIYRYvXq1cHR0VP8eFRUlbGxsRGZmprrso48+EkFBQSW2/d69ewKAOH/+vM526OLp6Sm6deumURYWFia6d++u/g5AjBs3TqNOSEiImDt3rkbZt99+KxQKhRBCiL179wpzc3Nx8+ZN9e8//fSTxtgWb19kZKTw9vYWeXl5Ots6dOhQ0adPH42yFStWiFq1aolHjx6py3788UdhZmYm0tLS1Nu5urqK3NzcEsdBl9L+hvU5f/NlqEREVH6P7gHHvgQK/++SyrX9T/97dR+w9/9mT8xqAMFjALs6ldYs8X8zPzLZ/38bfWBgYKnbXLp0CS1bttQoa9WqVZnH8vLygr29vfq7QqHAvXv31N+vXbuGadOm4fjx47h//7565ic1NVWvBdjBwcFa37/44guNsuJ9TExMxMmTJ9UzPgCgUqmQk5OD7OxsJCcno169enjppZdKPE5xZ8+eRUhICCwsLMrd9uTkZDRt2hS2trbqsjZt2qCwsBCXLl2Cq6srAKBx48awtLQs934NiQGIiIjKLysNOB4LqHIBmRmA/wschSrg+DJAFALmVoD/vwwagBo2bAiZTIakpCSddxz9+eefqFWrFmrXrq0ue/bkq4sQQiMwFZWVpXgQkMlkGpe3evXqBQ8PD8TFxcHd3R2FhYXw9/c3yCLf4u0t3sfCwkLMmDEDb775pta2crlcZ/+K77O44pccy0PX2Oo6Xln/RsbENUBERFR+iibAB4eA2v+3YFWoNP9b+5WnvyuaGPSwzs7O6NKlC5YtW4YnTzQXXKelpWH9+vUICwsr82T+LB8fH5w8eVKj7NSpU8/VzvT0dCQnJ2Pq1Kno1KkTfH198c8//1RoX8ePH9f67uPjU+o2LVq0wKVLl9CwYUOtj5mZGfz8/JCamoo7d+6otzl27Fip+2zSpAkOHz5c4l1mlpaWUKlUGmV+fn44e/YsHj9+rC47cuQIzMzMKn2xc0kYgIiISD8uvkD4z7p/C//56e9GsHTpUuTm5qJr16749ddfcfPmTezZswddunRB3bp1NS77lMcHH3yAP//8E5988gkuX76M7777DmvWrAFQ9qxISWrVqgVnZ2esWLECV69exYEDBzBhwoQK7evIkSNYsGABLl++jC+//BJbtmxBREREqdtMnz4d33zzDaKjo3Hx4kUkJydj8+bNmDp1KgCgc+fOaNSoEYYMGYJz587h8OHDZS78HjNmDDIzM9G/f3+cOnUKV65cwbfffotLly4BeHpJ8I8//sClS5dw//595OfnY+DAgZDL5Rg6dCguXLiAX375BR9++CEGDx6svvxlagxARESkv7+OPr3c9SxRCPxV+mzC83j55Zdx6tQpNGjQAGFhYWjQoAHef/99dOjQAceOHYOTk5Ne+/P29sbWrVvxww8/oEmTJoiNjVWHASsrqwq10czMDJs2bUJiYiL8/f0xfvx4LFy4sEL7mjhxIhITE9G8eXPMmjULn376Kbp27VrqNl27dsXu3buRkJCAli1b4rXXXsNnn30GT09Pdfu2bduG3NxctGrVCsOHDy8zODo7O+PAgQN49OgR2rdvj4CAAMTFxakvBY4YMQKNGjVCYGAg6tSpgyNHjsDGxgZ79+7FgwcP0LJlS7z11lvo1KkTli5dWqGxMAaZKM8FT4nJzMyEo6MjMjIy4ODgYOrmEBEZTE5ODlJSUuDt7Q25XF7xHW37ADi3CagXDHSZBfw8Fbh5HGj6DtBvueEaXMnmzJmD5cuX4+bNmyZth5eXF8aNG4dx48aZtB1VUWl/w/qcv7kImoiI9KdoDtTxBVp/CJiZA+/GA0eXADWeI1SZwLJly9CyZUs4OzvjyJEjWLhwodazaujFxABERET6e22k5nczc6DtOJM05XlcuXIFs2fPxoMHD1CvXj1MnDgRkZGRpm4WVQIGICIikqzPP/8cn3/+uamboeXGjRumbsILj4ugiYiISHIYgIiIiEhyGICIiIhIchiAiIiISHIYgIiIiEhyGICIiIhIchiAiIiIAAwbNkznm+Yrw40bNyCTyXD27FmTHF8XmUyG7du3A6ia7XteDEBERFQtvP766zpfDbF9+/YKv7z0ebyIoaAkHh4eUCqV8Pf3L1d9U4bJ8mIAIiIiekHl5+cbZD/m5uZwc3NDjRqV+/zkvLw8o+2bAYiIiPTyV+ZfSEpPKvHzV+ZfJm1fdHQ0mjVrhq+++goeHh6wsbHBv//9bzx8+FBdR6VSYcKECahZsyacnZ3x8ccfo/i7wffs2YO2bduq67zxxhu4du2a+ndvb28AQPPmzSGTyfD666+rf1u9ejV8fX0hl8vh4+ODZcuWaez7xIkTaN68OeRyOQIDA3HmzJky++Xl5YVZs2ZhwIABsLOzg7u7O5YsWaJRRyaTYfny5ejTpw9sbW0xe/ZsAMCuXbsQEBAAuVyO+vXrY8aMGSgoKFBvd+XKFbRr1w5yuRx+fn5ISEjQ2K+u2a6LFy+iZ8+ecHBwgL29PUJCQnDt2jVER0dj7dq12LFjB2QyGWQyGQ4ePAgAOH/+PDp27Ahra2s4Ozvj/fffx6NHj9T7LJo5iomJgbu7O1555ZUyx6Wi+CoMIiIqt78y/8Ib294os97ufrvh6eBZCS3S7erVq/juu++wa9cuZGZmIjw8HKNHj8b69esBAJ9++ilWrVqFlStXws/PD59++im2bduGjh07qvfx+PFjTJgwAY0bN8bjx48xffp09OvXD2fPnoWZmRlOnDiBVq1aYd++fXj11VdhaWkJAIiLi0NUVBSWLl2K5s2b48yZMxgxYgRsbW0xdOhQPH78GG+88QY6duyIdevWISUlBREREeXq18KFC/Hf//4X0dHR2Lt3L8aPHw8fHx906dJFXScqKgoxMTH4/PPPYW5ujr1792LQoEH43//+pw4p77//vrpuYWEh3nzzTdSuXRvHjx9HZmZmmW+hv337Ntq1a4fXX38dBw4cgIODA44cOYKCggJMmjQJycnJyMzMxOrVqwEATk5OyM7ORrdu3fDaa6/h5MmTuHfvHoYPH44xY8ZgzZo16n3v378fDg4OSEhI0AqlBiVIS0ZGhgAgMjIyDLrfAlWhOHr1vth+5pY4evW+KFAVGnT/RERlefLkiUhKShJPnjyp0PYX718U/mv8y/xcvH/RwC0Xon379iIiIkKrfNu2beLZ01lUVJQwNzcXN2/eVJf99NNPwszMTCiVSiGEEAqFQsybN0/9e35+vnjppZdEnz59Sjz+vXv3BABx/vx5IYQQKSkpAoA4c+aMRj0PDw+xYcMGjbJZs2aJ4OBgIYQQX331lXBychKPHz9W/x4bG6tzX8/y9PQU3bp10ygLCwsT3bt3V38HIMaNG6dRJyQkRMydO1ej7NtvvxUKhUIIIcTevXt1jhcAsW3bNp19jYyMFN7e3iIvL09nW4cOHao1litWrBC1atUSjx49Upf9+OOPwszMTKSlpam3c3V1Fbm5uSWOQ2l/w/qcvzkDVEn2XFBixq4kKDNy1GUKRzmievmhm7/ChC0jInrx1KtXDy+99JL6e3BwMAoLC3Hp0iVYW1tDqVQiODhY/XuNGjUQGBioMeNw7do1TJs2DcePH8f9+/dRWFgIAEhNTS1xMfDff/+NmzdvIjw8HCNGjFCXFxQUwNHREQCQnJyMpk2bwsbGRqN95VG8XnBwML744guNssDAQI3viYmJOHnyJObMmaMuU6lUyMnJQXZ2NpKTk3WOV2nOnj2LkJAQWFhYlKvdwP/vt62trbqsTZs26n8XV1dXAEDjxo3Vs2nGxABUCfZcUGLUutMoPpGXlpGDUetOI3ZQC4YgIqIyODg4ICMjQ6v84cOHcHBwKHXborvE9LlbrFevXvDw8EBcXBzc3d1RWFgIf3//UhfmFoWkuLg4BAUFafxmbm4OAAa/rFO8T88GjKI2zZgxA2+++abWtnK5XGd7yhona2trvdsphChxv8+WF2+/sXARtJGpCgVm7ErSCj8A1GUzdiVBVWjE65xERC8AHx8fnDp1Sqv85MmTaNSokUZZamoq7ty5o/5+7NgxmJmZ4ZVXXoGjoyMUCgWOHz+u/r2goACJiYnq7+np6UhOTsbUqVPRqVMn+Pr64p9//tE4RtEshUqlUpe5urqibt26uH79Oho2bKjxKVo07efnh3PnzuHJkyfq7Z5tS2mK1zt+/Dh8fHxK3aZFixa4dOmSVnsaNmwIMzMz+Pn56Ryv0jRp0gSHDx8u8S4zS0tLjXEBnvb77NmzePz4sbrsyJEj6n+XymbyALRs2TJ4e3tDLpcjICAAhw8fLrHusGHD1CvKn/28+uqrGvUePnyI0aNHQ6FQQC6Xw9fXF/Hx8cbuik4nUh5oXPYqTgBQZuTgRMqDymsUEVE19J///AfXrl3D6NGjce7cOVy+fBlffvklVq5ciY8++kijrlwux9ChQ3Hu3DkcPnwYY8eOxdtvvw03NzcAQEREBObNm4dt27bhzz//xH/+8x+Nu8Rq1aoFZ2dnrFixAlevXsWBAwcwYcIEjWO4uLjA2toae/bswd27d9WzU9HR0YiJicHixYtx+fJlnD9/HqtXr8Znn30GABgwYADMzMwQHh6OpKQkxMfHY9GiReUagyNHjmDBggXqvm/ZsqXMBdTTp0/HN998g+joaFy8eBHJycnYvHkzpk6dCgDo3LkzGjVqhCFDhqjHa8qUKaXuc8yYMcjMzET//v1x6tQpXLlyBd9++y0uXboE4Okda3/88QcuXbqE+/fvIz8/HwMHDlT/u1y4cAG//PILPvzwQwwePFh9+asymTQAbd68GePGjcOUKVNw5swZhISEoHv37khNTdVZf/HixVAqlerPzZs34eTkhH//+9/qOnl5eejSpQtu3LiBrVu34tKlS4iLi0PdunUrq1sa7mWVHH4qUo+ISKq8vLxw+PBhXLt2DaGhoWjZsiXWrFmDNWvWaJwHAKBhw4Z488030aNHD4SGhsLf31/jVvSJEydiyJAhGDZsGIKDg2Fvb49+/fqpfzczM8OmTZuQmJgIf39/jB8/HgsXLtQ4Ro0aNfC///0PX331Fdzd3dGnTx8AwPDhw/H1119jzZo1aNy4Mdq3b481a9aoZ4Ds7Oywa9cuJCUloXnz5pgyZQrmz59frjGYOHEiEhMT0bx5c8yaNQuffvopunbtWuo2Xbt2xe7du5GQkICWLVvitddew2effQZPT091X7dt24bc3Fy0atUKw4cP11gvpIuzszMOHDiAR48eoX379ggICEBcXJx6TdCIESPQqFEjBAYGok6dOjhy5AhsbGywd+9ePHjwAC1btsRbb72FTp06YenSpeXqu6HJhKEvRuohKCgILVq0QGxsrLrM19dX/QyAsmzfvh1vvvkmUlJS1P+Qy5cvx8KFC/Hnn3/qtTjrWZmZmXB0dERGRkaZ15XLcuxaOt6JK3tqc+OI1xDcwPm5jkVEVJacnBykpKSoZ971lZSehLDdYWXW2/zGZvg5+1Wkic8tOjoa27dvf+Ge0Ozl5YVx48aVeYv6i660v2F9zt8mmwHKy8tDYmIiQkNDNcpDQ0Nx9OjRcu1j5cqV6Ny5szr8AMDOnTsRHByM0aNHw9XVFf7+/pg7d67Wtchn5ebmIjMzU+NjKK28naBwlKOk5WQyPL0brJW3k8GOSURkLLYW5VugWt56RKZisrvA7t+/D5VKpXXdz9XVFWlpaWVur1Qq8dNPP2HDhg0a5devX8eBAwcwcOBAxMfH48qVKxg9ejQKCgowffp0nfuKiYnBjBkzKt6ZUpibyRDVyw+j1p2GDNBYDF0UiqJ6+cHcrPLfY0NEpC9PB0/s7rcbj/Mfl1jH1sLWpA9BJCoPk98GX/yWuNJuk3vWmjVrULNmTa2XrRUWFsLFxQUrVqyAubk5AgICcOfOHSxcuLDEABQZGamxuC0zMxMeHh76d6YE3fwViB3UQus5QG58DhARVUNVPdxER0cjOjra1M0wuBs3bpi6CS8UkwWg2rVrw9zcXGu25969e2WuBhdCYNWqVRg8eLDWw5IUCgUsLCzUz1sAnq4rSktLQ15ens6HK1lZWcHKyuo5elO2bv4KdPFzw4mUB7iXlQMX+6eXvTjzQ0REVPlMtgbI0tISAQEBWi9cS0hIQOvWrUvd9tChQ7h69SrCw8O1fmvTpg2uXr2qfhgVAFy+fBkKhaJSnixZGnMzGYIbOKNPs7oIbuDM8ENERGQiJr0NfsKECfj666+xatUqJCcnY/z48UhNTcXIkSMBPL00NWTIEK3tVq5ciaCgIJ2PIh81ahTS09MRERGBy5cv48cff8TcuXMxevRoo/eHiKi6MOENwETPxVB/uyZdAxQWFob09HTMnDkTSqUS/v7+iI+PV9/VpVQqtZ4JlJGRge+//x6LFy/WuU8PDw/8/PPPGD9+PJo0aYK6desiIiICn3zyidH7Q0RU1RU9HiQ7O7tCrzMgMrWiV5E8u9SlIkz6HKCqypDPASIiqmqUSiUePnwIFxcX2NjY6PV+LCJTKiwsxJ07d2BhYYF69epp/e3qc/42+V1gRERUuYpeB3Hv3j0Tt4RIf2ZmZjrDj74YgIiIJEYmk0GhUMDFxaXEl1kSVVWWlpYwM3v+JcwMQEREEmVubv7c6yiIqiuTvw2eiIiIqLIxABEREZHkMAARERGR5HANkA5FTwYw5FvhiYiIyLiKztvlecIPA5AOWVlZAGDQF6ISERFR5cjKyoKjo2OpdfggRB2KHrRkb2+v8ZyBorfE37x5kw9INCCOq3FwXA2PY2ocHFfjkOK4CiGQlZUFd3f3Mm+V5wyQDmZmZnjppZdK/N3BwUEyf0yVieNqHBxXw+OYGgfH1TikNq5lzfwU4SJoIiIikhwGICIiIpIcBiA9WFlZISoqClZWVqZuyguF42ocHFfD45gaB8fVODiupeMiaCIiIpIczgARERGR5DAAERERkeQwABEREZHkMAARERGR5Eg6AC1btgze3t6Qy+UICAjA4cOHS61/6NAhBAQEQC6Xo379+li+fLlWne+//x5+fn6wsrKCn58ftm3bZqzmV1mGHte4uDiEhISgVq1aqFWrFjp37owTJ04YswtVkjH+Xots2rQJMpkMffv2NXCrqz5jjOvDhw8xevRoKBQKyOVy+Pr6Ij4+3lhdqJKMMa5ffPEFGjVqBGtra3h4eGD8+PHIyckxVheqHH3GVKlUYsCAAWjUqBHMzMwwbtw4nfUkfc4SErVp0yZhYWEh4uLiRFJSkoiIiBC2trbir7/+0ln/+vXrwsbGRkRERIikpCQRFxcnLCwsxNatW9V1jh49KszNzcXcuXNFcnKymDt3rqhRo4Y4fvx4ZXXL5IwxrgMGDBBffvmlOHPmjEhOThbvvvuucHR0FLdu3aqsbpmcMca1yI0bN0TdunVFSEiI6NOnj5F7UrUYY1xzc3NFYGCg6NGjh/jtt9/EjRs3xOHDh8XZs2crq1smZ4xxXbdunbCyshLr168XKSkpYu/evUKhUIhx48ZVVrdMSt8xTUlJEWPHjhVr164VzZo1ExEREVp1pH7OkmwAatWqlRg5cqRGmY+Pj5g8ebLO+h9//LHw8fHRKPvggw/Ea6+9pv7+9ttvi27dumnU6dq1q+jfv7+BWl31GWNciysoKBD29vZi7dq1z9/gasJY41pQUCDatGkjvv76azF06FDJBSBjjGtsbKyoX7++yMvLM3yDqwljjOvo0aNFx44dNepMmDBBtG3b1kCtrtr0HdNntW/fXmcAkvo5S5KXwPLy8pCYmIjQ0FCN8tDQUBw9elTnNseOHdOq37VrV5w6dQr5+fml1ilpny8aY41rcdnZ2cjPz4eTk5NhGl7FGXNcZ86ciTp16iA8PNzwDa/ijDWuO3fuRHBwMEaPHg1XV1f4+/tj7ty5UKlUxulIFWOscW3bti0SExPVl7+vX7+O+Ph49OzZ0wi9qFoqMqblIfVzliRfhnr//n2oVCq4urpqlLu6uiItLU3nNmlpaTrrFxQU4P79+1AoFCXWKWmfLxpjjWtxkydPRt26ddG5c2fDNb4KM9a4HjlyBCtXrsTZs2eN1fQqzVjjev36dRw4cAADBw5EfHw8rly5gtGjR6OgoADTp083Wn+qCmONa//+/fH333+jbdu2EEKgoKAAo0aNwuTJk43Wl6qiImNaHlI/Z0kyABWRyWQa34UQWmVl1S9eru8+X0TGGNciCxYswMaNG3Hw4EHI5XIDtLb6MOS4ZmVlYdCgQYiLi0Pt2rUN39hqxNB/r4WFhXBxccGKFStgbm6OgIAA3LlzBwsXLpREACpi6HE9ePAg5syZg2XLliEoKAhXr15FREQEFAoFpk2bZuDWV03GOL9I+ZwlyQBUu3ZtmJuba6Xce/fuaaXhIm5ubjrr16hRA87OzqXWKWmfLxpjjWuRRYsWYe7cudi3bx+aNGli2MZXYcYY14sXL+LGjRvo1auX+vfCwkIAQI0aNXDp0iU0aNDAwD2pWoz196pQKGBhYQFzc3N1HV9fX6SlpSEvLw+WlpYG7knVYqxxnTZtGgYPHozhw4cDABo3bozHjx/j/fffx5QpU2Bm9uKu6KjImJaH1M9ZL+5fTCksLS0REBCAhIQEjfKEhAS0bt1a5zbBwcFa9X/++WcEBgbCwsKi1Dol7fNFY6xxBYCFCxdi1qxZ2LNnDwIDAw3f+CrMGOPq4+OD8+fP4+zZs+pP79690aFDB5w9exYeHh5G609VYay/1zZt2uDq1avqQAkAly9fhkKheOHDD2C8cc3OztYKOebm5hBPb+YxYA+qnoqMaXlI/Zwl2bvAim4pXLlypUhKShLjxo0Ttra24saNG0IIISZPniwGDx6srl90m+b48eNFUlKSWLlypdZtmkeOHBHm5uZi3rx5Ijk5WcybN09StxQKYZxxnT9/vrC0tBRbt24VSqVS/cnKyqr0/pmKMca1OCneBWaMcU1NTRV2dnZizJgx4tKlS2L37t3CxcVFzJ49u9L7ZyrGGNeoqChhb28vNm7cKK5fvy5+/vln0aBBA/H2229Xev9MQd8xFUKIM2fOiDNnzoiAgAAxYMAAcebMGXHx4kX171I/Z0k2AAkhxJdffik8PT2FpaWlaNGihTh06JD6t6FDh4r27dtr1D948KBo3ry5sLS0FF5eXiI2NlZrn1u2bBGNGjUSFhYWwsfHR3z//ffG7kaVY+hx9fT0FAC0PlFRUZXQm6rDGH+vz5JiABLCOON69OhRERQUJKysrET9+vXFnDlzREFBgbG7UqUYelzz8/NFdHS0aNCggZDL5cLDw0P85z//Ef/8808l9KZq0HdMdf3/pqenp0YdKZ+zZEK84HOHRERERMVIcg0QERERSRsDEBEREUkOAxARERFJDgMQERERSQ4DEBEREUkOAxARERFJDgMQERERSQ4DEBERgDVr1qBmzZql1hk2bBj69u1bKe3RR3R0NJo1a2bqZhBVKwxARBIwbNgwyGQyyGQyWFhYwNXVFV26dMGqVas03llFpVu8eDHWrFlT4e2L/zvUr18fkyZNwuPHj5+rXZMmTcL+/fs1jlMVgxpRVcIARCQR3bp1g1KpxI0bN/DTTz+hQ4cOiIiIwBtvvIGCggJTN69C8vLyKvV4jo6OZc4SlaXo3+H69euYPXs2li1bhkmTJlVoX0IIFBQUwM7OTv3WdCIqHwYgIomwsrKCm5sb6tatixYtWuC///0vduzYgZ9++kljViMjIwPvv/8+XFxc4ODggI4dO+LcuXPq34sut6xatQr16tWDnZ0dRo0aBZVKhQULFsDNzQ0uLi6YM2eOxvFTU1PRp08f2NnZwcHBAW+//Tbu3r2rUWf27NlwcXGBvb09hg8fjsmTJ2tc2ima2YiJiYG7uzteeeUVAMC6desQGBgIe3t7uLm5YcCAAbh37556u4MHD0Imk+HHH39E06ZNIZfLERQUhPPnz2uN0969e+Hr6ws7Ozt1WCl+/CKFhYWYP38+GjZsCCsrK9SrV0+r3yX9O3h4eGDAgAEYOHAgtm/frlc/9u7di8DAQFhZWeHw4cMal8Cio6Oxdu1a7NixQz3bdPDgQXTs2BFjxozRaEt6ejqsrKxw4MCBUttM9CJiACKSsI4dO6Jp06b44YcfADydUejZsyfS0tIQHx+PxMREtGjRAp06dcKDBw/U2127dg0//fQT9uzZg40bN2LVqlXo2bMnbt26hUOHDmH+/PmYOnUqjh8/rt5v37598eDBAxw6dAgJCQm4du0awsLC1Ptcv3495syZg/nz5yMxMRH16tVDbGysVpv379+P5ORkJCQkYPfu3QCezgTNmjUL586dw/bt25GSkoJhw4ZpbfvRRx9h0aJFOHnyJFxcXNC7d2/k5+erf8/OzsaiRYvw7bff4tdff0VqamqpszORkZGYP38+pk2bhqSkJGzYsAGurq56/RtYW1ur21Defnz88ceIiYlBcnIymjRpovHbpEmT8Pbbb6vDm1KpROvWrTF8+HBs2LABubm56rrr16+Hu7s7OnTooFebiV4IJn0VKxFVitLe9B4WFiZ8fX2FEELs379fODg4iJycHI06DRo0EF999ZUQQoioqChhY2MjMjMz1b937dpVeHl5CZVKpS5r1KiRiImJEUII8fPPPwtzc3ORmpqq/v3ixYsCgDhx4oQQQoigoCAxevRojeO2adNGNG3aVKMfrq6uIjc3t9T+njhxQgAQWVlZQgghfvnlFwFAbNq0SV0nPT1dWFtbi82bNwshhFi9erUAIK5evaqu8+WXXwpXV1eN4xeNY2ZmprCyshJxcXGltuVZxf8dfv/9d+Hs7Czefvttvfqxfft2jXpRUVFa41T83zsnJ0c4OTmp+yuEEM2aNRPR0dHlbj/Ri4QzQEQSJ4SATCYDACQmJuLRo0dwdnaGnZ2d+pOSkoJr166pt/Hy8oK9vb36u6urK/z8/GBmZqZRVnT5Jjk5GR4eHvDw8FD/7ufnh5o1ayI5ORkAcOnSJbRq1UqjbcW/A0Djxo1haWmpUXbmzBn06dMHnp6esLe3x+uvvw7g6WW3ZwUHB6v/t5OTExo1aqQ+PgDY2NigQYMG6u8KhULjEtSzkpOTkZubi06dOun8vSS7d++GnZ0d5HI5goOD0a5dOyxZskSvfgQGBup1TODppbdBgwZh1apVAICzZ8/i3LlzOmeYiKSghqkbQESmlZycDG9vbwBP17QoFAocPHhQq96zi38tLCw0fiu6q6l4WdEdZs+GrGcVLy9eRwihtY2tra3G98ePHyM0NBShoaFYt24d6tSpg9TUVHTt2rVci6SfPaauPuhqA/D00lVFdOjQAbGxsbCwsIC7u7v6mPr0o/gYlNfw4cPRrFkz3Lp1C6tWrUKnTp3g6elZoX0RVXecASKSsAMHDuD8+fP417/+BQBo0aIF0tLSUKNGDTRs2FDjU7t27Qofx8/PD6mpqbh586a6LCkpCRkZGfD19QUANGrUCCdOnNDY7tSpU2Xu+88//8T9+/cxb948hISEwMfHp8RZm6I1SQDwzz//4PLly/Dx8alIl/Dyyy/D2tpa4/bz8rC1tUXDhg3h6empEbj06UdZLC0toVKptMobN26MwMBAxMXFYcOGDXjvvfcqtH+iFwEDEJFE5ObmIi0tDbdv38bp06cxd+5c9OnTB2+88QaGDBkCAOjcuTOCg4PRt29f7N27Fzdu3MDRo0cxderUcoWRknTu3BlNmjTBwIEDcfr0aZw4cQJDhgxB+/bt1ZdzPvzwQ6xcuRJr167FlStXMHv2bPzxxx86Z46eVa9ePVhaWmLJkiW4fv06du7ciVmzZumsO3PmTOzfvx8XLlzAsGHDULt27Qo/L0cul+OTTz7Bxx9/jG+++QbXrl3D8ePHsXLlygrtT59+lMXLywt//PEHLl26hPv372ss9B4+fDjmzZsHlUqFfv36VWj/RC8CBiAiidizZw8UCgW8vLzQrVs3/PLLL/jf//6HHTt2wNzcHMDTSz7x8fFo164d3nvvPbzyyivo378/bty4offdTc+SyWTYvn07atWqhXbt2qFz586oX78+Nm/erK4zcOBAREZGYtKkSWjRooX6Dii5XF7qvuvUqYM1a9Zgy5Yt8PPzw7x587Bo0SKddefNm4eIiAgEBARAqVRi586dWuuJ9DFt2jRMnDgR06dPh6+vL8LCwio8a6NPP8oyYsQINGrUCIGBgahTpw6OHDmi/u2dd95BjRo1MGDAgDLHluhFJhMlXeAmIjKxLl26wM3NDd9+++1z7efgwYPo0KED/vnnn+d+kGF1d/PmTXh5eeHkyZNo0aKFqZtDZDJcBE1EVUJ2djaWL1+Orl27wtzcHBs3bsS+ffuQkJBg6qa9EPLz86FUKjF58mS89tprDD8keQxARFQlFF1+mz17NnJzc9GoUSN8//336Ny5s6mb9kI4cuQIOnTogFdeeQVbt241dXOITI6XwIiIiEhyuAiaiIiIJIcBiIiIiCSHAYiIiIgkhwGIiIiIJIcBiIiIiCSHAYiIiIgkhwGIiIiIJIcBiIiIiCSHAYiIiIgk5/8B0rdas0M1naAAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "#and on the test set\n",
    "fpredictor.plot_frontier(test_data) "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th>Accuracy</th>\n",
       "      <th>Balanced Accuracy</th>\n",
       "      <th>F1 score</th>\n",
       "      <th>MCC</th>\n",
       "      <th>Precision</th>\n",
       "      <th>Recall</th>\n",
       "      <th>ROC AUC</th>\n",
       "      <th>Number of Datapoints</th>\n",
       "      <th>Positive Count</th>\n",
       "      <th>Negative Count</th>\n",
       "      <th>Positive Label Rate</th>\n",
       "      <th>Positive Prediction Rate</th>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th></th>\n",
       "      <th>Groups</th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th rowspan=\"7\" valign=\"top\">original</th>\n",
       "      <th>Overall</th>\n",
       "      <td>0.870035</td>\n",
       "      <td>0.797288</td>\n",
       "      <td>0.707789</td>\n",
       "      <td>0.627843</td>\n",
       "      <td>0.766042</td>\n",
       "      <td>0.657769</td>\n",
       "      <td>0.922670</td>\n",
       "      <td>12211.0</td>\n",
       "      <td>2922.0</td>\n",
       "      <td>9289.0</td>\n",
       "      <td>0.239292</td>\n",
       "      <td>0.205470</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>0.936364</td>\n",
       "      <td>0.661716</td>\n",
       "      <td>0.461538</td>\n",
       "      <td>0.473568</td>\n",
       "      <td>0.750000</td>\n",
       "      <td>0.333333</td>\n",
       "      <td>0.885589</td>\n",
       "      <td>110.0</td>\n",
       "      <td>9.0</td>\n",
       "      <td>101.0</td>\n",
       "      <td>0.081818</td>\n",
       "      <td>0.036364</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>0.834734</td>\n",
       "      <td>0.773608</td>\n",
       "      <td>0.681081</td>\n",
       "      <td>0.572865</td>\n",
       "      <td>0.732558</td>\n",
       "      <td>0.636364</td>\n",
       "      <td>0.898089</td>\n",
       "      <td>357.0</td>\n",
       "      <td>99.0</td>\n",
       "      <td>258.0</td>\n",
       "      <td>0.277311</td>\n",
       "      <td>0.240896</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>0.926421</td>\n",
       "      <td>0.780790</td>\n",
       "      <td>0.669173</td>\n",
       "      <td>0.636968</td>\n",
       "      <td>0.780702</td>\n",
       "      <td>0.585526</td>\n",
       "      <td>0.949618</td>\n",
       "      <td>1196.0</td>\n",
       "      <td>152.0</td>\n",
       "      <td>1044.0</td>\n",
       "      <td>0.127090</td>\n",
       "      <td>0.095318</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>0.915966</td>\n",
       "      <td>0.740291</td>\n",
       "      <td>0.615385</td>\n",
       "      <td>0.590926</td>\n",
       "      <td>0.800000</td>\n",
       "      <td>0.500000</td>\n",
       "      <td>0.961165</td>\n",
       "      <td>119.0</td>\n",
       "      <td>16.0</td>\n",
       "      <td>103.0</td>\n",
       "      <td>0.134454</td>\n",
       "      <td>0.084034</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>0.863554</td>\n",
       "      <td>0.797954</td>\n",
       "      <td>0.712002</td>\n",
       "      <td>0.625900</td>\n",
       "      <td>0.766449</td>\n",
       "      <td>0.664777</td>\n",
       "      <td>0.919172</td>\n",
       "      <td>10429.0</td>\n",
       "      <td>2646.0</td>\n",
       "      <td>7783.0</td>\n",
       "      <td>0.253716</td>\n",
       "      <td>0.220059</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Maximum difference</th>\n",
       "      <td>0.101630</td>\n",
       "      <td>0.136238</td>\n",
       "      <td>0.250463</td>\n",
       "      <td>0.163400</td>\n",
       "      <td>0.067442</td>\n",
       "      <td>0.331444</td>\n",
       "      <td>0.075576</td>\n",
       "      <td>10319.0</td>\n",
       "      <td>2637.0</td>\n",
       "      <td>7682.0</td>\n",
       "      <td>0.195493</td>\n",
       "      <td>0.204533</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th rowspan=\"7\" valign=\"top\">updated</th>\n",
       "      <th>Overall</th>\n",
       "      <td>0.866432</td>\n",
       "      <td>0.786827</td>\n",
       "      <td>0.694398</td>\n",
       "      <td>0.614456</td>\n",
       "      <td>0.767288</td>\n",
       "      <td>0.634155</td>\n",
       "      <td>0.896317</td>\n",
       "      <td>12211.0</td>\n",
       "      <td>2922.0</td>\n",
       "      <td>9289.0</td>\n",
       "      <td>0.239292</td>\n",
       "      <td>0.197773</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>0.909091</td>\n",
       "      <td>0.748075</td>\n",
       "      <td>0.500000</td>\n",
       "      <td>0.453295</td>\n",
       "      <td>0.454545</td>\n",
       "      <td>0.555556</td>\n",
       "      <td>0.885589</td>\n",
       "      <td>110.0</td>\n",
       "      <td>9.0</td>\n",
       "      <td>101.0</td>\n",
       "      <td>0.081818</td>\n",
       "      <td>0.100000</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>0.834734</td>\n",
       "      <td>0.761158</td>\n",
       "      <td>0.666667</td>\n",
       "      <td>0.565863</td>\n",
       "      <td>0.756410</td>\n",
       "      <td>0.595960</td>\n",
       "      <td>0.898089</td>\n",
       "      <td>357.0</td>\n",
       "      <td>99.0</td>\n",
       "      <td>258.0</td>\n",
       "      <td>0.277311</td>\n",
       "      <td>0.218487</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>0.902174</td>\n",
       "      <td>0.859649</td>\n",
       "      <td>0.675900</td>\n",
       "      <td>0.630884</td>\n",
       "      <td>0.583732</td>\n",
       "      <td>0.802632</td>\n",
       "      <td>0.949618</td>\n",
       "      <td>1196.0</td>\n",
       "      <td>152.0</td>\n",
       "      <td>1044.0</td>\n",
       "      <td>0.127090</td>\n",
       "      <td>0.174749</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>0.865546</td>\n",
       "      <td>0.922330</td>\n",
       "      <td>0.666667</td>\n",
       "      <td>0.649869</td>\n",
       "      <td>0.500000</td>\n",
       "      <td>1.000000</td>\n",
       "      <td>0.961165</td>\n",
       "      <td>119.0</td>\n",
       "      <td>16.0</td>\n",
       "      <td>103.0</td>\n",
       "      <td>0.134454</td>\n",
       "      <td>0.268908</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>0.862978</td>\n",
       "      <td>0.784099</td>\n",
       "      <td>0.697950</td>\n",
       "      <td>0.618199</td>\n",
       "      <td>0.791847</td>\n",
       "      <td>0.623961</td>\n",
       "      <td>0.919172</td>\n",
       "      <td>10429.0</td>\n",
       "      <td>2646.0</td>\n",
       "      <td>7783.0</td>\n",
       "      <td>0.253716</td>\n",
       "      <td>0.199923</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Maximum difference</th>\n",
       "      <td>0.074357</td>\n",
       "      <td>0.174255</td>\n",
       "      <td>0.197950</td>\n",
       "      <td>0.196575</td>\n",
       "      <td>0.337301</td>\n",
       "      <td>0.444444</td>\n",
       "      <td>0.075576</td>\n",
       "      <td>10319.0</td>\n",
       "      <td>2637.0</td>\n",
       "      <td>7682.0</td>\n",
       "      <td>0.195493</td>\n",
       "      <td>0.168908</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "                             Accuracy  Balanced Accuracy  F1 score       MCC  \\\n",
       "         Groups                                                                \n",
       "original Overall             0.870035           0.797288  0.707789  0.627843   \n",
       "         0                   0.936364           0.661716  0.461538  0.473568   \n",
       "         1                   0.834734           0.773608  0.681081  0.572865   \n",
       "         2                   0.926421           0.780790  0.669173  0.636968   \n",
       "         3                   0.915966           0.740291  0.615385  0.590926   \n",
       "         4                   0.863554           0.797954  0.712002  0.625900   \n",
       "         Maximum difference  0.101630           0.136238  0.250463  0.163400   \n",
       "updated  Overall             0.866432           0.786827  0.694398  0.614456   \n",
       "         0                   0.909091           0.748075  0.500000  0.453295   \n",
       "         1                   0.834734           0.761158  0.666667  0.565863   \n",
       "         2                   0.902174           0.859649  0.675900  0.630884   \n",
       "         3                   0.865546           0.922330  0.666667  0.649869   \n",
       "         4                   0.862978           0.784099  0.697950  0.618199   \n",
       "         Maximum difference  0.074357           0.174255  0.197950  0.196575   \n",
       "\n",
       "                             Precision    Recall   ROC AUC  \\\n",
       "         Groups                                              \n",
       "original Overall              0.766042  0.657769  0.922670   \n",
       "         0                    0.750000  0.333333  0.885589   \n",
       "         1                    0.732558  0.636364  0.898089   \n",
       "         2                    0.780702  0.585526  0.949618   \n",
       "         3                    0.800000  0.500000  0.961165   \n",
       "         4                    0.766449  0.664777  0.919172   \n",
       "         Maximum difference   0.067442  0.331444  0.075576   \n",
       "updated  Overall              0.767288  0.634155  0.896317   \n",
       "         0                    0.454545  0.555556  0.885589   \n",
       "         1                    0.756410  0.595960  0.898089   \n",
       "         2                    0.583732  0.802632  0.949618   \n",
       "         3                    0.500000  1.000000  0.961165   \n",
       "         4                    0.791847  0.623961  0.919172   \n",
       "         Maximum difference   0.337301  0.444444  0.075576   \n",
       "\n",
       "                             Number of Datapoints  Positive Count  \\\n",
       "         Groups                                                     \n",
       "original Overall                          12211.0          2922.0   \n",
       "         0                                  110.0             9.0   \n",
       "         1                                  357.0            99.0   \n",
       "         2                                 1196.0           152.0   \n",
       "         3                                  119.0            16.0   \n",
       "         4                                10429.0          2646.0   \n",
       "         Maximum difference               10319.0          2637.0   \n",
       "updated  Overall                          12211.0          2922.0   \n",
       "         0                                  110.0             9.0   \n",
       "         1                                  357.0            99.0   \n",
       "         2                                 1196.0           152.0   \n",
       "         3                                  119.0            16.0   \n",
       "         4                                10429.0          2646.0   \n",
       "         Maximum difference               10319.0          2637.0   \n",
       "\n",
       "                             Negative Count  Positive Label Rate  \\\n",
       "         Groups                                                    \n",
       "original Overall                     9289.0             0.239292   \n",
       "         0                            101.0             0.081818   \n",
       "         1                            258.0             0.277311   \n",
       "         2                           1044.0             0.127090   \n",
       "         3                            103.0             0.134454   \n",
       "         4                           7783.0             0.253716   \n",
       "         Maximum difference          7682.0             0.195493   \n",
       "updated  Overall                     9289.0             0.239292   \n",
       "         0                            101.0             0.081818   \n",
       "         1                            258.0             0.277311   \n",
       "         2                           1044.0             0.127090   \n",
       "         3                            103.0             0.134454   \n",
       "         4                           7783.0             0.253716   \n",
       "         Maximum difference          7682.0             0.195493   \n",
       "\n",
       "                             Positive Prediction Rate  \n",
       "         Groups                                        \n",
       "original Overall                             0.205470  \n",
       "         0                                   0.036364  \n",
       "         1                                   0.240896  \n",
       "         2                                   0.095318  \n",
       "         3                                   0.084034  \n",
       "         4                                   0.220059  \n",
       "         Maximum difference                  0.204533  \n",
       "updated  Overall                             0.197773  \n",
       "         0                                   0.100000  \n",
       "         1                                   0.218487  \n",
       "         2                                   0.174749  \n",
       "         3                                   0.268908  \n",
       "         4                                   0.199923  \n",
       "         Maximum difference                  0.168908  "
      ]
     },
     "execution_count": 14,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "#We find that the demographic parity is very different on test data for a range of solutions found.\n",
    "#By looking at the per group decomposition, we can find out why.\n",
    "fpredictor.evaluate_groups(test_data)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th>Accuracy</th>\n",
       "      <th>Balanced Accuracy</th>\n",
       "      <th>F1 score</th>\n",
       "      <th>MCC</th>\n",
       "      <th>Precision</th>\n",
       "      <th>Recall</th>\n",
       "      <th>ROC AUC</th>\n",
       "      <th>Number of Datapoints</th>\n",
       "      <th>Positive Count</th>\n",
       "      <th>Negative Count</th>\n",
       "      <th>Positive Label Rate</th>\n",
       "      <th>Positive Prediction Rate</th>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th></th>\n",
       "      <th>Groups</th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th rowspan=\"7\" valign=\"top\">original</th>\n",
       "      <th>Overall</th>\n",
       "      <td>0.868550</td>\n",
       "      <td>0.794791</td>\n",
       "      <td>0.704038</td>\n",
       "      <td>0.623305</td>\n",
       "      <td>0.763295</td>\n",
       "      <td>0.653320</td>\n",
       "      <td>0.924039</td>\n",
       "      <td>12210.0</td>\n",
       "      <td>2922.0</td>\n",
       "      <td>9288.0</td>\n",
       "      <td>0.239312</td>\n",
       "      <td>0.204832</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>0.888000</td>\n",
       "      <td>0.666667</td>\n",
       "      <td>0.500000</td>\n",
       "      <td>0.542020</td>\n",
       "      <td>1.000000</td>\n",
       "      <td>0.333333</td>\n",
       "      <td>0.897436</td>\n",
       "      <td>125.0</td>\n",
       "      <td>21.0</td>\n",
       "      <td>104.0</td>\n",
       "      <td>0.168000</td>\n",
       "      <td>0.056000</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>0.818428</td>\n",
       "      <td>0.755598</td>\n",
       "      <td>0.663317</td>\n",
       "      <td>0.546588</td>\n",
       "      <td>0.741573</td>\n",
       "      <td>0.600000</td>\n",
       "      <td>0.902984</td>\n",
       "      <td>369.0</td>\n",
       "      <td>110.0</td>\n",
       "      <td>259.0</td>\n",
       "      <td>0.298103</td>\n",
       "      <td>0.241192</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>0.925613</td>\n",
       "      <td>0.754264</td>\n",
       "      <td>0.633333</td>\n",
       "      <td>0.608911</td>\n",
       "      <td>0.791667</td>\n",
       "      <td>0.527778</td>\n",
       "      <td>0.952000</td>\n",
       "      <td>1183.0</td>\n",
       "      <td>144.0</td>\n",
       "      <td>1039.0</td>\n",
       "      <td>0.121724</td>\n",
       "      <td>0.081150</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>0.948980</td>\n",
       "      <td>0.827519</td>\n",
       "      <td>0.761905</td>\n",
       "      <td>0.743516</td>\n",
       "      <td>0.888889</td>\n",
       "      <td>0.666667</td>\n",
       "      <td>0.978682</td>\n",
       "      <td>98.0</td>\n",
       "      <td>12.0</td>\n",
       "      <td>86.0</td>\n",
       "      <td>0.122449</td>\n",
       "      <td>0.091837</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>0.862865</td>\n",
       "      <td>0.797320</td>\n",
       "      <td>0.710030</td>\n",
       "      <td>0.623230</td>\n",
       "      <td>0.761739</td>\n",
       "      <td>0.664896</td>\n",
       "      <td>0.920673</td>\n",
       "      <td>10435.0</td>\n",
       "      <td>2635.0</td>\n",
       "      <td>7800.0</td>\n",
       "      <td>0.252516</td>\n",
       "      <td>0.220412</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Maximum difference</th>\n",
       "      <td>0.130551</td>\n",
       "      <td>0.160853</td>\n",
       "      <td>0.261905</td>\n",
       "      <td>0.201497</td>\n",
       "      <td>0.258427</td>\n",
       "      <td>0.333333</td>\n",
       "      <td>0.081246</td>\n",
       "      <td>10337.0</td>\n",
       "      <td>2623.0</td>\n",
       "      <td>7714.0</td>\n",
       "      <td>0.176379</td>\n",
       "      <td>0.185192</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th rowspan=\"7\" valign=\"top\">updated</th>\n",
       "      <th>Overall</th>\n",
       "      <td>0.868223</td>\n",
       "      <td>0.787069</td>\n",
       "      <td>0.696358</td>\n",
       "      <td>0.618672</td>\n",
       "      <td>0.776188</td>\n",
       "      <td>0.631417</td>\n",
       "      <td>0.898137</td>\n",
       "      <td>12210.0</td>\n",
       "      <td>2922.0</td>\n",
       "      <td>9288.0</td>\n",
       "      <td>0.239312</td>\n",
       "      <td>0.194676</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>0.848000</td>\n",
       "      <td>0.756639</td>\n",
       "      <td>0.577778</td>\n",
       "      <td>0.487206</td>\n",
       "      <td>0.541667</td>\n",
       "      <td>0.619048</td>\n",
       "      <td>0.897436</td>\n",
       "      <td>125.0</td>\n",
       "      <td>21.0</td>\n",
       "      <td>104.0</td>\n",
       "      <td>0.168000</td>\n",
       "      <td>0.192000</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>0.829268</td>\n",
       "      <td>0.750246</td>\n",
       "      <td>0.659459</td>\n",
       "      <td>0.568903</td>\n",
       "      <td>0.813333</td>\n",
       "      <td>0.554545</td>\n",
       "      <td>0.902984</td>\n",
       "      <td>369.0</td>\n",
       "      <td>110.0</td>\n",
       "      <td>259.0</td>\n",
       "      <td>0.298103</td>\n",
       "      <td>0.203252</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>0.914624</td>\n",
       "      <td>0.864657</td>\n",
       "      <td>0.694864</td>\n",
       "      <td>0.653662</td>\n",
       "      <td>0.614973</td>\n",
       "      <td>0.798611</td>\n",
       "      <td>0.952000</td>\n",
       "      <td>1183.0</td>\n",
       "      <td>144.0</td>\n",
       "      <td>1039.0</td>\n",
       "      <td>0.121724</td>\n",
       "      <td>0.158073</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>0.908163</td>\n",
       "      <td>0.911822</td>\n",
       "      <td>0.709677</td>\n",
       "      <td>0.682950</td>\n",
       "      <td>0.578947</td>\n",
       "      <td>0.916667</td>\n",
       "      <td>0.978682</td>\n",
       "      <td>98.0</td>\n",
       "      <td>12.0</td>\n",
       "      <td>86.0</td>\n",
       "      <td>0.122449</td>\n",
       "      <td>0.193878</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>0.864207</td>\n",
       "      <td>0.784772</td>\n",
       "      <td>0.698959</td>\n",
       "      <td>0.620283</td>\n",
       "      <td>0.793919</td>\n",
       "      <td>0.624288</td>\n",
       "      <td>0.920673</td>\n",
       "      <td>10435.0</td>\n",
       "      <td>2635.0</td>\n",
       "      <td>7800.0</td>\n",
       "      <td>0.252516</td>\n",
       "      <td>0.198563</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Maximum difference</th>\n",
       "      <td>0.085356</td>\n",
       "      <td>0.161576</td>\n",
       "      <td>0.131900</td>\n",
       "      <td>0.195743</td>\n",
       "      <td>0.271667</td>\n",
       "      <td>0.362121</td>\n",
       "      <td>0.081246</td>\n",
       "      <td>10337.0</td>\n",
       "      <td>2623.0</td>\n",
       "      <td>7714.0</td>\n",
       "      <td>0.176379</td>\n",
       "      <td>0.045179</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "                             Accuracy  Balanced Accuracy  F1 score       MCC  \\\n",
       "         Groups                                                                \n",
       "original Overall             0.868550           0.794791  0.704038  0.623305   \n",
       "         0                   0.888000           0.666667  0.500000  0.542020   \n",
       "         1                   0.818428           0.755598  0.663317  0.546588   \n",
       "         2                   0.925613           0.754264  0.633333  0.608911   \n",
       "         3                   0.948980           0.827519  0.761905  0.743516   \n",
       "         4                   0.862865           0.797320  0.710030  0.623230   \n",
       "         Maximum difference  0.130551           0.160853  0.261905  0.201497   \n",
       "updated  Overall             0.868223           0.787069  0.696358  0.618672   \n",
       "         0                   0.848000           0.756639  0.577778  0.487206   \n",
       "         1                   0.829268           0.750246  0.659459  0.568903   \n",
       "         2                   0.914624           0.864657  0.694864  0.653662   \n",
       "         3                   0.908163           0.911822  0.709677  0.682950   \n",
       "         4                   0.864207           0.784772  0.698959  0.620283   \n",
       "         Maximum difference  0.085356           0.161576  0.131900  0.195743   \n",
       "\n",
       "                             Precision    Recall   ROC AUC  \\\n",
       "         Groups                                              \n",
       "original Overall              0.763295  0.653320  0.924039   \n",
       "         0                    1.000000  0.333333  0.897436   \n",
       "         1                    0.741573  0.600000  0.902984   \n",
       "         2                    0.791667  0.527778  0.952000   \n",
       "         3                    0.888889  0.666667  0.978682   \n",
       "         4                    0.761739  0.664896  0.920673   \n",
       "         Maximum difference   0.258427  0.333333  0.081246   \n",
       "updated  Overall              0.776188  0.631417  0.898137   \n",
       "         0                    0.541667  0.619048  0.897436   \n",
       "         1                    0.813333  0.554545  0.902984   \n",
       "         2                    0.614973  0.798611  0.952000   \n",
       "         3                    0.578947  0.916667  0.978682   \n",
       "         4                    0.793919  0.624288  0.920673   \n",
       "         Maximum difference   0.271667  0.362121  0.081246   \n",
       "\n",
       "                             Number of Datapoints  Positive Count  \\\n",
       "         Groups                                                     \n",
       "original Overall                          12210.0          2922.0   \n",
       "         0                                  125.0            21.0   \n",
       "         1                                  369.0           110.0   \n",
       "         2                                 1183.0           144.0   \n",
       "         3                                   98.0            12.0   \n",
       "         4                                10435.0          2635.0   \n",
       "         Maximum difference               10337.0          2623.0   \n",
       "updated  Overall                          12210.0          2922.0   \n",
       "         0                                  125.0            21.0   \n",
       "         1                                  369.0           110.0   \n",
       "         2                                 1183.0           144.0   \n",
       "         3                                   98.0            12.0   \n",
       "         4                                10435.0          2635.0   \n",
       "         Maximum difference               10337.0          2623.0   \n",
       "\n",
       "                             Negative Count  Positive Label Rate  \\\n",
       "         Groups                                                    \n",
       "original Overall                     9288.0             0.239312   \n",
       "         0                            104.0             0.168000   \n",
       "         1                            259.0             0.298103   \n",
       "         2                           1039.0             0.121724   \n",
       "         3                             86.0             0.122449   \n",
       "         4                           7800.0             0.252516   \n",
       "         Maximum difference          7714.0             0.176379   \n",
       "updated  Overall                     9288.0             0.239312   \n",
       "         0                            104.0             0.168000   \n",
       "         1                            259.0             0.298103   \n",
       "         2                           1039.0             0.121724   \n",
       "         3                             86.0             0.122449   \n",
       "         4                           7800.0             0.252516   \n",
       "         Maximum difference          7714.0             0.176379   \n",
       "\n",
       "                             Positive Prediction Rate  \n",
       "         Groups                                        \n",
       "original Overall                             0.204832  \n",
       "         0                                   0.056000  \n",
       "         1                                   0.241192  \n",
       "         2                                   0.081150  \n",
       "         3                                   0.091837  \n",
       "         4                                   0.220412  \n",
       "         Maximum difference                  0.185192  \n",
       "updated  Overall                             0.194676  \n",
       "         0                                   0.192000  \n",
       "         1                                   0.203252  \n",
       "         2                                   0.158073  \n",
       "         3                                   0.193878  \n",
       "         4                                   0.198563  \n",
       "         Maximum difference                  0.045179  "
      ]
     },
     "execution_count": 15,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "#The groups with the smallest Positive Prediction Rate (corresponding to American-Indian-Eskimo, and Asian-Pacific-Islander) \n",
    "# only have around 100 samples making it impossible to accurately evaluate demographic parity, while the\n",
    "# group labelled 'Other' has even less data.\n",
    "# Moreover, on validation data we find that there are less than 90 people identified as American-Indian-Eskimo, \n",
    "# meaning that it is not possible to predict if the positive prediction rate will hold on new data. \n",
    "fpredictor.evaluate_groups()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Fairness on COMPAS using Inferred Attributes\n",
    "\n",
    "We demonstrate how to enforce a wide range of fairness definitions on the COMPAS dataset. This dataset records paroles caught violating the terms of parole. As it measures who was caught, it is strongly influenced by policing and environmental biases, and should not be confused with a measurement of who actually violated their terms of parole. See [this paper](https://datasets-benchmarks-proceedings.neurips.cc/paper/2021/file/92cc227532d17e56e07902b254dfad10-Paper-round1.pdf) for a discussion of its limitations and caveats. \n",
    "We use it because it is a standard fairness dataset that captures such strong differences in outcome between people identified as African-American and everyone else, that classifiers trained on this dataset violate most definitions of fairness.\n",
    "\n",
    "As many of the ethnic groups are too small for reliable statistical estimation, we only consider differences is in outcomes between African-Americans vs. everyone else (labeled as other).\n",
    "We load and preprocess the COMPAS dataset, splitting it into three roughly equal partitions of train, validation, and test:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [],
   "source": [
    "#Generate two sets of training, validation and test, with race and without.\n",
    "train, val, test = dataset_loader.compas('race', train_proportion=0.66, test_proportion=0.33, discard_groups=True, replace_groups={'Hispanic':'Other', 'Native American':'Other', 'Asian':'Other'})\n",
    "train_g, val_g, test_g = dataset_loader.compas('race', train_proportion=0.66, test_proportion=0.33, replace_groups={'Hispanic':'Other', 'Native American':'Other', 'Asian':'Other'})"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [],
   "source": [
    "predictor = xgboost.XGBClassifier().fit(X=train_g['data'],y=train_g['target'])\n",
    "predictor2 = xgboost.XGBClassifier().fit(X=train['data'],y=train['target'])\n",
    "protected = xgboost.XGBClassifier().fit(X=train['data'],y=2*(train['groups']=='Other')+(train['groups']=='Caucasian'))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {},
   "outputs": [],
   "source": [
    "fpredictor=FairPredictor(predictor2, train, inferred_groups=protected)\n",
    "fpredictor.fit(gm.accuracy, gm.demographic_parity, 0.025)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {},
   "outputs": [],
   "source": [
    "#However, instead we will show how a family of fairness measures can be individually optimized. \n",
    "# The following code plots a table showing the change in accuracy and the fairness measure on a held-out test set as we decrease\n",
    "# the fairness measure to less than 0.025 for all measures except for disparate impact which we raise to above 0.975.\n",
    "# We define a helper function for evaluation:\n",
    "\n",
    "def evaluate(fpredictor, use_metrics, test):\n",
    "    \"Print a table showing the accuracy drop that comes with enforcing fairness\"\n",
    "    extra_metrics= {**use_metrics, 'accuracy':gm.accuracy}\n",
    "    collect=pd.DataFrame(columns=['Measure (original)', 'Measure (updated)', 'Accuracy (original)', 'Accuracy (updated)'])\n",
    "    for d in use_metrics.items():\n",
    "        if d[1].greater_is_better is False:\n",
    "            fpredictor.fit(gm.accuracy,d[1], 0.025)\n",
    "        else:\n",
    "            fpredictor.fit(gm.accuracy,d[1], 1-0.025)\n",
    "        tmp=fpredictor.evaluate_fairness(test,metrics=extra_metrics)\n",
    "        collect.loc[d[1].name]=np.concatenate((np.asarray(tmp.loc[d[1].name]), np.asarray(tmp.loc[gm.accuracy.name])), 0)\n",
    "    return collect"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Index(['sex', 'age', 'juv_fel_count', 'juv_misd_count', 'juv_other_count',\n",
       "       'priors_count', 'age_cat', 'c_charge_degree'],\n",
       "      dtype='object')"
      ]
     },
     "execution_count": 20,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "train['data'].columns"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Groups passed twice to fairpredictor both as part of the dataset and as an argument. The argument will be used.\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>Measure (original)</th>\n",
       "      <th>Measure (updated)</th>\n",
       "      <th>Accuracy (original)</th>\n",
       "      <th>Accuracy (updated)</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>Demographic Parity</th>\n",
       "      <td>0.159466</td>\n",
       "      <td>0.011079</td>\n",
       "      <td>0.657563</td>\n",
       "      <td>0.646218</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Disparate Impact</th>\n",
       "      <td>0.623388</td>\n",
       "      <td>0.944255</td>\n",
       "      <td>0.657563</td>\n",
       "      <td>0.647899</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in Conditional Acceptance Rate</th>\n",
       "      <td>0.351803</td>\n",
       "      <td>0.064117</td>\n",
       "      <td>0.657563</td>\n",
       "      <td>0.650000</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in Conditional Rejectance Rate</th>\n",
       "      <td>0.052968</td>\n",
       "      <td>0.044369</td>\n",
       "      <td>0.657563</td>\n",
       "      <td>0.653361</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in Accuracy</th>\n",
       "      <td>0.017547</td>\n",
       "      <td>0.011044</td>\n",
       "      <td>0.657563</td>\n",
       "      <td>0.653361</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in Recall</th>\n",
       "      <td>0.166757</td>\n",
       "      <td>0.063227</td>\n",
       "      <td>0.657563</td>\n",
       "      <td>0.648319</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in Acceptance Rate</th>\n",
       "      <td>0.062387</td>\n",
       "      <td>0.093970</td>\n",
       "      <td>0.657563</td>\n",
       "      <td>0.651681</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in Specificity</th>\n",
       "      <td>0.107465</td>\n",
       "      <td>0.026887</td>\n",
       "      <td>0.657563</td>\n",
       "      <td>0.649580</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in Rejection Rate</th>\n",
       "      <td>0.050992</td>\n",
       "      <td>0.040492</td>\n",
       "      <td>0.657563</td>\n",
       "      <td>0.652941</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Treatment Equality</th>\n",
       "      <td>0.223708</td>\n",
       "      <td>0.074627</td>\n",
       "      <td>0.657563</td>\n",
       "      <td>0.647059</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "                                                    Measure (original)  \\\n",
       "Demographic Parity                                            0.159466   \n",
       "Disparate Impact                                              0.623388   \n",
       "Average Group Difference in Conditional Accepta...            0.351803   \n",
       "Average Group Difference in Conditional Rejecta...            0.052968   \n",
       "Average Group Difference in Accuracy                          0.017547   \n",
       "Average Group Difference in Recall                            0.166757   \n",
       "Average Group Difference in Acceptance Rate                   0.062387   \n",
       "Average Group Difference in Specificity                       0.107465   \n",
       "Average Group Difference in Rejection Rate                    0.050992   \n",
       "Treatment Equality                                            0.223708   \n",
       "\n",
       "                                                    Measure (updated)  \\\n",
       "Demographic Parity                                           0.011079   \n",
       "Disparate Impact                                             0.944255   \n",
       "Average Group Difference in Conditional Accepta...           0.064117   \n",
       "Average Group Difference in Conditional Rejecta...           0.044369   \n",
       "Average Group Difference in Accuracy                         0.011044   \n",
       "Average Group Difference in Recall                           0.063227   \n",
       "Average Group Difference in Acceptance Rate                  0.093970   \n",
       "Average Group Difference in Specificity                      0.026887   \n",
       "Average Group Difference in Rejection Rate                   0.040492   \n",
       "Treatment Equality                                           0.074627   \n",
       "\n",
       "                                                    Accuracy (original)  \\\n",
       "Demographic Parity                                             0.657563   \n",
       "Disparate Impact                                               0.657563   \n",
       "Average Group Difference in Conditional Accepta...             0.657563   \n",
       "Average Group Difference in Conditional Rejecta...             0.657563   \n",
       "Average Group Difference in Accuracy                           0.657563   \n",
       "Average Group Difference in Recall                             0.657563   \n",
       "Average Group Difference in Acceptance Rate                    0.657563   \n",
       "Average Group Difference in Specificity                        0.657563   \n",
       "Average Group Difference in Rejection Rate                     0.657563   \n",
       "Treatment Equality                                             0.657563   \n",
       "\n",
       "                                                    Accuracy (updated)  \n",
       "Demographic Parity                                            0.646218  \n",
       "Disparate Impact                                              0.647899  \n",
       "Average Group Difference in Conditional Accepta...            0.650000  \n",
       "Average Group Difference in Conditional Rejecta...            0.653361  \n",
       "Average Group Difference in Accuracy                          0.653361  \n",
       "Average Group Difference in Recall                            0.648319  \n",
       "Average Group Difference in Acceptance Rate                   0.651681  \n",
       "Average Group Difference in Specificity                       0.649580  \n",
       "Average Group Difference in Rejection Rate                    0.652941  \n",
       "Treatment Equality                                            0.647059  "
      ]
     },
     "execution_count": 21,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "#We can now contrast the behavior of a fair classifier that relies on access to the protected\n",
    "# attribtute at test time with one that infers it.\n",
    "\n",
    "#Create a fair predictor object, using the attribute 'race' on validation data\n",
    "fpredictor = FairPredictor(predictor, train_g, 'race')\n",
    "#and then evaluate it\n",
    "true_groups = evaluate(fpredictor, gm.clarify_metrics, test_g)\n",
    "\n",
    "true_groups"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>Measure (original)</th>\n",
       "      <th>Measure (updated)</th>\n",
       "      <th>Accuracy (original)</th>\n",
       "      <th>Accuracy (updated)</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>Demographic Parity</th>\n",
       "      <td>0.151470</td>\n",
       "      <td>0.084651</td>\n",
       "      <td>0.67395</td>\n",
       "      <td>0.627731</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Disparate Impact</th>\n",
       "      <td>0.677047</td>\n",
       "      <td>0.736657</td>\n",
       "      <td>0.67395</td>\n",
       "      <td>0.619328</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in Conditional Acceptance Rate</th>\n",
       "      <td>0.225513</td>\n",
       "      <td>0.176817</td>\n",
       "      <td>0.67395</td>\n",
       "      <td>0.657563</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in Conditional Rejectance Rate</th>\n",
       "      <td>0.067061</td>\n",
       "      <td>0.048793</td>\n",
       "      <td>0.67395</td>\n",
       "      <td>0.665546</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in Accuracy</th>\n",
       "      <td>0.019008</td>\n",
       "      <td>0.021466</td>\n",
       "      <td>0.67395</td>\n",
       "      <td>0.670588</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in Recall</th>\n",
       "      <td>0.145031</td>\n",
       "      <td>0.082055</td>\n",
       "      <td>0.67395</td>\n",
       "      <td>0.630672</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in Acceptance Rate</th>\n",
       "      <td>0.057459</td>\n",
       "      <td>0.060132</td>\n",
       "      <td>0.67395</td>\n",
       "      <td>0.680252</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in Specificity</th>\n",
       "      <td>0.103984</td>\n",
       "      <td>0.090335</td>\n",
       "      <td>0.67395</td>\n",
       "      <td>0.665126</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in Rejection Rate</th>\n",
       "      <td>0.051372</td>\n",
       "      <td>0.052749</td>\n",
       "      <td>0.67395</td>\n",
       "      <td>0.671849</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Treatment Equality</th>\n",
       "      <td>0.249687</td>\n",
       "      <td>0.168621</td>\n",
       "      <td>0.67395</td>\n",
       "      <td>0.666387</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "                                                    Measure (original)  \\\n",
       "Demographic Parity                                            0.151470   \n",
       "Disparate Impact                                              0.677047   \n",
       "Average Group Difference in Conditional Accepta...            0.225513   \n",
       "Average Group Difference in Conditional Rejecta...            0.067061   \n",
       "Average Group Difference in Accuracy                          0.019008   \n",
       "Average Group Difference in Recall                            0.145031   \n",
       "Average Group Difference in Acceptance Rate                   0.057459   \n",
       "Average Group Difference in Specificity                       0.103984   \n",
       "Average Group Difference in Rejection Rate                    0.051372   \n",
       "Treatment Equality                                            0.249687   \n",
       "\n",
       "                                                    Measure (updated)  \\\n",
       "Demographic Parity                                           0.084651   \n",
       "Disparate Impact                                             0.736657   \n",
       "Average Group Difference in Conditional Accepta...           0.176817   \n",
       "Average Group Difference in Conditional Rejecta...           0.048793   \n",
       "Average Group Difference in Accuracy                         0.021466   \n",
       "Average Group Difference in Recall                           0.082055   \n",
       "Average Group Difference in Acceptance Rate                  0.060132   \n",
       "Average Group Difference in Specificity                      0.090335   \n",
       "Average Group Difference in Rejection Rate                   0.052749   \n",
       "Treatment Equality                                           0.168621   \n",
       "\n",
       "                                                    Accuracy (original)  \\\n",
       "Demographic Parity                                              0.67395   \n",
       "Disparate Impact                                                0.67395   \n",
       "Average Group Difference in Conditional Accepta...              0.67395   \n",
       "Average Group Difference in Conditional Rejecta...              0.67395   \n",
       "Average Group Difference in Accuracy                            0.67395   \n",
       "Average Group Difference in Recall                              0.67395   \n",
       "Average Group Difference in Acceptance Rate                     0.67395   \n",
       "Average Group Difference in Specificity                         0.67395   \n",
       "Average Group Difference in Rejection Rate                      0.67395   \n",
       "Treatment Equality                                              0.67395   \n",
       "\n",
       "                                                    Accuracy (updated)  \n",
       "Demographic Parity                                            0.627731  \n",
       "Disparate Impact                                              0.619328  \n",
       "Average Group Difference in Conditional Accepta...            0.657563  \n",
       "Average Group Difference in Conditional Rejecta...            0.665546  \n",
       "Average Group Difference in Accuracy                          0.670588  \n",
       "Average Group Difference in Recall                            0.630672  \n",
       "Average Group Difference in Acceptance Rate                   0.680252  \n",
       "Average Group Difference in Specificity                       0.665126  \n",
       "Average Group Difference in Rejection Rate                    0.671849  \n",
       "Treatment Equality                                            0.666387  "
      ]
     },
     "execution_count": 22,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    " # Now using infered attributes\n",
    "#Note that group information is not present in train['data']\n",
    "#But contained in train['groups'] as such we can not specify groups='race'\n",
    "fpredictor2 = FairPredictor(predictor2, train, inferred_groups=protected) \n",
    "inferred = evaluate(fpredictor2, gm.clarify_metrics, test)\n",
    "inferred"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th>Accuracy</th>\n",
       "      <th>Balanced Accuracy</th>\n",
       "      <th>F1 score</th>\n",
       "      <th>MCC</th>\n",
       "      <th>Precision</th>\n",
       "      <th>Recall</th>\n",
       "      <th>ROC AUC</th>\n",
       "      <th>Number of Datapoints</th>\n",
       "      <th>Positive Count</th>\n",
       "      <th>Negative Count</th>\n",
       "      <th>Positive Label Rate</th>\n",
       "      <th>Positive Prediction Rate</th>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th></th>\n",
       "      <th>Groups</th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th rowspan=\"5\" valign=\"top\">original</th>\n",
       "      <th>Overall</th>\n",
       "      <td>0.673950</td>\n",
       "      <td>0.664093</td>\n",
       "      <td>0.609265</td>\n",
       "      <td>0.335820</td>\n",
       "      <td>0.662651</td>\n",
       "      <td>0.563840</td>\n",
       "      <td>0.714265</td>\n",
       "      <td>2380.0</td>\n",
       "      <td>1073.0</td>\n",
       "      <td>1307.0</td>\n",
       "      <td>0.450840</td>\n",
       "      <td>0.383613</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>African-American</th>\n",
       "      <td>0.660377</td>\n",
       "      <td>0.660899</td>\n",
       "      <td>0.660656</td>\n",
       "      <td>0.321783</td>\n",
       "      <td>0.679595</td>\n",
       "      <td>0.642743</td>\n",
       "      <td>0.706468</td>\n",
       "      <td>1219.0</td>\n",
       "      <td>627.0</td>\n",
       "      <td>592.0</td>\n",
       "      <td>0.514356</td>\n",
       "      <td>0.486464</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Caucasian</th>\n",
       "      <td>0.688889</td>\n",
       "      <td>0.649490</td>\n",
       "      <td>0.540146</td>\n",
       "      <td>0.324394</td>\n",
       "      <td>0.646288</td>\n",
       "      <td>0.463950</td>\n",
       "      <td>0.693601</td>\n",
       "      <td>810.0</td>\n",
       "      <td>319.0</td>\n",
       "      <td>491.0</td>\n",
       "      <td>0.393827</td>\n",
       "      <td>0.282716</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Other</th>\n",
       "      <td>0.686610</td>\n",
       "      <td>0.630009</td>\n",
       "      <td>0.495413</td>\n",
       "      <td>0.285117</td>\n",
       "      <td>0.593407</td>\n",
       "      <td>0.425197</td>\n",
       "      <td>0.715323</td>\n",
       "      <td>351.0</td>\n",
       "      <td>127.0</td>\n",
       "      <td>224.0</td>\n",
       "      <td>0.361823</td>\n",
       "      <td>0.259259</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Maximum difference</th>\n",
       "      <td>0.028512</td>\n",
       "      <td>0.030889</td>\n",
       "      <td>0.165243</td>\n",
       "      <td>0.039278</td>\n",
       "      <td>0.086189</td>\n",
       "      <td>0.217546</td>\n",
       "      <td>0.021722</td>\n",
       "      <td>868.0</td>\n",
       "      <td>500.0</td>\n",
       "      <td>368.0</td>\n",
       "      <td>0.152533</td>\n",
       "      <td>0.227205</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th rowspan=\"5\" valign=\"top\">updated</th>\n",
       "      <th>Overall</th>\n",
       "      <td>0.666387</td>\n",
       "      <td>0.651951</td>\n",
       "      <td>0.577210</td>\n",
       "      <td>0.319619</td>\n",
       "      <td>0.673292</td>\n",
       "      <td>0.505126</td>\n",
       "      <td>0.703703</td>\n",
       "      <td>2380.0</td>\n",
       "      <td>1073.0</td>\n",
       "      <td>1307.0</td>\n",
       "      <td>0.450840</td>\n",
       "      <td>0.338235</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>African-American</th>\n",
       "      <td>0.648893</td>\n",
       "      <td>0.651007</td>\n",
       "      <td>0.628472</td>\n",
       "      <td>0.304834</td>\n",
       "      <td>0.689524</td>\n",
       "      <td>0.577352</td>\n",
       "      <td>0.693327</td>\n",
       "      <td>1219.0</td>\n",
       "      <td>627.0</td>\n",
       "      <td>592.0</td>\n",
       "      <td>0.514356</td>\n",
       "      <td>0.430681</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Caucasian</th>\n",
       "      <td>0.688889</td>\n",
       "      <td>0.641803</td>\n",
       "      <td>0.515385</td>\n",
       "      <td>0.320808</td>\n",
       "      <td>0.666667</td>\n",
       "      <td>0.420063</td>\n",
       "      <td>0.687318</td>\n",
       "      <td>810.0</td>\n",
       "      <td>319.0</td>\n",
       "      <td>491.0</td>\n",
       "      <td>0.393827</td>\n",
       "      <td>0.248148</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Other</th>\n",
       "      <td>0.675214</td>\n",
       "      <td>0.607442</td>\n",
       "      <td>0.446602</td>\n",
       "      <td>0.247247</td>\n",
       "      <td>0.582278</td>\n",
       "      <td>0.362205</td>\n",
       "      <td>0.708960</td>\n",
       "      <td>351.0</td>\n",
       "      <td>127.0</td>\n",
       "      <td>224.0</td>\n",
       "      <td>0.361823</td>\n",
       "      <td>0.225071</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Maximum difference</th>\n",
       "      <td>0.039996</td>\n",
       "      <td>0.043566</td>\n",
       "      <td>0.181870</td>\n",
       "      <td>0.073562</td>\n",
       "      <td>0.107245</td>\n",
       "      <td>0.215148</td>\n",
       "      <td>0.021642</td>\n",
       "      <td>868.0</td>\n",
       "      <td>500.0</td>\n",
       "      <td>368.0</td>\n",
       "      <td>0.152533</td>\n",
       "      <td>0.205610</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "                             Accuracy  Balanced Accuracy  F1 score       MCC  \\\n",
       "         Groups                                                                \n",
       "original Overall             0.673950           0.664093  0.609265  0.335820   \n",
       "         African-American    0.660377           0.660899  0.660656  0.321783   \n",
       "         Caucasian           0.688889           0.649490  0.540146  0.324394   \n",
       "         Other               0.686610           0.630009  0.495413  0.285117   \n",
       "         Maximum difference  0.028512           0.030889  0.165243  0.039278   \n",
       "updated  Overall             0.666387           0.651951  0.577210  0.319619   \n",
       "         African-American    0.648893           0.651007  0.628472  0.304834   \n",
       "         Caucasian           0.688889           0.641803  0.515385  0.320808   \n",
       "         Other               0.675214           0.607442  0.446602  0.247247   \n",
       "         Maximum difference  0.039996           0.043566  0.181870  0.073562   \n",
       "\n",
       "                             Precision    Recall   ROC AUC  \\\n",
       "         Groups                                              \n",
       "original Overall              0.662651  0.563840  0.714265   \n",
       "         African-American     0.679595  0.642743  0.706468   \n",
       "         Caucasian            0.646288  0.463950  0.693601   \n",
       "         Other                0.593407  0.425197  0.715323   \n",
       "         Maximum difference   0.086189  0.217546  0.021722   \n",
       "updated  Overall              0.673292  0.505126  0.703703   \n",
       "         African-American     0.689524  0.577352  0.693327   \n",
       "         Caucasian            0.666667  0.420063  0.687318   \n",
       "         Other                0.582278  0.362205  0.708960   \n",
       "         Maximum difference   0.107245  0.215148  0.021642   \n",
       "\n",
       "                             Number of Datapoints  Positive Count  \\\n",
       "         Groups                                                     \n",
       "original Overall                           2380.0          1073.0   \n",
       "         African-American                  1219.0           627.0   \n",
       "         Caucasian                          810.0           319.0   \n",
       "         Other                              351.0           127.0   \n",
       "         Maximum difference                 868.0           500.0   \n",
       "updated  Overall                           2380.0          1073.0   \n",
       "         African-American                  1219.0           627.0   \n",
       "         Caucasian                          810.0           319.0   \n",
       "         Other                              351.0           127.0   \n",
       "         Maximum difference                 868.0           500.0   \n",
       "\n",
       "                             Negative Count  Positive Label Rate  \\\n",
       "         Groups                                                    \n",
       "original Overall                     1307.0             0.450840   \n",
       "         African-American             592.0             0.514356   \n",
       "         Caucasian                    491.0             0.393827   \n",
       "         Other                        224.0             0.361823   \n",
       "         Maximum difference           368.0             0.152533   \n",
       "updated  Overall                     1307.0             0.450840   \n",
       "         African-American             592.0             0.514356   \n",
       "         Caucasian                    491.0             0.393827   \n",
       "         Other                        224.0             0.361823   \n",
       "         Maximum difference           368.0             0.152533   \n",
       "\n",
       "                             Positive Prediction Rate  \n",
       "         Groups                                        \n",
       "original Overall                             0.383613  \n",
       "         African-American                    0.486464  \n",
       "         Caucasian                           0.282716  \n",
       "         Other                               0.259259  \n",
       "         Maximum difference                  0.227205  \n",
       "updated  Overall                             0.338235  \n",
       "         African-American                    0.430681  \n",
       "         Caucasian                           0.248148  \n",
       "         Other                               0.225071  \n",
       "         Maximum difference                  0.205610  "
      ]
     },
     "execution_count": 24,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "fpredictor2.evaluate_groups(test)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "metadata": {},
   "outputs": [],
   "source": [
    "from anonfair import conditional_group_metrics as cgm\n",
    "fpredictor2_cond = FairPredictor(predictor2, train, inferred_groups=protected,conditioning_factor='sex') \n",
    "fpredictor2_cond.fit(gm.accuracy,cgm.pos_pred_rate.diff,0.02)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>original</th>\n",
       "      <th>updated</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>Accuracy</th>\n",
       "      <td>0.728265</td>\n",
       "      <td>0.669467</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in Conditional Positive Prediction Rate</th>\n",
       "      <td>0.141604</td>\n",
       "      <td>0.019039</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "                                                    original   updated\n",
       "Accuracy                                            0.728265  0.669467\n",
       "Average Group Difference in Conditional Positiv...  0.141604  0.019039"
      ]
     },
     "execution_count": 27,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "fpredictor2_cond.evaluate(metrics={'a':gm.accuracy,'b':cgm.pos_pred_rate.diff})\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>original</th>\n",
       "      <th>updated</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>Accuracy</th>\n",
       "      <td>0.673950</td>\n",
       "      <td>0.626471</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in Conditional Positive Prediction Rate</th>\n",
       "      <td>0.150927</td>\n",
       "      <td>0.073203</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "                                                    original   updated\n",
       "Accuracy                                            0.673950  0.626471\n",
       "Average Group Difference in Conditional Positiv...  0.150927  0.073203"
      ]
     },
     "execution_count": 28,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "fpredictor2_cond.evaluate(test,metrics={'a':gm.accuracy,'b':cgm.pos_pred_rate.diff})\n"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "base",
   "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.10.13"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
