{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {
    "metadata": {}
   },
   "outputs": [],
   "source": [
    "import jax\n",
    "import jaxopt\n",
    "import jax.numpy as jnp\n",
    "import numpy as np\n",
    "import optax\n",
    "import equinox as eqx\n",
    "import matplotlib.pyplot as plt\n",
    "%matplotlib inline\n",
    "# import pandas as pd\n",
    "from time import process_time\n",
    "\n",
    "from src.linear_solvers_scan import forward_solve_SD, forward_solve_jacobi\n",
    "from src.BTCS_Stepper import BTCS_Stepper, RandomTruncatedFourierSeries, rollout, dataloader\n",
    "from src.prdp import should_refine\n",
    "\n",
    "# add magic comments for autoreload\n",
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "jax.devices()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "N_DOF = 30   # (along each dimension)\n",
    "MLP_WIDTH = 3000\n",
    "grid_1d = jnp.linspace(0, 1, N_DOF+2)[1:-1]"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Heat equation\n",
    "\n",
    "$$\n",
    "\\frac{\\partial u}{\\partial t} = \\alpha \\nabla^2 u\n",
    "$$\n",
    "\n",
    "where $u$ is the temperature, $\\alpha$ is the thermal diffusivity, and $\\nabla^2$ is the Laplacian operator.\n",
    "\n",
    "## 1D\n",
    "\n",
    "$$\n",
    "\\frac{\\partial u}{\\partial t} = \\alpha \\frac{\\partial^2 u}{\\partial x^2}\n",
    "$$\n",
    "\n",
    "#### Discretization\n",
    "\n",
    "$$\n",
    "\\frac{u_i^{n} - u_i^{n-1}}{\\Delta t} = \\alpha \\frac{u_{i+1}^n - 2u_i^n + u_{i-1}^n}{\\Delta x^2}\n",
    "$$\n",
    "\n",
    "Then\n",
    "$$\n",
    "u_i^{n} - \\frac{\\alpha \\Delta t}{\\Delta x^2} \\left( u_{i+1}^n - 2u_i^n + u_{i-1}^n \\right) = u_i^{n-1}\n",
    "$$\n",
    "\n",
    "## 2D\n",
    "\n",
    "$$\n",
    "\\frac{\\partial u}{\\partial t} = \\alpha \\left( \\frac{\\partial^2 u}{\\partial x^2} + \\frac{\\partial^2 u}{\\partial y^2} \\right)\n",
    "$$\n",
    "\n",
    "#### Discretization: BTCS\n",
    "\n",
    "$$\n",
    "\\frac{u_{i,j}^{n} - u_{i,j}^{n-1}}{\\Delta t} = \\alpha \\left( \\frac{u_{i+1,j}^{n} - 2u_{i,j}^{n} + u_{i-1,j}^{n}}{\\Delta x^2} + \\frac{u_{i,j+1}^{n} - 2u_{i,j}^{n} + u_{i,j-1}^{n}}{\\Delta y^2} \\right)\n",
    "$$\n",
    "\n",
    "Using $\\Delta y = \\Delta x$\n",
    "\n",
    "$$\n",
    "u_{i,j}^{n} - \\frac{\\alpha \\Delta t}{\\Delta x^2} \\left( u_{i+1,j}^{n} + u_{i-1,j}^{n} + u_{i,j+1}^{n} + u_{i,j-1}^{n} - 4u_{i,j}^{n} \\right) = u_{i,j}^{n-1}\n",
    "$$\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "$$\n",
    "u_0(x,y) = sin(2 \\pi x) sin(2 \\pi y)\n",
    "$$"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "metadata": {}
   },
   "outputs": [],
   "source": [
    "# create sinusoidal initial condition on a 2D grid\n",
    "grid_2d = jnp.meshgrid(grid_1d, grid_1d, indexing='ij')\n",
    "u_0_2d = jnp.sin(2 * jnp.pi * grid_2d[0]) * jnp.sin(2 * jnp.pi * grid_2d[1])\n",
    "plt.figure(); plt.title('Initial condition')\n",
    "plt.imshow(u_0_2d, origin='lower')\n",
    "plt.colorbar()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {
    "metadata": {}
   },
   "outputs": [],
   "source": [
    "btcs_stepper = BTCS_Stepper(num_points=N_DOF, dim=2)\n",
    "u_0 = jnp.reshape(u_0_2d, -1) # flattening the 2D initial condition\n",
    "assert u_0.shape == (N_DOF**2,)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {
    "metadata": {}
   },
   "outputs": [],
   "source": [
    "# test the forward solve - direct and solver\n",
    "N_STEPS = 50\n",
    "trj_direct = rollout(btcs_stepper, N_STEPS, include_init=True)(u_0)\n",
    "trj_solver = rollout(btcs_stepper.jacobi_dynamic, N_STEPS, include_init=True, solver_iterations=20)(u_0)\n",
    "assert trj_direct.shape == trj_solver.shape == (N_STEPS+1, N_DOF**2)\n",
    "\n",
    "rel_error = jnp.abs(trj_direct - trj_solver) / jnp.abs(trj_direct)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {
    "metadata": {}
   },
   "outputs": [],
   "source": [
    "# plot the time steps as 2D animations\n",
    "trj_direct_2d = jnp.reshape(trj_direct, (N_STEPS+1, N_DOF, N_DOF))\n",
    "trj_solver_2d = jnp.reshape(trj_solver, (N_STEPS+1, N_DOF, N_DOF))\n",
    "rel_error_2d = jnp.reshape(rel_error, (N_STEPS+1, N_DOF, N_DOF))\n",
    "\n",
    "# create a figure and axes\n",
    "fig = plt.figure(figsize=(12,5))\n",
    "ax1 = plt.subplot(1,2,1)   \n",
    "ax2 = plt.subplot(1,2,2)\n",
    "\n",
    "ax1.set_xlabel('X')\n",
    "ax1.set_ylabel('Y')\n",
    "ax1.set_title('Direct')\n",
    "ax2.set_xlabel('X')\n",
    "ax2.set_ylabel('Y')\n",
    "ax2.set_title('Linear Solver')\n",
    "txt_title = fig.suptitle('Time Step: 0')\n",
    "\n",
    "# create an animation\n",
    "im1 = ax1.imshow(trj_direct_2d[0], origin='lower', animated=True)\n",
    "im2 = ax2.imshow(trj_solver_2d[0], origin='lower', animated=True)\n",
    "plt.close()\n",
    "\n",
    "def animate(i):\n",
    "    im1.set_array(trj_direct_2d[i])\n",
    "    im2.set_array(trj_solver_2d[i])\n",
    "    txt_title.set_text(f'Time Step: {i}')\n",
    "    return (im1, im2)\n",
    "\n",
    "from matplotlib.animation import FuncAnimation\n",
    "anim = FuncAnimation(fig, animate, frames=N_STEPS+1, interval=100, blit=True)\n",
    "\n",
    "# from IPython.display import HTML\n",
    "# HTML(anim.to_html5_video())\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Define training and test data\n",
    "Each $u^0$ is the sum of first `num_modes` sine and cosine modes of the domain with random amplitudes.\n",
    "\n",
    "### 1D\n",
    "\n",
    "$$\n",
    "u_0(x) = \\sum_n ( a_n \\sin(2n \\pi x) + b_n \\cos(2n \\pi x) )\n",
    "$$\n",
    "\n",
    "### 2D\n",
    "\n",
    "$$\n",
    "u_0(x,y) = \\sum_n ( a_n \\sin(2n \\pi x) \\sin(2n \\pi y) + b_n \\cos(2n \\pi x) \\cos(2n \\pi y) + c_n \\sin(2n \\pi x) \\cos(2n \\pi y) + d_n \\cos(2n \\pi x) \\sin(2n \\pi y) )\n",
    "$$\n",
    "\n",
    "where $a_n$, $b_n$, ... are sampled from $U(-1,1)$"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Create training set and dataloader"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {
    "metadata": {}
   },
   "outputs": [],
   "source": [
    "ic_generator = RandomTruncatedFourierSeries(domain_extent=1.0, num_modes=5, dim=2) "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {
    "metadata": {}
   },
   "outputs": [],
   "source": [
    "NUM_TRAINING_SAMPLES = 200\n",
    "\n",
    "# Training Data: X's\n",
    "seed = 1337\n",
    "ic_master_key = jax.random.PRNGKey(seed)\n",
    "ic_keys = jax.random.split(ic_master_key, NUM_TRAINING_SAMPLES)\n",
    "ic_funs = jax.vmap(ic_generator)(ic_keys) # list of functions that generate initial conditions on given grid\n",
    "ic_set_2d = jax.vmap(lambda f: f(grid_2d))(ic_funs) # vmap the list of functions to generate many initial conditions\n",
    "\n",
    "assert ic_set_2d.shape == (NUM_TRAINING_SAMPLES, N_DOF, N_DOF)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "metadata": {}
   },
   "outputs": [],
   "source": [
    "# 1D: Plot a subset of the intial conditions\n",
    "# plt.plot(grid, ic_set[5:10].T);\n",
    "\n",
    "# 2D: Plot an initial condition\n",
    "fig,axs = plt.subplots(1,5)\n",
    "# set figure size\n",
    "fig.set_figwidth(15); fig.set_figheight(3)\n",
    "fig.suptitle('Some Initial Conditions from the training set')\n",
    "axs_index = 0\n",
    "for i in jax.random.randint(ic_master_key, shape=(5,), minval=0, maxval=NUM_TRAINING_SAMPLES):\n",
    "    axs[axs_index].imshow(ic_set_2d[i], origin='lower')\n",
    "    axs[axs_index].axis('off')\n",
    "    axs[axs_index].set_title(f'IC {i}')\n",
    "    axs_index += 1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {
    "metadata": {}
   },
   "outputs": [],
   "source": [
    "# flatten ic_set\n",
    "ic_set = jnp.reshape(ic_set_2d, (NUM_TRAINING_SAMPLES, N_DOF**2))\n",
    "# Training Data: Y's\n",
    "data_trjs_direct = jax.vmap(rollout(btcs_stepper, 2, include_init=True))(ic_set)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Generate validation set"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {
    "metadata": {}
   },
   "outputs": [],
   "source": [
    "N_TEST_SAMPLES = 5\n",
    "# Validation Data: X's\n",
    "test_ic_master_key = jax.random.PRNGKey(seed + 1)\n",
    "test_ic_keys = jax.random.split(test_ic_master_key, N_TEST_SAMPLES)\n",
    "test_ic_funs = jax.vmap(ic_generator)(test_ic_keys)\n",
    "test_ic_set_2d = jax.vmap(lambda f: f(grid_2d))(test_ic_funs)\n",
    "\n",
    "assert test_ic_set_2d.shape == (N_TEST_SAMPLES, N_DOF, N_DOF)\n",
    "\n",
    "val_ic_set = jnp.reshape(test_ic_set_2d, (N_TEST_SAMPLES, N_DOF**2))\n",
    "# Validation Data: Y's\n",
    "val_data_trjs = jax.vmap(rollout(btcs_stepper, 2, include_init=True))(val_ic_set)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "metadata": {}
   },
   "outputs": [],
   "source": [
    "val_data_trjs.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {
    "metadata": {}
   },
   "outputs": [],
   "source": [
    "@eqx.filter_jit\n",
    "def val_loss(m, val_data):\n",
    "    \"\"\"Compute the loss on the test set.\n",
    "\n",
    "    Args:\n",
    "        m: the model to evaluate\n",
    "        val_data: the test data, with shape (n_samples, n_steps, n_dof)\n",
    "    \"\"\"\n",
    "    print(\"compiling val_loss()\")\n",
    "    val_ic_set = val_data[:,0]\n",
    "    pred_trajectories = jax.vmap(rollout(m, 2, include_init=True))(val_ic_set)\n",
    "    pred_1_errors = jnp.linalg.norm(pred_trajectories[:, 1] - val_data[:, 1], axis=1) # normed over n_dof\n",
    "    pred_2_errors = jnp.linalg.norm(pred_trajectories[:, 2] - val_data[:, 2], axis=1) # normed over n_dof\n",
    "    \n",
    "    data_1_norms  = jnp.linalg.norm(val_data[:, 0], axis=1) # norm over n_dof\n",
    "    data_2_norms = jnp.linalg.norm(val_data[:, 1], axis=1) # norm over n_dof\n",
    "    \n",
    "    pred_1_mse_normalized = jnp.mean((pred_1_errors**2 / data_1_norms**2), axis=0) # mean squared for over all samples\n",
    "    pred_2_mse_normalized = jnp.mean((pred_2_errors**2 / data_2_norms**2), axis=0) # mean squared for over all samples\n",
    "    \n",
    "    return jnp.hstack((pred_1_mse_normalized, pred_2_mse_normalized))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {
    "metadata": {}
   },
   "outputs": [],
   "source": [
    "import jax.tree_util as jtu\n",
    "def count_parameters(model: eqx.Module):\n",
    "    return sum(p.size for p in jtu.tree_leaves(eqx.filter(model, eqx.is_array)))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Residuum plot"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "SOLVER_NAME = 'jacobi'\n",
    "def relative_residuum_hist(state, n_inner):\n",
    "    res_2 = btcs_stepper.residuum_history(state, SOLVER_NAME, n_inner) # (n_iter+1, n_dof)\n",
    "    rel_residuum_hist = jnp.linalg.norm(res_2, axis=1) / jnp.linalg.norm(state)\n",
    "    return rel_residuum_hist # (n_iter+1,)\n",
    "\n",
    "pred_1_set = data_trjs_direct[:, 1]\n",
    "res_hist_all = jax.vmap(relative_residuum_hist, in_axes=(0, None))(pred_1_set, 80) # shape: (n_samples, n_iter+1)\n",
    "res_hist_mean = jnp.mean(res_hist_all, axis=0)\n",
    "res_hist_std = jnp.std(res_hist_all, axis=0)\n",
    "\n",
    "fig, ax = plt.subplots(figsize=(3,3))\n",
    "ax.plot(res_hist_mean, label=\"primal residuum\")\n",
    "ax.fill_between(range(res_hist_mean.shape[0]), res_hist_mean - res_hist_std, res_hist_mean + res_hist_std, alpha=0.2)\n",
    "ax.set_yscale(\"log\")\n",
    "ax.set_xlabel(\"# iterations\")\n",
    "ax.set_title(f\"Heat 2D, Residuums, solver={SOLVER_NAME}\")\n",
    "\n",
    "ax.grid(which='major', axis='y')\n",
    "ax.minorticks_on()\n",
    "ax.grid(which='both', axis='x', linestyle='--', linewidth=0.5)\n",
    "ax.grid(which='major', axis='both', linestyle='-', linewidth=1.0)\n",
    "\n",
    "# fig.savefig(f\"figures/heat_2d__residuum__{SOLVER_NAME}.pdf\", bbox_inches=\"tight\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Mix-Chain differentiable physics training\n",
    "One network followed by one iterative solver step\n"
   ]
  },
  {
   "attachments": {
    "image-2.png": {
     "image/png": "iVBORw0KGgoAAAANSUhEUgAAAtYAAAFPCAYAAABpkYnQAAAABHNCSVQICAgIfAhkiAAAABl0RVh0U29mdHdhcmUAZ25vbWUtc2NyZWVuc2hvdO8Dvz4AAAApdEVYdENyZWF0aW9uIFRpbWUARGkgMjcgRmViIDIwMjQgMTk6MDc6NDggQ0VUec+VtQAAIABJREFUeJzs3XlgG+WBNvBHvm/Jh2LHpxznTojHCSEJkFiG0Iaj4HC0hS6xDNuWD9jW2cJCWUqUBdJ0ocRdml26XbBCu9By2aQcZZs0ckJDSEg8zn34kK/Et8e3ZFnW90ewmzuxI2k00vP7h1iRNY+DR/PMq3feUTmdTieIiIiIiOiqBMgdgIiIiIjIF7BYExERERG5AIs1EREREZELsFgTEREREbkAizURERERkQuwWBMRERERuQCLNRERERGRC7BYExERERG5QJDcAYgmqry8HJIkQRRFuaNAp9MhOzsbgiDIHYWIiIhkwmJNiiJJEn71q1+huLgYGRkZ0Gg0ckcaI4oiuru7UVRUhDVr1nhVNiIiInI/FmtSDFEUUVhYiIyMDFRUVECn08kd6TySJMFgMCAnJwelpaUcwSYiIvIjKqfT6ZQ7BNGVyMnJQUFBAYqKiuSOclkmkwmrV69GbW0tR66JiIj8BIs1KYLRaITZbIbZbJY7yhUrKipCXV0dSktL5Y5CREREHsBiTV5PkiTExsaitrbWK6d/XIwkSRAEAWVlZZwSQkRE5Ae43B55PVEUkZubq6hSDQAajQZ6vV5Ro+xEREQ0cSzW5PXMZrNiR3wFQfCK5QCJiIjI/VisSRGUegGgIAiwWCxyxyAiIiIPYLEmIiIiInIBFmsiIiIiIhdgsSYiIiIicgEWayIiIiIiF2CxJiIiIiJyARZrIiIiIiIXYLEmIiIiInIBFmvyeyMjI3A4HHLHICIiIoVjsSa/9s4772DVqlVYuXIlampq5I5DRERECsZiTX7r3Xffxe9//3v87ne/w4MPPgij0Sh3JCIiIlKwILkDEMmhqqoKP/jBD7B7926oVCqo1WreepyIiIiuCkesyS+VlJQgPT0dWVlZAIDt27eju7tb5lRERESkZCzW5HeGh4exceNGPPDAA3A4HLDb7aivr4darZY7GhERESkYp4KQ36moqEB3dzc+/vhjfPTRRwCAzz//HI899pjMyYiIiEjJWKzJ74iiCADYunUrgoKCsGfPHixatAgFBQUyJyMiIiIl41QQ8jsWiwXLli1DYGAgVCoVNm/ejKVLl2LBggVyRyMiIiIF44g1+R2NRgOn04mAgAA4HA78/ve/x29/+1sEBPA8k4iIiCaOTYL8Tn5+Purr6+FwOGAwGPAP//APuOWWW+SORURERArHEWvyO9OmTcNLL72EVatW4a677sK3v/1tuSMRERGRD2CxJr9033334b777pM7BhEREfkQTgUhIiIiInIBFmsiIiIiIhdgsSYiIiIicgEWayIiIiIiF2CxJiIiIiJyARZrIiIiIiIXYLEmIiIiInIBFmsiIiIiIhdgsSYiIiIicgEWayIiIiIiF2CxJiIiIiJyARZrIiIiIiIXYLEmRZAkSe4IEyKKInQ6ndwxiIiIyANYrMnr6fV6iKIod4wJEUURgiDIHYOIiIg8gMWavJ4gCCgvL4fFYpE7yrhIkoTy8nLo9Xq5oxAREZEHsFiT19NoNFizZg0KCwvljjIuRqMR2dnZHLEmIiLyEyqn0+mUOwTRlRAEAStXrsSaNWvkjnJZJpMJRUVFsFgs0Gg0cschIiIiD+CINSmGyWRCaWkpVq5c6bXTQiRJwsqVK2E0GmE2m1mqiYiI/EiQ3AGIrpQgCDCbzSguLoYgCMjMzPSq4iqKIiRJwo9//GOUlJR4VTYiIiJyP04FIcUym82QJOmqVwx5+eWXYTAYkJCQMOHXEAQBOp2O86mJiIj8GIs1+b2kpCSYzWbMnDlT7ihERESkYJxjTURERETkAizWREREREQuwGJNREREROQCLNZERERERC7AYk1ERERE5AIs1kRERERELsBiTURERETkAizWREREREQuwGJNREREROQCLNZERERERC7AYk1ERERE5AIs1kRERERELsBiTURERETkAizWREREREQuwGJNREREROQCLNZERERERC7AYk1ERERE5AIs1kRERERELsBiTURERETkAizWREREREQuwGJNBCAggLsCERERXR22CSIATqdT7ghERESkcCzW5PecTidUKpXcMYiIiEjhWKyJiIiIiFyAxZr8HkesiYiIyBVYrImIiIiIXIDFmvweR6yJiIjIFVisiYiIiIhcgMWa/B5HrImIiMgVWKyJiIiIiFyAxZr8HkesiYiIyBVYrImIiIiIXIDFmvyK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"
    }
   },
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "\n",
    "![image-2.png](attachment:image-2.png)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {
    "metadata": {}
   },
   "outputs": [],
   "source": [
    "learning_rate = optax.exponential_decay(1e-3, 100, 0.9)\n",
    "optimizer = optax.adam(learning_rate)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Constant solver iterations"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "metadata": {}
   },
   "outputs": [],
   "source": [
    "def loss_fn(model, data):\n",
    "    ic = data[:, 0]\n",
    "    # Notice the difference, that the target is now two steps into the future\n",
    "    target = data[:, 2]\n",
    "\n",
    "    prediction_1 = jax.vmap(model)(ic)\n",
    "\n",
    "    # Here, you would then have your differentiable BTCS solver\n",
    "    prediction_2 = jax.vmap(btcs_stepper.jacobi, in_axes=(0, 0, None))(prediction_1, jnp.zeros_like(prediction_1), 20)\n",
    "\n",
    "    return jnp.mean((prediction_2 - target)**2)\n",
    "\n",
    "@eqx.filter_jit\n",
    "def update_fn(model, state, data):\n",
    "    loss, grad = eqx.filter_value_and_grad(loss_fn)(model, data)\n",
    "    updates, new_state = optimizer.update(grad, state, model)\n",
    "    new_model= eqx.apply_updates(model, updates)\n",
    "    return new_model, new_state, loss\n",
    "\n",
    "model_MLP = eqx.nn.MLP(in_size=N_DOF**2, out_size=N_DOF**2, width_size=MLP_WIDTH, depth=3, activation=jax.nn.relu, key=jax.random.PRNGKey(92))\n",
    "print(f\"number of model parameters = {count_parameters(model_MLP)}\")\n",
    "\n",
    "opt_state = optimizer.init(eqx.filter(model_MLP, eqx.is_array))\n",
    "\n",
    "N_EPOCHS = 100\n",
    "BATCH_SIZE = 25\n",
    "loss_history_solver = []\n",
    "rel_error_history_solver = []\n",
    "\n",
    "shuffle_key = jax.random.PRNGKey(42)\n",
    "for epoch in range(N_EPOCHS):\n",
    "    shuffle_key, subkey = jax.random.split(shuffle_key)\n",
    "    loss_mini_batch = []\n",
    "    for batch in dataloader(data_trjs_direct, key=subkey, batch_size=BATCH_SIZE):\n",
    "        model_MLP, opt_state, loss = update_fn(model_MLP, opt_state, batch)\n",
    "        # loss_history_solver.append(loss)\n",
    "        loss_mini_batch.append(loss)\n",
    "    loss_history_solver.append(np.mean(loss_mini_batch))\n",
    "    rel_error = val_loss(model_MLP, val_data_trjs)\n",
    "    rel_error_history_solver.append(rel_error)\n",
    "\n",
    "    print(f\"Epoch {epoch+1}/{N_EPOCHS}, loss: {loss}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "metadata": {}
   },
   "outputs": [],
   "source": [
    "rel_error_history_solver = np.array(rel_error_history_solver)\n",
    "fig, ax = plt.subplots(1, 2, figsize=(12, 4))\n",
    "ax[0].plot(loss_history_solver)\n",
    "ax[0].set_yscale('log'); ax[0].grid()\n",
    "ax[0].set_xlabel('Epoch')\n",
    "ax[0].set_ylabel('Loss (minibatch averaged)')\n",
    "\n",
    "ax[1].plot(rel_error_history_solver[:, 0], label='1-step')\n",
    "ax[1].plot(rel_error_history_solver[:, 1], label='2-step')\n",
    "ax[1].set_yscale('log'); ax[1].grid()\n",
    "ax[1].set_xlabel('Epoch')\n",
    "ax[1].set_ylabel('Relative MSE Error')\n",
    "ax[1].legend()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "metadata": {}
   },
   "outputs": [],
   "source": [
    "print(f\"Mean relative error: {val_loss(model_MLP, val_ic_set, val_data_trjs)}\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Different constant N values"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "metadata": {},
   "outputs": [],
   "source": [
    "def loss_fn(model, data, inner_iterations):\n",
    "    print(\"compiling loss_fn\")\n",
    "    ic = data[:, 0]\n",
    "    target = data[:, 2]\n",
    "    prediction_1 = jax.vmap(model)(ic) # batched forward pass # (batch_size, N_DOF)\n",
    "    prediction_2 = jax.vmap(btcs_stepper.jacobi_dynamic, in_axes=(0, None))(prediction_1, inner_iterations)\n",
    "    return jnp.mean((prediction_2 - target)**2) # MSE over batches as well as space\n",
    "\n",
    "@eqx.filter_jit\n",
    "def update_fn(model, state, data, inner_iterations):\n",
    "    print(\"compiling update_fn\")\n",
    "    loss, grad = eqx.filter_value_and_grad(loss_fn)(model, data, inner_iterations)\n",
    "    updates, new_state = optimizer.update(grad, state, model)\n",
    "    new_model= eqx.apply_updates(model, updates)\n",
    "    return new_model, new_state, loss"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "metadata": {}
   },
   "outputs": [],
   "source": [
    "# SEED_LIST = [1, 2, 25, 50, 1000, 1337, 2668, 3999, 12345, 54321]\n",
    "SEED_LIST = [1]\n",
    "N_INNER_LIST = [1,2,3,4,5,6,7,8,9,10,12,15,20,40,60]\n",
    "SAVE_RESULTS = False\n",
    "\n",
    "N_EPOCHS = 70\n",
    "BATCH_SIZE = 25\n",
    "\n",
    "for seed_count, seed in enumerate(SEED_LIST):\n",
    "    \n",
    "    print(f\"Training with seed {seed} ({seed_count+1} of {len(SEED_LIST)})\")\n",
    "    key = jax.random.PRNGKey(seed)\n",
    "    key, model_init_key = jax.random.split(key)\n",
    "    \n",
    "    losses_all_n = []\n",
    "    errors_all_n = []\n",
    "\n",
    "    # Loop over n_inner\n",
    "    for n_inner in N_INNER_LIST:\n",
    "        print(f\"\\nTraining with {n_inner} inner iterations\\n\")\n",
    "        \n",
    "        # initialize model\n",
    "        model_MLP = eqx.nn.MLP(\n",
    "            in_size=N_DOF**2, out_size=N_DOF**2, \n",
    "            width_size=MLP_WIDTH, depth=3, \n",
    "            activation=jax.nn.relu, \n",
    "            key=model_init_key,\n",
    "        )\n",
    "        \n",
    "        # initialize optimizer\n",
    "        opt_state = optimizer.init(eqx.filter(model_MLP, eqx.is_array))\n",
    "\n",
    "        # init metrics\n",
    "        loss_history = [loss_fn(model_MLP, data_trjs_direct, n_inner)]\n",
    "        error_history = [val_loss(model_MLP, val_data_trjs)]\n",
    "\n",
    "        # training loop\n",
    "        key, shuffle_key = jax.random.split(key)\n",
    "        for epoch in range(N_EPOCHS):\n",
    "            shuffle_key, subkey = jax.random.split(shuffle_key)\n",
    "            loss_mini_batch = []\n",
    "            for batch in dataloader(data_trjs_direct, key=subkey, batch_size=BATCH_SIZE):\n",
    "                model_MLP, opt_state, loss = update_fn(model_MLP, opt_state, batch, n_inner)\n",
    "                loss_mini_batch.append(loss)\n",
    "            \n",
    "            loss_history.append(np.mean(loss_mini_batch))\n",
    "            error_history.append(val_loss(model_MLP, val_data_trjs))\n",
    "            print(f\"Epoch {epoch+1}/{N_EPOCHS}, loss: {loss_history[-1]}, rel error: {error_history[-1]}\")\n",
    "        \n",
    "        losses_all_n.append(loss_history)\n",
    "        errors_all_n.append(np.array(error_history))\n",
    "    \n",
    "    # save results\n",
    "    losses_all_n = np.array(losses_all_n)\n",
    "    errors_all_n = np.array(errors_all_n) # shape (len(N_INNER_LIST), N_EPOCHS, 2)\n",
    "    if SAVE_RESULTS:\n",
    "        df = pd.DataFrame({\n",
    "            \"max_iter\": N_INNER_LIST,\n",
    "            \"losses\": list(losses_all_n),\n",
    "            \"1-step errors\": list(errors_all_n[:,:,0]),\n",
    "            \"2-step errors\": list(errors_all_n[:,:,1]),\n",
    "            \"seed\": seed,\n",
    "        })\n",
    "        file_name = f\"results/heat_2d_sep17_jacobi_unrolled/maxiter_constant__seed_{seed}.pkl\"\n",
    "        df.to_pickle(file_name)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# plot results of last seed\n",
    "\n",
    "errors_all_n = np.array(errors_all_n) # shape (len(N_INNER_LIST), N_EPOCHS, 2)\n",
    "\n",
    "fig, axs = plt.subplots(1, 3, figsize=(15, 5))\n",
    "\n",
    "for i, n_inner in enumerate(N_INNER_LIST):\n",
    "    axs[0].plot(losses_all_n[i], label=f\"{N_INNER_LIST[i]}\")\n",
    "    axs[1].plot(errors_all_n[i,:,0], label=f\"{N_INNER_LIST[i]}\")\n",
    "    axs[2].plot(errors_all_n[i,:,1], label=f\"{N_INNER_LIST[i]}\")\n",
    "\n",
    "axs[0].set_xlabel('Epoch'); axs[0].set_ylabel('Loss (minibatch avg.)'); \n",
    "axs[0].set_yscale('log'); axs[0].grid(); axs[0].set_ylim(1e-5, 1)\n",
    "\n",
    "axs[1].set_xlabel('Epoch'); axs[1].set_ylabel('1 Step nMSE');\n",
    "axs[2].set_xlabel('Epoch'); axs[2].set_ylabel('2-Step nMSE');\n",
    "axs[1].set_yscale('log'); axs[2].set_yscale('log');\n",
    "axs[1].grid(); axs[2].grid()\n",
    "axs[1].set_ylim(1e-3, 10); axs[2].set_ylim(1e-3, 10)\n",
    "\n",
    "plt.legend(bbox_to_anchor=(1.05, 1), loc='upper left')"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### PRDP"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [],
   "source": [
    "N_MIN, N_STEP = 1,1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "SAVE_RESULTS = False\n",
    "SEED_LIST = [1, 2, 25, 50, 1000, 1337, 2668, 3999, 12345, 54321]\n",
    "SEED_LIST = [1]\n",
    "N_EPOCHS = 70\n",
    "BATCH_SIZE = 25\n",
    "\n",
    "for seed_count, seed in enumerate(SEED_LIST):\n",
    "    print(f\"Training with seed {seed} ({seed_count+1}/{len(SEED_LIST)})\")\n",
    "    key = jax.random.PRNGKey(seed)\n",
    "    \n",
    "    # init model to be trained\n",
    "    key, model_init_key = jax.random.split(key)    \n",
    "    model_mlp_prdp = eqx.nn.MLP(\n",
    "        in_size=N_DOF**2, out_size=N_DOF**2, \n",
    "        width_size=MLP_WIDTH, depth=3, \n",
    "        activation=jax.nn.relu, \n",
    "        key=model_init_key,\n",
    "    )\n",
    "\n",
    "    # init optimizer\n",
    "    opt_state = optimizer.init(eqx.filter(model_mlp_prdp, eqx.is_array))\n",
    "    \n",
    "    # initialize metrics\n",
    "    n_inner_tracker = N_MIN\n",
    "    loss_hist_prdp = [loss_fn(model_mlp_prdp, data_trjs_direct, n_inner_tracker)]\n",
    "    error_hist_prdp = [val_loss(model_mlp_prdp, val_data_trjs)]\n",
    "    n_inner_hist_prdp = [np.nan] # no value at zeroth epoch, but need same list length as loss_hist\n",
    "    \n",
    "    # initialize PRDP's Nmax checkpoint error\n",
    "    should_refine.error_checkpoint = error_hist_prdp[-1][1] #[last_error][which-step error]\n",
    "\n",
    "    # training loop\n",
    "    key, shuffle_key = jax.random.split(key)\n",
    "    for epoch in range(N_EPOCHS):\n",
    "        shuffle_key, subkey = jax.random.split(shuffle_key)\n",
    "        loss_mini_batch = []\n",
    "        for batch in dataloader(data_trjs_direct, key=subkey, batch_size=BATCH_SIZE):\n",
    "            model_mlp_prdp, opt_state, loss = update_fn(model_mlp_prdp, opt_state, batch, \n",
    "                                                        n_inner_tracker)\n",
    "            loss_mini_batch.append(loss)\n",
    "        \n",
    "        loss_hist_prdp.append(np.mean(loss_mini_batch))\n",
    "        error_hist_prdp.append(val_loss(model_mlp_prdp, val_data_trjs))\n",
    "        n_inner_hist_prdp.append(n_inner_tracker)\n",
    "        print(f\"Epoch {epoch+1}/{N_EPOCHS}, n_inner: {n_inner_tracker}, loss: {loss_hist_prdp[-1]}, error: {error_hist_prdp[-1]}\")\n",
    "\n",
    "        # PRDP\n",
    "        if should_refine(np.array(error_hist_prdp)[:, 1],   # [all epochs, 2 step error]\n",
    "                         0.9, 0.9, 6, 3): \n",
    "            n_inner_tracker += N_STEP\n",
    "    \n",
    "    # SAVE\n",
    "    loss_hist_prdp = np.array(loss_hist_prdp)\n",
    "    error_hist_prdp = np.array(error_hist_prdp)\n",
    "    if SAVE_RESULTS:\n",
    "        df = pd.DataFrame({\n",
    "            \"losses\": [loss_hist_prdp],\n",
    "            \"1-step errors\": [error_hist_prdp[:,0]],\n",
    "            \"2-step errors\": [error_hist_prdp[:,1]],\n",
    "            \"n_inner\": [n_inner_hist_prdp],\n",
    "            \"max_iter\": \"PRDP\",\n",
    "            \"auto_using\": \"two-step-error\",\n",
    "            \"seed\": seed,\n",
    "        })\n",
    "        file_name = f\"results/heat_2d_sep17_jacobi_unrolled/maxiter_auto__seed_{seed}.pkl\"\n",
    "        df.to_pickle(file_name)\n",
    "    "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "error_hist_prdp = np.array(error_hist_prdp)\n",
    "\n",
    "fig, axs = plt.subplots(1, 3, figsize=(20, 5))\n",
    "last = 70\n",
    "\n",
    "for i, n_inner in enumerate(N_INNER_LIST):\n",
    "    axs[0].plot(losses_all_n[i], label=f\"{N_INNER_LIST[i]}\")\n",
    "    axs[1].plot(errors_all_n[i,:,0], label=f\"{N_INNER_LIST[i]}\")\n",
    "    axs[2].plot(errors_all_n[i,:,1], label=f\"{N_INNER_LIST[i]}\")\n",
    "\n",
    "axs[0].plot(loss_hist_prdp, label='PRDP', color='black')\n",
    "axs[1].plot(error_hist_prdp[:last, 0], label='PRDP', color='black')\n",
    "axs[2].plot(error_hist_prdp[:last, 1], label='PRDP', color='black')\n",
    "\n",
    "axs[0].set_xlabel('Epoch'); axs[0].set_ylabel('Loss (minibatch avg.)');\n",
    "axs[0].set_yscale('log'); axs[0].grid(); axs[0].set_ylim(1e-5, 1)\n",
    "\n",
    "axs[1].set_xlabel('Epoch'); axs[1].set_ylabel('1-step nMSE');\n",
    "axs[2].set_xlabel('Epoch'); axs[2].set_ylabel('2-step nMSE');\n",
    "axs[1].set_yscale('log'); axs[2].set_yscale('log');\n",
    "axs[1].grid(); axs[2].grid()\n",
    "axs[1].set_ylim(1e-3, 10); axs[2].set_ylim(1e-3, 10)\n",
    "\n",
    "plt.legend(bbox_to_anchor=(1.05, 1), loc='upper left', title=\"n_inner\")"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "jax_fresh",
   "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.14"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
