{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "5f192207",
   "metadata": {},
   "outputs": [],
   "source": [
    "#Pytorch imports\n",
    "\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "\n",
    "from torch.utils.data import Dataset\n",
    "from torch.utils.data import DataLoader\n",
    "from tqdm import tqdm\n",
    "from tqdm import tqdm_notebook\n",
    "\n",
    "#Package for creating Actionable Counterfactuals\n",
    "\n",
    "import dice_ml as dml\n",
    "from dice_ml.explainer_interfaces.dice_pytorch import DicePyTorch\n",
    "from dice_ml.utils.exception import UserConfigValidationException\n",
    "\n",
    "# Package for creating Adversarial Examples\n",
    "\n",
    "import art\n",
    "\n",
    "#Package for creating amicable perturbations\n",
    "\n",
    "import amicable_pertubations as ap\n",
    "\n",
    "#Miscellaneous Utilities \n",
    "\n",
    "import numpy as np\n",
    "import scipy.linalg as la\n",
    "import pandas as pd\n",
    "import random\n",
    "\n",
    "from matplotlib import pyplot as plt\n",
    "\n",
    "import math\n",
    "\n",
    "import Utils as ut\n",
    "\n",
    "random.seed(0)\n",
    "np.random.seed(0)\n",
    "torch.manual_seed(0)\n",
    "torch.cuda.manual_seed(0)\n",
    "torch.backends.cudnn.deterministic = True\n",
    "device = torch.device('cuda:0' if torch.cuda.is_available() else 'cpu')"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "ddb2cd69",
   "metadata": {},
   "source": [
    "# Adult Income"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e9e4095f",
   "metadata": {},
   "source": [
    "Get and seperate data."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "e805c570",
   "metadata": {},
   "outputs": [],
   "source": [
    "adult_income_data = pd.read_csv('data_folder/numerisized_adult_income_data.csv')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "e7265e66",
   "metadata": {},
   "outputs": [],
   "source": [
    "names = list(adult_income_data)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "0e28d7ec",
   "metadata": {},
   "outputs": [],
   "source": [
    "adult_training_dataset, adult_testing_dataset, adult_validation_dataset = ut.get_datasets_from_dataframe(adult_income_data)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "5545b57e",
   "metadata": {},
   "outputs": [],
   "source": [
    "adult_income_array = adult_income_data.to_numpy()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "02adcdeb",
   "metadata": {},
   "outputs": [],
   "source": [
    "adult_training_array, adult_testing_array, adult_validation_array = ut.even_triple_split(adult_income_array)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "3f0c1e61",
   "metadata": {},
   "outputs": [],
   "source": [
    "maxes = np.max(adult_income_array,axis = 0)[:-1]\n",
    "mins = np.min(adult_income_array,axis = 0)[:-1]\n",
    "bounds = np.vstack((mins,maxes))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "a4fb5fbc",
   "metadata": {},
   "outputs": [],
   "source": [
    "low_income = adult_testing_array[np.where(adult_testing_array[:,-1]==0)]\n",
    "low_income = low_income[:,:-1]"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "aa0fa2b9",
   "metadata": {},
   "source": [
    "Make and test models"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "b0b71856",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Basic_fully_connected_Network(\n",
       "  (layer1): Linear(in_features=27, out_features=60, bias=True)\n",
       "  (layer2): Linear(in_features=60, out_features=60, bias=True)\n",
       "  (layer3): Linear(in_features=60, out_features=60, bias=True)\n",
       "  (layer4): Linear(in_features=60, out_features=2, bias=True)\n",
       ")"
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "classifier = ut.Basic_fully_connected_Network(27,2).to(device)\n",
    "classifier.eval()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "6ba7aa89",
   "metadata": {},
   "outputs": [],
   "source": [
    "#Train neural net if not already trained\n",
    "#Training_Accuracy_History, Validation_Accuracy_History, Training_Loss_History, Validation_Loss_History= train_neural_net(classifier, 'adult_classifier.pt', 100, 1000, 64 , adult_training_dataset,adult_validation_dataset,adult_testing_dataset, new_net = True)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "07fae73b",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<All keys matched successfully>"
      ]
     },
     "execution_count": 11,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "classifier.load_state_dict(torch.load('models_folder/adult_classifier.pt'))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "c8ba6c70",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Basic_fully_connected_Network(\n",
       "  (layer1): Linear(in_features=27, out_features=60, bias=True)\n",
       "  (layer2): Linear(in_features=60, out_features=60, bias=True)\n",
       "  (layer3): Linear(in_features=60, out_features=60, bias=True)\n",
       "  (layer4): Linear(in_features=60, out_features=2, bias=True)\n",
       ")"
      ]
     },
     "execution_count": 12,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "classifier_for_softening = ut.Basic_fully_connected_Network(27,2)\n",
    "classifier_for_softening.eval()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "ac95a1fd",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<All keys matched successfully>"
      ]
     },
     "execution_count": 13,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "classifier_for_softening.load_state_dict(torch.load('models_folder/adult_classifier.pt'))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "0d9892fa",
   "metadata": {},
   "outputs": [],
   "source": [
    "soft_classifier = ut.softmaxed_network(classifier_for_softening)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "e39a03cf",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 62/62 [00:00<00:00, 114.45it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model Accuracy : 80.12108980827448\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "0.8012108980827447"
      ]
     },
     "execution_count": 15,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "ut.check_net_accuracy(classifier, adult_testing_dataset, 64, verbose = True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "1d9df6bb",
   "metadata": {},
   "outputs": [],
   "source": [
    "low_income_estimates = classifier(torch.tensor(low_income).float().to(device)).detach().cpu().numpy()\n",
    "correctly_classified = low_income_estimates[:,0]>=low_income_estimates[:,1]\n",
    "certified_low_income = low_income[correctly_classified]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "81827302",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/u4/friedbaum/Utils.py:317: VisibleDeprecationWarning: Creating an ndarray from ragged nested sequences (which is a list-or-tuple of lists-or-tuples-or ndarrays with different lengths or shapes) is deprecated. If you meant to do this, you must specify 'dtype=object' when creating the ndarray\n",
      "  difference_training_data = MIMO_Dataset(np.array([sorted_training_data[0][:,:-1],sorted_training_data[1][:,:-1]]),portion_same_class = 0.5, portion_different_class = 0.5)\n",
      "/home/u4/friedbaum/Utils.py:318: VisibleDeprecationWarning: Creating an ndarray from ragged nested sequences (which is a list-or-tuple of lists-or-tuples-or ndarrays with different lengths or shapes) is deprecated. If you meant to do this, you must specify 'dtype=object' when creating the ndarray\n",
      "  difference_testing_data = MIMO_Dataset(np.array([sorted_testing_data[0][:,:-1],sorted_testing_data[1][:,:-1]]),portion_same_class = 0.5, portion_different_class = 0.5)\n",
      "/home/u4/friedbaum/Utils.py:319: VisibleDeprecationWarning: Creating an ndarray from ragged nested sequences (which is a list-or-tuple of lists-or-tuples-or ndarrays with different lengths or shapes) is deprecated. If you meant to do this, you must specify 'dtype=object' when creating the ndarray\n",
      "  difference_validation_data = MIMO_Dataset(np.array([sorted_validation_data[0][:,:-1],sorted_validation_data[1][:,:-1]]),portion_same_class = 0.5, portion_different_class = 0.5)\n"
     ]
    }
   ],
   "source": [
    "adult_training_difference_data, adult_testing_difference_data, adult_validation_difference_data = ut.get_difference_datasets_from_dataframe(adult_income_data)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "a3f796ad",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Basic_fully_connected_Network(\n",
       "  (layer1): Linear(in_features=54, out_features=60, bias=True)\n",
       "  (layer2): Linear(in_features=60, out_features=60, bias=True)\n",
       "  (layer3): Linear(in_features=60, out_features=60, bias=True)\n",
       "  (layer4): Linear(in_features=60, out_features=2, bias=True)\n",
       ")"
      ]
     },
     "execution_count": 18,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "verifier = ut.Basic_fully_connected_Network(54,2).to(device)\n",
    "verifier.eval()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "a9222c39",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<All keys matched successfully>"
      ]
     },
     "execution_count": 19,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "verifier.load_state_dict(torch.load('models_folder/adult_verifier.pt'))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "1ff518d8",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 41/41 [00:00<00:00, 67.10it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model Accuracy : 69.08602150537635\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "0.6908602150537635"
      ]
     },
     "execution_count": 20,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "ut.check_net_accuracy(verifier, adult_testing_difference_data, 64, verbose = True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "id": "36e0964d",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 41/41 [00:00<00:00, 52.98it/s]\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "text/plain": [
       "(0.6716589861751152,\n",
       " 0.67357910906298,\n",
       " array([0.46681261, 0.16006331, 0.38324776, ..., 0.24583812, 0.5937227 ,\n",
       "        0.75084001]),\n",
       " array([0.58988523, 0.39341164, 0.44256821, ..., 0.29179969, 0.60191679,\n",
       "        0.76511586]))"
      ]
     },
     "execution_count": 21,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "ut.check_verifier(verifier,classifier,adult_testing_difference_data, 64)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "4f4812a8",
   "metadata": {},
   "source": [
    "Define data set dependent functions"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "id": "06f16aa0",
   "metadata": {},
   "outputs": [],
   "source": [
    "def ai_conditioner(x):\n",
    "    output = x.detach().clone()\n",
    "    output[0] = torch.round(output[0])\n",
    "    \n",
    "    output[1] = torch.round(output[1])\n",
    "    \n",
    "    output[2] = torch.round(output[2])\n",
    "    \n",
    "    output[3] = torch.round(output[3])\n",
    "    \n",
    "    status = torch.argmax(output[4:9])\n",
    "    output[4:9] *= 0\n",
    "    output[4+status] += 1\n",
    "    \n",
    "    employer = torch.argmax(output[9:13])\n",
    "    output[9:13] *= 0\n",
    "    output[9+employer] += 1\n",
    "    \n",
    "    education = torch.argmax(output[13:21])\n",
    "    output[13:21] *= 0\n",
    "    output[13+education] += 1\n",
    "    \n",
    "    job = torch.argmax(output[21:])\n",
    "    output[21:] *= 0\n",
    "    output[21+job] += 1\n",
    "    \n",
    "    return output"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "id": "b6c960a8",
   "metadata": {},
   "outputs": [],
   "source": [
    "def ai_distance(x,y):\n",
    "    distance = 0\n",
    "    distance += 0.1*(y[3]-x[3])**2\n",
    "    \n",
    "    status_weights = torch.tensor([[0,2,1000,1000,1000],[100,0,1,1,2],[1000,2,0,1000,1000],[1000,3,2,0,2],[1000,2,1000,1000,0]]).double()    \n",
    "    distance += torch.dot(torch.mv(status_weights,y[4:9]),x[4:9])\n",
    "    \n",
    "    employer_weights  = 1-torch.eye(4).double()\n",
    "    distance += torch.dot(torch.mv(employer_weights,y[9:13]),x[9:13])\n",
    "    \n",
    "    edu_weights = torch.tensor([[0,2,10,3,4,6,8,11],[1000,0,8,1,2,4,6,9],[1000,1000,0,1000,1000,1000,2,5],[1000,1000,7,0,1,3,5,8],[1000,1000,6,1000,0,2,4,7],[1000,1000,4,1000,1000,0,2,5],[1000,1000,4,1000,1000,1000,0,3],[1000,1000,4,1000,1000,1000,1000,0]]).double()\n",
    "    distance += torch.dot(torch.mv(edu_weights,y[13:21]),x[13:21])\n",
    "    \n",
    "    field_weights = torch.tensor([[0,1,2,3,4,1],[1,0,1,2,3,1],[1,1,0,1,2,1],[1,1,1,0,1,1],[1,1,1,1,0,1],[1,1,1,1,1,0]]).double()\n",
    "    distance += torch.dot(torch.mv(field_weights,y[21:27]),x[21:27])\n",
    "    \n",
    "    return distance"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "id": "1208e126",
   "metadata": {},
   "outputs": [],
   "source": [
    "def ai_interpretor(x):\n",
    "    print('Age: '+str(x[0]))\n",
    "    \n",
    "    if x[1] == 1:\n",
    "        print('Race: White')\n",
    "    else:\n",
    "        print('Race: Not White')\n",
    "        \n",
    "    if x[2] == 1:\n",
    "        print('Sex: Male')\n",
    "    else:\n",
    "        print('Sex: Female')\n",
    "        \n",
    "    print('Hours worked per week: '+str(x[3]))\n",
    "        \n",
    "    statuses = ['Single','Married','Widowed','Seperated','Divorced']\n",
    "    status = np.argmax(x[4:9])\n",
    "    print('Marital Status: '+statuses[status])\n",
    "    \n",
    "    \n",
    "    employers = ['Government','Private', 'Self-Employed', 'Other/Unknown']\n",
    "    employer = np.argmax(x[9:13])\n",
    "    print('Employer: '+employers[employer])\n",
    "    \n",
    "    educations = ['School','HS-grad','Prof-school','Some-college','Assoc','Bachelors','Masters','Doctorate']\n",
    "    education = np.argmax(x[13:21])\n",
    "    print('Education: '+educations[education])\n",
    "    \n",
    "    jobs = ['Service','Sales','Blue-Collar','White-Collar','Professional','Other/Unknown_Occupation']\n",
    "    job = np.argmax(x[21:27])\n",
    "    print('Occupation: '+jobs[job])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "id": "9a054c0f",
   "metadata": {},
   "outputs": [],
   "source": [
    "def ai__change_interpretor(x,y):\n",
    "    age_change =  y[0]-x[0]    \n",
    "    if age_change > 0:\n",
    "        print('Age increased by '+str(age_change)+'.')\n",
    "    elif age_change < 0:\n",
    "        print('Age decreased by '+str(age_change*-1)+'.')\n",
    "    \n",
    "    race_change = y[1]-x[1]\n",
    "    if race_change ==1:\n",
    "        print('Changed from non-white to white.')\n",
    "    elif race_change ==-1:\n",
    "        print('Changed from white to non-white.')\n",
    "        \n",
    "    race_change = y[2]-x[2]\n",
    "    if race_change ==1:\n",
    "        print('Changed from female to male.')\n",
    "    elif race_change ==-1:\n",
    "        print('Changed from male to female.')\n",
    "        \n",
    "    hours_change =  y[3]-x[3]    \n",
    "    if hours_change > 0:\n",
    "        print('Hours worked increased by '+str(hours_change)+'.')\n",
    "    elif hours_change < 0:\n",
    "        print('Hours worked decreased by '+str(hours_change*-1)+'.')\n",
    "           \n",
    "    statuses = ['Single','Married','Widowed','Seperated','Divorced']\n",
    "    org_status = np.argmax(x[4:9])\n",
    "    new_status = np.argmax(y[4:9])\n",
    "    if np.abs(org_status - new_status)>.5:\n",
    "        print('Marital Status changed from '+statuses[org_status]+' to '+statuses[new_status]+'.')\n",
    "\n",
    "        \n",
    "    employers = ['Government','Private', 'Self-Employed', 'Other/Unknown']\n",
    "    org_employer = np.argmax(x[9:13])\n",
    "    new_employer = np.argmax(y[9:13])\n",
    "    if np.abs(org_employer - new_employer)>.5:\n",
    "        print('Employer type from '+employers[org_employer]+' to '+employers[new_employer]+'.')\n",
    "\n",
    "        \n",
    "    educations = ['School','HS-grad','Prof-school','Some-college','Assoc','Bachelors','Masters','Doctorate']\n",
    "    org_education = np.argmax(x[13:21])\n",
    "    new_education = np.argmax(y[13:21])\n",
    "    if np.abs(org_education - new_education)>.5:\n",
    "        print('Education level changed from '+educations[org_education]+' to '+educations[new_education]+'.')\n",
    "\n",
    "        \n",
    "    jobs = ['Service','Sales','Blue-Collar','White-Collar','Professional','Other/Unknown_Occupation']\n",
    "    org_job = np.argmax(x[21:27])\n",
    "    new_job = np.argmax(y[21:27])\n",
    "    if np.abs(org_job - new_job)>.5:\n",
    "        print('Job type changed from '+jobs[org_job]+' to '+jobs[new_job]+'.')\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "id": "a6ff9028",
   "metadata": {},
   "outputs": [],
   "source": [
    "immutables = torch.tensor([0,1,2,4,5,6,7,8])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "id": "37b368a9",
   "metadata": {},
   "outputs": [],
   "source": [
    "categoricals = [(4,8),(9,12),(13,20),(21,26)]"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "9e2f922b",
   "metadata": {},
   "source": [
    "Set up amicable perturbation creator"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "id": "0f860f7f",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "None\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/u4/friedbaum/amicable_pertubations.py:130: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  self.immutable = torch.tensor(immutable_indices)\n"
     ]
    }
   ],
   "source": [
    "adult_income_ap_creator = ap.AP_generator_Continuous(classifier.cpu().double(),verifier.cpu().double(),ai_distance,27,2,0.9,None,np.array([1]),np.array([0]),bounds,immutables,ai_conditioner,categoricals,initial_balance_param = 0,categorical_penalty=1,max_iters = 10000,learning_rate = 0.01,bound_penalty = 1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "id": "e04fc993",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Basic_fully_connected_Network(\n",
       "  (layer1): Linear(in_features=54, out_features=60, bias=True)\n",
       "  (layer2): Linear(in_features=60, out_features=60, bias=True)\n",
       "  (layer3): Linear(in_features=60, out_features=60, bias=True)\n",
       "  (layer4): Linear(in_features=60, out_features=2, bias=True)\n",
       ")"
      ]
     },
     "execution_count": 29,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "verifier.to(device)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "id": "46245901",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Using lambda 0.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/u4/friedbaum/amicable_pertubations.py:607: UserWarning: Implicit dimension choice for softmax has been deprecated. Change the call to include dim=X as an argument.\n",
      "  original_probs = torch.nn.functional.softmax(classifier(torch.tensor(x))).detach().cpu().numpy()\n",
      "/home/u4/friedbaum/amicable_pertubations.py:421: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  original = torch.tensor(x,dtype= float)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Using lambda 0.05\n",
      "Using lambda 0.1\n",
      "Using lambda 0.15000000000000002\n",
      "Using lambda 0.2\n",
      "Using lambda 0.25\n",
      "Using lambda 0.30000000000000004\n",
      "Using lambda 0.35000000000000003\n",
      "Using lambda 0.4\n",
      "Using lambda 0.45\n",
      "Using lambda 0.5\n",
      "Using lambda 0.55\n",
      "Using lambda 0.6000000000000001\n",
      "Using lambda 0.65\n",
      "Using lambda 0.7000000000000001\n",
      "Using lambda 0.75\n",
      "Using lambda 0.8\n",
      "Using lambda 0.8500000000000001\n",
      "Using lambda 0.9\n",
      "Using lambda 0.9500000000000001\n",
      "Using lambda 1.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/u4/friedbaum/amicable_pertubations.py:639: UserWarning: Implicit dimension choice for softmax has been deprecated. Change the call to include dim=X as an argument.\n",
      "  verification = torch.nn.functional.softmax(verifier(torch.tensor(np.concatenate((x,aps[idx]))).to(device).double()))\n",
      "/home/u4/friedbaum/amicable_pertubations.py:643: UserWarning: Implicit dimension choice for softmax has been deprecated. Change the call to include dim=X as an argument.\n",
      "  original_delta = creator.calc_wanted_PD(torch.nn.functional.softmax(classifier(torch.tensor(x))),shrinkage=1).item()\n"
     ]
    }
   ],
   "source": [
    "\n",
    "aps,probs,deltas,epsilons,ver,discrep = ap.create_pareto_graph(adult_income_ap_creator,classifier,verifier,low_income[0],0,1,21,2000, 1, np.array([]), np.array([]), np.array([]))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "id": "27d29314",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.collections.PathCollection at 0x7fb610038668>"
      ]
     },
     "execution_count": 31,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.scatter(epsilons,deltas)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "41554d30",
   "metadata": {},
   "source": [
    "Set up Counterfactual creator for comparison"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "id": "a367f9be",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_dice_data = dml.Data(dataframe=adult_income_data, continuous_features=['age', 'hours_per_week'], outcome_name='income')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "id": "48714159",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "softmaxed_network(\n",
       "  (classifier): Basic_fully_connected_Network(\n",
       "    (layer1): Linear(in_features=27, out_features=60, bias=True)\n",
       "    (layer2): Linear(in_features=60, out_features=60, bias=True)\n",
       "    (layer3): Linear(in_features=60, out_features=60, bias=True)\n",
       "    (layer4): Linear(in_features=60, out_features=2, bias=True)\n",
       "  )\n",
       "  (softener): Softmax(dim=-1)\n",
       ")"
      ]
     },
     "execution_count": 33,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "soft_classifier.cpu()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "id": "4c324c67",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_dice_model = dml.Model(soft_classifier, backend='PYT')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "id": "1f6e66db",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_dice_explainer = DicePyTorch(my_dice_data,my_dice_model)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 36,
   "id": "e9058cec",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_variables = [\n",
    " 'hours_per_week',\n",
    " 'Government',\n",
    " 'Private',\n",
    " 'Self-Employed',\n",
    " 'Other/Unknown_Work',\n",
    " 'School',\n",
    " 'HS-grad',\n",
    " 'Prof-school',\n",
    " 'Some-college',\n",
    " 'Assoc',\n",
    " 'Bachelors',\n",
    " 'Masters',\n",
    " 'Doctorate',\n",
    " 'Service',\n",
    " 'Sales',\n",
    " 'Blue-Collar',\n",
    " 'White-Collar',\n",
    " 'Professional',\n",
    " 'Other/Unknown_Occupation']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 37,
   "id": "8ab2fb67",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 1/1 [00:03<00:00,  3.29s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Diverse Counterfactuals found! total time taken: 00 min 03 sec\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "<dice_ml.counterfactual_explanations.CounterfactualExplanations at 0x7fb5fbd34588>"
      ]
     },
     "execution_count": 37,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "my_dice_explainer.generate_counterfactuals(adult_income_data.drop(columns=\"income\")[1:2], total_CFs=1, desired_class=\"opposite\",features_to_vary=my_variables)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 38,
   "id": "6aa816ed",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_categoricals = [[4,5,6,7,8],[9,10,11,12],[13,14,15,16,17,18,19,20],[21,22,23,24,25,26]]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 39,
   "id": "119c81a3",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_dice_explainer.encoded_categorical_feature_indexes = my_categoricals"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 40,
   "id": "d7f5dd59",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_dice_explainer.encoded_continuous_feature_indexes = [0,1,2,3]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 41,
   "id": "eca8a3b2",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_dice_explainer.cont_minx = [17,0,0,1]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 42,
   "id": "da05be41",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_dice_explainer.cont_maxx = [90,1,1,99]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 43,
   "id": "870ffb67",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_dice_explainer.minx[0,0] = 17\n",
    "my_dice_explainer.minx[0,3] = 1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 44,
   "id": "570c136b",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_dice_explainer.maxx[0,0] = 90\n",
    "my_dice_explainer.maxx[0,3] = 99"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 45,
   "id": "f4f10c48",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_dice_explainer.cont_precisions = [0,0,0,0]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 46,
   "id": "87e3fd1e",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_dice_explainer.feature_weights_list[0] /= 73\n",
    "my_dice_explainer.feature_weights_list[3] /= 98"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 47,
   "id": "59d24530",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "softmaxed_network(\n",
       "  (classifier): Basic_fully_connected_Network(\n",
       "    (layer1): Linear(in_features=27, out_features=60, bias=True)\n",
       "    (layer2): Linear(in_features=60, out_features=60, bias=True)\n",
       "    (layer3): Linear(in_features=60, out_features=60, bias=True)\n",
       "    (layer4): Linear(in_features=60, out_features=2, bias=True)\n",
       "  )\n",
       "  (softener): Softmax(dim=-1)\n",
       ")"
      ]
     },
     "execution_count": 47,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "soft_classifier.float()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "b8846f43",
   "metadata": {},
   "source": [
    "Test Individual and save results"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 48,
   "id": "4360606f",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Using lambda 0.0\n",
      "Using lambda 0.05\n",
      "Using lambda 0.1\n",
      "Using lambda 0.15000000000000002\n",
      "Using lambda 0.2\n",
      "Using lambda 0.25\n",
      "Using lambda 0.30000000000000004\n",
      "Using lambda 0.35000000000000003\n",
      "Using lambda 0.4\n",
      "Using lambda 0.45\n",
      "Using lambda 0.5\n",
      "Using lambda 0.55\n",
      "Using lambda 0.6000000000000001\n",
      "Using lambda 0.65\n",
      "Using lambda 0.7000000000000001\n",
      "Using lambda 0.75\n",
      "Using lambda 0.8\n",
      "Using lambda 0.8500000000000001\n",
      "Using lambda 0.9\n",
      "Using lambda 0.9500000000000001\n",
      "Using lambda 1.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 1/1 [00:02<00:00,  2.18s/it]\n",
      "/home/u4/friedbaum/amicable_pertubations.py:211: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  prediction = torch.nn.functional.softmax(self.classifier(torch.tensor(input)),dim = 0)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Diverse Counterfactuals found! total time taken: 00 min 02 sec\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 1/1 [00:15<00:00, 15.80s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Diverse Counterfactuals found! total time taken: 00 min 15 sec\n",
      "Age: 30.0\n",
      "Race: White\n",
      "Sex: Male\n",
      "Hours worked per week: 50.0\n",
      "Marital Status: Single\n",
      "Employer: Private\n",
      "Education: Bachelors\n",
      "Occupation: Professional\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAXMAAAEWCAYAAACUg3d7AAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjMuNCwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8QVMy6AAAACXBIWXMAAAsTAAALEwEAmpwYAAAyr0lEQVR4nO3deZwU1bXA8d+ZhZkBRtmNCgIqKgMzDMgmqIAswSWgIm64YBJREeNLkCcGY0gkiuKCJhqDSxBFBElAjCZqAsQVFXDYRcGHLC4MyD7ALH3eH1XdNkPPTM3WXdN9vp9Pf+iuqr51qrrncPvWrXtFVTHGGFO3JcU6AGOMMdVnydwYY+KAJXNjjIkDlsyNMSYOWDI3xpg4YMncGGPigCXzKhKRNiKiIpIS61gShYhMFJEX3edRP/8ico6IrI/W/uKdiCwWkZ/HOo54kRDJXEQ2ichBEdknIrtF5AMRuVlEfHn8ItLXTVTzSi3v5C5fHKPQIhKRX4vI/4nIfhHZKiKzYx1TZbn/URS535F9IvK5iPxJRI4PbqOq76rq6bGM009EZLyIvBNheTMRKRSRjrGIK1H5MpnVkp+oaibQGpgM3Ak8W5WCarI2WE5Z+cBZItI0bNn1wOc1te+aICLXA9cCA1S1IdAV+E9soypfOed8tvsdaQJcAvwIWBae0KMUR13xItBLRNqWWn4lsEpVV8cgpoSVSMkcAFXdo6oLgCuA64O1BxG5UEQ+FZG9IrJFRCYG3xP2k/5nIrIZWFi6XBEZ5v4C6CgiSW6tZaOI7BSROSLSxGtZrkJgPs4fBiKS7MY8s9R+zxCRt0XkexFZLyKXh63zckzXi8hmEdkhIhPC1ncXkaXue78TkUfKiLMb8KaqbnTP77eqOi2snMUiMsn9NbRfRF4TkaYiMtMt+xMRaRO2/WNurHtFZJmInFPGfo8gIseKyLMi8o2IbHP3meyuGyki74vIoyKyE5hYXlmqWqSqa3DOdz4w1i2nr4hsDdvnne6+9rnnvr+7PNn9tbLRXbdMRFq561REbhWRL4Av3GUXiUhe2K/GnLB9bBKRu0RkrYjsEpG/iki6u66xiPxDRPLddf8QkZalzv297rHvE5G3RKRZ2Pqz3f3tds/5SBHp5n7eyWHbXSoiKyKcp604399rS626DphRUXylPr9QE5r7+ohmtPI+X+NS1bh/AJtwao6ll28GbnGf9wWycf6DywG+Ay5217UBFJgBNAAywpalADcAG4BT3e1vB5YALYE04C/ArLLKihBXX2Ar0Av4yF12AfAm8HNgsbusAbDF3X8K0BnYAWRV4piedo+nE3AYaO+u/xC41n3eEOhZxrm9BvgeGIdTK08utX6xe25OAY4F1uL8uhjgxjwD+Gup8pq668YC3wLp7rqJwIul4k9xX89zz3MDoAXwMXCTu24kUAzc5pYb6ZyHyi61/Pdhn0FfYKv7/HT33J8QFs8p7vNxwCp3G3HPbVN3nQJv49T+M9zPbDvQA0jG+fW1CUgL++6uBlq573kfmOSuawoMA+oDmcArwPxS534jcJq7r8XAZHdda2AfcBWQ6paV665bC5wfVs48YGwZn/8I4Iuw16fjVESae4zv55HOf2U+X3u45yzWAUTlIMtO5kuACWW8ZyrwqPs8+MU6OWx9cNkd7pe/Zdi6dUD/sNfHA0U4ieSosiLsuy8/JI0v3D+Ql90/nPBkfgXwbqn3/gX4bSWOKTzuj4Er3efvAL8Dmnk4vyOAfwMHgJ3AnWHrFoefY+Bh4J9hr38C5JVT9i6gk/s89AfPkf+ZHofzH1FG2PuuAha5z0cCmys4hiOSSdjym3GTVanP5VScJDwASC31nvXA0DL2o8B5Ya//DNwb4f19wr67N4etuwDYWEbZucCuUuf+7rDXo4F/uc/vAuaVUc6dwEz3eROgADi+jG3rA3uBXu7rPwCvViK+CpN5RZ+vPZxHwjWzlHIiTq0SEekhIovcn4R7cP6Im5XafkuEMsYBT6jzkzOoNTDP/fm6Gye5l+B8KcsrK5IXgDFAP5zaSbjWQI/gftx9jcBp6/V6TN+GPS/AqYUD/AynRveZ2xRyUVkBqupMVR0ANHL3ca+I/Dhsk+/Cnh+M8Dq4T0TkDhFZJyJ73OM5NkLMpbXGqV1+E3Ye/oJTgwvyer5LC31HwqnqBuB/cJLQdhF5WUROcFe3wqkRlyU8ltbA2FKfYSvghDK2/yq4TkTqi8hfROQrEdmL8x9wo1LND2V9vuXF+CLwExFpAFyOU2H4JtKGqlqAU+O+TkQE5/s3oxLxeeHl8014CZvMRaQbzh/qe+6il4AFQCtVPRZ4CucncrhIQ0wOAu4WkWFhy7bg/ExtFPZIV9VtFZQVyQs4Nao33D+ccFuA/5baT0NVvaUSxxSRqn6hqlfh/ME8AMx1/7jLe0+Rqr4CrAQq3ZPBbR//X5wE0lhVGwF7PMS8Bafm1izsPByjqh3Cw6tCPEk4vxzejbReVV9S1bNxko3inKdgPKeUU3R4LFuAP5T6DOur6qywbVqFPT8J+Np9PhbnV1sPVT0GODcYesVHV3aM7vf0Q+BSnPbwFyoo63mcz2wgTnPKa1WI7wBOLT/oR6VirejzTXgJl8xF5Bi3lvkyzs+6Ve6qTOB7VT0kIt2Bqz0WuQYYDDwhIkPcZU8BfxCR1u4+m4vI0KrEq6r/B/QBJkRY/Q/gNBG5VkRS3Uc3EWlfzWNCRK4RkeaqGgB2u4sDEbYbKc6F1kxxLvyeD3QAPvJ+lCGZOG3b+UCKiNwDHFPRm9xa41vAw+7nmyQip4hInyrEgIikuOdwFk5SOerir4icLiLniUgacAjnF0bw/DyD8+uknThy5MheSeGeBm52f0WJiDQIns+wbW4VkZbiXESfAAS7fma6+93trvttJQ5zJjBARC53j7epiOSGrZ+B8x9rNvD3Csp6F+c7Mg14WVULqxBfHnCuiJwkIsfiNAMBNf/5xqtESuavicg+nP/lJ+D8gd4Qtn408Ht3m3uAOV4LVtUVwEXA024yewynRvyWW94SnAtcVaKq76nq1xGW78P5ZXAlTm3tW5zaYVp1jwnnP6g1IrIf53iuVNWDEbbbC/wa52LybuBBnIvK70XYtiJvAv/CuUD6FU6S9No8ch1QD+f6xS5gLs61isq4wj3ePTif307gzEjnHuccT8a54Pwtzi+YYAJ6BOdcv4Vzfp7FuQB5FFVdCtwI/MmNewNOG3+4l9yyvsRpGpnkLp/qlrsD5zv2L68Hqqqbcdrfx+I0I+XhXKgNmofbXBjhF2HpshQn+bd2/w3yHJ+qvo3zn9RKYBlORSVcTXy+cU2cz8EY40cisgnnIuG/Y7DvjTg9RqK+b1N5iVQzN8Z45F4DUsq+D8L4TF2/A80YU8PEGS4iC+c+g6Oukxh/smYWY4yJA9bMYowxcaDONrM0a9ZM27RpE+swjDF1wLJly3aoavPqlDH4vEa6Y2ext/2tPPCmqg6uzv4qq84m8zZt2rB06dJYh2GMqQNE5KvqlrFjZzGfvJXtadukHy2p6K7lGldnk7kxxkSXQsC/14MtmRtjjFeWzI0xpo5T8HPnP0vmJqqKiorYunUrhw4dinUoJg6lp6fTsmVLUlNTa6V8Vf92AEyIZK7FmyCwG1JOQ5LqV7S5qUVbt24lMzOTNm3a4IyYakzNUFV27tzJ1q1badu29Ex2NVA+Ysk8VrQkH3bfAcVfgaSAFqMNRyMNLq/4zaZWHDp0yBK5qRUiQtOmTcnPz6+1fQQsmcfI7jugeAMQAD3sLNv/ZzT1ZKRe15iGlsgskZvaUrvfLauZx4QWb4LizRw9BPchOPAyWDI3xlSCKgQC/p1D2r//zVRXYA+UNTtV4KhZwEyCmT9/PiLCZ599Vun3PvXUU8yYMaPiDUvZtGkTHTtGnoCpb9++lboJbuTIkbRt25bc3Fy6dOnChx9+WKlYpk6dSkFBucOURzRx4kQeeughz9vv3r2bJ598MvT666+/5rLLLqv0fv3BqZl7ecRC/Cbz1NNASyKsqAdpZ0c9HOMvs2bN4uyzz2bWrFkVb1zKzTffzHXXXVcLUVXOlClTyMvLY/Lkydx0002e31dSUlKlZF5c7O1W9nClk/kJJ5zA3LlzK12OXwRUPD1iIW6TuUgGZN4KpIctrQdJTaD+8FiFZSpp/qfb6D15IW3Hv07vyQuZ/+m2it9Ugf379/Pee+/x7LPP8vLLL4eWL168mD59+jB06FBOPvlkxo8fz8yZM+nevTvZ2dls3OjMfxxeO92wYQMDBgygU6dOdOnShY0bN7J//3769+9Ply5dyM7O5tVXXw3to7i4mBEjRtC+fXsuu+yyiAn1rbfe4qyzzqJLly4MHz6c/fv3l3s85557Lhs2bADgxRdfpHv37uTm5nLTTTdRUuJUaBo2bMjYsWPp1KkTf/jDH/j666/p168f/fr1C60Pmjt3LiNHjgScXwA333wzPXr04H//938BWLFiBWeddRbt2rXj6aefDp3TSMc8fvx4Nm7cSG5uLuPGjTvi18mhQ4e44YYbyM7OpnPnzixatAiA6dOnc+mllzJ48GDatWsX2m9JSQkjR46kY8eOZGdn8+ijj5b/QdcCP9fM47bNHEDqX4amnAwFL0PJ95DWC+oPR5IyK36zibn5n27jrr+v4mCRk5C27T7IXX93pmy9uPOJVS731VdfZfDgwZx22mk0bdqUZcuWceaZZwJOolq3bh1NmjTh5JNP5uc//zkff/wxjz32GH/84x+ZOnXqEWWNGDGC8ePHc8kll3Do0CECgQD16tVj3rx5HHPMMezYsYOePXsyZIgzPez69et59tln6d27Nz/96U958sknueOOO0Ll7dixg0mTJvHvf/+bBg0a8MADD/DII49wzz33lHk8r732GtnZ2axbt47Zs2fz/vvvk5qayujRo5k5cybXXXcdBw4coEePHjz88MMAPPfccyxatIhmzSoeQmTr1q188MEHJCcnM3HiRFauXMmSJUs4cOAAnTt35sILL6RFixYRj3ny5MmsXr2avLw8wGlqCnriiScQEVatWsVnn33GoEGD+PzzzwHIy8vj008/JS0tjdNPP53bbruN7du3s23bNlavXg04tf5oUgWNUa3bi7hO5gBSrwvU6xLrMEwVTHlzfSiRBx0sKmHKm+urlcxnzZrF7bffDsCVV17JrFmzQsm8W7duHH+8M7XkKaecwqBBgwDIzs4O1RyD9u3bx7Zt27jkkksA54YVcG6M+vWvf80777xDUlIS27Zt47vvvgOgVatW9O7dG4BrrrmGxx9//IhkvmTJEtauXRvaprCwkLPOOivicYwbN45JkybRvHlznn32Wf7zn/+wbNkyunXr5pyrgwdp0aIFAMnJyQwbNqxK52v48OEkJ/9w/Wno0KFkZGSQkZFBv379+Pjjj7nwwgvLPOayvPfee9x2220AnHHGGbRu3TqUzPv378+xxx4LQFZWFl999RUdOnTgyy+/5LbbbuPCCy8MfTbRI76+ABr3yTxIA3tAiyGpiXWNqyO+3h1p/uiyl3vx/fffs3DhQlatWoWIUFJSgogwZcoUANLS0kLbJiUlhV4nJSV5bjOeOXMm+fn5LFu2jNTUVNq0aRO647X0d6/0a1Vl4MCBntryp0yZcsTFxEWLFnH99ddz//33H7Vtenr6EQm5tPA4St+d26BBg3JjFpFyj7kqwj+H5ORkiouLady4MStWrODNN9/kqaeeYs6cOTz33HNV3kdV+LlmHrdt5kFash39/hbI/wnsuBR2XokWrYl1WMaDExpFnNC+zOVezJ07l2uvvZavvvqKTZs2sWXLFtq2bcu7775b6bIyMzNp2bIl8+fPB+Dw4cMUFBSwZ88eWrRoQWpqKosWLeKrr34YfXXz5s2hnicvvfQSZ5995MX4nj178v7774fawA8cOBCqrVakf//+zJ07l+3btwPOf1zh+y4d+759+0KvjzvuONatW0cgEGDe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\n",
      "text/plain": [
       "<Figure size 432x288 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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cM899xy1bXFxsecued9++20aN24c01jfeecdpkyZwnvvvceKFStYtGhRqL+YWDqWZB8IBJgxYwZ33303y5Yto27dyt3VXNVkHxcCPo9TIrJkbxLevoIiBj32T/YVVP2y1/z588nMzOSaa64BnH5XHn/8cZ5//nkOHTrE9OnTGTZsGOeeey4DBw4sVYI8dOgQl112GZ07d+biiy+mT58+oQFHsrOzyc/PZ/PmzXTq1InrrruOLl26MHjw4FCp95lnnqF3795069aNESNGVNif/IMPPsgjjzxC69atAae/mOuuuw5weoXs27cvubm5XHzxxXz33XdA6UFQ8vPzyc7OBsrvNnjChAkcPnyY7t27M3r0aMBbN8kPPvggs2fP5te//jWjR4+O2q3ziy++SG5uLt26deOqq67ik08+Yd68eYwfP57u3buzcePGcuPevHkzP/jBD+jZsyc9e/bkk08+OZY/e8z4UE9TIrJkbxLegrU72LDjAAvW7qjyvlatWhXq4bJEw4YNadeuXag/mqVLlzJnzhz++c9/ltruqaee4rjjjmP16tX87ne/Y8mSJRGPsX79em655RZWrVpF48aNef311wG45JJL+Oyzz1i+fDmdOnXiueeeixrrypUrj4q1xNVXX81DDz3EF198QU5ODr/97W8rfO3Lli1j1qxZrFixglmzZrFlyxYmTZpE3bp1WbZsGTNmzCjVTfKyZcvw+/2h/mtKuklevnw59957L8OGDWPy5MnMmDGDzMxM3njjDZYuXcqCBQu48847UVVWrVrFAw88wPz581m+fDlPPPEEZ555Zui5y5Yt4+STTy435hYtWvD++++zdOlSZs2axW233Vbh66wuQnKX7O0CrUlYt838nPdXf0tRwBlx7M7Zy5nw+grO69ySJy/vUW3HPe+882jSpMlRyz/66KNQ18hdu3YlNzfyiGclwwUCnH766aE+3FeuXMm9997Lnj17OHDgAD/60Y+OKb69e/eyZ88ezjnnHADGjBnDyJEjK3xepG6D27YtPY71sXaTrKoRuzieP38+I0eODPWbH+m8RlNUVMS4ceNCXzxeO4WrLomayL2wZG8S1h3ndWT1tn3kfXeI4qCS5hfaHFeXOwd3POZ9du7c+ag6+H379vH1119zyimnsHTp0qO69K2sst3zllTjjB07lrlz59KtWzemT5/OwoUL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\n",
      "text/plain": [
       "<Figure size 432x288 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Amicable Perturbations \n",
      "\n",
      "\n",
      "Distance to target: 0.1880311369895935\n",
      "Cost :6.1\n",
      "Dicrepancy value : 0.052569652883960316\n",
      "Verification Confidence: 0.11485901208508642\n",
      "Straight verification value: 0.4567151665687561\n",
      "Hours worked increased by 1.0.\n",
      "Employer type from Private to Self-Employed.\n",
      "Education level changed from Bachelors to Doctorate.\n",
      "\n",
      "Distance to target: 0.10499143600463867\n",
      "Cost :5.1\n",
      "Dicrepancy value : 0.07179035649082621\n",
      "Verification Confidence: 0.024723525644651467\n",
      "Straight verification value: 0.4408366084098816\n",
      "Hours worked increased by 1.0.\n",
      "Education level changed from Bachelors to Doctorate.\n",
      "\n",
      "Distance to target: 0.059387341141700745\n",
      "Cost :5.1\n",
      "Dicrepancy value : 0.0739477068604728\n",
      "Verification Confidence: -0.015894249789719483\n",
      "Straight verification value: 0.4409995675086975\n",
      "Hours worked increased by 1.0.\n",
      "Employer type from Private to Self-Employed.\n",
      "Education level changed from Bachelors to Prof-school.\n",
      "\n",
      "Distance to target: 0.06286952644586563\n",
      "Cost :7.1\n",
      "Dicrepancy value : 0.0815894363166656\n",
      "Verification Confidence: -0.02062069648476844\n",
      "Straight verification value: 0.4331592321395874\n",
      "Hours worked increased by 1.0.\n",
      "Employer type from Private to Self-Employed.\n",
      "Education level changed from Bachelors to Doctorate.\n",
      "Job type changed from Professional to White-Collar.\n",
      "\n",
      "Distance to target: 0.06831606477499008\n",
      "Cost :7.4\n",
      "Dicrepancy value : 0.08381119236216605\n",
      "Verification Confidence: -0.018273698291662532\n",
      "Straight verification value: 0.4306357502937317\n",
      "Hours worked increased by 2.0.\n",
      "Employer type from Private to Self-Employed.\n",
      "Education level changed from Bachelors to Doctorate.\n",
      "Job type changed from Professional to White-Collar.\n",
      "\n",
      "Distance to target: 0.041091591119766235\n",
      "Cost :7.6\n",
      "Dicrepancy value : 0.12763886944325264\n",
      "Verification Confidence: -0.08480948025136492\n",
      "Straight verification value: 0.38844501972198486\n",
      "Hours worked increased by 4.0.\n",
      "Employer type from Private to Self-Employed.\n",
      "Education level changed from Bachelors to Prof-school.\n",
      "Job type changed from Professional to White-Collar.\n",
      "\n",
      "Distance to target: 0.04277769476175308\n",
      "Cost :6.9\n",
      "Dicrepancy value : 0.129140501586118\n",
      "Verification Confidence: -0.08491601135641719\n",
      "Straight verification value: 0.38683071732521057\n",
      "Hours worked increased by 3.0.\n",
      "Employer type from Private to Self-Employed.\n",
      "Education level changed from Bachelors to Prof-school.\n",
      "Job type changed from Professional to White-Collar.\n",
      "\n",
      "Distance to target: 0.048589058220386505\n",
      "Cost :6.1\n",
      "Dicrepancy value : 0.12970216118895905\n",
      "Verification Confidence: -0.08065559182610865\n",
      "Straight verification value: 0.38589441776275635\n",
      "Hours worked increased by 1.0.\n",
      "Employer type from Private to Self-Employed.\n",
      "Education level changed from Bachelors to Prof-school.\n",
      "Job type changed from Professional to White-Collar.\n",
      "\n",
      "Distance to target: 0.044936999678611755\n",
      "Cost :6.4\n",
      "Dicrepancy value : 0.13056636206181693\n",
      "Verification Confidence: -0.0845524809126314\n",
      "Straight verification value: 0.3852633237838745\n",
      "Hours worked increased by 2.0.\n",
      "Employer type from Private to Self-Employed.\n",
      "Education level changed from Bachelors to Prof-school.\n",
      "Job type changed from Professional to White-Collar.\n",
      "\n",
      "Distance to target: 0.3932616114616394\n",
      "Cost :2.0\n",
      "Dicrepancy value : 0.4772324930935241\n",
      "Verification Confidence: -0.13258160257848753\n",
      "Straight verification value: 0.025750473141670227\n",
      "Education level changed from Bachelors to Masters.\n",
      "\n",
      "Distance to target: 0.5703204870223999\n",
      "Cost :0.0\n",
      "Dicrepancy value : 0.4984707134175288\n",
      "Verification Confidence: 0.00013110654253978282\n",
      "Straight verification value: 0.00013110654253978282\n",
      "\n",
      "Distance to target: 0.5458793640136719\n",
      "Cost :0.1\n",
      "Dicrepancy value : 0.4990085503648949\n",
      "Verification Confidence: -0.021696428661933057\n",
      "Straight verification value: 0.00015965579950716347\n",
      "Hours worked increased by 1.0.\n",
      "\n",
      "\n",
      "Single Counterfactual\n",
      "\n",
      "\n",
      "Distance to target: 0.10455541\n",
      "Cost :tensor(5., dtype=torch.float64)\n",
      "Dicrepancy value : 0.06746781\n",
      "Verification Confidence: 0.028676142394542692\n",
      "Straight verification value: 0.44517913\n",
      "Education level changed from Bachelors to Doctorate.\n",
      "\n",
      "\n",
      "Diverse Counterfactuals\n",
      "\n",
      "\n",
      "Distance to target: 0.1526663899421692\n",
      "Cost :24.6\n",
      "Dicrepancy value : 0.0014817715\n",
      "Verification Confidence: 0.13861666435885295\n",
      "Straight verification value: 0.5120972\n",
      "Hours worked decreased by 14.0.\n",
      "Education level changed from Bachelors to Doctorate.\n",
      "\n",
      "Distance to target: 0.07770955562591553\n",
      "Cost :7.6\n",
      "Dicrepancy value : 0.032457262\n",
      "Verification Confidence: 0.10589674026869395\n",
      "Straight verification value: 0.5464062\n",
      "Hours worked increased by 4.0.\n",
      "Employer type from Private to Self-Employed.\n",
      "Education level changed from Bachelors to Prof-school.\n",
      "Job type changed from Professional to Sales.\n",
      "\n",
      "Distance to target: 0.1366870403289795\n",
      "Cost :13.1\n",
      "Dicrepancy value : 0.06256494\n",
      "Verification Confidence: 0.06092222967172983\n",
      "Straight verification value: 0.44869202\n",
      "Hours worked increased by 9.0.\n",
      "Education level changed from Bachelors to Doctorate.\n",
      "\n",
      "Distance to target: 0.0466897189617157\n",
      "Cost :8.6\n",
      "Dicrepancy value : 0.0668838\n",
      "Verification Confidence: -0.019415266074836457\n",
      "Straight verification value: 0.4488332\n",
      "Hours worked decreased by 4.0.\n",
      "Employer type from Private to Self-Employed.\n",
      "Education level changed from Bachelors to Doctorate.\n",
      "Job type changed from Professional to White-Collar.\n",
      "\n",
      "Distance to target: 0.0647207647562027\n",
      "Cost :28.5\n",
      "Dicrepancy value : 0.08857581\n",
      "Verification Confidence: -0.02605534785477831\n",
      "Straight verification value: 0.42606914\n",
      "Hours worked increased by 15.0.\n",
      "Employer type from Private to Self-Employed.\n",
      "Education level changed from Bachelors to Prof-school.\n",
      "Job type changed from Professional to White-Collar.\n",
      "\n",
      "Distance to target: 0.42313095927238464\n",
      "Cost :5.0\n",
      "Dicrepancy value : 0.15198734\n",
      "Verification Confidence: 0.5225589369573165\n",
      "Straight verification value: 0.6541808\n",
      "Employer type from Private to Government.\n",
      "Education level changed from Bachelors to Prof-school.\n",
      "\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "(array([[ 3.00000000e+01,  1.00000000e+00,  1.00000000e+00,\n",
       "          5.00000000e+01,  1.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  1.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00, -0.00000000e+00, -0.00000000e+00,\n",
       "          0.00000000e+00, -0.00000000e+00, -0.00000000e+00,\n",
       "          0.00000000e+00,  1.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "         -0.00000000e+00,  1.00000000e+00,  0.00000000e+00,\n",
       "          4.43591863e-01,  5.56408167e-01,  3.93261611e-01,\n",
       "          2.00000000e+00,  2.57504731e-02,  4.77232493e-01],\n",
       "        [ 3.00000000e+01,  1.00000000e+00,  1.00000000e+00,\n",
       "          5.00000000e+01,  1.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "         -0.00000000e+00,  1.00000000e+00,  0.00000000e+00,\n",
       "         -0.00000000e+00, -0.00000000e+00, -0.00000000e+00,\n",
       "          0.00000000e+00, -0.00000000e+00, -0.00000000e+00,\n",
       "          1.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "         -0.00000000e+00, -0.00000000e+00, -0.00000000e+00,\n",
       "         -0.00000000e+00,  1.00000000e+00, -0.00000000e+00,\n",
       "          5.26440620e-01,  4.73559320e-01,  5.70320487e-01,\n",
       "          0.00000000e+00,  1.31106543e-04,  4.98470713e-01],\n",
       "        [ 3.00000000e+01,  1.00000000e+00,  1.00000000e+00,\n",
       "          5.10000000e+01,  1.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  1.00000000e+00,\n",
       "          0.00000000e+00, -0.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  1.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  1.00000000e+00,  0.00000000e+00,\n",
       "          3.24422330e-01,  6.75577641e-01,  1.88031137e-01,\n",
       "          6.10000000e+00,  4.56715167e-01,  5.25696529e-02],\n",
       "        [ 3.00000000e+01,  1.00000000e+00,  1.00000000e+00,\n",
       "          5.10000000e+01,  1.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  1.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "         -0.00000000e+00,  0.00000000e+00,  1.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          1.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          2.21099302e-01,  7.78900683e-01,  6.28695264e-02,\n",
       "          7.10000000e+00,  4.33159232e-01,  8.15894363e-02],\n",
       "        [ 3.00000000e+01,  1.00000000e+00,  1.00000000e+00,\n",
       "          5.10000000e+01,  1.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  1.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          1.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  1.00000000e+00,  0.00000000e+00,\n",
       "          2.17342809e-01,  7.82657266e-01,  5.93873411e-02,\n",
       "          5.10000000e+00,  4.40999568e-01,  7.39477069e-02],\n",
       "        [ 3.00000000e+01,  1.00000000e+00,  1.00000000e+00,\n",
       "          5.10000000e+01,  1.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  1.00000000e+00,\n",
       "          0.00000000e+00, -0.00000000e+00,  0.00000000e+00,\n",
       "          1.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00, -0.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          1.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          2.05063820e-01,  7.94936180e-01,  4.85890582e-02,\n",
       "          6.10000000e+00,  3.85894418e-01,  1.29702161e-01],\n",
       "        [ 3.00000000e+01,  1.00000000e+00,  1.00000000e+00,\n",
       "          5.10000000e+01,  1.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  1.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00, -0.00000000e+00, -0.00000000e+00,\n",
       "          0.00000000e+00, -0.00000000e+00, -0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  1.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  1.00000000e+00,  0.00000000e+00,\n",
       "          2.61220902e-01,  7.38779128e-01,  1.04991436e-01,\n",
       "          5.10000000e+00,  4.40836608e-01,  7.17903565e-02],\n",
       "        [ 3.00000000e+01,  1.00000000e+00,  1.00000000e+00,\n",
       "          5.10000000e+01,  1.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "         -0.00000000e+00,  1.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00, -0.00000000e+00, -0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          1.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  1.00000000e+00, -0.00000000e+00,\n",
       "          5.15728414e-01,  4.84271586e-01,  5.45879364e-01,\n",
       "          1.00000000e-01,  1.59655800e-04,  4.99008550e-01],\n",
       "        [ 3.00000000e+01,  1.00000000e+00,  1.00000000e+00,\n",
       "          5.20000000e+01,  1.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  1.00000000e+00,\n",
       "          0.00000000e+00, -0.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  1.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          1.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          2.26805881e-01,  7.73194134e-01,  6.83160648e-02,\n",
       "          7.40000000e+00,  4.30635750e-01,  8.38111924e-02],\n",
       "        [ 3.00000000e+01,  1.00000000e+00,  1.00000000e+00,\n",
       "          5.20000000e+01,  1.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  1.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          1.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          1.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          2.00656831e-01,  7.99343169e-01,  4.49369997e-02,\n",
       "          6.40000000e+00,  3.85263324e-01,  1.30566362e-01],\n",
       "        [ 3.00000000e+01,  1.00000000e+00,  1.00000000e+00,\n",
       "          5.30000000e+01,  1.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  1.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          1.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          1.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          1.97979972e-01,  8.02020013e-01,  4.27776948e-02,\n",
       "          6.90000000e+00,  3.86830717e-01,  1.29140502e-01],\n",
       "        [ 3.00000000e+01,  1.00000000e+00,  1.00000000e+00,\n",
       "          5.40000000e+01,  1.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  1.00000000e+00,\n",
       "         -0.00000000e+00, -0.00000000e+00, -0.00000000e+00,\n",
       "          1.00000000e+00, -0.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          0.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          1.00000000e+00,  0.00000000e+00,  0.00000000e+00,\n",
       "          1.95849776e-01,  8.04150224e-01,  4.10915911e-02,\n",
       "          7.60000000e+00,  3.88445020e-01,  1.27638869e-01]]),\n",
       " array([30.        ,  1.        ,  1.        , 50.        ,  1.        ,\n",
       "         0.        ,  0.        ,  0.        ,  0.        ,  0.        ,\n",
       "         1.        ,  0.        ,  0.        ,  0.        ,  0.        ,\n",
       "         0.        ,  0.        ,  0.        ,  0.        ,  0.        ,\n",
       "         1.        ,  0.        ,  0.        ,  0.        ,  0.        ,\n",
       "         1.        ,  0.        ,  0.26084408,  0.73915589,  0.10455541,\n",
       "         5.        ,  0.44517913,  0.06746781]),\n",
       " array([[3.00000000e+01, 1.00000000e+00, 1.00000000e+00, 4.60000000e+01,\n",
       "         1.00000000e+00, 0.00000000e+00, 0.00000000e+00, 0.00000000e+00,\n",
       "         0.00000000e+00, 0.00000000e+00, 0.00000000e+00, 1.00000000e+00,\n",
       "         0.00000000e+00, 0.00000000e+00, 0.00000000e+00, 0.00000000e+00,\n",
       "         0.00000000e+00, 0.00000000e+00, 0.00000000e+00, 0.00000000e+00,\n",
       "         1.00000000e+00, 0.00000000e+00, 0.00000000e+00, 0.00000000e+00,\n",
       "         1.00000000e+00, 0.00000000e+00, 0.00000000e+00, 2.02789813e-01,\n",
       "         7.97210157e-01, 4.66897190e-02, 8.60000000e+00, 4.48833197e-01,\n",
       "         6.68838024e-02],\n",
       "        [3.00000000e+01, 1.00000000e+00, 1.00000000e+00, 5.00000000e+01,\n",
       "         1.00000000e+00, 0.00000000e+00, 0.00000000e+00, 0.00000000e+00,\n",
       "         0.00000000e+00, 1.00000000e+00, 0.00000000e+00, 0.00000000e+00,\n",
       "         0.00000000e+00, 0.00000000e+00, 0.00000000e+00, 1.00000000e+00,\n",
       "         0.00000000e+00, 0.00000000e+00, 0.00000000e+00, 0.00000000e+00,\n",
       "         0.00000000e+00, 0.00000000e+00, 0.00000000e+00, 0.00000000e+00,\n",
       "         0.00000000e+00, 1.00000000e+00, 0.00000000e+00, 4.58520800e-01,\n",
       "         5.41479230e-01, 4.23130959e-01, 5.00000000e+00, 6.54180825e-01,\n",
       "         1.51987344e-01],\n",
       "        [3.00000000e+01, 1.00000000e+00, 1.00000000e+00, 5.40000000e+01,\n",
       "         1.00000000e+00, 0.00000000e+00, 0.00000000e+00, 0.00000000e+00,\n",
       "         0.00000000e+00, 0.00000000e+00, 0.00000000e+00, 1.00000000e+00,\n",
       "         0.00000000e+00, 0.00000000e+00, 0.00000000e+00, 1.00000000e+00,\n",
       "         0.00000000e+00, 0.00000000e+00, 0.00000000e+00, 0.00000000e+00,\n",
       "         0.00000000e+00, 0.00000000e+00, 1.00000000e+00, 0.00000000e+00,\n",
       "         0.00000000e+00, 0.00000000e+00, 0.00000000e+00, 2.36224189e-01,\n",
       "         7.63775826e-01, 7.77095556e-02, 7.60000000e+00, 5.46406209e-01,\n",
       "         3.24572623e-02],\n",
       "        [3.00000000e+01, 1.00000000e+00, 1.00000000e+00, 6.50000000e+01,\n",
       "         1.00000000e+00, 0.00000000e+00, 0.00000000e+00, 0.00000000e+00,\n",
       "         0.00000000e+00, 0.00000000e+00, 0.00000000e+00, 1.00000000e+00,\n",
       "         0.00000000e+00, 0.00000000e+00, 0.00000000e+00, 1.00000000e+00,\n",
       "         0.00000000e+00, 0.00000000e+00, 0.00000000e+00, 0.00000000e+00,\n",
       "         0.00000000e+00, 0.00000000e+00, 0.00000000e+00, 0.00000000e+00,\n",
       "         1.00000000e+00, 0.00000000e+00, 0.00000000e+00, 2.23060980e-01,\n",
       "         7.76939094e-01, 6.47207648e-02, 2.85000000e+01, 4.26069140e-01,\n",
       "         8.85758102e-02],\n",
       "        [3.00000000e+01, 1.00000000e+00, 1.00000000e+00, 3.60000000e+01,\n",
       "         1.00000000e+00, 0.00000000e+00, 0.00000000e+00, 0.00000000e+00,\n",
       "         0.00000000e+00, 0.00000000e+00, 1.00000000e+00, 0.00000000e+00,\n",
       "         0.00000000e+00, 0.00000000e+00, 0.00000000e+00, 0.00000000e+00,\n",
       "         0.00000000e+00, 0.00000000e+00, 0.00000000e+00, 0.00000000e+00,\n",
       "         1.00000000e+00, 0.00000000e+00, 0.00000000e+00, 0.00000000e+00,\n",
       "         0.00000000e+00, 1.00000000e+00, 0.00000000e+00, 2.99260288e-01,\n",
       "         7.00739741e-01, 1.52666390e-01, 2.46000000e+01, 5.12097180e-01,\n",
       "         1.48177147e-03],\n",
       "        [3.00000000e+01, 1.00000000e+00, 1.00000000e+00, 5.90000000e+01,\n",
       "         1.00000000e+00, 0.00000000e+00, 0.00000000e+00, 0.00000000e+00,\n",
       "         0.00000000e+00, 0.00000000e+00, 1.00000000e+00, 0.00000000e+00,\n",
       "         0.00000000e+00, 0.00000000e+00, 0.00000000e+00, 0.00000000e+00,\n",
       "         0.00000000e+00, 0.00000000e+00, 0.00000000e+00, 0.00000000e+00,\n",
       "         1.00000000e+00, 0.00000000e+00, 0.00000000e+00, 0.00000000e+00,\n",
       "         0.00000000e+00, 1.00000000e+00, 0.00000000e+00, 2.87129551e-01,\n",
       "         7.12870419e-01, 1.36687040e-01, 1.31000000e+01, 4.48692024e-01,\n",
       "         6.25649393e-02]]))"
      ]
     },
     "execution_count": 48,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "\n",
    "ap.test_individual(low_income[4],classifier,soft_classifier,verifier,adult_income_ap_creator,my_dice_explainer,names[:-1],my_variables,immutables,0,1,21,2000,ai_conditioner,ai_distance,ai_interpretor,ai__change_interpretor,6,intersection = 0.51,verbose = True)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "9de3c496",
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "ap.save_results(low_income,2,names[:-1],'adult_income_examples.csv',classifier,soft_classifier,verifier,adult_income_ap_creator,my_dice_explainer,my_variables,immutables,0,1,21,2000,ai_conditioner,ai_distance,ai_interpretor,ai__change_interpretor,6)\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "f437ba30",
   "metadata": {},
   "source": [
    "# Law School Success"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "46466655",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f4749cbf",
   "metadata": {},
   "outputs": [],
   "source": [
    "def ls_interpretor(x):\n",
    "    \n",
    "    \n",
    "    if x[0] == 1:\n",
    "        print('Sex: Male')\n",
    "    else:\n",
    "        print('Sex: Female')\n",
    "        \n",
    "    Races = ['Amerindian','Asian','Black','Hispanic','Mexican','Puertorican','White','Other']\n",
    "    race = np.argmax(x[4:12])\n",
    "    print('Race: '+Races[race])\n",
    "        \n",
    "    print('LAST Score: '+str(x[1]))\n",
    "    \n",
    "    print('Undergraduate GPA: '+str(x[2]))\n",
    "    \n",
    "    print('First semester grades: '+str(x[3]))\n",
    "    \n",
    "    Regions = ['Far West','Great Lakes','Mid-South','Mountain West','Mid-West','North East','New England','North West']\n",
    "    region = np.argmax(x[12:20])\n",
    "    print('Region: '+Regions[region])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "12fb9aed",
   "metadata": {},
   "outputs": [],
   "source": [
    "def ls_change_interpretor(x,y):\n",
    "    sex_change =  y[0]-x[0]    \n",
    "    if sex_change > 0:\n",
    "        print('Sex changed from female to male.')\n",
    "    elif sex_change < 0:\n",
    "        print('Sex changed from male to female.')\n",
    "        \n",
    "    Races = ['Amerindian','Asian','Black','Hispanic','Mexican','Puertorican','White','Other']\n",
    "    original_race = np.argmax(x[4:12])\n",
    "    new_race = np.argmax(y[4:12])\n",
    "    if np.abs(original_race - new_race)>.5:\n",
    "        print('Race changed from '+Races[original_race]+' to '+Races[new_race]+'.')\n",
    "        \n",
    "    LSAT_change =  y[1]-x[1]    \n",
    "    if LSAT_change > 0:\n",
    "        print('LSAT score increased by '+str(LSAT_change)+'.')\n",
    "    elif LSAT_change < 0:\n",
    "        print('LSAT score decreased by '+str(LSAT_change*-1)+'.')\n",
    "        \n",
    "    UGPA_change =  y[2]-x[2]    \n",
    "    if UGPA_change > 0:\n",
    "        print('Undergraduate GPA increased by '+str(UGPA_change)+'.')\n",
    "    elif UGPA_change < 0:\n",
    "        print('Undergraduate GPA decreased by '+str(UGPA_change*-1)+'.')\n",
    "        \n",
    "    FSG_change =  y[3]-x[3]    \n",
    "    if FSG_change > 0:\n",
    "        print('First Semester Grades increased by '+str(FSG_change)+'.')\n",
    "    elif FSG_change < 0:\n",
    "        print('First Semester Grades decreased by '+str(FSG_change*-1)+'.')\n",
    "    \n",
    "    Regions = ['Far West','Great Lakes','Mid-South','Mountain West','Mid-West','North East','New England','North West']\n",
    "    original_region = np.argmax(x[12:20])\n",
    "    new_region = np.argmax(y[12:20])\n",
    "    if np.abs(original_region - new_region)>.5:\n",
    "        print('Region changed from '+Regions[original_region]+' to '+Regions[new_region]+'.')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3b12e30f",
   "metadata": {},
   "outputs": [],
   "source": [
    "def ls_conditioner(invec):\n",
    "    output = invec.detach().clone()\n",
    "    output[0] = torch.round(output[0])\n",
    "    race = torch.argmax(output[4:12])\n",
    "    output[4:12] *= 0\n",
    "    output[4+race] += 1\n",
    "    region = torch.argmax(output[12:20])\n",
    "    output[12:20] *= 0\n",
    "    output[12+region] += 1\n",
    "    return output"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "97c307a8",
   "metadata": {},
   "outputs": [],
   "source": [
    "def ls_distance(a,b):\n",
    "    distance = 0\n",
    "    distance += b[1]-a[1]\n",
    "    distance += 10*(b[2]-a[2])\n",
    "    distance += b[3]-a[3]\n",
    "    travel_weights = torch.tensor([[0,3,4,1,2,6,5,1],[3,0,1,2,1,2,1,3],[4,1,0,2,1,2,1,5],[1,2,2,0,1,4,3,2],[2,1,1,1,0,3,2,3],[6,2,2,4,3,0,1,5],[5,1,1,3,2,1,0,5],[1,3,5,2,3,5,5,0]]).double()    \n",
    "    distance += torch.dot(torch.mv(travel_weights,b[12:20]),a[12:20])\n",
    "    \n",
    "    return distance"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e68bdd7f",
   "metadata": {},
   "outputs": [],
   "source": [
    "ls_immutables = np.array([0,1,2,4,5,6,7,8,9,10,11])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "5c13d8f7",
   "metadata": {},
   "outputs": [],
   "source": [
    "categoricals = [(4,11),(12,19)]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "407c2267",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_dice_data = dml.Data(dataframe=ls_data, continuous_features=['LSAT', 'UGPA', 'ZFYA'], outcome_name='first_pf')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3a78837d",
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "soft_classifier.cpu()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "72de1acc",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_dice_model = dml.Model(soft_classifier, backend='PYT')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "9f16886a",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_dice_explainer = DicePyTorch(my_dice_data,my_dice_model)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "4c5b5e45",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_variables = [\n",
    " 'ZFYA',\n",
    " 'FW',\n",
    " 'GL',\n",
    " 'MS',\n",
    " 'Mt',\n",
    " 'MW',\n",
    " 'NE',\n",
    " 'NG',\n",
    " 'NW']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "cc46f4ff",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_dice_explainer.generate_counterfactuals(ls_data.drop(columns=\"first_pf\")[1:2], total_CFs=1, desired_class=\"opposite\",features_to_vary=my_variables)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f0cbbf3d",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_categoricals = [[4,5,6,7,8,9,10,11],[12,13,14,15,16,17,18,19]]\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "0eae8083",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_dice_explainer.encoded_categorical_feature_indexes = my_categoricals"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "81a7b6e6",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_dice_explainer.encoded_continuous_feature_indexes = [1,2,3]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d04b1481",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_dice_explainer.cont_minx = mins[my_dice_explainer.encoded_continuous_feature_indexes]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "13f814e4",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_dice_explainer.cont_maxx = maxes[my_dice_explainer.encoded_continuous_feature_indexes]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "2e9e6703",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_dice_explainer.minx[0,my_dice_explainer.encoded_continuous_feature_indexes] = my_dice_explainer.cont_minx = mins[my_dice_explainer.encoded_continuous_feature_indexes]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "195d55e6",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_dice_explainer.maxx[0,my_dice_explainer.encoded_continuous_feature_indexes] = my_dice_explainer.cont_maxx = maxes[my_dice_explainer.encoded_continuous_feature_indexes]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "971b36c2",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_dice_explainer.cont_precisions = [100,100,100]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3fe9c4fa",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "dd00ed4b",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "markdown",
   "id": "994111fe",
   "metadata": {},
   "source": [
    "# Diabetes Prediction"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "5b747a6f",
   "metadata": {},
   "outputs": [],
   "source": [
    "def torch_dbts_conditioner(x):\n",
    "    new_x = x.detach().clone()\n",
    "    new_x = torch.round(new_x)\n",
    "    return new_x"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ec3123c9",
   "metadata": {},
   "outputs": [],
   "source": [
    "def dbts_dist(a,b):\n",
    "    dist = 0\n",
    "    dist += 3*(a[0]-b[0])**2\n",
    "    dist += 3*(a[1]-b[1])**2\n",
    "    dist += (a[3]-b[3])**2\n",
    "    dist += (a[7]-b[7])**2\n",
    "    dist += (a[8]-b[8])**2\n",
    "    dist += (a[9]-b[9])**2\n",
    "    dist += (a[10]-b[10])**2\n",
    "    dist += 4*(a[11]-b[11])**2\n",
    "    dist += 4*(a[13]-b[13])**2\n",
    "    dist += 0.25*(a[14]-b[14])**2\n",
    "    dist += 0.25*(a[15]-b[15])**2\n",
    "    dist += 5*(a[19]-b[19])**2\n",
    "    dist += 2*(a[20]-b[20])**2\n",
    "    return dist"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c2167b8f",
   "metadata": {},
   "outputs": [],
   "source": [
    "immutables = torch.tensor([0,1,2,5,6,12,13,14,15,16,17,18])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "9605a4c4",
   "metadata": {},
   "outputs": [],
   "source": [
    "categoricals = []"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "30fb7261",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_dice_data = dml.Data(dataframe=dbts_data, continuous_features=['BMI', 'GenHlth','MentHlth', 'PhysHlth', 'Age','Education','Income'], outcome_name='Any_Diabetes')"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "5b4ced12",
   "metadata": {},
   "source": [
    "# German Credit"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "6a65f3fe",
   "metadata": {},
   "outputs": [],
   "source": [
    "#For the german credit data we use the Enhanced_fully_connected_Network"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "bcc44c19",
   "metadata": {},
   "outputs": [],
   "source": [
    "def german_interpretor(x):\n",
    "    \n",
    "    print('Loan Duration: '+str(x[0]*deviations[0]+averages[0]))\n",
    "    \n",
    "    print('Loan Size: '+str(x[1]*deviations[1]+averages[1]))\n",
    "    \n",
    "    print('Disposable Income: '+str(x[2]*deviations[2]+averages[2]))\n",
    "    \n",
    "    print('Time at Current Residence: '+str(x[3]*deviations[3]+averages[3]))\n",
    "    \n",
    "    print('Age: '+str(x[4]*deviations[4]+averages[4]))\n",
    "    \n",
    "    print('Dependents: '+str(x[5]*deviations[5]+averages[5]))\n",
    "    \n",
    "    if x[7] == 1:\n",
    "        print('Telephone: Yes')\n",
    "    else:\n",
    "        print('Telephone: No')\n",
    "    \n",
    "    if x[8] == 1:\n",
    "        print('Foreign Worker: Yes')\n",
    "    else:\n",
    "        print('Foreign Worker: No')\n",
    "        \n",
    "    other_credits = ['Other Credits Critical','Other Credits Critical','All Other Credits Current','Other Credits Delayed','No Other Credits',]\n",
    "    other_credit = np.argmax(x[9:14])    \n",
    "    print(other_credits[other_credit])\n",
    "        \n",
    "    purposes = ['Appliances','Business','Car(new)','Car(used)','Education','Furniture/Equipment','Other','Radio/TV','Repairs','Retraining']\n",
    "    purpose = np.argmax(x[14:24])\n",
    "    print('Purpose: '+purposes[purpose])\n",
    "    \n",
    "    \n",
    "    statuses = ['Female','Divorced Male', 'Married Male', 'Single Male']\n",
    "    status = np.argmax(x[24:28])\n",
    "    print('Personal Status: '+statuses[status])\n",
    "    \n",
    "    files = ['Co-file','Guarantor','Single-file']\n",
    "    file = np.argmax(x[28:31])\n",
    "    print('Filing Type: '+files[file])\n",
    "    \n",
    "    collaterals = ['Car','Life Insurance','Real Estate','None']\n",
    "    collateral = np.argmax(x[31:35])\n",
    "    print('Collateral: '+collaterals[collateral])\n",
    "    \n",
    "    housings = ['Free','Own','Rent']\n",
    "    housing = np.argmax(x[35:38])\n",
    "    print('Housing Type: '+housings[housing])\n",
    "       \n",
    "    installments = ['Bank','None','Store']\n",
    "    installment = np.argmax(x[38:41])\n",
    "    print('Other Installments: '+installments[installment])\n",
    "    \n",
    "    checkings = ['Less than 0','Between 0 and 200','Greater than 200', 'No checking account']\n",
    "    checking = np.argmax(x[41:45])\n",
    "    print('Amount in Checkings: '+checkings[checking])\n",
    "    \n",
    "    savings = ['No savings account','Between 0 and 100','Between 500 and 1,000','Between 100 and 500', 'Greater than 1000']\n",
    "    saving = np.argmax(x[45:50])\n",
    "    print('Amount in Savings: '+savings[saving])\n",
    "    \n",
    "    employments = ['Less than a year','Between 1 and 4 years','Between 4 and 7 years','Greater than 7 years']\n",
    "    employment = np.argmax(x[50:54])\n",
    "    print('Employment length: '+employments[employment])\n",
    "    \n",
    "    jobs = ['Unemployed','Managemant','Skilled','Unskilled','Unskilled non-resident']\n",
    "    job = np.argmax(x[50:55])\n",
    "    print('Job Type: '+jobs[job])\n",
    "    \n",
    "    return    \n",
    "    "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "12cdc21d",
   "metadata": {},
   "outputs": [],
   "source": [
    "def german_change_interpretor(x,y):\n",
    "    \n",
    "    duration_change =  y[0]-x[0]    \n",
    "    if duration_change > 0:\n",
    "        print('Duration increased by '+str(duration_change*deviations[0])+' months.')\n",
    "    elif duration_change < 0:\n",
    "        print('Duration decreased by '+str(duration_change*deviations[0])+' months.')\n",
    "    \n",
    "    size_change =  y[1]-x[1]    \n",
    "    if size_change > 0:\n",
    "        print('Size increased by '+str(size_change*deviations[1])+' DM.')\n",
    "    elif size_change < 0:\n",
    "        print('Size decreased by '+str(size_change*deviations[1])+' DM.')\n",
    "        \n",
    "    income_change =  y[2]-x[2]    \n",
    "    if income_change > 0:\n",
    "        print('Income increased by '+str(income_change*deviations[2])+'.')\n",
    "    elif income_change < 0:\n",
    "        print('Income decreased by '+str(income_change*deviations[2])+'.')\n",
    "        \n",
    "    residence_change =  y[3]-x[3]    \n",
    "    if residence_change > 0:\n",
    "        print('Time at residence increased by '+str(residence_change*deviations[3])+' years.')\n",
    "    elif residence_change < 0:\n",
    "        print('Time at residence decreased by '+str(residence_change*deviations[4])+' years.')\n",
    "        \n",
    "    age_change =  y[4]-x[4]    \n",
    "    if age_change > 0:\n",
    "        print('Age increased by '+str(age_change*deviations[4])+' years.')\n",
    "    elif age_change < 0:\n",
    "        print('Age decreased by '+str(age_change*deviations[4])+' years.')\n",
    "        \n",
    "    credits_change =  y[5]-x[5]    \n",
    "    if credits_change > 0:\n",
    "        print('Existing credits increased by '+str(credits_change*deviations[5])+'.')\n",
    "    elif credits_change < 0:\n",
    "        print('Existing credits decreased by '+str(credits_change*deviations[5])+'.')\n",
    "        \n",
    "    dependents_change =  y[6]-x[6]    \n",
    "    if dependents_change > 0:\n",
    "        print('Dependents increased by '+str(dependents_change*deviations[6])+'.')\n",
    "    elif dependents_change < 0:\n",
    "        print('Dependents decreased by '+str(dependents_change*deviations[6])+'.')\n",
    "    \n",
    "    telephone_change = y[7]-x[7]\n",
    "    if telephone_change ==1:\n",
    "        print('Got Telephone.')\n",
    "    elif telephone_change ==-1:\n",
    "        print('Lost Telephone.')\n",
    "        \n",
    "    other_credits = ['Other Credits Critical','Other Credits Critical','All Other Credits Current','Other Credits Delayed','No Other Credits',]\n",
    "    org_other_credit = np.argmax(x[9:14])    \n",
    "    new_other_credit = np.argmax(y[9:14]) \n",
    "    if np.abs(org_other_credit - new_other_credit)>.5:\n",
    "        print('Other Credits changed from '+other_credits[org_other_credit]+' to '+other_credits[new_other_credit]+'.')\n",
    "        \n",
    "    purposes = ['Appliances','Business','Car(new)','Car(used)','Education','Furniture/Equipment','Other','Radio/TV','Repairs','Retraining']\n",
    "    org_purpose = np.argmax(x[14:24])\n",
    "    new_purpose = np.argmax(y[14:24])\n",
    "    if np.abs(org_purpose - new_purpose)>.5:\n",
    "        print('Marital Status changed from '+purposes[org_purpose]+' to '+purposes[new_purpose]+'.')\n",
    "    \n",
    "    \n",
    "    statuses = ['Female','Divorced Male', 'Married Male', 'Single Male']\n",
    "    org_status = np.argmax(x[24:28])\n",
    "    new_status = np.argmax(y[24:28])\n",
    "    if np.abs(org_status - new_status)>.5:\n",
    "        print('Marital Status changed from '+statuses[org_status]+' to '+statuses[new_status]+'.')\n",
    "    \n",
    "    files = ['Co-file','Guarantor','Single-file']\n",
    "    org_file = np.argmax(x[28:31])\n",
    "    new_file = np.argmax(y[28:31])\n",
    "    if np.abs(org_file - new_file)>.5:\n",
    "        print('Filing type changed from '+files[org_file]+' to '+files[new_file]+'.')\n",
    "    \n",
    "    collaterals = ['Car','Life Insurance','Real Estate','None']\n",
    "    org_collateral = np.argmax(x[31:35])\n",
    "    new_collateral = np.argmax(y[31:35])\n",
    "    if np.abs(org_collateral - new_collateral)>.5:\n",
    "        print('Collateral type changed from '+collaterals[org_collateral]+' to '+collaterals[new_collateral]+'.')\n",
    "    \n",
    "    housings = ['Free','Own','Rent']\n",
    "    org_housing = np.argmax(x[35:38])\n",
    "    new_housing = np.argmax(y[35:38])\n",
    "    if np.abs(org_housing - new_housing)>.5:\n",
    "        print('Housing type changed from '+housings[org_housing]+' to '+housings[new_housing]+'.')\n",
    "       \n",
    "    installments = ['Bank','None','Store']\n",
    "    org_installment = np.argmax(x[38:41])\n",
    "    new_installment = np.argmax(y[38:41])\n",
    "    if np.abs(org_installment - new_installment)>.5:\n",
    "        print('Installment type changed from '+installments[org_installment]+' to '+installments[new_installment]+'.')\n",
    "    \n",
    "    checkings = ['Less than 0','Between 0 and 200','Greater than 200', 'No checking account']\n",
    "    org_checking = np.argmax(x[41:45])\n",
    "    new_checking = np.argmax(y[41:45])\n",
    "    if np.abs(org_checking - new_checking)>.5:\n",
    "        print('Checking account changed from '+checkings[org_checking]+' to '+checkings[new_checking]+'.')\n",
    "    \n",
    "    savings = ['No savings account','Between 0 and 100','Between 500 and 1,000','Between 100 and 500', 'Greater than 1000']\n",
    "    org_saving = np.argmax(x[45:50])\n",
    "    new_saving = np.argmax(y[45:50])\n",
    "    if np.abs(org_saving - new_saving)>.5:\n",
    "        print('Saving account changed from '+savings[org_saving]+' to '+savings[new_saving]+'.')\n",
    "    \n",
    "    employments = ['Less than a year','Between 1 and 4 years','Between 4 and 7 years','Greater than 7 years']\n",
    "    org_employment = np.argmax(x[50:54])\n",
    "    new_employment = np.argmax(y[50:54])\n",
    "    if np.abs(org_employment - new_employment)>.5:\n",
    "        print('Time at employment changed from '+employments[org_employment]+' to '+employments[new_employment]+'.')\n",
    "    \n",
    "    jobs = ['Unemployed','Managemant','Skilled','Unskilled','Unskilled non-resident']\n",
    "    org_job = np.argmax(x[50:55])\n",
    "    new_job = np.argmax(x[50:55])\n",
    "    if np.abs(org_job - new_job)>.5:\n",
    "        print('Job type changed from '+jobs[org_job]+' to '+jobs[new_job]+'.')\n",
    "\n",
    "    return"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "776effd4",
   "metadata": {},
   "outputs": [],
   "source": [
    "def german_conditioner(x):\n",
    "\n",
    "    new_x = x.detach().clone()  \n",
    "    \n",
    "    true_duration = torch.round((new_x[0]*deviations[0])+averages[0])\n",
    "    new_x[0] = (true_duration-averages[0])/deviations[0]\n",
    "    \n",
    "    true_size = torch.round(new_x[1]*deviations[1]+averages[1])\n",
    "    new_x[1] = (true_size-averages[1])/deviations[1]\n",
    "    \n",
    "    new_x[7:9] = torch.round(x[7:9])\n",
    "    \n",
    "    credit_state= torch.argmax(new_x[9:14])\n",
    "    new_x[9:14] *= 0\n",
    "    new_x[9+credit_state] += 1\n",
    "    \n",
    "    purpose = torch.argmax(new_x[14:24])\n",
    "    new_x[14:24] *= 0\n",
    "    new_x[14+purpose] += 1\n",
    "    \n",
    "    personal = torch.argmax(new_x[24:28])\n",
    "    new_x[24:28] *= 0\n",
    "    new_x[24+personal] += 1\n",
    "    \n",
    "    file = torch.argmax(new_x[28:31])\n",
    "    new_x[28:31] *= 0\n",
    "    new_x[28+file] += 1\n",
    "    \n",
    "    collateral = torch.argmax(new_x[31:35])\n",
    "    new_x[31:35] *= 0\n",
    "    new_x[31+collateral] += 1\n",
    "       \n",
    "    housing = torch.argmax(new_x[35:38])\n",
    "    new_x[35:38] *= 0\n",
    "    new_x[35+housing] += 1\n",
    "    \n",
    "    installments = torch.argmax(new_x[38:41])\n",
    "    new_x[38:40] *= 0\n",
    "    new_x[38+installments] += 1\n",
    "    \n",
    "    checking = torch.argmax(new_x[41:45])\n",
    "    new_x[41:45] *= 0\n",
    "    new_x[41+checking] += 1\n",
    "    \n",
    "    savings = torch.argmax(new_x[45:50])\n",
    "    new_x[45:50] *= 0\n",
    "    new_x[45+savings] += 1\n",
    "    \n",
    "    employment = torch.argmax(new_x[50:55])\n",
    "    new_x[50:55] *= 0\n",
    "    new_x[50+employment] += 1\n",
    "    \n",
    "    job = torch.argmax(new_x[55:59])\n",
    "    new_x[55:59] *= 0\n",
    "    new_x[55+job] += 1\n",
    "    \n",
    "    new_x = torch.min(new_x,torch.tensor(maxes))\n",
    "    new_x = torch.max(new_x,torch.tensor(mins))\n",
    "    \n",
    "\n",
    "    \n",
    "    return new_x"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "cd19ca28",
   "metadata": {},
   "outputs": [],
   "source": [
    "def german_dist(a,b):\n",
    "    distance = 0\n",
    "    \n",
    "    distance += torch.abs((b[0]-a[0])*deviations[0])*(a[2]*deviations[2]+averages[2])/12\n",
    "    \n",
    "    distance += torch.abs((a[1]-b[1])*deviations[1])\n",
    "    \n",
    "    distance += (b[7]-a[7])*50\n",
    "    \n",
    "    \n",
    "    checking_matrix = torch.tensor([[0,200,400,200],[-200,0,200,0],[-400,-200,0,-200],[-200,0,200,0]]).double()    \n",
    "    distance += torch.dot(torch.mv(checking_matrix,b[41:45]),a[41:45])\n",
    "    \n",
    "    savings_matrix = torch.tensor([[0,0,500,100,1000],[0,0,500,100,1000],[-500,-500,0,-400,500],[-100,-100,-400,0,500],[-1000,-1000,-500,-900,0]]).double()    \n",
    "    distance += torch.dot(torch.mv(savings_matrix,b[45:50]),a[45:50])\n",
    "    \n",
    "    \n",
    "    \n",
    "    return distance"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "9ad29e71",
   "metadata": {},
   "outputs": [],
   "source": [
    "immutables = torch.tensor([2,3,4,5,6,]+[i for i in range(8,40)]+[i for i in range(50,59)])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "32779243",
   "metadata": {},
   "outputs": [],
   "source": [
    "categoricals = [(9,13),(14,23),(24,27),(28,30),(31,34),(35,37),(38,40),(41,44),(45,49),(50,53),(54,58)]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "257cd980",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_dice_model = dml.Model(soft_classifier, backend='PYT')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "9ccc5899",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_dice_explainer = DicePyTorch(my_dice_data,my_dice_model)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "6d02d042",
   "metadata": {},
   "outputs": [],
   "source": [
    "immutables"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "bdbfdf7a",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_variables = [\n",
    " 'Duration',\n",
    " 'Loan_Size',\n",
    " 'Telephone',\n",
    " 'Checking < 0 DM',\n",
    " 'Checking < 200 DM',\n",
    " 'Checking > 200 DM',\n",
    " 'No checking',\n",
    " 'No checking',\n",
    " 'Savings < 100 DM',\n",
    " 'Savings < 1000 DM',\n",
    " 'Savings < 500 DM'\n",
    " 'Savings > 1000 DM']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b2572395",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_dice_explainer.generate_counterfactuals(all_dataframe.drop(columns=\"Good_Risk\")[1:2], total_CFs=1, desired_class=\"opposite\",features_to_vary=my_variables)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ae64959c",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_categoricals = [(9,10,11,12,13),(14,15,16,17,18,19,20,21,22,23),(24,25,26,27),(28,29,30),(31,32,33,34),(35,36,37),(38,39,40),(41,42,43,44),(46,47,48,49),(50,51,52,53),(54,55,56,57,58)]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "5065a5c0",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_dice_explainer.encoded_categorical_feature_indexes = my_categoricals"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "7809385b",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_dice_explainer.encoded_continuous_feature_indexes = np.arange(9)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "dc0c2670",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_dice_explainer.cont_minx  = mins[:9]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "59594614",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_dice_explainer.cont_maxx = maxes[:9]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "5a74e18e",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_dice_explainer.minx[0] = mins"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "afed6592",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_dice_explainer.maxx[0] = maxes"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "4a2b42e2",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_dice_explainer.cont_precisions = np.ones(9,dtype = int)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "cf2495ab",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_dice_explainer.feature_weights_list /= maxes-mins"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "260a53c1",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "9d20397a",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "virpyenv",
   "language": "python",
   "name": "virpyenv"
  },
  "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.6.8"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
