{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Python Path non-sense\n",
    "%load_ext autoreload\n",
    "%autoreload 2\n",
    "\n",
    "import sys\n",
    "import os\n",
    "\n",
    "sys.path = [x for x in sys.path if 'bayes_gsl' not in x]\n",
    "new_path = '/Users/maxw/projects/gsl-bnn/' ## change this to the path of the src directory!\n",
    "if new_path not in sys.path:\n",
    "    sys.path.append(new_path)\n",
    "\n",
    "# Now try importing your module using the absolute path as a check\n",
    "from src.models import dpg_bnn\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "\"\"\"\n",
    "    Compare all models on NLL, BS, Error, ECE.\n",
    "    Plot the calibration curves and the distribution of probabilities within each bin.\n",
    "\n",
    "\"\"\"\n",
    "import os\n",
    "import pickle\n",
    "import arviz\n",
    "import matplotlib.pyplot as plt\n",
    "import numpyro.infer\n",
    "import seaborn as sns\n",
    "from sklearn.calibration import calibration_curve\n",
    "import numpy as np\n",
    "\n",
    "import jax.numpy as jnp\n",
    "import jax\n",
    "\n",
    "\n",
    "from src import TMLR_TEXTWIDTH, TMLR_PLOT_DEPTH, FIGURES_PATH\n",
    "\n",
    "from src.models import dpg_bnn, dpg_mimo_bnn, dpg_mimo_stochastic_head, pds_bnn\n",
    "#from src.data import load_data\n",
    "from src.metrics import compute_metrics\n",
    "\n",
    "from src.config import num_samples_to_generate, w_init_scale, lam_init_scale, altered_prior\n",
    "from experiments.iid_generalization.config import dpg_hyperparameters, pds_hyperparameters, dpg_mimo_hyperparameters, dpg_mimo_e_hyperparameters, experiment_settings\n",
    "from src import NUM_BINS"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "from experiments.iid_generalization.config import dpg_hyperparameters, pds_hyperparameters, dpg_mimo_hyperparameters, dpg_mimo_e_hyperparameters, experiment_settings\n",
    "models = {\n",
    "        'dpg': (dpg_bnn, dpg_hyperparameters, f\"dpg_D={dpg_hyperparameters['depth']}.pkl\"),\n",
    "        'pds': (pds_bnn, pds_hyperparameters, f\"pds_D={dpg_hyperparameters['depth']}.pkl\"),\n",
    "        'dpg_mimo': (dpg_mimo_bnn, dpg_mimo_hyperparameters, f\"dpg_mimo_D={dpg_mimo_hyperparameters['depth']}_C={dpg_mimo_hyperparameters['num_channels']}.pkl\"),\n",
    "        'dpg_mimo_e': (dpg_mimo_stochastic_head, dpg_mimo_e_hyperparameters, f\"dpg_mimo_e_D={dpg_mimo_e_hyperparameters['mimo_base']['depth']}_C={dpg_mimo_e_hyperparameters['mimo_base']['num_channels']}_stoch_head_C={dpg_mimo_e_hyperparameters['num_stochastic_channels']}_D={dpg_mimo_e_hyperparameters['stochastic_head_depth']}.pkl\")\n",
    "    }"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "def display_metrics(metrics_dict):\n",
    "    calibration_dict = metrics_dict['calibration_dict']\n",
    "    print(f'Test Error: {1 - metrics_dict[\"accuracies\"].mean():.5f} \\pm {metrics_dict[\"accuracies\"].std():.5f}')\n",
    "    print(f'Test NLL: {-1 * metrics_dict[\"log_likelihoods\"].mean():.3f} \\pm {metrics_dict[\"log_likelihoods\"].std():.3f}')\n",
    "    print(f'Test BS: {metrics_dict[\"brier_scores\"].mean():.5f} \\pm {metrics_dict[\"brier_scores\"].std():.5f}')\n",
    "    print(f'Test ECE:{calibration_dict[\"ece\"]:.5f}') \n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "**** dpg ****\n",
      "Metrics found at /Users/maxw/projects/gsl-bnn/results/iid_generalization/dpg_D=200.pkl.\n",
      "Test Error: 0.01574 \\pm 0.01008\n",
      "Test NLL: 10.333 \\pm 5.378\n",
      "Test BS: 0.01286 \\pm 0.00716\n",
      "Test ECE:0.00342\n",
      "\n",
      "**** pds ****\n",
      "Metrics found at /Users/maxw/projects/gsl-bnn/results/iid_generalization/pds_D=200.pkl.\n",
      "Test Error: 0.01589 \\pm 0.01026\n",
      "Test NLL: 10.317 \\pm 5.391\n",
      "Test BS: 0.01285 \\pm 0.00718\n",
      "Test ECE:0.00375\n",
      "\n",
      "**** dpg_mimo ****\n",
      "Metrics found at /Users/maxw/projects/gsl-bnn/results/iid_generalization/dpg_mimo_D=200_C=4.pkl.\n",
      "Test Error: 0.01416 \\pm 0.00941\n",
      "Test NLL: 9.191 \\pm 5.155\n",
      "Test BS: 0.01188 \\pm 0.00665\n",
      "Test ECE:0.00350\n",
      "\n",
      "**** dpg_mimo_e ****\n",
      "Metrics found at /Users/maxw/projects/gsl-bnn/results/iid_generalization/dpg_mimo_e_D=200_C=4_stoch_head_C=2_D=20.pkl.\n",
      "Test Error: 0.01289 \\pm 0.00823\n",
      "Test NLL: 8.354 \\pm 4.738\n",
      "Test BS: 0.01065 \\pm 0.00604\n",
      "Test ECE:0.00240\n"
     ]
    }
   ],
   "source": [
    "# load the metrics for each model\n",
    "model_metrics = {}\n",
    "# Run experiments for each model\n",
    "for name, (model, params, results_file) in models.items():\n",
    "    results_path = os.path.join(experiment_settings['results_path'], results_file)\n",
    "    print(f\"\\n**** {name} ****\")\n",
    "    if os.path.exists(results_path):\n",
    "        print(f\"Metrics found at {results_path}.\")\n",
    "        # Load results\n",
    "        with open(results_path, 'rb') as f:\n",
    "            results = pickle.load(f)\n",
    "            #samples_key = 'samples' if name != 'dpg_mimo' else 'map point estimates'\n",
    "            samples, metrics_dict = results['samples'], results['metrics']\n",
    "            model_metrics[name] = metrics_dict\n",
    "            display_metrics(metrics_dict)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "hi\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 650.127x175 with 5 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "\"\"\"\n",
    "    Make plot\n",
    "\"\"\"\n",
    "dpg_metrics = model_metrics['dpg']\n",
    "pds_metrics = model_metrics['pds']\n",
    "partial_stoch_metrics = model_metrics['dpg_mimo_e']\n",
    "\n",
    "num_plots = 5\n",
    "tick_fontsize = 6\n",
    "label_fontsize = 10\n",
    "fig, axes = plt.subplots(1, num_plots, figsize=(TMLR_TEXTWIDTH, TMLR_PLOT_DEPTH))\n",
    "nll_ax, bs_ax, err_ax, ece_ax, cal_ax = axes\n",
    "\n",
    "\n",
    "# first 4 plots: barplots (with stdv) of are the NLL, BS, Error, ECE for each model\n",
    "stdv_scale = 1/20\n",
    "colors = ['#1f77b4', '#ff7f0e', '#2ca02c']\n",
    "# NLL\n",
    "means = [-1*dpg_metrics['log_likelihoods'].mean(),\n",
    "         -1*pds_metrics['log_likelihoods'].mean(),\n",
    "         -1*partial_stoch_metrics['log_likelihoods'].mean()]\n",
    "stdvs = np.array([dpg_metrics['log_likelihoods'].std(),\n",
    "                  pds_metrics['log_likelihoods'].std(),\n",
    "                  partial_stoch_metrics['log_likelihoods'].std()])\n",
    "nll_ax.bar([0, 1, 2], means, yerr=stdv_scale*stdvs, capsize=5, color=colors)\n",
    "nll_ax.set_xlabel('NLL', fontsize=label_fontsize)\n",
    "nll_ax.set_ylim([7.5, 10.9])\n",
    "nll_ax.set_yticks([8, 9, 10])\n",
    "nll_ax.set_yticklabels([8, 9, 10], fontsize=tick_fontsize)\n",
    "\n",
    "# BS\n",
    "means = [dpg_metrics['brier_scores'].mean(),\n",
    "         pds_metrics['brier_scores'].mean(),\n",
    "         partial_stoch_metrics['brier_scores'].mean()]\n",
    "stdvs = np.array([dpg_metrics['brier_scores'].std(),\n",
    "                  pds_metrics['brier_scores'].std(),\n",
    "                  partial_stoch_metrics['brier_scores'].std()])\n",
    "bs_ax.clear()\n",
    "bs_ax.bar([0, 1, 2], means, yerr=stdv_scale*stdvs, capsize=5, color=colors)\n",
    "bs_ax.set_xlabel('BS', fontsize=label_fontsize)\n",
    "#bs_ax.set_yscale('log')\n",
    "bs_ax.set_ylim([.009, .0145])\n",
    "bs_ax.set_yticks([1.0e-2, 1.2e-2, 1.4e-2])\n",
    "bs_ax.set_yticklabels(['.01', '.012', '.014'], fontsize=tick_fontsize)\n",
    "\n",
    "# Error\n",
    "means = [1 - dpg_metrics['accuracies'].mean(),\n",
    "         1 - pds_metrics['accuracies'].mean(),\n",
    "         1 - partial_stoch_metrics['accuracies'].mean()]\n",
    "stdvs = np.array([\n",
    "    dpg_metrics['accuracies'].std(),\n",
    "    pds_metrics['accuracies'].std(),\n",
    "    partial_stoch_metrics['accuracies'].std()])\n",
    "err_ax.clear()\n",
    "err_ax.bar([0, 1, 2], means, yerr=stdv_scale*stdvs, capsize=5, color=colors)\n",
    "err_ax.set_xlabel('Error', fontsize=label_fontsize)\n",
    "#err_ax.set_yscale('log')\n",
    "err_ax.set_ylim([.0115, .017])\n",
    "err_ax.set_yticks([.012, .014, .016])\n",
    "err_ax.set_yticklabels(['.012', '.014', '.016'], fontsize=tick_fontsize)\n",
    "\n",
    "# ECE\n",
    "ece_ax.clear()\n",
    "ece_ax.bar([0, 1, 2],\n",
    "           [dpg_metrics['calibration_dict']['ece'],\n",
    "            pds_metrics['calibration_dict']['ece'],\n",
    "            partial_stoch_metrics['calibration_dict']['ece']],\n",
    "           #label=['dpg', 'pds', 'dpg-mimo-e'], \n",
    "           label=['DPG', 'PDS', 'DPG-MIMO-E'],\n",
    "           color=colors)\n",
    "ece_ax.set_xlabel('ECE', fontsize=label_fontsize)\n",
    "ece_ax.set_yscale('linear')\n",
    "ece_ax.set_ylim([.0005, .005])\n",
    "ece_ax.set_yticks([.001, .002, .003, .004])\n",
    "ece_ax.set_yticklabels(['.001', '.002', '.003', '.004'], fontsize=tick_fontsize)\n",
    "ece_ax.legend(loc='upper left', fontsize=tick_fontsize, framealpha=1.0)\n",
    "#ece_ax.set_ylim([.95, 1.05])\n",
    "\n",
    "# last plot: calibration plot\n",
    "cal_ax.clear()\n",
    "cal_ax.plot([0.45, 1], [.45, 1], 'k--') #, label='Perfect') # Calibration')\n",
    "# x is the center of the NUM_BINS uniform bins between 0.5 and 1\n",
    "x = np.linspace(0.5, 1, NUM_BINS, endpoint=False) + 1/(4*NUM_BINS)\n",
    "alpha=1.0\n",
    "for calibration_dict, label, marker in \\\n",
    "        zip([dpg_metrics['calibration_dict'],\n",
    "             pds_metrics['calibration_dict'],\n",
    "             partial_stoch_metrics['calibration_dict']],\n",
    "            ['base', 'pds', 'dpg-mimo-e'],\n",
    "            ['o', 'x', '*']):\n",
    "    bin_counts = calibration_dict['bin_counts']\n",
    "    mask = bin_counts > 0\n",
    "    cal_ax.plot(x[mask],\n",
    "                calibration_dict['bin_acc'][mask],\n",
    "                alpha=alpha,\n",
    "                marker=marker,\n",
    "                label=label)\n",
    "\n",
    "cal_ax.tick_params(axis='y', direction='in')\n",
    "# move y-label to right\n",
    "\n",
    "cal_ax.set_yticks([.4, .5, .6, .7, .8, .9, 1])\n",
    "cal_ax.set_yticklabels(['', '', '.6', '.7', '.8', '.9', '1'], fontsize=tick_fontsize)\n",
    "cal_ax.set_xticks([.4, .5, .6, .7, .8, .9, 1])\n",
    "cal_ax.set_xticklabels(['', '.5', '', '.7', '.8', '.9', '1'], fontsize=tick_fontsize)\n",
    "cal_ax.tick_params(axis='both', which='major', pad=-12, direction='in')\n",
    "cal_ax.set_ylabel('Accuracy', fontsize=label_fontsize, labelpad=1)\n",
    "cal_ax.set_xlabel('Confidence', fontsize=label_fontsize, labelpad=4)\n",
    "#cal_ax.legend(loc='upper left', fontsize=tick_fontsize, framealpha=1.0)\n",
    "\n",
    "for ax in [nll_ax, bs_ax, err_ax, ece_ax]:\n",
    "    ax.set_xticks([])\n",
    "    ax.set_xticklabels([])\n",
    "    ax.tick_params(axis='y', direction='in')\n",
    "    #ax.set_xlabel('')\n",
    "    # remove borders\n",
    "    #ax.spines['top'].set_visible(False)\n",
    "    #ax.spines['right'].set_visible(False)\n",
    "    #ax.spines['bottom'].set_visible(False)\n",
    "    #ax.spines['left'].set_visible(False)\n",
    "\n",
    "print('hi')\n",
    "space = .25\n",
    "plt.subplots_adjust(wspace=space, hspace=space, bottom=0.0, top=1, left=0.0, right=1.0) # SAVE FIG AND INSPECT BEFORE CHANGING!\n",
    "plt.savefig(FIGURES_PATH + 'iid_generalization', dpi=1000, bbox_inches=\"tight\", pad_inches=0.02)\n",
    "plt.show(block=False)"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "gsl-bnn-mac-m2",
   "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.18"
  },
  "orig_nbformat": 4
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
