{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Import Libraries"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "cuda\n"
     ]
    }
   ],
   "source": [
    "import torch\n",
    "import torch.autograd as autograd       \n",
    "from torch import Tensor                  \n",
    "import torch.nn as nn                  \n",
    "import torch.optim as optim              \n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "import matplotlib.gridspec as gridspec\n",
    "from mpl_toolkits.axes_grid1 import make_axes_locatable\n",
    "from mpl_toolkits.mplot3d import Axes3D\n",
    "import matplotlib.ticker\n",
    "\n",
    "import numpy as np\n",
    "import time\n",
    "from pyDOE import lhs         #Latin Hypercube Sampling\n",
    "import scipy.io\n",
    "\n",
    "#Set default dtype to float32\n",
    "torch.set_default_dtype(torch.float)\n",
    "\n",
    "#PyTorch random number generator\n",
    "torch.manual_seed(1234)\n",
    "\n",
    "# Random number generators in other libraries\n",
    "np.random.seed(1234)\n",
    "\n",
    "# Device configuration\n",
    "device = torch.device('cuda' if torch.cuda.is_available() else 'cpu')\n",
    "\n",
    "print(device)\n",
    "\n",
    "if device == 'cuda': \n",
    "    print(torch.cuda.get_device_name()) "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# *Data Prep*\n",
    "\n",
    "Training and Testing data is prepared from the solution file"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "data = scipy.io.loadmat('Data/burgers_shock_mu_01_pi.mat')  \t# Load data from file\n",
    "x = data['x']                                 \n",
    "t = data['t']                              \n",
    "usol = data['usol']                           \n",
    "\n",
    "X, T = np.meshgrid(x,t)                         "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Test Data\n",
    "\n",
    "We prepare the test data to compare against the solution produced by the PINN."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "colab": {},
    "colab_type": "code",
    "id": "yddknKA2Xohp"
   },
   "outputs": [],
   "source": [
    "''' X_u_test = [X[i],T[i]] [25600,2] for interpolation'''\n",
    "X_u_test = np.hstack((X.flatten()[:,None], T.flatten()[:,None]))\n",
    "\n",
    "# Domain bounds\n",
    "lb = X_u_test[0]  # [-1. 0.]\n",
    "ub = X_u_test[-1] # [1.  0.99]\n",
    "\n",
    "u_true = usol.flatten('F')[:,None] "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Training Data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "colab": {},
    "colab_type": "code",
    "id": "8UVJmvZbXjXb"
   },
   "outputs": [],
   "source": [
    "def trainingdata(N_u,N_f):\n",
    "\n",
    "    '''Boundary Conditions'''\n",
    "\n",
    "    #Initial Condition -1 =< x =<1 and t = 0  \n",
    "    leftedge_x = np.hstack((X[0,:][:,None], T[0,:][:,None])) \n",
    "    leftedge_u = usol[:,0][:,None]\n",
    "\n",
    "    #Boundary Condition x = -1 and 0 =< t =<1\n",
    "    bottomedge_x = np.hstack((X[:,0][:,None], T[:,0][:,None])) \n",
    "    bottomedge_u = usol[-1,:][:,None]\n",
    "\n",
    "    #Boundary Condition x = 1 and 0 =< t =<1\n",
    "    topedge_x = np.hstack((X[:,-1][:,None], T[:,0][:,None])) \n",
    "    topedge_u = usol[0,:][:,None]\n",
    "\n",
    "    all_X_u_train = np.vstack([leftedge_x, bottomedge_x, topedge_x]) \n",
    "    all_u_train = np.vstack([leftedge_u, bottomedge_u, topedge_u])   \n",
    "\n",
    "    #choose random N_u points for training\n",
    "    idx = np.random.choice(all_X_u_train.shape[0], N_u, replace=False) \n",
    "\n",
    "    X_u_train = all_X_u_train[idx, :] #choose indices from  set 'idx' (x,t)\n",
    "    u_train = all_u_train[idx,:]      #choose corresponding u\n",
    "\n",
    "    '''Collocation Points'''\n",
    "\n",
    "    # Latin Hypercube sampling for collocation points \n",
    "    # N_f sets of tuples(x,t)\n",
    "    X_f_train = lb + (ub-lb)*lhs(2,N_f) \n",
    "    X_f_train = np.vstack((X_f_train, X_u_train)) # append training points to collocation points \n",
    "\n",
    "    return X_f_train, X_u_train, u_train \n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Physics Informed Neural Network"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "class Sequentialmodel(nn.Module):\n",
    "    \n",
    "    def __init__(self, layers):\n",
    "        super().__init__()\n",
    "    \n",
    "        \n",
    "        assert len(layers) == 3, \"Layers must follow [input_dim, m, output_dim] structure\"\n",
    "    \n",
    "        self.activation = nn.Tanh()\n",
    "    \n",
    "        self.loss_function = nn.MSELoss(reduction='mean')\n",
    "    \n",
    "        self.w_layer = nn.Linear(layers[0], layers[1], bias=False)\n",
    "    \n",
    "        self.a_layer = nn.Linear(layers[1], layers[2], bias=False)\n",
    "    \n",
    "        self.w_layer.weight.data.normal_(mean=0.0, std=1.0)  \n",
    "\n",
    "        self.a_layer.weight.data.uniform_(-1.0, 1.0)  \n",
    "    \n",
    "        self.iter = 0\n",
    "\n",
    "    def forward(self, x):\n",
    "        \n",
    "        x = self.w_layer(x)        \n",
    "        x = self.activation(x)      \n",
    "        x = self.a_layer(x)         \n",
    "        return x\n",
    "                        \n",
    "    def loss_BC(self,x,y):\n",
    "                \n",
    "        loss_u = self.loss_function(self.forward(x), y)\n",
    "                \n",
    "        return loss_u\n",
    "    \n",
    "    def loss_PDE(self, x_to_train_f):\n",
    "        \n",
    "        nu = 0.01/np.pi\n",
    "                \n",
    "        x_1_f = x_to_train_f[:,[0]]\n",
    "        x_2_f = x_to_train_f[:,[1]]\n",
    "                        \n",
    "        g = x_to_train_f.clone()\n",
    "                        \n",
    "        g.requires_grad = True\n",
    "        \n",
    "        u = self.forward(g)\n",
    "                \n",
    "        u_x_t = autograd.grad(u,g,torch.ones([x_to_train_f.shape[0], 1]).to(device), retain_graph=True, create_graph=True)[0]\n",
    "                                \n",
    "        u_xx_tt = autograd.grad(u_x_t,g,torch.ones(x_to_train_f.shape).to(device), create_graph=True)[0]\n",
    "                                                            \n",
    "        u_x = u_x_t[:,[0]]\n",
    "        \n",
    "        u_t = u_x_t[:,[1]]\n",
    "        \n",
    "        u_xx = u_xx_tt[:,[0]]\n",
    "                                        \n",
    "        f = u_t + (self.forward(g))*(u_x) - (nu)*u_xx \n",
    "        \n",
    "        loss_f = self.loss_function(f,f_hat)\n",
    "                \n",
    "        return loss_f\n",
    "    \n",
    "    def loss(self,x,y,x_to_train_f):\n",
    "\n",
    "        loss_u = self.loss_BC(x,y)\n",
    "        loss_f = self.loss_PDE(x_to_train_f)\n",
    "        \n",
    "        loss_val = loss_u + loss_f\n",
    "        \n",
    "        return loss_val\n",
    "     \n",
    "    'callable for optimizer'                                       \n",
    "    def closure(self):\n",
    "        \n",
    "        optimizer.zero_grad()\n",
    "        \n",
    "        loss = self.loss(X_u_train, u_train, X_f_train)\n",
    "        \n",
    "        loss.backward()\n",
    "                \n",
    "        self.iter += 1\n",
    "        \n",
    "        if self.iter % 100 == 0:\n",
    "\n",
    "            error_vec, _ = PINN.test()\n",
    "        \n",
    "            print(loss,error_vec)\n",
    "\n",
    "        return loss        \n",
    "    \n",
    "    'test neural network'\n",
    "    def test(self):\n",
    "                \n",
    "        u_pred = self.forward(X_u_test_tensor)\n",
    "        \n",
    "        error_vec = torch.linalg.norm((u-u_pred),2)/torch.linalg.norm(u,2)        # Relative L2 Norm of the error (Vector)\n",
    "        \n",
    "        u_pred = u_pred.cpu().detach().numpy()\n",
    "        \n",
    "        u_pred = np.reshape(u_pred,(256,100),order='F')\n",
    "                \n",
    "        return error_vec, u_pred"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "colab_type": "text",
    "id": "bOjuHdzAhib-"
   },
   "source": [
    "# *Solution Plot*"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {
    "colab": {},
    "colab_type": "code",
    "id": "UWqNuRMLhg4m"
   },
   "outputs": [],
   "source": [
    "def solutionplot(u_pred,X_u_train,u_train):\n",
    "    \n",
    "    fig, ax = plt.subplots()\n",
    "    ax.axis('off')\n",
    "\n",
    "    gs0 = gridspec.GridSpec(1, 2)\n",
    "    gs0.update(top=1-0.06, bottom=1-1/3, left=0.15, right=0.85, wspace=0)\n",
    "    ax = plt.subplot(gs0[:, :])\n",
    "\n",
    "    h = ax.imshow(u_pred, interpolation='nearest', cmap='rainbow', \n",
    "                extent=[T.min(), T.max(), X.min(), X.max()], \n",
    "                origin='lower', aspect='auto')\n",
    "    divider = make_axes_locatable(ax)\n",
    "    cax = divider.append_axes(\"right\", size=\"5%\", pad=0.05)\n",
    "    fig.colorbar(h, cax=cax)\n",
    "    \n",
    "    ax.plot(X_u_train[:,1], X_u_train[:,0], 'kx', label = 'Data (%d points)' % (u_train.shape[0]), markersize = 4, clip_on = False)\n",
    "\n",
    "    line = np.linspace(x.min(), x.max(), 2)[:,None]\n",
    "    ax.plot(t[25]*np.ones((2,1)), line, 'w-', linewidth = 1)\n",
    "    ax.plot(t[50]*np.ones((2,1)), line, 'w-', linewidth = 1)\n",
    "    ax.plot(t[75]*np.ones((2,1)), line, 'w-', linewidth = 1)    \n",
    "\n",
    "    ax.set_xlabel('$t$')\n",
    "    ax.set_ylabel('$x$')\n",
    "    ax.legend(frameon=False, loc = 'best')\n",
    "    ax.set_title('$u(x,t)$', fontsize = 10)\n",
    "    \n",
    "    ''' \n",
    "    Slices of the solution at points t = 0.25, t = 0.50 and t = 0.75\n",
    "    '''\n",
    "    \n",
    "    ####### Row 1: u(t,x) slices ##################\n",
    "    gs1 = gridspec.GridSpec(1, 3)\n",
    "    gs1.update(top=1-1/3, bottom=0, left=0.1, right=0.9, wspace=0.5)\n",
    "\n",
    "    ax = plt.subplot(gs1[0, 0])\n",
    "    ax.plot(x,usol.T[25,:], 'b-', linewidth = 2, label = 'Exact')       \n",
    "    ax.plot(x,u_pred.T[25,:], 'r--', linewidth = 2, label = 'Prediction')\n",
    "    ax.set_xlabel('$x$')\n",
    "    ax.set_ylabel('$u(x,t)$')    \n",
    "    ax.set_title('$t = 0.25s$', fontsize = 10)\n",
    "    ax.axis('square')\n",
    "    ax.set_xlim([-1.1,1.1])\n",
    "    ax.set_ylim([-1.1,1.1])\n",
    "\n",
    "    ax = plt.subplot(gs1[0, 1])\n",
    "    ax.plot(x,usol.T[50,:], 'b-', linewidth = 2, label = 'Exact')       \n",
    "    ax.plot(x,u_pred.T[50,:], 'r--', linewidth = 2, label = 'Prediction')\n",
    "    ax.set_xlabel('$x$')\n",
    "    ax.set_ylabel('$u(x,t)$')\n",
    "    ax.axis('square')\n",
    "    ax.set_xlim([-1.1,1.1])\n",
    "    ax.set_ylim([-1.1,1.1])\n",
    "    ax.set_title('$t = 0.50s$', fontsize = 10)\n",
    "    ax.legend(loc='upper center', bbox_to_anchor=(0.5, -0.35), ncol=5, frameon=False)\n",
    "\n",
    "    ax = plt.subplot(gs1[0, 2])\n",
    "    ax.plot(x,usol.T[75,:], 'b-', linewidth = 2, label = 'Exact')       \n",
    "    ax.plot(x,u_pred.T[75,:], 'r--', linewidth = 2, label = 'Prediction')\n",
    "    ax.set_xlabel('$x$')\n",
    "    ax.set_ylabel('$u(x,t)$')\n",
    "    ax.axis('square')\n",
    "    ax.set_xlim([-1.1,1.1])\n",
    "    ax.set_ylim([-1.1,1.1])    \n",
    "    ax.set_title('$t = 0.75s$', fontsize = 10)\n",
    "    \n",
    "    plt.savefig('Burgers.png',dpi = 500)   "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "a10ecc15",
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "\n",
    "def assign_params(model, param_list):\n",
    "    for p, src in zip(model.parameters(), param_list):\n",
    "        p.data.copy_(src.data)\n",
    "\n",
    "def IGD_PINNa(PINN, X_u_train, u_train, X_f_train, theta_init, K0, K1, gamma, eta):\n",
    "    for p in PINN.w_layer.parameters():\n",
    "        p.requires_grad_(False)\n",
    "    for p in PINN.a_layer.parameters():\n",
    "        p.requires_grad_(True)\n",
    "    theta_n = [p.clone() for p in theta_init]\n",
    "    n = 0\n",
    "    loss_list = []\n",
    "    while n < K0:\n",
    "        assign_params(PINN, theta_n)\n",
    "        \n",
    "        optimizer_inner = torch.optim.LBFGS(PINN.a_layer.parameters(), \n",
    "                                          lr=gamma,        \n",
    "                                          max_iter=K1,    \n",
    "                                          max_eval=K1*2,   \n",
    "                                          history_size=100)\n",
    "        \n",
    "        def closure():\n",
    "            optimizer_inner.zero_grad()\n",
    "            \n",
    "            param_now = list(PINN.parameters())\n",
    "            loss_quad = 0.5 * sum([(a - b).pow(2).sum() for a, b in zip(param_now, theta_n)])\n",
    "           \n",
    "            loss_main = PINN.loss(X_u_train, u_train, X_f_train)\n",
    "          \n",
    "            total_loss = loss_quad + eta * loss_main\n",
    "            total_loss.backward()\n",
    "            return total_loss\n",
    "        \n",
    "        optimizer_inner.step(closure)\n",
    "        theta_n = [p.clone().detach() for p in PINN.parameters()]\n",
    "        loss = PINN.loss(X_u_train, u_train, X_f_train)\n",
    "        \n",
    "        loss_current = loss.item()\n",
    "        loss_list.append(loss_current)\n",
    "        \n",
    "        if n % 100 == 0:\n",
    "            loss = PINN.loss(X_u_train, u_train, X_f_train)\n",
    "            print(f'Iteration {n}, loss = {loss.item():.8e}')\n",
    "           \n",
    "        n += 1\n",
    "\n",
    "    assign_params(PINN, theta_n)\n",
    "    return [p.clone().detach() for p in PINN.parameters()], loss_list"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Main"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sequentialmodel(\n",
      "  (activation): Tanh()\n",
      "  (loss_function): MSELoss()\n",
      "  (w_layer): Linear(in_features=2, out_features=100, bias=False)\n",
      "  (a_layer): Linear(in_features=100, out_features=1, bias=False)\n",
      ")\n",
      "Iteration 0, loss = 2.07160358e+01\n",
      "Iteration 100, loss = 1.55732751e-01\n",
      "Iteration 200, loss = 1.32878929e-01\n",
      "Iteration 300, loss = 1.25316307e-01\n",
      "Iteration 400, loss = 1.21633992e-01\n",
      "Iteration 500, loss = 1.19219244e-01\n",
      "Iteration 600, loss = 1.17373206e-01\n",
      "Iteration 700, loss = 1.15865052e-01\n",
      "Iteration 800, loss = 1.14587687e-01\n",
      "Iteration 900, loss = 1.13489047e-01\n",
      "Iteration 1000, loss = 1.12531900e-01\n",
      "Iteration 1100, loss = 1.11688808e-01\n",
      "Iteration 1200, loss = 1.10933304e-01\n",
      "Iteration 1300, loss = 1.10253692e-01\n",
      "Iteration 1400, loss = 1.09635316e-01\n",
      "Iteration 1500, loss = 1.09068461e-01\n",
      "Iteration 1600, loss = 1.08546838e-01\n",
      "Iteration 1700, loss = 1.08062856e-01\n",
      "Iteration 1800, loss = 1.07612446e-01\n",
      "Iteration 1900, loss = 1.07191354e-01\n",
      "Iteration 2000, loss = 1.06796399e-01\n",
      "Iteration 2100, loss = 1.06424689e-01\n",
      "Iteration 2200, loss = 1.06074005e-01\n",
      "Iteration 2300, loss = 1.05742753e-01\n",
      "Iteration 2400, loss = 1.05429344e-01\n",
      "Iteration 2500, loss = 1.05132177e-01\n",
      "Iteration 2600, loss = 1.04850307e-01\n",
      "Iteration 2700, loss = 1.04581363e-01\n",
      "Iteration 2800, loss = 1.04324266e-01\n",
      "Iteration 2900, loss = 1.04079492e-01\n",
      "Iteration 3000, loss = 1.03844032e-01\n",
      "Iteration 3100, loss = 1.03618249e-01\n",
      "Iteration 3200, loss = 1.03402495e-01\n",
      "Iteration 3300, loss = 1.03195459e-01\n",
      "Iteration 3400, loss = 1.02998026e-01\n",
      "Iteration 3500, loss = 1.02807283e-01\n",
      "Iteration 3600, loss = 1.02623746e-01\n",
      "Iteration 3700, loss = 1.02447189e-01\n",
      "Iteration 3800, loss = 1.02276921e-01\n",
      "Iteration 3900, loss = 1.02113545e-01\n",
      "Iteration 4000, loss = 1.01954952e-01\n",
      "Iteration 4100, loss = 1.01802044e-01\n",
      "Iteration 4200, loss = 1.01653904e-01\n",
      "Iteration 4300, loss = 1.01510882e-01\n",
      "Iteration 4400, loss = 1.01371944e-01\n",
      "Iteration 4500, loss = 1.01237744e-01\n",
      "Iteration 4600, loss = 1.01107627e-01\n",
      "Iteration 4700, loss = 1.00981116e-01\n",
      "Iteration 4800, loss = 1.00859016e-01\n",
      "Iteration 4900, loss = 1.00739568e-01\n",
      "Iteration 5000, loss = 1.00623541e-01\n",
      "Iteration 5100, loss = 1.00510173e-01\n",
      "Iteration 5200, loss = 1.00400105e-01\n",
      "Iteration 5300, loss = 1.00293428e-01\n",
      "Iteration 5400, loss = 1.00189321e-01\n",
      "Iteration 5500, loss = 1.00088328e-01\n",
      "Iteration 5600, loss = 9.99894738e-02\n",
      "Iteration 5700, loss = 9.98933911e-02\n",
      "Iteration 5800, loss = 9.97985452e-02\n",
      "Iteration 5900, loss = 9.97059494e-02\n",
      "Iteration 6000, loss = 9.96162444e-02\n",
      "Iteration 6100, loss = 9.95290726e-02\n",
      "Iteration 6200, loss = 9.94462818e-02\n",
      "Iteration 6300, loss = 9.93655473e-02\n",
      "Iteration 6400, loss = 9.92868394e-02\n",
      "Iteration 6500, loss = 9.92075354e-02\n",
      "Iteration 6600, loss = 9.91313159e-02\n",
      "Iteration 6700, loss = 9.90556404e-02\n",
      "Iteration 6800, loss = 9.89826173e-02\n",
      "Iteration 6900, loss = 9.89106447e-02\n",
      "Iteration 7000, loss = 9.88412052e-02\n",
      "Iteration 7100, loss = 9.87723917e-02\n",
      "Iteration 7200, loss = 9.87055600e-02\n",
      "Iteration 7300, loss = 9.86382961e-02\n",
      "Iteration 7400, loss = 9.85742509e-02\n",
      "Iteration 7500, loss = 9.85110253e-02\n",
      "Iteration 7600, loss = 9.84513015e-02\n",
      "Iteration 7700, loss = 9.83916968e-02\n",
      "Iteration 7800, loss = 9.83368158e-02\n",
      "Iteration 7900, loss = 9.82796252e-02\n",
      "Iteration 8000, loss = 9.82249826e-02\n",
      "Iteration 8100, loss = 9.81711447e-02\n",
      "Iteration 8200, loss = 9.81170014e-02\n",
      "Iteration 8300, loss = 9.80647355e-02\n",
      "Iteration 8400, loss = 9.80132520e-02\n",
      "Iteration 8500, loss = 9.79645327e-02\n",
      "Iteration 8600, loss = 9.79178101e-02\n",
      "Iteration 8700, loss = 9.78685021e-02\n",
      "Iteration 8800, loss = 9.78218541e-02\n",
      "Iteration 8900, loss = 9.77801681e-02\n",
      "Iteration 9000, loss = 9.77376774e-02\n",
      "Iteration 9100, loss = 9.76897627e-02\n",
      "Iteration 9200, loss = 9.76458937e-02\n",
      "Iteration 9300, loss = 9.76043344e-02\n",
      "Iteration 9400, loss = 9.75648463e-02\n",
      "Iteration 9500, loss = 9.75245982e-02\n",
      "Iteration 9600, loss = 9.74846482e-02\n",
      "Iteration 9700, loss = 9.74440724e-02\n",
      "Iteration 9800, loss = 9.74060744e-02\n",
      "Iteration 9900, loss = 9.73675847e-02\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAkIAAAHHCAYAAABTMjf2AAAAOnRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjEwLjEsIGh0dHBzOi8vbWF0cGxvdGxpYi5vcmcvc2/+5QAAAAlwSFlzAAAPYQAAD2EBqD+naQAARExJREFUeJzt3Xl8VNX9//H3zCSZJIQkrAlLICzaGERANlFQlNRI0QouoKVsKlbFFopL9WcFd1wqX6xG0VoBlVqRKlpEFNmsFlkFWSyoICCQAEJIQiDLzPn9EeYmQwIETHIC83o+Hnkkc+72uRckb889516XMcYIAAAgBLltFwAAAGALQQgAAIQsghAAAAhZBCEAABCyCEIAACBkEYQAAEDIIggBAICQRRACAAAhiyAEAABCFkEIQK0yY8YM1a9fX3l5edV2jEWLFsnlcmnRokXVdoyq5HK59NBDD9kuo5wffvhBLpdLU6dOtV3KcU2ePFktWrRQQUGB7VJQCxGEgCOmTp0ql8ulFStW2C6lUlavXq3f/va3SkpKktfrVf369ZWWlqYpU6bI5/PZLu+U+Hw+jR8/Xr///e8VExNju5xKmzNnTq0MKigxfPhwFRYW6uWXX7ZdCmohghBwGnr11VfVpUsXLVy4UIMHD9aLL76ocePGKSoqSjfffLOeeuop2yWekn//+9/auHGjbr31VtulnJQ5c+bo4Ycfrrb9Hzp0SH/+85+rbf9nusjISA0bNkwTJ04Ur9fE0cJsFwDg5Hz55Ze67bbb1KNHD82ZM0d169Z1lo0ZM0YrVqzQunXrquRYBw8eVJ06dapkX5UxZcoUXXTRRWrWrFmNHbOmFRcXy+/3KyIiotLbREZGVmNFZ66yf38HDhyop59+WgsXLtRll11muTLUJvQIASfpq6++Ut++fRUbG6uYmBj16dNHX375ZdA6RUVFevjhh3XWWWcpMjJSDRo0UM+ePTVv3jxnnczMTI0YMULNmzeX1+tVkyZNdPXVV+uHH3447vEffvhhuVwuTZ8+PSgEBXTp0kXDhw+XdOyxMBWN7Rg+fLhiYmL0/fff61e/+pXq1q2rwYMH684771RMTIzy8/PLHevGG29UYmJi0K24jz76SL169VKdOnVUt25d9evXT+vXrz/uOUnS4cOHNXfuXKWlpZVbVlxcrEcffVRt2rSR1+tVcnKy/t//+3/lxnwkJyfryiuv1Oeff65u3bopMjJSrVu31uuvv37cY48fP17h4eHas2dPuWW33nqr4uPjdfjw4Qq3HT58uDIyMiSVjOUJfEml1/kvf/mLJk2a5NS/YcMGFRYWaty4cercubPi4uJUp04d9erVSwsXLix3jKPHCD300ENyuVz67rvvNHz4cMXHxysuLk4jRoyo8M/paP/5z390/fXXq0WLFvJ6vUpKStIf//hHHTp06ITbnsjXX3+t4cOHq3Xr1oqMjFRiYqJuuukm/fTTT846CxculMvl0nvvvVdu+3/84x9yuVxasmSJ0/a///1P1113nerXr6/IyEh16dJFH3zwQdB2gVvbixcv1h133KHGjRurefPmzvLOnTurfv36ev/993/2OeLMQo8QcBLWr1+vXr16KTY2Vvfee6/Cw8P18ssvq3fv3lq8eLG6d+8uqeQX1YQJE3TLLbeoW7duysnJ0YoVK7Rq1Sr98pe/lCRde+21Wr9+vX7/+98rOTlZu3fv1rx587Rt2zYlJydXePz8/HzNnz9fF198sVq0aFHl51dcXKz09HT17NlTf/nLXxQdHa3k5GRlZGToww8/1PXXXx9Uy7///W8NHz5cHo9HkvTGG29o2LBhSk9P11NPPaX8/Hy99NJL6tmzp7766qtjnpckrVy5UoWFhTr//PPLLbvllls0bdo0XXfddbrrrru0dOlSTZgwQd988025X6bfffedrrvuOt18880aNmyYXnvtNQ0fPlydO3dWu3btKjz2kCFD9Mgjj+jtt9/WnXfe6bQXFhZq5syZuvbaa4/ZK/O73/1OO3fu1Lx58/TGG29UuM6UKVN0+PBh3Xrrrc54rpycHL366qu68cYbNXLkSOXm5urvf/+70tPTtWzZMnXs2PGY1ypg4MCBatWqlSZMmKBVq1bp1VdfVePGjU94a/Sdd95Rfn6+br/9djVo0EDLli3T888/rx9//FHvvPPOCY97PPPmzdPmzZs1YsQIJSYmav369XrllVe0fv16ffnll3K5XOrdu7eSkpI0ffp0DRgwIGj76dOnq02bNurRo4ekkv/mAr2E9913n+rUqaMZM2aof//++te//lVu+zvuuEONGjXSuHHjdPDgwaBl559/vr744oufdX44AxkAxhhjpkyZYiSZ5cuXH3Od/v37m4iICPP99987bTt37jR169Y1F198sdPWoUMH069fv2PuZ//+/UaSeeaZZ06qxjVr1hhJZvTo0ZVaf+HChUaSWbhwYVD7li1bjCQzZcoUp23YsGFGkrnvvvuC1vX7/aZZs2bm2muvDWqfMWOGkWQ+++wzY4wxubm5Jj4+3owcOTJovczMTBMXF1eu/WivvvqqkWTWrl0b1L569Wojydxyyy1B7XfffbeRZBYsWOC0tWzZMqgmY4zZvXu38Xq95q677jrudenRo4fp3r170DHefffdCq/f0UaNGmUq+uc0cJ1jY2PN7t27g5YVFxebgoKCoLb9+/ebhIQEc9NNNwW1SzLjx493Po8fP95IKrfegAEDTIMGDY5bqzHG5Ofnl2ubMGGCcblcZuvWrSfcPqCiv0cV7futt94q9+dy//33G6/Xa7Kzs5223bt3m7CwsKBz7dOnj2nfvr05fPiw0+b3+82FF15ozjrrLKct8N9vz549TXFxcYX13nrrrSYqKqrS54fQwK0xoJJ8Pp8++eQT9e/fX61bt3bamzRpot/85jf6/PPPlZOTI0mKj4/X+vXr9e2331a4r6ioKEVERGjRokXav39/pWsI7L+iW2JV5fbbbw/67HK5dP3112vOnDlBU9rffvttNWvWTD179pRU0hOQnZ2tG2+8UXv37nW+PB6PunfvXuEtn7ICt07q1asX1D5nzhxJ0tixY4Pa77rrLknShx9+GNSempqqXr16OZ8bNWqkX/ziF9q8efNxjz906FAtXbpU33//vdM2ffp0JSUl6ZJLLjnutidy7bXXqlGjRkFtHo/HGSfk9/u1b98+FRcXq0uXLlq1alWl9nvbbbcFfe7Vq5d++ukn5+/JsURFRTk/Hzx4UHv37tWFF14oY4y++uqrSh27Mvs+fPiw9u7dqwsuuECSgs5r6NChKigo0MyZM522t99+W8XFxfrtb38rSdq3b58WLFiggQMHKjc31/k79dNPPyk9PV3ffvutduzYEXT8kSNHOj2UR6tXr54OHTpUqduHCB0EIaCS9uzZo/z8fP3iF78ot+ycc86R3+/X9u3bJUmPPPKIsrOzdfbZZ6t9+/a655579PXXXzvre71ePfXUU/roo4+UkJCgiy++WE8//bQyMzOPW0NsbKwkKTc3twrPrFRYWFjQuIqAQYMG6dChQ864jLy8PM2ZM0fXX3+9Mx4mEPouu+wyNWrUKOjrk08+0e7duytVgzlqVs/WrVvldrvVtm3boPbExETFx8dr69atQe0V3TKsV6/eCQPnoEGD5PV6NX36dEnSgQMHNHv2bA0ePNg5x1PVqlWrCtunTZum8847zxlH1qhRI3344Yc6cOBApfZ79LkGQuSJznXbtm0aPny46tevr5iYGDVq1MgJe5U99rHs27dPo0ePVkJCgqKiotSoUSPn/MvuOyUlRV27dnWut1QSPC+44ALnz/q7776TMUYPPvhgub9T48ePl6Ryf6+Oda2l0r9bP/fPE2cWxggB1eDiiy/W999/r/fff1+ffPKJXn31Vf3f//2fJk+erFtuuUVSyQyvq666SrNmzdLHH3+sBx98UBMmTNCCBQvUqVOnCvfbtm1bhYWFae3atZWq41j/4B/rOUNer1dud/n/P7rggguUnJysGTNm6De/+Y3+/e9/69ChQxo0aJCzjt/vl1QyTigxMbHcPsLCjv/PTYMGDSSV/BKvKIxV9pfXsXoDjg5YR6tXr56uvPJKTZ8+XePGjdPMmTNVUFDg9E78HGV7SQLefPNNDR8+XP3799c999yjxo0by+PxaMKECUG9UsdzKufq8/n0y1/+Uvv27dOf/vQnpaSkqE6dOtqxY4eGDx/u/DmeqoEDB+q///2v7rnnHnXs2FExMTHy+/264ooryu176NChGj16tH788UcVFBToyy+/1AsvvOAsD6x/9913Kz09vcLjHR2QK7rWAfv371d0dPRx10HoIQgBldSoUSNFR0dr48aN5Zb973//k9vtVlJSktNWv359jRgxQiNGjFBeXp4uvvhiPfTQQ04QkqQ2bdrorrvu0l133aVvv/1WHTt21LPPPqs333yzwhqio6N12WWXacGCBdq+fXvQ8SoS6CHIzs4Oaj+6F6UyBg4cqOeee045OTl6++23lZyc7NzyCJyLJDVu3LjCmV8nkpKSIknasmWL2rdv77S3bNlSfr9f3377rc455xynPSsrS9nZ2WrZsuVJH+tYhg4dqquvvlrLly/X9OnT1alTp2MOsC7rVHoYZs6cqdatW+vdd98N2j7Q01Fd1q5dq02bNmnatGkaOnSo0152RuOp2r9/v+bPn6+HH35Y48aNc9qPdYv4hhtu0NixY/XWW2/p0KFDCg8PDwrXgVvQ4eHhp/R36mhbtmwJ+jsESNwaAyrN4/Ho8ssv1/vvvx80xT0rK0v/+Mc/1LNnT+fWVdmpwpIUExOjtm3bOtO98/Pzy03HbtOmjerWrXvC1wCMHz9exhgNGTKkwtdQrFy5UtOmTZNUEiI8Ho8+++yzoHVefPHFyp10GYMGDVJBQYGmTZumuXPnauDAgUHL09PTFRsbqyeeeEJFRUXltq9oanpZnTt3VkRERLkne//qV7+SJE2aNCmofeLEiZKkfv36neypHFPfvn3VsGFDPfXUU1q8eHGle4MCz6o5OnAeT6A3p2zvzdKlS4OmjVeHio5rjNFzzz1XLfuWyv/ZBTRs2FB9+/bVm2++qenTp+uKK65Qw4YNneWNGzdW79699fLLL2vXrl3ltj/R36mjrVq1ShdeeOFJbYMzHz1CwFFee+01zZ07t1z76NGj9dhjj2nevHnq2bOn7rjjDoWFhenll19WQUGBnn76aWfd1NRU9e7d23l2yYoVKzRz5kxnavamTZvUp08fDRw4UKmpqQoLC9N7772nrKws3XDDDcet78ILL1RGRobuuOMOpaSkaMiQITrrrLOUm5urRYsW6YMPPtBjjz0mSYqLi9P111+v559/Xi6XS23atNHs2bMrPV6nrPPPP19t27bVAw88oIKCgqD/c5dKxi+99NJLGjJkiM4//3zdcMMNatSokbZt26YPP/xQF110UdBtj6NFRkbq8ssv16effqpHHnnEae/QoYOGDRumV155RdnZ2brkkku0bNkyTZs2Tf3799ell1560udyLOHh4brhhhv0wgsvyOPx6MYbb6zUdp07d5Yk/eEPf1B6ero8Hs8J/xyvvPJKvfvuuxowYID69eunLVu2aPLkyUpNTa3W96ylpKSoTZs2uvvuu7Vjxw7FxsbqX//610kN2j+W2NhYZ7xbUVGRmjVrpk8++URbtmw55jZDhw7VddddJ0l69NFHyy3PyMhQz5491b59e40cOVKtW7dWVlaWlixZoh9//FFr1qypVG0rV67Uvn37dPXVV5/ayeHMZWm2GlDrBKbfHutr+/btxhhjVq1aZdLT001MTIyJjo42l156qfnvf/8btK/HHnvMdOvWzcTHx5uoqCiTkpJiHn/8cVNYWGiMMWbv3r1m1KhRJiUlxdSpU8fExcWZ7t27mxkzZlS63pUrV5rf/OY3pmnTpiY8PNzUq1fP9OnTx0ybNs34fD5nvT179phrr73WREdH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NcusWbPG/PrXvzatWrUyhw4dcta54oorTIcOHcyXX35p/vOf/5i2bduaG2+80Vl+4MABk5CQYAYPHmzWrVtn3nrrLRMVFWVefvnlGj1fWx5//HHToEEDM3v2bLNlyxbzzjvvmJiYGPPcc88563CdT82cOXPMAw88YN59910jybz33ntBy2vqun7xxRfG4/GYp59+2mzYsMH8+c9/NuHh4Wbt2rUndT4EIQu6detmRo0a5Xz2+XymadOmZsKECRarOn3s3r3bSDKLFy82xhiTnZ1twsPDzTvvvOOs88033xhJZsmSJcaYkv9w3W63yczMdNZ56aWXTGxsrCkoKDDGGHPvvfeadu3aBR1r0KBBJj09vbpPqVbJzc01Z511lpk3b5655JJLnCDEda4af/rTn0zPnj2Pudzv95vExETzzDPPOG3Z2dnG6/Wat956yxhjzIYNG4wks3z5cmedjz76yLhcLrNjxw5jjDEvvviiqVevnnPdA8f+xS9+UdWnVCv169fP3HTTTUFt11xzjRk8eLAxhutcVY4OQjV5XQcOHGj69esXVE/37t3N7373u5M6B26N1bDCwkKtXLlSaWlpTpvb7VZaWpqWLFlisbLTx4EDByRJ9evXlyStXLlSRUVFQdc0JSVFLVq0cK7pkiVL1L59eyUkJDjrpKenKycnR+vXr3fWKbuPwDqh9ucyatQo9evXr9y14DpXjQ8++EBdunTR9ddfr8aNG6tTp07629/+5izfsmWLMjMzg65RXFycunfvHnSd4+Pj1aVLF2edtLQ0ud1uLV261Fnn4osvVkREhLNOenq6Nm7cqP3791f3aVp34YUXav78+dq0aZMkac2aNfr888/Vt29fSVzn6lKT17Wq/i0hCNWwvXv3yufzBf2ikKSEhARlZmZaqur04ff7NWbMGF100UU699xzJUmZmZmKiIhQfHx80Lplr2lmZmaF1zyw7Hjr5OTk6NChQ9VxOrXOP//5T61atUoTJkwot4zrXDU2b96sl156SWeddZY+/vhj3X777frDH/6gadOmSSq9Tsf7NyIzM1ONGzcOWh4WFqb69euf1J/Fmey+++7TDTfcoJSUFIWHh6tTp04aM2aMBg8eLInrXF1q8roea52Tve68fR6nlVGjRmndunX6/PPPbZdyxtm+fbtGjx6tefPmKTIy0nY5Zyy/368uXbroiSeekCR16tRJ69at0+TJkzVs2DDL1Z05ZsyYoenTp+sf//iH2rVrp9WrV2vMmDFq2rQp1xlB6BGqYQ0bNpTH4yk30yYrK0uJiYmWqjo93HnnnZo9e7YWLlyo5s2bO+2JiYkqLCxUdnZ20Pplr2liYmKF1zyw7HjrxMbGKioqqqpPp9ZZuXKldu/erfPPP19hYWEKCwvT4sWL9de//lVhYWFKSEjgOleBJk2aKDU1NajtnHPO0bZt2ySVXqfj/RuRmJio3bt3By0vLi7Wvn37TurP4kx2zz33OL1C7du315AhQ/THP/7R6e3kOlePmryux1rnZK87QaiGRUREqHPnzpo/f77T5vf7NX/+fPXo0cNiZbWXMUZ33nmn3nvvPS1YsECtWrUKWt65c2eFh4cHXdONGzdq27ZtzjXt0aOH1q5dG/Qf37x58xQbG+v8UurRo0fQPgLrhMqfS58+fbR27VqtXr3a+erSpYsGDx7s/Mx1/vkuuuiico9/2LRpk1q2bClJatWqlRITE4OuUU5OjpYuXRp0nbOzs7Vy5UpnnQULFsjv96t79+7OOp999pmKioqcdebNm6df/OIXqlevXrWdX22Rn58vtzv4V5zH45Hf75fEda4uNXldq+zfkpMaWo0q8c9//tN4vV4zdepUs2HDBnPrrbea+Pj4oJk2KHX77bebuLg4s2jRIrNr1y7nKz8/31nntttuMy1atDALFiwwK1asMD169DA9evRwlgemdV9++eVm9erVZu7cuaZRo0YVTuu+5557zDfffGMyMjJCalp3RcrOGjOG61wVli1bZsLCwszjjz9uvv32WzN9+nQTHR1t3nzzTWedJ5980sTHx5v333/ffP311+bqq6+ucPpxp06dzNKlS83nn39uzjrrrKDpx9nZ2SYhIcEMGTLErFu3zvzzn/800dHRZ/S07rKGDRtmmjVr5kyff/fdd03Dhg3Nvffe66zDdT41ubm55quvvjJfffWVkWQmTpxovvrqK7N161ZjTM1d1y+++MKEhYWZv/zlL+abb74x48ePZ/r86eT55583LVq0MBEREaZbt27myy+/tF1SrSWpwq8pU6Y46xw6dMjccccdpl69eiY6OtoMGDDA7Nq1K2g/P/zwg+nbt6+JiooyDRs2NHfddZcpKioKWmfhwoWmY8eOJiIiwrRu3TroGKHo6CDEda4a//73v825555rvF6vSUlJMa+88krQcr/fbx588EGTkJBgvF6v6dOnj9m4cWPQOj/99JO58cYbTUxMjImNjTUjRowwubm5QeusWbPG9OzZ03i9XtOsWTPz5JNPVvu51RY5OTlm9OjRpkWLFiYyMtK0bt3aPPDAA0HTsbnOp2bhwoUV/ps8bNgwY0zNXtcZM2aYs88+20RERJh27dqZDz/88KTPx2VMmcdsAgAAhBDGCAEAgJBFEAIAACGLIAQAAEIWQQgAAIQsghAAAAhZBCEAABCyCEIAACBkEYQA4DiSk5M1adIk22UAqCYEIQC1xvDhw9W/f39JUu/evTVmzJgaO/bUqVMVHx9frn358uW69dZba6wOADUrzHYBAFCdCgsLFRERccrbN2rUqAqrAVDb0CMEoNYZPny4Fi9erOeee04ul0sul0s//PCDJGndunXq27evYmJilJCQoCFDhmjv3r3Otr1799add96pMWPGqGHDhkpPT5ckTZw4Ue3bt1edOnWUlJSkO+64Q3l5eZKkRYsWacSIETpw4IBzvIceekhS+Vtj27Zt09VXX62YmBjFxsZq4MCBysrKcpY/9NBD6tixo9544w0lJycrLi5ON9xwg3Jzc6v3ogE4JQQhALXOc889px49emjkyJHatWuXdu3apaSkJGVnZ+uyyy5Tp06dtGLFCs2dO1dZWVkaOHBg0PbTpk1TRESEvvjiC02ePFmS5Ha79de//lXr16/XtGnTtGDBAt17772SpAsvvFCTJk1SbGysc7y77767XF1+v19XX3219u3bp8WLF2vevHnavHmzBg0aFLTe999/r1mzZmn27NmaPXu2Fi9erCeffLKarhaAn4NbYwBqnbi4OEVERCg6OlqJiYlO+wsvvKBOnTrpiSeecNpee+01JSUladOmTTr77LMlSWeddZaefvrpoH2WHW+UnJysxx57TLfddptefPFFRUREKC4uTi6XK+h4R5s/f77Wrl2rLVu2KCkpSZL0+uuvq127dlq+fLm6du0qqSQwTZ06VXXr1pUkDRkyRPPnz9fjjz/+8y4MgCpHjxCA08aaNWu0cOFCxcTEOF8pKSmSSnphAjp37lxu208//VR9+vRRs2bNVLduXQ0ZMkQ//fST8vPzK338b775RklJSU4IkqTU1FTFx8frm2++cdqSk5OdECRJTZo00e7du0/qXAHUDHqEAJw28vLydNVVV+mpp54qt6xJkybOz3Xq1Ala9sMPP+jKK6/U7bffrscff1z169fX559/rptvvlmFhYWKjo6u0jrDw8ODPrtcLvn9/io9BoCqQRACUCtFRETI5/MFtZ1//vn617/+peTkZIWFVf6fr5UrV8rv9+vZZ5+V213SET5jxowTHu9o55xzjrZv367t27c7vUIbNmxQdna2UlNTK10PgNqDW2MAaqXk5GQtXbpUP/zwg/bu3Su/369Ro0Zp3759uvHGG7V8+XJ9//33+vjjjzVixIjjhpi2bduqqKhIzz//vDZv3qw33njDGURd9nh5eXmaP3++9u7dW+Ets7S0NLVv316DBw/WqlWrtGzZMg0dOlSXXHKJunTpUuXXAED1IwgBqJXuvvtueTwepaamqlGjRtq2bZuaNm2qL774Qj6fT5dffrnat2+vMWPGKD4+3unpqUiHDh00ceJEPfXUUzr33HM1ffp0TZgwIWidCy+8ULfddpsGDRqkRo0alRtsLZXc4nr//fdVr149XXzxxUpLS1Pr1q319ttvV/n5A6gZLmOMsV0EAACADfQIAQCAkEUQAgAAIYsgBAAAQhZBCAAAhCyCEAAACFkEIQAAELIIQgAAIGQRhAAAQMgiCAEAgJBFEAIAACGLIAQAAEIWQQgAAISs/w9oYhUUE3k2sgAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test Error: 0.56854\n"
     ]
    }
   ],
   "source": [
    "'Generate Training data'\n",
    "N_u = 100 #Total number of data points for 'u'\n",
    "N_f = 10_000 #Total number of collocation points \n",
    "X_f_train_np_array, X_u_train_np_array, u_train_np_array = trainingdata(N_u,N_f)\n",
    "\n",
    "'Convert to tensor and send to GPU'\n",
    "X_f_train = torch.from_numpy(X_f_train_np_array).float().to(device)\n",
    "X_u_train = torch.from_numpy(X_u_train_np_array).float().to(device)\n",
    "u_train = torch.from_numpy(u_train_np_array).float().to(device)\n",
    "X_u_test_tensor = torch.from_numpy(X_u_test).float().to(device)\n",
    "u = torch.from_numpy(u_true).float().to(device)\n",
    "f_hat = torch.zeros(X_f_train.shape[0],1).to(device)\n",
    "\n",
    "layers = np.array([2,100,1])\n",
    "\n",
    "PINN = Sequentialmodel(layers)\n",
    "       \n",
    "PINN.to(device)\n",
    "\n",
    "'Neural Network Summary'\n",
    "print(PINN)\n",
    "\n",
    "params = list(PINN.parameters())\n",
    "\n",
    "'''Optimization'''\n",
    "\n",
    "theta_init = [p.data.clone() for p in PINN.parameters()]\n",
    "K0 = 10000       \n",
    "K1 = 20       \n",
    "gamma = 1e-2 \n",
    "eta = 0.5   \n",
    "\n",
    "final_params, loss_list = IGD_PINNa(PINN, X_u_train, u_train, X_f_train, theta_init, K0, K1, gamma, eta)\n",
    "\n",
    "# 绘图\n",
    "import matplotlib.pyplot as plt\n",
    "plt.plot(loss_list)\n",
    "plt.yscale('log')\n",
    "plt.xlabel('Iteration')\n",
    "plt.ylabel('Loss')\n",
    "plt.title('Loss Curve (only train a_layer)')\n",
    "plt.show()\n",
    "''' Model Accuracy ''' \n",
    "error_vec, u_pred = PINN.test()\n",
    "\n",
    "print('Test Error: %.5f'  % (error_vec))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "dde46591",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Gradient Norm: 2.35394156e-03\n"
     ]
    }
   ],
   "source": [
    "PINN.zero_grad()\n",
    "loss = PINN.loss(X_u_train, u_train, X_f_train)\n",
    "loss.backward()\n",
    "grad_list = [p.grad.view(-1) for p in PINN.parameters() if p.grad is not None]\n",
    "if grad_list:\n",
    "    grad_norm = torch.norm(torch.cat(grad_list))\n",
    "    print(f'Gradient Norm: {grad_norm.item():.8e}')\n",
    "else:\n",
    "    print('No gradients found.')"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "python_31015",
   "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.15"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
