{
 "cells": [
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Training GNS on the 2D Taylor Green Vortex\n",
    "This tutorial notebook will show how to train a GNS model on the 2D Taylor Green Vortex. It will give a brief overview of LagrangeBench on these topics:\n",
    "- Datasets and case setup\n",
    "- Models\n",
    "- Training tricks and strategies (random-walk noise and pushforward)\n",
    "- Training and inference"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "2023-10-24 20:02:59.973089: E external/xla/xla/stream_executor/cuda/cuda_driver.cc:276] failed call to cuInit: CUDA_ERROR_NO_DEVICE: no CUDA-capable device is detected\n",
      "CUDA backend failed to initialize: FAILED_PRECONDITION: No visible GPU devices. (Set TF_CPP_MIN_LOG_LEVEL=0 and rerun for more info.)\n"
     ]
    }
   ],
   "source": [
    "import os\n",
    "os.environ[\"CUDA_VISIBLE_DEVICES\"] = \"\"\n",
    "os.environ[\"JAX_ENABLE_X64\"] = \"True\"\n",
    "\n",
    "import lagrangebench\n",
    "import haiku as hk\n",
    "import jax.numpy as jnp\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "import pickle\n",
    "import matplotlib.animation as animation"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Datasets\n",
    "First thing to do is to load the dataset. The simplest way to do this is by using e.g. the `lagrangebench.data.TGV2D` class for the 2-dimensional Taylor-Green vortex problem. It will automatically download the HDF5 files if they are not found in the respective folder, and it will take care of setting up the dataset. Note that for the validation/test set you need to specify a positive number of rollout steps, e.g. `n_rollout_steps=5`. This means that the dataset will not split the trajectory into subsequences and keep whole rollouts for evaluation."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "This is a 2D dataset called TGV.\n",
      "Train snapshot have shape (2500, 7, 2) (n_nodes, seq_len, xy pos).\n",
      "Val snapshot have shape (2500, 26, 2) (n_nodes, rollout, xy pos).\n",
      "\n"
     ]
    }
   ],
   "source": [
    "tgv2d_train = lagrangebench.data.TGV2D(\"train\")\n",
    "tgv2d_eval = lagrangebench.data.TGV2D(\"valid\", n_rollout_steps=20)\n",
    "\n",
    "print(\n",
    "    f\"This is a {tgv2d_train.metadata['dim']}D dataset \"\n",
    "    f\"called {tgv2d_train.metadata['case']}.\\n\"\n",
    "    f\"Train snapshot have shape {tgv2d_train[0][0].shape} (n_nodes, seq_len, xy pos).\\n\"\n",
    "    f\"Val snapshot have shape {tgv2d_eval[0][0].shape} (n_nodes, rollout, xy pos).\\n\"\n",
    ")"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Similarly, for other datasets one can use the respective subclass, for example\n",
    "```python\n",
    "rpf_3d_data = lagrangebench.data.RPF3D(\"train\") # 3D Reverse Poiseuille flow\n",
    "dam_2d_data = lagrangebench.data.DAM2D(\"train\") # 2D Dam break\n",
    "# etc.\n",
    "# and in general: \n",
    "lagrangebench.data.H5Dataset(\"train\", dataset_path=\"path/to/dataset\")\n",
    "```"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Models\n",
    "All models should inherit from [`models.BaseModel`](/lagrangebench/models/base.py), and generally include a `_transform` function for feature engineering and graph building. \n",
    "\n",
    "Here we use a small GNS model, with latent dimension of 16 and 4 message passing layers and predicting 2D accelerations. Note that we use a function wrapper beause `haiku.Modules` must be initialized inside `haiku.transform`.\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "def gns(x):\n",
    "    return lagrangebench.GNS(\n",
    "        particle_dimension=tgv2d_train.metadata[\"dim\"],\n",
    "        latent_size=16,\n",
    "        blocks_per_step=2,\n",
    "        num_mp_steps=4,\n",
    "        particle_type_embedding_size=8,\n",
    "    )(x)"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Then apply `haiku.transform` to the model function"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "gns = hk.without_apply_rng(hk.transform_with_state(gns))"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Training tricks and strategies\n",
    "To improve performance and stabilize training we can use a number of tricks and strategies. In this notebook we will cover random-walk noise and pushforward, which are included by default in LagrangeBench. \n",
    "\n",
    "### Random-walk noise\n",
    "Briefly, random-walk noise adds noise to the velocities in the input sequence and adjusts the target positions accordingly. The standard deviation is passed as a parameter, and each noise step is rescaled so that the cumulated noise has the input standard deviation. It applied to GNNs in physics in the [\"Learning to Simulate Complex Physics with Graph Networks\"](https://arxiv.org/abs/2002.09405) paper (2020).\n",
    "\n",
    "### Pushforward\n",
    "Instead of training always on the next step, the pushforward trick will randomly sample a number of \"unroll\" steps to perform and compute the loss only on the last one. Ideally this should inject the \"right amount\" of noise, since it comes from the physical distribution. Because it adds some overhead, the maximum number of unroll steps should not be too large and the probability of unrolling large sequences should be reasonably small. It comes from the [\"Message Passing Neural PDE Solvers\"](https://arxiv.org/pdf/2202.03376.pdf) paper (2022)."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "noise_std = 3e-4\n",
    "\n",
    "pf_config = lagrangebench.PushforwardConfig(\n",
    "    steps=[-1, 500, 700],  # training steps to unlock the relative stage\n",
    "    unrolls=[0, 2, 5],  # number of unroll steps per stage\n",
    "    probs=[7, 2, 1],  # relative probabilities to unroll to the relative stage\n",
    ")"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "For example, this configuration would apply noise with `std=1e-5` and the pushforward trick with three unroll stages (0, 2 and 5), \"unlocking\" the second stage after 5000 training steps and the third stage after 10000 training steps. After 10000 steps, 0-step unroll (normal training) will happen with a probability of 70%, 2-step unroll with a probability of 20% and finally 5-step unroll with a probability of 10%."
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Case\n",
    "The `CaseSetupFn` is an essential part in LagrangeBench. Most of the operations during training, such as data preprocessing, time integration for the ouput and neighbor computation are managed by `CaseSetupFn`. The easies way to initialize it is through the `lagrangebench.case_builder` function."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "bounds = np.array(tgv2d_train.metadata[\"bounds\"])\n",
    "box = bounds[:, 1] - bounds[:, 0]\n",
    "\n",
    "tgv2d_case = lagrangebench.case_builder(\n",
    "    box=box,  # (x,y) array with the world size along each axis. (1.0, 1.0) for 2D TGV\n",
    "    metadata=tgv2d_train.metadata,  # metadata dictionary\n",
    "    input_seq_length=6,  # number of consecutive time steps fed to the model\n",
    "    isotropic_norm=False,  # whether to normalize each dimension independently\n",
    "    noise_std=noise_std,  # noise standard deviation used by the random-walk noise\n",
    ")"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Training and inference\n",
    "Finally, to train GNS on Taylor Green (with noise and pushforward) the `lagrangebench.Trainer` methods comes to hand"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0000, train/loss: 1.68755.\n",
      "0100, train/loss: 0.17377.\n",
      "0200, train/loss: 0.17768.\n",
      "0300, train/loss: 0.16432.\n",
      "0400, train/loss: 3.47414.\n",
      "0500, train/loss: 0.25281.\n",
      "{'val/loss': 0.006483794655650854, 'val/mse1': 3.545727849996183e-06, 'val/mse5': 0.0001387525990139693, 'val/mse10': 0.0009877493139356375, 'val/stdloss': 0.0, 'val/stdmse1': 0.0, 'val/stdmse5': 0.0, 'val/stdmse10': 0.0}\n",
      "0600, train/loss: 0.17975.\n",
      "0700, train/loss: 0.17109.\n",
      "0800, train/loss: 0.21826.\n",
      "0900, train/loss: 0.18034.\n",
      "1000, train/loss: 0.22316.\n",
      "{'val/loss': 0.004053703509271145, 'val/mse1': 2.7963376396655804e-06, 'val/mse5': 9.380515984958038e-05, 'val/mse10': 0.0006204654346220195, 'val/stdloss': 0.0, 'val/stdmse1': 0.0, 'val/stdmse5': 0.0, 'val/stdmse10': 0.0}\n"
     ]
    }
   ],
   "source": [
    "trainer = lagrangebench.Trainer(\n",
    "    model=gns,\n",
    "    case=tgv2d_case,\n",
    "    data_train=tgv2d_train,\n",
    "    data_eval=tgv2d_eval,\n",
    "    pushforward=pf_config,\n",
    "    noise_std=noise_std,\n",
    "    metrics=[\"mse\"],\n",
    "    n_rollout_steps=20,\n",
    "    eval_n_trajs=1,\n",
    "    lr_start=5e-4,\n",
    "    log_steps=100,\n",
    "    eval_steps=500,\n",
    ")\n",
    "\n",
    "params, state, _ = trainer(step_max=1000)"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Now let's see what the trained GNS can do. First, let's get the test data. As for the validation set, `n_rollout_steps=20`."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "tgv2d_test = lagrangebench.TGV2D(\"test\", n_rollout_steps=20)"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Then let's run then evaluation over 20 timesteps. Here we use three evaluation metrics: __position MSE__ and __Sinkhorn__ distance."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "metrics = lagrangebench.infer(\n",
    "    gns,\n",
    "    tgv2d_case,\n",
    "    tgv2d_test,\n",
    "    params,\n",
    "    state,\n",
    "    metrics=[\"mse\", \"sinkhorn\"],\n",
    "    eval_n_trajs=1,\n",
    "    n_rollout_steps=20,\n",
    "    rollout_dir=\"rollouts/\",\n",
    "    out_type=\"pkl\",\n",
    ")[\"rollout_0\"]\n",
    "rollout = pickle.load(open(\"rollouts/rollout_0.pkl\", \"rb\"))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAA+cAAAF2CAYAAAAMW+lzAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjguMCwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy81sbWrAAAACXBIWXMAAA9hAAAPYQGoP6dpAACK80lEQVR4nOzdd3gUVdvH8e+mJ0ASagoECBB6SWghiIIaDYJKFBGwISLYaAZFUIoFX5SiNH2wIVggiCIoYBQpjwoxCCRA6EgvG0pINgTSduf9A9nHSCjBwKb8Ptc1V9yZe2buGZCTe+fMOSbDMAxERERERERExGGcHJ2AiIiIiIiISFmn4lxERERERETEwVSci4iIiIiIiDiYinMRERERERERB1NxLiIiIiIiIuJgKs5FREREREREHEzFuYiIiIiIiIiDqTgXERERERERcTAV5yIiIiIiIiIOpuJcRERERKQEefzxx6ldu/Y171u+fPkrxtWuXZu77777ms4hItdGxbmIiIiIiIiIg7k4OgEREREREbl6H330ETabzdFpiEgRU3EuIiIiIlKCuLq6OjqFIpGVlYWbmxtOTurMKwLq1i5SJr366quYTCZ27drFI488go+PD1WrVmX06NEYhsGhQ4fo1q0b3t7e+Pv7M3ny5Hz7T58+nSZNmuDl5UXFihVp3bo1c+fOzRdz5MgRnnjiCfz8/HB3d6dJkybMmjXrRl6miIhIiZSRkcHQoUOpXbs27u7uVKtWjTvuuIONGzcCF79zvn//fkwmE5MmTeLDDz+kbt26uLu706ZNG/74448rni8pKYmqVavSqVMnzpw5k2/bb7/9Rtu2bfHw8KBOnTp89tlnF+2/d+9eevToQaVKlfDy8qJdu3YsXbo0X8zq1asxmUzExsYyatQoqlevjpeXFxaLxf4e/JEjR4iOjqZ8+fJUrVqVF154AavVeg13UKRk0pNzkTKsZ8+eNGrUiLfeeoulS5cybtw4KlWqxAcffMBtt93G22+/zZdffskLL7xAmzZtuOWWW/joo48YPHgwDzzwAEOGDCErK4vNmzeTkJDAQw89BEBKSgrt2rXDZDIxcOBAqlatyg8//EC/fv2wWCwMHTrUsRcuIiJSjD399NN8/fXXDBw4kMaNG3Pq1Cl+++03tm/fTsuWLS+539y5c8nIyOCpp57CZDIxYcIE7r//fvbu3XvJp+1//PEHUVFRtG7dmsWLF+Pp6WnftmfPHh544AH69etHnz59mDVrFo8//jitWrWiSZMmwPk2v3379pw9e5bBgwdTuXJl5syZw7333svXX3/Nfffdl+98b7zxBm5ubrzwwgtkZ2fj5uYGgNVqJSoqivDwcCZNmsTPP//M5MmTqVu3Ls8888y/vaUiJYMhImXO2LFjDcAYMGCAfV1eXp5Ro0YNw2QyGW+99ZZ9/enTpw1PT0+jT58+hmEYRrdu3YwmTZpc9vj9+vUzAgICjJMnT+Zb36tXL8PHx8c4e/Zs0V2MiIhIKePj42M899xzl9zep08fo1atWvbP+/btMwCjcuXKRmpqqn394sWLDcD4/vvv8+1brlw5wzAM47fffjO8vb2Nrl27GllZWfnOUatWLQMwfvnlF/u648ePG+7u7sawYcPs64YOHWoAxq+//mpfl5GRYQQHBxu1a9c2rFarYRiGsWrVKgMw6tSpc9HvAX369DEA4/XXX8+3PiwszGjVqtUl74NIaaNu7SJl2JNPPmn/b2dnZ1q3bo1hGPTr18++3tfXlwYNGrB3717758OHD1+ym5xhGHzzzTfcc889GIbByZMn7UtUVBTp6en2bnkiIiJyMV9fXxISEjh69Gih9uvZsycVK1a0f7755psB7G34361atYqoqChuv/12Fi5ciLu7+0UxjRs3th8DoGrVqvl+JwBYtmwZbdu2pUOHDvZ15cuXZ8CAAezfv59t27blO2afPn3yPZ3/u6effjrf55tvvrnA3EVKKxXnImVYzZo183328fHBw8ODKlWqXLT+9OnTALz00kuUL1+etm3bEhISwnPPPceaNWvssSdOnCAtLY0PP/yQqlWr5lv69u0LwPHjx6/zlYmIiJRcEyZMIDk5maCgINq2bcurr756VUXqP9v1C4X6hTb8gqysLLp27UpYWBhfffWVvWv5lY534Zh/P96BAwdo0KDBRXGNGjWyb/+74ODgAs/l4eFB1apVL3sukdJOxblIGebs7HxV6+D8E3E439ju3LmT2NhYOnTowDfffEOHDh0YO3YsgH1ql0ceeYTly5cXuNx0003X6YpERERKvgcffJC9e/cyffp0AgMDmThxIk2aNOGHH3647H5XasMvcHd3p2vXriQkJBAXF/evj1cYl3pqfqlziZQlGhBORAqtXLly9OzZk549e5KTk8P999/Pm2++yciRI6latSoVKlTAarUSGRnp6FRFRERKpICAAJ599lmeffZZjh8/TsuWLXnzzTe56667/vWxTSYTX375Jd26daNHjx788MMPdOrU6ZqOVatWLXbu3HnR+h07dti3i8jV0ZNzESmUU6dO5fvs5uZG48aNMQyD3NxcnJ2d6d69O9988w3JyckX7X/ixIkblaqIiEiJY7VaSU9Pz7euWrVqBAYGkp2dXWTncXNzY+HChbRp04Z77rmHdevWXdNxunTpwrp164iPj7evy8zM5MMPP6R27do0bty4qFIWKfX05FxECuXOO+/E39+fm266CT8/P7Zv386MGTPo2rUrFSpUAOCtt95i1apVhIeH079/fxo3bkxqaiobN27k559/JjU11cFXISIiUjxlZGRQo0YNHnjgAVq0aEH58uX5+eef+eOPP5g8eXKRnsvT05MlS5Zw2223cdddd/Hf//6Xpk2bFuoYI0aMYN68edx1110MHjyYSpUqMWfOHPbt28c333yDk5OeBYpcLRXnIlIoTz31FF9++SXvvPMOZ86coUaNGgwePJhRo0bZY/z8/Fi3bh2vv/46Cxcu5P3336dy5co0adKEt99+24HZi4iIFG9eXl48++yz/PTTTyxcuBCbzUa9evV4//33r8t8397e3vz444/ccsst3HHHHfz666/Uq1fvqvf38/Nj7dq1vPTSS0yfPp2srCyaN2/O999/T9euXYs8X5HSzGT8mxEdRERERERERORfUz8TEREREREREQdTcS4iIiIiIiLiYCrORURERERERBxMxbmIiIiIiIiIg6k4FxEREREREXEwFeciIiIiIiIiDlZm5jm32WwcPXqUChUqYDKZHJ2OiIgIhmGQkZFBYGAgTk5F9335e++9x8SJEzGbzbRo0YLp06fTtm3bS8YvWLCA0aNHs3//fkJCQnj77bfp0qVLvjzHjh3LRx99RFpaGjfddBP/+c9/CAkJsce8+eabLF26lKSkJNzc3EhLS8t3jtmzZ9O3b98Cz5+SkkK1atVYvXo1t95660Xbjx07hr+//xWvW229iIgUN4Vq640y4tChQwagRYsWLVq0FLvl0KFDRdbexcbGGm5ubsasWbOMrVu3Gv379zd8fX2NlJSUAuPXrFljODs7GxMmTDC2bdtmjBo1ynB1dTW2bNlij3nrrbcM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",
      "text/plain": [
       "<Figure size 1200x400 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# plot metrics\n",
    "fig, ax = plt.subplots(1, 2, figsize=(12, 4))\n",
    "\n",
    "# plot the metrics\n",
    "for i, m in enumerate([\"mse\", \"sinkhorn\"]):\n",
    "    ax[i].plot(metrics[m])\n",
    "    ax[i].set_title(m)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "rollout of shape (26, 2500, 2) (steps, nodes, xy pos)\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<img src=\"media/scatter.gif\"/>"
      ],
      "text/plain": [
       "<IPython.core.display.Image object>"
      ]
     },
     "execution_count": 11,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "print(f\"rollout of shape {rollout['predicted_rollout'].shape} (steps, nodes, xy pos)\")\n",
    "\n",
    "fig, ax = plt.subplots(1, 2)\n",
    "ax[0].set_xlim([0, 1.0])\n",
    "ax[0].set_ylim([0, 1.0])\n",
    "ax[1].set_xlim([0, 1.0])\n",
    "ax[1].set_ylim([0, 1.0])\n",
    "fig.set_size_inches(10, 5, forward=True)\n",
    "ax[0].set_title(\"GNS\")\n",
    "ax[1].set_title(\"Ground Truth\")\n",
    "\n",
    "rollout_len = rollout[\"predicted_rollout\"].shape[0] - 1\n",
    "\n",
    "scat0 = ax[0].scatter(\n",
    "    rollout[\"predicted_rollout\"][0, :, 0], rollout[\"predicted_rollout\"][0, :, 1]\n",
    ")\n",
    "scat1 = ax[1].scatter(\n",
    "    rollout[\"ground_truth_rollout\"][0, :, 0], rollout[\"ground_truth_rollout\"][0, :, 1]\n",
    ")\n",
    "\n",
    "\n",
    "def animate(i):\n",
    "    scat0.set_offsets(rollout[\"predicted_rollout\"][i])\n",
    "    scat1.set_offsets(rollout[\"ground_truth_rollout\"][i])\n",
    "    return scat0, scat1\n",
    "\n",
    "\n",
    "ani = animation.FuncAnimation(\n",
    "    fig, animate, repeat=True, frames=rollout_len, interval=50\n",
    ")\n",
    "\n",
    "plt.close(fig)\n",
    "\n",
    "writer = animation.PillowWriter(fps=10, metadata=dict(artist=\"Me\"), bitrate=1800)\n",
    "ani.save(\"media/scatter.gif\", writer=writer)\n",
    "\n",
    "from IPython.display import Image\n",
    "\n",
    "Image(url=\"media/scatter.gif\")"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": ".venv",
   "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.10.12"
  },
  "orig_nbformat": 4
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
