{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "6b3ccdb8-4fd8-4224-b23e-979fa8228059",
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "os.environ['CUDA_VISIBLE_DEVICES'] = '1'\n",
    "\n",
    "import numpy as np\n",
    "import torch\n",
    "import matplotlib.pyplot as plt\n",
    "from models.FNO2d import FNO2d\n",
    "from models.DiverseFNO2d import DiverseFNO2d\n",
    "from models.UncertainNO import *\n",
    "import utils\n",
    "from einops import rearrange, reduce, repeat\n",
    "import os\n",
    "from docopt import docopt\n",
    "import dill\n",
    "from datasets import *\n",
    "import probconserv\n",
    "import sys\n",
    "import torch.optim as optim\n",
    "\n",
    "# args = docopt(__doc__)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "5fdcc8cd-82d9-4cec-888d-806541259f2c",
   "metadata": {},
   "outputs": [],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "1d5eb530-ab43-4054-8953-9efdb86b3106",
   "metadata": {},
   "outputs": [],
   "source": [
    "args = {'--batch_size': '20',\n",
    " '--dataset': 'StefanPME_1D',\n",
    " '--dataset_params': '0.6,0.65',\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': '0.6,0.65'}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "a69b4337-c84b-4454-af1f-979e654417e3",
   "metadata": {},
   "outputs": [],
   "source": [
    "# lambdas_pinn = {}\n",
    "# lambdas_pinn['Heat_Eq]\n",
    "# heq = 1e4"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "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": 9,
   "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": 8,
   "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": 1142,
   "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": 1143,
   "id": "a7e5c61c-2b59-466f-9b4a-6157fa29ad4b",
   "metadata": {},
   "outputs": [],
   "source": [
    "uq = False\n",
    "model_name = args[\"--model\"]\n",
    "n_models = 1\n",
    "fno_modes2 = min(fno_modes, 12)\n",
    "if args[\"--model\"].lower() == \"FNO2d\".lower():\n",
    "    FNO2d_params = {\"modes1\": fno_modes, \"modes2\": fno_modes2, \"width\": fno_width, \"output_var\": True}\n",
    "    model = FNO2d(**FNO2d_params).to(device)\n",
    "elif args[\"--model\"].lower().startswith(\"EnsembleFNO2d\".lower()):\n",
    "    FNO2d_params = {\"modes1\": fno_modes, \"modes2\": fno_modes2, \"width\": fno_width}\n",
    "    n_models = int(args[\"--m.n_models\"])\n",
    "    utils.filter_config(args, [\"--m.n_models\"], mode=\"add\", new_config=save_args)\n",
    "    model = EnsembleNO(base_model_class=FNO2d, base_model_params=FNO2d_params, n_models=n_models)\n",
    "    uq = True\n",
    "elif args[\"--model\"].lower().startswith(\"BayesianFNO2d\".lower()):\n",
    "    FNO2d_params = {\"modes1\": fno_modes, \"modes2\": fno_modes2, \"width\": fno_width}\n",
    "    model = BayesianNO(base_model_class=FNO2d, base_model_params=FNO2d_params)\n",
    "    uq = True\n",
    "elif args[\"--model\"].lower().startswith(\"MCDropoutFNO2d\".lower()):\n",
    "    FNO2d_params = {\"modes1\": fno_modes, \"modes2\": fno_modes2, \"width\": fno_width}\n",
    "    dropout = float(args[\"--m.drop_prob\"])\n",
    "    n_dropouts = int(args[\"--m.n_models\"])\n",
    "    utils.filter_config(args, [\"--m.n_models\", \"--m.drop_prob\"], mode=\"add\", new_config=save_args)\n",
    "    model = MCDropoutNO(base_model_class=FNO2d, base_model_params=FNO2d_params, dropout=dropout, n_dropouts=n_dropouts)\n",
    "    uq = True\n",
    "elif args[\"--model\"].lower().startswith(\"OutputVarFNO2d\".lower()):\n",
    "    FNO2d_params = {\"modes1\": fno_modes, \"modes2\": fno_modes2, \"width\": fno_width}\n",
    "    model = SoftOutputVarNO(base_model_class=FNO2d, probconserv=False, base_model_params=FNO2d_params, constraint_context=constraint_context, lambda_pinn=1e-4)\n",
    "    uq = True\n",
    "elif args[\"--model\"].lower().startswith(\"DiverseFNO2d\".lower()):\n",
    "    FNO2d_params = {\"modes1\": fno_modes, \"modes2\": fno_modes2, \"width\": fno_width}\n",
    "    lam = float(args[\"--m.reg_strength\"])\n",
    "    reg_type = args[\"--m.reg_type\"]\n",
    "    n_models = int(args[\"--m.n_models\"])\n",
    "    n_regularize = int(args[\"--m.n_regularize\"])\n",
    "    utils.filter_config(args, [\"--m.n_models\", \"--m.reg_strength\", \"--m.reg_type\", \"--m.n_regularize\"], mode=\"add\", new_config=save_args)\n",
    "    model = DiverseFNO2d(reg_loss=reg_type, n_outputs=n_models, bias_last=False, lam=lam, n_regularize=n_regularize, **FNO2d_params).to(device)\n",
    "    uq = True\n",
    "else:\n",
    "    raise NotImplementedError"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1144,
   "id": "4333d6f8-fe2c-4caa-a889-efae589c31ff",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([160, 100, 20, 1])"
      ]
     },
     "execution_count": 1144,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "y_train.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1145,
   "id": "96aa9ee0-a30e-458b-85e2-5e7dbbd89dc4",
   "metadata": {},
   "outputs": [],
   "source": [
    "mu_true = torch.mean(y_train, dim = 0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1146,
   "id": "fc1ac32d-6fb2-4293-8ea7-f90486a32dce",
   "metadata": {},
   "outputs": [],
   "source": [
    "var_true = torch.var(y_train, dim = 0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1147,
   "id": "f07c25ce-49e4-4331-81a3-ee678808a6b4",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([100, 20, 1])"
      ]
     },
     "execution_count": 1147,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "mu_true.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1148,
   "id": "0f4d18b3-1988-4417-9122-3a8709048fc2",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor([0.0000e+00, 2.1284e-04, 4.2556e-04, 6.3808e-04, 8.5027e-04, 1.0620e-03,\n",
      "        1.2733e-03, 1.4838e-03, 1.6937e-03, 1.9027e-03, 2.1107e-03, 2.3178e-03,\n",
      "        2.5236e-03, 2.7282e-03, 2.9315e-03, 3.1333e-03, 3.3337e-03, 3.5324e-03,\n",
      "        3.7294e-03, 3.9246e-03, 4.1180e-03, 4.3094e-03, 4.4988e-03, 4.6861e-03,\n",
      "        4.8712e-03, 5.0541e-03, 5.2348e-03, 5.4130e-03, 5.5889e-03, 5.7623e-03,\n",
      "        5.9331e-03, 2.7728e-01, 3.0200e-01, 1.1456e-01, 0.0000e+00, 0.0000e+00,\n",
      "        0.0000e+00, 0.0000e+00, 0.0000e+00, 0.0000e+00, 0.0000e+00, 0.0000e+00,\n",
      "        0.0000e+00, 0.0000e+00, 0.0000e+00, 0.0000e+00, 0.0000e+00, 0.0000e+00,\n",
      "        0.0000e+00, 0.0000e+00, 0.0000e+00, 0.0000e+00, 0.0000e+00, 0.0000e+00,\n",
      "        0.0000e+00, 0.0000e+00, 0.0000e+00, 0.0000e+00, 0.0000e+00, 0.0000e+00,\n",
      "        0.0000e+00, 0.0000e+00, 0.0000e+00, 0.0000e+00, 0.0000e+00, 0.0000e+00,\n",
      "        0.0000e+00, 0.0000e+00, 0.0000e+00, 0.0000e+00, 0.0000e+00, 0.0000e+00,\n",
      "        0.0000e+00, 0.0000e+00, 0.0000e+00, 0.0000e+00, 0.0000e+00, 0.0000e+00,\n",
      "        0.0000e+00, 0.0000e+00, 0.0000e+00, 0.0000e+00, 0.0000e+00, 0.0000e+00,\n",
      "        0.0000e+00, 0.0000e+00, 0.0000e+00, 0.0000e+00, 0.0000e+00, 0.0000e+00,\n",
      "        0.0000e+00, 0.0000e+00, 0.0000e+00, 0.0000e+00, 0.0000e+00, 0.0000e+00,\n",
      "        0.0000e+00, 0.0000e+00, 0.0000e+00, 0.0000e+00])\n"
     ]
    },
    {
     "data": {
      "image/png": "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iPj4ePHll18alDl27Jj4+9//LgIDA4VCoRCNGzcWgwcPFrt377Zwpkwf44ULF8STTz4p/Pz8hFKpFPHx8WLbtm1G63z379aUixcvitmzZ4sHHnhAhISECDc3NxEcHCwGDBgg9uzZY1DW3OtbiMopJeLi4oSHh4fw9vYWbdu2Fa+88oq4fv26voypqQO+/fZb0a5dO6FUKkVUVJR4++23xapVqwQAkZ6eri9n6rVp7d/43cy95u5k69QBtn486/7Wjd3u/JvW/a3qbnK5XISFhYmHH35YfPDBB0KlUtn0vPcKiRAOHu1K5AIGDRqEP/74Q39lEhER3bs4ZonueSUlJQY/nzt3Dtu3bze5ajcREd1b2LJE97zw8HCMGjUK0dHRuHz5MpYtWwa1Wo1jx45VmR+IiKi2aDQaiwP6vby8amUqiXsdB3jTPa9v37748ssvkZWVBYVCgc6dO2PBggUMSkRUpzIyMixODpqcnGwwiJ5qBluWiIiI6qHS0lKLKxxER0fbfbUqWY9hiYiIiMgMDvAmIiIiMoNjlhxAq9Xi+vXr8Pb2rpFlOYiIiMjxhBAoKChARESE2YlZGZYc4Pr161VWmCciIiLnkJGRgUaNGpm8n2HJAby9vQFUnmwfH586rg0RERFZQ6VSITIyUv85bgrDkgPout58fHwYloiIiJyMpSE0HOBNREREZAbDEhEREZEZDEtEREREZjAsEREREZnBsERERERkBsMSERERkRkMS0RERERmMCwRERERmcGwRERERGQGwxIRERGRGQxLRERERGYwLBERERGZwbBEREREZAbDEhEREZEZDEtELkZdoanrKhARuRSGJSIXU1LGsERE5EgMS0QupohhiYjIoZwqLO3btw8DBw5EREQEJBIJtm7darb8li1b8PDDDyM4OBg+Pj7o3LkzfvzxR4Myc+bMgUQiMbjFxsbW4FEQ1axidUVdV4GIyKU4VVgqKirC/fffj6VLl1pVft++fXj44Yexfft2HDlyBA899BAGDhyIY8eOGZRr3bo1MjMz9bf9+/fXRPWJakWZRouyCm1dV4OIyGW41XUFbNGvXz/069fP6vKLFy82+HnBggX45ptv8N1336FDhw767W5ubggLC3NUNYnqlBBAaYUGcjen+i5ERFRv3VPvplqtFgUFBQgICDDYfu7cOURERCA6OhrDhw/HlStX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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": 1149,
   "id": "6fd587e6-4cf8-4cc4-92bd-ac09c7be790a",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<function datasets.StefanPME_1D.get_mass_rhs_func.<locals>.mass_rhs_func(inputs)>"
      ]
     },
     "execution_count": 1149,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "dataset_class.get_mass_rhs_func(x=x_train)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1150,
   "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": 1151,
   "id": "b339b6e5",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "device(type='cpu')"
      ]
     },
     "execution_count": 1151,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "x_train.device"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1152,
   "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": 1153,
   "id": "d71ec298-4bbc-4186-a938-8675f0526890",
   "metadata": {},
   "outputs": [],
   "source": [
    "import time"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1154,
   "id": "87e15df0-69a3-4d0e-82f0-f04540d1a1e8",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([0.0000, 0.0101, 0.0202, 0.0303, 0.0404, 0.0505, 0.0606, 0.0707, 0.0808,\n",
       "        0.0909, 0.1010, 0.1111, 0.1212, 0.1313, 0.1414, 0.1515, 0.1616, 0.1717,\n",
       "        0.1818, 0.1919, 0.2020, 0.2121, 0.2222, 0.2323, 0.2424, 0.2525, 0.2626,\n",
       "        0.2727, 0.2828, 0.2929, 0.3030, 0.3131, 0.3232, 0.3333, 0.3434, 0.3535,\n",
       "        0.3636, 0.3737, 0.3838, 0.3939, 0.4040, 0.4141, 0.4242, 0.4343, 0.4444,\n",
       "        0.4545, 0.4646, 0.4747, 0.4848, 0.4949, 0.5051, 0.5152, 0.5253, 0.5354,\n",
       "        0.5455, 0.5556, 0.5657, 0.5758, 0.5859, 0.5960, 0.6061, 0.6162, 0.6263,\n",
       "        0.6364, 0.6465, 0.6566, 0.6667, 0.6768, 0.6869, 0.6970, 0.7071, 0.7172,\n",
       "        0.7273, 0.7374, 0.7475, 0.7576, 0.7677, 0.7778, 0.7879, 0.7980, 0.8081,\n",
       "        0.8182, 0.8283, 0.8384, 0.8485, 0.8586, 0.8687, 0.8788, 0.8889, 0.8990,\n",
       "        0.9091, 0.9192, 0.9293, 0.9394, 0.9495, 0.9596, 0.9697, 0.9798, 0.9899,\n",
       "        1.0000])"
      ]
     },
     "execution_count": 1154,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "grid"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1155,
   "id": "2165975d",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[0, -1, 5]"
      ]
     },
     "execution_count": 1155,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "tpred"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1156,
   "id": "1ad2c532",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([ 0, -1,  5])"
      ]
     },
     "execution_count": 1156,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "torch.tensor(tpred)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1157,
   "id": "18a1c208-a4df-4f27-8dad-f2466164e855",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([160, 100, 20, 1])"
      ]
     },
     "execution_count": 1157,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "x_train.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1158,
   "id": "6ff40f4d-d8b7-45f7-8e3a-8115564540df",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 0: Train loss=16.109852, Validation loss=-294.001013 (saved)\n",
      "Epoch 1: Train loss=-558.559494, Validation loss=-905.707178 (saved)\n",
      "Epoch 2: Train loss=-1243.330554, Validation loss=-1750.230078 (saved)\n",
      "Epoch 3: Train loss=-2382.025269, Validation loss=-3684.852539 (saved)\n",
      "Epoch 4: Train loss=-5612.254980, Validation loss=-8276.633594 (saved)\n",
      "Epoch 5: Train loss=-8251.976660, Validation loss=-8590.891406 (saved)\n",
      "Epoch 6: Train loss=-9634.435352, Validation loss=-10985.509766 (saved)\n",
      "Epoch 7: Train loss=-9857.744238, Validation loss=-10068.807422 \n",
      "Epoch 8: Train loss=-11060.216699, Validation loss=-12285.261328 (saved)\n",
      "Epoch 9: Train loss=-9684.647620, Validation loss=-5361.115430 \n",
      "Epoch 10: Train loss=-9372.945312, Validation loss=-9717.806250 \n",
      "Epoch 11: Train loss=-9451.340137, Validation loss=-9562.763672 \n",
      "Epoch 12: Train loss=-10002.001367, Validation loss=-10622.125000 \n",
      "Epoch 13: Train loss=-11512.677246, Validation loss=-12632.004297 (saved)\n",
      "Epoch 14: Train loss=-12217.706836, Validation loss=49908.260938 \n",
      "Epoch 15: Train loss=7485.756017, Validation loss=-6722.389844 \n",
      "Epoch 16: Train loss=-6266.037793, Validation loss=-6144.145508 \n",
      "Epoch 17: Train loss=-6004.481396, Validation loss=-5847.110742 \n",
      "Epoch 18: Train loss=-5803.102783, Validation loss=-5797.904102 \n",
      "Epoch 19: Train loss=-5851.824219, Validation loss=-5946.055469 \n",
      "Epoch 20: Train loss=-6041.468994, Validation loss=-6173.489258 \n",
      "Epoch 21: Train loss=-6283.375879, Validation loss=-6431.582227 \n",
      "Epoch 22: Train loss=-6553.660498, Validation loss=-6718.623437 \n",
      "Epoch 23: Train loss=-6852.297656, Validation loss=-7032.388281 \n",
      "Epoch 24: Train loss=-7176.505469, Validation loss=-7371.766406 \n",
      "Epoch 25: Train loss=-7528.432910, Validation loss=-7741.386719 \n",
      "Epoch 26: Train loss=-7911.838574, Validation loss=-8144.415234 \n",
      "Epoch 27: Train loss=-8331.405371, Validation loss=-8588.678516 \n",
      "Epoch 28: Train loss=-8795.512891, Validation loss=-9082.613281 \n",
      "Epoch 29: Train loss=-9314.287793, Validation loss=-9640.023437 \n",
      "Epoch 30: Train loss=-9904.121387, Validation loss=-10281.928125 \n",
      "Epoch 31: Train loss=-10592.604883, Validation loss=-11042.482422 \n",
      "Epoch 32: Train loss=-11420.682812, Validation loss=-11980.918359 \n",
      "Epoch 33: Train loss=-12460.666895, Validation loss=-13190.663281 (saved)\n",
      "Epoch 34: Train loss=-13826.940430, Validation loss=-14604.164062 (saved)\n",
      "Epoch 35: Train loss=4613.835767, Validation loss=-10196.835547 \n",
      "Epoch 36: Train loss=-7342.887207, Validation loss=-6331.854883 \n",
      "Epoch 37: Train loss=-6728.445801, Validation loss=-7135.227734 \n",
      "Epoch 38: Train loss=-7256.369238, Validation loss=-7320.513672 \n",
      "Epoch 39: Train loss=-7318.002441, Validation loss=-7360.327344 \n",
      "Epoch 40: Train loss=-7472.359570, Validation loss=-7642.358203 \n",
      "Epoch 41: Train loss=-7767.868848, Validation loss=-7927.019141 \n",
      "Epoch 42: Train loss=-8048.801855, Validation loss=-8219.957422 \n",
      "Epoch 43: Train loss=-8360.039844, Validation loss=-8552.381250 \n",
      "Epoch 44: Train loss=-8701.277344, Validation loss=-8905.685937 \n",
      "Epoch 45: Train loss=-9065.456836, Validation loss=-9287.123047 \n",
      "Epoch 46: Train loss=-9458.635840, Validation loss=-9699.776953 \n",
      "Epoch 47: Train loss=-9885.502148, Validation loss=-10148.099219 \n",
      "Epoch 48: Train loss=-10350.327539, Validation loss=-10638.807812 \n",
      "Epoch 49: Train loss=-10861.305957, Validation loss=-11179.691016 \n",
      "Epoch 50: Train loss=-11295.438770, Validation loss=-11476.907031 \n",
      "Epoch 51: Train loss=-11601.337500, Validation loss=-11797.031641 \n",
      "Epoch 52: Train loss=-11931.283301, Validation loss=-12141.919141 \n",
      "Epoch 53: Train loss=-12286.065625, Validation loss=-12514.809766 \n",
      "Epoch 54: Train loss=-12671.645020, Validation loss=-12918.442969 \n",
      "Epoch 55: Train loss=-13088.860645, Validation loss=-13359.473437 \n",
      "Epoch 56: Train loss=-13547.891406, Validation loss=-13845.423437 \n",
      "Epoch 57: Train loss=-14051.849609, Validation loss=-14377.567969 \n",
      "Epoch 58: Train loss=-14610.712305, Validation loss=-14979.093750 (saved)\n",
      "Epoch 59: Train loss=-15229.782617, Validation loss=-15587.898437 (saved)\n",
      "Epoch 60: Train loss=-15187.041992, Validation loss=-6499.311328 \n",
      "Epoch 61: Train loss=-8636.676385, Validation loss=-13241.926562 \n",
      "Epoch 62: Train loss=-13483.394727, Validation loss=-14008.705469 \n",
      "Epoch 63: Train loss=-14004.913477, Validation loss=-14351.457812 \n",
      "Epoch 64: Train loss=-14271.855664, Validation loss=-14606.046094 \n",
      "Epoch 65: Train loss=-14622.926758, Validation loss=-14832.170312 \n",
      "Epoch 66: Train loss=-15044.996289, Validation loss=-15394.986719 \n",
      "Epoch 67: Train loss=-15537.390430, Validation loss=-15846.824219 (saved)\n",
      "Epoch 68: Train loss=-16022.140234, Validation loss=-16330.555469 (saved)\n",
      "Epoch 69: Train loss=-16580.167187, Validation loss=-16846.650781 (saved)\n",
      "Epoch 70: Train loss=-4592.778125, Validation loss=2523.268945 \n",
      "Epoch 71: Train loss=-9080.608740, Validation loss=-13263.599219 \n",
      "Epoch 72: Train loss=-12639.589160, Validation loss=-12202.327344 \n",
      "Epoch 73: Train loss=-12585.420898, Validation loss=-12474.807812 \n",
      "Epoch 74: Train loss=-12755.169727, Validation loss=-12798.041016 \n",
      "Epoch 75: Train loss=-12981.885156, Validation loss=-13137.986719 \n",
      "Epoch 76: Train loss=-13275.911523, Validation loss=-13474.287500 \n",
      "Epoch 77: Train loss=-13597.114063, Validation loss=-13799.168750 \n",
      "Epoch 78: Train loss=-13947.992578, Validation loss=-14173.619531 \n",
      "Epoch 79: Train loss=-14327.223633, Validation loss=-14563.807812 \n",
      "Epoch 80: Train loss=-14725.836523, Validation loss=-14978.393750 \n",
      "Epoch 81: Train loss=-15148.023633, Validation loss=-15414.407031 \n",
      "Epoch 82: Train loss=-15591.027344, Validation loss=-15876.771875 \n",
      "Epoch 83: Train loss=-16058.667773, Validation loss=-16346.451562 \n",
      "Epoch 84: Train loss=-16544.049023, Validation loss=-16830.262500 \n",
      "Epoch 85: Train loss=-17032.207422, Validation loss=-17191.390625 (saved)\n",
      "Epoch 86: Train loss=-17323.090625, Validation loss=-13545.001562 \n",
      "Epoch 87: Train loss=-285.845459, Validation loss=-7914.697656 \n",
      "Epoch 88: Train loss=-12724.866211, Validation loss=-13537.492969 \n",
      "Epoch 89: Train loss=-13489.413965, Validation loss=-13247.685937 \n",
      "Epoch 90: Train loss=-13553.735547, Validation loss=-13632.367969 \n",
      "Epoch 91: Train loss=-13763.935156, Validation loss=-13942.112500 \n",
      "Epoch 92: Train loss=-14019.291211, Validation loss=-14206.103125 \n",
      "Epoch 93: Train loss=-14308.712305, Validation loss=-14488.147656 \n",
      "Epoch 94: Train loss=-14641.600781, Validation loss=-14864.871094 \n",
      "Epoch 95: Train loss=-14999.433984, Validation loss=-15230.470312 \n",
      "Epoch 96: Train loss=-15377.492383, Validation loss=-15622.716406 \n",
      "Epoch 97: Train loss=-15777.595898, Validation loss=-16038.989844 \n",
      "Epoch 98: Train loss=-16198.616602, Validation loss=-16467.755469 \n",
      "Epoch 99: Train loss=-16637.561719, Validation loss=-16908.707812 \n",
      "Epoch 100: Train loss=-16988.165625, Validation loss=-17149.547656 \n",
      "Epoch 101: Train loss=-17208.730078, Validation loss=-17393.468750 (saved)\n",
      "Epoch 102: Train loss=-17415.855859, Validation loss=-17608.792187 (saved)\n",
      "Epoch 103: Train loss=-17690.388281, Validation loss=-17854.313281 (saved)\n",
      "Epoch 104: Train loss=-17941.895898, Validation loss=-18118.891406 (saved)\n",
      "Epoch 105: Train loss=-18158.766992, Validation loss=-18252.142187 (saved)\n",
      "Epoch 106: Train loss=-18312.886914, Validation loss=-18502.182812 (saved)\n",
      "Epoch 107: Train loss=-18561.595508, Validation loss=-18415.810937 \n",
      "Epoch 108: Train loss=-18207.867188, Validation loss=-19043.352344 (saved)\n",
      "Epoch 109: Train loss=-17869.373633, Validation loss=-6336.762500 \n",
      "Epoch 110: Train loss=-12883.405957, Validation loss=-18635.139062 \n",
      "Epoch 111: Train loss=-17455.465430, Validation loss=-18344.842187 \n",
      "Epoch 112: Train loss=-18150.872461, Validation loss=-18330.168750 \n",
      "Epoch 113: Train loss=-18295.564844, Validation loss=-18527.344531 \n",
      "Epoch 114: Train loss=-18458.625586, Validation loss=-18457.597656 \n",
      "Epoch 115: Train loss=-18584.108008, Validation loss=-18638.659375 \n",
      "Epoch 116: Train loss=-18644.069531, Validation loss=-18983.312500 \n",
      "Epoch 117: Train loss=-18832.600586, Validation loss=-19017.305469 \n",
      "Epoch 118: Train loss=-18909.241797, Validation loss=-16090.416406 \n",
      "Epoch 119: Train loss=-12482.737109, Validation loss=-15784.366406 \n",
      "Epoch 120: Train loss=-16140.259570, Validation loss=-18749.385937 \n",
      "Epoch 121: Train loss=-17409.067578, Validation loss=-17837.868750 \n",
      "Epoch 122: Train loss=-17997.752930, Validation loss=-18311.810156 \n",
      "Epoch 123: Train loss=-18315.570703, Validation loss=-18322.618750 \n",
      "Epoch 124: Train loss=-18525.562109, Validation loss=-18726.962500 \n",
      "Epoch 125: Train loss=-18625.824805, Validation loss=-18953.604687 \n",
      "Epoch 126: Train loss=-18830.242383, Validation loss=-19074.017969 (saved)\n",
      "Epoch 127: Train loss=-19164.872070, Validation loss=-19314.181250 (saved)\n",
      "Epoch 128: Train loss=-19240.152148, Validation loss=-19018.609375 \n",
      "Epoch 129: Train loss=-19150.132812, Validation loss=-19630.465625 (saved)\n",
      "Epoch 130: Train loss=-18739.131445, Validation loss=-19097.971875 \n",
      "Epoch 131: Train loss=-14606.944482, Validation loss=1200.985986 \n",
      "Epoch 132: Train loss=-11587.688965, Validation loss=-16814.889062 \n",
      "Epoch 133: Train loss=-16892.174609, Validation loss=-16951.948437 \n",
      "Epoch 134: Train loss=-17729.661328, Validation loss=-17710.211719 \n",
      "Epoch 135: Train loss=-17957.856250, Validation loss=-18256.706250 \n",
      "Epoch 136: Train loss=-18224.934180, Validation loss=-18499.993750 \n",
      "Epoch 137: Train loss=-18464.237305, Validation loss=-18675.699219 \n",
      "Epoch 138: Train loss=-18684.906250, Validation loss=-18869.550781 \n",
      "Epoch 139: Train loss=-18891.242773, Validation loss=-19045.545312 \n",
      "Epoch 140: Train loss=-19064.384766, Validation loss=-19170.793750 \n",
      "Epoch 141: Train loss=-19208.705859, Validation loss=-19109.031250 \n",
      "Epoch 142: Train loss=-19016.404883, Validation loss=-19447.330469 \n",
      "Epoch 143: Train loss=-19155.341406, Validation loss=-19768.415625 (saved)\n",
      "Epoch 144: Train loss=-19154.471875, Validation loss=-19650.868750 \n",
      "Epoch 145: Train loss=-19534.402734, Validation loss=-19788.569531 (saved)\n",
      "Epoch 146: Train loss=-19189.270703, Validation loss=-17078.974219 \n",
      "Epoch 147: Train loss=-12317.740649, Validation loss=-19498.128125 \n",
      "Epoch 148: Train loss=-18343.174023, Validation loss=-18288.648437 \n",
      "Epoch 149: Train loss=-18705.362305, Validation loss=-18241.768750 \n",
      "Epoch 150: Train loss=-18824.716211, Validation loss=-19289.571875 \n",
      "Epoch 151: Train loss=-19126.595117, Validation loss=-19392.876562 \n",
      "Epoch 152: Train loss=-19374.502930, Validation loss=-19450.510937 \n",
      "Epoch 153: Train loss=-19430.119336, Validation loss=-19513.078906 \n",
      "Epoch 154: Train loss=-19569.825586, Validation loss=-19703.750781 \n",
      "Epoch 155: Train loss=-19703.347461, Validation loss=-19815.042969 (saved)\n",
      "Epoch 156: Train loss=-19803.894922, Validation loss=-19910.982812 (saved)\n",
      "Epoch 157: Train loss=-19891.344922, Validation loss=-20008.472656 (saved)\n",
      "Epoch 158: Train loss=-19960.061914, Validation loss=-20097.835156 (saved)\n",
      "Epoch 159: Train loss=-20061.963086, Validation loss=-20206.994531 (saved)\n",
      "Epoch 160: Train loss=-20092.841406, Validation loss=-20118.130469 \n",
      "Epoch 161: Train loss=-19744.273828, Validation loss=-20313.892187 (saved)\n",
      "Epoch 162: Train loss=-20093.356445, Validation loss=-20362.455469 (saved)\n",
      "Epoch 163: Train loss=-20169.577539, Validation loss=-19991.525000 \n",
      "Epoch 164: Train loss=-20334.145508, Validation loss=-20464.175000 (saved)\n",
      "Epoch 165: Train loss=-20457.184961, Validation loss=-20588.271875 (saved)\n",
      "Epoch 166: Train loss=-20511.169141, Validation loss=-20660.639844 (saved)\n",
      "Epoch 167: Train loss=-20502.033984, Validation loss=-20175.101562 \n",
      "Epoch 168: Train loss=-19451.356641, Validation loss=-19098.897656 \n",
      "Epoch 169: Train loss=-19023.961719, Validation loss=-20522.226562 \n",
      "Epoch 170: Train loss=-19094.656445, Validation loss=-19205.414062 \n",
      "Epoch 171: Train loss=-19974.203125, Validation loss=-20007.829687 \n",
      "Epoch 172: Train loss=-20094.168359, Validation loss=-20414.232812 \n",
      "Epoch 173: Train loss=-20535.538281, Validation loss=-20498.992187 \n",
      "Epoch 174: Train loss=-20356.943359, Validation loss=-20615.136719 \n",
      "Epoch 175: Train loss=-20411.865039, Validation loss=-20156.536719 \n",
      "Epoch 176: Train loss=-20583.184766, Validation loss=-20499.726562 \n",
      "Epoch 177: Train loss=-19606.463477, Validation loss=-20875.826562 (saved)\n",
      "Epoch 178: Train loss=-20793.718164, Validation loss=-20962.264062 (saved)\n",
      "Epoch 179: Train loss=-20388.477734, Validation loss=-20275.729687 \n",
      "Epoch 180: Train loss=-20261.879492, Validation loss=-20502.375000 \n",
      "Epoch 181: Train loss=-20259.561719, Validation loss=-20947.645312 \n",
      "Epoch 182: Train loss=-20281.731445, Validation loss=-20679.957812 \n",
      "Epoch 183: Train loss=-20565.701367, Validation loss=-21038.099219 (saved)\n",
      "Epoch 184: Train loss=-18892.340234, Validation loss=-20097.009375 \n",
      "Epoch 185: Train loss=-20087.205469, Validation loss=-20298.428906 \n",
      "Epoch 186: Train loss=-19052.433203, Validation loss=-19945.578906 \n",
      "Epoch 187: Train loss=-19871.491211, Validation loss=-20671.867969 \n",
      "Epoch 188: Train loss=-19941.198437, Validation loss=-20068.720312 \n",
      "Epoch 189: Train loss=-20429.523633, Validation loss=-20415.389062 \n",
      "Epoch 190: Train loss=-20487.944531, Validation loss=-20821.958594 \n",
      "Epoch 191: Train loss=-20798.897266, Validation loss=-21003.641406 \n",
      "Epoch 192: Train loss=-20509.872656, Validation loss=-20573.943750 \n",
      "Epoch 193: Train loss=-20569.102734, Validation loss=-20034.673437 \n",
      "Epoch 194: Train loss=-20619.749219, Validation loss=-21155.960937 (saved)\n",
      "Epoch 195: Train loss=-20680.314453, Validation loss=-19480.681250 \n",
      "Epoch 196: Train loss=-20337.609961, Validation loss=-20331.502344 \n",
      "Epoch 197: Train loss=-19829.012695, Validation loss=-18214.562500 \n",
      "Epoch 198: Train loss=-20478.859570, Validation loss=-20487.649219 \n",
      "Epoch 199: Train loss=-20166.871094, Validation loss=-19991.720312 \n",
      "Finished training with best train loss: -21078.617578 and validation loss: -21155.960937\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": 1159,
   "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": 1160,
   "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": 1161,
   "id": "58fe7954-8db9-47db-ba7c-39fde51a4146",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Here\n",
      "Here\n",
      "Here\n",
      "Here\n",
      "Train results\n",
      "MSE: 0.004368497431278229\n",
      "n-MeRCI: 0.8420261740684509\n",
      "RMSCE: 0.15552745759487152\n",
      "Cerr: 0.1712425947189331\n",
      "In-domain results\n",
      "MSE: 0.004094430059194565\n",
      "n-MeRCI: 0.6205651164054871\n",
      "RMSCE: 0.15925250947475433\n",
      "ProbConserv Results\n",
      "MSE: 0.002505195438861847\n",
      "n-MeRCI: 0.6378424763679504\n",
      "RMSCE: 0.15341053903102875\n",
      "Cerr: 0.159877210855484\n",
      "Prob_Cerr: 3.51461608261161e-07\n",
      "Here\n",
      "\n",
      "\n",
      "Out-of-domain results\n",
      "MSE: 0.004290626794099808\n",
      "n-MeRCI: 0.6105079054832458\n",
      "RMSCE: 0.15304656326770782\n",
      "ProbConserv Results\n",
      "MSE: 0.0026433950662612914\n",
      "n-MeRCI: 0.745917558670044\n",
      "RMSCE: 0.1475571244955063\n",
      "Cerr: 0.16875244677066803\n",
      "Prob_Cerr: 3.907984194029268e-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": 1162,
   "id": "98acb797",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.002505195438861847"
      ]
     },
     "execution_count": 1162,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "id_results['pc.mse']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1163,
   "id": "062f8da4-99eb-4f91-97c7-8e711746a05d",
   "metadata": {},
   "outputs": [],
   "source": [
    "def compute_statistics(\n",
    "    model, \n",
    "    x_data, \n",
    "    y_data, \n",
    "    t, \n",
    "    tpred, \n",
    "    grid, \n",
    "    dataset_class, \n",
    "    apply_probconserv=False, \n",
    "    plot=False,\n",
    "    x_data_test=None, \n",
    "    y_data_test=None,\n",
    "    return_latex=False,\n",
    "    name=\"Model\"\n",
    "):\n",
    "    import torch\n",
    "    import utils\n",
    "    import probconserv\n",
    "    import matplotlib.pyplot as plt\n",
    "\n",
    "    device = next(model.parameters()).device\n",
    "    x_data = x_data.to(device)\n",
    "\n",
    "    with torch.no_grad():\n",
    "        out = model(x_data)\n",
    "\n",
    "    if isinstance(out, tuple):\n",
    "        mu, var = out[0].cpu(), out[1].cpu()\n",
    "        std = torch.sqrt(var)\n",
    "    else:\n",
    "        mu = out.cpu()\n",
    "        std = torch.zeros_like(mu)\n",
    "        var = torch.square(std)\n",
    "\n",
    "    x_cpu = x_data.cpu()\n",
    "    mass_rhs_func = dataset_class.get_mass_rhs_func(x=x_cpu)\n",
    "\n",
    "    if apply_probconserv:\n",
    "        new_mu, new_std, _, mass_rhs = probconserv.apply_constraint(\n",
    "            mu=mu[:, :, :, 0],\n",
    "            std=std[:, :, :, 0],\n",
    "            mass_rhs_func=mass_rhs_func,\n",
    "            t=t,\n",
    "            tpred=tpred,\n",
    "            grid_train=grid,\n",
    "            precis_g=float('inf'),\n",
    "            second_deriv_alpha=None,\n",
    "        )\n",
    "        mu = new_mu.unsqueeze(-1)\n",
    "        std = new_std.unsqueeze(-1)\n",
    "        var = torch.square(std)\n",
    "        cerr = (probconserv.get_empirical_mass_rhs(mu[:, :, :, 0]) - mass_rhs).abs().sum(dim=-1)\n",
    "    else:\n",
    "        t_sliced = t[slice(*tpred)]\n",
    "        ts = repeat(t_sliced, \"nt -> nf nt\", nf=mu.shape[0])\n",
    "        xs = repeat(grid, \"nx -> nf nx\", nf=mu.shape[0])\n",
    "        inputs = meshgrid(ts, xs)\n",
    "        cerr = (probconserv.get_empirical_mass_rhs(mu[:, :, :, 0]) - mass_rhs_func(inputs)).abs().sum(dim=-1)\n",
    "\n",
    "    stats = utils.compute_all_metrics_avg((mu, var), y_data, {})\n",
    "    stats[\"nMeRCI_all\"] = utils.compute_nMeRCI(mu, var, y_data).item()\n",
    "    stats[\"rmsce_all\"] = utils.compute_rmsce(mu, var, y_data).item()\n",
    "    stats[\"cerr_by_example\"] = cerr.tolist()\n",
    "    stats[\"mcerr\"] = cerr.mean().item()\n",
    "\n",
    "    # --- Test dataset ---\n",
    "    test_stats = None\n",
    "    if x_data_test is not None and y_data_test is not None:\n",
    "        x_data_test = x_data_test.to(device)\n",
    "        with torch.no_grad():\n",
    "            test_out = model(x_data_test)\n",
    "\n",
    "        if isinstance(test_out, tuple):\n",
    "            mu_test, var_test = test_out[0].cpu(), test_out[1].cpu()\n",
    "            std_test = torch.sqrt(var_test)\n",
    "        else:\n",
    "            mu_test = test_out.cpu()\n",
    "            std_test = torch.zeros_like(mu_test)\n",
    "            var_test = torch.square(std_test)\n",
    "\n",
    "        x_test_cpu = x_data_test.cpu()\n",
    "        test_mass_rhs_func = dataset_class.get_mass_rhs_func(x=x_test_cpu)\n",
    "\n",
    "        if apply_probconserv:\n",
    "            new_mu_test, new_std_test, _, test_mass_rhs = probconserv.apply_constraint(\n",
    "                mu=mu_test[:, :, :, 0],\n",
    "                std=std_test[:, :, :, 0],\n",
    "                mass_rhs_func=test_mass_rhs_func,\n",
    "                t=t,\n",
    "                tpred=tpred,\n",
    "                grid_train=grid,\n",
    "                precis_g=float('inf'),\n",
    "                second_deriv_alpha=None,\n",
    "            )\n",
    "            mu_test = new_mu_test.unsqueeze(-1)\n",
    "            std_test = new_std_test.unsqueeze(-1)\n",
    "            var_test = torch.square(std_test)\n",
    "            cerr_test = (probconserv.get_empirical_mass_rhs(mu_test[:, :, :, 0]) - test_mass_rhs).abs().sum(dim=-1)\n",
    "        else:\n",
    "            t_sliced = t[slice(*tpred)]\n",
    "            ts = repeat(t_sliced, \"nt -> nf nt\", nf=mu_test.shape[0])\n",
    "            xs = repeat(grid, \"nx -> nf nx\", nf=mu_test.shape[0])\n",
    "            inputs = meshgrid(ts, xs)\n",
    "            cerr_test = (probconserv.get_empirical_mass_rhs(mu_test[:, :, :, 0]) - test_mass_rhs_func(inputs)).abs().sum(dim=-1)\n",
    "\n",
    "        test_stats = utils.compute_all_metrics_avg((mu_test, var_test), y_data_test, {})\n",
    "        test_stats[\"nMeRCI_all\"] = utils.compute_nMeRCI(mu_test, var_test, y_data_test).item()\n",
    "        test_stats[\"rmsce_all\"] = utils.compute_rmsce(mu_test, var_test, y_data_test).item()\n",
    "        test_stats[\"cerr_by_example\"] = cerr_test.tolist()\n",
    "        test_stats[\"mcerr\"] = cerr_test.mean().item()\n",
    "\n",
    "    # --- Optional plot ---\n",
    "    if plot:\n",
    "        t_idx = 1\n",
    "        param_idx = 0\n",
    "        with torch.no_grad():\n",
    "            plt.ylabel(f\"u(x, t={t[slice(*tpred)][t_idx]:.2f})\")\n",
    "            plt.xlabel(\"x\")\n",
    "            plt.title(f\"Predicted vs True (param = {x_data[param_idx,0,0,0].item():.2f})\")\n",
    "            mu_plot = mu[param_idx, :, t_idx, 0]\n",
    "            std_plot = std[param_idx, :, t_idx, 0]\n",
    "            y_true_plot = y_data[param_idx, :, t_idx, 0]\n",
    "            plt.plot(grid, mu_plot, '--', lw=2, label=\"μ ± 3σ\")\n",
    "            plt.fill_between(grid, mu_plot + 3*std_plot, mu_plot - 3*std_plot, alpha=0.2)\n",
    "            plt.plot(grid, y_true_plot, color=\"green\", label=\"true\")\n",
    "            plt.legend()\n",
    "            plt.show()\n",
    "\n",
    "    # --- Optional LaTeX row ---\n",
    "    latex_row = None\n",
    "    if return_latex and test_stats:\n",
    "        latex_row = (\n",
    "            f\"{name} & \"\n",
    "            f\"{stats['mse']:.3E} & {stats['nMeRCI_all']:.3E} & {stats['rmsce_all']:.3E} & {stats['mcerr']:.3E} & {stats['crps']:.3E} & \"\n",
    "            f\"{test_stats['mse']:.3E} & {test_stats['nMeRCI_all']:.3E} & {test_stats['rmsce_all']:.3E} & {test_stats['mcerr']:.3E} & {test_stats['crps']:.3E} \\\\\\\\\"\n",
    "        )\n",
    "\n",
    "    return (stats, test_stats, latex_row) if return_latex else (stats, test_stats)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1164,
   "id": "aa23f617-8a23-4bbc-83e1-e3ed3cdf5498",
   "metadata": {},
   "outputs": [],
   "source": [
    "train_stats, test_stats, latex = compute_statistics(\n",
    "    model,\n",
    "    x_train, y_train,\n",
    "    x_data_test=x_ood_test, \n",
    "    y_data_test=y_ood_test,\n",
    "    t=t, tpred=tpred, grid=grid,\n",
    "    dataset_class=dataset_class,\n",
    "    apply_probconserv=False,\n",
    "    plot=False,\n",
    "    return_latex=True,\n",
    "    name=\"Unconstrained\"\n",
    ")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1165,
   "id": "3a5ebe0f-9094-408e-8162-db12f3b800dc",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.004290626347064972"
      ]
     },
     "execution_count": 1165,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "test_stats['mse']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1166,
   "id": "7b0e9a1a-673d-4601-8ecd-0b6be435b39c",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.16875241696834564"
      ]
     },
     "execution_count": 1166,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "test_stats['mcerr']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1167,
   "id": "42b92e3f-5318-4bac-b650-3dff39acc228",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.010062500238418579"
      ]
     },
     "execution_count": 1167,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "test_stats['crps']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1168,
   "id": "3505923a-8f51-44c6-9053-31951d11bf9a",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "'Unconstrained & 4.368E-03 & 8.420E-01 & 1.555E-01 & 1.712E-01 & 1.003E-02 & 4.291E-03 & 6.105E-01 & 1.530E-01 & 1.688E-01 & 1.006E-02 \\\\\\\\'"
      ]
     },
     "execution_count": 1168,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "latex"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "6f43028d-91c3-407f-8bcd-ca4bcce22141",
   "metadata": {},
   "source": [
    "## CRPS\n",
    "Heat = 1e4 \n",
    "\n",
    "PME = 1e1 \n",
    "\n",
    "Stefan = 1e1 \n",
    "\n",
    "Advection = 1e-2 \n",
    "\n",
    "\n",
    "\n",
    "## NLL\n",
    "Heat = 1e2 \n",
    "\n",
    "PME = -- \n",
    "\n",
    "Stefan = 1e1 \n",
    "\n",
    "Advection = 1e-4 \n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3c034735-f46e-4356-a34c-ccda4a81b704",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "optprobconserv",
   "language": "python",
   "name": "optprobconserv"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.9.19"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
