{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 179,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The autoreload extension is already loaded. To reload it, use:\n",
      "  %reload_ext autoreload\n",
      "ERROR! Session/line number was not unique in database. History logging moved to new session 234\n"
     ]
    }
   ],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2\n",
    "    \n",
    "import os\n",
    "import sys\n",
    "\n",
    "import pandas as pd\n",
    "import pickle\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "import seaborn as sns\n",
    "\n",
    "from tqdm import tqdm\n",
    "src_dir = os.path.join(os.path.abspath(os.path.join(os.getcwd(), os.pardir)), 'src')\n",
    "sys.path.append(src_dir)\n",
    "\n",
    "from dataset import find_unique_element\n",
    "from atomizer import text_to_sentences\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 180,
   "metadata": {},
   "outputs": [],
   "source": [
    "def load_dataset_from_file(dataset = None, filepath = None, cache_path = \"/Users/cherian/Projects/conformal-safety/.cache\"):\n",
    "    if dataset is not None:\n",
    "        with open(f\"{cache_path}/{dataset}_responses.pkl\", 'rb') as fp:\n",
    "            res_cache = pickle.load(fp)\n",
    "        with open(f\"{cache_path}/{dataset}_parsed.pkl\", 'rb') as fp:\n",
    "            af_cache = pickle.load(fp)\n",
    "        with open(f\"{cache_path}/{dataset}_scored.pkl\", 'rb') as fp:\n",
    "            ann_cache = pickle.load(fp)\n",
    "            \n",
    "        prompts = []\n",
    "        messages = []\n",
    "        for key, val in res_cache.items():\n",
    "            prompts.append(key[:-2])\n",
    "            messages.append(val[0]['message'])\n",
    "\n",
    "        dataset = [\n",
    "            {\n",
    "                'prompt': key[:-2],\n",
    "                'response': val[0]['message'],\n",
    "                'atomic_facts': []\n",
    "            }\n",
    "            for key, val in res_cache.items()\n",
    "        ]\n",
    "        idx_guess = 0\n",
    "        atomic_facts = [[] for _ in range(len(messages))]\n",
    "        for k in af_cache.keys():\n",
    "            atomized_msg = af_cache[k][0]['message']\n",
    "            atomized_facts = text_to_sentences(atomized_msg)\n",
    "            sentence = k.split('\\n')[-1].split('facts:')[-1].strip()[:-2]\n",
    "            cur_idx = find_unique_element(messages, lambda x: sentence in x, approx_index=idx_guess)\n",
    "            if cur_idx is None:\n",
    "                raise ValueError(\"Should regenerate responses and atomic facts I guess because they don't match :(\")\n",
    "            idx_guess = cur_idx\n",
    "            atomic_facts[cur_idx].extend(atomized_facts)\n",
    "        \n",
    "        idx_guess = 0\n",
    "        for key, val in ann_cache.items():\n",
    "            af, _ = key.split(\"Input: \")[-1].split(\"True or False?\\nOutput:_\")\n",
    "            af = af.strip()\n",
    "            ann = bool(val[0]['message'])\n",
    "            \n",
    "            cur_idx = find_unique_element(atomic_facts, lambda x: af in x, approx_index=idx_guess)\n",
    "            if cur_idx is None:\n",
    "                if af.startswith(\"He retired with a career record of\"): # special case because of some crazy pickle bug??\n",
    "                    cur_idx = idx_guess\n",
    "                else:\n",
    "                    raise ValueError(\"Huh weird...probably should have failed in the last cache recovery step.\")\n",
    "            dataset[cur_idx]['atomic_facts'].append(\n",
    "                {'atom': af, 'is_supported': ann}\n",
    "            )\n",
    "            idx_guess = cur_idx\n",
    "    elif filepath is not None:\n",
    "        with open(filepath, 'rb') as fp:\n",
    "            dataset = pickle.load(fp)\n",
    "    else:\n",
    "        raise ValueError(\"No dataset to load.\")\n",
    "    return dataset\n",
    "\n",
    "def load_features(dataset = None, filepath = None, cache_path = \"/Users/cherian/Projects/conformal-safety/.cache\"):\n",
    "    if dataset is not None:\n",
    "        with open(f\"{cache_path}/{dataset}_frequency.pkl\", 'rb') as fp:\n",
    "            freq_cache = pickle.load(fp)\n",
    "        with open(f\"{cache_path}/{dataset}_self_evals.pkl\", 'rb') as fp:\n",
    "            eval_cache = pickle.load(fp)\n",
    "        return freq_cache, eval_cache"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 181,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "8474 8474 8474 8474 8474\n"
     ]
    }
   ],
   "source": [
    "dataset_name = \"factscore_final\"\n",
    "data_path = \"/Users/isaacgibbs/Documents/safety/conformal-safety/data\"\n",
    "dataset_path = f\"{data_path}/factscore_final_dataset.pkl\"\n",
    "freq_path = f\"{data_path}/factscore_final_frequencies_2.npz\"\n",
    "selfevals_path = f\"{data_path}/factscore_final_self_evals.npz\"\n",
    "metadata_path = f\"{data_path}/factscore_final.csv\"\n",
    "\n",
    "dataset = load_dataset_from_file(filepath=dataset_path)\n",
    "frequencies = np.load(freq_path)\n",
    "selfevals = np.load(selfevals_path)\n",
    "metadata = pd.read_csv(metadata_path, index_col=0).reset_index(drop=True).drop_duplicates()\n",
    "## need to filter the dataset so that all information is available for every data point\n",
    "\n",
    "drop_prompts = []\n",
    "for k in frequencies:\n",
    "    if frequencies[k].ndim != 1:\n",
    "        drop_prompts.append(k)\n",
    "    elif np.allclose(selfevals[k], -1):\n",
    "        drop_prompts.append(k)\n",
    "\n",
    "# drop and match ordering of dataset\n",
    "dataset = [dat for dat in dataset if dat['prompt'] not in drop_prompts]\n",
    "\n",
    "prompts_to_keep = [dat['prompt'] for dat in dataset]\n",
    "names_to_keep = [p.split('about')[-1].strip()[:-1] for p in prompts_to_keep]\n",
    "\n",
    "metadata = metadata.set_index(\"Name\")\n",
    "metadata = metadata.loc[names_to_keep].reset_index()\n",
    "\n",
    "frequencies_arr = [frequencies[p] for p in prompts_to_keep]\n",
    "selfevals_arr = [selfevals[p] for p in prompts_to_keep]\n",
    "annotations_arr = [np.asarray([af[\"is_supported\"] for af in dat[\"atomic_facts\"]]) for dat in dataset]\n",
    "\n",
    "\n",
    "print(len(frequencies_arr), len(selfevals_arr), len(metadata), len(dataset), len(annotations_arr))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 182,
   "metadata": {},
   "outputs": [],
   "source": [
    "metadata.loc[663, \"count_bins\"] = \"Very Rare\"\n",
    "metadata.loc[663, \"max_counts\"] = 0\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 183,
   "metadata": {},
   "outputs": [],
   "source": [
    "from typing import List\n",
    "from conditionalconformal import CondConf\n",
    "\n",
    "def split_dataset(dataset, rng, train_frac = 0.8):\n",
    "    x, y = dataset\n",
    "    ind = np.arange(len(x))\n",
    "    rng.shuffle(ind)\n",
    "    train_num = int(train_frac * len(x))\n",
    "    train_ind = ind[0:train_num]\n",
    "    calib_ind = ind[train_num:]\n",
    "\n",
    "    x_train = [x[i] for i in train_ind]\n",
    "    y_train = [y[i] for i in train_ind]\n",
    "\n",
    "    x_calib = [x[i] for i in calib_ind]\n",
    "    y_calib = [y[i] for i in calib_ind]\n",
    "    return (x_train, y_train), (x_calib, y_calib), train_ind, calib_ind\n",
    "    \n",
    "def score_func(\n",
    "    claim_scores : List[np.ndarray],\n",
    "    annotations : List[np.ndarray],\n",
    "    method : str = \"max\"\n",
    "):\n",
    "    if method == \"max\":\n",
    "        min_score = -1\n",
    "        scores = np.zeros((len(claim_scores),))\n",
    "        for i, (cs, a) in enumerate(zip(claim_scores, annotations)):\n",
    "            scores[i] = np.max(cs[~a]) if np.sum(~a) >= 1 else min_score\n",
    "    if isinstance(method, int):\n",
    "        min_score = -1\n",
    "        scores = np.zeros((len(claim_scores),))\n",
    "        for i, (cs, a) in enumerate(zip(claim_scores, annotations)):\n",
    "            scores[i] = np.sort(cs[~a])[::-1][method] if np.sum(~a) > method else min_score\n",
    "    return scores\n",
    "\n",
    "def split_threshold(\n",
    "    conf_scores : np.ndarray,\n",
    "    quantile\n",
    "):\n",
    "    n = len(conf_scores)\n",
    "    threshold = np.sort(conf_scores)[int(np.ceil(quantile * (n + 1)))]\n",
    "    return threshold\n",
    "\n",
    "def get_retained_claims(claim_scores, thresholds):\n",
    "    claims_retained = []\n",
    "    for cs, t in zip(claim_scores, thresholds):\n",
    "        claims_retained.append(np.mean(cs > t))\n",
    "    return claims_retained\n",
    "\n",
    "def get_validity(claim_scores, annotations, threshold, method):\n",
    "    conf_scores = score_func(claim_scores, annotations, method)\n",
    "    validity = conf_scores <= threshold\n",
    "    return validity\n",
    "\n",
    "def run_split_conformal(x_arr, y_arr, method, quantile):\n",
    "    conf_scores = score_func(x_arr, y_arr, method=k)\n",
    "    threshold = split_threshold(conf_scores, quantile)\n",
    "    return conf_scores, threshold\n",
    "    \n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 184,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[[1.         0.37656178]\n",
      " [0.37656178 1.        ]]\n",
      "[[1.         0.16207942]\n",
      " [0.16207942 1.        ]]\n",
      "[[ 1.         -0.00960692]\n",
      " [-0.00960692  1.        ]]\n",
      "[[ 1.         -0.00251182]\n",
      " [-0.00251182  1.        ]]\n"
     ]
    }
   ],
   "source": [
    "ordinal_arr = [np.arange(len(f)) for f in frequencies_arr]\n",
    "\n",
    "rng = np.random.default_rng(seed=0)\n",
    "rand_scores = rng.standard_normal(size=(len(np.concatenate(selfevals_arr)),))\n",
    "\n",
    "print(np.corrcoef(np.concatenate(frequencies_arr), np.concatenate(annotations_arr)))\n",
    "print(np.corrcoef(np.concatenate(selfevals_arr), np.concatenate(annotations_arr)))\n",
    "print(np.corrcoef(np.concatenate(ordinal_arr), np.concatenate(annotations_arr)))\n",
    "print(np.corrcoef(rand_scores, np.concatenate(annotations_arr)))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 185,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████| 5/5 [00:10<00:00,  2.14s/it]\n"
     ]
    }
   ],
   "source": [
    "continent_dummies = pd.get_dummies(metadata[\"continent\"])\n",
    "freq_dummies = pd.get_dummies(metadata[\"count_bins\"])\n",
    "freq_dummies = freq_dummies[[\"Very Frequent\", \"Frequent\", \"Medium\", \"Rare\", \"Very Rare\"]]\n",
    "continent_names = continent_dummies.columns\n",
    "freq_names = freq_dummies.columns\n",
    "\n",
    "def run_coverage_trial(x_arr, y_arr, z_arr, rng, method, quantile, continent_dummies=continent_dummies, freq_dummies=freq_dummies):\n",
    "    data_calib, data_test, idx_calib, idx_test = split_dataset((x_arr, y_arr), rng, train_frac=0.7)\n",
    "    scores_calib = score_func(*data_calib, method=method)\n",
    "    scores_test = score_func(*data_test, method=method)\n",
    "    \n",
    "    condconf = CondConf(lambda x,y: y, lambda x: x)\n",
    "    condconf.setup_problem(z_arr[idx_calib], scores_calib)\n",
    "    \n",
    "    # scores, threshold = run_split_conformal(*data_calib, method=method, quantile=quantile)\n",
    "    valid_inds = []\n",
    "    claim_perc = []\n",
    "    for i, j in enumerate(idx_test):\n",
    "        try:\n",
    "            threshold = condconf.predict(quantile, z_arr[j].reshape(1,-1), lambda c, x: c, randomize=True)\n",
    "        except:\n",
    "            threshold = [np.inf]\n",
    "        valid_inds.append(threshold >= scores_test[i])\n",
    "        claim_perc.append(get_retained_claims([data_test[0][i]], threshold)[0])\n",
    "                          \n",
    "    valid_inds = np.asarray(valid_inds).flatten()\n",
    "    claim_perc = np.asarray(claim_perc).flatten()\n",
    "\n",
    "    # valid_inds = get_validity(*data_calib, threshold, method)\n",
    "    all_covs = np.zeros((1,))\n",
    "    marginal_cov = np.mean(valid_inds)\n",
    "    all_covs[0] = marginal_cov\n",
    "\n",
    "    all_claims = np.zeros((1,))\n",
    "    all_claims[0] = np.mean(claim_perc)\n",
    "\n",
    "    continent_covs = np.zeros(len(continent_names))\n",
    "    continent_claims = np.zeros(len(continent_names))\n",
    "\n",
    "    for c_idx in range(len(continent_names)):\n",
    "        continent_cov = np.sum(continent_dummies.to_numpy()[idx_test,c_idx] * valid_inds) / np.sum(continent_dummies.to_numpy()[idx_test,c_idx])\n",
    "        continent_covs[c_idx] = continent_cov\n",
    "        continent_perc = np.sum(continent_dummies.to_numpy()[idx_test,c_idx] * claim_perc) / np.sum(continent_dummies.to_numpy()[idx_test,c_idx])\n",
    "        continent_claims[c_idx] = continent_perc\n",
    "        \n",
    "    freq_covs = np.zeros(len(freq_names))\n",
    "    freq_claims = np.zeros(len(freq_names))\n",
    "    for f_idx in range(len(freq_names)):\n",
    "        freq_cov = np.sum(freq_dummies.to_numpy()[idx_test,f_idx] * valid_inds) / np.sum(freq_dummies.to_numpy()[idx_test,f_idx])\n",
    "        freq_covs[f_idx] = freq_cov\n",
    "        freq_perc = np.sum(freq_dummies.to_numpy()[idx_test,f_idx] * claim_perc) / np.sum(freq_dummies.to_numpy()[idx_test,f_idx])\n",
    "        freq_claims[f_idx] = freq_perc\n",
    "    \n",
    "    return np.concatenate((all_covs, continent_covs, freq_covs)), np.concatenate((all_claims, continent_claims, freq_claims))\n",
    "\n",
    "# selfevals_arr another option\n",
    "n_trials = 5\n",
    "all_covs = np.zeros((n_trials, 1 + len(continent_names) + len(freq_names)))\n",
    "all_claims = np.zeros((n_trials, 1 + len(continent_names) + len(freq_names)))\n",
    "\n",
    "rng = np.random.default_rng(seed=0)\n",
    "k = 3\n",
    "quantile = 0.9\n",
    "method = \"frequency\"\n",
    "\n",
    "z_ones = np.ones((len(frequencies_arr), 1))\n",
    "views = metadata[\"max_counts\"].to_numpy()\n",
    "views += 1\n",
    "z_views = views.clip(0, np.quantile(views, 0.95)).reshape(-1,1)\n",
    "z_views = z_views / np.mean(z_views)\n",
    "z_dummies = freq_dummies.to_numpy().astype(int)\n",
    "z_freq = np.asarray([freq.std() for freq in frequencies_arr]).reshape(-1,1)\n",
    "z_se = np.asarray([se.std() for se in selfevals_arr]).reshape(-1,1)\n",
    "z_arr = np.concatenate((z_ones, z_views, z_views**2, z_views**3), axis=1)\n",
    "#z_arr = z_dummies\n",
    "frequencies_arr_jitter = [freq + rng.uniform(low=0, high=1e-3, size=freq.shape) for freq in frequencies_arr]\n",
    "o_arr_jitter = [o + rng.uniform(low=0, high=1e-3, size=o.shape) for o in ordinal_arr]\n",
    "rng = np.random.default_rng(seed=1)\n",
    "for trial in tqdm(range(n_trials)):\n",
    "    all_covs[trial], all_claims[trial] = run_coverage_trial(frequencies_arr_jitter, annotations_arr, z_arr, rng, k, quantile)\n",
    "    #all_covs[trial], all_claims[trial] = run_coverage_trial(o_arr_jitter, annotations_arr, z_arr, rng, k, quantile)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 186,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████████████| 50/50 [02:03<00:00,  2.47s/it]\n"
     ]
    }
   ],
   "source": [
    "n_trials = 50\n",
    "all_covs_mh = np.zeros((n_trials, 1 + len(continent_names) + len(freq_names)))\n",
    "all_claims_mh = np.zeros((n_trials, 1 + len(continent_names) + len(freq_names)))\n",
    "\n",
    "rng = np.random.default_rng(seed=0)\n",
    "k = 3\n",
    "quantile = 0.9\n",
    "method = \"frequency\"\n",
    "\n",
    "z_ones = np.ones((len(frequencies_arr), 1))\n",
    "views = metadata[\"max_counts\"].to_numpy()\n",
    "views += 1\n",
    "z_views = views.clip(0, np.quantile(views, 0.95)).reshape(-1,1)\n",
    "z_views = z_views / np.mean(z_views)\n",
    "z_dummies = freq_dummies.to_numpy().astype(int)\n",
    "z_freq = np.asarray([freq.std() for freq in frequencies_arr]).reshape(-1,1)\n",
    "z_se = np.asarray([se.std() for se in selfevals_arr]).reshape(-1,1)\n",
    "# z_arr = np.concatenate((z_ones, z_views, z_views**2, z_views**3), axis=1)\n",
    "z_arr = z_ones\n",
    "frequencies_arr_jitter = [freq + rng.uniform(low=0, high=1e-3, size=freq.shape) for freq in frequencies_arr]\n",
    "rng = np.random.default_rng(seed=1)\n",
    "for trial in tqdm(range(n_trials)):\n",
    "    all_covs_mh[trial], all_claims_mh[trial] = run_coverage_trial(frequencies_arr_jitter, annotations_arr, z_arr, rng, k, quantile)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 187,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.legend.Legend at 0x17b7959d0>"
      ]
     },
     "execution_count": 187,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAtcAAAFzCAYAAAD16yU4AAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjUuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/YYfK9AAAACXBIWXMAAAsTAAALEwEAmpwYAAA6kUlEQVR4nO3de5zVdZ348debAWHwtnGRahBBIa+YF7QsMy2xqVzMzVKrFXfb3DKRbMu0NU0y16xfiaYltSa0pZa30CVc8F5byqAoeMvRUBmN1dHwAggzfH5/nO/QEQfmDHy/nDnD6/l48Jjv9XPeh++c7/c9n/O5REoJSZIkSZuuT7UDkCRJknoLk2tJkiQpJybXkiRJUk5MriVJkqScmFxLkiRJOTG5liRJknLSt9oB5GXIkCFp5MiR1Q5DkiRJvdz8+fNfSCkN7Wxfr0muR44cSVNTU7XDkCRJUi8XEU+tb5/NQiRJkqScmFxLkiRJOTG5liRJknLSa9pcd2b16tUsWbKElStXVjsU9SIDBgxg+PDh9OvXr9qhSJKkHqZXJ9dLlixh2223ZeTIkUREtcNRL5BSorW1lSVLljBq1KhqhyNJknqYXt0sZOXKlQwePNjEWrmJCAYPHuy3IZIkqVO9OrkGTKyVO3+nJEnS+vT65Lravv3tb7Pnnnuy9957s88++3DPPfds8PhvfvObfO973wPg7LPPZu7cuQBcdNFFLF++vNNzVq9ezRlnnMGYMWPYb7/9OOigg/jtb3+7UfE+//zzvOtd72Lffffl7rvv3qgyumvx4sXstddem+W1JEmSitSr21yv65Qvf5X/e+HF3MrbYcggfvj97653/x/+8Aduvvlm7rvvPvr3788LL7zAqlWrKi5/ypQpa5cvuugiPvOZzzBw4MA3HfeNb3yD5557jkWLFtG/f3+WLl3KnXfe2b03k7n11lsZO3YsP/3pTys+p729nbq6uo16PUmSpN5ki0qu/++FF3li2PvzK3DphhPY5557jiFDhtC/f38AhgwZsnbfyJEj+eQnP8lvf/tb6uvr+eUvf8no0aPfcP6JJ57IkUceybPPPsuzzz7LYYcdxpAhQ7j99tvXHrN8+XJ+8pOf8Oc//3nt6wwbNoxPfvKTAFx11VWcf/75pJT46Ec/yne+8x0AttlmGyZPnszNN99MfX09v/nNb3juuec4/fTTWbFiBU1NTfzhD3/gxhtvXO/5//qv/8rcuXO59NJLaWxs5Atf+AKzZs3ibW97G+effz6nn346Tz/9NBdddBETJkxg8eLF/OM//iOvvfYaAD/84Q95z3vesylXQJIkqUexWUiBjjjiCJ555hne8Y53cPLJJ7+pNnn77bdn4cKFnHLKKXzpS19abzmnnnoqb3/727n99tvfkFgDNDc3M2LECLbbbrs3nffss8/yta99jdtuu40FCxYwb948brzxRgBee+013v3ud/PAAw9wyCGH8JOf/IR99tmHKVOmcOyxx7JgwQJeeumlDZ7/rne9iwceeICDDz6Y1157jQ984AM89NBDbLvttpx11lnMmTOHG264gbPPPhuAHXbYgTlz5nDfffdxzTXXcOqpp278f64kSVIPtEXVXG9u22yzDfPnz+fuu+/m9ttv59hjj+WCCy7gxBNPBOD4449f+/O0007L/fXnzZvHoYceytChQwH49Kc/zV133cXHPvYxttpqK4488kgA9t9/f+bMmdOt8+vq6vj4xz++9titttqKxsZGAMaOHUv//v3p168fY8eOZfHixUCpbfgpp5zCggULqKur409/+lPu71mSpC3ZJZdcQnNzc0XHtrS0ANDQ0FDR8aNHj2bSpEkbHduWwuS6YHV1dRx66KEceuihjB07lunTp69NrstHndjYEShGjx7N008/zcsvv9xp7fX69OvXb+1r1tXV0dbW1q3XHTBgwBvaWZeX16dPn7VNVPr06bO27B/84AcMGzaMBx54gDVr1jBgwIBuvaYkScrPihUrqh1Cr2RyXaDHHnuMPn36MGbMGAAWLFjATjvttHb/NddcwxlnnME111zDQQcdtMGytt12W1555ZU3tNsGGDhwIJ/97GeZPHkyl19+OVtttRXPP/88d9xxBwcffDCnnnoqL7zwAm95y1u46qqruvUX54EHHrhJ569r2bJlDB8+nD59+jB9+nTa29s3uixJkvRm3XlOT548GYCpU6cWFc4WyeS6QK+++iqTJk3ir3/9K3379mX06NFMmzZt7f6XXnqJvffem/79+3PVVVdtsKyTTjqJxsbGtW2vy5133nmcddZZ7LHHHgwYMICtt96aKVOm8La3vY0LLriAww47bG2HxKOOOqri+Df1/HWdfPLJfPzjH2fGjBk0Njay9dZbb3RZkiRJPVGklKodQy7GjRuXmpqa3rDtkUceYffdd1+7vrmH4tuQkSNH0tTU9KaaaNWGdX+3JEmqNdZcb7yImJ9SGtfZvi2q5npjE2FJkiSpEltUct2TdIygIUmSpN7Dca4lSZKknJhcS5IkSTkxuZYkSZJyYnItSZIk5cTkumB/+ctfOO6449hll13Yf//9+chHPpLLtN933HHH2unLZ86cyQUXXADAjTfeyMMPP7z2uLPPPpu5c+du8uut77XXde+993LIIYew6667su+++/Iv//IvLF++vKKyyt/HiSeeyLXXXrvRMf71r3/lsssuW7v+7LPPcswxx2x0eZIkSZXYokYL+fq/ncKyF5bmVt72Q4Zx/v/74Xr3p5Q4+uijmThxIldffTUADzzwAEuXLuUd73hHbnFMmDCBCRMmAKXk+sgjj2SPPfYAYMqUKbm9TleWLl3KJz7xCa6++uq1M05ee+21vPLKKwwcOLDL88vfRyXa2tro27fzX+GO5Prkk08G4O1vf/smJeuSJEmV2KKS62UvLOVruzyaW3nfeWLD+2+//Xb69evH5z//+bXb3vnOdwKlxPv000/nt7/9LRHBWWedxbHHHssdd9zBN7/5TYYMGcKiRYvYf//9+a//+i8igtmzZ/OlL32JgQMHcvDBB68t88orr6SpqYlPfepTzJw5kzvvvJPzzjuP6667jm9961sceeSRHHPMMdx666185Stfoa2tjQMOOIAf/ehH9O/fn5EjRzJx4kRuuukmVq9eza9//Wt222037r33XiZPnszKlSupr6/nZz/7Gbvuuut63++ll17KxIkT3zCVe0dtcSVldbyPH/6w9AfL3LlzueCCC3j55Zf5/ve/z5FHHsmVV17J9ddfz6uvvkp7ezv//d//zVFHHcVLL73E6tWrOe+88zjqqKM444wzeOKJJ9hnn30YP348X/ziFznyyCNZtGgRK1eu5Atf+AJNTU307duX73//+xx22GFceeWVzJw5k+XLl/PEE09w9NFHc+GFF1b2yyBJksQWllxvbh3JcWeuv/56FixYwAMPPMALL7zAAQccwCGHHALA/fffz0MPPcTb3/523vve9/L73/+ecePG8bnPfY7bbruN0aNHc+yxx76pzPe85z1MmDBhbTJdbuXKlZx44onceuutvOMd7+CEE07gRz/6EV/60pcAGDJkCPfddx+XXXYZ3/ve9/jpT3/Kbrvtxt13303fvn2ZO3cuX//617nuuus2+H4nTpzY6b7ulgWlscDvvfdennjiCQ477DCam5sBuO+++3jwwQcZNGgQbW1t3HDDDWy33Xa88MILvPvd72bChAlccMEFLFq0iAULFqwtq8Oll15KRLBw4UIeffRRjjjiiLVNdRYsWMD9999P//792XXXXZk0aRI77rjjBuOUJEnqYHJdJb/73e84/vjjqaurY9iwYbz//e9n3rx5bLfddhx44IEMHz4cgH322YfFixezzTbbMGrUKMaMGQPAZz7zGaZNm1bx6z322GOMGjVqbXOUiRMncumll65Nrv/hH/4BgP3335/rr78egGXLljFx4kQef/xxIoLVq1dv9PvdmLI++clP0qdPH8aMGcPOO+/Mo4+WvnUYP348gwYNAkrfAHz961/nrrvuok+fPrS0tLB06Yab/vzud79j0qRJQCnp32mnndYm1x/84AfZfvvtAdhjjz146qmnTK4lSVLF7NBYoD333JP58+d3+7z+/fuvXa6rq6OtrS3PsDb4muWv941vfIPDDjuMRYsWcdNNN7Fy5coNlrGh99vdsgAiotP1rbfeeu22X/ziFzz//PPMnz+fBQsWMGzYsIrKXp9q/N9LkqTew+S6QB/4wAd4/fXX31DD/OCDD3L33Xfzvve9j2uuuYb29naef/557rrrLg488MD1lrXbbruxePFinnii1ND7qquu6vS4bbfdlldeeeVN23fddVcWL168tmnFz3/+c97//vdvMP5ly5bR0NAAlNpDd+WUU05h+vTp3HPPPWu3XX/99SxdurTbZQH8+te/Zs2aNTzxxBM8+eSTnbb3XrZsGTvssAP9+vXj9ttv56mnngLW//8A8L73vY9f/OIXAPzpT3/i6aef3mBbckmSpEoVmlxHRGNEPBYRzRFxRif7Px8RCyNiQUT8LiL2yLaPjIgV2fYFEfHjIuMsSkRwww03MHfuXHbZZRf23HNPzjzzTN761rdy9NFHs/fee/POd76TD3zgA1x44YW89a1vXW9ZAwYMYNq0aXz0ox9lv/32Y4cdduj0uOOOO47vfve77LvvvmsT8Y7zf/azn/GJT3yCsWPH0qdPnzd0tOzM6aefzplnnsm+++5bUQ3usGHDuPrqq/nKV77Crrvuyu67784tt9zCtttu2+2yAEaMGMGBBx7I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\n",
      "text/plain": [
       "<Figure size 864x432 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Create a DataFrame for each method\n",
    "names = [\"Marginal\"] + continent_names.tolist() + freq_names.tolist()\n",
    "indices_to_plot = [0, -5, -4, -3, -2, -1]\n",
    "\n",
    "names = [names[i] for i in indices_to_plot]\n",
    "df1 = pd.DataFrame(1 - all_covs_mh[:,indices_to_plot], columns=names)\n",
    "df2 = pd.DataFrame(1 - all_covs[:,indices_to_plot], columns=names)\n",
    "\n",
    "# Add a column to identify the method\n",
    "df1['Method'] = 'Split Conformal'\n",
    "df2['Method'] = 'Conditional Calibration'\n",
    "\n",
    "# Combine the DataFrames\n",
    "df = pd.concat([df1, df2])\n",
    "\n",
    "# Melt the combined DataFrame\n",
    "df_melted = df.melt(id_vars='Method', var_name='Group', value_name='Miscoverage')\n",
    "\n",
    "# Create the boxplot\n",
    "plt.figure(figsize=(12, 6))\n",
    "sns.boxplot(x='Group', y='Miscoverage', hue='Method', data=df_melted)\n",
    "plt.axhline(0.1, color='red')\n",
    "# Customize the plot\n",
    "plt.legend()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 188,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.legend.Legend at 0x154664f70>"
      ]
     },
     "execution_count": 188,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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xRXU1dXx2c9+lptvvpm7776b8847j+uuu47vfve7G3p477jjDs444wyampr44Ac/yKWXXsqAAQMYNWoUU6ZM4ZZbbmHdunX8/ve/Z8yYMdx///1Mnz6dNWvWUFNTw69+9St22WWXdt/vJZdcwpQpU962NHpLz3Ix52p5Hz/5SeEPlHnz5nHBBRfw2muv8aMf/YgjjjiCK664guuvv57XX3+d5uZm/vu//5ujjjqKV155hXXr1nHeeedx1FFHcdZZZ/HUU08xbtw4DjvsML70pS9xxBFHsHjxYtasWcMXv/hF6urq6NevHz/60Y846KCDuOKKK7j55ptZtWoVTz31FMcccww/+MEPirsY2mGYliRJyqklDLfl+uuvZ9GiRTz00EMsX76cD37wg+y///4APPjggzz66KO8973v5SMf+Qh//vOfmThxIv/yL//CnXfeSW1tLccff/w7zvnhD3+YI488ss3hEWvWrOGkk07ijjvuYOedd+bEE0/k0ksv5fTTTwdg2LBhLFy4kJ/+9KdceOGFXH755YwZM4Z7772Xfv36MW/ePM4++2yuu+66Dt/vlClT2nyts+eCwlzc999/P0899RQHHXTQhk/MFy5cyMMPP8w222xDU1MTN9xwA0OGDGH58uXss88+HHnkkVxwwQUsXryYRYsWbThXi0suuYSI4JFHHuHxxx/nYx/7GE888QQAixYt4sEHH2TAgAHssssuTJ06lfe9730d1tkRw7QkSVIJ/OlPf+KEE06gurqa4cOHc8ABB/DAAw8wZMgQJk2axA477ADAuHHjWLJkCVtttRU77rgjo0ePBuDzn/88l112WdHt/e1vf2PHHXfcMLxkypQpXHLJJRvC9Cc/+UkAJkyYwPXXXw/AihUrmDJlCk8++SQRwbp163K/3zznOu6446iqqmL06NHstNNOPP544VOFww47jG222QYo9PCfffbZ3HPPPVRVVdHQ0MDSpR0P5fnTn/604RP4MWPG8P73v39DmD7kkEPYeuutAdhtt9145plnuhSmnc1DkiQpp913350FCxZ0+rgBAwZseFxdXU1TU1N3ltVhm63b++Y3v8lBBx3E4sWLueWWW1izZk2H5+jo/Xb2XACx0VRkLc+33HLLDdt++9vfsmzZMhYsWMCiRYsYPnx4UeduT3f/tzdMS5Ik5XTwwQezdu3at/UgP/zw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\n",
      "text/plain": [
       "<Figure size 864x432 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Create a DataFrame for each method\n",
    "names = [\"Marginal\"] + continent_names.tolist() + freq_names.tolist()\n",
    "indices_to_plot = [-5, -4, -3, -2, -1]\n",
    "\n",
    "names = [names[i] for i in indices_to_plot]\n",
    "df1 = pd.DataFrame(all_claims_mh[:,indices_to_plot] * 100, columns=names)\n",
    "df2 = pd.DataFrame(all_claims[:,indices_to_plot] * 100, columns=names)\n",
    "\n",
    "# Add a column to identify the method\n",
    "df1['Method'] = 'Split Conformal'\n",
    "df2['Method'] = 'Conditional Calibration'\n",
    "\n",
    "# Combine the DataFrames\n",
    "df = pd.concat([df1, df2])\n",
    "\n",
    "# Melt the combined DataFrame\n",
    "df_melted = df.melt(id_vars='Method', var_name='Group', value_name='% Claims Retained')\n",
    "\n",
    "# Create the boxplot\n",
    "plt.figure(figsize=(12, 6))\n",
    "sns.boxplot(x='Group', y='% Claims Retained', hue='Method', data=df_melted)\n",
    "# Customize the plot\n",
    "plt.legend()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 189,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████████████| 50/50 [00:41<00:00,  1.19it/s]\n"
     ]
    }
   ],
   "source": [
    "## now I should try to estimate the alpha at which the conditional conformal method retains 70% of the claims\n",
    "rng = np.random.default_rng(seed=0)\n",
    "k = 3\n",
    "\n",
    "dataset = (frequencies_arr, annotations_arr)\n",
    "# split dataset\n",
    "data_alpha, data_conf, ind_alpha, ind_conf = split_dataset(dataset, rng, train_frac=0.1)\n",
    "metadata_alpha = metadata.iloc[ind_alpha].reset_index(drop=True)\n",
    "z_ones = np.ones((len(ind_alpha), 1))\n",
    "views = metadata_alpha[\"max_counts\"].to_numpy()\n",
    "views += 1\n",
    "z_views = views.clip(0, np.quantile(views, 0.95)).reshape(-1,1)\n",
    "z_views = z_views / np.mean(z_views)\n",
    "phi_alpha = lambda x: np.concatenate((np.ones((len(x), 1)), x, x**2, x**3), axis=1)\n",
    "\n",
    "# use data alpha to estimate critical alpha for which fact retention goal is achieved\n",
    "data_calib, data_test, ind_calib, ind_test = split_dataset(data_alpha, rng, train_frac=0.5)\n",
    "z_calib = z_views[ind_calib]\n",
    "z_test = z_views[ind_test]\n",
    "\n",
    "scores_calib = score_func(*data_calib, method=k)\n",
    "\n",
    "x_test, y_test = data_test\n",
    "n_grid = 50\n",
    "percents = np.empty((n_grid, len(x_test)))\n",
    "\n",
    "condconf = CondConf(lambda x,y: y, phi_alpha)\n",
    "condconf.setup_problem(z_calib, scores_calib)\n",
    "\n",
    "quantile_grid = np.linspace(0.01, 0.99, 50)\n",
    "i = 0\n",
    "for q in tqdm(quantile_grid):\n",
    "    perc_q = []\n",
    "    for x_t, z_t in zip(x_test, z_test):\n",
    "        cutoff = condconf.predict(quantile=q, x_test=z_t.reshape(1,-1), score_inv_fn = lambda c, x: c, randomize=True)\n",
    "        perc_q.append(get_retained_claims([x_t], cutoff)[0])\n",
    "    percents[i] = perc_q\n",
    "    i += 1 \n",
    "\n",
    "\n",
    "perc_booleans = percents >= 0.8\n",
    "\n",
    "first_violating_indices = np.argmax(~perc_booleans, axis=0) - 1\n",
    "first_violating_indices[~np.any(~perc_booleans, axis=0)] = -1  # Use -1 to indicate no False found\n",
    "\n",
    "max_quantiles = quantile_grid[first_violating_indices]\n",
    "\n",
    "# truncate the alpha estimate?\n",
    "# max_quantiles = np.clip(max_quantiles, a_max=0.9, a_min=0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 190,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.collections.PathCollection at 0x17b837ca0>"
      ]
     },
     "execution_count": 190,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "condconf_alpha = CondConf(score_fn = lambda x, y: y, Phi_fn = phi_alpha)\n",
    "condconf_alpha.setup_problem(z_test, max_quantiles)\n",
    "_, primals = condconf_alpha._get_calibration_solution(quantile=0.25)\n",
    "\n",
    "quantile_fn = lambda x: np.clip(phi_alpha(x) @ primals, a_min=0.01, a_max=0.9)\n",
    "plt.scatter(phi_alpha(z_test)[:,1], quantile_fn(z_test))\n",
    "plt.scatter(phi_alpha(z_test)[:,1], max_quantiles)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 193,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "7627"
      ]
     },
     "execution_count": 193,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "len(data_conf[0])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 206,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████| 100/100 [01:38<00:00,  1.01it/s]\n"
     ]
    }
   ],
   "source": [
    "seed = 1\n",
    "nominal = []\n",
    "realized = []\n",
    "percents = []\n",
    "zs = []\n",
    "\n",
    "metadata_conf = metadata.iloc[ind_conf].reset_index(drop=True)\n",
    "views = metadata_conf[\"max_counts\"].to_numpy()\n",
    "views += 1\n",
    "z_views = views.clip(0, np.quantile(views, 0.95)).reshape(-1,1)\n",
    "z_views = z_views / np.mean(z_views)\n",
    "\n",
    "nTrials = 100\n",
    "nominal_avg = []\n",
    "for bin_left in np.linspace(0.65, .90, 11):\n",
    "    nominal_avg.append(bin_left + 0.0125)\n",
    "\n",
    "realized_covs = np.zeros((nTrials,len(nominal_avg)))\n",
    "perc_retained = np.zeros((nTrials,len(nominal_avg)))\n",
    "\n",
    "for seed in tqdm(range(nTrials)):\n",
    "    rng = np.random.default_rng(seed=seed)\n",
    "    \n",
    "    data_calib, data_test, ind_calib, ind_test = split_dataset(data_conf, rng, train_frac=0.95)\n",
    "    z_calib = z_views[ind_calib]\n",
    "    z_test = z_views[ind_test]\n",
    "    \n",
    "    x_calib, y_calib = data_calib\n",
    "    scores_calib = score_func(*data_calib, method=k)\n",
    "    score_fn = lambda x, y: y\n",
    "    phi_fn = lambda x : np.concatenate((np.ones((len(x), 1)), x, x**2, x**3, (0.9 - quantile_fn(x)).reshape(-1,1)**2),\n",
    "                                       axis=1)\n",
    "    # phi_fn = lambda x : np.ones((len(x), 1))\n",
    "    condconf = CondConf(score_fn = score_fn, Phi_fn = phi_fn, quantile_fn = quantile_fn)\n",
    "    condconf.setup_problem(z_calib, scores_calib)\n",
    "    \n",
    "    x_test, y_test = data_test\n",
    "    scores_test = score_func(*data_test, method=k)\n",
    "    percs = []\n",
    "    thresholds = []\n",
    "    coverages = []\n",
    "    for x_t, z_t, s_t in (zip(x_test, z_test, scores_test)):\n",
    "        threshold = condconf.predict(quantile=None, x_test=z_t.reshape(1,-1), score_inv_fn = lambda c, x: c, randomize=False)\n",
    "        percs.append(get_retained_claims([x_t], threshold)[0])\n",
    "        thresholds.append(threshold)\n",
    "        coverages.append(s_t <= threshold)\n",
    "    \n",
    "    coverages = np.array(coverages)\n",
    "    percs = np.array(percs)\n",
    "    count = 0\n",
    "    for bin_left in np.linspace(0.65, .90, 11):\n",
    "        bin_ind = np.where((condconf.quantile_fn(z_test) >= bin_left) & (condconf.quantile_fn(z_test) <= bin_left + 0.025))[0]\n",
    "        realized_covs[seed,count] = np.mean(coverages[bin_ind])\n",
    "        perc_retained[seed,count] = np.mean(percs[bin_ind].mean())\n",
    "        count += 1\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 212,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[0.6457172256654182,\n",
       " 0.7210207321479893,\n",
       " 0.7137934020507918,\n",
       " 0.7186817567338549,\n",
       " 0.7430025276592768,\n",
       " 0.7981987871818436,\n",
       " 0.8622287438033467,\n",
       " 0.8384763601336219,\n",
       " 0.8266613300699989,\n",
       " 0.8919014451292991,\n",
       " 0.9041452762532443]"
      ]
     },
     "execution_count": 212,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "realized_means"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 210,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[<matplotlib.lines.Line2D at 0x180baef10>]"
      ]
     },
     "execution_count": 210,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "stds=[]\n",
    "realized_means = []\n",
    "for i in range(len(nominal_avg)):\n",
    "    stds.append(1.96 * np.std(realized_covs[:,i])*np.sqrt(1/nTrials))\n",
    "    realized_means.append(np.mean(realized_covs[:,i]))\n",
    "plt.errorbar(nominal_avg, realized_means, stds, marker='o')\n",
    "plt.plot(np.linspace(0.6, 0.95, 100), np.linspace(0.6, 0.95, 100), '--', color='grey')\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "metadata": {},
   "outputs": [
    {
     "ename": "NameError",
     "evalue": "name 'dummies' is not defined",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mNameError\u001b[0m                                 Traceback (most recent call last)",
      "Input \u001b[0;32mIn [33]\u001b[0m, in \u001b[0;36m<module>\u001b[0;34m\u001b[0m\n\u001b[1;32m      1\u001b[0m \u001b[38;5;66;03m# whoops deleted the code that actually got the dummies...\u001b[39;00m\n\u001b[0;32m----> 2\u001b[0m all_dummies \u001b[38;5;241m=\u001b[39m np\u001b[38;5;241m.\u001b[39mconcatenate(\u001b[43mdummies\u001b[49m)\u001b[38;5;241m.\u001b[39mtolist()\n\u001b[1;32m      3\u001b[0m freq_claims \u001b[38;5;241m=\u001b[39m np\u001b[38;5;241m.\u001b[39mzeros(\u001b[38;5;28mlen\u001b[39m(freq_names))\n\u001b[1;32m      4\u001b[0m z_bin_vals \u001b[38;5;241m=\u001b[39m []\n",
      "\u001b[0;31mNameError\u001b[0m: name 'dummies' is not defined"
     ]
    }
   ],
   "source": [
    "\n",
    "# whoops deleted the code that actually got the dummies...\n",
    "all_dummies = np.concatenate(dummies).tolist()\n",
    "freq_claims = np.zeros(len(freq_names))\n",
    "z_bin_vals = []\n",
    "percent_vals = []\n",
    "for freq_name in freq_names:\n",
    "    ind = [i for i in range(len(all_dummies)) if all_dummies[i] == freq_name]\n",
    "    z_bin_vals.append(np.mean(zs[ind]))\n",
    "    percent_vals.append(np.mean([percents[i] for i in ind]))\n",
    "    print(freq_name)\n",
    "print(np.mean(np.asarray(percents) < 0.8))\n",
    "\n",
    "# Function to compute the moving average\n",
    "def moving_average(data, window_size):\n",
    "    return np.convolve(data, np.ones(window_size) / window_size, mode='valid')\n",
    "\n",
    "window_size = 50\n",
    "# zs = np.concatenate(zs)\n",
    "sorted_ind = np.argsort(zs)\n",
    "sorted_percents = np.asarray(percents)[sorted_ind]\n",
    "sorted_zs = zs[sorted_ind]\n",
    "plt.plot(sorted_zs[window_size - 1:], moving_average(sorted_percents, window_size))\n",
    "# plt.scatter(z_bin_vals, percent_vals, marker='x', color='red')\n",
    "plt.axhline(0.8, color='red')\n",
    "plt.xscale('log')\n",
    "plt.xlabel(\"Scaled View Count\")\n",
    "plt.ylabel(\"% Claims Retained\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 176,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor([0.2000, 0.2000, 0.2000, 0.2000, 0.2000, 0.2000], requires_grad=True)\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  0%|                                            | 1/1000 [00:00<09:10,  1.81it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor([0.4065]) tensor([0.2100, 0.2100, 0.1900, 0.2100, 0.2100, 0.1900], requires_grad=True) tensor([-5.2585e-02, -3.9245e-04,  1.7962e-05, -7.9548e-04, -1.4311e-03,\n",
      "         9.5514e-04])\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 10%|████▏                                     | 101/1000 [00:47<06:32,  2.29it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor([0.3752]) tensor([ 1.1299,  0.3971,  0.1711,  0.5962,  0.8372, -0.2969],\n",
      "       requires_grad=True) tensor([-3.5494e-02,  4.8702e-04,  4.4307e-04, -1.8554e-04,  2.1507e-05,\n",
      "         7.3314e-04])\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 20%|████████▍                                 | 201/1000 [01:36<06:43,  1.98it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor([0.3600]) tensor([ 1.8313,  0.4466,  0.0787,  0.8880,  1.2841, -0.5406],\n",
      "       requires_grad=True) tensor([-1.4621e-02,  2.6165e-06,  2.1343e-04, -3.1038e-04, -6.6194e-04,\n",
      "         2.7984e-04])\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 30%|████████████▋                             | 301/1000 [02:23<05:45,  2.02it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor([0.3545]) tensor([ 2.3467,  0.4652, -0.0134,  1.1547,  1.4701, -0.5570],\n",
      "       requires_grad=True) tensor([-0.0114, -0.0002, -0.0002, -0.0008,  0.0011,  0.0004])\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 40%|████████████████▊                         | 401/1000 [03:14<04:46,  2.09it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor([0.3454]) tensor([ 2.7656,  0.5738, -0.1919,  1.1932,  1.5071, -0.5506],\n",
      "       requires_grad=True) tensor([-0.0113,  0.0003, -0.0004, -0.0012,  0.0006, -0.0002])\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 50%|█████████████████████                     | 501/1000 [04:01<03:52,  2.15it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor([0.3569]) tensor([ 3.0848,  0.9086, -0.4177,  1.1429,  1.4306, -0.3446],\n",
      "       requires_grad=True) tensor([-5.6445e-03, -1.6202e-04,  7.5157e-05, -1.4987e-04, -1.6782e-04,\n",
      "         7.8899e-05])\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 60%|█████████████████████████▏                | 601/1000 [04:48<04:21,  1.52it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor([0.3409]) tensor([ 3.3728,  1.0428, -0.6764,  1.3569,  1.3519, -0.2451],\n",
      "       requires_grad=True) tensor([-0.0069, -0.0007,  0.0002,  0.0010, -0.0002, -0.0001])\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 70%|█████████████████████████████▍            | 701/1000 [05:36<02:11,  2.28it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor([0.3309]) tensor([ 3.5940,  1.2552, -0.8935,  1.5224,  1.3639, -0.1549],\n",
      "       requires_grad=True) tensor([-2.0453e-02,  3.1061e-04,  1.0276e-03, -1.7279e-03, -6.6058e-05,\n",
      "         1.7200e-03])\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 80%|█████████████████████████████████▋        | 801/1000 [06:23<01:36,  2.05it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor([0.3470]) tensor([ 3.8008,  1.3686, -1.1177,  1.4762,  1.2483,  0.0149],\n",
      "       requires_grad=True) tensor([-4.9504e-03, -1.0739e-03,  9.4815e-04, -3.9306e-04,  6.9500e-05,\n",
      "        -2.0042e-04])\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 90%|█████████████████████████████████████▊    | 901/1000 [07:09<00:42,  2.35it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor([0.3373]) tensor([ 4.0479,  1.7219, -1.2787,  1.6055,  0.9842,  0.4249],\n",
      "       requires_grad=True) tensor([-1.1780e-03, -1.3735e-04, -2.1796e-05, -2.9665e-04, -8.0297e-05,\n",
      "        -1.8183e-05])\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████████| 1000/1000 [07:51<00:00,  2.12it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor([0.3391], grad_fn=<AddBackward0>) tensor([ 4.2244,  1.8887, -1.4540,  1.8550,  0.6473,  0.8826],\n",
      "       requires_grad=True)\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "# need to boost now...\n",
    "# features to boost on: frequencies_arr, selfevals_arr, ordinals_arr\n",
    "\n",
    "import torch\n",
    "from scipy.optimize import linprog\n",
    "\n",
    "def estimate_true_error_cond(rng, q, n_eval, p, beta1, beta2):\n",
    "    dataset = generate_dataset(rng, n_eval, p)\n",
    "    data_calib, data_test = split_dataset(dataset, rng, train_frac=0.2)\n",
    "    x, y = data_calib\n",
    "    x_test, y_test = data_test\n",
    "    condconf_up = CondConf(\n",
    "        score_fn = lambda x, y: np.abs(y) / np.abs(x @ beta1),\n",
    "        Phi_fn = lambda x: np.concatenate((np.ones((len(x), 1)), x, x**2), axis=1)\n",
    "    )\n",
    "    condconf_down = CondConf(\n",
    "        score_fn = lambda x, y: np.abs(y) / np.abs(x @ beta2),\n",
    "        Phi_fn = lambda x: np.concatenate((np.ones((len(x), 1)), x, x**2), axis=1)\n",
    "    )\n",
    "\n",
    "    condconf_up.setup_problem(x, y)\n",
    "    condconf_down.setup_problem(x,y)\n",
    "    \n",
    "    lengths = []\n",
    "    for x_t in x_test:\n",
    "        raw_cutoff_up = condconf_up.predict(0.975, x_t.reshape(1,-1),\n",
    "                                         score_inv_fn = lambda cutoff, x: cutoff * np.abs(x.T @ beta1))\n",
    "        raw_cutoff_down = condconf_down.predict(0.025, x_t.reshape(1,-1),\n",
    "                                           score_inv_fn = lambda cutoff, x: cutoff * np.abs(x.T @ beta2))\n",
    "\n",
    "        lengths.append(raw_cutoff_up - raw_cutoff_down)\n",
    "        \n",
    "    lengths = np.asarray(lengths).flatten()\n",
    "\n",
    "    lengths = np.mean(lengths)\n",
    "\n",
    "    condconf = CondConf(\n",
    "        score_fn = lambda x, y: np.abs(y) / np.abs(x @ beta_opt),\n",
    "        Phi_fn = lambda x: np.concatenate((np.ones((len(x), 1)), x, x**2), axis=1)\n",
    "    )\n",
    "    condconf.setup_problem(x, y)\n",
    "    raw_cutoffs = []\n",
    "    for x_t in x_test:\n",
    "        raw_cutoff = condconf.predict(0.95, x_t.reshape(1,-1), score_inv_fn = lambda cutoff, x: cutoff * np.abs(x.T @ beta_opt))\n",
    "        raw_cutoffs.append(raw_cutoff)\n",
    "    raw_cutoffs = np.asarray(raw_cutoffs).flatten()\n",
    "\n",
    "    lengths_opt = np.mean(raw_cutoffs)\n",
    "    \n",
    "    return lengths, lengths_opt\n",
    "    \n",
    "\n",
    "def get_current_basis(primals, duals, Phi, S, quantile):\n",
    "    interp_bools = np.logical_and(~np.isclose(duals, quantile - 1),~np.isclose(duals, quantile))\n",
    "    if np.sum(interp_bools) == Phi.shape[1]:\n",
    "        return interp_bools\n",
    "    preds = (Phi @ primals).flatten()\n",
    "    active_indices = np.where(interp_bools)[0]\n",
    "    interp_indices = np.argsort(np.abs(S - preds))[0:Phi.shape[1]]\n",
    "    diff_indices = np.setdiff1d(interp_indices, active_indices)\n",
    "    num_missing = Phi.shape[1] - np.sum(interp_bools)\n",
    "    if num_missing < len(diff_indices):\n",
    "        from itertools import combinations\n",
    "        for cand_indices in combinations(diff_indices, num_missing):\n",
    "            cand_phi = Phi[np.concatenate((active_indices, cand_indices))]\n",
    "            if np.isfinite(np.linalg.cond(cand_phi)):\n",
    "                interp_bools[np.asarray(cand_indices)] = True\n",
    "                break\n",
    "    else:\n",
    "        interp_bools[diff_indices] = True\n",
    "    return interp_bools\n",
    "            \n",
    "def solve_qr(Phi, s, q, dtype):\n",
    "    S = s.detach().numpy()\n",
    "    zeros = np.zeros((Phi.shape[1],))\n",
    "    bounds = [(q - 1, q)] * len(S)\n",
    "    res = linprog(-1 * S, A_eq=Phi.T, b_eq=zeros, bounds=bounds,\n",
    "                    method='highs-ds', options={'presolve': False})\n",
    "\n",
    "    tau = -1 * res.eqlin.marginals\n",
    "\n",
    "    basis = get_current_basis(tau, res.x, Phi, S, q)\n",
    "    \n",
    "    tau = as_tensor(np.linalg.inv(Phi[basis].T @ Phi[basis]) @ Phi[basis].T, dtype=dtype) @ s[basis]\n",
    "    return tau.reshape(-1,1)\n",
    "\n",
    "def as_tensor(x, dtype, requires_grad=False):\n",
    "    return torch.tensor(x, dtype=dtype, requires_grad=requires_grad)\n",
    "    \n",
    "def cond_score_length(beta, dataset, metadata, rng, q=0.9):\n",
    "    data_train, data_calib, ind_train, ind_calib = split_dataset(dataset, rng, train_frac = 0.5)\n",
    "    x_train, y_train = data_train\n",
    "    x_calib, y_calib = data_calib\n",
    "\n",
    "    views = metadata[\"max_counts\"].to_numpy()\n",
    "    views += 1\n",
    "    z_views = views.clip(0, np.quantile(views, 0.95)).reshape(-1,1)\n",
    "    z_views = z_views / np.mean(z_views)\n",
    "    z_dummies = freq_dummies.to_numpy().astype(int)\n",
    "\n",
    "    z_arr = z_views\n",
    "    #z_arr = z_dummies \n",
    "    z_train = z_arr[ind_train]\n",
    "    z_calib = z_arr[ind_calib]\n",
    "    \n",
    "    scores_train = torch_score_func(x_train, y_train, beta, method=3)\n",
    "\n",
    "    Phi_fn = lambda x: np.concatenate((np.ones((len(x), 1)), x, x**2, x**3), axis=1)\n",
    "    Phi_fn = lambda x : x\n",
    "    \n",
    "    # replace cutoff computation with quantile regression\n",
    "    phi_train = Phi_fn(z_train)\n",
    "    try:\n",
    "        tau = solve_qr(phi_train, scores_train, q, beta.dtype)\n",
    "    except:\n",
    "        print(\"gradient broke\")\n",
    "        tau = torch.zeros((phi_train.shape[1],), dtype=beta.dtype)\n",
    "\n",
    "    phi_calib = Phi_fn(z_calib)\n",
    "    cutoffs = torch.flatten(as_tensor(phi_calib, dtype=beta.dtype) @ tau)\n",
    "\n",
    "    total = torch.zeros(1, requires_grad=True)\n",
    "    for x_c, cutoff in zip(x_calib, cutoffs):\n",
    "        perc = torch.sigmoid((cutoff - torch.sigmoid(torch.as_tensor(x_c, dtype=beta.dtype) @ beta)))\n",
    "        total = total + torch.mean(perc) / len(cutoffs)\n",
    "\n",
    "    return total, (x_calib, cutoffs), tau\n",
    "\n",
    "def torch_score_func(features, annotations, beta, method):\n",
    "    scores = as_tensor(np.zeros((len(features),)), dtype=beta.dtype)\n",
    "    for i, (f, a) in enumerate(zip(features, annotations)):\n",
    "        cs = torch.sigmoid(as_tensor(f, dtype=beta.dtype) @ beta)\n",
    "        at = as_tensor(a, dtype=bool)\n",
    "        scores[i] = torch.sort(cs[~at], descending=True)[0][method] if torch.sum(~at) > method else -1\n",
    "    return scores\n",
    "    \n",
    "rng = np.random.default_rng(seed=0)\n",
    "z_feat_arr = [z * np.ones_like(f_arr) for z, f_arr in zip(z_views, frequencies_arr)]\n",
    "#features_arr = [np.concatenate((f_arr.reshape(-1,1), s_arr.reshape(-1,1), o_arr.reshape(-1,1) / o_arr.max(),\n",
    "#                               z_arr.reshape(-1,1)), axis=1)\n",
    "#                for f_arr, s_arr, o_arr, z_arr in zip(frequencies_arr, selfevals_arr, ordinal_arr, z_feat_arr)]\n",
    "#features_arr = [np.concatenate((s_arr.reshape(-1,1), o_arr.reshape(-1,1) / o_arr.max(),\n",
    "#                               z_arr.reshape(-1,1)), axis=1)\n",
    "#                for s_arr, o_arr, z_arr in zip(selfevals_arr, ordinal_arr, z_feat_arr)]\n",
    "#features_arr = [np.concatenate((f_arr.reshape(-1,1), f_arr.reshape(-1,1)**2, f_arr.reshape(-1,1)**3,\n",
    "#                               f_arr.reshape(-1,1)**4), axis=1)\n",
    "#                for f_arr in frequencies_arr]\n",
    "#features_arr = [np.concatenate((s_arr.reshape(-1,1), s_arr.reshape(-1,1)**2, s_arr.reshape(-1,1)**3 , o_arr.reshape(-1,1),\n",
    "#                               o_arr.reshape(-1,1)**2, o_arr.reshape(-1,1)**3), axis=1)\n",
    "#                for s_arr, o_arr  in  zip(selfevals_arr, ordinal_arr)]\n",
    "prefix = 'Please write one biographical paragraph about '\n",
    "embeddings = np.load(\"/Users/isaacgibbs/Downloads/factscore_final_embeddings.npz\")\n",
    "features_arr = [np.concatenate((f_arr.reshape(-1,1),embeddings[prefix + x + '.']), axis = 1)\n",
    "                for (x, f_arr) in zip(metadata['Name'], frequencies_arr)]\n",
    "\n",
    "\n",
    "dataset = (features_arr, annotations_arr)\n",
    "# split dataset\n",
    "data_alpha, data_conf, ind_alpha, ind_conf = split_dataset(dataset, rng, train_frac=0.5)\n",
    "metadata_alpha = metadata.iloc[ind_alpha].reset_index(drop=True)\n",
    "\n",
    "\n",
    "beta_cond = torch.tensor(np.asarray([1/5]*6), dtype=torch.float, requires_grad=True)\n",
    "print(beta_cond)\n",
    "optimizer = torch.optim.Adam([beta_cond], lr = 0.01)\n",
    "\n",
    "obj_percs = []\n",
    "# true_lengths = []\n",
    "# n_eval = 1000\n",
    "for i in tqdm(range(1000)):\n",
    "    optimizer.zero_grad()\n",
    "    perc, (x_calib, cutoffs), tau = cond_score_length(beta_cond, data_alpha, metadata_alpha, rng)\n",
    "    perc.backward()\n",
    "    optimizer.step()\n",
    "    obj_percs.append(perc.detach())\n",
    "    if i % 100 == 0:\n",
    "        # true_error, true_error_baseline = estimate_true_error_cond(rng, 0.95, n_eval, p, beta_cond.detach().numpy())\n",
    "        # print(f\"iteration {i}\", true_error - true_error_baseline)\n",
    "        # true_lengths.append(true_error - true_error_baseline)\n",
    "        print(obj_percs[-1], beta_cond, beta_cond.grad)\n",
    "print(perc,beta_cond)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 177,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████| 5/5 [00:05<00:00,  1.04s/it]\n"
     ]
    }
   ],
   "source": [
    "features_arr_test = data_conf[0]\n",
    "annotations_arr_test = data_conf[1]\n",
    "\n",
    "scores_arr = [f @ beta_cond.detach().numpy() for f in features_arr_test]\n",
    "\n",
    "n_trials = 5 #50\n",
    "all_covs_jc = np.zeros((n_trials, 1 + len(continent_names) + len(freq_names)))\n",
    "all_claims_jc = np.zeros((n_trials, 1 + len(continent_names) + len(freq_names)))\n",
    "\n",
    "rng = np.random.default_rng(seed=0)\n",
    "k = 3\n",
    "quantile = 0.9\n",
    "\n",
    "z_ones = np.ones((len(features_arr_test), 1))\n",
    "views_test = metadata.iloc[ind_conf][\"max_counts\"].to_numpy()\n",
    "views_test += 1\n",
    "z_views_test = views_test.clip(0, np.quantile(views_test, 0.95)).reshape(-1,1)\n",
    "z_views_test = z_views_test / np.mean(z_views_test)\n",
    "z_dummies = freq_dummies.iloc[ind_conf].to_numpy().astype(int)\n",
    "z_arr_test = np.concatenate((z_ones, z_views_test, z_views_test**2, z_views_test**3), axis=1)\n",
    "#z_arr = z_dummies\n",
    "scores_arr_jitter = [score + rng.uniform(low=0, high=1e-3, size=score.shape) for score in scores_arr]\n",
    "rng = np.random.default_rng(seed=1)\n",
    "for trial in tqdm(range(n_trials)):\n",
    "    all_covs_jc[trial], all_claims_jc[trial] = run_coverage_trial(scores_arr_jitter,\n",
    "                                                                  annotations_arr_test, z_arr_test,\n",
    "                                                                  rng, k, quantile, continent_dummies.iloc[ind_conf,:],\n",
    "                                                                 freq_dummies.iloc[ind_conf,:])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 178,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.8230055394143413 0.827438538964389\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAtMAAAFzCAYAAAD8AIVCAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjUuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/YYfK9AAAACXBIWXMAAAsTAAALEwEAmpwYAAA4vUlEQVR4nO3de5xd873/8dcnk0gmIi5B6GiETgiCkJGi7kFp1e1HUW1GT0tpJaFo0dNTbR1H8WulqVI/5xCtllONurTS4xa3o8gNcQ9NHHMiIgiRSWSS7++PvWaMmJnMrJk9e2byej4e85i9v3ut9f3sycrs93z3d39XpJSQJEmS1Ha9Sl2AJEmS1F0ZpiVJkqScDNOSJElSToZpSZIkKSfDtCRJkpSTYVqSJEnKqXepC2iPTTfdNA0dOrTUZUiSJKmHmzFjxlsppc3WbO/WYXro0KFMnz691GVIkiSph4uI+U21O81DkiRJyskwLUmSJOVkmJYkSZJy6tZzpiVJUsdauXIlr7/+OsuXLy91KVJJ9OvXj6222oo+ffq0avuihemI+A/gCODNlNKIrG0T4BZgKDAP+HJK6Z2ICGAi8AVgGXBKSmlmsWqTJElNe/3119lggw0YOnQohZdnad2RUmLx4sW8/vrrbLPNNq3ap5jTPG4ADluj7XzgvpTSMOC+7D7A4cCw7Os04Ooi1iVJkpqxfPlyBg0aZJDWOikiGDRoUJvemSlamE4pPQS8vUbzUcDk7PZk4OhG7Temgr8DG0XElsWqTZIkNc8grXVZW8//zv4A4uCU0oLs9hvA4Ox2BfA/jbZ7PWv7hIg4LSKmR8T0RYsWFa9SSZJUEmVlZYwcOZJdd92V3Xffnf/+7//u0ONfcsklbd7nhhtu4Mwzz2zysbvvvpuqqip23HFHdtttN84555z2lgjAKaecwq233grAN7/5TZ577jngk/XvvffeHdJfc32v6YorrmD48OGMHDmSPfbYgxtvvLHVx2r8PAYMGNCuGqdNm/axc+Oaa65Zay3FULIPIKaUUkSkHPtdC1wLUFVV1eb9JUlS65353fN4860132jOb/NNN+FXP7+8xW3Ky8uZPXs2AH/729+44IILePDBBzushksuuYQLL7ywQ441Z84czjzzTP7yl78wfPhwVq1axbXXXtshx27suuuua7i9Zv0d/cdGS6655hruuecennjiCQYOHMh7773Hbbfd1ur9Gz+P1qirq6N376bj6rRp0xgwYEDDHxOnn356m47dUTp7ZHph/fSN7PubWXsN8OlG222VtUmSpBJ68623eWXw/h321dZg/t5777HxxhsDhQ+HnXfeeYwYMYKdd96ZW265pcX2BQsWsN9++zFy5EhGjBjBww8/zPnnn09tbS0jR47k5JNPBuB3v/sdo0ePZuTIkXzrW99i1apVAFx//fVst912jB49mkcffbTJ+i677DJ+8IMfMHz4cKAwqn7GGWcAMG/ePA466CB22WUXxowZw2uvvQYURmrHjx/P3nvvzbbbbtswaptS4swzz2T77bfn4IMP5s0332zo54ADDmD69OlN1l8/wtvcz2HatGkccMABHHfccQwfPpyTTz6ZlArjkT/5yU/YY489GDFiBKeddlpDe3MuueQSrr76agYOHAjAwIEDqa6ubvWx6p9HvbPPPpuddtqJMWPGUD/j4IADDuCss86iqqqKiRMncuedd/LZz36W3XbbjYMPPpiFCxcyb948rrnmGn7xi18wcuRIHn74YS666CKuuOIKAGbPns2ee+7JLrvswjHHHMM777zTcOzvf//7jB49mu22246HH364xefbGp0dpu8AqrPb1cDtjdrHRsGewJJG00EkSdI6pD4sDh8+nG9+85v88Ic/BGDKlCnMnj2bp556invvvZfzzjuPBQsWNNv++9//ns9//vMNj40cOZJLL720YeT7pptu4vnnn+eWW27h0UcfZfbs2ZSVlXHTTTexYMECfvSjH/Hoo4/yyCOPNExNWNOcOXMYNWpUk4+NGzeO6upqnn76aU4++WTGjx/f8NiCBQt45JFHuOuuuzj//MJ6DLfddhsvvvgizz33HDfeeGOTI85r1t9Ycz8HgFmzZnHllVfy3HPP8eqrrzb8cXDmmWfy5JNPMmfOHGpra7nrrrua/Xd57733eP/999l2222bfLwtxwL44IMPqKqq4tlnn2X//ffnxz/+ccNjH374IdOnT+ecc85hn3324e9//zuzZs3ixBNP5LLLLmPo0KGcfvrpnH322cyePZt99933Y8ceO3YsP/vZz3j66afZeeedP3bsuro6nnjiCa688sqPtedVzKXx/gAcAGwaEa8DPwIuBf4zIr4BzAe+nG3+VwrL4s2lsDTe14tVlz4yadIk5s6d26Z9amoKbxhUVDQ5pb1ZlZWVjBs3rk37SJLWTY2neTz22GOMHTuWOXPm8Mgjj3DSSSdRVlbG4MGD2X///XnyySebbd9jjz34p3/6J1auXMnRRx/NyJEjP9HXfffdx4wZM9hjjz2AQpDffPPNefzxxznggAPYbLPNADjhhBN46aWX2vQ8HnvsMaZMmQLA1772Nb73ve81PHb00UfTq1cvdtxxRxYuXAjAQw891PA8PvWpT3HQQQe1qb/mfg4DBw5k9OjRbLXVVgCMHDmSefPmsc8++/DAAw9w2WWXsWzZMt5++2122mknvvSlL7Wp33ptPVavXr044YQTAPjqV7/Kscce2/BYfTsUlms84YQTWLBgAR9++OFal6xbsmQJ7777Lvvvvz8A1dXVHH/88Q2P1/czatQo5s2b1+bn+Ynn0e4jNCOldFJKacuUUp+U0lYppX9PKS1OKY1JKQ1LKR2cUno72zallL6TUvpMSmnnlNL0tR1fpVFbW0ttbW2py5AkrSP22msv3nrrLfIsOrDffvvx0EMPUVFRwSmnnNLkh9NSSlRXVzN79mxmz57Niy++yEUXXdTqPnbaaSdmzJjR5tr69u37sRqKrXF/ZWVl1NXVsXz5cr797W9z66238swzz3Dqqae2uCTcwIEDGTBgAK+++uonHmvrsZrSeBWN9ddfv+H2uHHjOPPMM3nmmWf4zW9+0+4LCtX/LOp/Du3lFRDXYXlGiidMmADAxIkTO7ocSZI+4YUXXmDVqlUMGjSIfffdl9/85jdUV1fz9ttv89BDD3H55ZdTV1fXZPv8+fPZaqutOPXUU1mxYgUzZ85k7Nix9OnTh5UrV9KnTx/GjBnDUUcdxdlnn83mm2/O22+/zfvvv89nP/tZJkyYwOLFixk4cCB//OMf2XXXXT9R33nnncexxx7LPvvsw3bbbcfq1au59tprOf3009l77725+eab+drXvsZNN930iakIa9pvv/0ansebb77JAw88wFe+8pVPbNe4/saa+/m88MILTfZXH0o33XRTli5dyq233spxxx3XYo0XXHAB3/nOd7jlllsYOHAgS5cuZcqUKRx55JFtPtbq1au59dZbOfHEE/n973/PPvvs0+R2S5YsaXhHfPLkyQ3tG2ywAe+9994ntt9www3ZeOONefjhh9l333357W9/2zBKXQyGaUmS1KXUz5mGwqjt5MmTKSsr45hjjuGxxx5j1113JSK47LLL2GKLLZptnzx5Mpdffjl9+vRhwIABDSPTp512Grvssgu77747N910ExdffDGHHnooq1evpk+fPlx11VXsueeeXHTRRey1115stNFGTU4RAdhll1248sorOemkk1i2bBkRwRFHHAEUplN+/etf5/LLL2ezzTbj+uuvb/F5H3PMMdx///3suOOODBkyhL322qvJ7dasv/H+Tf0cmgvTG220EaeeeiojRoxgiy22aJjq0pIzzjiDpUuXsscee9CnTx/69OnDOeeck+tY66+/Pk888QQXX3wxm2++ecMHJtd00UUXcfzxx7Pxxhtz0EEH8Y9//AOAL33pSxx33HHcfvvtTJo06WP7TJ48mdNPP51ly5ax7bbbrvVn3x7RGW8tFEtVVVVq/InQdVme+c951PdRWVlZ9L6cZy1Jne/5559nhx12aLhfiqXxpFJb8/8BQETMSClVrbmtI9M9xIMPPsjbby2ib1lx/zhaubown+mlObOK2s+KVUFNTY1hWpJKzOArtcww3YP0LUtsvcGqUpfRIea/X1bqEiRJktbKMN0K3WEJuYqKCl5+9821b9hOC5cVFoAZ3H91UfuJaPvPTpIkqbOtc2E6bzBu63Jw9du3db+ampo211dZWdkpc5gBPsxq67t1cfsbRufMy5YkSWqPdS5MP/jggyx6azGUFfmpZ1OXly7/sE27LV3+IYveWdL6HVbVUVNT03Ap0mJzaTxJkqSPdPblxCVJkqQeY50bmd5///07dZpHeb/12rRfeXl5rnnWeeSZ8lK/ff0IdWu5zJ0kqbXeeOMNzjrrLJ588kk22mgjBg8ezJVXXsl2223XruNOmzaNK664grvuuos77riD5557jvPPP58///nPbLfdduy4444A/Mu//Av77bcfBx98cEc8nU/0vaYnnniCc889l4ULF9K/f39GjRrFL3/5S/r379+m53HKKadwxBFHrPViKc159913+f3vf8+3v/1tAP73f/+X8ePHd9q7393VOhem8wS67vABxM5SXl5e6hIkSZ3ownPOZMlbCzvseBtuOphL/u+vmn08pcQxxxxDdXU1N998MwBPPfUUCxcubHeYbuzII49suGrfn//8Z4444oiGMP2Tn/ykw/pZm4ULF3L88cdz8803N1yk5dZbb+X9999vNkw31vh5tEZdXR29ezcd/959911+/etfN4TpT33qUwbpVljnwnQeXTnctkdPfV6SpI6z5K2FfP8zTV9BL4+fvdLy4w888AB9+vTh9NNPb2irv4x3Sonvfe973H333UQE//zP/8wJJ5zAtGnTuOiii9h0002ZM2cOo0aN4ne/+x0RwdSpUznrrLPo37//xy5XfcMNNzB9+nS+8pWvcMcdd/Dggw9y8cUX86c//Ymf/vSnDSO89913H+eeey51dXXsscceXH311fTt25ehQ4dSXV3NnXfeycqVK/njH//I8OHDeeKJJ5gwYQLLly+nvLyc66+/nu23377Z53vVVVdRXV39sasd1o8st+ZY9c/jV78q/IFy7733cumll/Lee+/x85//nCOOOIIbbriBKVOmsHTpUlatWsVf/vIXjjrqKN555x1WrlzJxRdfzFFHHcX555/PK6+8wsiRIznkkEP4zne+wxFHHMGcOXNYvnw5Z5xxBtOnT6d37978/Oc/58ADD+SGG27gjjvuYNmyZbzyyiscc8wxXHbZZa07GXoIw7QkSeoy6sNwU6ZMmcLs2bN56qmneOutt9hjjz3Yb7/9AJg1axbPPvssn/rUp/jc5z7Ho48+SlVVFaeeeir3338/lZWVnHDCCZ845t57782RRx7Z5PSI5cuXc8opp3Dfffex3XbbMXbsWK6++mrOOussADbddFNmzpzJr3/9a6644gquu+46hg8fzsMPP0zv3r259957ufDCC/nTn/7U4vOtrq5u8rG2Hgtg3rx5PPHEE7zyyisceOCBDe+sz5w5k6effppNNtmEuro6brvtNgYOHMhbb73FnnvuyZFHHsmll17KnDlzmD17dsOx6l111VVEBM888wwvvPAChx56KC+99BIAs2fPZtasWfTt25ftt9+ecePG8elPf7rFOnsSw7QkSeoWHnnkEU466STKysoYPHgw+++/P08++SQDBw5k9OjRbLXVVgCMHDmSefPmMWDAALbZZhuGDRsGwFe/+lWuvfbaVvf34osvss022zRML6muruaqq65qCNPHHnssAKNGjWLKlCkALFmyhOrqal5++WUigpUrV+Z+vnmO9eUvf5levXoxbNgwtt12W154ofCuwiGHHMImm2wCFEb4L7zwQh566CF69epFTU0NCxe2PJXnkUceaXhHe/jw4Wy99dYNYXrMmDFsuOGGAOy4447Mnz9/nQrTruYhSZK6jJ122okZM2a0eb++ffs23C4rK6Ourq4jy2qxz8b9/fCHP+TAAw9kzpw53HnnnSxfvrzFY7T0fNt6LICIaPL++uuv39B20003sWjRImbMmMHs2bMZPHhwq47dnFL87LsSw7QkSeoyDjroIFasWPGxEeSnn36ahx9+mH333ZdbbrmFVatWsWjRIh566CFGjx7d7LGGDx/OvHnzeOWVwkTtP/zhD01ut8EGG/D+++9/on377bdn3rx5DVMlfvvb37L//vu3WP+SJUsaFh+44YYbWtwW4Mwzz2Ty5Mk8/vjjDW1Tpkxh4cKFbT4WwB//+EdWr17NK6+8wquvvtrkfO0lS5aw+eab06dPHx544AHmz58PNP9zANh333256aabAHjppZd47bXXWpwLvi4xTEuSpC4jIrjtttu49957+cxnPsNOO+3EBRdcwBZbbMExxxzDLrvswq677spBBx3EZZddxhZbbNHssfr168e1117LF7/4RXbffXc233zzJrc78cQTufzyy9ltt90agnf9/tdffz3HH388O++8M7169frYByOb8r3vfY8LLriA3XbbrVUjtIMHD+bmm2/m3HPPZfvtt2eHHXbgb3/7GxtssEGbjwUwZMgQRo8ezeGHH84111xDv379PrHNySefzPTp09l555258cYbGT58OACDBg3ic5/7HCNGjOC888772D7f/va3Wb16NTvvvDMnnHACN9xww8dGpNdlkVIqdQ25VVVVpenTp5e6DEmSeoznn3+eHXbYoeF+Zy+NJ3UFa/4/AIiIGSmlqjW39QOIkiSpWQZfqWVO85AkSZJyMkxLkiRJORmmJUnSx3Tnz1NJ7dXW898wLUmSGvTr14/FixcbqLVOSimxePHiJldBaY4fQJQkSQ222morXn/9dRYtWlTqUqSS6NevX8PVNFvDMC1Jkhr06dOHbbbZptRlSN2G0zwkSZKknAzTkiRJUk6GaUmSJCknw7QkSZKUk2FakiRJysnVPKScJk2axNy5c9u0T01NDQAVFRVt2q+yspJx48a1aZ+8eurzkiSpGAzTUieqra0tdQlF0VOflyRJaxPd+QpHVVVVafr06aUuQ2q1CRMmADBx4sSi95VnhDmv+n4qKyuL3pej2ZKkUoiIGSmlqjXbHZmWeqi5c+fy8rOzGDJgVdH7Wm9l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\n",
      "text/plain": [
       "<Figure size 864x432 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Create a DataFrame for each method\n",
    "names = [\"Marginal\"] + continent_names.tolist() + freq_names.tolist()\n",
    "indices_to_plot = [-5, -4, -3, -2, -1]\n",
    "\n",
    "names = [names[i] for i in indices_to_plot]\n",
    "df1 = pd.DataFrame(all_claims_jc[:,indices_to_plot] * 100, columns=names)\n",
    "df2 = pd.DataFrame(all_claims[:,indices_to_plot] * 100, columns=names)\n",
    "\n",
    "# Add a column to identify the method\n",
    "df1['Method'] = 'Boosted Conditional Calibration'\n",
    "df2['Method'] = 'Conditional Calibration'\n",
    "\n",
    "# Combine the DataFrames\n",
    "df = pd.concat([df1, df2])\n",
    "\n",
    "# Melt the combined DataFrame\n",
    "df_melted = df.melt(id_vars='Method', var_name='Group', value_name='% Claims Retained')\n",
    "\n",
    "# Create the boxplot\n",
    "plt.figure(figsize=(12, 6))\n",
    "sns.boxplot(x='Group', y='% Claims Retained', hue='Method', data=df_melted)\n",
    "# Customize the plot\n",
    "plt.legend()\n",
    "\n",
    "print(np.mean(all_claims_jc[:,0]), np.mean(all_claims[:,0]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 126,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(array([[0.81904708, 0.79930868, 0.84777225, 0.85388823, 0.78933617,\n",
       "         0.75921314, 0.96982432, 0.89912396, 0.81348683, 0.66631874,\n",
       "         0.51921578],\n",
       "        [0.82747089, 0.82913459, 0.84275367, 0.83979566, 0.81447809,\n",
       "         0.75620749, 0.96291105, 0.91066679, 0.80872932, 0.68700868,\n",
       "         0.31725913],\n",
       "        [0.82827283, 0.77570058, 0.85259104, 0.82300521, 0.8131433 ,\n",
       "         0.72603448, 0.96461757, 0.91641   , 0.8019857 , 0.69276059,\n",
       "         0.48563006],\n",
       "        [0.81767048, 0.83701582, 0.84122923, 0.80290075, 0.80016196,\n",
       "         0.76966802, 0.96558571, 0.90588489, 0.80991663, 0.66688949,\n",
       "         0.49346151],\n",
       "        [0.82097018, 0.78388803, 0.83867993, 0.81859381, 0.807534  ,\n",
       "         0.76232692, 0.9729174 , 0.90014497, 0.82003034, 0.65261369,\n",
       "         0.41638749]]),\n",
       " array([[0.82367197, 0.76264288, 0.85635753, 0.80998055, 0.80076785,\n",
       "         0.74930409, 0.97391676, 0.89965515, 0.79950123, 0.68833418,\n",
       "         0.45519984],\n",
       "        [0.82765015, 0.78369948, 0.85783801, 0.82412709, 0.80206215,\n",
       "         0.77936582, 0.97748292, 0.90295259, 0.80654342, 0.69991146,\n",
       "         0.38225003],\n",
       "        [0.82214963, 0.7981219 , 0.85328795, 0.82686923, 0.7938807 ,\n",
       "         0.78132665, 0.98882024, 0.90599702, 0.81156279, 0.65254291,\n",
       "         0.50630006],\n",
       "        [0.82633467, 0.83387136, 0.85376875, 0.80316754, 0.80415142,\n",
       "         0.78069153, 0.96633849, 0.92220107, 0.80222372, 0.68171157,\n",
       "         0.51724792],\n",
       "        [0.82570485, 0.77631792, 0.85555237, 0.80466752, 0.80469878,\n",
       "         0.76301112, 0.9689168 , 0.91765039, 0.80220598, 0.6863893 ,\n",
       "         0.47435641],\n",
       "        [0.83252443, 0.78409334, 0.86518133, 0.78897614, 0.80976776,\n",
       "         0.7933892 , 0.97828165, 0.9183252 , 0.80593198, 0.6949503 ,\n",
       "         0.5565409 ],\n",
       "        [0.82305475, 0.76696855, 0.84513785, 0.82027315, 0.80946129,\n",
       "         0.72237753, 0.97031944, 0.91881005, 0.79792441, 0.67499483,\n",
       "         0.54590324],\n",
       "        [0.82607644, 0.80064104, 0.85105314, 0.81653207, 0.80690425,\n",
       "         0.77603369, 0.97560104, 0.91267493, 0.80157226, 0.67659169,\n",
       "         0.55194258],\n",
       "        [0.83442319, 0.78877608, 0.8548059 , 0.83870113, 0.81714722,\n",
       "         0.79560908, 0.97558667, 0.91831368, 0.8123356 , 0.69835757,\n",
       "         0.45625805],\n",
       "        [0.82964354, 0.76441558, 0.86113733, 0.82395079, 0.80531754,\n",
       "         0.7650977 , 0.97556523, 0.91682089, 0.81014005, 0.68247989,\n",
       "         0.55935643],\n",
       "        [0.82731044, 0.76071987, 0.85150521, 0.82546256, 0.80653748,\n",
       "         0.79839744, 0.97353028, 0.90980175, 0.80412575, 0.69435109,\n",
       "         0.53338942],\n",
       "        [0.82133505, 0.79886416, 0.84701488, 0.79954472, 0.80236342,\n",
       "         0.78487619, 0.97832819, 0.9148677 , 0.79843496, 0.66986833,\n",
       "         0.41361973],\n",
       "        [0.82291842, 0.77027379, 0.84859358, 0.80541041, 0.80411252,\n",
       "         0.78020466, 0.98154451, 0.89792055, 0.80404433, 0.68548159,\n",
       "         0.48875236],\n",
       "        [0.82823748, 0.78219531, 0.85527801, 0.8064681 , 0.80733904,\n",
       "         0.79969833, 0.98399368, 0.91770718, 0.80480363, 0.67840125,\n",
       "         0.52758245],\n",
       "        [0.83023585, 0.79116953, 0.85098269, 0.79952244, 0.81677917,\n",
       "         0.7957475 , 0.9834702 , 0.9117044 , 0.8094595 , 0.69226819,\n",
       "         0.58024682],\n",
       "        [0.82602618, 0.78401852, 0.85763663, 0.82049542, 0.80342044,\n",
       "         0.72286897, 0.97918309, 0.91294322, 0.80046068, 0.69247664,\n",
       "         0.51285576],\n",
       "        [0.83434378, 0.79479376, 0.8581315 , 0.82653679, 0.81634898,\n",
       "         0.80460942, 0.98921206, 0.91323373, 0.81060173, 0.69623084,\n",
       "         0.50994147],\n",
       "        [0.82549711, 0.79382217, 0.85231786, 0.83912247, 0.80202117,\n",
       "         0.75010832, 0.98067029, 0.91150571, 0.79979705, 0.69539763,\n",
       "         0.44488187],\n",
       "        [0.82896851, 0.78992493, 0.85022856, 0.83305594, 0.80970324,\n",
       "         0.79891866, 0.97744698, 0.90670602, 0.81214266, 0.69047842,\n",
       "         0.53113402],\n",
       "        [0.82849737, 0.78006235, 0.85120797, 0.83752816, 0.80811378,\n",
       "         0.79232525, 0.96908343, 0.8972564 , 0.80222018, 0.71628714,\n",
       "         0.55091759],\n",
       "        [0.83043253, 0.80514428, 0.84821107, 0.84517562, 0.81552996,\n",
       "         0.77219206, 0.9719426 , 0.91040098, 0.80570745, 0.70370897,\n",
       "         0.47098248],\n",
       "        [0.8267205 , 0.77339588, 0.84238785, 0.8253196 , 0.812101  ,\n",
       "         0.83044556, 0.98765466, 0.90588373, 0.79661567, 0.6980573 ,\n",
       "         0.53298113],\n",
       "        [0.82085872, 0.77073363, 0.84689796, 0.8209607 , 0.8010216 ,\n",
       "         0.73864858, 0.97481465, 0.91311951, 0.79991055, 0.66602614,\n",
       "         0.4579202 ],\n",
       "        [0.82566315, 0.77496727, 0.85736582, 0.79912457, 0.803173  ,\n",
       "         0.77055959, 0.98738203, 0.90470981, 0.79842263, 0.68569998,\n",
       "         0.44993204],\n",
       "        [0.83240021, 0.79592456, 0.85928461, 0.83997895, 0.80752782,\n",
       "         0.79228643, 0.98696896, 0.91270272, 0.80971623, 0.68878623,\n",
       "         0.53379188],\n",
       "        [0.83261921, 0.8287012 , 0.85715363, 0.84401541, 0.80970143,\n",
       "         0.7781185 , 0.98233312, 0.91830966, 0.81132984, 0.68730711,\n",
       "         0.4905724 ],\n",
       "        [0.82996148, 0.8246878 , 0.85549629, 0.80602048, 0.81231685,\n",
       "         0.76739599, 0.97188286, 0.91662209, 0.80771062, 0.69534129,\n",
       "         0.5085116 ],\n",
       "        [0.82873956, 0.78104529, 0.85394845, 0.80444641, 0.80963055,\n",
       "         0.79772494, 0.98261437, 0.91508205, 0.80505369, 0.68460857,\n",
       "         0.51172809],\n",
       "        [0.8274692 , 0.80872922, 0.85335775, 0.82523138, 0.80721396,\n",
       "         0.75368031, 0.98727063, 0.90977514, 0.79988336, 0.7018909 ,\n",
       "         0.52726842],\n",
       "        [0.82853266, 0.78253602, 0.85138016, 0.80440409, 0.81414824,\n",
       "         0.77674278, 0.97941921, 0.91110664, 0.80091396, 0.70368687,\n",
       "         0.46567083],\n",
       "        [0.83494944, 0.77198684, 0.85442146, 0.83495458, 0.81951978,\n",
       "         0.79808609, 0.98326336, 0.9079418 , 0.80775563, 0.73330752,\n",
       "         0.53797682],\n",
       "        [0.82536719, 0.78118655, 0.84843127, 0.79968064, 0.81182366,\n",
       "         0.75072934, 0.97885224, 0.90100492, 0.80150235, 0.70392385,\n",
       "         0.56594419],\n",
       "        [0.83040575, 0.77908838, 0.85442894, 0.82266029, 0.81292861,\n",
       "         0.77369884, 0.97496516, 0.91052557, 0.80491802, 0.70214278,\n",
       "         0.44561779],\n",
       "        [0.8221794 , 0.79437067, 0.84223673, 0.81306475, 0.80828213,\n",
       "         0.76606229, 0.98092467, 0.90638814, 0.80075448, 0.68477299,\n",
       "         0.51166997],\n",
       "        [0.82259903, 0.77002411, 0.84728937, 0.84274432, 0.80115194,\n",
       "         0.75391651, 0.97925809, 0.90417269, 0.80184831, 0.69455852,\n",
       "         0.4886962 ],\n",
       "        [0.83046842, 0.76181862, 0.86024409, 0.8137311 , 0.80962856,\n",
       "         0.7627001 , 0.98122186, 0.91691595, 0.8045603 , 0.69632031,\n",
       "         0.52799781],\n",
       "        [0.82086947, 0.76268314, 0.84699867, 0.80941211, 0.80102975,\n",
       "         0.76679673, 0.98131019, 0.8959296 , 0.79943736, 0.6823675 ,\n",
       "         0.49501391],\n",
       "        [0.82762055, 0.7801449 , 0.85559866, 0.82913794, 0.80471122,\n",
       "         0.73749884, 0.98032282, 0.91322362, 0.81124394, 0.6791218 ,\n",
       "         0.5518311 ],\n",
       "        [0.82946283, 0.79082028, 0.85655908, 0.82391548, 0.80755275,\n",
       "         0.77141379, 0.98397253, 0.91474619, 0.79960972, 0.69056881,\n",
       "         0.56335121],\n",
       "        [0.82874578, 0.81898277, 0.85341039, 0.8245155 , 0.80779613,\n",
       "         0.77176025, 0.9750495 , 0.91562799, 0.80874852, 0.68802556,\n",
       "         0.47384476],\n",
       "        [0.83400479, 0.79458666, 0.85888578, 0.80056207, 0.8193297 ,\n",
       "         0.77745286, 0.97565754, 0.92115337, 0.80194883, 0.70761643,\n",
       "         0.61138299],\n",
       "        [0.82928885, 0.78911478, 0.84673491, 0.83155416, 0.81593718,\n",
       "         0.77151373, 0.97676938, 0.91694846, 0.8066324 , 0.70123619,\n",
       "         0.4795992 ],\n",
       "        [0.82813608, 0.80058328, 0.86066192, 0.80217706, 0.80342447,\n",
       "         0.77956012, 0.98602259, 0.91695779, 0.80110498, 0.70207741,\n",
       "         0.43578514],\n",
       "        [0.83202055, 0.77943579, 0.8554734 , 0.81197592, 0.82015506,\n",
       "         0.73744354, 0.97890966, 0.90793326, 0.80547248, 0.71349406,\n",
       "         0.46084578],\n",
       "        [0.82391264, 0.83789391, 0.84786433, 0.82194627, 0.8034914 ,\n",
       "         0.75298989, 0.98170853, 0.90588948, 0.80313081, 0.67826894,\n",
       "         0.51072599],\n",
       "        [0.83033246, 0.77229773, 0.85389916, 0.81806633, 0.81339333,\n",
       "         0.76445296, 0.97577604, 0.91564257, 0.80786105, 0.69062371,\n",
       "         0.56028954],\n",
       "        [0.81908522, 0.7865657 , 0.84573512, 0.79159819, 0.80446989,\n",
       "         0.74926942, 0.97596091, 0.89859137, 0.79702499, 0.68838257,\n",
       "         0.49990461],\n",
       "        [0.82271416, 0.77573522, 0.84864439, 0.81280269, 0.8028305 ,\n",
       "         0.78899405, 0.98417444, 0.9134011 , 0.80306716, 0.67208598,\n",
       "         0.52527001],\n",
       "        [0.81986696, 0.73956144, 0.84791522, 0.80918208, 0.80467126,\n",
       "         0.68209217, 0.97353231, 0.91378848, 0.7999562 , 0.68119421,\n",
       "         0.47284313],\n",
       "        [0.83189632, 0.80548083, 0.85230629, 0.8263503 , 0.81464538,\n",
       "         0.78789661, 0.97715721, 0.9068825 , 0.8043036 , 0.71703802,\n",
       "         0.56052871]]))"
      ]
     },
     "execution_count": 126,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "all_claims_jc, all_claims"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 92,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.legend.Legend at 0x17b8e95e0>"
      ]
     },
     "execution_count": 92,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAtAAAAFzCAYAAAAXNz5BAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjUuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/YYfK9AAAACXBIWXMAAAsTAAALEwEAmpwYAAA5OElEQVR4nO3de5yUdd3/8deHXQQUsFxPhZoHFERQlINaeUihqEyz21KzWLtTbw8geZulVp7y58/Un6Vmmj9/6VKWlmlB0XqDZ71LBUUBD7USGtxGuKZgHHSX7++PmV3XdVnmghlmZ/f1fDx4MHPNNd/5zFw7c73nO9/re0VKCUmSJEmF6VXuAiRJkqRKYoCWJEmSMjBAS5IkSRkYoCVJkqQMDNCSJElSBgZoSZIkKYPqcheQ1dZbb5123nnncpchSZKkbm7OnDmvppS2ab+84gL0zjvvzOzZs8tdhiRJkrq5iHipo+UO4ZAkSZIyMEBLkiRJGRigJUmSpAwqbgx0R95++20WL17M6tWry12KVBZ9+/Zlhx12oHfv3uUuRZKkbq9bBOjFixczYMAAdt55ZyKi3OVIm1RKicbGRhYvXswuu+xS7nIkSer2usUQjtWrV1NTU2N4Vo8UEdTU1PgLjCRJm0i3CNCA4Vk9mn//kiRtOt0mQJdbVVUVI0eOZJ999mG//fbjv//7v4va/mWXXZb5PrfeeiuTJk3q8LY//OEPjB49mmHDhrHvvvty9tlnb2yJAJx44onceeedAJx00kk8++yzwHvr//CHP1yUx1vXY7d31VVXMXToUEaOHMmYMWOYOnVqwW21fR79+/ffqBofeOCBd/1t3HjjjeutRZIkdS3dYgx0e5P+8xz+8eprRWtv26234odXX9npOv369WPu3LkA3HPPPZx33nk8+OCDRavhsssu4/zzzy9KW/Pnz2fSpEn8/ve/Z+jQoTQ3N3PTTTcVpe22br755tbL7esv9heMztx4443MnDmTxx9/nIEDB7J8+XLuvvvugu/f9nkUoqmpierqjt9aDzzwAP3792/9AnHqqadmaluSpKwaGxu5+OKLufDCC6mpqSl3Od1CSXugI2JCRLwQEQ0RcW4Ht58YEcsiYm7+30nFeNx/vPoaL253SNH+ZQ3jy5cv5/3vfz+QO8DrnHPOYfjw4YwYMYI77rij0+WvvPIKBx98MCNHjmT48OE8/PDDnHvuuaxatYqRI0dywgknAPCzn/2MsWPHMnLkSP7jP/6D5uZmAG655Rb22GMPxo4dy6OPPtphfVdccQXf+ta3GDp0KJDrPT/ttNMAWLRoEYcddhh77703hx9+OC+//DKQ65E988wz+fCHP8yuu+7a2jubUmLSpEkMGTKEcePG8Y9//KP1cQ499FBmz57dYf0tPbnreh0eeOABDj30UI455hiGDh3KCSecQEoJgEsuuYQxY8YwfPhwTjnllNbl63LZZZdxww03MHDgQAAGDhxIbW1twW21PI8WZ511FnvttReHH344y5Yta13na1/7GqNHj+aaa65h+vTp7L///uy7776MGzeOpUuXsmjRIm688Ua+//3vM3LkSB5++GEuuugirrrqKgDmzp3LAQccwN57783RRx/NP//5z9a2v/nNbzJ27Fj22GMPHn744U6fryRJbdXV1TFv3jx/8SyikgXoiKgCrgc+CQwDjo+IYR2sekdKaWT+X7auvi6kJSAOHTqUk046ie985zsA3HXXXcydO5enn36aWbNmcc455/DKK6+sc/nPf/5zPvGJT7TeNnLkSC6//PLWHu7bbruN5557jjvuuINHH32UuXPnUlVVxW233cYrr7zChRdeyKOPPsojjzzSOuygvfnz5zNq1KgOb5s8eTK1tbU888wznHDCCZx55pmtt73yyis88sgj/O53v+Pcc3Pfh+6++25eeOEFnn32WaZOndphz3L7+tta1+sA8NRTT/GDH/yAZ599loULF7Z+IZg0aRJPPPEE8+fPZ9WqVfzud79b53ZZvnw5K1asYNddd+3w9ixtAfzrX/9i9OjRLFiwgEMOOYSLL7649ba33nqL2bNnc/bZZ/PRj36UP/3pTzz11FMcd9xxXHHFFey8886ceuqpnHXWWcydO5eDDjroXW1PnDiR733vezzzzDOMGDHiXW03NTXx+OOP84Mf/OBdyyVJ6kxjYyP19fWklKivr6exsbHcJXULpeyBHgs0pJQWppTeAm4Hjirh45VVS0B8/vnnqa+vZ+LEiaSUeOSRRzj++OOpqqpiu+2245BDDuGJJ55Y5/IxY8Zwyy23cNFFFzFv3jwGDBjwnse69957mTNnDmPGjGHkyJHce++9LFy4kMcee4xDDz2UbbbZhs0224xjjz028/P44x//yBe/+EUAvvzlL/PII4+03vbZz36WXr16MWzYMJYuXQrAQw891Po8PvjBD3LYYYdlerx1vQ4AY8eOZYcddqBXr16MHDmSRYsWAXD//fez//77M2LECO677z4WLFiQ+Xm2yNpWr169Wl/XL33pS+96fdq+3osXL+YTn/gEI0aM4Morr1xvu2+88Qavv/46hxxyCAC1tbU89NBDrbd/7nOfA2DUqFGtr4MkSetTV1fH2rVrAWhubrYXukhKGaAHAX9rc31xfll7/xYRz0TEnRGxY0cNRcQpETE7Ima3/GTelR144IG8+uqrbEitBx98MA899BCDBg3ixBNP7PAPPaVEbW0tc+fOZe7cubzwwgtcdNFFBT/GXnvtxZw5czLX1qdPn3fVUGptH6+qqoqmpiZWr17N6aefzp133sm8efM4+eSTO52+beDAgfTv35+FCxe+57asbXWk7ewXW2yxRevlyZMnM2nSJObNm8ePf/zjjZ5iruW1aHkdJEkqxKxZs1r3G01NTcycObPMFXUP5Z6FYzqwc0ppb2AmUNfRSimlm1JKo1NKo7fZZptNWuCGeP7552lubqampoaDDjqIO+64g+bmZpYtW8ZDDz3E2LFj17n8pZdeYrvttuPkk0/mpJNO4sknnwSgd+/evP322wAcfvjh3Hnnna3jjV977TVeeukl9t9/fx588EEaGxt5++23+dWvftVhfeeccw6XXXYZf/7znwFYu3YtN954I5CbHeP2228H4LbbbnvPMIP2Dj744Nbn8corr3D//fd3uF7b+tta1+uwLi1BdOutt+bNN99c56wbbZ133nmcccYZLF++HIA333yTqVOnblBba9eubV3v5z//OR/96Ec7XO+NN95g0KDc98W6unf+rAcMGMCKFSves/6WW27J+9///tbxzT/96U9be6MlSdpQ48aNaz2wvbq6mvHjx5e5ou6hlLNwLAHa9ijvkF/WKqXUdiDOzcAVJaynpFrGQEOud7auro6qqiqOPvpo/vjHP7LPPvsQEVxxxRVsv/3261xeV1fHlVdeSe/evenfv39rD/Qpp5zC3nvvzX777cdtt93GpZdeysc//nHWrl1L7969uf766znggAO46KKLOPDAA3nf+97XWk97e++9Nz/4wQ84/vjjWblyJRHBEUccAcB1113HV77yFa688kq22WYbbrnllk6f99FHH819993HsGHD2GmnnTjwwAM7XK99/W3v39Hr8Pzzz3fYzvve9z5OPvlkhg8fzvbbb8+YMWM6rQ/gtNNO480332TMmDH07t2b3r17c/bZZ29QW1tssQWPP/44l156Kdtuu23rQY/tXXTRRXz+85/n/e9/P4cddhh//etfAfjMZz7DMcccw29/+1uuu+66d92nrq6OU089lZUrV7Lrrruu97WXJGl9amtrqa+vB3K/Yk6cOLHMFXUPUaqf4iOiGvgzcDi54PwE8MWU0oI263wgpfRK/vLRwDdTSgd01u7o0aNT2xkRAJ577jn23HPP1uvlmMZOKrf27wNJkgCuvvpqpk+fzpFHHslZZ51V7nIqSkTMSSmNbr+8ZD3QKaWmiJgE3ANUAT9JKS2IiEuA2SmlacCZEXEk0AS8BpxYjMc27EqSJOXU1tayaNEie5+LqKQnUkkpzQBmtFt2QZvL5wHnlbIGSZKknqympoZrr7223GV0K+U+iFCSJEmqKAZoSZIkKQMDtCRJkpSBAVqSJEnKwABdJH//+9857rjj2G233Rg1ahSf+tSnWk9UsjEeeOCB1jmap02bxuWXXw7Ab37zG5599tnW9S644AJmzZq10Y+3rsdu7/HHH+fggw9myJAh7Lvvvpx00kmsXLmyoLbaPo8TTzyxoBOYrMvrr7/Oj370o9br//M//8Mxxxyzwe1JkiStT0ln4SiX88+exBuvLi1ae1tuvR2X/Z8frvP2lBJHH300tbW1rWfxe/rpp1m6dCl77LFH0eo48sgjOfLII4FcgD7iiCMYNmwYAJdccknRHmd9li5dyuc//3luv/321hOn3HnnnaxYsYLNN998vfdv+zwK0dTU1HoWpfZaAvTpp58OwAc/+MGNCuSSJEnr0y0D9BuvLuWbu3V8JrsN8b0XO7/9/vvvp3fv3px66qmty/bZZx8gF66/8Y1v8Ic//IGI4Nvf/jbHHnssDzzwABdddBFbb7018+fPZ9SoUfzsZz8jIqivr+drX/sam2+++btOFX3rrbcye/ZsvvjFLzJt2jQefPBBLr30Un7961/z3e9+lyOOOIJjjjmGe++9l69//es0NTUxZswYbrjhBvr06cPOO+9MbW0t06dPbz3V99ChQ3n88ceZMmUKq1evpl+/ftxyyy0MGTJknc/3+uuvp7a29l1nHWzp9S2krZbn8cMf5r6UzJo1i8svv5zly5dz9dVXc8QRR3Drrbdy11138eabb9Lc3Mzvf/97jjrqKP75z3/y9ttvc+mll3LUUUdx7rnn8uKLLzJy5EjGjx/PGWecwRFHHMH8+fNZvXo1p512GrNnz6a6upqrr76aj33sY9x6661MmzaNlStX8uKLL3L00UdzxRUVexJMSZK0iXXLAL2ptQTgjtx1113MnTuXp59+mldffZUxY8Zw8MEHA/DUU0+xYMECPvjBD/KRj3yERx99lNGjR3PyySdz3333MXjwYI499tj3tPnhD3+YI488sjUwt7V69WpOPPFE7r33XvbYYw8mTpzIDTfcwNe+9jUAtt56a5588kl+9KMfcdVVV3HzzTczdOhQHn74Yaqrq5k1axbnn38+v/71rzt9vrW1tR3elrUtgEWLFvH444/z4osv8rGPfYyGhgYAnnzySZ555hm22mormpqauPvuuxk4cCCvvvoqBxxwAEceeSSXX3458+fPZ+7cua1ttbj++uuJCObNm8fzzz/Pxz/+8dZhNXPnzuWpp56iT58+DBkyhMmTJ7Pjjju2L02SJOk9DNAl9sgjj3D88cdTVVXFdtttxyGHHMITTzzBwIEDGTt2LDvssAMAI0eOZNGiRfTv359ddtmF3XffHYAvfelL3HTTTQU/3gsvvMAuu+zSOnSktraW66+/vjVAf+5znwNg1KhR3HXXXQC88cYb1NbW8pe//IWI4O23397g57shbX3hC1+gV69e7L777uy66648/3zu14Px48ez1VZbAbme/PPPP5+HHnqIXr16sWTJEpYu7XyYziOPPMLkyZOBXLD/0Ic+1BqgDz/8cLbccksAhg0bxksvvWSAliRJBfEgwiLYa6+9mDNnTub79enTp/VyVVUVTU1NxSyr08ds+3jf+c53+NjHPsb8+fOZPn06q1ev7rSNzp5v1rYAIqLD61tssUXrsttuu41ly5YxZ84c5s6dy3bbbVdQ2+tSjtdekiR1DwboIjjssMNYs2bNu3qKn3nmGR5++GEOOugg7rjjDpqbm1m2bBkPPfQQY8eOXWdbQ4cOZdGiRbz4Ym7g9S9+8YsO1xswYAArVqx4z/IhQ4awaNGi1mEQP/3pTznkkEM6rf+NN95g0KBBQG588vpMmjSJuro6HnvssdZld911F0uXLs3cFsCvfvUr1q5dy4svvsjChQs7HH/9xhtvsO22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\n",
      "text/plain": [
       "<Figure size 864x432 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Create a DataFrame for each method\n",
    "names = [\"Marginal\"] + continent_names.tolist() + freq_names.tolist()\n",
    "indices_to_plot = [0, -5, -4, -3, -2, -1]\n",
    "\n",
    "names = [names[i] for i in indices_to_plot]\n",
    "df1 = pd.DataFrame(1 - all_covs_jc[:,indices_to_plot], columns=names)\n",
    "df2 = pd.DataFrame(1 - all_covs[:,indices_to_plot], columns=names)\n",
    "\n",
    "# Add a column to identify the method\n",
    "df1['Method'] = 'Boosted Conditional Calibration'\n",
    "df2['Method'] = 'Conditional Calibration'\n",
    "\n",
    "# Combine the DataFrames\n",
    "df = pd.concat([df1, df2])\n",
    "\n",
    "# Melt the combined DataFrame\n",
    "df_melted = df.melt(id_vars='Method', var_name='Group', value_name='Miscoverage')\n",
    "\n",
    "# Create the boxplot\n",
    "plt.figure(figsize=(12, 6))\n",
    "sns.boxplot(x='Group', y='Miscoverage', hue='Method', data=df_melted)\n",
    "plt.axhline(0.1, color='red')\n",
    "# Customize the plot\n",
    "plt.legend()"
   ]
  }
 ],
 "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": 5
}
