{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import time\n",
    "import argparse\n",
    "import jax\n",
    "import matplotlib.pyplot as plt\n",
    "import optax\n",
    "import matfree\n",
    "import tree_math as tm\n",
    "from flax import linen as nn\n",
    "from jax import nn as jnn\n",
    "from jax import numpy as jnp\n",
    "from jax import random, jit\n",
    "import pickle\n",
    "from src.losses import cross_entropy_loss, accuracy_preds, nll\n",
    "from src.helper import calculate_exact_ggn, tree_random_normal_like, compute_num_params\n",
    "from src.sampling.predictive_samplers import sample_predictive, sample_hessian_predictive\n",
    "from src.models import ResNet, ResNetBlock, PreActResNetBlock\n",
    "from jax import flatten_util\n",
    "import matplotlib.pyplot as plt\n",
    "from src.data import CIFAR10, n_classes\n",
    "import torch\n",
    "from src.data.torch_datasets import MNIST, numpy_collate_fn\n",
    "import flax\n",
    "from src.data.datasets import get_rotated_cifar_loaders, get_cifar10_ood_loaders, load_corrupted_cifar10_per_type, load_corrupted_cifar10\n",
    "from src.ood_functions.evaluate import evaluate, evaluate_map, evaluate_samples\n",
    "from src.ood_functions.metrics import compute_metrics\n",
    "\n",
    "\n",
    "    "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Model Loading"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "param_dict = pickle.load(open(\"../checkpoints/CIFAR-10/ResNet/epoch200_seed0_params.pickle\", \"rb\"))\n",
    "params = param_dict['params']\n",
    "batch_stats = param_dict['batch_stats']\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "param_list = []\n",
    "param_list.append(pickle.load(open(\"../checkpoints/CIFAR-10/ResNet/good_params_seed0.pickle\", \"rb\")))\n",
    "param_list.append(pickle.load(open(\"../checkpoints/CIFAR-10/ResNet/good_params_seed1.pickle\", \"rb\")))\n",
    "param_list.append(pickle.load(open(\"../checkpoints/CIFAR-10/ResNet/good_params_seed2.pickle\", \"rb\")))\n",
    "param_list.append(pickle.load(open(\"../checkpoints/CIFAR-10/ResNet/good_params_seed3.pickle\", \"rb\")))\n",
    "param_list.append(pickle.load(open(\"../checkpoints/CIFAR-10/ResNet/good_params_seed4.pickle\", \"rb\")))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "output_dim = 10\n",
    "model = ResNet(\n",
    "            num_classes = output_dim,\n",
    "            c_hidden =(16, 32, 64),\n",
    "            num_blocks = (3, 3, 3),\n",
    "            act_fn = nn.relu,\n",
    "            block_class = ResNetBlock #PreActResNetBlock #\n",
    "        )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Files already downloaded and verified\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/xxx/geometric-laplace/src/data/cifar10.py:50: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  self.targets = F.one_hot(torch.tensor(self.dataset.targets), len(cls)).numpy()\n"
     ]
    }
   ],
   "source": [
    "n_samples_per_class = None\n",
    "cls=list(range(n_classes(\"CIFAR-10\")))\n",
    "dataset = CIFAR10(path_root='/xxx/data', train=True, n_samples_per_class=n_samples_per_class, download=True, cls=cls, seed=0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "data_array = jnp.array([data[0] for data in dataset])\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "# posterior_dict = pickle.load(open(\"../checkpoints/CIFAR-10/old_samples/posterior_samples_deep_resnet_prec05_seed0_type_non-kernel-eigvals_params.pickle\", \"rb\"))\n",
    "# posterior_dict = pickle.load(open(\"../checkpoints/CIFAR-10/posterior_samples_just_in_case_seed0_type_non-kernel-eigvals_params.pickle\", \"rb\"))\n",
    "# posterior_dict = pickle.load(open(\"../checkpoints/CIFAR-10/posterior_samples_deep_resnet_prec05_seed0_type_non-kernel-eigvals_params.pickle\", \"rb\"))\n",
    "# posterior_dict = pickle.load(open(\"../checkpoints/CIFAR-10/proj_posterior_samples_prec2_seed0_params.pickle\", \"rb\"))\n",
    "\n",
    "# posterior_params = posterior_dict['posterior_samples']\n",
    "# posterior_dict = pickle.load(open(\"../checkpoints/CIFAR-10/posterior_samples_posterior_resnet_250iters_prec_5__seed0_type_full-ggn_params.pickle\", \"rb\"))\n",
    "posterior_dict = pickle.load(open(\"../checkpoints/CIFAR-10/posterior_samples__1500_5_seed0_type_non-kernel-eigvals_iter1500_steps3_samples1_prec5.0_params.pickle\", \"rb\"))\n",
    "\n",
    "posterior_params = posterior_dict['posterior_samples']\n",
    "params = param_list[0]\n",
    "map_params = params\n",
    "x_train = data_array[:500]\n",
    "y_train = jnp.array([data[1] for data in dataset])[:500]\n",
    "# For the rebuttals only\n",
    "x_train = jnp.moveaxis(x_train, 2, 3)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "posterior_dict = pickle.load(open(\"../checkpoints/CIFAR-10/posterior_samples_1_1500_10_seed0_type_non-kernel-eigvals_iter1500_steps3_samples3_prec10.0_params.pickle\", \"rb\"))\n",
    "\n",
    "posterior_params = posterior_dict['posterior_samples']\n",
    "params = param_list[0]\n",
    "map_params = params\n",
    "x_train = data_array[:500]\n",
    "y_train = jnp.array([data[1] for data in dataset])[:500]\n",
    "# For the rebuttals only\n",
    "x_train = jnp.moveaxis(x_train, 2, 3)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy for seed 0: 99.2\n",
      "Accuracy for seed 1: 99.6\n",
      "Accuracy for seed 2: 98.6\n",
      "Accuracy for seed 3: 98.799995\n",
      "Accuracy for seed 4: 99.2\n"
     ]
    }
   ],
   "source": [
    "for seed, param_dict in enumerate(param_list):\n",
    "    params = param_dict['params']\n",
    "    batch_stats = param_dict['batch_stats']\n",
    "    map_preds = model.apply({'params': params, 'batch_stats': batch_stats},\n",
    "                                    x_train,\n",
    "                                    train=False,\n",
    "                                    mutable=False)\n",
    "    print(f\"Accuracy for seed {seed}:\",accuracy_preds(map_preds, y_train)/y_train.shape[0] * 100)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [],
   "source": [
    "# model_fn = lambda p, x: model.apply({'params': p, 'batch_stats': batch_stats},\n",
    "#                                         x,\n",
    "#                                         train=False,\n",
    "#                                         mutable=False)\n",
    "# pred_posterior = sample_predictive(posterior_samples=posterior_params,\n",
    "#                                     params=params,\n",
    "#                                     model_fn=model_fn,\n",
    "#                                     x_test=x_train,\n",
    "#                                     linearised_laplace=True,\n",
    "#                                     posterior_sample_type=\"Pytree\",\n",
    "#                                 )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [],
   "source": [
    "# print(jax.vmap(lambda x,y : jnp.linalg.norm(x - y)/jnp.linalg.norm(y), in_axes=(0,None))(pred_posterior, map_preds))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [],
   "source": [
    "# print(jax.vmap(accuracy_preds, in_axes=(0,None))(pred_posterior, y_train)/y_train.shape[0] * 100)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Array([12.8, 12.8, 13.2], dtype=float32)"
      ]
     },
     "execution_count": 13,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "def accuracy_samples(sample):\n",
    "    posterior_preds = model.apply({'params': sample, 'batch_stats': batch_stats},\n",
    "                                    x_train,\n",
    "                                    train=False,\n",
    "                                    mutable=False)\n",
    "    return accuracy_preds(posterior_preds, y_train)/y_train.shape[0] * 100\n",
    "jax.vmap(accuracy_samples)(posterior_params)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Files already downloaded and verified\n",
      "map Accuracy:  99.2\n",
      "sample Accuracy:  297.8\n"
     ]
    }
   ],
   "source": [
    "nonker_posterior_samples = posterior_params\n",
    "params = param_list[0]['params']\n",
    "batch_stats = param_list[0]['batch_stats']\n",
    "dataset = CIFAR10(path_root='/xxx/data', train=True, n_samples_per_class=n_samples_per_class, download=True, cls=cls, seed=0)\n",
    "dataset_size = len(dataset)\n",
    "output_dim = 10\n",
    "model = ResNet(\n",
    "    num_classes = output_dim,\n",
    "    c_hidden =(16, 32, 64),\n",
    "    num_blocks = (3, 3, 3),\n",
    "    act_fn = nn.relu,\n",
    "    block_class = ResNetBlock #PreActResNetBlock #\n",
    ")\n",
    "x_train = jnp.array([data[0] for data in dataset])\n",
    "# Only for rebuttal experiments:\n",
    "x_train = jnp.moveaxis(x_train, 2, 3)\n",
    "model_on_data = lambda p: model.apply({'params': p, 'batch_stats': batch_stats}, x_train[:500], train=False, mutable=False)\n",
    "sample_preds = jax.vmap(model_on_data)(nonker_posterior_samples)\n",
    "sample_preds = sample_preds.squeeze()\n",
    "y_train = jnp.array([data[1] for data in dataset])[:500]\n",
    "accuracy = accuracy_preds(sample_preds, y_train)/500 * 100.\n",
    "preds = model_on_data(params)\n",
    "map_accuracy = accuracy_preds(preds, y_train)/500 * 100.\n",
    "print(\"map Accuracy: \", map_accuracy)\n",
    "print(\"sample Accuracy: \", accuracy)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(50000, 3, 32, 32)"
      ]
     },
     "execution_count": 15,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "x_train.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [
    {
     "ename": "XlaRuntimeError",
     "evalue": "RESOURCE_EXHAUSTED: Out of memory allocating 9830400000 bytes.",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mXlaRuntimeError\u001b[0m                           Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[16], line 7\u001b[0m\n\u001b[1;32m      2\u001b[0m     posterior_preds \u001b[38;5;241m=\u001b[39m model\u001b[38;5;241m.\u001b[39mapply({\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mparams\u001b[39m\u001b[38;5;124m'\u001b[39m: sample, \u001b[38;5;124m'\u001b[39m\u001b[38;5;124mbatch_stats\u001b[39m\u001b[38;5;124m'\u001b[39m: batch_stats},\n\u001b[1;32m      3\u001b[0m                                     x_train,\n\u001b[1;32m      4\u001b[0m                                     train\u001b[38;5;241m=\u001b[39m\u001b[38;5;28;01mFalse\u001b[39;00m,\n\u001b[1;32m      5\u001b[0m                                     mutable\u001b[38;5;241m=\u001b[39m\u001b[38;5;28;01mFalse\u001b[39;00m)\n\u001b[1;32m      6\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m jnp\u001b[38;5;241m.\u001b[39mlinalg\u001b[38;5;241m.\u001b[39mnorm(posterior_preds \u001b[38;5;241m-\u001b[39m map_preds)\u001b[38;5;241m/\u001b[39m(jnp\u001b[38;5;241m.\u001b[39mlinalg\u001b[38;5;241m.\u001b[39mnorm(map_preds))\n\u001b[0;32m----> 7\u001b[0m \u001b[43mjax\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mvmap\u001b[49m\u001b[43m(\u001b[49m\u001b[43mdistance_from_maps\u001b[49m\u001b[43m)\u001b[49m\u001b[43m(\u001b[49m\u001b[43mposterior_params\u001b[49m\u001b[43m)\u001b[49m\n",
      "    \u001b[0;31m[... skipping hidden 3 frame]\u001b[0m\n",
      "Cell \u001b[0;32mIn[16], line 2\u001b[0m, in \u001b[0;36mdistance_from_maps\u001b[0;34m(sample)\u001b[0m\n\u001b[1;32m      1\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m \u001b[38;5;21mdistance_from_maps\u001b[39m(sample):\n\u001b[0;32m----> 2\u001b[0m     posterior_preds \u001b[38;5;241m=\u001b[39m \u001b[43mmodel\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mapply\u001b[49m\u001b[43m(\u001b[49m\u001b[43m{\u001b[49m\u001b[38;5;124;43m'\u001b[39;49m\u001b[38;5;124;43mparams\u001b[39;49m\u001b[38;5;124;43m'\u001b[39;49m\u001b[43m:\u001b[49m\u001b[43m \u001b[49m\u001b[43msample\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;124;43m'\u001b[39;49m\u001b[38;5;124;43mbatch_stats\u001b[39;49m\u001b[38;5;124;43m'\u001b[39;49m\u001b[43m:\u001b[49m\u001b[43m \u001b[49m\u001b[43mbatch_stats\u001b[49m\u001b[43m}\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m      3\u001b[0m \u001b[43m                                    \u001b[49m\u001b[43mx_train\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m      4\u001b[0m \u001b[43m                                    \u001b[49m\u001b[43mtrain\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43;01mFalse\u001b[39;49;00m\u001b[43m,\u001b[49m\n\u001b[1;32m      5\u001b[0m \u001b[43m                                    \u001b[49m\u001b[43mmutable\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43;01mFalse\u001b[39;49;00m\u001b[43m)\u001b[49m\n\u001b[1;32m      6\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m jnp\u001b[38;5;241m.\u001b[39mlinalg\u001b[38;5;241m.\u001b[39mnorm(posterior_preds \u001b[38;5;241m-\u001b[39m map_preds)\u001b[38;5;241m/\u001b[39m(jnp\u001b[38;5;241m.\u001b[39mlinalg\u001b[38;5;241m.\u001b[39mnorm(map_preds))\n",
      "    \u001b[0;31m[... skipping hidden 6 frame]\u001b[0m\n",
      "File \u001b[0;32m/xxx/geometric-laplace/src/models/resnet.py:73\u001b[0m, in \u001b[0;36mResNet.__call__\u001b[0;34m(self, x, train)\u001b[0m\n\u001b[1;32m     70\u001b[0m x \u001b[38;5;241m=\u001b[39m jnp\u001b[38;5;241m.\u001b[39mtranspose(x, (\u001b[38;5;241m0\u001b[39m, \u001b[38;5;241m2\u001b[39m, \u001b[38;5;241m3\u001b[39m, \u001b[38;5;241m1\u001b[39m))\n\u001b[1;32m     72\u001b[0m \u001b[38;5;66;03m# A first convolution on the original image to scale up the channel size\u001b[39;00m\n\u001b[0;32m---> 73\u001b[0m x \u001b[38;5;241m=\u001b[39m \u001b[43mnn\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mConv\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mc_hidden\u001b[49m\u001b[43m[\u001b[49m\u001b[38;5;241;43m0\u001b[39;49m\u001b[43m]\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mkernel_size\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43m(\u001b[49m\u001b[38;5;241;43m3\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m3\u001b[39;49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mkernel_init\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mresnet_kernel_init\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43muse_bias\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43;01mFalse\u001b[39;49;00m\u001b[43m)\u001b[49m\u001b[43m(\u001b[49m\u001b[43mx\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m     74\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mblock_class \u001b[38;5;241m==\u001b[39m ResNetBlock:  \u001b[38;5;66;03m# If pre-activation block, we do not apply non-linearities yet\u001b[39;00m\n\u001b[1;32m     75\u001b[0m     x \u001b[38;5;241m=\u001b[39m nn\u001b[38;5;241m.\u001b[39mBatchNorm()(x, use_running_average\u001b[38;5;241m=\u001b[39m\u001b[38;5;129;01mnot\u001b[39;00m train)\n",
      "    \u001b[0;31m[... skipping hidden 2 frame]\u001b[0m\n",
      "File \u001b[0;32m/xxx/geometric-laplace/geom/lib64/python3.9/site-packages/flax/linen/linear.py:529\u001b[0m, in \u001b[0;36m_Conv.__call__\u001b[0;34m(self, inputs)\u001b[0m\n\u001b[1;32m    527\u001b[0m   \u001b[38;5;28;01melse\u001b[39;00m:\n\u001b[1;32m    528\u001b[0m     conv_general_dilated \u001b[38;5;241m=\u001b[39m lax\u001b[38;5;241m.\u001b[39mconv_general_dilated\n\u001b[0;32m--> 529\u001b[0m   y \u001b[38;5;241m=\u001b[39m \u001b[43mconv_general_dilated\u001b[49m\u001b[43m(\u001b[49m\n\u001b[1;32m    530\u001b[0m \u001b[43m      \u001b[49m\u001b[43minputs\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    531\u001b[0m \u001b[43m      \u001b[49m\u001b[43mkernel\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    532\u001b[0m \u001b[43m      \u001b[49m\u001b[43mstrides\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    533\u001b[0m \u001b[43m      \u001b[49m\u001b[43mpadding_lax\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    534\u001b[0m \u001b[43m      \u001b[49m\u001b[43mlhs_dilation\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43minput_dilation\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    535\u001b[0m \u001b[43m      \u001b[49m\u001b[43mrhs_dilation\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mkernel_dilation\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    536\u001b[0m \u001b[43m      \u001b[49m\u001b[43mdimension_numbers\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mdimension_numbers\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    537\u001b[0m \u001b[43m      \u001b[49m\u001b[43mfeature_group_count\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mfeature_group_count\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    538\u001b[0m \u001b[43m      \u001b[49m\u001b[43mprecision\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mprecision\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    539\u001b[0m \u001b[43m  \u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    540\u001b[0m \u001b[38;5;28;01melse\u001b[39;00m:\n\u001b[1;32m    541\u001b[0m   y \u001b[38;5;241m=\u001b[39m lax\u001b[38;5;241m.\u001b[39mconv_general_dilated_local(\n\u001b[1;32m    542\u001b[0m       lhs\u001b[38;5;241m=\u001b[39minputs,\n\u001b[1;32m    543\u001b[0m       rhs\u001b[38;5;241m=\u001b[39mkernel,\n\u001b[0;32m   (...)\u001b[0m\n\u001b[1;32m    550\u001b[0m       precision\u001b[38;5;241m=\u001b[39m\u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mprecision,\n\u001b[1;32m    551\u001b[0m   )\n",
      "    \u001b[0;31m[... skipping hidden 20 frame]\u001b[0m\n",
      "File \u001b[0;32m/xxx/geometric-laplace/geom/lib64/python3.9/site-packages/jax/_src/interpreters/pxla.py:1159\u001b[0m, in \u001b[0;36mExecuteReplicated.__call__\u001b[0;34m(self, *args)\u001b[0m\n\u001b[1;32m   1157\u001b[0m   \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_handle_token_bufs(result_token_bufs, sharded_runtime_token)\n\u001b[1;32m   1158\u001b[0m \u001b[38;5;28;01melse\u001b[39;00m:\n\u001b[0;32m-> 1159\u001b[0m   results \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mxla_executable\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mexecute_sharded\u001b[49m\u001b[43m(\u001b[49m\u001b[43minput_bufs\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m   1160\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m dispatch\u001b[38;5;241m.\u001b[39mneeds_check_special():\n\u001b[1;32m   1161\u001b[0m   out_arrays \u001b[38;5;241m=\u001b[39m results\u001b[38;5;241m.\u001b[39mdisassemble_into_single_device_arrays()\n",
      "\u001b[0;31mXlaRuntimeError\u001b[0m: RESOURCE_EXHAUSTED: Out of memory allocating 9830400000 bytes."
     ]
    }
   ],
   "source": [
    "def distance_from_maps(sample):\n",
    "    posterior_preds = model.apply({'params': sample, 'batch_stats': batch_stats},\n",
    "                                    x_train,\n",
    "                                    train=False,\n",
    "                                    mutable=False)\n",
    "    return jnp.linalg.norm(posterior_preds - map_preds)/(jnp.linalg.norm(map_preds))\n",
    "jax.vmap(distance_from_maps)(posterior_params)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "args_dict = {}\n",
    "args_dict[\"optimizer\"] = \"sgd\"\n",
    "args_dict[\"learning_rate\"] = 0.1\n",
    "args_dict[\"momentum\"] = 0.9\n",
    "args_dict[\"weight_decay\"] = 1e-4\n",
    "# args_dict[\"n_steps_per_epoch\"] = int(len(train_loader.dataset) / args.batch_size)\n",
    "args_dict[\"opt_hp\"] = {\n",
    "    \"lr\": 0.1,\n",
    "    \"momentum\": 0.9,\n",
    "    \"weight_decay\": 1e-4\n",
    "}\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(512, 3, 32, 32)\n"
     ]
    }
   ],
   "source": [
    "X = data_array[:512]\n",
    "outs = model.apply({'params': params, 'batch_stats': batch_stats},\n",
    "                        X,\n",
    "                        train=False,\n",
    "                        mutable=False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Array(99.21875, dtype=float32)"
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "Y = jnp.array([data[1] for data in dataset])[:512]\n",
    "accuracy_preds(outs, Y)/512 * 100."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "272378"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "compute_num_params(params)\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## OOD METRICS"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "from sklearn.metrics import roc_auc_score, roc_curve\n",
    "import numpy as np\n",
    "\n",
    "def auroc(scores_id, scores_ood):\n",
    "    labels = np.zeros(len(scores_id) + len(scores_ood), dtype=\"int32\")\n",
    "    labels[len(scores_id) :] = 1\n",
    "    scores = np.concatenate([scores_id, scores_ood])\n",
    "    return roc_auc_score(labels, scores)\n",
    "\n",
    "def plot_roc(scores_id, scores_ood, label=None):\n",
    "    labels = np.zeros(len(scores_id) + len(scores_ood), dtype=\"int32\")\n",
    "    labels[len(scores_id) :] = 1\n",
    "    scores = np.concatenate([scores_id, scores_ood])\n",
    "    fpr, tpr, thresholds = roc_curve(labels, scores)\n",
    "    plt.plot(fpr, tpr, label=label)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Files already downloaded and verified\n"
     ]
    }
   ],
   "source": [
    "param_dict = pickle.load(open(\"../checkpoints/CIFAR-10/ResNet/epoch200_seed0_params.pickle\", \"rb\"))\n",
    "\n",
    "params = param_dict['params']\n",
    "batch_stats = param_dict['batch_stats']\n",
    "\n",
    "# lr_posterior_dict = pickle.load(open(\"../checkpoints/CIFAR-10/posterior_samples_deep_resnet_prec05_seed0_type_non-kernel-eigvals_params.pickle\", \"rb\"))\n",
    "# Good one:\n",
    "lr_posterior_dict = pickle.load(open(\"../checkpoints/CIFAR-10/old_samples/posterior_samples_just_in_case_seed0_type_non-kernel-eigvals_params.pickle\", \"rb\"))\n",
    "# lr_posterior_dict = pickle.load(open(\"../checkpoints/CIFAR-10/old_samples/posterior_samples_pray_seed0_type_non-kernel-eigvals_params.pickle\", \"rb\"))\n",
    "# lr_posterior_dict = pickle.load(open(\"../checkpoints/CIFAR-10/proj_posterior_samples_prec2_seed0_params.pickle\", \"rb\"))\n",
    "# lr_posterior_dict = pickle.load(open(\"../checkpoints/CIFAR-10/posterior_samples_pray1_seed0_type_non-kernel-eigvals_iter5000_steps3_samples1_prec0.5_params.pickle\", \"rb\"))\n",
    "lr_posterior = lr_posterior_dict['posterior_samples']\n",
    "\n",
    "posterior_dict = pickle.load(open(\"../checkpoints/CIFAR-10/old_samples/posterior_samples_baseline_1000_large_prec_seed0_type_full-ggn_params.pickle\", \"rb\"))\n",
    "posterior = posterior_dict['posterior_samples']\n",
    "\n",
    "output_dim = 10\n",
    "model = ResNet(\n",
    "            num_classes = output_dim,\n",
    "            c_hidden =(16, 32, 64),\n",
    "            num_blocks = (3, 3, 3),\n",
    "            act_fn = nn.relu,\n",
    "            block_class = ResNetBlock #PreActResNetBlock #\n",
    "        )\n",
    "\n",
    "n_samples_per_class = None\n",
    "cls=list(range(10))\n",
    "dataset = CIFAR10(path_root='/xxx/data', train=True, n_samples_per_class=n_samples_per_class, download=True, cls=cls, seed=0)\n",
    "\n",
    "data_array = jnp.array([data[0] for data in dataset])\n",
    "\n",
    "model_fn = lambda p, x: model.apply({'params': p, 'batch_stats': batch_stats},\n",
    "                                        x,\n",
    "                                        train=False,\n",
    "                                        mutable=False)\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Rotated CIFAR"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "metadata": {},
   "outputs": [],
   "source": [
    "# eval_args = {}\n",
    "# eval_args[\"linearised_laplace\"] = True\n",
    "# eval_args[\"posterior_sample_type\"] = \"Pytree\"\n",
    "# eval_args[\"likelihood\"] = \"classification\"\n",
    "\n",
    "# ids = [0, 15, 30, 60, 90, 120, 150, 180]\n",
    "# n_datapoint=500\n",
    "# ood_batch_size = 50\n",
    "# metrics_lr_sampled = []\n",
    "# for i, id in enumerate(ids):    \n",
    "#     _, test_loader = get_rotated_cifar_loaders(id, data_path=\"data\", download=True, batch_size=ood_batch_size, n_datapoint=n_datapoint)\n",
    "#     some_metrics, all_y_prob, all_y_true, all_y_var = evaluate_samples(test_loader, lr_posterior, params, model_fn, eval_args)\n",
    "#     if i == 0:\n",
    "#         all_y_prob_in = all_y_prob\n",
    "#     more_metrics = list(map(lambda x,y: compute_metrics(\n",
    "#             i, id, x, test_loader, y, all_y_var, benchmark=\"R-CIFAR\", supress=True\n",
    "#         ), all_y_prob, all_y_prob_in))\n",
    "#     more_metrics = {k: [dic[k] for dic in more_metrics] for k in more_metrics[0]}\n",
    "#     metrics_lr_sampled.append({**some_metrics, **more_metrics})\n",
    "#     print(\", \".join([f\"{k}: {jnp.mean(jnp.array(v)):.4f}\" for k, v in metrics_lr_sampled[-1].items()]))\n",
    "\n",
    "# metrics_lr_sampled = [{k: jnp.array(dic[k]).mean() for k in dic} for dic in metrics_lr_sampled]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {},
   "outputs": [],
   "source": [
    "params = pickle.load(open(\"../checkpoints/CIFAR-10/ResNet/good_params_seed0.pickle\", \"rb\"))['params']\n",
    "batch_stats = pickle.load(open(\"../checkpoints/CIFAR-10/ResNet/good_params_seed0.pickle\", \"rb\"))['batch_stats']\n",
    "model_fn = lambda p, x: model.apply({'params': p, 'batch_stats': batch_stats},\n",
    "                                        x,\n",
    "                                        train=False,\n",
    "                                        mutable=False)\n",
    "# lr_posterior_dict = pickle.load(open(\"../checkpoints/CIFAR-10/old_samples/posterior_samples_pray_seed0_type_non-kernel-eigvals_params.pickle\", \"rb\"))\n",
    "lr_posterior_dict = pickle.load(open(\"../checkpoints/CIFAR-10/posterior_samples_moresteps_1500_10_0_seed0_type_non-kernel-eigvals_iter1000_steps10_samples3_prec10.0_params.pickle\", \"rb\"))\n",
    "# lr_posterior_dict = pickle.load(open(\"../checkpoints/CIFAR-10/posterior_samples__1500_10_seed0_type_non-kernel-eigvals_iter1500_steps3_samples1_prec10.0_params.pickle\", \"rb\"))\n",
    "lr_posterior = lr_posterior_dict['posterior_samples']\n",
    "lr_posterior_list = [lr_posterior]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/xxx/geometric-laplace/src/ood_functions/metrics.py:90: RuntimeWarning: invalid value encountered in divide\n",
      "  ECE = np.average(np.abs(confs_bin - accs_bin), weights=nitems_bin / nitems_bin.sum())\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "conf: 0.9999, nll: 38.1479, acc: 0.0820, brier: 1.8359, ece: nan, mce: 0.0000\n",
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 1\n",
      "conf: 0.9998, nll: 37.4249, acc: 0.0820, brier: 1.8356, ece: 0.9700, mce: 0.9700\n",
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 2\n",
      "conf: 0.9999, nll: 37.6830, acc: 0.0820, brier: 1.8358, ece: nan, mce: 0.0000\n",
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 3\n",
      "conf: 0.9999, nll: 37.6901, acc: 0.0820, brier: 1.8358, ece: nan, mce: 0.0000\n",
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 4\n",
      "conf: 0.9999, nll: 37.8537, acc: 0.0820, brier: 1.8358, ece: 0.9895, mce: 0.9895\n",
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 5\n",
      "conf: 0.9999, nll: 37.2496, acc: 0.0820, brier: 1.8357, ece: 0.9883, mce: 0.9883\n",
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 6\n",
      "conf: 0.9999, nll: 37.2427, acc: 0.0820, brier: 1.8358, ece: nan, mce: 0.0000\n",
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 7\n",
      "conf: 0.9999, nll: 37.5478, acc: 0.0820, brier: 1.8358, ece: nan, mce: 0.0000\n"
     ]
    }
   ],
   "source": [
    "eval_args = {}\n",
    "eval_args[\"linearised_laplace\"] = False\n",
    "eval_args[\"posterior_sample_type\"] = \"Pytree\"\n",
    "eval_args[\"likelihood\"] = \"classification\"\n",
    "\n",
    "ids = [0, 15, 30, 60, 90, 120, 150, 180]#, 210, 240, 270, 300, 330, 345, 360]\n",
    "n_datapoint=500\n",
    "ood_batch_size = 50\n",
    "metrics_lr = []\n",
    "for seed, lr_posterior in enumerate(lr_posterior_list):\n",
    "    for i, id in enumerate(ids):    \n",
    "        _, test_loader = get_rotated_cifar_loaders(id, data_path=\"data\", download=True, batch_size=ood_batch_size, n_datapoint=n_datapoint)\n",
    "        some_metrics, all_y_prob, all_y_true, all_y_var = evaluate(test_loader, lr_posterior, params, model_fn, eval_args)\n",
    "        if i == 0:\n",
    "            all_y_prob_in = all_y_prob\n",
    "        more_metrics = compute_metrics(\n",
    "                i, id, all_y_prob, test_loader, all_y_prob_in, all_y_var, benchmark=\"R-CIFAR\"\n",
    "            )\n",
    "        metrics_lr.append({**some_metrics, **more_metrics})\n",
    "        print(\", \".join([f\"{k}: {v:.4f}\" for k, v in metrics_lr[-1].items()]))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 0\n",
      "conf: 0.9668, nll: 50.4464, acc: 0.1040, brier: 1.7366, ece: 0.7818, mce: 0.9546\n",
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 1\n",
      "conf: 0.9545, nll: 53.3963, acc: 0.1020, brier: 1.7295, ece: 0.7555, mce: 0.9748\n",
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 2\n",
      "conf: 0.9572, nll: 54.1502, acc: 0.0880, brier: 1.7670, ece: 0.7567, mce: 0.9753\n",
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 3\n",
      "conf: 0.9535, nll: 56.8677, acc: 0.0620, brier: 1.8002, ece: 0.8034, mce: 0.9543\n",
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 4\n",
      "conf: 0.9508, nll: 57.5300, acc: 0.0780, brier: 1.7743, ece: 0.7347, mce: 0.9861\n",
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 5\n",
      "conf: 0.9536, nll: 55.1770, acc: 0.0680, brier: 1.7875, ece: 0.7909, mce: 0.9746\n",
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 6\n",
      "conf: 0.9518, nll: 53.9369, acc: 0.1020, brier: 1.7310, ece: 0.7428, mce: 0.9855\n",
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 7\n",
      "conf: 0.9540, nll: 50.7933, acc: 0.1060, brier: 1.7235, ece: 0.7449, mce: 0.9660\n"
     ]
    }
   ],
   "source": [
    "eval_args = {}\n",
    "eval_args[\"linearised_laplace\"] = False\n",
    "eval_args[\"posterior_sample_type\"] = \"Pytree\"\n",
    "eval_args[\"likelihood\"] = \"classification\"\n",
    "\n",
    "ids = [0, 15, 30, 60, 90, 120, 150, 180]#, 210, 240, 270, 300, 330, 345, 360]\n",
    "n_datapoint=500\n",
    "ood_batch_size = 50\n",
    "metrics_posterior = []\n",
    "for i, id in enumerate(ids):    \n",
    "    _, test_loader = get_rotated_cifar_loaders(id, data_path=\"data\", download=True, batch_size=ood_batch_size, n_datapoint=n_datapoint)\n",
    "    some_metrics, all_y_prob, all_y_true, all_y_var = evaluate(test_loader, posterior, params, model_fn, eval_args)\n",
    "    if i == 0:\n",
    "        all_y_prob_in = all_y_prob\n",
    "    more_metrics = compute_metrics(\n",
    "            i, id, all_y_prob, test_loader, all_y_prob_in, all_y_var, benchmark=\"R-CIFAR\"\n",
    "        )\n",
    "    metrics_posterior.append({**some_metrics, **more_metrics})\n",
    "    print(\", \".join([f\"{k}: {v:.4f}\" for k, v in metrics_posterior[-1].items()]))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 0\n",
      "conf: 0.9798, nll: 109.0327, acc: 0.0460, brier: 1.8791, ece: 0.8192, mce: 0.9860\n",
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 1\n",
      "conf: 0.9831, nll: 117.2469, acc: 0.0440, brier: 1.8823, ece: 0.8469, mce: 0.9866\n",
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 2\n",
      "conf: 0.9789, nll: 117.7819, acc: 0.0700, brier: 1.8274, ece: 0.8263, mce: 0.9862\n",
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 3\n",
      "conf: 0.9832, nll: 114.2189, acc: 0.0900, brier: 1.7978, ece: 0.7428, mce: 0.9766\n",
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 4\n",
      "conf: 0.9757, nll: 79.9972, acc: 0.0920, brier: 1.7834, ece: 0.7673, mce: 0.9761\n",
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 5\n",
      "conf: 0.9811, nll: 105.2350, acc: 0.0900, brier: 1.7934, ece: 0.7700, mce: 0.9859\n",
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 6\n",
      "conf: 0.9829, nll: 114.7986, acc: 0.0780, brier: 1.8206, ece: 0.7626, mce: 0.9846\n",
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 7\n",
      "conf: 0.9787, nll: 85.7438, acc: 0.1020, brier: 1.7654, ece: 0.7814, mce: 0.9637\n"
     ]
    }
   ],
   "source": [
    "eval_args = {}\n",
    "eval_args[\"linearised_laplace\"] = True\n",
    "eval_args[\"posterior_sample_type\"] = \"Pytree\"\n",
    "eval_args[\"likelihood\"] = \"classification\"\n",
    "\n",
    "ids = [0, 15, 30, 60, 90, 120, 150, 180]#, 210, 240, 270, 300, 330, 345, 360]\n",
    "n_datapoint=500\n",
    "ood_batch_size = 50\n",
    "metrics_lienarised = []\n",
    "for i, id in enumerate(ids):    \n",
    "    _, test_loader = get_rotated_cifar_loaders(id, data_path=\"data\", download=True, batch_size=ood_batch_size, n_datapoint=n_datapoint)\n",
    "    some_metrics, all_y_prob, all_y_true, all_y_var = evaluate(test_loader, posterior, params, model_fn, eval_args)\n",
    "    if i == 0:\n",
    "        all_y_prob_in = all_y_prob\n",
    "    more_metrics = compute_metrics(\n",
    "            i, id, all_y_prob, test_loader, all_y_prob_in, all_y_var, benchmark=\"R-CIFAR\"\n",
    "        )\n",
    "    metrics_lienarised.append({**some_metrics, **more_metrics})\n",
    "    print(\", \".join([f\"{k}: {v:.4f}\" for k, v in metrics_lienarised[-1].items()]))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 0\n",
      "conf: 0.9566, nll: 0.3032, acc: 0.9200, brier: 0.1339, ece: 0.2796, mce: 0.9388\n",
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 1\n",
      "conf: 0.8868, nll: 1.0825, acc: 0.7200, brier: 0.4236, ece: 0.2718, mce: 0.6794\n",
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 2\n",
      "conf: 0.8172, nll: 3.2163, acc: 0.4280, brier: 0.9309, ece: 0.4357, mce: 0.8258\n",
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 3\n",
      "conf: 0.8256, nll: 5.5042, acc: 0.2620, brier: 1.2518, ece: 0.5586, mce: 0.8845\n",
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 4\n",
      "conf: 0.8071, nll: 3.8256, acc: 0.3580, brier: 1.0416, ece: 0.4664, mce: 0.8408\n",
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 5\n",
      "conf: 0.8102, nll: 6.4565, acc: 0.2080, brier: 1.3261, ece: 0.5837, mce: 0.9761\n",
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 6\n",
      "conf: 0.8285, nll: 5.4965, acc: 0.2620, brier: 1.2526, ece: 0.5527, mce: 0.9452\n",
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 7\n",
      "conf: 0.8229, nll: 3.8181, acc: 0.3900, brier: 1.0146, ece: 0.4373, mce: 0.8721\n"
     ]
    }
   ],
   "source": [
    "eval_args = {}\n",
    "eval_args[\"linearised_laplace\"] = False\n",
    "eval_args[\"posterior_sample_type\"] = \"Pytree\"\n",
    "eval_args[\"likelihood\"] = \"classification\"\n",
    "\n",
    "ids = [0, 15, 30, 60, 90, 120, 150, 180]#, 210, 240, 270, 300, 330, 345, 360]\n",
    "n_datapoint=500\n",
    "ood_batch_size = 50\n",
    "metrics_map = []\n",
    "for i, id in enumerate(ids):    \n",
    "    _, test_loader = get_rotated_cifar_loaders(id, data_path=\"data\", download=True, batch_size=ood_batch_size, n_datapoint=n_datapoint)\n",
    "    some_metrics, all_y_prob, all_y_true, all_y_var = evaluate_map(test_loader, params, model_fn, eval_args)\n",
    "    if i == 0:\n",
    "        all_y_prob_in = all_y_prob\n",
    "    more_metrics = compute_metrics(\n",
    "            i, id, all_y_prob, test_loader, all_y_prob_in, all_y_var, benchmark=\"R-CIFAR\"\n",
    "        )\n",
    "    metrics_map.append({**some_metrics, **more_metrics})\n",
    "    print(\", \".join([f\"{k}: {v:.4f}\" for k, v in metrics_map[-1].items()]))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 0\n",
      "conf: 0.8229, nll: 3.8181, acc: 0.3900, brier: 1.0146, ece: 0.4373, mce: 0.8721\n",
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 1\n",
      "conf: 0.8229, nll: 3.8181, acc: 0.3900, brier: 1.0146, ece: 0.4373, mce: 0.8721\n",
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 2\n",
      "conf: 0.8229, nll: 3.8181, acc: 0.3900, brier: 1.0146, ece: 0.4373, mce: 0.8721\n",
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 3\n",
      "conf: 0.8229, nll: 3.8181, acc: 0.3900, brier: 1.0146, ece: 0.4373, mce: 0.8721\n",
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 4\n",
      "conf: 0.8229, nll: 3.8181, acc: 0.3900, brier: 1.0146, ece: 0.4373, mce: 0.8721\n",
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 5\n",
      "conf: 0.8229, nll: 3.8181, acc: 0.3900, brier: 1.0146, ece: 0.4373, mce: 0.8721\n",
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 6\n",
      "conf: 0.8229, nll: 3.8181, acc: 0.3900, brier: 1.0146, ece: 0.4373, mce: 0.8721\n",
      "Files already downloaded and verified\n",
      "R-CIFAR with distribution shift intensity 7\n",
      "conf: 0.8229, nll: 3.8181, acc: 0.3900, brier: 1.0146, ece: 0.4373, mce: 0.8721\n"
     ]
    }
   ],
   "source": [
    "class random_classifier():\n",
    "    def __init__(self, key, num_classes):\n",
    "        self.n_classes = num_classes\n",
    "        self.key = key\n",
    "    \n",
    "    def apply(self, param, x):\n",
    "        self.key, _ = jax.random.split(self.key)\n",
    "        pred = jax.random.randint(self.key, shape=(x.shape[0],), minval=0, maxval=self.n_classes)\n",
    "        one_hot = jax.nn.one_hot(pred, self.n_classes)\n",
    "        return one_hot\n",
    "    \n",
    "key = jax.random.PRNGKey(5000)\n",
    "n_classes = 10\n",
    "rndm_classifier = random_classifier(key, n_classes)\n",
    "rand_fn = rndm_classifier.apply\n",
    "eval_args = {}\n",
    "eval_args[\"linearised_laplace\"] = False\n",
    "eval_args[\"posterior_sample_type\"] = \"Pytree\"\n",
    "eval_args[\"likelihood\"] = \"classification\"\n",
    "\n",
    "ids = [0, 15, 30, 60, 90, 120, 150, 180]#, 210, 240, 270, 300, 330, 345, 360]\n",
    "n_datapoint=500\n",
    "ood_batch_size = 50\n",
    "metrics_random = []\n",
    "for i, id in enumerate(ids):    \n",
    "    _, test_loader = get_rotated_cifar_loaders(id, data_path=\"data\", download=True, batch_size=ood_batch_size, n_datapoint=n_datapoint)\n",
    "    some_metrics, all_y_prob, all_y_true, all_y_var = evaluate_map(test_loader, params, rand_fn, eval_args)\n",
    "    if i == 0:\n",
    "        all_y_prob_in = all_y_prob\n",
    "    more_metrics = compute_metrics(\n",
    "            i, id, all_y_prob, test_loader, all_y_prob_in, all_y_var, benchmark=\"R-CIFAR\"\n",
    "        )\n",
    "    metrics_random.append({**some_metrics, **more_metrics})\n",
    "    print(\", \".join([f\"{k}: {v:.4f}\" for k, v in metrics_map[-1].items()]))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.legend.Legend at 0x7f5e3b61d310>"
      ]
     },
     "execution_count": 29,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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fHy8oqOjlZaW1qYDMhJkAACW4ufnp5deekn//ve/9frrr+uTTz5x+9CXzD4kv/3tb/XGG29oy5Ytys/P1913393kNouKijR16lRt3rxZn376qfr06aNbbrlFRUVFkiSn06kJEyZoy5YtevPNN7V3714988wzrkuGDx8+rPHjx2vy5MnatWuX3n77bW3evFkzZ8684P2LiYnRlClT9Pe//13V1dWN5q9cuVIzZsxQamqqsrOztXLlSo9lzfHSSy/p73//u9555x1lZWXprbfeUvfu3T0uW1JSonHjxumyyy7T9u3b9e6772rt2rWN9nH9+vU6fPiw1q9fr9dff13Lli1zBae2wKklALhEGIahs5WNPxjbWmigf6tePVW/02v37t31m9/8Rg8++KBefvllV3llZaUWLVqkESNGSDL7lPTr10+fffaZhg8f3mibN910k9vjP/3pT4qMjNTGjRv1/e9/X2vXrtVnn32mffv26YorrpAk9ezZ07V8enq6pkyZ4qpbnz599NJLL+mGG27QK6+8opCQkAvax759+6qoqEinTp1STEyM27yoqCiFhYUpKCjI7dSPp7LzOXbsmPr06aPrrrtONptN3bp1a3LZ5cuXq6ysTG+88YY6deokSVq0aJFuvfVW/f73v1dsbKwk6bLLLtOiRYvk7++vvn37auLEiVq3bp1mzJhxIX+CZiPIAMAl4mxlta58ck27P+/ep8cpLKj1Pm7Wrl2r9PR07d+/X4WFhaqqqlJZWZlKS0td45IEBARo2LBhrnX69u2ryMhI7du3z2OQyc3N1RNPPKENGzYoLy9P1dXVKi0t1bFjxyRJmZmZ6tq1qyvENPTll19q165deuutt1xlhmHI6XTqyJEj6tev3wXto2EYktr+8vlp06bp5ptvVnJyssaPH6/vf//7Gjt2rMdl9+3bp4EDB7pCjCRde+21cjqdysrKcgWZq666ym1wu/j4eO3evbvN9sGrp5bS09M1bNgwRUREKCYmRpMmTVJWVpbbMmVlZUpLS1N0dLTCw8M1efJk5ebmeqnGAABvOnr0qL7//e8rJSVFf/vb37Rz504tXrxYktnfo6WmTp2qzMxMvfjii9q6dasyMzMVHR3t2mbtKLRNKS4u1k9/+lNlZma6pi+//FIHDx5Ur169Lrg++/btk91uv+hRmW02mysU1arfX+Xqq6/WkSNH9Otf/1pnz57VnXfeqTvuuOOinrP+1VW1dXA6nRe1zXPxaovMxo0blZaWpmHDhqmqqkq//OUvNXbsWO3du9eV+B599FF98MEHevfdd+VwODRz5kzdfvvt2rJlizerDgCWExror71Pj/PK87aWnTt3yul06g9/+IP8/Mzv4u+8806j5aqqqrRjxw5X60tWVpby8/ObbBnZsmWLXn75Zd1yyy2SpOPHj+vkyZOu+SkpKfrmm2904MABj60yV199tfbu3avevXtf9D7m5eVp+fLlmjRpkmsfW6pLly7Kzs52PT548KBKS0vdlrHb7brrrrt011136Y477tD48eN1+vTpRldK9evXT8uWLVNJSYnrM3rLli3y8/NTcnLyRdXzYng1yHz00Uduj5ctW6aYmBjt3LlTI0eOVEFBgZYsWaLly5e7zl8uXbpU/fr106effqprrrnGG9UGAEuy2WyteoqnLRUUFCgzM9OtLDo6Wr1791ZlZaX++Mc/6tZbb9WWLVv06quvNlo/MDBQs2bN0ksvvaSAgADNnDlT11xzjcfTSpLZp+V///d/NXToUBUWFurnP/+5WyvMDTfcoJEjR2ry5MlauHChevfurf3798tms2n8+PGaO3eurrnmGs2cOVM/+clP1KlTJ+3du1cff/yxFi1a1OR+1v5Cee3l1xkZGfrd734nh8OhZ555pmV/vHpuuukmLVq0SKmpqaqurtbcuXPdWkwWLlyo+Ph4DR48WH5+fnr33XcVFxenyMjIRtuaMmWKnnrqKU2dOlXz58/Xd999p1mzZuk///M/XaeVvMGnrloqKCiQVHe9/M6dO1VZWakxY8a4lunbt6+SkpKUkZHhcRvl5eUqLCx0mwAA1rJhwwYNHjzYbVqwYIEGDhyohQsX6ve//7369++vt956y+1y4VphYWGaO3eu7rnnHl177bUKDw/X22+/3eTzLVmyRGfOnNHVV1+t//zP/9QjjzzSqJPt3/72Nw0bNkw/+tGPdOWVV+rxxx93XVWUkpKijRs36sCBA7r++us1ePBgPfnkk65xYJpSWFio+Ph4XX755UpNTdVrr72mqVOn6osvvrioX4Su9Yc//EGJiYm6/vrrdc899+ixxx5z+/2tiIgIPfvssxo6dKiGDRumo0eP6p///KfHlqCwsDCtWbNGp0+f1rBhw3THHXdo9OjR5wxq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tar09HQNGzZMERERiomJ0aRJk5SVleW2zKhRo2Sz2dymBx980Es1bj3z589vtF99+/Z1zS8rK1NaWpqio6MVHh6uyZMnKzc314s1bj3du3dvtO82m01paWmSOs4x37Rpk2699VYlJCTIZrNp9erVbvMNw9CTTz6p+Ph4hYaGasyYMTp48KDbMqdPn9aUKVNkt9sVGRmp6dOnq7i4uB33omXOte+VlZWaO3euBgwYoE6dOikhIUH33XefTpw44bYNT6+TZ555pp335MKd77hPmzat0X6NHz/ebRkrHvfz7ben97zNZtNzzz3nWsaXjzlBpgXefvttzZkzR0899ZQ+//xzDRw4UOPGjVNeXp63q9ZqNm7cqLS0NH366af6+OOPVVlZqbFjx6qkpMRtuRkzZig7O9s1Pfvss16qceu66qqr3PZr8+bNrnmPPvqo/vGPf+jdd9/Vxo0bdeLECd1+++1erG3r2b59u9t+f/zxx5Kk//iP/3At0xGOeUlJiQYOHKjFixd7nP/ss8/qpZde0quvvqpt27apU6dOGjdunMrKylzLTJkyRf/+97/18ccf6/3339emTZv0wAMPtNcutNi59r20tFSff/65fvWrX+nzzz/XypUrlZWVpR/84AeNln366afdXgezZs1qj+pflPMdd0kaP36823799a9/dZtvxeN+vv2uv7/Z2dn6y1/+IpvNpsmTJ7st57PH3MAFGz58uJGWluZ6XF1dbSQkJBjp6elerFXbysvLMyQZGzdudJXdcMMNxs9+9jPvVaqNPPXUU8bAgQM9zsvPzzcCAwONd99911W2b98+Q5KRkZHRTjVsPz/72c+MXr16GU6n0zCMjnnMJRmrVq1yPXY6nUZcXJzx3HPPucry8/ON4OBg469//athGIaxd+9eQ5Kxfft21zIffvihYbPZjG+//bbd6n6xGu67J5999pkhyfj6669dZd26dTOef/75tq1cG/O071OnTjVuu+22JtfpCMe9Ocf8tttuM2666Sa3Ml8+5rTIXKCKigrt3LlTY8aMcZX5+flpzJgxysjI8GLN2lZBQYEkKSoqyq38rbfeUufOndW/f3/NmzdPpaWl3qheqzt48KASEhLUs2dPTZkyRceOHZMk7dy5U5WVlW7Hv2/fvkpKSupwx7+iokJvvvmmfvzjH7v9TllHPea1jhw5opycHLdj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2WSz2XT06FFVV1dr+vTp6tGjh0JDQ5WcnKwXX3zR9Tzz58/X66+/rvfee8+13oYNGzyeWjrX89fW+ZFHHtHjjz+uqKgoxcXFaf78+a75hmFo/vz5SkpKUnBwsBISEvTII4+0+d8SQPPx1QVAq3n99df10EMPacuWLcrJydEtt9yiadOm6Y033tD+/fs1Y8YMhYSEaP78+XrxxRd14MAB9e/fX08//bQkqUuXLnI6nerataveffddRUdHa+vWrXrggQcUHx+vO++8U4899pj27dunwsJCLV26VJIUFRWlEydOuNXl22+/Pefz16/znDlztG3bNmVkZGjatGm69tprdfPNN+tvf/ubnn/+ea1YsUJXXXWVcnJy9OWXX7bb3xPA+RFkALSaPn366Nlnn5UkvfHGG0pMTNSiRYtks9nUt29fnThxQnPnztWTTz4ph8OhoKAghYWFKS4uzrUNf39/LViwwPW4R48eysjI0DvvvKM777xT4eHhCg0NVXl5udt6Db388svnfH4/P7NBOiUlRU899ZSr/osWLdK6det0880369ixY4qLi9OYMWMUGBiopKQkDR8+vC3+dABaiFNLAFrNkCFDXPf37dun1NRU2Ww2V9m1116r4uJiffPNN+fczuLFizVkyBB16dJF4eHh+tOf/qRjx45dUF2a+/wpKSlu68XHxysvL0+S9B//8R86e/asevbsqRkzZmjVqlVup6YAeB9BBkCr6dSp00VvY8WKFXrsscc0ffp0/d///Z8yMzN1//33t1nn4cDAQLfHNptNTqdTknklVFZWll5++WWFhobq4Ycf1siRI1VZWdkmdQFw4QgyANpEv379lJGRIcMwXGVbtmxRRESEunbtKkkKCgpSdXW123pbtmzR9773PT388MMaPHiwevfurcOHD7st42m9ljx/c4SGhurWW2/VSy+9pA0bNigjI0O7d+9u9voA2hZBBkCbePjhh3X8+HHNmjVL+/fv13vvvaennnpKc+bMcfVP6d69u7Zt26ajR4/q5MmTcjqd6tOnj3bs2KE1a9bowIED+tWvfqXt27e7bbt79+7atWuXsrKydPLkSY8tJM15/vNZtmyZlixZoj179uirr77Sm2++qdDQUHXr1u3i/0AAWgVBBkCbuPzyy/XPf/5Tn332mQYOHKgHH3xQ06dP1xNPPOFa5rHHHpO/v7+uvPJKdenSRceOHdNPf/pT3X777brrrrs0YsQInTp1Sg8//LDbtmfMmKHk5GQNHTpUXbp00ZYtW1r0/OcTGRmpP//5z7r22muVkpKitWvX6h//+Ieio6Nb/ocB0KpsRv12VwAAAAuhRQYAAFgWQQYAAFgWQQYAAFgWQQYAAFgWQQYAAFgWQQYAAFgWQQYAAFgWQQYAAFgWQQYAAFgWQQYAAFgWQQYAAFgWQQYAAFjW/weSN3Yy6LVwtAAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "uq_metric = 'nll'\n",
    "plt.plot(ids, [lr[uq_metric] for lr in metrics_lr], label=\"Laplace Diffusion\")\n",
    "# plt.plot(ids, [post[uq_metric] for post in metrics_posterior], label=\"Sampled Laplace\")\n",
    "# plt.plot(ids, [lin[uq_metric] for lin in metrics_lienarised], label=\"Linearised Laplace\")\n",
    "plt.plot(ids, [map[uq_metric] for map in metrics_map], label=\"MAP\")\n",
    "# plt.plot(ids, [rand[uq_metric]/n_datapoint for rand in metrics_random], label=\"Random Classifier\")\n",
    "\n",
    "# plt.title(uq_metric)\n",
    "plt.xlabel(\"rotations\")\n",
    "plt.ylabel(uq_metric)\n",
    "plt.legend()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.legend.Legend at 0x7f5e39fa4f40>"
      ]
     },
     "execution_count": 25,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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6dWrl+Xrtm3bEhQURP369fnuu+9wcHDQMFnFmj9/Pr169cLX19fSVlU/c3GlwsJCBgwYgKqqfPbZZ8VeCw8Pt3zdtm1b7OzseOqpp4iIiKh0SyhcbuDAgZav27RpQ9u2bWncuDFRUVH06NFDw2QVZ8GCBQwePBh7e/ti7db+mcstsArg4eGBXq+/4qmf5ORkvL29NUpVfsaOHcuqVatYv349devWvW7foKAgAI4ePVoR0SqMm5sbzZo14+jRo3h7e1NQUEB6enqxPlXt84+Pj2fdunWMHDnyuv2q4md+6XO83t9xb2/vKx56KCoqIjU1tUr8HFwqfuLj4/njjz+KXf25mqCgIIqKioiLi6uYgBWkUaNGeHh4WH6+q/rnvnHjRmJjY2/49x6s7zOXAqgC2NnZERAQQGRkpKXNZDIRGRlJ586dNUxWtlRVZezYsfz000/8+eefNGzY8Ib77N69GwAfH59yTlexsrOzOXbsGD4+PgQEBGBra1vs84+NjSUhIaFKff5ffvklnp6e9OnT57r9quJn3rBhQ7y9vYt9xpmZmWzbts3yGXfu3Jn09HRiYmIsff78809MJpOlKKysLhU/R44cYd26ddSqVeuG++zevRudTnfF7aHK7tSpU5w/f97y812VP3cwX/UNCAigXbt2N+xrdZ+51qOwq4tvv/1WNRgM6sKFC9UDBw6oTz75pOrm5qYmJSVpHa3MjB49WnV1dVWjoqLUxMREy5abm6uqqqoePXpUnT59uhodHa2eOHFCXblypdqoUSO1W7duGicvvRdeeEGNiopST5w4oW7evFkNDg5WPTw81JSUFFVVVfXpp59W69Wrp/75559qdHS02rlzZ7Vz584apy47RqNRrVevnjpp0qRi7VXpM8/KylJ37dql7tq1SwXUmTNnqrt27bI86fTOO++obm5u6sqVK9U9e/ao/fr1Uxs2bKheuHDBcozQ0FC1ffv26rZt29RNmzapTZs2VQcNGqTVKd206517QUGB2rdvX7Vu3brq7t27i/3dz8/PV1VVVbds2aJ++OGH6u7du9Vjx46pS5YsUWvXrq0OGTJE4zO7seude1ZWljphwgR169at6okTJ9R169apHTp0UJs2barm5eVZjlEZP/cb/byrqqpmZGSojo6O6meffXbF/pXhM5cCqAJ98sknar169VQ7Ozu1U6dO6t9/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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "uq_metric = 'ece'\n",
    "plt.plot(ids, [lr[uq_metric] for lr in metrics_lr], label=\"Laplace Diffusion\")\n",
    "# plt.plot(ids, [post[uq_metric] for post in metrics_posterior], label=\"Sampled Laplace\")\n",
    "# plt.plot(ids, [lin[uq_metric] for lin in metrics_lienarised], label=\"Linearised Laplace\")\n",
    "plt.plot(ids, [map[uq_metric] for map in metrics_map], label=\"MAP\")\n",
    "# plt.plot(ids, [rand[uq_metric]/n_datapoint for rand in metrics_random], label=\"Random Classifier\")\n",
    "\n",
    "# plt.title(uq_metric)\n",
    "plt.xlabel(\"rotations\")\n",
    "plt.ylabel(uq_metric)\n",
    "plt.legend()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_229173/488607537.py:1: DeprecationWarning: \n",
      "Pyarrow will become a required dependency of pandas in the next major release of pandas (pandas 3.0),\n",
      "(to allow more performant data types, such as the Arrow string type, and better interoperability with other libraries)\n",
      "but was not found to be installed on your system.\n",
      "If this would cause problems for you,\n",
      "please provide us feedback at https://github.com/pandas-dev/pandas/issues/54466\n",
      "        \n",
      "  import pandas as pd\n"
     ]
    },
    {
     "ename": "NameError",
     "evalue": "name 'metrics_posterior' is not defined",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mNameError\u001b[0m                                 Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[11], line 4\u001b[0m\n\u001b[1;32m      2\u001b[0m columns \u001b[38;5;241m=\u001b[39m [\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mconf\u001b[39m\u001b[38;5;124m'\u001b[39m, \u001b[38;5;124m'\u001b[39m\u001b[38;5;124mnll\u001b[39m\u001b[38;5;124m'\u001b[39m, \u001b[38;5;124m'\u001b[39m\u001b[38;5;124macc\u001b[39m\u001b[38;5;124m'\u001b[39m, \u001b[38;5;124m'\u001b[39m\u001b[38;5;124mbrier\u001b[39m\u001b[38;5;124m'\u001b[39m, \u001b[38;5;124m'\u001b[39m\u001b[38;5;124mece\u001b[39m\u001b[38;5;124m'\u001b[39m, \u001b[38;5;124m'\u001b[39m\u001b[38;5;124mmce\u001b[39m\u001b[38;5;124m'\u001b[39m]\n\u001b[1;32m      3\u001b[0m method_list \u001b[38;5;241m=\u001b[39m [\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mLaplace Diffusion\u001b[39m\u001b[38;5;124m\"\u001b[39m, \u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mSampled Laplace\u001b[39m\u001b[38;5;124m\"\u001b[39m, \u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mLinearised Laplace\u001b[39m\u001b[38;5;124m\"\u001b[39m, \u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mMAP\u001b[39m\u001b[38;5;124m\"\u001b[39m]\n\u001b[0;32m----> 4\u001b[0m method_dict \u001b[38;5;241m=\u001b[39m {\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mProjected Laplace\u001b[39m\u001b[38;5;124m\"\u001b[39m:metrics_lr, \u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mSampled Laplace\u001b[39m\u001b[38;5;124m\"\u001b[39m:\u001b[43mmetrics_posterior\u001b[49m, \u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mLinearised Laplace\u001b[39m\u001b[38;5;124m\"\u001b[39m:metrics_lienarised, \u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mMAP\u001b[39m\u001b[38;5;124m\"\u001b[39m:metrics_map}\n\u001b[1;32m      5\u001b[0m df_data \u001b[38;5;241m=\u001b[39m {k: [method_dict[k][\u001b[38;5;241m0\u001b[39m][dic] \u001b[38;5;28;01mfor\u001b[39;00m dic \u001b[38;5;129;01min\u001b[39;00m method_dict[k][\u001b[38;5;241m0\u001b[39m]] \u001b[38;5;28;01mfor\u001b[39;00m k \u001b[38;5;129;01min\u001b[39;00m method_dict}\n\u001b[1;32m      6\u001b[0m df \u001b[38;5;241m=\u001b[39m pd\u001b[38;5;241m.\u001b[39mDataFrame\u001b[38;5;241m.\u001b[39mfrom_dict(df_data, orient\u001b[38;5;241m=\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mindex\u001b[39m\u001b[38;5;124m'\u001b[39m,\n\u001b[1;32m      7\u001b[0m                        columns\u001b[38;5;241m=\u001b[39mcolumns)\n",
      "\u001b[0;31mNameError\u001b[0m: name 'metrics_posterior' is not defined"
     ]
    }
   ],
   "source": [
    "import pandas as pd\n",
    "columns = ['conf', 'nll', 'acc', 'brier', 'ece', 'mce']\n",
    "method_list = [\"Laplace Diffusion\", \"Sampled Laplace\", \"Linearised Laplace\", \"MAP\"]\n",
    "method_dict = {\"Projected Laplace\":metrics_lr, \"Sampled Laplace\":metrics_posterior, \"Linearised Laplace\":metrics_lienarised, \"MAP\":metrics_map}\n",
    "df_data = {k: [method_dict[k][0][dic] for dic in method_dict[k][0]] for k in method_dict}\n",
    "df = pd.DataFrame.from_dict(df_data, orient='index',\n",
    "                       columns=columns)\n",
    "print(df.to_latex(index=True,\n",
    "                  formatters={\"name\": str.upper},\n",
    "                  float_format=\"{:.3f}\".format,\n",
    "))  \n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>conf</th>\n",
       "      <th>nll</th>\n",
       "      <th>acc</th>\n",
       "      <th>brier</th>\n",
       "      <th>ece</th>\n",
       "      <th>mce</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>Projected Laplace</th>\n",
       "      <td>0.939361</td>\n",
       "      <td>140.97946</td>\n",
       "      <td>0.910</td>\n",
       "      <td>0.134793</td>\n",
       "      <td>0.187769</td>\n",
       "      <td>0.813319</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Sampled Laplace</th>\n",
       "      <td>0.865207</td>\n",
       "      <td>338.75055</td>\n",
       "      <td>0.792</td>\n",
       "      <td>0.308853</td>\n",
       "      <td>0.153966</td>\n",
       "      <td>0.667497</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Linearised Laplace</th>\n",
       "      <td>0.952717</td>\n",
       "      <td>243.31937</td>\n",
       "      <td>0.878</td>\n",
       "      <td>0.193803</td>\n",
       "      <td>0.304401</td>\n",
       "      <td>0.843402</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>MAP</th>\n",
       "      <td>0.958564</td>\n",
       "      <td>120.65207</td>\n",
       "      <td>0.932</td>\n",
       "      <td>0.104418</td>\n",
       "      <td>0.207458</td>\n",
       "      <td>0.858538</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "                        conf        nll    acc     brier       ece       mce\n",
       "Projected Laplace   0.939361  140.97946  0.910  0.134793  0.187769  0.813319\n",
       "Sampled Laplace     0.865207  338.75055  0.792  0.308853  0.153966  0.667497\n",
       "Linearised Laplace  0.952717  243.31937  0.878  0.193803  0.304401  0.843402\n",
       "MAP                 0.958564  120.65207  0.932  0.104418  0.207458  0.858538"
      ]
     },
     "execution_count": 14,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "df"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### CIFAR-OOD"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "lr_posterior_dict = pickle.load(open(\"../checkpoints/CIFAR-10/old_samples/posterior_samples_posterior_resnet_500iters_prec10_seed0_type_non-kernel-eigvals_params.pickle\", \"rb\"))\n",
    "lr_posterior = lr_posterior_dict['posterior_samples']\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "lr_posterior_dict = pickle.load(open(\"../checkpoints/CIFAR-10/old_samples/posterior_samples_pray_seed0_type_non-kernel-eigvals_params.pickle\", \"rb\"))\n",
    "\n",
    "lr_posterior = lr_posterior_dict['posterior_samples']\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "ename": "FileNotFoundError",
     "evalue": "[Errno 2] No such file or directory: '../checkpoints/CIFAR-10/proj_posterior_samples_prec2_seed0_params.pickle'",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mFileNotFoundError\u001b[0m                         Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[10], line 1\u001b[0m\n\u001b[0;32m----> 1\u001b[0m lr_posterior_dict \u001b[38;5;241m=\u001b[39m pickle\u001b[38;5;241m.\u001b[39mload(\u001b[38;5;28;43mopen\u001b[39;49m\u001b[43m(\u001b[49m\u001b[38;5;124;43m\"\u001b[39;49m\u001b[38;5;124;43m../checkpoints/CIFAR-10/proj_posterior_samples_prec2_seed0_params.pickle\u001b[39;49m\u001b[38;5;124;43m\"\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;124;43m\"\u001b[39;49m\u001b[38;5;124;43mrb\u001b[39;49m\u001b[38;5;124;43m\"\u001b[39;49m\u001b[43m)\u001b[49m)\n\u001b[1;32m      2\u001b[0m \u001b[38;5;66;03m# lr_posterior_dict = pickle.load(open(\"../checkpoints/CIFAR-10/proj_posterior_samples_500pts_seed0_params.pickle\", \"rb\"))\u001b[39;00m\n\u001b[1;32m      3\u001b[0m lr_posterior \u001b[38;5;241m=\u001b[39m lr_posterior_dict[\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mposterior_samples\u001b[39m\u001b[38;5;124m'\u001b[39m]\n",
      "File \u001b[0;32m/xxx/geometric-laplace/geom/lib64/python3.9/site-packages/IPython/core/interactiveshell.py:310\u001b[0m, in \u001b[0;36m_modified_open\u001b[0;34m(file, *args, **kwargs)\u001b[0m\n\u001b[1;32m    303\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m file \u001b[38;5;129;01min\u001b[39;00m {\u001b[38;5;241m0\u001b[39m, \u001b[38;5;241m1\u001b[39m, \u001b[38;5;241m2\u001b[39m}:\n\u001b[1;32m    304\u001b[0m     \u001b[38;5;28;01mraise\u001b[39;00m \u001b[38;5;167;01mValueError\u001b[39;00m(\n\u001b[1;32m    305\u001b[0m         \u001b[38;5;124mf\u001b[39m\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mIPython won\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mt let you open fd=\u001b[39m\u001b[38;5;132;01m{\u001b[39;00mfile\u001b[38;5;132;01m}\u001b[39;00m\u001b[38;5;124m by default \u001b[39m\u001b[38;5;124m\"\u001b[39m\n\u001b[1;32m    306\u001b[0m         \u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mas it is likely to crash IPython. If you know what you are doing, \u001b[39m\u001b[38;5;124m\"\u001b[39m\n\u001b[1;32m    307\u001b[0m         \u001b[38;5;124m\"\u001b[39m\u001b[38;5;124myou can use builtins\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;124m open.\u001b[39m\u001b[38;5;124m\"\u001b[39m\n\u001b[1;32m    308\u001b[0m     )\n\u001b[0;32m--> 310\u001b[0m \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[43mio_open\u001b[49m\u001b[43m(\u001b[49m\u001b[43mfile\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43margs\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43mkwargs\u001b[49m\u001b[43m)\u001b[49m\n",
      "\u001b[0;31mFileNotFoundError\u001b[0m: [Errno 2] No such file or directory: '../checkpoints/CIFAR-10/proj_posterior_samples_prec2_seed0_params.pickle'"
     ]
    }
   ],
   "source": [
    "lr_posterior_dict = pickle.load(open(\"../checkpoints/CIFAR-10/proj_posterior_samples_prec2_seed0_params.pickle\", \"rb\"))\n",
    "# lr_posterior_dict = pickle.load(open(\"../checkpoints/CIFAR-10/proj_posterior_samples_500pts_seed0_params.pickle\", \"rb\"))\n",
    "lr_posterior = lr_posterior_dict['posterior_samples']\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [],
   "source": [
    "# lr_posterior_dict = pickle.load(open(\"../checkpoints/CIFAR-10/posterior_samples_seed0_seed0_type_non-kernel-eigvals_iter2500_steps3_samples5_prec1.0_params.pickle\", \"rb\"))\n",
    "# lr_posterior_dict = pickle.load(open(\"../checkpoints/CIFAR-10/posterior_samples_seed1_seed1_type_non-kernel-eigvals_iter2500_steps3_samples5_prec1.0_params.pickle\", \"rb\"))\n",
    "lr_posterior_dict = pickle.load(open(\"../checkpoints/CIFAR-10/posterior_samples_seed2_seed2_type_non-kernel-eigvals_iter1500_steps3_samples5_prec1.0_params.pickle\", \"rb\"))\n",
    "# lr_posterior_dict = pickle.load(open(\"../checkpoints/CIFAR-10/posterior_samples_seed3_seed3_type_non-kernel-eigvals_iter2500_steps3_samples5_prec1.0_params.pickle\", \"rb\"))\n",
    "# lr_posterior_dict = pickle.load(open(\"../checkpoints/CIFAR-10/posterior_samples_seed4_seed4_type_non-kernel-eigvals_iter2500_steps3_samples5_prec1.0_params.pickle\", \"rb\"))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [],
   "source": [
    "# eval_args = {}\n",
    "# eval_args[\"linearised_laplace\"] = True\n",
    "# # eval_args[\"linearised_laplace\"] = False\n",
    "\n",
    "# eval_args[\"posterior_sample_type\"] = \"Pytree\"\n",
    "# eval_args[\"likelihood\"] = \"classification\"\n",
    "\n",
    "# ids = [\"CIFAR-10\", \"CIFAR-100\", \"SVHN\"]\n",
    "# n_datapoint=500\n",
    "# ood_batch_size = 50\n",
    "# metrics_lr_sampled = []\n",
    "# for i, id in enumerate(ids):    \n",
    "#     _, test_loader = get_cifar10_ood_loaders(id, data_path=\"data\", download=True, batch_size=ood_batch_size, n_datapoint=n_datapoint)\n",
    "#     some_metrics, all_y_prob, all_y_true, all_y_var = evaluate_samples(test_loader, lr_posterior, params, model_fn, eval_args)\n",
    "#     if i == 0:\n",
    "#         all_y_prob_in = all_y_prob\n",
    "#     more_metrics = list(map(lambda x,y: compute_metrics(\n",
    "#             i, id, x, test_loader, y, all_y_var, benchmark=\"CIFAR-10-OOD\", supress=True\n",
    "#         ), all_y_prob, all_y_prob_in))\n",
    "#     more_metrics = {k: [dic[k] for dic in more_metrics] for k in more_metrics[0]}\n",
    "#     metrics_lr_sampled.append({**some_metrics, **more_metrics})\n",
    "#     print(\", \".join([f\"{k}: {jnp.mean(jnp.array(v)):.4f}\" for k, v in metrics_lr_sampled[-1].items()]))\n",
    "\n",
    "# metrics_lr_sampled = [{k: jnp.array(dic[k]).mean() for k in dic} for dic in metrics_lr_sampled]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "ename": "NameError",
     "evalue": "name 'get_cifar10_ood_loaders' is not defined",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mNameError\u001b[0m                                 Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[2], line 11\u001b[0m\n\u001b[1;32m      9\u001b[0m metrics_lr \u001b[38;5;241m=\u001b[39m []\n\u001b[1;32m     10\u001b[0m \u001b[38;5;28;01mfor\u001b[39;00m i, \u001b[38;5;28mid\u001b[39m \u001b[38;5;129;01min\u001b[39;00m \u001b[38;5;28menumerate\u001b[39m(ids):    \n\u001b[0;32m---> 11\u001b[0m     _, test_loader \u001b[38;5;241m=\u001b[39m \u001b[43mget_cifar10_ood_loaders\u001b[49m(\u001b[38;5;28mid\u001b[39m, data_path\u001b[38;5;241m=\u001b[39m\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mdata\u001b[39m\u001b[38;5;124m\"\u001b[39m, download\u001b[38;5;241m=\u001b[39m\u001b[38;5;28;01mTrue\u001b[39;00m, batch_size\u001b[38;5;241m=\u001b[39mood_batch_size, n_datapoint\u001b[38;5;241m=\u001b[39mn_datapoint)\n\u001b[1;32m     12\u001b[0m     some_metrics, all_y_prob, all_y_true, all_y_var \u001b[38;5;241m=\u001b[39m evaluate(test_loader, lr_posterior, params, model_fn, eval_args)\n\u001b[1;32m     13\u001b[0m     \u001b[38;5;28;01mif\u001b[39;00m i \u001b[38;5;241m==\u001b[39m \u001b[38;5;241m0\u001b[39m:\n",
      "\u001b[0;31mNameError\u001b[0m: name 'get_cifar10_ood_loaders' is not defined"
     ]
    }
   ],
   "source": [
    "eval_args = {}\n",
    "eval_args[\"linearised_laplace\"] = False\n",
    "eval_args[\"posterior_sample_type\"] = \"Pytree\"\n",
    "eval_args[\"likelihood\"] = \"classification\"\n",
    "\n",
    "ids = [\"CIFAR-10\", \"CIFAR-100\", \"SVHN\"]\n",
    "n_datapoint=500\n",
    "ood_batch_size = 50\n",
    "metrics_lr = []\n",
    "for i, id in enumerate(ids):    \n",
    "    _, test_loader = get_cifar10_ood_loaders(id, data_path=\"data\", download=True, batch_size=ood_batch_size, n_datapoint=n_datapoint)\n",
    "    some_metrics, all_y_prob, all_y_true, all_y_var = evaluate(test_loader, lr_posterior, params, model_fn, eval_args)\n",
    "    if i == 0:\n",
    "        all_y_prob_in = all_y_prob\n",
    "    more_metrics = compute_metrics(\n",
    "            i, id, all_y_prob, test_loader, all_y_prob_in, all_y_var, benchmark=\"CIFAR-10-OOD\"\n",
    "        )\n",
    "    metrics_lr.append({**some_metrics, **more_metrics})\n",
    "    print(\", \".join([f\"{k}: {v:.4f}\" for k, v in metrics_lr[-1].items()]))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Files already downloaded and verified\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/xxx/geometric-laplace/src/data/torch_datasets.py:158: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  self.targets = F.one_hot(torch.tensor(self.dataset.targets), len(cls)).numpy()\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "CIFAR-10-OOD - dataset: CIFAR-10\n",
      "conf: 0.8652, nll: 338.7505, acc: 0.7920\n",
      "Files already downloaded and verified\n",
      "CIFAR-10-OOD - dataset: CIFAR-100\n",
      "conf: 0.7029, nll: 285.3149, acc: 0.1920, auroc: 0.7527, fpr95: 0.8220\n",
      "Using downloaded and verified file: data/test_32x32.mat\n",
      "CIFAR-10-OOD - dataset: SVHN\n",
      "conf: 0.7620, nll: 3391.9841, acc: 0.1000, auroc: 0.6908, fpr95: 0.8960\n"
     ]
    }
   ],
   "source": [
    "eval_args = {}\n",
    "eval_args[\"linearised_laplace\"] = False\n",
    "eval_args[\"posterior_sample_type\"] = \"Pytree\"\n",
    "eval_args[\"likelihood\"] = \"classification\"\n",
    "\n",
    "ids = [\"CIFAR-10\", \"CIFAR-100\", \"SVHN\"]\n",
    "n_datapoint=500\n",
    "ood_batch_size = 50\n",
    "metrics_posterior = []\n",
    "for i, id in enumerate(ids):    \n",
    "    _, test_loader = get_cifar10_ood_loaders(id, data_path=\"data\", download=True, batch_size=ood_batch_size, n_datapoint=n_datapoint)\n",
    "    some_metrics, all_y_prob, all_y_true, all_y_var = evaluate(test_loader, posterior, params, model_fn, eval_args)\n",
    "    if i == 0:\n",
    "        all_y_prob_in = all_y_prob\n",
    "    more_metrics = compute_metrics(\n",
    "            i, id, all_y_prob, test_loader, all_y_prob_in, all_y_var, benchmark=\"CIFAR-10-OOD\"\n",
    "        )\n",
    "    metrics_posterior.append({**some_metrics, **more_metrics})\n",
    "    print(\", \".join([f\"{k}: {v:.4f}\" for k, v in metrics_posterior[-1].items()]))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Files already downloaded and verified\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/xxx/geometric-laplace/src/data/torch_datasets.py:158: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  self.targets = F.one_hot(torch.tensor(self.dataset.targets), len(cls)).numpy()\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "CIFAR-10-OOD - dataset: CIFAR-10\n",
      "conf: 0.9527, nll: 243.3194, acc: 0.8780\n",
      "Files already downloaded and verified\n",
      "CIFAR-10-OOD - dataset: CIFAR-100\n",
      "conf: 0.8246, nll: 462.7240, acc: 0.1160, auroc: 0.8248, fpr95: 0.7740\n",
      "Using downloaded and verified file: data/test_32x32.mat\n",
      "CIFAR-10-OOD - dataset: SVHN\n",
      "conf: 0.8211, nll: 4398.6895, acc: 0.0920, auroc: 0.8169, fpr95: 0.7540\n"
     ]
    }
   ],
   "source": [
    "eval_args = {}\n",
    "eval_args[\"linearised_laplace\"] = True\n",
    "eval_args[\"posterior_sample_type\"] = \"Pytree\"\n",
    "eval_args[\"likelihood\"] = \"classification\"\n",
    "\n",
    "ids = [\"CIFAR-10\", \"CIFAR-100\", \"SVHN\"]\n",
    "n_datapoint=500\n",
    "ood_batch_size = 50\n",
    "metrics_linearised = []\n",
    "for i, id in enumerate(ids):    \n",
    "    _, test_loader = get_cifar10_ood_loaders(id, data_path=\"data\", download=True, batch_size=ood_batch_size, n_datapoint=n_datapoint)\n",
    "    some_metrics, all_y_prob, all_y_true, all_y_var = evaluate(test_loader, posterior, params, model_fn, eval_args)\n",
    "    if i == 0:\n",
    "        all_y_prob_in = all_y_prob\n",
    "    more_metrics = compute_metrics(\n",
    "            i, id, all_y_prob, test_loader, all_y_prob_in, all_y_var, benchmark=\"CIFAR-10-OOD\"\n",
    "        )\n",
    "    metrics_linearised.append({**some_metrics, **more_metrics})\n",
    "    print(\", \".join([f\"{k}: {v:.4f}\" for k, v in metrics_linearised[-1].items()]))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Files already downloaded and verified\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/xxx/geometric-laplace/src/data/torch_datasets.py:158: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  self.targets = F.one_hot(torch.tensor(self.dataset.targets), len(cls)).numpy()\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "CIFAR-10-OOD - dataset: CIFAR-10\n",
      "conf: 0.9566, nll: 151.5962, acc: 0.9200\n",
      "Files already downloaded and verified\n",
      "CIFAR-10-OOD - dataset: CIFAR-100\n",
      "conf: 0.8004, nll: 415.8189, acc: 0.1160, auroc: 0.8633, fpr95: 0.7580\n",
      "Using downloaded and verified file: data/test_32x32.mat\n",
      "CIFAR-10-OOD - dataset: SVHN\n",
      "conf: 0.7576, nll: 3643.8286, acc: 0.0800, auroc: 0.8858, fpr95: 0.6780\n"
     ]
    }
   ],
   "source": [
    "eval_args = {}\n",
    "eval_args[\"linearised_laplace\"] = False\n",
    "eval_args[\"posterior_sample_type\"] = \"Pytree\"\n",
    "eval_args[\"likelihood\"] = \"classification\"\n",
    "\n",
    "ids = [\"CIFAR-10\", \"CIFAR-100\", \"SVHN\"]\n",
    "n_datapoint=500\n",
    "ood_batch_size = 50\n",
    "metrics_map = []\n",
    "for i, id in enumerate(ids):    \n",
    "    _, test_loader = get_cifar10_ood_loaders(id, data_path=\"data\", download=True, batch_size=ood_batch_size, n_datapoint=n_datapoint)\n",
    "    some_metrics, all_y_prob, all_y_true, all_y_var = evaluate_map(test_loader, params, model_fn, eval_args)\n",
    "    if i == 0:\n",
    "        all_y_prob_in = all_y_prob\n",
    "    more_metrics = compute_metrics(\n",
    "            i, id, all_y_prob, test_loader, all_y_prob_in, all_y_var, benchmark=\"CIFAR-10-OOD\"\n",
    "        )\n",
    "    metrics_map.append({**some_metrics, **more_metrics})\n",
    "    print(\", \".join([f\"{k}: {v:.4f}\" for k, v in metrics_map[-1].items()]))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_1085966/41536951.py:1: DeprecationWarning: \n",
      "Pyarrow will become a required dependency of pandas in the next major release of pandas (pandas 3.0),\n",
      "(to allow more performant data types, such as the Arrow string type, and better interoperability with other libraries)\n",
      "but was not found to be installed on your system.\n",
      "If this would cause problems for you,\n",
      "please provide us feedback at https://github.com/pandas-dev/pandas/issues/54466\n",
      "        \n",
      "  import pandas as pd\n"
     ]
    },
    {
     "ename": "NameError",
     "evalue": "name 'metrics_lr' is not defined",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mNameError\u001b[0m                                 Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[1], line 4\u001b[0m\n\u001b[1;32m      2\u001b[0m metrics_dict \u001b[38;5;241m=\u001b[39m [\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mconf\u001b[39m\u001b[38;5;124m'\u001b[39m, \u001b[38;5;124m'\u001b[39m\u001b[38;5;124mauroc\u001b[39m\u001b[38;5;124m'\u001b[39m]\n\u001b[1;32m      3\u001b[0m method_list \u001b[38;5;241m=\u001b[39m [\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mLaplace Diffusion\u001b[39m\u001b[38;5;124m\"\u001b[39m, \u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mSampled Laplace\u001b[39m\u001b[38;5;124m\"\u001b[39m, \u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mLinearised Laplace\u001b[39m\u001b[38;5;124m\"\u001b[39m, \u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mMAP\u001b[39m\u001b[38;5;124m\"\u001b[39m]\n\u001b[0;32m----> 4\u001b[0m method_dict \u001b[38;5;241m=\u001b[39m {\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mLaplace Diffusion\u001b[39m\u001b[38;5;124m\"\u001b[39m:\u001b[43mmetrics_lr\u001b[49m, \u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mSampled Laplace\u001b[39m\u001b[38;5;124m\"\u001b[39m:metrics_posterior, \u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mLinearised Laplace\u001b[39m\u001b[38;5;124m\"\u001b[39m:metrics_linearised, \u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mMAP\u001b[39m\u001b[38;5;124m\"\u001b[39m:metrics_map}\n\u001b[1;32m      5\u001b[0m mux \u001b[38;5;241m=\u001b[39m pd\u001b[38;5;241m.\u001b[39mMultiIndex\u001b[38;5;241m.\u001b[39mfrom_product([ids[\u001b[38;5;241m1\u001b[39m:], metrics_dict])\n\u001b[1;32m      6\u001b[0m df_data \u001b[38;5;241m=\u001b[39m {k: [dic[metric] \u001b[38;5;28;01mfor\u001b[39;00m metric \u001b[38;5;129;01min\u001b[39;00m metrics_dict \u001b[38;5;28;01mfor\u001b[39;00m dic \u001b[38;5;129;01min\u001b[39;00m method_dict[k][\u001b[38;5;241m1\u001b[39m:]] \u001b[38;5;28;01mfor\u001b[39;00m k \u001b[38;5;129;01min\u001b[39;00m method_dict}\n",
      "\u001b[0;31mNameError\u001b[0m: name 'metrics_lr' is not defined"
     ]
    }
   ],
   "source": [
    "import pandas as pd\n",
    "metrics_dict = ['conf', 'auroc']\n",
    "method_list = [\"Laplace Diffusion\", \"Sampled Laplace\", \"Linearised Laplace\", \"MAP\"]\n",
    "method_dict = {\"Laplace Diffusion\":metrics_lr, \"Sampled Laplace\":metrics_posterior, \"Linearised Laplace\":metrics_linearised, \"MAP\":metrics_map}\n",
    "mux = pd.MultiIndex.from_product([ids[1:], metrics_dict])\n",
    "df_data = {k: [dic[metric] for metric in metrics_dict for dic in method_dict[k][1:]] for k in method_dict}\n",
    "df = pd.DataFrame.from_dict(df_data, orient='index',\n",
    "                       columns=mux)\n",
    "    "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead tr th {\n",
       "        text-align: left;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr>\n",
       "      <th></th>\n",
       "      <th colspan=\"2\" halign=\"left\">CIFAR-100</th>\n",
       "      <th colspan=\"2\" halign=\"left\">SVHN</th>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th></th>\n",
       "      <th>conf</th>\n",
       "      <th>auroc</th>\n",
       "      <th>conf</th>\n",
       "      <th>auroc</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>Laplace Diffusion</th>\n",
       "      <td>0.753719</td>\n",
       "      <td>0.773558</td>\n",
       "      <td>0.848340</td>\n",
       "      <td>0.833040</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Sampled Laplace</th>\n",
       "      <td>0.702883</td>\n",
       "      <td>0.762028</td>\n",
       "      <td>0.752650</td>\n",
       "      <td>0.690840</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>Linearised Laplace</th>\n",
       "      <td>0.824627</td>\n",
       "      <td>0.821055</td>\n",
       "      <td>0.824760</td>\n",
       "      <td>0.816894</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>MAP</th>\n",
       "      <td>0.795880</td>\n",
       "      <td>0.805988</td>\n",
       "      <td>0.863354</td>\n",
       "      <td>0.860866</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "                   CIFAR-100                SVHN          \n",
       "                        conf     auroc      conf     auroc\n",
       "Laplace Diffusion   0.753719  0.773558  0.848340  0.833040\n",
       "Sampled Laplace     0.702883  0.762028  0.752650  0.690840\n",
       "Linearised Laplace  0.824627  0.821055  0.824760  0.816894\n",
       "MAP                 0.795880  0.805988  0.863354  0.860866"
      ]
     },
     "execution_count": 27,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "df"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Corrupted CIFAR"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "ename": "NameError",
     "evalue": "name 'pickle' is not defined",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mNameError\u001b[0m                                 Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[1], line 1\u001b[0m\n\u001b[0;32m----> 1\u001b[0m lr_posterior_dict \u001b[38;5;241m=\u001b[39m \u001b[43mpickle\u001b[49m\u001b[38;5;241m.\u001b[39mload(\u001b[38;5;28mopen\u001b[39m(\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124m../checkpoints/CIFAR-10/old_samples/posterior_samples_pray_seed0_type_non-kernel-eigvals_params.pickle\u001b[39m\u001b[38;5;124m\"\u001b[39m, \u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mrb\u001b[39m\u001b[38;5;124m\"\u001b[39m))\n\u001b[1;32m      3\u001b[0m lr_posterior \u001b[38;5;241m=\u001b[39m lr_posterior_dict[\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mposterior_samples\u001b[39m\u001b[38;5;124m'\u001b[39m]\n",
      "\u001b[0;31mNameError\u001b[0m: name 'pickle' is not defined"
     ]
    }
   ],
   "source": [
    "lr_posterior_dict = pickle.load(open(\"../checkpoints/CIFAR-10/old_samples/posterior_samples_pray_seed0_type_non-kernel-eigvals_params.pickle\", \"rb\"))\n",
    "\n",
    "lr_posterior = lr_posterior_dict['posterior_samples']\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "from src.data.datasets import load_corrupted_cifar10_per_type\n",
    "noise_types = [corruption[:-4] for corruption in os.listdir('../src/data/CIFAR-10-C')]\n",
    "noise_types.remove('labels')\n",
    "ids = [(0, None)]\n",
    "ids += [(5, t) for t in noise_types]\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "18"
      ]
     },
     "execution_count": 26,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "len(noise_types)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "['saturate',\n",
       " 'shot_noise',\n",
       " 'gaussian_noise',\n",
       " 'contrast',\n",
       " 'snow',\n",
       " 'glass_blur',\n",
       " 'defocus_blur',\n",
       " 'spatter',\n",
       " 'impulse_noise',\n",
       " 'elastic_transform',\n",
       " 'speckle_noise',\n",
       " 'jpeg_compression',\n",
       " 'fog',\n",
       " 'zoom_blur',\n",
       " 'motion_blur',\n",
       " 'frost',\n",
       " 'pixelate',\n",
       " 'gaussian_blur']"
      ]
     },
     "execution_count": 27,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "noise_types"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Files already downloaded and verified\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/xxx/geometric-laplace/src/data/torch_datasets.py:158: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  self.targets = F.one_hot(torch.tensor(self.dataset.targets), len(cls)).numpy()\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "CIFAR-10-C with distribution shift intensity 0\n",
      "conf: 0.9394, nll: 140.9795, acc: 0.9100, brier: 0.1348, ece: 0.1878, mce: 0.8133\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/xxx/geometric-laplace/geom/lib64/python3.9/site-packages/numpy/core/fromnumeric.py:3504: RuntimeWarning: Mean of empty slice.\n",
      "  return _methods._mean(a, axis=axis, dtype=dtype,\n",
      "/xxx/geometric-laplace/geom/lib64/python3.9/site-packages/numpy/core/_methods.py:129: RuntimeWarning: invalid value encountered in scalar divide\n",
      "  ret = ret.dtype.type(ret / rcount)\n",
      "/xxx/geometric-laplace/geom/lib64/python3.9/site-packages/numpy/core/_methods.py:129: RuntimeWarning: invalid value encountered in divide\n",
      "  ret = ret.dtype.type(ret / rcount)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "CIFAR-10-C with distribution shift intensity 1\n",
      "conf: 0.8843, nll: 365.8628, acc: 0.7780, brier: 0.3167, ece: 0.2390, mce: 0.7155\n",
      "CIFAR-10-C with distribution shift intensity 2\n",
      "conf: 0.7487, nll: 1291.7693, acc: 0.4400, brier: 0.8044, ece: 0.3496, mce: 0.7759\n",
      "CIFAR-10-C with distribution shift intensity 3\n",
      "conf: 0.7187, nll: 1313.5364, acc: 0.4500, brier: 0.8066, ece: 0.3066, mce: 0.7793\n",
      "CIFAR-10-C with distribution shift intensity 4\n",
      "conf: 0.7285, nll: 1644.9664, acc: 0.3340, brier: 0.9941, ece: 0.4019, mce: 0.7062\n",
      "CIFAR-10-C with distribution shift intensity 5\n",
      "conf: 0.8610, nll: 560.9661, acc: 0.6920, brier: 0.4488, ece: 0.2994, mce: 0.8571\n",
      "CIFAR-10-C with distribution shift intensity 6\n",
      "conf: 0.7886, nll: 989.9955, acc: 0.5280, brier: 0.7086, ece: 0.3141, mce: 0.8337\n",
      "CIFAR-10-C with distribution shift intensity 7\n",
      "conf: 0.8940, nll: 357.0130, acc: 0.7780, brier: 0.3261, ece: 0.2448, mce: 0.8953\n",
      "CIFAR-10-C with distribution shift intensity 8\n",
      "conf: 0.8758, nll: 662.8196, acc: 0.7180, brier: 0.4430, ece: 0.2332, mce: 0.6185\n",
      "CIFAR-10-C with distribution shift intensity 9\n",
      "conf: 0.7387, nll: 1491.2211, acc: 0.3420, brier: 0.9901, ece: 0.4136, mce: 0.8027\n",
      "CIFAR-10-C with distribution shift intensity 10\n",
      "conf: 0.8492, nll: 536.7590, acc: 0.6940, brier: 0.4535, ece: 0.2492, mce: 0.7531\n",
      "CIFAR-10-C with distribution shift intensity 11\n",
      "conf: 0.7566, nll: 1208.7373, acc: 0.4640, brier: 0.7874, ece: 0.3408, mce: 0.7056\n",
      "CIFAR-10-C with distribution shift intensity 12\n",
      "conf: 0.8532, nll: 565.5759, acc: 0.6840, brier: 0.4737, ece: 0.2588, mce: 0.8365\n",
      "CIFAR-10-C with distribution shift intensity 13\n",
      "conf: 0.8255, nll: 732.4495, acc: 0.6140, brier: 0.5686, ece: 0.2885, mce: 0.8665\n",
      "CIFAR-10-C with distribution shift intensity 14\n",
      "conf: 0.8812, nll: 378.4203, acc: 0.7620, brier: 0.3410, ece: 0.2370, mce: 0.7608\n",
      "CIFAR-10-C with distribution shift intensity 15\n",
      "conf: 0.8543, nll: 519.1296, acc: 0.7000, brier: 0.4439, ece: 0.2416, mce: 0.7626\n",
      "CIFAR-10-C with distribution shift intensity 16\n",
      "conf: 0.8156, nll: 766.4399, acc: 0.6220, brier: 0.5431, ece: 0.2694, mce: 0.7031\n",
      "CIFAR-10-C with distribution shift intensity 17\n",
      "conf: 0.8126, nll: 1666.4465, acc: 0.4120, brier: 0.9470, ece: 0.4543, mce: 0.7779\n",
      "CIFAR-10-C with distribution shift intensity 18\n",
      "conf: 0.8891, nll: 447.8246, acc: 0.7340, brier: 0.3949, ece: 0.2689, mce: 0.7419\n"
     ]
    }
   ],
   "source": [
    "eval_args = {}\n",
    "eval_args[\"linearised_laplace\"] = False\n",
    "eval_args[\"posterior_sample_type\"] = \"Pytree\"\n",
    "eval_args[\"likelihood\"] = \"classification\"\n",
    "\n",
    "n_datapoint=500\n",
    "ood_batch_size = 50\n",
    "data_path = \"../src/data\"\n",
    "metrics_lr = []\n",
    "for i, id in enumerate(ids):    \n",
    "    test_loader = load_corrupted_cifar10_per_type(id[0], id[1], data_path=data_path, batch_size=ood_batch_size, n_datapoint=n_datapoint)\n",
    "    some_metrics, all_y_prob, all_y_true, all_y_var = evaluate(test_loader, lr_posterior, params, model_fn, eval_args)\n",
    "    if i == 0:\n",
    "        all_y_prob_in = all_y_prob\n",
    "    more_metrics = compute_metrics(\n",
    "            i, id, all_y_prob, test_loader, all_y_prob_in, all_y_var, benchmark=\"CIFAR-10-C\"\n",
    "        )\n",
    "    metrics_lr.append({**some_metrics, **more_metrics})\n",
    "    print(\", \".join([f\"{k}: {v:.4f}\" for k, v in metrics_lr[-1].items()]))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Files already downloaded and verified\n",
      "CIFAR-10-C with distribution shift intensity 0\n",
      "conf: 0.8652, nll: 338.7505, acc: 0.7920, brier: 0.3089, ece: 0.1540, mce: 0.6675\n",
      "CIFAR-10-C with distribution shift intensity 1\n",
      "conf: 0.7860, nll: 637.9795, acc: 0.6400, brier: 0.5098, ece: 0.2012, mce: 0.8538\n",
      "CIFAR-10-C with distribution shift intensity 2\n",
      "conf: 0.6900, nll: 1173.5796, acc: 0.4100, brier: 0.8615, ece: 0.3233, mce: 0.9133\n",
      "CIFAR-10-C with distribution shift intensity 3\n",
      "conf: 0.6788, nll: 1245.5637, acc: 0.3460, brier: 0.9159, ece: 0.3429, mce: 0.8839\n",
      "CIFAR-10-C with distribution shift intensity 4\n",
      "conf: 0.7426, nll: 2553.1362, acc: 0.2180, brier: 1.2223, ece: 0.5241, mce: 0.8154\n",
      "CIFAR-10-C with distribution shift intensity 5\n",
      "conf: 0.7846, nll: 943.2003, acc: 0.5120, brier: 0.7282, ece: 0.3137, mce: 0.8247\n",
      "CIFAR-10-C with distribution shift intensity 6\n",
      "conf: 0.7540, nll: 1376.8655, acc: 0.3680, brier: 0.9691, ece: 0.4028, mce: 0.7446\n",
      "CIFAR-10-C with distribution shift intensity 7\n",
      "conf: 0.7886, nll: 646.1942, acc: 0.6460, brier: 0.5279, ece: 0.1984, mce: 0.6522\n",
      "CIFAR-10-C with distribution shift intensity 8\n",
      "conf: 0.7763, nll: 1042.7153, acc: 0.5480, brier: 0.6946, ece: 0.2840, mce: 0.7550\n",
      "CIFAR-10-C with distribution shift intensity 9\n",
      "conf: 0.7399, nll: 2167.1646, acc: 0.2000, brier: 1.2231, ece: 0.5418, mce: 0.8952\n",
      "CIFAR-10-C with distribution shift intensity 10\n",
      "conf: 0.7526, nll: 814.2997, acc: 0.5260, brier: 0.6903, ece: 0.2800, mce: 0.7189\n",
      "CIFAR-10-C with distribution shift intensity 11\n",
      "conf: 0.6854, nll: 1182.1589, acc: 0.4000, brier: 0.8605, ece: 0.3054, mce: 0.8251\n",
      "CIFAR-10-C with distribution shift intensity 12\n",
      "conf: 0.7576, nll: 733.6227, acc: 0.5460, brier: 0.6271, ece: 0.2478, mce: 0.7073\n",
      "CIFAR-10-C with distribution shift intensity 13\n",
      "conf: 0.7383, nll: 1230.6744, acc: 0.4400, brier: 0.8586, ece: 0.3352, mce: 0.8863\n",
      "CIFAR-10-C with distribution shift intensity 14\n",
      "conf: 0.7898, nll: 754.2441, acc: 0.5760, brier: 0.6321, ece: 0.2618, mce: 0.7548\n",
      "CIFAR-10-C with distribution shift intensity 15\n",
      "conf: 0.7737, nll: 862.4518, acc: 0.5400, brier: 0.6786, ece: 0.2694, mce: 0.7793\n",
      "CIFAR-10-C with distribution shift intensity 16\n",
      "conf: 0.7343, nll: 1151.6510, acc: 0.4320, brier: 0.8267, ece: 0.3248, mce: 0.7944\n",
      "CIFAR-10-C with distribution shift intensity 17\n",
      "conf: 0.7753, nll: 1673.9932, acc: 0.3600, brier: 0.9963, ece: 0.4495, mce: 0.8346\n",
      "CIFAR-10-C with distribution shift intensity 18\n",
      "conf: 0.7648, nll: 835.4083, acc: 0.5420, brier: 0.6762, ece: 0.2641, mce: 0.7403\n"
     ]
    }
   ],
   "source": [
    "eval_args = {}\n",
    "eval_args[\"linearised_laplace\"] = False\n",
    "eval_args[\"posterior_sample_type\"] = \"Pytree\"\n",
    "eval_args[\"likelihood\"] = \"classification\"\n",
    "\n",
    "n_datapoint=500\n",
    "ood_batch_size = 50\n",
    "data_path = \"../src/data\"\n",
    "metrics_posterior = []\n",
    "for i, id in enumerate(ids):    \n",
    "    test_loader = load_corrupted_cifar10_per_type(id[0], id[1], data_path=data_path, batch_size=ood_batch_size, n_datapoint=n_datapoint)\n",
    "    some_metrics, all_y_prob, all_y_true, all_y_var = evaluate(test_loader, posterior, params, model_fn, eval_args)\n",
    "    if i == 0:\n",
    "        all_y_prob_in = all_y_prob\n",
    "    more_metrics = compute_metrics(\n",
    "            i, id, all_y_prob, test_loader, all_y_prob_in, all_y_var, benchmark=\"CIFAR-10-C\"\n",
    "        )\n",
    "    metrics_posterior.append({**some_metrics, **more_metrics})\n",
    "    print(\", \".join([f\"{k}: {v:.4f}\" for k, v in metrics_posterior[-1].items()]))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Files already downloaded and verified\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "CIFAR-10-C with distribution shift intensity 0\n",
      "conf: 0.9527, nll: 243.3194, acc: 0.8780, brier: 0.1938, ece: 0.3044, mce: 0.8434\n",
      "CIFAR-10-C with distribution shift intensity 1\n",
      "conf: 0.9181, nll: 568.9250, acc: 0.7740, brier: 0.3623, ece: 0.3115, mce: 0.9335\n",
      "CIFAR-10-C with distribution shift intensity 2\n",
      "conf: 0.8317, nll: 1735.4540, acc: 0.4360, brier: 0.9048, ece: 0.4627, mce: 0.8830\n",
      "CIFAR-10-C with distribution shift intensity 3\n",
      "conf: 0.8257, nll: 1993.4885, acc: 0.4180, brier: 0.9547, ece: 0.4526, mce: 0.7959\n",
      "CIFAR-10-C with distribution shift intensity 4\n",
      "conf: 0.7658, nll: 1896.4753, acc: 0.4020, brier: 0.9326, ece: 0.3917, mce: 0.8839\n",
      "CIFAR-10-C with distribution shift intensity 5\n",
      "conf: 0.8854, nll: 863.2042, acc: 0.6760, brier: 0.5144, ece: 0.3292, mce: 0.8065\n",
      "CIFAR-10-C with distribution shift intensity 6\n",
      "conf: 0.8580, nll: 1730.3289, acc: 0.4700, brier: 0.8922, ece: 0.4321, mce: 0.8668\n",
      "CIFAR-10-C with distribution shift intensity 7\n",
      "conf: 0.9279, nll: 548.8822, acc: 0.7620, brier: 0.3884, ece: 0.3395, mce: 0.7941\n",
      "CIFAR-10-C with distribution shift intensity 8\n",
      "conf: 0.8981, nll: 977.3506, acc: 0.6500, brier: 0.5703, ece: 0.3910, mce: 0.8960\n",
      "CIFAR-10-C with distribution shift intensity 9\n",
      "conf: 0.8230, nll: 2103.6963, acc: 0.3400, brier: 1.0796, ece: 0.5024, mce: 0.8528\n",
      "CIFAR-10-C with distribution shift intensity 10\n",
      "conf: 0.8811, nll: 947.7552, acc: 0.6480, brier: 0.5823, ece: 0.3246, mce: 0.9468\n",
      "CIFAR-10-C with distribution shift intensity 11\n",
      "conf: 0.8520, nll: 1663.4673, acc: 0.4940, brier: 0.8415, ece: 0.4170, mce: 0.9173\n",
      "CIFAR-10-C with distribution shift intensity 12\n",
      "conf: 0.9010, nll: 871.5920, acc: 0.6580, brier: 0.5760, ece: 0.3843, mce: 0.9421\n",
      "CIFAR-10-C with distribution shift intensity 13\n",
      "conf: 0.8659, nll: 994.6486, acc: 0.6160, brier: 0.6021, ece: 0.3474, mce: 0.7056\n",
      "CIFAR-10-C with distribution shift intensity 14\n",
      "conf: 0.9082, nll: 656.2048, acc: 0.7340, brier: 0.4318, ece: 0.3003, mce: 0.7263\n",
      "CIFAR-10-C with distribution shift intensity 15\n",
      "conf: 0.8909, nll: 945.8001, acc: 0.6320, brier: 0.6138, ece: 0.4102, mce: 0.8851\n",
      "CIFAR-10-C with distribution shift intensity 16\n",
      "conf: 0.8612, nll: 1285.5745, acc: 0.5820, brier: 0.6800, ece: 0.3724, mce: 0.7798\n",
      "CIFAR-10-C with distribution shift intensity 17\n",
      "conf: 0.8798, nll: 2230.3875, acc: 0.4220, brier: 1.0001, ece: 0.5286, mce: 0.8655\n",
      "CIFAR-10-C with distribution shift intensity 18\n",
      "conf: 0.8992, nll: 648.0941, acc: 0.7320, brier: 0.4339, ece: 0.3311, mce: 0.7347\n"
     ]
    }
   ],
   "source": [
    "eval_args = {}\n",
    "eval_args[\"linearised_laplace\"] = True\n",
    "eval_args[\"posterior_sample_type\"] = \"Pytree\"\n",
    "eval_args[\"likelihood\"] = \"classification\"\n",
    "\n",
    "n_datapoint=500\n",
    "ood_batch_size = 50\n",
    "data_path = \"../src/data\"\n",
    "metrics_linearised = []\n",
    "for i, id in enumerate(ids):    \n",
    "    test_loader = load_corrupted_cifar10_per_type(id[0], id[1], data_path=data_path, batch_size=ood_batch_size, n_datapoint=n_datapoint)\n",
    "    some_metrics, all_y_prob, all_y_true, all_y_var = evaluate(test_loader, posterior, params, model_fn, eval_args)\n",
    "    if i == 0:\n",
    "        all_y_prob_in = all_y_prob\n",
    "    more_metrics = compute_metrics(\n",
    "            i, id, all_y_prob, test_loader, all_y_prob_in, all_y_var, benchmark=\"CIFAR-10-C\"\n",
    "        )\n",
    "    metrics_linearised.append({**some_metrics, **more_metrics})\n",
    "    print(\", \".join([f\"{k}: {v:.4f}\" for k, v in metrics_linearised[-1].items()]))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Files already downloaded and verified\n",
      "CIFAR-10-C with distribution shift intensity 0\n",
      "conf: 0.9586, nll: 120.6521, acc: 0.9320, brier: 0.1044, ece: 0.2075, mce: 0.8585\n",
      "CIFAR-10-C with distribution shift intensity 1\n",
      "conf: 0.9360, nll: 337.9069, acc: 0.8500, brier: 0.2477, ece: 0.2608, mce: 0.7687\n",
      "CIFAR-10-C with distribution shift intensity 2\n",
      "conf: 0.8223, nll: 1376.8699, acc: 0.4940, brier: 0.8057, ece: 0.3890, mce: 0.7346\n",
      "CIFAR-10-C with distribution shift intensity 3\n",
      "conf: 0.7954, nll: 1476.8350, acc: 0.4820, brier: 0.8173, ece: 0.3716, mce: 0.7456\n",
      "CIFAR-10-C with distribution shift intensity 4\n",
      "conf: 0.7622, nll: 1516.1951, acc: 0.4000, brier: 0.9012, ece: 0.3764, mce: 0.8357\n",
      "CIFAR-10-C with distribution shift intensity 5\n",
      "conf: 0.8976, nll: 588.0447, acc: 0.7340, brier: 0.4143, ece: 0.3302, mce: 0.8552\n",
      "CIFAR-10-C with distribution shift intensity 6\n",
      "conf: 0.8394, nll: 1145.6726, acc: 0.5360, brier: 0.7381, ece: 0.3686, mce: 0.8141\n",
      "CIFAR-10-C with distribution shift intensity 7\n",
      "conf: 0.9342, nll: 306.2891, acc: 0.8380, brier: 0.2593, ece: 0.2804, mce: 0.8255\n",
      "CIFAR-10-C with distribution shift intensity 8\n",
      "conf: 0.9075, nll: 645.5288, acc: 0.7680, brier: 0.3873, ece: 0.2473, mce: 0.7588\n",
      "CIFAR-10-C with distribution shift intensity 9\n",
      "conf: 0.7814, nll: 1417.8950, acc: 0.4300, brier: 0.8960, ece: 0.4005, mce: 0.8980\n",
      "CIFAR-10-C with distribution shift intensity 10\n",
      "conf: 0.8925, nll: 618.1788, acc: 0.7140, brier: 0.4546, ece: 0.3017, mce: 0.6867\n",
      "CIFAR-10-C with distribution shift intensity 11\n",
      "conf: 0.8289, nll: 1280.3405, acc: 0.5280, brier: 0.7581, ece: 0.3828, mce: 0.8819\n",
      "CIFAR-10-C with distribution shift intensity 12\n",
      "conf: 0.8948, nll: 596.6155, acc: 0.7240, brier: 0.4415, ece: 0.3002, mce: 0.7663\n",
      "CIFAR-10-C with distribution shift intensity 13\n",
      "conf: 0.8799, nll: 706.2683, acc: 0.6700, brier: 0.5157, ece: 0.3404, mce: 0.8564\n",
      "CIFAR-10-C with distribution shift intensity 14\n",
      "conf: 0.9256, nll: 388.4565, acc: 0.8140, brier: 0.2874, ece: 0.2744, mce: 0.8577\n",
      "CIFAR-10-C with distribution shift intensity 15\n",
      "conf: 0.8919, nll: 507.3625, acc: 0.7420, brier: 0.3966, ece: 0.2979, mce: 0.8567\n",
      "CIFAR-10-C with distribution shift intensity 16\n",
      "conf: 0.8648, nll: 793.6117, acc: 0.6780, brier: 0.5089, ece: 0.2802, mce: 0.8450\n",
      "CIFAR-10-C with distribution shift intensity 17\n",
      "conf: 0.8540, nll: 1583.7080, acc: 0.4680, brier: 0.8606, ece: 0.4701, mce: 0.8174\n",
      "CIFAR-10-C with distribution shift intensity 18\n",
      "conf: 0.9242, nll: 370.3598, acc: 0.8020, brier: 0.3022, ece: 0.2872, mce: 0.9360\n"
     ]
    }
   ],
   "source": [
    "eval_args = {}\n",
    "eval_args[\"linearised_laplace\"] = True\n",
    "eval_args[\"posterior_sample_type\"] = \"Pytree\"\n",
    "eval_args[\"likelihood\"] = \"classification\"\n",
    "\n",
    "n_datapoint=500\n",
    "ood_batch_size = 50\n",
    "data_path = \"../src/data\"\n",
    "metrics_map = []\n",
    "for i, id in enumerate(ids):    \n",
    "    test_loader = load_corrupted_cifar10_per_type(id[0], id[1], data_path=data_path, batch_size=ood_batch_size, n_datapoint=n_datapoint)\n",
    "    some_metrics, all_y_prob, all_y_true, all_y_var = evaluate_map(test_loader, params, model_fn, eval_args)\n",
    "    if i == 0:\n",
    "        all_y_prob_in = all_y_prob\n",
    "    more_metrics = compute_metrics(\n",
    "            i, id, all_y_prob, test_loader, all_y_prob_in, all_y_var, benchmark=\"CIFAR-10-C\"\n",
    "        )\n",
    "    metrics_map.append({**some_metrics, **more_metrics})\n",
    "    print(\", \".join([f\"{k}: {v:.4f}\" for k, v in metrics_map[-1].items()]))\n"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "geom",
   "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"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
