{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "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": 2,
   "id": "3f650c83-1845-4ef1-9b8b-2aaf18d83d08",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "'2.6.0+cu118'"
      ]
     },
     "execution_count": 2,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "torch.__version__"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "dabd741a-54b6-4556-8585-7ed8dfaff61c",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "'Tesla V100S-PCIE-32GB'"
      ]
     },
     "execution_count": 3,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "torch.cuda.get_device_name()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "5fdcc8cd-82d9-4cec-888d-806541259f2c",
   "metadata": {},
   "outputs": [],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "1d5eb530-ab43-4054-8953-9efdb86b3106",
   "metadata": {},
   "outputs": [],
   "source": [
    "args = {'--batch_size': '20',\n",
    " '--dataset': 'PME_1D',\n",
    " '--dataset_params': '3,4',\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': '3,4'}"
   ]
  },
  {
   "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": 7,
   "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": 9,
   "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": 10,
   "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)\n",
    "    model = ProbHardE2ETVD(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": 11,
   "id": "4333d6f8-fe2c-4caa-a889-efae589c31ff",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([160, 100, 20, 1])"
      ]
     },
     "execution_count": 11,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "y_train.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "96aa9ee0-a30e-458b-85e2-5e7dbbd89dc4",
   "metadata": {},
   "outputs": [],
   "source": [
    "mu_true = torch.mean(y_train, dim = 0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "fc1ac32d-6fb2-4293-8ea7-f90486a32dce",
   "metadata": {},
   "outputs": [],
   "source": [
    "var_true = torch.var(y_train, dim = 0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "f07c25ce-49e4-4331-81a3-ee678808a6b4",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([100, 20, 1])"
      ]
     },
     "execution_count": 14,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "mu_true.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "0f4d18b3-1988-4417-9122-3a8709048fc2",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor([0.0402, 0.0407, 0.0411, 0.0411, 0.0400, 0.0000, 0.0000, 0.0000, 0.0000,\n",
      "        0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000,\n",
      "        0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000,\n",
      "        0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000,\n",
      "        0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000,\n",
      "        0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000,\n",
      "        0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000,\n",
      "        0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000,\n",
      "        0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000,\n",
      "        0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000,\n",
      "        0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000,\n",
      "        0.0000])\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAjcAAAHHCAYAAABDUnkqAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/bCgiHAAAACXBIWXMAAA9hAAAPYQGoP6dpAABM7UlEQVR4nO3de3yT9f3//2eSNmk5tBwKbYGOCggIKMUCHXjAabUqU3FTC6hAPU4OUzv9KeqooqNM0eEBYSKnL1NBEd2miMMOPop2QwEVRVChHIS1gECLFHpIrt8fmEBo2iYhp4bH/XbL7dZeua5cr1xtyZP36TIZhmEIAAAgSpjDXQAAAEAgEW4AAEBUIdwAAICoQrgBAABRhXADAACiCuEGAABEFcINAACIKoQbAAAQVQg3AAAgqhBuEFKrVq2SyWTSqlWrQnK+Rx99VCaTKSTnQuBcdNFFuuiii8JdhhuTyaRHH3003GWcsrKyMl133XVq27atTCaTpk+fHu6SgIAj3CAg5s+fL5PJ5HrExcWpe/fuGj9+vMrKygJyjmXLlnn8cKmsrNSjjz4assAENGX33nuv3n//fU2cOFELFy7U5ZdfHtTznfjvgtlsVocOHXTZZZfV+XtNT0+XyWRSdna2x9eZPXu263U+++wz13bnf2Dqe5SWlnpd65o1azR27FhlZmYqNja23v8Ybdu2ze0csbGxSkpK0uDBg/XQQw9px44dXp8TwRET7gIQXSZPnqwzzjhDR48e1erVqzVz5kwtW7ZMX331lZo1a3ZKr71s2TLNmDGjTsCprKzUY489Jkl1/rf/yCOP6MEHHzyl8wKSdOTIEcXENP1/Mv/973/rmmuu0X333Reyc1566aUaNWqUDMNQSUmJXnzxRV188cV69913dcUVV7j2i4uL08qVK1VaWqqUlBS313jllVcUFxeno0ePejzHzJkz1aJFizrbW7Vq5XWdy5Yt08svv6xzzjlHXbp00bffftvg/iNGjNCVV14ph8OhAwcO6NNPP9X06dP17LPPas6cORo+fLjX50ZgNf2/VESUK664Qv3795ck3XbbbWrbtq2eeeYZ/f3vf9eIESNCXk9MTExUfCAhPBwOh6qrqxUXF6e4uLhwlxMQe/bs8ekDvzFHjx6V1WqV2Vx/R0D37t110003ub6/9tprdc4552j69Olu4ea8887Tp59+qsWLF+vuu+92bf/hhx/00Ucf6dprr9Wbb77p8RzXXXedkpKSTum93HXXXXrggQcUHx+v8ePHNxpuzj33XLf3JUnbt2/XZZddptGjR+uss85S3759T6km+IduKQTVxRdfLEkqKSlpcL833nhDmZmZio+PV1JSkm666Sbt2rXL9fyYMWM0Y8YMSe7N3Nu2bVO7du0kSY899phru7N1x9OYG5PJpPHjx+vtt99Wnz59ZLPZ1Lt3by1fvrxOXatWrVL//v0VFxenrl276q9//avX43guuugi9enTR19++aWGDBmiZs2aqVu3blqyZIkk6f/+7/+UlZWl+Ph49ejRQx988EGd19i1a5duueUWJScnu+qcO3eu2z7V1dWaNGmSMjMzlZiYqObNm+uCCy7QypUr3fZzNqVPmzZNL730krp27SqbzaYBAwbo008/bfT97N+/X/fdd5/OPvtstWjRQgkJCbriiiv0xRdf1LlmJpNJr7/+uv70pz+pU6dOiouL0yWXXKLvv/++zus6a4mPj9fAgQP10UcfNVqLJPXp00e/+tWv6mx3OBzq2LGjrrvuOte2adOmafDgwWrbtq3i4+OVmZnp+jmcyPm78corr6h3796y2Wyu34uTx9xs375dY8eOVY8ePRQfH6+2bdvq+uuv17Zt29xe09ll+/HHHys/P1/t2rVT8+bNde2112rv3r11anjvvfc0ZMgQtWzZUgkJCRowYIBeffVVt33++9//6vLLL1diYqKaNWumIUOG6OOPP27wejnrMAxDM2bMcP2tOG3dulXXX3+92rRpo2bNmumXv/yl3n33XbfXcP5sFy1apEceeUQdO3ZUs2bNVFFR0eC5T3b22WcrKSmpzr8LcXFx+s1vflPn/b722mtq3bq1cnJyfDqPr5KTkxUfH39Kr9G5c2fNnz9f1dXVevLJJwNUGXzFf2kRVFu2bJEktW3btt595s+fr7y8PA0YMECFhYUqKyvTs88+q48//ljr169Xq1atdOedd2r37t1asWKFFi5c6Dq2Xbt2mjlzpu666y5de+21+s1vfiNJOueccxqsa/Xq1Vq6dKnGjh2rli1b6rnnntNvf/tb7dixw1Xr+vXrdfnllys1NVWPPfaY7Ha7Jk+e7ApT3jhw4IB+/etfa/jw4br++us1c+ZMDR8+XK+88oruuece/e53v9PIkSP11FNP6brrrtPOnTvVsmVLSccGfv7yl790feC2a9dO7733nm699VZVVFTonnvukSRVVFTo5Zdf1ogRI3T77bfr0KFDmjNnjnJycrRmzRplZGS41fTqq6/q0KFDuvPOO2UymfTkk0/qN7/5jbZu3arY2Nh638vWrVv19ttv6/rrr9cZZ5yhsrIy/fWvf9WQIUO0ceNGdejQwW3/qVOnymw267777lN5ebmefPJJ3Xjjjfrvf//r2mfOnDm68847NXjwYN1zzz3aunWrrr76arVp00ZpaWkNXtvc3Fw9+uijdbowVq9erd27d7t1CTz77LO6+uqrdeONN6q6ulqLFi3S9ddfr3feeUdDhw51e91///vfev311zV+/HglJSUpPT3d4/k//fRTffLJJxo+fLg6deqkbdu2aebMmbrooou0cePGOt2wEyZMUOvWrVVQUKBt27Zp+vTpGj9+vBYvXuzaZ/78+brlllvUu3dvTZw4Ua1atdL69eu1fPlyjRw50lXfFVdcoczMTBUUFMhsNmvevHm6+OKL9dFHH2ngwIEe673wwgu1cOFC3Xzzza5uIqeysjINHjxYlZWV+v3vf6+2bdtqwYIFuvrqq7VkyRJde+21bq/1+OOPy2q16r777lNVVZWsVmsDP6m6Dhw4oAMHDqhbt251nhs5cqQuu+wybdmyRV27dpV07Hf2uuuua/D3c//+/XW2xcTEBLSVyluDBg1S165dtWLFipCfGz8zgACYN2+eIcn44IMPjL179xo7d+40Fi1aZLRt29aIj483fvjhB8MwDGPlypWGJGPlypWGYRhGdXW10b59e6NPnz7GkSNHXK/3zjvvGJKMSZMmubaNGzfO8PQru3fvXkOSUVBQUOe5goKCOsdIMqxWq/H999+7tn3xxReGJOP55593bbvqqquMZs2aGbt27XJt++6774yYmBiPdZxsyJAhhiTj1VdfdW3btGmTIckwm83Gf/7zH9f2999/35BkzJs3z7Xt1ltvNVJTU419+/a5ve7w4cONxMREo7Ky0jAMw6itrTWqqqrc9jlw4ICRnJxs3HLLLa5tJSUlhiSjbdu2xv79+13b//73vxuSjH/+858Nvp+jR48adrvdbVtJSYlhs9mMyZMnu7Y5f8ZnnXWWW13PPvusIcnYsGGDYRjHf/YZGRlu+7300kuGJGPIkCEN1rN58+Y6PzPDMIyxY8caLVq0cF0fwzDcvnaeu0+fPsbFF1/stt35s/n666/rnO/k37GTX9MwDKO4uNiQZPy///f/XNucfxvZ2dmGw+Fwbb/33nsNi8ViHDx40DAMwzh48KDRsmVLIysry+1vwTAM13EOh8M488wzjZycHLfXqqysNM444wzj0ksvrVOTp/cxbtw4t2333HOPIcn46KOPXNsOHTpknHHGGUZ6errr5+782Xbp0sXj+6/vfLfeequxd+9eY8+ePcZ///tf45JLLjEkGU8//bRrv86dOxtDhw41amtrjZSUFOPxxx83DMMwNm7caEgy/u///s91LT/99FPXcc6/cU+PHj16eFWjJ/X9e2MYx/+WnnrqqXqPv+aaawxJRnl5ud81wH90SyGgsrOz1a5dO6WlpWn48OFq0aKF3nrrLXXs2NHj/p999pn27NmjsWPHuo1pGDp0qHr27FmnWTyQdTr/Vygda+lJSEjQ1q1bJUl2u10ffPCBhg0b5tYi0a1bN7cxAo1p0aKFWwtCjx491KpVK5111lnKyspybXd+7Ty/YRh68803ddVVV8kwDO3bt8/1yMnJUXl5udatWydJslgsrv85OxwO7d+/X7W1terfv79rnxPl5uaqdevWru8vuOACt3PXx2azucZV2O12/fjjj2rRooV69Ojh8Tx5eXlu/6M/+TzOn/3vfvc7t/3GjBmjxMTEBmuRjo3jyMjIcGv5sNvtWrJkia666iq37oUTvz5w4IDKy8t1wQUXeKx7yJAh6tWrV6PnP/E1a2pq9OOPP6pbt25q1aqVx9e944473LqBLrjgAtntdm3fvl2StGLFCh06dEgPPvhgnfE9zuM+//xzfffddxo5cqR+/PFH1+/E4cOHdckll+jDDz+Uw+FotPaTLVu2TAMHDtT555/v2taiRQvdcccd2rZtmzZu3Oi2/+jRo33qvpkzZ47atWun9u3bKysry9VF52x9PJHFYtENN9yg1157TdKxgcRpaWmu35/6vPnmm1qxYoXbY968eV7XGGjOwc2HDh0KWw2nM7qlEFAzZsxQ9+7dFRMTo+TkZPXo0aPBgYbOf9h79OhR57mePXtq9erVQanzF7/4RZ1trVu31oEDByQdG3R55MgRj83mnrbVp1OnTnXG5yQmJtbpcnF+mDvPv3fvXh08eFAvvfSSXnrpJY+vvWfPHtfXCxYs0NNPP61NmzappqbGtf2MM86oc9zJ790ZdJznro/D4dCzzz6rF198USUlJbLb7a7nPHU7NnYe58/+zDPPdNsvNjZWXbp0abAWp9zcXD300EPatWuXOnbsqFWrVmnPnj3Kzc112++dd97RE088oc8//1xVVVWu7Z7GTnm6Zp4cOXJEhYWFmjdvnnbt2iXDMFzPlZeX19m/sevh7MLt06dPvef87rvvJB0LF/UpLy93C6/e2L59u1vYdjrrrLNcz59Yl7fXyOmaa67R+PHjZTKZ1LJlS/Xu3VvNmzevd/+RI0fqueee0xdffKFXX31Vw4cPb3Sc24UXXnjKA4oD6aeffpIkVzczQotwg4AaOHCga7ZUJLNYLB63n/gBFczzNHZ+5/++b7rppno/yJzjiv72t79pzJgxGjZsmO6//361b99eFotFhYWFrg9MX85dnylTpuiPf/yjbrnlFj3++ONq06aNzGaz7rnnHo+tBaG4xrm5uZo4caLeeOMN3XPPPXr99deVmJjotnbLRx99pKuvvloXXnihXnzxRaWmpio2Nlbz5s2rM3BVktctEhMmTNC8efN0zz33aNCgQUpMTJTJZNLw4cODdj2cr/vUU0/VGUvl5Gk6dKD5Oui2U6dO9a5f40lWVpa6du2qe+65RyUlJa7xRk3JV199pfbt2yshISHcpZyWCDcIq86dO0uSNm/e7JpZ5bR582bX85Ln/2U3tP1UtG/fXnFxcR5n93jaFmjt2rVTy5YtZbfbG/1QWLJkibp06aKlS5e6XYuCgoKA1rRkyRL96le/0pw5c9y2Hzx40K//MTt/tt99953bz76mpkYlJSVeTaE944wzNHDgQC1evFjjx4/X0qVLNWzYMNlsNtc+b775puLi4vT++++7bT/VLoslS5Zo9OjRevrpp13bjh49qoMHD/r1es5u0q+++qre1kHnPgkJCT6FhcZ07txZmzdvrrN906ZNrudDbcSIEXriiSd01lln1RvkIlVxcbG2bNlSZ5o4QocxNwir/v37q3379po1a5Zbd8F7772nb775xm0mi7MZ++QPD+esFH8/VDyxWCzKzs7W22+/rd27d7u2f//993rvvfcCdp6Gzv/b3/5Wb775pr766qs6z584hdjZInBiC8B///tfFRcXB7ymk1sZ3njjDbcp+77o37+/2rVrp1mzZqm6utq1ff78+T79LHNzc/Wf//xHc+fO1b59++p0SVksFplMJrdutG3btuntt9/2q+4TX/fk6/H888+7nccXl112mVq2bKnCwsI6C9U5z5OZmamuXbtq2rRprm6PE3maWu6NK6+8UmvWrHH7nTl8+LBeeuklpaenezUGKdBuu+02FRQUuIXHpmD79u0aM2aMrFar7r///nCXc9qi5QZhFRsbqz//+c/Ky8vTkCFDNGLECNdU8PT0dN17772ufTMzMyVJv//975WTkyOLxaLhw4crPj5evXr10uLFi9W9e3e1adNGffr0aXDsgjceffRR/etf/9J5552nu+66S3a7XS+88IL69Omjzz///JRe2xtTp07VypUrlZWVpdtvv129evXS/v37tW7dOn3wwQeuqa+//vWvtXTpUl177bUaOnSoSkpKNGvWLPXq1cvjB6C/fv3rX2vy5MnKy8vT4MGDtWHDBr3yyitej485WWxsrJ544gndeeeduvjii5Wbm6uSkhLNmzfPp9e84YYbdN999+m+++5TmzZt6rRoDB06VM8884wuv/xyjRw5Unv27NGMGTPUrVs3ffnll37VLh27HgsXLlRiYqJ69eql4uJiffDBBw0ue9CQhIQE/eUvf9Ftt92mAQMGaOTIkWrdurW++OILVVZWasGCBTKbzXr55Zd1xRVXqHfv3srLy1PHjh21a9curVy5UgkJCfrnP//p87kffPBBvfbaa7riiiv0+9//Xm3atNGCBQtUUlKiN998s8Fxc8HSuXNnn+7ltWTJEo9dcpdeeqmSk5O9eo3t27e7lppw3uLhiSeecNVz8803u+2/bt06/e1vf5PD4dDBgwf16aef6s0335TJZNLChQsbXZICQRSeSVqINp6maHpy8lRwp8WLFxv9+vUzbDab0aZNG+PGG290TR93qq2tNSZMmGC0a9fOMJlMbtM0P/nkEyMzM9OwWq1uU3brmwp+8lRYwzg2FXX06NFu24qKiox+/foZVqvV6Nq1q/Hyyy8bf/jDH4y4uLhGrsixqeC9e/f2eJ6hQ4fW2e6prrKyMmPcuHFGWlqaERsba6SkpBiXXHKJ8dJLL7n2cTgcxpQpU4zOnTsbNpvN6Nevn/HOO+8Yo0ePNjp37uzar6Hpqydes/ocPXrU+MMf/mCkpqYa8fHxxnnnnWcUFxcbQ4YMcZu27fwZv/HGG27HO89/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xqpVq4zPP/88xJU3Lb5e51deecWw2WzGK6+8YpSUlBjvv/++kZqaatx7770hrrxpWbZsmfHwww8bS5cuNSQZb731VoP7b9261WjWrJmRn59vbNy40Xj++ecNi8ViLF++PKh1Em58MHDgQGPcuHGu7+12u9GhQwejsLDQ4/433HCDMXToULdtWVlZxp133hnUOps6X6/zyWpra42WLVsaCxYsCFaJUcGf61xbW2sMHjzYePnll43Ro0cTbrzg63WeOXOm0aVLF6O6ujpUJUYFX6/zuHHjjIsvvthtW35+vnHeeecFtc5o4k24+f/+v//P6N27t9u23NxcIycnJ4iVGQbdUl6qrq7W2rVrlZ2d7dpmNpuVnZ2t4uJij8cUFxe77S9JOTk59e4P/67zySorK1VTU6M2bdoEq8wmz9/rPHnyZLVv31633nprKMps8vy5zv/4xz80aNAgjRs3TsnJyerTp4+mTJkiu90eqrKbHH+u8+DBg7V27VpX19XWrVu1bNkyXXnllSGp+XQRrs/B0+7Gmf7at2+f7Ha7kpOT3bYnJydr06ZNHo8pLS31uH9paWnQ6mzq/LnOJ3vggQfUoUOHOn9QOM6f67x69WrNmTNHn3/+eQgqjA7+XOetW7fq3//+t2688UYtW7ZM33//vcaOHauamhoVFBSEouwmx5/rPHLkSO3bt0/nn3++DMNQbW2tfve73+mhhx4KRcmnjfo+BysqKnTkyBHFx8cH5by03CCqTJ06VYsWLdJbb72luLi4cJcTNQ4dOqSbb75Zs2fPVlJSUrjLiWoOh0Pt27fXSy+9pMzMTOXm5urhhx/WrFmzwl1aVFm1apWmTJmiF198UevWrdPSpUv17rvv6vHHHw93aQgAWm68lJSUJIvForKyMrftZWVlSklJ8XhMSkqKT/vDv+vsNG3aNE2dOlUffPCBzjnnnGCW2eT5ep23bNmibdu26aqrrnJtczgckqSYmBht3rxZXbt2DW7RTZA/v8+pqamKjY2VxWJxbTvrrLNUWlqq6upqWa3WoNbcFPlznf/4xz/q5ptv1m233SZJOvvss3X48GHdcccdevjhh2U283//QKjvczAhISForTYSLTdes1qtyszMVFFRkWubw+FQUVGRBg0a5PGYQYMGue0vSStWrKh3f/h3nSXpySef1OOPP67ly5erf//+oSi1SfP1Ovfs2VMbNmzQ559/7npcffXV+tWvfqXPP/9caWlpoSy/yfDn9/m8887T999/7wqPkvTtt98qNTWVYFMPf65zZWVlnQDjDJQGt1wMmLB9DgZ1uHKUWbRokWGz2Yz58+cbGzduNO644w6jVatWRmlpqWEYhnHzzTcbDz74oGv/jz/+2IiJiTGmTZtmfPPNN0ZBQQFTwb3g63WeOnWqYbVajSVLlhj/+9//XI9Dhw6F6y00Cb5e55MxW8o7vl7nHTt2GC1btjTGjx9vbN682XjnnXeM9u3bG0888US43kKT4Ot1LigoMFq2bGm89tprxtatW41//etfRteuXY0bbrghXG+hSTh06JCxfv16Y/369YYk45lnnjHWr19vbN++3TAMw3jwwQeNm2++2bW/cyr4/fffb3zzzTfGjBkzmAoeiZ5//nnjF7/4hWG1Wo2BAwca//nPf1zPDRkyxBg9erTb/q+//rrRvXt3w2q1Gr179zbefffdEFfcNPlynTt37mxIqvMoKCgIfeFNjK+/zyci3HjP1+v8ySefGFlZWYbNZjO6dOli/OlPfzJqa2tDXHXT48t1rqmpMR599FGja9euRlxcnJGWlmaMHTvWOHDgQOgLb0JWrlzp8d9b57UdPXq0MWTIkDrHZGRkGFar1ejSpYsxb968oNdpMgza3wAAQPRgzA0AAIgqhBsAABBVCDcAACCqEG4AAEBUIdwAAICoQrgBAABRhXADAACiCuEGAABEFcINAACIKoQbAAAQVQg3AJq8vXv3KiUlRVOmTHFt++STT2S1WuvckRhA9OPeUgCiwrJlyzRs2DB98skn6tGjhzIyMnTNNdfomWeeCXdpAEKMcAMgaowbN04ffPCB+vfvrw0bNujTTz+VzWYLd1kAQoxwAyBqHDlyRH369NHOnTu1du1anX322eEuCUAYMOYGQNTYsmWLdu/eLYfDoW3btoW7HABhQssNgKhQXV2tgQMHKiMjQz169ND06dO1YcMGtW/fPtylAQgxwg2AqHD//fdryZIl+uKLL9SiRQsNGTJEiYmJeuedd8JdGoAQo1sKQJO3atUqTZ8+XQsXLlRCQoLMZrMWLlyojz76SDNnzgx3eQBCjJYbAAAQVWi5AQAAUYVwAwAAogrhBgAARBXCDQAAiCqEGwAAEFUINwAAIKoQbgAAQFQh3AAAgKhCuAEAAFGFcAMAAKIK4QYAAEQVwg0AAIgq/z8Bt7nTxSf5nwAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "t_idx = 1\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": 16,
   "id": "6fd587e6-4cf8-4cc4-92bd-ac09c7be790a",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<function datasets.PME_1D.get_mass_rhs_func.<locals>.mass_rhs_func(inputs)>"
      ]
     },
     "execution_count": 16,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "dataset_class.get_mass_rhs_func(x=x_train)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "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": 18,
   "id": "b339b6e5",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "device(type='cpu')"
      ]
     },
     "execution_count": 18,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "x_train.device"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "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": 20,
   "id": "b202791b-cc81-4a0a-b82b-db3c5a831a77",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<function datasets.PME_1D.get_mass_rhs_func.<locals>.mass_rhs_func(inputs)>"
      ]
     },
     "execution_count": 20,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "mass_rhs_func"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "id": "d71ec298-4bbc-4186-a938-8675f0526890",
   "metadata": {},
   "outputs": [],
   "source": [
    "import time"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "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": 22,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "grid"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "id": "2165975d",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[0, -1, 5]"
      ]
     },
     "execution_count": 23,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "tpred"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "id": "1ad2c532",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([ 0, -1,  5])"
      ]
     },
     "execution_count": 24,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "torch.tensor(tpred)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "id": "18a1c208-a4df-4f27-8dad-f2466164e855",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([160, 100, 20, 1])"
      ]
     },
     "execution_count": 25,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "x_train.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "id": "6ff40f4d-d8b7-45f7-8e3a-8115564540df",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 0: Train loss=0.339774, Validation loss=0.297572 (saved)\n",
      "Epoch 1: Train loss=0.262302, Validation loss=0.211976 (saved)\n",
      "Epoch 2: Train loss=0.190790, Validation loss=0.164604 (saved)\n",
      "Epoch 3: Train loss=0.155245, Validation loss=0.136212 (saved)\n",
      "Epoch 4: Train loss=0.111748, Validation loss=0.072202 (saved)\n",
      "Epoch 5: Train loss=0.051281, Validation loss=0.045226 (saved)\n",
      "Epoch 6: Train loss=0.033732, Validation loss=0.023927 (saved)\n",
      "Epoch 7: Train loss=0.025498, Validation loss=0.019853 (saved)\n",
      "Epoch 8: Train loss=0.021430, Validation loss=0.016612 (saved)\n",
      "Epoch 9: Train loss=0.017913, Validation loss=0.016423 (saved)\n",
      "Epoch 10: Train loss=0.015524, Validation loss=0.015511 (saved)\n",
      "Epoch 11: Train loss=0.016071, Validation loss=0.024900 \n",
      "Epoch 12: Train loss=0.017915, Validation loss=0.013986 (saved)\n",
      "Epoch 13: Train loss=0.014804, Validation loss=0.015110 \n",
      "Epoch 14: Train loss=0.012612, Validation loss=0.010383 (saved)\n",
      "Epoch 15: Train loss=0.011024, Validation loss=0.012103 \n",
      "Epoch 16: Train loss=0.012892, Validation loss=0.018775 \n",
      "Epoch 17: Train loss=0.012466, Validation loss=0.012859 \n",
      "Epoch 18: Train loss=0.009401, Validation loss=0.011628 \n",
      "Epoch 19: Train loss=0.010880, Validation loss=0.007250 (saved)\n",
      "Epoch 20: Train loss=0.009484, Validation loss=0.007990 \n",
      "Epoch 21: Train loss=0.009954, Validation loss=0.007208 (saved)\n",
      "Epoch 22: Train loss=0.008183, Validation loss=0.016058 \n",
      "Epoch 23: Train loss=0.012075, Validation loss=0.020032 \n",
      "Epoch 24: Train loss=0.015853, Validation loss=0.019182 \n",
      "Epoch 25: Train loss=0.016753, Validation loss=0.011890 \n",
      "Epoch 26: Train loss=0.012377, Validation loss=0.011846 \n",
      "Epoch 27: Train loss=0.010306, Validation loss=0.007640 \n",
      "Epoch 28: Train loss=0.007305, Validation loss=0.006469 (saved)\n",
      "Epoch 29: Train loss=0.010621, Validation loss=0.006104 (saved)\n",
      "Epoch 30: Train loss=0.008969, Validation loss=0.006399 \n",
      "Epoch 31: Train loss=0.009851, Validation loss=0.007288 \n",
      "Epoch 32: Train loss=0.009286, Validation loss=0.007492 \n",
      "Epoch 33: Train loss=0.009352, Validation loss=0.013105 \n",
      "Epoch 34: Train loss=0.008702, Validation loss=0.009038 \n",
      "Epoch 35: Train loss=0.007053, Validation loss=0.005893 (saved)\n",
      "Epoch 36: Train loss=0.006901, Validation loss=0.009905 \n",
      "Epoch 37: Train loss=0.009869, Validation loss=0.006580 \n",
      "Epoch 38: Train loss=0.006936, Validation loss=0.006733 \n",
      "Epoch 39: Train loss=0.007364, Validation loss=0.007584 \n",
      "Epoch 40: Train loss=0.007558, Validation loss=0.006782 \n",
      "Epoch 41: Train loss=0.005467, Validation loss=0.010364 \n",
      "Epoch 42: Train loss=0.008977, Validation loss=0.005965 \n",
      "Epoch 43: Train loss=0.007503, Validation loss=0.006432 \n",
      "Epoch 44: Train loss=0.006921, Validation loss=0.008527 \n",
      "Epoch 45: Train loss=0.007834, Validation loss=0.005612 (saved)\n",
      "Epoch 46: Train loss=0.008063, Validation loss=0.005561 (saved)\n",
      "Epoch 47: Train loss=0.006577, Validation loss=0.005608 \n",
      "Epoch 48: Train loss=0.007001, Validation loss=0.004872 (saved)\n",
      "Epoch 49: Train loss=0.006855, Validation loss=0.010086 \n",
      "Epoch 50: Train loss=0.007455, Validation loss=0.008621 \n",
      "Epoch 51: Train loss=0.008307, Validation loss=0.005105 \n",
      "Epoch 52: Train loss=0.006029, Validation loss=0.004762 (saved)\n",
      "Epoch 53: Train loss=0.005108, Validation loss=0.004687 (saved)\n",
      "Epoch 54: Train loss=0.004674, Validation loss=0.004114 (saved)\n",
      "Epoch 55: Train loss=0.004117, Validation loss=0.004015 (saved)\n",
      "Epoch 56: Train loss=0.004035, Validation loss=0.003791 (saved)\n",
      "Epoch 57: Train loss=0.004803, Validation loss=0.004239 \n",
      "Epoch 58: Train loss=0.004667, Validation loss=0.003936 \n",
      "Epoch 59: Train loss=0.004975, Validation loss=0.003994 \n",
      "Epoch 60: Train loss=0.004495, Validation loss=0.003598 (saved)\n",
      "Epoch 61: Train loss=0.004581, Validation loss=0.004839 \n",
      "Epoch 62: Train loss=0.004763, Validation loss=0.003627 \n",
      "Epoch 63: Train loss=0.004443, Validation loss=0.003989 \n",
      "Epoch 64: Train loss=0.004554, Validation loss=0.003734 \n",
      "Epoch 65: Train loss=0.004878, Validation loss=0.005089 \n",
      "Epoch 66: Train loss=0.004626, Validation loss=0.005023 \n",
      "Epoch 67: Train loss=0.004673, Validation loss=0.004767 \n",
      "Epoch 68: Train loss=0.004740, Validation loss=0.006113 \n",
      "Epoch 69: Train loss=0.004653, Validation loss=0.003809 \n",
      "Epoch 70: Train loss=0.004218, Validation loss=0.003854 \n",
      "Epoch 71: Train loss=0.004357, Validation loss=0.003775 \n",
      "Epoch 72: Train loss=0.004542, Validation loss=0.006462 \n",
      "Epoch 73: Train loss=0.004719, Validation loss=0.003780 \n",
      "Epoch 74: Train loss=0.004223, Validation loss=0.007582 \n",
      "Epoch 75: Train loss=0.004937, Validation loss=0.004196 \n",
      "Epoch 76: Train loss=0.004202, Validation loss=0.004339 \n",
      "Epoch 77: Train loss=0.004926, Validation loss=0.003376 (saved)\n",
      "Epoch 78: Train loss=0.004020, Validation loss=0.003893 \n",
      "Epoch 79: Train loss=0.004326, Validation loss=0.004823 \n",
      "Epoch 80: Train loss=0.004350, Validation loss=0.004804 \n",
      "Epoch 81: Train loss=0.004304, Validation loss=0.004353 \n",
      "Epoch 82: Train loss=0.003726, Validation loss=0.004336 \n",
      "Epoch 83: Train loss=0.004482, Validation loss=0.004386 \n",
      "Epoch 84: Train loss=0.005036, Validation loss=0.005602 \n",
      "Epoch 85: Train loss=0.003843, Validation loss=0.004802 \n",
      "Epoch 86: Train loss=0.004839, Validation loss=0.003239 (saved)\n",
      "Epoch 87: Train loss=0.004247, Validation loss=0.003358 \n",
      "Epoch 88: Train loss=0.003709, Validation loss=0.005032 \n",
      "Epoch 89: Train loss=0.004366, Validation loss=0.003235 (saved)\n",
      "Epoch 90: Train loss=0.004231, Validation loss=0.003500 \n",
      "Epoch 91: Train loss=0.004218, Validation loss=0.006099 \n",
      "Epoch 92: Train loss=0.004790, Validation loss=0.003451 \n",
      "Epoch 93: Train loss=0.004237, Validation loss=0.004019 \n",
      "Epoch 94: Train loss=0.003994, Validation loss=0.005508 \n",
      "Epoch 95: Train loss=0.004714, Validation loss=0.003460 \n",
      "Epoch 96: Train loss=0.003851, Validation loss=0.005231 \n",
      "Epoch 97: Train loss=0.004403, Validation loss=0.004109 \n",
      "Epoch 98: Train loss=0.004847, Validation loss=0.006690 \n",
      "Epoch 99: Train loss=0.005897, Validation loss=0.005201 \n",
      "Epoch 100: Train loss=0.004467, Validation loss=0.004093 \n",
      "Epoch 101: Train loss=0.003674, Validation loss=0.003427 \n",
      "Epoch 102: Train loss=0.003358, Validation loss=0.003387 \n",
      "Epoch 103: Train loss=0.003249, Validation loss=0.003119 (saved)\n",
      "Epoch 104: Train loss=0.003228, Validation loss=0.003476 \n",
      "Epoch 105: Train loss=0.003112, Validation loss=0.003016 (saved)\n",
      "Epoch 106: Train loss=0.002962, Validation loss=0.002828 (saved)\n",
      "Epoch 107: Train loss=0.002911, Validation loss=0.002807 (saved)\n",
      "Epoch 108: Train loss=0.002854, Validation loss=0.002754 (saved)\n",
      "Epoch 109: Train loss=0.002894, Validation loss=0.003398 \n",
      "Epoch 110: Train loss=0.003104, Validation loss=0.003601 \n",
      "Epoch 111: Train loss=0.003116, Validation loss=0.003069 \n",
      "Epoch 112: Train loss=0.002821, Validation loss=0.002767 \n",
      "Epoch 113: Train loss=0.002965, Validation loss=0.002618 (saved)\n",
      "Epoch 114: Train loss=0.002905, Validation loss=0.002629 \n",
      "Epoch 115: Train loss=0.003032, Validation loss=0.004125 \n",
      "Epoch 116: Train loss=0.003256, Validation loss=0.004382 \n",
      "Epoch 117: Train loss=0.003225, Validation loss=0.002833 \n",
      "Epoch 118: Train loss=0.003171, Validation loss=0.003388 \n",
      "Epoch 119: Train loss=0.003135, Validation loss=0.002672 \n",
      "Epoch 120: Train loss=0.002934, Validation loss=0.003890 \n",
      "Epoch 121: Train loss=0.002975, Validation loss=0.002710 \n",
      "Epoch 122: Train loss=0.003128, Validation loss=0.003029 \n",
      "Epoch 123: Train loss=0.002832, Validation loss=0.002805 \n",
      "Epoch 124: Train loss=0.003170, Validation loss=0.003494 \n",
      "Epoch 125: Train loss=0.003264, Validation loss=0.004721 \n",
      "Epoch 126: Train loss=0.003883, Validation loss=0.002762 \n",
      "Epoch 127: Train loss=0.003296, Validation loss=0.003482 \n",
      "Epoch 128: Train loss=0.003386, Validation loss=0.003464 \n",
      "Epoch 129: Train loss=0.003294, Validation loss=0.003103 \n",
      "Epoch 130: Train loss=0.003146, Validation loss=0.003001 \n",
      "Epoch 131: Train loss=0.003200, Validation loss=0.002713 \n",
      "Epoch 132: Train loss=0.003071, Validation loss=0.002628 \n",
      "Epoch 133: Train loss=0.003041, Validation loss=0.002731 \n",
      "Epoch 134: Train loss=0.003051, Validation loss=0.002867 \n",
      "Epoch 135: Train loss=0.003354, Validation loss=0.002941 \n",
      "Epoch 136: Train loss=0.003106, Validation loss=0.002727 \n",
      "Epoch 137: Train loss=0.002927, Validation loss=0.002577 (saved)\n",
      "Epoch 138: Train loss=0.003030, Validation loss=0.002954 \n",
      "Epoch 139: Train loss=0.002739, Validation loss=0.002872 \n",
      "Epoch 140: Train loss=0.003164, Validation loss=0.002761 \n",
      "Epoch 141: Train loss=0.003605, Validation loss=0.003192 \n",
      "Epoch 142: Train loss=0.003311, Validation loss=0.002980 \n",
      "Epoch 143: Train loss=0.003286, Validation loss=0.003263 \n",
      "Epoch 144: Train loss=0.003948, Validation loss=0.004552 \n",
      "Epoch 145: Train loss=0.004662, Validation loss=0.003746 \n",
      "Epoch 146: Train loss=0.004092, Validation loss=0.003377 \n",
      "Epoch 147: Train loss=0.004154, Validation loss=0.004086 \n",
      "Epoch 148: Train loss=0.004076, Validation loss=0.003732 \n",
      "Epoch 149: Train loss=0.003351, Validation loss=0.002895 \n",
      "Epoch 150: Train loss=0.002925, Validation loss=0.002905 \n",
      "Epoch 151: Train loss=0.002773, Validation loss=0.002692 \n",
      "Epoch 152: Train loss=0.002657, Validation loss=0.002578 \n",
      "Epoch 153: Train loss=0.002657, Validation loss=0.002582 \n",
      "Epoch 154: Train loss=0.002601, Validation loss=0.002572 (saved)\n",
      "Epoch 155: Train loss=0.002566, Validation loss=0.002527 (saved)\n",
      "Epoch 156: Train loss=0.002579, Validation loss=0.002663 \n",
      "Epoch 157: Train loss=0.002601, Validation loss=0.002494 (saved)\n",
      "Epoch 158: Train loss=0.002533, Validation loss=0.002528 \n",
      "Epoch 159: Train loss=0.002498, Validation loss=0.002443 (saved)\n",
      "Epoch 160: Train loss=0.002515, Validation loss=0.002419 (saved)\n",
      "Epoch 161: Train loss=0.002455, Validation loss=0.002435 \n",
      "Epoch 162: Train loss=0.002443, Validation loss=0.002412 (saved)\n",
      "Epoch 163: Train loss=0.002455, Validation loss=0.002421 \n",
      "Epoch 164: Train loss=0.002545, Validation loss=0.002539 \n",
      "Epoch 165: Train loss=0.002467, Validation loss=0.002523 \n",
      "Epoch 166: Train loss=0.002547, Validation loss=0.002432 \n",
      "Epoch 167: Train loss=0.002523, Validation loss=0.002470 \n",
      "Epoch 168: Train loss=0.002467, Validation loss=0.002651 \n",
      "Epoch 169: Train loss=0.002463, Validation loss=0.002421 \n",
      "Epoch 170: Train loss=0.002491, Validation loss=0.002496 \n",
      "Epoch 171: Train loss=0.002499, Validation loss=0.002401 (saved)\n",
      "Epoch 172: Train loss=0.002386, Validation loss=0.002404 \n",
      "Epoch 173: Train loss=0.002482, Validation loss=0.002492 \n",
      "Epoch 174: Train loss=0.002376, Validation loss=0.002330 (saved)\n",
      "Epoch 175: Train loss=0.002408, Validation loss=0.002332 \n",
      "Epoch 176: Train loss=0.002480, Validation loss=0.002600 \n",
      "Epoch 177: Train loss=0.002782, Validation loss=0.002489 \n",
      "Epoch 178: Train loss=0.002735, Validation loss=0.002344 \n",
      "Epoch 179: Train loss=0.002499, Validation loss=0.002318 (saved)\n",
      "Epoch 180: Train loss=0.002399, Validation loss=0.002437 \n",
      "Epoch 181: Train loss=0.002419, Validation loss=0.002317 (saved)\n",
      "Epoch 182: Train loss=0.002406, Validation loss=0.002322 \n",
      "Epoch 183: Train loss=0.002716, Validation loss=0.003118 \n",
      "Epoch 184: Train loss=0.003433, Validation loss=0.003725 \n",
      "Epoch 185: Train loss=0.003107, Validation loss=0.004008 \n",
      "Epoch 186: Train loss=0.003428, Validation loss=0.002776 \n",
      "Epoch 187: Train loss=0.003047, Validation loss=0.003888 \n",
      "Epoch 188: Train loss=0.003481, Validation loss=0.004751 \n",
      "Epoch 189: Train loss=0.004330, Validation loss=0.003606 \n",
      "Epoch 190: Train loss=0.003824, Validation loss=0.003337 \n",
      "Epoch 191: Train loss=0.003463, Validation loss=0.004279 \n",
      "Epoch 192: Train loss=0.003692, Validation loss=0.003178 \n",
      "Epoch 193: Train loss=0.003028, Validation loss=0.003339 \n",
      "Epoch 194: Train loss=0.002850, Validation loss=0.003379 \n",
      "Epoch 195: Train loss=0.002719, Validation loss=0.002480 \n",
      "Epoch 196: Train loss=0.002481, Validation loss=0.002589 \n",
      "Epoch 197: Train loss=0.002581, Validation loss=0.002527 \n",
      "Epoch 198: Train loss=0.002578, Validation loss=0.002729 \n",
      "Epoch 199: Train loss=0.002460, Validation loss=0.002444 \n",
      "Finished training with best train loss: 0.002341 and validation loss: 0.002317\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": 27,
   "id": "61efde5e-a31a-4ec9-87ee-17cf2402b98d",
   "metadata": {},
   "outputs": [],
   "source": [
    "# torch.save(model.state_dict(), \"./pme_e2e_var_update_crps.pt\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "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": 29,
   "id": "41b11e17-4840-42fe-9a5b-34d65f0a77d7",
   "metadata": {},
   "outputs": [],
   "source": [
    "from nonlinear_projection import project_and_stats\n",
    "\n",
    "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",
    "            # nf,nx,nt,_ = _mu.shape\n",
    "\n",
    "            # _mu = _mu.view(nf, -1)\n",
    "            # _var = _var.view(nf, -1)\n",
    "            # _m = x.view(nf, -1)\n",
    "\n",
    "            # # print(_m)\n",
    "\n",
    "            # u_proj, u_var = project_and_stats(torch.relu(_mu), _var, _m, model.full_residual, max_iter=30)\n",
    "\n",
    "            # # print(u_proj, u_var)\n",
    "\n",
    "            # if  u_proj.isnan().any().item() or  u_var.isnan().any().item():\n",
    "            #     print(\"any NaN in new_mu?\", u_proj.isnan().any().item())\n",
    "            #     # print(\"min new_var before clamp:\", u_var.min().item())\n",
    "            #     print(\"any NaN in new_var before clamp?\", u_var.isnan().any().item())\n",
    "            #     # new_var = new_var.clamp(min=eps)\n",
    "            #     # print(\"min new_var after clamp:\", new_var.min().item())\n",
    "\n",
    "            # out = (u_proj.view(nf,nx,nt,1), u_var.view(nf,nx,nt,1))\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": 30,
   "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.00015381800185423344\n",
      "n-MeRCI: 1.4556134939193726\n",
      "RMSCE: 0.2141520082950592\n",
      "Cerr: 0.016453618183732033\n",
      "In-domain results\n",
      "MSE: 0.0001540168677456677\n",
      "n-MeRCI: 1.2143707275390625\n",
      "RMSCE: 0.21474553644657135\n",
      "ProbConserv Results\n",
      "MSE: 0.00018028581514954568\n",
      "n-MeRCI: 1.2749927043914795\n",
      "RMSCE: 0.21800251305103302\n",
      "Cerr: 0.016252165660262108\n",
      "Prob_Cerr: 3.2295832852469175e-07\n",
      "Here\n",
      "\n",
      "\n",
      "Out-of-domain results\n",
      "MSE: 0.00016222022473812105\n",
      "n-MeRCI: 1.380208969116211\n",
      "RMSCE: 0.21964138746261597\n",
      "ProbConserv Results\n",
      "MSE: 0.00018453938886523247\n",
      "n-MeRCI: 1.4384989738464355\n",
      "RMSCE: 0.22281640768051147\n",
      "Cerr: 0.014854798093438148\n",
      "Prob_Cerr: 3.150005056795635e-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": 31,
   "id": "00406482-eb2b-41df-b458-1d230c92ff1a",
   "metadata": {},
   "outputs": [],
   "source": [
    "def compute_statistics(\n",
    "    model, \n",
    "    x_data, \n",
    "    y_data, \n",
    "    t, \n",
    "    tpred, \n",
    "    grid, \n",
    "    dataset_class, \n",
    "    apply_probconserv=False, \n",
    "    plot=False,\n",
    "    x_data_test=None, \n",
    "    y_data_test=None,\n",
    "    return_latex=False,\n",
    "    name=\"Model\"\n",
    "):\n",
    "    import torch\n",
    "    import utils\n",
    "    import probconserv\n",
    "    import matplotlib.pyplot as plt\n",
    "\n",
    "    device = next(model.parameters()).device\n",
    "    x_data = x_data.to(device)\n",
    "\n",
    "    with torch.no_grad():\n",
    "        out = model(x_data)\n",
    "\n",
    "    if isinstance(out, tuple):\n",
    "        mu, var = out[0].cpu(), out[1].cpu()\n",
    "        std = torch.sqrt(var)\n",
    "    else:\n",
    "        mu = out.cpu()\n",
    "        std = torch.zeros_like(mu)\n",
    "        var = torch.square(std)\n",
    "\n",
    "    x_cpu = x_data.cpu()\n",
    "    mass_rhs_func = dataset_class.get_mass_rhs_func(x=x_cpu)\n",
    "\n",
    "    if apply_probconserv:\n",
    "        new_mu, new_std, _, mass_rhs = probconserv.apply_constraint(\n",
    "            mu=mu[:, :, :, 0],\n",
    "            std=std[:, :, :, 0],\n",
    "            mass_rhs_func=mass_rhs_func,\n",
    "            t=t,\n",
    "            tpred=tpred,\n",
    "            grid_train=grid,\n",
    "            precis_g=float('inf'),\n",
    "            second_deriv_alpha=None,\n",
    "        )\n",
    "        mu = new_mu.unsqueeze(-1)\n",
    "        std = new_std.unsqueeze(-1)\n",
    "        var = torch.square(std)\n",
    "        cerr = (probconserv.get_empirical_mass_rhs(mu[:, :, :, 0]) - mass_rhs).abs().sum(dim=-1)\n",
    "    else:\n",
    "        t_sliced = t[slice(*tpred)]\n",
    "        ts = repeat(t_sliced, \"nt -> nf nt\", nf=mu.shape[0])\n",
    "        xs = repeat(grid, \"nx -> nf nx\", nf=mu.shape[0])\n",
    "        inputs = meshgrid(ts, xs)\n",
    "        cerr = (probconserv.get_empirical_mass_rhs(mu[:, :, :, 0]) - mass_rhs_func(inputs)).abs().sum(dim=-1)\n",
    "\n",
    "    stats = utils.compute_all_metrics_avg((mu, var), y_data, {})\n",
    "    stats[\"nMeRCI_all\"] = utils.compute_nMeRCI(mu, var, y_data).item()\n",
    "    stats[\"rmsce_all\"] = utils.compute_rmsce(mu, var, y_data).item()\n",
    "    stats[\"cerr_by_example\"] = cerr.tolist()\n",
    "    stats[\"mcerr\"] = cerr.mean().item()\n",
    "\n",
    "    # --- Test dataset ---\n",
    "    test_stats = None\n",
    "    if x_data_test is not None and y_data_test is not None:\n",
    "        x_data_test = x_data_test.to(device)\n",
    "        with torch.no_grad():\n",
    "            test_out = model(x_data_test)\n",
    "\n",
    "        if isinstance(test_out, tuple):\n",
    "            mu_test, var_test = test_out[0].cpu(), test_out[1].cpu()\n",
    "            std_test = torch.sqrt(var_test)\n",
    "        else:\n",
    "            mu_test = test_out.cpu()\n",
    "            std_test = torch.zeros_like(mu_test)\n",
    "            var_test = torch.square(std_test)\n",
    "\n",
    "        x_test_cpu = x_data_test.cpu()\n",
    "        test_mass_rhs_func = dataset_class.get_mass_rhs_func(x=x_test_cpu)\n",
    "\n",
    "        if apply_probconserv:\n",
    "            new_mu_test, new_std_test, _, test_mass_rhs = probconserv.apply_constraint(\n",
    "                mu=mu_test[:, :, :, 0],\n",
    "                std=std_test[:, :, :, 0],\n",
    "                mass_rhs_func=test_mass_rhs_func,\n",
    "                t=t,\n",
    "                tpred=tpred,\n",
    "                grid_train=grid,\n",
    "                precis_g=float('inf'),\n",
    "                second_deriv_alpha=None,\n",
    "            )\n",
    "            mu_test = new_mu_test.unsqueeze(-1)\n",
    "            std_test = new_std_test.unsqueeze(-1)\n",
    "            var_test = torch.square(std_test)\n",
    "            cerr_test = (probconserv.get_empirical_mass_rhs(mu_test[:, :, :, 0]) - test_mass_rhs).abs().sum(dim=-1)\n",
    "        else:\n",
    "            t_sliced = t[slice(*tpred)]\n",
    "            ts = repeat(t_sliced, \"nt -> nf nt\", nf=mu_test.shape[0])\n",
    "            xs = repeat(grid, \"nx -> nf nx\", nf=mu_test.shape[0])\n",
    "            inputs = meshgrid(ts, xs)\n",
    "            cerr_test = (probconserv.get_empirical_mass_rhs(mu_test[:, :, :, 0]) - test_mass_rhs_func(inputs)).abs().sum(dim=-1)\n",
    "\n",
    "        test_stats = utils.compute_all_metrics_avg((mu_test, var_test), y_data_test, {})\n",
    "        test_stats[\"nMeRCI_all\"] = utils.compute_nMeRCI(mu_test, var_test, y_data_test).item()\n",
    "        test_stats[\"rmsce_all\"] = utils.compute_rmsce(mu_test, var_test, y_data_test).item()\n",
    "        test_stats[\"cerr_by_example\"] = cerr_test.tolist()\n",
    "        test_stats[\"mcerr\"] = cerr_test.mean().item()\n",
    "\n",
    "    # --- Optional plot ---\n",
    "    if plot:\n",
    "        t_idx = 1\n",
    "        param_idx = 0\n",
    "        with torch.no_grad():\n",
    "            plt.ylabel(f\"u(x, t={t[slice(*tpred)][t_idx]:.2f})\")\n",
    "            plt.xlabel(\"x\")\n",
    "            plt.title(f\"Predicted vs True (param = {x_data[param_idx,0,0,0].item():.2f})\")\n",
    "            mu_plot = mu[param_idx, :, t_idx, 0]\n",
    "            std_plot = std[param_idx, :, t_idx, 0]\n",
    "            y_true_plot = y_data[param_idx, :, t_idx, 0]\n",
    "            plt.plot(grid, mu_plot, '--', lw=2, label=\"μ ± 3σ\")\n",
    "            plt.fill_between(grid, mu_plot + 3*std_plot, mu_plot - 3*std_plot, alpha=0.2)\n",
    "            plt.plot(grid, y_true_plot, color=\"green\", label=\"true\")\n",
    "            plt.legend()\n",
    "            plt.show()\n",
    "\n",
    "    # --- Optional LaTeX row ---\n",
    "    latex_row = None\n",
    "    if return_latex and test_stats:\n",
    "        latex_row = (\n",
    "            f\"{name} & \"\n",
    "            f\"{stats['mse']:.2E} & {stats['nMeRCI_all']:.2E} & {stats['rmsce_all']:.2E} & {stats['mcerr']:.2E} & {stats['crps']:.2E} & \"\n",
    "            f\"{test_stats['mse']:.2E} & {test_stats['nMeRCI_all']:.2E} & {test_stats['rmsce_all']:.2E} & {test_stats['mcerr']:.2E} & {test_stats['crps']:.2E} \\\\\\\\\"\n",
    "        )\n",
    "\n",
    "    return (stats, test_stats, latex_row) if return_latex else (stats, test_stats)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 44,
   "id": "a48efcc2-03a0-47c2-8af5-62cd2fd6e4e4",
   "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=\"Unconstrained\"\n",
    ")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 45,
   "id": "ea1e01de-5ed3-489d-ab73-3173bd8181b0",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "184.5393143594265"
      ]
     },
     "execution_count": 45,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "test_stats['mse']*1e6"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 51,
   "id": "837265e6-c08a-4bf7-9200-8835499a555b",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "24.73994344472885"
      ]
     },
     "execution_count": 51,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "test_stats['crps']*1e4"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 52,
   "id": "aa075212-97ba-4823-bac0-e43558b27ce2",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.00031796210464563046"
      ]
     },
     "execution_count": 52,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "test_stats['mcerr']*1e3"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 53,
   "id": "ca88a17e-fc02-4347-8c26-74fa8bd4d1d2",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "'Unconstrained & 1.78E-04 & 1.53E+00 & 2.17E-01 & 3.11E-07 & 2.44E-03 & 1.85E-04 & 1.44E+00 & 2.23E-01 & 3.18E-07 & 2.47E-03 \\\\\\\\'"
      ]
     },
     "execution_count": 53,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "latex"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 54,
   "id": "98acb797",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.00018028581514954568"
      ]
     },
     "execution_count": 54,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "id_results['pc.mse']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 55,
   "id": "1ec3729e-d8e6-454f-8e4e-6b05f44a94f4",
   "metadata": {},
   "outputs": [],
   "source": [
    "out = model(x_ood_test.to(device))\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",
    "\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\n",
    "nf,nx,nt,_ = mu.shape\n",
    "\n",
    "# _mu = mu.view(nf, -1)\n",
    "# _var = var.view(nf, -1)\n",
    "# _m = x.view(nf, -1).to(device)\n",
    "\n",
    "# # print(_m)\n",
    "\n",
    "# u_proj, u_var = project_and_stats(torch.relu(_mu), _var, _m, model.full_residual, max_iter=30)\n",
    "\n",
    "# out = (u_proj.view(nf,nx,nt,1), u_var .view(nf,nx,nt,1))\n",
    "\n",
    "mu, var, = out[0].cpu(), out[1].cpu()\n",
    "\n",
    "std = torch.sqrt(var)\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 56,
   "id": "fa65af1d-7356-4892-acff-81125e64347f",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor(0.0239, grad_fn=<LinalgVectorNormBackward0>)"
      ]
     },
     "execution_count": 56,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "torch.norm(utils.compute_crps_by_example(mu, var, y_ood_test))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 57,
   "id": "11e418ff-cbf6-4455-a062-676e2df0a1cc",
   "metadata": {},
   "outputs": [],
   "source": [
    "# u_proj, u_var = project_and_stats(nmu.view(nf,-1).to(device), nvar.view(nf,-1).to(device), _m, model.full_residual, max_iter=30)\n",
    "# nmu, nvar = (u_proj.view(nf,nx,nt,1), u_var.view(nf,nx,nt,1))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 58,
   "id": "b4d043cc-dd31-4a21-ad06-663438462c21",
   "metadata": {},
   "outputs": [],
   "source": [
    "# torch.norm(utils.compute_crps_by_example(mu.cpu(), nvar.cpu(), y_train))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 59,
   "id": "a2ecdfcd-84ec-49f2-9d09-51549d332c28",
   "metadata": {},
   "outputs": [],
   "source": [
    "from torch.func import vmap"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 60,
   "id": "5e96a231-5af5-4a84-af56-f3069403e0b1",
   "metadata": {},
   "outputs": [],
   "source": [
    "# torch.norm(vmap(model.full_residual)(torch.relu(mu.to(device)), _m),dim = 0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 61,
   "id": "4cd9a8f9-b2d7-43c3-982b-f4e547fcaf20",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor(0.0002, grad_fn=<MseLossBackward0>)"
      ]
     },
     "execution_count": 61,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "torch.nn.MSELoss()(y_ood_test[0,:,1,:],mu[0,:,1,:])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 62,
   "id": "995e5a6f-63e8-4cf3-85f9-73f8d7d30009",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(0.2104)\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# t_idx = len(t[slice(*tpred)])//2\n",
    "t_idx = 10\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$ (probconserv)\")\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",
    "        \n",
    "        plt.plot(grid, mu[parameters_idx,:,t_idx,0], '--', lw=2, label = \"predicted $\\mu$ and $\\pm 3\\sigma$ (probharde2e)\")\n",
    "        plt.fill_between(grid, mu[parameters_idx,:,t_idx,0]+3*std[parameters_idx,:,t_idx,0], mu[parameters_idx,:,t_idx,0]-3*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",
    "        \n",
    "        print(torch.norm(y_train[parameters_idx,:,t_idx,0] - mu[parameters_idx,:,t_idx,0]))\n",
    "        plt.legend()\n",
    "        # plt.ylim(-1.0,1.5)\n",
    "        plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 63,
   "id": "4dfa6e03-c09a-406f-abf8-d41d68afa9f6",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([100, 100, 20, 1])"
      ]
     },
     "execution_count": 63,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "mu.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 64,
   "id": "6376bd34-380d-43f0-b9ae-e21dfa77d686",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([100, 100, 20, 1])"
      ]
     },
     "execution_count": 64,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "var.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 65,
   "id": "1824da2b-18e6-4b1f-80d5-4e5d7e167a9d",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([100])"
      ]
     },
     "execution_count": 65,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "grid.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 66,
   "id": "01feae7c-7aba-49ca-9723-f00545566493",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "slice(0, -1, 5)"
      ]
     },
     "execution_count": 66,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "slice(*tpred)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 67,
   "id": "67a10613-e7ed-48d3-bd48-3cd1f6ace4b1",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([0.0000, 0.0505, 0.1010, 0.1515, 0.2020, 0.2525, 0.3030, 0.3535, 0.4040,\n",
       "        0.4545, 0.5051, 0.5556, 0.6061, 0.6566, 0.7071, 0.7576, 0.8081, 0.8586,\n",
       "        0.9091, 0.9596])"
      ]
     },
     "execution_count": 67,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "t[slice(*tpred)]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 68,
   "id": "eecfe6c0-2ee6-4918-b2b8-7856bf097fdf",
   "metadata": {},
   "outputs": [],
   "source": [
    "dt = t[slice(*tpred)][1] - t[slice(*tpred)][0]\n",
    "dx = grid[1] - grid[0]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 69,
   "id": "7e3bc062-c816-49f6-bc44-103fa68cadbc",
   "metadata": {},
   "outputs": [],
   "source": [
    "def conservation_residual_all_times(u_flat, m):\n",
    "    u = u_flat.view(nx, nt)\n",
    "\n",
    "    # Initial mass\n",
    "    mass_0 = torch.sum(u[:, 0]) * dx  # shape: scalar\n",
    "\n",
    "    # Compute u^m * ∂u/∂x at left and right boundary for all time steps\n",
    "    um = u ** m\n",
    "    ux_left = (u[1, :] - u[0, :]) / dx         # shape: (nt,)\n",
    "    ux_right = (u[-1, :] - u[-2, :]) / dx      # shape: (nt,)\n",
    "    \n",
    "    flux_left = um[0, :] * ux_left             # u^m * u_x at x = 0\n",
    "    flux_right = um[-1, :] * ux_right          # u^m * u_x at x = 1\n",
    "\n",
    "    net_flux = torch.cumsum(flux_right - flux_left, dim=0) * dt  # ∫₀^t (flux_right - flux_left)\n",
    "\n",
    "    # Total mass at each time\n",
    "    mass_t = torch.sum(u, dim=0) * dx          # shape: (nt,)\n",
    "\n",
    "    # Conservation residual\n",
    "    residue = mass_t - mass_0 + net_flux       # sign flipped because outflow reduces mass\n",
    "\n",
    "    return residue  # shape: (nt,)\n",
    "\n",
    "# Combine\n",
    "def full_residual(u_flat):\n",
    "    return torch.cat([conservation_residual_all_times(u_flat)])\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 70,
   "id": "7567e79a-911c-4214-8d55-f2e05643e4e1",
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "from torch.func import vmap, jacrev\n",
    "\n",
    "# --- Projection code ---\n",
    "def fast_project_batched(xi_batch, h_func, max_iter=30):\n",
    "    B, n = xi_batch.shape\n",
    "\n",
    "    def newton_step(u, xi):\n",
    "        h_val = h_func(u)\n",
    "        if h_val.ndim == 1:\n",
    "            h_val = h_val.unsqueeze(-1)\n",
    "        J = jacrev(h_func)(u)\n",
    "        if J.ndim == 1:\n",
    "            J = J.unsqueeze(0)\n",
    "        delta = (xi - u).unsqueeze(-1)\n",
    "        JJt = J @ J.transpose(-2, -1)\n",
    "        rhs = J @ delta + h_val\n",
    "        lambda_ = torch.linalg.solve(JJt, rhs)\n",
    "        du = delta - J.transpose(-2, -1) @ lambda_\n",
    "        return u + du.squeeze(-1)\n",
    "\n",
    "    def loop(xi):\n",
    "        u = xi.clone()\n",
    "        for _ in range(max_iter):\n",
    "            u = newton_step(u, xi)\n",
    "        return u\n",
    "\n",
    "    return vmap(loop)(xi_batch)\n",
    "\n",
    "def fast_project_weighted(xi_batch, sigma_batch, h_func, max_iter=30):\n",
    "    \"\"\"\n",
    "    Solve: argmin_u ||u - xi||^2_{sigma^{-1}} s.t. h(u) = 0\n",
    "    Args:\n",
    "        xi_batch: (B, n)\n",
    "        sigma_batch: (B, n)\n",
    "        h_func: function u → ℝ^m\n",
    "    Returns:\n",
    "        u_proj: (B, n)\n",
    "    \"\"\"\n",
    "    B, n = xi_batch.shape\n",
    "\n",
    "    def newton_step(u, xi, sigma):\n",
    "        sigma_inv = 1.0 / (sigma)         # shape (n,)\n",
    "        h_val = h_func(u)\n",
    "        if h_val.ndim == 1:\n",
    "            h_val = h_val.unsqueeze(-1)\n",
    "        J = jacrev(h_func)(u)\n",
    "        if J.ndim == 1:\n",
    "            J = J.unsqueeze(0)\n",
    "\n",
    "        delta = (xi - u).unsqueeze(-1)           # shape (n, 1)\n",
    "\n",
    "        # Use elementwise weighting: JW = J * sigma_inv[None, :]\n",
    "        JW = J * sigma_inv.unsqueeze(0)              # shape (m, n)\n",
    "        JJt = JW @ J.transpose(-2, -1)               # (m, m)\n",
    "        rhs = JW @ delta + h_val                     # (m, 1)\n",
    "\n",
    "        lambda_ = torch.linalg.solve(JJt, rhs)       # (m, 1)\n",
    "        Jt_lambda = J.transpose(-2, -1) @ lambda_    # (n, 1)\n",
    "\n",
    "        du = sigma_inv.unsqueeze(-1) * (delta - Jt_lambda)  # elementwise multiply\n",
    "\n",
    "\n",
    "        return u + du.squeeze(-1)\n",
    "\n",
    "    def loop(xi, sigma):\n",
    "        u = xi.clone()\n",
    "        for _ in range(max_iter):\n",
    "            u = newton_step(u, xi, sigma)\n",
    "        return u\n",
    "\n",
    "    return vmap(loop)(xi_batch, sigma_batch)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 71,
   "id": "7ab81cad-a983-472d-a561-627d397b2938",
   "metadata": {},
   "outputs": [],
   "source": [
    "nf,nx,nt, _ = mu.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 72,
   "id": "5d8bdf1f-e427-4098-87bb-11017ac65088",
   "metadata": {},
   "outputs": [],
   "source": [
    "mu_flat = mu.view(nf, -1).to(device)\n",
    "var_flat = var.view(nf, -1).to(device)\n",
    "m_flat = x_ood_test.view(nf, -1).to(device)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 73,
   "id": "319cd929-a2b9-4410-ab6e-eae65b91ee5c",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([100, 100, 20, 1])"
      ]
     },
     "execution_count": 73,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "x_ood_test.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 74,
   "id": "5d9cbb76-f1a6-4bd0-8a1f-1808abb5539e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([100, 2000])"
      ]
     },
     "execution_count": 74,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "mu_flat.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 75,
   "id": "27fa38f0-9d5f-49b6-bcc8-0f455cd0ff99",
   "metadata": {},
   "outputs": [],
   "source": [
    "# def conservation_residual_all_times(u_flat, m_flat):\n",
    "#     u = u_flat.view(nx, nt)\n",
    "#     m = m_flat.view(nx, nt)\n",
    "\n",
    "#     # Initial mass\n",
    "#     mass_0 = torch.sum(u[:, 0]) * dx  # shape: scalar\n",
    "\n",
    "#     # Compute u^m * ∂u/∂x at left and right boundary for all time steps\n",
    "#     um = u ** m\n",
    "#     ux_left = (u[1, :] - u[0, :]) / dx         # shape: (nt,)\n",
    "#     ux_right = (u[-1, :] - u[-2, :]) / dx      # shape: (nt,)\n",
    "    \n",
    "#     flux_left = um[0, :] * ux_left             # u^m * u_x at x = 0\n",
    "#     flux_right = um[-1, :] * ux_right          # u^m * u_x at x = 1\n",
    "\n",
    "#     net_flux = torch.cumsum(flux_right - flux_left, dim=0) * dt  # ∫₀^t (flux_right - flux_left)\n",
    "\n",
    "#     # Total mass at each time\n",
    "#     mass_t = torch.sum(u, dim=0) * dx          # shape: (nt,)\n",
    "\n",
    "#     # Conservation residual\n",
    "#     residue = mass_t - mass_0 + net_flux       # sign flipped because outflow reduces mass\n",
    "\n",
    "#     return residue[1:]  # shape: (nt,)\n",
    "\n",
    "def conservation_residual_all_times(u_flat, m_flat):\n",
    "    u = u_flat.view(nx, nt)\n",
    "    m = m_flat.view(nx, nt)\n",
    "\n",
    "    def trapz_space(u):\n",
    "        weight = torch.ones_like(u)\n",
    "        weight[0, ...] *= 0.5\n",
    "        weight[-1, ...] *= 0.5\n",
    "        return torch.sum(weight * u, dim=0) * dx\n",
    "\n",
    "    # Initial mass\n",
    "    mass_0 = trapz_space(u[:, 0])  # scalar\n",
    "\n",
    "    # Compute u^m * u_x at both ends using backward differences\n",
    "    um = u ** m\n",
    "    ux_left = (u[0, :] - u[1, :]) / dx\n",
    "    ux_right = (u[-2, :] - u[-1, :]) / dx\n",
    "\n",
    "    flux_diff = um[-1, :] * ux_right - um[0, :] * ux_left  # shape: (nt,)\n",
    "\n",
    "    # Left Riemann sum: net flux up to each time step\n",
    "    flux_increments = torch.cat([\n",
    "        torch.zeros(1, device=u.device),        # net_flux[0] = 0\n",
    "        flux_diff[:-1] * dt\n",
    "    ], dim=0)\n",
    "\n",
    "    net_flux = torch.cumsum(flux_increments, dim=0)\n",
    "\n",
    "    # Mass at each time\n",
    "    mass_t = trapz_space(u)\n",
    "\n",
    "    # Residual = mass_t - mass_0 + net_flux (should be ≈ 0)\n",
    "    residue = mass_t - mass_0 + net_flux\n",
    "\n",
    "    return residue[1:]  # skip t=0\n",
    "\n",
    "\n",
    "def ic_residual(u_flat, u0):\n",
    "    u = u_flat.view(nx, nt)           # shape: (nx, nt)\n",
    "    return u[1:-1, 0]              # shape: (nx,)\n",
    "\n",
    "\n",
    "t_grid = t[slice(*tpred)].clone().to(device)\n",
    "def bc_residual_dirichlet(u_flat, m_flat):\n",
    "    u = u_flat.view(nx, nt)\n",
    "    m = m_flat[0] # for broadcasting\n",
    "\n",
    "    # Construct target boundary profile for left boundary\n",
    "    # t_grid = torch.linspace(0, 1, nt, device=u.device)  # shape: (nt,)\n",
    "    left_bc_target = (m * t_grid) ** (1.0 / m)  # shape: (nt,)\n",
    "    \n",
    "    left_bc_actual = u[0, :]                               # u(x=0, t)\n",
    "    right_bc_actual = u[-1, :]                             # u(x=1, t)\n",
    "\n",
    "    h_left = left_bc_actual - left_bc_target               # (nt,)\n",
    "    h_right = right_bc_actual                              # (nt,)\n",
    "\n",
    "    return torch.cat([h_left, h_right], dim=0)             # shape: (2 × nt,)\n",
    "\n",
    "# Combine\n",
    "def full_residual(u_flat, m_flat):\n",
    "    return torch.cat([ic_residual(u_flat, m_flat),\n",
    "                      conservation_residual_all_times(u_flat, m_flat),                          # (nx,)\n",
    "                      bc_residual_dirichlet(u_flat, m_flat),\n",
    "                     ])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 76,
   "id": "5f9489c5-b02d-4719-b675-af1c38df18fe",
   "metadata": {},
   "outputs": [],
   "source": [
    "def fast_project_weighted_with_m(xi_batch, sigma_batch, m_batch, h_func, max_iter=30):\n",
    "    def newton_step(u, xi, sigma, m):\n",
    "        sigma_inv = 1.0 / (sigma +1e-6)\n",
    "        sigma_inv = torch.ones_like(sigma)\n",
    "        h_val = h_func(u,m)\n",
    "        if h_val.ndim == 1:\n",
    "            h_val = h_val.unsqueeze(-1)\n",
    "        J = jacrev(lambda u_: h_func(u_, m))(u)  # (nt, n)\n",
    "        if J.ndim == 1:\n",
    "            J = J.unsqueeze(0)\n",
    "\n",
    "        # print(J)\n",
    "\n",
    "        delta = (xi - u).unsqueeze(-1)           # shape (n, 1)\n",
    "\n",
    "        # Use elementwise weighting: JW = J * sigma_inv[None, :]\n",
    "        JW = J * sigma_inv.unsqueeze(0)              # shape (m, n)\n",
    "        JJt = JW @ J.transpose(-2, -1)               # (m, m)\n",
    "        rhs = JW @ delta + h_val                     # (m, 1)\n",
    "\n",
    "        lambda_ = torch.linalg.solve(JJt, rhs)       # (m, 1)\n",
    "        Jt_lambda = J.transpose(-2, -1) @ lambda_    # (n, 1)\n",
    "        \n",
    "        du = sigma_inv.unsqueeze(-1) * (delta - Jt_lambda)\n",
    "\n",
    "        # return u + du.squeeze(-1)\n",
    "        return torch.clamp(u + du.squeeze(-1),min=0.0)\n",
    "\n",
    "    def loop(xi, sigma, m):\n",
    "        u = xi.clone()\n",
    "        for _ in range(max_iter):\n",
    "            u = newton_step(u, xi, sigma, m)\n",
    "        return u\n",
    "\n",
    "    return vmap(loop)(xi_batch, sigma_batch, m_batch)\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 77,
   "id": "681dc3a6-49ef-40ad-b78c-a359b0aa81af",
   "metadata": {},
   "outputs": [],
   "source": [
    "def fast_project_weighted_with_m(xi_batch, sigma_batch, m_batch, h_func, max_iter=30):\n",
    "    def newton_step(u_tilde, xi_tilde, sqrt_sigma, m):\n",
    "        # back-transform: u = Σ^{1/2} ũ\n",
    "        u = u_tilde * sqrt_sigma\n",
    "\n",
    "        h_val = h_func(u, m)\n",
    "        if h_val.ndim == 1:\n",
    "            h_val = h_val.unsqueeze(-1)\n",
    "\n",
    "        # J: dh/dũ = dh/du × du/dũ = J_u × diag(sqrt_sigma)\n",
    "        J_u = jacrev(lambda u_: h_func(u_, m))(u)  # (k, p)\n",
    "        J = J_u * sqrt_sigma.unsqueeze(0)          # chain rule\n",
    "\n",
    "        delta = (xi_tilde - u_tilde).unsqueeze(-1)\n",
    "\n",
    "        JJt = J @ J.transpose(-2, -1)\n",
    "        rhs = J @ delta + h_val\n",
    "\n",
    "        #I = torch.eye(JJt.shape[-1], device=JJt.device, dtype=JJt.dtype)\n",
    "        # + 1e-6 * I\n",
    "        lambda_ = torch.linalg.solve(JJt, rhs)\n",
    "\n",
    "        du = delta - J.transpose(-2, -1) @ lambda_\n",
    "        u_tilde_next = u_tilde + du.squeeze(-1)\n",
    "        \n",
    "        return torch.clamp(u_tilde_next,min=0.0)\n",
    "\n",
    "    def loop(xi, sigma, m):\n",
    "        eps = 1e-6\n",
    "        sigma = sigma.clamp(min=eps)\n",
    "        # sigma = torch.ones_like(sigma)\n",
    "        sqrt_sigma = sigma.sqrt()\n",
    "        sqrt_sigma_inv = 1.0 / sqrt_sigma\n",
    "\n",
    "        xi_tilde = xi * sqrt_sigma_inv\n",
    "        u_tilde = xi_tilde.clone()\n",
    "\n",
    "        for _ in range(max_iter):\n",
    "            u_tilde = newton_step(u_tilde, xi_tilde, sqrt_sigma, m)\n",
    "\n",
    "        u_proj = u_tilde * sqrt_sigma  # back-transform\n",
    "        return torch.clamp(u_proj, min=0.0)\n",
    "\n",
    "    return vmap(loop)(xi_batch, sigma_batch, m_batch)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 78,
   "id": "e883c9c9-bb46-4702-baf8-26e3b24a989c",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([[1.1776e-04, 2.8311e-05, 1.1950e-05,  ..., 4.1896e-05, 6.3852e-06,\n",
       "         7.2833e-05],\n",
       "        [6.1618e-05, 1.2370e-05, 4.8593e-06,  ..., 1.8290e-05, 2.3802e-06,\n",
       "         3.2969e-05],\n",
       "        [1.2217e-04, 2.9668e-05, 1.2578e-05,  ..., 4.3943e-05, 6.7564e-06,\n",
       "         7.6216e-05],\n",
       "        ...,\n",
       "        [8.1909e-05, 1.7815e-05, 7.2143e-06,  ..., 2.6262e-05, 3.6662e-06,\n",
       "         4.6641e-05],\n",
       "        [1.2961e-04, 3.1982e-05, 1.3656e-05,  ..., 4.7448e-05, 7.3987e-06,\n",
       "         8.1986e-05],\n",
       "        [3.4969e-05, 5.9676e-06, 2.2178e-06,  ..., 9.0063e-06, 1.0156e-06,\n",
       "         1.6633e-05]], device='cuda:0', grad_fn=<ToCopyBackward0>)"
      ]
     },
     "execution_count": 78,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "var_flat"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 79,
   "id": "6dfe6a62-07b8-4f10-9815-58ee606220b8",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "5058"
      ]
     },
     "execution_count": 79,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "import gc\n",
    "gc.collect()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 80,
   "id": "9decf451-14e0-4e6f-8f4f-bdd4fc1821e1",
   "metadata": {},
   "outputs": [],
   "source": [
    "torch.cuda.empty_cache()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 81,
   "id": "51f5fde6-fd55-4c0b-8d1e-d1883f63600d",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([157])"
      ]
     },
     "execution_count": 81,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "full_residual(mu_flat[0,:], m_flat[0,:]).shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 82,
   "id": "fc460f0d-5920-4f3e-886d-657a7d747b65",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([[3.2588, 3.2588, 3.2588,  ..., 3.2588, 3.2588, 3.2588],\n",
       "        [3.5898, 3.5898, 3.5898,  ..., 3.5898, 3.5898, 3.5898],\n",
       "        [3.2403, 3.2403, 3.2403,  ..., 3.2403, 3.2403, 3.2403],\n",
       "        ...,\n",
       "        [3.4431, 3.4431, 3.4431,  ..., 3.4431, 3.4431, 3.4431],\n",
       "        [3.2107, 3.2107, 3.2107,  ..., 3.2107, 3.2107, 3.2107],\n",
       "        [3.8865, 3.8865, 3.8865,  ..., 3.8865, 3.8865, 3.8865]],\n",
       "       device='cuda:0')"
      ]
     },
     "execution_count": 82,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "m_flat"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 83,
   "id": "23d5d420-215b-4320-add6-4d58bf746878",
   "metadata": {},
   "outputs": [],
   "source": [
    "import gc\n",
    "gc.collect()\n",
    "torch.cuda.empty_cache()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 84,
   "id": "3be0869f-f0db-41df-94f4-814aaf315d79",
   "metadata": {},
   "outputs": [],
   "source": [
    "u_proj = fast_project_weighted_with_m(torch.relu(mu_flat), var_flat, m_flat, full_residual)\n",
    "u_proj_reshaped = u_proj.view(nf, nx, nt, 1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 85,
   "id": "e8ce1384-f11f-4d5a-8f9d-b96339ac6788",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([[0.0000e+00, 5.7483e-01, 7.1107e-01,  ..., 1.8838e-13, 0.0000e+00,\n",
       "         0.0000e+00],\n",
       "        [0.0000e+00, 6.2147e-01, 7.5383e-01,  ..., 1.2447e-13, 1.6838e-14,\n",
       "         8.3555e-14],\n",
       "        [0.0000e+00, 5.7201e-01, 7.0844e-01,  ..., 0.0000e+00, 0.0000e+00,\n",
       "         2.6018e-10],\n",
       "        ...,\n",
       "        [0.0000e+00, 6.0166e-01, 7.3584e-01,  ..., 1.8643e-13, 2.8532e-11,\n",
       "         0.0000e+00],\n",
       "        [1.4267e-07, 5.6744e-01, 7.0418e-01,  ..., 2.0048e-13, 0.0000e+00,\n",
       "         0.0000e+00],\n",
       "        [0.0000e+00, 6.5774e-01, 7.8616e-01,  ..., 1.7468e-13, 7.3325e-15,\n",
       "         0.0000e+00]], device='cuda:0', grad_fn=<ClampBackward1>)"
      ]
     },
     "execution_count": 85,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "u_proj"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 86,
   "id": "fb3a9c7e-06bc-49b6-898c-21ea991483c9",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([], device='cuda:0', grad_fn=<IndexBackward0>)"
      ]
     },
     "execution_count": 86,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "u_proj[torch.isnan(u_proj)]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 87,
   "id": "957e7c15-a613-4e67-a9cb-1856e51e8dd1",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([0.0000e+00, 0.0000e+00, 1.7829e-09, 2.2592e-09, 1.2037e-09, 7.3103e-10,\n",
       "        6.4110e-10, 6.4164e-10, 4.9749e-10, 6.1466e-10, 2.6209e-10, 5.5071e-10,\n",
       "        9.0641e-10, 0.0000e+00, 1.7528e-09, 0.0000e+00, 0.0000e+00, 2.2365e-09,\n",
       "        8.7670e-10, 1.3286e-09, 6.9547e-10, 8.7811e-10, 0.0000e+00, 6.9559e-11,\n",
       "        0.0000e+00, 5.6545e-10, 5.7543e-10, 0.0000e+00, 0.0000e+00, 3.3577e-10,\n",
       "        2.2641e-10, 0.0000e+00, 0.0000e+00, 1.0024e-09, 0.0000e+00, 1.4548e-09,\n",
       "        0.0000e+00, 0.0000e+00, 1.0735e-09, 3.3476e-10, 6.9709e-10, 0.0000e+00,\n",
       "        5.4307e-10, 1.9582e-10, 3.8416e-10, 1.2144e-09, 1.6187e-11, 0.0000e+00,\n",
       "        4.1883e-10, 2.8094e-10, 0.0000e+00, 6.3639e-10, 7.9287e-10, 0.0000e+00,\n",
       "        6.9023e-10, 1.4296e-09, 0.0000e+00, 0.0000e+00, 4.7423e-10, 1.9154e-10,\n",
       "        3.2242e-10, 6.6377e-10, 0.0000e+00, 0.0000e+00, 0.0000e+00, 2.1451e-09,\n",
       "        8.9442e-10, 0.0000e+00, 2.5344e-10, 4.1394e-10, 3.1635e-10, 4.8813e-10,\n",
       "        0.0000e+00, 3.4777e-10, 6.2273e-10, 0.0000e+00, 3.2164e-10, 6.8631e-10,\n",
       "        0.0000e+00, 9.2378e-11, 0.0000e+00, 0.0000e+00, 5.0132e-11, 7.0990e-10,\n",
       "        2.5019e-10, 0.0000e+00, 2.2141e-10, 1.3175e-10, 0.0000e+00, 0.0000e+00,\n",
       "        0.0000e+00, 0.0000e+00, 6.2459e-10, 0.0000e+00, 0.0000e+00, 0.0000e+00,\n",
       "        0.0000e+00, 2.3676e-10, 1.2199e-11, 0.0000e+00], device='cuda:0',\n",
       "       grad_fn=<SelectBackward0>)"
      ]
     },
     "execution_count": 87,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "u_proj_reshaped[0,:,0,0]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 88,
   "id": "1ccd11e3-69f6-435f-8a39-0dcc9acb9003",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([ 1.4931e-03, -5.0416e-03,  1.3960e-03,  9.5251e-04,  5.5682e-05,\n",
       "        -6.6515e-04, -4.0019e-04,  3.4453e-04,  4.1491e-04,  1.9071e-04,\n",
       "        -8.6328e-04, -1.5856e-03, -3.5506e-05,  1.6776e-03,  8.1876e-04,\n",
       "        -1.0565e-03, -9.5557e-04,  7.3386e-04,  1.2757e-03,  1.5995e-04,\n",
       "        -5.5669e-04,  5.2761e-05,  7.9487e-04,  5.3826e-04, -3.1434e-04,\n",
       "        -4.8088e-04,  3.1443e-04,  1.0053e-03,  8.6376e-04,  4.5029e-04,\n",
       "         5.3154e-04,  8.5768e-04,  8.1957e-04,  4.8278e-04,  2.8727e-04,\n",
       "         3.2827e-04,  2.9492e-04, -8.3335e-06, -2.4336e-04, -3.7998e-07,\n",
       "         4.8552e-04,  7.2373e-04,  7.3280e-04,  7.3298e-04,  6.6771e-04,\n",
       "         4.7180e-04,  2.5283e-04,  9.1210e-05,  5.2527e-05,  2.3472e-04,\n",
       "         4.7296e-04,  3.8082e-04, -6.9067e-06, -1.3101e-04,  2.9188e-04,\n",
       "         8.1575e-04,  8.7086e-04,  5.2566e-04,  3.3518e-04,  5.4389e-04,\n",
       "         7.3439e-04,  4.3251e-04, -1.4414e-04, -3.5113e-04, -2.3648e-05,\n",
       "         3.0091e-04,  2.0100e-04, -1.4357e-05,  1.2023e-04,  5.3300e-04,\n",
       "         7.6106e-04,  5.2333e-04,  1.1708e-04,  7.7233e-05,  3.3429e-04,\n",
       "         3.0982e-04, -4.3459e-05, -1.2220e-04,  2.7138e-04,  5.2417e-04,\n",
       "         1.8685e-04, -1.8721e-04,  1.5546e-04,  7.9982e-04,  7.1279e-04,\n",
       "        -3.8892e-06, -2.5926e-04,  1.8927e-04,  4.1411e-04, -7.2267e-05,\n",
       "        -5.7557e-04, -3.4603e-04,  8.9895e-05, -3.9046e-04, -1.0294e-03,\n",
       "        -4.2738e-04,  4.8001e-04,  4.6413e-05, -1.0601e-03, -3.8265e-04],\n",
       "       grad_fn=<SelectBackward0>)"
      ]
     },
     "execution_count": 88,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "mu[0,:,0,0]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 89,
   "id": "2b9e3bb2-41d7-49f0-916c-246826661660",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([ 0.0000e+00,  1.7829e-09,  2.2592e-09,  1.2037e-09,  7.3103e-10,\n",
       "         6.4110e-10,  6.4164e-10,  4.9749e-10,  6.1466e-10,  2.6209e-10,\n",
       "         5.5071e-10,  9.0641e-10,  0.0000e+00,  1.7528e-09,  0.0000e+00,\n",
       "         0.0000e+00,  2.2365e-09,  8.7670e-10,  1.3286e-09,  6.9547e-10,\n",
       "         8.7811e-10,  0.0000e+00,  6.9559e-11,  0.0000e+00,  5.6545e-10,\n",
       "         5.7543e-10,  0.0000e+00,  0.0000e+00,  3.3577e-10,  2.2641e-10,\n",
       "         0.0000e+00,  0.0000e+00,  1.0024e-09,  0.0000e+00,  1.4548e-09,\n",
       "         0.0000e+00,  0.0000e+00,  1.0735e-09,  3.3476e-10,  6.9709e-10,\n",
       "         0.0000e+00,  5.4307e-10,  1.9582e-10,  3.8416e-10,  1.2144e-09,\n",
       "         1.6187e-11,  0.0000e+00,  4.1883e-10,  2.8094e-10,  0.0000e+00,\n",
       "         6.3639e-10,  7.9287e-10,  0.0000e+00,  6.9023e-10,  1.4296e-09,\n",
       "         0.0000e+00,  0.0000e+00,  4.7423e-10,  1.9154e-10,  3.2242e-10,\n",
       "         6.6377e-10,  0.0000e+00,  0.0000e+00,  0.0000e+00,  2.1451e-09,\n",
       "         8.9442e-10,  0.0000e+00,  2.5344e-10,  4.1394e-10,  3.1635e-10,\n",
       "         4.8813e-10,  0.0000e+00,  3.4777e-10,  6.2273e-10,  0.0000e+00,\n",
       "         3.2164e-10,  6.8631e-10,  0.0000e+00,  9.2378e-11,  0.0000e+00,\n",
       "         0.0000e+00,  5.0132e-11,  7.0990e-10,  2.5019e-10,  0.0000e+00,\n",
       "         2.2141e-10,  1.3175e-10,  0.0000e+00,  0.0000e+00,  0.0000e+00,\n",
       "         0.0000e+00,  6.2459e-10,  0.0000e+00,  0.0000e+00,  0.0000e+00,\n",
       "         0.0000e+00,  2.3676e-10,  1.2199e-11,  1.5860e-02,  1.9863e-05,\n",
       "         2.4006e-05,  7.0333e-06, -8.9407e-08,  7.9572e-06,  2.3842e-07,\n",
       "         6.7055e-06,  4.1425e-06,  1.0431e-06,  8.5235e-06,  2.9802e-07,\n",
       "         3.3379e-06,  1.3113e-06, -3.5763e-07, -1.7881e-07,  2.9802e-07,\n",
       "        -5.9605e-08,  1.1921e-06,  0.0000e+00,  0.0000e+00,  0.0000e+00,\n",
       "         0.0000e+00,  0.0000e+00,  0.0000e+00,  0.0000e+00, -1.1921e-07,\n",
       "         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,\n",
       "         0.0000e+00,  0.0000e+00,  0.0000e+00,  0.0000e+00,  0.0000e+00,\n",
       "         0.0000e+00,  0.0000e+00,  5.4894e-11,  0.0000e+00,  0.0000e+00,\n",
       "         0.0000e+00,  8.6515e-11,  0.0000e+00,  1.2178e-10,  1.6931e-11,\n",
       "         0.0000e+00,  0.0000e+00,  0.0000e+00,  1.0492e-10,  1.8838e-13,\n",
       "         0.0000e+00,  0.0000e+00], device='cuda:0', grad_fn=<CatBackward0>)"
      ]
     },
     "execution_count": 89,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "full_residual(u_proj_reshaped[0,...].flatten().to(device), m_flat[0,...])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 90,
   "id": "9e7908c2-1b72-40eb-922f-35bfab9a071e",
   "metadata": {},
   "outputs": [],
   "source": [
    "std = torch.sqrt(var)\n",
    "\n",
    "out = model(x_ood_test.to(device))\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",
    "\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": 91,
   "id": "cedca976-7263-4789-aeaf-1850bca3d083",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([0.0000e+00, 1.7829e-09, 2.2592e-09, 1.2037e-09, 7.3103e-10, 6.4110e-10,\n",
       "        6.4164e-10, 4.9749e-10, 6.1466e-10, 2.6209e-10, 5.5071e-10, 9.0641e-10,\n",
       "        0.0000e+00, 1.7528e-09, 0.0000e+00, 0.0000e+00, 2.2365e-09, 8.7670e-10,\n",
       "        1.3286e-09, 6.9547e-10, 8.7811e-10, 0.0000e+00, 6.9559e-11, 0.0000e+00,\n",
       "        5.6545e-10, 5.7543e-10, 0.0000e+00, 0.0000e+00, 3.3577e-10, 2.2641e-10,\n",
       "        0.0000e+00, 0.0000e+00, 1.0024e-09, 0.0000e+00, 1.4548e-09, 0.0000e+00,\n",
       "        0.0000e+00, 1.0735e-09, 3.3476e-10, 6.9709e-10, 0.0000e+00, 5.4307e-10,\n",
       "        1.9582e-10, 3.8416e-10, 1.2144e-09, 1.6187e-11, 0.0000e+00, 4.1883e-10,\n",
       "        2.8094e-10, 0.0000e+00, 6.3639e-10, 7.9287e-10, 0.0000e+00, 6.9023e-10,\n",
       "        1.4296e-09, 0.0000e+00, 0.0000e+00, 4.7423e-10, 1.9154e-10, 3.2242e-10,\n",
       "        6.6377e-10, 0.0000e+00, 0.0000e+00, 0.0000e+00, 2.1451e-09, 8.9442e-10,\n",
       "        0.0000e+00, 2.5344e-10, 4.1394e-10, 3.1635e-10, 4.8813e-10, 0.0000e+00,\n",
       "        3.4777e-10, 6.2273e-10, 0.0000e+00, 3.2164e-10, 6.8631e-10, 0.0000e+00,\n",
       "        9.2378e-11, 0.0000e+00, 0.0000e+00, 5.0132e-11, 7.0990e-10, 2.5019e-10,\n",
       "        0.0000e+00, 2.2141e-10, 1.3175e-10, 0.0000e+00, 0.0000e+00, 0.0000e+00,\n",
       "        0.0000e+00, 6.2459e-10, 0.0000e+00, 0.0000e+00, 0.0000e+00, 0.0000e+00,\n",
       "        2.3676e-10, 1.2199e-11], device='cuda:0', grad_fn=<SelectBackward0>)"
      ]
     },
     "execution_count": 91,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "ic_residual(u_proj[0,...].to(device), m_flat[0,...])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 92,
   "id": "ac4ef7b7-025a-439c-b21a-8fbd9f037f80",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([5.8893e-03, 9.8622e-04, 6.7450e-04, 1.7054e-04, 8.4708e-04, 5.4010e-04,\n",
       "        2.3379e-04, 3.1904e-04, 9.9503e-05, 9.5682e-04, 1.6888e-03, 1.5981e-04,\n",
       "        1.5153e-03, 6.0018e-04, 1.3348e-03, 1.2671e-03, 4.3197e-04, 1.0140e-03,\n",
       "        5.1065e-05, 7.2135e-04, 7.8513e-05, 6.8232e-04, 4.3284e-04, 4.1983e-04,\n",
       "        5.8972e-04, 2.0171e-04, 8.8932e-04, 7.4500e-04, 3.2899e-04, 4.0745e-04,\n",
       "        7.2890e-04, 6.8187e-04, 3.3076e-04, 1.1743e-04, 1.4239e-04, 1.0111e-04,\n",
       "        1.9806e-04, 4.1771e-04, 1.5297e-04, 3.5482e-04, 6.1084e-04, 6.3240e-04,\n",
       "        6.4015e-04, 5.7840e-04, 3.8280e-04, 1.6149e-04, 5.3841e-06, 5.3635e-05,\n",
       "        1.1243e-04, 3.2588e-04, 2.0074e-04, 2.2091e-04, 3.6504e-04, 6.4058e-05,\n",
       "        6.1785e-04, 7.1215e-04, 4.0181e-04, 2.3519e-04, 4.5507e-04, 6.4379e-04,\n",
       "        3.2615e-04, 2.8102e-04, 5.2883e-04, 2.3789e-04, 7.3571e-05, 8.3208e-06,\n",
       "        1.8585e-04, 1.2255e-05, 4.2930e-04, 6.7384e-04, 4.4222e-04, 3.4031e-05,\n",
       "        1.3973e-05, 2.2998e-04, 1.8846e-04, 1.8433e-04, 2.8255e-04, 9.4760e-05,\n",
       "        3.3824e-04, 2.9774e-06, 3.5491e-04, 1.5279e-05, 6.9003e-04, 6.2755e-04,\n",
       "        7.4518e-05, 3.2602e-04, 1.1457e-04, 3.1568e-04, 2.1682e-04, 7.8923e-04,\n",
       "        6.2416e-04, 1.9108e-04, 5.9848e-04, 1.1496e-03, 4.9154e-04, 4.4034e-04,\n",
       "        1.1029e-05, 1.1132e-03], device='cuda:0', grad_fn=<SelectBackward0>)"
      ]
     },
     "execution_count": 92,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "ic_residual(torch.abs(new_mu[0,...]).flatten().to(device), m_flat[0,...])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 93,
   "id": "66fe7479-1fa6-4b3a-9603-43c98c62b13e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor(0.0002, grad_fn=<MseLossBackward0>)"
      ]
     },
     "execution_count": 93,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "torch.nn.MSELoss()(mu,y_ood_test)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 94,
   "id": "840f0465-2661-4b3e-81b6-04a03a432fcb",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor(0.0002, grad_fn=<MseLossBackward0>)"
      ]
     },
     "execution_count": 94,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "torch.nn.MSELoss()(u_proj_reshaped.cpu(), y_ood_test)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 95,
   "id": "5ae9a844-eb88-48d7-a4f0-411d7cb8fb9b",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor(0.0002, grad_fn=<MseLossBackward0>)"
      ]
     },
     "execution_count": 95,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "torch.nn.MSELoss()(new_mu.cpu(), y_ood_test)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 96,
   "id": "48c8eb7f-8a71-4fb9-b007-12fb7d0aa798",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([[0.0000e+00, 1.7829e-09, 2.2592e-09,  ..., 0.0000e+00, 2.3676e-10,\n",
       "         1.2199e-11],\n",
       "        [0.0000e+00, 8.4580e-10, 6.1198e-10,  ..., 0.0000e+00, 1.1921e-10,\n",
       "         5.4183e-11],\n",
       "        [0.0000e+00, 0.0000e+00, 0.0000e+00,  ..., 1.7572e-10, 1.2584e-10,\n",
       "         2.0748e-10],\n",
       "        ...,\n",
       "        [0.0000e+00, 0.0000e+00, 1.4533e-09,  ..., 2.5657e-10, 1.2039e-10,\n",
       "         0.0000e+00],\n",
       "        [0.0000e+00, 7.2507e-10, 1.5553e-09,  ..., 0.0000e+00, 9.2605e-12,\n",
       "         0.0000e+00],\n",
       "        [1.6248e-09, 0.0000e+00, 0.0000e+00,  ..., 0.0000e+00, 6.0290e-11,\n",
       "         6.0290e-11]], device='cuda:0', grad_fn=<SelectBackward0>)"
      ]
     },
     "execution_count": 96,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "vmap(ic_residual)(u_proj,m_flat)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 97,
   "id": "1197d13e-3821-4d20-a7fc-9c4a313ce623",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([ 3.5369e-04, -5.8893e-03,  9.8622e-04,  6.7450e-04, -1.7054e-04,\n",
       "        -8.4708e-04, -5.4010e-04,  2.3379e-04,  3.1904e-04,  9.9503e-05,\n",
       "        -9.5682e-04, -1.6888e-03, -1.5981e-04,  1.5153e-03,  6.0018e-04,\n",
       "        -1.3348e-03, -1.2671e-03,  4.3197e-04,  1.0140e-03, -5.1065e-05,\n",
       "        -7.2135e-04, -7.8513e-05,  6.8232e-04,  4.3284e-04, -4.1983e-04,\n",
       "        -5.8972e-04,  2.0171e-04,  8.8932e-04,  7.4500e-04,  3.2899e-04,\n",
       "         4.0745e-04,  7.2890e-04,  6.8187e-04,  3.3076e-04,  1.1743e-04,\n",
       "         1.4239e-04,  1.0111e-04, -1.9806e-04, -4.1771e-04, -1.5297e-04,\n",
       "         3.5482e-04,  6.1084e-04,  6.3240e-04,  6.4015e-04,  5.7840e-04,\n",
       "         3.8280e-04,  1.6149e-04, -5.3841e-06, -5.3635e-05,  1.1243e-04,\n",
       "         3.2588e-04,  2.0074e-04, -2.2091e-04, -3.6504e-04,  6.4058e-05,\n",
       "         6.1785e-04,  7.1215e-04,  4.0181e-04,  2.3519e-04,  4.5507e-04,\n",
       "         6.4379e-04,  3.2615e-04, -2.8102e-04, -5.2883e-04, -2.3789e-04,\n",
       "         7.3571e-05, -8.3208e-06, -1.8585e-04, -1.2255e-05,  4.2930e-04,\n",
       "         6.7384e-04,  4.4222e-04,  3.4031e-05, -1.3973e-05,  2.2998e-04,\n",
       "         1.8846e-04, -1.8433e-04, -2.8255e-04,  9.4760e-05,  3.3824e-04,\n",
       "         2.9774e-06, -3.5491e-04,  1.5279e-05,  6.9003e-04,  6.2755e-04,\n",
       "        -7.4518e-05, -3.2602e-04,  1.1457e-04,  3.1568e-04, -2.1682e-04,\n",
       "        -7.8923e-04, -6.2416e-04, -1.9108e-04, -5.9848e-04, -1.1496e-03,\n",
       "        -4.9154e-04,  4.4034e-04,  1.1029e-05, -1.1132e-03, -4.5220e-04],\n",
       "       grad_fn=<SelectBackward0>)"
      ]
     },
     "execution_count": 97,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "new_mu[0,:,0,0]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 98,
   "id": "31182cc9-f3fa-4b1d-aa63-5f91b505463e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0.,\n",
       "        0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0.,\n",
       "        0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0.,\n",
       "        0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0.,\n",
       "        0., 0., 0., 0.])"
      ]
     },
     "execution_count": 98,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "y_train[0,:,0,0]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 99,
   "id": "53fd25a7-c6b3-4d3e-943c-39e0e2c5ad11",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([0.0000e+00, 0.0000e+00, 1.7829e-09, 2.2592e-09, 1.2037e-09, 7.3103e-10,\n",
       "        6.4110e-10, 6.4164e-10, 4.9749e-10, 6.1466e-10, 2.6209e-10, 5.5071e-10,\n",
       "        9.0641e-10, 0.0000e+00, 1.7528e-09, 0.0000e+00, 0.0000e+00, 2.2365e-09,\n",
       "        8.7670e-10, 1.3286e-09, 6.9547e-10, 8.7811e-10, 0.0000e+00, 6.9559e-11,\n",
       "        0.0000e+00, 5.6545e-10, 5.7543e-10, 0.0000e+00, 0.0000e+00, 3.3577e-10,\n",
       "        2.2641e-10, 0.0000e+00, 0.0000e+00, 1.0024e-09, 0.0000e+00, 1.4548e-09,\n",
       "        0.0000e+00, 0.0000e+00, 1.0735e-09, 3.3476e-10, 6.9709e-10, 0.0000e+00,\n",
       "        5.4307e-10, 1.9582e-10, 3.8416e-10, 1.2144e-09, 1.6187e-11, 0.0000e+00,\n",
       "        4.1883e-10, 2.8094e-10, 0.0000e+00, 6.3639e-10, 7.9287e-10, 0.0000e+00,\n",
       "        6.9023e-10, 1.4296e-09, 0.0000e+00, 0.0000e+00, 4.7423e-10, 1.9154e-10,\n",
       "        3.2242e-10, 6.6377e-10, 0.0000e+00, 0.0000e+00, 0.0000e+00, 2.1451e-09,\n",
       "        8.9442e-10, 0.0000e+00, 2.5344e-10, 4.1394e-10, 3.1635e-10, 4.8813e-10,\n",
       "        0.0000e+00, 3.4777e-10, 6.2273e-10, 0.0000e+00, 3.2164e-10, 6.8631e-10,\n",
       "        0.0000e+00, 9.2378e-11, 0.0000e+00, 0.0000e+00, 5.0132e-11, 7.0990e-10,\n",
       "        2.5019e-10, 0.0000e+00, 2.2141e-10, 1.3175e-10, 0.0000e+00, 0.0000e+00,\n",
       "        0.0000e+00, 0.0000e+00, 6.2459e-10, 0.0000e+00, 0.0000e+00, 0.0000e+00,\n",
       "        0.0000e+00, 2.3676e-10, 1.2199e-11, 0.0000e+00], device='cuda:0',\n",
       "       grad_fn=<SelectBackward0>)"
      ]
     },
     "execution_count": 99,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "u_proj_reshaped[0,:,0,0]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 100,
   "id": "5e51ff78-8e4f-40e9-8457-27dcacdafbaa",
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "from torch.func import vmap, jacrev\n",
    "\n",
    "class ProjectWeightedImplicit(torch.autograd.Function):\n",
    "    @staticmethod\n",
    "    def forward(ctx, xi_batch, sigma_batch, m_batch, h_func, max_iter=30):\n",
    "        \"\"\"\n",
    "        Batched Newton‐projection with implicit‐diff backward pass.\n",
    "        \n",
    "        Args:\n",
    "            xi_batch:   Tensor of shape (B, p), unconstrained points\n",
    "            sigma_batch: Tensor of shape (B, p), positive weights\n",
    "            m_batch:    Tensor of extra parameters per sample (e.g. (B, …))\n",
    "            h_func:     callable h(u, m) → Tensor of shape (k,) or scalar\n",
    "            max_iter:   number of Newton steps\n",
    "        \n",
    "        Returns:\n",
    "            u_proj: Tensor of shape (B, p), projections s.t. h(u_proj, m)=0\n",
    "        \"\"\"\n",
    "        B, p = xi_batch.shape\n",
    "\n",
    "        def project_one(xi, sigma, m):\n",
    "            # ensure positivity\n",
    "            eps = 1e-6\n",
    "            sigma = sigma.clamp(min=eps)\n",
    "            sqrt_sigma = sigma.sqrt()\n",
    "            inv_sqrt = 1.0 / sqrt_sigma\n",
    "\n",
    "            # transform to tilde‐space\n",
    "            xi_tilde = xi * inv_sqrt\n",
    "            u_tilde  = xi_tilde.clone()\n",
    "\n",
    "            def newton_step(u_tilde):\n",
    "                # back to u‐space\n",
    "                u = u_tilde * sqrt_sigma\n",
    "                h = h_func(u, m)\n",
    "                # make h a vector of shape (k,)\n",
    "                if h.ndim == 0:\n",
    "                    h = h.unsqueeze(0)\n",
    "\n",
    "                # Jacobian ∂h/∂u at u: shape (k, p)\n",
    "                J_u = jacrev(lambda u_: h_func(u_, m))(u)\n",
    "                if J_u.ndim == 1:\n",
    "                    J_u = J_u.unsqueeze(0)\n",
    "                # chain‐rule for tilde: dh/dũ = J_u * diag(sqrt_sigma)\n",
    "                J = J_u * sqrt_sigma.unsqueeze(0)\n",
    "\n",
    "                # compute Newton update in tilde‐space\n",
    "                δ = (xi_tilde - u_tilde).unsqueeze(-1)        # (p,1)\n",
    "                JJt = J @ J.transpose(-2, -1)                  # (k,k)\n",
    "                rhs = J @ δ + h.unsqueeze(-1)                  # (k,1)\n",
    "                I   = torch.eye(JJt.shape[-1], device=JJt.device)\n",
    "                λ   = torch.linalg.solve(JJt + 1e-6*I, rhs)    # (k,1)\n",
    "\n",
    "                du = δ - J.transpose(-2, -1) @ λ               # (p,1)\n",
    "                return torch.clamp(u_tilde + du.squeeze(-1), min=0.0)\n",
    "\n",
    "            # run Newton iterations\n",
    "            for _ in range(max_iter):\n",
    "                u_tilde = newton_step(u_tilde)\n",
    "\n",
    "            # back to original space and clamp\n",
    "            return torch.clamp(u_tilde * sqrt_sigma, min=0.0)\n",
    "\n",
    "        # vectorize over batch\n",
    "        u_proj = vmap(project_one)(xi_batch, sigma_batch, m_batch)\n",
    "\n",
    "        # save for backward\n",
    "        ctx.save_for_backward(xi_batch, sigma_batch, m_batch, u_proj)\n",
    "        ctx.h_func = h_func\n",
    "        return u_proj\n",
    "\n",
    "    @staticmethod\n",
    "    def backward(ctx, grad_u):\n",
    "        \"\"\"\n",
    "        Implicit differentiation via KKT system:\n",
    "          A = [[Σ⁻¹,    Jᵀ],\n",
    "               [  J,      0]]\n",
    "        Stationarity & primal feasibility → solve Aᵀ w = [grad_u; 0],\n",
    "        then ∂u/∂xi^T g = Σ⁻¹ w_u.\n",
    "        \"\"\"\n",
    "        xi_batch, sigma_batch, m_batch, u_proj = ctx.saved_tensors\n",
    "        h_func = ctx.h_func\n",
    "\n",
    "        B, p = xi_batch.shape\n",
    "        grad_xi_list = []\n",
    "\n",
    "        for i in range(B):\n",
    "            xi    = xi_batch[i]    # (p,)\n",
    "            sigma = sigma_batch[i] # (p,)\n",
    "            u     = u_proj[i]      # (p,)\n",
    "            m     = m_batch[i]\n",
    "            g     = grad_u[i]      # (p,)\n",
    "\n",
    "            # compute J_u = ∂h/∂u at the solution\n",
    "            h = h_func(u, m)\n",
    "            if h.ndim == 0:\n",
    "                h = h.unsqueeze(0)\n",
    "            J_u = jacrev(lambda u_: h_func(u_, m))(u)\n",
    "            if J_u.ndim == 1:\n",
    "                J_u = J_u.unsqueeze(0)\n",
    "            k = J_u.shape[0]\n",
    "\n",
    "            # build KKT matrix A and solve Aᵀ w = [g; 0]\n",
    "            σ_inv = 1.0 / sigma\n",
    "            # top block: [Σ⁻¹, Jᵀ]\n",
    "            top    = torch.cat([torch.diag(σ_inv), J_u.transpose(0,1)], dim=1)  # (p, p+k)\n",
    "            # bottom: [J, 0]\n",
    "            bottom = torch.cat([J_u, torch.zeros(k, k, device=u.device)], dim=1)  # (k, p+k)\n",
    "            A      = torch.cat([top, bottom], dim=0)                             # (p+k, p+k)\n",
    "\n",
    "            # right‐hand side\n",
    "            rhs = torch.cat([g, torch.zeros(k, device=g.device)], dim=0)         # (p+k,)\n",
    "\n",
    "            # solve Aᵀ w = rhs\n",
    "            w = torch.linalg.solve(A.transpose(0,1), rhs)                        # (p+k,)\n",
    "\n",
    "            # gradient wrt xi: Σ⁻¹ * w_u\n",
    "            w_u       = w[:p]\n",
    "            grad_xi   = σ_inv * w_u\n",
    "            grad_xi_list.append(grad_xi)\n",
    "\n",
    "        grad_xi_batch = torch.stack(grad_xi_list, dim=0)\n",
    "\n",
    "        # return gradients for each forward argument\n",
    "        return grad_xi_batch, None, None, None, None\n",
    "\n",
    "\n",
    "# ------------------------------------------------------------------------------\n",
    "# Usage example:\n",
    "\n",
    "# Suppose:\n",
    "#   xi_batch   = torch.randn(B, p, requires_grad=True)\n",
    "#   sigma_batch= torch.rand(B, p)\n",
    "#   m_batch    = some tensor of shape (B, …)\n",
    "#   def full_residual(u, m):  return ...  # returns a vector (k,) constraint\n",
    "\n",
    "# Then project + differentiate:\n",
    "# u_proj = ProjectWeightedImplicit.apply(xi_batch, sigma_batch, m_batch, full_residual, 30)\n",
    "# loss = some_loss(u_proj)\n",
    "# loss.backward()\n",
    "# ------------------------------------------------------------------------------\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 101,
   "id": "4b1fc975-ec2a-42bd-b41d-23cb58640675",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([[0.0000e+00, 5.7483e-01, 7.1107e-01,  ..., 1.8838e-13, 0.0000e+00,\n",
       "         0.0000e+00],\n",
       "        [0.0000e+00, 6.2147e-01, 7.5383e-01,  ..., 1.2447e-13, 1.6838e-14,\n",
       "         8.3555e-14],\n",
       "        [0.0000e+00, 5.7201e-01, 7.0844e-01,  ..., 0.0000e+00, 0.0000e+00,\n",
       "         2.6018e-10],\n",
       "        ...,\n",
       "        [0.0000e+00, 6.0166e-01, 7.3584e-01,  ..., 1.8643e-13, 2.8532e-11,\n",
       "         0.0000e+00],\n",
       "        [1.4267e-07, 5.6744e-01, 7.0418e-01,  ..., 2.0048e-13, 0.0000e+00,\n",
       "         0.0000e+00],\n",
       "        [0.0000e+00, 6.5774e-01, 7.8616e-01,  ..., 1.7468e-13, 7.3325e-15,\n",
       "         0.0000e+00]], device='cuda:0', grad_fn=<ClampBackward1>)"
      ]
     },
     "execution_count": 101,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "u_proj"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 102,
   "id": "7406d28a-ad26-4a4c-a2b9-6e1f27a7c48f",
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "new_u_proj = ProjectWeightedImplicit.apply(torch.relu(mu_flat), var_flat, m_flat, full_residual, 30)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 103,
   "id": "a26ec383-d9da-43b4-bee8-f6253b0c39e7",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor(2.8406, device='cuda:0', grad_fn=<LinalgVectorNormBackward0>)"
      ]
     },
     "execution_count": 103,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "torch.norm(u_proj - new_u_proj)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 104,
   "id": "170963ba-2e5c-4ab8-be48-3eb9a5156e14",
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "from torch.func import vmap, jacrev\n",
    "\n",
    "def diag_JSigmaJT(sigma: torch.Tensor, J_u: torch.Tensor, eps: float = 1e-6) -> torch.Tensor:\n",
    "    \"\"\"\n",
    "    Compute diag(J Σ J^T) efficiently for one sample, given:\n",
    "      - sigma: Tensor (p,) forming Σ = diag(sigma)\n",
    "      - J_u:   Tensor (k, p) = ∂h/∂u at u*\n",
    "    Returns:\n",
    "      Tensor (p,) containing the diagonal of J Σ J^T.\n",
    "    \"\"\"\n",
    "    # 1) build S = J_u Σ J_u^T  (shape k×k)\n",
    "    S = (J_u * sigma.unsqueeze(0)) @ J_u.T\n",
    "    S = S + eps * torch.eye(S.shape[0], device=S.device)  # regularize\n",
    "\n",
    "    # 2) invert S  (k×k)  — cheap if k ≪ p\n",
    "    S_inv = torch.linalg.inv(S)\n",
    "\n",
    "    # 3) compute vᵢ = J_u[:,i], then vᵢᵀ S⁻¹ vᵢ for each i\n",
    "    #    but we can do it all at once:\n",
    "    T = S_inv @ J_u          # (k, p)\n",
    "    quad = (J_u * T).sum(dim=0)  # (p,)\n",
    "\n",
    "    # diag(J Σ J^T) = sigma - sigma² * quad\n",
    "    return sigma - sigma**2 * quad\n",
    "\n",
    "\n",
    "def batch_diag_JSigmaJT(u_proj: torch.Tensor,\n",
    "                        sigma_batch: torch.Tensor,\n",
    "                        m_batch: torch.Tensor,\n",
    "                        h_func,\n",
    "                        eps: float = 1e-6) -> torch.Tensor:\n",
    "    \"\"\"\n",
    "    Vectorized: for each sample in the batch, compute diag(J Σ J^T).\n",
    "    \n",
    "    Args:\n",
    "      u_proj:      (B, p)  — projected outputs u*\n",
    "      sigma_batch: (B, p)  — weight diagonals\n",
    "      m_batch:     (B, …)  — extra per-sample params\n",
    "      h_func:      callable h(u, m) → Tensor (k,) or scalar\n",
    "    Returns:\n",
    "      Tensor (B, p) of diag(J Σ J^T) for each batch element.\n",
    "    \"\"\"\n",
    "    def one(sigma, u, m):\n",
    "        # 1) build Jacobian of constraints J_u at u*\n",
    "        J_u = jacrev(lambda u_: h_func(u_, m))(u)  # (k,p) or (p,) if k=1\n",
    "        if J_u.ndim == 1:\n",
    "            J_u = J_u.unsqueeze(0)\n",
    "        # 2) compute diag(J Σ J^T)\n",
    "        return diag_JSigmaJT(sigma, J_u, eps)\n",
    "\n",
    "    # vmap over batch\n",
    "    return vmap(one)(sigma_batch, u_proj, m_batch)\n",
    "\n",
    "\n",
    "# -----------------------------------------------\n",
    "# Example usage, after your forward‐pass:\n",
    "\n",
    "# u_proj = ProjectWeightedImplicit.apply(xi_batch, sigma_batch, m_batch, h_func, max_iter)\n",
    "\n",
    "# Now get the variances:\n",
    "# variances = batch_diag_JSigmaJT(u_proj, sigma_batch, m_batch, h_func)\n",
    "\n",
    "# `variances[b,i]` is (J Σ J^T)_{ii} for sample b and dimension i.\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 105,
   "id": "c4af1764-71b4-4c60-8959-f4f08cf5fe42",
   "metadata": {},
   "outputs": [],
   "source": [
    "variances = batch_diag_JSigmaJT(new_u_proj, var_flat, m_flat, full_residual)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 106,
   "id": "0845b57b-0aeb-4ed6-a197-ee3afa9bd03d",
   "metadata": {},
   "outputs": [],
   "source": [
    "variances = variances.view(nf, nx, nt, 1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 107,
   "id": "1f791304-76ea-4e1f-8a8c-ce2d2937821b",
   "metadata": {},
   "outputs": [],
   "source": [
    "stds = torch.sqrt(variances)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 108,
   "id": "5825100a-2722-4f1b-b053-eed6dd9098e5",
   "metadata": {},
   "outputs": [],
   "source": [
    "stds = stds.detach().cpu()\n",
    "u_proj_reshaped = u_proj_reshaped.detach().cpu()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 109,
   "id": "31f5f22d-34f1-4b5a-819d-c553113c7722",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(0.1169)\n"
     ]
    },
    {
     "data": {
      "image/png": "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0tPTXe6fnp5eZ3F6u9pc57Kee+45xMfHl/uDolK1uc5btmzBRx99hL1799ZDhL6hNtf5xIkT+L//+z888MADWLNmDY4dO4axY8fCZDJh+vTp9RG216nNdb7//vuRmZmJvn37QggBs9mMJ554Ai+88EJ9hOw3KvoczMnJQWFhIQICAurkftlyQz5l9uzZWLFiBb755hsYDAZPh+MzcnNz8dBDD2HJkiWIjIz0dDg+TVVVREdH44MPPkBiYiKGDRuGF198EYsXL/Z0aD5l48aNmDlzJhYuXIjdu3fj66+/xurVq/Haa695OjRyA7bcVFNkZCQURUFGRobT9oyMDMTGxro8JjY2tkb7U+2us83bb7+N2bNnY926dejSpUtdhun1anqdjx8/jrS0NNx22232baqqAgA0Gg0OHz6M1q1b123QXqg2r+e4uDhotVooimLf1qFDB6Snp8NoNEKn09VpzN6oNtf55ZdfxkMPPYRHHnkEAHD11VcjPz8fjz32GF588UXIMr/7u0NFn4OhoaF11moDsOWm2nQ6HRITE7F+/Xr7NlVVsX79eiQlJbk8JikpyWl/APjpp58q3J9qd50BYM6cOXjttdewdu1a9OjRoz5C9Wo1vc5XXXUV9u/fj71799p/br/9dlx//fXYu3cvEhIS6jN8r1Gb13OfPn1w7Ngxe/IIAEeOHEFcXBwTmwrU5joXFBSUS2BsCaXgkotu47HPwTotV/YxK1asEHq9XixdulQcPHhQPPbYYyI8PFykp6cLIYR46KGHxPPPP2/f/5dffhEajUa8/fbb4tChQ2L69OkcCl4NNb3Os2fPFjqdTqxatUqcO3fO/pObm+uph+AVanqdy+Joqeqp6XU+deqUCAkJEePHjxeHDx8W3333nYiOjhavv/66px6CV6jpdZ4+fboICQkRX375pThx4oT48ccfRevWrcW9997rqYfgFXJzc8WePXvEnj17BADxzjvviD179oiTJ08KIYR4/vnnxUMPPWTf3zYUfMqUKeLQoUPivffe41DwhmjBggWiWbNmQqfTiZ49e4pff/3VfltycrIYOXKk0/7/+te/RLt27YROpxOdOnUSq1evrueIvVNNrnPz5s0FgHI/06dPr//AvUxNX8+OmNxUX02v89atW0WvXr2EXq8XrVq1Em+88YYwm831HLX3qcl1NplM4pVXXhGtW7cWBoNBJCQkiLFjx4pLly7Vf+BeZMOGDS7fb23XduTIkSI5ObncMd26dRM6nU60atVKfPLJJ3UepyQE29+IiIjId7DmhoiIiHwKkxsiIiLyKUxuiIiIyKcwuSEiIiKfwuSGiIiIfAqTGyIiIvIpTG6IiIjIpzC5ISIiIp/C5IaIiIh8CpMbIiIi8ilMbojI6124cAGxsbGYOXOmfdvWrVuh0+nKrUhMRL6Pa0sRkU9Ys2YNhg4diq1bt6J9+/bo1q0bhgwZgnfeecfToRFRPWNyQ0Q+Y9y4cVi3bh169OiB/fv3Y8eOHdDr9Z4Oi4jqGZMbIvIZhYWF6Ny5M06fPo1du3bh6quv9nRIROQBrLkhIp9x/PhxnD17FqqqIi0tzdPhEJGHsOWGiHyC0WhEz5490a1bN7Rv3x7z5s3D/v37ER0d7enQiKieMbkhIp8wZcoUrFq1Cvv27UNwcDCSk5MRFhaG7777ztOhEVE9Y7cUEXm9jRs3Yt68efjss88QGhoKWZbx2WefYfPmzVi0aJGnwyOiesaWGyIiIvIpbLkhIiIin8LkhoiIiHwKkxsiIiLyKUxuiIiIyKcwuSEiIiKfwuSGiIiIfAqTGyIiIvIpTG6IiIjIpzC5ISIiIp/C5IaIiIh8CpMbIiIi8ilMboiIiMin/D9maWRmQarUEQAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# t_idx = len(t[slice(*tpred)])//2\n",
    "t_idx = 2\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, u_proj_reshaped[parameters_idx,:,t_idx,0], '--', lw=2, label = \"predicted $\\mu$ and $\\pm 3\\sigma$ (varFNO)\")\n",
    "        plt.fill_between(grid, u_proj_reshaped[parameters_idx,:,t_idx,0]+3*stds[parameters_idx,:,t_idx,0], u_proj_reshaped[parameters_idx,:,t_idx,0]-3*stds[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] - u_proj_reshaped[parameters_idx,:,t_idx,0]))\n",
    "        plt.legend()\n",
    "        # plt.ylim(-1.0,1.5)\n",
    "        plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 110,
   "id": "9f02c9bc-41d9-42b5-b108-ab8af7c7bf6a",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAABVYAAANrCAYAAACtDcNbAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/bCgiHAAAACXBIWXMAAC4jAAAuIwF4pT92AAEAAElEQVR4nOzdd5xcdb3/8df0nZ2Z3Sk7szXbsumdhBbAhI4CAoKAgICiXDv3il6v16s/vTYseMWKogKKIghIL0EgoQVI720323ub7Ts77ffHmjFLtpItyeb9fDzyeJwzc873fD+z+5nJfuZ7vl9DPB6PIyIiIiIiIiIiIiKjZpzqDoiIiIiIiIiIiIgcb1RYFRERERERERERERkjFVZFRERERERERERExkiFVREREREREREREZExUmFVREREREREREREZIxUWBUREREREREREREZIxVWRURERERERERERMZIhVURERERERERERGRMVJhVURERERERERERGSMVFgVERERERERERERGSMVVkVERERERERERETGSIVVERERERERERERkTFSYVVERERERERERERkjFRYFRERERERERERERkjFVZFRERERERERERExkiFVREREREREREREZExUmFVREREREREREREZIxUWBUREREREREREREZIxVWRURERERERERERMZIhVURERERERERERGRMVJhVURERERERERERGSMVFgVERERERERERERGSMVVkVERERERERERETGSIVVERERERERERERkTFSYVVERERERERERERkjFRYFRERERERERERERkjFVZFRERERERERERExsg81R2QE0MwGGTdunWJ/RkzZmCz2aawRyIiIiIiIiIicrwLhUJUVlYm9letWoXb7Z6Ua6uwKpNi3bp1XH755VPdDRERERERERERmcYef/xxLrvsskm5lqYCEBERERERERERERkjFVZFRERERERERERExkhTAcikmDFjxoD9xx9/nKKioinqjYiIiIiIiIiITAfFxcUDpp98dw1qIqmwKpPi3QtVFRUVsWDBginqjYiIiIiIiIiITEeTuVi6pgIQERERERERERERGSMVVkVERERERERERETGSIVVERERERERERERkTHSHKsiRyEcDtPQ0JDYDwQCWCyWKeyRyPSg3BIZf8orkYmh3BIZf8orkYmh3Bp/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5vH3v/+duXPn8vbbb9PQ0EAmk+kyxrZtRowYwQknnMDJJ5/MhRdeyPTp04s0YxEREZG+Meb4OkIRm1h4Jd+Z/wrOiPzF1df9Gbh/nMutpsGY4+r6eKYiIiIiIoNXvyqsLl26lEcffZTHHnuMLVu2dPksCIJu413XZe3ataxdu5Ynn3ySL3/5ywwbNoxPfOITXHXVVUyePLmvpi4iIiLSp+qPruGcsEU0vIaZc16F+jw9V02bNzmFb//+dW43DEZNGdbHMxURERERGZzMYk8A4Pnnn+eiiy7iuOOO495772XLli0EQdDlkc97x23ZsoV7772X448/nosuuojnn3++DyMRERER6Ts1oys5/YKj+Y+z4jhbX8s7zjAt3nZO4+u/XcGGpVv7cIYiIiIiIoNXUVeszp8/ny9+8Yu88MILwN5VqYZhMGXKFM466yymTp3Ksccey/Dhw6muriYWixEEAYlEgm3btrFx40aWLVvG22+/zcsvv8ySJUuy53n22Wd59tlnOffcc/nud7/LtGnTihWqDFKe59Ha2pp9HY/HsSyriDMSGRyUWyIHb8iwOKecP57/cNZxzyuzaa8+Nec4w7RYEjmD23/5Ml/+cIKjjx+pvBIpAP2ZJVJ4yiuR3qHcKryiFVavv/56fvazn+H7PkEQYBgG559/Ph/96Ee54oorqKmp2e/x8XiceDzOuHHjOOuss7Lvb9u2jSeeeILf/e53PPfccwRBwAsvvMApp5zCNddcw0MPPdTbockRxPM8mpubs69jsZh+UxIpAOWWSM+UVpVw0gXjudtZx9dfnM2uyjzFVcNkTek5fPXXs/heJMLoY9UWQORw6c8skcJTXon0DuVW4RWtFcDDDz+M53nE43Fuu+02Vq5cyd///neuvfbaAxZV96empoZrr72Wv/3tb6xatYovfvGLxONxPM/j4YcfLmAEIiIiIv1HNB5m+vlj+Y/zq6hqnr3fsZsr3seXH57D5ncb+2h2IiIiIiKDT9EKq/F4nLvvvpv169fzrW99i7Fjxxb8GmPGjOHb3/4269ev56677qK0tLTg1xARERHpL0JRhxPOG8vd769haMucvOPaNs7n6b88ytf+83/YsnJ7H85QRERERGTwKFphddWqVXz1q1+lrKys169VVlbG1772NVatWtXr1xIREREpJidkc/y5Y7n74lqGt3Uvria2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6/7+5V/+xQ4ODs66v2w2az/72c+2dZ/LFSNWccZ69NFHddlll2nfvn0yxujNb36zvvnNb+quu+7Sxz/+ca1bt05xHOsDH/iA/vZv/3ZB+/rEJz6hD3zgA4rjWOvWrdMnPvEJ3XXXXfrmN7+pN7/5zTLGaN++fbrsssv06KOPtqmHwOJbjLoql8s6fPiwJOmcc87Ru9/9bt122226++67dffdd+umm27SeeedJ0m69957demll9aXB5arxfw/60RJkui3f/u3VSqVtGbNmrZuG+i2xawta61+93d/V9u2bVO5XNZTn/pU/cVf/IW+9a1v6f7779d3v/td3XTTTXrVq16lfD7fph4Ci2+x6uqJJ57Qr/zKr2jPnj2SpAsuuEC33HKLvve97+n222/Xpz71KW3dulWStGPHDr3kJS/R5ORkW/oILDZrbf3ywMCALrnkEq1du7Yj+7rnnnv0mte8RqOjowqCQH/4h3+oO+64Q3feeaf+/M//XP39/SqVSrrmmmv01a9+tSNtWFa6newC3fLKV76y/m3LTTfdNOPve/bssatXr7aSbD6ft0888cS89vPEE0/YfD5vJdnh4WG7Z8+eGcvceOON9ba86lWvmtd+gKVgMepq37599pJLLrHf+ta3Zl2mXC5Pa8tb3/rWlvcDLCWL9X9WM3/9139tJdn169fbj33sY4xYxWllMWursX7e8pa32EqlMuuy5XJ53vsBum2x6uqd73xnfT+vf/3rbZIkM5Ypl8v2RS96UX25j33sY/PaF9BtH//4x+0Xv/hFu2vXrvpz/eKLL277iNUkSexzn/tcK8kaY+z/+T//Z8Yy999/f/1IpvXr19vJycm27Hu5IljFGWnHjh31F6AXvvCFsy53880315d773vfO699/cEf/EF9GycbKv/CF76wvtzOnTvntS+gmxazruZiZGTEBkFgJdmhoaGmb7aB5aCbtfWzn/3M5nI5K8l+9atftbfccgvBKk4bi1lb+/fvr3/Rfskll/B/Ek5bi1lX559/fn0bu3btmnW5f//3f68vd8UVV8xrX8BS1Ilg9Rvf+EZ9m294wxtmXe7aa6+tL3fDDTe0Zd/LFVMB4Ix066231i9fffXVsy73G7/xG/VDsRrXmc++8vm8Xv/618+63FVXXVW//OUvf3le+wK6aTHrai5Wr16tZz3rWZKko0eP6siR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F+jRRx9VHMf62c9+puc///kt7Q84E/T09Ojss8/WY489pocfflhhGMr3/VmXr9VhEATaunXrYjVzyeEYL5yRLr744vrl22+/fdblisWi7rrrLknSli1btGnTppb2c9ZZZ9Xn0Lrrrrvqb6ibaWxHY/uA5WKx6qrRgQMHdOmll2rXrl2S0jPDcrIPnG66UVvAmWCxast1Xb3oRS+qX699ETibhx9+uH55w4YNLe0L6LbFqqvGLx3CMDzl8o3LnCwoAs50tRpurNFmdu/erUcffVSS9Iu/+Itn9BeBBKs4I73mNa+pX7755ptnXe6LX/yiCoXCjHXms6/JyUl94QtfmHW5xnbMd19ANy1mXUkzQ9U/+IM/0Ic+9KF5bw9YqhartkZHR2WtPenPLbfcUl9+27Zt9duvu+66lvcHdNti/r/1ute9rn75H//xH2dd7qGHHtIPf/hDSdLP/dzPae3atfPaH9Ati1VXT3nKU+qXTxbgSukRGXfeeackyXGcWU8eB2B6PX7mM5+ZdbnPfvazTdc5I1ngDPWKV7zCSrKS7E033TTj73v27LHDw8NWks3lcnbfvn0zltm2bVt9G9u2bWu6n3379tlcLmcl2eHhYbt3794Zy9x444317Vx++eUL7hvQLYtVVwcOHLBPf/rT68u95z3vaXdXgCVlsWrrVG655ZYFbwNYShartkqlkt2yZYuVZIMgsHfeeeeMZYrFor344ovr2/rkJz+54P4B3bAYdZUkiT377LPry/z1X/9107YkSWLf/va315e75JJLFtw/YKlo/D/jscceO+Xyt99+e335iy++uOkycRzb5zznOVaSNcbYf/u3f5uxzP3332+z2ayVZNetW2cnJycX2JPl7cwdq4sz3kc/+lHdeeedOnr0qN7ylrfo/vvv15VXXqne3l7dc8892r59u0ZGRiRJ27dvn/ehWBs2bNCf//mf613vepdGRkZ04YUX6o//+I914YUXamJiQl/+8pd10003SZKGhob00Y9+tG19BBbbYtTVkSNH9OIXv1g/+9nPJEmXX3653vSmN+mBBx446XpbtmxRT09P650CloDF+j8LONMsVm1lMhnddNNNuuyyy1SpVPSSl7xE73jHO/Srv/qr6unp0QMPPKCPfOQj+slPfiJJuuSSS/TmN7+5bf0EFtNi1JUxRh/96Ed1xRVXyFqr97znPfrOd76jN7zhDdq6dauSJNHOnTv12c9+Vt/73vckpXX44Q9/uK19BRbLww8/XH8u1zz55JP1y7feeqtWrVpVv97b2zuvkaSO4+iGG27QpZdeqnK5rMsvv1zvete79Gu/9mvyPE/f+c539Jd/+ZcqlUoyxuiTn/yk8vn8/Dt2Ouh2sgt00w9+8AO7fv36+rc2J/44jnPSETmtjP75wAc+YB3HmXVf69evtz/4wQ/a20GgCzpdV43ftLbyc/vtt3esz8BiWMz/s2bDiFWcjhaztm699VY7MDBw0v+vfu3Xfs2Ojo62t5PAIlusuvqf//N/2p6enlO+D1y9erX9xje+0f6OAouk8T3YXH42b948YxtzGbFa87Wvfe2k/19ls9mmI9LPRIxYxRntoosu0o4dO/SpT31KX/nKV/TII4+oVCpp3bp1uvTSS/XWt771lGeZnKvrr79er3jFK/TpT39at99+uw4cOKBsNqunPOUpevWrX623ve1tGhwcbMu+gG5azLoCziTUFtAZi1lbV1xxhV7wghfo05/+tP7lX/5Fjz/+uAqFgoaHh3XRRRfpTW96k17+8pe3ZV9ANy1WXf32b/+2XvrSl+qWW27RN7/5Te3cuVOjo6MyxmjlypU677zz9Gu/9mt64xvfqBUrVrShZ8CZ4fLLL9eOHTv0yU9+Uv/yL/+i3bt3K0kSbdiwQS996Uv1tre9TU9/+tO73cwlwVhrbbcbAQAAAAAAAADLidPtBgAAAAAAAADAckOwCgAAAAAAAAAtIlgFAAAAAAAAgBYRrAIAAAAAAABAiwhWAQAAAAAAAKBFBKsAAAAAAAAA0CKCVQAAAAAAAABoEcEqAAAAAAAAALSIYBUAAAAAAAAAWkSwCgAAAAAAAAAtIlgFAAAAAAAAgBYRrAIAAAAAAABAiwhWAQAAAAAAAKBFBKsAAAAAAAAA0CKCVQAAAAAAAABoEcEqAAAAAAAAALSIYBUAAAAAAAAAWkSwCgAAAAAAAAAtIlgFAAAAAAAAgBYRrAIAAAAAAABAiwhWAQAAAAAAAKBFBKsAAAAAAAAA0CKCVQAAAAAAAABoEcEqAAAAAAAAALSIYBUAAAAAAAAAWkSwCgAAAAAAAAAtIlgFAAAAAAAAgBYRrAIAAAAAAABAiwhWAQAAAAAAAKBFBKsAAABAlzz66KPq7++XMUbGGH3oQx865Tq///u/X19+06ZNOnbs2CK0FAAAACcy1lrb7UYAAAAAZ6rPf/7zeuMb3yhJ8n1fP/jBD3T++ec3XfYb3/iGXv7yl0uSHMfRt771LV1yySWL1VQAAAA0YMQqAAAA0EW/+Zu/qV//9V+XJIVhqNe//vWanJycsdyTTz6p3/7t365ff+9730uoCgAA0EWMWAUAAAC67Pjx43r2s5+t3bt3S5Kuuuoq3XzzzfW/W2v1q7/6q/r3f/93SdIFF1yg73//+/J9vyvtBQAAACNWAQAAgK4bGBjQ3//938t1XUnSZz/7WX3lK1+p//1jH/tYPVTt6enRF77wBUJVAACALmPEKgAAALBEbNu2TX/6p38qSRoaGtKPf/xjHT58WM9//vNVLpclSTfffLOuuuqqbjYTAAAAIlgFAAAAlow4jvXf//t/1/e//31J0qWXXqqDBw9q586dkqT/8T/+h2677bZuNhEAAABVBKsAAADAEvLYY4/pOc95jsbGxqbdvmHDBv34xz/W0NBQl1oGAACARsyxCgAAACwhW7Zs0Q033DDtNmOM/u7v/o5QFQAAYAkhWAUAAACWmOHh4WnX161bp+c///ldag0AAACaIVgFAAAAlpDDhw/rTW9607Tb9u/fr3e+853daRAAAACaIlgFAAAAlpCrrrpKBw4ckCRt3bpVrutKkj772c/qq1/9ahdbBgAAgEYEqwAAAMAScdNNN+lrX/uaJCmXy+nrX/+6/uRP/qT+96uuukr79+/vVvMAAADQwFhrbbcbAQAAAJzpfvazn+kXfuEXVCgUJEk33HCD3vrWtyqOY73oRS/SD37wA0nSL/3SL+n//b//J2NMN5sLAABwxmPEKgAAANBllUpFr3/96+uh6ite8Qq99a1vlSS5rqu///u/V19fnyTpW9/6lj7ykY90ra0AAABIEawCAAAAXfYnf/In+q//+i9J0tq1a/W5z31u2t/POeccffKTn6xfv/baa/XDH/5wMZsIAACAEzAVAAAAANBF3/72t/XLv/zLSpJExhj967/+q172spc1XfZ1r3udvvSlL0mSnvGMZ+i+++5TLpdbzOYCAACgihGrAAAAQJccPXpUb3zjG5UkiSTp7W9/+6yhqiTdeOONOuussyRJDz74oN797ncvSjsBAAAwEyNWAQAAgC55zWteo9tuu02SdN555+k///M/lc1mT7rOd7/7XV166aX1MPaf//mf9YpXvKLjbQUAAMB0BKsAAAAAAAAA0CKmAgAAAAAAAACAFhGsAgAAAAAAAECLCFYBAAAAAAAAoEUEqwAAAAAAAADQIoJVAAAAAAAAAGgRwSoAAAAAAAAAtIhgFQAAAAAAAABaRLAKAAAAAAAAAC0iWAUAAAAAAACAFhGsAgAAAAAAAECLCFYBAAAAAAAAoEUEqwAAAAAAAADQIoJVAAAAAAAAAGgRwSoAAAAAAAAAtIhgFQAAAAAAAABaRLAKAAAAAAAAAC0iWAUAAAAAAACAFhGsAgAAAAAAAECLCFYBAAAAAAAAoEUEqwAAAAAAAADQIoJVAAAAAAAAAGgRwSoAAAAAAAAAtIhgFQAAAAAAAABaRLAKAAAAAAAAAC0iWAUAAAAAAACAFhGsAgAAAAAAAECLCFYBAAAAAAAAoEUEqwAAAAAAAADQIoJVAAAAAAAAAGgRwSoAAAAAAAAAtIhgFQAAAAAAAABaRLAKAAAAAAAAAC0iWAUAAAAAAACAFhGsAgAAAAAAAECLCFYBAAAAAAAAoEUEqwAAAAAAAADQov8fglnEVv9vScMAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 1350x840 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "L2 error: 0.11060722917318344\n"
     ]
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "\n",
    "t_idx = 10\n",
    "\n",
    "for parameters_idx in range(0, 1, 5):\n",
    "    with torch.no_grad():\n",
    "        # Set compact figure size and high DPI\n",
    "        fig = plt.figure(figsize=(4.5, 2.8), dpi=300)\n",
    "\n",
    "        time_label = t[slice(*tpred)][t_idx].item()\n",
    "        param_val = x_ood_test[parameters_idx, 0, 0, 0].item()\n",
    "        true_vals = y_ood_test[parameters_idx, :, t_idx, 0]\n",
    "        pred_mu_conserv = new_mu[parameters_idx, :, t_idx, 0]\n",
    "        pred_std_conserv = new_std[parameters_idx, :, t_idx, 0]\n",
    "        pred_mu_e2e = u_proj_reshaped[parameters_idx, :, t_idx, 0]\n",
    "        pred_std_e2e = stds[parameters_idx, :, t_idx, 0]\n",
    "\n",
    "        # Plot true solution\n",
    "        plt.plot(grid, true_vals, color=\"black\", lw=1.5, label=\"True\", zorder=3)\n",
    "\n",
    "        # ProbConserv\n",
    "        plt.plot(grid, pred_mu_conserv, '--', lw=1.2, color=\"#ef233c\",\n",
    "                 label=r\"ProbConserv\", zorder=2)\n",
    "        plt.fill_between(grid,\n",
    "                         pred_mu_conserv + 3 * pred_std_conserv,\n",
    "                         pred_mu_conserv - 3 * pred_std_conserv,\n",
    "                         color=\"#ef233c\", alpha=0.3, label=\"_nolegend_\", zorder=1)\n",
    "\n",
    "        # ProbHardE2E\n",
    "        plt.plot(grid, pred_mu_e2e, '--', lw=1.5, color=\"#3a86ff\",\n",
    "                 label=r\"ProbHardE2E\", zorder=4)\n",
    "        plt.fill_between(grid,\n",
    "                         pred_mu_e2e + 3 * pred_std_e2e,\n",
    "                         pred_mu_e2e - 3 * pred_std_e2e,\n",
    "                         color=\"#3a86ff\", alpha=0.3, label=\"_nolegend_\", zorder=3)\n",
    "\n",
    "        # Labels\n",
    "        plt.xlabel(\"x\", fontsize=8)\n",
    "        plt.ylabel(r\"$u(x, t={:.2f})$\".format(time_label), fontsize=8)\n",
    "\n",
    "        # Ticks and grid\n",
    "        plt.xticks(fontsize=7)\n",
    "        plt.yticks(fontsize=7)\n",
    "        plt.grid(True, linestyle=\"--\", alpha=0.4)\n",
    "\n",
    "        # Legend\n",
    "        plt.legend(fontsize=7, loc=\"upper right\", frameon=False)\n",
    "\n",
    "        # Layout for Overleaf\n",
    "        plt.tight_layout(pad=0.3)\n",
    "        plt.show()\n",
    "\n",
    "        print(\"L2 error:\", torch.norm(true_vals - pred_mu_e2e).item())\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 111,
   "id": "e2918b7b-cab2-4da1-8179-9d49cdd14170",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(0.1675)\n"
     ]
    },
    {
     "data": {
      "image/png": "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bJnNZitXtVRV6du3L+3atePrr78udZuOHTvy7rvvcv/999O8eXP++te/ApCcnExeXh4DBw5k27ZtjB49mjZt2vDxxx8zcuRIjh8/zn333RdyrJNdHyVZunQpH330EWPGjMHpdPLyyy9z4YUXsnLlSjp37gzAqlWr+PHHH7n22mtp3rw5O3fu5JVXXmHgwIFs3ryZ6OjokGPec889JCcnM2nSJHJycip0jHvvvZfU1FSeeOIJfvrpJ1577TUaNGjAjz/+SMuWLXn22WdZuHAhzz//PJ07d2b48OHBfe+8807mzp3LzTffzJgxY0hLS+Mf//gHa9eu5YcffsButzNjxgzuvfdeYmNjg+1GGjduDPjbowwYMIA9e/Zw55130rJlS3788UcmTJjAvn37mDFjRoXPeUZGBl6v96SvCfiT+5KqVkty5MgRTNMkPT2dJ598EoBBgwaVad8TFbRLS0pKCi775ptvAHA6nfTq1YvVq1fjcDi4/PLLefnll095jYlqFu6iIxE5ylItdeutt6omTZqow4cPhyy/9tprVUJCQrB41+fzFSvGPnbsmGrcuLG65ZZbgssKqj/i4+PVwYMHQ7YvqI4pur1SSl1++eWqUaNGIctatWqlRowYUey5DB48WFmWFVx+//33K8Mw1PHjx5VS/moam82mLrvsspDjPf744woIOWZpAHXrrbeqQ4cOqYMHD6oVK1aoQYMGKUBNnz49JMaLL75Y+Xw+lZqaqp566imllFKbN29WgFq6dGmJr0HBeSjpr0OHDqeMrzQVrZYqMGzYMAWojIyMkz5OwfMuasaMGQpQ7733XnCZx+NRffv2VbGxsSozMzMkjpKuj9IUnJuff/45uGzXrl3K5XKpyy+/PLispKqI5cuXK0C98847wWUFr0n//v2Vz+cL2b68xxg6dGjI9di3b1+laZq66667gst8Pp9q3rx5SNXS999/rwA1b968kMdatGhRseWlVUs99dRTKiYmRm3dujVk+cMPP6wMw1Dp6elKqYqd8wEDBpR6jRb9K8v7qYDT6Qzu16hRI/Xiiy+Wed+i8vPzVadOnVSbNm2U1+sNLv/zn/8cPPYNN9yg5s+fryZOnKhsNps655xzQl4nUfOk5EbUGKUUn3zyCVdffTVKKQ4fPhxcN3ToUD788EPWrFlDv379MAwDwzAAf3H68ePHsSyLXr16sWbNmmLHvvLKK0lOTi7xce+6666Q++eeey6fffYZmZmZxMfHnzTmO+64I6RH0Lnnnsvf//53du3aRdeuXVm8eDE+n4977rknZL97772Xxx9//KTHLuqNN97gjTfeCN53uVyMGzeuxG7PhmFw9dVX88EHH/DYY48xb948WrRowbnnnsuOHTtKfYxPPvmk2PONiYkpc4xVreAXeFZW1ilfhxMtXLiQ1NRUrrvuuuAyu93OmDFjuO6661i6dCmXXHJJcN3Jro+S9O3bl549ewbvt2zZkmHDhvF///d/mKaJYRgh1XNer5fMzExOO+00GjRowJo1a4r16rr99tuD13SB8h7j1ltvDbke+/Tpw/Lly7n11luDywzDCJYkFPj4449JSEjgggsuCHnf9ezZk9jYWL799luuv/76k56Tjz/+mHPPPZfExMSQYwwePJipU6fy3XffccMNNwSXl+ecT58+PaR6rDQnawx8oi+++AK3282vv/7Ke++9FywtK6/Ro0ezefNmFixYgM1W+JWZnZ0NwFlnncV7770H+J9zdHQ0EyZMYPHixaX2bBTVT5IbUWMOHTrE8ePHee2113jttddK3ObgwYPB22+//TbTp0/nt99+CymybtOmTbH9SlpWoGXLliH3ExMTAX/Vz6m+VE+2L/jbTADFqo8aNmwY3LYshg0bxujRo9E0jbi4OM4888yTJh7XX389L774IuvXr+f999/n2muvLbVbdoHzzjsvpFg93Aq+HOLi4sq9765du2jfvn2xHm4dO3YMri/qZNdHSdq3b19s2emnn05ubi6HDh0iNTWVvLw8pkyZwltvvcWePXtC2rhkZGQU27+kGMp7jBOvx4SEBABatGhRbHnRZOH3338nIyODlJSUEp9v0fddaX7//Xd++eWXUhOWE49RnnNeNJGsKn/84x8B+NOf/sSwYcPo3LkzsbGxjB49uszHeP7555kzZw5PPfUUF110Uci6gsS0aIIN/vfmhAkT+PHHHyW5CSNJbkSNKWh0eOONNzJixIgSt+natSsA7733HiNHjuSyyy5j/PjxpKSkYBgGU6ZMYfv27cX2O1kj1xN/LRdQJzS4rOp9y6N58+bl+iDs06cP7dq1Y+zYsaSlpZ3yV3dttHHjRlJSUspdalMRlW0EXZJ7772Xt956i7Fjx9K3b18SEhLQNI1rr702pIHtyWIo7zFKux5LWl70GrUsi5SUFObNm1fi/mUpYbEsiwsuuIAHH3ywxPWnn356yP3ynPOjR4/i8XhOuV1UVFQwoSuPdu3a0aNHD+bNm1fm5Gbu3Lk89NBD3HXXXTz22GPF1heUIhW0SSpQkECWpSRKVB9JbkSNSU5OJi4uDtM0T/lFPn/+fNq2bcunn34aUiJR28Y6adWqFQDbtm0L+aV65MiRav9wu+6663j66afp2LFjsQbWtd3y5cvZvn07N954Y4X2b9WqFb/88guWZYWU3vz222/B9ZXx+++/F1u2detWoqOjg4nA/PnzGTFiBNOnTw9u43a7T9kjp6iqOEZZtGvXjm+++YZ+/fqdMukorQSwXbt2ZGdnV0tpxBVXXMHSpUtPud2IESMqPMBgXl4e+fn5Zdr2X//6F7fddhtXXHEFL730Uonb9OzZkzlz5rBnz56Q5Xv37gXKljCK6iNdwUWNMQyDK6+8kk8++YSNGzcWW1+0i3XBL9Givz5XrFjB8uXLqz/Qchg0aBA2m41XXnklZPk//vGPan/s2267jcmTJ4d8MdYFu3btYuTIkTgcDsaPH1+hY1x00UXs378/pNeaz+dj1qxZxMbGMmDAgErFuHz58pC2Xbt37+Zf//oXQ4YMCV6bhmEUK8GbNWtWuboDV8UxyuLqq6/GNE2eeuqpYut8Pl9IMhUTE1NicnX11VezfPlyvvzyy2Lrjh8/js/nq3B806dP5+uvvz7lX2mlRkWfS0k/KlauXMmGDRvo1atXyPLffvut2GjZ3333Hddeey3nnXce8+bNK3Vwz2HDhuF0OnnrrbdCStlef/11wD8wpwgfKbkRVe7NN99k0aJFxZbfd999TJ06lW+//ZY+ffpw++2306lTJ44ePcqaNWv45ptvgt2VL7nkEj799FMuv/xyLr74YtLS0pg9ezadOnUKttWoDRo3bsx9993H9OnT+fOf/8yFF17I+vXr+eKLL0hKSjplO5jKaNWqVbkaLc+fP7/EbrQXXHBBsaL10uzatYt3330XIDjFw9NPPx2M58QGsGvWrOG9994LNgpftWoVn3zyCZqm8e677warIcvrjjvu4NVXX2XkyJGsXr2a1q1bM3/+fH744QdmzJhRoXY8RXXu3JmhQ4eGdAUHeOKJJ4LbXHLJJbz77rskJCTQqVMnli9fzjfffEOjRo3K/DhVcYyyGDBgAHfeeSdTpkxh3bp1DBkyBLvdzu+//87HH3/MzJkzueqqqwB/icQrr7zC008/zWmnnUZKSgrnn38+48eP59///jeXXHIJI0eOpGfPnuTk5LBhwwbmz5/Pzp07K9ymq6ra3GRnZ9OiRQuuueaaYLu1DRs28NZbb5GQkMDEiRNDtu/YsSMDBgwITnWya9cu/vznP6NpGldddRUff/xxyPZdu3YNXrOpqak8+uijTJo0iQsvvJDLLruM9evXM2fOHK677jrOOuusKnlOomIkuRFV7sRSjAIjR46kefPmrFy5kieffJJPP/2Ul19+mUaNGnHmmWeGjDszcuRI9u/fz6uvvsqXX35Jp06deO+99/j444+LzbkUbs899xzR0dHMmTOHb775hr59+/LVV1/Rv39/XC5XuMMLuvvuu0tc/u2335Y5uUlLSyv2BVFwf8CAAcWSmw8++IAPPvgAm81GfHw87du3Z+zYsdx1113FGseWR1RUFEuWLOHhhx/m7bffJjMzkw4dOvDWW29VybQBAwYMoG/fvjzxxBOkp6fTqVMn5s6dG5KMzZw5E8MwmDdvHm63m379+vHNN9+UayDFqjhGWc2ePZuePXvy6quv8sgjj2Cz2WjdujU33ngj/fr1C243adIkdu3axbRp08jKymLAgAGcf/75REdHs3TpUp599lk+/vhj3nnnHeLj4zn99NN54oknKtQWpqpFR0dz22238e233zJ//nzy8vJo2rQp1113HY899lhwML/SpKWlBRtyjxo1qtj6yZMnh1wDjz32GImJicyaNYuxY8eGJDwivDRV1S0jhRAcP36cxMREnn76aZlEr47RNI1Ro0bVSNWiEKJ6SJsbISopLy+v2LKC0VqLDl0vhBCiZki1lBCV9NFHHzF37lwuuugiYmNjWbZsGR988AFDhgwJKe6vzQ4dOnTSRqwOh0OGkxdC1BmS3AhRSV27dsVmszFt2jQyMzODjYwLGtrWBWeddVaxge+KKtroUgghajtpcyOE4Icffiixeq1AYmJitYwiK4QQ1UGSGyGEEEJEFGlQLIQQQoiIUu/a3FiWxd69e4mLi6vWAdaEEEIIUXWUUmRlZdG0adNSR44uUO+Sm7179xabQVcIIYQQdcPu3btp3rz5Sbepd8lNwbDsu3fvrpHZiIUQQghReZmZmbRo0aJM06vUu+SmoCoqPj5ekhshhBCijilLkxJpUCyEEEKIiCLJjRBCCCEiiiQ3QgghhIgo9a7NjRD1nWmaeL3ecIchhBDFOByOU3bzLgtJboSoJ5RS7N+/n+PHj4c7FCGEKJGu67Rp0waHw1Gp40hyI0Q9UZDYpKSkEB0dLYNYCiFqlYJBdvft20fLli0r9RklyY0Q9YBpmsHEplGjRuEORwghSpScnMzevXvx+XzY7fYKH0caFAtRDxS0sYmOjg5zJEIIUbqC6ijTNCt1HEluhKhHpCpKCFGbVdVnlCQ3QgghhIgoktwIIYQQIqJIciOEEEKIiCLJTRVavns5n/36WbjDEEJU0sCBAxk7dmyx2+GMoy6oK/EeOXKElJQUdu7cWSOPV1fOS3W49tprmT59eo0/riQ3VWT8V+M5581zGP75cCzLCnc4Qogq8umnn/LUU0+Vefv6/EVWlV555RW6du1KfHw88fHx9O3bly+++KJKjv3MM88wbNgwWrduXSXHE6V77LHHeOaZZ8jIyKjRx5Xkporc1O0mALI92UxcMjHM0QhRv3k8nio7VsOGDYmLi6uy44lCAwcOZO7cuSWua968OVOnTmX16tX8/PPPnH/++QwbNoxNmzZV6jFzc3N54403uPXWWyt1HKja66yuKetz79y5M+3ateO9996r5ohCSXJTRbo27krf5n0BmP7jdDy++nvRC1HVBg4cyOjRoxk9ejQJCQkkJSUxceJElFIh68eOHUtSUhJDhw4F/COeTpkyhTZt2hAVFUW3bt2YP39+yLFzcnIYPnw4sbGxNGnSpFgR+oklMZZlMW3aNE477TScTictW7bkmWeeAWDkyJEsXbqUmTNnomkamqaxc+fOKomjJMuWLcNut+N2u4PLdu7ciaZp7Nq1q8R9Fi1aRP/+/WnQoAGNGjXikksuYfv27cWe85gxY3jwwQdp2LAhqampPP7445WOtzwuvfRSLrroItq3b8/pp5/OM888Q2xsLD/99FPIdj/99BODBg2iUaNGwXNe8JeZmVnsuAsXLsTpdHL22WcXe84nu8aKblP0OsvPz2fMmDGkpKTgcrno378/q1atKva4Pp/vpMc+2XVVlscoy2s2f/58unTpQlRUFI0aNWLw4MHk5OQEH/9k12hJz/21116jadOmxWorhg0bxi233BLyWn744YfFzkm1UvVMRkaGAlRGRkaVH3vH0R1Ke1xTPI66+z93V/nxhaiovLw8tXnzZpWXlxfuUCpkwIABKjY2Vt13333qt99+U++9956Kjo5Wr732Wsj68ePHq99++0399ttvSimlnn76aXXGGWeoRYsWqe3bt6u33npLOZ1OtWTJkuCx7777btWyZUv1zTffqF9++UVdcsklKi4uTt13333BYxfcVkqpBx98UCUmJqq5c+eqbdu2qe+//17NmTNHKaXU8ePHVd++fdXtt9+u9u3bp/bt26d8Pl+VxFGSWbNmqS5duoQs+/TTT1ViYmKp+8yfP1998skn6vfff1dr165Vl156qerSpYsyTTPkfMfHx6vHH39cbd26Vb399ttK0zT11VdfVSreEw0YMEC99dZbp9zO5/OpDz74QDkcDrVp06bg8nXr1imn06n++te/qk2bNqlFixaphg0bqkGDBqmPPvqoxGONGTNGXXjhhSXGcrJrrOg2Ra+zMWPGqKZNm6qFCxeqTZs2qREjRqjExER15MiRch37ZNdVWR/jZK/Z3r17lc1mU3/7299UWlqa+uWXX9RLL72ksrKylFKnfq+U9NyPHj2qHA6H+uabb4JxHDlypNiyL774QjkcDuV2u0/5Wp/ss6o839+S3FSxIe8MUTyOsj1pU1n5WdXyGEKU18k+MOZ8t131eeabU/7dOndlsX1vnbuyTPvO+W57peIfMGCA6tixo7IsK7jsoYceUh07dgyu79GjR8g+brdbRUdHqx9//DE05ltvVdddd51SSqmsrCzlcDjUP//5z+D6I0eOqKioqBKTm8zMTOV0OoNfOqXFWvQLvqriKMltt92mhg8fHrJs0qRJauDAgaXuc6JDhw4pQG3YsCHkOfTv3z9ku7POOks99NBDlYr3mWeeUTExMcE/XdeV0+kMWbZr167g9r/88ouKiYlRhmGohIQEtWDBgpDjnXfeecFzWGDUqFHq7LPPLjWGYcOGqVtuuaXY8lNdYwXbFL3OsrOzld1uV/PmzQsu83g8qmnTpmratGllPvbJrqvyPMbJXrPVq1crQO3cubPYY5TlGi3pPaZU8fP56quvqqZNm4Yky+vXry/1sU9UVcmNzC1Vxd69/F2a/K0JPsvH7f++nQ+u+iDcIQlxUlluH/sz3afcrkkDV7FlR3I8Zdo3y+2rUGxFnX322SGjl/bt25fp06cHh2nv2bNnyPbbtm0j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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# t_idx = len(t[slice(*tpred)])//2\n",
    "t_idx = 2\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$ (probconserv)\")\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",
    "        \n",
    "        plt.plot(grid, u_proj_reshaped[parameters_idx,:,t_idx,0], '--', lw=2, label = \"predicted $\\mu$ and $\\pm 3\\sigma$ (probharde2e)\")\n",
    "        plt.fill_between(grid, u_proj_reshaped[parameters_idx,:,t_idx,0]+3*stds[parameters_idx,:,t_idx,0], u_proj_reshaped[parameters_idx,:,t_idx,0]-3*stds[parameters_idx,:,t_idx,0], alpha=0.2)\n",
    "        plt.plot(grid, y_train[parameters_idx,:,t_idx,0], color = \"green\", label = \"true\")\n",
    "        \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": 112,
   "id": "2614e9d3-c4bb-4f3a-8ed6-bb81c2881208",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([100, 2000])"
      ]
     },
     "execution_count": 112,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "var_flat.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 113,
   "id": "a35cc21f-5ab7-47a0-9460-ac32a0975993",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([100, 100, 20, 1])"
      ]
     },
     "execution_count": 113,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "variances.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 114,
   "id": "6659e913-9ab1-45fd-8c9b-e22aac0f3134",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([100, 100, 20, 1])"
      ]
     },
     "execution_count": 114,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "u_proj_reshaped.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 115,
   "id": "f53de336-182c-4ed7-9bde-3a493a1f3ddc",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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1ssvv2xed911+R53uVxmhQoVzPfff9+9Ly4uznQ4HObUqVOLIsSrxp133mk++uijOfbde++9Zo8ePUzT1LO+VP6djBTkuW7fvt0EzD///NNd5tdffzUNwzCPHDlyUfGUyGaajIwMNmzYQMeOHd37LBYLHTt2ZPXq1R6M7OoTHx8PQOnSpQHYsGEDmZmZOZ593bp1qVq1qp79BejXrx933nlnjucJes6X2uzZs2nRogUPPPAA5cuXp2nTpkycONF9fP/+/URHR+d43kFBQbRu3VrPu5Datm3L4sWL2bVrFwCbN29mxYoV3H777YCe9eVSkOe6evVqgoODadGihbtMx44dsVgsrF279qLuXywWyrvUTpw4gdPpJCQkJMf+kJAQdu7c6aGorj4ul4uBAwfSrl07GjZsCEB0dDR2u53g4OAcZUNCQoiOjvZAlMXXtGnT2LhxI3/++WeuY3rOl9a+ffsYN24cgwYN4tVXX+XPP//kueeew26306tXL/czzet3ip534QwZMoSEhATq1q2L1WrF6XTy9ttv06NHDwA968ukIM81Ojqa8uXL5zhus9koXbr0RT/7EpmMSNHo168fW7duZcWKFZ4O5apz6NAhBgwYwKJFi/D29vZ0OFc9l8tFixYtGDVqFABNmzZl69atjB8/nl69enk4uqvLjz/+yPfff8+UKVNo0KABERERDBw4kEqVKulZX8VKZDNN2bJlsVqtuUYWxMTEUKFCBQ9FdXXp378/c+bMYcmSJVSpUsW9v0KFCmRkZBAXF5ejvJ594WzYsIHY2FiaNWuGzWbDZrOxbNkyPv30U2w2GyEhIXrOl1DFihWpX79+jn316tUjKioKwP1M9Tvl4r344osMGTKEbt260ahRIx555BGef/55Ro8eDehZXy4Fea4VKlQgNjY2x/GsrCxOnTp10c++RCYjdrud5s2bs3jxYvc+l8vF4sWLadOmjQcjK/5M06R///7MnDmT33//nerVq+c43rx5c7y8vHI8+8jISKKiovTsC+Hmm29my5YtREREuF8tWrSgR48e7m0950unXbt2uYao79q1i2rVqgFQvXp1KlSokON5JyQksHbtWj3vQkpJScFiyfnVZLVacblcgJ715VKQ59qmTRvi4uLYsGGDu8zvv/+Oy+WidevWFxfARXV/LcamTZtmOhwOc9KkSeb27dvNJ5980gwODjajo6M9HVqx1rdvXzMoKMhcunSpeezYMfcrJSXFXebpp582q1atav7+++/m+vXrzTZt2pht2rTxYNRXh3+OpjFNPedLad26dabNZjPffvttc/fu3eb3339v+vr6mt999527zDvvvGMGBweb//vf/8y//vrL/L//+z8NN70AvXr1MitXruwe2jtjxgyzbNmy5ksvveQuo2d9YRITE81NmzaZmzZtMgHzo48+Mjdt2mQePHjQNM2CPdfbbrvNbNq0qbl27VpzxYoVZq1atTS092J99tlnZtWqVU273W62atXKXLNmjadDKvaAPF///e9/3WVSU1PNZ555xixVqpTp6+tr3nPPPeaxY8c8F/RV4t/JiJ7zpfXLL7+YDRs2NB0Oh1m3bl1zwoQJOY67XC5z6NChZkhIiOlwOMybb77ZjIyM9FC0xVdCQoI5YMAAs2rVqqa3t7dZo0YN87XXXjPT09PdZfSsL8ySJUvy/P3cq1cv0zQL9lxPnjxpdu/e3fT39zcDAwPNPn36mImJiRcdm2Ga/5jWTkRERKSIlcg+IyIiInLlUDIiIiIiHqVkRERERDxKyYiIiIh4lJIRERER8SglIyIiIuJRSkZERETEo5SMiIiIiEcpGRERERGPUjIiIiIiHqVkRERERDxKyYiIiIh41P8Dus9g7f89iqIAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "t_idx = 10\n",
    "plt.plot(y_train[0,:,t_idx,0],lw=4,label=\"true solution\")\n",
    "plt.plot(mu[0,:,t_idx,0].cpu().detach(),'--',lw=2,label=\"unconstrained\")\n",
    "plt.plot(new_mu[0,:,t_idx,0].cpu().detach(),label=\"probconserv\")\n",
    "plt.plot(u_proj_reshaped[0,:,t_idx,0].cpu().detach(),'-.',lw=2,label=\"probharde2e\")\n",
    "plt.legend()\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 116,
   "id": "93b5f5a1-f5a7-4b0c-9b69-f963d3756103",
   "metadata": {},
   "outputs": [],
   "source": [
    "# t_idx = 2\n",
    "# plt.plot(y_train[0,:,t_idx,0],lw=4,label=\"true solution\")\n",
    "# plt.plot(mu[0,:,t_idx,0].cpu().detach(),'--',lw=2,label=\"unconstrained\")\n",
    "# plt.plot(new_mu[0,:,t_idx,0].cpu().detach(),label=\"probconserv\")\n",
    "# plt.plot(new_u_proj_reshaped[0,:,t_idx,0].cpu().detach(),'-.',lw=2,label=\"probharde2e\")\n",
    "# plt.legend()\n",
    "# plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 117,
   "id": "e995547a-8fdd-4a15-9096-fdd970b9b220",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor(2.8406, device='cuda:0', grad_fn=<LinalgVectorNormBackward0>)"
      ]
     },
     "execution_count": 117,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "torch.norm(new_u_proj - u_proj)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "16a81126-a48a-4424-8290-6a9cd29b85f6",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 118,
   "id": "b5f2258f-3c7f-4302-b420-fca2acb5541a",
   "metadata": {},
   "outputs": [],
   "source": [
    "# --- Constraint function ---\n",
    "def h(u):\n",
    "    return u[0]**2 + u[1]**2 - 1.0\n",
    "\n",
    "# Check backprop\n",
    "xi_batch = torch.tensor([[2.0, 1.0],\n",
    "                         [0.1, 0.1],\n",
    "                         [-1.5, -0.5]], dtype=torch.float32, device=device)\n",
    "xi_leaf = xi_batch.clone().detach().requires_grad_(True)\n",
    "\n",
    "u_projected = fast_project_batched(mu_flat, h)\n",
    "u_proj = fast_project_weighted(mu_flat, var_flat, h_func=h)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 119,
   "id": "321130f8-924f-4198-8715-49e8f4797d4b",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor(0.0240, grad_fn=<LinalgVectorNormBackward0>)"
      ]
     },
     "execution_count": 119,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "torch.norm(utils.compute_sampling_crps_by_example(mu, var, y,nbins=500))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 120,
   "id": "6df72eff-6964-4de7-9326-6b9458282089",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor(0.0249, grad_fn=<LinalgVectorNormBackward0>)"
      ]
     },
     "execution_count": 120,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "torch.norm(utils.compute_sampling_crps_by_example(new_mu, new_std.square(), y,nbins=500))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 121,
   "id": "260d83c1-f32c-4693-8694-35b22ab1e6a4",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor(0.0246, grad_fn=<LinalgVectorNormBackward0>)"
      ]
     },
     "execution_count": 121,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "torch.norm(utils.compute_sampling_crps_by_example(u_proj_reshaped, variances.cpu(), y,nbins=500))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 122,
   "id": "0a2ba91c-01a6-4903-8b3d-89cd9dc39757",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([0.0000, 0.0505, 0.1010, 0.1515, 0.2020, 0.2525, 0.3030, 0.3535, 0.4040,\n",
       "        0.4545, 0.5051, 0.5556, 0.6061, 0.6566, 0.7071, 0.7576, 0.8081, 0.8586,\n",
       "        0.9091, 0.9596], device='cuda:0')"
      ]
     },
     "execution_count": 122,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "t_grid"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 123,
   "id": "633aa3b5-ea6a-4acf-9b78-63adc39d6418",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor(3.7377, grad_fn=<LinalgVectorNormBackward0>)"
      ]
     },
     "execution_count": 123,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "torch.norm(mu-u_proj_reshaped)"
   ]
  },
  {
   "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": 103,
   "id": "f3f1485f-07fb-4039-a0c4-d0480edbd8bd",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor(0.0021, device='cuda:0', grad_fn=<DivBackward0>)"
      ]
     },
     "execution_count": 103,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "model.loss_func(out, y.to(device))/len(out[0])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 104,
   "id": "7f826c2e-eee5-4b4a-abc0-daf2e505aa50",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([160, 100, 20, 1])"
      ]
     },
     "execution_count": 104,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "out[0].shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "dd27a2af-c705-4621-b36b-614bf61fc40c",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 105,
   "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": 106,
   "id": "400f807a-3676-42a2-82fd-f0ee6999c64e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor(0.0021, grad_fn=<MeanBackward0>)"
      ]
     },
     "execution_count": 106,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "torch.mean(crps_by_sample)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 107,
   "id": "913e8446-4753-427f-8d64-d69dbed2b866",
   "metadata": {},
   "outputs": [],
   "source": [
    "std = torch.sqrt(var)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 108,
   "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": 53,
   "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": 54,
   "id": "8d06a5a3-4116-421c-bd9a-e01ead11928c",
   "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": 55,
   "id": "95fa4119-7947-412c-9c9b-e79e9d97f6b1",
   "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": 56,
   "id": "e13b2903-c201-478d-b374-b1b80f813d72",
   "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)\n",
    "    model = ProbHardE2ETVD(base_model_class=FNO2d, probconserv=False, base_model_params=FNO2d_params, constraint_context=constraint_context, noneq_constraint_e2e=True)\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": 57,
   "id": "dd02c190-49e0-490f-85a4-1189103cef61",
   "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": 57,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "t"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 58,
   "id": "11c859b7-92f8-4125-b562-fad7c2a4683c",
   "metadata": {},
   "outputs": [],
   "source": [
    "import time"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 59,
   "id": "4945a696-f59c-4d0b-a800-08616a447547",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Updating Sigma now\n",
      "Epoch 0: Train loss=-1939.974829, Validation loss=-2218.755371 (saved)\n",
      "Finished training with best train loss: -2217.094727 and validation loss: -2218.755371\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=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",
    "stop = time.time()\n",
    "# print(stop-start)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 60,
   "id": "fea6bf3c-dd2f-4877-b902-345a59828a3f",
   "metadata": {},
   "outputs": [],
   "source": [
    "# torch.save(model.state_dict(), \"./pme_e2e_var_update_crps_05_05_3_4.pt\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 61,
   "id": "221859cb-662e-4e38-ad03-cd466b1c3e74",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<All keys matched successfully>"
      ]
     },
     "execution_count": 61,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "model.load_state_dict(torch.load(\"./pme_e2e_var_update_nll_05_04.pt\", weights_only=True))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 62,
   "id": "e82f2228-6888-4796-8b40-f13e419d3ff7",
   "metadata": {},
   "outputs": [],
   "source": [
    "from nonlinear_projection import project_and_stats\n",
    "\n",
    "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",
    "            nf,nx,nt,_ = _mu.shape\n",
    "\n",
    "            _mu = _mu.view(nf, -1)\n",
    "            _var = _var.view(nf, -1)\n",
    "            _m = x.view(nf, -1)\n",
    "\n",
    "            if model.noneq_constraint_e2e:\n",
    "                u_proj, u_var = project_and_stats(torch.relu(_mu), _var, _m, model.full_residual, max_iter=30)\n",
    "                out = (u_proj.view(nf,nx,nt,1), u_var.view(nf,nx,nt,1))\n",
    "\n",
    "\n",
    "            # out = model.base_model._apply_constraints(_mu, _std, x, t, tpred, grid, dataset_class)\n",
    "\n",
    "\n",
    "            # nf,nx,nt,_ = _mu.shape\n",
    "\n",
    "            # _mu = _mu.view(nf, -1)\n",
    "            # _var = _var.view(nf, -1)\n",
    "            # _m = x.view(nf, -1)\n",
    "\n",
    "            # # print(_m)\n",
    "\n",
    "            # u_proj, u_var = project_and_stats(torch.relu(_mu), _var, _m, model.full_residual, max_iter=30)\n",
    "\n",
    "            # # print(u_proj, u_var)\n",
    "\n",
    "            # if  u_proj.isnan().any().item() or  u_var.isnan().any().item():\n",
    "            #     print(\"any NaN in new_mu?\", u_proj.isnan().any().item())\n",
    "            #     # print(\"min new_var before clamp:\", u_var.min().item())\n",
    "            #     print(\"any NaN in new_var before clamp?\", u_var.isnan().any().item())\n",
    "            #     # new_var = new_var.clamp(min=eps)\n",
    "            #     # print(\"min new_var after clamp:\", new_var.min().item())\n",
    "\n",
    "            # out = (u_proj.view(nf,nx,nt,1), u_var.view(nf,nx,nt,1))\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": 63,
   "id": "4f58dc93-0b91-4195-b51e-126ad1212b6f",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Here\n",
      "Here\n",
      "Here\n",
      "Here\n",
      "In-domain results\n",
      "MSE: 0.00020112702623009682\n",
      "n-MeRCI: 0.3938933312892914\n",
      "RMSCE: 0.20225894451141357\n",
      "ProbConserv Results\n",
      "MSE: 0.0001839534379541874\n",
      "n-MeRCI: 0.3280416429042816\n",
      "RMSCE: 0.1996416449546814\n",
      "Cerr: 0.04093601182103157\n",
      "Prob_Cerr: 2.9514544053199643e-07\n",
      "Here\n",
      "\n",
      "\n",
      "Out-of-domain results\n",
      "MSE: 0.00022153704892843962\n",
      "n-MeRCI: 0.3724527955055237\n",
      "RMSCE: 0.19517181813716888\n",
      "ProbConserv Results\n",
      "MSE: 0.0001768207922577858\n",
      "n-MeRCI: 0.32482367753982544\n",
      "RMSCE: 0.19327029585838318\n",
      "Cerr: 0.04008578881621361\n",
      "Prob_Cerr: 2.8105452543059073e-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(\"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": 64,
   "id": "d3c74561-cafe-4068-b5ec-92d2c4634cc9",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.00020112702623009682"
      ]
     },
     "execution_count": 64,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "id_results['mse']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 65,
   "id": "c29252c6-3147-4b6a-b320-ee566b03e910",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "False"
      ]
     },
     "execution_count": 65,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "model.probconserv"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 66,
   "id": "b13f9495-85ed-4ba9-91fd-301557685c8a",
   "metadata": {},
   "outputs": [],
   "source": [
    "out = model(x_ood_test.to(device))\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",
    "\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\n",
    "nf,nx,nt,_ = mu.shape\n",
    "\n",
    "_mu = mu.view(nf, -1)\n",
    "_var = var.view(nf, -1)\n",
    "_m = x.view(nf, -1).to(device)\n",
    "\n",
    "# print(_m)\n",
    "\n",
    "u_proj, u_var = project_and_stats(torch.relu(_mu), _var, _m, model.full_residual, max_iter=30)\n",
    "\n",
    "out = (u_proj.view(nf,nx,nt,1), u_var .view(nf,nx,nt,1))\n",
    "\n",
    "mu, var, = out[0].cpu(), out[1].cpu()\n",
    "\n",
    "std = torch.sqrt(var)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 67,
   "id": "58602c51-77ce-4e25-ac60-b976ccd541c0",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([[[[1.9943e-06],\n",
       "          [1.9800e-06],\n",
       "          [1.9907e-06],\n",
       "          ...,\n",
       "          [1.9830e-06],\n",
       "          [1.9823e-06],\n",
       "          [1.9916e-06]],\n",
       "\n",
       "         [[1.9729e-06],\n",
       "          [1.2352e-04],\n",
       "          [3.4335e-05],\n",
       "          ...,\n",
       "          [2.0949e-06],\n",
       "          [2.0955e-06],\n",
       "          [2.1314e-04]],\n",
       "\n",
       "         [[1.7583e-06],\n",
       "          [9.5170e-02],\n",
       "          [3.2288e-04],\n",
       "          ...,\n",
       "          [9.7999e-05],\n",
       "          [1.2467e-04],\n",
       "          [1.1014e-04]],\n",
       "\n",
       "         ...,\n",
       "\n",
       "         [[1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          ...,\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06]],\n",
       "\n",
       "         [[1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          ...,\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06]],\n",
       "\n",
       "         [[1.9488e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          ...,\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06]]],\n",
       "\n",
       "\n",
       "        [[[1.9918e-06],\n",
       "          [2.0079e-06],\n",
       "          [1.9817e-06],\n",
       "          ...,\n",
       "          [1.9742e-06],\n",
       "          [1.9732e-06],\n",
       "          [1.9865e-06]],\n",
       "\n",
       "         [[1.9537e-06],\n",
       "          [8.9306e-05],\n",
       "          [2.4989e-05],\n",
       "          ...,\n",
       "          [2.0683e-06],\n",
       "          [2.0663e-06],\n",
       "          [1.2766e-04]],\n",
       "\n",
       "         [[1.5275e-06],\n",
       "          [8.1594e-02],\n",
       "          [1.3972e-04],\n",
       "          ...,\n",
       "          [5.7969e-05],\n",
       "          [7.6553e-05],\n",
       "          [6.4101e-05]],\n",
       "\n",
       "         ...,\n",
       "\n",
       "         [[1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          ...,\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06]],\n",
       "\n",
       "         [[1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          ...,\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06]],\n",
       "\n",
       "         [[1.8826e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          ...,\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06]]],\n",
       "\n",
       "\n",
       "        [[[1.9943e-06],\n",
       "          [1.8738e-06],\n",
       "          [1.9914e-06],\n",
       "          ...,\n",
       "          [1.9833e-06],\n",
       "          [1.9828e-06],\n",
       "          [1.9919e-06]],\n",
       "\n",
       "         [[1.9738e-06],\n",
       "          [1.2593e-04],\n",
       "          [3.4962e-05],\n",
       "          ...,\n",
       "          [2.0950e-06],\n",
       "          [2.0957e-06],\n",
       "          [2.1944e-04]],\n",
       "\n",
       "         [[1.7683e-06],\n",
       "          [9.5925e-02],\n",
       "          [3.3913e-04],\n",
       "          ...,\n",
       "          [1.0103e-04],\n",
       "          [1.2825e-04],\n",
       "          [1.1359e-04]],\n",
       "\n",
       "         ...,\n",
       "\n",
       "         [[1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          ...,\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06]],\n",
       "\n",
       "         [[1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          ...,\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06]],\n",
       "\n",
       "         [[1.9512e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          ...,\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06]]],\n",
       "\n",
       "\n",
       "        ...,\n",
       "\n",
       "\n",
       "        [[[1.9930e-06],\n",
       "          [1.8515e-06],\n",
       "          [1.9874e-06],\n",
       "          ...,\n",
       "          [1.9787e-06],\n",
       "          [1.9778e-06],\n",
       "          [1.9891e-06]],\n",
       "\n",
       "         [[1.9633e-06],\n",
       "          [1.0270e-04],\n",
       "          [2.8642e-05],\n",
       "          ...,\n",
       "          [2.0769e-06],\n",
       "          [2.0768e-06],\n",
       "          [1.5995e-04]],\n",
       "\n",
       "         [[1.6418e-06],\n",
       "          [8.7514e-02],\n",
       "          [2.0063e-04],\n",
       "          ...,\n",
       "          [7.2840e-05],\n",
       "          [9.4647e-05],\n",
       "          [8.1289e-05]],\n",
       "\n",
       "         ...,\n",
       "\n",
       "         [[1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          ...,\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06]],\n",
       "\n",
       "         [[1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          ...,\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06]],\n",
       "\n",
       "         [[1.9183e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          ...,\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06]]],\n",
       "\n",
       "\n",
       "        [[[1.9945e-06],\n",
       "          [1.9744e-06],\n",
       "          [1.9930e-06],\n",
       "          ...,\n",
       "          [1.9835e-06],\n",
       "          [1.9834e-06],\n",
       "          [1.9922e-06]],\n",
       "\n",
       "         [[1.9751e-06],\n",
       "          [1.2976e-04],\n",
       "          [3.5988e-05],\n",
       "          ...,\n",
       "          [2.0952e-06],\n",
       "          [2.1006e-06],\n",
       "          [2.2995e-04]],\n",
       "\n",
       "         [[1.7837e-06],\n",
       "          [9.7117e-02],\n",
       "          [3.6706e-04],\n",
       "          ...,\n",
       "          [1.0612e-04],\n",
       "          [1.3424e-04],\n",
       "          [1.1938e-04]],\n",
       "\n",
       "         ...,\n",
       "\n",
       "         [[1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          ...,\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06]],\n",
       "\n",
       "         [[1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          ...,\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06]],\n",
       "\n",
       "         [[1.9548e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          ...,\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06]]],\n",
       "\n",
       "\n",
       "        [[[1.9890e-06],\n",
       "          [2.0303e-06],\n",
       "          [1.9705e-06],\n",
       "          ...,\n",
       "          [1.9623e-06],\n",
       "          [1.9610e-06],\n",
       "          [1.9795e-06]],\n",
       "\n",
       "         [[1.9276e-06],\n",
       "          [7.0423e-05],\n",
       "          [1.9797e-05],\n",
       "          ...,\n",
       "          [2.0450e-06],\n",
       "          [2.0423e-06],\n",
       "          [8.1288e-05]],\n",
       "\n",
       "         [[1.2465e-06],\n",
       "          [7.0981e-02],\n",
       "          [7.0115e-05],\n",
       "          ...,\n",
       "          [3.7086e-05],\n",
       "          [5.0469e-05],\n",
       "          [3.9919e-05]],\n",
       "\n",
       "         ...,\n",
       "\n",
       "         [[1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          ...,\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06]],\n",
       "\n",
       "         [[1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          ...,\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06]],\n",
       "\n",
       "         [[1.7630e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          ...,\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06],\n",
       "          [1.0000e-06]]]], grad_fn=<ToCopyBackward0>)"
      ]
     },
     "execution_count": 67,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "var"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 68,
   "id": "6d7837ba-f720-43d2-b8d0-8252c41e8ad3",
   "metadata": {},
   "outputs": [],
   "source": [
    "std = torch.sqrt(var)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 69,
   "id": "cb722479-afdf-4b05-a156-c8a2bbbf245a",
   "metadata": {},
   "outputs": [
    {
     "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",
    "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,0], color = \"green\", label = \"true\")\n",
    "    plt.legend(loc=\"upper right\")\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 70,
   "id": "70f6d09c-1fb6-4295-b33b-4000e11722d0",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor(2.2588)"
      ]
     },
     "execution_count": 70,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    " x_ood_test[parameters_idx, 0, 0, 0]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 71,
   "id": "4f6352ec-3b12-445d-837e-e619b6815350",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor(2.8865)"
      ]
     },
     "execution_count": 71,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "x_ood_test[-1, 0, 0, 0]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 72,
   "id": "a2b9fc32-f85d-4ea2-80b9-2019a23d48da",
   "metadata": {},
   "outputs": [
    {
     "ename": "NameError",
     "evalue": "name 'new_mu' is not defined",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mNameError\u001b[0m                                 Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[72], line 13\u001b[0m\n\u001b[1;32m     11\u001b[0m param_val \u001b[38;5;241m=\u001b[39m x_ood_test[parameters_idx, \u001b[38;5;241m0\u001b[39m, \u001b[38;5;241m0\u001b[39m, \u001b[38;5;241m0\u001b[39m]\u001b[38;5;241m.\u001b[39mitem()\n\u001b[1;32m     12\u001b[0m true_vals \u001b[38;5;241m=\u001b[39m y_ood_test[parameters_idx, :, t_idx, \u001b[38;5;241m0\u001b[39m]\n\u001b[0;32m---> 13\u001b[0m pred_mu_conserv \u001b[38;5;241m=\u001b[39m \u001b[43mnew_mu\u001b[49m[parameters_idx, :, t_idx, \u001b[38;5;241m0\u001b[39m]\n\u001b[1;32m     14\u001b[0m pred_std_conserv \u001b[38;5;241m=\u001b[39m new_std[parameters_idx, :, t_idx, \u001b[38;5;241m0\u001b[39m]\n\u001b[1;32m     15\u001b[0m pred_mu_inf \u001b[38;5;241m=\u001b[39m u_proj_reshaped[parameters_idx, :, t_idx, \u001b[38;5;241m0\u001b[39m]\n",
      "\u001b[0;31mNameError\u001b[0m: name 'new_mu' is not defined"
     ]
    },
    {
     "data": {
      "text/plain": [
       "<Figure size 1350x840 with 0 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "\n",
    "t_idx = 10\n",
    "\n",
    "for parameters_idx in [0,-1]:\n",
    "    with torch.no_grad():\n",
    "        # Set compact figure size and high DPI\n",
    "        fig = plt.figure(figsize=(4.5, 2.8), dpi=300)\n",
    "\n",
    "        time_label = t[slice(*tpred)][t_idx].item()\n",
    "        param_val = x_ood_test[parameters_idx, 0, 0, 0].item()\n",
    "        true_vals = y_ood_test[parameters_idx, :, t_idx, 0]\n",
    "        pred_mu_conserv = new_mu[parameters_idx, :, t_idx, 0]\n",
    "        pred_std_conserv = new_std[parameters_idx, :, t_idx, 0]\n",
    "        pred_mu_inf = u_proj_reshaped[parameters_idx, :, t_idx, 0]\n",
    "        pred_std_inf = stds[parameters_idx, :, t_idx, 0]\n",
    "\n",
    "        pred_mu_e2e = mu[parameters_idx, :, t_idx, 0]\n",
    "        pred_std_e2e = std[parameters_idx, :, t_idx, 0]\n",
    "\n",
    "    \n",
    "\n",
    "        # Plot true solution\n",
    "        plt.plot(grid, true_vals, color=\"black\", lw=1.5, label=\"True\", zorder=3)\n",
    "\n",
    "        # ProbConserv\n",
    "        plt.plot(grid, pred_mu_conserv, '--', lw=1.2, color=\"#ef233c\",\n",
    "                 label=r\"ProbConserv\", zorder=2)\n",
    "        plt.fill_between(grid,\n",
    "                         pred_mu_conserv + 3 * pred_std_conserv,\n",
    "                         pred_mu_conserv - 3 * pred_std_conserv,\n",
    "                         color=\"#ef233c\", alpha=0.3, label=\"_nolegend_\", zorder=1)\n",
    "\n",
    "        # ProbHardE2E\n",
    "        plt.plot(grid, pred_mu_inf, '--', lw=1.5, color=\"#3a86ff\",\n",
    "                 label=r\"ProbHardInference\", zorder=4)\n",
    "        plt.fill_between(grid,\n",
    "                         pred_mu_inf + 3 * pred_std_inf,\n",
    "                         pred_mu_inf - 3 * pred_std_inf,\n",
    "                         color=\"#3a86ff\", alpha=0.3, label=\"_nolegend_\", zorder=3)\n",
    "\n",
    "        # ProbHardE2E\n",
    "        plt.plot(grid, pred_mu_e2e, ':', lw=1.5, color=\"#06d6a0\",\n",
    "                 label=r\"ProbHardE2E\", zorder=4)\n",
    "        plt.fill_between(grid,\n",
    "                         pred_mu_e2e + 3 * pred_std_e2e,\n",
    "                         pred_mu_e2e - 3 * pred_std_e2e,\n",
    "                         color=\"#06d6a0\", alpha=0.2, label=\"_nolegend_\", zorder=4)\n",
    "\n",
    "        # Labels\n",
    "        plt.xlabel(\"x\", fontsize=8)\n",
    "        plt.ylabel(r\"$u(x, t={:.2f})$\".format(time_label), fontsize=8)\n",
    "\n",
    "        # Ticks and grid\n",
    "        plt.xticks(fontsize=7)\n",
    "        plt.yticks(fontsize=7)\n",
    "        plt.grid(True, linestyle=\"--\", alpha=0.4)\n",
    "        # plt.xlim((0.45,0.58))\n",
    "\n",
    "        # Legend\n",
    "        plt.legend(fontsize=7, loc=\"upper right\", frameon=False)\n",
    "\n",
    "        # Layout for Overleaf\n",
    "        plt.tight_layout(pad=0.3)\n",
    "        plt.savefig(\"nonlinear_PME_CRPS_m\" + str(param_val) + \".pdf\")\n",
    "        plt.show()\n",
    "        print(\"L2 error ProbHardE2E:\", torch.norm(true_vals - pred_mu_e2e).item())\n",
    "        print(\"L2 error ProbConserv:\", torch.norm(true_vals - pred_mu_conserv).item())\n",
    "        print(\"L2 error ProbHardInference:\", torch.norm(true_vals - pred_mu_inf).item())\n",
    "        \n",
    "        \n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 124,
   "id": "3a4babe7-d69b-4f4e-92a3-4c19c8527702",
   "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",
    "        nf,nx,nt, _ = mu.shape\n",
    "        _mu = mu.view(nf, -1).to(device)\n",
    "        _var = var.view(nf, -1).to(device)\n",
    "        _m = x_data.view(nf, -1).to(device)\n",
    "        \n",
    "        # print(_m)\n",
    "        \n",
    "        u_proj, u_var = project_and_stats(torch.relu(_mu), _var, _m, model.full_residual, max_iter=30)\n",
    "        \n",
    "        out = (u_proj.view(nf,nx,nt,1), u_var .view(nf,nx,nt,1))\n",
    "        \n",
    "        mu, var, = out[0].cpu(), out[1].cpu()\n",
    "\n",
    "        std = torch.sqrt(var)\n",
    "        var = torch.square(std)\n",
    "\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",
    "    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",
    "            nf,nx,nt, _ = mu_test.shape\n",
    "            _mu = mu_test.view(nf, -1).to(device)\n",
    "            _var = var_test.view(nf, -1).to(device)\n",
    "            _m = x_data_test.view(nf, -1).to(device)\n",
    "            \n",
    "            # print(_m)\n",
    "            \n",
    "            u_proj, u_var = project_and_stats(torch.relu(_mu), _var, _m, model.full_residual, max_iter=30)\n",
    "            \n",
    "            out = (u_proj.view(nf,nx,nt,1), u_var .view(nf,nx,nt,1))\n",
    "            \n",
    "            mu_test, var_test, = out[0].cpu(), out[1].cpu()\n",
    "    \n",
    "            std_test = torch.sqrt(var_test)\n",
    "            var_test = torch.square(std_test)\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",
    "        else:\n",
    "            t_sliced = t[slice(*tpred)]\n",
    "            ts = repeat(t_sliced, \"nt -> nf nt\", nf=mu_test.shape[0])\n",
    "            xs = repeat(grid, \"nx -> nf nx\", nf=mu_test.shape[0])\n",
    "            inputs = meshgrid(ts, xs)\n",
    "            cerr_test = (probconserv.get_empirical_mass_rhs(mu_test[:, :, :, 0]) - test_mass_rhs_func(inputs)).abs().sum(dim=-1)\n",
    "\n",
    "        test_stats = utils.compute_all_metrics_avg((mu_test, var_test), y_data_test, {})\n",
    "        test_stats[\"nMeRCI_all\"] = utils.compute_nMeRCI(mu_test, var_test, y_data_test).item()\n",
    "        test_stats[\"rmsce_all\"] = utils.compute_rmsce(mu_test, var_test, y_data_test).item()\n",
    "        test_stats[\"cerr_by_example\"] = cerr_test.tolist()\n",
    "        test_stats[\"mcerr\"] = cerr_test.mean().item()\n",
    "\n",
    "    # --- Optional plot ---\n",
    "    if plot:\n",
    "        t_idx = 1\n",
    "        param_idx = 0\n",
    "        with torch.no_grad():\n",
    "            plt.ylabel(f\"u(x, t={t[slice(*tpred)][t_idx]:.2f})\")\n",
    "            plt.xlabel(\"x\")\n",
    "            plt.title(f\"Predicted vs True (param = {x_data[param_idx,0,0,0].item():.2f})\")\n",
    "            mu_plot = mu[param_idx, :, t_idx, 0]\n",
    "            std_plot = std[param_idx, :, t_idx, 0]\n",
    "            y_true_plot = y_data[param_idx, :, t_idx, 0]\n",
    "            plt.plot(grid, mu_plot, '--', lw=2, label=\"μ ± 3σ\")\n",
    "            plt.fill_between(grid, mu_plot + 3*std_plot, mu_plot - 3*std_plot, alpha=0.2)\n",
    "            plt.plot(grid, y_true_plot, color=\"green\", label=\"true\")\n",
    "            plt.legend()\n",
    "            plt.show()\n",
    "\n",
    "    # --- Optional LaTeX row ---\n",
    "    latex_row = None\n",
    "    if return_latex and test_stats:\n",
    "        latex_row = (\n",
    "            f\"{name} & \"\n",
    "            f\"{stats['mse']:.2E} & {stats['nMeRCI_all']:.2E} & {stats['rmsce_all']:.2E} & {stats['mcerr']:.2E} & {stats['crps']:.2E} & \"\n",
    "            f\"{test_stats['mse']:.2E} & {test_stats['nMeRCI_all']:.2E} & {test_stats['rmsce_all']:.2E} & {test_stats['mcerr']:.2E} & {test_stats['crps']:.2E} \\\\\\\\\"\n",
    "        )\n",
    "\n",
    "    return (stats, test_stats, latex_row) if return_latex else (stats, test_stats)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 125,
   "id": "5e892f52-ecfa-4592-88ef-08d411f65be9",
   "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=\"ProbHardE2E\"\n",
    ")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 126,
   "id": "6119a5a8-6ac8-4d29-809a-25c3218a9c4b",
   "metadata": {},
   "outputs": [],
   "source": [
    "import gc\n",
    "gc.collect()\n",
    "torch.cuda.empty_cache()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 128,
   "id": "1c2eb841-f9bc-4b49-81de-58620aee2a4a",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.00015788378193974496"
      ]
     },
     "execution_count": 128,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "test_stats['mse']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 129,
   "id": "fa62601e-6198-464e-99e7-5b2efa38e8eb",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.0022609339654445647"
      ]
     },
     "execution_count": 129,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "test_stats['crps']*1e4"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 79,
   "id": "1c54d2f0-cfae-4b5a-ac1f-6d323f0d3d00",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "'ProbHardE2E & 2.05E-04 & 6.81E-01 & 1.96E-01 & 4.28E-02 & 2.00E-03 & 2.22E-04 & 3.72E-01 & 1.95E-01 & 4.01E-02 & 1.89E-03 \\\\\\\\'"
      ]
     },
     "execution_count": 79,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "latex"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f44a1aec-0c82-40ef-b406-13f57d0bb081",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 134,
   "id": "3c9f4d06-2687-4058-8c02-941af51595a7",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor(0.1478, device='cuda:0', grad_fn=<MeanBackward0>)"
      ]
     },
     "execution_count": 134,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "torch.mean(torch.norm(vmap(full_residual)(torch.relu(new_mu.reshape(nf,-1)).to(device), _m), dim=0))"
   ]
  },
  {
   "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": 172,
   "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",
    "            t_sliced = t[slice(*tpred)]\n",
    "            ts = repeat(t_sliced, \"nt -> nf nt\", nf=mu.shape[0])\n",
    "            xs = repeat(grid_train, \"nx -> nf nx\", nf=mu.shape[0])\n",
    "            inputs = meshgrid(ts, xs)\n",
    "            inputs = inputs.to(mu.device)\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": "optprobconserv2",
   "language": "python",
   "name": "optprobconserv2"
  },
  "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
}
