{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "ename": "ImportError",
     "evalue": "cannot import name 'mse_loss' from 'src.losses' (/xxx/geometric-laplace/src/losses.py)",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mImportError\u001b[0m                               Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[1], line 13\u001b[0m\n\u001b[1;32m     11\u001b[0m \u001b[38;5;28;01mfrom\u001b[39;00m \u001b[38;5;21;01mjax\u001b[39;00m \u001b[38;5;28;01mimport\u001b[39;00m random, jit\n\u001b[1;32m     12\u001b[0m \u001b[38;5;28;01mimport\u001b[39;00m \u001b[38;5;21;01mpickle\u001b[39;00m\n\u001b[0;32m---> 13\u001b[0m \u001b[38;5;28;01mfrom\u001b[39;00m \u001b[38;5;21;01msrc\u001b[39;00m\u001b[38;5;21;01m.\u001b[39;00m\u001b[38;5;21;01mlosses\u001b[39;00m \u001b[38;5;28;01mimport\u001b[39;00m mse_loss, accuracy_preds, nll\n\u001b[1;32m     14\u001b[0m \u001b[38;5;28;01mfrom\u001b[39;00m \u001b[38;5;21;01msrc\u001b[39;00m\u001b[38;5;21;01m.\u001b[39;00m\u001b[38;5;21;01mhelper\u001b[39;00m \u001b[38;5;28;01mimport\u001b[39;00m calculate_exact_ggn, tree_random_normal_like\n\u001b[1;32m     15\u001b[0m \u001b[38;5;28;01mfrom\u001b[39;00m \u001b[38;5;21;01msrc\u001b[39;00m\u001b[38;5;21;01m.\u001b[39;00m\u001b[38;5;21;01msampling\u001b[39;00m\u001b[38;5;21;01m.\u001b[39;00m\u001b[38;5;21;01mpredictive_samplers\u001b[39;00m \u001b[38;5;28;01mimport\u001b[39;00m sample_predictive, sample_hessian_predictive\n",
      "\u001b[0;31mImportError\u001b[0m: cannot import name 'mse_loss' from 'src.losses' (/xxx/geometric-laplace/src/losses.py)"
     ]
    }
   ],
   "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 mse_loss, accuracy_preds, nll\n",
    "from src.helper import calculate_exact_ggn, tree_random_normal_like\n",
    "from src.sampling.predictive_samplers import sample_predictive, sample_hessian_predictive\n",
    "from jax import flatten_util\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "import torch\n",
    "from src.data.torch_datasets import MNIST, numpy_collate_fn"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Train ConvNet"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "train_samples = 10#1000\n",
    "train_samples = 300#1000\n",
    "classes_train = [0,1,2,3,4,5,6,7,8,9]\n",
    "n_classes = 10\n",
    "batch_size = 256#20\n",
    "test_batch_size = 256\n",
    "\n",
    "data_train = MNIST(path_root= \"/xxx/data/\",\n",
    "            train=True, n_samples=train_samples if train_samples > 0 else None, cls=classes_train\n",
    "        )\n",
    "data_test = MNIST(path_root = \"/xxx/data/\", train=False, cls=classes_train)\n",
    "\n",
    "if train_samples > 0:\n",
    "    N = train_samples * n_classes\n",
    "else:\n",
    "    N = len(data_train)\n",
    "N_test = len(data_test)\n",
    "if test_batch_size > 0:\n",
    "    test_batch_size = test_batch_size\n",
    "else:\n",
    "    test_batch_size = len(data_test)\n",
    "\n",
    "n_test_batches = int(N_test / test_batch_size)\n",
    "n_batches = int(N / batch_size)\n",
    "\n",
    "train_loader = torch.utils.data.DataLoader(\n",
    "    data_train, batch_size=batch_size, shuffle=True, collate_fn=numpy_collate_fn, drop_last=True,\n",
    ")\n",
    "\n",
    "valid_loader = torch.utils.data.DataLoader(\n",
    "    data_test, batch_size=test_batch_size, shuffle=True, collate_fn=numpy_collate_fn, drop_last=True,\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "class ConvNet(nn.Module):\n",
    "    output_dim: int = 10\n",
    "\n",
    "    @nn.compact\n",
    "    def __call__(self, x):\n",
    "        if len(x.shape) != 4:\n",
    "            x = jnp.expand_dims(x, 0)\n",
    "        x = jnp.transpose(x, (0, 2, 3, 1))\n",
    "        x = nn.Conv(features=4, kernel_size=(3, 3), strides=(2, 2), padding=1)(x)\n",
    "        x = nn.tanh(x)\n",
    "        x = nn.max_pool(x, window_shape=(2, 2), strides=(2, 2))\n",
    "        x = nn.Conv(features=4, kernel_size=(3, 3), strides=(2, 2), padding=1)(x)\n",
    "        x = nn.tanh(x)\n",
    "        x = nn.max_pool(x, window_shape=(2, 2), strides=(2, 2))\n",
    "        x = x.reshape((x.shape[0], -1))\n",
    "        return nn.Dense(features=self.output_dim)(x)\n",
    "\n",
    "def compute_num_params(pytree):\n",
    "    return sum(x.size if hasattr(x, \"size\") else 0 for x in jax.tree_util.tree_leaves(pytree))\n",
    "\n",
    "\n",
    "model = ConvNet()\n",
    "batch = next(iter(train_loader))\n",
    "x_init, y_init = batch[\"image\"], batch[\"label\"]\n",
    "output_dim = y_init.shape[-1]\n",
    "key, split_key = random.split(jax.random.PRNGKey(0))\n",
    "params = model.init(key, x_init)\n",
    "alpha = 1.\n",
    "optim = optax.chain(\n",
    "        optax.clip(1.),\n",
    "        getattr(optax, \"adam\")(1e-2),\n",
    "    )\n",
    "opt_state = optim.init(params)\n",
    "n_params = compute_num_params(params)\n",
    "n_epochs = 100"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "def cross_entropy_loss(preds, y, rho=1.0):\n",
    "    \"\"\"\n",
    "    preds: (n_samples, n_classes) (logits)\n",
    "    y: (n_samples, n_classes) (one-hot labels)\n",
    "    \"\"\"\n",
    "    preds = preds * rho\n",
    "    preds = jax.nn.log_softmax(preds, axis=-1)\n",
    "    return -jnp.sum(jnp.sum(preds * y, axis=-1))\n",
    "\n",
    "def accuracy(params, model, batch_x, batch_y):\n",
    "    preds = model.apply(params, batch_x)\n",
    "    return jnp.sum(preds.argmax(axis=-1) == batch_y.argmax(axis=-1))\n",
    "\n",
    "\n",
    "def map_loss(\n",
    "    params,\n",
    "    model,\n",
    "    x_batch,\n",
    "    y_batch,\n",
    "    alpha,\n",
    "    n_params: int,\n",
    "    N_datapoints_max: int,\n",
    "):\n",
    "    # define dict for logging purposes\n",
    "    B = x_batch.shape[0]\n",
    "    O = y_batch.shape[-1]\n",
    "    D = n_params\n",
    "    N = N_datapoints_max\n",
    "\n",
    "    # hessian_scaler = 1\n",
    "\n",
    "    vparams = tm.Vector(params)\n",
    "\n",
    "    rho = 1.\n",
    "    nll = lambda x, y, rho: 1/B * cross_entropy_loss(x, y, rho)\n",
    "\n",
    "    y_pred = model.apply(params, x_batch)\n",
    "\n",
    "    loglike_loss = nll(y_pred, y_batch, rho) #* hessian_scaler\n",
    "\n",
    "    log_prior_term = -D / 2 * jnp.log(2 * jnp.pi) - (1 / 2) * alpha * (vparams @ vparams) + D / 2 * jnp.log(alpha)\n",
    "    # log_det_term = 0\n",
    "    loss = loglike_loss - 0. * log_prior_term\n",
    "\n",
    "    return loss\n",
    "\n",
    "def make_step(params, alpha, opt_state, x, y):\n",
    "    grad_fn = jax.value_and_grad(map_loss, argnums=0, has_aux=False)\n",
    "    loss, grads = grad_fn(params, model, x, y, alpha, n_params, N)\n",
    "    param_updates, opt_state = optim.update(grads, opt_state)\n",
    "    params = optax.apply_updates(params, param_updates)\n",
    "    return loss, params, opt_state\n",
    "\n",
    "jit_make_step = jit(make_step)\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "epoch=1, loss=25.162, , accuracy=0.17, alpha=1.00, time=1.506s\n",
      "epoch=2, loss=23.608, , accuracy=0.25, alpha=1.00, time=0.348s\n",
      "epoch=3, loss=21.387, , accuracy=0.45, alpha=1.00, time=0.351s\n",
      "epoch=4, loss=18.559, , accuracy=0.53, alpha=1.00, time=0.345s\n",
      "epoch=5, loss=15.972, , accuracy=0.57, alpha=1.00, time=0.346s\n",
      "epoch=6, loss=13.639, , accuracy=0.64, alpha=1.00, time=0.349s\n",
      "epoch=7, loss=12.058, , accuracy=0.68, alpha=1.00, time=0.346s\n",
      "epoch=8, loss=10.933, , accuracy=0.71, alpha=1.00, time=0.351s\n",
      "epoch=9, loss=10.108, , accuracy=0.73, alpha=1.00, time=0.338s\n",
      "epoch=10, loss=9.513, , accuracy=0.74, alpha=1.00, time=0.339s\n",
      "epoch=11, loss=8.994, , accuracy=0.76, alpha=1.00, time=0.341s\n",
      "epoch=12, loss=8.700, , accuracy=0.76, alpha=1.00, time=0.350s\n",
      "epoch=13, loss=8.272, , accuracy=0.77, alpha=1.00, time=0.346s\n",
      "epoch=14, loss=8.000, , accuracy=0.79, alpha=1.00, time=0.338s\n",
      "epoch=15, loss=7.700, , accuracy=0.78, alpha=1.00, time=0.341s\n",
      "epoch=16, loss=7.675, , accuracy=0.79, alpha=1.00, time=0.345s\n",
      "epoch=17, loss=7.449, , accuracy=0.80, alpha=1.00, time=0.341s\n",
      "epoch=18, loss=7.165, , accuracy=0.81, alpha=1.00, time=0.339s\n",
      "epoch=19, loss=6.999, , accuracy=0.81, alpha=1.00, time=0.351s\n",
      "epoch=20, loss=6.835, , accuracy=0.81, alpha=1.00, time=0.344s\n",
      "epoch=21, loss=6.760, , accuracy=0.81, alpha=1.00, time=0.338s\n",
      "epoch=22, loss=6.568, , accuracy=0.82, alpha=1.00, time=0.352s\n",
      "epoch=23, loss=6.507, , accuracy=0.82, alpha=1.00, time=0.343s\n",
      "epoch=24, loss=6.306, , accuracy=0.82, alpha=1.00, time=0.346s\n",
      "epoch=25, loss=6.293, , accuracy=0.82, alpha=1.00, time=0.341s\n",
      "epoch=26, loss=6.196, , accuracy=0.82, alpha=1.00, time=0.345s\n",
      "epoch=27, loss=6.052, , accuracy=0.83, alpha=1.00, time=0.342s\n",
      "epoch=28, loss=6.089, , accuracy=0.83, alpha=1.00, time=0.343s\n",
      "epoch=29, loss=6.059, , accuracy=0.82, alpha=1.00, time=0.343s\n",
      "epoch=30, loss=5.988, , accuracy=0.83, alpha=1.00, time=0.350s\n",
      "epoch=31, loss=5.828, , accuracy=0.83, alpha=1.00, time=0.344s\n",
      "epoch=32, loss=5.874, , accuracy=0.83, alpha=1.00, time=0.346s\n",
      "epoch=33, loss=5.766, , accuracy=0.83, alpha=1.00, time=0.354s\n",
      "epoch=34, loss=5.730, , accuracy=0.83, alpha=1.00, time=0.339s\n",
      "epoch=35, loss=5.729, , accuracy=0.83, alpha=1.00, time=0.346s\n",
      "epoch=36, loss=5.615, , accuracy=0.84, alpha=1.00, time=0.347s\n",
      "epoch=37, loss=5.610, , accuracy=0.84, alpha=1.00, time=0.343s\n",
      "epoch=38, loss=5.588, , accuracy=0.84, alpha=1.00, time=0.341s\n",
      "epoch=39, loss=5.536, , accuracy=0.84, alpha=1.00, time=0.348s\n",
      "epoch=40, loss=5.495, , accuracy=0.84, alpha=1.00, time=0.354s\n",
      "epoch=41, loss=5.377, , accuracy=0.84, alpha=1.00, time=0.350s\n",
      "epoch=42, loss=5.414, , accuracy=0.84, alpha=1.00, time=0.339s\n",
      "epoch=43, loss=5.323, , accuracy=0.85, alpha=1.00, time=0.343s\n",
      "epoch=44, loss=5.293, , accuracy=0.85, alpha=1.00, time=0.348s\n",
      "epoch=45, loss=5.170, , accuracy=0.85, alpha=1.00, time=0.342s\n",
      "epoch=46, loss=5.246, , accuracy=0.85, alpha=1.00, time=0.342s\n",
      "epoch=47, loss=5.217, , accuracy=0.85, alpha=1.00, time=0.347s\n",
      "epoch=48, loss=5.169, , accuracy=0.85, alpha=1.00, time=0.340s\n",
      "epoch=49, loss=5.054, , accuracy=0.86, alpha=1.00, time=0.340s\n",
      "epoch=50, loss=5.179, , accuracy=0.85, alpha=1.00, time=0.343s\n",
      "epoch=51, loss=5.240, , accuracy=0.85, alpha=1.00, time=0.339s\n",
      "epoch=52, loss=5.027, , accuracy=0.85, alpha=1.00, time=0.340s\n",
      "epoch=53, loss=5.040, , accuracy=0.86, alpha=1.00, time=0.347s\n",
      "epoch=54, loss=4.898, , accuracy=0.86, alpha=1.00, time=0.365s\n",
      "epoch=55, loss=4.978, , accuracy=0.86, alpha=1.00, time=0.336s\n",
      "epoch=56, loss=4.984, , accuracy=0.86, alpha=1.00, time=0.339s\n",
      "epoch=57, loss=5.056, , accuracy=0.86, alpha=1.00, time=0.340s\n",
      "epoch=58, loss=5.010, , accuracy=0.86, alpha=1.00, time=0.333s\n",
      "epoch=59, loss=4.992, , accuracy=0.85, alpha=1.00, time=0.349s\n",
      "epoch=60, loss=4.884, , accuracy=0.86, alpha=1.00, time=0.337s\n",
      "epoch=61, loss=4.987, , accuracy=0.86, alpha=1.00, time=0.337s\n",
      "epoch=62, loss=4.917, , accuracy=0.86, alpha=1.00, time=0.345s\n",
      "epoch=63, loss=4.813, , accuracy=0.86, alpha=1.00, time=0.339s\n",
      "epoch=64, loss=4.886, , accuracy=0.86, alpha=1.00, time=0.345s\n",
      "epoch=65, loss=4.867, , accuracy=0.86, alpha=1.00, time=0.343s\n",
      "epoch=66, loss=4.875, , accuracy=0.86, alpha=1.00, time=0.344s\n",
      "epoch=67, loss=4.818, , accuracy=0.86, alpha=1.00, time=0.344s\n",
      "epoch=68, loss=4.729, , accuracy=0.86, alpha=1.00, time=0.338s\n",
      "epoch=69, loss=4.843, , accuracy=0.87, alpha=1.00, time=0.339s\n",
      "epoch=70, loss=4.758, , accuracy=0.86, alpha=1.00, time=0.339s\n",
      "epoch=71, loss=4.771, , accuracy=0.86, alpha=1.00, time=0.341s\n",
      "epoch=72, loss=4.763, , accuracy=0.86, alpha=1.00, time=0.340s\n",
      "epoch=73, loss=4.724, , accuracy=0.86, alpha=1.00, time=0.347s\n",
      "epoch=74, loss=4.669, , accuracy=0.86, alpha=1.00, time=0.348s\n",
      "epoch=75, loss=4.816, , accuracy=0.86, alpha=1.00, time=0.346s\n",
      "epoch=76, loss=4.608, , accuracy=0.87, alpha=1.00, time=0.340s\n",
      "epoch=77, loss=4.647, , accuracy=0.87, alpha=1.00, time=0.353s\n",
      "epoch=78, loss=4.640, , accuracy=0.86, alpha=1.00, time=0.343s\n",
      "epoch=79, loss=4.622, , accuracy=0.87, alpha=1.00, time=0.345s\n",
      "epoch=80, loss=4.687, , accuracy=0.87, alpha=1.00, time=0.343s\n",
      "epoch=81, loss=4.622, , accuracy=0.87, alpha=1.00, time=0.346s\n",
      "epoch=82, loss=4.556, , accuracy=0.87, alpha=1.00, time=0.341s\n",
      "epoch=83, loss=4.577, , accuracy=0.87, alpha=1.00, time=0.344s\n",
      "epoch=84, loss=4.541, , accuracy=0.87, alpha=1.00, time=0.347s\n",
      "epoch=85, loss=4.536, , accuracy=0.87, alpha=1.00, time=0.340s\n",
      "epoch=86, loss=4.470, , accuracy=0.88, alpha=1.00, time=0.342s\n",
      "epoch=87, loss=4.442, , accuracy=0.87, alpha=1.00, time=0.340s\n",
      "epoch=88, loss=4.467, , accuracy=0.87, alpha=1.00, time=0.343s\n",
      "epoch=89, loss=4.508, , accuracy=0.87, alpha=1.00, time=0.343s\n",
      "epoch=90, loss=4.462, , accuracy=0.88, alpha=1.00, time=0.337s\n",
      "epoch=91, loss=4.459, , accuracy=0.87, alpha=1.00, time=0.339s\n",
      "epoch=92, loss=4.545, , accuracy=0.87, alpha=1.00, time=0.340s\n",
      "epoch=93, loss=4.438, , accuracy=0.88, alpha=1.00, time=0.341s\n",
      "epoch=94, loss=4.484, , accuracy=0.87, alpha=1.00, time=0.358s\n",
      "epoch=95, loss=4.416, , accuracy=0.88, alpha=1.00, time=0.336s\n",
      "epoch=96, loss=4.401, , accuracy=0.87, alpha=1.00, time=0.339s\n",
      "epoch=97, loss=4.429, , accuracy=0.88, alpha=1.00, time=0.349s\n",
      "epoch=98, loss=4.448, , accuracy=0.88, alpha=1.00, time=0.348s\n",
      "epoch=99, loss=4.387, , accuracy=0.88, alpha=1.00, time=0.362s\n",
      "epoch=100, loss=4.369, , accuracy=0.88, alpha=1.00, time=0.354s\n"
     ]
    }
   ],
   "source": [
    "for epoch in range(1, n_epochs + 1):\n",
    "    epoch_loss = 0\n",
    "    epoch_accuracy = 0\n",
    "    start_time = time.time()\n",
    "    for _, batch in zip(range(n_batches), train_loader):\n",
    "        X = batch[\"image\"]\n",
    "        y = batch[\"label\"]\n",
    "        B = X.shape[0]\n",
    "        train_key, split_key = random.split(split_key)\n",
    "\n",
    "        loss, params, opt_state = jit_make_step(params, alpha, opt_state, X, y)\n",
    "        loss = loss\n",
    "        epoch_loss += loss.item()\n",
    "\n",
    "        epoch_accuracy += accuracy(params, model, X, y).item()\n",
    "\n",
    "    epoch_accuracy /= (n_batches * B)\n",
    "    epoch_time = time.time() - start_time\n",
    "    print(\n",
    "        f\"epoch={epoch}, loss={epoch_loss:.3f}, , accuracy={epoch_accuracy:.2f}, alpha={alpha:.2f}, time={epoch_time:.3f}s\"\n",
    "    )\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Sampling"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "sampling_train_loader = torch.utils.data.DataLoader(\n",
    "    data_train, batch_size=N, shuffle=True, collate_fn=numpy_collate_fn, drop_last=True,\n",
    ")\n",
    "data = next(iter(sampling_train_loader))\n",
    "x_train = jnp.array(data[\"image\"])\n",
    "y_train = jnp.array(data[\"label\"])\n",
    "sampling_val_loader = torch.utils.data.DataLoader(\n",
    "    data_test, batch_size=N_test, shuffle=True, collate_fn=numpy_collate_fn, drop_last=True,\n",
    ")\n",
    "data = next(iter(sampling_val_loader))\n",
    "x_val = jnp.array(data[\"image\"])\n",
    "y_val = jnp.array(data[\"label\"])\n",
    "\n",
    "sample_key = jax.random.PRNGKey(0)\n",
    "n_posterior_samples = 200\n",
    "num_iterations = 1\n",
    "n_sample_batch_size = 1\n",
    "n_sample_batches = N // n_sample_batch_size"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "from src.sampling.exact_ggn import exact_ggn_laplace\n",
    "from src.sampling.laplace_ode import ode_ggn\n",
    "from src.sampling.lanczos_diffusion import lanczos_diffusion\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "_model_fn = lambda params, x: model.apply(params, x[None, ...])[0]\n",
    "ggn = calculate_exact_ggn(cross_entropy_loss, _model_fn, params, x_train[:120], y_train[:120], n_params)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Array(226, dtype=int32)"
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "jnp.linalg.matrix_rank(ggn)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(3000, 1, 28, 28)"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "x_train.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [],
   "source": [
    "sample_key = jax.random.PRNGKey(0)\n",
    "model_fn = lambda params, x: model.apply(params, x[None, ...])[0]\n",
    "rank = 10\n",
    "alpha = 1.0\n",
    "n_posterior_samples = 200\n",
    "var = 0.1\n",
    "\n",
    "lr_posterior_samples, posterior_samples, isotropic_posterior_samples = exact_ggn_laplace(cross_entropy_loss, \n",
    "                                                                                         model_fn,\n",
    "                                                                                         params,\n",
    "                                                                                         x_train[:30],\n",
    "                                                                                         y_train[:30],\n",
    "                                                                                         n_params,\n",
    "                                                                                         rank,\n",
    "                                                                                         alpha,\n",
    "                                                                                         n_posterior_samples,\n",
    "                                                                                         sample_key,\n",
    "                                                                                         var,\n",
    "                                                                                         \"all\"\n",
    "                                                                                         )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [],
   "source": [
    "from functools import partial\n",
    "\n",
    "# @partial(jax.jit, static_argnames=['n_data_pts'])\n",
    "def rank_expt(n_data_pts, train_stats=True, mean_acc=True):\n",
    "    sample_key = jax.random.PRNGKey(0)\n",
    "    model_fn = lambda params, x: model.apply(params, x[None, ...])[0]\n",
    "    rank = 100\n",
    "    alpha = 1.0\n",
    "    n_posterior_samples = 200\n",
    "    var = 0.1\n",
    "\n",
    "    lr_posterior_samples, posterior_samples, isotropic_posterior_samples = exact_ggn_laplace(cross_entropy_loss, \n",
    "                                                                   model_fn,\n",
    "                                                                   params,\n",
    "                                                                   x_train[:n_data_pts],\n",
    "                                                                   y_train[:n_data_pts],\n",
    "                                                                   n_params,\n",
    "                                                                   rank,\n",
    "                                                                   alpha,\n",
    "                                                                   n_posterior_samples,\n",
    "                                                                   sample_key,\n",
    "                                                                   var,\n",
    "                                                                   \"all\"\n",
    "                                                                   )\n",
    "    \n",
    "    if train_stats:\n",
    "        n_data_pts = 3000\n",
    "        lr_predictive = sample_predictive(lr_posterior_samples, params, model, x_train[:n_data_pts], False, \"Pytree\")\n",
    "        predictive = sample_predictive(posterior_samples, params, model, x_train[:n_data_pts], False, \"Pytree\")\n",
    "        linearised_predictive = sample_predictive(posterior_samples, params, model, x_train[:n_data_pts], True, \"Pytree\")\n",
    "        isotropic_predictive = sample_predictive(isotropic_posterior_samples, params, model, x_train[:n_data_pts], False, \"Pytree\")\n",
    "        map_acc = accuracy(params, model, x_train[:n_data_pts], y_train[:n_data_pts])/n_data_pts\n",
    "        if mean_acc:\n",
    "            lr_acc = jnp.mean(jax.vmap(accuracy_preds, in_axes=(0,None))(lr_predictive, y_train[:n_data_pts])/n_data_pts)\n",
    "            linearised_acc = jnp.mean(jax.vmap(accuracy_preds, in_axes=(0,None))(linearised_predictive, y_train[:n_data_pts])/n_data_pts)\n",
    "            full_acc = jnp.mean(jax.vmap(accuracy_preds, in_axes=(0,None))(predictive, y_train[:n_data_pts])/n_data_pts)\n",
    "            isotropic_acc = jnp.mean(jax.vmap(accuracy_preds, in_axes=(0,None))(isotropic_predictive, y_train[:n_data_pts])/n_data_pts)\n",
    "        else:\n",
    "            lr_acc = accuracy_preds(lr_predictive.mean(axis=0), y_train[:n_data_pts])/ n_data_pts \n",
    "            linearised_acc = accuracy_preds(linearised_predictive.mean(axis=0), y_train[:n_data_pts])/ n_data_pts \n",
    "            full_acc = accuracy_preds(predictive.mean(axis=0), y_train[:n_data_pts])/ n_data_pts \n",
    "            isotropic_acc = accuracy_preds(isotropic_predictive.mean(axis=0), y_train[:n_data_pts])/ n_data_pts\n",
    "        return map_acc, lr_acc, linearised_acc, full_acc, isotropic_acc\n",
    "        \n",
    "    else:\n",
    "        N_val = 1000\n",
    "        lr_predictive = sample_predictive(lr_posterior_samples, params, model, x_val[:N_val], False, \"Pytree\")\n",
    "        predictive = sample_predictive(posterior_samples, params, model, x_val[:N_val], False, \"Pytree\")\n",
    "        linearised_predictive = sample_predictive(posterior_samples, params, model, x_val[:N_val], True, \"Pytree\")\n",
    "        isotropic_predictive = sample_predictive(isotropic_posterior_samples, params, model, x_val[:N_val], False, \"Pytree\")\n",
    "        map_acc = accuracy(params, model, x_train[:n_data_pts], y_train[:n_data_pts])/n_data_pts\n",
    "        if mean_acc:\n",
    "            lr_acc = jnp.mean(jax.vmap(accuracy_preds, in_axes=(0,None))(lr_predictive, y_train[:N_val])/N_val)\n",
    "            linearised_acc = jnp.mean(jax.vmap(accuracy_preds, in_axes=(0,None))(linearised_predictive, y_train[:N_val])/N_val)\n",
    "            full_acc = jnp.mean(jax.vmap(accuracy_preds, in_axes=(0,None))(predictive, y_train[:N_val])/N_val)\n",
    "            isotropic_acc = jnp.mean(jax.vmap(accuracy_preds, in_axes=(0,None))(isotropic_predictive, y_train[:N_val])/N_val)\n",
    "        else:\n",
    "            lr_acc = accuracy_preds(lr_predictive.mean(axis=0), y_train[:N_val])/ N_val \n",
    "            linearised_acc = accuracy_preds(linearised_predictive.mean(axis=0), y_train[:N_val])/ N_val \n",
    "            full_acc = accuracy_preds(predictive.mean(axis=0), y_train[:N_val])/ N_val \n",
    "            isotropic_acc = accuracy_preds(isotropic_predictive.mean(axis=0), y_train[:N_val])/ N_val\n",
    "        return map_acc, lr_acc, linearised_acc, full_acc, isotropic_acc\n",
    "    \n",
    "\n",
    "    \n",
    "        "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [],
   "source": [
    "n_data_pt_list = jnp.array([ 50, 75, 100, 120, 150, 200, 250, 300, 500, 1000])\n",
    "map_acc_list, lr_acc_list, linearised_acc_list, full_acc_list, isotropic_acc_list = [], [], [], [], []\n",
    "for n_data_pt in n_data_pt_list:\n",
    "    map_acc, lr_acc, linearised_acc, full_acc, isotropic_acc = rank_expt(n_data_pt)\n",
    "    map_acc_list.append(map_acc)\n",
    "    lr_acc_list.append(lr_acc)\n",
    "    linearised_acc_list.append(linearised_acc)\n",
    "    full_acc_list.append(full_acc)\n",
    "    isotropic_acc_list.append(isotropic_acc)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.legend.Legend at 0x7f8997db7820>"
      ]
     },
     "execution_count": 17,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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fv364ejoiLu7O3feeSfHjh0zalNQUMBTTz1F69atcXBw4J577iElJcWoTUJCAqNHj8bOzg53d3deeOEFSkpKGvKtNHlRUVGoVCqmT59uOCb3vv4kJiby0EMP0bp1a2xtbenRowd79+41nFcUhblz5+Ll5YWtrS1hYWGcOHHC6Brp6emMGzcOJycnXFxceOSRR8jJyWnot9KkaLVaXn75Zfz9/bG1taVjx44sWrTIaK8hufem8ffffzNmzBi8vb1RqVT88MMPRudNdZ//+ecfBg8ejI2NDb6+vrz66qumeQOKENcpPDxc+fjjj5XDhw8rBw4cUEaNGqW0a9dOycnJMbR54oknFF9fX2XLli3K3r17lf79+ysDBgwwnC8pKVG6d++uhIWFKfv371d+/fVXpU2bNsqsWbPM8ZaapJiYGMXPz0/p2bOnMm3aNMNxuff1Iz09XWnfvr0yceJEZffu3crp06eV33//XTl58qShTVRUlOLs7Kz88MMPysGDB5Xbb79d8ff3V/Lz8w1tRowYofTq1UvZtWuXsnXrVqVTp07Kgw8+aI631GS88sorSuvWrZWff/5ZOXPmjPLNN98oDg4OyltvvWVoI/feNH799Vdl9uzZyvr16xVA+f77743Om+I+Z2ZmKh4eHsq4ceOUw4cPK1999ZVia2urvP/++3WOX5IbYTKpqakKoPz111+KoijK5cuXFUtLS+Wbb74xtDly5IgCKDt37lQURf8DpFarleTkZEObVatWKU5OTkphYWHDvoEmKDs7W+ncubOyadMmZejQoYbkRu59/ZkxY4YyaNCgSs/rdDrF09NTee211wzHLl++rFhbWytfffWVoiiKEhcXpwDKnj17DG1+++03RaVSKYmJifUXfBM3evRoZfLkyUbH7r77bmXcuHGKosi9ry/XJjemus/vvvuu0qpVK6PfNzNmzFC6dOlS55hlWEqYTGZmJgCurq4A7Nu3j+LiYsLCwgxtunbtSrt27di5cycAO3fupEePHnh4eBjahIeHk5WVxb///tuA0TdNTz31FKNHjza6xyD3vj5t2LCBoKAg7rvvPtzd3enTpw8ffvih4fyZM2dITk42uvfOzs6EhIQY3XsXFxeCgoIMbcLCwlCr1ezevbvh3kwTM2DAALZs2cLx48cBOHjwINu2bWPkyJGA3PuGYqr7vHPnToYMGYKV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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "map_acc_list, lr_acc_list, linearised_acc_list, full_acc_list, isotropic_acc_list = jnp.array(map_acc_list), jnp.array(lr_acc_list), jnp.array(linearised_acc_list), jnp.array(full_acc_list), jnp.array(isotropic_acc_list)\n",
    "plt.plot(n_data_pt_list, lr_acc_list, label=\"Low Rank\", marker=\"o\")\n",
    "plt.plot(n_data_pt_list, linearised_acc_list, label=\"Linearised\", marker=\"o\")\n",
    "plt.plot(n_data_pt_list, full_acc_list, label=\"Sampled Laplace\", marker=\"o\")\n",
    "plt.plot(n_data_pt_list, map_acc_list, label=\"Map accuracy\", marker=\"o\")\n",
    "# plt.plot(n_data_pt_list, isotropic_acc_list, label=\"Isotropic accuracy\", marker=\"o\")\n",
    "plt.xlabel(\"Rank\")\n",
    "# plt.xticks(n_data_pt_list)\n",
    "plt.ylabel(\"Train Accuracy\")\n",
    "plt.legend()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 50,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "MAP acc: 0.8333333\n",
      "lr acc: 0.79800004\n",
      "full acc: 0.14366667\n",
      "isotropic acc: 0.7725\n"
     ]
    }
   ],
   "source": [
    "N_val = 1000\n",
    "lr_predictive = sample_predictive(lr_posterior_samples, params, model, x_train[:30], False, \"Pytree\")\n",
    "predictive = sample_predictive(posterior_samples, params, model, x_train[:30], False, \"Pytree\")\n",
    "isotropic_predictive = sample_predictive(isotropic_posterior_samples, params, model, x_train[:30], False, \"Pytree\")\n",
    "preds = model.apply(params, x_train[:30])\n",
    "print(\"MAP acc:\", accuracy(params, model, x_train[:30], y_train[:30])/x_train[:30].shape[0])\n",
    "print(\"lr acc:\", jnp.mean(jax.vmap(accuracy_preds, in_axes=(0,None))(lr_predictive, y_train[:30])/x_train[:30].shape[0]))\n",
    "print(\"full acc:\", jnp.mean(jax.vmap(accuracy_preds, in_axes=(0,None))(predictive, y_train[:30])/x_train[:30].shape[0]))\n",
    "print(\"isotropic acc:\", jnp.mean(jax.vmap(accuracy_preds, in_axes=(0,None))(isotropic_predictive, y_train[:30])/x_train[:30].shape[0]))\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 59,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Array(0.8333333, dtype=float32)"
      ]
     },
     "execution_count": 59,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "accuracy_preds(isotropic_predictive.mean(axis=0), y_train[:30])/30"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 48,
   "metadata": {},
   "outputs": [],
   "source": [
    "N_val = 1000\n",
    "lr_predictive = sample_predictive(lr_posterior_samples, params, model, x_val[:N_val], False, \"Pytree\")\n",
    "predictive = sample_predictive(posterior_samples, params, model, x_val[:N_val], False, \"Pytree\")\n",
    "isotropic_predictive = sample_predictive(isotropic_posterior_samples, params, model, x_val[:N_val], False, \"Pytree\")\n",
    "preds = model.apply(params, x_val[:N_val])\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 49,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "MAP acc: 0.853\n",
      "lr acc: 0.81091\n",
      "full acc: 0.15518999\n",
      "isotropic acc: 0.78494996\n"
     ]
    }
   ],
   "source": [
    "print(\"MAP acc:\", accuracy(params, model, x_val[:N_val], y_val[:N_val])/x_val[:N_val].shape[0])\n",
    "print(\"lr acc:\", jnp.mean(jax.vmap(accuracy_preds, in_axes=(0,None))(lr_predictive, y_val[:N_val])/x_val[:N_val].shape[0]))\n",
    "print(\"full acc:\", jnp.mean(jax.vmap(accuracy_preds, in_axes=(0,None))(predictive, y_val[:N_val])/x_val[:N_val].shape[0]))\n",
    "print(\"isotropic acc:\", jnp.mean(jax.vmap(accuracy_preds, in_axes=(0,None))(isotropic_predictive, y_val[:N_val])/x_val[:N_val].shape[0]))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 46,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "MAP nll: 473.9449\n",
      "lr nll: 670.9823\n",
      "full nll: 7545.1475\n",
      "isotropic nll: 685.7618\n"
     ]
    }
   ],
   "source": [
    "print(\"MAP nll:\", nll(preds, y_val[:N_val]))\n",
    "print(\"lr nll:\", nll(lr_predictive, y_val[:N_val]))\n",
    "print(\"full nll:\", nll(predictive, y_val[:N_val]))\n",
    "print(\"isotropic nll:\", nll(isotropic_predictive, y_val[:N_val]))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [],
   "source": [
    "lr_predictive = sample_predictive(lr_posterior_samples, params, model, x_val[:N_val], True, \"Pytree\")\n",
    "predictive = sample_predictive(posterior_samples, params, model, x_val[:N_val], True, \"Pytree\")\n",
    "isotropic_predictive = sample_predictive(isotropic_posterior_samples, params, model, x_val[:N_val], True, \"Pytree\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(200, 1000, 10)"
      ]
     },
     "execution_count": 15,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "isotropic_predictive.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "MAP acc: 0.843\n",
      "lr acc: 0.842125\n",
      "full acc: 0.812935\n",
      "isotropic acc: 0.77391493\n"
     ]
    }
   ],
   "source": [
    "print(\"MAP acc:\", accuracy(params, model, x_val[:N_val], y_val[:N_val])/x_val[:N_val].shape[0])\n",
    "print(\"lr acc:\", jnp.mean(jax.vmap(accuracy_preds, in_axes=(0,None))(lr_predictive, y_val[:N_val])/x_val[:N_val].shape[0]))\n",
    "print(\"full acc:\", jnp.mean(jax.vmap(accuracy_preds, in_axes=(0,None))(predictive, y_val[:N_val])/x_val[:N_val].shape[0]))\n",
    "print(\"isotropic acc:\", jnp.mean(jax.vmap(accuracy_preds, in_axes=(0,None))(isotropic_predictive, y_val[:N_val])/x_val[:N_val].shape[0]))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "MAP nll: 537.5515\n",
      "lr nll: 540.08307\n",
      "full nll: 624.62225\n",
      "isotropic nll: 747.6211\n"
     ]
    }
   ],
   "source": [
    "print(\"MAP nll:\", nll(preds, y_val[:N_val]))\n",
    "print(\"lr nll:\", nll(lr_predictive, y_val[:N_val]))\n",
    "print(\"full nll:\", nll(predictive, y_val[:N_val]))\n",
    "print(\"isotropic nll:\", nll(isotropic_predictive, y_val[:N_val]))\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## OOD"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {},
   "outputs": [],
   "source": [
    "from src.data.datasets import (\n",
    "    get_rotated_mnist_loaders,\n",
    "    load_corrupted_cifar10,\n",
    "    get_mnist_ood_loaders,\n",
    "    get_cifar10_ood_loaders,\n",
    "    get_cifar10_train_set,\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {},
   "outputs": [],
   "source": [
    "eval_args = {}\n",
    "eval_args[\"linearised_laplace\"] = False\n",
    "eval_args[\"posterior_sample_type\"] = \"Pytree\"\n",
    "eval_args[\"likelihood\"] = \"classification\"\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 52,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "R-MNIST with distribution shift intensity 0\n",
      "conf: 0.8293, nll: 297.9814, acc: 0.8140, brier: 0.2763, ece: 0.1219, mce: 0.7098\n",
      "R-MNIST with distribution shift intensity 1\n",
      "conf: 0.7904, nll: 402.6505, acc: 0.7320, brier: 0.3799, ece: 0.1402, mce: 0.6985\n",
      "R-MNIST with distribution shift intensity 2\n",
      "conf: 0.7345, nll: 886.8191, acc: 0.4980, brier: 0.7143, ece: 0.2762, mce: 0.7698\n",
      "R-MNIST with distribution shift intensity 3\n",
      "conf: 0.6773, nll: 1994.0592, acc: 0.1980, brier: 1.1895, ece: 0.4839, mce: 0.9056\n",
      "R-MNIST with distribution shift intensity 4\n",
      "conf: 0.6894, nll: 2437.9873, acc: 0.1620, brier: 1.2652, ece: 0.5363, mce: 0.9752\n",
      "R-MNIST with distribution shift intensity 5\n",
      "conf: 0.6485, nll: 2278.5176, acc: 0.1620, brier: 1.2186, ece: 0.4859, mce: 0.8357\n",
      "R-MNIST with distribution shift intensity 6\n",
      "conf: 0.6566, nll: 2078.4587, acc: 0.1760, brier: 1.1976, ece: 0.4802, mce: 0.9545\n",
      "R-MNIST with distribution shift intensity 7\n",
      "conf: 0.7467, nll: 2250.4639, acc: 0.2840, brier: 1.1244, ece: 0.4850, mce: 0.8760\n",
      "R-MNIST with distribution shift intensity 8\n",
      "conf: 0.7026, nll: 2193.2632, acc: 0.2060, brier: 1.1950, ece: 0.5045, mce: 0.8464\n",
      "R-MNIST with distribution shift intensity 9\n",
      "conf: 0.6688, nll: 2475.7488, acc: 0.0940, brier: 1.3525, ece: 0.5679, mce: 0.9546\n",
      "R-MNIST with distribution shift intensity 10\n",
      "conf: 0.6455, nll: 2425.1375, acc: 0.1180, brier: 1.2952, ece: 0.5245, mce: 0.9265\n",
      "R-MNIST with distribution shift intensity 11\n",
      "conf: 0.6277, nll: 2180.3149, acc: 0.1460, brier: 1.2035, ece: 0.4833, mce: 0.9335\n",
      "R-MNIST with distribution shift intensity 12\n",
      "conf: 0.6645, nll: 997.9722, acc: 0.4720, brier: 0.7187, ece: 0.2311, mce: 0.7249\n",
      "R-MNIST with distribution shift intensity 13\n",
      "conf: 0.7573, nll: 474.7219, acc: 0.6980, brier: 0.4231, ece: 0.1392, mce: 0.6969\n",
      "R-MNIST with distribution shift intensity 14\n",
      "conf: 0.8293, nll: 297.9814, acc: 0.8140, brier: 0.2763, ece: 0.1219, mce: 0.7098\n"
     ]
    }
   ],
   "source": [
    "from src.ood_functions.evaluate import evaluate\n",
    "from src.ood_functions.metrics import compute_metrics\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 = []\n",
    "for i, id in enumerate(ids):    \n",
    "    _, test_loader = get_rotated_mnist_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_samples, params, model, 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-MNIST\"\n",
    "        )\n",
    "    metrics.append({**some_metrics, **more_metrics})\n",
    "    print(\", \".join([f\"{k}: {v:.4f}\" for k, v in metrics[-1].items()]))\n",
    "    \n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 53,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "R-MNIST with distribution shift intensity 0\n",
      "conf: 0.8126, nll: 286.0074, acc: 0.8160, brier: 0.2681, ece: 0.1080, mce: 0.7015\n",
      "R-MNIST with distribution shift intensity 1\n",
      "conf: 0.7734, nll: 385.7637, acc: 0.7460, brier: 0.3662, ece: 0.1366, mce: 0.8287\n",
      "R-MNIST with distribution shift intensity 2\n",
      "conf: 0.7114, nll: 842.7404, acc: 0.5000, brier: 0.7018, ece: 0.2386, mce: 0.6550\n",
      "R-MNIST with distribution shift intensity 3\n",
      "conf: 0.6617, nll: 1906.2975, acc: 0.1900, brier: 1.1863, ece: 0.4762, mce: 0.8752\n",
      "R-MNIST with distribution shift intensity 4\n",
      "conf: 0.6748, nll: 2343.8108, acc: 0.1620, brier: 1.2489, ece: 0.5192, mce: 0.8548\n",
      "R-MNIST with distribution shift intensity 5\n",
      "conf: 0.6342, nll: 2185.4668, acc: 0.1640, brier: 1.1982, ece: 0.4690, mce: 0.9887\n",
      "R-MNIST with distribution shift intensity 6\n",
      "conf: 0.6332, nll: 1971.8813, acc: 0.1800, brier: 1.1739, ece: 0.4579, mce: 0.9133\n",
      "R-MNIST with distribution shift intensity 7\n",
      "conf: 0.7272, nll: 2125.1313, acc: 0.2820, brier: 1.1018, ece: 0.4448, mce: 0.8240\n",
      "R-MNIST with distribution shift intensity 8\n",
      "conf: 0.6844, nll: 2088.2256, acc: 0.1980, brier: 1.1944, ece: 0.5016, mce: 0.8848\n",
      "R-MNIST with distribution shift intensity 9\n",
      "conf: 0.6539, nll: 2327.7085, acc: 0.0980, brier: 1.3283, ece: 0.5571, mce: 0.9158\n",
      "R-MNIST with distribution shift intensity 10\n",
      "conf: 0.6328, nll: 2302.7290, acc: 0.1320, brier: 1.2758, ece: 0.5190, mce: 0.9379\n",
      "R-MNIST with distribution shift intensity 11\n",
      "conf: 0.6141, nll: 2076.3010, acc: 0.1600, brier: 1.1770, ece: 0.4531, mce: 0.9720\n",
      "R-MNIST with distribution shift intensity 12\n",
      "conf: 0.6422, nll: 972.1418, acc: 0.4580, brier: 0.7149, ece: 0.2090, mce: 0.6534\n",
      "R-MNIST with distribution shift intensity 13\n",
      "conf: 0.7300, nll: 461.4109, acc: 0.6920, brier: 0.4180, ece: 0.1220, mce: 0.7707\n",
      "R-MNIST with distribution shift intensity 14\n",
      "conf: 0.8126, nll: 286.0074, acc: 0.8160, brier: 0.2681, ece: 0.1080, mce: 0.7015\n"
     ]
    }
   ],
   "source": [
    "from src.ood_functions.evaluate import evaluate\n",
    "# from src.ood_functions.metrics import compute_metrics\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 = []\n",
    "for i, id in enumerate(ids):    \n",
    "    _, test_loader = get_rotated_mnist_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_samples, params, model, 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-MNIST\"\n",
    "        )\n",
    "    metrics.append({**some_metrics, **more_metrics})\n",
    "    print(\", \".join([f\"{k}: {v:.4f}\" for k, v in metrics[-1].items()]))\n",
    "    \n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Lanczos diffusion"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 59,
   "metadata": {},
   "outputs": [],
   "source": [
    "n_steps = 2\n",
    "n_samples = 50\n",
    "alpha = 10.0\n",
    "rank = 10\n",
    "nonker_posterior_samples = lanczos_diffusion(cross_entropy_loss, model.apply, params,n_steps,n_samples,alpha,sample_key,n_params,rank,x_train,y_train,1.0,\"non-kernel-eigvals\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 63,
   "metadata": {},
   "outputs": [],
   "source": [
    "nonker_predictive = sample_predictive(nonker_posterior_samples, params, model, x_val[:N_val], False, \"Pytree\")\n",
    "preds = model.apply(params, x_val[:N_val])\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 64,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "MAP acc: 0.832\n",
      "nonker acc: 0.83283997\n",
      "MAP nll: 520.5774\n",
      "nonker nll: 522.54913\n"
     ]
    }
   ],
   "source": [
    "print(\"MAP acc:\", accuracy(params, model, x_val[:N_val], y_val[:N_val])/x_val[:N_val].shape[0])\n",
    "print(\"nonker acc:\", jnp.mean(jax.vmap(accuracy_preds, in_axes=(0,None))(nonker_predictive, y_val[:N_val])/x_val[:N_val].shape[0]))\n",
    "print(\"MAP nll:\", nll(preds, y_val[:N_val]))\n",
    "print(\"nonker nll:\", nll(nonker_predictive, y_val[:N_val]))\n",
    "\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### ODE"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": []
  }
 ],
 "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.11"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
