{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# FairPredictor Autogluon 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",
    "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",
      "No path specified. Models will be saved in: \"AutogluonModels/ag-20240522_162305\"\n",
      "No presets specified! To achieve strong results with AutoGluon, it is recommended to use the available presets.\n",
      "\tRecommended Presets (For more details refer to https://auto.gluon.ai/stable/tutorials/tabular/tabular-essentials.html#presets):\n",
      "\tpresets='best_quality'   : Maximize accuracy. Default time_limit=3600.\n",
      "\tpresets='high_quality'   : Strong accuracy with fast inference speed. Default time_limit=3600.\n",
      "\tpresets='good_quality'   : Good accuracy with very fast inference speed. Default time_limit=3600.\n",
      "\tpresets='medium_quality' : Fast training time, ideal for initial prototyping.\n",
      "Beginning AutoGluon training ... Time limit = 5s\n",
      "AutoGluon will save models to \"AutogluonModels/ag-20240522_162305\"\n",
      "=================== System Info ===================\n",
      "AutoGluon Version:  1.1.0\n",
      "Python Version:     3.10.13\n",
      "Operating System:   Darwin\n",
      "Platform Machine:   arm64\n",
      "Platform Version:   Darwin Kernel Version 23.5.0: Wed May  1 20:14:38 PDT 2024; root:xnu-10063.121.3~5/RELEASE_ARM64_T6020\n",
      "CPU Count:          10\n",
      "Memory Avail:       5.16 GB / 16.00 GB (32.3%)\n",
      "Disk Space Avail:   393.46 GB / 460.43 GB (85.5%)\n",
      "===================================================\n",
      "Train Data Rows:    39073\n",
      "Train Data Columns: 14\n",
      "Label Column:       class\n",
      "AutoGluon infers your prediction problem is: 'binary' (because only two unique label-values observed).\n",
      "\t2 unique label values:  [' <=50K', ' >50K']\n",
      "\tIf 'binary' is not the correct problem_type, please manually specify the problem_type parameter during predictor init (You may specify problem_type as one of: ['binary', 'multiclass', 'regression'])\n",
      "Problem Type:       binary\n",
      "Preprocessing data ...\n",
      "Selected class <--> label mapping:  class 1 =  >50K, class 0 =  <=50K\n",
      "\tNote: For your binary classification, AutoGluon arbitrarily selected which label-value represents positive ( >50K) vs negative ( <=50K) class.\n",
      "\tTo explicitly set the positive_class, either rename classes to 1 and 0, or specify positive_class in Predictor init.\n",
      "Using Feature Generators to preprocess the data ...\n",
      "Fitting AutoMLPipelineFeatureGenerator...\n",
      "\tAvailable Memory:                    5299.46 MB\n",
      "\tTrain Data (Original)  Memory Usage: 21.86 MB (0.4% of available memory)\n",
      "\tInferring data type of each feature based on column values. Set feature_metadata_in to manually specify special dtypes of the features.\n",
      "\tStage 1 Generators:\n",
      "\t\tFitting AsTypeFeatureGenerator...\n",
      "\t\t\tNote: Converting 1 features to boolean dtype as they only contain 2 unique values.\n",
      "\tStage 2 Generators:\n",
      "\t\tFitting FillNaFeatureGenerator...\n",
      "\tStage 3 Generators:\n",
      "\t\tFitting IdentityFeatureGenerator...\n",
      "\t\tFitting CategoryFeatureGenerator...\n",
      "\t\t\tFitting CategoryMemoryMinimizeFeatureGenerator...\n",
      "\tStage 4 Generators:\n",
      "\t\tFitting DropUniqueFeatureGenerator...\n",
      "\tStage 5 Generators:\n",
      "\t\tFitting DropDuplicatesFeatureGenerator...\n",
      "\tTypes of features in original data (raw dtype, special dtypes):\n",
      "\t\t('int', [])    : 6 | ['age', 'fnlwgt', 'education-num', 'capital-gain', 'capital-loss', ...]\n",
      "\t\t('object', []) : 8 | ['workclass', 'education', 'marital-status', 'occupation', 'relationship', ...]\n",
      "\tTypes of features in processed data (raw dtype, special dtypes):\n",
      "\t\t('category', [])  : 7 | ['workclass', 'education', 'marital-status', 'occupation', 'relationship', ...]\n",
      "\t\t('int', [])       : 6 | ['age', 'fnlwgt', 'education-num', 'capital-gain', 'capital-loss', ...]\n",
      "\t\t('int', ['bool']) : 1 | ['sex']\n",
      "\t0.1s = Fit runtime\n",
      "\t14 features in original data used to generate 14 features in processed data.\n",
      "\tTrain Data (Processed) Memory Usage: 2.09 MB (0.0% of available memory)\n",
      "Data preprocessing and feature engineering runtime = 0.16s ...\n",
      "AutoGluon will gauge predictive performance using evaluation metric: 'accuracy'\n",
      "\tTo change this, specify the eval_metric parameter of Predictor()\n",
      "Automatically generating train/validation split with holdout_frac=0.0639828014229775, Train Rows: 36573, Val Rows: 2500\n",
      "User-specified model hyperparameters to be fit:\n",
      "{\n",
      "\t'NN_TORCH': {},\n",
      "\t'GBM': [{'extra_trees': True, 'ag_args': {'name_suffix': 'XT'}}, {}, 'GBMLarge'],\n",
      "\t'CAT': {},\n",
      "\t'XGB': {},\n",
      "\t'FASTAI': {},\n",
      "\t'RF': [{'criterion': 'gini', 'ag_args': {'name_suffix': 'Gini', 'problem_types': ['binary', 'multiclass']}}, {'criterion': 'entropy', 'ag_args': {'name_suffix': 'Entr', 'problem_types': ['binary', 'multiclass']}}, {'criterion': 'squared_error', 'ag_args': {'name_suffix': 'MSE', 'problem_types': ['regression', 'quantile']}}],\n",
      "\t'XT': [{'criterion': 'gini', 'ag_args': {'name_suffix': 'Gini', 'problem_types': ['binary', 'multiclass']}}, {'criterion': 'entropy', 'ag_args': {'name_suffix': 'Entr', 'problem_types': ['binary', 'multiclass']}}, {'criterion': 'squared_error', 'ag_args': {'name_suffix': 'MSE', 'problem_types': ['regression', 'quantile']}}],\n",
      "\t'KNN': [{'weights': 'uniform', 'ag_args': {'name_suffix': 'Unif'}}, {'weights': 'distance', 'ag_args': {'name_suffix': 'Dist'}}],\n",
      "}\n",
      "Fitting 13 L1 models ...\n",
      "Fitting model: KNeighborsUnif ... Training model for up to 4.84s of the 4.84s of remaining time.\n",
      "\t0.7752\t = Validation score   (accuracy)\n",
      "\t5.87s\t = Training   runtime\n",
      "\t0.06s\t = Validation runtime\n",
      "Fitting model: WeightedEnsemble_L2 ... Training model for up to 4.84s of the -1.12s of remaining time.\n",
      "\tEnsemble Weights: {'KNeighborsUnif': 1.0}\n",
      "\t0.7752\t = Validation score   (accuracy)\n",
      "\t0.0s\t = Training   runtime\n",
      "\t0.0s\t = Validation runtime\n",
      "AutoGluon training complete, total runtime = 6.15s ... Best model: \"WeightedEnsemble_L2\"\n",
      "TabularPredictor saved. To load, use: predictor = TabularPredictor.load(\"AutogluonModels/ag-20240522_162305\")\n"
     ]
    }
   ],
   "source": [
    "# Load and train a baseline classifier\n",
    "\n",
    "from autogluon.tabular import TabularDataset, TabularPredictor\n",
    "from anonfair import FairPredictor \n",
    "from anonfair.utils import group_metrics as gm\n",
    "train_data = TabularDataset('https://autogluon.s3.amazonaws.com/datasets/Inc/train.csv')\n",
    "test_data = TabularDataset('https://autogluon.s3.amazonaws.com/datasets/Inc/test.csv')\n",
    "predictor = TabularPredictor(label='class').fit(train_data=train_data, time_limit=5)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Modify predictor to enforce fairness over the train_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)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0        <=50K\n",
       "1        <=50K\n",
       "2        <=50K\n",
       "3        <=50K\n",
       "4        <=50K\n",
       "         ...  \n",
       "9764     <=50K\n",
       "9765     <=50K\n",
       "9766     <=50K\n",
       "9767     <=50K\n",
       "9768     <=50K\n",
       "Length: 9769, dtype: object"
      ]
     },
     "execution_count": 3,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# Evaluate on test data\n",
    "fpredictor.predict(test_data)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "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.773365</td>\n",
       "      <td>0.782270</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Balanced Accuracy</th>\n",
       "      <td>0.621548</td>\n",
       "      <td>0.594546</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>F1 score</th>\n",
       "      <td>0.410543</td>\n",
       "      <td>0.340874</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>MCC</th>\n",
       "      <td>0.291887</td>\n",
       "      <td>0.276910</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Precision</th>\n",
       "      <td>0.536161</td>\n",
       "      <td>0.605061</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Recall</th>\n",
       "      <td>0.332614</td>\n",
       "      <td>0.237274</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>ROC AUC</th>\n",
       "      <td>0.670449</td>\n",
       "      <td>0.595486</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "                   original   updated\n",
       "Accuracy           0.773365  0.782270\n",
       "Balanced Accuracy  0.621548  0.594546\n",
       "F1 score           0.410543  0.340874\n",
       "MCC                0.291887  0.276910\n",
       "Precision          0.536161  0.605061\n",
       "Recall             0.332614  0.237274\n",
       "ROC AUC            0.670449  0.595486"
      ]
     },
     "execution_count": 4,
     "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": 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>Statistical Parity</th>\n",
       "      <td>0.068820</td>\n",
       "      <td>0.012816</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Predictive Parity</th>\n",
       "      <td>0.253791</td>\n",
       "      <td>0.416944</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Equal Opportunity</th>\n",
       "      <td>0.011405</td>\n",
       "      <td>0.101063</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in False Negative Rate</th>\n",
       "      <td>0.011405</td>\n",
       "      <td>0.101063</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Equalized Odds</th>\n",
       "      <td>0.017738</td>\n",
       "      <td>0.072604</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Conditional Use Accuracy</th>\n",
       "      <td>0.208108</td>\n",
       "      <td>0.297498</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in Accuracy</th>\n",
       "      <td>0.131277</td>\n",
       "      <td>0.117851</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Treatment Equality</th>\n",
       "      <td>0.578759</td>\n",
       "      <td>0.829593</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "                                                 original   updated\n",
       "Statistical Parity                               0.068820  0.012816\n",
       "Predictive Parity                                0.253791  0.416944\n",
       "Equal Opportunity                                0.011405  0.101063\n",
       "Average Group Difference in False Negative Rate  0.011405  0.101063\n",
       "Equalized Odds                                   0.017738  0.072604\n",
       "Conditional Use Accuracy                         0.208108  0.297498\n",
       "Average Group Difference in Accuracy             0.131277  0.117851\n",
       "Treatment Equality                               0.578759  0.829593"
      ]
     },
     "execution_count": 5,
     "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": 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></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.773365</td>\n",
       "      <td>0.621548</td>\n",
       "      <td>0.410543</td>\n",
       "      <td>0.291887</td>\n",
       "      <td>0.536161</td>\n",
       "      <td>0.332614</td>\n",
       "      <td>0.670449</td>\n",
       "      <td>9769.0</td>\n",
       "      <td>2318.0</td>\n",
       "      <td>7451.0</td>\n",
       "      <td>0.237281</td>\n",
       "      <td>0.147200</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Female</th>\n",
       "      <td>0.860444</td>\n",
       "      <td>0.624007</td>\n",
       "      <td>0.331878</td>\n",
       "      <td>0.254150</td>\n",
       "      <td>0.341317</td>\n",
       "      <td>0.322946</td>\n",
       "      <td>0.651070</td>\n",
       "      <td>3289.0</td>\n",
       "      <td>353.0</td>\n",
       "      <td>2936.0</td>\n",
       "      <td>0.107327</td>\n",
       "      <td>0.101551</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Male</th>\n",
       "      <td>0.729167</td>\n",
       "      <td>0.617674</td>\n",
       "      <td>0.428152</td>\n",
       "      <td>0.287744</td>\n",
       "      <td>0.595109</td>\n",
       "      <td>0.334351</td>\n",
       "      <td>0.667435</td>\n",
       "      <td>6480.0</td>\n",
       "      <td>1965.0</td>\n",
       "      <td>4515.0</td>\n",
       "      <td>0.303241</td>\n",
       "      <td>0.170370</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Maximum difference</th>\n",
       "      <td>0.131277</td>\n",
       "      <td>0.006333</td>\n",
       "      <td>0.096275</td>\n",
       "      <td>0.033594</td>\n",
       "      <td>0.253791</td>\n",
       "      <td>0.011405</td>\n",
       "      <td>0.016364</td>\n",
       "      <td>3191.0</td>\n",
       "      <td>1612.0</td>\n",
       "      <td>1579.0</td>\n",
       "      <td>0.195913</td>\n",
       "      <td>0.068820</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th rowspan=\"4\" valign=\"top\">updated</th>\n",
       "      <th>Overall</th>\n",
       "      <td>0.782270</td>\n",
       "      <td>0.594546</td>\n",
       "      <td>0.340874</td>\n",
       "      <td>0.276910</td>\n",
       "      <td>0.605061</td>\n",
       "      <td>0.237274</td>\n",
       "      <td>0.595486</td>\n",
       "      <td>9769.0</td>\n",
       "      <td>2318.0</td>\n",
       "      <td>7451.0</td>\n",
       "      <td>0.237281</td>\n",
       "      <td>0.093049</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Female</th>\n",
       "      <td>0.860444</td>\n",
       "      <td>0.624007</td>\n",
       "      <td>0.331878</td>\n",
       "      <td>0.254150</td>\n",
       "      <td>0.341317</td>\n",
       "      <td>0.322946</td>\n",
       "      <td>0.651070</td>\n",
       "      <td>3289.0</td>\n",
       "      <td>353.0</td>\n",
       "      <td>2936.0</td>\n",
       "      <td>0.107327</td>\n",
       "      <td>0.101551</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Male</th>\n",
       "      <td>0.742593</td>\n",
       "      <td>0.595548</td>\n",
       "      <td>0.343307</td>\n",
       "      <td>0.308901</td>\n",
       "      <td>0.758261</td>\n",
       "      <td>0.221883</td>\n",
       "      <td>0.667435</td>\n",
       "      <td>6480.0</td>\n",
       "      <td>1965.0</td>\n",
       "      <td>4515.0</td>\n",
       "      <td>0.303241</td>\n",
       "      <td>0.088735</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Maximum difference</th>\n",
       "      <td>0.117851</td>\n",
       "      <td>0.028459</td>\n",
       "      <td>0.011429</td>\n",
       "      <td>0.054751</td>\n",
       "      <td>0.416944</td>\n",
       "      <td>0.101063</td>\n",
       "      <td>0.016364</td>\n",
       "      <td>3191.0</td>\n",
       "      <td>1612.0</td>\n",
       "      <td>1579.0</td>\n",
       "      <td>0.195913</td>\n",
       "      <td>0.012816</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "                             Accuracy  Balanced Accuracy  F1 score       MCC  \\\n",
       "         Groups                                                                \n",
       "original Overall             0.773365           0.621548  0.410543  0.291887   \n",
       "          Female             0.860444           0.624007  0.331878  0.254150   \n",
       "          Male               0.729167           0.617674  0.428152  0.287744   \n",
       "         Maximum difference  0.131277           0.006333  0.096275  0.033594   \n",
       "updated  Overall             0.782270           0.594546  0.340874  0.276910   \n",
       "          Female             0.860444           0.624007  0.331878  0.254150   \n",
       "          Male               0.742593           0.595548  0.343307  0.308901   \n",
       "         Maximum difference  0.117851           0.028459  0.011429  0.054751   \n",
       "\n",
       "                             Precision    Recall   ROC AUC  \\\n",
       "         Groups                                              \n",
       "original Overall              0.536161  0.332614  0.670449   \n",
       "          Female              0.341317  0.322946  0.651070   \n",
       "          Male                0.595109  0.334351  0.667435   \n",
       "         Maximum difference   0.253791  0.011405  0.016364   \n",
       "updated  Overall              0.605061  0.237274  0.595486   \n",
       "          Female              0.341317  0.322946  0.651070   \n",
       "          Male                0.758261  0.221883  0.667435   \n",
       "         Maximum difference   0.416944  0.101063  0.016364   \n",
       "\n",
       "                             Number of Datapoints  Positive Count  \\\n",
       "         Groups                                                     \n",
       "original Overall                           9769.0          2318.0   \n",
       "          Female                           3289.0           353.0   \n",
       "          Male                             6480.0          1965.0   \n",
       "         Maximum difference                3191.0          1612.0   \n",
       "updated  Overall                           9769.0          2318.0   \n",
       "          Female                           3289.0           353.0   \n",
       "          Male                             6480.0          1965.0   \n",
       "         Maximum difference                3191.0          1612.0   \n",
       "\n",
       "                             Negative Count  Positive Label Rate  \\\n",
       "         Groups                                                    \n",
       "original Overall                     7451.0             0.237281   \n",
       "          Female                     2936.0             0.107327   \n",
       "          Male                       4515.0             0.303241   \n",
       "         Maximum difference          1579.0             0.195913   \n",
       "updated  Overall                     7451.0             0.237281   \n",
       "          Female                     2936.0             0.107327   \n",
       "          Male                       4515.0             0.303241   \n",
       "         Maximum difference          1579.0             0.195913   \n",
       "\n",
       "                             Positive Prediction Rate  \n",
       "         Groups                                        \n",
       "original Overall                             0.147200  \n",
       "          Female                             0.101551  \n",
       "          Male                               0.170370  \n",
       "         Maximum difference                  0.068820  \n",
       "updated  Overall                             0.093049  \n",
       "          Female                             0.101551  \n",
       "          Male                               0.088735  \n",
       "         Maximum difference                  0.012816  "
      ]
     },
     "execution_count": 6,
     "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": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "#We can repeat the same analysis using the feature 'race' instead of 'sex'\n",
    "fpredictor = FairPredictor(predictor,train_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": 8,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>original</th>\n",
       "      <th>updated</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>Statistical Parity</th>\n",
       "      <td>0.043922</td>\n",
       "      <td>0.024707</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Predictive Parity</th>\n",
       "      <td>0.156746</td>\n",
       "      <td>0.155182</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Equal Opportunity</th>\n",
       "      <td>0.062128</td>\n",
       "      <td>0.038268</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in False Negative Rate</th>\n",
       "      <td>0.062128</td>\n",
       "      <td>0.038268</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Equalized Odds</th>\n",
       "      <td>0.041638</td>\n",
       "      <td>0.025259</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Conditional Use Accuracy</th>\n",
       "      <td>0.112009</td>\n",
       "      <td>0.116499</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in Accuracy</th>\n",
       "      <td>0.063525</td>\n",
       "      <td>0.076979</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Treatment Equality</th>\n",
       "      <td>0.228601</td>\n",
       "      <td>0.065512</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "                                                 original   updated\n",
       "Statistical Parity                               0.043922  0.024707\n",
       "Predictive Parity                                0.156746  0.155182\n",
       "Equal Opportunity                                0.062128  0.038268\n",
       "Average Group Difference in False Negative Rate  0.062128  0.038268\n",
       "Equalized Odds                                   0.041638  0.025259\n",
       "Conditional Use Accuracy                         0.112009  0.116499\n",
       "Average Group Difference in Accuracy             0.063525  0.076979\n",
       "Treatment Equality                               0.228601  0.065512"
      ]
     },
     "execution_count": 8,
     "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, verbose=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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MTEx8PLyQtu2bctsKzQ0FFlZWaotLS2tXH0nIiKimsnYUCeuV68e5HK5xmxPRkaGxqxQcUIIrFixAoMHD4aJiUmJdXJycrBhwwZMmzbtuX0xNTWFqalp+TtPRERENZrBZoBMTEzg7e2NhIQEtfKEhAT4+fmVeezBgwdx9epVhISElFrnp59+Qm5uLgYNGqST/hIREdGLw2AzQAAwbtw4DB48GD4+PvD19cXy5cuRmpqKkSNHAnh6aer27dtYvXq12nExMTFo164dvLy8Sm07JiYGgYGBqFu3rl7HQERERDWPQQNQUFAQMjMzMW3aNCiVSnh5eSEuLk51V5dSqdR4JlBWVhZ+/vlnLFy4sNR2L1++jMOHD2PPnj167T8RERHVTAZ9DlB1xecAERER1Tw14jlARERERIbCAERERESSwwBEREREksMARERERJLDAERERESSwwBEREREksMARERERJLDAERERESSwwBEREREksMARERERJLDAERERESSwwBEREREksMARERERJLDAERERESSwwBEREREksMARERERJLDAERERESSwwBEREREksMARERERJLDAERERESSwwBEREREksMARERERJLDAERERESSwwBEREREksMARERERJLDAERERESSwwBEREREksMARERERJLDAERERESSwwBEREREksMARERERJLDAERERESSY2zoDkhJQaHAiRv3kHH/MewsFWjrZgu5kczQ3SIiIpIcBqAqEn9Biak7kqHMeqwqc7RWILyvJ3p6ORqwZ0RERNJj8EtgS5YsgZubGxQKBby9vXHo0KFS6wYHB0Mmk2lsr7zyilq9f//9F59++ikcHR2hUCjg4eGBuLg4fQ+lVPEXlPh47Wm18AMA6VmP8fHa04i/oDRQz4iIiKTJoAFo48aNGDt2LMLCwnDmzBn4+/ujV69eSE1NLbH+woULoVQqVVtaWhpsbW3xzjvvqOrk5eWhe/fuuHnzJjZv3oxLly4hOjoaDRo0qKphqSkoFJi6IxmihH1FZVN3JKOgsKQaREREpA8GvQQ2f/58hISEYPjw4QCABQsWYPfu3YiKikJkZKRGfWtra1hbW6teb9u2Df/88w+GDRumKluxYgXu3buHo0ePolatWgAAFxcXPY+kdCdu3NOY+XmWAKDMeowTN+7B96W6VdcxIiIiCTPYDFBeXh4S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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 training set\n",
    "fpredictor.plot_frontier() "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAkkAAAHFCAYAAADmGm0KAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjguNCwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8fJSN1AAAACXBIWXMAAA9hAAAPYQGoP6dpAABfSElEQVR4nO3deVxUVeM/8M8wCMOuoGyKgLkirqCIWy6Ie6ItamlaWGlaLvmUhgauZItplpgE4pJL5ZamKGqauYtoKoSAGqhDBBq4sZ/fH/6Yr+NckIEZBvDzfr3m9Txz7rn3njMg8+ncc8+VCSEEiIiIiEiNkaEbQERERFQdMSQRERERSWBIIiIiIpLAkEREREQkgSGJiIiISAJDEhEREZEEhiQiIiIiCQxJRERERBIYkoiIiIgkMCQRUblFRUVBJpNJvmbOnFmlbXnw4AFCQkJw+PDhUtt5/fp1vbfj8OHDpX4mL730kt7PX1FV+RkR1VTGhm4AEdU8a9asQcuWLdXKnJ2dq7QNDx48wLx58wAAvXr1Uts2ePBgnDhxAk5OTlXWnsWLF6N3795qZXZ2dlV2fiLSPYYkItKap6cnvL29y1W3oKAAMpkMxsZV9+emQYMGaNCggc6O9+DBA5ibm5dZp1mzZujSpYvOzklEhsfLbUSkMyWXntavX48PPvgADRs2hKmpKZKTkwEAkZGRaNeuHRQKBWxtbTF8+HAkJCSoHWP8+PGwtLREcnIyBg0aBEtLS7i4uOCDDz5AXl4eAOD69euqEDRv3jzV5a3x48cDKP1S0oEDB9C3b19YW1vD3Nwc3bp1w8GDB9XqhISEQCaT4dy5c3jppZdQr149PPfcc5X+bP744w/07dsXVlZWMDc3R9euXfHrr79KnvtJUv1xc3PDkCFDEB0djY4dO8LMzAwtW7ZEZGSkxv4nT55Et27doFAo4OzsjNmzZ6OgoKDSfSKq7RiSiEhrRUVFKCwsVHs9bvbs2UhNTcWqVauwa9cu2NvbIzQ0FIGBgWjdujW2bduG5cuX488//4Svry+SkpLU9i8oKMALL7yAvn37YufOnXjzzTfx1VdfYcmSJQAAJycnREdHAwACAwNx4sQJnDhxAnPnzi21zRs2bIC/vz+sra2xdu1a/Pjjj7C1tUX//v01ghIAjBgxAk2bNsVPP/2EVatWPfUzKS4uLvUzOXLkCPr06YPs7GxERERg06ZNsLKywtChQ7Fly5anHrs0Fy5cwAcffIDp06dj586daNu2LQIDA/H777+r6sTHx6Nv377477//EBUVhVWrViEuLg4LFy6s8HmJnhmCiKic1qxZIwBIvgoKCsRvv/0mAIiePXuq7Xfnzh1hZmYmBg0apFaempoqTE1NxauvvqoqGzdunAAgfvzxR7W6gwYNEi1atFC9//fffwUAERwcXGo7r127JoQQ4v79+8LW1lYMHTpUrV5RUZFo166d6Ny5s6osODhYABCffPJJuT6Tkj5LvZKSkoQQQnTp0kXY29uLu3fvqvYrLCwUnp6eolGjRqK4uFjt3E/rjxBCuLq6CoVCIf7++29V2cOHD4Wtra145513VGUjR44UZmZmIj09Xe3cLVu21DgmEanjSBIRaW3dunU4c+aM2uvxOUcvvviiWv0TJ07g4cOHqsthJVxcXNCnTx+NkRyZTIahQ4eqlbVt2xZ///13hdp7/Phx3L59G+PGjVMb6SkuLsaAAQNw5swZ3L9/X22fJ/vwNEuWLNH4TFxcXHD//n2cOnUKL730EiwtLVX15XI5xo4dixs3biAxMbFC/Wrfvj0aN26seq9QKNC8eXO1z+m3335D37594eDgoHbukSNHVuicRM8STtwmIq21atWqzInbT95VlpWVJVkOPLorLiYmRq3M3NwcCoVCrczU1BS5ubkVau8///wDAGXekn/79m1YWFio3mt7Z1yTJk0kP5N///0XQohS+w783+ejLam750xNTfHw4UPV+6ysLDg6OmrUkyojInUMSUSkc09OPi75MlcqlRp1b926hfr16+u1PSXHX7FiRal3oD0+0gJo9qGi6tWrByMjo1L7/nj7SoJhXl4eTE1NVfUyMzMrfH47Ozukp6drlEuVEZE6Xm4jIr3z9fWFmZkZNmzYoFZ+48YNHDp0CH379tX6mCUh4vFRk9J069YNdevWRXx8PLy9vSVfJiYmWrehPCwsLODj44Nt27aptbW4uBgbNmxAo0aN0Lx5cwCP7lgDgD///FPtGLt27arw+Xv37o2DBw+qRtOARxPvKzNhnOhZwZEkItK7unXrYu7cufj444/x+uuvY/To0cjKysK8efOgUCgQHBys9TGtrKzg6uqKnTt3om/fvrC1tUX9+vVVQeNxlpaWWLFiBcaNG4fbt2/jpZdegr29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NxcPHjxAQkICGjdujEaNGpV6niedP38ePXr00Gq+U0JCAtq1awcLi/9bk61bt24oLi5GYmKias3ENm3aVOk8pMcxJBERkW45tQXeOQL8+DqQlQyI//9YKFH0KDTVbw68sg6wb6XT0zZt2hQymQzx8fGSd1L99ddfqFevHurXr68qe/wLWoqQeApESdgqy5NhQSaTqYIQAAwdOhQuLi4IDw+Hs7MziouL4enpifz8/Kce+2mebO+TfSwuLsa8efMwYsQIjX0VCoVk/572JIyKTFyX+mylzve0n5E+ceI2ERHpnn0rIHC/9LbA/ToPSABgZ2eHfv36YeXKlXj48KHatvT0dPzwww8YOXJkuR59VaJly5Y4c+aMWtnZs2cr1c6srCwkJCRgzpw56Nu3L1q1aoU7d+5U6FgnT57UeN+yZcsy9+nYsSMSExPRtGlTjZeRkRE8PDyQmpqKW7duqfY5ceJEmcds27Ytjh49ioKCAsntJiYmas9QBQAPDw+cP38e9+/fV5UdO3YMRkZGVT9BuxQMSUREpB9/H390ae1xohj4u+wv3Mr45ptvkJeXh/79++P3339HWloaoqOj0a9fPzRs2FDtElN5vPPOO/jrr7/w0Ucf4cqVK/jxxx9Vy9NoE7YeV69ePdjZ2WH16tVITk7GoUOHMGPGjAod69ixY/jss89w5coVfPvtt/jpp58wderUMvf55JNPsG7dOoSEhODy5ctISEjAli1bMGfOHACAn58fWrRogddffx0XLlzA0aNH1e4KlDJlyhTk5ORg1KhROHv2LJKSkrB+/XokJiYCeHT58c8//0RiYiIyMzNRUFCA1157DQqFAuPGjcOlS5fw22+/4b333sPYsWOf+niyqsKQRERE+pHwy6P/bewLBB4AXLqol+tBs2bNcPbsWTz33HMYOXIknnvuObz99tvo3bs3Tpw4UeoaSaVxd3fHzz//jG3btqFt27YICwtTBQZTU9MKtdHIyAibN29GbGwsPD09MX36dHz++ecVOtYHH3yA2NhYdOjQAQsWLMCXX36J/v37l7lP//79sXv3bsTExKBTp07o0qULli5dqlpQ2cjICNu3b0deXh46d+6MCRMmPDVc2tnZ4dChQ7h37x6ef/55eHl5ITw8XHXZ8a233kKLFi3g7e2NBg0a4NixYzA3N8e+fftw+/ZtdOrUCS+99BL6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",
      "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": 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></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.773365</td>\n",
       "      <td>0.621548</td>\n",
       "      <td>0.410543</td>\n",
       "      <td>0.291887</td>\n",
       "      <td>0.536161</td>\n",
       "      <td>0.332614</td>\n",
       "      <td>0.670449</td>\n",
       "      <td>9769.0</td>\n",
       "      <td>2318.0</td>\n",
       "      <td>7451.0</td>\n",
       "      <td>0.237281</td>\n",
       "      <td>0.147200</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Amer-Indian-Eskimo</th>\n",
       "      <td>0.886179</td>\n",
       "      <td>0.568588</td>\n",
       "      <td>0.222222</td>\n",
       "      <td>0.168968</td>\n",
       "      <td>0.285714</td>\n",
       "      <td>0.181818</td>\n",
       "      <td>0.709010</td>\n",
       "      <td>123.0</td>\n",
       "      <td>11.0</td>\n",
       "      <td>112.0</td>\n",
       "      <td>0.089431</td>\n",
       "      <td>0.056911</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Asian-Pac-Islander</th>\n",
       "      <td>0.786585</td>\n",
       "      <td>0.617478</td>\n",
       "      <td>0.396552</td>\n",
       "      <td>0.307672</td>\n",
       "      <td>0.589744</td>\n",
       "      <td>0.298701</td>\n",
       "      <td>0.685595</td>\n",
       "      <td>328.0</td>\n",
       "      <td>77.0</td>\n",
       "      <td>251.0</td>\n",
       "      <td>0.234756</td>\n",
       "      <td>0.118902</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Black</th>\n",
       "      <td>0.855219</td>\n",
       "      <td>0.615094</td>\n",
       "      <td>0.331606</td>\n",
       "      <td>0.253388</td>\n",
       "      <td>0.372093</td>\n",
       "      <td>0.299065</td>\n",
       "      <td>0.638655</td>\n",
       "      <td>891.0</td>\n",
       "      <td>107.0</td>\n",
       "      <td>784.0</td>\n",
       "      <td>0.120090</td>\n",
       "      <td>0.096521</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Other</th>\n",
       "      <td>0.851852</td>\n",
       "      <td>0.614789</td>\n",
       "      <td>0.333333</td>\n",
       "      <td>0.253135</td>\n",
       "      <td>0.375000</td>\n",
       "      <td>0.300000</td>\n",
       "      <td>0.478169</td>\n",
       "      <td>81.0</td>\n",
       "      <td>10.0</td>\n",
       "      <td>71.0</td>\n",
       "      <td>0.123457</td>\n",
       "      <td>0.098765</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>White</th>\n",
       "      <td>0.761682</td>\n",
       "      <td>0.621156</td>\n",
       "      <td>0.416887</td>\n",
       "      <td>0.290739</td>\n",
       "      <td>0.547766</td>\n",
       "      <td>0.336488</td>\n",
       "      <td>0.670081</td>\n",
       "      <td>8346.0</td>\n",
       "      <td>2113.0</td>\n",
       "      <td>6233.0</td>\n",
       "      <td>0.253175</td>\n",
       "      <td>0.155524</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Maximum difference</th>\n",
       "      <td>0.124497</td>\n",
       "      <td>0.052568</td>\n",
       "      <td>0.194664</td>\n",
       "      <td>0.138704</td>\n",
       "      <td>0.304029</td>\n",
       "      <td>0.154670</td>\n",
       "      <td>0.230841</td>\n",
       "      <td>8265.0</td>\n",
       "      <td>2103.0</td>\n",
       "      <td>6162.0</td>\n",
       "      <td>0.163744</td>\n",
       "      <td>0.098613</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th rowspan=\"7\" valign=\"top\">updated</th>\n",
       "      <th>Overall</th>\n",
       "      <td>0.795168</td>\n",
       "      <td>0.596612</td>\n",
       "      <td>0.336318</td>\n",
       "      <td>0.319341</td>\n",
       "      <td>0.727403</td>\n",
       "      <td>0.218723</td>\n",
       "      <td>0.670275</td>\n",
       "      <td>9769.0</td>\n",
       "      <td>2318.0</td>\n",
       "      <td>7451.0</td>\n",
       "      <td>0.237281</td>\n",
       "      <td>0.071348</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Amer-Indian-Eskimo</th>\n",
       "      <td>0.918699</td>\n",
       "      <td>0.586445</td>\n",
       "      <td>0.285714</td>\n",
       "      <td>0.319833</td>\n",
       "      <td>0.666667</td>\n",
       "      <td>0.181818</td>\n",
       "      <td>0.709010</td>\n",
       "      <td>123.0</td>\n",
       "      <td>11.0</td>\n",
       "      <td>112.0</td>\n",
       "      <td>0.089431</td>\n",
       "      <td>0.024390</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Asian-Pac-Islander</th>\n",
       "      <td>0.789634</td>\n",
       "      <td>0.565452</td>\n",
       "      <td>0.241758</td>\n",
       "      <td>0.274479</td>\n",
       "      <td>0.785714</td>\n",
       "      <td>0.142857</td>\n",
       "      <td>0.685595</td>\n",
       "      <td>328.0</td>\n",
       "      <td>77.0</td>\n",
       "      <td>251.0</td>\n",
       "      <td>0.234756</td>\n",
       "      <td>0.042683</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Black</th>\n",
       "      <td>0.893378</td>\n",
       "      <td>0.600461</td>\n",
       "      <td>0.326241</td>\n",
       "      <td>0.340917</td>\n",
       "      <td>0.676471</td>\n",
       "      <td>0.214953</td>\n",
       "      <td>0.638655</td>\n",
       "      <td>891.0</td>\n",
       "      <td>107.0</td>\n",
       "      <td>784.0</td>\n",
       "      <td>0.120090</td>\n",
       "      <td>0.038159</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Other</th>\n",
       "      <td>0.901235</td>\n",
       "      <td>0.600000</td>\n",
       "      <td>0.333333</td>\n",
       "      <td>0.423966</td>\n",
       "      <td>1.000000</td>\n",
       "      <td>0.200000</td>\n",
       "      <td>0.478169</td>\n",
       "      <td>81.0</td>\n",
       "      <td>10.0</td>\n",
       "      <td>71.0</td>\n",
       "      <td>0.123457</td>\n",
       "      <td>0.024691</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>White</th>\n",
       "      <td>0.782051</td>\n",
       "      <td>0.596941</td>\n",
       "      <td>0.340225</td>\n",
       "      <td>0.315932</td>\n",
       "      <td>0.728261</td>\n",
       "      <td>0.221959</td>\n",
       "      <td>0.670081</td>\n",
       "      <td>8346.0</td>\n",
       "      <td>2113.0</td>\n",
       "      <td>6233.0</td>\n",
       "      <td>0.253175</td>\n",
       "      <td>0.077163</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Maximum difference</th>\n",
       "      <td>0.136648</td>\n",
       "      <td>0.035009</td>\n",
       "      <td>0.098467</td>\n",
       "      <td>0.149486</td>\n",
       "      <td>0.333333</td>\n",
       "      <td>0.079102</td>\n",
       "      <td>0.230841</td>\n",
       "      <td>8265.0</td>\n",
       "      <td>2103.0</td>\n",
       "      <td>6162.0</td>\n",
       "      <td>0.163744</td>\n",
       "      <td>0.052772</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "                              Accuracy  Balanced Accuracy  F1 score       MCC  \\\n",
       "         Groups                                                                 \n",
       "original Overall              0.773365           0.621548  0.410543  0.291887   \n",
       "          Amer-Indian-Eskimo  0.886179           0.568588  0.222222  0.168968   \n",
       "          Asian-Pac-Islander  0.786585           0.617478  0.396552  0.307672   \n",
       "          Black               0.855219           0.615094  0.331606  0.253388   \n",
       "          Other               0.851852           0.614789  0.333333  0.253135   \n",
       "          White               0.761682           0.621156  0.416887  0.290739   \n",
       "         Maximum difference   0.124497           0.052568  0.194664  0.138704   \n",
       "updated  Overall              0.795168           0.596612  0.336318  0.319341   \n",
       "          Amer-Indian-Eskimo  0.918699           0.586445  0.285714  0.319833   \n",
       "          Asian-Pac-Islander  0.789634           0.565452  0.241758  0.274479   \n",
       "          Black               0.893378           0.600461  0.326241  0.340917   \n",
       "          Other               0.901235           0.600000  0.333333  0.423966   \n",
       "          White               0.782051           0.596941  0.340225  0.315932   \n",
       "         Maximum difference   0.136648           0.035009  0.098467  0.149486   \n",
       "\n",
       "                              Precision    Recall   ROC AUC  \\\n",
       "         Groups                                               \n",
       "original Overall               0.536161  0.332614  0.670449   \n",
       "          Amer-Indian-Eskimo   0.285714  0.181818  0.709010   \n",
       "          Asian-Pac-Islander   0.589744  0.298701  0.685595   \n",
       "          Black                0.372093  0.299065  0.638655   \n",
       "          Other                0.375000  0.300000  0.478169   \n",
       "          White                0.547766  0.336488  0.670081   \n",
       "         Maximum difference    0.304029  0.154670  0.230841   \n",
       "updated  Overall               0.727403  0.218723  0.670275   \n",
       "          Amer-Indian-Eskimo   0.666667  0.181818  0.709010   \n",
       "          Asian-Pac-Islander   0.785714  0.142857  0.685595   \n",
       "          Black                0.676471  0.214953  0.638655   \n",
       "          Other                1.000000  0.200000  0.478169   \n",
       "          White                0.728261  0.221959  0.670081   \n",
       "         Maximum difference    0.333333  0.079102  0.230841   \n",
       "\n",
       "                              Number of Datapoints  Positive Count  \\\n",
       "         Groups                                                      \n",
       "original Overall                            9769.0          2318.0   \n",
       "          Amer-Indian-Eskimo                 123.0            11.0   \n",
       "          Asian-Pac-Islander                 328.0            77.0   \n",
       "          Black                              891.0           107.0   \n",
       "          Other                               81.0            10.0   \n",
       "          White                             8346.0          2113.0   \n",
       "         Maximum difference                 8265.0          2103.0   \n",
       "updated  Overall                            9769.0          2318.0   \n",
       "          Amer-Indian-Eskimo                 123.0            11.0   \n",
       "          Asian-Pac-Islander                 328.0            77.0   \n",
       "          Black                              891.0           107.0   \n",
       "          Other                               81.0            10.0   \n",
       "          White                             8346.0          2113.0   \n",
       "         Maximum difference                 8265.0          2103.0   \n",
       "\n",
       "                              Negative Count  Positive Label Rate  \\\n",
       "         Groups                                                     \n",
       "original Overall                      7451.0             0.237281   \n",
       "          Amer-Indian-Eskimo           112.0             0.089431   \n",
       "          Asian-Pac-Islander           251.0             0.234756   \n",
       "          Black                        784.0             0.120090   \n",
       "          Other                         71.0             0.123457   \n",
       "          White                       6233.0             0.253175   \n",
       "         Maximum difference           6162.0             0.163744   \n",
       "updated  Overall                      7451.0             0.237281   \n",
       "          Amer-Indian-Eskimo           112.0             0.089431   \n",
       "          Asian-Pac-Islander           251.0             0.234756   \n",
       "          Black                        784.0             0.120090   \n",
       "          Other                         71.0             0.123457   \n",
       "          White                       6233.0             0.253175   \n",
       "         Maximum difference           6162.0             0.163744   \n",
       "\n",
       "                              Positive Prediction Rate  \n",
       "         Groups                                         \n",
       "original Overall                              0.147200  \n",
       "          Amer-Indian-Eskimo                  0.056911  \n",
       "          Asian-Pac-Islander                  0.118902  \n",
       "          Black                               0.096521  \n",
       "          Other                               0.098765  \n",
       "          White                               0.155524  \n",
       "         Maximum difference                   0.098613  \n",
       "updated  Overall                              0.071348  \n",
       "          Amer-Indian-Eskimo                  0.024390  \n",
       "          Asian-Pac-Islander                  0.042683  \n",
       "          Black                               0.038159  \n",
       "          Other                               0.024691  \n",
       "          White                               0.077163  \n",
       "         Maximum difference                   0.052772  "
      ]
     },
     "execution_count": 11,
     "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, verbose=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "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.833184</td>\n",
       "      <td>0.706551</td>\n",
       "      <td>0.571128</td>\n",
       "      <td>0.495060</td>\n",
       "      <td>0.744553</td>\n",
       "      <td>0.463230</td>\n",
       "      <td>0.860978</td>\n",
       "      <td>39073.0</td>\n",
       "      <td>9369.0</td>\n",
       "      <td>29704.0</td>\n",
       "      <td>0.239782</td>\n",
       "      <td>0.149182</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Amer-Indian-Eskimo</th>\n",
       "      <td>0.893372</td>\n",
       "      <td>0.686394</td>\n",
       "      <td>0.493151</td>\n",
       "      <td>0.448225</td>\n",
       "      <td>0.620690</td>\n",
       "      <td>0.409091</td>\n",
       "      <td>0.830221</td>\n",
       "      <td>347.0</td>\n",
       "      <td>44.0</td>\n",
       "      <td>303.0</td>\n",
       "      <td>0.126801</td>\n",
       "      <td>0.083573</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Asian-Pac-Islander</th>\n",
       "      <td>0.815281</td>\n",
       "      <td>0.708405</td>\n",
       "      <td>0.584906</td>\n",
       "      <td>0.501991</td>\n",
       "      <td>0.782828</td>\n",
       "      <td>0.466867</td>\n",
       "      <td>0.867011</td>\n",
       "      <td>1191.0</td>\n",
       "      <td>332.0</td>\n",
       "      <td>859.0</td>\n",
       "      <td>0.278757</td>\n",
       "      <td>0.166247</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Black</th>\n",
       "      <td>0.889299</td>\n",
       "      <td>0.681515</td>\n",
       "      <td>0.471033</td>\n",
       "      <td>0.417252</td>\n",
       "      <td>0.558209</td>\n",
       "      <td>0.407407</td>\n",
       "      <td>0.855751</td>\n",
       "      <td>3794.0</td>\n",
       "      <td>459.0</td>\n",
       "      <td>3335.0</td>\n",
       "      <td>0.120980</td>\n",
       "      <td>0.088297</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Other</th>\n",
       "      <td>0.892308</td>\n",
       "      <td>0.691447</td>\n",
       "      <td>0.492754</td>\n",
       "      <td>0.441252</td>\n",
       "      <td>0.586207</td>\n",
       "      <td>0.425000</td>\n",
       "      <td>0.851404</td>\n",
       "      <td>325.0</td>\n",
       "      <td>40.0</td>\n",
       "      <td>285.0</td>\n",
       "      <td>0.123077</td>\n",
       "      <td>0.089231</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>White</th>\n",
       "      <td>0.826251</td>\n",
       "      <td>0.707703</td>\n",
       "      <td>0.577192</td>\n",
       "      <td>0.497487</td>\n",
       "      <td>0.756586</td>\n",
       "      <td>0.466565</td>\n",
       "      <td>0.860305</td>\n",
       "      <td>33416.0</td>\n",
       "      <td>8494.0</td>\n",
       "      <td>24922.0</td>\n",
       "      <td>0.254190</td>\n",
       "      <td>0.156751</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Maximum difference</th>\n",
       "      <td>0.078090</td>\n",
       "      <td>0.026890</td>\n",
       "      <td>0.113873</td>\n",
       "      <td>0.084739</td>\n",
       "      <td>0.224619</td>\n",
       "      <td>0.059460</td>\n",
       "      <td>0.036790</td>\n",
       "      <td>33091.0</td>\n",
       "      <td>8454.0</td>\n",
       "      <td>24637.0</td>\n",
       "      <td>0.157777</td>\n",
       "      <td>0.082673</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th rowspan=\"7\" valign=\"top\">updated</th>\n",
       "      <th>Overall</th>\n",
       "      <td>0.819671</td>\n",
       "      <td>0.632851</td>\n",
       "      <td>0.421416</td>\n",
       "      <td>0.439171</td>\n",
       "      <td>0.913492</td>\n",
       "      <td>0.273882</td>\n",
       "      <td>0.860331</td>\n",
       "      <td>39073.0</td>\n",
       "      <td>9369.0</td>\n",
       "      <td>29704.0</td>\n",
       "      <td>0.239782</td>\n",
       "      <td>0.071891</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Amer-Indian-Eskimo</th>\n",
       "      <td>0.904899</td>\n",
       "      <td>0.644427</td>\n",
       "      <td>0.440678</td>\n",
       "      <td>0.472619</td>\n",
       "      <td>0.866667</td>\n",
       "      <td>0.295455</td>\n",
       "      <td>0.830221</td>\n",
       "      <td>347.0</td>\n",
       "      <td>44.0</td>\n",
       "      <td>303.0</td>\n",
       "      <td>0.126801</td>\n",
       "      <td>0.043228</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Asian-Pac-Islander</th>\n",
       "      <td>0.769941</td>\n",
       "      <td>0.587349</td>\n",
       "      <td>0.297436</td>\n",
       "      <td>0.363937</td>\n",
       "      <td>1.000000</td>\n",
       "      <td>0.174699</td>\n",
       "      <td>0.867011</td>\n",
       "      <td>1191.0</td>\n",
       "      <td>332.0</td>\n",
       "      <td>859.0</td>\n",
       "      <td>0.278757</td>\n",
       "      <td>0.048699</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Black</th>\n",
       "      <td>0.904586</td>\n",
       "      <td>0.625392</td>\n",
       "      <td>0.394649</td>\n",
       "      <td>0.435314</td>\n",
       "      <td>0.848921</td>\n",
       "      <td>0.257081</td>\n",
       "      <td>0.855751</td>\n",
       "      <td>3794.0</td>\n",
       "      <td>459.0</td>\n",
       "      <td>3335.0</td>\n",
       "      <td>0.120980</td>\n",
       "      <td>0.036637</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Other</th>\n",
       "      <td>0.904615</td>\n",
       "      <td>0.644737</td>\n",
       "      <td>0.436364</td>\n",
       "      <td>0.453247</td>\n",
       "      <td>0.800000</td>\n",
       "      <td>0.300000</td>\n",
       "      <td>0.851404</td>\n",
       "      <td>325.0</td>\n",
       "      <td>40.0</td>\n",
       "      <td>285.0</td>\n",
       "      <td>0.123077</td>\n",
       "      <td>0.046154</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>White</th>\n",
       "      <td>0.810091</td>\n",
       "      <td>0.634862</td>\n",
       "      <td>0.427049</td>\n",
       "      <td>0.439820</td>\n",
       "      <td>0.915957</td>\n",
       "      <td>0.278432</td>\n",
       "      <td>0.860305</td>\n",
       "      <td>33416.0</td>\n",
       "      <td>8494.0</td>\n",
       "      <td>24922.0</td>\n",
       "      <td>0.254190</td>\n",
       "      <td>0.077268</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Maximum difference</th>\n",
       "      <td>0.134958</td>\n",
       "      <td>0.057387</td>\n",
       "      <td>0.143242</td>\n",
       "      <td>0.108682</td>\n",
       "      <td>0.200000</td>\n",
       "      <td>0.125301</td>\n",
       "      <td>0.036790</td>\n",
       "      <td>33091.0</td>\n",
       "      <td>8454.0</td>\n",
       "      <td>24637.0</td>\n",
       "      <td>0.157777</td>\n",
       "      <td>0.040632</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "                              Accuracy  Balanced Accuracy  F1 score       MCC  \\\n",
       "         Groups                                                                 \n",
       "original Overall              0.833184           0.706551  0.571128  0.495060   \n",
       "          Amer-Indian-Eskimo  0.893372           0.686394  0.493151  0.448225   \n",
       "          Asian-Pac-Islander  0.815281           0.708405  0.584906  0.501991   \n",
       "          Black               0.889299           0.681515  0.471033  0.417252   \n",
       "          Other               0.892308           0.691447  0.492754  0.441252   \n",
       "          White               0.826251           0.707703  0.577192  0.497487   \n",
       "         Maximum difference   0.078090           0.026890  0.113873  0.084739   \n",
       "updated  Overall              0.819671           0.632851  0.421416  0.439171   \n",
       "          Amer-Indian-Eskimo  0.904899           0.644427  0.440678  0.472619   \n",
       "          Asian-Pac-Islander  0.769941           0.587349  0.297436  0.363937   \n",
       "          Black               0.904586           0.625392  0.394649  0.435314   \n",
       "          Other               0.904615           0.644737  0.436364  0.453247   \n",
       "          White               0.810091           0.634862  0.427049  0.439820   \n",
       "         Maximum difference   0.134958           0.057387  0.143242  0.108682   \n",
       "\n",
       "                              Precision    Recall   ROC AUC  \\\n",
       "         Groups                                               \n",
       "original Overall               0.744553  0.463230  0.860978   \n",
       "          Amer-Indian-Eskimo   0.620690  0.409091  0.830221   \n",
       "          Asian-Pac-Islander   0.782828  0.466867  0.867011   \n",
       "          Black                0.558209  0.407407  0.855751   \n",
       "          Other                0.586207  0.425000  0.851404   \n",
       "          White                0.756586  0.466565  0.860305   \n",
       "         Maximum difference    0.224619  0.059460  0.036790   \n",
       "updated  Overall               0.913492  0.273882  0.860331   \n",
       "          Amer-Indian-Eskimo   0.866667  0.295455  0.830221   \n",
       "          Asian-Pac-Islander   1.000000  0.174699  0.867011   \n",
       "          Black                0.848921  0.257081  0.855751   \n",
       "          Other                0.800000  0.300000  0.851404   \n",
       "          White                0.915957  0.278432  0.860305   \n",
       "         Maximum difference    0.200000  0.125301  0.036790   \n",
       "\n",
       "                              Number of Datapoints  Positive Count  \\\n",
       "         Groups                                                      \n",
       "original Overall                           39073.0          9369.0   \n",
       "          Amer-Indian-Eskimo                 347.0            44.0   \n",
       "          Asian-Pac-Islander                1191.0           332.0   \n",
       "          Black                             3794.0           459.0   \n",
       "          Other                              325.0            40.0   \n",
       "          White                            33416.0          8494.0   \n",
       "         Maximum difference                33091.0          8454.0   \n",
       "updated  Overall                           39073.0          9369.0   \n",
       "          Amer-Indian-Eskimo                 347.0            44.0   \n",
       "          Asian-Pac-Islander                1191.0           332.0   \n",
       "          Black                             3794.0           459.0   \n",
       "          Other                              325.0            40.0   \n",
       "          White                            33416.0          8494.0   \n",
       "         Maximum difference                33091.0          8454.0   \n",
       "\n",
       "                              Negative Count  Positive Label Rate  \\\n",
       "         Groups                                                     \n",
       "original Overall                     29704.0             0.239782   \n",
       "          Amer-Indian-Eskimo           303.0             0.126801   \n",
       "          Asian-Pac-Islander           859.0             0.278757   \n",
       "          Black                       3335.0             0.120980   \n",
       "          Other                        285.0             0.123077   \n",
       "          White                      24922.0             0.254190   \n",
       "         Maximum difference          24637.0             0.157777   \n",
       "updated  Overall                     29704.0             0.239782   \n",
       "          Amer-Indian-Eskimo           303.0             0.126801   \n",
       "          Asian-Pac-Islander           859.0             0.278757   \n",
       "          Black                       3335.0             0.120980   \n",
       "          Other                        285.0             0.123077   \n",
       "          White                      24922.0             0.254190   \n",
       "         Maximum difference          24637.0             0.157777   \n",
       "\n",
       "                              Positive Prediction Rate  \n",
       "         Groups                                         \n",
       "original Overall                              0.149182  \n",
       "          Amer-Indian-Eskimo                  0.083573  \n",
       "          Asian-Pac-Islander                  0.166247  \n",
       "          Black                               0.088297  \n",
       "          Other                               0.089231  \n",
       "          White                               0.156751  \n",
       "         Maximum difference                   0.082673  \n",
       "updated  Overall                              0.071891  \n",
       "          Amer-Indian-Eskimo                  0.043228  \n",
       "          Asian-Pac-Islander                  0.048699  \n",
       "          Black                               0.036637  \n",
       "          Other                               0.046154  \n",
       "          White                               0.077268  \n",
       "         Maximum difference                   0.040632  "
      ]
     },
     "execution_count": 12,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "#The groups with the smallest Positive Prediction Rate (American-Indian-Eskimo, and Asian-Pacific-Islander) \n",
    "# only have around 100-300 samples making it impossible to accurately evaluate demographic parity, while the\n",
    "# group labelled 'Other' has even less data.\n",
    "# Moreover, on training data we find that there are less than 350 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(verbose=True)"
   ]
  },
  {
   "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": 13,
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import pandas as pd\n",
    "from anonfair import inferred_attribute_builder\n",
    "all_data = pd.read_csv('https://github.com/propublica/compas-analysis/raw/master/compas-scores-two-years.csv')\n",
    "condensed_data=all_data[['sex','race','age', 'juv_fel_count', 'juv_misd_count', 'juv_other_count', 'priors_count', 'age_cat', 'c_charge_degree','two_year_recid']].copy()\n",
    "condensed_data.replace({'Caucasian':'Other', 'Hispanic':'Other', 'Native American':'Other', 'Asian':'Other'},inplace=True)\n",
    "train=condensed_data.sample(frac=0.3, random_state=0)\n",
    "val_and_test=condensed_data.drop(train.index)\n",
    "val=val_and_test.sample(frac=0.5, random_state=0)\n",
    "test=val_and_test.drop(val.index)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "No path specified. Models will be saved in: \"AutogluonModels/ag-20240522_162316\"\n",
      "No presets specified! To achieve strong results with AutoGluon, it is recommended to use the available presets.\n",
      "\tRecommended Presets (For more details refer to https://auto.gluon.ai/stable/tutorials/tabular/tabular-essentials.html#presets):\n",
      "\tpresets='best_quality'   : Maximize accuracy. Default time_limit=3600.\n",
      "\tpresets='high_quality'   : Strong accuracy with fast inference speed. Default time_limit=3600.\n",
      "\tpresets='good_quality'   : Good accuracy with very fast inference speed. Default time_limit=3600.\n",
      "\tpresets='medium_quality' : Fast training time, ideal for initial prototyping.\n",
      "Beginning AutoGluon training ... Time limit = 5s\n",
      "AutoGluon will save models to \"AutogluonModels/ag-20240522_162316\"\n",
      "=================== System Info ===================\n",
      "AutoGluon Version:  1.1.0\n",
      "Python Version:     3.10.13\n",
      "Operating System:   Darwin\n",
      "Platform Machine:   arm64\n",
      "Platform Version:   Darwin Kernel Version 23.5.0: Wed May  1 20:14:38 PDT 2024; root:xnu-10063.121.3~5/RELEASE_ARM64_T6020\n",
      "CPU Count:          10\n",
      "Memory Avail:       5.09 GB / 16.00 GB (31.8%)\n",
      "Disk Space Avail:   393.45 GB / 460.43 GB (85.5%)\n",
      "===================================================\n",
      "Train Data Rows:    2164\n",
      "Train Data Columns: 8\n",
      "Label Column:       two_year_recid\n",
      "AutoGluon infers your prediction problem is: 'binary' (because only two unique label-values observed).\n",
      "\t2 unique label values:  [0, 1]\n",
      "\tIf 'binary' is not the correct problem_type, please manually specify the problem_type parameter during predictor init (You may specify problem_type as one of: ['binary', 'multiclass', 'regression'])\n",
      "Problem Type:       binary\n",
      "Preprocessing data ...\n",
      "Selected class <--> label mapping:  class 1 = 1, class 0 = 0\n",
      "Using Feature Generators to preprocess the data ...\n",
      "Fitting AutoMLPipelineFeatureGenerator...\n",
      "\tAvailable Memory:                    5210.76 MB\n",
      "\tTrain Data (Original)  Memory Usage: 0.47 MB (0.0% of available memory)\n",
      "\tInferring data type of each feature based on column values. Set feature_metadata_in to manually specify special dtypes of the features.\n",
      "\tStage 1 Generators:\n",
      "\t\tFitting AsTypeFeatureGenerator...\n",
      "\t\t\tNote: Converting 2 features to boolean dtype as they only contain 2 unique values.\n",
      "\tStage 2 Generators:\n",
      "\t\tFitting FillNaFeatureGenerator...\n",
      "\tStage 3 Generators:\n",
      "\t\tFitting IdentityFeatureGenerator...\n",
      "\t\tFitting CategoryFeatureGenerator...\n",
      "\t\t\tFitting CategoryMemoryMinimizeFeatureGenerator...\n",
      "\tStage 4 Generators:\n",
      "\t\tFitting DropUniqueFeatureGenerator...\n",
      "\tStage 5 Generators:\n",
      "\t\tFitting DropDuplicatesFeatureGenerator...\n",
      "\tTypes of features in original data (raw dtype, special dtypes):\n",
      "\t\t('int', [])    : 5 | ['age', 'juv_fel_count', 'juv_misd_count', 'juv_other_count', 'priors_count']\n",
      "\t\t('object', []) : 3 | ['sex', 'age_cat', 'c_charge_degree']\n",
      "\tTypes of features in processed data (raw dtype, special dtypes):\n",
      "\t\t('category', [])  : 1 | ['age_cat']\n",
      "\t\t('int', [])       : 5 | ['age', 'juv_fel_count', 'juv_misd_count', 'juv_other_count', 'priors_count']\n",
      "\t\t('int', ['bool']) : 2 | ['sex', 'c_charge_degree']\n",
      "\t0.0s = Fit runtime\n",
      "\t8 features in original data used to generate 8 features in processed data.\n",
      "\tTrain Data (Processed) Memory Usage: 0.09 MB (0.0% of available memory)\n",
      "Data preprocessing and feature engineering runtime = 0.03s ...\n",
      "AutoGluon will gauge predictive performance using evaluation metric: 'accuracy'\n",
      "\tTo change this, specify the eval_metric parameter of Predictor()\n",
      "Automatically generating train/validation split with holdout_frac=0.2, Train Rows: 1731, Val Rows: 433\n",
      "User-specified model hyperparameters to be fit:\n",
      "{\n",
      "\t'NN_TORCH': {},\n",
      "\t'GBM': [{'extra_trees': True, 'ag_args': {'name_suffix': 'XT'}}, {}, 'GBMLarge'],\n",
      "\t'CAT': {},\n",
      "\t'XGB': {},\n",
      "\t'FASTAI': {},\n",
      "\t'RF': [{'criterion': 'gini', 'ag_args': {'name_suffix': 'Gini', 'problem_types': ['binary', 'multiclass']}}, {'criterion': 'entropy', 'ag_args': {'name_suffix': 'Entr', 'problem_types': ['binary', 'multiclass']}}, {'criterion': 'squared_error', 'ag_args': {'name_suffix': 'MSE', 'problem_types': ['regression', 'quantile']}}],\n",
      "\t'XT': [{'criterion': 'gini', 'ag_args': {'name_suffix': 'Gini', 'problem_types': ['binary', 'multiclass']}}, {'criterion': 'entropy', 'ag_args': {'name_suffix': 'Entr', 'problem_types': ['binary', 'multiclass']}}, {'criterion': 'squared_error', 'ag_args': {'name_suffix': 'MSE', 'problem_types': ['regression', 'quantile']}}],\n",
      "\t'KNN': [{'weights': 'uniform', 'ag_args': {'name_suffix': 'Unif'}}, {'weights': 'distance', 'ag_args': {'name_suffix': 'Dist'}}],\n",
      "}\n",
      "Fitting 13 L1 models ...\n",
      "Fitting model: KNeighborsUnif ... Training model for up to 4.97s of the 4.97s of remaining time.\n",
      "\t0.6351\t = Validation score   (accuracy)\n",
      "\t0.0s\t = Training   runtime\n",
      "\t0.01s\t = Validation runtime\n",
      "Fitting model: KNeighborsDist ... Training model for up to 4.95s of the 4.95s of remaining time.\n",
      "\t0.6259\t = Validation score   (accuracy)\n",
      "\t0.0s\t = Training   runtime\n",
      "\t0.01s\t = Validation runtime\n",
      "Fitting model: LightGBMXT ... Training model for up to 4.93s of the 4.93s of remaining time.\n",
      "/opt/miniconda3/envs/ag/lib/python3.10/site-packages/dask/dataframe/__init__.py:31: FutureWarning: \n",
      "Dask dataframe query planning is disabled because dask-expr is not installed.\n",
      "\n",
      "You can install it with `pip install dask[dataframe]` or `conda install dask`.\n",
      "This will raise in a future version.\n",
      "\n",
      "  warnings.warn(msg, FutureWarning)\n",
      "\t0.6882\t = Validation score   (accuracy)\n",
      "\t3.56s\t = Training   runtime\n",
      "\t0.0s\t = Validation runtime\n",
      "Fitting model: LightGBM ... Training model for up to 1.36s of the 1.36s of remaining time.\n",
      "\tRan out of time, early stopping on iteration 315. Best iteration is:\n",
      "\t[27]\tvalid_set's binary_error: 0.323326\n",
      "\t0.6767\t = Validation score   (accuracy)\n",
      "\t1.37s\t = Training   runtime\n",
      "\t0.0s\t = Validation runtime\n",
      "Fitting model: WeightedEnsemble_L2 ... Training model for up to 4.97s of the -0.03s of remaining time.\n",
      "\tEnsemble Weights: {'LightGBM': 0.667, 'LightGBMXT': 0.333}\n",
      "\t0.6928\t = Validation score   (accuracy)\n",
      "\t0.02s\t = Training   runtime\n",
      "\t0.0s\t = Validation runtime\n",
      "AutoGluon training complete, total runtime = 5.08s ... Best model: \"WeightedEnsemble_L2\"\n",
      "TabularPredictor saved. To load, use: predictor = TabularPredictor.load(\"AutogluonModels/ag-20240522_162316\")\n",
      "No path specified. Models will be saved in: \"AutogluonModels/ag-20240522_162321\"\n",
      "No presets specified! To achieve strong results with AutoGluon, it is recommended to use the available presets.\n",
      "\tRecommended Presets (For more details refer to https://auto.gluon.ai/stable/tutorials/tabular/tabular-essentials.html#presets):\n",
      "\tpresets='best_quality'   : Maximize accuracy. Default time_limit=3600.\n",
      "\tpresets='high_quality'   : Strong accuracy with fast inference speed. Default time_limit=3600.\n",
      "\tpresets='good_quality'   : Good accuracy with very fast inference speed. Default time_limit=3600.\n",
      "\tpresets='medium_quality' : Fast training time, ideal for initial prototyping.\n",
      "Beginning AutoGluon training ... Time limit = 5s\n",
      "AutoGluon will save models to \"AutogluonModels/ag-20240522_162321\"\n",
      "=================== System Info ===================\n",
      "AutoGluon Version:  1.1.0\n",
      "Python Version:     3.10.13\n",
      "Operating System:   Darwin\n",
      "Platform Machine:   arm64\n",
      "Platform Version:   Darwin Kernel Version 23.5.0: Wed May  1 20:14:38 PDT 2024; root:xnu-10063.121.3~5/RELEASE_ARM64_T6020\n",
      "CPU Count:          10\n",
      "Memory Avail:       5.11 GB / 16.00 GB (31.9%)\n",
      "Disk Space Avail:   393.44 GB / 460.43 GB (85.5%)\n",
      "===================================================\n",
      "Train Data Rows:    2164\n",
      "Train Data Columns: 8\n",
      "Label Column:       race\n",
      "AutoGluon infers your prediction problem is: 'binary' (because only two unique label-values observed).\n",
      "\t2 unique label values:  ['African-American', 'Other']\n",
      "\tIf 'binary' is not the correct problem_type, please manually specify the problem_type parameter during predictor init (You may specify problem_type as one of: ['binary', 'multiclass', 'regression'])\n",
      "Problem Type:       binary\n",
      "Preprocessing data ...\n",
      "Selected class <--> label mapping:  class 1 = Other, class 0 = African-American\n",
      "\tNote: For your binary classification, AutoGluon arbitrarily selected which label-value represents positive (Other) vs negative (African-American) class.\n",
      "\tTo explicitly set the positive_class, either rename classes to 1 and 0, or specify positive_class in Predictor init.\n",
      "Using Feature Generators to preprocess the data ...\n",
      "Fitting AutoMLPipelineFeatureGenerator...\n",
      "\tAvailable Memory:                    5231.54 MB\n",
      "\tTrain Data (Original)  Memory Usage: 0.47 MB (0.0% of available memory)\n",
      "\tInferring data type of each feature based on column values. Set feature_metadata_in to manually specify special dtypes of the features.\n",
      "\tStage 1 Generators:\n",
      "\t\tFitting AsTypeFeatureGenerator...\n",
      "\t\t\tNote: Converting 2 features to boolean dtype as they only contain 2 unique values.\n",
      "\tStage 2 Generators:\n",
      "\t\tFitting FillNaFeatureGenerator...\n",
      "\tStage 3 Generators:\n",
      "\t\tFitting IdentityFeatureGenerator...\n",
      "\t\tFitting CategoryFeatureGenerator...\n",
      "\t\t\tFitting CategoryMemoryMinimizeFeatureGenerator...\n",
      "\tStage 4 Generators:\n",
      "\t\tFitting DropUniqueFeatureGenerator...\n",
      "\tStage 5 Generators:\n",
      "\t\tFitting DropDuplicatesFeatureGenerator...\n",
      "\tTypes of features in original data (raw dtype, special dtypes):\n",
      "\t\t('int', [])    : 5 | ['age', 'juv_fel_count', 'juv_misd_count', 'juv_other_count', 'priors_count']\n",
      "\t\t('object', []) : 3 | ['sex', 'age_cat', 'c_charge_degree']\n",
      "\tTypes of features in processed data (raw dtype, special dtypes):\n",
      "\t\t('category', [])  : 1 | ['age_cat']\n",
      "\t\t('int', [])       : 5 | ['age', 'juv_fel_count', 'juv_misd_count', 'juv_other_count', 'priors_count']\n",
      "\t\t('int', ['bool']) : 2 | ['sex', 'c_charge_degree']\n",
      "\t0.0s = Fit runtime\n",
      "\t8 features in original data used to generate 8 features in processed data.\n",
      "\tTrain Data (Processed) Memory Usage: 0.09 MB (0.0% of available memory)\n",
      "Data preprocessing and feature engineering runtime = 0.03s ...\n",
      "AutoGluon will gauge predictive performance using evaluation metric: 'accuracy'\n",
      "\tTo change this, specify the eval_metric parameter of Predictor()\n",
      "Automatically generating train/validation split with holdout_frac=0.2, Train Rows: 1731, Val Rows: 433\n",
      "User-specified model hyperparameters to be fit:\n",
      "{\n",
      "\t'NN_TORCH': {},\n",
      "\t'GBM': [{'extra_trees': True, 'ag_args': {'name_suffix': 'XT'}}, {}, 'GBMLarge'],\n",
      "\t'CAT': {},\n",
      "\t'XGB': {},\n",
      "\t'FASTAI': {},\n",
      "\t'RF': [{'criterion': 'gini', 'ag_args': {'name_suffix': 'Gini', 'problem_types': ['binary', 'multiclass']}}, {'criterion': 'entropy', 'ag_args': {'name_suffix': 'Entr', 'problem_types': ['binary', 'multiclass']}}, {'criterion': 'squared_error', 'ag_args': {'name_suffix': 'MSE', 'problem_types': ['regression', 'quantile']}}],\n",
      "\t'XT': [{'criterion': 'gini', 'ag_args': {'name_suffix': 'Gini', 'problem_types': ['binary', 'multiclass']}}, {'criterion': 'entropy', 'ag_args': {'name_suffix': 'Entr', 'problem_types': ['binary', 'multiclass']}}, {'criterion': 'squared_error', 'ag_args': {'name_suffix': 'MSE', 'problem_types': ['regression', 'quantile']}}],\n",
      "\t'KNN': [{'weights': 'uniform', 'ag_args': {'name_suffix': 'Unif'}}, {'weights': 'distance', 'ag_args': {'name_suffix': 'Dist'}}],\n",
      "}\n",
      "Fitting 13 L1 models ...\n",
      "Fitting model: KNeighborsUnif ... Training model for up to 4.97s of the 4.97s of remaining time.\n",
      "\t0.6051\t = Validation score   (accuracy)\n",
      "\t0.0s\t = Training   runtime\n",
      "\t0.01s\t = Validation runtime\n",
      "Fitting model: KNeighborsDist ... Training model for up to 4.95s of the 4.95s of remaining time.\n",
      "\t0.5889\t = Validation score   (accuracy)\n",
      "\t0.0s\t = Training   runtime\n",
      "\t0.01s\t = Validation runtime\n",
      "Fitting model: LightGBMXT ... Training model for up to 4.92s of the 4.92s of remaining time.\n",
      "\t0.6559\t = Validation score   (accuracy)\n",
      "\t1.86s\t = Training   runtime\n",
      "\t0.0s\t = Validation runtime\n",
      "Fitting model: LightGBM ... Training model for up to 3.05s of the 3.05s of remaining time.\n",
      "\t0.6351\t = Validation score   (accuracy)\n",
      "\t1.35s\t = Training   runtime\n",
      "\t0.0s\t = Validation runtime\n",
      "Fitting model: RandomForestGini ... Training model for up to 1.69s of the 1.69s of remaining time.\n",
      "\t0.5727\t = Validation score   (accuracy)\n",
      "\t0.58s\t = Training   runtime\n",
      "\t0.03s\t = Validation runtime\n",
      "Fitting model: RandomForestEntr ... Training model for up to 1.06s of the 1.06s of remaining time.\n",
      "\t0.5681\t = Validation score   (accuracy)\n",
      "\t0.27s\t = Training   runtime\n",
      "\t0.03s\t = Validation runtime\n",
      "Fitting model: CatBoost ... Training model for up to 0.74s of the 0.73s of remaining time.\n",
      "\t0.6697\t = Validation score   (accuracy)\n",
      "\t0.72s\t = Training   runtime\n",
      "\t0.0s\t = Validation runtime\n",
      "Fitting model: WeightedEnsemble_L2 ... Training model for up to 4.97s of the -0.01s of remaining time.\n",
      "\tEnsemble Weights: {'CatBoost': 1.0}\n",
      "\t0.6697\t = Validation score   (accuracy)\n",
      "\t0.03s\t = Training   runtime\n",
      "\t0.0s\t = Validation runtime\n",
      "AutoGluon training complete, total runtime = 5.06s ... Best model: \"WeightedEnsemble_L2\"\n",
      "TabularPredictor saved. To load, use: predictor = TabularPredictor.load(\"AutogluonModels/ag-20240522_162321\")\n"
     ]
    }
   ],
   "source": [
    "# To enforce fairness constraints without access to protected attributes at test time, we train two classifiers to infer the 2-year recidivism rate, and ethnicity.\n",
    "  \n",
    "predictor2, protected = inferred_attribute_builder(train, 'two_year_recid', 'race', time_limit=5)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [],
   "source": [
    "# From these a single predictor that maximizes acuracy while reducing the demographic parity violation to less than 2.5% can be trained by running:\n",
    "\n",
    "fpredictor=FairPredictor(predictor2, val, 'race', inferred_groups=protected)\n",
    "fpredictor.fit(gm.accuracy, gm.demographic_parity, 0.025)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "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):\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": 17,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "No path specified. Models will be saved in: \"AutogluonModels/ag-20240522_162327\"\n",
      "No presets specified! To achieve strong results with AutoGluon, it is recommended to use the available presets.\n",
      "\tRecommended Presets (For more details refer to https://auto.gluon.ai/stable/tutorials/tabular/tabular-essentials.html#presets):\n",
      "\tpresets='best_quality'   : Maximize accuracy. Default time_limit=3600.\n",
      "\tpresets='high_quality'   : Strong accuracy with fast inference speed. Default time_limit=3600.\n",
      "\tpresets='good_quality'   : Good accuracy with very fast inference speed. Default time_limit=3600.\n",
      "\tpresets='medium_quality' : Fast training time, ideal for initial prototyping.\n",
      "Beginning AutoGluon training ... Time limit = 5s\n",
      "AutoGluon will save models to \"AutogluonModels/ag-20240522_162327\"\n",
      "=================== System Info ===================\n",
      "AutoGluon Version:  1.1.0\n",
      "Python Version:     3.10.13\n",
      "Operating System:   Darwin\n",
      "Platform Machine:   arm64\n",
      "Platform Version:   Darwin Kernel Version 23.5.0: Wed May  1 20:14:38 PDT 2024; root:xnu-10063.121.3~5/RELEASE_ARM64_T6020\n",
      "CPU Count:          10\n",
      "Memory Avail:       5.04 GB / 16.00 GB (31.5%)\n",
      "Disk Space Avail:   393.40 GB / 460.43 GB (85.4%)\n",
      "===================================================\n",
      "Train Data Rows:    2164\n",
      "Train Data Columns: 9\n",
      "Label Column:       two_year_recid\n",
      "AutoGluon infers your prediction problem is: 'binary' (because only two unique label-values observed).\n",
      "\t2 unique label values:  [0, 1]\n",
      "\tIf 'binary' is not the correct problem_type, please manually specify the problem_type parameter during predictor init (You may specify problem_type as one of: ['binary', 'multiclass', 'regression'])\n",
      "Problem Type:       binary\n",
      "Preprocessing data ...\n",
      "Selected class <--> label mapping:  class 1 = 1, class 0 = 0\n",
      "Using Feature Generators to preprocess the data ...\n",
      "Fitting AutoMLPipelineFeatureGenerator...\n",
      "\tAvailable Memory:                    5166.17 MB\n",
      "\tTrain Data (Original)  Memory Usage: 0.61 MB (0.0% of available memory)\n",
      "\tInferring data type of each feature based on column values. Set feature_metadata_in to manually specify special dtypes of the features.\n",
      "\tStage 1 Generators:\n",
      "\t\tFitting AsTypeFeatureGenerator...\n",
      "\t\t\tNote: Converting 3 features to boolean dtype as they only contain 2 unique values.\n",
      "\tStage 2 Generators:\n",
      "\t\tFitting FillNaFeatureGenerator...\n",
      "\tStage 3 Generators:\n",
      "\t\tFitting IdentityFeatureGenerator...\n",
      "\t\tFitting CategoryFeatureGenerator...\n",
      "\t\t\tFitting CategoryMemoryMinimizeFeatureGenerator...\n",
      "\tStage 4 Generators:\n",
      "\t\tFitting DropUniqueFeatureGenerator...\n",
      "\tStage 5 Generators:\n",
      "\t\tFitting DropDuplicatesFeatureGenerator...\n",
      "\tTypes of features in original data (raw dtype, special dtypes):\n",
      "\t\t('int', [])    : 5 | ['age', 'juv_fel_count', 'juv_misd_count', 'juv_other_count', 'priors_count']\n",
      "\t\t('object', []) : 4 | ['sex', 'race', 'age_cat', 'c_charge_degree']\n",
      "\tTypes of features in processed data (raw dtype, special dtypes):\n",
      "\t\t('category', [])  : 1 | ['age_cat']\n",
      "\t\t('int', [])       : 5 | ['age', 'juv_fel_count', 'juv_misd_count', 'juv_other_count', 'priors_count']\n",
      "\t\t('int', ['bool']) : 3 | ['sex', 'race', 'c_charge_degree']\n",
      "\t0.0s = Fit runtime\n",
      "\t9 features in original data used to generate 9 features in processed data.\n",
      "\tTrain Data (Processed) Memory Usage: 0.09 MB (0.0% of available memory)\n",
      "Data preprocessing and feature engineering runtime = 0.03s ...\n",
      "AutoGluon will gauge predictive performance using evaluation metric: 'accuracy'\n",
      "\tTo change this, specify the eval_metric parameter of Predictor()\n",
      "Automatically generating train/validation split with holdout_frac=0.2, Train Rows: 1731, Val Rows: 433\n",
      "User-specified model hyperparameters to be fit:\n",
      "{\n",
      "\t'NN_TORCH': {},\n",
      "\t'GBM': [{'extra_trees': True, 'ag_args': {'name_suffix': 'XT'}}, {}, 'GBMLarge'],\n",
      "\t'CAT': {},\n",
      "\t'XGB': {},\n",
      "\t'FASTAI': {},\n",
      "\t'RF': [{'criterion': 'gini', 'ag_args': {'name_suffix': 'Gini', 'problem_types': ['binary', 'multiclass']}}, {'criterion': 'entropy', 'ag_args': {'name_suffix': 'Entr', 'problem_types': ['binary', 'multiclass']}}, {'criterion': 'squared_error', 'ag_args': {'name_suffix': 'MSE', 'problem_types': ['regression', 'quantile']}}],\n",
      "\t'XT': [{'criterion': 'gini', 'ag_args': {'name_suffix': 'Gini', 'problem_types': ['binary', 'multiclass']}}, {'criterion': 'entropy', 'ag_args': {'name_suffix': 'Entr', 'problem_types': ['binary', 'multiclass']}}, {'criterion': 'squared_error', 'ag_args': {'name_suffix': 'MSE', 'problem_types': ['regression', 'quantile']}}],\n",
      "\t'KNN': [{'weights': 'uniform', 'ag_args': {'name_suffix': 'Unif'}}, {'weights': 'distance', 'ag_args': {'name_suffix': 'Dist'}}],\n",
      "}\n",
      "Fitting 13 L1 models ...\n",
      "Fitting model: KNeighborsUnif ... Training model for up to 4.97s of the 4.97s of remaining time.\n",
      "\t0.6351\t = Validation score   (accuracy)\n",
      "\t0.0s\t = Training   runtime\n",
      "\t0.01s\t = Validation runtime\n",
      "Fitting model: KNeighborsDist ... Training model for up to 4.95s of the 4.95s of remaining time.\n",
      "\t0.6259\t = Validation score   (accuracy)\n",
      "\t0.0s\t = Training   runtime\n",
      "\t0.01s\t = Validation runtime\n",
      "Fitting model: LightGBMXT ... Training model for up to 4.93s of the 4.93s of remaining time.\n",
      "\t0.6975\t = Validation score   (accuracy)\n",
      "\t1.15s\t = Training   runtime\n",
      "\t0.0s\t = Validation runtime\n",
      "Fitting model: LightGBM ... Training model for up to 3.76s of the 3.76s of remaining time.\n",
      "\t0.6744\t = Validation score   (accuracy)\n",
      "\t1.23s\t = Training   runtime\n",
      "\t0.0s\t = Validation runtime\n",
      "Fitting model: RandomForestGini ... Training model for up to 2.53s of the 2.52s of remaining time.\n",
      "\t0.6259\t = Validation score   (accuracy)\n",
      "\t0.26s\t = Training   runtime\n",
      "\t0.03s\t = Validation runtime\n",
      "Fitting model: RandomForestEntr ... Training model for up to 2.21s of the 2.21s of remaining time.\n",
      "\t0.6305\t = Validation score   (accuracy)\n",
      "\t0.26s\t = Training   runtime\n",
      "\t0.03s\t = Validation runtime\n",
      "Fitting model: CatBoost ... Training model for up to 1.9s of the 1.9s of remaining time.\n",
      "\t0.7067\t = Validation score   (accuracy)\n",
      "\t0.57s\t = Training   runtime\n",
      "\t0.0s\t = Validation runtime\n",
      "Fitting model: ExtraTreesGini ... Training model for up to 1.33s of the 1.33s of remaining time.\n",
      "\t0.6259\t = Validation score   (accuracy)\n",
      "\t0.27s\t = Training   runtime\n",
      "\t0.03s\t = Validation runtime\n",
      "Fitting model: ExtraTreesEntr ... Training model for up to 1.01s of the 1.0s of remaining time.\n",
      "\t0.6282\t = Validation score   (accuracy)\n",
      "\t0.25s\t = Training   runtime\n",
      "\t0.03s\t = Validation runtime\n",
      "Fitting model: NeuralNetFastAI ... Training model for up to 0.7s of the 0.7s of remaining time.\n",
      "\tRan out of time, stopping training early. (Stopping on epoch 9)\n",
      "\t0.7067\t = Validation score   (accuracy)\n",
      "\t1.58s\t = Training   runtime\n",
      "\t0.01s\t = Validation runtime\n",
      "Fitting model: WeightedEnsemble_L2 ... Training model for up to 4.97s of the -0.94s of remaining time.\n",
      "\tEnsemble Weights: {'CatBoost': 0.412, 'NeuralNetFastAI': 0.294, 'LightGBM': 0.235, 'KNeighborsUnif': 0.059}\n",
      "\t0.7206\t = Validation score   (accuracy)\n",
      "\t0.04s\t = Training   runtime\n",
      "\t0.0s\t = Validation runtime\n",
      "AutoGluon training complete, total runtime = 6.01s ... Best model: \"WeightedEnsemble_L2\"\n",
      "TabularPredictor saved. To load, use: predictor = TabularPredictor.load(\"AutogluonModels/ag-20240522_162327\")\n"
     ]
    }
   ],
   "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",
    "# we first create a classifer using the protected attribute\n",
    "predictor=TabularPredictor(label='two_year_recid').fit(train_data=train, time_limit=5)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "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.322816</td>\n",
       "      <td>0.002713</td>\n",
       "      <td>0.666535</td>\n",
       "      <td>0.661386</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Disparate Impact</th>\n",
       "      <td>0.449970</td>\n",
       "      <td>0.994385</td>\n",
       "      <td>0.666535</td>\n",
       "      <td>0.674059</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in Conditional Acceptance Rate</th>\n",
       "      <td>0.556190</td>\n",
       "      <td>0.001191</td>\n",
       "      <td>0.666535</td>\n",
       "      <td>0.662178</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in Conditional Rejectance Rate</th>\n",
       "      <td>0.343438</td>\n",
       "      <td>0.015404</td>\n",
       "      <td>0.666535</td>\n",
       "      <td>0.656634</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in Accuracy</th>\n",
       "      <td>0.047797</td>\n",
       "      <td>0.051771</td>\n",
       "      <td>0.666535</td>\n",
       "      <td>0.666931</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in Recall</th>\n",
       "      <td>0.286660</td>\n",
       "      <td>0.023849</td>\n",
       "      <td>0.666535</td>\n",
       "      <td>0.665347</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in Acceptance Rate</th>\n",
       "      <td>0.003154</td>\n",
       "      <td>0.053462</td>\n",
       "      <td>0.666535</td>\n",
       "      <td>0.664950</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in Specificity</th>\n",
       "      <td>0.284533</td>\n",
       "      <td>0.096372</td>\n",
       "      <td>0.666535</td>\n",
       "      <td>0.656238</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in Rejection Rate</th>\n",
       "      <td>0.050291</td>\n",
       "      <td>0.051620</td>\n",
       "      <td>0.666535</td>\n",
       "      <td>0.670891</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Treatment Equality</th>\n",
       "      <td>1.101769</td>\n",
       "      <td>0.009207</td>\n",
       "      <td>0.666535</td>\n",
       "      <td>0.678812</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "                                                         Measure (original)  \\\n",
       "Demographic Parity                                                 0.322816   \n",
       "Disparate Impact                                                   0.449970   \n",
       "Average Group Difference in Conditional Acceptance Rate            0.556190   \n",
       "Average Group Difference in Conditional Rejectance Rate            0.343438   \n",
       "Average Group Difference in Accuracy                               0.047797   \n",
       "Average Group Difference in Recall                                 0.286660   \n",
       "Average Group Difference in Acceptance Rate                        0.003154   \n",
       "Average Group Difference in Specificity                            0.284533   \n",
       "Average Group Difference in Rejection Rate                         0.050291   \n",
       "Treatment Equality                                                 1.101769   \n",
       "\n",
       "                                                         Measure (updated)  \\\n",
       "Demographic Parity                                                0.002713   \n",
       "Disparate Impact                                                  0.994385   \n",
       "Average Group Difference in Conditional Acceptance Rate           0.001191   \n",
       "Average Group Difference in Conditional Rejectance Rate           0.015404   \n",
       "Average Group Difference in Accuracy                              0.051771   \n",
       "Average Group Difference in Recall                                0.023849   \n",
       "Average Group Difference in Acceptance Rate                       0.053462   \n",
       "Average Group Difference in Specificity                           0.096372   \n",
       "Average Group Difference in Rejection Rate                        0.051620   \n",
       "Treatment Equality                                                0.009207   \n",
       "\n",
       "                                                         Accuracy (original)  \\\n",
       "Demographic Parity                                                  0.666535   \n",
       "Disparate Impact                                                    0.666535   \n",
       "Average Group Difference in Conditional Acceptance Rate             0.666535   \n",
       "Average Group Difference in Conditional Rejectance Rate             0.666535   \n",
       "Average Group Difference in Accuracy                                0.666535   \n",
       "Average Group Difference in Recall                                  0.666535   \n",
       "Average Group Difference in Acceptance Rate                         0.666535   \n",
       "Average Group Difference in Specificity                             0.666535   \n",
       "Average Group Difference in Rejection Rate                          0.666535   \n",
       "Treatment Equality                                                  0.666535   \n",
       "\n",
       "                                                         Accuracy (updated)  \n",
       "Demographic Parity                                                 0.661386  \n",
       "Disparate Impact                                                   0.674059  \n",
       "Average Group Difference in Conditional Acceptance Rate            0.662178  \n",
       "Average Group Difference in Conditional Rejectance Rate            0.656634  \n",
       "Average Group Difference in Accuracy                               0.666931  \n",
       "Average Group Difference in Recall                                 0.665347  \n",
       "Average Group Difference in Acceptance Rate                        0.664950  \n",
       "Average Group Difference in Specificity                            0.656238  \n",
       "Average Group Difference in Rejection Rate                         0.670891  \n",
       "Treatment Equality                                                 0.678812  "
      ]
     },
     "execution_count": 18,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "#Create a fair predictor object, using the attribute 'race' on validation data\n",
    "fpredictor = FairPredictor(predictor, val, 'race')\n",
    "#and then evaluate it\n",
    "evaluate(fpredictor, gm.clarify_metrics)\n",
    "#Note that Class Imbalance is a property of the dataset and can not be altered."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "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.234362</td>\n",
       "      <td>0.019982</td>\n",
       "      <td>0.671287</td>\n",
       "      <td>0.571089</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Disparate Impact</th>\n",
       "      <td>0.561681</td>\n",
       "      <td>0.979621</td>\n",
       "      <td>0.671287</td>\n",
       "      <td>0.454653</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in Conditional Acceptance Rate</th>\n",
       "      <td>0.300350</td>\n",
       "      <td>0.002536</td>\n",
       "      <td>0.671287</td>\n",
       "      <td>0.592079</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in Conditional Rejectance Rate</th>\n",
       "      <td>0.165646</td>\n",
       "      <td>0.007828</td>\n",
       "      <td>0.671287</td>\n",
       "      <td>0.677228</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in Accuracy</th>\n",
       "      <td>0.043198</td>\n",
       "      <td>0.044679</td>\n",
       "      <td>0.671287</td>\n",
       "      <td>0.668119</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in Recall</th>\n",
       "      <td>0.190611</td>\n",
       "      <td>0.010143</td>\n",
       "      <td>0.671287</td>\n",
       "      <td>0.560396</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in Acceptance Rate</th>\n",
       "      <td>0.033289</td>\n",
       "      <td>0.111111</td>\n",
       "      <td>0.671287</td>\n",
       "      <td>0.560792</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in Specificity</th>\n",
       "      <td>0.197344</td>\n",
       "      <td>0.038921</td>\n",
       "      <td>0.671287</td>\n",
       "      <td>0.621386</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Average Group Difference in Rejection Rate</th>\n",
       "      <td>0.074006</td>\n",
       "      <td>0.078283</td>\n",
       "      <td>0.671287</td>\n",
       "      <td>0.668119</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Treatment Equality</th>\n",
       "      <td>0.561487</td>\n",
       "      <td>0.044549</td>\n",
       "      <td>0.671287</td>\n",
       "      <td>0.622178</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "                                                         Measure (original)  \\\n",
       "Demographic Parity                                                 0.234362   \n",
       "Disparate Impact                                                   0.561681   \n",
       "Average Group Difference in Conditional Acceptance Rate            0.300350   \n",
       "Average Group Difference in Conditional Rejectance Rate            0.165646   \n",
       "Average Group Difference in Accuracy                               0.043198   \n",
       "Average Group Difference in Recall                                 0.190611   \n",
       "Average Group Difference in Acceptance Rate                        0.033289   \n",
       "Average Group Difference in Specificity                            0.197344   \n",
       "Average Group Difference in Rejection Rate                         0.074006   \n",
       "Treatment Equality                                                 0.561487   \n",
       "\n",
       "                                                         Measure (updated)  \\\n",
       "Demographic Parity                                                0.019982   \n",
       "Disparate Impact                                                  0.979621   \n",
       "Average Group Difference in Conditional Acceptance Rate           0.002536   \n",
       "Average Group Difference in Conditional Rejectance Rate           0.007828   \n",
       "Average Group Difference in Accuracy                              0.044679   \n",
       "Average Group Difference in Recall                                0.010143   \n",
       "Average Group Difference in Acceptance Rate                       0.111111   \n",
       "Average Group Difference in Specificity                           0.038921   \n",
       "Average Group Difference in Rejection Rate                        0.078283   \n",
       "Treatment Equality                                                0.044549   \n",
       "\n",
       "                                                         Accuracy (original)  \\\n",
       "Demographic Parity                                                  0.671287   \n",
       "Disparate Impact                                                    0.671287   \n",
       "Average Group Difference in Conditional Acceptance Rate             0.671287   \n",
       "Average Group Difference in Conditional Rejectance Rate             0.671287   \n",
       "Average Group Difference in Accuracy                                0.671287   \n",
       "Average Group Difference in Recall                                  0.671287   \n",
       "Average Group Difference in Acceptance Rate                         0.671287   \n",
       "Average Group Difference in Specificity                             0.671287   \n",
       "Average Group Difference in Rejection Rate                          0.671287   \n",
       "Treatment Equality                                                  0.671287   \n",
       "\n",
       "                                                         Accuracy (updated)  \n",
       "Demographic Parity                                                 0.571089  \n",
       "Disparate Impact                                                   0.454653  \n",
       "Average Group Difference in Conditional Acceptance Rate            0.592079  \n",
       "Average Group Difference in Conditional Rejectance Rate            0.677228  \n",
       "Average Group Difference in Accuracy                               0.668119  \n",
       "Average Group Difference in Recall                                 0.560396  \n",
       "Average Group Difference in Acceptance Rate                        0.560792  \n",
       "Average Group Difference in Specificity                            0.621386  \n",
       "Average Group Difference in Rejection Rate                         0.668119  \n",
       "Treatment Equality                                                 0.622178  "
      ]
     },
     "execution_count": 19,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    " # Now using infered attributes\n",
    "fpredictor2 = FairPredictor(predictor2, val, 'race', inferred_groups=protected)\n",
    "evaluate(fpredictor2, gm.clarify_metrics)"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3.9.15 64-bit",
   "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"
  },
  "orig_nbformat": 4,
  "vscode": {
   "interpreter": {
    "hash": "397704579725e15f5c7cb49fe5f0341eb7531c82d19f2c29d197e8b64ab5776b"
   }
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
