{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 813,
   "id": "6b3ccdb8-4fd8-4224-b23e-979fa8228059",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[autoreload of models.UncertainNO failed: Traceback (most recent call last):\n",
      "  File \"/home/utkarsh530/anaconda3/envs/optprobconserv/lib/python3.9/site-packages/IPython/extensions/autoreload.py\", line 276, in check\n",
      "    superreload(m, reload, self.old_objects)\n",
      "  File \"/home/utkarsh530/anaconda3/envs/optprobconserv/lib/python3.9/site-packages/IPython/extensions/autoreload.py\", line 475, in superreload\n",
      "    module = reload(module)\n",
      "  File \"/home/utkarsh530/anaconda3/envs/optprobconserv/lib/python3.9/importlib/__init__.py\", line 169, in reload\n",
      "    _bootstrap._exec(spec, module)\n",
      "  File \"<frozen importlib._bootstrap>\", line 613, in _exec\n",
      "  File \"<frozen importlib._bootstrap_external>\", line 846, in exec_module\n",
      "  File \"<frozen importlib._bootstrap_external>\", line 983, in get_code\n",
      "  File \"<frozen importlib._bootstrap_external>\", line 913, in source_to_code\n",
      "  File \"<frozen importlib._bootstrap>\", line 228, in _call_with_frames_removed\n",
      "  File \"/home/utkarsh530/e2e-hardconstraints/models/UncertainNO.py\", line 545\n",
      "    out = \n",
      "          ^\n",
      "SyntaxError: invalid syntax\n",
      "]\n"
     ]
    }
   ],
   "source": [
    "import os\n",
    "os.environ['CUDA_VISIBLE_DEVICES'] = '1'\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": 814,
   "id": "14229855-e486-4513-b332-55c13ba531fa",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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HEBkZKQcnzRpEnbv3o20tLT79jFu3Djs2LEDp0+fhrOzMwBg7dq1qFOnDgIDA5GVlYX/+7//g6urK1JTU6HVak36MHfGybhzurnd+YfOmn+lvJr0k83POC1LXKhob4szTiVpmxW5Pi5/edXEvyaflJxeXfqjzc84La0wd2x1xqkkbXON3k7Myf5y+seyn+S4rc44GeeOLc84lR7aYpPtVFRUBE9PTxQWFsrH/apU+6W6hzFnzhysXbsWKSkpctEEAEOHDpX/v23btggNDUWzZs2QkpKC3r17m/Tj5OQEJycnk7hWqzUptDSayrvOg8VFCFXEzTMfF6qMlxtMl4iiBLHShADu7JzmmOujYrzye2OuKLU0LgiCRXFrbY+q4szJvnKSIJjZg607nyrv96IkQTSYrvVh5lPF96gmbifmZH85mZsj1p5PFfdxSTJ/DHnY+WTM0drbqarlZtehuqUNeHl5QavVIj8/XxHPz8+/7/1JH374IebMmYMffvgBoaGh92zbtGlTeHl54ezZsw89ZiIiInpyVWvh5OjoiI4dOyI5OVmOiaKI5ORkxaW7yubOnYuZM2di+/btCA8Pv+96Lly4gGvXrsHHx8cq4yYiIqInU7V/HUF8fDz+/e9/Y+XKlThz5gzGjh0LvV4vf8puxIgRipvHP/jgA0ydOhXLli1DQEAA8vLykJeXJ9/YVVxcjH/96184ePAgcnJykJycjAEDBqB58+aIiYmplhyJiIioZqj2e5yGDBmCK1euYNq0acjLy0O7du2wfft2eHt7AwDOnz+vuGb52WefoaysDIMHD1b0M336dMyYMQNarRYnTpzAypUrcePGDfj6+uKZZ57BzJkzzd7HRERERKRWtRdOADB+/HiMHz/e7LKUlBTF85ycnHv2Vbt2bezYscNKIyMiIiK6q9ov1RERERE9Llg4EREREanEwomIiIhIJRZORERERCqxcCIiIiJSiYUTERERkUosnIiIiIhUYuFEREREpBILJyIiIiKVWDgRERERqcTCiYiIiEglFk5EREREKrFwIiIiIlKJhRMRERGRSiyciIiIiFRi4URERESkEgsnIiIiIpVYOBERERGpxMKJiIiISCUWTkREREQqsXAiIiIiUomFExEREZFKLJyIiIiIVGLhRERERKQSCyciIiIilVg4EREREanEwomIiIhIJRZORERE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",
      "text/plain": [
       "<Figure size 600x400 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "\n",
    "# Data\n",
    "pdes = ['Heat', 'PME', 'Advection']\n",
    "time_no_sampling = [0.454, 0.487, 0.454]\n",
    "time_with_sampling = [1.62, 1.75, 1.63]\n",
    "\n",
    "x = np.arange(len(pdes))\n",
    "width = 0.12  # Narrower bars\n",
    "\n",
    "# Plot\n",
    "fig, ax = plt.subplots(figsize=(6, 4))\n",
    "\n",
    "# New non-blue/orange colors\n",
    "ax.bar(x - width/2, time_no_sampling, width, label='No Sampling', color='#669bbc')   # green\n",
    "ax.bar(x + width/2, time_with_sampling, width, label='With Sampling (100)', color='#003049')  # magenta\n",
    "\n",
    "# Labels and layout\n",
    "ax.set_ylabel('Time per Epoch (s)')\n",
    "ax.set_xticks(x)\n",
    "ax.set_xticklabels(pdes)\n",
    "ax.grid(axis='y', linestyle='--', alpha=0.5)\n",
    "\n",
    "# Legend at center top, horizontal layout\n",
    "ax.legend(loc='upper center', bbox_to_anchor=(0.5, 1.12),\n",
    "          ncol=2, frameon=False)\n",
    "\n",
    "plt.tight_layout()\n",
    "plt.savefig(\"pde_timing_sampling.pdf\", dpi=300)\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 815,
   "id": "5fdcc8cd-82d9-4cec-888d-806541259f2c",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The autoreload extension is already loaded. To reload it, use:\n",
      "  %reload_ext autoreload\n"
     ]
    }
   ],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 816,
   "id": "1d5eb530-ab43-4054-8953-9efdb86b3106",
   "metadata": {},
   "outputs": [],
   "source": [
    "args = {'--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'}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 817,
   "id": "a69b4337-c84b-4454-af1f-979e654417e3",
   "metadata": {},
   "outputs": [],
   "source": [
    "# lambdas_pinn = {}\n",
    "# lambdas_pinn['Heat_Eq]\n",
    "# heq = 1e4"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 818,
   "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": 819,
   "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": 820,
   "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": 821,
   "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": 822,
   "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 = SoftOutputVarNO(base_model_class=FNO2d, probconserv=False, base_model_params=FNO2d_params, constraint_context=constraint_context, lambda_pinn=1e-4)\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": 823,
   "id": "4333d6f8-fe2c-4caa-a889-efae589c31ff",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([160, 100, 20, 1])"
      ]
     },
     "execution_count": 823,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "y_train.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 824,
   "id": "96aa9ee0-a30e-458b-85e2-5e7dbbd89dc4",
   "metadata": {},
   "outputs": [],
   "source": [
    "mu_true = torch.mean(y_train, dim = 0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 825,
   "id": "fc1ac32d-6fb2-4293-8ea7-f90486a32dce",
   "metadata": {},
   "outputs": [],
   "source": [
    "var_true = torch.var(y_train, dim = 0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 826,
   "id": "f07c25ce-49e4-4331-81a3-ee678808a6b4",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([100, 20, 1])"
      ]
     },
     "execution_count": 826,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "mu_true.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 827,
   "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.2538, 0.3701, 0.4510,\n",
      "        0.4887, 0.5009, 0.4949, 0.4698, 0.4230, 0.3245, 0.2052, 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])\n"
     ]
    },
    {
     "data": {
      "image/png": "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jl0VM2bMwM6dOxEXF4dXXnkFLVq0wO3bt3H48GFs27YNt2/fBlCyDsm6devw1FNPoW/fvkhLS0NycjJatGhh9pdDdT3++OOYMmUKhg4dio4dO+LYsWNYvnx5lcfzlOXp6Ylp06bh1VdfRffu3ZGUlIS0tDQsXbrUqmMOGDAAY8eOxdixY1GnTp1yfwX37dsXs2fPRq9evfDcc8/h+vXrWLBgARo3bozff/+9WrUDJefjq6++glarRYsWLZCSkoJt27aZHS9VFQEBAZgzZw5efvllPPTQQ3juuedQu3ZtHD16FLm5ufjiiy+gVCqxZMkS9O7dGy1btsTQoUNRr149pKenY+fOnQgICMD3339v9WuPGzcO//3vf9G7d2+8+eabqFOnDr744gukpaXhm2++qdFiiUDJwPhp06aV2961a9ca/ZVuzfmo6c/ru+++w7179/DEE0+Yffzhhx9GUFAQli9fjqSkJPzjH/8w3sLk73//O3x9fbF48WJERkaave769OkDf39/jB071viHUGmNGjXCtGnTMH78eFy8eBH9+/eHv78/0tLSsH79egwbNgxjx46FUqnEokWL0K9fP8TExGDo0KEICwvD6dOnceLECfz0008ALH+emjNt2jRs3boVnTp1wvDhw+Hh4YH//Oc/KCgowEcffVSl81cdn3zyCe7evWv8Y/T777/Hn3/+CaCkq7P0pIqsrCx8/fXXAEruNmBYWfr8+fMYOHAgpk6darc67caRU9So5iytI1SWuXWEhBBi1apVol27dkKj0Yg6deqI559/3jjl3qC4uFi88cYbIigoSCgUCpOpn3v37hWxsbFCrVabTP20NH3e3HTeyMhIMXjwYJNt27dvF+3atRNqtVo0atRILFmyRLz11lvCy8urkjNieUpnZGSk6Nu3b7nt5urKzMwUI0aMEBEREcLT01OEhoaKHj16iMWLFxv30ev14oMPPhCRkZFCo9GIdu3aiY0bN4rBgweLyMhI436G6fOl1yUp/drmlh8oLT8/X7z11lsiLCxMeHt7i0ceeUSkpKRYnPpceh2f0q9fdgrswoULjWuStG/fXuzevbvcMSvzyCOPCADi5ZdfNvv4Z599Jpo0aSI0Go2Ijo4WS5cuteraMDxW+hzduXNHDB06VAQGBgo/Pz/Rs2dPcfr06XLXkaX3hqX3wnfffSc6duwovL29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",
      "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": 828,
   "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": 828,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "dataset_class.get_mass_rhs_func(x=x_train)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 829,
   "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": 830,
   "id": "b339b6e5",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "device(type='cpu')"
      ]
     },
     "execution_count": 830,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "x_train.device"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 831,
   "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": 832,
   "id": "d71ec298-4bbc-4186-a938-8675f0526890",
   "metadata": {},
   "outputs": [],
   "source": [
    "import time"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 833,
   "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": 833,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "grid"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 834,
   "id": "2165975d",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[0, -1, 5]"
      ]
     },
     "execution_count": 834,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "tpred"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 835,
   "id": "1ad2c532",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([ 0, -1,  5])"
      ]
     },
     "execution_count": 835,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "torch.tensor(tpred)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 836,
   "id": "18a1c208-a4df-4f27-8dad-f2466164e855",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([160, 100, 20, 1])"
      ]
     },
     "execution_count": 836,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "x_train.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 837,
   "id": "6ff40f4d-d8b7-45f7-8e3a-8115564540df",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 0: Train loss=1028.576648, Validation loss=439.560231 (saved)\n",
      "Epoch 1: Train loss=-174.807974, Validation loss=-920.623291 (saved)\n",
      "Epoch 2: Train loss=-1121.867725, Validation loss=-1593.616846 (saved)\n",
      "Epoch 3: Train loss=-1927.318323, Validation loss=-2568.021387 (saved)\n",
      "Epoch 4: Train loss=-3531.583203, Validation loss=-5510.720508 (saved)\n",
      "Epoch 5: Train loss=-6902.640186, Validation loss=-9006.164062 (saved)\n",
      "Epoch 6: Train loss=-8676.645801, Validation loss=-9443.957031 (saved)\n",
      "Epoch 7: Train loss=-9659.240918, Validation loss=-10468.813672 (saved)\n",
      "Epoch 8: Train loss=-5271.716113, Validation loss=9015.691797 \n",
      "Epoch 9: Train loss=-5228.387500, Validation loss=-6835.995703 \n",
      "Epoch 10: Train loss=-6586.420996, Validation loss=-5980.885547 \n",
      "Epoch 11: Train loss=-5801.724707, Validation loss=-5773.838867 \n",
      "Epoch 12: Train loss=-5973.136328, Validation loss=-6272.569727 \n",
      "Epoch 13: Train loss=-6495.194092, Validation loss=-6746.075781 \n",
      "Epoch 14: Train loss=-6942.167578, Validation loss=-7239.555469 \n",
      "Epoch 15: Train loss=-7506.972168, Validation loss=-7852.526953 \n",
      "Epoch 16: Train loss=-8143.285352, Validation loss=-8552.426172 \n",
      "Epoch 17: Train loss=-8903.837891, Validation loss=-9386.729687 \n",
      "Epoch 18: Train loss=-9802.167871, Validation loss=-10387.080859 \n",
      "Epoch 19: Train loss=-10891.221191, Validation loss=-11614.101953 (saved)\n",
      "Epoch 20: Train loss=-12240.968750, Validation loss=-13162.466406 (saved)\n",
      "Epoch 21: Train loss=-11678.768567, Validation loss=55795.200000 \n",
      "Epoch 22: Train loss=2293.939453, Validation loss=-5440.127734 \n",
      "Epoch 23: Train loss=-5644.460449, Validation loss=-6289.471484 \n",
      "Epoch 24: Train loss=-6669.816211, Validation loss=-7002.342969 \n",
      "Epoch 25: Train loss=-7065.150293, Validation loss=-7075.049219 \n",
      "Epoch 26: Train loss=-7102.488574, Validation loss=-7198.383594 \n",
      "Epoch 27: Train loss=-7337.681641, Validation loss=-7519.654297 \n",
      "Epoch 28: Train loss=-7651.821387, Validation loss=-7817.968359 \n",
      "Epoch 29: Train loss=-7962.230176, Validation loss=-8158.356641 \n",
      "Epoch 30: Train loss=-8321.431641, Validation loss=-8532.864453 \n",
      "Epoch 31: Train loss=-8704.031836, Validation loss=-8934.298047 \n",
      "Epoch 32: Train loss=-9121.418457, Validation loss=-9371.748828 \n",
      "Epoch 33: Train loss=-9574.584766, Validation loss=-9850.119922 \n",
      "Epoch 34: Train loss=-10072.950781, Validation loss=-10376.446094 \n",
      "Epoch 35: Train loss=-10622.627930, Validation loss=-10962.235156 \n",
      "Epoch 36: Train loss=-11238.790918, Validation loss=-11618.813281 \n",
      "Epoch 37: Train loss=-11931.948242, Validation loss=-12366.001172 \n",
      "Epoch 38: Train loss=-12726.070703, Validation loss=-13225.884375 (saved)\n",
      "Epoch 39: Train loss=-13645.959375, Validation loss=-14233.837500 (saved)\n",
      "Epoch 40: Train loss=-14727.596875, Validation loss=-15368.512500 (saved)\n",
      "Epoch 41: Train loss=35424.126172, Validation loss=-7198.086719 \n",
      "Epoch 42: Train loss=-7078.479541, Validation loss=-6172.903320 \n",
      "Epoch 43: Train loss=-6245.529395, Validation loss=-6116.857031 \n",
      "Epoch 44: Train loss=-5941.284375, Validation loss=-5772.578125 \n",
      "Epoch 45: Train loss=-5733.481885, Validation loss=-5729.341602 \n",
      "Epoch 46: Train loss=-5766.423340, Validation loss=-5822.985742 \n",
      "Epoch 47: Train loss=-5868.912842, Validation loss=-5924.560937 \n",
      "Epoch 48: Train loss=-5962.600684, Validation loss=-6009.428711 \n",
      "Epoch 49: Train loss=-6046.247949, Validation loss=-6094.167578 \n",
      "Epoch 50: Train loss=-6114.093555, Validation loss=-6139.576953 \n",
      "Epoch 51: Train loss=-6160.151563, Validation loss=-6186.425977 \n",
      "Epoch 52: Train loss=-6207.202734, Validation loss=-6233.571289 \n",
      "Epoch 53: Train loss=-6254.283105, Validation loss=-6280.988672 \n",
      "Epoch 54: Train loss=-6301.897656, Validation loss=-6328.665820 \n",
      "Epoch 55: Train loss=-6349.760449, Validation loss=-6376.972461 \n",
      "Epoch 56: Train loss=-6398.305615, Validation loss=-6425.938672 \n",
      "Epoch 57: Train loss=-6447.487451, Validation loss=-6475.616406 \n",
      "Epoch 58: Train loss=-6497.445117, Validation loss=-6525.980273 \n",
      "Epoch 59: Train loss=-6547.984961, Validation loss=-6577.157031 \n",
      "Epoch 60: Train loss=-6599.484180, Validation loss=-6629.040625 \n",
      "Epoch 61: Train loss=-6651.729785, Validation loss=-6681.717187 \n",
      "Epoch 62: Train loss=-6704.645020, Validation loss=-6735.338281 \n",
      "Epoch 63: Train loss=-6758.706934, Validation loss=-6789.724609 \n",
      "Epoch 64: Train loss=-6813.435840, Validation loss=-6845.123828 \n",
      "Epoch 65: Train loss=-6869.233301, Validation loss=-6901.437109 \n",
      "Epoch 66: Train loss=-6925.967285, Validation loss=-6958.751172 \n",
      "Epoch 67: Train loss=-6983.682324, Validation loss=-7017.106250 \n",
      "Epoch 68: Train loss=-7042.458496, Validation loss=-7076.519141 \n",
      "Epoch 69: Train loss=-7102.310547, Validation loss=-7137.001953 \n",
      "Epoch 70: Train loss=-7163.256250, Validation loss=-7198.619531 \n",
      "Epoch 71: Train loss=-7225.310254, Validation loss=-7261.435937 \n",
      "Epoch 72: Train loss=-7288.691016, Validation loss=-7325.326562 \n",
      "Epoch 73: Train loss=-7353.113867, Validation loss=-7390.507812 \n",
      "Epoch 74: Train loss=-7418.766602, Validation loss=-7457.021484 \n",
      "Epoch 75: Train loss=-7485.838184, Validation loss=-7524.809375 \n",
      "Epoch 76: Train loss=-7554.151172, Validation loss=-7593.937891 \n",
      "Epoch 77: Train loss=-7623.891504, Validation loss=-7664.422266 \n",
      "Epoch 78: Train loss=-7695.023730, Validation loss=-7736.306250 \n",
      "Epoch 79: Train loss=-7767.504883, Validation loss=-7809.726562 \n",
      "Epoch 80: Train loss=-7841.585156, Validation loss=-7884.658594 \n",
      "Epoch 81: Train loss=-7917.189160, Validation loss=-7961.125000 \n",
      "Epoch 82: Train loss=-7994.394727, Validation loss=-8039.242187 \n",
      "Epoch 83: Train loss=-8073.200195, Validation loss=-8119.155469 \n",
      "Epoch 84: Train loss=-8153.817383, Validation loss=-8200.776172 \n",
      "Epoch 85: Train loss=-8236.122168, Validation loss=-8284.234766 \n",
      "Epoch 86: Train loss=-8320.350879, Validation loss=-8369.450391 \n",
      "Epoch 87: Train loss=-8406.399219, Validation loss=-8456.819531 \n",
      "Epoch 88: Train loss=-8494.540332, Validation loss=-8546.172266 \n",
      "Epoch 89: Train loss=-8584.634668, Validation loss=-8637.748828 \n",
      "Epoch 90: Train loss=-8677.204590, Validation loss=-8731.278906 \n",
      "Epoch 91: Train loss=-8771.833105, Validation loss=-8827.398828 \n",
      "Epoch 92: Train loss=-8868.937695, Validation loss=-8925.903125 \n",
      "Epoch 93: Train loss=-8968.465723, Validation loss=-9027.071875 \n",
      "Epoch 94: Train loss=-9070.764063, Validation loss=-9130.762500 \n",
      "Epoch 95: Train loss=-9175.712891, Validation loss=-9237.548437 \n",
      "Epoch 96: Train loss=-9283.645801, Validation loss=-9347.376953 \n",
      "Epoch 97: Train loss=-9394.678516, Validation loss=-9460.958984 \n",
      "Epoch 98: Train loss=-9509.438867, Validation loss=-9577.490625 \n",
      "Epoch 99: Train loss=-9627.538770, Validation loss=-9697.828125 \n",
      "Epoch 100: Train loss=-9722.516016, Validation loss=-9759.693359 \n",
      "Epoch 101: Train loss=-9785.012988, Validation loss=-9823.293750 \n",
      "Epoch 102: Train loss=-9848.972363, Validation loss=-9888.128125 \n",
      "Epoch 103: Train loss=-9914.383691, Validation loss=-9954.292578 \n",
      "Epoch 104: Train loss=-9981.157129, Validation loss=-10021.923047 \n",
      "Epoch 105: Train loss=-10049.348926, Validation loss=-10091.309375 \n",
      "Epoch 106: Train loss=-10119.199023, Validation loss=-10161.959766 \n",
      "Epoch 107: Train loss=-10190.634375, Validation loss=-10234.319141 \n",
      "Epoch 108: Train loss=-10263.518359, Validation loss=-10308.807031 \n",
      "Epoch 109: Train loss=-10338.529590, Validation loss=-10384.694922 \n",
      "Epoch 110: Train loss=-10415.160547, Validation loss=-10462.386719 \n",
      "Epoch 111: Train loss=-10493.728906, Validation loss=-10542.192187 \n",
      "Epoch 112: Train loss=-10574.227441, Validation loss=-10624.286719 \n",
      "Epoch 113: Train loss=-10656.997070, Validation loss=-10708.163672 \n",
      "Epoch 114: Train loss=-10741.691992, Validation loss=-10794.259375 \n",
      "Epoch 115: Train loss=-10828.981543, Validation loss=-10883.467578 \n",
      "Epoch 116: Train loss=-10918.672656, Validation loss=-10974.736719 \n",
      "Epoch 117: Train loss=-11011.014063, Validation loss=-11068.535156 \n",
      "Epoch 118: Train loss=-11105.725586, Validation loss=-11165.594531 \n",
      "Epoch 119: Train loss=-11203.772559, Validation loss=-11264.917187 \n",
      "Epoch 120: Train loss=-11304.565039, Validation loss=-11367.576562 \n",
      "Epoch 121: Train loss=-11408.511230, Validation loss=-11473.837891 \n",
      "Epoch 122: Train loss=-11516.026270, Validation loss=-11583.530859 \n",
      "Epoch 123: Train loss=-11627.013867, Validation loss=-11696.848047 \n",
      "Epoch 124: Train loss=-11741.691211, Validation loss=-11814.067578 \n",
      "Epoch 125: Train loss=-11860.392578, Validation loss=-11934.951172 \n",
      "Epoch 126: Train loss=-11983.537891, Validation loss=-12061.066016 \n",
      "Epoch 127: Train loss=-12110.890137, Validation loss=-12191.886328 \n",
      "Epoch 128: Train loss=-12243.510156, Validation loss=-12326.907422 \n",
      "Epoch 129: Train loss=-12380.944238, Validation loss=-12467.468750 \n",
      "Epoch 130: Train loss=-12524.204980, Validation loss=-12614.708594 \n",
      "Epoch 131: Train loss=-12673.303516, Validation loss=-12767.473437 \n",
      "Epoch 132: Train loss=-12828.700781, Validation loss=-12926.841016 \n",
      "Epoch 133: Train loss=-12991.330469, Validation loss=-13094.658594 \n",
      "Epoch 134: Train loss=-13161.996484, Validation loss=-13269.421875 \n",
      "Epoch 135: Train loss=-13340.829297, Validation loss=-13453.165625 \n",
      "Epoch 136: Train loss=-13529.506641, Validation loss=-13647.081250 \n",
      "Epoch 137: Train loss=-13728.400977, Validation loss=-13854.067969 \n",
      "Epoch 138: Train loss=-13939.850000, Validation loss=-14073.019531 \n",
      "Epoch 139: Train loss=-14164.907617, Validation loss=-14306.122656 \n",
      "Epoch 140: Train loss=-14405.559570, Validation loss=-14557.554687 \n",
      "Epoch 141: Train loss=-14664.479687, Validation loss=-14819.922656 \n",
      "Epoch 142: Train loss=-14945.269922, Validation loss=-15125.704687 \n",
      "Epoch 143: Train loss=-15255.665039, Validation loss=-15450.244531 (saved)\n",
      "Epoch 144: Train loss=-15597.507227, Validation loss=-15810.246875 (saved)\n",
      "Epoch 145: Train loss=-15972.989258, Validation loss=-16212.094531 (saved)\n",
      "Epoch 146: Train loss=-16394.447461, Validation loss=-16652.865625 (saved)\n",
      "Epoch 147: Train loss=-16794.233984, Validation loss=-16940.505469 (saved)\n",
      "Epoch 148: Train loss=-17280.083008, Validation loss=-17459.973437 (saved)\n",
      "Epoch 149: Train loss=-14618.501855, Validation loss=-14071.960937 \n",
      "Epoch 150: Train loss=-16282.110352, Validation loss=-16219.982812 \n",
      "Epoch 151: Train loss=-17085.718750, Validation loss=-17432.821875 \n",
      "Epoch 152: Train loss=-17487.755469, Validation loss=-17791.342969 (saved)\n",
      "Epoch 153: Train loss=-17754.160742, Validation loss=-17919.459375 (saved)\n",
      "Epoch 154: Train loss=-17940.348438, Validation loss=-18077.692187 (saved)\n",
      "Epoch 155: Train loss=-18114.024805, Validation loss=-18213.440625 (saved)\n",
      "Epoch 156: Train loss=-18292.428516, Validation loss=-18402.538281 (saved)\n",
      "Epoch 157: Train loss=-18472.647656, Validation loss=-18593.938281 (saved)\n",
      "Epoch 158: Train loss=-18657.428320, Validation loss=-18787.506250 (saved)\n",
      "Epoch 159: Train loss=-18854.780078, Validation loss=-18977.725781 (saved)\n",
      "Epoch 160: Train loss=-19055.829883, Validation loss=-19187.403906 (saved)\n",
      "Epoch 161: Train loss=-19265.392969, Validation loss=-19396.401562 (saved)\n",
      "Epoch 162: Train loss=-19468.731836, Validation loss=-19560.596875 (saved)\n",
      "Epoch 163: Train loss=-18884.850000, Validation loss=-13857.385156 \n",
      "Epoch 164: Train loss=-17660.592969, Validation loss=-18314.564062 \n",
      "Epoch 165: Train loss=-18983.253516, Validation loss=-18382.757031 \n",
      "Epoch 166: Train loss=-19286.858008, Validation loss=-19491.184375 \n",
      "Epoch 167: Train loss=-19645.223828, Validation loss=-19982.421094 (saved)\n",
      "Epoch 168: Train loss=-19895.901758, Validation loss=-20001.707812 (saved)\n",
      "Epoch 169: Train loss=-20154.312695, Validation loss=-20311.430469 (saved)\n",
      "Epoch 170: Train loss=-20362.643359, Validation loss=-20465.396875 (saved)\n",
      "Epoch 171: Train loss=-20296.451953, Validation loss=-16786.202344 \n",
      "Epoch 172: Train loss=-12868.404346, Validation loss=-13246.078125 \n",
      "Epoch 173: Train loss=-17624.273633, Validation loss=-17207.960937 \n",
      "Epoch 174: Train loss=-19085.070703, Validation loss=-19483.276562 \n",
      "Epoch 175: Train loss=-19731.083594, Validation loss=-20215.183594 \n",
      "Epoch 176: Train loss=-20053.464453, Validation loss=-20131.676562 \n",
      "Epoch 177: Train loss=-20224.975391, Validation loss=-20303.767969 \n",
      "Epoch 178: Train loss=-20404.843359, Validation loss=-20498.573437 (saved)\n",
      "Epoch 179: Train loss=-20551.222656, Validation loss=-20645.589062 (saved)\n",
      "Epoch 180: Train loss=-20690.056445, Validation loss=-20779.611719 (saved)\n",
      "Epoch 181: Train loss=-20821.926367, Validation loss=-20919.097656 (saved)\n",
      "Epoch 182: Train loss=-20958.610156, Validation loss=-21054.117187 (saved)\n",
      "Epoch 183: Train loss=-21109.333984, Validation loss=-21208.646875 (saved)\n",
      "Epoch 184: Train loss=-21246.540234, Validation loss=-21359.332031 (saved)\n",
      "Epoch 185: Train loss=-21387.257617, Validation loss=-21275.116406 \n",
      "Epoch 186: Train loss=-17717.595117, Validation loss=-9134.205469 \n",
      "Epoch 187: Train loss=-17772.034473, Validation loss=-20267.312500 \n",
      "Epoch 188: Train loss=-20251.524023, Validation loss=-20193.210156 \n",
      "Epoch 189: Train loss=-20677.424023, Validation loss=-20603.893750 \n",
      "Epoch 190: Train loss=-21140.726953, Validation loss=-21209.466406 \n",
      "Epoch 191: Train loss=-21364.612305, Validation loss=-21484.275000 (saved)\n",
      "Epoch 192: Train loss=-21522.474609, Validation loss=-21580.267187 (saved)\n",
      "Epoch 193: Train loss=-21579.247656, Validation loss=-21694.343750 (saved)\n",
      "Epoch 194: Train loss=-21684.509375, Validation loss=-21868.345312 (saved)\n",
      "Epoch 195: Train loss=-21832.039453, Validation loss=-22003.371875 (saved)\n",
      "Epoch 196: Train loss=-21803.517383, Validation loss=-20169.514062 \n",
      "Epoch 197: Train loss=-13763.076123, Validation loss=-15190.598437 \n",
      "Epoch 198: Train loss=-18398.841992, Validation loss=-19716.335937 \n",
      "Epoch 199: Train loss=-20399.063086, Validation loss=-20756.865625 \n",
      "Finished training with best train loss: -21988.515820 and validation loss: -22003.371875\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": 838,
   "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": 839,
   "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",
    "\n",
    "            # out = model.base_model._apply_constraints(_mu, _std, x, t, tpred, grid, dataset_class)\n",
    "\n",
    "\n",
    "\n",
    "            if model.probconserv:\n",
    "                _mu, _var = out\n",
    "                _std = torch.sqrt(_var)\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",
    "                out = (new_mu.unsqueeze(-1), torch.square(new_std).unsqueeze(-1))\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": 840,
   "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.0062325683422386645\n",
      "n-MeRCI: 0.2902601957321167\n",
      "RMSCE: 0.21963952481746674\n",
      "Cerr: 0.19116143882274628\n",
      "In-domain results\n",
      "MSE: 0.005855180025100708\n",
      "n-MeRCI: 0.2542012929916382\n",
      "RMSCE: 0.22335682809352875\n",
      "ProbConserv Results\n",
      "MSE: 0.0027483826875686645\n",
      "n-MeRCI: 0.1816856563091278\n",
      "RMSCE: 0.22512495517730713\n",
      "Cerr: 0.18004390597343445\n",
      "Prob_Cerr: 3.6090611388317484e-07\n",
      "Here\n",
      "\n",
      "\n",
      "Out-of-domain results\n",
      "MSE: 0.0059910941869020466\n",
      "n-MeRCI: 0.25686484575271606\n",
      "RMSCE: 0.22187484800815582\n",
      "ProbConserv Results\n",
      "MSE: 0.0027010133862495423\n",
      "n-MeRCI: 0.1876482516527176\n",
      "RMSCE: 0.22379235923290253\n",
      "Cerr: 0.18219928443431854\n",
      "Prob_Cerr: 3.680586928567209e-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": 841,
   "id": "98acb797",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.0027483826875686645"
      ]
     },
     "execution_count": 841,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "id_results['pc.mse']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 842,
   "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_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_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']:.2E} & {stats['nMeRCI_all']:.2E} & {stats['rmsce_all']:.2E} & {stats['mcerr']:.2E} & {stats['crps']:.2E} & \"\n",
    "            f\"{test_stats['mse']:.2E} & {test_stats['nMeRCI_all']:.2E} & {test_stats['rmsce_all']:.2E} & {test_stats['mcerr']:.2E} & {test_stats['crps']:.2E} \\\\\\\\\"\n",
    "        )\n",
    "\n",
    "    return (stats, test_stats, latex_row) if return_latex else (stats, test_stats)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 843,
   "id": "aa23f617-8a23-4bbc-83e1-e3ed3cdf5498",
   "metadata": {},
   "outputs": [],
   "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=False,\n",
    "    plot=False,\n",
    "    return_latex=True,\n",
    "    name=\"Unconstrained\"\n",
    ")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 844,
   "id": "3a5ebe0f-9094-408e-8162-db12f3b800dc",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.0059910941123962405"
      ]
     },
     "execution_count": 844,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "test_stats['mse']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 845,
   "id": "7b0e9a1a-673d-4601-8ecd-0b6be435b39c",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.18219928443431854"
      ]
     },
     "execution_count": 845,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "test_stats['mcerr']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 846,
   "id": "42b92e3f-5318-4bac-b650-3dff39acc228",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.00978479266166687"
      ]
     },
     "execution_count": 846,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "test_stats['crps']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 847,
   "id": "3505923a-8f51-44c6-9053-31951d11bf9a",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "'Unconstrained & 6.23E-03 & 2.90E-01 & 2.20E-01 & 1.91E-01 & 1.03E-02 & 5.99E-03 & 2.57E-01 & 2.22E-01 & 1.82E-01 & 9.78E-03 \\\\\\\\'"
      ]
     },
     "execution_count": 847,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "latex"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "6f43028d-91c3-407f-8bcd-ca4bcce22141",
   "metadata": {},
   "source": [
    "## CRPS\n",
    "Heat = 1e4 \n",
    "\n",
    "PME = 1e1 \n",
    "\n",
    "Stefan = 1e1 \n",
    "\n",
    "Advection = 1e-2 \n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3c034735-f46e-4356-a34c-ccda4a81b704",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "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
}
