{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "30aae7c9",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "2024-08-05 00:02:53.013916: E external/local_xla/xla/stream_executor/cuda/cuda_dnn.cc:9373] Unable to register cuDNN factory: Attempting to register factory for plugin cuDNN when one has already been registered\n",
      "2024-08-05 00:02:53.013988: E external/local_xla/xla/stream_executor/cuda/cuda_fft.cc:607] Unable to register cuFFT factory: Attempting to register factory for plugin cuFFT when one has already been registered\n",
      "2024-08-05 00:02:53.015875: E external/local_xla/xla/stream_executor/cuda/cuda_blas.cc:1534] Unable to register cuBLAS factory: Attempting to register factory for plugin cuBLAS when one has already been registered\n",
      "2024-08-05 00:02:53.024621: I tensorflow/core/platform/cpu_feature_guard.cc:183] This TensorFlow binary is optimized to use available CPU instructions in performance-critical operations.\n",
      "To enable the following instructions: SSE3 SSE4.1 SSE4.2 AVX, in other operations, rebuild TensorFlow with the appropriate compiler flags.\n"
     ]
    }
   ],
   "source": [
    "import os\n",
    "import time\n",
    "import numpy as np\n",
    "import tensorflow as tf\n",
    "import matplotlib.pyplot as plt\n",
    "import seaborn as sns\n",
    "\n",
    "from src.datasets import load_dataset, preprocess_dataset, prefetch_dataset\n",
    "import src.utils as utils"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "id": "7e8a1c5d",
   "metadata": {},
   "outputs": [],
   "source": [
    "cfg = { 'dataset' : 'fashion_mnist',\n",
    "       'model' : 'cnn',\n",
    "       'batch_size' : 512,\n",
    "       }\n",
    "\n",
    "model_name = cfg['model']\n",
    "dataset_name = cfg['dataset']\n",
    "\n",
    "ds_train, ds_val, ds_test, ds_info = load_dataset(cfg)\n",
    "n_classes = ds_info.features['label'].num_classes\n",
    "ds_train = preprocess_dataset(ds_train, cfg, n_classes, resize=False, normalize=True, onehot=True)\n",
    "ds_val = preprocess_dataset(ds_val, cfg, n_classes, resize=False, normalize=True, onehot=True)\n",
    "ds_test = preprocess_dataset(ds_test, cfg, n_classes, resize=False, normalize=True, onehot=True)\n",
    "\n",
    "true_y_val = np.argmax([y for x,y in ds_val], axis=1)\n",
    "true_y_test = np.argmax([y for x,y in ds_test], axis=1)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "117a4b7c",
   "metadata": {},
   "source": [
    "### Softmax"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "id": "386d73ac",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run: 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "2024-08-05 00:41:39.441010: I external/local_xla/xla/stream_executor/cuda/cuda_dnn.cc:467] Loaded cuDNN version 90100\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run: 2\n",
      "Run: 3\n",
      "Run: 4\n",
      "Run: 5\n"
     ]
    }
   ],
   "source": [
    "n_examples = [int(i/1000*10000) for i in range(0,11,2)] + [int(i/100*10000) for i in range(2,10,2)] + [int(i/10*10000) for i in range(1,10,2)]\n",
    "\n",
    "for run in range(5):\n",
    "    print('Run:',run+1)\n",
    "    tf.keras.utils.set_random_seed(run+10) # set random seed for Python, NumPy, and TensorFlow\n",
    "    \n",
    "    exp_name = f'../results/PI_Explainability/{model_name}_{dataset_name}/run_{run+1}/calibration'\n",
    "    model = tf.keras.models.load_model(f'../results/PI_Explainability/{model_name}_{dataset_name}/run_{run+1}/saved_models/trained_model.keras')\n",
    "    pred_y_val = np.argmax(model.predict(ds_val.batch(cfg['batch_size']), verbose=0), axis=1)\n",
    "    true_label_val = np.equal(true_y_val, pred_y_val).astype(int)\n",
    "    pred_y_test = np.argmax(model.predict(ds_test.batch(cfg['batch_size']), verbose=0), axis=1)\n",
    "    true_label_test = np.equal(true_y_test, pred_y_test).astype(int)\n",
    "    \n",
    "    logits_val = model.predict(ds_val.batch(512), verbose=0)\n",
    "    opt_temp = utils.temp_scaling_2(logits_val, pred_y_val, true_label_val)\n",
    "    np.save(f'{exp_name}/softmax_opt_temp.npy', opt_temp)\n",
    "    softmax_test = np.max(tf.nn.softmax(model.predict(ds_test.batch(512), verbose=0)/opt_temp), axis=1)\n",
    "\n",
    "    softmax_error_list = []\n",
    "    for n in n_examples:\n",
    "        err = 1 - utils.compute_selective_pred_acc(softmax_test, ds_test, model, n)\n",
    "        softmax_error_list.append(err)\n",
    "    softmax_error = np.array(softmax_error_list)\n",
    "    np.save(f'{exp_name}/softmax_selective_pred.npy', softmax_error)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "109ea0c6",
   "metadata": {},
   "source": [
    "### PMI"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "id": "2e612c1d",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run: 1\n",
      "Run: 2\n",
      "Run: 3\n",
      "Run: 4\n",
      "Run: 5\n"
     ]
    }
   ],
   "source": [
    "estimator = 'separable_variational_f_js'\n",
    "n_examples = [int(i/1000*10000) for i in range(0,11,2)] + [int(i/100*10000) for i in range(2,10,2)] + [int(i/10*10000) for i in range(1,10,2)]\n",
    "\n",
    "for run in range(5):\n",
    "    print('Run:',run+1)\n",
    "    tf.keras.utils.set_random_seed(run+10) # set random seed for Python, NumPy, and TensorFlow\n",
    "    \n",
    "    exp_name = f'../results/PI_Explainability/{model_name}_{dataset_name}/run_{run+1}/calibration/pmi/{estimator}'\n",
    "    model = tf.keras.models.load_model(f'../results/PI_Explainability/{model_name}_{dataset_name}/run_{run+1}/saved_models/trained_model.keras')\n",
    "    pred_y_val = np.argmax(model.predict(ds_val.batch(cfg['batch_size']), verbose=0), axis=1)\n",
    "    true_label_val = np.equal(true_y_val, pred_y_val).astype(int)\n",
    "    pred_y_test = np.argmax(model.predict(ds_test.batch(cfg['batch_size']), verbose=0), axis=1)\n",
    "    true_label_test = np.equal(true_y_test, pred_y_test).astype(int)\n",
    "    \n",
    "    pmi_class_val = np.load(f'{exp_name}/pmi_output_class_val.npy')\n",
    "    opt_temp = utils.temp_scaling_2(pmi_class_val, pred_y_val, true_label_val)\n",
    "    np.save(f'{exp_name}/pmi_opt_temp.npy', opt_temp)\n",
    "    pmi_class_test = np.load(f'{exp_name}/pmi_output_class_test.npy')\n",
    "    pmi_class = np.array([utils.softmax(x/opt_temp) for x in pmi_class_test])\n",
    "    pmi_test = np.array([pmi_value[pred_value] for pmi_value, pred_value in zip(pmi_class, pred_y_test)])\n",
    "\n",
    "    pmi_error_list = []\n",
    "    for n in n_examples:\n",
    "        err = 1 - utils.compute_selective_pred_acc(pmi_test, ds_test, model, n)\n",
    "        pmi_error_list.append(err)\n",
    "    pmi_error = np.array(pmi_error_list)\n",
    "    np.save(f'{exp_name}/pmi_selective_pred.npy', pmi_error)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "66c64437",
   "metadata": {},
   "source": [
    "### PSI"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "id": "ca89ce65",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run: 1\n",
      "Run: 2\n",
      "Run: 3\n",
      "Run: 4\n",
      "Run: 5\n"
     ]
    }
   ],
   "source": [
    "estimator = 'gaussian'\n",
    "n_projs = 500\n",
    "\n",
    "n_examples = [int(i/1000*10000) for i in range(0,11,2)] + [int(i/100*10000) for i in range(2,10,2)] + [int(i/10*10000) for i in range(1,10,2)]\n",
    "\n",
    "for run in range(5):\n",
    "    tf.keras.utils.set_random_seed(run+10) # set random seed for Python, NumPy, and TensorFlow\n",
    "    print(f'Run: {run+1}')\n",
    "    \n",
    "    exp_name = f'../results/PI_Explainability/{model_name}_{dataset_name}/run_{run+1}/calibration/psi/{estimator}'\n",
    "    model = tf.keras.models.load_model(f'../results/PI_Explainability/{model_name}_{dataset_name}/run_{run+1}/saved_models/trained_model.keras')\n",
    "    \n",
    "    pred_y_val = np.argmax(model.predict(ds_val.batch(cfg['batch_size']), verbose=0), axis=1)\n",
    "    true_label_val = np.equal(true_y_val, pred_y_val).astype(int)\n",
    "    pred_y_test = np.argmax(model.predict(ds_test.batch(cfg['batch_size']), verbose=0), axis=1)\n",
    "    true_label_test = np.equal(true_y_test, pred_y_test).astype(int)\n",
    "    \n",
    "    psi_class_val = np.load(f'{exp_name}/psi_output_class_{n_projs}_projs_val.npy')\n",
    "    opt_temp = utils.temp_scaling_2(psi_class_val, pred_y_val, true_label_val)\n",
    "    np.save(f'{exp_name}/psi_opt_temp.npy', opt_temp)\n",
    "    psi_class_test = np.load(f'{exp_name}/psi_output_class_{n_projs}_projs_test.npy')\n",
    "    psi_class = np.array([utils.softmax(x/opt_temp) for x in psi_class_test])\n",
    "    psi_test = np.array([psi_value[pred_value] for psi_value, pred_value in zip(psi_class, pred_y_test)])\n",
    "\n",
    "    psi_error_list = []\n",
    "    for n in n_examples:\n",
    "        err = 1 - utils.compute_selective_pred_acc(psi_test, ds_test, model, n)\n",
    "        psi_error_list.append(err) \n",
    "    psi_error = np.array(psi_error_list)\n",
    "    np.save(f'{exp_name}/psi_selective_pred.npy', psi_error)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "9b6082d3",
   "metadata": {},
   "source": [
    "### PVI"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "id": "cc1b6545",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run: 1\n",
      "Run: 2\n",
      "Run: 3\n",
      "Run: 4\n",
      "Run: 5\n"
     ]
    }
   ],
   "source": [
    "estimator = 'no_training'\n",
    "\n",
    "n_examples = [int(i/1000*10000) for i in range(0,11,2)] + [int(i/100*10000) for i in range(2,10,2)] + [int(i/10*10000) for i in range(1,10,2)]\n",
    "\n",
    "for run in range(5):\n",
    "    tf.keras.utils.set_random_seed(run+10) # set random seed for Python, NumPy, and TensorFlow\n",
    "    print(f'Run: {run+1}')\n",
    "    \n",
    "    exp_name = f'../results/PI_Explainability/{model_name}_{dataset_name}/run_{run+1}/calibration/pvi/{estimator}'\n",
    "    model = tf.keras.models.load_model(f'../results/PI_Explainability/{model_name}_{dataset_name}/run_{run+1}/saved_models/trained_model.keras')\n",
    "    pred_y_val = np.argmax(model.predict(ds_val.batch(cfg['batch_size']), verbose=0), axis=1)\n",
    "    true_label_val = np.equal(true_y_val, pred_y_val).astype(int)\n",
    "    pred_y_test = np.argmax(model.predict(ds_test.batch(cfg['batch_size']), verbose=0), axis=1)\n",
    "    true_label_test = np.equal(true_y_test, pred_y_test).astype(int)\n",
    "    \n",
    "    pvi_class_val = np.load(f'{exp_name}/pvi_class_val.npy')\n",
    "    opt_temp = utils.temp_scaling_2(pvi_class_val, pred_y_val, true_label_val)\n",
    "    np.save(f'{exp_name}/pvi_opt_temp.npy', opt_temp)\n",
    "    pvi_class_test = np.load(f'{exp_name}/pvi_class_test.npy')\n",
    "    pvi_class = np.array([utils.softmax(x/opt_temp) for x in pvi_class_test])\n",
    "    pvi_test = np.array([pvi_value[pred_value] for pvi_value, pred_value in zip(pvi_class, pred_y_test)])\n",
    "\n",
    "    pvi_error_list = []\n",
    "    for n in n_examples:\n",
    "        err = 1 - utils.compute_selective_pred_acc(pvi_test, ds_test, model, n)\n",
    "        pvi_error_list.append(err)\n",
    "    pvi_error = np.array(pvi_error_list)\n",
    "    np.save(f'{exp_name}/pvi_selective_pred.npy', pvi_error)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "45948358",
   "metadata": {},
   "source": [
    "### Plot"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "id": "81973c62",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run: 1\n",
      "Run: 2\n",
      "Run: 3\n",
      "Run: 4\n",
      "Run: 5\n"
     ]
    }
   ],
   "source": [
    "all_softmax = []\n",
    "all_pmi = []\n",
    "all_pvi = []\n",
    "all_psi = []\n",
    "\n",
    "for run in range(5):\n",
    "    print(f'Run: {run+1}')\n",
    "    exp_name = f'../results/PI_Explainability/{model_name}_{dataset_name}/run_{run+1}/calibration'\n",
    "    softmax = np.load(f'{exp_name}/softmax_selective_pred.npy')\n",
    "    all_softmax.append(softmax)\n",
    "    \n",
    "    estimator = 'separable_variational_f_js'\n",
    "    exp_name = f'../results/PI_Explainability/{model_name}_{dataset_name}/run_{run+1}/calibration/pmi/{estimator}'\n",
    "    pmi = np.load(f'{exp_name}/pmi_selective_pred.npy')\n",
    "    all_pmi.append(pmi)\n",
    "    \n",
    "    estimator = 'gaussian'\n",
    "    exp_name = f'../results/PI_Explainability/{model_name}_{dataset_name}/run_{run+1}/calibration/psi/{estimator}'\n",
    "    psi = np.load(f'{exp_name}/psi_selective_pred.npy')\n",
    "    all_psi.append(psi)\n",
    "    \n",
    "    estimator = 'no_training'\n",
    "    exp_name = f'../results/PI_Explainability/{model_name}_{dataset_name}/run_{run+1}/calibration/pvi/{estimator}'\n",
    "    pvi = np.load(f'{exp_name}/pvi_selective_pred.npy')\n",
    "    all_pvi.append(pvi)\n",
    "    \n",
    "softmax_val = np.mean(np.array(all_softmax), axis=0)\n",
    "pmi = np.mean(np.array(all_pmi), axis=0)\n",
    "pvi = np.mean(np.array(all_pvi), axis=0)\n",
    "psi = np.mean(np.array(all_psi), axis=0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "id": "5d56aea9",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAYcAAAEXCAYAAABGeIg9AAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjkuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/TGe4hAAAACXBIWXMAAA9hAAAPYQGoP6dpAABhiklEQVR4nO3dd3wURf/A8c9ev0snlUBC6L0JgqGDdARBpSpdrIDKIyhYKBYQFEHlAUEpPx8RBQVpAhFBunSV3gkQEkhIr1fm90fISbwEkpDkQjLv1+te5GZnd79zF/ab3dmdUYQQAkmSJEm6g8rZAUiSJEklj0wOkiRJkgOZHCRJkiQHMjlIkiRJDmRykCRJkhzI5CBJkiQ5kMlBkiRJciCTgyRJkuRAJgdJkiTJgUwOklTEpkyZgqIoREdH37NuSEgIw4YNK/qgJOkeZHKQ7tv58+d5/vnnqVKlCgaDAXd3d1q2bMncuXNJTU211wsJCUFRFMaMGeOwje3bt6MoCqtWrbKXLV26FEVRMBgMXLt2zWGddu3aUa9evQLFnLW/nF4DBgwo0DYfZFmftaIo7Nq1y2G5EIKgoCAUReGxxx7LtixrvU8++STX7R48eNBelluyXLduHW3btsXPzw+TyUSVKlXo168fmzZtAjK/79y+sztfU6ZMKYRPRNI4OwDpwbZhwwb69u2LXq9nyJAh1KtXj4yMDHbt2sX48eM5fvw4CxcuzLbOokWLmDhxIoGBgXnaR3p6OjNmzODzzz8v9PjHjh3Lww8/nK0sJCSk0PeTV6dPn0alct7fbAaDgeXLl9OqVats5b///jtXr15Fr9fnuu6sWbN48cUXMZlM+d7vxx9/zPjx42nbti0TJ07EZDJx7tw5fv31V1asWEHXrl156623ePbZZ+3rHDhwgM8++4xJkyZRu3Zte3mDBg3yvX/JkUwOUoFdvHiRAQMGUKlSJX777TfKly9vX/byyy9z7tw5NmzYkG2dunXrcvr0aWbMmMFnn32Wp/00atQo3wklr1q3bs1TTz1VqNu8H3c7+BaH7t27s3LlSj777DM0mn8OD8uXL6dJkya5Xhpr1KgRR48eZcGCBYwbNy5f+7RYLLz33nt06tSJLVu2OCy/ceMGAJ06dcpWbjAY+Oyzz+jUqRPt2rXL1z6le5OXlaQCmzlzJklJSXz99dfZEkOWatWq8corr2QrCwkJYciQISxatIiIiIg87WfSpElYrVZmzJhRKHHnxa1bt3j99depX78+rq6uuLu7061bN/7880+Hup9//jl169bFZDLh5eVF06ZNWb58uUO9uLg4hg0bhqenJx4eHgwfPpyUlJRsdXLqc7hw4QJ9+/alXLlymEwmHnnkEYekm3WZ7IcffuCDDz6gYsWKGAwGHn30Uc6dO5fndg8cOJCYmBjCwsLsZRkZGaxatYpBgwblul7Lli3p0KEDM2fOzHYpMS+io6NJSEigZcuWOS738/PL1/akwiGTg1Rg69ato0qVKrRo0SJf67311ltYLJY8H+wrV66c74SSV4mJiURHR2d72Ww2Lly4wJo1a3jssceYPXs248eP5++//6Zt27bZYli0aBFjx46lTp06zJkzh6lTp9KoUSP++OMPh33169ePxMREpk+fTr9+/Vi6dClTp069a3xRUVG0aNGCzZs389JLL/HBBx+QlpZGr169WL16tUP9GTNmsHr1al5//XUmTpzIvn37ePrpp/P8eYSEhBAaGsp3331nL/vll1+Ij4+/Z1/MlClTiIqKYv78+XneH2Qe/I1GI+vWrePWrVv5WlcqQkKSCiA+Pl4A4vHHH8/zOpUqVRI9evQQQggxfPhwYTAYREREhBBCiG3btglArFy50l5/yZIlAhAHDhwQ58+fFxqNRowdO9a+vG3btqJu3boFij9rfzm9Ll68KNLS0oTVas22zsWLF4VerxfTpk2zlz3++OP3jGHy5MkCECNGjMhW3qdPH+Ht7Z2trFKlSmLo0KH296+++qoAxM6dO+1liYmJonLlyiIkJMQeY1Z7ateuLdLT0+11586dKwDx999/3zXGOz/rL774Qri5uYmUlBQhhBB9+/YV7du3t8eX9R1mAcTLL78shBCiffv2IiAgwL7undv99+dx8+ZNe9m7774rAOHi4iK6desmPvjgA3Ho0KG7xrxy5UoBiG3btt21nlQw8sxBKpCEhAQA3NzcCrT+22+/na+zhypVqjB48GAWLlzI9evXC7TPnLz77ruEhYVlewUEBKDX6+0dw1arlZiYGFxdXalZsyaHDx+2r+/p6cnVq1c5cODAPff1wgsvZHvfunVrYmJi7J9lTjZu3EizZs2ydRC7urry3HPPcenSJU6cOJGt/vDhw9HpdNn2AZmXpvKqX79+pKamsn79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VKlSRSxZsuS+45BDdkuSJEkOZJ+DJEmS5EAmB0mSJMmBTA6SJEmSA5kcJEmSJAcyOUiSJEkOZHKQJEmSHMjkIEmSJDmQyUEq89auXUvnzp0pV64cOp2OypUr8/zzz3PmzBlnhyZJTiMfgpPKtDfffJOPPvqIp556igEDBuDr68v58+dZvHgxSUlJHDlyxNkhSpJTyOQglVkbN26kR48evPPOO0ybNs1h+fr163nsscecEFl29zMmvyQVlLysJJVZn3zyCf7+/rzzzjs5Ls9KDGlpaYwbN47AwEAMBgONGjVi9erV9npLly5Fo9E4DJt869YtdDodX375pb1s7969dOjQARcXFzw8PBg0aBA3btywL7906RKKorB06VJGjRqFt7c3zZo1A2DDhg106tQJPz8/3N3dad68OZs2bXKIe9euXTRu3BiDwUCDBg0ICwujUaNGDBs2LFu9e8UilW0yOUhlksViYffu3Tz66KNotdq71n366af58ssvmTBhAmvWrKFOnTo8+eSTrF27FoA+ffqg0WhYuXJltvV+/PFHAPr27QtkHozbtWuHh4cH33//PQsXLuTAgQM8/vjjDvucOHEiQgi+++47Zs2aBcDFixfp2bMn33zzDT/++CMtW7ake/fubN++3b7e9evX6dq1K25ubvzwww+MHz+eF198kWvXrmXbfn5ikcqo+x66T5IeQJGRkQK46zDfQgjx559/CkAsWLAgW3loaKh46KGH7O/79OkjWrRoka1O+/btRY8ePezv27RpI1q0aCFsNpu97Pjx40JRFLFhwwYhhBAXL14UgOjatetd47JarcJsNovOnTuLgQMH2svHjx8vPDw8REJCgr1s586d2Ub1zGssUtkmzxykMk1RlLsu37lzJ/DPX/9Z+vfvz5EjR0hOTgZg4MCB7N27l/DwcCDzL/jff/+dgQMHApkT0ezevZu+fftitVqxWCxYLBZq1KhBUFAQBw4cyLb9Hj16OMRy9epVhg4dSoUKFdBoNGi1WrZs2ZLtrqoDBw7Qvn173Nzc7GWtWrWiXLly9vf5jUUqm2RykMokb29vDAaD/WCem9jYWLRabbaDK2RO4C6EIC4uDsjsn3BxcWHFihUA/PDDDxgMBnr37m3fjtVq5bXXXkOr1WZ7hYeHc+XKFYft38lms9GrVy927drFtGnT2LZtGwcOHKBbt26kpaXZ612/fh1fX1+Hdvj5+WVrU35ikcomOYe0VCZpNBpatmzJ1q1bsVgsaDQ5/1coV64cZrOZ2NhYvLy87OVRUVEoioKnpycARqOR3r17s2LFCiZMmMCKFSvo2bMnLi4uAHh6eqIoCpMmTbInjDv5+Phke//vM5pz585x5MgR1qxZk61fIDU1NVu98uXLc/PmTYft39nRnN9YpLJJnjlIZda4ceOIjIzkgw8+yHH5xo0badWqFYBDZ/PKlStp3Lix/eAPmZeWjhw5wubNm9m3b5/9khKAi4sLoaGhnDx5kqZNmzq8QkJC7hprVhK4cw7oy5cvs3v37mz1Hn74YX777TcSExPtZTt37uTWrVuFFotURji700OSnGnChAkCEP369RM//fST2LFjh1i2bJlo27ataNSokRBCiCeeeEK4uLiIOXPmiF9++UU8/fTTQlEUsXbt2mzbysjIEN7e3iIwMFB4enqK9PT0bMt3794t9Hq9fV/btm0T33zzjRgyZIjYtm2bEOKfDumVK1dmWzctLU1UrFhRNGjQQKxbt0589913okaNGiIkJETUrVvXXi8iIkK4uLiI1q1bi7Vr14r/+7//E1WrVhU+Pj5i+PDh+YpFKttkcpDKvDVr1oiOHTsKT09PodVqRUhIiHj++efF2bNnhRBCpKSkiFdffVUEBAQInU4nGjRoIH788ccct/X8888LQIwcOTLH5QcOHBDdu3cXHh4ewmg0iurVq4sXXnhBXLlyRQiRe3IQQoj9+/eLhx9+WBgMBlG9enWxbNkyMXTo0GzJQQghduzYYZ9PuHbt2mL9+vUiJCREvPrqq/mKRSrb5BPSklTKnT17llq1arF48WKGDh3q7HCkB4RMDpJUykycOJEGDRoQGBjIhQsX+PDDD0lNTeXUqVO4uro6OzzpASHvVpKkUiYjI4M33niDqKgojEYj7dq1Y9asWTIxSPkizxwkSZIkB/JWVkmSJMmBTA6SJEmSA5kcJEmSJAcyOUiSJEkOZHKQJEmSHMjkIEmSJDmQyUGSJElyIJODJEmS5EAmB0mSJMnB/wNrIxILPLLAmgAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 400x250 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.figure(figsize=(4,2.5),dpi=100)\n",
    "cp = sns.color_palette('colorblind', as_cmap=True)\n",
    "length_test = 10000\n",
    "n_examples = [int(i/1000*length_test) for i in range(0,11,2)] + [int(i/100*length_test) for i in range(2,10,2)] + [int(i/10*length_test) for i in range(1,10,2)]\n",
    "n_examples = [1-i/length_test for i in n_examples]\n",
    "plt.plot(n_examples,softmax_val, marker='s', label='softmax', c=cp[0], ms=4)\n",
    "plt.plot(n_examples,pmi, marker='o', label='PMI', c=cp[1], ms=4)\n",
    "plt.plot(n_examples,pvi, marker='s', label='PVI', c=cp[2], ms=5,  markerfacecolor=\"None\")\n",
    "plt.plot(n_examples,psi, marker='o', label='PSI', c=cp[3], ms=5,  markerfacecolor=\"None\")\n",
    "plt.grid(True)\n",
    "# plt.xscale('log')\n",
    "# plt.xticks([20,100,1000,9000],[20,100,1000,9000])\n",
    "plt.xlabel('Coverage', fontsize=11)\n",
    "plt.ylabel('Risk', fontsize=11)\n",
    "plt.title(f'CNN, Fashion MNIST', fontsize=12)\n",
    "plt.legend(loc='upper left')\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "1bb3f02e",
   "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.10.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
