{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "72a27a16",
   "metadata": {},
   "source": [
    "# Import libraries"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "f031f752",
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "from sklearn.model_selection import train_test_split\n",
    "\n",
    "from dirrac.data.synthetic_data import DataSynthesizer\n",
    "from dirrac.classifier.logistic import logistic_classifier\n",
    "from dirrac.optim.opt import Optimization\n",
    "from roar.gen_counterfactual import ROAR\n",
    "from dirrac.gen_counterfactual import DRRA\n",
    "from utils import cal_validity"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "af770047",
   "metadata": {},
   "outputs": [],
   "source": [
    "if not os.path.exists('result/figure9/'):\n",
    "    os.makedirs('result/figure9/')"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "cb6a6b67",
   "metadata": {},
   "source": [
    "# Synthesize data and train original classifier"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "b8c0ebac",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Synthesize data\n",
    "mean_0 = np.ones(2) * (-3)\n",
    "mean_1 = np.ones(2) * 3\n",
    "cov_0 = cov_1 = np.identity(2)\n",
    "n = 1000\n",
    "p = [0.4, 0.2, 0.4]\n",
    "\n",
    "sd = DataSynthesizer(mean_0, cov_0, mean_1, cov_1, n)\n",
    "features, labels = sd.synthesize_modes_data(100, [0.4, 0.2, 0.4], [0.1, 0.1])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "c5b9e79a",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Train\n",
    "all_coef = np.zeros((101, 2))\n",
    "for i in range(101):\n",
    "    coef = logistic_classifier(features[i], labels[i])[1].T\n",
    "    all_coef[i] = np.squeeze(coef)\n",
    "\n",
    "# Get theta\n",
    "mean_shift = all_coef[1:41]\n",
    "cov_shift = all_coef[41:61]\n",
    "both_shift = all_coef[61:101]\n",
    "\n",
    "theta = np.zeros((3, 2))\n",
    "sigma = np.zeros((3, 2, 2))\n",
    "\n",
    "theta[0], sigma[0] = np.mean(mean_shift, axis=0), np.cov(mean_shift.T)\n",
    "theta[1], sigma[1] = np.mean(cov_shift, axis=0), np.cov(cov_shift.T)\n",
    "theta[2], sigma[2] = np.mean(both_shift, axis=0), np.cov(both_shift.T)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "4a983835",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Train a classifier\n",
    "X_train, X_test, y_train, y_test = train_test_split(features[0], labels[0], test_size=0.1, random_state=42)\n",
    "\n",
    "# Train and get theta\n",
    "clf, coef = logistic_classifier(X_train, y_train)\n",
    "\n",
    "yhat = clf.predict(X_test)\n",
    "X_recourse = X_test[yhat == 0]"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "6fe1cb22",
   "metadata": {},
   "source": [
    "# Define function"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "479c80a8",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Optimization module\n",
    "delta_l = [1e-3, 0.02, 0.04, 0.06, 0.08, 0.1, 0.12, 0.14, 0.16, 0.18, 0.2]\n",
    "k = 3\n",
    "dim = 2\n",
    "rho = np.array([0, 0, 0])\n",
    "lmbda = 0.7\n",
    "zeta = 1\n",
    "\n",
    "features_shift, labels_shift = sd.synthesize_modes_data(1000, [0.1, 0.8, 0.1], [0.3, 0.3])\n",
    "clf_shift = [logistic_classifier(features_shift[i + 1], labels_shift[i + 1])[0] for i in range(len(features_shift) - 1)]\n",
    "\n",
    "def recourse_cost(X_recourse):\n",
    "    \n",
    "    cost_dirrac, cost_roar = [], []\n",
    "    val_dirrac, val_roar = [], []\n",
    "    \n",
    "    # delta_l = [1e-3, 7e-3, 8e-3, 0.05, 0.1]\n",
    "    delta_l = [1e-3, 5e-3, 0.0065, 7e-3, 0.0075, 0.0076, 1e-2, 0.1, 0.3, 0.5, 0.7, 0.9, 1, 2.5, 5]\n",
    "    dirrac_l = [DRRA(delta_l[i], k, dim, p, theta, sigma, rho, lmbda, zeta, dist_type='l1') for i in range(len(delta_l))]\n",
    "    \n",
    "    iter_l = [20, 21, 22, 21, 24, 25, 26, 27, 28, 29, 30, 31, 32, 33, 34]\n",
    "    roar_l = [ROAR(X_recourse, all_coef[0], np.zeros(1), 1e-3, 0.1, delta_max=0.1, alpha=0.1, dist_type=1, max_iter=iter_l[i]) for i in range(len(iter_l))]\n",
    "    \n",
    "    for i in range(len(delta_l)):\n",
    "        all_val_dirrac = np.zeros(len(features_shift) - 1)\n",
    "        all_val_roar = np.zeros(len(features_shift) - 1)\n",
    "        \n",
    "        x_drra = dirrac_l[i].fit_data(X_recourse)\n",
    "        x_roar = roar_l[i].fit_data(roar_l[i].data)\n",
    "        \n",
    "        l1 = np.mean(np.linalg.norm((x_drra - X_recourse), ord=1, axis=1))\n",
    "        l2 = np.mean(np.linalg.norm((x_roar - X_recourse), ord=1, axis=1))\n",
    "\n",
    "        cost_dirrac.append(l1)\n",
    "        cost_roar.append(l2)\n",
    "    \n",
    "        for j in range(len(features_shift) - 1):\n",
    "            # Train and get theta\n",
    "            all_val_dirrac[j] = np.mean(clf_shift[j].predict(x_drra))\n",
    "            all_val_roar[j] = np.mean(clf_shift[j].predict(x_roar))\n",
    "        \n",
    "        val_dirrac.append(np.mean(all_val_dirrac))\n",
    "        val_roar.append(np.mean(all_val_roar))\n",
    "    \n",
    "    return cost_dirrac, val_dirrac, cost_roar, val_roar"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "0d0443cb",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 20/20 [00:21<00:00,  1.07s/it]\n",
      "100%|█████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 20/20 [00:00<00:00, 205.38it/s]\n",
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 20/20 [00:56<00:00,  2.83s/it]\n",
      "100%|█████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 20/20 [00:00<00:00, 202.31it/s]\n",
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 20/20 [00:47<00:00,  2.37s/it]\n",
      "100%|█████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 20/20 [00:00<00:00, 191.29it/s]\n",
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 20/20 [00:45<00:00,  2.28s/it]\n",
      "100%|█████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 20/20 [00:00<00:00, 208.72it/s]\n",
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 20/20 [00:49<00:00,  2.47s/it]\n",
      "100%|█████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 20/20 [00:00<00:00, 188.62it/s]\n",
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 20/20 [00:47<00:00,  2.35s/it]\n",
      "100%|█████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 20/20 [00:00<00:00, 166.73it/s]\n",
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 20/20 [00:42<00:00,  2.14s/it]\n",
      "100%|█████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 20/20 [00:00<00:00, 172.29it/s]\n",
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 20/20 [00:20<00:00,  1.04s/it]\n",
      "100%|█████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 20/20 [00:00<00:00, 163.26it/s]\n",
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 20/20 [00:19<00:00,  1.04it/s]\n",
      "100%|█████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 20/20 [00:00<00:00, 157.87it/s]\n",
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 20/20 [00:20<00:00,  1.05s/it]\n",
      "100%|█████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 20/20 [00:00<00:00, 159.93it/s]\n",
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 20/20 [00:20<00:00,  1.00s/it]\n",
      "100%|█████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 20/20 [00:00<00:00, 151.12it/s]\n",
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 20/20 [00:21<00:00,  1.08s/it]\n",
      "100%|█████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 20/20 [00:00<00:00, 143.84it/s]\n",
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 20/20 [00:20<00:00,  1.05s/it]\n",
      "100%|█████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 20/20 [00:00<00:00, 144.74it/s]\n",
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 20/20 [00:21<00:00,  1.09s/it]\n",
      "100%|█████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 20/20 [00:00<00:00, 137.81it/s]\n",
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 20/20 [00:22<00:00,  1.12s/it]\n",
      "100%|█████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 20/20 [00:00<00:00, 123.11it/s]\n"
     ]
    }
   ],
   "source": [
    "cost_dirrac, val_dirrac, cost_roar, val_roar = recourse_cost(X_recourse[:20])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "1a336003",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Matplotlib config\n",
    "SMALL_SIZE = 8\n",
    "MEDIUM_SIZE = 12\n",
    "BIGGER_SIZE = 18\n",
    "\n",
    "plt.rc('font', size=BIGGER_SIZE + 2)          # controls default text sizes\n",
    "plt.rc('axes', titlesize=BIGGER_SIZE)     # fontsize of the axes title\n",
    "plt.rc('axes', labelsize=BIGGER_SIZE - 4)    # fontsize of the x and y labels\n",
    "plt.rc('xtick', labelsize=MEDIUM_SIZE)    # fontsize of the tick labels\n",
    "plt.rc('ytick', labelsize=MEDIUM_SIZE)    # fontsize of the tick labels\n",
    "plt.rc('legend', fontsize=BIGGER_SIZE)    # legend fontsize"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "id": "fb36d1d7",
   "metadata": {},
   "outputs": [
    {
     "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": [
    "fig, ax = plt.subplots()\n",
    "ax.plot(cost_dirrac[:10], val_dirrac[:10], marker='o',label='DiRRAc', alpha=1, linewidth=3)\n",
    "ax.plot(cost_roar[4:12], val_roar[4:12], marker='^', label='ROAR', alpha=1, linewidth=3)\n",
    "\n",
    "ax.set(xlabel='$l_{1}$ cost', ylabel='$M_{2}$ validity')\n",
    "ax.grid()\n",
    "ax.legend(loc='lower right', frameon=False)\n",
    "ax.set_xlim([6.0, 7.1])\n",
    "ax.set_ylim([0.5, 1.01])\n",
    "# ax.set_xscale('log')\n",
    "\n",
    "\n",
    "plt.savefig('result/figure9/cost_robust_dirrac_roar.pdf', dpi=400, transparent=True)\n",
    "plt.show()"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.7.11"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
