{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "metadata": {},
   "outputs": [],
   "source": [
    "from folktables import ACSDataSource\n",
    "import pandas as pd\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "import math\n",
    "from sklearn.linear_model import LinearRegression\n",
    "from sklearn.model_selection import train_test_split\n",
    "\n",
    "import seaborn as sns\n",
    "\n",
    "import cvxpy as cp\n",
    "import mosek\n",
    "\n",
    "from tqdm import tqdm\n",
    "\n",
    "from rkhs import setup_full_cvx_dual, runCV, compute_shifted_coverage\n",
    "from group import compute_split_coverages, compute_group_coverages\n",
    "from density_estimation import compute_coverage_under_density_shift"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Construct Dataset"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "data_source = ACSDataSource(survey_year='2018', horizon='1-Year', survey='person')\n",
    "ca_data = data_source.get_data(states=[\"CA\"], download=True)\n",
    "ny_data = data_source.get_data(states=[\"NY\"], download=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>AGEP</th>\n",
       "      <th>POVPIP</th>\n",
       "      <th>SEX_1</th>\n",
       "      <th>SEX_2</th>\n",
       "      <th>RAC1P_1</th>\n",
       "      <th>RAC1P_2</th>\n",
       "      <th>RAC1P_3</th>\n",
       "      <th>RAC1P_4</th>\n",
       "      <th>RAC1P_5</th>\n",
       "      <th>RAC1P_6</th>\n",
       "      <th>RAC1P_7</th>\n",
       "      <th>RAC1P_8</th>\n",
       "      <th>RAC1P_9</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>17</th>\n",
       "      <td>0.176987</td>\n",
       "      <td>-0.839138</td>\n",
       "      <td>0</td>\n",
       "      <td>1</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>27</th>\n",
       "      <td>0.389267</td>\n",
       "      <td>-1.709452</td>\n",
       "      <td>0</td>\n",
       "      <td>1</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>51</th>\n",
       "      <td>-0.502308</td>\n",
       "      <td>-1.280256</td>\n",
       "      <td>0</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>75</th>\n",
       "      <td>0.856282</td>\n",
       "      <td>-1.023931</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>1</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>157</th>\n",
       "      <td>-0.714588</td>\n",
       "      <td>-1.858479</td>\n",
       "      <td>0</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "         AGEP    POVPIP  SEX_1  SEX_2  RAC1P_1  RAC1P_2  RAC1P_3  RAC1P_4  \\\n",
       "17   0.176987 -0.839138      0      1        1        0        0        0   \n",
       "27   0.389267 -1.709452      0      1        1        0        0        0   \n",
       "51  -0.502308 -1.280256      0      1        0        0        0        0   \n",
       "75   0.856282 -1.023931      1      0        0        0        0        0   \n",
       "157 -0.714588 -1.858479      0      1        0        1        0        0   \n",
       "\n",
       "     RAC1P_5  RAC1P_6  RAC1P_7  RAC1P_8  RAC1P_9  \n",
       "17         0        0        0        0        0  \n",
       "27         0        0        0        0        0  \n",
       "51         0        0        0        1        0  \n",
       "75         0        0        0        0        1  \n",
       "157        0        0        0        0        0  "
      ]
     },
     "execution_count": 22,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "#features = [\"AGEP\",\"SCHL\",\"SEX\",\"RAC1P\",\"JWTR\",\"POVPIP\",\"JWMNP\"]\n",
    "features = [\"AGEP\",\"SEX\",\"RAC1P\",\"POVPIP\",\"JWMNP\"]\n",
    "\n",
    "ca_data_sub = ca_data[features].copy()\n",
    "ny_data_sub = ny_data[features].copy()\n",
    "#ca_data_sub.loc[np.isnan(ca_data_sub['SCHL']),'SCHL'] = 0\n",
    "#ca_data_sub.loc[np.isnan(ca_data_sub['JWTR']),'JWTR'] = 0\n",
    "\n",
    "#ca_data_sub.loc[ca_data_sub['SCHL'] <= 15,'SCHL'] = 0\n",
    "#ca_data_sub.loc[np.logical_and(ca_data_sub['SCHL'] <= 19, ca_data_sub['SCHL'] > 15),'SCHL'] = 1\n",
    "#ca_data_sub.loc[np.logical_or(ca_data_sub['SCHL'] == 20, ca_data_sub['SCHL'] == 21),'SCHL'] = 2\n",
    "#ca_data_sub.loc[ca_data_sub['SCHL'] > 21,'SCHL'] = 3\n",
    "\n",
    "#ca_data_sub['SCHL'] = ca_data_sub['SCHL'].astype('category')\n",
    "ca_data_sub['SEX'] = ca_data_sub['SEX'].astype('category')\n",
    "ca_data_sub['RAC1P'] = ca_data_sub['RAC1P'].astype('category')\n",
    "\n",
    "ny_data_sub['SEX'] = ny_data_sub['SEX'].astype('category')\n",
    "ny_data_sub['RAC1P'] = ny_data_sub['RAC1P'].astype('category')\n",
    "#ca_data_sub['JWTR'] = ca_data_sub['JWTR'].astype('category')\n",
    "\n",
    "ca_data_sub['POVPIP'] = (ca_data_sub['POVPIP'] - np.mean(ca_data_sub['POVPIP']))/np.std(ca_data_sub['POVPIP'])\n",
    "ca_data_sub['AGEP'] = (ca_data_sub['AGEP'] - np.mean(ca_data_sub['AGEP']))/np.std(ca_data_sub['AGEP'])\n",
    "\n",
    "ny_data_sub['POVPIP'] = (ny_data_sub['POVPIP'] - np.mean(ny_data_sub['POVPIP']))/np.std(ny_data_sub['POVPIP'])\n",
    "ny_data_sub['AGEP'] = (ny_data_sub['AGEP'] - np.mean(ny_data_sub['AGEP']))/np.std(ny_data_sub['AGEP'])\n",
    "\n",
    "ca_data_sub = ca_data_sub[~ca_data_sub.isnull().any(axis=1)]\n",
    "ny_data_sub = ny_data_sub[~ny_data_sub.isnull().any(axis=1)]\n",
    "\n",
    "Y_ny = ny_data_sub['JWMNP']\n",
    "X_ny = pd.get_dummies(ny_data_sub)\n",
    "X_ny = X_ny.drop(columns = ['JWMNP'])\n",
    "\n",
    "Y_ca = ca_data_sub['JWMNP']\n",
    "X_ca = pd.get_dummies(ca_data_sub)\n",
    "X_ca = X_ca.drop(columns = ['JWMNP'])\n",
    "\n",
    "X_ny.head()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "def make_train_test_split(X,Y,num_calib=1500,num_test=200):\n",
    "\n",
    "    x_train, x_calib, y_train, y_calib = train_test_split(X, Y, test_size=num_calib + num_test)\n",
    "\n",
    "    reg = LinearRegression().fit(x_train, y_train)\n",
    "    scores = np.array(np.abs(reg.predict(x_calib) - y_calib))\n",
    "\n",
    "    x_calib = pd.DataFrame.to_numpy(x_calib)\n",
    "\n",
    "    calibration_set_size = num_calib  \n",
    "    x_calib_final, x_test, scores_calib, scores_test =  train_test_split(x_calib, scores,\n",
    "                                                            test_size=len(scores) - calibration_set_size)\n",
    "    x_calib_cont = x_calib_final[:,:2]\n",
    "    x_calib_cat = x_calib_final[:,2:]\n",
    "\n",
    "    x_test_cont = x_test[:,:2]\n",
    "    x_test_cat = x_test[:,2:]\n",
    "    \n",
    "    return x_calib_cont, x_calib_cat, x_test_cont, x_test_cat, scores_calib, scores_test, reg"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Compute Coverage On Test Set"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "metadata": {},
   "outputs": [],
   "source": [
    "def compute_adaptive_and_split_coverage(x_calib_cont,x_calib_cat,x_test_cont,x_test_cat,scores_calib,scores_test,\n",
    "                                       gamma=2, alpha=0.9, k=5, min_radius = 1, max_radius = 4000, num_radii = 40,\n",
    "                                        radius = None\n",
    "                                       ):\n",
    "    if radius is None:\n",
    "        allLosses, radii = runCV(x_calib_cont,scores_calib,\"rbf\",gamma,alpha,k,min_radius,\n",
    "                                  max_radius, num_radii, x_calib_cat)\n",
    "        selectedRadius = radii[np.argmin(allLosses)]\n",
    "    else:\n",
    "        selectedRadius = radius\n",
    "        \n",
    "    prob = setup_full_cvx_dual(\n",
    "                x_calib_cont,\n",
    "                scores_calib,\n",
    "                kernel=\"rbf\",\n",
    "                gamma=gamma,\n",
    "                alpha=alpha,\n",
    "                z_calib = x_calib_cat\n",
    "            )\n",
    "    prob.param_dict['radius'].value = np.asarray([[selectedRadius]])\n",
    "    prob.solve(\n",
    "            solver='MOSEK', \n",
    "            verbose=False, \n",
    "            mosek_params={mosek.iparam.intpnt_solve_form: mosek.solveform.dual}\n",
    "          )\n",
    "\n",
    "    est_coverage, weighted_coverage, thresholds = compute_shifted_coverage(\n",
    "            scores_test,\n",
    "            scores_calib,\n",
    "            x_calib_cont,\n",
    "            x_test_cont,\n",
    "            None,\n",
    "            \"rbf\",\n",
    "            alpha,\n",
    "            selectedRadius,\n",
    "            gamma,\n",
    "            x_calib_cat,\n",
    "            x_test_cat\n",
    "        )\n",
    "\n",
    "    adaptive_coverages = scores_test <= thresholds\n",
    "    \n",
    "    split_threshold = np.quantile(scores_calib, [alpha * (1 + 1/len(scores_calib))])\n",
    "    split_coverages = scores_test <= split_threshold\n",
    "    \n",
    "    return adaptive_coverages, split_coverages, selectedRadius"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Plot Coverage Over Groups For One Run"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████| 40/40 [05:08<00:00,  7.72s/it]\n",
      "100%|█████████████████████████████████████████| 200/200 [07:48<00:00,  2.34s/it]\n"
     ]
    },
    {
     "data": {
      "image/png": "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tmct1k5broe1WDtt/dqgdtTya8PrgvWlll8XSW2/Nlr1uth663S8/bZ1MxAb/9neAIANT/t29amvr1vfY7yrSqWSurranK9SaU5D4wb7u22oqg36nZ+FC5emufm9x+/Ro5tT+q6hvcf+IPPnL2mz7VmLtdOW62Et1k5bvzaoHbX82ijtdVHLa5HU9npYi9rRmrWor6/Lllt2XUcT4TM/AABAEcQPAABQBPEDAAAUQfwAAABFED8AAEARxA8AAFAE8QMAABRB/AAAAEVov74HAChdt+6d07lTh/U9xjtqaliZdh06ru8x3lXjyrey+PWV63sMADYQ4gdgPevcqUPNfnt6LX9zevK3b09PxA8AreOwNwAAoAjiBwAAKIL4AQAAiiB+AACAIogfAACgCOIHAAAogvgBAACKIH4AAIAiiB8AAKAI4gcAACiC+AEAAIogfgAAgCKIHwAAoAjiBwAAKIL4AQAAiiB+AACAIogfAACgCOIHAAAogvgBAACKIH4AAIAiiB8AAKAI4gcAACiC+AEAAIogfgAAgCKIHwAAoAjiBwAAKIL4AQAAiiB+AACAIogfAACgCOIHAAAogvgBAACKIH4AAIAiiB8AAKAI4gcAACiC+AEAAIogfgAAgCJUNX6mT5+egQMHpn///pk8efJqt//hD3/I4MGDc8wxx+Scc87JG2+8Uc1xAACAglUtfubOnZsJEyZkypQpueuuu3LrrbfmueeeW+U+V199dcaMGZO7774722+/fW666aZqjQMAABSuavHz6KOPpm/fvtlss83SpUuXDBgwIDNmzFjlPs3NzVm2bFmSZPny5encuXO1xgEAAArXvlobnjdvXnr06NFyuWfPnpk1a9Yq97nooovyuc99Lt/4xjeyySabZOrUqR9oH1tu2bVNZuXd9ejRbX2PwNtYj9phLWqHtagd1qJ2WIvaYS1qS9Xip1KprHZdXV1dy88rVqzIxRdfnB//+MfZY489cvPNN+fCCy/MxIkTW72PhQuXprl59f28nb9wa2f+/CVtti1rsfbaaj2sxdrz2qgd1qJ2WIvaYS1qx/utRX19nV/or0NVO+ytV69eWbBgQcvlefPmpWfPni2X//SnP6VTp07ZY489kiQnn3xynnjiiWqNAwAAFK5q8XPAAQfksccey6JFi7J8+fLMnDkz/fr1a7l92223zZw5c/L8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/Pjx2XHHHas9GwAAQJtpVfxsscUW6dmzZz7+8Y9n9uzZOfbYY/PCCy9UezYAAIA206r4ad++fV588cV8/OMfz5NPPpnGxsa88cYb1Z4NAACgzbQqfkaOHJlLL700hxxySH7xi1/kkEMOSd++fas9GwAAQJtp1QkPGhsb8+Mf/zhJcuedd+aFF17ILrvsUtXBAAAA2lKr3vmZMGFCy8+bbLJJevfunbq6uqoNBQAA0NZa9c7PzjvvnO9973vZZ5990qVLl5brd99996oNBgAA0JZaFT9PP/10nn766dx2220t19XV1eWXv/xl1QYDAABoS62Kn/vvv7/acwAAAFRVqz7zs2zZslxxxRX57Gc/m9deey3jxo3LsmXLqj0bAABAm2lV/Fx11VXp1q1bFi5cmE6dOmXp0qUZN25ctWcDAABoM62Kn2effTbnnXde2rdvn0022STXXXddnn322WrPBgAA0GZaFT/19averampabXrAAAAalmrTniw77775pvf/GZWrFiRhx56KJMnT85+++1X7dkAAADaTKvevjn//PPTpUuXdOvWLRMmTMguu+ySsWPHVns2AACANtOqd34ef/zxjB49OqNHj672PAAAAFXRqnd+brjhhhx66KH57ne/m7lz51Z7JgAAgDbXqvi59dZbM2nSpLz55ps56aSTcs455+S+++6r9mwAAABtptWnbNthhx1ywQUX5Dvf+U4WL16cr3zlK9WcCwAAoE216jM/CxcuzN13352f//znaWpqypAhQ3LjjTdWezYAAIA206r4OeKII3LEEUfk61//evbee+9qzwQAANDmWhU/v/71r9O1a9e88soreeGFF7LttttWey4AAIA21erD3k4++eTMnTs3lUolm2++eW688cbssMMO1Z4PAACgTbTqhAdXXHFFhg8fnieffDJPPfVUzj333Fx++eXVng0AAKDNtCp+Fi5cmOOPP77l8uDBg7N48eKqDQUAANDWWhU/TU1Nee2111ouL1q0qFrzAAAAVEWrPvNz+umn5+STT85RRx2VJPn3f//3fPazn63qYAAAAG2pVe/8HHzwwUmShoaGPP/885k7d2769+9f1cEAAADaUqve+bnooosydOjQDBs2LG+99VZ++tOf5mtf+1omTZpU7fkAAADaRKve+Vm8eHGGDRuWJOnUqVPOPPPMzJ8/v6qDAQAA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cjsAAEBbqlr8HHDAAXnssceyaNGiLF++PDNnzky/fv1abt9mm23SqVOnPPXUU0mSO++8c5XbAQAA2lLVPvOT/O1U1zfeeGMaGhoyZMiQjBgxIiNGjMiYMWPSp0+fzJ49O5dcckmWLVuW3XbbLddcc006duxYrXEAAICCVTV+AAAAakVVv+QUAACgVogfAACgCOIHAAAogvgBAACKIH4AAIAiiB8AAKAI7df3ABu6GTNmZOLEiWlsbEylUsmxxx6b4cOH54wzzsicOXPSpUuXlvtutdVWuemmm/LVr341lUol48ePT5KsXLkyp5xySkaPHp3DDjvsPffX0NCQ4cOHZ9SoUdlvv/2q+tw2NOtyLW699dbccsstqauryyc+8YlcfvnlRX1H1csvv5wjjzwyO+ywQ5Kkubk5y5Yty3HHHZcxY8YkSf70pz9l0KBB+fa3v50BAwas8vg777wzt9xySxobG9Pc3JwTTzwxw4YNW+U+119/ferr6/PFL34xSXLRRRfl8ccfz4c+9KEkf1uroUOH5vTTT3/feZcuXZpTTjkl3//+9/PRj350rZ9/rdmQ1uOGG27Iv//7vydJDj744IwdO3bt/wBqyIa0Fvfff39uuOGGvPnmmznwwANzySWXtMmfQa3YkNbirrvuysSJE5Mk/fr1y4UXXrj2fwA1ZENai4kTJ+b2229Px44dM3DgwJx77rlt8mdADamwxubMmVM55JBDKosWLapUKpXK0qVLK8cff3zlvvvuq5x++umVxx9//B0ft2TJksphhx1WmTlzZqVSqVQuv/zyypVXXvm++/vLX/5SOfnkkyt9+vR5122Xal2uxfPPP1/p379/ZcmSJZXm5ubK2LFjKzfffHObPp9a99JLL1U+85nPrHLdnDlzKnvuuWflueeeq1Qqlco111xT+eIXv1g588wzV7nfz372s8pxxx1XmTt3bqVSqVRef/31yuDBgytTp06tVCqVyhtvvFH56le/Wtljjz0q3/72t1sed+GFF1Zuv/32lsvz58+v7LPPPpVnnnnmPWf97W9/Wzn66KMru+++e+Wll15a8yddwzaU9XjkkUcqJ598cuWtt96qrFy5sjJs2LCW197GYkNZixdffLFy4IEHVl599dXKypUrK6eeemrlgQceWLsnX2M2lLV48803K/vuu29l4cKFlYaGhsqQIUMqjzzyyNo9+RqzoazFI488Ujn66KMrS5YsqTQ2NlbOOeecyr333rt2T56a47C3tbB48eI0NDRkxYoVSZJNN90048ePz4477viej+vatWuuvfbaXHbZZZk2bVp++9vftuq3n9OmTcvw4cOz5557tsn8G5N1uRYdO3bMZZddlq5du6auri4777xz/vu//7vNnsuGav78+alUKtl0003T2NiYu+++O+edd16eeeaZvPjiiy33+973vpeLLrooPXv2TJJ07949//RP/5Sdd945SfLLX/4y2223XT73uc+95/622mqrbLfddvnrX//6nvebOnVqvv71r7fsrxS1uB49evTIRRddlI4dO6ZDhw7ZYYcdinjt1OJa/OIXv8jAgQOz9dZbp0OHDpkwYUIR/7bU4lo0NTWlubk5y5cvT2NjYxobG9OpU6e1f7I1rhbX4plnnsmBBx6Yrl27pl27djnooINy3333rf2TpaY47G0t9O7dO4cddlgOP/zw7Lrrrtlvv/0yaNCgbLvttkmSSy65ZJVDrY488siWt0/32muvHH/88Rk3blzuueeeVh0y9ff/lP/4xz+uwrPZsK3Ltdhmm22yzTbbJEkWLVqUyZMn55prrqnSM6td8+bNy7HHHpu33norixcvTp8+fXLDDTdk6623zn333ZePfOQj2X777XP44YfnZz/7WcaOHZtFixbl1VdfXe0/WX8/FCJJjjvuuCTJd77znffc/+zZs/Nf//Vf6dOnz3ve7+qrr16zJ7iB2RDWY6eddmr5+a9//Wv+7d/+LT/72c/W4NnWtg1hLV544YV06NAhZ599dubPn5/PfOYz+fKXv7zGz7lWbQhr0bVr13zpS1/KUUcdlc6dO+dTn/pU9tprrzV/0jVqQ1iL3XffPd/4xjdyzjnnZJNNNsn999+fSqWy5k+amiR+1tLll1+eUaNG5eGHH87DDz+ck046Kdddd12S5KqrrnrXz+U0NTXlN7/5TTbffPM88sgj2X777dfl2Buldb0Wc+fOzfDhwzN48OAiP3/Vs2fP3HXXXWlubs748ePzxz/+MX379k2S3HHHHTn66KOTJAMHDsz555+fL3/5y6mv/9ubzWv6j8m3v/3t/PjHP05zc3M6d+6cK664YqP8DM+a2JDW489//nPOOeecXHjhhdluu+3WaN+1bENYi6ampjz55JO55ZZb0qVLl4waNSo///nPc8IJJ6zR/mvVhrAWs2fPzu23355f/epX6datW84///zcdNNNGT58+Brtv1ZtCGux//7754QTTsgZZ5yRzTbbLPvvv3+efvrpNdo3tUv8rIUHHnggb775ZgYOHJjBgwdn8ODBmTp1aqZNm/a+j/3ud7+b7t2750c/+lGGDh2a/ffff5XfZPDBrOu1+Mtf/pIRI0bk9NNPz1lnndVWT2ODVF9fn7Fjx+a4447LD3/4wwwZMiQPPvhgfv/73+cnP/lJKpVK3njjjcycOTNHH310Pvaxj+X3v/999t1335ZtPPHEE3nwwQdz/vnnv+e+xowZs9H956yt1fp6PPXUUxkzZky+9rWv5R//8R/X6DluKGp5Lbbaaqvsv//+2WKLLZIkhx12WGbNmrXRvr5qeS0efvjh7L///tlyyy2TJCeccEKmTJmy0cXP39XyWixdujT9+/dvOYTu5ptvzsc+9rE1e6LULJ/5WQudO3fOP//zP+fll19O8rffTDz33HPZdddd3/NxTzzxRKZNm5arr746O+20U0aNGpXzzz8/K1euXBdjb5TW5VosXbo0Z599dr70pS8VHz5/1759+4wdOzbf//73c9ttt6Vv37558MEHc//99+dXv/pVRo4cmVtvvTVJcvbZZ2f8+PGZP39+kr8dOjh+/PiWQxRZe7W6Hq+++mpGjx6d6667bqMPn7+r1bX4zGc+k4cffjhvvPFGmpqa8tBDD2X33Xdv8/3Uklpdi969e+fRRx/Nm2++mUqlkvvvv/99D+fd0NXqWrz88ssZPXp0Ghsbs2TJktx222056qij2nw/rF/e+VkLffv2zRe+8IWMHDkyDQ0NSZKDDjooo0ePztlnn73a50ySv31eZ+zYsbn66qtbfuP22c9+Ng888ECuv/76XHDBBev8eWwM1uVaTJs2LQsWLMgPf/jD/PCHP0ySHHroofnSl75UxWdY+/r165dPfvKTmTp16mqnzD3ttNPygx/8IH/5y19y6qmnpqGhIWeddVbq6upSqVRy8skn58QTT1xPk2+canE9brrpprz11lstp5ZPklNOOSWnnnpqm++rltTiWuy5554ZPnx4TjvttDQ0NOTTn/50Bg8e3Ob7qTW1uBYHHnhgnnnmmZxwwgnp0KFD+vTpk89//vNtvp9aU4tr0bt37xxxxBE55phj0tTUlDPPPDN77713m++H9auu4pNcAABAAbzzU0NefPHFli/n+t+uuuqqjf5t8FpiLTYcTz75ZK688sp3vG3ixInp1avXOp6obNajdliL2mEtaoe1wDs/AABAEZzwAAAAKIL4AQAAiiB+AACAIogfAACgCM72BlCAiRMnZtq0adl0002zzz775Je//GU+9alP5bXXXstLL72UQw45JCNHjszll1+e2bNnp66uLgcddFC+8pWvpH379tlll13y2GOPtXwn1t8v//nPf861116bXr165aWXXkrnzp0zfvz47LDDDuv5GQPA6rzzA7CRe+ihh3LHHXdk2rRpueOOO7Js2bKW21asWJF77rknF1xwQa666qpsttlmmT59em6//fb88Y9/bPki3/fyzDPP5Kyzzsr06dNzwgkn+LJmAGqW+AHYyP3617/OkUceme7du6euri5Dhw5tue3t317+4IMP5vTTT09dXV06duyYU045JQ8++OD7br93797ZZ599kiSDBw/Os88+m8WLF7f9EwGAtSR+ADZy7du3z9u/0q1du3YtP3fp0qXl5+bm5lUe19zcnMbGxtW2t3LlylUuv317SVKpVFa7DgBqgfgB2MgdfPDBmTlzZpYsWZIkmTZt2jve78ADD8zkyZNTqVSycuXKTJ06NQcccECSZIsttsjvfve7JMkvfvGLVR43e/bszJ49O0ly6623Zq+99kr37t2r9XQAYI054QHARm7//ffPSSedlJNPPjmdO3fOTjvtlE022WS1+11yySW56qqrMmjQoDQ0NOSggw7KyJEjW2674oor0r179xxwwAHp0aNHy+O22mqrfOtb38orr7ySLbbYItdee+06e24A8EHUVd5+LAQAG53f/e53+c1vfpNhw4YlSW6++eY8/fTT+da3vrXW2/6P//iPXHnllfnXf/3Xtd4WAFSbd34ANnLbb799Jk2alKlTp6auri4f/vCHc+WVV67vsQBgnfPODwAAUAQnPAAAAIogfgAAgCKIHwAAoAjiBwAAKIL4AQAAivD/A9IhK+HSl5KyAAAAAElFTkSuQmCC\n",
      "text/plain": [
       "<Figure size 842.4x595.44 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "gamma = 0.5\n",
    "alpha = 0.9\n",
    "k = 5\n",
    "min_radius = 1\n",
    "max_radius = 3000\n",
    "num_radii = 40\n",
    "num_calib=1500\n",
    "num_test=200\n",
    "\n",
    "x_calib_cont, x_calib_cat, x_test_cont, x_test_cat, scores_calib, scores_test, _ = make_train_test_split(X_ny,Y_ny,\n",
    "                                                                                                      num_calib,\n",
    "                                                                                                      num_test)\n",
    "adaptive_coverages, split_coverages, _ = compute_adaptive_and_split_coverage(x_calib_cont,x_calib_cat,\n",
    "                                                                          x_test_cont,x_test_cat,\n",
    "                                                                          scores_calib,scores_test,\n",
    "                                                                          gamma, alpha, k,\n",
    "                                                                          min_radius, max_radius,\n",
    "                                                                          num_radii\n",
    "                                                                          )\n",
    "\n",
    "group_data = pd.DataFrame()\n",
    "\n",
    "for i in range(x_test_cat.shape[1]):\n",
    "    if sum(x_test_cat[:,i]) > 0:\n",
    "        group_cov_adaptive = (x_test_cat[:,i] @ adaptive_coverages)/sum(x_test_cat[:,i])\n",
    "        group_split_adaptive = (x_test_cat[:,i] @ adaptive_coverages)/sum(x_test_cat[:,i])\n",
    "    \n",
    "        df = pd.DataFrame({'group' : [X.columns[i+2],X.columns[i+2]],\n",
    "                            'coverage': [group_cov_adaptive,group_split_adaptive], \n",
    "                            'type' : ['robust','split'],\n",
    "                            'sample_size' : [sum(x_test_cat[:,i]),sum(x_test_cat[:,i])]\n",
    "                          })\n",
    "\n",
    "        group_data = pd.concat([group_data, df], axis=0)\n",
    "    \n",
    "sns.set(rc={'figure.figsize':(11.7,8.27)})\n",
    "\n",
    "fig = sns.barplot(group_data, \n",
    "            x=\"group\", y=\"coverage\", hue=\"type\")\n",
    "fig.axhline(alpha, color='red')\n",
    "plt.legend(bbox_to_anchor=(1.02, 1), loc='upper left', borderaxespad=0)\n",
    "plt.tight_layout()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Look at coverage over just groups"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 12%|████▊                                   | 181/1500 [00:04<00:32, 40.61it/s]\n"
     ]
    },
    {
     "ename": "KeyboardInterrupt",
     "evalue": "",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m                         Traceback (most recent call last)",
      "Input \u001b[0;32mIn [28]\u001b[0m, in \u001b[0;36m<module>\u001b[0;34m\u001b[0m\n\u001b[1;32m     16\u001b[0m x_test_cont \u001b[38;5;241m=\u001b[39m x_test[:,:\u001b[38;5;241m2\u001b[39m]\n\u001b[1;32m     17\u001b[0m x_test_cat \u001b[38;5;241m=\u001b[39m x_test[:,\u001b[38;5;241m2\u001b[39m:]\n\u001b[0;32m---> 18\u001b[0m adaptive_coverages \u001b[38;5;241m=\u001b[39m \u001b[43mcompute_group_coverages\u001b[49m\u001b[43m(\u001b[49m\n\u001b[1;32m     19\u001b[0m \u001b[43m    \u001b[49m\u001b[43mx_calib_cat\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mscores_calib\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mscores_test\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mx_test_cat\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mx_test_cat\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43malpha\u001b[49m\n\u001b[1;32m     20\u001b[0m \u001b[43m\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m     21\u001b[0m split_coverages \u001b[38;5;241m=\u001b[39m compute_split_coverages(scores_calib,scores_test,x_test_cat,alpha)\n\u001b[1;32m     22\u001b[0m df \u001b[38;5;241m=\u001b[39m pd\u001b[38;5;241m.\u001b[39mDataFrame({\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mgroup\u001b[39m\u001b[38;5;124m'\u001b[39m : \u001b[38;5;28mlist\u001b[39m(adaptive_coverages\u001b[38;5;241m.\u001b[39mkeys()) \u001b[38;5;241m+\u001b[39m  \u001b[38;5;28mlist\u001b[39m(split_coverages\u001b[38;5;241m.\u001b[39mkeys()),\n\u001b[1;32m     23\u001b[0m                     \u001b[38;5;124m'\u001b[39m\u001b[38;5;124mcoverage\u001b[39m\u001b[38;5;124m'\u001b[39m: \u001b[38;5;28mlist\u001b[39m(adaptive_coverages\u001b[38;5;241m.\u001b[39mvalues()) \u001b[38;5;241m+\u001b[39m \u001b[38;5;28mlist\u001b[39m(split_coverages\u001b[38;5;241m.\u001b[39mvalues()), \n\u001b[1;32m     24\u001b[0m                     \u001b[38;5;124m'\u001b[39m\u001b[38;5;124mtype\u001b[39m\u001b[38;5;124m'\u001b[39m : [\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mrobust\u001b[39m\u001b[38;5;124m'\u001b[39m]\u001b[38;5;241m*\u001b[39m\u001b[38;5;28mlen\u001b[39m(adaptive_coverages) \u001b[38;5;241m+\u001b[39m [\u001b[38;5;124m'\u001b[39m\u001b[38;5;124msplit\u001b[39m\u001b[38;5;124m'\u001b[39m]\u001b[38;5;241m*\u001b[39m\u001b[38;5;28mlen\u001b[39m(split_coverages)\n\u001b[1;32m     25\u001b[0m                     })\n",
      "File \u001b[0;32m~/Documents/ConformalGans/Code/conformal-gan/group.py:72\u001b[0m, in \u001b[0;36mcompute_group_coverages\u001b[0;34m(x_calib, scores_calib, scores_test, groups_test, x_test, alpha, exact)\u001b[0m\n\u001b[1;32m     70\u001b[0m \u001b[38;5;28;01mfor\u001b[39;00m i \u001b[38;5;129;01min\u001b[39;00m tqdm(\u001b[38;5;28mrange\u001b[39m(\u001b[38;5;28mlen\u001b[39m(x_test))):\n\u001b[1;32m     71\u001b[0m     \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;129;01mnot\u001b[39;00m exact:\n\u001b[0;32m---> 72\u001b[0m         threshold, _ \u001b[38;5;241m=\u001b[39m \u001b[43m_compute_adaptive_threshold\u001b[49m\u001b[43m(\u001b[49m\n\u001b[1;32m     73\u001b[0m \u001b[43m            \u001b[49m\u001b[43mprob\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mx_test\u001b[49m\u001b[43m[\u001b[49m\u001b[43mi\u001b[49m\u001b[43m,\u001b[49m\u001b[43m:\u001b[49m\u001b[43m]\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mnp\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mmax\u001b[49m\u001b[43m(\u001b[49m\u001b[43mscores_calib\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m     74\u001b[0m \u001b[43m        \u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m     75\u001b[0m     \u001b[38;5;28;01melse\u001b[39;00m:\n\u001b[1;32m     76\u001b[0m         threshold, _ \u001b[38;5;241m=\u001b[39m _compute_adaptive_threshold(\n\u001b[1;32m     77\u001b[0m             prob, x_test[i,:], scores_test[i]\n\u001b[1;32m     78\u001b[0m         )\n",
      "File \u001b[0;32m~/Documents/ConformalGans/Code/conformal-gan/group.py:52\u001b[0m, in \u001b[0;36m_compute_adaptive_threshold\u001b[0;34m(prob, x_test, M)\u001b[0m\n\u001b[1;32m     50\u001b[0m prob\u001b[38;5;241m.\u001b[39mparam_dict[\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mscore_impute\u001b[39m\u001b[38;5;124m\"\u001b[39m]\u001b[38;5;241m.\u001b[39mvalue \u001b[38;5;241m=\u001b[39m np\u001b[38;5;241m.\u001b[39masarray([[M]])\n\u001b[1;32m     51\u001b[0m prob\u001b[38;5;241m.\u001b[39mparam_dict[\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mx_test\u001b[39m\u001b[38;5;124m\"\u001b[39m]\u001b[38;5;241m.\u001b[39mvalue \u001b[38;5;241m=\u001b[39m x_test\u001b[38;5;241m.\u001b[39mreshape(\u001b[38;5;241m1\u001b[39m,\u001b[38;5;241m-\u001b[39m\u001b[38;5;241m1\u001b[39m)\n\u001b[0;32m---> 52\u001b[0m \u001b[43mprob\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43msolve\u001b[49m\u001b[43m(\u001b[49m\u001b[43msolver\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;124;43m'\u001b[39;49m\u001b[38;5;124;43mMOSEK\u001b[39;49m\u001b[38;5;124;43m'\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mverbose\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43;01mFalse\u001b[39;49;00m\u001b[43m)\u001b[49m\n\u001b[1;32m     54\u001b[0m var_dict \u001b[38;5;241m=\u001b[39m {}\n\u001b[1;32m     55\u001b[0m var_dict[\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mparams\u001b[39m\u001b[38;5;124m'\u001b[39m] \u001b[38;5;241m=\u001b[39m prob\u001b[38;5;241m.\u001b[39mvar_dict[\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mparams\u001b[39m\u001b[38;5;124m'\u001b[39m]\u001b[38;5;241m.\u001b[39mvalue\n",
      "File \u001b[0;32m/usr/local/lib/python3.9/site-packages/cvxpy/problems/problem.py:473\u001b[0m, in \u001b[0;36mProblem.solve\u001b[0;34m(self, *args, **kwargs)\u001b[0m\n\u001b[1;32m    471\u001b[0m \u001b[38;5;28;01melse\u001b[39;00m:\n\u001b[1;32m    472\u001b[0m     solve_func \u001b[38;5;241m=\u001b[39m Problem\u001b[38;5;241m.\u001b[39m_solve\n\u001b[0;32m--> 473\u001b[0m \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[43msolve_func\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43margs\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43mkwargs\u001b[49m\u001b[43m)\u001b[49m\n",
      "File \u001b[0;32m/usr/local/lib/python3.9/site-packages/cvxpy/problems/problem.py:975\u001b[0m, in \u001b[0;36mProblem._solve\u001b[0;34m(self, solver, warm_start, verbose, gp, qcp, requires_grad, enforce_dpp, **kwargs)\u001b[0m\n\u001b[1;32m    971\u001b[0m     s\u001b[38;5;241m.\u001b[39mLOGGER\u001b[38;5;241m.\u001b[39minfo(\n\u001b[1;32m    972\u001b[0m             \u001b[38;5;124m'\u001b[39m\u001b[38;5;124mInvoking solver \u001b[39m\u001b[38;5;132;01m%s\u001b[39;00m\u001b[38;5;124m  to obtain a solution.\u001b[39m\u001b[38;5;124m'\u001b[39m,\n\u001b[1;32m    973\u001b[0m             solving_chain\u001b[38;5;241m.\u001b[39mreductions[\u001b[38;5;241m-\u001b[39m\u001b[38;5;241m1\u001b[39m]\u001b[38;5;241m.\u001b[39mname())\n\u001b[1;32m    974\u001b[0m start \u001b[38;5;241m=\u001b[39m time\u001b[38;5;241m.\u001b[39mtime()\n\u001b[0;32m--> 975\u001b[0m solution \u001b[38;5;241m=\u001b[39m \u001b[43msolving_chain\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43msolve_via_data\u001b[49m\u001b[43m(\u001b[49m\n\u001b[1;32m    976\u001b[0m \u001b[43m    \u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mdata\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mwarm_start\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mverbose\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mkwargs\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    977\u001b[0m end \u001b[38;5;241m=\u001b[39m time\u001b[38;5;241m.\u001b[39mtime()\n\u001b[1;32m    978\u001b[0m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_solve_time \u001b[38;5;241m=\u001b[39m end \u001b[38;5;241m-\u001b[39m start\n",
      "File \u001b[0;32m/usr/local/lib/python3.9/site-packages/cvxpy/reductions/solvers/solving_chain.py:343\u001b[0m, in \u001b[0;36mSolvingChain.solve_via_data\u001b[0;34m(self, problem, data, warm_start, verbose, solver_opts)\u001b[0m\n\u001b[1;32m    307\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m \u001b[38;5;21msolve_via_data\u001b[39m(\u001b[38;5;28mself\u001b[39m, problem, data, warm_start: \u001b[38;5;28mbool\u001b[39m \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mFalse\u001b[39;00m, verbose: \u001b[38;5;28mbool\u001b[39m \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mFalse\u001b[39;00m,\n\u001b[1;32m    308\u001b[0m                    solver_opts\u001b[38;5;241m=\u001b[39m{}):\n\u001b[1;32m    309\u001b[0m     \u001b[38;5;124;03m\"\"\"Solves the problem using the data output by the an apply invocation.\u001b[39;00m\n\u001b[1;32m    310\u001b[0m \n\u001b[1;32m    311\u001b[0m \u001b[38;5;124;03m    The semantics are:\u001b[39;00m\n\u001b[0;32m   (...)\u001b[0m\n\u001b[1;32m    341\u001b[0m \u001b[38;5;124;03m        a Solution object.\u001b[39;00m\n\u001b[1;32m    342\u001b[0m \u001b[38;5;124;03m    \"\"\"\u001b[39;00m\n\u001b[0;32m--> 343\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43msolver\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43msolve_via_data\u001b[49m\u001b[43m(\u001b[49m\u001b[43mdata\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mwarm_start\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mverbose\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    344\u001b[0m \u001b[43m                                      \u001b[49m\u001b[43msolver_opts\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mproblem\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_solver_cache\u001b[49m\u001b[43m)\u001b[49m\n",
      "File \u001b[0;32m/usr/local/lib/python3.9/site-packages/cvxpy/reductions/solvers/conic_solvers/mosek_conif.py:238\u001b[0m, in \u001b[0;36mMOSEK.solve_via_data\u001b[0;34m(self, data, warm_start, verbose, solver_opts, solver_cache)\u001b[0m\n\u001b[1;32m    236\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m save_file:\n\u001b[1;32m    237\u001b[0m     task\u001b[38;5;241m.\u001b[39mwritedata(save_file)\n\u001b[0;32m--> 238\u001b[0m \u001b[43mtask\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43moptimize\u001b[49m\u001b[43m(\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    240\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m verbose:\n\u001b[1;32m    241\u001b[0m     task\u001b[38;5;241m.\u001b[39msolutionsummary(mosek\u001b[38;5;241m.\u001b[39mstreamtype\u001b[38;5;241m.\u001b[39mmsg)\n",
      "File \u001b[0;32m/usr/local/lib/python3.9/site-packages/mosek/__init__.py:10777\u001b[0m, in \u001b[0;36mTask.optimize\u001b[0;34m(self, *args, **kwds)\u001b[0m\n\u001b[1;32m  10770\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m \u001b[38;5;21moptimize\u001b[39m(\u001b[38;5;28mself\u001b[39m,\u001b[38;5;241m*\u001b[39margs,\u001b[38;5;241m*\u001b[39m\u001b[38;5;241m*\u001b[39mkwds):\n\u001b[1;32m  10771\u001b[0m   \u001b[38;5;124;03m\"\"\"\u001b[39;00m\n\u001b[1;32m  10772\u001b[0m \u001b[38;5;124;03m  Optimizes the problem.\u001b[39;00m\n\u001b[1;32m  10773\u001b[0m \u001b[38;5;124;03m\u001b[39;00m\n\u001b[1;32m  10774\u001b[0m \u001b[38;5;124;03m  optimize() -> (trmcode)\u001b[39;00m\n\u001b[1;32m  10775\u001b[0m \u001b[38;5;124;03m    [trmcode : mosek.rescode]  Is either OK or a termination response code.  \u001b[39;00m\n\u001b[1;32m  10776\u001b[0m \u001b[38;5;124;03m  \"\"\"\u001b[39;00m\n\u001b[0;32m> 10777\u001b[0m   \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m__optimizetrm__1\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43margs\u001b[49m\u001b[43m,\u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43mkwds\u001b[49m\u001b[43m)\u001b[49m\n",
      "File \u001b[0;32m/usr/local/lib/python3.9/site-packages/mosek/__init__.py:10763\u001b[0m, in \u001b[0;36mTask.__optimizetrm__1\u001b[0;34m(self)\u001b[0m\n\u001b[1;32m  10762\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m \u001b[38;5;21m__optimizetrm__1\u001b[39m(\u001b[38;5;28mself\u001b[39m):\n\u001b[0;32m> 10763\u001b[0m   _res_optimizetrm,_retargs_optimizetrm \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m__obj\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43moptimizetrm__1\u001b[49m\u001b[43m(\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m  10764\u001b[0m   \u001b[38;5;28;01mif\u001b[39;00m _res_optimizetrm \u001b[38;5;241m!=\u001b[39m \u001b[38;5;241m0\u001b[39m:\n\u001b[1;32m  10765\u001b[0m     _,_msg_optimizetrm \u001b[38;5;241m=\u001b[39m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m__getlasterror(_res_optimizetrm)\n",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m: "
     ]
    }
   ],
   "source": [
    "group_data = pd.DataFrame()\n",
    "for i in range(25):\n",
    "    x_train, x_calib, y_train, y_calib = train_test_split(X_ny, Y_ny, test_size=3000)\n",
    "\n",
    "    reg = LinearRegression().fit(x_train, y_train)\n",
    "    scores = np.array(np.abs(reg.predict(x_calib) - y_calib))\n",
    "\n",
    "    x_calib = pd.DataFrame.to_numpy(x_calib)\n",
    "\n",
    "    calibration_set_size = 1500  \n",
    "    x_calib_final, x_test, scores_calib, scores_test =  train_test_split(x_calib, scores,\n",
    "                                                            test_size=len(scores) - calibration_set_size)\n",
    "    x_calib_cont = x_calib_final[:,:2]\n",
    "    x_calib_cat = x_calib_final[:,2:]\n",
    "\n",
    "    x_test_cont = x_test[:,:2]\n",
    "    x_test_cat = x_test[:,2:]\n",
    "    adaptive_coverages = compute_group_coverages(\n",
    "        x_calib_cat, scores_calib, scores_test, x_test_cat, x_test_cat, alpha\n",
    "    )\n",
    "    split_coverages = compute_split_coverages(scores_calib,scores_test,x_test_cat,alpha)\n",
    "    df = pd.DataFrame({'group' : list(adaptive_coverages.keys()) +  list(split_coverages.keys()),\n",
    "                        'coverage': list(adaptive_coverages.values()) + list(split_coverages.values()), \n",
    "                        'type' : ['robust']*len(adaptive_coverages) + ['split']*len(split_coverages)\n",
    "                        })\n",
    "    group_data = pd.concat([group_data, df], axis=0)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "group_data_drop = group_data[np.logical_not(np.isnan(group_data['coverage']))].copy()\n",
    "\n",
    "sns.set(rc={'figure.figsize':(11.7,8.27)})\n",
    "\n",
    "fig = sns.barplot(group_data_drop, \n",
    "            x=\"group\", y=\"coverage\", hue=\"type\", errorbar=('ci', 95))\n",
    "fig.axhline(alpha, color='red')\n",
    "plt.legend(bbox_to_anchor=(1.02, 1), loc='upper left', borderaxespad=0)\n",
    "plt.tight_layout()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Run Full Experiment Multiple Times"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 87,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/var/folders/03/1mcjgjk12rv51t4kkk97lq5r0000gn/T/ipykernel_37806/1856564329.py:75: FutureWarning: The behavior of `series[i:j]` with an integer-dtype index is deprecated. In a future version, this will be treated as *label-based* indexing, consistent with e.g. `series[i]` lookups. To retain the old behavior, use `series.iloc[i:j]`. To get the future behavior, use `series.loc[i:j]`.\n",
      "  df_group, df_state_shift = run_one_synthetic_experiment(X_ny, Y_ny, X_ca_mod[0:num_target,:], Y_ca[0:num_target],\n",
      "/usr/local/lib/python3.9/site-packages/sklearn/base.py:409: UserWarning: X does not have valid feature names, but LinearRegression was fitted with feature names\n",
      "  warnings.warn(\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Trial number 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 20/20 [00:55<00:00,  2.78s/it]\n",
      "100%|██████████████████████████████████████| 200/200 [03:46<00:00,  1.13s/it]\n",
      "100%|████████████████████████████████████████| 20/20 [01:52<00:00,  5.62s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[-0.50044043 -0.52218653 -0.54145323 -0.55861117 -0.5739582  -0.58773055\n",
      " -0.60012598 -0.61130626 -0.62140635 -0.63055265 -0.63880019 -0.64627053\n",
      " -0.65301928 -0.65910631 -0.66458322 -0.6694943  -0.67387905 -0.67777137\n",
      " -0.6812014  -0.68419652]\n",
      "Selected Radius For Density Fit: 10.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████| 200/200 [04:10<00:00,  1.25s/it]\n",
      "/var/folders/03/1mcjgjk12rv51t4kkk97lq5r0000gn/T/ipykernel_37806/1856564329.py:75: FutureWarning: The behavior of `series[i:j]` with an integer-dtype index is deprecated. In a future version, this will be treated as *label-based* indexing, consistent with e.g. `series[i]` lookups. To retain the old behavior, use `series.iloc[i:j]`. To get the future behavior, use `series.loc[i:j]`.\n",
      "  df_group, df_state_shift = run_one_synthetic_experiment(X_ny, Y_ny, X_ca_mod[0:num_target,:], Y_ca[0:num_target],\n",
      "/usr/local/lib/python3.9/site-packages/sklearn/base.py:409: UserWarning: X does not have valid feature names, but LinearRegression was fitted with feature names\n",
      "  warnings.warn(\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Trial number 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 20/20 [00:56<00:00,  2.84s/it]\n",
      "100%|██████████████████████████████████████| 200/200 [03:45<00:00,  1.13s/it]\n",
      "100%|████████████████████████████████████████| 20/20 [01:56<00:00,  5.85s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[-0.50047989 -0.52413794 -0.54503063 -0.56357746 -0.58010421 -0.5948818\n",
      " -0.60812583 -0.62001998 -0.63071287 -0.64033061 -0.64897921 -0.65674851\n",
      " -0.66371616 -0.66994827 -0.67550262 -0.68042811 -0.68476927 -0.68856436\n",
      " -0.69184594 -0.69465454]\n",
      "Selected Radius For Density Fit: 10.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████| 200/200 [04:12<00:00,  1.26s/it]\n",
      "/var/folders/03/1mcjgjk12rv51t4kkk97lq5r0000gn/T/ipykernel_37806/1856564329.py:75: FutureWarning: The behavior of `series[i:j]` with an integer-dtype index is deprecated. In a future version, this will be treated as *label-based* indexing, consistent with e.g. `series[i]` lookups. To retain the old behavior, use `series.iloc[i:j]`. To get the future behavior, use `series.loc[i:j]`.\n",
      "  df_group, df_state_shift = run_one_synthetic_experiment(X_ny, Y_ny, X_ca_mod[0:num_target,:], Y_ca[0:num_target],\n",
      "/usr/local/lib/python3.9/site-packages/sklearn/base.py:409: UserWarning: X does not have valid feature names, but LinearRegression was fitted with feature names\n",
      "  warnings.warn(\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Trial number 2\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 20/20 [01:00<00:00,  3.02s/it]\n",
      "100%|██████████████████████████████████████| 200/200 [03:44<00:00,  1.12s/it]\n",
      "100%|████████████████████████████████████████| 20/20 [01:57<00:00,  5.88s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[-0.50039398 -0.51975062 -0.53672612 -0.55176953 -0.56512264 -0.57704922\n",
      " -0.58774913 -0.59740193 -0.60608481 -0.61396618 -0.6211205  -0.62762767\n",
      " -0.63355286 -0.63897155 -0.64388528 -0.64838499 -0.65249195 -0.65623826\n",
      " -0.65965472 -0.66276412]\n",
      "Selected Radius For Density Fit: 10.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████| 200/200 [03:58<00:00,  1.19s/it]\n",
      "/var/folders/03/1mcjgjk12rv51t4kkk97lq5r0000gn/T/ipykernel_37806/1856564329.py:75: FutureWarning: The behavior of `series[i:j]` with an integer-dtype index is deprecated. In a future version, this will be treated as *label-based* indexing, consistent with e.g. `series[i]` lookups. To retain the old behavior, use `series.iloc[i:j]`. To get the future behavior, use `series.loc[i:j]`.\n",
      "  df_group, df_state_shift = run_one_synthetic_experiment(X_ny, Y_ny, X_ca_mod[0:num_target,:], Y_ca[0:num_target],\n",
      "/usr/local/lib/python3.9/site-packages/sklearn/base.py:409: UserWarning: X does not have valid feature names, but LinearRegression was fitted with feature names\n",
      "  warnings.warn(\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Trial number 3\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 20/20 [01:16<00:00,  3.81s/it]\n",
      "100%|██████████████████████████████████████| 200/200 [04:24<00:00,  1.32s/it]\n",
      "100%|████████████████████████████████████████| 20/20 [02:00<00:00,  6.03s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[-0.50039552 -0.51982738 -0.53688456 -0.55195067 -0.56533142 -0.57727263\n",
      " -0.58797515 -0.59760289 -0.60629388 -0.61416035 -0.62129873 -0.62779059\n",
      " -0.6337049  -0.63910095 -0.64403273 -0.6485361  -0.65265823 -0.65643071\n",
      " -0.66014695 -0.66303729]\n",
      "Selected Radius For Density Fit: 10.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████| 200/200 [04:08<00:00,  1.24s/it]\n",
      "/var/folders/03/1mcjgjk12rv51t4kkk97lq5r0000gn/T/ipykernel_37806/1856564329.py:75: FutureWarning: The behavior of `series[i:j]` with an integer-dtype index is deprecated. In a future version, this will be treated as *label-based* indexing, consistent with e.g. `series[i]` lookups. To retain the old behavior, use `series.iloc[i:j]`. To get the future behavior, use `series.loc[i:j]`.\n",
      "  df_group, df_state_shift = run_one_synthetic_experiment(X_ny, Y_ny, X_ca_mod[0:num_target,:], Y_ca[0:num_target],\n",
      "/usr/local/lib/python3.9/site-packages/sklearn/base.py:409: UserWarning: X does not have valid feature names, but LinearRegression was fitted with feature names\n",
      "  warnings.warn(\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Trial number 4\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 20/20 [01:06<00:00,  3.35s/it]\n",
      "100%|██████████████████████████████████████| 200/200 [03:45<00:00,  1.13s/it]\n",
      "100%|████████████████████████████████████████| 20/20 [01:53<00:00,  5.69s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[-0.50031756 -0.51576166 -0.52904569 -0.54047386 -0.55038027 -0.55896712\n",
      " -0.56641264 -0.57287001 -0.57845753 -0.58327493 -0.58740804 -0.59092672\n",
      " -0.59389372 -0.59635778 -0.59836475 -0.59995264 -0.6011487  -0.60200008\n",
      " -0.60251453 -0.602721  ]\n",
      "Selected Radius For Density Fit: 10.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████| 200/200 [04:03<00:00,  1.22s/it]\n",
      "/var/folders/03/1mcjgjk12rv51t4kkk97lq5r0000gn/T/ipykernel_37806/1856564329.py:75: FutureWarning: The behavior of `series[i:j]` with an integer-dtype index is deprecated. In a future version, this will be treated as *label-based* indexing, consistent with e.g. `series[i]` lookups. To retain the old behavior, use `series.iloc[i:j]`. To get the future behavior, use `series.loc[i:j]`.\n",
      "  df_group, df_state_shift = run_one_synthetic_experiment(X_ny, Y_ny, X_ca_mod[0:num_target,:], Y_ca[0:num_target],\n",
      "/usr/local/lib/python3.9/site-packages/sklearn/base.py:409: UserWarning: X does not have valid feature names, but LinearRegression was fitted with feature names\n",
      "  warnings.warn(\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Trial number 5\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 20/20 [01:07<00:00,  3.37s/it]\n",
      "100%|██████████████████████████████████████| 200/200 [03:49<00:00,  1.15s/it]\n",
      "100%|████████████████████████████████████████| 20/20 [01:57<00:00,  5.90s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[-0.50038213 -0.51915367 -0.53563274 -0.55019712 -0.56314629 -0.57472064\n",
      " -0.58511607 -0.59449156 -0.60298193 -0.61069508 -0.61772419 -0.62414833\n",
      " -0.63003782 -0.63543708 -0.64040865 -0.64498957 -0.64921804 -0.65312614\n",
      " -0.65674225 -0.66009138]\n",
      "Selected Radius For Density Fit: 10.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████| 200/200 [04:22<00:00,  1.31s/it]\n",
      "/var/folders/03/1mcjgjk12rv51t4kkk97lq5r0000gn/T/ipykernel_37806/1856564329.py:75: FutureWarning: The behavior of `series[i:j]` with an integer-dtype index is deprecated. In a future version, this will be treated as *label-based* indexing, consistent with e.g. `series[i]` lookups. To retain the old behavior, use `series.iloc[i:j]`. To get the future behavior, use `series.loc[i:j]`.\n",
      "  df_group, df_state_shift = run_one_synthetic_experiment(X_ny, Y_ny, X_ca_mod[0:num_target,:], Y_ca[0:num_target],\n",
      "/usr/local/lib/python3.9/site-packages/sklearn/base.py:409: UserWarning: X does not have valid feature names, but LinearRegression was fitted with feature names\n",
      "  warnings.warn(\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Trial number 6\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 20/20 [01:11<00:00,  3.59s/it]\n",
      "100%|██████████████████████████████████████| 200/200 [05:02<00:00,  1.51s/it]\n",
      "100%|████████████████████████████████████████| 20/20 [01:56<00:00,  5.81s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[-0.50043695 -0.52210938 -0.54151062 -0.55900114 -0.57486706 -0.58938226\n",
      " -0.60261123 -0.61483589 -0.62614419 -0.63663493 -0.64641781 -0.65557955\n",
      " -0.66411994 -0.67216553 -0.67974215 -0.68689204 -0.69365236 -0.70005585\n",
      " -0.70613021 -0.71190083]\n",
      "Selected Radius For Density Fit: 10.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████| 200/200 [05:06<00:00,  1.53s/it]\n",
      "/var/folders/03/1mcjgjk12rv51t4kkk97lq5r0000gn/T/ipykernel_37806/1856564329.py:75: FutureWarning: The behavior of `series[i:j]` with an integer-dtype index is deprecated. In a future version, this will be treated as *label-based* indexing, consistent with e.g. `series[i]` lookups. To retain the old behavior, use `series.iloc[i:j]`. To get the future behavior, use `series.loc[i:j]`.\n",
      "  df_group, df_state_shift = run_one_synthetic_experiment(X_ny, Y_ny, X_ca_mod[0:num_target,:], Y_ca[0:num_target],\n",
      "/usr/local/lib/python3.9/site-packages/sklearn/base.py:409: UserWarning: X does not have valid feature names, but LinearRegression was fitted with feature names\n",
      "  warnings.warn(\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Trial number 7\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 20/20 [01:22<00:00,  4.15s/it]\n",
      "100%|██████████████████████████████████████| 200/200 [04:47<00:00,  1.44s/it]\n",
      "100%|████████████████████████████████████████| 20/20 [01:58<00:00,  5.92s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[-0.50038835 -0.51957485 -0.53662369 -0.55188605 -0.56563893 -0.57810272\n",
      " -0.58945648 -0.59984196 -0.60939041 -0.61820222 -0.62632487 -0.63386875\n",
      " -0.64087463 -0.6474116  -0.65349928 -0.65918881 -0.66450901 -0.66949328\n",
      " -0.67415458 -0.67852607]\n",
      "Selected Radius For Density Fit: 10.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████| 200/200 [04:56<00:00,  1.48s/it]\n",
      "/var/folders/03/1mcjgjk12rv51t4kkk97lq5r0000gn/T/ipykernel_37806/1856564329.py:75: FutureWarning: The behavior of `series[i:j]` with an integer-dtype index is deprecated. In a future version, this will be treated as *label-based* indexing, consistent with e.g. `series[i]` lookups. To retain the old behavior, use `series.iloc[i:j]`. To get the future behavior, use `series.loc[i:j]`.\n",
      "  df_group, df_state_shift = run_one_synthetic_experiment(X_ny, Y_ny, X_ca_mod[0:num_target,:], Y_ca[0:num_target],\n",
      "/usr/local/lib/python3.9/site-packages/sklearn/base.py:409: UserWarning: X does not have valid feature names, but LinearRegression was fitted with feature names\n",
      "  warnings.warn(\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Trial number 8\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 20/20 [01:13<00:00,  3.68s/it]\n",
      "100%|██████████████████████████████████████| 200/200 [03:59<00:00,  1.20s/it]\n",
      "100%|████████████████████████████████████████| 20/20 [01:53<00:00,  5.65s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[-0.50037095 -0.51847961 -0.53416358 -0.54781927 -0.55976176 -0.57024275\n",
      " -0.5794663  -0.58759379 -0.59478214 -0.60112652 -0.60673162 -0.61167688\n",
      " -0.61603145 -0.61985532 -0.62319824 -0.62610291 -0.62861234 -0.63075565\n",
      " -0.63256052 -0.63403513]\n",
      "Selected Radius For Density Fit: 10.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████| 200/200 [03:53<00:00,  1.17s/it]\n",
      "/var/folders/03/1mcjgjk12rv51t4kkk97lq5r0000gn/T/ipykernel_37806/1856564329.py:75: FutureWarning: The behavior of `series[i:j]` with an integer-dtype index is deprecated. In a future version, this will be treated as *label-based* indexing, consistent with e.g. `series[i]` lookups. To retain the old behavior, use `series.iloc[i:j]`. To get the future behavior, use `series.loc[i:j]`.\n",
      "  df_group, df_state_shift = run_one_synthetic_experiment(X_ny, Y_ny, X_ca_mod[0:num_target,:], Y_ca[0:num_target],\n",
      "/usr/local/lib/python3.9/site-packages/sklearn/base.py:409: UserWarning: X does not have valid feature names, but LinearRegression was fitted with feature names\n",
      "  warnings.warn(\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Trial number 9\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 20/20 [01:04<00:00,  3.22s/it]\n",
      "100%|██████████████████████████████████████| 200/200 [03:03<00:00,  1.09it/s]\n",
      "100%|████████████████████████████████████████| 20/20 [01:57<00:00,  5.87s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[-0.50036508 -0.51830771 -0.5340749  -0.54802264 -0.56043593 -0.5715387\n",
      " -0.58151286 -0.59051213 -0.59865812 -0.60605587 -0.61278808 -0.61893165\n",
      " -0.62454771 -0.62968955 -0.63440287 -0.63872755 -0.64269826 -0.64634471\n",
      " -0.64969405 -0.65276884]\n",
      "Selected Radius For Density Fit: 10.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████| 200/200 [03:11<00:00,  1.05it/s]\n",
      "/var/folders/03/1mcjgjk12rv51t4kkk97lq5r0000gn/T/ipykernel_37806/1856564329.py:75: FutureWarning: The behavior of `series[i:j]` with an integer-dtype index is deprecated. In a future version, this will be treated as *label-based* indexing, consistent with e.g. `series[i]` lookups. To retain the old behavior, use `series.iloc[i:j]`. To get the future behavior, use `series.loc[i:j]`.\n",
      "  df_group, df_state_shift = run_one_synthetic_experiment(X_ny, Y_ny, X_ca_mod[0:num_target,:], Y_ca[0:num_target],\n",
      "/usr/local/lib/python3.9/site-packages/sklearn/base.py:409: UserWarning: X does not have valid feature names, but LinearRegression was fitted with feature names\n",
      "  warnings.warn(\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Trial number 10\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 20/20 [01:00<00:00,  3.02s/it]\n",
      "100%|██████████████████████████████████████| 200/200 [03:15<00:00,  1.02it/s]\n",
      "100%|████████████████████████████████████████| 20/20 [02:01<00:00,  6.07s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[-0.50035802 -0.5179049  -0.53324468 -0.54675195 -0.55872223 -0.56939213\n",
      " -0.57895036 -0.58755208 -0.59532375 -0.60237012 -0.60877881 -0.6146234\n",
      " -0.61996682 -0.62486055 -0.62935066 -0.63347608 -0.63727027 -0.64076252\n",
      " -0.64397844 -0.6469399 ]\n",
      "Selected Radius For Density Fit: 10.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████| 200/200 [03:18<00:00,  1.01it/s]\n",
      "/var/folders/03/1mcjgjk12rv51t4kkk97lq5r0000gn/T/ipykernel_37806/1856564329.py:75: FutureWarning: The behavior of `series[i:j]` with an integer-dtype index is deprecated. In a future version, this will be treated as *label-based* indexing, consistent with e.g. `series[i]` lookups. To retain the old behavior, use `series.iloc[i:j]`. To get the future behavior, use `series.loc[i:j]`.\n",
      "  df_group, df_state_shift = run_one_synthetic_experiment(X_ny, Y_ny, X_ca_mod[0:num_target,:], Y_ca[0:num_target],\n",
      "/usr/local/lib/python3.9/site-packages/sklearn/base.py:409: UserWarning: X does not have valid feature names, but LinearRegression was fitted with feature names\n",
      "  warnings.warn(\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Trial number 11\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 20/20 [01:11<00:00,  3.56s/it]\n",
      "100%|██████████████████████████████████████| 200/200 [04:02<00:00,  1.21s/it]\n",
      "100%|████████████████████████████████████████| 20/20 [01:57<00:00,  5.86s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[-0.50040391 -0.52025698 -0.53771235 -0.55315812 -0.56690604 -0.57920531\n",
      " -0.59025934 -0.60022863 -0.60925356 -0.61744715 -0.62490407 -0.6317048\n",
      " -0.63791387 -0.64360371 -0.64881104 -0.65358412 -0.65796273 -0.66198004\n",
      " -0.66566189 -0.66904286]\n",
      "Selected Radius For Density Fit: 10.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████| 200/200 [04:00<00:00,  1.20s/it]\n",
      "/var/folders/03/1mcjgjk12rv51t4kkk97lq5r0000gn/T/ipykernel_37806/1856564329.py:75: FutureWarning: The behavior of `series[i:j]` with an integer-dtype index is deprecated. In a future version, this will be treated as *label-based* indexing, consistent with e.g. `series[i]` lookups. To retain the old behavior, use `series.iloc[i:j]`. To get the future behavior, use `series.loc[i:j]`.\n",
      "  df_group, df_state_shift = run_one_synthetic_experiment(X_ny, Y_ny, X_ca_mod[0:num_target,:], Y_ca[0:num_target],\n",
      "/usr/local/lib/python3.9/site-packages/sklearn/base.py:409: UserWarning: X does not have valid feature names, but LinearRegression was fitted with feature names\n",
      "  warnings.warn(\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Trial number 12\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 20/20 [01:18<00:00,  3.91s/it]\n",
      "100%|██████████████████████████████████████| 200/200 [04:51<00:00,  1.46s/it]\n",
      "100%|████████████████████████████████████████| 20/20 [02:03<00:00,  6.18s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[-0.5002923  -0.5145207  -0.52677096 -0.53737212 -0.54658355 -0.55461318\n",
      " -0.56162895 -0.56776815 -0.57314387 -0.57785014 -0.58196564 -0.58555697\n",
      " -0.58868054 -0.59138472 -0.5937108  -0.59567607 -0.59736789 -0.59875706\n",
      " -0.59988633 -0.60077776]\n",
      "Selected Radius For Density Fit: 10.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████| 200/200 [05:19<00:00,  1.60s/it]\n",
      "/var/folders/03/1mcjgjk12rv51t4kkk97lq5r0000gn/T/ipykernel_37806/1856564329.py:75: FutureWarning: The behavior of `series[i:j]` with an integer-dtype index is deprecated. In a future version, this will be treated as *label-based* indexing, consistent with e.g. `series[i]` lookups. To retain the old behavior, use `series.iloc[i:j]`. To get the future behavior, use `series.loc[i:j]`.\n",
      "  df_group, df_state_shift = run_one_synthetic_experiment(X_ny, Y_ny, X_ca_mod[0:num_target,:], Y_ca[0:num_target],\n",
      "/usr/local/lib/python3.9/site-packages/sklearn/base.py:409: UserWarning: X does not have valid feature names, but LinearRegression was fitted with feature names\n",
      "  warnings.warn(\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Trial number 13\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 20/20 [00:57<00:00,  2.90s/it]\n",
      "100%|██████████████████████████████████████| 200/200 [03:50<00:00,  1.15s/it]\n",
      "100%|████████████████████████████████████████| 20/20 [01:59<00:00,  5.96s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[-0.50042051 -0.52125447 -0.53986428 -0.556603   -0.57175215 -0.58553844\n",
      " -0.59814708 -0.60972991 -0.62041668 -0.63029999 -0.63948219 -0.64803378\n",
      " -0.65604321 -0.66349315 -0.67050362 -0.67709176 -0.68329375 -0.68913998\n",
      " -0.69466015 -0.69987587]\n",
      "Selected Radius For Density Fit: 10.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████| 200/200 [04:02<00:00,  1.21s/it]\n",
      "/var/folders/03/1mcjgjk12rv51t4kkk97lq5r0000gn/T/ipykernel_37806/1856564329.py:75: FutureWarning: The behavior of `series[i:j]` with an integer-dtype index is deprecated. In a future version, this will be treated as *label-based* indexing, consistent with e.g. `series[i]` lookups. To retain the old behavior, use `series.iloc[i:j]`. To get the future behavior, use `series.loc[i:j]`.\n",
      "  df_group, df_state_shift = run_one_synthetic_experiment(X_ny, Y_ny, X_ca_mod[0:num_target,:], Y_ca[0:num_target],\n",
      "/usr/local/lib/python3.9/site-packages/sklearn/base.py:409: UserWarning: X does not have valid feature names, but LinearRegression was fitted with feature names\n",
      "  warnings.warn(\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Trial number 14\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████| 20/20 [01:11<00:00,  3.59s/it]\n",
      "100%|██████████████████████████████████████| 200/200 [04:59<00:00,  1.50s/it]\n",
      "100%|████████████████████████████████████████| 20/20 [01:56<00:00,  5.81s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[-0.50043515 -0.52188059 -0.54083415 -0.55770809 -0.57287476 -0.58642618\n",
      " -0.59875131 -0.60996313 -0.62020593 -0.6295997  -0.63824507 -0.6462274\n",
      " -0.65363792 -0.66048267 -0.66687337 -0.6728308  -0.67840235 -0.68362048\n",
      " -0.68851614 -0.69311644]\n",
      "Selected Radius For Density Fit: 10.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████| 200/200 [04:45<00:00,  1.43s/it]\n"
     ]
    }
   ],
   "source": [
    "def run_one_synthetic_experiment(X_base, Y_base, X_target, Y_target,\n",
    "                                 num_calib, num_test, gamma, alpha, k, min_radius, max_radius, num_radii,\n",
    "                                min_radius_density_shift, max_radius_density_shift):\n",
    "    \n",
    "    x_calib_cont, x_calib_cat, x_test_cont, x_test_cat, scores_calib, scores_test, reg = make_train_test_split(\n",
    "                                                                                                    X_base,Y_base,\n",
    "                                                                                                    num_calib,\n",
    "                                                                                                    num_test)\n",
    "    scores_target = np.array(np.abs(reg.predict(X_target) - Y_target))\n",
    "    x_target_cont = X_target[:,:2]\n",
    "    x_target_cat = X_target[:,2:]\n",
    "\n",
    "\n",
    "    adaptive_coverages_test, split_coverages_test, radius = compute_adaptive_and_split_coverage(x_calib_cont,\n",
    "                                                                          x_calib_cat,\n",
    "                                                                          x_test_cont,x_test_cat,\n",
    "                                                                          scores_calib,scores_test,\n",
    "                                                                          gamma, alpha, k,\n",
    "                                                                          min_radius, max_radius,\n",
    "                                                                          num_radii\n",
    "                                                                          )\n",
    "    ### Get group coverage\n",
    "    group_data = pd.DataFrame()\n",
    "\n",
    "    for i in range(x_test_cat.shape[1]):\n",
    "        if sum(x_test_cat[:,i]) > 0:\n",
    "            group_cov_adaptive = (x_test_cat[:,i] @ adaptive_coverages_test)/sum(x_test_cat[:,i])\n",
    "            group_split_adaptive = (x_test_cat[:,i] @ split_coverages_test)/sum(x_test_cat[:,i])\n",
    "    \n",
    "            df = pd.DataFrame({'group' : [X_base.columns[i+2],X_base.columns[i+2]],\n",
    "                            'coverage': [group_cov_adaptive,group_split_adaptive], \n",
    "                            'type' : ['robust','split'],\n",
    "                            'sample_size' : [sum(x_test_cat[:,i]),sum(x_test_cat[:,i])]\n",
    "                          })\n",
    "\n",
    "            group_data = pd.concat([group_data, df], axis=0)\n",
    "            \n",
    "            \n",
    "    ### Compare shift in POVPIP + Age for two states\n",
    "    est_cov = compute_coverage_under_density_shift(x_calib_cont,scores_calib,x_target_cont,\n",
    "                                                   \"rbf\",gamma,alpha,radius,\n",
    "                                                   k, min_radius_density_shift, max_radius_density_shift, num_radii)\n",
    "    adaptive_coverages_target, split_coverages_target, _ = compute_adaptive_and_split_coverage(x_calib_cont,\n",
    "                                                                          x_calib_cat,\n",
    "                                                                          x_target_cont[0:num_test,:],\n",
    "                                                                          x_target_cat[0:num_test,:],\n",
    "                                                                          scores_calib,scores_target[0:num_test],\n",
    "                                                                          gamma, alpha, radius = radius\n",
    "                                                                          )\n",
    "    state_shift_df = pd.DataFrame({'coverage': [sum(adaptive_coverages_target)/len(adaptive_coverages_target),\n",
    "                                                sum(split_coverages_target)/len(adaptive_coverages_target),est_cov], \n",
    "                            'type' : ['robust','split','estimated'],\n",
    "                          })\n",
    "    \n",
    "    return group_data, state_shift_df\n",
    "\n",
    "gamma = 0.5\n",
    "alpha = 0.9\n",
    "k = 5\n",
    "min_radius = 1\n",
    "max_radius = 2000\n",
    "num_radii = 20\n",
    "num_calib = 1000\n",
    "num_test = 200\n",
    "num_target = 1000\n",
    "min_radius_density_shift = 0.01\n",
    "max_radius_density_shift = 100\n",
    "                                 \n",
    "n_trials = 15\n",
    "\n",
    "X_ca_mod = pd.DataFrame.to_numpy(X_ca)\n",
    "group_data = pd.DataFrame()\n",
    "state_shift_data = pd.DataFrame()\n",
    "for i in range(n_trials):\n",
    "    print(\"Trial number\",i)\n",
    "    df_group, df_state_shift = run_one_synthetic_experiment(X_ny, Y_ny, X_ca_mod[0:num_target,:], Y_ca[0:num_target],\n",
    "                                                            num_calib, num_test, gamma, alpha, k, min_radius,\n",
    "                                                            max_radius, num_radii, min_radius_density_shift, \n",
    "                                                            max_radius_density_shift)\n",
    "    group_data = pd.concat([group_data, df_group], axis=0)\n",
    "    state_shift_data = pd.concat([state_shift_data,df_state_shift], axis=0)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 88,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAz8AAAJHCAYAAAC+U+hbAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjUuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/YYfK9AAAACXBIWXMAAAsTAAALEwEAmpwYAAA4CUlEQVR4nO3de5zWdZ03/tcMw0EEFRVQ225PpKChpaZ4TgtRVzxBnjCylCQtyh6Kbhp5TGO9b7LsV2Fl2UKKZCq1q2SWZ7dFSyulUsvTylGUAYGZYa7fH66zEmbDOBczzOf5/Oea7/U9vd/MNRfX6/p8DzWVSqUSAACALq62owsAAABYH4QfAACgCMIPAABQBOEHAAAogvADAAAUoa6jCwAAANZdY2Njnn/++axYsbKjS+k0unXrls0375ctt9wytbVrj/PUbMiXul6yZHmamzfY8gEAKFxtbU369du4Tes+88wzqavrmT59Nk1NTU07V7bhqVQqWb26KUuXLkldXW223XbbtZbZoEd+mpsrwg8AAEVasWJltt56gODzP2pqalJX1z39+m2ZefOef8tlnPMDAAAbKMFnbTU1tfl7x7YJPwAAQBE26MPeAACA/9V7457p2aP9P+KvamjKa8tXtft21zfhBwAAuoiePepyysRp7b7d6ZPHtEv4+elPb8+jjz6SSZMueUfbue++e/L888/nlFNOXaf1hB8AAGCDMnfuk21aT/gBAADesUcemZNvfOOarF69Ou961z+lrq57nnrqT6mpqcmYMWNz5JFHJUleeOH5jB9/el599dUccMBBOeusz+Sll17KWWeNy623/ixJct1130qSfPzjp+fyyy/JM888nSQ5/viPZPfd35ef/OTHSZKtt946Rx11TKtrFH4AAIB28dxzz+bWW3+W66//bhoaGjJ9+s155ZUl+cQnxuY979kpSfLSSy/mhhtuTJ8+fXL22WfmvvvuyaBBO73l9h5//PEsXbo0N9zwo7z66iu55popOfbY43PccaOSZJ2CT+JqbwAAQDv5P/9nu/Tp0zePPPJfOfroY5Mkm23WLwcddHAeffSRJMkBBxycfv36pXv37vnQh4bn0Ufn/N3t7bjjjnnuub/ms589K3fc8e85++wJ76g+4QcAAGgXPXv2TJI0N695o51KpZLVq1cnSbp16/bmOenWrS6v367of9dpampKkmy66WaZPn1mPvKRk/Lss8/mtNNOSX19fZvrE34AAIB2tddeH8jtt9+aJHnllSW5555fZY899kySPPTQA6mvr8+qVasye/Yd2XvvfdK3b98sXVqfJUuWpKGhIQ8//GCS5N5778nFF1+U/fc/MJ///HnZaKPemT9/Xrp1q2sJU+vCOT8AANBFrGpoyvTJY6qy3XVx+unjMnnylRkz5oSsXr06p512egYPHpKnnvpztt12u3z+859JfX19DjvsiOyzz75JklNPHZuPf/zUDBw4MLvs8t4kyX777Zdf/vKunHzy6PTo0TMf/OChGTToPamvX5pLL/1SNt98i5xwwkmtrqumUqlU/vFindPixcvWGlIDAIANRW1tTbbYok+b1v3DH57INtts284VdQ3//d/PZtddd1nreYe9AQAARRB+AACAIgg/AABAEYQfAACgCMIPAABQBOEHAAAogvv8AABAF9F34+6p69Gj3bfb1NCQ+uWN7ba9T31qXM4448z07t07t9wyMxdeOCm33vrj9O69cQ477PB228/fEn4AAKCLqOvRI49MPqPdt7vnxO8k7Rh+3jBkyC658MJJSZLHH388e+yxZ7vv482qftjbsmXLctRRR+WFF15Ya96TTz6ZUaNGZcSIEbnwwgvT1LRud44FAAA6hwUL5udTnzojp502Jp/4xEfz+98/nmOP/ef8v//3rxk79uSMHXty/vjHuWus88gjc/KpT43Lr3/9n7n//nty3XXfysMPP1i1Gqsafh577LGcfPLJ+etf//qW888777x88YtfzJ133plKpZIZM2ZUsxwAAKBKbr/91uy//4H5/ven5eyzP5vHHvttkmSTTTbJDTf8KOPGjc+ll056y3X33nufHHDAwRk3bnyGDduvajVWNfzMmDEjX/rSlzJgwIC15r344otZuXJl3ve+9yVJjj/++Nxxxx3VLAcAivPoo3NyySUX5tFH53R0KUAX94EP7JNp036YSZO+kIULF2T06BOTJMcee3yS5MADD87ChQvyyitLOqzGqp7zc8UVV/zdeQsWLEj//v1bpvv375/58+ev0/a32KJPm2sDuoaGxtXp0b1bVfexurEh3bq3/8mjb9bc1Jjauu5V3Qdl+slPbsqf//znNDU1ZMSIQzq6nHdkffy9J53zb17vZfa+odl99/flxhtn5v7778tdd83Oz342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fgAAgCIIPwAAQBGEHwAAoAjCDwAAUAThBwAAKILwAwAAFEH4AQAAiiD8AAAARRB+AACAIgg/AABAEYQfAACgCMIPAABQBOEHAAAogvADAAAUQfgBAACKIPwAAABFEH4AAIAiCD8AAEARhB8AAKAIwg8AAFAE4QcAACiC8AMAABRB+AEAAIog/AAAAEUQfgAAgCIIPwAAQBGEHwAAoAjCDwAAUAThBwAAKEJVw8+sWbNy5JFHZvjw4Zk2bdpa85955pl89KMfzdFHH53TTz89r776ajXLAQAACla18DN//vxMmTIl06dPz2233ZabbropTz31VMv8SqWST33qUxk3blxuv/32DBkyJFOnTq1WOQAAQOGqFn4efPDBDBs2LJtttll69+6dESNG5I477miZ/4c//CG9e/fOQQcdlCQZP358xowZU61yAACAwtVVa8MLFixI//79W6YHDBiQxx9/vGX6ueeey5Zbbpnzzz8/TzzxRHbaaad88YtfXKd9bLFFn3art7Pq379vR5fQYfReJr2XqeTek7L713uZ9E5HqVr4qVQqaz1XU1PT8nNTU1N+/etf59/+7d8ydOjQfPWrX81VV12Vq666qtX7WLx4WZqb195PV3pRLVxYv07L671r0Hvr6b1rKLn3ZN3613vXoffW6eq919bWFPGFfmdRtcPeBg4cmEWLFrVML1iwIAMGDGiZ7t+/f7bddtsMHTo0SXLUUUetMTIEAADQnqoWfvbbb7889NBDefnll7NixYrMnj275fyeJHn/+9+fl19+OXPnzk2S3H333dl1112rVQ4AAFC4qh32NnDgwJxzzjkZO3ZsGhsbM3r06Oy2224ZN25cJkyYkKFDh+Yb3/hGLrrooqxYsSJbbbVVJk+eXK1yAACAwlUt/CTJyJEjM3LkyDWeu+6661p+3n333TNz5sxqlgAAAJCkyjc5BQAA6CyEHwAAoAjCDwAAUAThBwAAKILwAwAAFEH4AQAAiiD8AAAARRB+AACAIgg/AABAEYQfAACgCMIPAABQhFaFn+XLl+eSSy7Jxz72sbzyyiuZNGlSli9fXu3aAAAA2k2rws/ll1+eTTbZJIsXL07Pnj2zbNmyTJo0qdq1AQAAtJtWhZ8nn3wy55xzTurq6rLRRhvl6quvzpNPPlnt2gAAANpNq8JPbe2ai61evXqt5wAAADqzutYs9IEPfCD/+q//mpUrV+a+++7LtGnTss8++1S7NgAAgHbTquGbc889N717907fvn0zZcqU7Lzzzpk4cWK1awMAAGg3rRr56d69e84+++ycffbZ1a4HAACgKloVfg499NDU1NS0TNfU1GSjjTbKe97znlxwwQUZMGBA1QoEAABoD60KPx/+8IezfPnyjBkzJrW1tZk5c2aWL1+enXfeOZMmTcq3vvWtatcJAADwjrTqnJ85c+bkiiuuyC677JLBgwfnoosuyp///OecdtppefHFF6tdIwAAwDvWqvCzfPnyLFu2rGV62bJlWblyZdWKAgAAaG+tOuxt1KhROeGEE3L44YenUqlk9uzZ+chHPpIf/vCH2WGHHapdIwAAwDvWqvDzyU9+MkOGDMm9996burq6fPGLX8ywYcPy+9//Pscdd1y1awQAAHjHWhV+kmTo0KEZNGhQKpVKVq9enQceeCD7779/NWsDAABoN60KP9dcc02mTp36+gp1dWloaMigQYMya9asqhYHAADQXlp1wYPbbrstv/zlLzNixIjceeedueqqqzJo0KBq1wYAANBuWhV+Nt988wwYMCA77LBD5s6dm2OOOSbPPvtstWsDAABoN60KP3V1dXnuueeyww47ZM6cOWlqasrSpUurXRsAAEC7aVX4GT9+fL74xS/mgx/8YH7+85/ngx/8YIYNG1bt2gAAANpNqy540NTUlB/84AdJkltvvTXPPvtsdt5556oWBgAA0J5aNfIzZcqUlp832mijDB48ODU1NVUrCgAAoL21auRnp512yje/+c3stdde6d27d8vzu+66a9UKAwAAaE+tCj+PPfZYHnvssdx8880tz9XU1OQXv/hF1QoDAABoT60KP3fffXe16wAAAKiqVp3zs3z58lx66aX52Mc+lldeeSWTJk3K8uXLq10bAABAu2lV+Ln88svTt2/fLF68OD179syyZcsyadKkatcGAADQbloVfp588smcc845qaury0YbbZSrr746Tz75ZLVrAwAAaDetCj+1tWsutnr16rWeAwAA6MxadcGDD3zgA/nXf/3XrFy5Mvfdd1+mTZuWffbZp9q1AQAAtJtWDd+ce+656d27d/r27ZspU6Zk5513zsSJE6tdGwAAQLtp1cjPww8/nLPPPjtnn312tesBAACoilaN/Fx77bU59NBD841vfCPz58+vdk0AAADtrlXh56abbsp1112X1157LSeccELOPPPM3HXXXdWuDQAAoN20+pJtO+64Y84777x8/etfz5IlS/L5z3++mnUBAAC0q1ad87N48eLcfvvt+clPfpLVq1dn9OjR+fa3v13t2gAAANpNq8LPYYcdlsMOOyxf+tKXsueee1a7JgAAgHbXqvBzzz33pE+fPnnxxRfz7LPPZtttt612XQAAAO2q1Ye9nXjiiZk/f34qlUr69euXb3/729lxxx2rXR8AAEC7aNUFDy699NKcccYZmTNnTh555JF86lOfyiWXXFLt2gAAANpNq8LP4sWLc9xxx7VMjxo1KkuWLKlaUQAAAO2tVeFn9erVeeWVV1qmX3755WrVAwAAUBWtOufn1FNPzYknnpgjjjgiSfIf//Ef+djHPlbVwgAAANpTq0Z+Dj744CRJY2NjnnnmmcyfPz/Dhw+vamEAAADtqVUjPxdccEHGjBmTsWPHZtWqVfnRj36UL3zhC7nuuuuqXR8AAEC7aNXIz5IlSzJ27NgkSc+ePXPaaadl4cKFVS0MAACgPbX6ggfz589vmV60aFEqlUrVigIAAGhvrTrs7bTTTsuxxx6bAw88MDU1NXnwwQczceLEatcGAADQbloVfkaPHp33vve9efjhh9OtW7ecfvrp2WmnnapdGwAAQLtpVfhJksGDB2fw4MHVrAUAAKBqWnXODwAAwIZO+AEAAIog/AAAAEUQfgAAgCIIPwAAQBGEHwAAoAjCDwAAUAThBwAAKILwAwAAFEH4AQAAiiD8AAAARRB+AACAIgg/AABAEYQfAACgCMIPAABQBOEHAAAogvADAAAUQfgBAACKIPwAAABFEH4AAIAiCD8AAEARhB8AAKAIwg8AAFAE4QcAACiC8AMAABRB+AEAAIog/AAAAEUQfgAAgCIIPwAAQBGEHwAAoAjCDwAAUAThBwAAKILwAwAAFKGq4WfWrFk58sgjM3z48EybNu3vLverX/0qhx56aDVLAQAACldXrQ3Pnz8/U6ZMyS233JIePXrkpJNOyj777JNBgwatsdyiRYvyla98pVplAAAAJKniyM+DDz6YYcOGZbPNNkvv3r0zYsSI3HHHHWstd9FFF+XTn/50tcoAAABIUsWRnwULFqR///4t0wMGDMjjjz++xjI33HBDdtlll+y+++5t2scWW/R5RzVuCPr379vRJXQYvZdJ72Uqufek7P71Xia901GqFn4qlcpaz9XU1LT8/Kc//SmzZ8/O97///cybN69N+1i8eFmam9feT1d6US1cWL9Oy+u9a9B76+m9ayi592Td+td716H31unqvdfW1hTxhX5nUbXD3gYOHJhFixa1TC9YsCADBgxomb7jjjuycOHCjBo1Kp/85CezYMGCnHLKKdUqBwAAKFzVws9+++2Xhx56KC+//HJWrFiR2bNn56CDDmqZP2HChNx555257bbbMnXq1AwYMCDTp0+vVjkAAEDhqjryc84552Ts2LE59thjc9RRR2W33XbLuHHj8rvf/a5auwUAAHhLVTvnJ0lGjhyZkSNHrvHcddddt9Zy//RP/5S77767mqUAAACFq+pNTgEAADoL4QcAACiC8AMAABRB+AEAAIog/AAAAEUQfgAAgCIIPwAAQBGEHwAAoAjCDwAAUAThBwAAKILwAwAAFEH4AQAAiiD8AAAARRB+AACAIgg/AABAEYQfAACgCMIPAABQBOEHAAAogvADAAAUQfgBAACKIPwAAABFEH4AAIAiCD8AAEARhB8AAKAIwg8AAFAE4QcAACiC8AMAABRB+AEAAIog/AAAAEUQfgAAgCIIPwAAQBGEHwAAoAjCDwAAUAThBwAAKILwAwAAFEH4AQAAiiD8AAAARRB+AACAIgg/AABAEYQfAACgCMIPAABQBOEHAAAogvADAAAUQfgBAACKIPwAAABFEH4AAIAiCD8AAEARhB8AAKAIwg8AAFAE4QcAACiC8AMAABRB+AEAAIog/AAAAEUQfgAAgCIIPwAAQBGEHwAAoAjCDwAAUAThBwAAKILwAwAAFEH4AQAAiiD8AAAARRB+AACAIgg/AABAEYQfAACgCMIPAABQBOEHAAAogvADAAAUQfgBAACKIPwAAABFEH4AAIAiCD8AAEARhB8AAKAIwg8AAFAE4QcAACiC8AMAABRB+AEAAIog/AAAAEUQfgAAgCIIPwAAQBGEHwAAoAjCDwAAUAThBwAAKILwAwAAFEH4AQAAiiD8AAAARRB+AACAIgg/AABAEYQfAACgCMIPAABQBOEHAAAogvADAAAUQfgBAACKIPwAAABFqGr4mTVrVo488sgMHz4806ZNW2v+XXfdlWOOOSZHH310zjrrrLz66qvVLAcAAChY1cLP/PnzM2XKlEyfPj233XZbbrrppjz11FMt85ctW5aLL744U6dOze23356dd945X//616tVDgAAULiqhZ8HH3www4YNy2abbZbevXtnxIgRueOOO1rmNzY25uKLL87AgQOTJDvvvHNeeumlapUDAAAUrq5aG16wYEH69+/fMj1gwIA8/vjjLdP9+vXLhz/84STJypUrM3Xq1Hz0ox9dp31ssUWf9im2E+vfv29Hl9Bh9F4mvZep5N6TsvvXe5n0TkepWvipVCprPVdTU7PWc/X19TnrrLMyePDgHHfcceu0j8WLl6W5ee39dKUX1cKF9eu0vN67Br23nt67hpJ7T9atf713HXpvna7ee21tTRFf6HcWVTvsbeDAgVm0aFHL9IIFCzJgwIA1llmwYEFOOeWUDB48OFdccUW1SgEAAKhe+Nlvv/3y0EMP5eWXX86KFSsye/bsHHTQQS3zV69enfHjx+eII47IhRde+JajQgAAAO2laoe9DRw4MOecc07Gjh2bxsbGjB49OrvttlvGjRuXCRMmZN68eXniiSeyevXq3HnnnUmS9773vUaAAACAqqha+EmSkSNHZuTIkWs8d9111yVJhg4dmrlz51Zz9wAAAC2qepNTAACAzkL4AQAAiiD8AAAARRB+AACAIgg/AABAEYQfAACgCMIPAABQBOEHAAAogvADAAAUQfgBAACKIPwAAABFEH4AAIAiCD8AAEARhB8AAKAIwg8AAFAE4QcAACiC8AMAABRB+AEAAIog/AAAAEUQfgAAgCIIPwAAQBGEHwAAoAjCDwAAUAThBwAAKILwAwAAFEH4AQAAiiD8AAAARRB+AACAIgg/AABAEYQfAACgCMIPAABQBOEHAAAogvADAAAUQfgBAACKIPwAAABFEH4AAIAiCD8AAEARhB8AAKAIwg8AAFAE4QcAACiC8AMAABRB+AEAAIog/AAAAEUQfgAAgCIIPwAAQBGEHwAAoAjCDwAAUAThBwAAKILwAwAAFEH4AQAAiiD8AAAARRB+AACAIgg/AABAEYQfAACgCMIPAABQBOEHAAAogvADAAAUQfgBAACKIPwAAABFEH4AAIAiCD8AAEARhB8AAKAIwg8AAFAE4QcAACiC8AMAABRB+AEAAIog/AAAAEUQfgAAgCIIPwAAQBGEHwAAoAjCDwAAUAThBwAAKILwAwAAFEH4AQAAiiD8AAAARRB+AACAIgg/AABAEYQfAACgCMIPAABQBOEHAAAogvADAAAUQfgBAACKIPwAAABFEH4AAIAiCD8AAEARhB8AAKAIwg8AAFAE4QcAACiC8AMAABRB+AEAAIog/AAAAEUQfgAAgCIIPwAAQBGqGn5mzZqVI488MsOHD8+0adPWmv/kk09m1KhRGTFiRC688MI0NTVVsxwAAKBgVQs/8+fPz5QpUzJ9+vTcdtttuemmm/LUU0+tscx5552XL37xi7nzzjtTqVQyY8aMapUDAAAUrq5aG37wwQczbNiwbLbZZkmSESNG5I477sinP/3pJMmLL76YlStX5n3ve1+S5Pjjj8/Xvva1nHLKKa3eR21tzd+dt2W/jdtce2v12GSLqu/j7Xr8e/ReXXqvLr2vG71X1/roPVn3/vVefZ3xda/36uuI3tvy/kfb1VQqlUo1Nvztb387r732Ws4555wkyc0335zHH388l112WZLkN7/5TSZPnpwf/ehHSZJnn302n/zkJ3PnnXdWoxwAAKBwVTvs7a0yVU1NTavnAwAAtKeqhZ+BAwdm0aJFLdMLFizIgAED/u78hQsXrjEfAACgPVUt/Oy333556KGH8vLLL2fFihWZPXt2DjrooJb573rXu9KzZ8888sgjSZJbb711jfkAAADtqWrn/CSvX+r629/+dhobGzN69OiMGzcu48aNy4QJEzJ06NDMnTs3F110UZYvX55ddtklV155ZXr06FGtcgAAgIJVNfwAAAB0FlW9ySkAAEBnIfwAAABFEH4AAIAiCD8AAEARhB8AAKAIwg8AAFCEuo4uoKPccccdmTp1apqamlKpVHLMMcfkjDPOyEc/+tHMmzcvvXv3bll2yy23zHe/+938y7/8SyqVSq666qokSUNDQ0466aScffbZ+dCHPvS2+2tsbMwZZ5yRs846K/vss09Ve/tH1mfvN910U374wx+mpqYm733ve3PJJZdU9V5OL7zwQg4//PDsuOOOSZLm5uYsX748xx57bCZMmJAk+dOf/pSRI0fma1/7WkaMGLHG+rfeemt++MMfpqmpKc3NzfnIRz6SsWPHrrHMNddck9ra2nzmM59JklxwwQV5+OGHs+mmmyZ5/d9mzJgxOfXUU/9hvcuWLctJJ52Ub33rW/mnf/qnYnq/9tpr8x//8R9JkoMPPjgTJ04spvdrrrkmd955Z2pqajJ69Oh8/OMff0e9b2j933333bn22mvz2muv5YADDshFF11UTO+33XZbpk6dmiQ56KCDcv755xfT+9SpU/PjH/84PXr0yJFHHplPfepTRfR+880359/+7d/WqPuYY47JpEmTunzvb/aVr3wlS5YsafkMUULvY8eOzeLFi1NX9/pH7UsvvTS77777O+qfdlIp0Lx58yof/OAHKy+//HKlUqlUli1bVjnuuOMqd911V+XUU0+tPPzww2+5Xn19feVDH/pQZfbs2ZVKpVK55JJLKpdddtk/3N/TTz9dOfHEEytDhw79u9teX9Zn788880xl+PDhlfr6+kpzc3Nl4sSJleuvv75d+/lbzz//fOWQQw5Z47l58+ZVdt9998pTTz1VqVQqlSuvvLLymc98pnLaaaetsdyNN95YOfbYYyvz58+vVCqVyquvvloZNWpUZcaMGZVKpVJZunRp5V/+5V8qu+22W+VrX/tay3rnn39+5cc//nHL9MKFCyt77bVX5YknnnjbWn/7299WjjrqqMquu+5aef7559ve9P/YUHp/4IEHKieeeGJl1apVlYaGhsrYsWNbXlddvff//M//rJx00kmVxsbGyooVKyqHHHJI5emnn35HvVcqG07/zz33XOWAAw6ovPTSS5WGhobKySefXPnVr35VRO+vvfZa5QMf+EBl8eLFlcbGxsro0aMrDzzwQBG9P/DAA5WjjjqqUl9fX2lqaqqceeaZlTvvvLOI3t/sT3/6U2X48OGVxYsXr3vDb7Kh9f7ggw9W9tlnn8r555/ftobfZEPpvbm5ubL//vtXGhsb31nDVEWRh70tWbIkjY2NWblyZZJk4403zlVXXZVBgwa97Xp9+vTJ5MmTc/HFF2fmzJn57W9/26pvrGfOnJkzzjijUyT+9dl7jx49cvHFF6dPnz6pqanJTjvtlP/+7/9ut15aa+HChalUKtl4443T1NSU22+/Peecc06eeOKJPPfccy3LffOb38wFF1yQAQMGJEk22WSTfOUrX8lOO+2UJPnFL36R7bbb7h9+U7/llltmu+22y1//+te3XW7GjBn50pe+1LK/auiMvffv3z8XXHBBevToke7du2fHHXesyuuiM/a+995754YbbkhdXV0WL16c1atXrzHS2p46Y/8///nPc+SRR2arrbZK9+7dM2XKlKq8L3bG3levXp3m5uasWLEiTU1NaWpqSs+ePd95s3+jM/b+xBNP5IADDkifPn3SrVu3HHjggbnrrrveebN/ozP2/mYXX3xxzjnnnGy++eZta/BtdNbeX3nllUyZMiXjx49/Zw2+jc7Y+zPPPJOampqMGzcuRx999Bqjf3S8Ig97Gzx4cD70oQ/lwx/+cIYMGZJ99tknI0eOzLbbbpskueiii9b4QHL44Ye3DNHvscceOe644zJp0qT87Gc/a9UhXG+EhB/84AdV6GbdrM/e3/Wud+Vd73pXkuTll1/OtGnTcuWVV1aps/+1YMGCHHPMMVm1alWWLFmSoUOH5tprr81WW22Vu+66K9tss0223377fPjDH86NN96YiRMn5uWXX85LL7201gexN4bWk+TYY49Nknz9619/2/3PnTs3f/nLXzJ06NC3Xe6KK65oW4NvY0Po/T3veU/Lz3/961/z7//+77nxxhvb0O2aNoTek6R79+752te+lu9973s5/PDDM3DgwLY1/Dc2hP6fffbZdO/ePaeffnoWLlyYQw45JJ/73Ofa3PMbNoTe+/Tpk89+9rM54ogj0qtXr+y9997ZY4892t70/9gQet91113z5S9/OWeeeWY22mij3H333alUKm1v+n9sCL2/4cEHH8zKlStzxBFHrHujb2FD6X3SpEk555xz8tJLL7Wt0bewIfS+dOnS7Lvvvrn44ouzcuXKjB07Nttvv33233//tjdOuyky/CTJJZdckrPOOiv3339/7r///pxwwgm5+uqrkySXX3753z0vZ/Xq1fnNb36Tfv365YEHHsj222+/PstuF+u79/nz5+eMM87IqFGj1sv5TgMGDMhtt92W5ubmXHXVVfnjH/+YYcOGJUluueWWHHXUUUmSI488Mueee24+97nPpbb29UHQtv6H/LWvfS0/+MEP0tzcnF69euXSSy99x+fwtMWG1Puf//znnHnmmTn//POz3XbbtWnfb7Yh9T5hwoSMGzcu48ePz4wZM3LiiSe2af9vtiH0v3r16syZMyc//OEP07t375x11ln5yU9+kuOPP75N+3/DhtD73Llz8+Mf/zi//OUv07dv35x77rn57ne/mzPOOKNN+3/DhtD7vvvum+OPPz4f/ehHs9lmm2XffffNY4891qZ9v9mG0PsbbrzxxnY5v+8NG0LvN998c7beeuvsu+++ueWWW9q0z7eyIfT+/ve/P+9///uTJL17987o0aNzzz33CD+dRJHh51e/+lVee+21HHnkkRk1alRGjRqVGTNmZObMmf9w3W984xvZZJNN8v3vfz9jxozJvvvuu8Y3B53d+u796aefzrhx43LqqafmE5/4RHu10Sq1tbWZOHFijj322Hzve9/L6NGjc++99+b3v/99brjhhlQqlSxdujSzZ8/OUUcdlXe/+935/e9/nw984AMt2/j1r3+de++9N+eee+7b7mvChAnv+ANce+rsvT/yyCOZMGFCvvCFL+Sf//mf29Tj39OZe3/66afT0NCQIUOGZKONNsphhx2WP/7xj23u9a105v633HLL7Lvvvi2H/XzoQx/K448/3m5/O5259/vvvz/77rtvtthiiyTJ8ccfn+nTp7/j8POGztz7smXLMnz48JYP/9dff33e/e53t63Rt9CZe09eP0H+v/7rv97xyf5vpTP3/u///u9ZuHBhjjnmmLz66qt57bXX8uUvfzlf+MIX2tzvm3Xm3ufMmZPGxsbsu+++SV4PXW9c+ICOV+Q5P7169cr//b//Ny+88EKS11+UTz31VIYMGfK26/3617/OzJkzc8UVV+Q973lPzjrrrJx77rlpaGhYH2W3i/XZ+7Jly3L66afns5/97HoPPm+oq6vLxIkT861vfSs333xzhg0blnvvvTd33313fvnLX2b8+PG56aabkiSnn356rrrqqixcuDDJ64fqXXXVVS2HBG5oOmvvL730Us4+++xcffXV7R583tBZe3/hhRdy0UUXpaGhIQ0NDfnFL36RPffcs93301n7P+SQQ3L//fdn6dKlWb16de67777suuuu7bqPztr74MGD8+CDD+a1115LpVLJ3Xff3arDpdZFZ+39hRdeyNlnn52mpqbU19fn5ptvbrfDv97QWXtPkj/+8Y/ZbrvtqnZ+X2ft/frrr89Pf/rT3HbbbZkwYUIOPfTQdgs+b+isvdfX12fy5MlZtWpVli1blp/85CcZPnx4u++Htikyhg4bNiyf/vSnM378+DQ2NiZJDjzwwJx99tk5/fTT1zrvJXn9fJ2JEyfmiiuuaPnW8mMf+1h+9atf5Zprrsl555233vtoi/XZ+8yZM7No0aJ873vfy/e+970kyaGHHprPfvazVexwbQcddFDe9773ZcaMGWtdVveUU07Jd77znTz99NM5+eST09jYmE984hOpqalJpVLJiSeemI985CPrtd721Bl7/+53v5tVq1at8S3oSSedlJNPPrld99MZez/44IPz2GOP5dhjj023bt1y2GGHVS0Adsb+d99995xxxhk55ZRT0tjYmP333z+jRo1q9/10xt4POOCAPPHEEzn++OPTvXv3DB06NJ/85CfbfT+dsffBgwfnsMMOy9FHH53Vq1fntNNOq0ro74y9J8nzzz+frbbaqirbfkNn7X196Iy9H3LIIS3v9c3NzTnllFNaDoOj49VU2uOsQwAAgE6uyJGf9vbcc8+13Azrb11++eXtfmhDZ1Jy7//InDlzctlll73lvKlTp7bbVb46I72X2XtSdv961/vf0rve6XyM/AAAAEUo8oIHAABAeYQfAACgCMIPAABQBOEHAAAogqu9ARRg6tSpmTlzZjbeeOPstdde+cUvfpG99947r7zySp5//vl88IMfzPjx43PJJZdk7ty5qampyYEHHpjPf/7zqaury84775yHHnqo5V5fb0z/+c9/zuTJkzNw4MA8//zz6dWrV6666qrsuOOOHdwxAKzNyA9AF3fffffllltuycyZM3PLLbdk+fLlLfNWrlyZn/3sZznvvPNy+eWXZ7PNNsusWbPy4x//OH/84x9bblD8dp544ol84hOfyKxZs3L88cdvMDd9BqA8wg9AF3fPPffk8MMPzyabbJKampqMGTOmZd6ee+7Z8vO9996bU089NTU1NenRo0dOOumk3Hvvvf9w+4MHD85ee+2VJBk1alSefPLJLFmypP0bAYB3SPgB6OLq6ury5lu6devWreXn3r17t/zc3Ny8xnrNzc1pampaa3sNDQ1rTL95e0lSqVTWeg4AOgPhB6CLO/jggzN79uzU19cnSWbOnPmWyx1wwAGZNm1aKpVKGhoaMmPGjOy3335Jks033zy/+93vkiQ///nP11hv7ty5mTt3bpLkpptuyh577JFNNtmkWu0AQJu54AFAF7fvvvvmhBNOyIknnphevXrlPe95TzbaaKO1lrvoooty+eWXZ+TIkWlsbMyBBx6Y8ePHt8y79NJLs8kmm2S//fZL//79W9bbcsst89WvfjUvvvhiNt9880yePHm99QYA66Km8uZjIQDocn73u9/lN7/5TcaOHZskuf766/PYY4/lq1/96jve9n/+53/msssuy09/+tN3vC0AqDYjPwBd3Pbbb5/rrrsuM2bMSE1NTbbeeutcdtllHV0WAKx3Rn4AAIAiuOABAABQBOEHAAAogvADAAAUQfgBAACKIPwAAABF+P8BsR6lnqhEC0sAAAAASUVORK5CYII=\n",
      "text/plain": [
       "<Figure size 842.4x595.44 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "sns.set(rc={'figure.figsize':(11.7,8.27)})\n",
    "\n",
    "fig_group = sns.barplot(group_data, \n",
    "            x=\"group\", y=\"coverage\", hue=\"type\")\n",
    "fig_group.axhline(alpha, color='red')\n",
    "plt.legend(bbox_to_anchor=(1.02, 1), loc='upper left', borderaxespad=0)\n",
    "plt.tight_layout()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 91,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/var/folders/03/1mcjgjk12rv51t4kkk97lq5r0000gn/T/ipykernel_37806/663409929.py:2: SettingWithCopyWarning: \n",
      "A value is trying to be set on a copy of a slice from a DataFrame\n",
      "\n",
      "See the caveats in the documentation: https://pandas.pydata.org/pandas-docs/stable/user_guide/indexing.html#returning-a-view-versus-a-copy\n",
      "  state_shift_data_mod['coverage'][state_shift_data_mod['type'] == 'robust'] = state_shift_data_mod['coverage'][state_shift_data_mod['type'] == 'robust']/200\n",
      "/var/folders/03/1mcjgjk12rv51t4kkk97lq5r0000gn/T/ipykernel_37806/663409929.py:2: SettingWithCopyWarning: \n",
      "A value is trying to be set on a copy of a slice from a DataFrame\n",
      "\n",
      "See the caveats in the documentation: https://pandas.pydata.org/pandas-docs/stable/user_guide/indexing.html#returning-a-view-versus-a-copy\n",
      "  state_shift_data_mod['coverage'][state_shift_data_mod['type'] == 'robust'] = state_shift_data_mod['coverage'][state_shift_data_mod['type'] == 'robust']/200\n",
      "/var/folders/03/1mcjgjk12rv51t4kkk97lq5r0000gn/T/ipykernel_37806/663409929.py:3: SettingWithCopyWarning: \n",
      "A value is trying to be set on a copy of a slice from a DataFrame\n",
      "\n",
      "See the caveats in the documentation: https://pandas.pydata.org/pandas-docs/stable/user_guide/indexing.html#returning-a-view-versus-a-copy\n",
      "  state_shift_data_mod['coverage'][state_shift_data_mod['type'] == 'split'] = state_shift_data_mod['coverage'][state_shift_data_mod['type'] == 'split']/200\n",
      "/var/folders/03/1mcjgjk12rv51t4kkk97lq5r0000gn/T/ipykernel_37806/663409929.py:3: SettingWithCopyWarning: \n",
      "A value is trying to be set on a copy of a slice from a DataFrame\n",
      "\n",
      "See the caveats in the documentation: https://pandas.pydata.org/pandas-docs/stable/user_guide/indexing.html#returning-a-view-versus-a-copy\n",
      "  state_shift_data_mod['coverage'][state_shift_data_mod['type'] == 'split'] = state_shift_data_mod['coverage'][state_shift_data_mod['type'] == 'split']/200\n"
     ]
    },
    {
     "data": {
      "image/png": "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6+vT1NT0Fx1j0HsUJNf847HvyXGgNIMH96/2CN1q+JRvVHsE2GaVtB5cdfzF1R4BtlnVXAsqdqlbR0fHVvfV1NR0eT8AAEB3qVj4DBkyJKtXr+7cbm5uTkNDw9vuX7Vq1Rb7AQAAukvFwmfUqFG577770tLSkg0bNmTJkiUZPXp05/7ddtstffr0ycMPP5wkufXWW7fYDwAA0F1qOt7qmrNusnjx4txwww3ZtGlTJkyYkMmTJ2fy5MmZNm1ahg8fnieffDIzZ87M+vXrs+++++aKK65I7969KzUOAACwnapo+AAAAPQEFf0FpgAAAD2B8AEAAIonfAAAgOIJHwAAoHjCBwAAKJ7wYZvw4x//ODNmzHjXz7N06dLceOON3TAR0NOcfPLJeeCBB/Kb3/wmF154YZJkwYIF+clPflLlyYB36n//DF999dW5++67u/U5u2rMmDF54YUX3vWxqa66ag8A76XHHnus2iMAFTZ8+PAMHz48SfLoo4/mwAMPrPJEwDv1v3+Gv/zlL3f7c7J9ET5U3QMPPJBvfetbaW9vz+67754ddtghv/vd71JTU5PTTz89xx57bJLkueeey0knnZRXXnklf/d3f5ezzz47L774YiZNmpSlS5cmSa699tokyZQpU3LBBRfk97//fZJk4sSJ+djHPpYf/vCHSZK/+qu/yvjx49/7Fwv8RVauXJlzzjknr7/+emprazNz5sx89atfzZgxY/LQQw8lSS6//PLsu+++nd/zwAMP5Lvf/W7OPPPMLF26NPfff3/q6+tz8MEHV+tlAG8yZ86c3HHHHWlra8snP/nJnHnmmTn77LOzevXqJMnUqVPTt2/fLX6Gb7/99hx44IE58MADM3Xq1HzgAx/IU089lY985CM58MADc8stt+TVV1/Nddddlz333DN33HFHbrzxxrS2tmbjxo259NJLs2nTpi2ec5999smsWbOycuXK1NTU5Oyzz86oUaPyyiuv5Nxzz83KlSuz5557ZuPGjVX+J0Z3ED70CH/84x/z85//PN/73vfyxhtv5Cc/+UlaWlpy/PHHZ9iwYUmSF154IYsWLUr//v1zyimn5O677+7c92aPPvpoXn311dx66615+eWX841vfCMnnHBCTjzxxCQRPbCNWLhwYQ499NB84QtfyAMPPJCHH344STJw4MDceuutWbp0ac4///wsXrx4q+8dNWpUxowZkwMPPFD0QA/yy1/+Mr/97W+zcOHC1NTU5Nxzz83Pfvaz7LbbbpkzZ07+8Ic/ZOHChTn//PO3+Bm+/fbbO5/jd7/7Xa644ooMGzYs48aNy2677ZYFCxbku9/9bhYsWJAZM2bkhz/8Ya6//vrsvPPOWbhwYb7//e/n+uuv3+I5p0+fnvHjx+ewww5Lc3NzJk6cmFtvvTXXXHNN9t1338ydOze/+tWvcscdd1TxnxjdRfjQI3zoQx/KgAEDcv/99+fyyy9Pkuy888457LDD8uCDD6Z///4ZM2ZMdt555yTJEUcckQcffPBtw+fDH/5wnn322Zx++ukZPXp0zjnnnPfstQDd56CDDso//MM/5IknnsghhxySz3/+85k3b15OOOGEJP9z3f2MGTPS0tJS5UmBrrrvvvuyYsWKfPazn02StLa2pqOjI7/61a/S1NSUQw89NFOnTv0/n2OXXXbpPNO766675qCDDkryP1d0vPDCC6mtrc11112XpUuX5tlnn82DDz6Y2tqt39q+fPnyPPPMM7nmmmuSJJs3b85///d/58EHH8w//dM/JUkOOOCAfOADH+i210/1CB96hB133DFJ0tHRscX9HR0daWtrS5LU1dVtcX9dXV1qamq2+J7Nmzenrq4ugwYNyu233557770399xzT4477rgt/qYI2Dbsv//+uf322/OLX/wi//Vf/5VbbrklyZbrQXt7e3r16lWtEYG/UFtbW0455ZScdtppSZK1a9emV69e6ejoyLJly/Lzn/88//Iv//J/nmXp3bv3FttvXgPWr1+f8ePH55hjjskBBxyQoUOHZt68eVs9T3t7e2666aYMHDgwSdLU1JRddtllq/+/sMaUwae60aOMHDkyCxcuTJK0tLTk7rvv7nwD4j333JO1a9dm48aNuf322zNq1KjstNNOefXVV9PS0pI33ngjy5YtS5LcfffdOeecc3LooYdm5syZ6devX1566aX06tUrmzdvrtrrA/4y3/zmN7No0aIcd9xxmTVrVh5//PEk6fyLjJ/+9KfZc8898/73v/8tv79Xr16df3kC9AwjR47MokWLsn79+mzevDlTp07NLbfckmuvvTZHHHFELr744rS0tOS11157xz/Df/zjH1NbW5spU6Zk5MiR+eUvf9n5PP/7OUeOHJn58+cnSZ5++ukcffTR2bBhQw466KAsWrQoSbJixYo8//zz3fTqqSZnfOhRpk6dmq997WtpbGxMW1tbpkyZkv322y+/+93vsscee+SMM87I2rVrc9RRR+WTn/xkkuT000/PhAkTsuuuu3Z+ktPo0aNz11135TOf+Uz69OmTww8/PEOHDs3atWtz/vnnZ5dddsnJJ59czZcKdMHJJ5+cs88+O7fcckt69eqViy++OFdddVUeeeSRLFy4MH379s2VV175tt8/atSofPvb386AAQPy6U9/+j2cHHg7Y8aMyZNPPpkTTjghbW1tOfjgg3PMMcfk7LPPTmNjY+rq6vKlL30pO+200xY/w3+JYcOGZZ999skRRxyRHXfcMQcccED+9Kc/JdlyXZg5c2ZmzZqVxsbGJP/zly39+/fPtGnTMmPGjHzmM5/JHnvs4VK3QtR0vPnaIgDowcaMGZMf/OAH2X333as9CgDbEJe6AQAAxXPGBwAAKJ4zPgAAQPGEDwAAUDzhAwAAFE/4ANBj/f3f/31aWlqqPQYABRA+APRY9957b7VHAKAQwgeAHukf//EfkyQHH3xwhg4dmvb29iTp/K3qa9asyZgxY3L55Zfns5/9bMaOHdv5G9iTZOnSpTn++ONz7LHH5sQTT8yjjz5aldcBQM9QV+0BAOCtXHHFFfnxj3+cZcuW5bTTTsuyZctyyCGH5Pbbb89BBx2UwYMHJ0laW1vzn//5n2lqasqxxx6b/fffP3369Mns2bPzgx/8IIMGDcrvf//7nHbaaVmyZEn69etX5VcGQDUIHwB6vJNOOik333xzDjnkkCxYsCDnnXde576JEyempqYmu+66aw4++ODce++96dOnT5qbm3Pqqad2Pq6mpibPP/98hg0bVoVXAEC1CR8AerzGxsZ8+9vfzv3335/XX389BxxwQOe+urr//z9l7e3tqa2tTXt7ew466KB85zvf6dz30ksvpaGh4b0cG4AexHt8AOixevXqlc2bN6dv3745+uijc8EFF+TEE0/c4jG33nprkuRPf/pT7r333owePTojR47Mvffemz/84Q9JknvuuSdHH310Nm7c+F6/BAB6iJqOjo6Oag8BAG/ly1/+ch577LH88z//c9rb23PiiSdm2bJlGTBgQJJkzJgxGTFiRJ5//vm0trbmrLPOylFHHZUkueOOO3L99deno6MjdXV1ueCCC/Lxj3+8mi8HgCoSPgD0eB0dHZk7d25efPHFXHLJJZ33jxkzJldffXWGDx9exekA2BZ4jw8APd5hhx2WnXfeOd/73veqPQoA2yhnfAAAgOL5cAMAAKB4wgcAACie8AEAAIonfAAAgOIJHwAAoHj/H9zmW56ibGmnAAAAAElFTkSuQmCC\n",
      "text/plain": [
       "<Figure size 842.4x595.44 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "state_shift_data_mod = state_shift_data.copy()\n",
    "state_shift_data_mod['coverage'][state_shift_data_mod['type'] == 'robust'] = state_shift_data_mod['coverage'][state_shift_data_mod['type'] == 'robust']/200\n",
    "state_shift_data_mod['coverage'][state_shift_data_mod['type'] == 'split'] = state_shift_data_mod['coverage'][state_shift_data_mod['type'] == 'split']/200\n",
    "\n",
    "\n",
    "fig_group = sns.barplot(state_shift_data_mod, \n",
    "            x=\"type\", y=\"coverage\")\n",
    "fig_group.axhline(alpha, color='red')\n",
    "plt.tight_layout()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.9.5"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
