{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "6b3ccdb8-4fd8-4224-b23e-979fa8228059",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/opt/conda/envs/optprobconserv/lib/python3.9/site-packages/tqdm/auto.py:21: TqdmWarning: IProgress not found. Please update jupyter and ipywidgets. See https://ipywidgets.readthedocs.io/en/stable/user_install.html\n",
      "  from .autonotebook import tqdm as notebook_tqdm\n"
     ]
    }
   ],
   "source": [
    "import numpy as np\n",
    "import torch\n",
    "import matplotlib.pyplot as plt\n",
    "from models.FNO2d import FNO2d\n",
    "from models.DiverseFNO2d import DiverseFNO2d\n",
    "from models.UncertainNO import *\n",
    "import utils\n",
    "from einops import rearrange, reduce, repeat\n",
    "import os\n",
    "from docopt import docopt\n",
    "import dill\n",
    "from datasets import *\n",
    "import probconserv\n",
    "import sys\n",
    "import torch.optim as optim\n",
    "\n",
    "# args = docopt(__doc__)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "5fdcc8cd-82d9-4cec-888d-806541259f2c",
   "metadata": {},
   "outputs": [],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "1d5eb530-ab43-4054-8953-9efdb86b3106",
   "metadata": {},
   "outputs": [],
   "source": [
    "args = {'--batch_size': '20',\n",
    " '--dataset': 'LinearAdvection_1D',\n",
    " '--dataset_params': '1,2',\n",
    " '--epochs': '1',\n",
    " '--fno_modes': '10',\n",
    " '--fno_width': '32',\n",
    " '--grid_len': '50',\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,10',\n",
    " '--seed': '0',\n",
    " '--time_len': '100',\n",
    " '--tplot': '0.5',\n",
    " '--train_ood_dataset_params': '1,2'}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "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": 5,
   "id": "b02f6033-3977-4d51-8f39-024a376dbabf",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Here 160\n",
      "torch.Size([160, 50, 1]) torch.Size([160, 50, 10, 1])\n",
      "torch.Size([160, 50, 10, 1]) torch.Size([160, 50, 10, 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": 6,
   "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=True, base_model_params=FNO2d_params)\n",
    "    uq = True\n",
    "elif args[\"--model\"].lower().startswith(\"DiverseFNO2d\".lower()):\n",
    "    FNO2d_params = {\"modes1\": fno_modes, \"modes2\": fno_modes2, \"width\": fno_width}\n",
    "    lam = float(args[\"--m.reg_strength\"])\n",
    "    reg_type = args[\"--m.reg_type\"]\n",
    "    n_models = int(args[\"--m.n_models\"])\n",
    "    n_regularize = int(args[\"--m.n_regularize\"])\n",
    "    utils.filter_config(args, [\"--m.n_models\", \"--m.reg_strength\", \"--m.reg_type\", \"--m.n_regularize\"], mode=\"add\", new_config=save_args)\n",
    "    model = DiverseFNO2d(reg_loss=reg_type, n_outputs=n_models, bias_last=False, lam=lam, n_regularize=n_regularize, **FNO2d_params).to(device)\n",
    "    uq = True\n",
    "else:\n",
    "    raise NotImplementedError"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "ac0b8967",
   "metadata": {},
   "source": [
    "## Running VarianceNO out of the box"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "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": 8,
   "id": "b339b6e5",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "device(type='cpu')"
      ]
     },
     "execution_count": 8,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "x_train.device"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "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": 10,
   "id": "d71ec298-4bbc-4186-a938-8675f0526890",
   "metadata": {},
   "outputs": [],
   "source": [
    "import time"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "87e15df0-69a3-4d0e-82f0-f04540d1a1e8",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([0.0000, 0.0204, 0.0408, 0.0612, 0.0816, 0.1020, 0.1224, 0.1429, 0.1633,\n",
       "        0.1837, 0.2041, 0.2245, 0.2449, 0.2653, 0.2857, 0.3061, 0.3265, 0.3469,\n",
       "        0.3673, 0.3878, 0.4082, 0.4286, 0.4490, 0.4694, 0.4898, 0.5102, 0.5306,\n",
       "        0.5510, 0.5714, 0.5918, 0.6122, 0.6327, 0.6531, 0.6735, 0.6939, 0.7143,\n",
       "        0.7347, 0.7551, 0.7755, 0.7959, 0.8163, 0.8367, 0.8571, 0.8776, 0.8980,\n",
       "        0.9184, 0.9388, 0.9592, 0.9796, 1.0000])"
      ]
     },
     "execution_count": 11,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "grid"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "2165975d",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[0, -1, 10]"
      ]
     },
     "execution_count": 12,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "tpred"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "1ad2c532",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([ 0, -1, 10])"
      ]
     },
     "execution_count": 13,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "torch.tensor(tpred)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "6ff40f4d-d8b7-45f7-8e3a-8115564540df",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 0: Train loss=361631980.800000, Validation loss=249764541.600000 (saved)\n",
      "Finished training with best train loss: 620920096.800000 and validation loss: 510149760.000000\n",
      "41.8882315158844\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": null,
   "id": "41b11e17-4840-42fe-9a5b-34d65f0a77d7",
   "metadata": {},
   "outputs": [],
   "source": [
    "def test(model, test_loader, **test_params):\n",
    "    test_type = test_params.get(\"test_type\", \"id\")\n",
    "    mu = []\n",
    "    var = []\n",
    "    results = {}\n",
    "    results[\"loss\"] = 0.0\n",
    "\n",
    "    model = model.to(device)\n",
    "\n",
    "    with torch.no_grad():\n",
    "        for batch_idx, batch in enumerate(test_loader):\n",
    "            x, y = batch\n",
    "            x, y = x.to(device), y.to(device)\n",
    "\n",
    "            out = model(x)\n",
    "\n",
    "            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",
    "    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": null,
   "id": "58fe7954-8db9-47db-ba7c-39fde51a4146",
   "metadata": {},
   "outputs": [],
   "source": [
    "is_probconserv = True\n",
    "\n",
    "train_loader_no_shuffle = torch.utils.data.DataLoader(train_loader.dataset, batch_size=batch_size, shuffle=False)\n",
    "train_results = test(model, train_loader_no_shuffle, test_type=\"train\")\n",
    "id_results = test(model, id_test_loader, test_type=\"id\")\n",
    "\n",
    "if is_train:\n",
    "    train_loader_no_shuffle = torch.utils.data.DataLoader(train_loader.dataset, batch_size=batch_size, shuffle=False)\n",
    "    train_results = test(model, train_loader_no_shuffle, test_type=\"train\")\n",
    "    id_results = test(model, id_test_loader, test_type=\"id\")\n",
    "\n",
    "    print(\"In-domain results\")\n",
    "    print(f\"MSE: {id_results['mse']}\")\n",
    "    print(f\"n-MeRCI: {id_results['nMeRCI_all']}\")\n",
    "    print(f\"RMSCE: {id_results['rmsce_all']}\")\n",
    "\n",
    "    if is_probconserv:\n",
    "        print(\"ProbConserv Results\")\n",
    "        print(f\"MSE: {id_results['pc.mse']}\")\n",
    "        print(f\"n-MeRCI: {id_results['pc.nMeRCI_all']}\")\n",
    "        print(f\"RMSCE: {id_results['pc.rmsce_all']}\")\n",
    "        print(f\"Cerr: {id_results['mcerr']}\")\n",
    "        print(f\"Prob_Cerr: {id_results['pc.mcerr']}\")\n",
    "        \n",
    "\n",
    "ood_results = test(model, ood_test_loader, test_type=\"ood\")\n",
    "\n",
    "print(\"\\n\")\n",
    "print(\"Out-of-domain results\")\n",
    "print(f\"MSE: {ood_results['mse']}\")\n",
    "print(f\"n-MeRCI: {ood_results['nMeRCI_all']}\")\n",
    "print(f\"RMSCE: {ood_results['rmsce_all']}\")\n",
    "\n",
    "if is_probconserv:\n",
    "    print(\"ProbConserv Results\")\n",
    "    print(f\"MSE: {ood_results['pc.mse']}\")\n",
    "    print(f\"n-MeRCI: {ood_results['pc.nMeRCI_all']}\")\n",
    "    print(f\"RMSCE: {ood_results['pc.rmsce_all']}\")\n",
    "    print(f\"Cerr: {ood_results['mcerr']}\")\n",
    "    print(f\"Prob_Cerr: {ood_results['pc.mcerr']}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "98acb797",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.04798305988311768"
      ]
     },
     "execution_count": 17,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "id_results['pc.mse']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "cc667c1d-e1d1-43b2-ad5a-5d542828f7df",
   "metadata": {},
   "outputs": [],
   "source": [
    "mu, var =  model(x_id_test.to(device))\n",
    "mu = mu.cpu()\n",
    "var = var.cpu()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "6648c558-a342-46a2-a3fe-96cf7fc7736e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([100, 50, 10, 1])"
      ]
     },
     "execution_count": 19,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "mu.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "473b5c5e-0293-4259-8dc0-e7742cd1f2e2",
   "metadata": {},
   "outputs": [],
   "source": [
    "std = torch.sqrt(var)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "id": "64b75faa-0ce0-47f9-ae71-2aae64386daf",
   "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_id_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_id_test[parameter_idx,:,t_idx,:], color = \"green\", label = \"true\")\n",
    "    plt.legend(loc=\"upper right\")\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "id": "9f8b7bc5-6147-492a-bd22-6fe91fe7eb2e",
   "metadata": {},
   "outputs": [],
   "source": [
    "x = id_test_loader.dataset.tensors[0]\n",
    "y = id_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",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "id": "4e4c98b6-a59b-49e2-a903-7d6173f11246",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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cs1qM3v2bMiyjMcffxwFBQWYM2cOxowZg02bNjn2+fXXXzF8+HD07t0b06ZNgyzLWLhwIS6//HKsXbsWF198MQDgr7/+woYNGzB69Gi0aNECR48exXvvvYfBgwdjz5491T5B3n///UhISMDzzz+PkpISp/sWLVqEG264AWq1Grfccgvee+89/PXXX06fCDMzM3HZZZfBbDbjqaeeQnh4OD744APodDqnc40aNQpjx46tdvyxY8fw559/4tVXX3Vse+mll/Dcc8/h5ptvxt13342cnBzMmzcPl156KbZt2+aoZKxcuRLXXHMNUlJS8PDDDyM5ORl79+7FDz/8gIcffhj33HMPTp8+jZUrV+KLL76o8++we/duDBw4EJGRkZgyZQpUKhXef/99DB48GL///nu1flLuPK8ai0KhwC233ILnnnsO69atw9VXX+1yv7feeguff/45li1bhvfeew96vR7du3cHYHvTnj59OoYOHYr77rsP+/fvd/zN169f79TklpeXh+HDh2P06NG47bbbkJSUVGt8FosFw4YNwyWXXII5c+ZgxYoVmDZtGsxmM2bMmOHYb+7cubj22msxZswYGI1GLF68GDfddBN++OGHao/p119/xddff40HHngA8fHxaN26tcfnWLt2Lb7//ntMmjQJADBr1ixcc801mDJlCubPn4/7778f586dw5w5c3DnnXfi119/dfr9df2bvOGGG3DgwAH85z//wZtvvon4+HgAtiQUcP+57uk1NxgMKCoqqvVvYmePydcuu+wyFBcXQ61WIz09Ha+//jrat29f53H2fkSNFafXCQo6CxcuFADEX3/9VeM+d911l0hJSRG5ublO20ePHi2ioqJEaWmpEEIIs9ksDAaD0z7nzp0TSUlJ4s4773Rsy8jIEABEZGSkyM7Odtp/2rRpAoDT/kIIcf3114u4uDinba1atRLjxo2r9liGDh0qrFarY/ujjz4qFAqFyM/PF0IIkZmZKZRKpRg5cqTT+V544QUBwOmcNQEgJk2aVOs+gwYNEoMGDXLc/u233wQA0alTJ6frNHfuXAFA7Nq1SwghhNVqFe3btxfp6elOj6O0tFSkpaWJK664wmlbVRs3bhQAxOeff+7YZr82AwYMEGazudoxf//9twAgVq5c6YihRYsW4uGHH3ba75FHHhEAxKZNmxzbsrOzRVRUlAAg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KoyFCxe6/Xu8cQ53ePJvsqbnoLvP9fpYsmSJ281o/krcjxw54rIrhH3er0ceeQRPPvmkHyLzLvYRaoIUCgVuvPFG/Pe//3V8Aq2s8vBh+6ecyv/QNm3ahI0bN/o+UA8MGTIESqUS7733ntP2d955x08ROevduzfatm2L1157zdH8UVnVa171hW3evHk1DlmtqqysDEuXLsU111yDf//739W+HnjgARQVFeH7778HAFx11VX4888/sXnzZqd4ahpqP2rUKJw+fRofffQRduzYgVGjRjndf8MNN0ChUGD69OnVHocQAnl5eQCACy+8EGlpaXjrrbeqJXmVj7PPO1JXIqhQKHDllVfif//7n1NTTVZWFr766isMGDCgziZYf7FYLHjooYewd+9ePPTQQ/WKc+jQoVCr1Xj77bedrt/HH3+MgoKCGkeheaLyvychBN555x2oVCoMGTIEgO1vIEmS03P16NGj+O6779z+Hd44hzs8+TdZ03PQ3ed6ffirj9CZM2eqdYNwNaXE8uXLsWXLFgwbNsxp+5IlS/DQQw9hzJgxeOONN7wam7+wIhTEPvnkE8e8HJU9/PDDmD17Nn777Tf07dsXEyZMQOfOnXH27Fls3boVq1atwtmzZwHY5o5ZunQprr/+elx99dXIyMjAggUL0LlzZ5cvHv6SlJSEhx9+GK+//jquvfZaDBs2DDt27MBPP/2E+Ph4t6sKf//9N1588cVq2wcPHlyvT+l2sizjo48+wvDhw9GlSxfccccdaN68OU6dOoXffvsNkZGR+L//+z8Atmv+xRdfICoqCp07d8bGjRuxatUqxMXFufW7vv/+exQVFeHaa691ef8ll1yChIQELFq0CKNGjcKUKVPwxRdfYNiwYXj44Ycdw+dbtWqFnTt3Vjv+qquuQkREBB5//HFHUl1Z27Zt8eKLL2Lq1Kk4evQoRo4ciYiICGRkZGDZsmWYOHEiHn/8cciyjPfeew8jRoxAz549cccddyAlJQX79u3D7t278fPPPwOwvWEBwEMPPYT09HQoFAqMHj3a5WN78cUXsXLlSgwYMAD3338/lEol3n//fRgMBsyZM8et61cf77zzDvLz83H69GkAwP/93//h5MmTAGxNnfb+KYCt38mXX34JwDahnn1m6cOHD2P06NGYOXNmvWJISEjA1KlTMX36dAwbNgzXXnst9u/f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efNmGAwGt+Igqq9g7CekVclom6BHcpQ24GeTriw40rUg89HaI/hobUad+3VtHomPxl3ktO3uz/7CP6eqf+qu6u6Babh7YJt6xwgAqampePPNNyFJEjp06IBdu3bhzTffxIQJEwAA7du3x5w5cxz7GwwGvPzyy1i1apXjzbNNmzZYt24d3n//fQwaNAjFxcX4+OOP8eWXX2LIkCEAbG9kLVq0cBlDUVER5s6di3feeQfjxo0DALRt2xYDBgwAAERFRUGtViMsLAzJyck+i8Nux44d6NWrl9O27du3o0ePHjUec+ONNzrd/uSTT5CQkIA9e/aga9euju3du3fHtGnTANiu7TvvvIPVq1fjiiuuqHe8L7/8Ml5++WXH7bKyMvz555944IEHHNv27NmDli1bAgB27dqFfv36oby8HHq9HsuWLUPnzp0d+z700EO44YYb8NprrwEAOnfujFtuuQVbtmzBzTff7DKGY8eOoVmzZk7bYmJiMHz4cHz11VeOx/Ptt98iPj4el112mUfXrerzEAB++OEH6PV6x+3hw4fjm2++cdrn+uuvR8+ePTFt2jR8/PHH1eI+ePAghBDo1KmTy8fVqVMnnDt3Djk5OUhMTAQANGvWDEajEZmZmTUmxkTeEEyJkCQBiREaJERognIuLSZCPlBUbkZmYXmd+6VEa6ttyysxunVsUbm5XrFVdskllzg9afv164fXX38dFosFAKp9yj906BBKS0txxRVXOG03Go2O5OHw4cMwGo3o27ev4/7Y2Fh06NDBZQx79+6FwWBwvFm6wxdx2G3fvh233nqr07Zt27ahZ8+eNR5z8OBBPP/889i0aRNyc3MdFY3jx49XS4QqS0lJcaxNVd947733XqcEZcyYMbjxxhtxww03OLZVTlI6dOiA7du3o6CgAN9++y3GjRuH33//HZ07d0ZWVhbWrVvnVFUDgPDw8Fpf3MrKypwqaJVjmTBhAubPnw+NRoNFixZh9OjRkCsWYHT3ulV9HgLAZZddhvfee88pRldeeeUVXH755Xj88cdrjL9yxacu9spRaWmp28cQ1Ye9w7TJ7P7z0x90agVaxOiCoi9QTZgI+UCEVonkyOpvDFXFhatdbnPn2Ait7/90Vd9ciouLAQA//vgjmjdv7nSfRqOp1++o2iThDl/EAdiai/75559qFaGtW7dWq15UNmLECLRq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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_id_test[parameters_idx,0,0,0], dataset = dataset))\n",
    "        plt.xlabel(\"x\")\n",
    "        plt.plot(grid, new_mu[parameters_idx,:,t_idx], '--', lw=2, label = \"predicted $\\mu$ and $\\pm 3\\sigma$ (varFNO)\")\n",
    "        plt.fill_between(grid, new_mu[parameters_idx,:,t_idx]+3*new_std[parameters_idx,:,t_idx], new_mu[parameters_idx,:,t_idx]-3*new_std[parameters_idx,:,t_idx], alpha=0.2)\n",
    "        plt.plot(grid, y_id_test[parameters_idx,:,t_idx,:], color = \"green\", label = \"true\")\n",
    "        plt.legend(loc=\"lower left\")\n",
    "        # plt.ylim(-1.0,1.5)\n",
    "        plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "id": "23d782f6-1d92-4ede-be84-48ec124c9414",
   "metadata": {},
   "outputs": [],
   "source": [
    "t_sliced = t[slice(*tpred)]\n",
    "mu = x_train[:,:,:,0]\n",
    "grid_train = torch.clone(grid)\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",
    "\n",
    "target_inputs = torch.clone(inputs)\n",
    "nf, nt, nx, _ = target_inputs.shape\n",
    "\n",
    "input_grid = rearrange(target_inputs, \"nf nt nx d -> nf nt nx d\", nt=nt, nx=nx)\n",
    "x_inp = input_grid[:, :, :, 1]\n",
    "\n",
    "x_delta = probconserv._get_riemman_delta(x_inp)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ada62ce2-9242-461e-8a76-95c06054ed2d",
   "metadata": {},
   "outputs": [],
   "source": [
    "g = rearrange(x_delta, \"nf nt nx -> nf nt 1 nx\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f19b36e9-4720-40a8-b5b5-c1840e1bbe09",
   "metadata": {},
   "outputs": [],
   "source": [
    "g.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "04a57f50-3a30-49c0-a627-78110bf0276f",
   "metadata": {},
   "outputs": [],
   "source": [
    "y_train_t = rearrange(y_train, \"nf nx nt 1 -> nf nt nx 1\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "42032c17-23ff-40db-8cbf-95996526c083",
   "metadata": {},
   "outputs": [],
   "source": [
    "(g@y_train_t).shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3c614d6c-032a-42f4-9bf6-93918beb21df",
   "metadata": {},
   "outputs": [],
   "source": [
    "b = dataset_class.get_mass_rhs_func(x_train)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "49c5569b-9971-465c-a83f-52a6c9ec9e55",
   "metadata": {},
   "outputs": [],
   "source": [
    "b(inputs)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "01cbf0be-518f-403c-a87e-3bdff9a28308",
   "metadata": {},
   "outputs": [],
   "source": [
    "b(inputs).shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "2cf9e8c2-83a5-4c95-87ab-60ec0d5220ea",
   "metadata": {},
   "outputs": [],
   "source": [
    "import cvxpy as cp\n",
    "import torch\n",
    "from cvxpylayers.torch import CvxpyLayer\n",
    "\n",
    "# n, m = 2, 3\n",
    "# x = cp.Variable(n)\n",
    "# A = cp.Parameter((m, n))\n",
    "# b = cp.Parameter(m)\n",
    "# constraints = [x >= 0]\n",
    "# objective = cp.Minimize(0.5 * cp.pnorm(A @ x - b, p=1))\n",
    "# problem = cp.Problem(objective, constraints)\n",
    "# assert problem.is_dpp()\n",
    "\n",
    "# cvxpylayer = CvxpyLayer(problem, parameters=[A, b], variables=[x])\n",
    "# A_tch = torch.randn(m, n, requires_grad=True)\n",
    "# b_tch = torch.randn(m, requires_grad=True)\n",
    "\n",
    "# # solve the problem\n",
    "# solution, = cvxpylayer(A_tch, b_tch)\n",
    "\n",
    "# # compute the gradient of the sum of the solution with respect to A, b\n",
    "# solution.sum().backward()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "903a6bdc-a23c-4237-bff3-8b05799cb9df",
   "metadata": {},
   "outputs": [],
   "source": [
    "x_delta.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "16791329-3ad1-4dd9-874b-ea200fab65d2",
   "metadata": {},
   "outputs": [],
   "source": [
    "mat = torch.zeros((20,20))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "6c00c7c3-9c48-49a1-a832-34eae8c92c4a",
   "metadata": {},
   "outputs": [],
   "source": [
    "tt = repeat(g, \"nf nt 1 nx -> nf c (nt nx)\", c = nt)[0,0,0:100]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "2fedab6c-bc38-4b5a-83f7-35eb7e4938ed",
   "metadata": {},
   "outputs": [],
   "source": [
    "tt = torch.zeros((160, 20, 2000))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "03976597-e3e5-479d-8b23-dd3f6fd37472",
   "metadata": {},
   "outputs": [],
   "source": [
    "for i in range(20):\n",
    "    tt[:,i,100*i:100*i+100] = x_delta[:,i,:]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "cb46b879-fb6e-4bf9-ac54-91bb0a18fc94",
   "metadata": {},
   "outputs": [],
   "source": [
    "tt = tt.requires_grad_(True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "8990baa2-6233-404b-b028-0c6c15e64d2a",
   "metadata": {},
   "outputs": [],
   "source": [
    "mu, var = model(x_train.to(device))\n",
    "std = torch.sqrt(var)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ea1db30d-5433-4b29-b541-09022e5a09b7",
   "metadata": {},
   "outputs": [],
   "source": [
    "mu.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "400aa953-720c-4203-8691-0e4726dd4c71",
   "metadata": {},
   "outputs": [],
   "source": [
    "mu_shaped = rearrange(mu, \"nf nx nt 1->nf nt nx 1\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "902de5f1-05a8-4bc4-af4e-b863a9f5f93d",
   "metadata": {},
   "outputs": [],
   "source": [
    "g = g.to(device)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "1e821970-b49d-4245-b9c9-5f02bab87055",
   "metadata": {},
   "outputs": [],
   "source": [
    "(g@mu_shaped).shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "63323e76-3afd-48a4-8806-c8f8b62f2769",
   "metadata": {},
   "outputs": [],
   "source": [
    "mu_r = rearrange(mu, \"nf nx nt 1 -> nf (nt nx) 1\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "a3c5cf8d-587c-4de9-86b5-39b288f51597",
   "metadata": {},
   "outputs": [],
   "source": [
    "tt = tt.to(device)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3fbc4ebe-a1c7-4894-be7c-ea1e397688a9",
   "metadata": {},
   "outputs": [],
   "source": [
    "mu_r.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "fce7c6a6-0e2f-428e-b978-aea4aea88660",
   "metadata": {},
   "outputs": [],
   "source": [
    "tt.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "cf778fee-f55e-4300-baca-4275bd6012fa",
   "metadata": {},
   "outputs": [],
   "source": [
    "g@mu_shaped"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c71d3a2e-6a21-40be-850b-ba7cfdc1e1a8",
   "metadata": {},
   "outputs": [],
   "source": [
    "tt@mu_r"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "7ee7ebca-ca87-405d-9142-15f1aaf1a13f",
   "metadata": {},
   "outputs": [],
   "source": [
    "torch.norm(((g@mu_shaped)).squeeze(-1) - tt@mu_r)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ff452222-c366-47a9-81da-aa284d153215",
   "metadata": {},
   "outputs": [],
   "source": [
    "A = cp.Parameter((tt.shape[1], tt.shape[2]))\n",
    "x = cp.Variable(tt.shape[2])\n",
    "y = cp.Variable(tt.shape[2])\n",
    "b = cp.Parameter(tt.shape[1])\n",
    "eta = cp.Parameter(tt.shape[2])\n",
    "weight_mat = cp.Parameter(tt.shape[2])\n",
    "\n",
    "constraints = [A@x == b,\n",
    "               y == x - eta\n",
    "              ]\n",
    "objective = cp.Minimize(cp.norm(cp.multiply(weight_mat, y)))\n",
    "problem = cp.Problem(objective, constraints)\n",
    "problem.is_dpp()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "11f2061d-1576-43ad-b5c9-2d5d8779eaf8",
   "metadata": {},
   "outputs": [],
   "source": [
    "cvxpylayer = CvxpyLayer(problem, parameters=[A, b, eta, weight_mat], variables=[x, y])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d3dcc940-0689-4c2e-a14e-40244da2c85e",
   "metadata": {},
   "outputs": [],
   "source": [
    "A_tch = tt.to(device)\n",
    "b_tch = dataset_class.get_mass_rhs_func(x_train)(inputs)\n",
    "b_tch = b_tch.requires_grad_(True)\n",
    "b_tch = b_tch.to(device)\n",
    "\n",
    "eta_tch = rearrange(mu, \"nf nx nt 1 -> nf (nt nx)\")\n",
    "weight_mat_tch = rearrange(1/std, \"nf nx nt 1 -> nf (nt nx)\")\n",
    "\n",
    "\n",
    "print(A_tch.shape, b_tch.shape, eta_tch.shape, weight_mat_tch.shape)\n",
    "\n",
    "solution, _ = cvxpylayer(A_tch, b_tch, eta_tch, weight_mat_tch)\n",
    "\n",
    "# %timeit solution, _ = cvxpylayer(A_tch, b_tch, eta_tch, weight_mat_tch)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "6a6929a3-e760-4e0e-aae5-0f29d57e6bb9",
   "metadata": {},
   "outputs": [],
   "source": [
    "train_loader.dataset.tensors[2].shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "deb49518-5b87-4dbf-ae43-817f34652ad6",
   "metadata": {},
   "outputs": [],
   "source": [
    "data_cvxpylayer = dataset_class.get_cvxpylayer(x_train)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b1e93b63-286e-4c03-acbb-ee30f8604d04",
   "metadata": {},
   "outputs": [],
   "source": [
    "A_tch_new, b_tch_new = probconserv.get_convex_params(train_loader.dataset.tensors[0], mass_rhs_func, t, tpred, grid_train, precis_g=np.inf, second_deriv_alpha=None)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "36033c43-4569-4600-8e6b-f25d3f54a112",
   "metadata": {},
   "outputs": [],
   "source": [
    "data_cvxpylayer(A_tch_new.to(device), b_tch_new.to(device), eta_tch, weight_mat_tch)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3b13743f-29ed-45e4-b511-900a9cfc941a",
   "metadata": {},
   "outputs": [],
   "source": [
    "data_cvxpylayer(A_tch, b_tch, eta_tch, weight_mat_tch)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3508db28-d035-49fa-b4cc-fa585e4d6f34",
   "metadata": {},
   "outputs": [],
   "source": [
    "solution"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d9a8f396-9cca-4076-91cf-e936f9af094c",
   "metadata": {},
   "outputs": [],
   "source": [
    "solution_t = rearrange(solution, \"nf (nt nx) -> nf nx nt\", nx = mu.shape[1], nt = mu.shape[2]).cpu()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e99f1e28-359a-411a-8e80-f7a6f6be5612",
   "metadata": {},
   "outputs": [],
   "source": [
    "new_mu, new_std, _, mass_rhs = probconserv.apply_constraint(mu=mu[:, :, :, 0].cpu(), std=std[:, :, :, 0].cpu(), mass_rhs_func=mass_rhs_func, t=t, tpred=tpred, grid_train=grid, precis_g=np.inf, second_deriv_alpha=None)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e2f50c62-d5b3-49e5-b2e0-dda3b9c59867",
   "metadata": {},
   "outputs": [],
   "source": [
    "# %timeit new_mu, new_std, _, mass_rhs = probconserv.apply_constraint(mu=mu[:, :, :, 0], std=std[:, :, :, 0], mass_rhs_func=mass_rhs_func, t=t, tpred=tpred, grid_train=grid, precis_g=np.inf, second_deriv_alpha=None)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "62344632-e78d-4d75-aab6-a6cc752e9212",
   "metadata": {},
   "outputs": [],
   "source": [
    "solution_t[1,:,:]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "46a83396-68e1-4a19-84ae-aeaeef821d80",
   "metadata": {},
   "outputs": [],
   "source": [
    "new_mu[1,:,:]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "68d09dda-7580-4137-9955-cc47e5ab990d",
   "metadata": {},
   "outputs": [],
   "source": [
    "torch.norm(solution_t.cpu() - new_mu)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "dcac6ddf-e4cb-402c-8d4c-06724a9e9f59",
   "metadata": {},
   "outputs": [],
   "source": [
    "A = cp.Parameter((tt.shape[1], tt.shape[2]))\n",
    "x = cp.Variable(tt.shape[2])\n",
    "y = cp.Variable(tt.shape[2])\n",
    "b = cp.Parameter(tt.shape[1])\n",
    "eta = cp.Parameter(tt.shape[2])\n",
    "lbda = 0.0\n",
    "weight_mat = cp.Parameter(tt.shape[2])\n",
    "\n",
    "constraints = [A@x == b,\n",
    "               y == x - eta\n",
    "              ]\n",
    "# for i in range(20):\n",
    "#     xs1 = x[i*100:100*(i+1)]\n",
    "#     # xs2 = x[(i+1)*100:100*(i+2)]\n",
    "#     # tvd2 = cp.norm1(xs2[1:] - xs2[:-1])\n",
    "#     tvd1 = cp.norm2(xs1[1:])\n",
    "#     constraints.append(tvd1 >= 1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "9fe1c71d-159b-40ae-95ea-566afa90a9c7",
   "metadata": {},
   "outputs": [],
   "source": [
    "objective = cp.Minimize(cp.norm(cp.multiply(weight_mat, y)) \n",
    "                        #+ lbda*cp.norm1(x[1:] - x[:-1])\n",
    "                       )\n",
    "problem = cp.Problem(objective, constraints)\n",
    "problem.is_dpp()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "37f1d7f9-27ff-4fa7-8db8-353d27a851ff",
   "metadata": {},
   "outputs": [],
   "source": [
    "cvxpylayer = CvxpyLayer(problem, parameters=[A, b, eta, weight_mat], variables=[x, y])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e5b8067c-7f54-403e-bbca-46c067dd75b5",
   "metadata": {},
   "outputs": [],
   "source": [
    "A_tch = tt.to(device)\n",
    "b_tch = dataset_class.get_mass_rhs_func(x_train)(inputs)\n",
    "b_tch = b_tch.requires_grad_(True)\n",
    "b_tch = b_tch.to(device)\n",
    "\n",
    "eta_tch = rearrange(mu, \"nf nx nt 1 -> nf (nt nx)\")\n",
    "weight_mat_tch = rearrange(1/std, \"nf nx nt 1 -> nf (nt nx)\")\n",
    "\n",
    "\n",
    "print(A_tch.shape, b_tch.shape, eta_tch.shape, weight_mat_tch.shape)\n",
    "\n",
    "solution, _ = cvxpylayer(A_tch, b_tch, eta_tch, weight_mat_tch)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "996903ab-f46c-4aae-91b2-d4db9a64341e",
   "metadata": {},
   "outputs": [],
   "source": [
    "solution"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "27970a91-ce3f-4138-9a9c-3df71313e3c2",
   "metadata": {},
   "outputs": [],
   "source": [
    "solution_t = rearrange(solution, \"nf (nt nx) -> nf nx nt\", nx = mu.shape[1], nt = mu.shape[2]).cpu()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "10ba3674-f19b-421f-a7db-63f1d2b33d58",
   "metadata": {},
   "outputs": [],
   "source": [
    "torch.norm(solution_t.cpu() - new_mu)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "a30ccbc4-5627-490e-a1fb-9b0c994af935",
   "metadata": {},
   "outputs": [],
   "source": [
    "nn.MSELoss()(y_train[:,:,:,0], solution_t)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "708391cc-f819-4b82-9b44-56750ff5ce1b",
   "metadata": {},
   "outputs": [],
   "source": [
    "nn.MSELoss()(y_train[:,:,:,0], new_mu)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ae180450-5567-43aa-ab2c-e36f5cd43447",
   "metadata": {},
   "outputs": [],
   "source": [
    "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_train[parameters_idx,0,0,0], dataset = dataset))\n",
    "        plt.xlabel(\"x\")\n",
    "        plt.plot(grid, solution_t[parameters_idx,:,t_idx], '--', lw=2, label = \"predicted $\\mu$ (TVD, Nonlinear)\")\n",
    "        plt.plot(grid, new_mu[parameters_idx,:,t_idx], '--', lw=2, label = \"predicted $\\mu$\")\n",
    "        # plt.fill_between(grid, new_mu[parameters_idx,:,t_idx]+3*new_std[parameters_idx,:,t_idx], new_mu[parameters_idx,:,t_idx]-3*new_std[parameters_idx,:,t_idx], alpha=0.2)\n",
    "        plt.plot(grid, y_train[parameters_idx,:,t_idx,:], color = \"green\", label = \"true\")\n",
    "        plt.legend(loc=\"lower left\")\n",
    "        # plt.ylim(-1.0,1.5)\n",
    "        plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "191e957c-4b9c-487b-8e4c-00685d51d108",
   "metadata": {},
   "outputs": [],
   "source": [
    "from torch.distributions import MultivariateNormal"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d179af31-01fd-424b-b96c-5659049aecae",
   "metadata": {},
   "outputs": [],
   "source": [
    "dist = torch.distributions.Normal(rearrange(mu, \"nf nx nt 1 -> nf (nt nx)\"), rearrange(std, \"nf nx nt 1 -> nf (nt nx)\"))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "398e5533-f537-482f-a722-c9af6cd1f64d",
   "metadata": {},
   "outputs": [],
   "source": [
    "n_samples = 50"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "215f431e-01a2-4feb-9535-004127c1a9e7",
   "metadata": {},
   "outputs": [],
   "source": [
    "samples = dist.sample((n_samples,))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "42d0ee62-c22a-454b-a60a-24ab14a29b7e",
   "metadata": {},
   "outputs": [],
   "source": [
    "r_samples = rearrange(samples, \"ns nf ntnx -> (ns nf) ntnx \")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c89fd3a3-9fae-463e-8c4a-f7e5e619f969",
   "metadata": {},
   "outputs": [],
   "source": [
    "r_samples.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "887fce27-f7e6-4c33-9634-df5d617b81cf",
   "metadata": {},
   "outputs": [],
   "source": [
    "A_tch_1 = repeat(A_tch, \"nf nx nt -> (nf1 nf) nx nt\", nf1 = n_samples)\n",
    "b_tch_1 = repeat(b_tch, \"nf nx -> (nf1 nf) nx\", nf1 = n_samples)\n",
    "weight_mat_tch_1 = repeat(weight_mat_tch, \"nf nxnt -> (nf1 nf) nxnt\", nf1 = n_samples)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "19b5a7ad-0891-4a13-b940-9e9c442d75c5",
   "metadata": {},
   "outputs": [],
   "source": [
    "solutions, _ = cvxpylayer(A_tch_1, b_tch_1, r_samples, weight_mat_tch_1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "964880ea-06da-4cea-a594-b343a5fd3922",
   "metadata": {},
   "outputs": [],
   "source": [
    "solutions_r = rearrange(solutions, \"(ns n1) n2 -> ns n1 n2\", ns = n_samples)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b8265651-a70b-4886-875c-e1bf9fcdf63e",
   "metadata": {},
   "outputs": [],
   "source": [
    "sample_mu = solutions_r.mean(dim = 0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "19c4c2b7-aa2b-4acb-87cd-67746f7f4d1f",
   "metadata": {},
   "outputs": [],
   "source": [
    "sample_std = solutions_r.std(dim = 0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "9cb6a78d-e5d5-4053-9397-d797a43c6a1e",
   "metadata": {},
   "outputs": [],
   "source": [
    "sample_mu"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "1f91d6db-36fa-4974-aaf0-0414a90809ca",
   "metadata": {},
   "outputs": [],
   "source": [
    "solution"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "8c23b8ae-455f-4c18-93bc-485396199f13",
   "metadata": {},
   "outputs": [],
   "source": [
    "torch.norm(sample_mu - solution)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e23b8fb0-13fa-41ac-89ea-7fe56bd6d201",
   "metadata": {},
   "outputs": [],
   "source": [
    "sample_mu_r = rearrange(sample_mu, \"nf (nt nx) -> nf nx nt\", nx = mu.shape[1], nt = mu.shape[2]).cpu()\n",
    "sample_std_r = rearrange(sample_std, \"nf (nt nx) -> nf nx nt\", nx = mu.shape[1], nt = mu.shape[2]).cpu()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e16100a2-cc32-40c0-94a3-fa04c1c712b6",
   "metadata": {},
   "outputs": [],
   "source": [
    "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_train[parameters_idx,0,0,0], dataset = dataset))\n",
    "        plt.xlabel(\"x\")\n",
    "        plt.plot(grid, new_mu[parameters_idx,:,t_idx], '--', lw=2, label = \"predicted $\\mu$ and $\\pm 3\\sigma$\")\n",
    "        plt.fill_between(grid, new_mu[parameters_idx,:,t_idx]+3*new_std[parameters_idx,:,t_idx], new_mu[parameters_idx,:,t_idx]-3*new_std[parameters_idx,:,t_idx], alpha=0.2)\n",
    "        plt.plot(grid, sample_mu_r[parameters_idx,:,t_idx], '--', lw=2, label = \"predicted $\\mu$ and $\\pm 3\\sigma$ (convex sampling)\")\n",
    "        plt.fill_between(grid, sample_mu_r[parameters_idx,:,t_idx]+3*sample_std_r[parameters_idx,:,t_idx], sample_mu_r[parameters_idx,:,t_idx]-3*sample_std_r[parameters_idx,:,t_idx], alpha=0.2)\n",
    "        plt.plot(grid, y_train[parameters_idx,:,t_idx,:], color = \"green\", label = \"true\")\n",
    "        plt.legend(loc=\"lower left\")\n",
    "        # plt.ylim(-1.0,1.5)\n",
    "        plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ae1017b1-f022-4233-98ec-4f7354b0cb6a",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c5db5020-c2d3-4382-9b77-bdde8b865fa5",
   "metadata": {},
   "outputs": [],
   "source": [
    "samples[1].shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c7ef72e7-5d80-42d6-9e3e-ec2cc87b467b",
   "metadata": {},
   "outputs": [],
   "source": [
    "convex_solve(samples[0])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "5f85cd5b-719e-4657-b1d0-f99452f19bd5",
   "metadata": {},
   "outputs": [],
   "source": [
    "batched_convex_solve = torch.vmap(convex_solve)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ab18bf8e-ed9b-46a8-874a-38642b73c1aa",
   "metadata": {},
   "outputs": [],
   "source": [
    "with torch.no_grad():\n",
    "    batched_convex_solve(samples)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ab7c00e4-5a79-45c4-878f-90471cb2ffa0",
   "metadata": {},
   "outputs": [],
   "source": [
    "torch.norm(samples.var(dim = 0) - var)"
   ]
  },
  {
   "cell_type": "raw",
   "id": "4a43f91e-8478-4a9e-8ba9-212dad2471bc",
   "metadata": {},
   "source": [
    "solution_t[1,:,:]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f1c5b10f-47ed-42f5-a97e-6f61fa7a7b66",
   "metadata": {},
   "outputs": [],
   "source": [
    "new_mu[1,:,:]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "6a867638-cfc0-44a5-98e5-9a328cb5560c",
   "metadata": {},
   "outputs": [],
   "source": [
    "optimizer = optim.Adam(model.parameters(), lr=0.01)\n",
    "for epoch in range(200):\n",
    "    mu, var = model(x_train)\n",
    "    \n",
    "    model_loss = nn.GaussianNLLLoss()(mu, y_train, var)\n",
    "    \n",
    "    optimizer.zero_grad()\n",
    "    model_loss.backward()\n",
    "    optimizer.step()\n",
    "    \n",
    "    if epoch % 10 == 0:\n",
    "        # print(target.shape, z.shape)\n",
    "        print(\"loss: \",model_loss.item(), \"epoch: \", epoch)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c4525b5f-580a-4abd-9e83-2ae40bbcbe24",
   "metadata": {},
   "outputs": [],
   "source": [
    "def train_model(model, loss_func = nn.MSELoss(), epochs=200):\n",
    "    optimizer = optim.Adam(model.parameters(), lr=0.01)\n",
    "    for epoch in range(epochs):\n",
    "        \n",
    "        mu, var = model(x_train)\n",
    "        std = torch.sqrt(var)\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",
    "    \n",
    "        # ## Do loss function on var\n",
    "    \n",
    "        # # nn.NLLLoss()()\n",
    "        \n",
    "        # new_mu = new_mu.unsqueeze(3)\n",
    "        # new_std = new_std.unsqueeze(3)\n",
    "        \n",
    "        model_loss = loss_func(y_train, new_mu)\n",
    "\n",
    "        optimizer.zero_grad()\n",
    "        model_loss.backward()\n",
    "        optimizer.step()\n",
    "        \n",
    "        if epoch % 10 == 0:\n",
    "            # print(target.shape, z.shape)\n",
    "            print(\"loss: \",model_loss.item(), \"epoch: \", epoch)\n",
    "    return new_mu, new_std"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e9af2566-1b0d-4e9b-9bcc-020001cbe877",
   "metadata": {},
   "outputs": [],
   "source": [
    "utils.set_seed(0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "aed63b58-e1b3-4784-8bc5-2e418a96a6bf",
   "metadata": {},
   "outputs": [],
   "source": [
    "new_mu_output_var, new_std_output_var = train_model(model, epochs=500)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b84c4802-e2c6-4464-aba6-81190d5d0208",
   "metadata": {},
   "outputs": [],
   "source": [
    "# optimizer = optim.Adam(model.parameters(), lr=0.01)\n",
    "\n",
    "# for epoch in range(200):\n",
    "#     mu, std = model(x_train)\n",
    "#     # std = torch.abs(std)\n",
    "#     # mu = rearrange(mu, \"nf (nx nt) -> nf nx nt 1\", nx = x_train.shape[1], nt = x_train.shape[2])\n",
    "#     # std = rearrange(std, \"nf (nx nt) -> nf nx nt 1\", nx = x_train.shape[1], nt = x_train.shape[2])\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",
    "\n",
    "#     ## Do loss function on var\n",
    "\n",
    "#     # nn.NLLLoss()()\n",
    "    \n",
    "#     new_mu = new_mu.unsqueeze(3)\n",
    "#     new_std = new_std.unsqueeze(3)\n",
    "#     # z = torch.normal(mean = new_mu.flatten(), std = new_std.flatten())\n",
    "#     # # print(z.shape)\n",
    "#     # z = rearrange(z, \"(nf nx nt) -> nf nx nt 1\", nf = x_train.shape[0], nx = x_train.shape[1], nt = x_train.shape[2])\n",
    "    \n",
    "#     model_loss = nn.MSELoss()(y_train, new_mu)\n",
    "#     # model_loss = utils.nll_mu_var((new_mu, torch.square(new_std)), y_train)\n",
    "#     # mse_loss = nn.MSELoss()(y_train, z)\n",
    "\n",
    "#     optimizer.zero_grad()\n",
    "#     model_loss.backward()\n",
    "#     optimizer.step()\n",
    "    \n",
    "#     if epoch % 10 == 0:\n",
    "#         # print(target.shape, z.shape)\n",
    "#         print(\"loss: \",model_loss.item(), \"epoch: \", epoch)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "0947eca8-ad15-46c1-99cf-2ffde0d31fe2",
   "metadata": {},
   "outputs": [],
   "source": [
    "torch.norm(y_train - new_mu_output_var)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "6f3a586e-b280-4d34-9389-06fd6750685f",
   "metadata": {},
   "outputs": [],
   "source": [
    "from collections import OrderedDict\n",
    "utils.set_seed(0)\n",
    "\n",
    "mlp_model = nn.Sequential(OrderedDict([\n",
    "    ('dense1', nn.Linear(x_train_reshaped.shape[1], 2*x_train_reshaped.shape[1])),\n",
    "    ('act1', nn.ReLU()),\n",
    "    ('dense2', nn.Linear(2*x_train_reshaped.shape[1], x_train_reshaped.shape[1])),\n",
    "    ('act2', nn.ReLU()),\n",
    "    ('output', nn.Linear(x_train_reshaped.shape[1], 2*x_train_reshaped.shape[1])),\n",
    "    # ('outact', nn.Sigmoid()),\n",
    "]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f436788e-dfc2-4480-a9b8-061ca4d84a40",
   "metadata": {},
   "outputs": [],
   "source": [
    "class CombinedModel(nn.Module):\n",
    "    def __init__(self, model1, model2):\n",
    "        super(CombinedModel, self).__init__()\n",
    "        self.model1 = model1\n",
    "        self.model2 = model2\n",
    "    def forward(self, x):\n",
    "        output1 = self.model1(x)\n",
    "        reshaped_output = rearrange(output1, \" nf nx nt 1 -> nf (nx nt)\")\n",
    "        output2 = self.model2(reshaped_output)\n",
    "        mu, var = output2.split(x_train_reshaped.shape[1], dim=1)\n",
    "        mu = rearrange(mu, \"nf (nx nt) -> nf nx nt 1\", nx = x.shape[1], nt = x.shape[2])\n",
    "        var = rearrange(var, \"nf (nx nt) -> nf nx nt 1\", nx = x.shape[1], nt = x.shape[2])\n",
    "        return mu, torch.abs(var)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "17c0beec-8356-497f-a6c9-90f2e842b669",
   "metadata": {},
   "outputs": [],
   "source": [
    "FNO2d_params = {\"modes1\": fno_modes, \"modes2\": fno_modes2, \"width\": fno_width, \"output_var\": False}\n",
    "fno_model = FNO2d(**FNO2d_params).to(device)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d7d41c13-255f-495b-9777-51fb02165b37",
   "metadata": {},
   "outputs": [],
   "source": [
    "comb_model = CombinedModel(fno_model, mlp_model)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "1bc5ab92-38dd-4ceb-8c53-393d97408f84",
   "metadata": {},
   "outputs": [],
   "source": [
    "new_mu, new_std = train_model(comb_model)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f8caf43b-f5c5-418b-b74a-bb7e80763500",
   "metadata": {},
   "outputs": [],
   "source": [
    "# optimizer = optim.Adam(comb_model.parameters(), lr=0.01)\n",
    "\n",
    "# for epoch in range(200):\n",
    "#     mu, std = comb_model(x_train)\n",
    "#     # mu, std = output.split(x_train_reshaped.shape[1], dim=1)\n",
    "#     # std = torch.abs(std)\n",
    "#     # mu = rearrange(mu, \"nf (nx nt) -> nf nx nt 1\", nx = x_train.shape[1], nt = x_train.shape[2])\n",
    "#     # std = rearrange(std, \"nf (nx nt) -> nf nx nt 1\", nx = x_train.shape[1], nt = x_train.shape[2])\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",
    "\n",
    "#     ## Do loss function on var\n",
    "\n",
    "#     # nn.NLLLoss()()\n",
    "    \n",
    "#     new_mu = new_mu.unsqueeze(3)\n",
    "#     new_std = new_std.unsqueeze(3)\n",
    "#     # z = torch.normal(mean = new_mu.flatten(), std = new_std.flatten())\n",
    "#     # # print(z.shape)\n",
    "#     # z = rearrange(z, \"(nf nx nt) -> nf nx nt 1\", nf = x_train.shape[0], nx = x_train.shape[1], nt = x_train.shape[2])\n",
    "    \n",
    "#     model_loss = nn.MSELoss()(y_train, new_mu)\n",
    "#     # model_loss = utils.nll_mu_var((new_mu, torch.square(new_std)), y_train)\n",
    "#     # mse_loss = nn.MSELoss()(y_train, z)\n",
    "\n",
    "#     optimizer.zero_grad()\n",
    "#     model_loss.backward()\n",
    "#     optimizer.step()\n",
    "    \n",
    "#     if epoch % 10 == 0:\n",
    "#         # print(target.shape, z.shape)\n",
    "#         print(\"loss: \",model_loss.item(), \"epoch: \", epoch)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b244124a-a61c-4e8f-8613-e5fdd24ce420",
   "metadata": {},
   "outputs": [],
   "source": [
    "torch.norm(y_train - new_mu)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "94f792d8-a8f6-44d2-af79-1e8c6248493c",
   "metadata": {},
   "outputs": [],
   "source": [
    "with torch.no_grad():\n",
    "    plt.ylabel(\"u(x,t=1.0)\")\n",
    "    plt.title(\"Learning Heat Equation for k = {k:.2f}\".format(k = x_train[1,0,0,0]))\n",
    "    plt.xlabel(\"x\")\n",
    "    plt.plot(y_train[1,:,-1,:])\n",
    "    plt.plot(new_mu[1,:,-1,0], '--')\n",
    "    plt.plot(new_mu_output_var[1,:,-1,0], '--')\n",
    "    plt.legend([\"true\", \"predicted_MLP\", \"predicted_var_FNO\"])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "31eb8ddb-a3f1-4cc0-bead-f46005e78837",
   "metadata": {},
   "outputs": [],
   "source": [
    "with torch.no_grad():\n",
    "    plt.ylabel(\"u(x,t=1.0)\")\n",
    "    plt.title(\"Learning Heat Equation for k = {k:.2f}\".format(k = x_train[1,0,0,0]))\n",
    "    plt.xlabel(\"x\")\n",
    "    plt.plot(grid, new_mu[1,:,-1,0], lw=2, label = \"predicted $\\mu$ and $\\pm 3\\sigma$ (FNO + MLP)\")\n",
    "    plt.fill_between(grid, new_mu[1,:,-1,0]+3*new_std[1,:,-1,0], new_mu[1,:,-1,0]-3*new_std[1,:,-1,0], alpha=0.2)\n",
    "    plt.plot(grid, new_mu_output_var[1,:,-1,0], lw=2, label = \"predicted $\\mu$ and $\\pm 3\\sigma$ (varFNO)\")\n",
    "    plt.fill_between(grid, new_mu_output_var[1,:,-1,0]+3*new_std_output_var[1,:,-1,0], new_mu_output_var[1,:,-1,0]-3*new_std_output_var[1,:,-1,0], alpha=0.2)\n",
    "    plt.plot(grid, y_train[1,:,-1,:], color = \"green\", label = \"true\")\n",
    "    plt.legend(loc=\"upper right\")\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c5b75056-7e4a-437a-a7a8-56dba6d29981",
   "metadata": {},
   "source": [
    "## Training Models with Negative Log Likelihood functions"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "23755c06-785b-44d4-bcdd-a53f764a49ba",
   "metadata": {},
   "outputs": [],
   "source": [
    "utils.set_seed(0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3d72b16b-54dc-41f7-9b0a-15e4cba4a48a",
   "metadata": {},
   "outputs": [],
   "source": [
    "FNO2d_params = {\"modes1\": fno_modes, \"modes2\": fno_modes2, \"width\": fno_width, \"output_var\": True}\n",
    "model = FNO2d(**FNO2d_params).to(device)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b2928f32-96b8-43da-b3c6-90aa60c20b73",
   "metadata": {},
   "outputs": [],
   "source": [
    "optimizer = optim.Adam(model.parameters(), lr=0.01)\n",
    "for epoch in range(200):\n",
    "    mu, var = model(x_train)\n",
    "    \n",
    "    std = torch.sqrt(var)\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",
    "    \n",
    "    ## Do loss function on var\n",
    "\n",
    "    # nn.NLLLoss()()\n",
    "    \n",
    "    new_mu = new_mu.unsqueeze(3)\n",
    "    new_std = new_std.unsqueeze(3)\n",
    "    new_var = torch.square(new_std)\n",
    "\n",
    "    model_loss = nn.GaussianNLLLoss()(new_mu, y_train, new_var)\n",
    "    \n",
    "    optimizer.zero_grad()\n",
    "    model_loss.backward()\n",
    "    optimizer.step()\n",
    "    \n",
    "    if epoch % 10 == 0:\n",
    "        # print(target.shape, z.shape)\n",
    "        print(\"loss: \",model_loss.item(), \"epoch: \", epoch)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "da3c7cf9-1d4f-4571-9750-d75c53b872e8",
   "metadata": {},
   "outputs": [],
   "source": [
    "new_mu_varfno_nll = torch.clone(new_mu)\n",
    "new_std_varfno_nll = torch.clone(new_std)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "2b7c0a66-52c5-4957-a0e6-49f9c48566d7",
   "metadata": {},
   "outputs": [],
   "source": [
    "torch.norm(y_train-new_mu_varfno_nll)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f9655397-a250-4b3d-af60-a16866065a35",
   "metadata": {},
   "outputs": [],
   "source": [
    "utils.set_seed(0)\n",
    "\n",
    "mlp_model = nn.Sequential(OrderedDict([\n",
    "    ('dense1', nn.Linear(x_train_reshaped.shape[1], 2*x_train_reshaped.shape[1])),\n",
    "    ('act1', nn.ReLU()),\n",
    "    ('dense2', nn.Linear(2*x_train_reshaped.shape[1], x_train_reshaped.shape[1])),\n",
    "    ('act2', nn.ReLU()),\n",
    "    ('output', nn.Linear(x_train_reshaped.shape[1], 2*x_train_reshaped.shape[1])),\n",
    "    # ('outact', nn.Sigmoid()),\n",
    "]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b9de0730-b4e3-4239-89d7-edf1361221c7",
   "metadata": {},
   "outputs": [],
   "source": [
    "FNO2d_params = {\"modes1\": fno_modes, \"modes2\": fno_modes2, \"width\": fno_width, \"output_var\": False}\n",
    "fno_model = FNO2d(**FNO2d_params).to(device)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c8b50265-6126-44bc-b376-f6a332816e86",
   "metadata": {},
   "outputs": [],
   "source": [
    "class CombinedModel(nn.Module):\n",
    "    def __init__(self, model1, model2):\n",
    "        super(CombinedModel, self).__init__()\n",
    "        self.model1 = model1\n",
    "        self.model2 = model2\n",
    "    def forward(self, x):\n",
    "        output1 = self.model1(x)\n",
    "        reshaped_output = rearrange(output1, \" nf nx nt 1 -> nf (nx nt)\")\n",
    "        output2 = self.model2(reshaped_output)\n",
    "        mu, var = output2.split(x_train_reshaped.shape[1], dim=1)\n",
    "        mu = rearrange(mu, \"nf (nx nt) -> nf nx nt 1\", nx = x.shape[1], nt = x.shape[2])\n",
    "        var = rearrange(var, \"nf (nx nt) -> nf nx nt 1\", nx = x.shape[1], nt = x.shape[2])\n",
    "        return mu, torch.abs(var)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b85c4980-5ea4-450d-b09e-c028a3dd93fa",
   "metadata": {},
   "outputs": [],
   "source": [
    "comb_model = CombinedModel(fno_model, mlp_model)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "138c720e-65c6-4e2b-98a3-c468c52312ab",
   "metadata": {},
   "outputs": [],
   "source": [
    "optimizer = optim.Adam(comb_model.parameters(), lr=0.01)\n",
    "for epoch in range(200):\n",
    "    mu, var = comb_model(x_train)\n",
    "    \n",
    "    std = torch.sqrt(var)\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",
    "    \n",
    "    ## Do loss function on var\n",
    "\n",
    "    # nn.NLLLoss()()\n",
    "    \n",
    "    new_mu = new_mu.unsqueeze(3)\n",
    "    new_std = new_std.unsqueeze(3)\n",
    "    new_var = torch.square(new_std)\n",
    "\n",
    "    model_loss = nn.GaussianNLLLoss()(new_mu, y_train, new_var)\n",
    "\n",
    "    optimizer.zero_grad()\n",
    "    model_loss.backward()\n",
    "    optimizer.step()\n",
    "    \n",
    "    if epoch % 10 == 0:\n",
    "        # print(target.shape, z.shape)\n",
    "        print(\"loss: \",model_loss.item(), \"epoch: \", epoch)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "538b1e13-6ee5-4700-942c-31ae8782cd02",
   "metadata": {},
   "outputs": [],
   "source": [
    "torch.norm(y_train-new_mu)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "44bce8ad-04fa-41c1-89ca-40068f7e0923",
   "metadata": {},
   "outputs": [],
   "source": [
    "with torch.no_grad():\n",
    "    plt.ylabel(\"u(x,t=1.0)\")\n",
    "    plt.title(\"Learning Heat Equation for k = {k:.2f}\".format(k = x_train[1,0,0,0]))\n",
    "    plt.xlabel(\"x\")\n",
    "    plt.plot(y_train[1,:,-1,:])\n",
    "    plt.plot(new_mu[1,:,-1,0], '--')\n",
    "    plt.plot(new_mu_varfno_nll[1,:,-1,0], '--')\n",
    "    plt.legend([\"true\", \"predicted_MLP\", \"predicted_var_FNO\"])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "daabc6b2-bafe-4af5-94ae-9e20f2bdb91d",
   "metadata": {},
   "outputs": [],
   "source": [
    "with torch.no_grad():\n",
    "    plt.ylabel(\"u(x,t=1.0)\")\n",
    "    plt.title(\"Learning Heat Equation for k = {k:.2f}\".format(k = x_train[1,0,0,0]))\n",
    "    plt.xlabel(\"x\")\n",
    "    plt.plot(grid, new_mu[1,:,-1,0], lw=2, label = \"predicted $\\mu$ and $\\pm 3\\sigma$ (FNO + MLP)\")\n",
    "    plt.fill_between(grid, new_mu[1,:,-1,0]+3*new_std[1,:,-1,0], new_mu[1,:,-1,0]-3*new_std[1,:,-1,0], alpha=0.2)\n",
    "    plt.plot(grid, new_mu_varfno_nll[1,:,-1,0], lw=2, label = \"predicted $\\mu$ and $\\pm 3\\sigma$ (varFNO)\")\n",
    "    plt.fill_between(grid, new_mu_varfno_nll[1,:,-1,0]+3*new_std_varfno_nll[1,:,-1,0], new_mu_varfno_nll[1,:,-1,0]-3*new_std_varfno_nll[1,:,-1,0], alpha=0.2)\n",
    "    plt.plot(grid, y_train[1,:,-1,:], color = \"green\", label = \"true\")\n",
    "    plt.legend(loc=\"upper right\")\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "941dfe9d-2e17-4e17-a523-4ede084d6f24",
   "metadata": {},
   "source": [
    "## Estabilishing the connection between varFNO and MNO+FLP\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "34b91265-db88-4255-8dab-b8ddb40bc068",
   "metadata": {},
   "outputs": [],
   "source": [
    "from collections import OrderedDict\n",
    "utils.set_seed(0)\n",
    "\n",
    "mlp_model = nn.Sequential(OrderedDict([\n",
    "    ('dense1', nn.Linear(x_train_reshaped.shape[1], 2*x_train_reshaped.shape[1])),\n",
    "    # ('act1', nn.ReLU()),\n",
    "    # ('dense2', nn.Linear(2*x_train_reshaped.shape[1], x_train_reshaped.shape[1])),\n",
    "    # ('act2', nn.ReLU()),\n",
    "    # ('output', nn.Linear(x_train_reshaped.shape[1], 2*x_train_reshaped.shape[1])),\n",
    "    # ('outact', nn.Sigmoid()),\n",
    "]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d7f1f5df-9a55-4bf8-a16d-d745efedb206",
   "metadata": {},
   "outputs": [],
   "source": [
    "class CombinedModel(nn.Module):\n",
    "    def __init__(self, model1, model2):\n",
    "        super(CombinedModel, self).__init__()\n",
    "        self.model1 = model1\n",
    "        self.model2 = model2\n",
    "    def forward(self, x):\n",
    "        output1 = self.model1(x)\n",
    "        reshaped_output = rearrange(output1, \" nf nx nt 1 -> nf (nx nt)\")\n",
    "        output2 = self.model2(reshaped_output)\n",
    "        mu, std = output2.split(x_train_reshaped.shape[1], dim=1)\n",
    "        mu = rearrange(mu, \"nf (nx nt) -> nf nx nt 1\", nx = x.shape[1], nt = x.shape[2])\n",
    "        std = rearrange(std, \"nf (nx nt) -> nf nx nt 1\", nx = x.shape[1], nt = x.shape[2])\n",
    "        return mu, torch.abs(std)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "077f65d4-1866-4514-b659-5b69e6dc3552",
   "metadata": {},
   "outputs": [],
   "source": [
    "FNO2d_params = {\"modes1\": fno_modes, \"modes2\": fno_modes2, \"width\": fno_width, \"output_var\": False}\n",
    "fno_model = FNO2d(**FNO2d_params).to(device)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b8aa2cb0-d31a-4c33-8fd3-1398864ddb08",
   "metadata": {},
   "outputs": [],
   "source": [
    "comb_model = CombinedModel(fno_model, mlp_model)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "50d8ef61-9023-4e0e-98a7-e03be73cb1e3",
   "metadata": {},
   "outputs": [],
   "source": [
    "new_mu, new_std = train_model(comb_model)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "5d771b4e-4860-4715-8401-350214ca6a81",
   "metadata": {},
   "outputs": [],
   "source": [
    "with torch.no_grad():\n",
    "    plt.ylabel(\"u(x,t=1.0)\")\n",
    "    plt.title(\"Learning Heat Equation for k = {k:.2f}\".format(k = x_train[1,0,0,0]))\n",
    "    plt.xlabel(\"x\")\n",
    "    plt.plot(grid, new_mu[1,:,-1,0], lw=2, label = \"predicted $\\mu$ and $\\pm 3\\sigma$ (FNO + MLP)\")\n",
    "    plt.fill_between(grid, new_mu[1,:,-1,0]+3*new_std[1,:,-1,0], new_mu[1,:,-1,0]-3*new_std[1,:,-1,0], alpha=0.2)\n",
    "    plt.plot(grid, new_mu_output_var[1,:,-1,0], lw=2, label = \"predicted $\\mu$ and $\\pm 3\\sigma$ (varFNO)\")\n",
    "    plt.fill_between(grid, new_mu_output_var[1,:,-1,0]+3*new_std_output_var[1,:,-1,0], new_mu_output_var[1,:,-1,0]-3*new_std_output_var[1,:,-1,0], alpha=0.2)\n",
    "    plt.plot(grid, y_train[1,:,-1,:], color = \"green\", label = \"true\")\n",
    "    plt.legend(loc=\"upper right\")\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c80209fd-4219-450b-98af-7946b0e7183d",
   "metadata": {},
   "source": [
    "## Analysis of imposing positive constraints to predict variance"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "0985aa6c-96d0-4584-8265-731cfa7328d9",
   "metadata": {},
   "outputs": [],
   "source": [
    "from collections import OrderedDict\n",
    "utils.set_seed(0)\n",
    "\n",
    "mlp_model = nn.Sequential(OrderedDict([\n",
    "    ('dense1', nn.Linear(x_train_reshaped.shape[1], 2*x_train_reshaped.shape[1])),\n",
    "    ('act1', nn.ReLU()),\n",
    "    ('dense2', nn.Linear(2*x_train_reshaped.shape[1], x_train_reshaped.shape[1])),\n",
    "    ('act2', nn.ReLU()),\n",
    "    ('output', nn.Linear(x_train_reshaped.shape[1], 2*x_train_reshaped.shape[1])),\n",
    "    # ('outact', nn.Sigmoid()),\n",
    "]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "6644f1ca-c440-4804-a557-376052ef9197",
   "metadata": {},
   "outputs": [],
   "source": [
    "class CombinedModel(nn.Module):\n",
    "    def __init__(self, model1, model2):\n",
    "        super(CombinedModel, self).__init__()\n",
    "        self.model1 = model1\n",
    "        self.model2 = model2\n",
    "    def forward(self, x):\n",
    "        output1 = self.model1(x)\n",
    "        reshaped_output = rearrange(output1, \" nf nx nt 1 -> nf (nx nt)\")\n",
    "        output2 = self.model2(reshaped_output)\n",
    "        mu, var = output2.split(x_train_reshaped.shape[1], dim=1)\n",
    "        mu = rearrange(mu, \"nf (nx nt) -> nf nx nt 1\", nx = x.shape[1], nt = x.shape[2])\n",
    "        var = rearrange(var, \"nf (nx nt) -> nf nx nt 1\", nx = x.shape[1], nt = x.shape[2])\n",
    "        return mu, F.softplus(var)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "82097422-3f23-4aa7-acd0-bb8d663e2dfe",
   "metadata": {},
   "outputs": [],
   "source": [
    "FNO2d_params = {\"modes1\": fno_modes, \"modes2\": fno_modes2, \"width\": fno_width, \"output_var\": False}\n",
    "fno_model = FNO2d(**FNO2d_params).to(device)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "22308625-bb7c-4bcc-a2fe-39f4fa8a3d2d",
   "metadata": {},
   "outputs": [],
   "source": [
    "comb_model = CombinedModel(fno_model, mlp_model)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c0f794d2-6b0b-4f67-9c9e-c437c5b268ac",
   "metadata": {},
   "outputs": [],
   "source": [
    "new_mu, new_std = train_model(comb_model)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "1569b449-5c1e-4f5a-995b-039633850b67",
   "metadata": {},
   "outputs": [],
   "source": [
    "torch.norm(y_train - new_mu)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "1de7fa22-7418-418a-a144-f0e4fb8ac3b9",
   "metadata": {},
   "outputs": [],
   "source": [
    "with torch.no_grad():\n",
    "    plt.ylabel(\"u(x,t=1.0)\")\n",
    "    plt.title(\"Learning Heat Equation for k = {k:.2f}\".format(k = x_train[1,0,0,0]))\n",
    "    plt.xlabel(\"x\")\n",
    "    plt.plot(y_train[1,:,-1,:])\n",
    "    plt.plot(new_mu[1,:,-1,0], '--')\n",
    "    plt.plot(new_mu_output_var[1,:,-1,0], '--')\n",
    "    plt.legend([\"true\", \"predicted_MLP\", \"predicted_var_FNO\"])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "1ba607de-b6ef-480c-9f49-91f95656d4b9",
   "metadata": {},
   "outputs": [],
   "source": [
    "with torch.no_grad():\n",
    "    plt.ylabel(\"u(x,t=1.0)\")\n",
    "    plt.title(\"Learning Heat Equation for k = {k:.2f}\".format(k = x_train[1,0,0,0]))\n",
    "    plt.xlabel(\"x\")\n",
    "    plt.plot(grid, new_mu[1,:,-1,0], lw=2, label = \"predicted $\\mu$ and $\\pm 3\\sigma$ (FNO + MLP)\")\n",
    "    plt.fill_between(grid, new_mu[1,:,-1,0]+3*new_std[1,:,-1,0], new_mu[1,:,-1,0]-3*new_std[1,:,-1,0], alpha=0.2)\n",
    "    plt.plot(grid, new_mu_output_var[1,:,-1,0], lw=2, label = \"predicted $\\mu$ and $\\pm 3\\sigma$ (varFNO)\")\n",
    "    plt.fill_between(grid, new_mu_output_var[1,:,-1,0]+3*new_std_output_var[1,:,-1,0], new_mu_output_var[1,:,-1,0]-3*new_std_output_var[1,:,-1,0], alpha=0.2)\n",
    "    plt.plot(grid, y_train[1,:,-1,:], color = \"green\", label = \"true\")\n",
    "    plt.legend(loc=\"upper right\")\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "09bf6d02-ea60-44a6-850f-aad6a2735fe9",
   "metadata": {},
   "outputs": [],
   "source": [
    "from collections import OrderedDict\n",
    "utils.set_seed(0)\n",
    "\n",
    "mlp_model = nn.Sequential(OrderedDict([\n",
    "    ('dense1', nn.Linear(x_train_reshaped.shape[1], 2*x_train_reshaped.shape[1])),\n",
    "    ('act1', nn.ReLU()),\n",
    "    ('dense2', nn.Linear(2*x_train_reshaped.shape[1], x_train_reshaped.shape[1])),\n",
    "    ('act2', nn.ReLU()),\n",
    "    ('output', nn.Linear(x_train_reshaped.shape[1], 2*x_train_reshaped.shape[1])),\n",
    "    # ('outact', nn.Sigmoid()),\n",
    "]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b11de612-2de1-4ba8-ba69-8e1c404919ca",
   "metadata": {},
   "outputs": [],
   "source": [
    "class CombinedModel(nn.Module):\n",
    "    def __init__(self, model1, model2):\n",
    "        super(CombinedModel, self).__init__()\n",
    "        self.model1 = model1\n",
    "        self.model2 = model2\n",
    "    def forward(self, x):\n",
    "        output1 = self.model1(x)\n",
    "        reshaped_output = rearrange(output1, \" nf nx nt 1 -> nf (nx nt)\")\n",
    "        output2 = self.model2(reshaped_output)\n",
    "        mu, var = output2.split(x_train_reshaped.shape[1], dim=1)\n",
    "        mu = rearrange(mu, \"nf (nx nt) -> nf nx nt 1\", nx = x.shape[1], nt = x.shape[2])\n",
    "        var = rearrange(var, \"nf (nx nt) -> nf nx nt 1\", nx = x.shape[1], nt = x.shape[2])\n",
    "        return mu, torch.square(var)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "dcb1f02d-9147-48f9-b138-0008d7d2b4b0",
   "metadata": {},
   "outputs": [],
   "source": [
    "FNO2d_params = {\"modes1\": fno_modes, \"modes2\": fno_modes2, \"width\": fno_width, \"output_var\": False}\n",
    "fno_model = FNO2d(**FNO2d_params).to(device)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3cdf7e34-41e1-4b09-90a0-10b81fe619b9",
   "metadata": {},
   "outputs": [],
   "source": [
    "comb_model = CombinedModel(fno_model, mlp_model)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "770f849d-f351-4a98-84bb-668e20e6b763",
   "metadata": {},
   "outputs": [],
   "source": [
    "new_mu, new_std = train_model(comb_model)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "553a25b6-bfaa-4021-b8c1-17b7a72c95a7",
   "metadata": {},
   "outputs": [],
   "source": [
    "torch.norm(y_train - new_mu)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "6adeb7d4-bf99-4769-bef9-fa09b05b5945",
   "metadata": {},
   "outputs": [],
   "source": [
    "with torch.no_grad():\n",
    "    plt.ylabel(\"u(x,t=1.0)\")\n",
    "    plt.title(\"Learning Heat Equation for k = {k:.2f}\".format(k = x_train[1,0,0,0]))\n",
    "    plt.xlabel(\"x\")\n",
    "    plt.plot(y_train[1,:,-1,:])\n",
    "    plt.plot(new_mu[1,:,-1,0], '--')\n",
    "    plt.plot(new_mu_output_var[1,:,-1,0], '--')\n",
    "    plt.legend([\"true\", \"predicted_MLP\", \"predicted_var_FNO\"])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "48dd7a71-4063-437c-9fab-a553f27548e2",
   "metadata": {},
   "outputs": [],
   "source": [
    "with torch.no_grad():\n",
    "    plt.ylabel(\"u(x,t=1.0)\")\n",
    "    plt.title(\"Learning Heat Equation for k = {k:.2f}\".format(k = x_train[1,0,0,0]))\n",
    "    plt.xlabel(\"x\")\n",
    "    plt.plot(grid, new_mu[1,:,-1,0], lw=2, label = \"predicted $\\mu$ and $\\pm 3\\sigma$ (FNO + MLP)\")\n",
    "    plt.fill_between(grid, new_mu[1,:,-1,0]+3*new_std[1,:,-1,0], new_mu[1,:,-1,0]-3*new_std[1,:,-1,0], alpha=0.2)\n",
    "    plt.plot(grid, new_mu_output_var[1,:,-1,0], lw=2, label = \"predicted $\\mu$ and $\\pm 3\\sigma$ (varFNO)\")\n",
    "    plt.fill_between(grid, new_mu_output_var[1,:,-1,0]+3*new_std_output_var[1,:,-1,0], new_mu_output_var[1,:,-1,0]-3*new_std_output_var[1,:,-1,0], alpha=0.2)\n",
    "    plt.plot(grid, y_train[1,:,-1,:], color = \"green\", label = \"true\")\n",
    "    plt.legend(loc=\"upper right\")\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "1ae5b480-a746-4459-9f5e-6ef9d8648f2d",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "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
}
