{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "1220bb3c-ba92-4dbc-8694-5ef71fad0f13",
   "metadata": {},
   "outputs": [],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "62fea57b-cfdf-43d3-b811-be5c3f80782d",
   "metadata": {},
   "source": [
    "# Global Fairness Verification example\n",
    "\n",
    "*Based on Calzavara et al. Explainable Global Fairness Veriﬁcation of Tree-Based Classiﬁers\" (2022).*\n",
    "\n",
    "This example focuses on fairness that expresses (a lack of) causal discrimination. It is a global fairness (or verification of global robustness) task which means that is does not rely on the choice of a specific test set, but (implicitly) ranges over all possible instances."
   ]
  },
  {
   "cell_type": "markdown",
   "id": "227fe8d9-320e-4e4e-a156-830f0c8f46b0",
   "metadata": {},
   "source": [
    "## Preliminaries"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "14f9a281-69d4-4990-8870-f973202cf064",
   "metadata": {},
   "outputs": [],
   "source": [
    "import time\n",
    "from sklearn import datasets\n",
    "from sklearn.model_selection import train_test_split\n",
    "from sklearn.preprocessing import OrdinalEncoder\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "from tqdm import tqdm\n",
    "import matplotlib.pyplot as plt"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "07cc3f8b-21aa-45b8-a950-20efdb0c3a37",
   "metadata": {},
   "source": [
    "## German dataset"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "3dcb5180-6551-43f8-b369-c4f6e0177686",
   "metadata": {},
   "source": [
    "We will use the [German Credit Data](https://archive.ics.uci.edu/dataset/144/statlog+german+credit+data) benchmark dataset."
   ]
  },
  {
   "cell_type": "markdown",
   "id": "4e94fa0c-2399-4a7f-a460-61314d62cdad",
   "metadata": {},
   "source": [
    "### Load the data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "3eb318d6-9113-49f3-ba83-472a18f63d7a",
   "metadata": {},
   "outputs": [],
   "source": [
    "X_df, y_df = datasets.fetch_openml(data_id=31, return_X_y=True, as_frame=True)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c523b787-b246-446f-9f5f-ec5b5e01a279",
   "metadata": {},
   "source": [
    "Make the target numerical"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "9c561cd5-5d82-4127-ac3a-4c6f91ed37c4",
   "metadata": {},
   "outputs": [],
   "source": [
    "y = (y_df=='good').astype(int)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "6a369f1b-a207-41ad-b67f-4b5f9b7603cf",
   "metadata": {},
   "outputs": [],
   "source": [
    "ordinal_feature_indexes = [0, 2, 3, 5, 6, 8, 9, 11, 13, 14, 16, 18, 19]\n",
    "ord_enc = OrdinalEncoder(categories=[\n",
    "    ['<0', '0<=X<200', '>=200', 'no checking'], # 0: checking_status\n",
    "    ['no credits/all paid', 'all paid', 'existing paid', 'delayed previously', 'critical/other existing credit'], # 2: credit_history\n",
    "    ['domestic appliance', 'furniture/equipment', 'radio/tv', 'repairs',\n",
    "         'new car', 'used car', 'business', 'education', 'retraining', 'other'], # 3: purpose, likely better as one-hot\n",
    "    ['<100', '100<=X<500', '500<=X<1000', '>=1000', 'no known savings'], # 5: saving_status\n",
    "    ['<1', '1<=X<4', '4<=X<7', '>=7', 'unemployed'], # 6: employment\n",
    "    ['female div/dep/mar', 'male div/sep', 'male mar/wid', 'male single'], # 8: personal_status\n",
    "    ['none', 'co applicant', 'guarantor'], # 9: other parties\n",
    "    ['no known property', 'car', 'real estate', 'life insurance'], # 11: property_magnitude\n",
    "    ['none', 'stores', 'bank'], # 13: other_payment_plans\n",
    "    ['for free', 'rent', 'own'], # 14: housing\n",
    "    ['unemp/unskilled non res', 'unskilled resident', 'skilled', 'high qualif/self emp/mgmt'], # 16: job\n",
    "    ['none', 'yes'], # 18: own_telephone\n",
    "    ['no', 'yes'], # 19: foreign_worker\n",
    "])\n",
    "ordinals = ord_enc.fit_transform(X_df.iloc[:, ordinal_feature_indexes])\n",
    "columns = []\n",
    "for k in range(X_df.shape[1]):\n",
    "    if k in ordinal_feature_indexes:\n",
    "        u = [u for u, v in enumerate(ordinal_feature_indexes) if v == k][0]\n",
    "        columns.append(ordinals[:, u])\n",
    "    else:\n",
    "        columns.append(X_df.iloc[:, k].values)\n",
    "X_df = pd.DataFrame(np.array(columns).T, columns=X_df.columns)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "2dd8ee74-bc89-4496-98c1-1aea260f0460",
   "metadata": {},
   "outputs": [],
   "source": [
    "X = X_df.to_numpy()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "3d8f52c1-4b83-4186-855b-81f10173e7bd",
   "metadata": {},
   "source": [
    "We will focus on sex as a relevant parameter, thus will split up column 9 (index 8) 'personal_status'."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "da0e0c8c-29e8-40fa-8a4e-39598a023cdd",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "['female div/dep/mar', 'male div/sep', 'male mar/wid', 'male single']\n"
     ]
    }
   ],
   "source": [
    "# Simplify the personal_status feature to just female/male for simplicity\n",
    "print(ord_enc.categories[5])\n",
    "featmap = [1, 0, 0, 0] # 1 => female, 0 => male\n",
    "X[:, 8] = [featmap[int(val)] for val in X[:, 8]]"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "5e221d13-a20e-4357-a2f6-236d20f64676",
   "metadata": {},
   "source": [
    "Introduce bias: we can optionally introduce bias to make the effect easier to see."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "237137fe-3a95-4ccf-bee1-b6e5c2045a8d",
   "metadata": {},
   "outputs": [],
   "source": [
    "introduce_bias = 100\n",
    "if introduce_bias > 0:\n",
    "    X[0:introduce_bias, :] = X[0, :]\n",
    "    half_idx = int(introduce_bias/2)\n",
    "    X[:half_idx, 8] = 0\n",
    "    y[:half_idx] = 1\n",
    "    X[half_idx:introduce_bias, 8] = 1\n",
    "    y[half_idx:introduce_bias] = 0"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "b07ca660-a037-4a89-a91b-969f6ddabb31",
   "metadata": {},
   "source": [
    "Use train and test set"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "1f2bbd35-8240-4dc3-a82c-e48a8ea66f9e",
   "metadata": {},
   "outputs": [],
   "source": [
    "xtrain, xtest, ytrain, ytest = train_test_split(X, y, test_size=0.2, shuffle=True, random_state=73)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "0fa6ebb8-3d9a-4e56-a51b-c5ad9b63d31d",
   "metadata": {},
   "source": [
    "### Train the classifier"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "75023e65-bc8d-444e-a324-4b43091e0ff0",
   "metadata": {},
   "outputs": [],
   "source": [
    "import veritas\n",
    "import xgboost as xgb\n",
    "import matplotlib.pyplot as plt\n",
    "from sklearn.metrics import accuracy_score"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "9dbb8f8d-9fe3-487f-a45e-eaa1abf27ea1",
   "metadata": {},
   "source": [
    "The original paper uses 12 trees of depth 6."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "2ab71998-aae8-42bd-a39d-8a901dfcc7d0",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "XGB trained in 0.02 seconds\n"
     ]
    }
   ],
   "source": [
    "n_estimators = 12 # 50\n",
    "params = {\n",
    "    \"n_estimators\": n_estimators,\n",
    "    \"eval_metric\": \"error\",\n",
    "    \"random_state\": 73,\n",
    "    \n",
    "    \"tree_method\": \"hist\",\n",
    "    \"max_depth\": 6,\n",
    "    \"learning_rate\": 0.5,\n",
    "    \"colsample_bynode\": 0.75,\n",
    "    \"subsample\": 1.0,\n",
    "}\n",
    "\n",
    "model = xgb.XGBClassifier(**params)\n",
    "\n",
    "t = time.time()\n",
    "model.fit(X, y)\n",
    "print(f\"XGB trained in {time.time()-t:.2f} seconds\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "698b2a65-d0af-438b-be01-6489ccff5531",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train acc: 96.0%, test acc: 98.5%\n"
     ]
    }
   ],
   "source": [
    "ytrain_pred = model.predict(xtrain)\n",
    "ytest_pred = model.predict(xtest)\n",
    "acc_train = accuracy_score(ytrain, ytrain_pred)\n",
    "acc_test = accuracy_score(ytest, ytest_pred)\n",
    "\n",
    "print(f\"Train acc: {acc_train*100:.1f}%, test acc: {acc_test*100:.1f}%\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "2047df41-4d14-421d-bc76-ae06067171e0",
   "metadata": {},
   "source": [
    "### Global fairness"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "99a91d09-4996-4666-96d5-b6fd239cae4c",
   "metadata": {},
   "source": [
    "The `personal_status` (converted to female/male) is considered sensitive. Who are the subset of people whose credit risk prediction does not change by ﬂipping their sex. This maps to a query where we want to contrast two instances."
   ]
  },
  {
   "cell_type": "markdown",
   "id": "762f38cf-292a-4d15-952e-90e9a658ed9d",
   "metadata": {},
   "source": [
    "Start with loading the trained model, veritas figures out automatically it is a XGB model."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "2b7a3376-924c-49eb-8dac-fbf1555c8436",
   "metadata": {},
   "outputs": [],
   "source": [
    "at = veritas.get_addtree(model, silent=True)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "8aca69b3-5ead-4a24-8c00-5f1743cac4f8",
   "metadata": {},
   "source": [
    "For two instances X0 and X1, allowing only feature 9 (index 8) 'personal_status' to be different between the two instances, what is the maximum output difference at(X1)-at(X0)?\n",
    "The values that occur are 'male single', 'female div/dep/mar', 'male div/sep' and 'male mar/wid'. Thus only the 2nd (index 1) is female."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "6a585481-598d-4245-b3f2-d60be0578d45",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Feature IDs used by instance 0\n",
      "   and instance 1 respectively:\n",
      "---------------------------------\n",
      "checking_status             0   0 \n",
      "duration                    1   1 \n",
      "credit_history              2   2 \n",
      "purpose                     3   3 \n",
      "credit_amount               4   4 \n",
      "savings_status              5   5 \n",
      "employment                  6   6 \n",
      "installment_commitment      7   7 \n",
      "personal_status             8  28 *\n",
      "other_parties               9   9 \n",
      "residence_since            10  10 \n",
      "property_magnitude         11  11 \n",
      "age                        12  12 \n",
      "other_payment_plans        13  13 \n",
      "housing                    14  14 \n",
      "existing_credits           15  15 \n",
      "job                        16  16 \n",
      "num_dependents             17  17 \n",
      "own_telephone              18  18 \n",
      "foreign_worker             19  19 \n"
     ]
    }
   ],
   "source": [
    "def constrast_two_examples(at, columns, nonfixed_columns):\n",
    "    \"\"\"Create an `veritas.AddTree` that contrast two instances.\n",
    "    \n",
    "    The new AddTree outputs difference between the original AddTree's outputs\n",
    "    for instances 0 and 1.\n",
    "\n",
    "    at: The original veritas.AddTree tree ensemble model\n",
    "    columns: array with column names\n",
    "    nonfixed_columns: columns that are allowed to change between the two instances\n",
    "    \"\"\"\n",
    "    feat_map = veritas.FeatMap(columns)\n",
    "    for column in columns:\n",
    "        if column not in nonfixed_cols:\n",
    "            index_for_instance0 = feat_map.get_index(column, 0)\n",
    "            index_for_instance1 = feat_map.get_index(column, 1)\n",
    "            feat_map.use_same_id_for(index_for_instance0, index_for_instance1)\n",
    "\n",
    "    at_for_instance1 = feat_map.transform(at, 1)\n",
    "    at_contrast = at.concat_negated(at_for_instance1)\n",
    "\n",
    "    print(\"  Feature IDs used by instance 0\\n   and instance 1 respectively:\")\n",
    "    print(\"-\"*(25+4+4))\n",
    "    for column in columns:\n",
    "        mark = \"*\" if column in nonfixed_columns else \"\"\n",
    "        feat_id_instance0 = feat_map.get_feat_id(column, 0)\n",
    "        feat_id_instance1 = feat_map.get_feat_id(column, 1)\n",
    "        print(f\"{column:25s} {feat_id_instance0:3d} {feat_id_instance1:3d}\", mark)\n",
    "    \n",
    "    return at_contrast, feat_map\n",
    "\n",
    "nonfixed_cols = {'personal_status'}\n",
    "at_contrast, feat_map = constrast_two_examples(at, X_df.columns, nonfixed_cols)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "904645d2-4100-4782-95f5-8e7748018b7d",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "lower 6.951, upper 6.916, #sols 1012: : 2016it [00:05, 363.39it/s]\n"
     ]
    }
   ],
   "source": [
    "# We optimize the output of the contrasting ensemble `at_contrast`\n",
    "config = veritas.Config(veritas.HeuristicType.MAX_OUTPUT)\n",
    "\n",
    "# We are interested in cases where the output of instance 0 is greater\n",
    "# than the output of instance 1, i.e., their output difference is greater\n",
    "# than 0.0.\n",
    "config.ignore_state_when_worse_than = 0.0\n",
    "\n",
    "# Veritas uses an approximate search.\n",
    "# We modify the parameters of the approximate search to more aggressively work\n",
    "# on lowering the upper bound instead of also trying to find suboptimal solutions.\n",
    "config.focal_eps = 0.95\n",
    "config.max_focal_size = 100\n",
    "\n",
    "# Obtain the search object from the search configuration\n",
    "# We can optionally constrain certain feature values (e.g., look only for\n",
    "# cases where instance 0 describes a male person)\n",
    "#prune_box = {feat_map.get_feat_id(\"personal_status\", 0): veritas.Interval(0, 0.5)}\n",
    "prune_box = {}\n",
    "search = config.get_search(at_contrast, prune_box)\n",
    "\n",
    "bounds = []\n",
    "num_search_steps_per_iteration = 100\n",
    "stop_reason = veritas.StopReason.NONE\n",
    "\n",
    "with tqdm() as pbar:\n",
    "    #while search.num_solutions() < 10:  # search until we find 10 solutions\n",
    "    while stop_reason != stop_reason.OPTIMAL:  # search until it is certain that the best solution is optimal\n",
    "        stop_reason = search.steps(num_search_steps_per_iteration)\n",
    "        bound_lh = search.current_bounds()\n",
    "        bounds.append((bound_lh.atleast, bound_lh.top_of_open))\n",
    "        pbar.update(1)\n",
    "        pbar.set_description(f\"lower {bound_lh.atleast:.3f}, \"\n",
    "                             f\"upper {bound_lh.top_of_open:.3f}, \"\n",
    "                             f\"#sols {search.num_solutions():<4d}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "d1e115ec-b7ba-4da9-8a22-ee019300f855",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 500x200 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "lo, hi = zip(*bounds)\n",
    "i = list(range(len(bounds)))\n",
    "sol = search.get_solution(0) # best solution so far\n",
    "plt.figure(figsize=(5,2))\n",
    "plt.plot(i, lo)\n",
    "plt.plot(i, hi);\n",
    "plt.xlabel(f'Nb steps / {num_search_steps_per_iteration}')\n",
    "plt.ylabel('at(X1)-at(X0)')\n",
    "plt.scatter(len(bounds)-1, sol.output, marker=\"x\", color=\"red\");"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "549dfd43-0768-4b4b-880f-a11ef483c8c4",
   "metadata": {},
   "source": [
    "We found multiple solutions."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "b46354ad-cf82-4174-8cf8-52bb3b75e512",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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xfHjx8WRI0dEp06dmsw0iNpERkaKtm3bKqeHbN26Vdjb24s33nhDWedh75fS0lJx6tQpcerUKQFArFixQpw6dUpcvXpVCKGf8y8uLhZOTk5i3LhxIjMzU2zevFm0bNmS00OM5ZNPPhHt27cXUqlU9OnTRxw7dszYIRkMgFo/69atU9a5c+eOmDx5srCzsxMtW7YUo0aNEtevX1dp58qVK2LYsGHCwsJC2NvbixkzZoh79+418NkYzr8TZXPtkx9//FH4+voKmUwmunTpIuLj41XKFQqFmDdvnnBychIymUw89thj4sKFCyp1bt68KcaMGSMsLS2FtbW1eOmll0RpaWlDnoZeyeVyMXXqVNG+fXthbm4uOnToIN58802VaQwPe7/s37+/1r8jkZGRQgj9nX96eroYMGCAkMlkom3btmLJkiX1jp2v2SIiIlKDY5RERERqMFESERGpwURJRESkBhMlERGRGkyUREREajBREhERqcFESUREpAYTJdFDZsGCBejRo0e921m/fn2Nt58QNUdMlESNzB9//IFJkyahffv2kMlkcHZ2RmhoKJKSkgx2TA8PD6xcuVJl2+jRo5XvTyRqzviaLaJG5rnnnkNFRQW++uordOjQATdu3MDevXtx8+bNBo3DwsICFhYWDXpMosaIV5REjUhxcTEOHz6MpUuXIjg4GO7u7ujTpw/mzJmDESNGAAByc3MxcuRIWFpawtraGuHh4TVeLfRPgwcPxrRp01S2PfPMM4iKilKWX716FbGxsZBIJJBIJABqv/W6atUqeHl5QSqVonPnzvjmm29UyiUSCdasWYNRo0ahZcuW6NSpE3bs2FG/TiEyMiZKokbE0tISlpaWSEhIQHl5eY1yhUKBkSNHoqioCAcPHkRiYiIuX76M0aNH63zMrVu3ol27dli0aBGuX7+O69ev11pv27ZtmDp1KmbMmIHMzExMnDgRL730Evbv369Sb+HChQgPD0dGRgaefPJJREREoKioSOf4iIyNiZKoETEzM8P69evx1VdfwdbWFkFBQZg7dy4yMjIAAHv37sWZM2fw7bffolevXggMDMTXX3+NgwcPIjU1Vadjtm7dGqamprCysoKzs3OdL7hdvnw5oqKiMHnyZDzyyCOYPn06nn32WSxfvlylXlRUFMaMGYOOHTti8eLFuH37NlJSUnSKjagxYKIkamSee+45XLt2DTt27MDQoUNx4MAB9OzZE+vXr8e5c+fg5uYGNzc3ZX0fHx/Y2tri3LlzBo3r3LlzCAoKUtkWFBRU47jdunVT/tyqVStYW1vXeDM9UVPCREnUCJmbm+Pxxx/HvHnzcPToUURFRWH+/Pk6tWViYoJ/v03vn2+A17cWLVqofJdIJFAoFAY7HpGhMVESNQE+Pj7466+/4O3tjby8POTl5SnLsrKyUFxcDB8fn1r3dXBwUBl3rKqqQmZmpkodqVSKqqoqtTF4e3vXmKKSlJRU53GJHhacHkLUiNy8eRNhYWEYP348unXrBisrK6SlpeH999/HyJEjERISAj8/P0RERGDlypWorKzE5MmTMWjQIAQEBNTa5pAhQzB9+nT89NNP8PLywooVK1BcXKxSx8PDA4cOHcLzzz8PmUwGe3v7Gu3MnDkT4eHh8Pf3R0hICH788Uds3boVv/76qyG6gqjRYKIkakQsLS0RGBiIDz/8EJcuXcK9e/fg5uaG6OhozJ07FxKJBNu3b8f//d//4dFHH4WJiQmGDh2KTz75pM42x48fj/T0dLz44oswMzNDbGwsgoODVeosWrQIEydOhJeXF8rLy2vcqgWqp5R89NFHWL58OaZOnQpPT0+sW7cOgwcP1nc3EDUqElHbbwQREREB4BglERGRWkyUREREajBREhERqcFESUREpAYTJRERkRpMlERERGowURIREanBRElERKQGEyUREZEaTJRERERqMFESERGpwURJRESkxv8DnynHNVnspL0AAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 500x200 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.figure(figsize=(5,2))\n",
    "sol_i = list(range(search.num_solutions()))\n",
    "sol_values = [search.get_solution(i).output for i in sol_i]\n",
    "plt.xlabel('Solution')\n",
    "plt.ylabel('at(X1)-at(X0)')\n",
    "plt.scatter(sol_i, sol_values, s=10, marker=\"x\", lw=0.1, color=\"red\");"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "587a250d-a6b5-4d21-a3d1-fba15608f718",
   "metadata": {},
   "source": [
    "We will focus on that solution where the model is most sure that the target should be different even though only sex is different."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "2206d114-9296-44f7-bd65-ec62a5b4c839",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Maximum difference between instance0 and instance1\n",
      "- current best solution: 6.950628373759333 -> optimal solution\n",
      "- feature value ranges:\n",
      "       0           checking_status Interval(<1)\n",
      "       1                  duration Interval(>=28)\n",
      "       2            credit_history Interval(>=4)\n",
      "       3                   purpose Interval(5,6)\n",
      "       4             credit_amount Interval(1047,1107)\n",
      "       5            savings_status Interval(>=4)\n",
      "       6                employment Interval(>=4)\n",
      "       7    installment_commitment Interval(<3)\n",
      "    *  8           personal_status Interval(<1)\n",
      "       9             other_parties Interval(<1)\n",
      "      10           residence_since Interval(>=3)\n",
      "      11        property_magnitude Interval(<3)\n",
      "      12                       age Interval(>=67)\n",
      "      13       other_payment_plans Interval(<1)\n",
      "      14                   housing Interval(>=2)\n",
      "      15          existing_credits Interval(>=2)\n",
      "      16                       job Interval(<2)\n",
      "      17            num_dependents Interval(>=2)\n",
      "      19            foreign_worker Interval(>=1)\n",
      "    * 28           personal_status Interval(>=1)\n"
     ]
    }
   ],
   "source": [
    "print(\"Maximum difference between instance0 and instance1\")\n",
    "if search.num_solutions() > 0:\n",
    "    sol = search.get_solution(0)  # best solution so far\n",
    "    print(\"- current best solution:\", sol.output, \"->\",\n",
    "          \"optimal\" if search.is_optimal() else \"suboptimal\", \"solution\")\n",
    "    print(\"- feature value ranges:\")\n",
    "    for feat_id, ival in sol.box().items():\n",
    "        column = feat_map.get_name(feat_id)\n",
    "        mark = \"*\" if column in nonfixed_cols else \" \"\n",
    "        print(f\"    {mark}{feat_id:>3d} {column:>25s}\", ival)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "4b068924-936d-4c31-a4c7-616e17750891",
   "metadata": {},
   "outputs": [],
   "source": [
    "def contrasting_examples_from_solutions(feat_map, sol, columns):\n",
    "    nb_features = len(feat_map)\n",
    "    two_examples = np.zeros((2, nb_features))\n",
    "    box = sol.box()\n",
    "\n",
    "    for instance in (0, 1):\n",
    "        for column in columns:\n",
    "            feat_id = feat_map.get_feat_id(column, instance)\n",
    "            feat_id_untrasformed = feat_map.get_feat_id(column, 0)\n",
    "            if feat_id in box:\n",
    "                interval = box[feat_id]\n",
    "                if interval.lo_is_unbound():\n",
    "                    assert not interval.hi_is_unbound()\n",
    "                    value = interval.hi - 1e-4  # not inclusive\n",
    "                else:\n",
    "                    value = interval.lo\n",
    "                if feat_id_untrasformed in ordinal_feature_indexes:  # ordinal feature\n",
    "                    value = np.floor(value)\n",
    "                two_examples[instance, feat_id_untrasformed] = value\n",
    "\n",
    "    return pd.DataFrame(two_examples, columns=columns, index=[\"instance 0\", \"instance 1\"])\n",
    "\n",
    "two_examples = contrasting_examples_from_solutions(feat_map, sol, X_df.columns)\n",
    "predictions = pd.Series(model.predict(two_examples), index=[\"instance 0\", \"instance 1\"])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "c20ecb30-7087-4bbe-b8c8-9e3b8d967f1d",
   "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>instance 0</th>\n",
       "      <th>instance 1</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>checking_status</th>\n",
       "      <td>0.0</td>\n",
       "      <td>0.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>duration</th>\n",
       "      <td>28.0</td>\n",
       "      <td>28.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>credit_history</th>\n",
       "      <td>4.0</td>\n",
       "      <td>4.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>purpose</th>\n",
       "      <td>5.0</td>\n",
       "      <td>5.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>credit_amount</th>\n",
       "      <td>1047.0</td>\n",
       "      <td>1047.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>savings_status</th>\n",
       "      <td>4.0</td>\n",
       "      <td>4.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>employment</th>\n",
       "      <td>4.0</td>\n",
       "      <td>4.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>installment_commitment</th>\n",
       "      <td>3.0</td>\n",
       "      <td>3.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>personal_status</th>\n",
       "      <td>0.0</td>\n",
       "      <td>1.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>other_parties</th>\n",
       "      <td>0.0</td>\n",
       "      <td>0.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>residence_since</th>\n",
       "      <td>3.0</td>\n",
       "      <td>3.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>property_magnitude</th>\n",
       "      <td>2.0</td>\n",
       "      <td>2.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>age</th>\n",
       "      <td>67.0</td>\n",
       "      <td>67.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>other_payment_plans</th>\n",
       "      <td>0.0</td>\n",
       "      <td>0.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>housing</th>\n",
       "      <td>2.0</td>\n",
       "      <td>2.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>existing_credits</th>\n",
       "      <td>2.0</td>\n",
       "      <td>2.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>job</th>\n",
       "      <td>1.0</td>\n",
       "      <td>1.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>num_dependents</th>\n",
       "      <td>2.0</td>\n",
       "      <td>2.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>own_telephone</th>\n",
       "      <td>0.0</td>\n",
       "      <td>0.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>foreign_worker</th>\n",
       "      <td>1.0</td>\n",
       "      <td>1.0</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "                        instance 0  instance 1\n",
       "checking_status                0.0         0.0\n",
       "duration                      28.0        28.0\n",
       "credit_history                 4.0         4.0\n",
       "purpose                        5.0         5.0\n",
       "credit_amount               1047.0      1047.0\n",
       "savings_status                 4.0         4.0\n",
       "employment                     4.0         4.0\n",
       "installment_commitment         3.0         3.0\n",
       "personal_status                0.0         1.0\n",
       "other_parties                  0.0         0.0\n",
       "residence_since                3.0         3.0\n",
       "property_magnitude             2.0         2.0\n",
       "age                           67.0        67.0\n",
       "other_payment_plans            0.0         0.0\n",
       "housing                        2.0         2.0\n",
       "existing_credits               2.0         2.0\n",
       "job                            1.0         1.0\n",
       "num_dependents                 2.0         2.0\n",
       "own_telephone                  0.0         0.0\n",
       "foreign_worker                 1.0         1.0"
      ]
     },
     "execution_count": 20,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "two_examples.T.round(1)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "3e1abc87-664f-4e0d-b03a-d38090f5c772",
   "metadata": {},
   "source": [
    "The only value that differs between these two exampes are `personal_status`.\n",
    "\n",
    "When we compute the predictions for these two examples, we obtain different classes, even though these two examples only differ in `personal_status`."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "id": "7e5ac9d4-4d8e-4f1d-aeb1-de6cefd589c8",
   "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>instance 0</th>\n",
       "      <th>instance 1</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>class</th>\n",
       "      <td>1.0</td>\n",
       "      <td>0.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>probability%</th>\n",
       "      <td>97.8</td>\n",
       "      <td>8.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>raw_score</th>\n",
       "      <td>3.8</td>\n",
       "      <td>-2.4</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "              instance 0  instance 1\n",
       "class                1.0         0.0\n",
       "probability%        97.8         8.0\n",
       "raw_score            3.8        -2.4"
      ]
     },
     "execution_count": 21,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "predictions = model.predict(two_examples)\n",
    "probabilities = model.predict_proba(two_examples)[:, 1]       # probability of class 1\n",
    "raw_scores = model.predict(two_examples, output_margin=True)  # pre-sigmoid raw scores\n",
    "\n",
    "outputs = pd.DataFrame(np.vstack((\n",
    "                            predictions,\n",
    "                            100*probabilities,\n",
    "                            raw_scores)).T,\n",
    "                       columns=[\"class\", \"probability%\", \"raw_score\"],\n",
    "                       index=[\"instance 0\", \"instance 1\"])\n",
    "outputs.T.round(1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d803b33c-9fa4-4564-a276-b7cd28f0b96d",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "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.11.6"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
