{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "aa6a82a5-2260-42ed-92d0-65016223e5be",
   "metadata": {},
   "source": [
    "## Orthogonal Projection Baseline for PDEs (E2E Projection)\n",
    "\n",
    "This notebook implements an **orthogonal projection baseline** for constrained PDE modeling. Unlike the HardC baseline (which applies projection only at test time), this method applies a differentiable **orthogonal projection end-to-end (E2E)** during both training and inference. The projection enforces hard physical constraints—such as initial conditions or global conservation—directly on the predicted **mean** at every forward pass.\n",
    "\n",
    "The projection layer is fully differentiable, allowing gradients to flow through the projection operator, and enabling the model to learn constraint-compatible dynamics directly. Both mean and variance are updated.\n",
    "\n",
    "---\n",
    "\n",
    "### Running Different PDE Benchmarks\n",
    "\n",
    "To run experiments on different PDE datasets, modify the `args` dictionary. For example:\n",
    "\n",
    "```python\n",
    "args = {\n",
    " '--batch_size': '20',\n",
    " '--dataset': 'LinearAdvection_1D',\n",
    " '--dataset_params': '1,2',\n",
    " '--epochs': '200',\n",
    " '--fno_modes': '12',\n",
    " '--fno_width': '32',\n",
    " '--grid_len': '100',\n",
    " '--lr': '1e-3',\n",
    " '--m.drop_prob': '0.1',\n",
    " '--m.n_models': '10',\n",
    " '--m.n_regularize': '5',\n",
    " '--m.reg_strength': '1',\n",
    " '--m.reg_type': 'weights_l2',\n",
    " '--model': 'OutputVarFNO2d',\n",
    " '--n_samples': '200',\n",
    " '--no_train': False,\n",
    " '--ood_dataset_params': None,\n",
    " '--predict_time': '0,-1,5',\n",
    " '--seed': '0',\n",
    " '--time_len': '100',\n",
    " '--tplot': '0.5',\n",
    " '--train_ood_dataset_params': '1,2'\n",
    "}\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "id": "6b3ccdb8-4fd8-4224-b23e-979fa8228059",
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "os.environ['CUDA_VISIBLE_DEVICES'] = '2'\n",
    "\n",
    "import numpy as np\n",
    "import torch\n",
    "import matplotlib.pyplot as plt\n",
    "from models.FNO2d import FNO2d\n",
    "from models.DiverseFNO2d import DiverseFNO2d\n",
    "from models.UncertainNO import *\n",
    "import utils\n",
    "from einops import rearrange, reduce, repeat\n",
    "import os\n",
    "from docopt import docopt\n",
    "import dill\n",
    "from datasets import *\n",
    "import probconserv\n",
    "import sys\n",
    "import torch.optim as optim\n",
    "\n",
    "# args = docopt(__doc__)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "5fdcc8cd-82d9-4cec-888d-806541259f2c",
   "metadata": {},
   "outputs": [],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "1d5eb530-ab43-4054-8953-9efdb86b3106",
   "metadata": {},
   "outputs": [],
   "source": [
    "args = {'--batch_size': '20',\n",
    " '--dataset': 'LinearAdvection_1D',\n",
    " '--dataset_params': '2,3',\n",
    " '--epochs': '200',\n",
    " '--fno_modes': '12',\n",
    " '--fno_width': '32',\n",
    " '--grid_len': '100',\n",
    " '--lr': '1e-3',\n",
    " '--m.drop_prob': '0.1',\n",
    " '--m.n_models': '10',\n",
    " '--m.n_regularize': '5',\n",
    " '--m.reg_strength': '1',\n",
    " '--m.reg_type': 'weights_l2',\n",
    " '--model': 'OutputVarFNO2d',\n",
    " '--n_samples': '200',\n",
    " '--no_train': False,\n",
    " '--ood_dataset_params': None,\n",
    " '--predict_time': '0,-1,5',\n",
    " '--seed': '0',\n",
    " '--time_len': '100',\n",
    " '--tplot': '0.5',\n",
    " '--train_ood_dataset_params': '2,3'}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "a69b4337-c84b-4454-af1f-979e654417e3",
   "metadata": {},
   "outputs": [],
   "source": [
    "# lambdas_pinn = {}\n",
    "# lambdas_pinn['Heat_Eq]\n",
    "# heq = 1e4"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "9ac91626-a64a-4ae2-ae5f-f63f492a7d4d",
   "metadata": {},
   "outputs": [],
   "source": [
    "device = \"cuda\" if torch.cuda.is_available() else \"cpu\" \n",
    "experiment_name = \"trial\"\n",
    "# print(f\"Experiment: {experiment_name}\")\n",
    "# print(args)\n",
    "save_args = utils.filter_config(args, [\"generate\", \"--no_train\", \"--ood_dataset_params\", \"--tplot\"], mode=\"remove\")  # Also removes \".\" keys\n",
    "\n",
    "is_train = not bool(args[\"--no_train\"])\n",
    "\n",
    "# Parameters\n",
    "n_x = int(args[\"--grid_len\"])\n",
    "n_t = int(args[\"--time_len\"])\n",
    "n_samples = int(args[\"--n_samples\"])\n",
    "n_train = int(0.8 * n_samples)\n",
    "n_valid = int(0.2 * n_samples)\n",
    "n_test = n_samples // 2\n",
    "\n",
    "is_markov = False\n",
    "\n",
    "dataset = args[\"--dataset\"]\n",
    "dataset_params = [float(val) for val in args[\"--dataset_params\"].split(\",\")]\n",
    "train_ood_dataset_params = [float(val) for val in args[\"--train_ood_dataset_params\"].split(\",\")]\n",
    "ood_dataset_params = train_ood_dataset_params\n",
    "if not is_train:\n",
    "    ood_dataset_params = [float(val) for val in args[\"--ood_dataset_params\"].split(\",\")]\n",
    "\n",
    "tpred = [int(val) for val in args[\"--predict_time\"].split(\",\")]\n",
    "\n",
    "fno_modes = int(args[\"--fno_modes\"])\n",
    "fno_width = int(args[\"--fno_width\"])\n",
    "\n",
    "batch_size = int(args[\"--batch_size\"])\n",
    "lr = float(args[\"--lr\"])\n",
    "epochs = int(args[\"--epochs\"])\n",
    "step_size = 50\n",
    "gamma = 0.5\n",
    "# ################\n",
    "\n",
    "# Set seed\n",
    "utils.set_seed(int(args[\"--seed\"]))\n",
    "\n",
    "# Generate dataset\n",
    "if dataset.lower() == \"HeatEquation_1D\".lower():\n",
    "    t = torch.linspace(0, 1, n_t)\n",
    "    grid = torch.linspace(0, 2 * np.pi, n_x)\n",
    "    dataset_class = HeatEquation_1D\n",
    "elif dataset.lower() == \"PME_1D\".lower():\n",
    "    t = torch.linspace(0, 1, n_t)\n",
    "    grid = torch.linspace(0, 1, n_x)\n",
    "    dataset_class = PME_1D\n",
    "elif dataset.lower() == \"StefanPME_1D\".lower():\n",
    "    t = torch.linspace(0, 1, n_t)\n",
    "    grid = torch.linspace(0, 1, n_x)\n",
    "    dataset_class = StefanPME_1D\n",
    "elif dataset.lower() == \"LinearAdvection_1D\".lower():\n",
    "    t = torch.linspace(0, 1, n_t)\n",
    "    grid = torch.linspace(0, 1, n_x)\n",
    "    dataset_class = LinearAdvection_1D\n",
    "else:\n",
    "    raise NotImplementedError\n",
    "\n",
    "t_sliced = t[slice(*tpred)]\n",
    "T = len(t_sliced)\n",
    "\n",
    "def get_xy_from_pu(p, u, is_markov=False):\n",
    "    T = u.shape[2]\n",
    "    #TODO: What does is_markov do here?\n",
    "    if is_markov:\n",
    "        x0, y0 = p, u\n",
    "        \n",
    "        y0_vectorized = rearrange(y0[:, :, 0:T-1], \"nf nx nt 1 -> (nf nt) nx 1\")\n",
    "        x0 = repeat(x0, \"nf nx 1 -> (nf nt) nx 1\", nt=T-1)\n",
    "        x = torch.cat([x0, y0_vectorized], dim=-1)\n",
    "        \n",
    "        y = rearrange(y0[:, :, 1:T], \"nf nx nt 1 -> (nf nt) nx 1\")\n",
    "    else:\n",
    "        x, y = p, u\n",
    "        x = repeat(x, \"nf nx 1 -> nf nx T 1\", T=T)\n",
    "    return x, y\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "b02f6033-3977-4d51-8f39-024a376dbabf",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Here 160\n",
      "torch.Size([160, 100, 1]) torch.Size([160, 100, 20, 1])\n",
      "torch.Size([160, 100, 20, 1]) torch.Size([160, 100, 20, 1])\n"
     ]
    }
   ],
   "source": [
    "if is_train:\n",
    "    # Train data\n",
    "    print(\"Here\", n_train)\n",
    "    a, u, p = dataset_class.generate_dataset(n_train, grid, t, tpred, *dataset_params)\n",
    "    print(a.shape, u.shape)\n",
    "    x_train, y_train = get_xy_from_pu(p, u, is_markov=is_markov)\n",
    "\n",
    "    # Validation data\n",
    "    a, u, p = dataset_class.generate_dataset(n_valid, grid, t, tpred, *dataset_params)\n",
    "    x_valid, y_valid = get_xy_from_pu(p, u, is_markov=is_markov)\n",
    "\n",
    "    # In-distribution test data\n",
    "    a, u, p = dataset_class.generate_dataset(n_test, grid, t, tpred, *dataset_params)\n",
    "    x_id_test, y_id_test = get_xy_from_pu(p, u, is_markov=is_markov)\n",
    "\n",
    "    # Out-of-distribution inputs only\n",
    "    a, u, p = dataset_class.generate_dataset(n_test, grid, t, tpred, *train_ood_dataset_params)\n",
    "    x_ood_test, y_ood_test = get_xy_from_pu(p, u, is_markov=is_markov)\n",
    "\n",
    "    # Data loaders\n",
    "    train_loader = torch.utils.data.DataLoader(torch.utils.data.TensorDataset(x_train, y_train), \n",
    "                                            batch_size=batch_size, shuffle=True)\n",
    "    valid_loader = torch.utils.data.DataLoader(torch.utils.data.TensorDataset(x_valid, y_valid), \n",
    "                                            batch_size=batch_size, shuffle=False)\n",
    "    id_test_loader = torch.utils.data.DataLoader(torch.utils.data.TensorDataset(x_id_test, y_id_test), \n",
    "                                            batch_size=batch_size, shuffle=False)\n",
    "    ood_test_loader = torch.utils.data.DataLoader(torch.utils.data.TensorDataset(x_ood_test, y_ood_test), \n",
    "                                            batch_size=batch_size, shuffle=False)\n",
    "else:\n",
    "    # OOD test data\n",
    "    a, u, p = dataset_class.generate_dataset(n_test, grid, t, tpred, *ood_dataset_params)\n",
    "    x_ood_test, y_ood_test = get_xy_from_pu(p, u, is_markov=is_markov)\n",
    "    ood_test_loader = torch.utils.data.DataLoader(torch.utils.data.TensorDataset(x_ood_test, y_ood_test), \n",
    "                                            batch_size=batch_size, shuffle=False)\n",
    "\n",
    "print(x_train.shape, y_train.shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "b64f1eeb-0e6e-4167-a082-5479ac453cb9",
   "metadata": {},
   "outputs": [],
   "source": [
    "# tpred = torch.tensor(tpred).to(device), dataset_class = dataset_class, t=t.to(device), grid_train=grid.to(device))\n",
    "# stop = time.time()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "2bbd9b90-2b82-439d-b7e1-a642fdd2d348",
   "metadata": {},
   "outputs": [],
   "source": [
    "constraint_context = {\n",
    "    \"t\": t.to(device),\n",
    "    \"tpred\": torch.tensor(tpred).to(device),\n",
    "    \"grid_train\": grid.to(device),\n",
    "    \"dataset_class\": dataset_class\n",
    "}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "a7e5c61c-2b59-466f-9b4a-6157fa29ad4b",
   "metadata": {},
   "outputs": [],
   "source": [
    "uq = False\n",
    "model_name = args[\"--model\"]\n",
    "n_models = 1\n",
    "fno_modes2 = min(fno_modes, 12)\n",
    "if args[\"--model\"].lower() == \"FNO2d\".lower():\n",
    "    FNO2d_params = {\"modes1\": fno_modes, \"modes2\": fno_modes2, \"width\": fno_width, \"output_var\": True}\n",
    "    model = FNO2d(**FNO2d_params).to(device)\n",
    "elif args[\"--model\"].lower().startswith(\"EnsembleFNO2d\".lower()):\n",
    "    FNO2d_params = {\"modes1\": fno_modes, \"modes2\": fno_modes2, \"width\": fno_width}\n",
    "    n_models = int(args[\"--m.n_models\"])\n",
    "    utils.filter_config(args, [\"--m.n_models\"], mode=\"add\", new_config=save_args)\n",
    "    model = EnsembleNO(base_model_class=FNO2d, base_model_params=FNO2d_params, n_models=n_models)\n",
    "    uq = True\n",
    "elif args[\"--model\"].lower().startswith(\"BayesianFNO2d\".lower()):\n",
    "    FNO2d_params = {\"modes1\": fno_modes, \"modes2\": fno_modes2, \"width\": fno_width}\n",
    "    model = BayesianNO(base_model_class=FNO2d, base_model_params=FNO2d_params)\n",
    "    uq = True\n",
    "elif args[\"--model\"].lower().startswith(\"MCDropoutFNO2d\".lower()):\n",
    "    FNO2d_params = {\"modes1\": fno_modes, \"modes2\": fno_modes2, \"width\": fno_width}\n",
    "    dropout = float(args[\"--m.drop_prob\"])\n",
    "    n_dropouts = int(args[\"--m.n_models\"])\n",
    "    utils.filter_config(args, [\"--m.n_models\", \"--m.drop_prob\"], mode=\"add\", new_config=save_args)\n",
    "    model = MCDropoutNO(base_model_class=FNO2d, base_model_params=FNO2d_params, dropout=dropout, n_dropouts=n_dropouts)\n",
    "    uq = True\n",
    "elif args[\"--model\"].lower().startswith(\"OutputVarFNO2d\".lower()):\n",
    "    FNO2d_params = {\"modes1\": fno_modes, \"modes2\": fno_modes2, \"width\": fno_width}\n",
    "    model = HardE2EOutputVarNO(base_model_class=FNO2d, probconserv=False, base_model_params=FNO2d_params, constraint_context=constraint_context)\n",
    "    uq = True\n",
    "elif args[\"--model\"].lower().startswith(\"DiverseFNO2d\".lower()):\n",
    "    FNO2d_params = {\"modes1\": fno_modes, \"modes2\": fno_modes2, \"width\": fno_width}\n",
    "    lam = float(args[\"--m.reg_strength\"])\n",
    "    reg_type = args[\"--m.reg_type\"]\n",
    "    n_models = int(args[\"--m.n_models\"])\n",
    "    n_regularize = int(args[\"--m.n_regularize\"])\n",
    "    utils.filter_config(args, [\"--m.n_models\", \"--m.reg_strength\", \"--m.reg_type\", \"--m.n_regularize\"], mode=\"add\", new_config=save_args)\n",
    "    model = DiverseFNO2d(reg_loss=reg_type, n_outputs=n_models, bias_last=False, lam=lam, n_regularize=n_regularize, **FNO2d_params).to(device)\n",
    "    uq = True\n",
    "else:\n",
    "    raise NotImplementedError"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "4333d6f8-fe2c-4caa-a889-efae589c31ff",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([160, 100, 20, 1])"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "y_train.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "96aa9ee0-a30e-458b-85e2-5e7dbbd89dc4",
   "metadata": {},
   "outputs": [],
   "source": [
    "mu_true = torch.mean(y_train, dim = 0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "fc1ac32d-6fb2-4293-8ea7-f90486a32dce",
   "metadata": {},
   "outputs": [],
   "source": [
    "var_true = torch.var(y_train, dim = 0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "f07c25ce-49e4-4331-81a3-ee678808a6b4",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([100, 20, 1])"
      ]
     },
     "execution_count": 13,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "mu_true.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "0f4d18b3-1988-4417-9122-3a8709048fc2",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor([0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000,\n",
      "        0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000,\n",
      "        0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000,\n",
      "        0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000,\n",
      "        0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000,\n",
      "        0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000,\n",
      "        0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000,\n",
      "        0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.2538, 0.3701,\n",
      "        0.4510, 0.4887, 0.5009, 0.4949, 0.4698, 0.4230, 0.3245, 0.2052, 0.0000,\n",
      "        0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000,\n",
      "        0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000,\n",
      "        0.0000])\n"
     ]
    },
    {
     "data": {
      "image/png": "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tm7divbt2+OFF15A06ZNce3aNRw4cACbNm3CtWvXANyeh2TVqlV45JFH0KdPH2RkZGDBggVo2rSp2S+HqnrwwQcxceJEDB48GB07dsSRI0ewZMkSq/vz3Mnb2xuTJ0/Giy++iG7duiElJQUZGRlYtGiRTcfs378/3njjDbzxxhsICwsr919wnz59MGPGDPTs2RNPPPEELl++jLlz56J+/fo4fPhwlWIHbp+PL7/8ElqtFk2bNkVqaio2bdpktr+UNYKDgzFz5kw8//zzuOeee/DEE08gNDQUhw4dQkFBAb744guoVCp8+umn6NWrF5o1a4bBgwejZs2ayMzMxNatWxEcHIwffvjB5tcePXo0vv76a/Tq1QuvvvoqwsLC8MUXXyAjIwPffvtttSZLBG53jJ88eXK57V26dKnWf+m2nI/qXq/vv/8eeXl5eOihh8w+36FDB0RERGDJkiVISUnBv/71L+MSJq+99hoCAgKwcOFCxMfHm73vevfujaCgILzxxhvGf4TKqlevHiZPnowxY8bg7Nmz6NevH4KCgpCRkYHVq1djyJAheOONN6BSqTB//nz07dsXCQkJGDx4MGJiYnDixAkcO3YMP//8MwDLn6fmTJ48GRs3bkSnTp0wdOhQeHl54eOPP0ZRURE++OADq85fVXz00Ue4ceOG8Z/RH374AX/++SeA202dZQdV5OTk4KuvvgJwe7UBw8zSp0+fxoABAzBp0iSHxekwzhyiRtVnaR6hO5mbR0gIIZYvXy5at24tNBqNCAsLE08++aRxyL1BaWmpeOWVV0RERISQJMlk6OfOnTtFmzZthI+Pj8nQT0vD580N542PjxcDBw402bZ582bRunVr4ePjI+rVqyc+/fRT8frrrwtfX99KzojlIZ3x8fGiT58+5babiys7O1sMGzZMxMXFCW9vbxEdHS26d+8uFi5caCyj1+vFe++9J+Lj44VGoxGtW7cWa9euFQMHDhTx8fHGcobh82XnJSn72uamHyjr1q1b4vXXXxcxMTHCz89P3HvvvSI1NdXi0Oey8/iUff07h8DOmzfPOCdJ27Ztxfbt28sdszL33nuvACCef/55s89/9tlnokGDBkKj0YjGjRuLRYsW2XRvGJ4re46uX78uBg8eLMLDw0VgYKDo0aOHOHHiRLn7yNLfhqW/he+//1507NhR+Pn5ieDgYNGu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em3oNlvvaYNly5ZBkiT069fPsQF6CFvP840bNzBs2DDExMRAo9GgYcOG/Pywgq3nedasWWjUqBH8/PwQFxeHkSNH4tatW06K1j1t374dffv2RWxsLCRJwpo1ayrdZ9u2bbj77ruh0WhQv359LF682LFBCnKIZcuWCR8fH/H555+LY8eOiRdeeEGEhISI7Oxss+V/++03oVarxQcffCB+//138c477whvb29x5MgRJ0fufmw910888YSYO3euOHjwoDh+/LgYNGiQ0Gq14s8//3Ry5O7F1vNskJGRIWrWrCnuu+8+8fDDDzsnWDdm63kuKioSbdu2Fb179xY7duwQGRkZYtu2bSItLc3JkbsXW8/zkiVLhEajEUuWLBEZGRni559/FjExMWLkyJFOjty9rFu3Trz99tti1apVAoBYvXp1heXPnDkj/P39xahRo8Tvv/8u5syZI9RqtVi/fr3DYmQi5CDt2rUTw4YNM/6u0+lEbGysmDJlitny/fv3F3369DHZ1r59e/Hiiy86NE5PYOu5vlNpaakICgoSX3zxhaNC9AhVOc+lpaWiY8eO4tNPPxUDBw5kImQFW8/z/PnzRd26dUVxcbGzQvQItp7nYcOGiW7duplsGzVqlLj33nsdGqcnsSYR+te//iWaNWtmsi0lJUX06NHDYXGxacwBiouLsX//fiQnJxu3qVQqJCcnIzU11ew+qampJuUBoEePHhbL021VOdd3KigoQElJCcLCwhwVptur6nmeOHEiIiMj8dxzzzkjTLdXlfP8/fffIzExEcOGDUNUVBSaN2+O9957Dzqdzllhu52qnOeOHTti//79xuazM2fOYN26dejdu7dTYlYKV3wXctFVB7h69Sp0Oh2ioqJMtkdFReHEiRNm98nKyjJbPisry2FxeoKqnOs7/d///R9iY2PL/fHR/1TlPO/YsQOfffYZ0tLSnBChZ6jKeT5z5gy2bNmCJ598EuvWrcOpU6cwdOhQlJSUYNy4cc4I2+1U5Tw/8cQTuHr1Kjp16gQhBEpLS/HSSy/hrbfeckbIimHpuzA3NxeFhYXw8/Oz+2uyRogUberUqVi2bBlWr14NX19fV4fjMfLy8vD000/jk08+QXh4uKvD8Wh6vR6RkZFYuHAh2rRpg5SUFLz99ttYsGCBq0PzKNu2bcN7772HefPm4cCBA1i1ahV+/PFHTJo0ydWhUTWxRsgBwsPDoVarkZ2dbbI9Ozsb0dHRZveJjo62qTzdVpVzbTBt2jRMnToVmzZtQsuWLR0Zptuz9TyfPn0aZ8+eRd++fY3b9Ho9AMDLywvp6emoV6+eY4N2Q1W5n2NiYuDt7Q21Wm3c1qRJE2RlZaG4uBg+Pj4OjdkdVeU8//vf/8bTTz+N559/HgDQokUL5OfnY8iQIXj77behUrFewR4sfRcGBwc7pDYIYI2QQ/j4+KBNmzbYvHmzcZter8fmzZuRmJhodp/ExEST8gCwceNGi+XptqqcawD44IMPMGnSJKxfvx5t27Z1Rqhuzdbz3LhxYxw5cgRpaWnGx0MPPYSuXbsiLS0NcXFxzgzfbVTlfr733ntx6tQpY6IJAH/88QdiYmKYBFlQlfNcUFBQLtkxJJ+CS3bajUu+Cx3WDVvhli1bJjQajVi8eLH4/fffxZAhQ0RISIjIysoSQgjx9NNPi9GjRxvL//bbb8LLy0tMmzZNHD9+XIwbN47D561k67meOnWq8PHxEStXrhSXLl0yPvLy8lz1FtyCref5Thw1Zh1bz/P58+dFUFCQGD58uEhPTxdr164VkZGRYvLkya56C27B1vM8btw4ERQUJL7++mtx5swZsWHDBlGvXj3Rv39/V70Ft5CXlycOHjwoDh48KACIGTNmiIMHD4pz584JIYQYPXq0ePrpp43lDcPn33zzTXH8+HExd+5cDp93Z3PmzBF33XWX8PHxEe3atRO7du0yPpeUlCQGDhxoUv6bb74RDRs2FD4+PqJZs2bixx9/dHLE7suWcx0fHy8AlHuMGzfO+YG7GVvv6bKYCFnP1vO8c+dO0b59e6HRaETdunXFu+++K0pLS50ctfux5TyXlJSI8ePHi3r16glfX18RFxcnhg4dKq5fv+78wN3I1q1bzX7eGs7twIEDRVJSUrl9EhIShI+Pj6hbt65YtGiRQ2OUhGCdHhERESkT+wgRERGRYjERIiIiIsViIkRERESKxUSIiIiIFIuJEBERESkWEyEiIiJSLCZCREREpFhMhIiIiEixmAgRERGRYjERIiIiIsViIkREinLlyhVER0fjvffeM27buXMnfHx8yq16TUSej2uNEZHirFu3Dv369cPOnTvRqFEjJCQk4OGHH8aMGTNcHRoRORkTISJSpGHDhmHTpk1o27Ytjhw5gr1790Kj0bg6LCJyMiZCRKRIhYWFaN68OS5cuID9+/ejRYsWrg6JiFyAfYSISJFOnz6NixcvQq/X4+zZs64Oh4hchDVCRKQ4xcXFaNeuHRISEtCoUSPMmjULR44cQWRkpKtDIyInYyJERIrz5ptvYuXKlTh06BACAwORlJQErVaLtWvXujo0InIyNo0RkaJs27YNs2bNwpdffong4GCoVCp8+eWX+PXXXzF//nxXh0dETsYaISIiIlIs1ggRERGRYjERIiIiIsViIkRERESKxUSIiIiIFIuJEBERESkWEyEiIiJSLCZCREREpFhMhIiIiEixmAgRERGRYjERIiIiIsViIkRERESKxUSIiIiIFOv/AVV08TU7gEpUAAAAAElFTkSuQmCC",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "t_idx = 2\n",
    "with torch.no_grad():\n",
    "    plt.ylabel(\"u(x,t={t:.2f})\".format(t=t[slice(*tpred)][t_idx]))\n",
    "    plt.title(\"Plotting mean and variance for {dataset}\".format(k = np.mean(dataset_params), dataset = dataset))\n",
    "    plt.xlabel(\"x\")\n",
    "    mu =  mu_true[:,t_idx,:].squeeze(-1)\n",
    "    plt.plot(grid, mu)\n",
    "    std = torch.sqrt(var_true[:,t_idx,:]).squeeze(-1)\n",
    "    print(std)\n",
    "    plt.fill_between(grid, mu + 3*std, mu - 3*std, alpha = 0.2)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "ac0b8967",
   "metadata": {},
   "source": [
    "## Running VarianceNO out of the box"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "6fd587e6-4cf8-4cc4-92bd-ac09c7be790a",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<function datasets.LinearAdvection_1D.get_mass_rhs_func.<locals>.mass_rhs_func(inputs)>"
      ]
     },
     "execution_count": 15,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "dataset_class.get_mass_rhs_func(x=x_train)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "385681cf-a2fa-4ba9-8a9b-462fabb915ee",
   "metadata": {},
   "outputs": [],
   "source": [
    "x_train_reshaped = rearrange(x_train, \" nf nx nt 1 -> nf (nx nt)\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "b339b6e5",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "device(type='cpu')"
      ]
     },
     "execution_count": 17,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "x_train.device"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "f3073741-14be-4b35-b8cc-30ed22fbc4bc",
   "metadata": {},
   "outputs": [],
   "source": [
    "mass_rhs_func = dataset_class.get_mass_rhs_func(x=x_train)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "d71ec298-4bbc-4186-a938-8675f0526890",
   "metadata": {},
   "outputs": [],
   "source": [
    "import time"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "87e15df0-69a3-4d0e-82f0-f04540d1a1e8",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([0.0000, 0.0101, 0.0202, 0.0303, 0.0404, 0.0505, 0.0606, 0.0707, 0.0808,\n",
       "        0.0909, 0.1010, 0.1111, 0.1212, 0.1313, 0.1414, 0.1515, 0.1616, 0.1717,\n",
       "        0.1818, 0.1919, 0.2020, 0.2121, 0.2222, 0.2323, 0.2424, 0.2525, 0.2626,\n",
       "        0.2727, 0.2828, 0.2929, 0.3030, 0.3131, 0.3232, 0.3333, 0.3434, 0.3535,\n",
       "        0.3636, 0.3737, 0.3838, 0.3939, 0.4040, 0.4141, 0.4242, 0.4343, 0.4444,\n",
       "        0.4545, 0.4646, 0.4747, 0.4848, 0.4949, 0.5051, 0.5152, 0.5253, 0.5354,\n",
       "        0.5455, 0.5556, 0.5657, 0.5758, 0.5859, 0.5960, 0.6061, 0.6162, 0.6263,\n",
       "        0.6364, 0.6465, 0.6566, 0.6667, 0.6768, 0.6869, 0.6970, 0.7071, 0.7172,\n",
       "        0.7273, 0.7374, 0.7475, 0.7576, 0.7677, 0.7778, 0.7879, 0.7980, 0.8081,\n",
       "        0.8182, 0.8283, 0.8384, 0.8485, 0.8586, 0.8687, 0.8788, 0.8889, 0.8990,\n",
       "        0.9091, 0.9192, 0.9293, 0.9394, 0.9495, 0.9596, 0.9697, 0.9798, 0.9899,\n",
       "        1.0000])"
      ]
     },
     "execution_count": 20,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "grid"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "id": "2165975d",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[0, -1, 5]"
      ]
     },
     "execution_count": 21,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "tpred"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "id": "1ad2c532",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([ 0, -1,  5])"
      ]
     },
     "execution_count": 22,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "torch.tensor(tpred)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "id": "18a1c208-a4df-4f27-8dad-f2466164e855",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([160, 100, 20, 1])"
      ]
     },
     "execution_count": 23,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "x_train.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "id": "6ff40f4d-d8b7-45f7-8e3a-8115564540df",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Projecting Now\n",
      "Epoch 0: Train loss=0.205926, Validation loss=0.196379 (saved)\n",
      "Epoch 1: Train loss=0.181162, Validation loss=0.151701 (saved)\n",
      "Epoch 2: Train loss=0.113000, Validation loss=0.058098 (saved)\n",
      "Epoch 3: Train loss=0.032997, Validation loss=0.019765 (saved)\n",
      "Epoch 4: Train loss=0.019038, Validation loss=0.016134 (saved)\n",
      "Epoch 5: Train loss=0.015568, Validation loss=0.013226 (saved)\n",
      "Epoch 6: Train loss=0.012892, Validation loss=0.011708 (saved)\n",
      "Epoch 7: Train loss=0.011199, Validation loss=0.010900 (saved)\n",
      "Epoch 8: Train loss=0.010553, Validation loss=0.010149 (saved)\n",
      "Epoch 9: Train loss=0.010011, Validation loss=0.009597 (saved)\n",
      "Epoch 10: Train loss=0.009724, Validation loss=0.009472 (saved)\n",
      "Epoch 11: Train loss=0.009550, Validation loss=0.009439 (saved)\n",
      "Epoch 12: Train loss=0.009280, Validation loss=0.008815 (saved)\n",
      "Epoch 13: Train loss=0.008889, Validation loss=0.008688 (saved)\n",
      "Epoch 14: Train loss=0.008631, Validation loss=0.008038 (saved)\n",
      "Epoch 15: Train loss=0.008152, Validation loss=0.008003 (saved)\n",
      "Epoch 16: Train loss=0.008114, Validation loss=0.007652 (saved)\n",
      "Epoch 17: Train loss=0.007795, Validation loss=0.007967 \n",
      "Epoch 18: Train loss=0.007402, Validation loss=0.007183 (saved)\n",
      "Epoch 19: Train loss=0.007086, Validation loss=0.006636 (saved)\n",
      "Epoch 20: Train loss=0.007098, Validation loss=0.006859 \n",
      "Epoch 21: Train loss=0.006769, Validation loss=0.006251 (saved)\n",
      "Epoch 22: Train loss=0.006611, Validation loss=0.006480 \n",
      "Epoch 23: Train loss=0.006664, Validation loss=0.006887 \n",
      "Epoch 24: Train loss=0.006484, Validation loss=0.006656 \n",
      "Epoch 25: Train loss=0.006147, Validation loss=0.005772 (saved)\n",
      "Epoch 26: Train loss=0.005724, Validation loss=0.005803 \n",
      "Epoch 27: Train loss=0.005795, Validation loss=0.005479 (saved)\n",
      "Epoch 28: Train loss=0.005649, Validation loss=0.006158 \n",
      "Epoch 29: Train loss=0.005489, Validation loss=0.004980 (saved)\n",
      "Epoch 30: Train loss=0.005220, Validation loss=0.004929 (saved)\n",
      "Epoch 31: Train loss=0.005232, Validation loss=0.005505 \n",
      "Epoch 32: Train loss=0.005236, Validation loss=0.004564 (saved)\n",
      "Epoch 33: Train loss=0.004900, Validation loss=0.004700 \n",
      "Epoch 34: Train loss=0.004919, Validation loss=0.004595 \n",
      "Epoch 35: Train loss=0.004918, Validation loss=0.004575 \n",
      "Epoch 36: Train loss=0.004786, Validation loss=0.004814 \n",
      "Epoch 37: Train loss=0.004669, Validation loss=0.004451 (saved)\n",
      "Epoch 38: Train loss=0.004504, Validation loss=0.005642 \n",
      "Epoch 39: Train loss=0.005205, Validation loss=0.004741 \n",
      "Epoch 40: Train loss=0.004974, Validation loss=0.005463 \n",
      "Epoch 41: Train loss=0.005156, Validation loss=0.004486 \n",
      "Epoch 42: Train loss=0.005101, Validation loss=0.005015 \n",
      "Epoch 43: Train loss=0.005196, Validation loss=0.005173 \n",
      "Epoch 44: Train loss=0.004789, Validation loss=0.004319 (saved)\n",
      "Epoch 45: Train loss=0.004634, Validation loss=0.005229 \n",
      "Epoch 46: Train loss=0.004858, Validation loss=0.004727 \n",
      "Epoch 47: Train loss=0.004650, Validation loss=0.004530 \n",
      "Epoch 48: Train loss=0.004592, Validation loss=0.004609 \n",
      "Epoch 49: Train loss=0.004474, Validation loss=0.004553 \n",
      "Epoch 50: Train loss=0.004341, Validation loss=0.004130 (saved)\n",
      "Epoch 51: Train loss=0.004100, Validation loss=0.003899 (saved)\n",
      "Epoch 52: Train loss=0.003906, Validation loss=0.003792 (saved)\n",
      "Epoch 53: Train loss=0.003794, Validation loss=0.003715 (saved)\n",
      "Epoch 54: Train loss=0.003716, Validation loss=0.003645 (saved)\n",
      "Epoch 55: Train loss=0.003706, Validation loss=0.003641 (saved)\n",
      "Epoch 56: Train loss=0.003687, Validation loss=0.003729 \n",
      "Epoch 57: Train loss=0.003746, Validation loss=0.003611 (saved)\n",
      "Epoch 58: Train loss=0.003733, Validation loss=0.003774 \n",
      "Epoch 59: Train loss=0.003774, Validation loss=0.003712 \n",
      "Epoch 60: Train loss=0.003752, Validation loss=0.003811 \n",
      "Epoch 61: Train loss=0.003771, Validation loss=0.003689 \n",
      "Epoch 62: Train loss=0.003896, Validation loss=0.003887 \n",
      "Epoch 63: Train loss=0.003819, Validation loss=0.003758 \n",
      "Epoch 64: Train loss=0.003715, Validation loss=0.003577 (saved)\n",
      "Epoch 65: Train loss=0.003651, Validation loss=0.003670 \n",
      "Epoch 66: Train loss=0.003645, Validation loss=0.003614 \n",
      "Epoch 67: Train loss=0.003654, Validation loss=0.003629 \n",
      "Epoch 68: Train loss=0.003783, Validation loss=0.003895 \n",
      "Epoch 69: Train loss=0.003749, Validation loss=0.003689 \n",
      "Epoch 70: Train loss=0.003673, Validation loss=0.003546 (saved)\n",
      "Epoch 71: Train loss=0.003621, Validation loss=0.003478 (saved)\n",
      "Epoch 72: Train loss=0.003599, Validation loss=0.003650 \n",
      "Epoch 73: Train loss=0.003658, Validation loss=0.003566 \n",
      "Epoch 74: Train loss=0.003737, Validation loss=0.004187 \n",
      "Epoch 75: Train loss=0.004099, Validation loss=0.004384 \n",
      "Epoch 76: Train loss=0.004103, Validation loss=0.003689 \n",
      "Epoch 77: Train loss=0.003803, Validation loss=0.003757 \n",
      "Epoch 78: Train loss=0.003935, Validation loss=0.003516 \n",
      "Epoch 79: Train loss=0.003821, Validation loss=0.004131 \n",
      "Epoch 80: Train loss=0.003823, Validation loss=0.003698 \n",
      "Epoch 81: Train loss=0.003760, Validation loss=0.003608 \n",
      "Epoch 82: Train loss=0.003528, Validation loss=0.003411 (saved)\n",
      "Epoch 83: Train loss=0.003472, Validation loss=0.003451 \n",
      "Epoch 84: Train loss=0.003572, Validation loss=0.003564 \n",
      "Epoch 85: Train loss=0.003610, Validation loss=0.003720 \n",
      "Epoch 86: Train loss=0.003992, Validation loss=0.004503 \n",
      "Epoch 87: Train loss=0.004440, Validation loss=0.003959 \n",
      "Epoch 88: Train loss=0.004121, Validation loss=0.003573 \n",
      "Epoch 89: Train loss=0.003831, Validation loss=0.003827 \n",
      "Epoch 90: Train loss=0.003736, Validation loss=0.003663 \n",
      "Epoch 91: Train loss=0.003627, Validation loss=0.003593 \n",
      "Epoch 92: Train loss=0.003747, Validation loss=0.003492 \n",
      "Epoch 93: Train loss=0.003734, Validation loss=0.003776 \n",
      "Epoch 94: Train loss=0.003663, Validation loss=0.003478 \n",
      "Epoch 95: Train loss=0.003566, Validation loss=0.003366 (saved)\n",
      "Epoch 96: Train loss=0.003503, Validation loss=0.003415 \n",
      "Epoch 97: Train loss=0.003534, Validation loss=0.003819 \n",
      "Epoch 98: Train loss=0.003623, Validation loss=0.003610 \n",
      "Epoch 99: Train loss=0.003732, Validation loss=0.003515 \n",
      "Epoch 100: Train loss=0.003438, Validation loss=0.003298 (saved)\n",
      "Epoch 101: Train loss=0.003335, Validation loss=0.003271 (saved)\n",
      "Epoch 102: Train loss=0.003285, Validation loss=0.003251 (saved)\n",
      "Epoch 103: Train loss=0.003283, Validation loss=0.003249 (saved)\n",
      "Epoch 104: Train loss=0.003247, Validation loss=0.003184 (saved)\n",
      "Epoch 105: Train loss=0.003214, Validation loss=0.003166 (saved)\n",
      "Epoch 106: Train loss=0.003244, Validation loss=0.003250 \n",
      "Epoch 107: Train loss=0.003247, Validation loss=0.003214 \n",
      "Epoch 108: Train loss=0.003222, Validation loss=0.003203 \n",
      "Epoch 109: Train loss=0.003223, Validation loss=0.003149 (saved)\n",
      "Epoch 110: Train loss=0.003243, Validation loss=0.003262 \n",
      "Epoch 111: Train loss=0.003243, Validation loss=0.003187 \n",
      "Epoch 112: Train loss=0.003204, Validation loss=0.003139 (saved)\n",
      "Epoch 113: Train loss=0.003172, Validation loss=0.003131 (saved)\n",
      "Epoch 114: Train loss=0.003176, Validation loss=0.003150 \n",
      "Epoch 115: Train loss=0.003174, Validation loss=0.003129 (saved)\n",
      "Epoch 116: Train loss=0.003206, Validation loss=0.003201 \n",
      "Epoch 117: Train loss=0.003225, Validation loss=0.003188 \n",
      "Epoch 118: Train loss=0.003252, Validation loss=0.003118 (saved)\n",
      "Epoch 119: Train loss=0.003189, Validation loss=0.003127 \n",
      "Epoch 120: Train loss=0.003199, Validation loss=0.003189 \n",
      "Epoch 121: Train loss=0.003196, Validation loss=0.003134 \n",
      "Epoch 122: Train loss=0.003184, Validation loss=0.003238 \n",
      "Epoch 123: Train loss=0.003204, Validation loss=0.003138 \n",
      "Epoch 124: Train loss=0.003162, Validation loss=0.003130 \n",
      "Epoch 125: Train loss=0.003191, Validation loss=0.003185 \n",
      "Epoch 126: Train loss=0.003255, Validation loss=0.003212 \n",
      "Epoch 127: Train loss=0.003182, Validation loss=0.003117 (saved)\n",
      "Epoch 128: Train loss=0.003123, Validation loss=0.003070 (saved)\n",
      "Epoch 129: Train loss=0.003115, Validation loss=0.003083 \n",
      "Epoch 130: Train loss=0.003172, Validation loss=0.003175 \n",
      "Epoch 131: Train loss=0.003159, Validation loss=0.003054 (saved)\n",
      "Epoch 132: Train loss=0.003118, Validation loss=0.003066 \n",
      "Epoch 133: Train loss=0.003137, Validation loss=0.003146 \n",
      "Epoch 134: Train loss=0.003158, Validation loss=0.003090 \n",
      "Epoch 135: Train loss=0.003177, Validation loss=0.003142 \n",
      "Epoch 136: Train loss=0.003169, Validation loss=0.003054 (saved)\n",
      "Epoch 137: Train loss=0.003204, Validation loss=0.003138 \n",
      "Epoch 138: Train loss=0.003144, Validation loss=0.003192 \n",
      "Epoch 139: Train loss=0.003154, Validation loss=0.003200 \n",
      "Epoch 140: Train loss=0.003148, Validation loss=0.003029 (saved)\n",
      "Epoch 141: Train loss=0.003149, Validation loss=0.003112 \n",
      "Epoch 142: Train loss=0.003116, Validation loss=0.003024 (saved)\n",
      "Epoch 143: Train loss=0.003105, Validation loss=0.003044 \n",
      "Epoch 144: Train loss=0.003137, Validation loss=0.003076 \n",
      "Epoch 145: Train loss=0.003123, Validation loss=0.003051 \n",
      "Epoch 146: Train loss=0.003090, Validation loss=0.003073 \n",
      "Epoch 147: Train loss=0.003130, Validation loss=0.003185 \n",
      "Epoch 148: Train loss=0.003105, Validation loss=0.003173 \n",
      "Epoch 149: Train loss=0.003100, Validation loss=0.003044 \n",
      "Epoch 150: Train loss=0.003016, Validation loss=0.002951 (saved)\n",
      "Epoch 151: Train loss=0.002968, Validation loss=0.002915 (saved)\n",
      "Epoch 152: Train loss=0.002951, Validation loss=0.002910 (saved)\n",
      "Epoch 153: Train loss=0.002948, Validation loss=0.002931 \n",
      "Epoch 154: Train loss=0.002970, Validation loss=0.002944 \n",
      "Epoch 155: Train loss=0.002971, Validation loss=0.002989 \n",
      "Epoch 156: Train loss=0.002973, Validation loss=0.002911 \n",
      "Epoch 157: Train loss=0.002964, Validation loss=0.002907 (saved)\n",
      "Epoch 158: Train loss=0.002938, Validation loss=0.002895 (saved)\n",
      "Epoch 159: Train loss=0.002921, Validation loss=0.002897 \n",
      "Epoch 160: Train loss=0.002940, Validation loss=0.002937 \n",
      "Epoch 161: Train loss=0.002952, Validation loss=0.002923 \n",
      "Epoch 162: Train loss=0.002953, Validation loss=0.002916 \n",
      "Epoch 163: Train loss=0.002939, Validation loss=0.002908 \n",
      "Epoch 164: Train loss=0.002931, Validation loss=0.002880 (saved)\n",
      "Epoch 165: Train loss=0.002919, Validation loss=0.002871 (saved)\n",
      "Epoch 166: Train loss=0.002932, Validation loss=0.002868 (saved)\n",
      "Epoch 167: Train loss=0.002933, Validation loss=0.002868 (saved)\n",
      "Epoch 168: Train loss=0.002936, Validation loss=0.002892 \n",
      "Epoch 169: Train loss=0.002923, Validation loss=0.002879 \n",
      "Epoch 170: Train loss=0.002915, Validation loss=0.002879 \n",
      "Epoch 171: Train loss=0.002915, Validation loss=0.002865 (saved)\n",
      "Epoch 172: Train loss=0.002910, Validation loss=0.002869 \n",
      "Epoch 173: Train loss=0.002904, Validation loss=0.002891 \n",
      "Epoch 174: Train loss=0.002917, Validation loss=0.002866 \n",
      "Epoch 175: Train loss=0.002920, Validation loss=0.002919 \n",
      "Epoch 176: Train loss=0.002924, Validation loss=0.002861 (saved)\n",
      "Epoch 177: Train loss=0.002909, Validation loss=0.002903 \n",
      "Epoch 178: Train loss=0.002948, Validation loss=0.002901 \n",
      "Epoch 179: Train loss=0.002939, Validation loss=0.002859 (saved)\n",
      "Epoch 180: Train loss=0.002942, Validation loss=0.002877 \n",
      "Epoch 181: Train loss=0.002930, Validation loss=0.002882 \n",
      "Epoch 182: Train loss=0.002909, Validation loss=0.002889 \n",
      "Epoch 183: Train loss=0.002902, Validation loss=0.002843 (saved)\n",
      "Epoch 184: Train loss=0.002890, Validation loss=0.002878 \n",
      "Epoch 185: Train loss=0.002892, Validation loss=0.002845 \n",
      "Epoch 186: Train loss=0.002885, Validation loss=0.002843 \n",
      "Epoch 187: Train loss=0.002880, Validation loss=0.002821 (saved)\n",
      "Epoch 188: Train loss=0.002889, Validation loss=0.002861 \n",
      "Epoch 189: Train loss=0.002899, Validation loss=0.002836 \n",
      "Epoch 190: Train loss=0.002898, Validation loss=0.002834 \n",
      "Epoch 191: Train loss=0.002852, Validation loss=0.002810 (saved)\n",
      "Epoch 192: Train loss=0.002840, Validation loss=0.002812 \n",
      "Epoch 193: Train loss=0.002860, Validation loss=0.002856 \n",
      "Epoch 194: Train loss=0.002852, Validation loss=0.002811 \n",
      "Epoch 195: Train loss=0.002843, Validation loss=0.002783 (saved)\n",
      "Epoch 196: Train loss=0.002837, Validation loss=0.002787 \n",
      "Epoch 197: Train loss=0.002827, Validation loss=0.002785 \n",
      "Epoch 198: Train loss=0.002821, Validation loss=0.002780 (saved)\n",
      "Epoch 199: Train loss=0.002832, Validation loss=0.002790 \n",
      "Finished training with best train loss: 0.002810 and validation loss: 0.002780\n"
     ]
    }
   ],
   "source": [
    "x_ood_test = x_ood_test.to(device)\n",
    "start = time.time()\n",
    "model.fit(train_loader, valid_loader, x_test=x_ood_test, epochs=epochs, lr=lr, step_size=step_size, gamma=gamma, tpred = torch.tensor(tpred).to(device), dataset_class = dataset_class, t=t.to(device), grid_train=grid.to(device))\n",
    "stop = time.time()\n",
    "# print(stop-start)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "id": "fb522e5a",
   "metadata": {},
   "outputs": [],
   "source": [
    "# %timeit model.fit(train_loader, valid_loader, x_test=x_ood_test, epochs=1, lr=lr, step_size=step_size, gamma=gamma, tpred = torch.tensor(tpred).to(device), dataset_class = dataset_class, t=t.to(device), grid_train=grid.to(device))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "id": "41b11e17-4840-42fe-9a5b-34d65f0a77d7",
   "metadata": {},
   "outputs": [],
   "source": [
    "def test(model, test_loader, **test_params):\n",
    "    test_type = test_params.get(\"test_type\", \"id\")\n",
    "    mu = []\n",
    "    var = []\n",
    "    results = {}\n",
    "    results[\"loss\"] = 0.0\n",
    "\n",
    "    model = model.to(device)\n",
    "\n",
    "    with torch.no_grad():\n",
    "        for batch_idx, batch in enumerate(test_loader):\n",
    "            x, y = batch\n",
    "            x, y = x.to(device), y.to(device)\n",
    "\n",
    "            out = model(x)\n",
    "\n",
    "            _mu, _var = out\n",
    "            _std = torch.sqrt(_var)\n",
    "            mass_rhs_func = dataset_class.get_mass_rhs_func(x=x)\n",
    "            \n",
    "            new_mu, new_std, _, _ = probconserv.apply_ortho_constraint(\n",
    "                                                mu=_mu[:, :, :, 0], \n",
    "                                                std=_std[:, :, :, 0], \n",
    "                                                mass_rhs_func=mass_rhs_func, \n",
    "                                                t=t, \n",
    "                                                tpred=tpred, \n",
    "                                                grid_train=grid, \n",
    "                                                precis_g=np.inf,\n",
    "                                                second_deriv_alpha=None,\n",
    "                                                )\n",
    "            \n",
    "            out = (new_mu.unsqueeze(-1), torch.square(new_std).unsqueeze(-1))\n",
    "                \n",
    "\n",
    "            results[\"loss\"] += model.loss_func(out, y).item()\n",
    "            utils.compute_all_metrics(out, y, results)\n",
    "\n",
    "            if uq:\n",
    "                mu.append(out[0].detach().cpu())\n",
    "                var.append(out[1].detach().cpu())\n",
    "            else:\n",
    "                mu.append(out.detach().cpu())\n",
    "\n",
    "    # print(results['mse'])\n",
    "    # print(len(test_loader.dataset))\n",
    "\n",
    "    for key in results.keys():\n",
    "        if not key.endswith(\"by_example\"):\n",
    "            results[key] /= len(test_loader.dataset)\n",
    "        if type(results[key]) == torch.Tensor:\n",
    "            results[key] = results[key].tolist()\n",
    "\n",
    "    # Plot\n",
    "    mu = torch.cat(mu, dim=0)\n",
    "    if uq:\n",
    "        var = torch.cat(var, dim=0)\n",
    "        std = torch.sqrt(var)\n",
    "    else:\n",
    "        var = None\n",
    "        std = None\n",
    "    x = test_loader.dataset.tensors[0]\n",
    "    y = test_loader.dataset.tensors[1]\n",
    "\n",
    "    if uq:\n",
    "        results[\"nMeRCI_all\"] = utils.compute_nMeRCI(mu, var, y).item()\n",
    "        results[\"rmsce_all\"] = utils.compute_rmsce(mu, var, y).item()\n",
    "\n",
    "        if is_probconserv:\n",
    "            print(\"Here\")\n",
    "            mass_rhs_func = dataset_class.get_mass_rhs_func(x=x)\n",
    "            new_mu, new_std, _, mass_rhs = probconserv.apply_constraint(\n",
    "                mu=mu[:, :, :, 0], \n",
    "                std=std[:, :, :, 0], \n",
    "                mass_rhs_func=mass_rhs_func, \n",
    "                t=t, \n",
    "                tpred=tpred, \n",
    "                grid_train=grid, \n",
    "                precis_g=np.inf,\n",
    "                second_deriv_alpha=None,\n",
    "            )\n",
    "            new_mu = new_mu[:, :, :, None]\n",
    "            new_std = new_std[:, :, :, None]\n",
    "            new_var = new_std**2\n",
    "\n",
    "            probconserv_results = utils.compute_all_metrics((new_mu, new_var), y, {})\n",
    "            for key in probconserv_results.keys():\n",
    "                if not key.endswith(\"by_example\"):\n",
    "                    probconserv_results[key] /= len(test_loader.dataset)\n",
    "                if type(probconserv_results[key]) == torch.Tensor:\n",
    "                    probconserv_results[key] = probconserv_results[key].tolist()\n",
    "\n",
    "            probconserv_results[\"nMeRCI_all\"] = utils.compute_nMeRCI(new_mu, new_var, y).item()\n",
    "            probconserv_results[\"rmsce_all\"] = utils.compute_rmsce(new_mu, new_var, y).item()\n",
    "\n",
    "            cerr = (probconserv.get_empirical_mass_rhs(mu[:, :,  :, 0]) - mass_rhs).abs().sum(dim=-1)\n",
    "            new_cerr = (probconserv.get_empirical_mass_rhs(new_mu[:, :, :, 0]) - mass_rhs).abs().sum(dim=-1)\n",
    "\n",
    "            results[\"cerr_by_example\"] = cerr.tolist()\n",
    "            results[\"mcerr\"] = cerr.mean().item()\n",
    "            probconserv_results[\"cerr_by_example\"] = new_cerr.tolist()\n",
    "            probconserv_results[\"mcerr\"] = new_cerr.mean().item()\n",
    "\n",
    "            for key in probconserv_results.keys():\n",
    "                results[f\"pc.{key}\"] = probconserv_results[key]\n",
    "    \n",
    "    # results[\"time\"] = utils.compute_forward_time(model, x[:batch_size].to(device), repetitions=10)\n",
    "    results[\"n_params\"] = utils.compute_n_params(model)\n",
    "    results[\"n_flops\"] = utils.compute_n_flops(model_name, Np=n_x*n_t, fno_modes=fno_modes, fno_width=fno_width, n_layers=4, n_models=n_models)\n",
    "\n",
    "    dataset_params_correct_type = dataset_params if test_type == \"id\" or test_type == \"train\" else ood_dataset_params\n",
    "\n",
    "    mse_by_example = torch.tensor(results[\"mse_by_example\"])\n",
    "    random_idx = np.random.choice(mse_by_example.shape[0])\n",
    "    _, worst_idx = mse_by_example.max(dim=0)\n",
    "    _, best_idx = mse_by_example.min(dim=0)\n",
    "    _, median_idx = mse_by_example.median(dim=0)\n",
    "\n",
    "    for example_name, example_idx in zip([\"random\", \"worst\", \"best\", \"median\"], [random_idx, worst_idx, best_idx, median_idx]):\n",
    "        if uq:\n",
    "            results[f\"examples.{example_name}\"] = (mu[example_idx].tolist(), var[example_idx].tolist(), y[example_idx].tolist(), x[example_idx].tolist())\n",
    "            if is_probconserv:\n",
    "                results[f\"pc.examples.{example_name}\"] = (new_mu[example_idx].tolist(), new_var[example_idx].tolist(), y[example_idx].tolist(), x[example_idx].tolist())\n",
    "        else:\n",
    "            results[f\"examples.{example_name}\"] = (mu[example_idx].tolist(), None, y[example_idx].tolist(), x[example_idx].tolist())\n",
    "\n",
    "        # prefix = f\"{test_type}_{example_name}_params={dataset_params_correct_type}\"\n",
    "        # plot_and_save(prefix, example_idx, x.squeeze(-1), y.squeeze(-1), mu.squeeze(-1), std.squeeze(-1) if std is not None else None)\n",
    "\n",
    "    # utils.dict_to_file({\"test_type\": test_type, \"params\": dataset_params_correct_type, \"results\": results}, \n",
    "    #                    f\"{run_folder}/results_{test_type}_params={dataset_params_correct_type}.json\")\n",
    "\n",
    "    return results\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "id": "58fe7954-8db9-47db-ba7c-39fde51a4146",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Here\n",
      "Here\n",
      "Here\n",
      "Here\n",
      "Train results\n",
      "MSE: 0.0010409918730147183\n",
      "n-MeRCI: 1.1782644987106323\n",
      "RMSCE: 0.21788200736045837\n",
      "Cerr: 3.503635639390268e-07\n",
      "In-domain results\n",
      "MSE: 0.000979968961328268\n",
      "n-MeRCI: 0.979526698589325\n",
      "RMSCE: 0.22339452803134918\n",
      "ProbConserv Results\n",
      "MSE: 0.0009799689799547195\n",
      "n-MeRCI: 1.0282279253005981\n",
      "RMSCE: 0.22366289794445038\n",
      "Cerr: 3.7372112160483084e-07\n",
      "Prob_Cerr: 3.7312508993636584e-07\n",
      "Here\n",
      "\n",
      "\n",
      "Out-of-domain results\n",
      "MSE: 0.0009631957858800888\n",
      "n-MeRCI: 1.0520555973052979\n",
      "RMSCE: 0.22707891464233398\n",
      "ProbConserv Results\n",
      "MSE: 0.0009631958603858948\n",
      "n-MeRCI: 1.1003276109695435\n",
      "RMSCE: 0.22731609642505646\n",
      "Cerr: 3.5405159337642544e-07\n",
      "Prob_Cerr: 3.728270598912786e-07\n"
     ]
    }
   ],
   "source": [
    "is_probconserv = True\n",
    "\n",
    "train_loader_no_shuffle = torch.utils.data.DataLoader(train_loader.dataset, batch_size=batch_size, shuffle=False)\n",
    "train_results = test(model, train_loader_no_shuffle, test_type=\"train\")\n",
    "id_results = test(model, id_test_loader, test_type=\"id\")\n",
    "\n",
    "if is_train:\n",
    "    train_loader_no_shuffle = torch.utils.data.DataLoader(train_loader.dataset, batch_size=batch_size, shuffle=False)\n",
    "    train_results = test(model, train_loader_no_shuffle, test_type=\"train\")\n",
    "    id_results = test(model, id_test_loader, test_type=\"id\")\n",
    "\n",
    "    print(\"Train results\")\n",
    "    print(f\"MSE: {train_results['mse']}\")\n",
    "    print(f\"n-MeRCI: {train_results['nMeRCI_all']}\")\n",
    "    print(f\"RMSCE: {train_results['rmsce_all']}\")\n",
    "    print(f\"Cerr: {train_results['mcerr']}\")\n",
    "\n",
    "    \n",
    "\n",
    "    print(\"In-domain results\")\n",
    "    print(f\"MSE: {id_results['mse']}\")\n",
    "    print(f\"n-MeRCI: {id_results['nMeRCI_all']}\")\n",
    "    print(f\"RMSCE: {id_results['rmsce_all']}\")\n",
    "\n",
    "    if is_probconserv:\n",
    "        print(\"ProbConserv Results\")\n",
    "        print(f\"MSE: {id_results['pc.mse']}\")\n",
    "        print(f\"n-MeRCI: {id_results['pc.nMeRCI_all']}\")\n",
    "        print(f\"RMSCE: {id_results['pc.rmsce_all']}\")\n",
    "        print(f\"Cerr: {id_results['mcerr']}\")\n",
    "        print(f\"Prob_Cerr: {id_results['pc.mcerr']}\")\n",
    "        \n",
    "\n",
    "ood_results = test(model, ood_test_loader, test_type=\"ood\")\n",
    "\n",
    "print(\"\\n\")\n",
    "print(\"Out-of-domain results\")\n",
    "print(f\"MSE: {ood_results['mse']}\")\n",
    "print(f\"n-MeRCI: {ood_results['nMeRCI_all']}\")\n",
    "print(f\"RMSCE: {ood_results['rmsce_all']}\")\n",
    "\n",
    "if is_probconserv:\n",
    "    print(\"ProbConserv Results\")\n",
    "    print(f\"MSE: {ood_results['pc.mse']}\")\n",
    "    print(f\"n-MeRCI: {ood_results['pc.nMeRCI_all']}\")\n",
    "    print(f\"RMSCE: {ood_results['pc.rmsce_all']}\")\n",
    "    print(f\"Cerr: {ood_results['mcerr']}\")\n",
    "    print(f\"Prob_Cerr: {ood_results['pc.mcerr']}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "id": "98acb797",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.0009799689799547195"
      ]
     },
     "execution_count": 28,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "id_results['pc.mse']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "id": "062f8da4-99eb-4f91-97c7-8e711746a05d",
   "metadata": {},
   "outputs": [],
   "source": [
    "def compute_statistics(\n",
    "    model, \n",
    "    x_data, \n",
    "    y_data, \n",
    "    t, \n",
    "    tpred, \n",
    "    grid, \n",
    "    dataset_class, \n",
    "    apply_probconserv=False, \n",
    "    plot=False,\n",
    "    x_data_test=None, \n",
    "    y_data_test=None,\n",
    "    return_latex=False,\n",
    "    name=\"Model\"\n",
    "):\n",
    "    import torch\n",
    "    import utils\n",
    "    import probconserv\n",
    "    import matplotlib.pyplot as plt\n",
    "\n",
    "    device = next(model.parameters()).device\n",
    "    x_data = x_data.to(device)\n",
    "\n",
    "    with torch.no_grad():\n",
    "        out = model(x_data)\n",
    "\n",
    "    if isinstance(out, tuple):\n",
    "        mu, var = out[0].cpu(), out[1].cpu()\n",
    "        std = torch.sqrt(var)\n",
    "    else:\n",
    "        mu = out.cpu()\n",
    "        std = torch.zeros_like(mu)\n",
    "        var = torch.square(std)\n",
    "\n",
    "    x_cpu = x_data.cpu()\n",
    "    mass_rhs_func = dataset_class.get_mass_rhs_func(x=x_cpu)\n",
    "\n",
    "    if apply_probconserv:\n",
    "        new_mu, new_std, _, mass_rhs = probconserv.apply_ortho_constraint(\n",
    "            mu=mu[:, :, :, 0],\n",
    "            std=std[:, :, :, 0],\n",
    "            mass_rhs_func=mass_rhs_func,\n",
    "            t=t,\n",
    "            tpred=tpred,\n",
    "            grid_train=grid,\n",
    "            precis_g=float('inf'),\n",
    "            second_deriv_alpha=None,\n",
    "        )\n",
    "        mu = new_mu.unsqueeze(-1)\n",
    "        std = new_std.unsqueeze(-1)\n",
    "        var = torch.square(std)\n",
    "        cerr = (probconserv.get_empirical_mass_rhs(mu[:, :, :, 0]) - mass_rhs).abs().sum(dim=-1)\n",
    "    else:\n",
    "        t_sliced = t[slice(*tpred)]\n",
    "        ts = repeat(t_sliced, \"nt -> nf nt\", nf=mu.shape[0])\n",
    "        xs = repeat(grid, \"nx -> nf nx\", nf=mu.shape[0])\n",
    "        inputs = meshgrid(ts, xs)\n",
    "        cerr = (probconserv.get_empirical_mass_rhs(mu[:, :, :, 0]) - mass_rhs_func(inputs)).abs().sum(dim=-1)\n",
    "\n",
    "    stats = utils.compute_all_metrics_avg((mu, var), y_data, {})\n",
    "    stats[\"nMeRCI_all\"] = utils.compute_nMeRCI(mu, var, y_data).item()\n",
    "    stats[\"rmsce_all\"] = utils.compute_rmsce(mu, var, y_data).item()\n",
    "    stats[\"cerr_by_example\"] = cerr.tolist()\n",
    "    stats[\"mcerr\"] = cerr.mean().item()\n",
    "\n",
    "    # --- Test dataset ---\n",
    "    test_stats = None\n",
    "    if x_data_test is not None and y_data_test is not None:\n",
    "        x_data_test = x_data_test.to(device)\n",
    "        with torch.no_grad():\n",
    "            test_out = model(x_data_test)\n",
    "\n",
    "        if isinstance(test_out, tuple):\n",
    "            mu_test, var_test = test_out[0].cpu(), test_out[1].cpu()\n",
    "            std_test = torch.sqrt(var_test)\n",
    "        else:\n",
    "            mu_test = test_out.cpu()\n",
    "            std_test = torch.zeros_like(mu_test)\n",
    "            var_test = torch.square(std_test)\n",
    "\n",
    "        x_test_cpu = x_data_test.cpu()\n",
    "        test_mass_rhs_func = dataset_class.get_mass_rhs_func(x=x_test_cpu)\n",
    "\n",
    "        if apply_probconserv:\n",
    "            new_mu_test, new_std_test, _, test_mass_rhs = probconserv.apply_ortho_constraint(\n",
    "                mu=mu_test[:, :, :, 0],\n",
    "                std=std_test[:, :, :, 0],\n",
    "                mass_rhs_func=test_mass_rhs_func,\n",
    "                t=t,\n",
    "                tpred=tpred,\n",
    "                grid_train=grid,\n",
    "                precis_g=float('inf'),\n",
    "                second_deriv_alpha=None,\n",
    "            )\n",
    "            mu_test = new_mu_test.unsqueeze(-1)\n",
    "            std_test = new_std_test.unsqueeze(-1)\n",
    "            var_test = torch.square(std_test)\n",
    "            cerr_test = (probconserv.get_empirical_mass_rhs(mu_test[:, :, :, 0]) - test_mass_rhs).abs().sum(dim=-1)\n",
    "        else:\n",
    "            t_sliced = t[slice(*tpred)]\n",
    "            ts = repeat(t_sliced, \"nt -> nf nt\", nf=mu_test.shape[0])\n",
    "            xs = repeat(grid, \"nx -> nf nx\", nf=mu_test.shape[0])\n",
    "            inputs = meshgrid(ts, xs)\n",
    "            cerr_test = (probconserv.get_empirical_mass_rhs(mu_test[:, :, :, 0]) - test_mass_rhs_func(inputs)).abs().sum(dim=-1)\n",
    "\n",
    "        test_stats = utils.compute_all_metrics_avg((mu_test, var_test), y_data_test, {})\n",
    "        test_stats[\"nMeRCI_all\"] = utils.compute_nMeRCI(mu_test, var_test, y_data_test).item()\n",
    "        test_stats[\"rmsce_all\"] = utils.compute_rmsce(mu_test, var_test, y_data_test).item()\n",
    "        test_stats[\"cerr_by_example\"] = cerr_test.tolist()\n",
    "        test_stats[\"mcerr\"] = cerr_test.mean().item()\n",
    "\n",
    "    # --- Optional plot ---\n",
    "    if plot:\n",
    "        t_idx = 1\n",
    "        param_idx = 0\n",
    "        with torch.no_grad():\n",
    "            plt.ylabel(f\"u(x, t={t[slice(*tpred)][t_idx]:.2f})\")\n",
    "            plt.xlabel(\"x\")\n",
    "            plt.title(f\"Predicted vs True (param = {x_data[param_idx,0,0,0].item():.2f})\")\n",
    "            mu_plot = mu[param_idx, :, t_idx, 0]\n",
    "            std_plot = std[param_idx, :, t_idx, 0]\n",
    "            y_true_plot = y_data[param_idx, :, t_idx, 0]\n",
    "            plt.plot(grid, mu_plot, '--', lw=2, label=\"μ ± 3σ\")\n",
    "            plt.fill_between(grid, mu_plot + 3*std_plot, mu_plot - 3*std_plot, alpha=0.2)\n",
    "            plt.plot(grid, y_true_plot, color=\"green\", label=\"true\")\n",
    "            plt.legend()\n",
    "            plt.show()\n",
    "\n",
    "    # --- Optional LaTeX row ---\n",
    "    latex_row = None\n",
    "    if return_latex and test_stats:\n",
    "        latex_row = (\n",
    "            f\"{name} & \"\n",
    "            f\"{stats['mse']:.3E} & {stats['nMeRCI_all']:.3E} & {stats['rmsce_all']:.3E} & {stats['mcerr']:.3E} & {stats['crps']:.3E} & \"\n",
    "            f\"{test_stats['mse']:.3E} & {test_stats['nMeRCI_all']:.3E} & {test_stats['rmsce_all']:.3E} & {test_stats['mcerr']:.3E} & {test_stats['crps']:.3E} \\\\\\\\\"\n",
    "        )\n",
    "\n",
    "    return (stats, test_stats, latex_row) if return_latex else (stats, test_stats)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "id": "aa23f617-8a23-4bbc-83e1-e3ed3cdf5498",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAksAAAHHCAYAAACvJxw8AAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/bCgiHAAAACXBIWXMAAA9hAAAPYQGoP6dpAABndklEQVR4nO3deXwTdf4/8NdMzt4t9IZKOeQUqIIggoJSKIKs6LKCF4eKiiIqiyIeIMICuv5cFFlwvRBFQfyiroIooKwieCEoyCFIuWlpOZqeuebz+yPNNGmOpiVtSfN6PjZrM5mZfGZamlc/n/d8RhJCCBARERGRV3JjN4CIiIjoQsawREREROQHwxIRERGRHwxLRERERH4wLBERERH5wbBERERE5AfDEhEREZEfDEtEREREfjAsEREREfnBsER0AcjMzMS4cePU55s2bYIkSdi0aVOjtam66m0k/+6//34MGjSosZtB9eD06dOIiorC2rVrG7sp1EAYlijsLV26FJIkqQ+j0Yj27dtj0qRJyM/Pb+zm1cratWvxzDPPNHYz6sWAAQPcvk++HhfC8efm5uL111/HE0880dhNCXllZWVYtGgRBg8ejLS0NMTExODSSy/F4sWLYbfbA9pHZmam15+V++67z2Pdc+fO4Z577kFSUhKioqJwzTXX4JdffnFbp3nz5rj77rvx9NNPB+UY6cKnbewGEF0onn32WbRu3RoVFRXYvHkzFi9ejLVr12LXrl2IjIxs0LZcffXVKC8vh16vr9V2a9euxaJFiy6IwBBsTz75JO6++271+U8//YSXX34ZTzzxBDp16qQu79atW2M0z81LL72E1q1b45prrmnspoS8gwcP4sEHH8TAgQMxZcoUxMbG4osvvsD999+P77//Hm+//XZA+8nKysLf//53t2Xt27d3e64oCoYNG4Zff/0Vjz76KBITE/Hvf/8bAwYMwLZt23DxxRer69533314+eWX8dVXX+Haa689/wOlC5sgCnNvvfWWACB++uknt+VTpkwRAMR7773nc9uSkpKgtKFVq1Zi7Nix572fBx54QNTXP+tgtTFYVq1aJQCIr7/+2u96wfoeBcpisYjExETx1FNPNej7OjX08da3goICsWvXLo/l48ePFwDE/v37a9xHq1atxLBhw2pcb+XKlQKAWLVqlbrs1KlTIj4+Xtxyyy0e619yySXijjvuqHG/FPo4DEfkg/OvxdzcXADAuHHjEB0djT///BNDhw5FTEwMbrvtNgCOv0gXLFiALl26wGg0IiUlBffeey/Onj3rtk8hBObMmYOWLVsiMjIS11xzDX7//XeP9/ZVs/TDDz9g6NChSEhIQFRUFLp164aXXnpJbd+iRYsAwG2owSnYbazOarWiWbNmGD9+vMdrJpMJRqMRU6dOVZctXLgQXbp0QWRkJBISEtCzZ0+89957Nb6PP8888wwkScLu3btx6623IiEhAf369QPgGMYbMGCAxzbjxo1DZmam27JAz5U3mzdvRmFhIbKzs92WO7+nK1euxBNPPIHU1FRERUXhL3/5C44ePeq27rfffou//e1vuOiii2AwGJCRkYFHHnkE5eXlHm339TNZ230cOXIE119/PaKjo9GiRQv1Z2nnzp249tprERUVhVatWp3396i2EhMT0aVLF4/lN954IwBgz549Ae/LYrGgtLTU5+sffvghUlJScNNNN6nLkpKScPPNN+OTTz6B2Wx2W3/QoEH49NNPIYQIuA0UmjgMR+TDn3/+CcBRn+Bks9mQk5ODfv364YUXXlCH5+69914sXboU48ePx+TJk5Gbm4tXXnkF27dvx3fffQedTgcAmDFjBubMmYOhQ4di6NCh+OWXXzB48GBYLJYa27N+/Xpcf/31SEtLw0MPPYTU1FTs2bMHn332GR566CHce++9OHHiBNavX4933nnHY/v6bqNOp8ONN96I1atX49VXX3UbQvz4449hNpsxevRoAMBrr72GyZMnY+TIkXjooYdQUVGB3377DT/88ANuvfXWGs9FTf72t7/h4osvxty5c+v0QRboufJmy5YtkCQJl156qdfX//GPf0CSJEybNg2nTp3CggULkJ2djR07diAiIgIAsGrVKpSVlWHixIlo3rw5fvzxRyxcuBDHjh3DqlWr3Pbn62eyNvuw2+247rrrcPXVV+P555/H8uXLMWnSJERFReHJJ5/EbbfdhptuuglLlizBmDFj0KdPH7Ru3drvOTx79mxANUWRkZF1GubOy8sD4AhTgfjqq68QGRkJu92OVq1a4ZFHHsFDDz3kts727dtx2WWXQZbd+xF69eqF//znP/jjjz/QtWtXdXmPHj3wr3/9C7///jsuueSSWh8DhZDG7dgianzOYbgNGzaIgoICcfToUbFixQrRvHlzERERIY4dOyaEEGLs2LECgHj88cfdtv/2228FALF8+XK35evWrXNbfurUKaHX68WwYcOEoijqek888YQA4DbE9fXXX7sNMdlsNtG6dWvRqlUrcfbsWbf3cd2Xr2G4+mijN1988YUAID799FO35UOHDhVt2rRRn99www2iS5cufvdVE2/DcDNnzhQAvA6Z9O/fX/Tv399j+dixY0WrVq3U54GeK19uv/120bx5c4/lzu9pixYthMlkUpd/8MEHAoB46aWX1GVlZWUe28+bN09IkiQOHz7s1nZvP5N12cfcuXPVZWfPnhURERFCkiSxYsUKdfnevXsFADFz5kw/Z8ChVatWAkCNj0D2VZ3ZbBadO3cWrVu3Flartcb1hw8fLp577jnx8ccfizfeeENcddVVAoB47LHH3NaLiooSd955p8f2a9asEQDEunXr3JZv2bJFABArV66s9TFQaGHPElGl6sMmrVq1wvLly9GiRQu35RMnTnR7vmrVKsTFxWHQoEEoLCxUl/fo0QPR0dH4+uuvceutt2LDhg2wWCx48MEH3YbHHn74YcydO9dv27Zv347c3Fz861//Qnx8vNtrrvvypSHaCDiGLhMTE7Fy5Upcf/31ABw9DOvXr3cbgouPj8exY8fw008/4fLLL69xv7Xl7SqnQAV6rnw5ffo0EhISfL4+ZswYxMTEqM9HjhyJtLQ0rF27FpMnTwYAtYcJAEpLS1FeXo4rr7wSQghs374dF110kds+q/9M1mUfrsXz8fHx6NChAw4cOICbb75ZXd6hQwfEx8fj4MGDPo/Pafny5R5Dft60adOmxnWqmzRpEnbv3o01a9ZAq635Y+y///2v2/Px48fjuuuuw4svvogHH3wQLVu2BACUl5fDYDB4bG80GtXXXTm/z64/J9Q0MSwRVVq0aBHat28PrVaLlJQUdOjQwaM7XqvVqr9Ynfbv34+ioiIkJyd73e+pU6cAAIcPHwYAtytqAEdNhL8PV6BqSLCuXf0N0UbAcX7++te/4r333oPZbIbBYMDq1athtVoxatQodb1p06Zhw4YN6NWrF9q1a4fBgwfj1ltvRd++fet0fNXVNETkT6Dnyh/hZ+iv+rmVJAnt2rXDoUOH1GVHjhzBjBkz8N///tejTqqoqMjtubefydruw2g0IikpyW1ZXFwcWrZs6RHG4+LiAqrd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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "train_stats, test_stats, latex = compute_statistics(\n",
    "    model,\n",
    "    x_train, y_train,\n",
    "    x_data_test=x_ood_test, \n",
    "    y_data_test=y_ood_test,\n",
    "    t=t, tpred=tpred, grid=grid,\n",
    "    dataset_class=dataset_class,\n",
    "    apply_probconserv=True,\n",
    "    plot=True,\n",
    "    return_latex=True,\n",
    "    name=\"Unconstrained\"\n",
    ")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "id": "3a5ebe0f-9094-408e-8162-db12f3b800dc",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.0009631958603858948"
      ]
     },
     "execution_count": 31,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "test_stats['mse']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "id": "7b0e9a1a-673d-4601-8ecd-0b6be435b39c",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "3.957748333505151e-07"
      ]
     },
     "execution_count": 32,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "test_stats['mcerr']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "id": "42b92e3f-5318-4bac-b650-3dff39acc228",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.0026733049750328063"
      ]
     },
     "execution_count": 33,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "test_stats['crps']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "id": "3505923a-8f51-44c6-9053-31951d11bf9a",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "'Unconstrained & 1.041E-03 & 1.178E+00 & 2.179E-01 & 3.764E-07 & 2.810E-03 & 9.632E-04 & 1.052E+00 & 2.271E-01 & 3.958E-07 & 2.673E-03 \\\\\\\\'"
      ]
     },
     "execution_count": 34,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "latex"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "85984cad-04a3-4b74-92cc-57ecdb4f7f47",
   "metadata": {},
   "source": [
    "'Unconstrained & 1.721E-04 & 3.738E+00 & 2.018E-01 & 3.451E-07 & 2.206E-03 & 1.824E-04 & 3.368E+00 & 2.073E-01 & 3.259E-07 & 2.256E-03 \\\\\\\\'\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "ec02049b-bf1f-446b-ae11-59d5499aa69c",
   "metadata": {},
   "source": [
    "Q=Sigma_inv \n",
    "'Unconstrained & 8.431E-04 & 3.203E-01 & 2.139E-01 & 3.422E-07 & 2.640E-03 & 7.829E-04 & 3.003E-01 & 2.208E-01 & 3.809E-07 & 2.524E-03 \\\\\\\\'\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "bd289bf3-c510-471c-a918-f55601def88d",
   "metadata": {},
   "source": [
    "Q= I 'Unconstrained & 1.041E-03 & 1.178E+00 & 2.179E-01 & 3.764E-07 & 2.810E-03 & 9.632E-04 & 1.052E+00 & 2.271E-01 & 3.958E-07 & 2.673E-03 \\\\\\\\'\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d70de7a5-9e21-48bc-8ed8-490030344715",
   "metadata": {},
   "source": [
    "Probconserv Failure Mode\n",
    "\n",
    "'Unconstrained & 1.810E-03 & 6.771E-01 & 2.012E-01 & 3.705E-07 & 7.401E-03 & 2.128E-02 & 8.291E-01 & 3.712E-01 & 4.638E-07 & 3.411E-02 \\\\\\\\'\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "19db2bf2-4994-4532-b05b-5210f63113dd",
   "metadata": {},
   "source": [
    "\n",
    "    Unconstrained (VarianceNO) & 4.34E-03 & 8.53E-01 & 1.47E-01 & 1.73E-01 & 1.01E-02 & 1.56E-02 & 8.49E-01 & 4.18E-01 & 6.34E-01 & 2.96E-02 \\\\\n",
    "    \\texttt{ProbConserv} & 2.77E-03 & 8.21E-01 & 1.42E-01 & 3.62E-07 & 8.26E-03 & 2.66E-02 & 8.49E-01 & 4.15E-01 & 4.96E-07 & 3.39E-02 \\\\\n",
    "    \\ourmethod{} & 2.10E-03 & 7.83E-01 & 1.73E-01 & 3.61E-07 & 7.83E-03 & 2.26E-02 & 8.72E-01 & 4.19E-01 & 4.97E-07 & 3.41E-02 \\\\\n",
    "    \\ourmethod{} + \\texttt{ProbConserv} & 2.10E-03 & 7.82E-01 & 1.73E-01 & 3.59E-07 & 7.81E-03 & 2.26E-02 & 8.71E-01 & 4.19E-01 & 5.01E-07 & 3.45E-02 \\\\"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "optprobconserv",
   "language": "python",
   "name": "optprobconserv"
  },
  "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.19"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
