{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "a24c8b32-9ec7-474b-927f-8fb7bdd5da54",
   "metadata": {},
   "source": [
    "\n",
    "## Hard-Constrained (HardC) Baseline for PDEs\n",
    "\n",
    "This notebook implements the **HardC** baseline referenced in **Table 1** of the main paper. It evaluates a simple E2E constraint enforcement strategy: train an unconstrained model and **project the predicted mean** E2E to satisfy hard physical constraints (e.g., initial conditions, mass conservation). **The variance is not updated** during this projection.\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",
    "````\n",
    "\n",
    "Change the following keys to switch PDE datasets:\n",
    "\n",
    "* `--dataset`: one of\n",
    "\n",
    "  * `HeatEquation_1D`\n",
    "  * `PME_1D`\n",
    "  * `StefanPME_1D`\n",
    "  * `LinearAdvection_1D`\n",
    "\n",
    "* `--dataset_params` and `--train_ood_dataset_params`: refer to the training parameter ranges for each PDE as defined in the paper.\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "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": 36,
   "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": 37,
   "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": 38,
   "id": "a69b4337-c84b-4454-af1f-979e654417e3",
   "metadata": {},
   "outputs": [],
   "source": [
    "# lambdas_pinn = {}\n",
    "# lambdas_pinn['Heat_Eq]\n",
    "# heq = 1e4"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 39,
   "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": 40,
   "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": 41,
   "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": 42,
   "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": 43,
   "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 = HardOutputVarNO(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": 44,
   "id": "4333d6f8-fe2c-4caa-a889-efae589c31ff",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([160, 100, 20, 1])"
      ]
     },
     "execution_count": 44,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "y_train.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 45,
   "id": "96aa9ee0-a30e-458b-85e2-5e7dbbd89dc4",
   "metadata": {},
   "outputs": [],
   "source": [
    "mu_true = torch.mean(y_train, dim = 0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 46,
   "id": "fc1ac32d-6fb2-4293-8ea7-f90486a32dce",
   "metadata": {},
   "outputs": [],
   "source": [
    "var_true = torch.var(y_train, dim = 0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 47,
   "id": "f07c25ce-49e4-4331-81a3-ee678808a6b4",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([100, 20, 1])"
      ]
     },
     "execution_count": 47,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "mu_true.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 48,
   "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": 49,
   "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": 49,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "dataset_class.get_mass_rhs_func(x=x_train)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 50,
   "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": 51,
   "id": "b339b6e5",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "device(type='cpu')"
      ]
     },
     "execution_count": 51,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "x_train.device"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 52,
   "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": 53,
   "id": "d71ec298-4bbc-4186-a938-8675f0526890",
   "metadata": {},
   "outputs": [],
   "source": [
    "import time"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 54,
   "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": 54,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "grid"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 55,
   "id": "2165975d",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[0, -1, 5]"
      ]
     },
     "execution_count": 55,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "tpred"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 56,
   "id": "1ad2c532",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([ 0, -1,  5])"
      ]
     },
     "execution_count": 56,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "torch.tensor(tpred)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 57,
   "id": "18a1c208-a4df-4f27-8dad-f2466164e855",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([160, 100, 20, 1])"
      ]
     },
     "execution_count": 57,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "x_train.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 58,
   "id": "6ff40f4d-d8b7-45f7-8e3a-8115564540df",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 0: Train loss=0.218540, Validation loss=0.210542 (saved)\n",
      "Epoch 1: Train loss=0.198999, Validation loss=0.176234 (saved)\n",
      "Epoch 2: Train loss=0.145546, Validation loss=0.098695 (saved)\n",
      "Epoch 3: Train loss=0.065873, Validation loss=0.035620 (saved)\n",
      "Epoch 4: Train loss=0.034169, Validation loss=0.031008 (saved)\n",
      "Epoch 5: Train loss=0.028612, Validation loss=0.026082 (saved)\n",
      "Epoch 6: Train loss=0.025862, Validation loss=0.024273 (saved)\n",
      "Epoch 7: Train loss=0.023792, Validation loss=0.022257 (saved)\n",
      "Epoch 8: Train loss=0.022057, Validation loss=0.020762 (saved)\n",
      "Epoch 9: Train loss=0.020527, Validation loss=0.019567 (saved)\n",
      "Epoch 10: Train loss=0.019271, Validation loss=0.018412 (saved)\n",
      "Epoch 11: Train loss=0.018014, Validation loss=0.016844 (saved)\n",
      "Epoch 12: Train loss=0.016790, Validation loss=0.016163 (saved)\n",
      "Epoch 13: Train loss=0.015877, Validation loss=0.015267 (saved)\n",
      "Epoch 14: Train loss=0.015084, Validation loss=0.014561 (saved)\n",
      "Epoch 15: Train loss=0.014348, Validation loss=0.014757 \n",
      "Epoch 16: Train loss=0.014116, Validation loss=0.014028 (saved)\n",
      "Epoch 17: Train loss=0.013394, Validation loss=0.013604 (saved)\n",
      "Epoch 18: Train loss=0.013273, Validation loss=0.012984 (saved)\n",
      "Epoch 19: Train loss=0.012595, Validation loss=0.012553 (saved)\n",
      "Epoch 20: Train loss=0.012528, Validation loss=0.013585 \n",
      "Epoch 21: Train loss=0.012346, Validation loss=0.011391 (saved)\n",
      "Epoch 22: Train loss=0.012295, Validation loss=0.011142 (saved)\n",
      "Epoch 23: Train loss=0.011567, Validation loss=0.011207 \n",
      "Epoch 24: Train loss=0.010992, Validation loss=0.010443 (saved)\n",
      "Epoch 25: Train loss=0.010598, Validation loss=0.010150 (saved)\n",
      "Epoch 26: Train loss=0.010705, Validation loss=0.010643 \n",
      "Epoch 27: Train loss=0.010771, Validation loss=0.010099 (saved)\n",
      "Epoch 28: Train loss=0.010891, Validation loss=0.009943 (saved)\n",
      "Epoch 29: Train loss=0.010099, Validation loss=0.010223 \n",
      "Epoch 30: Train loss=0.009910, Validation loss=0.009233 (saved)\n",
      "Epoch 31: Train loss=0.009588, Validation loss=0.009431 \n",
      "Epoch 32: Train loss=0.009785, Validation loss=0.009039 (saved)\n",
      "Epoch 33: Train loss=0.010262, Validation loss=0.010334 \n",
      "Epoch 34: Train loss=0.009813, Validation loss=0.009075 \n",
      "Epoch 35: Train loss=0.009187, Validation loss=0.008813 (saved)\n",
      "Epoch 36: Train loss=0.008815, Validation loss=0.009591 \n",
      "Epoch 37: Train loss=0.009252, Validation loss=0.008904 \n",
      "Epoch 38: Train loss=0.008922, Validation loss=0.008942 \n",
      "Epoch 39: Train loss=0.009368, Validation loss=0.008213 (saved)\n",
      "Epoch 40: Train loss=0.009578, Validation loss=0.008506 \n",
      "Epoch 41: Train loss=0.008995, Validation loss=0.008073 (saved)\n",
      "Epoch 42: Train loss=0.008366, Validation loss=0.008047 (saved)\n",
      "Epoch 43: Train loss=0.008031, Validation loss=0.007982 (saved)\n",
      "Epoch 44: Train loss=0.007916, Validation loss=0.007519 (saved)\n",
      "Epoch 45: Train loss=0.007976, Validation loss=0.007400 (saved)\n",
      "Epoch 46: Train loss=0.008082, Validation loss=0.007737 \n",
      "Epoch 47: Train loss=0.008126, Validation loss=0.007691 \n",
      "Epoch 48: Train loss=0.007797, Validation loss=0.007969 \n",
      "Epoch 49: Train loss=0.007615, Validation loss=0.008443 \n",
      "Epoch 50: Train loss=0.007602, Validation loss=0.007158 (saved)\n",
      "Epoch 51: Train loss=0.007077, Validation loss=0.006881 (saved)\n",
      "Epoch 52: Train loss=0.006727, Validation loss=0.006515 (saved)\n",
      "Epoch 53: Train loss=0.006528, Validation loss=0.006436 (saved)\n",
      "Epoch 54: Train loss=0.006336, Validation loss=0.006111 (saved)\n",
      "Epoch 55: Train loss=0.006339, Validation loss=0.006736 \n",
      "Epoch 56: Train loss=0.006349, Validation loss=0.006329 \n",
      "Epoch 57: Train loss=0.006178, Validation loss=0.006082 (saved)\n",
      "Epoch 58: Train loss=0.006131, Validation loss=0.006040 (saved)\n",
      "Epoch 59: Train loss=0.006150, Validation loss=0.006110 \n",
      "Epoch 60: Train loss=0.006327, Validation loss=0.006080 \n",
      "Epoch 61: Train loss=0.006284, Validation loss=0.006323 \n",
      "Epoch 62: Train loss=0.006157, Validation loss=0.005981 (saved)\n",
      "Epoch 63: Train loss=0.006069, Validation loss=0.005802 (saved)\n",
      "Epoch 64: Train loss=0.005856, Validation loss=0.005901 \n",
      "Epoch 65: Train loss=0.005768, Validation loss=0.005756 (saved)\n",
      "Epoch 66: Train loss=0.005816, Validation loss=0.005832 \n",
      "Epoch 67: Train loss=0.005975, Validation loss=0.005817 \n",
      "Epoch 68: Train loss=0.005930, Validation loss=0.005862 \n",
      "Epoch 69: Train loss=0.005696, Validation loss=0.006210 \n",
      "Epoch 70: Train loss=0.005879, Validation loss=0.005438 (saved)\n",
      "Epoch 71: Train loss=0.005508, Validation loss=0.005379 (saved)\n",
      "Epoch 72: Train loss=0.005583, Validation loss=0.005585 \n",
      "Epoch 73: Train loss=0.005651, Validation loss=0.005818 \n",
      "Epoch 74: Train loss=0.005641, Validation loss=0.005303 (saved)\n",
      "Epoch 75: Train loss=0.005748, Validation loss=0.005790 \n",
      "Epoch 76: Train loss=0.006070, Validation loss=0.005414 \n",
      "Epoch 77: Train loss=0.006290, Validation loss=0.006127 \n",
      "Epoch 78: Train loss=0.006120, Validation loss=0.005616 \n",
      "Epoch 79: Train loss=0.005894, Validation loss=0.005528 \n",
      "Epoch 80: Train loss=0.005919, Validation loss=0.005620 \n",
      "Epoch 81: Train loss=0.005703, Validation loss=0.005542 \n",
      "Epoch 82: Train loss=0.005637, Validation loss=0.005392 \n",
      "Epoch 83: Train loss=0.005664, Validation loss=0.005490 \n",
      "Epoch 84: Train loss=0.005394, Validation loss=0.005149 (saved)\n",
      "Epoch 85: Train loss=0.005628, Validation loss=0.005252 \n",
      "Epoch 86: Train loss=0.005428, Validation loss=0.005711 \n",
      "Epoch 87: Train loss=0.005437, Validation loss=0.005027 (saved)\n",
      "Epoch 88: Train loss=0.005480, Validation loss=0.005545 \n",
      "Epoch 89: Train loss=0.005445, Validation loss=0.005040 \n",
      "Epoch 90: Train loss=0.005420, Validation loss=0.005618 \n",
      "Epoch 91: Train loss=0.005262, Validation loss=0.005233 \n",
      "Epoch 92: Train loss=0.004997, Validation loss=0.004741 (saved)\n",
      "Epoch 93: Train loss=0.005000, Validation loss=0.004667 (saved)\n",
      "Epoch 94: Train loss=0.005063, Validation loss=0.004999 \n",
      "Epoch 95: Train loss=0.005053, Validation loss=0.004708 \n",
      "Epoch 96: Train loss=0.005134, Validation loss=0.004787 \n",
      "Epoch 97: Train loss=0.005321, Validation loss=0.006402 \n",
      "Epoch 98: Train loss=0.006179, Validation loss=0.007004 \n",
      "Epoch 99: Train loss=0.006058, Validation loss=0.007863 \n",
      "Epoch 100: Train loss=0.006570, Validation loss=0.005046 \n",
      "Epoch 101: Train loss=0.005726, Validation loss=0.005740 \n",
      "Epoch 102: Train loss=0.005270, Validation loss=0.005231 \n",
      "Epoch 103: Train loss=0.004955, Validation loss=0.004721 \n",
      "Epoch 104: Train loss=0.004699, Validation loss=0.004608 (saved)\n",
      "Epoch 105: Train loss=0.004556, Validation loss=0.004414 (saved)\n",
      "Epoch 106: Train loss=0.004426, Validation loss=0.004342 (saved)\n",
      "Epoch 107: Train loss=0.004359, Validation loss=0.004378 \n",
      "Epoch 108: Train loss=0.004382, Validation loss=0.004271 (saved)\n",
      "Epoch 109: Train loss=0.004331, Validation loss=0.004281 \n",
      "Epoch 110: Train loss=0.004282, Validation loss=0.004229 (saved)\n",
      "Epoch 111: Train loss=0.004265, Validation loss=0.004228 (saved)\n",
      "Epoch 112: Train loss=0.004259, Validation loss=0.004159 (saved)\n",
      "Epoch 113: Train loss=0.004216, Validation loss=0.004169 \n",
      "Epoch 114: Train loss=0.004204, Validation loss=0.004112 (saved)\n",
      "Epoch 115: Train loss=0.004163, Validation loss=0.004087 (saved)\n",
      "Epoch 116: Train loss=0.004147, Validation loss=0.004102 \n",
      "Epoch 117: Train loss=0.004122, Validation loss=0.004066 (saved)\n",
      "Epoch 118: Train loss=0.004140, Validation loss=0.004064 (saved)\n",
      "Epoch 119: Train loss=0.004081, Validation loss=0.004037 (saved)\n",
      "Epoch 120: Train loss=0.004080, Validation loss=0.004038 \n",
      "Epoch 121: Train loss=0.004108, Validation loss=0.004126 \n",
      "Epoch 122: Train loss=0.004074, Validation loss=0.003980 (saved)\n",
      "Epoch 123: Train loss=0.004066, Validation loss=0.003966 (saved)\n",
      "Epoch 124: Train loss=0.004097, Validation loss=0.003944 (saved)\n",
      "Epoch 125: Train loss=0.004042, Validation loss=0.004142 \n",
      "Epoch 126: Train loss=0.004125, Validation loss=0.003949 \n",
      "Epoch 127: Train loss=0.004049, Validation loss=0.003977 \n",
      "Epoch 128: Train loss=0.004084, Validation loss=0.004227 \n",
      "Epoch 129: Train loss=0.004207, Validation loss=0.004095 \n",
      "Epoch 130: Train loss=0.004278, Validation loss=0.004043 \n",
      "Epoch 131: Train loss=0.004762, Validation loss=0.005170 \n",
      "Epoch 132: Train loss=0.004995, Validation loss=0.004484 \n",
      "Epoch 133: Train loss=0.004497, Validation loss=0.004628 \n",
      "Epoch 134: Train loss=0.004392, Validation loss=0.004065 \n",
      "Epoch 135: Train loss=0.004227, Validation loss=0.004029 \n",
      "Epoch 136: Train loss=0.004175, Validation loss=0.004312 \n",
      "Epoch 137: Train loss=0.004200, Validation loss=0.004370 \n",
      "Epoch 138: Train loss=0.004245, Validation loss=0.004076 \n",
      "Epoch 139: Train loss=0.004103, Validation loss=0.004028 \n",
      "Epoch 140: Train loss=0.004102, Validation loss=0.004030 \n",
      "Epoch 141: Train loss=0.004167, Validation loss=0.003990 \n",
      "Epoch 142: Train loss=0.004144, Validation loss=0.004023 \n",
      "Epoch 143: Train loss=0.004050, Validation loss=0.003987 \n",
      "Epoch 144: Train loss=0.003932, Validation loss=0.003799 (saved)\n",
      "Epoch 145: Train loss=0.003855, Validation loss=0.003902 \n",
      "Epoch 146: Train loss=0.003903, Validation loss=0.003750 (saved)\n",
      "Epoch 147: Train loss=0.003836, Validation loss=0.004009 \n",
      "Epoch 148: Train loss=0.003917, Validation loss=0.003839 \n",
      "Epoch 149: Train loss=0.003906, Validation loss=0.003921 \n",
      "Epoch 150: Train loss=0.003806, Validation loss=0.003735 (saved)\n",
      "Epoch 151: Train loss=0.003721, Validation loss=0.003630 (saved)\n",
      "Epoch 152: Train loss=0.003683, Validation loss=0.003666 \n",
      "Epoch 153: Train loss=0.003674, Validation loss=0.003610 (saved)\n",
      "Epoch 154: Train loss=0.003646, Validation loss=0.003591 (saved)\n",
      "Epoch 155: Train loss=0.003632, Validation loss=0.003584 (saved)\n",
      "Epoch 156: Train loss=0.003621, Validation loss=0.003578 (saved)\n",
      "Epoch 157: Train loss=0.003621, Validation loss=0.003563 (saved)\n",
      "Epoch 158: Train loss=0.003609, Validation loss=0.003563 \n",
      "Epoch 159: Train loss=0.003602, Validation loss=0.003550 (saved)\n",
      "Epoch 160: Train loss=0.003592, Validation loss=0.003554 \n",
      "Epoch 161: Train loss=0.003601, Validation loss=0.003573 \n",
      "Epoch 162: Train loss=0.003609, Validation loss=0.003559 \n",
      "Epoch 163: Train loss=0.003590, Validation loss=0.003547 (saved)\n",
      "Epoch 164: Train loss=0.003585, Validation loss=0.003525 (saved)\n",
      "Epoch 165: Train loss=0.003562, Validation loss=0.003515 (saved)\n",
      "Epoch 166: Train loss=0.003548, Validation loss=0.003509 (saved)\n",
      "Epoch 167: Train loss=0.003544, Validation loss=0.003512 \n",
      "Epoch 168: Train loss=0.003544, Validation loss=0.003529 \n",
      "Epoch 169: Train loss=0.003542, Validation loss=0.003488 (saved)\n",
      "Epoch 170: Train loss=0.003530, Validation loss=0.003477 (saved)\n",
      "Epoch 171: Train loss=0.003524, Validation loss=0.003520 \n",
      "Epoch 172: Train loss=0.003530, Validation loss=0.003495 \n",
      "Epoch 173: Train loss=0.003537, Validation loss=0.003472 (saved)\n",
      "Epoch 174: Train loss=0.003554, Validation loss=0.003579 \n",
      "Epoch 175: Train loss=0.003582, Validation loss=0.003586 \n",
      "Epoch 176: Train loss=0.003530, Validation loss=0.003519 \n",
      "Epoch 177: Train loss=0.003534, Validation loss=0.003511 \n",
      "Epoch 178: Train loss=0.003520, Validation loss=0.003567 \n",
      "Epoch 179: Train loss=0.003530, Validation loss=0.003466 (saved)\n",
      "Epoch 180: Train loss=0.003494, Validation loss=0.003466 (saved)\n",
      "Epoch 181: Train loss=0.003477, Validation loss=0.003498 \n",
      "Epoch 182: Train loss=0.003483, Validation loss=0.003512 \n",
      "Epoch 183: Train loss=0.003508, Validation loss=0.003531 \n",
      "Epoch 184: Train loss=0.003533, Validation loss=0.003430 (saved)\n",
      "Epoch 185: Train loss=0.003466, Validation loss=0.003491 \n",
      "Epoch 186: Train loss=0.003472, Validation loss=0.003409 (saved)\n",
      "Epoch 187: Train loss=0.003517, Validation loss=0.003462 \n",
      "Epoch 188: Train loss=0.003556, Validation loss=0.003444 \n",
      "Epoch 189: Train loss=0.003472, Validation loss=0.003424 \n",
      "Epoch 190: Train loss=0.003480, Validation loss=0.003435 \n",
      "Epoch 191: Train loss=0.003502, Validation loss=0.003388 (saved)\n",
      "Epoch 192: Train loss=0.003441, Validation loss=0.003370 (saved)\n",
      "Epoch 193: Train loss=0.003474, Validation loss=0.003455 \n",
      "Epoch 194: Train loss=0.003428, Validation loss=0.003380 \n",
      "Epoch 195: Train loss=0.003433, Validation loss=0.003381 \n",
      "Epoch 196: Train loss=0.003547, Validation loss=0.003556 \n",
      "Epoch 197: Train loss=0.003595, Validation loss=0.003565 \n",
      "Epoch 198: Train loss=0.003534, Validation loss=0.003454 \n",
      "Epoch 199: Train loss=0.003462, Validation loss=0.003482 \n",
      "Finished training with best train loss: 0.003398 and validation loss: 0.003370\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": 59,
   "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": 60,
   "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_hardc_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": 61,
   "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.0010838542715646327\n",
      "n-MeRCI: 0.37027138471603394\n",
      "RMSCE: 0.20611515641212463\n",
      "Cerr: 3.945082482914586e-07\n",
      "In-domain results\n",
      "MSE: 0.001069478578865528\n",
      "n-MeRCI: 0.16918982565402985\n",
      "RMSCE: 0.21512822806835175\n",
      "ProbConserv Results\n",
      "MSE: 0.001069478616118431\n",
      "n-MeRCI: 0.19851702451705933\n",
      "RMSCE: 0.2152755707502365\n",
      "Cerr: 4.073977493135317e-07\n",
      "Prob_Cerr: 4.085898410721711e-07\n",
      "Here\n",
      "\n",
      "\n",
      "Out-of-domain results\n",
      "MSE: 0.0010378452204167843\n",
      "n-MeRCI: 0.36269593238830566\n",
      "RMSCE: 0.22642281651496887\n",
      "ProbConserv Results\n",
      "MSE: 0.0010378451645374298\n",
      "n-MeRCI: 0.4000171720981598\n",
      "RMSCE: 0.22661568224430084\n",
      "Cerr: 3.722309998011042e-07\n",
      "Prob_Cerr: 4.041195040827006e-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": 62,
   "id": "98acb797",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.001069478616118431"
      ]
     },
     "execution_count": 62,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "id_results['pc.mse']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 183,
   "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_hardc_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_hardc_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": 184,
   "id": "aa23f617-8a23-4bbc-83e1-e3ed3cdf5498",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAksAAAHHCAYAAACvJxw8AAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/bCgiHAAAACXBIWXMAAA9hAAAPYQGoP6dpAABlhklEQVR4nO3dd3hUVf4/8Pednl5Ih0goSocoCBtBAQmEIisqK1bKKtgQlUUBXSnCEnT5sSAiWMECgrjqWhAElK8iKIogHSmhk4SWTOrU8/tjMpNMpmQymZKB9+t55oE5c+6dc++0Tz7n3HMkIYQAERERETklC3YDiIiIiBozBktEREREbjBYIiIiInKDwRIRERGRGwyWiIiIiNxgsERERETkBoMlIiIiIjcYLBERERG5wWCJiIiIyA0GS0SNQEZGBkaPHm27v3nzZkiShM2bNwetTbXVbiO59/jjj6N///7Bbgb5wcWLFxEREYG1a9cGuykUIAyW6Kq3fPlySJJku2k0Glx33XUYP348CgoKgt28elm7di1mzJgR7Gb4RZ8+fexeJ1e3xnD8eXl5ePvtt/H8888HuylXhG+//RYPPfQQOnbsCLlcjoyMjHptn5GR4fS98uijjzrULSoqwrhx45CYmIiIiAj07dsXv//+u12dJk2a4OGHH8aLL77YkMOiEKIIdgOIGouXXnoJLVq0QGVlJbZs2YIlS5Zg7dq12Lt3L8LDwwPalltuuQUVFRVQqVT12m7t2rVYvHhxowgYfO2FF17Aww8/bLv/66+/4tVXX8Xzzz+Pdu3a2co7d+4cjObZWbhwIVq0aIG+ffsGuylXhJUrV2L16tW44YYbkJaW5tU+MjMz8Y9//MOu7LrrrrO7bzabMWTIEPzxxx949tlnkZCQgNdffx19+vTBjh07cO2119rqPvroo3j11Vfx3Xff4dZbb/WqTRRCBNFVbtmyZQKA+PXXX+3KJ06cKACIlStXuty2tLTUJ21o3ry5GDVqVIP388QTTwh/fax91UZfWbNmjQAgvv/+e7f1fPUaeUqv14uEhATxz3/+M6DPaxXo4w2EM2fOCL1eL4QQYsiQIaJ58+b12r558+ZiyJAhddZbvXq1ACDWrFljKyssLBSxsbHi3nvvdajfsWNH8eCDD9arLRSa2A1H5IL1r8W8vDwAwOjRoxEZGYmjR49i8ODBiIqKwv333w/A8hfpggUL0KFDB2g0GiQnJ+ORRx7B5cuX7fYphMDs2bPRrFkzhIeHo2/fvti3b5/Dc7sas/TLL79g8ODBiIuLQ0REBDp37oyFCxfa2rd48WIAsOtqsPJ1G2szGAyIj4/HmDFjHB7TarXQaDSYNGmSrWzRokXo0KEDwsPDERcXh27dumHlypV1Po87M2bMgCRJ2L9/P+677z7ExcWhV69eACzdeH369HHYZvTo0Q7dOp6eK2e2bNmCCxcuIDs7267c+pquXr0azz//PFJSUhAREYG//vWvOHXqlF3dH3/8EX/7299wzTXXQK1WIz09Hc888wwqKioc2u7qPVnffZw8eRK33XYbIiMj0bRpU9t7ac+ePbj11lsRERGB5s2bN/g18kZaWhqUSmWD96PX61FWVuby8U8++QTJycm48847bWWJiYm4++678b///Q86nc6ufv/+/fHll19CCNHgtlHjxm44IheOHj0KwDI+wcpoNCInJwe9evXCvHnzbN1zjzzyCJYvX44xY8ZgwoQJyMvLw2uvvYadO3fip59+sn3RT5s2DbNnz8bgwYMxePBg/P777xgwYAD0en2d7dmwYQNuu+02pKam4qmnnkJKSgoOHDiAr776Ck899RQeeeQRnD17Fhs2bMAHH3zgsL2/26hUKnHHHXfg008/xRtvvGHXhfj5559Dp9PhnnvuAQC89dZbmDBhAoYPH46nnnoKlZWV2L17N3755Rfcd999dZ6Luvztb3/Dtddeizlz5nj1Q+bpuXJm69atkCQJ119/vdPH//Wvf0GSJEyePBmFhYVYsGABsrOzsWvXLoSFhQEA1qxZg/Lycjz22GNo0qQJtm/fjkWLFuH06dNYs2aN3f5cvSfrsw+TyYRBgwbhlltuwSuvvIIVK1Zg/PjxiIiIwAsvvID7778fd955J5YuXYqRI0ciKysLLVq0cHsOL1++DJPJVOe5Dg8PD0g393fffYfw8HCYTCY0b94czzzzDJ566im7Ojt37sQNN9wAmcw+j9C9e3e8+eab+PPPP9GpUydbedeuXfGf//wH+/btQ8eOHf1+DBREwU1sEQWftRtu48aN4vz58+LUqVNi1apVokmTJiIsLEycPn1aCCHEqFGjBAAxZcoUu+1//PFHAUCsWLHCrnzdunV25YWFhUKlUokhQ4YIs9lsq/f8888LAHZd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Uq63jDExmMxBbgld7YYOHYoJEyagU6dOuHTpEmbPno1PP/0UGzdudFp/9erVSE1NdXvbunWry+crKyvD448/jg4dOqBnz57473//iw8//BAPP/ywrc7bb7+NG2+8EbfddhuysrIAAF9//TUUCv/mfiRRu/OP6k2r1SImJgbFxcWIjo4OdnOIKIQVaitRoNVBkoAOadFcgDXEVVZWIi8vDy1atIBGU/cYo8YiIyMDTz/9NJ5++ulgN6XB3L0Gnv5+M7NERNSIWK+EEwIwmPi3LFFjwGCJiKgRqTltgNHMrjiixoADvImIGgmd0QSTuTqbZDAKwPlYWCK/On78eLCb0Kgws0RE1EhU1JqMUs9B3kSNAoMlIqJGovbM3bwijqhxCKlg6YcffsDQoUORlpYGSZLw+eefu63/6aefon///khMTER0dDSysrKwfv16uzozZsxwWLm4bdu2fjwKIiLnGCwRNU4hFSyVlZWhS5cuWLx4sUf1f/jhB/Tv3x9r167Fjh070LdvXwwdOhQ7d+60q9ehQwecO3fOdtuyZYs/mk9E5JIQApUGBktEjVFIDfAeNGgQBg0a5HH9BQsW2N2fM2cO/ve//+HLL7/E9ddfbytXKBRISUnxVTOJiOqt0mBG7Vnv9EZOHUDUGIRUZqmhzGYzSkpKEB8fb1d++PBhpKWloWXLlrj//vtx8uRJt/vR6XTQarV2NyKihrDO3F2Tyey4aCgRBd5VFSzNmzcPpaWluPvuu21lPXr0wPLly7Fu3TosWbIEeXl5uPnmm1FSUuJyP7m5uYiJibHd0tPTA9F8IrqC1R6vZMUr4oiC76oJllauXImZM2fi448/RlJSkq180KBB+Nvf/obOnTsjJycHa9euRVFRET7++GOX+5o6dSqKi4ttt2CsxExEV5YKg/NgibN4U7D06dPniljuxBdCasySt1atWoWHH34Ya9asQXZ2ttu6sbGxuO6663DkyBGXddRqNdRqta+bSURXKZNZQGdwnkEyGM0Av26oERJCwGQy+X0R28bgis8sffTRRxgzZgw++ugjDBkypM76paWlOHr0KFJTUwPQOiIi5+OVrAxc8oSCYPTo0fi///s/LFy40DatzvLlyyFJEr755ht07doVarUaW7ZswejRozFs2DC77Z9++mn06dPHdt9sNiM3NxctWrRAWFgYunTpgk8++SSw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V4BbHnrqe65XrFgh1Gq1WLFihcjLyxPr168Xqamp4plnnglwy0PL2rVrxQsvvCA+/fRTAUB89tlnbusfO3ZMhIeHi4kTJ4r9+/eLRYsWCblcLtatW+e3NjJYCqLu3buLJ554wnbfZDKJtLQ0kZub67T+3XffLYYMGWJX1qNHD/HII4/4tZ2hrr7nuTaj0SiioqLEe++9568mXhG8Oc9Go1HcdNNN4u233xajRo1isOSB+p7nJUuWiJYtWwq9Xh+oJl4x6nuun3jiCXHrrbfalU2cOFH07NnTr+28kngSLD333HOiQ4cOdmUjRowQOTk5fmsXu+GCRK/XY8eOHcjOzraVyWQyZGdnY9u2bU632bZtm119AMjJyXFZn7w7z7WVl5fDYDAgPj7eX80Med6e55deeglJSUl46KGHAtHMkOfNef7iiy+QlZWFJ554AsnJyejYsSPmzJkDk8kUqGaHJG/O9U033YQdO3bYuuqOHTuGtWvXYvDgwQFp89UiGL+FXEg3SC5cuACTyYTk5GS78uTkZBw8eNDpNvn5+U7r5+fn+62doc6b81zb5MmTkZaW5vDhpGrenOctW7bgnXfewa5duwLQwiuDN+f52LFj+O6773D//fdj7dq1OHLkCB5//HEYDAZMnz49EM0OSd6c6/vuuw8XLlxAr169IISA0WjEo48+iueffz4QTb5quPot1Gq1qKioQFhYmM+fk5klIjfmzp2LVatW4bPPPoNGowl2c64YJSUlePDBB/HWW28hISEh2M25opnNZiQlJeHNN99E165dMWLECLzwwgtYunRpsJt2xdm8eTPmzJmD119/Hb///js+/fRTfP3115g1a1awm0YNxMxSkCQkJEAul6OgoMCuvKCgACkpKU63SUlJqVd98u48W82bNw9z587Fxo0b0blzZ382M+TV9zwfPXoUx48fx9ChQ21lZrMZAKBQKHDo0CG0atXKv40OQd68n1NTU6FUKiGXy21l7dq1Q35+PvR6PVQqlV/bHKq8OdcvvvgiHnzwQTz88MMAgE6dOqGsrAzjxo3DCy+8AJmM+QlfcPVbGB0d7ZesEsDMUtCoVCp07doVmzZtspWZzWZs2rQJWVlZTrfJysqyqw8AGzZscFmfvDvPAPDKK69g1qxZWLduHbp16xaIpoa0+p7ntm3bYs+ePdi1a5ft9te//hV9+/bFrl27kJ6eHsjmhwxv3s89e/bEkSNHbMEoAPz5559ITU1loOSGN+e6vLzcISCyBqmCy7D6TFB+C/02dJzqtGrVKqFWq8Xy5cvF/v37xbhx40RsbKzIz88XQgjx4IMPiilTptjq//TTT0KhUIh58+aJAwcOiOnTp3PqAA/U9zzPnTtXqFQq8cknn4hz587ZbiUlJcE6hJBQ3/NcG6+G80x9z/PJkydFVFSUGD9+vDh06JD46quvRFJSkpg9e3awDiFk1PdcT58+XURFRYmPPvpIHDt2THz77beiVatW4u677w7WIYSEkpISsXPnTrFz504BQMyfP1/s3LlTnDhxQgghxJQpU8SDDz5oq2+dOuDZZ58VBw4cEIsXL+bUAVe6RYsWiWuuuUaoVCrRvXt38fPPP9se6927txg1apRd/Y8//lhcd911QqVSiQ4dOoivv/46wC0OTfU5z82bNxcAHG7Tp08PfMNDTH3fzzUxWPJcfc/z1q1bRY8ePYRarRYtW7YU//rXv4TRaAxwq0NTfc61wWAQM2bMEK1atRIajUakp6eLxx9/XFy+fDnwDQ8h33//vdPvXOu5HTVqlOjdu7fDNpmZmUKlUomWLVuKZcuW+bWNkhDMDRIRERG5wjFLRERERG4wWCIiIiJyg8ESERERkRsMloiIiIjcYLBERERE5AaDJSIiIiI3GCwRERERucFgiYiIiMgNBktEREREbjBYIiIiInKDwRIRUS3nz59HSkoK5syZYyvbunUrVCqVw2rnRHTl49pwREROrF27FsOGDcPWrVvRpk0bZGZm4vbbb8f8+fOD3TQiCjAGS0RELjzxxBPYuHEjunXrhj179uDXX3+FWq0OdrOIKMAYLBERuVBRUYGOHTvi1KlT2LFjBzp16hTsJhFREHDMEhGRC0ePHsXZs2dhNptx/PjxYDeHiIKEmSUiIif0ej26d++OzMxMtGnTBgsWLMCePXuQlJQU7KYRUYAxWCIicuLZZ5/FJ598gj/++AORkZHo3bs3YmJi8NVXXwW7aUQUYOyGIyKqZfPmzViwYAE++OADREdHQyaT4YMPPsCPP/6IJUuWBLt5RBRgzCwRERERucHMEhEREZEbDJaIiIiI3GCwREREROQGgyUiIiIiNxgsEREREbnBYImIiIjIDQZLRERERG4wWCIiIiJyg8ESERERkRsMloiIiIjcYLBERERE5AaDJSIiIiI3/j8F0iuWXtbh9QAAAABJRU5ErkJggg==",
      "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": 177,
   "id": "3a5ebe0f-9094-408e-8162-db12f3b800dc",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.0010378451645374298"
      ]
     },
     "execution_count": 177,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "test_stats['mse']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 178,
   "id": "7b0e9a1a-673d-4601-8ecd-0b6be435b39c",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "4.282593693005765e-07"
      ]
     },
     "execution_count": 178,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "test_stats['mcerr']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 179,
   "id": "42b92e3f-5318-4bac-b650-3dff39acc228",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.0032365545630455017"
      ]
     },
     "execution_count": 179,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "test_stats['crps']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 180,
   "id": "3505923a-8f51-44c6-9053-31951d11bf9a",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "'Unconstrained & 1.08E-03 & 3.70E-01 & 2.06E-01 & 4.37E-07 & 3.40E-03 & 1.04E-03 & 3.63E-01 & 2.26E-01 & 4.28E-07 & 3.24E-03 \\\\\\\\'"
      ]
     },
     "execution_count": 180,
     "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
}
