{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Imports and preliminaries"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 23548,
     "status": "ok",
     "timestamp": 1648492251609,
     "user": {
      "displayName": "DaEto Georgy",
      "userId": "07578216823519586108"
     },
     "user_tz": 240
    },
    "id": "zAOPTJsJylLV",
    "outputId": "780637ec-b5ad-4bd7-8589-9da90bca9e46"
   },
   "outputs": [],
   "source": [
    "import sys, os\n",
    "sys.path.append(os.path.join(os.getcwd(), '../src'))\n",
    "\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "from sklearn.model_selection import train_test_split\n",
    "\n",
    "!pip install folktables\n",
    "!pip install optax\n",
    "\n",
    "import splitConformalPrediction\n",
    "from recursiveLeastSquares import *\n",
    "import folktables\n",
    "from CalibrationScorers.customResidualCalibrationScorer import customResidualCalibrationScorer\n",
    "from MultivalidAlgorithms.MultivalidCoverage import multivalid_coverage, eval_fn\n",
    "from DatasetGeneration.DivisibleDataset import divisible_dataset\n",
    "from MultivalidAlgorithms.GroupCoverage import group_coverage"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Downloading census data from Folktables\n",
    "\n",
    "state_list = ['AL', 'AK', 'AZ', 'AR', 'CA', 'CO', 'CT', 'DE', 'FL', 'GA', 'HI',\n",
    "              'ID', 'IL', 'IN', 'IA', 'KS', 'KY', 'LA', 'ME', 'MD', 'MA', 'MI',\n",
    "              'MN', 'MS', 'MO', 'MT', 'NE', 'NV', 'NH', 'NJ', 'NM', 'NY', 'NC',\n",
    "              'ND', 'OH', 'OK', 'OR', 'PA', 'RI', 'SC', 'SD', 'TN', 'TX', 'UT',\n",
    "              'VT', 'VA', 'WA', 'WV', 'WI', 'WY', 'PR']\n",
    "curr_state = [state_list[4]] # Currently set to 'CA'\n",
    "data_source = folktables.ACSDataSource(survey_year='2018', horizon='1-Year', survey='person')\n",
    "acs_data = data_source.get_data(states = ['CA'], download=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "def adult_and_race_filter(data):\n",
    "    \"\"\"Mimic the filters in place for Adult data.\n",
    "    Adult documentation notes: Extraction was done by Barry Becker from\n",
    "    the 1994 Census database. A set of reasonably clean records was extracted\n",
    "    using the following conditions:\n",
    "    ((AAGE>16) && (AGI>100) && (AFNLWGT>1)&& (HRSWK>0))\n",
    "    \"\"\"\n",
    "    df = data\n",
    "    df = df[df['AGEP'] > 16]\n",
    "    df = df[df['PINCP'] > 100]\n",
    "    df = df[df['WKHP'] > 0]\n",
    "    # df = df[df['PWGTP'] >= 1]\n",
    "    # df = df[df['RAC1P'] != 3]\n",
    "    # df = df[df['RAC1P'] != 4]\n",
    "    # df = df[df['RAC1P'] != 5]\n",
    "    # df = df[df['RAC1P'] != 7]\n",
    "    return df\n",
    "\n",
    "ACSIncomeNew = folktables.BasicProblem(\n",
    "    features=[\n",
    "        'AGEP',\n",
    "        'COW',\n",
    "        'SCHL',\n",
    "        'MAR',\n",
    "        'OCCP',\n",
    "        'POBP',\n",
    "        'RELP',\n",
    "        'WKHP',\n",
    "        'SEX',\n",
    "        'RAC1P',\n",
    "    ],\n",
    "    target='PINCP',\n",
    "    #target_transform=lambda x: x / normalizingFactor,\n",
    "    group='RAC1P',\n",
    "    preprocess=adult_and_race_filter,\n",
    "    postprocess=lambda x: np.nan_to_num(x, -1),\n",
    ")\n",
    "\n",
    "features, label, group = ACSIncomeNew.df_to_numpy(acs_data)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "def produce_group(feat_index, feat_value):\n",
    "    '''\n",
    "        Input: \n",
    "            - feat_index: index of desired input feature\n",
    "            - feat_value: desired value of that feature\n",
    "        Output:\n",
    "            - f - function which defines a group; f(x) returns True\n",
    "                  iff x[feat_index] == feat_value and returns False otherwise.\n",
    "    '''\n",
    "    def f(x):\n",
    "      return (x[feat_index] == feat_value)\n",
    "    \n",
    "    return f\n",
    "\n",
    "race_groups = list()\n",
    "num_groups_race = 0\n",
    "# From FolksTable Census data,\n",
    "# 1. White\n",
    "# 2. Black / African-American\n",
    "# 3. American Indian - very few\n",
    "# 4. Alaska Native - extremely few\n",
    "# 5. American Indian - also very minimal (slightly more than 4.) \n",
    "# 6. Asian\n",
    "# 7. Native Haiwaan and Other Pacific Islander - very few\n",
    "# 8. Some other race (one)\n",
    "# 9. Two or more races\n",
    "list_of_races = [1, 2, 6, 8, 9]\n",
    "for j in list_of_races:\n",
    "    curr_group_fun = produce_group(9, j)\n",
    "    race_groups.append(curr_group_fun)\n",
    "    num_groups_race += 1\n",
    "\n",
    "sex_groups = list()\n",
    "num_groups_sex = 0\n",
    "# From FolksTable Census data,\n",
    "# 1. Male\n",
    "# 2. Female\n",
    "list_of_sex = [1, 2]\n",
    "for j in list_of_sex:\n",
    "    curr_group_fun = produce_group(8, j)\n",
    "    sex_groups.append(curr_group_fun)\n",
    "    num_groups_sex += 1\n",
    "\n",
    "def all_points(x):\n",
    "    return True\n",
    "\n",
    "basic_group = [all_points]\n",
    "num_groups_basic = 1\n",
    "\n",
    "group_list = race_groups + sex_groups\n",
    "num_groups = num_groups_race + num_groups_sex\n",
    "\n",
    "group_fn_race_sex =(lambda x: [group(x) for group in group_list])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "X_train, X_test, y_train, y_test, group_train, group_test = train_test_split(features, label, group, train_size = 0.8, test_size=0.2, shuffle = True)\n",
    "training_size, numFeatures = np.shape(X_train)\n",
    "\n",
    "# Separate training data into training (for point-predictor) and calibration\n",
    "print(np.shape(X_train))\n",
    "training_set_size = len(y_train)\n",
    "calibration_set_size = int(0.25 * training_set_size)\n",
    "new_train_set_size = len(y_train) - calibration_set_size\n",
    "x_train_final = X_train[ : new_train_set_size]\n",
    "x_calib = X_train[new_train_set_size : ]\n",
    "y_train_final = y_train[ : new_train_set_size]\n",
    "y_calib = y_train[new_train_set_size : ]\n",
    "print(new_train_set_size)\n",
    "print(calibration_set_size)\n",
    "print(len(y_test))\n",
    "print(len(y_train))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Synthetic Experiment with groups - Comparing conformal prediction methods: Single trial"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "##### Set all parameters for a single trial"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {
    "executionInfo": {
     "elapsed": 8,
     "status": "ok",
     "timestamp": 1648492392939,
     "user": {
      "displayName": "DaEto Georgy",
      "userId": "07578216823519586108"
     },
     "user_tz": 240
    },
    "id": "mNqAcKetylLY"
   },
   "outputs": [],
   "source": [
    "# Parameters for our uncertainty quantifier(s)\n",
    "tau = 0.9\n",
    "delta = 1 - tau\n",
    "num_grid = 300"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "##### Taking census data and splitting into train (0.6), calibration (0.2) and test (0.2) data."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {
    "executionInfo": {
     "elapsed": 11,
     "status": "ok",
     "timestamp": 1648492395061,
     "user": {
      "displayName": "DaEto Georgy",
      "userId": "07578216823519586108"
     },
     "user_tz": 240
    },
    "id": "yZXQudeJylLZ"
   },
   "outputs": [],
   "source": [
    "X_train, X_test, y_train, y_test, group_train, group_test = train_test_split(features, label, group, train_size = 0.8, test_size=0.2, shuffle = True)\n",
    "training_size, numFeatures = np.shape(X_train)\n",
    "\n",
    "# Separate training data into training (for point-predictor), calibration and test.\n",
    "training_set_size = len(y_train)\n",
    "calibration_set_size = int(0.25 * training_set_size)\n",
    "new_train_set_size = len(y_train) - calibration_set_size\n",
    "x_train_final = X_train[ : new_train_set_size]\n",
    "x_calib = X_train[new_train_set_size : ]\n",
    "y_train_final = y_train[ : new_train_set_size]\n",
    "y_calib = y_train[new_train_set_size : ]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Train point-predictor using x_train_final and y_train_final\n",
    "myRLS = RLS(numFeatures, 1.0, 1)\n",
    "for i in range(new_train_set_size):\n",
    "    myRLS.add_obs(x_train_final[i], y_train_final[i])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Convert y_calib values into threshold values in order to train for conformal prediction \n",
    "from CalibrationScorers.customResidualCalibrationScorer import customResidualCalibrationScorer\n",
    "mult_factor = 800000 # Set to value which ensures majority of threshold values are within [0, 1]\n",
    "residualCalibrationScorer = customResidualCalibrationScorer(mult_factor)\n",
    "residualCalibrationScorer.update(myRLS.predict)\n",
    "\n",
    "# These are the scores which we want to achieve threshold-calibration with respect to\n",
    "w_calib = np.array([residualCalibrationScorer.calc_score(x_calib[i], y_calib[i]) for i in range(calibration_set_size)])\n",
    "\n",
    "# Checking if values in w_calib are outside desired range.\n",
    "for val in w_calib:\n",
    "    if val < 0 or val > 1:\n",
    "        print(val)\n",
    "print(min(w_calib), max(w_calib))\n",
    "\n",
    "w_test = np.array([residualCalibrationScorer.calc_score(X_test[i], y_test[i]) for i in range(len(y_test))])"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "##### Run each of the 4 methods using x_calib and w_calib to train conformal predictors"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Max violation in round 0  :  5686.337444769763\n",
      "Update: (17, 0, 22)\n",
      "Max violation in round 1  :  5398.9493401015225\n",
      "Update: (16, 0, 16)\n",
      "Max violation in round 2  :  11.072115280043509\n",
      "Update: (18, 16, 12)\n",
      "Max violation in round 3  :  15.805184404636465\n",
      "Update: (15, 16, 20)\n",
      "Max violation in round 4  :  7.382891440501038\n",
      "Update: (18, 22, 19)\n",
      "Max violation in round 5  :  14.85090660841073\n",
      "Update: (15, 22, 27)\n",
      "Max violation in round 6  :  5.8906194690265465\n",
      "Update: (15, 12, 13)\n",
      "Max violation in round 7  :  5.439313304721022\n",
      "Update: (10, 12, 8)\n",
      "Max violation in round 8  :  4.826475770925107\n",
      "Update: (11, 13, 16)\n",
      "Max violation in round 9  :  4.732661678246918\n",
      "Update: (14, 19, 15)\n",
      "Max violation in round 10  :  4.302250770811928\n",
      "Update: (13, 19, 22)\n",
      "Max violation in round 11  :  2.3386446886446848\n",
      "Update: (12, 16, 13)\n",
      "Max violation in round 12  :  3.7685955056179745\n",
      "Update: (9, 16, 20)\n",
      "Max violation in round 13  :  1.7294759825327561\n",
      "Update: (9, 13, 15)\n",
      "Max violation in round 14  :  2.6931905465287995\n",
      "Update: (10, 13, 10)\n",
      "Max violation in round 15  :  1.9959379407616393\n",
      "Update: (11, 15, 17)\n",
      "Max violation in round 16  :  2.101250000000001\n",
      "Update: (4, 15, 12)\n",
      "Max violation in round 17  :  1.688578553615959\n",
      "Update: (13, 12, 13)\n",
      "Max violation in round 18  :  2.012576687116559\n",
      "Update: (18, 12, 9)\n",
      "Max violation in round 19  :  1.5347008547008556\n",
      "Update: (12, 8, 6)\n",
      "Max violation in round 20  :  2.2115450643776833\n",
      "Update: (9, 8, 10)\n",
      "Max violation in round 21  :  1.6904184100418433\n",
      "Update: (9, 6, 7)\n",
      "Max violation in round 22  :  1.190206327372767\n",
      "Update: (11, 22, 23)\n",
      "Max violation in round 23  :  1.8408333333333329\n",
      "Update: (12, 22, 17)\n",
      "Max violation in round 24  :  1.0473376623376651\n",
      "Update: (7, 13, 15)\n",
      "Max violation in round 25  :  1.025641025641027\n",
      "Update: (7, 9, 10)\n",
      "Max violation in round 26  :  1.2033333333333316\n",
      "Update: (8, 9, 7)\n",
      "Max violation in round 27  :  0.989871244635195\n",
      "Update: (11, 20, 21)\n",
      "Max violation in round 28  :  1.2645161290322584\n",
      "Update: (0, 20, 18)\n",
      "Max violation in round 29  :  0.9508695652173911\n",
      "Update: (7, 12, 15)\n",
      "Max violation in round 30  :  0.9308130081300822\n",
      "Update: (7, 7, 8)\n",
      "Max violation in round 31  :  0.8496153846153849\n",
      "Update: (6, 6, 3)\n",
      "Max violation in round 32  :  0.8426548672566404\n",
      "Update: (7, 27, 30)\n",
      "Max violation in round 33  :  1.4210572687224654\n",
      "Update: (12, 27, 23)\n",
      "Max violation in round 34  :  0.8152320675105489\n",
      "Update: (6, 17, 15)\n",
      "Max violation in round 35  :  1.2267841409691622\n",
      "Update: (5, 17, 19)\n",
      "Max violation in round 36  :  0.9799999999999955\n",
      "Update: (8, 15, 13)\n",
      "Max violation in round 37  :  1.7419787985865733\n",
      "Update: (13, 15, 18)\n",
      "Max violation in round 38  :  0.7838860103626933\n",
      "Update: (4, 8, 7)\n",
      "Max violation in round 39  :  1.9129702970297033\n",
      "Update: (7, 8, 10)\n",
      "Max violation in round 40  :  0.802536023054757\n",
      "Update: (3, 10, 11)\n",
      "Max violation in round 41  :  0.7838709677419365\n",
      "Update: (9, 23, 25)\n",
      "Max violation in round 42  :  0.7056701030927826\n",
      "Update: (15, 23, 21)\n",
      "Max violation in round 43  :  0.6334020618556699\n",
      "Update: (10, 19, 18)\n",
      "Max violation in round 44  :  1.0126506024096373\n",
      "Update: (3, 19, 21)\n",
      "Max violation in round 45  :  0.5759999999999984\n",
      "Update: (2, 7, 5)\n",
      "Max violation in round 46  :  0.5157142857142859\n",
      "Update: (5, 9, 12)\n",
      "Max violation in round 47  :  0.4310256410256416\n"
     ]
    }
   ],
   "source": [
    "# Iterative algorithm for multivalidity\n",
    "\n",
    "patches = multivalid_coverage(tau=tau, x_train=x_calib, y_train=w_calib, num_grid=num_grid, group_fn=group_fn_race_sex)\n",
    "multivalid_model = lambda x: eval_fn(x, patches=patches, num_grid=num_grid, group_fn=group_fn_race_sex)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# One-shot algorithm for group coverage\n",
    "group_accurate_model, opt_theta = group_coverage(tau=tau, x_train=x_calib, y_train=w_calib, num_groups=num_groups, group_fn=group_fn_race_sex)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Training naive conformal predictor and conservative group-wise conformal predictor\n",
    "marginalConformalPredictor = splitConformalPrediction.splitConformal(1, basic_group, delta)\n",
    "groupConformalPredictor = splitConformalPrediction.splitConformal(num_groups, group_list, delta)\n",
    "group_cov_marginal = np.zeros(num_groups)\n",
    "group_num_marginal = np.zeros(num_groups)\n",
    "group_cov_group = np.zeros(num_groups)\n",
    "group_num_group = np.zeros(num_groups)\n",
    "group_pred_set_size_marginal = np.zeros(num_groups)\n",
    "group_pred_set_size_group = np.zeros(num_groups)\n",
    "\n",
    "for i, curr_x in enumerate(x_calib):\n",
    "    curr_y = y_calib[i]\n",
    "    marginalConformalPredictor.update_calibration_data(curr_x, curr_y)\n",
    "    groupConformalPredictor.update_calibration_data(curr_x, curr_y)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Running naive conformal predictor and conservative group-wise conformal predictor on test-data\n",
    "\n",
    "grid_size = 2 * num_grid\n",
    "marginal_sizes = np.zeros((num_groups, grid_size))\n",
    "marginal_coverage_total = np.zeros((num_groups, grid_size))\n",
    "marginal_averaged_coverage = np.zeros((num_groups, grid_size))\n",
    "conformal_sizes = np.zeros((num_groups, grid_size))\n",
    "conformal_coverage_total = np.zeros((num_groups, grid_size))\n",
    "conformal_averaged_coverage = np.zeros((num_groups, grid_size))\n",
    "\n",
    "for i, curr_x in enumerate(X_test):\n",
    "    curr_y = y_test[i]\n",
    "    curr_w = w_test[i]\n",
    "    w_marginal_conformal = marginalConformalPredictor.select_best_width(residualCalibrationScorer, curr_x)\n",
    "    w_group_conformal = groupConformalPredictor.select_best_width(residualCalibrationScorer, curr_x)\n",
    "    pred_index1 = int(w_marginal_conformal * grid_size)\n",
    "    pred_index2 = int(w_group_conformal * grid_size)\n",
    "    for j in range(num_groups):\n",
    "        if group_list[j](curr_x) == True:\n",
    "            group_num_marginal[j] += 1\n",
    "            group_num_group[j] += 1\n",
    "            group_cov_marginal[j] += int(w_marginal_conformal > curr_w)\n",
    "            group_cov_group[j] += int(w_group_conformal > curr_w)\n",
    "            group_pred_set_size_marginal[j] += 2 * mult_factor * w_marginal_conformal\n",
    "            group_pred_set_size_group[j] += 2 * mult_factor * w_group_conformal\n",
    "            marginal_sizes[j][pred_index1] += 1\n",
    "            marginal_coverage_total[j][pred_index1] += int(w_marginal_conformal > w_test[i])\n",
    "            conformal_sizes[j][pred_index2] += 1\n",
    "            conformal_coverage_total[j][pred_index2] += int(w_group_conformal > w_test[i])\n",
    "\n",
    "marginal_conformal_coverage = [group_cov_marginal[j]/group_num_marginal[j] for j in range(num_groups)]\n",
    "group_conformal_coverage = [group_cov_group[j]/group_num_group[j] for j in range(num_groups)]\n",
    "marginal_conformal_size = [group_pred_set_size_marginal[j]/group_num_marginal[j] for j in range(num_groups)]\n",
    "group_conformal_size = [group_pred_set_size_group[j]/group_num_group[j] for j in range(num_groups)]"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Plotting Results"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "##### Plot single method results for either BatchGCP or BatchMVP"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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pZde+po8li5dyle+8hXe8Y53MHLkSE4//XQArr32WkaNGsW2227LEUccwUsvvcT//M//cPDBBy9773XXXbcs9jXXXMMuu+zC6NGjOfjgg3n22Wc7zP3cc8/lne98J9tttx0HHnjgsiL0b3/7GzvvvDPbbrstxx9/POuuu+6yON/73vd45zvfyciRI1+zXiStJjJzlXrtsMMOKalv22yzzXLRokXL+p944onMzFyyZEnuvvvuefvtty+b7pRTTsnMzBdeeCGHDBmS8+bNy8zMcePG5X777ZeZmccdd1xedNFFmZn51FNP5YgRI/LZZ5/Nn/3sZ3n00Ud3mMNZZ52VBx54YC5evHhZDm0x5syZk5mZn/jEJ/KHP/xhLl68OIcOHZrPPvtsZmYeddRRedFFF+WiRYtyt912Wzb85JNPzhNPPPF1uWdmPv7448u6v/71r+ePfvSjzMzcb7/98pJLLsnMzB//+Mc5cODAzMy8+uqr88gjj8xXXnklly5dmvvtt1/+6U9/Wt5VLamPA2ZkJzWNR8IkVe6yyy5j9OjRjBo1ilmzZjF79uxl4z72sY8BcPfdd/PWt76V4cOHAzB+/Phl01xzzTWcfPLJbL/99uyxxx68+OKLPPjgg13G/P3vf8+nP/1p+vfvD8CGG27InDlzGD58OFtssQUAhx12GNdffz39+/dn7Nix/OY3v2HJkiX89re/5YADDuCmm25i9uzZ7Lrrrmy//fZccMEFPPDAA6/LHeDOO+9kt912Y9ttt+Xiiy9m1qxZANx4443LjrIdcsghr1mma665hlGjRjF69Gjuvvtu7r333uVfuZJWWf3rTkBSa7vvvvs49dRTmT59Om984xs5/PDDefHFF5eNHzhwYLfzyEx++ctfsuWWW75m+M0339xjeY4bN44zzjiDDTfckDFjxjBo0CAyk7333pvJkyd3+J7G3A8//HCuuOIKtttuO84//3yuu+66LuNlJscddxyf/vSne2wZJK1aPBImqccNGjSIZ555BoCnn36agQMHsv766/Poo49y1VVXdfieLbfcknnz5nH//fcDvKaN1wc+8AFOP/10iiP7cOutt74uTnt77703Z599NkuWLAHgySefZMstt+T+++9n7ty5AFx00UXsvvvuAOy+++7ccsstnHvuuYwbNw6AnXfemRtuuGHZ9M899xz33HNPh/Ge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\n",
      "text/plain": [
       "<Figure size 720x504 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Coverage on group 1 : 0.9015825914935707\n",
      "Coverage on group 2 : 0.8864509605662285\n",
      "Coverage on group 3 : 0.9006522829904666\n",
      "Coverage on group 4 : 0.887593423019432\n",
      "Coverage on group 5 : 0.9007793764988009\n",
      "Coverage on group 6 : 0.8874099279423538\n",
      "Coverage on group 7 : 0.9073073073073074\n",
      "Coverage on group 8 : 0.8809190809190809\n",
      "Coverage on group 9 : 0.9009764276555874\n",
      "Coverage on group 10 : 0.8870297130108509\n",
      "Coverage on group 11 : 0.9023859023859024\n",
      "Coverage on group 12 : 0.8856450146479442\n",
      "Coverage on group 13 : 0.9136400322841001\n",
      "Coverage on group 14 : 0.8749008723235527\n",
      "Coverage on group 15 : 0.8939771030363365\n",
      "Coverage on group 16 : 0.8942240080361628\n",
      "Coverage on group 17 : 0.8937256079255479\n",
      "Coverage on group 18 : 0.8944738682921954\n",
      "Coverage on group 19 : 0.8980942828485456\n",
      "Coverage on group 20 : 0.8901296111665005\n"
     ]
    }
   ],
   "source": [
    "from utils.MultivalidPlotting import plot_group_coverage, plot_all_group_coverage\n",
    "\n",
    "# To plot single method results - set model param to group_accurate_model, or to multivalid_model\n",
    "plot_group_coverage(model=multivalid_model, tau=tau, x_test=X_test, y_test=w_test, num_groups=num_groups, group_fn=group_fn_race_sex, multivalid=False, verbose=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 720x504 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Average prediction set size on group 1 : 92.56330365974283\n",
      "Average prediction set size on group 2 : 92.98736097067746\n",
      "Average prediction set size on group 3 : 92.23858504766683\n",
      "Average prediction set size on group 4 : 93.30368709516692\n",
      "Average prediction set size on group 5 : 91.66616706634693\n",
      "Average prediction set size on group 6 : 93.88160528422738\n",
      "Average prediction set size on group 7 : 88.77577577577577\n",
      "Average prediction set size on group 8 : 96.76223776223776\n",
      "Average prediction set size on group 9 : 87.45561692474602\n",
      "Average prediction set size on group 10 : 98.24029003143697\n",
      "Average prediction set size on group 11 : 85.9021384021384\n",
      "Average prediction set size on group 12 : 99.78406909788868\n",
      "Average prediction set size on group 13 : 85.56093623890234\n",
      "Average prediction set size on group 14 : 99.85923869944489\n",
      "Average prediction set size on group 15 : 79.64285714285714\n",
      "Average prediction set size on group 16 : 106.02184831742842\n",
      "Average prediction set size on group 17 : 77.33638546982888\n",
      "Average prediction set size on group 18 : 108.188018387129\n",
      "Average prediction set size on group 19 : 77.4383149448345\n",
      "Average prediction set size on group 20 : 108.0159521435693\n"
     ]
    }
   ],
   "source": [
    "from utils.MultivalidPlotting import plot_pred_set_size, plot_all_pred_set_sizes\n",
    "\n",
    "plot_pred_set_size(model=multivalid_model, tau=tau, x_test=X_test, y_test=w_test, num_groups=num_groups, group_fn=group_fn_race_sex, mult_width = mult_factor, multivalid=False, verbose=True)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "##### Plot all method results against each other"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Plotting group-wise coverage\n",
    "\n",
    "plot_all_group_coverage(group_accurate_model, multivalid_model, tau, X_test, w_test, num_groups, group_fn_race_sex, marginal_conformal_coverage, group_conformal_coverage, multivalid=False, bar_width = 0.2)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Plotting group-wise prediction set size\n",
    "\n",
    "plot_all_pred_set_sizes(group_accurate_model, multivalid_model, tau, X_test, w_test, num_groups, group_fn_race_sex, mult_factor, marginal_conformal_size, group_conformal_size, multivalid=False, bar_width=0.2)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "##### Scatterplot of group-threshold pair coverage against size"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "grid_size = 2 * num_grid\n",
    "multivalid_sizes = np.zeros((num_groups, grid_size))\n",
    "multivalid_coverage_total = np.zeros((num_groups, grid_size))\n",
    "group_sizes = np.zeros((num_groups, grid_size))\n",
    "group_coverage_total = np.zeros((num_groups, grid_size))\n",
    "\n",
    "group_accurate_multivalid_sizes = np.zeros(num_groups)\n",
    "group_accurate_multivalid_coverage = np.zeros(num_groups)\n",
    "group_accurate_multivalid_pred_set_size = np.zeros(num_groups)\n",
    "group_accurate_group_sizes = np.zeros(num_groups)\n",
    "group_accurate_group_coverage = np.zeros(num_groups)\n",
    "group_accurate_group_pred_set_size = np.zeros(num_groups)\n",
    "\n",
    "for i, test_val in enumerate(X_test):\n",
    "    pred1 = multivalid_model(test_val)\n",
    "    pred_index1 = int(pred1 * grid_size)\n",
    "    pred2 = group_accurate_model(test_val)\n",
    "    pred_index2 = int(pred2 * grid_size)\n",
    "    for k in range(num_groups):\n",
    "        if group_fn_race_sex(test_val)[k]:\n",
    "            multivalid_sizes[k][pred_index1] += 1\n",
    "            multivalid_coverage_total[k][pred_index1] += int(pred1 > w_test[i])\n",
    "            group_sizes[k][pred_index2] += 1\n",
    "            group_coverage_total[k][pred_index2] += int(pred2 > w_test[i])\n",
    "            group_accurate_multivalid_sizes[k] += 1\n",
    "            group_accurate_multivalid_coverage[k] += int(pred1 > w_test[i])\n",
    "            group_accurate_multivalid_pred_set_size[k] += (2 * pred1 * mult_factor)\n",
    "            group_accurate_group_sizes[k] += 1\n",
    "            group_accurate_group_coverage[k] += int(pred2 > w_test[i])\n",
    "            group_accurate_group_pred_set_size[k] += (2 * pred2 * mult_factor)\n",
    "\n",
    "final_group_coverage = [group_accurate_group_coverage[j]/group_accurate_group_sizes[j] for j in range(num_groups)]\n",
    "final_multivalid_coverage = [group_accurate_multivalid_coverage[j]/group_accurate_multivalid_sizes[j] for j in range(num_groups)]\n",
    "final_group_pred_set_size = [group_accurate_group_pred_set_size[j]/group_accurate_group_sizes[j] for j in range(num_groups)]\n",
    "final_multivalid_pred_set_size = [group_accurate_multivalid_pred_set_size[j]/group_accurate_multivalid_sizes[j] for j in range(num_groups)]   \n",
    "\n",
    "x_scatter1 = list()\n",
    "y_scatter1 = list()\n",
    "x_scatter2 = list()\n",
    "y_scatter2 = list()\n",
    "x_scatter3 = list()\n",
    "y_scatter3 = list()\n",
    "x_scatter4 = list()\n",
    "y_scatter4 = list()\n",
    "\n",
    "for k in range(num_groups):\n",
    "    for j in range(grid_size):\n",
    "        if multivalid_sizes[k][j] > 0:\n",
    "            x_scatter1.append(multivalid_sizes[k][j])\n",
    "            y_scatter1.append(multivalid_coverage_total[k][j] / multivalid_sizes[k][j])\n",
    "        if group_sizes[k][j] > 0:\n",
    "            x_scatter2.append(group_sizes[k][j])\n",
    "            y_scatter2.append(group_coverage_total[k][j] / group_sizes[k][j])\n",
    "        if marginal_sizes[k][j] > 0:\n",
    "            x_scatter3.append(marginal_sizes[k][j])\n",
    "            y_scatter3.append(marginal_coverage_total[k][j] / marginal_sizes[k][j])\n",
    "        if conformal_sizes[k][j] > 0:\n",
    "            x_scatter4.append(conformal_sizes[k][j])\n",
    "            y_scatter4.append(conformal_coverage_total[k][j] / conformal_sizes[k][j])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Plot scatter\n",
    "plt.figure(figsize=(10, 7))\n",
    "plt.scatter(x_scatter3, y_scatter3, color='b', alpha = 0.5, label = 'Split-Conformal: Without groups')\n",
    "plt.scatter(x_scatter4, y_scatter4, color='m', alpha = 0.5, label = 'Split-Conformal: With groups, conservative approach')\n",
    "plt.scatter(x_scatter1, y_scatter1, color='g', alpha = 0.8, label = 'BatchMVP')\n",
    "plt.scatter(x_scatter2, y_scatter2, color='c', alpha = 0.5, label = 'BatchGCP')\n",
    "plt.axhline(y=tau, c='r', linestyle='--', linewidth=2)\n",
    "plt.text(810, tau + 0.02, '  target coverage')\n",
    "plt.xlabel('Size of group conditional on threshold')\n",
    "plt.ylabel('Realized threshold and group conditional coverage')\n",
    "plt.legend()\n",
    "plt.show()"
   ]
  }
 ],
 "metadata": {
  "colab": {
   "name": "Linear Regression Experiment.ipynb",
   "provenance": []
  },
  "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.9.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 1
}
