{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 469,
   "id": "6b3ccdb8-4fd8-4224-b23e-979fa8228059",
   "metadata": {},
   "outputs": [],
   "source": [
    "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": 470,
   "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": 471,
   "id": "1d5eb530-ab43-4054-8953-9efdb86b3106",
   "metadata": {},
   "outputs": [],
   "source": [
    "args = {'--batch_size': '20',\n",
    " '--dataset': 'HeatEquation_1D',\n",
    " '--dataset_params': '1,5,0,0',\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,5,0,0'}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 472,
   "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": 473,
   "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": 474,
   "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": 447,
   "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": 475,
   "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 = OutputVarNO(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": 476,
   "id": "4333d6f8-fe2c-4caa-a889-efae589c31ff",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([160, 100, 20, 1])"
      ]
     },
     "execution_count": 476,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "y_train.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 477,
   "id": "96aa9ee0-a30e-458b-85e2-5e7dbbd89dc4",
   "metadata": {},
   "outputs": [],
   "source": [
    "mu_true = torch.mean(y_train, dim = 0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 478,
   "id": "fc1ac32d-6fb2-4293-8ea7-f90486a32dce",
   "metadata": {},
   "outputs": [],
   "source": [
    "var_true = torch.var(y_train, dim = 0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 479,
   "id": "f07c25ce-49e4-4331-81a3-ee678808a6b4",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([100, 20, 1])"
      ]
     },
     "execution_count": 479,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "mu_true.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 480,
   "id": "0f4d18b3-1988-4417-9122-3a8709048fc2",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor([0.0003, 0.0053, 0.0104, 0.0154, 0.0204, 0.0252, 0.0300, 0.0347, 0.0392,\n",
      "        0.0436, 0.0478, 0.0518, 0.0557, 0.0593, 0.0626, 0.0657, 0.0686, 0.0712,\n",
      "        0.0735, 0.0754, 0.0771, 0.0785, 0.0796, 0.0803, 0.0807, 0.0808, 0.0806,\n",
      "        0.0800, 0.0792, 0.0780, 0.0764, 0.0746, 0.0725, 0.0701, 0.0674, 0.0644,\n",
      "        0.0612, 0.0577, 0.0540, 0.0500, 0.0459, 0.0416, 0.0371, 0.0324, 0.0277,\n",
      "        0.0228, 0.0178, 0.0128, 0.0077, 0.0026, 0.0026, 0.0077, 0.0128, 0.0178,\n",
      "        0.0228, 0.0277, 0.0324, 0.0371, 0.0416, 0.0459, 0.0500, 0.0540, 0.0577,\n",
      "        0.0612, 0.0644, 0.0674, 0.0701, 0.0725, 0.0746, 0.0764, 0.0780, 0.0792,\n",
      "        0.0800, 0.0806, 0.0808, 0.0807, 0.0803, 0.0796, 0.0785, 0.0771, 0.0754,\n",
      "        0.0735, 0.0712, 0.0686, 0.0657, 0.0626, 0.0593, 0.0557, 0.0518, 0.0478,\n",
      "        0.0436, 0.0392, 0.0347, 0.0300, 0.0252, 0.0204, 0.0154, 0.0104, 0.0053,\n",
      "        0.0003])\n"
     ]
    },
    {
     "data": {
      "image/png": "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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": 481,
   "id": "6fd587e6-4cf8-4cc4-92bd-ac09c7be790a",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<function datasets.HeatEquation_1D.get_mass_rhs_func.<locals>.mass_rhs_func(inputs)>"
      ]
     },
     "execution_count": 481,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "dataset_class.get_mass_rhs_func(x=x_train)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 482,
   "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": 483,
   "id": "b339b6e5",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "device(type='cpu')"
      ]
     },
     "execution_count": 483,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "x_train.device"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 484,
   "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": 485,
   "id": "d71ec298-4bbc-4186-a938-8675f0526890",
   "metadata": {},
   "outputs": [],
   "source": [
    "import time"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 486,
   "id": "87e15df0-69a3-4d0e-82f0-f04540d1a1e8",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([0.0000, 0.0635, 0.1269, 0.1904, 0.2539, 0.3173, 0.3808, 0.4443, 0.5077,\n",
       "        0.5712, 0.6347, 0.6981, 0.7616, 0.8251, 0.8885, 0.9520, 1.0155, 1.0789,\n",
       "        1.1424, 1.2059, 1.2693, 1.3328, 1.3963, 1.4597, 1.5232, 1.5867, 1.6501,\n",
       "        1.7136, 1.7771, 1.8405, 1.9040, 1.9675, 2.0309, 2.0944, 2.1579, 2.2213,\n",
       "        2.2848, 2.3483, 2.4117, 2.4752, 2.5387, 2.6021, 2.6656, 2.7291, 2.7925,\n",
       "        2.8560, 2.9195, 2.9829, 3.0464, 3.1099, 3.1733, 3.2368, 3.3003, 3.3637,\n",
       "        3.4272, 3.4907, 3.5541, 3.6176, 3.6811, 3.7445, 3.8080, 3.8715, 3.9349,\n",
       "        3.9984, 4.0619, 4.1253, 4.1888, 4.2523, 4.3157, 4.3792, 4.4427, 4.5061,\n",
       "        4.5696, 4.6331, 4.6965, 4.7600, 4.8235, 4.8869, 4.9504, 5.0139, 5.0773,\n",
       "        5.1408, 5.2043, 5.2677, 5.3312, 5.3947, 5.4581, 5.5216, 5.5851, 5.6485,\n",
       "        5.7120, 5.7755, 5.8389, 5.9024, 5.9659, 6.0293, 6.0928, 6.1563, 6.2197,\n",
       "        6.2832])"
      ]
     },
     "execution_count": 486,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "grid"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 487,
   "id": "2165975d",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[0, -1, 5]"
      ]
     },
     "execution_count": 487,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "tpred"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 488,
   "id": "1ad2c532",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([ 0, -1,  5])"
      ]
     },
     "execution_count": 488,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "torch.tensor(tpred)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 489,
   "id": "18a1c208-a4df-4f27-8dad-f2466164e855",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([160, 100, 20, 1])"
      ]
     },
     "execution_count": 489,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "x_train.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 490,
   "id": "6ff40f4d-d8b7-45f7-8e3a-8115564540df",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 0: Train loss=-470.526483, Validation loss=-651.526318 (saved)\n",
      "Epoch 1: Train loss=-951.780402, Validation loss=-1725.686719 (saved)\n",
      "Epoch 2: Train loss=-3315.339355, Validation loss=2256.940283 \n",
      "Epoch 3: Train loss=-2522.587012, Validation loss=-3889.209180 (saved)\n",
      "Epoch 4: Train loss=-3917.589990, Validation loss=-3770.378906 \n",
      "Epoch 5: Train loss=-3590.666699, Validation loss=-3890.203125 (saved)\n",
      "Epoch 6: Train loss=-4024.755615, Validation loss=-4494.108984 (saved)\n",
      "Epoch 7: Train loss=-4765.260400, Validation loss=-5036.612305 (saved)\n",
      "Epoch 8: Train loss=-5723.063672, Validation loss=-6347.496094 (saved)\n",
      "Epoch 9: Train loss=-6960.100684, Validation loss=-7822.101172 (saved)\n",
      "Epoch 10: Train loss=-8334.166797, Validation loss=-8090.631641 (saved)\n",
      "Epoch 11: Train loss=-10206.614648, Validation loss=-10371.355469 (saved)\n",
      "Epoch 12: Train loss=-1917.314160, Validation loss=-4389.698242 \n",
      "Epoch 13: Train loss=-4789.339014, Validation loss=-4549.107813 \n",
      "Epoch 14: Train loss=-4368.115723, Validation loss=-4287.366797 \n",
      "Epoch 15: Train loss=-4291.552393, Validation loss=-4409.315820 \n",
      "Epoch 16: Train loss=-4520.083740, Validation loss=-4748.656445 \n",
      "Epoch 17: Train loss=-4928.371143, Validation loss=-5227.011523 \n",
      "Epoch 18: Train loss=-5460.529590, Validation loss=-5816.944922 \n",
      "Epoch 19: Train loss=-6106.371973, Validation loss=-6530.490625 \n",
      "Epoch 20: Train loss=-6889.509570, Validation loss=-7419.819531 \n",
      "Epoch 21: Train loss=-7858.710254, Validation loss=-8504.382812 \n",
      "Epoch 22: Train loss=-9076.753613, Validation loss=-9890.676563 \n",
      "Epoch 23: Train loss=-10649.043164, Validation loss=-11386.477344 (saved)\n",
      "Epoch 24: Train loss=-4618.442090, Validation loss=-2009.161230 \n",
      "Epoch 25: Train loss=-6867.145703, Validation loss=-6662.587109 \n",
      "Epoch 26: Train loss=-6692.485107, Validation loss=-6472.869531 \n",
      "Epoch 27: Train loss=-6355.043555, Validation loss=-6303.983203 \n",
      "Epoch 28: Train loss=-6436.634424, Validation loss=-6639.567969 \n",
      "Epoch 29: Train loss=-6864.777344, Validation loss=-7134.303906 \n",
      "Epoch 30: Train loss=-7396.293652, Validation loss=-7724.464844 \n",
      "Epoch 31: Train loss=-8051.819531, Validation loss=-8453.250000 \n",
      "Epoch 32: Train loss=-8836.892773, Validation loss=-9320.891016 \n",
      "Epoch 33: Train loss=-9782.630957, Validation loss=-10376.996484 \n",
      "Epoch 34: Train loss=-10946.498047, Validation loss=-11685.454687 (saved)\n",
      "Epoch 35: Train loss=-12405.060742, Validation loss=-13326.611719 (saved)\n",
      "Epoch 36: Train loss=-254.210449, Validation loss=231809.650000 \n",
      "Epoch 37: Train loss=26897.624121, Validation loss=-4328.124023 \n",
      "Epoch 38: Train loss=-3847.364307, Validation loss=-3680.188867 \n",
      "Epoch 39: Train loss=-3637.928027, Validation loss=-3706.614453 \n",
      "Epoch 40: Train loss=-3729.148340, Validation loss=-3815.284180 \n",
      "Epoch 41: Train loss=-3851.651709, Validation loss=-3929.205078 \n",
      "Epoch 42: Train loss=-3968.832373, Validation loss=-4035.752539 \n",
      "Epoch 43: Train loss=-4077.675732, Validation loss=-4137.238867 \n",
      "Epoch 44: Train loss=-4180.836523, Validation loss=-4236.866406 \n",
      "Epoch 45: Train loss=-4281.112305, Validation loss=-4335.842969 \n",
      "Epoch 46: Train loss=-4379.518652, Validation loss=-4433.910547 \n",
      "Epoch 47: Train loss=-4476.345898, Validation loss=-4530.869531 \n",
      "Epoch 48: Train loss=-4571.965625, Validation loss=-4627.025391 \n",
      "Epoch 49: Train loss=-4666.893604, Validation loss=-4723.044336 \n",
      "Epoch 50: Train loss=-4741.298047, Validation loss=-4771.135352 \n",
      "Epoch 51: Train loss=-4789.134180, Validation loss=-4819.637109 \n",
      "Epoch 52: Train loss=-4837.408691, Validation loss=-4868.727148 \n",
      "Epoch 53: Train loss=-4886.246875, Validation loss=-4918.319727 \n",
      "Epoch 54: Train loss=-4935.604980, Validation loss=-4968.669141 \n",
      "Epoch 55: Train loss=-4985.706348, Validation loss=-5019.521680 \n",
      "Epoch 56: Train loss=-5036.362207, Validation loss=-5071.230469 \n",
      "Epoch 57: Train loss=-5087.801855, Validation loss=-5123.522266 \n",
      "Epoch 58: Train loss=-5139.847705, Validation loss=-5176.688672 \n",
      "Epoch 59: Train loss=-5192.778857, Validation loss=-5230.698438 \n",
      "Epoch 60: Train loss=-5246.551367, Validation loss=-5285.450000 \n",
      "Epoch 61: Train loss=-5301.262207, Validation loss=-5341.010938 \n",
      "Epoch 62: Train loss=-5356.597656, Validation loss=-5397.881641 \n",
      "Epoch 63: Train loss=-5413.125928, Validation loss=-5455.605469 \n",
      "Epoch 64: Train loss=-5470.791211, Validation loss=-5514.426172 \n",
      "Epoch 65: Train loss=-5529.351270, Validation loss=-5574.453125 \n",
      "Epoch 66: Train loss=-5589.300098, Validation loss=-5635.777344 \n",
      "Epoch 67: Train loss=-5650.425684, Validation loss=-5698.378516 \n",
      "Epoch 68: Train loss=-5713.004785, Validation loss=-5762.537695 \n",
      "Epoch 69: Train loss=-5777.135547, Validation loss=-5828.162891 \n",
      "Epoch 70: Train loss=-5842.807861, Validation loss=-5895.221484 \n",
      "Epoch 71: Train loss=-5910.238330, Validation loss=-5964.301367 \n",
      "Epoch 72: Train loss=-5979.322998, Validation loss=-6035.471484 \n",
      "Epoch 73: Train loss=-6050.721045, Validation loss=-6108.876758 \n",
      "Epoch 74: Train loss=-6124.576025, Validation loss=-6184.548633 \n",
      "Epoch 75: Train loss=-6200.711621, Validation loss=-6262.834180 \n",
      "Epoch 76: Train loss=-6279.671289, Validation loss=-6344.271094 \n",
      "Epoch 77: Train loss=-6361.477881, Validation loss=-6429.272070 \n",
      "Epoch 78: Train loss=-6447.359961, Validation loss=-6518.044336 \n",
      "Epoch 79: Train loss=-6536.968750, Validation loss=-6611.439453 \n",
      "Epoch 80: Train loss=-6631.703809, Validation loss=-6710.144531 \n",
      "Epoch 81: Train loss=-6731.531055, Validation loss=-6814.802734 \n",
      "Epoch 82: Train loss=-6838.061133, Validation loss=-6926.966016 \n",
      "Epoch 83: Train loss=-6952.596094, Validation loss=-7048.147266 \n",
      "Epoch 84: Train loss=-7075.684668, Validation loss=-7179.877344 \n",
      "Epoch 85: Train loss=-7211.448242, Validation loss=-7324.222656 \n",
      "Epoch 86: Train loss=-7360.387598, Validation loss=-7485.008984 \n",
      "Epoch 87: Train loss=-7525.910156, Validation loss=-7665.714453 \n",
      "Epoch 88: Train loss=-7713.624805, Validation loss=-7868.140234 \n",
      "Epoch 89: Train loss=-7927.839160, Validation loss=-8097.997266 \n",
      "Epoch 90: Train loss=-8170.293945, Validation loss=-8371.699219 \n",
      "Epoch 91: Train loss=-8457.538867, Validation loss=-8695.638281 \n",
      "Epoch 92: Train loss=-8800.561523, Validation loss=-9094.950781 \n",
      "Epoch 93: Train loss=-9226.549219, Validation loss=-9600.412891 \n",
      "Epoch 94: Train loss=-9767.899219, Validation loss=-10264.247656 \n",
      "Epoch 95: Train loss=-10483.351367, Validation loss=-10984.723438 \n",
      "Epoch 96: Train loss=-11108.596387, Validation loss=-7184.183984 \n",
      "Epoch 97: Train loss=-11367.190918, Validation loss=-12517.892187 \n",
      "Epoch 98: Train loss=-12203.853711, Validation loss=-12209.396094 \n",
      "Epoch 99: Train loss=-11985.788379, Validation loss=-13396.364844 (saved)\n",
      "Epoch 100: Train loss=-13102.958203, Validation loss=-13772.060156 (saved)\n",
      "Epoch 101: Train loss=-13570.117383, Validation loss=-13673.657031 \n",
      "Epoch 102: Train loss=-13653.703516, Validation loss=-12446.321094 \n",
      "Epoch 103: Train loss=-13346.663770, Validation loss=-8403.346484 \n",
      "Epoch 104: Train loss=-9541.034570, Validation loss=-14343.776562 (saved)\n",
      "Epoch 105: Train loss=-13461.170215, Validation loss=-13400.459375 \n",
      "Epoch 106: Train loss=-13451.688086, Validation loss=-14124.005469 \n",
      "Epoch 107: Train loss=-13606.428320, Validation loss=-13591.255469 \n",
      "Epoch 108: Train loss=-13948.214453, Validation loss=-14647.300781 (saved)\n",
      "Epoch 109: Train loss=-14362.892578, Validation loss=-14950.078906 (saved)\n",
      "Epoch 110: Train loss=-14741.809766, Validation loss=-15418.741406 (saved)\n",
      "Epoch 111: Train loss=-15173.154688, Validation loss=-15017.067188 \n",
      "Epoch 112: Train loss=-14775.907812, Validation loss=-16091.561719 (saved)\n",
      "Epoch 113: Train loss=-6701.396191, Validation loss=-7376.140625 \n",
      "Epoch 114: Train loss=-6164.946411, Validation loss=-9105.911328 \n",
      "Epoch 115: Train loss=-12110.796875, Validation loss=-13015.923047 \n",
      "Epoch 116: Train loss=-12600.014941, Validation loss=-12937.975781 \n",
      "Epoch 117: Train loss=-12633.209473, Validation loss=-13039.274219 \n",
      "Epoch 118: Train loss=-12777.181348, Validation loss=-13168.699219 \n",
      "Epoch 119: Train loss=-12968.556348, Validation loss=-13338.508594 \n",
      "Epoch 120: Train loss=-13179.411035, Validation loss=-13560.047656 \n",
      "Epoch 121: Train loss=-13407.446582, Validation loss=-13817.153906 \n",
      "Epoch 122: Train loss=-13651.107422, Validation loss=-14077.686719 \n",
      "Epoch 123: Train loss=-13909.307617, Validation loss=-14349.261719 \n",
      "Epoch 124: Train loss=-14180.543555, Validation loss=-14636.680469 \n",
      "Epoch 125: Train loss=-14463.209375, Validation loss=-14934.252344 \n",
      "Epoch 126: Train loss=-14758.007812, Validation loss=-15248.239844 \n",
      "Epoch 127: Train loss=-15070.712695, Validation loss=-15578.154688 \n",
      "Epoch 128: Train loss=-15394.481055, Validation loss=-15931.110937 \n",
      "Epoch 129: Train loss=-15732.939453, Validation loss=-16295.028125 (saved)\n",
      "Epoch 130: Train loss=-16092.651367, Validation loss=-16648.450781 (saved)\n",
      "Epoch 131: Train loss=-16441.020313, Validation loss=-17072.389844 (saved)\n",
      "Epoch 132: Train loss=-16782.112695, Validation loss=-16578.460156 \n",
      "Epoch 133: Train loss=24849.300586, Validation loss=28578.956250 \n",
      "Epoch 134: Train loss=46364.667871, Validation loss=28000.652344 \n",
      "Epoch 135: Train loss=-4080.247461, Validation loss=-10259.764844 \n",
      "Epoch 136: Train loss=-8777.563281, Validation loss=-7754.342969 \n",
      "Epoch 137: Train loss=-8217.452148, Validation loss=-8814.378125 \n",
      "Epoch 138: Train loss=-9231.622461, Validation loss=-9701.739063 \n",
      "Epoch 139: Train loss=-9754.228809, Validation loss=-9956.357422 \n",
      "Epoch 140: Train loss=-9882.845801, Validation loss=-9986.555078 \n",
      "Epoch 141: Train loss=-9916.220996, Validation loss=-10023.695312 \n",
      "Epoch 142: Train loss=-9967.681348, Validation loss=-10093.594141 \n",
      "Epoch 143: Train loss=-10030.092773, Validation loss=-10162.630078 \n",
      "Epoch 144: Train loss=-10086.009668, Validation loss=-10222.248438 \n",
      "Epoch 145: Train loss=-10136.865527, Validation loss=-10276.167187 \n",
      "Epoch 146: Train loss=-10187.241504, Validation loss=-10328.057422 \n",
      "Epoch 147: Train loss=-10237.866602, Validation loss=-10379.790234 \n",
      "Epoch 148: Train loss=-10289.109766, Validation loss=-10431.532031 \n",
      "Epoch 149: Train loss=-10340.631348, Validation loss=-10484.233984 \n",
      "Epoch 150: Train loss=-10381.338086, Validation loss=-10510.946875 \n",
      "Epoch 151: Train loss=-10407.654980, Validation loss=-10538.110547 \n",
      "Epoch 152: Train loss=-10434.382812, Validation loss=-10565.154297 \n",
      "Epoch 153: Train loss=-10461.087988, Validation loss=-10592.794531 \n",
      "Epoch 154: Train loss=-10488.069531, Validation loss=-10620.779688 \n",
      "Epoch 155: Train loss=-10515.541406, Validation loss=-10648.650000 \n",
      "Epoch 156: Train loss=-10543.080859, Validation loss=-10676.828125 \n",
      "Epoch 157: Train loss=-10570.938379, Validation loss=-10705.469531 \n",
      "Epoch 158: Train loss=-10598.972168, Validation loss=-10734.484375 \n",
      "Epoch 159: Train loss=-10627.329297, Validation loss=-10763.587109 \n",
      "Epoch 160: Train loss=-10656.043750, Validation loss=-10792.898438 \n",
      "Epoch 161: Train loss=-10684.949316, Validation loss=-10822.538672 \n",
      "Epoch 162: Train loss=-10714.130176, Validation loss=-10852.482031 \n",
      "Epoch 163: Train loss=-10743.677246, Validation loss=-10882.645313 \n",
      "Epoch 164: Train loss=-10773.418359, Validation loss=-10913.276953 \n",
      "Epoch 165: Train loss=-10803.535645, Validation loss=-10944.017578 \n",
      "Epoch 166: Train loss=-10833.927637, Validation loss=-10975.053125 \n",
      "Epoch 167: Train loss=-10864.634863, Validation loss=-11006.483203 \n",
      "Epoch 168: Train loss=-10895.584473, Validation loss=-11038.387109 \n",
      "Epoch 169: Train loss=-10926.991602, Validation loss=-11070.393359 \n",
      "Epoch 170: Train loss=-10958.531934, Validation loss=-11102.994141 \n",
      "Epoch 171: Train loss=-10990.425977, Validation loss=-11136.016016 \n",
      "Epoch 172: Train loss=-11022.882910, Validation loss=-11168.788672 \n",
      "Epoch 173: Train loss=-11055.435938, Validation loss=-11202.220312 \n",
      "Epoch 174: Train loss=-11088.351465, Validation loss=-11236.280859 \n",
      "Epoch 175: Train loss=-11121.724316, Validation loss=-11270.627734 \n",
      "Epoch 176: Train loss=-11155.417578, Validation loss=-11305.092578 \n",
      "Epoch 177: Train loss=-11189.594434, Validation loss=-11339.732422 \n",
      "Epoch 178: Train loss=-11223.826367, Validation loss=-11375.119141 \n",
      "Epoch 179: Train loss=-11258.676855, Validation loss=-11410.666797 \n",
      "Epoch 180: Train loss=-11293.789258, Validation loss=-11446.742969 \n",
      "Epoch 181: Train loss=-11329.412500, Validation loss=-11482.970312 \n",
      "Epoch 182: Train loss=-11365.294824, Validation loss=-11519.988281 \n",
      "Epoch 183: Train loss=-11401.653711, Validation loss=-11557.252344 \n",
      "Epoch 184: Train loss=-11438.270605, Validation loss=-11595.046094 \n",
      "Epoch 185: Train loss=-11475.321875, Validation loss=-11633.128125 \n",
      "Epoch 186: Train loss=-11512.985937, Validation loss=-11671.649609 \n",
      "Epoch 187: Train loss=-11550.837695, Validation loss=-11710.770313 \n",
      "Epoch 188: Train loss=-11589.365723, Validation loss=-11749.916406 \n",
      "Epoch 189: Train loss=-11628.177734, Validation loss=-11789.626953 \n",
      "Epoch 190: Train loss=-11667.335840, Validation loss=-11830.251562 \n",
      "Epoch 191: Train loss=-11707.176563, Validation loss=-11870.797656 \n",
      "Epoch 192: Train loss=-11747.255176, Validation loss=-11912.298828 \n",
      "Epoch 193: Train loss=-11787.837500, Validation loss=-11954.036719 \n",
      "Epoch 194: Train loss=-11829.101855, Validation loss=-11995.712500 \n",
      "Epoch 195: Train loss=-11870.560352, Validation loss=-12038.630859 \n",
      "Epoch 196: Train loss=-11912.705566, Validation loss=-12082.122656 \n",
      "Epoch 197: Train loss=-11955.255078, Validation loss=-12126.073437 \n",
      "Epoch 198: Train loss=-11998.391016, Validation loss=-12170.037500 \n",
      "Epoch 199: Train loss=-12042.026074, Validation loss=-12214.566406 \n",
      "Finished training with best train loss: -16697.251758 and validation loss: -17072.389844\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": 491,
   "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": 492,
   "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": 493,
   "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: 1.7351333644910484e-05\n",
      "n-MeRCI: 0.15508320927619934\n",
      "RMSCE: 0.21950307488441467\n",
      "Cerr: 0.0318121463060379\n",
      "In-domain results\n",
      "MSE: 2.2757325205020606e-05\n",
      "n-MeRCI: 0.32411494851112366\n",
      "RMSCE: 0.2061372697353363\n",
      "ProbConserv Results\n",
      "MSE: 1.4251739485189318e-05\n",
      "n-MeRCI: 0.2687947154045105\n",
      "RMSCE: 0.21836964786052704\n",
      "Cerr: 0.0353960245847702\n",
      "Prob_Cerr: 2.3849631247685465e-07\n",
      "Here\n",
      "\n",
      "\n",
      "Out-of-domain results\n",
      "MSE: 2.0100733527215196e-05\n",
      "n-MeRCI: 0.2769961357116699\n",
      "RMSCE: 0.21126341819763184\n",
      "ProbConserv Results\n",
      "MSE: 1.2615041341632605e-05\n",
      "n-MeRCI: 0.22199302911758423\n",
      "RMSCE: 0.22264768183231354\n",
      "Cerr: 0.03430234268307686\n",
      "Prob_Cerr: 2.4290119426950696e-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": 494,
   "id": "98acb797",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "1.4251739485189318e-05"
      ]
     },
     "execution_count": 494,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "id_results['pc.mse']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "1ec3729e-d8e6-454f-8e4e-6b05f44a94f4",
   "metadata": {},
   "outputs": [],
   "source": [
    "out = model(x_train.to(device))\n",
    "x = train_loader.dataset.tensors[0]\n",
    "y = train_loader.dataset.tensors[1]\n",
    "mass_rhs_func = dataset_class.get_mass_rhs_func(x=x)\n",
    "\n",
    "if model.probconserv:\n",
    "    _mu, _var, = out[0].cpu(), out[1].cpu()\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",
    "mu, var, = out[0].cpu(), out[1].cpu()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "45271205-293d-4a07-8c1f-3fa249cfe744",
   "metadata": {},
   "source": [
    "## Experiments to check CRPS (sampling) and CRPS without sampling"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f3f1485f-07fb-4039-a0c4-d0480edbd8bd",
   "metadata": {},
   "outputs": [],
   "source": [
    "model.loss_func(out, y.to(device))/len(out[0])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "7f826c2e-eee5-4b4a-abc0-daf2e505aa50",
   "metadata": {},
   "outputs": [],
   "source": [
    "out[0].shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "4dd459b4-d1b8-4d83-8540-074bb8374287",
   "metadata": {},
   "outputs": [],
   "source": [
    "crps_by_sample = utils.compute_sampling_crps_by_example(mu, var, y,nbins=500)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "400f807a-3676-42a2-82fd-f0ee6999c64e",
   "metadata": {},
   "outputs": [],
   "source": [
    "torch.mean(crps_by_sample)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "913e8446-4753-427f-8d64-d69dbed2b866",
   "metadata": {},
   "outputs": [],
   "source": [
    "std = torch.sqrt(var)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "9e8ad8c6-28e9-45ff-a9e3-a7c828ed5f2a",
   "metadata": {},
   "outputs": [],
   "source": [
    "out = model(x_train.to(device))\n",
    "x = train_loader.dataset.tensors[0]\n",
    "y = train_loader.dataset.tensors[1]\n",
    "mass_rhs_func = dataset_class.get_mass_rhs_func(x=x)\n",
    "\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]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "64b75faa-0ce0-47f9-ae71-2aae64386daf",
   "metadata": {},
   "outputs": [],
   "source": [
    "t_idx = 1\n",
    "parameter_idx = 0\n",
    "with torch.no_grad():\n",
    "    plt.ylabel(\"u(x,t={t:.2f})\".format(t=t[slice(*tpred)][t_idx]))\n",
    "    plt.title(\"Learning Heat Equation for parameter = {k:.2f}\".format(k = x_train[parameter_idx,0,0,0]))\n",
    "    plt.xlabel(\"x\")\n",
    "    plt.plot(grid, mu[parameter_idx,:,t_idx,0], '--', lw=2, label = \"predicted $\\mu$ and $\\pm 3\\sigma$ (varFNO)\")\n",
    "    plt.fill_between(grid, mu[parameter_idx,:,t_idx,0]+3*std[parameter_idx,:,t_idx,0], mu[parameter_idx,:,t_idx,0]-3*std[parameter_idx,:,t_idx,0], alpha=0.2)\n",
    "    plt.plot(grid, y_train[parameter_idx,:,t_idx,0], color = \"green\", label = \"true\")\n",
    "    plt.legend(loc=\"upper right\")\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "fbeef4c3-208f-490b-9f76-24ba47485adc",
   "metadata": {},
   "outputs": [],
   "source": [
    "t_idx = 1\n",
    "\n",
    "for parameters_idx in range(0, 1, 5):\n",
    "    with torch.no_grad():\n",
    "        plt.ylabel(\"u(x,t={t:.2f})\".format(t=t[slice(*tpred)][t_idx]))\n",
    "        plt.title(\"Learning {dataset} for parameter = {k:.2f}\".format(k = x_train[parameters_idx,0,0,0], dataset = dataset))\n",
    "        plt.xlabel(\"x\")\n",
    "        plt.plot(grid, new_mu[parameters_idx,:,t_idx,0], '--', lw=2, label = \"predicted $\\mu$ and $\\pm 3\\sigma$ (varFNO)\")\n",
    "        plt.fill_between(grid, new_mu[parameters_idx,:,t_idx,0]+3*new_std[parameters_idx,:,t_idx,0], new_mu[parameters_idx,:,t_idx,0]-3*new_std[parameters_idx,:,t_idx,0], alpha=0.2)\n",
    "        plt.plot(grid, y_train[parameters_idx,:,t_idx,0], color = \"green\", label = \"true\")        \n",
    "        plt.legend()\n",
    "        # plt.ylim(-1.0,1.5)\n",
    "        plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "5ea3ace5-4141-4975-9823-0ec6d7aab7cd",
   "metadata": {},
   "outputs": [],
   "source": [
    "ucons_stats_train = utils.compute_all_metrics_avg((mu, torch.square(std)), y_train, {})\n",
    "ucons_stats_train[\"nMeRCI_all\"] = utils.compute_nMeRCI(mu, torch.square(std), y_train).item()\n",
    "ucons_stats_train[\"rmsce_all\"] = utils.compute_rmsce(mu, torch.square(std), y_train).item()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "bbc1aeab-f5a3-45e5-8f17-c47458e866f2",
   "metadata": {},
   "outputs": [],
   "source": [
    "ucons_stats_train"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "6090d363-b486-478d-9694-278e7526154d",
   "metadata": {},
   "outputs": [],
   "source": [
    "probconserv_stats_train = utils.compute_all_metrics_avg((new_mu, torch.square(new_std)), y_train, {})\n",
    "probconserv_stats_train[\"nMeRCI_all\"] = utils.compute_nMeRCI(new_mu, torch.square(new_std), y_train).item()\n",
    "probconserv_stats_train[\"rmsce_all\"] = utils.compute_rmsce(new_mu, torch.square(new_std), y_train).item()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "fbc2b1ac-a8a6-4991-bc08-aad3f87982e0",
   "metadata": {},
   "outputs": [],
   "source": [
    "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",
    "ucons_stats_train[\"cerr_by_example\"] = cerr.tolist()\n",
    "ucons_stats_train[\"mcerr\"] = cerr.mean().item()\n",
    "probconserv_stats_train[\"cerr_by_example\"] = new_cerr.tolist()\n",
    "probconserv_stats_train[\"mcerr\"] = new_cerr.mean().item()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "aa190002-f493-40a0-857d-79de73857da7",
   "metadata": {},
   "outputs": [],
   "source": [
    "out = model(x_ood_test.to(device))\n",
    "\n",
    "x = ood_test_loader.dataset.tensors[0]\n",
    "y = ood_test_loader.dataset.tensors[1]\n",
    "mass_rhs_func = dataset_class.get_mass_rhs_func(x=x)\n",
    "if model.probconserv:\n",
    "    _mu, _var, = out[0].cpu(), out[1].cpu()\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",
    "mu, var, = out[0].cpu(), out[1].cpu()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "39e9bde3-2eba-4801-aa30-88c2c043ec20",
   "metadata": {},
   "outputs": [],
   "source": [
    "std = torch.sqrt(var)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "2384c2df-e413-4a42-8107-977afa42306e",
   "metadata": {},
   "outputs": [],
   "source": [
    "t_idx = 1\n",
    "parameter_idx = 0\n",
    "with torch.no_grad():\n",
    "    plt.ylabel(\"u(x,t={t:.2f})\".format(t=t[slice(*tpred)][t_idx]))\n",
    "    plt.title(\"Learning Heat Equation for parameter = {k:.2f}\".format(k = x_ood_test[parameter_idx,0,0,0]))\n",
    "    plt.xlabel(\"x\")\n",
    "    plt.plot(grid, mu[parameter_idx,:,t_idx,0], '--', lw=2, label = \"predicted $\\mu$ and $\\pm 3\\sigma$ (varFNO)\")\n",
    "    plt.fill_between(grid, mu[parameter_idx,:,t_idx,0]+3*std[parameter_idx,:,t_idx,0], mu[parameter_idx,:,t_idx,0]-3*std[parameter_idx,:,t_idx,0], alpha=0.2)\n",
    "    plt.plot(grid, y_ood_test[parameter_idx,:,t_idx,:], color = \"green\", label = \"true\")\n",
    "    plt.legend(loc=\"upper right\")\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "bbca5941-d33a-4608-b00c-99f5024fd3aa",
   "metadata": {},
   "outputs": [],
   "source": [
    "x = ood_test_loader.dataset.tensors[0]\n",
    "y = ood_test_loader.dataset.tensors[1]\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]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "6b809290-3cb4-4d19-a03a-ed6435a705a7",
   "metadata": {},
   "outputs": [],
   "source": [
    "# t_idx = len(t[slice(*tpred)])//2\n",
    "t_idx = 1\n",
    "\n",
    "for parameters_idx in range(0, 1, 5):\n",
    "    with torch.no_grad():\n",
    "        plt.ylabel(\"u(x,t={t:.2f})\".format(t=t[slice(*tpred)][t_idx]))\n",
    "        plt.title(\"Learning {dataset} for parameter = {k:.2f}\".format(k = x_ood_test[parameters_idx,0,0,0], dataset = dataset))\n",
    "        plt.xlabel(\"x\")\n",
    "        plt.plot(grid, new_mu[parameters_idx,:,t_idx,0], '--', lw=2, label = \"predicted $\\mu$ and $\\pm 3\\sigma$ (varFNO)\")\n",
    "        plt.fill_between(grid, new_mu[parameters_idx,:,t_idx,0]+3*new_std[parameters_idx,:,t_idx,0], new_mu[parameters_idx,:,t_idx,0]-3*new_std[parameters_idx,:,t_idx,0], alpha=0.2)\n",
    "        plt.plot(grid, y_ood_test[parameters_idx,:,t_idx,0], color = \"green\", label = \"true\")\n",
    "        print(torch.norm(y_ood_test[parameters_idx,:,t_idx,0] - new_mu[parameters_idx,:,t_idx,0]))\n",
    "        plt.legend()\n",
    "        # plt.ylim(-1.0,1.5)\n",
    "        plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f5762b3e-644a-4cae-ac0f-5ab97ce326bf",
   "metadata": {},
   "outputs": [],
   "source": [
    "ucons_stats_test = utils.compute_all_metrics_avg((mu, torch.square(std)), y_ood_test, {})\n",
    "ucons_stats_test[\"nMeRCI_all\"] = utils.compute_nMeRCI(mu, torch.square(std), y_ood_test).item()\n",
    "ucons_stats_test[\"rmsce_all\"] = utils.compute_rmsce(mu, torch.square(std), y_ood_test).item()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "053bdcfd-e155-4a61-9532-50a72beda9c8",
   "metadata": {},
   "outputs": [],
   "source": [
    "probconserv_stats_test = utils.compute_all_metrics_avg((new_mu, torch.square(new_std)), y_ood_test, {})\n",
    "probconserv_stats_test[\"nMeRCI_all\"] = utils.compute_nMeRCI(new_mu, torch.square(new_std), y_ood_test).item()\n",
    "probconserv_stats_test[\"rmsce_all\"] = utils.compute_rmsce(new_mu, torch.square(new_std), y_ood_test).item()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "973a1ea0-a774-4d8a-9947-fabe05d76bfc",
   "metadata": {},
   "outputs": [],
   "source": [
    "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",
    "ucons_stats_test[\"cerr_by_example\"] = cerr.tolist()\n",
    "ucons_stats_test[\"mcerr\"] = cerr.mean().item()\n",
    "probconserv_stats_test[\"cerr_by_example\"] = new_cerr.tolist()\n",
    "probconserv_stats_test[\"mcerr\"] = new_cerr.mean().item()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "da438412-1398-430d-ba74-82db34174f39",
   "metadata": {},
   "outputs": [],
   "source": [
    "ucons_stats_train"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "09f0df61-ed39-4fe8-8463-87a363b11ef6",
   "metadata": {},
   "source": [
    "## E2E Training"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c36a3814-61c7-4f14-a367-bad3828213de",
   "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": null,
   "id": "8d06a5a3-4116-421c-bd9a-e01ead11928c",
   "metadata": {},
   "outputs": [],
   "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": null,
   "id": "e11c51cf-7916-4324-83b5-d62666a03138",
   "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 = OutputVarNO(base_model_class=FNO2d, probconserv=True, base_model_params=FNO2d_params)\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": null,
   "id": "dd02c190-49e0-490f-85a4-1189103cef61",
   "metadata": {},
   "outputs": [],
   "source": [
    "t"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "7d1e8d19-c876-4e02-a95c-36987cb27c62",
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "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": null,
   "id": "4f58dc93-0b91-4195-b51e-126ad1212b6f",
   "metadata": {},
   "outputs": [],
   "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(\"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": null,
   "id": "d3c74561-cafe-4068-b5ec-92d2c4634cc9",
   "metadata": {},
   "outputs": [],
   "source": [
    "id_results['pc.mse']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c29252c6-3147-4b6a-b320-ee566b03e910",
   "metadata": {},
   "outputs": [],
   "source": [
    "model.probconserv"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d3c1bdb4-7418-4ecf-9309-5639193fa8d6",
   "metadata": {},
   "outputs": [],
   "source": [
    "out = model(x_train.to(device))\n",
    "x = train_loader.dataset.tensors[0]\n",
    "y = train_loader.dataset.tensors[1]\n",
    "mass_rhs_func = dataset_class.get_mass_rhs_func(x=x)\n",
    "if model.probconserv:\n",
    "    _mu, _var, = out[0].cpu(), out[1].cpu()\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",
    "mu, var, = out[0].cpu(), out[1].cpu()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "6d7837ba-f720-43d2-b8d0-8252c41e8ad3",
   "metadata": {},
   "outputs": [],
   "source": [
    "std = torch.sqrt(var)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "cb722479-afdf-4b05-a156-c8a2bbbf245a",
   "metadata": {},
   "outputs": [],
   "source": [
    "t_idx = 1\n",
    "parameter_idx = 0\n",
    "with torch.no_grad():\n",
    "    plt.ylabel(\"u(x,t={t:.2f})\".format(t=t[slice(*tpred)][t_idx]))\n",
    "    plt.title(\"Learning Heat Equation for parameter = {k:.2f}\".format(k = x_train[parameter_idx,0,0,0]))\n",
    "    plt.xlabel(\"x\")\n",
    "    plt.plot(grid, mu[parameter_idx,:,t_idx,0], '--', lw=2, label = \"predicted $\\mu$ and $\\pm 3\\sigma$ (varFNO)\")\n",
    "    plt.fill_between(grid, mu[parameter_idx,:,t_idx,0]+3*std[parameter_idx,:,t_idx,0], mu[parameter_idx,:,t_idx,0]-3*std[parameter_idx,:,t_idx,0], alpha=0.2)\n",
    "    plt.plot(grid, y_train[parameter_idx,:,t_idx,0], color = \"green\", label = \"true\")\n",
    "    plt.legend(loc=\"upper right\")\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ad73c5e5-51d1-41a9-868b-aa0a101bd2b2",
   "metadata": {},
   "outputs": [],
   "source": [
    "x = train_loader.dataset.tensors[0]\n",
    "y = train_loader.dataset.tensors[1]\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]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "4e1d8ff4-50fc-4cdc-b824-ea2e4d850060",
   "metadata": {},
   "outputs": [],
   "source": [
    "# t_idx = len(t[slice(*tpred)])//2\n",
    "t_idx = 1\n",
    "\n",
    "for parameters_idx in range(0, 1, 5):\n",
    "    with torch.no_grad():\n",
    "        plt.ylabel(\"u(x,t={t:.2f})\".format(t=t[slice(*tpred)][t_idx]))\n",
    "        plt.title(\"Learning {dataset} for parameter = {k:.2f}\".format(k = x_train[parameters_idx,0,0,0], dataset = dataset))\n",
    "        plt.xlabel(\"x\")\n",
    "        plt.plot(grid, new_mu[parameters_idx,:,t_idx,0], '--', lw=2, label = \"predicted $\\mu$ and $\\pm 3\\sigma$ (varFNO)\")\n",
    "        plt.fill_between(grid, new_mu[parameters_idx,:,t_idx,0]+3*new_std[parameters_idx,:,t_idx,0], new_mu[parameters_idx,:,t_idx,0]-3*new_std[parameters_idx,:,t_idx,0], alpha=0.2)\n",
    "        plt.plot(grid, y_train[parameters_idx,:,t_idx,0], color = \"green\", label = \"true\")\n",
    "        print(torch.norm(y_train[parameters_idx,:,t_idx,0] - new_mu[parameters_idx,:,t_idx,0]))\n",
    "        plt.legend()\n",
    "        # plt.ylim(-1.0,1.5)\n",
    "        plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "a48dcabf-7ee7-4671-9050-8e8e5340a020",
   "metadata": {},
   "outputs": [],
   "source": [
    "e2e_stats_train = utils.compute_all_metrics_avg((mu, torch.square(std)), y_train, {})\n",
    "e2e_stats_train[\"nMeRCI_all\"] = utils.compute_nMeRCI(mu, torch.square(std), y_train).item()\n",
    "e2e_stats_train[\"rmsce_all\"] = utils.compute_rmsce(mu, torch.square(std), y_train).item()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "1bf0a238-c09a-4d2d-b666-01b025ab0f3d",
   "metadata": {},
   "outputs": [],
   "source": [
    "e2e_probconserv_stats_train = utils.compute_all_metrics_avg((new_mu, torch.square(new_std)), y_train, {})\n",
    "e2e_probconserv_stats_train[\"nMeRCI_all\"] = utils.compute_nMeRCI(new_mu, torch.square(new_std), y_train).item()\n",
    "e2e_probconserv_stats_train[\"rmsce_all\"] = utils.compute_rmsce(new_mu, torch.square(new_std), y_train).item()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d75eddde-47c9-434f-ab24-e999e03cb312",
   "metadata": {},
   "outputs": [],
   "source": [
    "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",
    "e2e_stats_train[\"cerr_by_example\"] = cerr.tolist()\n",
    "e2e_stats_train[\"mcerr\"] = cerr.mean().item()\n",
    "e2e_probconserv_stats_train[\"cerr_by_example\"] = new_cerr.tolist()\n",
    "e2e_probconserv_stats_train[\"mcerr\"] = new_cerr.mean().item()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "50627394-b489-4944-a427-98f04dde6b82",
   "metadata": {},
   "outputs": [],
   "source": [
    "out = model(x_ood_test.to(device))\n",
    "\n",
    "x = ood_test_loader.dataset.tensors[0]\n",
    "y = ood_test_loader.dataset.tensors[1]\n",
    "mass_rhs_func = dataset_class.get_mass_rhs_func(x=x)\n",
    "if model.probconserv:\n",
    "    _mu, _var, = out[0].cpu(), out[1].cpu()\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",
    "mu, var, = out[0].cpu(), out[1].cpu()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3639a00a-4839-4317-98f9-549ba4ab4656",
   "metadata": {},
   "outputs": [],
   "source": [
    "std = torch.sqrt(var)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d1eddcf4-0349-4c22-819c-d8bf5fd81cdf",
   "metadata": {},
   "outputs": [],
   "source": [
    "t_idx = 1\n",
    "parameter_idx = 0\n",
    "with torch.no_grad():\n",
    "    plt.ylabel(\"u(x,t={t:.2f})\".format(t=t[slice(*tpred)][t_idx]))\n",
    "    plt.title(\"Learning Heat Equation for parameter = {k:.2f}\".format(k = x_ood_test[parameter_idx,0,0,0]))\n",
    "    plt.xlabel(\"x\")\n",
    "    plt.plot(grid, mu[parameter_idx,:,t_idx,0], '--', lw=2, label = \"predicted $\\mu$ and $\\pm 3\\sigma$ (varFNO)\")\n",
    "    plt.fill_between(grid, mu[parameter_idx,:,t_idx,0]+3*std[parameter_idx,:,t_idx,0], mu[parameter_idx,:,t_idx,0]-3*std[parameter_idx,:,t_idx,0], alpha=0.2)\n",
    "    plt.plot(grid, y_ood_test[parameter_idx,:,t_idx,:], color = \"green\", label = \"true\")\n",
    "    plt.legend(loc=\"upper right\")\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ee646529-35a6-40d2-898a-288fd1c8240b",
   "metadata": {},
   "outputs": [],
   "source": [
    "x = ood_test_loader.dataset.tensors[0]\n",
    "y = ood_test_loader.dataset.tensors[1]\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]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "dd8c116a-f7e0-4af0-a6d3-b506dee7945b",
   "metadata": {},
   "outputs": [],
   "source": [
    "# t_idx = len(t[slice(*tpred)])//2\n",
    "t_idx = 1\n",
    "\n",
    "for parameters_idx in range(0, 1, 5):\n",
    "    with torch.no_grad():\n",
    "        plt.ylabel(\"u(x,t={t:.2f})\".format(t=t[slice(*tpred)][t_idx]))\n",
    "        plt.title(\"Learning {dataset} for parameter = {k:.2f}\".format(k = x_ood_test[parameters_idx,0,0,0], dataset = dataset))\n",
    "        plt.xlabel(\"x\")\n",
    "        plt.plot(grid, new_mu[parameters_idx,:,t_idx,0], '--', lw=2, label = \"predicted $\\mu$ and $\\pm 3\\sigma$ (varFNO)\")\n",
    "        plt.fill_between(grid, new_mu[parameters_idx,:,t_idx,0]+3*new_std[parameters_idx,:,t_idx,0], new_mu[parameters_idx,:,t_idx,0]-3*new_std[parameters_idx,:,t_idx,0], alpha=0.2)\n",
    "        plt.plot(grid, y_ood_test[parameters_idx,:,t_idx,0], color = \"green\", label = \"true\")\n",
    "        print(torch.norm(y_ood_test[parameters_idx,:,t_idx,0] - new_mu[parameters_idx,:,t_idx,0]))\n",
    "        plt.legend()\n",
    "        # plt.ylim(-1.0,1.5)\n",
    "        plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "463e36f8-0ccd-4319-a639-b3bccdc998a3",
   "metadata": {},
   "outputs": [],
   "source": [
    "e2e_stats_test = utils.compute_all_metrics_avg((mu, torch.square(std)), y_ood_test, {})\n",
    "e2e_stats_test[\"nMeRCI_all\"] = utils.compute_nMeRCI(mu, torch.square(std), y_ood_test).item()\n",
    "e2e_stats_test[\"rmsce_all\"] = utils.compute_rmsce(mu, torch.square(std), y_ood_test).item()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "052adf56-3cac-4fd5-a5d8-83ca0e91a969",
   "metadata": {},
   "outputs": [],
   "source": [
    "e2e_probconserv_stats_test = utils.compute_all_metrics_avg((new_mu, torch.square(new_std)), y_ood_test, {})\n",
    "e2e_probconserv_stats_test[\"nMeRCI_all\"] = utils.compute_nMeRCI(new_mu, torch.square(new_std), y_ood_test).item()\n",
    "e2e_probconserv_stats_test[\"rmsce_all\"] = utils.compute_rmsce(new_mu, torch.square(new_std), y_ood_test).item()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ef17475e-fdca-446c-9b90-1f90beec49d0",
   "metadata": {},
   "outputs": [],
   "source": [
    "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",
    "e2e_stats_test[\"cerr_by_example\"] = cerr.tolist()\n",
    "e2e_stats_test[\"mcerr\"] = cerr.mean().item()\n",
    "e2e_probconserv_stats_test[\"cerr_by_example\"] = new_cerr.tolist()\n",
    "e2e_probconserv_stats_test[\"mcerr\"] = new_cerr.mean().item()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c8f87a91-2bad-44c5-9633-1f6a03f77d68",
   "metadata": {},
   "outputs": [],
   "source": [
    "from decimal import Decimal"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "82377954-2227-47c4-bbc3-7c1c95682b6c",
   "metadata": {},
   "outputs": [],
   "source": [
    " f\"{ucons_stats_train['mcerr']:.2}\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "bd09fb69-7adb-4f8f-811a-45a934dc46d0",
   "metadata": {},
   "outputs": [],
   "source": [
    "e2e_stats_test"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ffa40a8c-b09f-4f57-bbb9-67b241d88629",
   "metadata": {},
   "outputs": [],
   "source": [
    "def dump_to_latex(ucons_stats_train, ucons_stats_test,  probconserv_stats_train, probconserv_stats_test, e2e_stats_train, e2e_stats_test, e2e_probconserv_stats_train, e2e_probconserv_stats_test):\n",
    "    table_str = f\"\"\"\n",
    "    Unconstrained (VarianceNO) & {ucons_stats_train['mse']:.2E} & {ucons_stats_train['nMeRCI_all']:.2E} & {ucons_stats_train['rmsce_all']:.2E} & {ucons_stats_train['mcerr']:.2E} & {ucons_stats_train['crps']:.2E} & {ucons_stats_test['mse']:.2E} & {ucons_stats_test['nMeRCI_all']:.2E} & {ucons_stats_test['rmsce_all']:.2E} & {ucons_stats_test['mcerr']:.2E} & {ucons_stats_test['crps']:.2E} \\\\\\\\\n",
    "    \\\\texttt{{ProbConserv}} & {probconserv_stats_train['mse']:.2E} & {probconserv_stats_train['nMeRCI_all']:.2E} & {probconserv_stats_train['rmsce_all']:.2E} & {probconserv_stats_train['mcerr']:.2E} & {probconserv_stats_train['crps']:.2E} & {probconserv_stats_test['mse']:.2E} & {probconserv_stats_test['nMeRCI_all']:.2E} & {probconserv_stats_test['rmsce_all']:.2E} & {probconserv_stats_test['mcerr']:.2E} & {probconserv_stats_test['crps']:.2E} \\\\\\\\\n",
    "    \\\\ourmethod{{}} & {e2e_stats_train['mse']:.2E} & {e2e_stats_train['nMeRCI_all']:.2E} & {e2e_stats_train['rmsce_all']:.2E} & {e2e_stats_train['mcerr']:.2E} & {e2e_stats_train['crps']:.2E} & {e2e_stats_test['mse']:.2E} & {e2e_stats_test['nMeRCI_all']:.2E} & {e2e_stats_test['rmsce_all']:.2E} & {e2e_stats_test['mcerr']:.2E} & {e2e_stats_test['crps']:.2E} \\\\\\\\\n",
    "    \\\\ourmethod{{}} + \\\\texttt{{ProbConserv}} & {e2e_probconserv_stats_train['mse']:.2E} & {e2e_probconserv_stats_train['nMeRCI_all']:.2E} & {e2e_probconserv_stats_train['rmsce_all']:.2E} & {e2e_probconserv_stats_train['mcerr']:.2E} & {e2e_probconserv_stats_train['crps']:.2E} & {e2e_probconserv_stats_test['mse']:.2E} & {e2e_probconserv_stats_test['nMeRCI_all']:.2E} & {e2e_probconserv_stats_test['rmsce_all']:.2E} & {e2e_probconserv_stats_test['mcerr']:.2E} & {e2e_probconserv_stats_test['crps']:.2E} \\\\\\\\\n",
    "    \"\"\"\n",
    "    return table_str"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c95a912a-bd68-459a-abf7-1c5f03b5661a",
   "metadata": {},
   "outputs": [],
   "source": [
    "print(dump_to_latex(ucons_stats_train, ucons_stats_test,  probconserv_stats_train, probconserv_stats_test, e2e_stats_train, e2e_stats_test, e2e_probconserv_stats_train, e2e_probconserv_stats_test))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "654cef07-235c-492e-9960-9b3d236949dd",
   "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",
    "        cerr = (probconserv.get_empirical_mass_rhs(mu[:, :, :, 0]) - mass_rhs_func(rearrange(x_cpu, \"nf nx nt 1-> nf nt nx 1\"))).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",
    "            cerr_test = (probconserv.get_empirical_mass_rhs(mu_test[:, :, :, 0]) - test_mass_rhs_func(rearrange(x_test_cpu, \"nf nx nt 1-> nf nt nx 1\"))).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": null,
   "id": "47db742d-7441-4c5f-a60c-a7c6bfd76632",
   "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=True,\n",
    "    plot=False,\n",
    "    return_latex=True,\n",
    "    name=\"ProbConserv\"\n",
    ")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "a6a1914c-1c99-4c23-9fe7-fafe7a2a5d7c",
   "metadata": {},
   "outputs": [],
   "source": [
    "latex"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "af0c8f58-4cae-495a-8cfb-ee81163a961c",
   "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
}
