{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "60f5c649",
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "import os.path as osp\n",
    "import numpy as np\n",
    "import torch\n",
    "import matplotlib.pyplot as plt\n",
    "import seaborn as sns\n",
    "from class_separator import *\n",
    "from ccns import *\n",
    "from torch_geometric.nn import knn_graph"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "154176a2",
   "metadata": {},
   "source": [
    "## First, check if all pairwise deltaSED (- (SED_H - SED_X)) are positive => SED_X > SED_H. \n",
    "### This could be an indication that it might never be possible to have SED_H > SED_X"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "8803a2e3",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "{'best_epoch': 653, 'best_sed_tuples': ([(0, 1, tensor(0.0003, dtype=torch.float64), tensor(0.0003, dtype=torch.float64))],), 'best_deltaSED': -4.336808689942018e-18, 'best_lb_ccns': None, 'C': 2, 'M': 10, 'D': 2, 'lambda': 0.0, 'seed': 42} tensor([False])\n",
      "\n",
      "{'best_epoch': 1144, 'best_sed_tuples': ([(0, 1, tensor(9.2839e-05, dtype=torch.float64), tensor(9.2839e-05, dtype=torch.float64)), (0, 2, tensor(0.0004, dtype=torch.float64), tensor(0.0004, dtype=torch.float64)), (0, 3, tensor(6.6951e-05, dtype=torch.float64), tensor(6.6951e-05, dtype=torch.float64)), (0, 4, tensor(0.0002, dtype=torch.float64), tensor(0.0002, dtype=torch.float64)), (1, 2, tensor(0.0006, dtype=torch.float64), tensor(0.0006, dtype=torch.float64)), (1, 3, tensor(0.0001, dtype=torch.float64), tensor(0.0001, dtype=torch.float64)), (1, 4, tensor(0.0002, dtype=torch.float64), tensor(0.0002, dtype=torch.float64)), (2, 3, tensor(0.0005, dtype=torch.float64), tensor(0.0005, dtype=torch.float64)), (2, 4, tensor(0.0007, dtype=torch.float64), tensor(0.0007, dtype=torch.float64)), (3, 4, tensor(0.0002, dtype=torch.float64), tensor(0.0002, dtype=torch.float64))],), 'best_deltaSED': -2.4936649967166602e-18, 'best_lb_ccns': None, 'C': 5, 'M': 2, 'D': 2, 'lambda': 0.0, 'seed': 42} tensor([False,  True,  True, False,  True, False,  True, False,  True, False])\n",
      "\n"
     ]
    }
   ],
   "source": [
    "seed = 42\n",
    "for num_features in [1, 2, 5, 10]:\n",
    "    for num_classes in [2, 5]:\n",
    "        for num_mixtures in [2, 5, 10]:\n",
    "            for lambda1 in [0., 1., 5., 10.]:\n",
    "                \n",
    "                model_ckpt_pth = f\"C_{num_classes}_M_{num_mixtures}_\" \\\n",
    "                                 f\"D_{num_features}_\" \\\n",
    "                                 f\"lambda_{lambda1}_\" \\\n",
    "                                 f\"seed_{seed}.pth\"\n",
    "                results_ckpt_pth = f\"C_{num_classes}_M_{num_mixtures}_\" \\\n",
    "                                 f\"D_{num_features}_\" \\\n",
    "                                 f\"lambda_{lambda1}_\" \\\n",
    "                                 f\"seed_{seed}.pt\"\n",
    "                \n",
    "                res = torch.load(osp.join('RESULTS', results_ckpt_pth))\n",
    "                list_bool = torch.tensor([s[2] <= s[3] for s in res['best_sed_tuples'][0]])\n",
    "                bool_test = torch.all(list_bool)  # sed_h = s[2], sed_x = s[3]\n",
    "                if not bool_test:\n",
    "                    print(res, list_bool)\n",
    "                    print()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "a7aec70e",
   "metadata": {},
   "source": [
    "#### 2 out of 96 show a SED_X < SED_H, but the difference is too small that it might be due to numerical errors. In fact, out of 10k iterations the best epoch selected for thes few cases is relatively at the beginning of the training, and it never improves later. Plus these configs use lambda = 0., so it is likely that they have found a degenerate case (we can check). In any case, the delta is not enough to say that we have found a good distribution."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 79,
   "id": "e04a19e5",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 400x200 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "seed = 42\n",
    "\n",
    "plt.figure(figsize=(4,2))\n",
    "for num_features in [1, 2, 5, 10, 50]:\n",
    "    for num_classes in [2]: # only use 2 for now\n",
    "        for num_mixtures in [2, 5, 10]:\n",
    "            for lambda1 in [0., 1., 5., 10.]:\n",
    "                \n",
    "                model_ckpt_pth = f\"C_{num_classes}_M_{num_mixtures}_\" \\\n",
    "                                 f\"D_{num_features}_\" \\\n",
    "                                 f\"lambda_{lambda1}_\" \\\n",
    "                                 f\"seed_{seed}.pth\"\n",
    "                results_ckpt_pth = f\"C_{num_classes}_M_{num_mixtures}_\" \\\n",
    "                                 f\"D_{num_features}_\" \\\n",
    "                                 f\"lambda_{lambda1}_\" \\\n",
    "                                 f\"seed_{seed}.pt\"\n",
    "                \n",
    "                # k, sigma, epsilon are optimized by the method\n",
    "                model = ClassSeparator(\n",
    "                    num_classes=num_classes,\n",
    "                    num_mixtures=num_mixtures,\n",
    "                    num_features=num_features,\n",
    "                    use_full_covariance=(lambda1==0. and num_features>1)\n",
    "                )\n",
    "                \n",
    "                model.load_state_dict(torch.load(osp.join('checkpoints', model_ckpt_pth)))\n",
    "                model.eval()\n",
    "                               \n",
    "                sed_x = model.compute_SED_X(0,1)\n",
    "                \n",
    "                list_delta_sed = []\n",
    "                \n",
    "                for k in range(1, 501):\n",
    "                    model.k.data = torch.tensor([k], dtype=torch.float)\n",
    "                    sed_h = model.compute_SED_H(0,1)\n",
    "                    list_delta_sed.append(sed_h.detach().numpy()/sed_x.detach().numpy())\n",
    "                \n",
    "                plt.plot(list(range(1, 501)), list_delta_sed)\n",
    "                # specifying horizontal line type\n",
    "    #            break\n",
    "    #        break\n",
    "    #    break\n",
    "    #break\n",
    "plt.axhline(y = 1, color = 'black', linestyle = '--')\n",
    "plt.xlabel(r'$k$')\n",
    "plt.ylabel(r'$SED(H_0, H_1)$')\n",
    "plt.tight_layout()\n",
    "plt.savefig('sed_h_plot_2_classes.pdf')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 80,
   "id": "422b9efe",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 400x200 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "seed = 42\n",
    "\n",
    "epsilons = []\n",
    "\n",
    "plt.figure(figsize=(4,2))\n",
    "for num_features in [1, 2, 5, 10]:\n",
    "    for num_classes in [2, 5]:\n",
    "        for num_mixtures in [2, 5, 10]:\n",
    "            for lambda1 in [0., 1., 5., 10.]:\n",
    "                \n",
    "                model_ckpt_pth = f\"C_{num_classes}_M_{num_mixtures}_\" \\\n",
    "                                 f\"D_{num_features}_\" \\\n",
    "                                 f\"lambda_{lambda1}_\" \\\n",
    "                                 f\"seed_{seed}.pth\"\n",
    "                results_ckpt_pth = f\"C_{num_classes}_M_{num_mixtures}_\" \\\n",
    "                                 f\"D_{num_features}_\" \\\n",
    "                                 f\"lambda_{lambda1}_\" \\\n",
    "                                 f\"seed_{seed}.pt\"\n",
    "                \n",
    "                # k, sigma, epsilon are optimized by the method\n",
    "                model = ClassSeparator(\n",
    "                    num_classes=num_classes,\n",
    "                    num_mixtures=num_mixtures,\n",
    "                    num_features=num_features,\n",
    "                    use_full_covariance=(lambda1==0. and num_features>1)\n",
    "                )\n",
    "                \n",
    "                model.load_state_dict(torch.load(osp.join('checkpoints', model_ckpt_pth)))\n",
    "                model.eval()\n",
    "                               \n",
    "                prior, weights, mean, std, epsilon = model.get_parameters()\n",
    "                \n",
    "                epsilons.append(epsilon.detach().squeeze().numpy())\n",
    "                \n",
    "                # specifying horizontal line type\n",
    "    #            break\n",
    "    #        break\n",
    "    #    break\n",
    "    #break\n",
    "\n",
    "sns.histplot(np.array(epsilons), bins=100)\n",
    "    \n",
    "plt.xlabel(r'$\\varepsilon$')\n",
    "plt.tight_layout()\n",
    "plt.savefig('epsilon.pdf')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 81,
   "id": "0ce062d3",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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durrX//6gL988+bNkqTHHnvMNH/btm368Y9/LEnasGGDoqKitHjxYnk8HuXm5urll18e8HcDAHoXUDH87Gc/U0VFhVavXj2gLzcM45bvGTZsmJxOp5xO54C+CwDQPwEVQ0dHh/70pz9p//79yszMVExMjGn5+vXrgxIOADD4AiqG999/XzNnzpQk1dbWmpbZbLYBhwIAWCegYjh48GCwcwAAwgTDbgMATALaY5g7d26fh4wOHDgQcCAAgLUCKoae8ws9urq65HK5VFtbe8PgegCAoSWgYtiwYUOv859//nldu3ZtQIEAANYK6jmGH/zgB36PkwQACC9BLYajR49q2LBhwVwlAGCQBXQo6auP1jQMQ5cvX9aJEycGfDc0AMBaARXD9Y/0lKSoqChlZGRo7dq1WrBgQVCCAQCsEVAxbNu2Ldg5AABhIuAH9UhfPpP5P//5jyRpxowZeuCBB4ISCgBgnYCKobm5WU888YQOHTrke7JaS0uL5s6dqx07duiuu+4KZkYAwCAK6Kqk5cuXq62tTWfOnNHVq1d19epV1dbWyu1265e//GWwMwIABlFAewx79uzR/v37NW3aNN+86dOny+l0cvIZAIa4gPYYuru7b3gGgyTFxMSou7t7wKEAANYJqBjmzZunFStWqKGhwTfv0qVLWrVqlebPnx+0cACAwRdQMbz00ktyu92aPHmy7r77bt19991KT0+X2+3Wpk2bgp0RADCIAjrHMHHiRJ08eVL79+/XuXPnJEnTpk1TTk5OUMMBAAafX3sMBw4c0PTp0+V2u2Wz2fT4449r+fLlWr58uWbPnq0ZM2boX//6V6iyAgAGgV/FsHHjRi1btkwJCQk3LLPb7frFL36h9evXBy0cAGDw+VUMp06d0re+9a2bLl+wYIFqamoGHAoAYB2/iqGpqanXy1R73HHHHfrkk08GHAoAYB2/imHChAmqra296fL3339f48ePH3AoAIB1/CqGb3/721q9erU6OjpuWPbFF1+orKxM3/3ud4MWDgAw+Py6XPXZZ5/V66+/rvvuu0/FxcXKyMiQJJ07d05Op1Ner1e/+c1vQhIUADA4/CqGpKQkHTlyRE899ZRKS0tlGIYkyWazKTc3V06nU0lJSSEJCgAYHH7f+Txp0iS98847unLlio4dO6bq6mpduXJF77zzjtLT0/1a1+HDh5Wfn6+U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",
      "text/plain": [
       "<Figure size 400x200 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "seed = 42\n",
    "\n",
    "ks = []\n",
    "\n",
    "plt.figure(figsize=(4,2))\n",
    "for num_features in [1, 2, 5, 10]:\n",
    "    for num_classes in [2, 5]:\n",
    "        for num_mixtures in [2, 5, 10]:\n",
    "            for lambda1 in [0., 1., 5., 10.]:\n",
    "                \n",
    "                model_ckpt_pth = f\"C_{num_classes}_M_{num_mixtures}_\" \\\n",
    "                                 f\"D_{num_features}_\" \\\n",
    "                                 f\"lambda_{lambda1}_\" \\\n",
    "                                 f\"seed_{seed}.pth\"\n",
    "                results_ckpt_pth = f\"C_{num_classes}_M_{num_mixtures}_\" \\\n",
    "                                 f\"D_{num_features}_\" \\\n",
    "                                 f\"lambda_{lambda1}_\" \\\n",
    "                                 f\"seed_{seed}.pt\"\n",
    "                \n",
    "                # k, sigma, epsilon are optimized by the method\n",
    "                model = ClassSeparator(\n",
    "                    num_classes=num_classes,\n",
    "                    num_mixtures=num_mixtures,\n",
    "                    num_features=num_features,\n",
    "                    use_full_covariance=(lambda1==0. and num_features>1)\n",
    "                )\n",
    "                \n",
    "                model.load_state_dict(torch.load(osp.join('checkpoints', model_ckpt_pth)))\n",
    "                model.eval()\n",
    "                                               \n",
    "                ks.append(model.k.data.detach().squeeze().numpy())\n",
    "                \n",
    "                # specifying horizontal line type\n",
    "    #            break\n",
    "    #        break\n",
    "    #    break\n",
    "    #break\n",
    "\n",
    "sns.histplot(np.array(ks), bins=100)\n",
    "plt.xticks(range(6))\n",
    "    \n",
    "plt.xlabel(r'$k$')\n",
    "plt.tight_layout()\n",
    "plt.savefig('k.pdf')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "03a8ade2",
   "metadata": {},
   "outputs": [],
   "source": [
    "# sample data from model of Figure 1\n",
    "\n",
    "fig1_means = torch.Tensor([[0.3, 1.2], [2, 3]]).unsqueeze(2).double()\n",
    "fig1_std = torch.Tensor([[0.2, 0.4], [0.4, 0.2]]).unsqueeze(2).double()\n",
    "fig1_weights = torch.Tensor([[0.3, 0.7], [0.5, 0.5]]).double()\n",
    "fig1_prior = torch.Tensor([0.5, 0.5]).double()\n",
    "\n",
    "seed = 0\n",
    "torch.manual_seed(seed)\n",
    "\n",
    "num_features = 1\n",
    "num_classes = 2\n",
    "num_mixtures = 2\n",
    "\n",
    "model = ClassSeparator(\n",
    "    num_classes=num_classes,\n",
    "    num_mixtures=num_mixtures,\n",
    "    num_features=num_features,\n",
    "    use_full_covariance=(num_features>1)\n",
    ")                                              \n",
    "\n",
    "model.prior.data = fig1_prior\n",
    "model.weights.data = fig1_weights\n",
    "model.mean.data = fig1_means\n",
    "model.std.data = fig1_std\n",
    "\n",
    "num_nodes = 10000\n",
    "\n",
    "# get_parameters() method of ClassSeparator adds the quantity STD_MIN to the learned parameter std for numerical reasons\n",
    "# in this case, we want to remove it to stick to our example of Fig 1\n",
    "X, y = model.sample_data(num_samples=num_nodes, remove_std_min=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "915786e0",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(0.0, 0.01)"
      ]
     },
     "execution_count": 4,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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kNVqldPp1LPPPqvhw4crLy8vaH9tNpuampp8vgAAQP8RtaDpxIkTcrlcyszM9NmemZkZdJqsrq6u0/aefztrk5CQoL/85S/au3evBg8erNTUVC1ZskSVlZUaNmxY0P4uWrRI6enp3q9Ro0aFdsIAACCuRT0RvLcZhqHZs2dr+PDh+utf/6odO3aouLhYP/7xj/XFF18E3W/+/PlqbGz0fh09erQXew0AAKItakFTRkaGLBaL6uvrfbbX19crKysr4D5ZWVmdtvf821mbLVu2aNOmTVq3bp2+853v6LLLLtPTTz+ttLQ0/e53vwva35SUFA0ZMsTnCwAA9B9RC5qSk5OVl5en6upq7za3263q6moVFBQE3KegoMCnvSRVVVV52+fk5CgrK8unTVNTk7Zv3+5t09LSIqktf6q9xMREud3u7p8YAADokwZE883LyspUUlKiSZMmafLkyVq6dKmsVqtKS0slSbfddptGjhypRYsWSZLmzZunqVOn6sknn9S0adO0bt067dq1S6tXr5bUlq9055136pFHHtG4ceOUk5Oj+++/X9nZ2SouLpbUFngNGzZMJSUlWrhwodLS0vSf//mfOnLkiKZNmxaVcQAAALEvqkHT9OnTdfz4cS1cuFB1dXWaOHGiKisrvYnctbW1PneEpkyZorVr12rBggW67777NG7cOG3cuFGXXHKJt80999wjq9WqO+64Qw0NDbryyitVWVmp1NRUSW3TgpWVlfr1r3+tq6++Wg6HQxdffLFefvll5ebm9u4AAACAuBHVOk3xjDpNAADEn7is0wQAABBPCJoAAABMIGgCAAAwgaAJAADABIImAAAAEwiaAAAATCBoAgAAMIGgCQAAwASCJgAAABMImgAAAEwgaAIAADCBoAkAAMAEgiYAAAATCJoAAABMIGgCAAAwgaAJAADABIImAAAAEwiaAAAATCBoAgAAMIGgCQAAwASCJgAAABMImgAAAEwgaAIAADCBoAkAAMAEgiYAAAATCJoAAABMIGgCAAAwgaAJAADABIImAAAAEwiaAAAATCBoAgAAMIGgCQAAwASCJgAAABMImgAAAEwgaAIAADCBoAkAAMAEgiYAAAATCJoAAABMIGgCAAAwgaAJAADABIImAAAAEwiaAAAATCBoAgAAMIGgCQAAwASCJgAAABMImgAAAEwgaAIAADCBoAkA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p7R8+XLl5ORo27Ztke4jAABA1IU1PTdhwgQVFBTomWeekcVikSS5XC7927/9m95++2299957Ee9orGF6DgCA+NPry6gcPnxYd911lzdgkiSLxaKysjIdPnw4nEMCAADEtLCCpssuu8yby9TeBx98oNzc3G53CgAAINaYTgR/9913vf/9y1/+UvPmzdPhw4d1xRVXSJLeeecdrVq1So899ljkewkAABBlpnOaEhMTlZCQoK6a95filuQ0AQAQf3ql5MCRI0dC7hgAAEBfYTpoGjNmTE/2AwAAIKaFXdxSkt5//33V1tbKbrf7bP/JT37SrU4BAADEmrCCpr///e/6h3/4B7333ns+eU6eRXz7Q04TAADoX8IqOTBv3jzl5OTo2LFjGjhwoA4ePKht27Zp0qRJevPNNyPcRQAAgOgL605TTU2NtmzZooyMDCUmJioxMVFXXnmlFi1apF/+8pfau3dvpPsJAAAQVWHdaXK5XBo8eLAkKSMjQ59//rmktmTxDz/8MHK9AwAAiBFh3Wm65JJLtH//fuXk5Cg/P1+LFy9WcnKyVq9erfPPPz/SfQQAAIi6sIKmBQsWyGq1SpIeeugh/ehHP9JVV12lc845R+vXr49oBwEAAGKB6YrgXTl58qSGDRvmfYKur6MiOAAA8adXKoIHc/ToUUnSqFGjunsoAACAmBVWIrjT6dT999+v9PR0jR07VmPHjlV6eroWLFggh8MR6T4CAABEXVh3mubOnasXX3xRixcvVkFBgaS2MgQPPPCAvvzySz3zzDMR7SQAAEC0hZXTlJ6ernXr1um6667z2b5582bNmDFDjY2NEetgrCKnCQCA+NOdv99hTc+lpKRo7NixHbbn5OQoOTk5nEMCAADEtLCCpjlz5ujhhx+WzWbzbrPZbHr00Uc1Z86ciHUOAAAgVpjOabrhhht8vv/LX/6i8847T7m5uZKk/fv3y26365prrolsDwEAAGKA6aApPT3d5/sbb7zR53tKDgAAgL7MdNC0Zs2anuwHAABATOtWccvjx497F+i94IILdO6550akUwAAALEmrERwq9Wqn/3sZxoxYoS++93v6rvf/a6ys7M1c+ZMtbS0RLqPAAAAURdW0FRWVqa33npLr7zyihoaGtTQ0KCXX35Zb731lu66665I9xEAACDqwipumZGRoT/96U/63ve+57N969at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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# double check this is the right distribution \n",
    "sns.histplot(X.numpy(), bins=1000, stat='probability')\n",
    "plt.xlim([-1, 5])\n",
    "plt.ylim([0, 0.01])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "b470222d",
   "metadata": {},
   "outputs": [],
   "source": [
    "# build knn graph for a given k\n",
    "\n",
    "k = 5\n",
    "\n",
    "knn_edge_index = knn_graph(X, k, loop=False, batch=None)  # do not use self loop\n",
    "\n",
    "src_x = X[knn_edge_index[0]]\n",
    "dst_x = X[knn_edge_index[1]]\n",
    "\n",
    "euclidean_distance_x = torch.norm(dst_x - src_x, p=2, dim=1)\n",
    "\n",
    "# set epsilon to the average euclidean distance\n",
    "eps = euclidean_distance_x.mean()\n",
    "\n",
    "model.epsilon.data = torch.tensor([eps])\n",
    "\n",
    "                           \n",
    "# compute CCNS lower bound\n",
    "# get_parameters() method of ClassSeparator adds the quantity STD_MIN to the learned parameter std for numerical reasons\n",
    "# in this case, we want to remove it to stick to our example of Fig 1\n",
    "ccns_lb, _ = model.compute_CCNS_LB(remove_std_min=True)\n",
    "\n",
    "# compute ccns\n",
    "ccns_true = ccns(knn_edge_index, y, num_nodes)\n",
    "\n",
    "# compute ccns via Monte Carlo for the intra-class similarity\n",
    "ccns_mc = monte_carlo_ccns(num_classes,\n",
    "                             num_samples = 1000,\n",
    "                             epsilon=eps,\n",
    "                             p_c=fig1_prior,\n",
    "                             p_m_given_c=fig1_weights,\n",
    "                             gaussian_mean=fig1_means,\n",
    "                             gaussian_std=fig1_std)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "id": "25aca281",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 500x250 with 3 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# plot lower bound vs ccns as two subplots\n",
    "\n",
    "fig, axes = plt.subplots(1, 3, sharey=True, figsize=(5, 2.5))\n",
    "\n",
    "s = sns.heatmap(ccns_lb.detach(), annot=True, cbar=False, ax=axes[0], vmin=0., vmax=ccns_true.max())\n",
    "s.set(xlabel='(a)')\n",
    "s = sns.heatmap(ccns_mc.detach(), annot=True, cbar=False, ax=axes[2], vmin=0., vmax=ccns_true.max())\n",
    "s.set(xlabel='(b)')\n",
    "s = sns.heatmap(ccns_true.detach(), annot=True, cbar=False, ax=axes[1], vmin=0., vmax=ccns_true.max())\n",
    "s.set(xlabel='(c)')\n",
    "# fig.text(0.52, -0.01, 'classes', ha='center', fontsize=12)\n",
    "# fig.text(0.0, 0.56, 'classes', va='center', rotation='vertical', fontsize=12)\n",
    "plt.tight_layout()\n",
    "plt.savefig('ccns_fig1.pdf')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "7e483e29",
   "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.8.10"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
