{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Import Libraries"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "cuda\n"
     ]
    }
   ],
   "source": [
    "import numpy as np\n",
    "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",
    "import time\n",
    "from pyDOE import lhs  \n",
    "import matplotlib.pyplot as plt\n",
    "import matplotlib.ticker\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': 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": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "x_1 = np.linspace(-1, 1, 256)   \n",
    "x_2 = np.linspace(1, -1, 256)   \n",
    "t = np.linspace(0, 1, 100)      \n",
    "\n",
    "X, Y, T = np.meshgrid(x_1, x_2, t, indexing='ij')  \n"
   ]
  },
  {
   "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": 3,
   "id": "cb125bf4",
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "X_u_test = np.hstack((\n",
    "    X.flatten(order='F')[:, None],\n",
    "    Y.flatten(order='F')[:, None],\n",
    "    T.flatten(order='F')[:, None]\n",
    "))\n",
    "\n",
    "lb = np.array([-1, -1, 0])  \n",
    "ub = np.array([1, 1, 1])    \n",
    "\n",
    "usol = np.exp(-X**2-Y**2-T)\n",
    "\n",
    "u_true = usol.flatten('F')[:, None]"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Training Data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "12930ed0",
   "metadata": {},
   "outputs": [],
   "source": [
    "def trainingdata(N_u, N_f):\n",
    "   \n",
    "    # x = -1 \n",
    "    leftedge_x = np.hstack((\n",
    "        X[0, :, :].flatten()[:, None],   # x=-1\n",
    "        Y[0, :, :].flatten()[:, None],\n",
    "        T[0, :, :].flatten()[:, None]\n",
    "    ))\n",
    "    leftedge_u = usol[0, :, :].flatten()[:, None]\n",
    "\n",
    "    # x = 1 \n",
    "    rightedge_x = np.hstack((\n",
    "        X[-1, :, :].flatten()[:, None],  # x=1\n",
    "        Y[-1, :, :].flatten()[:, None],\n",
    "        T[-1, :, :].flatten()[:, None]\n",
    "    ))\n",
    "    rightedge_u = usol[-1, :, :].flatten()[:, None]\n",
    "\n",
    "    # y = 1 \n",
    "    topedge_x = np.hstack((\n",
    "        X[:, 0, :].flatten()[:, None],   # y=1\n",
    "        Y[:, 0, :].flatten()[:, None],\n",
    "        T[:, 0, :].flatten()[:, None]\n",
    "    ))\n",
    "    topedge_u = usol[:, 0, :].flatten()[:, None]\n",
    "\n",
    "    # y = -1 \n",
    "    bottomedge_x = np.hstack((\n",
    "        X[:, -1, :].flatten()[:, None],  # y=-1\n",
    "        Y[:, -1, :].flatten()[:, None],\n",
    "        T[:, -1, :].flatten()[:, None]\n",
    "    ))\n",
    "    bottomedge_u = usol[:, -1, :].flatten()[:, None]\n",
    "\n",
    "    # t = 0 \n",
    "    initial_x = np.hstack((\n",
    "        X[:, :, 0].flatten()[:, None],\n",
    "        Y[:, :, 0].flatten()[:, None],\n",
    "        T[:, :, 0].flatten()[:, None]\n",
    "    ))\n",
    "    initial_u = usol[:, :, 0].flatten()[:, None]\n",
    "\n",
    "    all_X_u_train = np.vstack([leftedge_x, rightedge_x, bottomedge_x, topedge_x, initial_x])\n",
    "    all_u_train = np.vstack([leftedge_u, rightedge_u, bottomedge_u, topedge_u, initial_u])\n",
    "\n",
    "    idx = np.random.choice(all_X_u_train.shape[0], N_u, replace=False)\n",
    "    X_u_train = all_X_u_train[idx, :]\n",
    "    u_train = all_u_train[idx, :]\n",
    "\n",
    "    '''Collocation Points'''\n",
    "    X_f = lb + (ub - lb) * lhs(3, N_f)\n",
    "    X_f_train = np.vstack((X_f, X_u_train))  \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": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "class Sequentialmodel(nn.Module):\n",
    "    \n",
    "    def __init__(self, layers):\n",
    "        super().__init__()\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.m = layers[1]\n",
    "        self.scale = 1 / torch.sqrt(torch.tensor(self.m, dtype=torch.float))\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",
    "        x = x * self.scale        \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 calc_source(self, x, y, t):\n",
    "       \n",
    "        u = torch.exp(- x**2 - y**2 - t)\n",
    "\n",
    "        S = -4*(x**2 + y**2)*u + 2*u + u**2\n",
    "        return S\n",
    "\n",
    "    def loss_PDE(self, x_to_train_f):\n",
    "        x_1_f = x_to_train_f[:, [0]]   # x\n",
    "        x_2_f = x_to_train_f[:, [1]]   # y\n",
    "        t_f   = x_to_train_f[:, [2]]   # t\n",
    "\n",
    "        g = x_to_train_f.clone()\n",
    "        g.requires_grad = True\n",
    "\n",
    "        u = self.forward(g)           \n",
    "\n",
    "        u_t = autograd.grad(u, g, torch.ones_like(u), retain_graph=True, create_graph=True)[0][:, [2]]\n",
    "\n",
    "        u_x_y = autograd.grad(u, g, torch.ones_like(u), retain_graph=True, create_graph=True)[0]\n",
    "        u_x = u_x_y[:, [0]]\n",
    "        u_y = u_x_y[:, [1]]\n",
    "\n",
    "        u_xx = autograd.grad(u_x, g, torch.ones_like(u_x), retain_graph=True, create_graph=True)[0][:, [0]]\n",
    "        u_yy = autograd.grad(u_y, g, torch.ones_like(u_y), retain_graph=True, create_graph=True)[0][:, [1]]\n",
    "\n",
    "        S = self.calc_source(x_1_f, x_2_f, t_f)   \n",
    "\n",
    "        pde_f = u_t - (u_xx + u_yy) - u* (1-u) - S\n",
    "\n",
    "        # loss\n",
    "        loss_f = self.loss_function(pde_f, torch.zeros_like(pde_f))\n",
    "\n",
    "        return pde_f, 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",
    "    \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 = np.reshape(u_pred.cpu().detach().numpy(),(256,256,100),order='F') \n",
    "        \n",
    "        return error_vec, u_pred"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b5f94d60",
   "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",
    "\n",
    "def IGD_PINN(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",
    "    n = 0\n",
    "    loss_list = []\n",
    "\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 % 1000 == 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": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sequentialmodel(\n",
      "  (activation): Tanh()\n",
      "  (loss_function): MSELoss()\n",
      "  (w_layer): Linear(in_features=3, out_features=1000, bias=False)\n",
      "  (a_layer): Linear(in_features=1000, out_features=1, bias=False)\n",
      ")\n",
      "Iteration 0, loss : 7.73490548e-01\n",
      "Iteration 10000, loss : 2.26191849e-01\n",
      "Iteration 20000, loss : 2.19932020e-01\n",
      "Iteration 30000, loss : 2.15886414e-01\n",
      "Iteration 40000, loss : 2.13067695e-01\n",
      "Iteration 50000, loss : 2.11005479e-01\n",
      "Iteration 60000, loss : 2.09423110e-01\n",
      "Iteration 70000, loss : 2.08127484e-01\n",
      "Iteration 80000, loss : 2.07076341e-01\n",
      "Iteration 90000, loss : 2.06154853e-01\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAl0AAAHHCAYAAACFl+2TAAAAOnRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjEwLjEsIGh0dHBzOi8vbWF0cGxvdGxpYi5vcmcvc2/+5QAAAAlwSFlzAAAPYQAAD2EBqD+naQAASdFJREFUeJzt3Xl8VNX9//H3LJnsGyEkhC0gKAQw7BRlM6SmlKJo64oS+fYrVWIFqbX257fq17phqw9cKC5VUNSC+sUdtSEsFoqyoxALsgkKSQiQnaxzfn8kGTImQAhhZpi8no/HPDJz7pl7P/cGzftx7pkzFmOMEQAAAM4pq7cLAAAAaAsIXQAAAB5A6AIAAPAAQhcAAIAHELoAAAA8gNAFAADgAYQuAAAADyB0AQAAeAChCwAAwAMIXQC8avr06frpT396To/x4IMPymKxnNNjtJaVK1fKYrFo5cqV3i6lkQULFshisWjfvn3eLuWUrr/+el177bXeLgNohNAF+Jn6P4wbNmzwdimntXfvXv3973/X//t//8/bpZyRv/3tb1qwYIG3y8BJ/OEPf9D//d//aevWrd4uBXBD6ALgNU8//bS6d++uyy67zNulnJFzGbpGjx6t48ePa/To0edk/23BwIEDNWTIED355JPeLgVwQ+gC4BVVVVV64403/P42UGlp6Rn1t1qtCgoKktXK/57PVMNrfe2112rJkiUqKSnxYkWAO/6rBtqozZs3a/z48YqIiFBYWJjGjRunL774wq1PVVWV/vd//1e9evVSUFCQYmJiNHLkSGVmZrr65OTkaOrUqercubMCAwPVsWNHXXnllaed97N69Wrl5+crNTW10ba8vDz9+te/VlxcnIKCgpScnKxXX33Vrc++fftksVj017/+VS+++KIuuOACBQYGaujQoVq/fv0pjz1mzBglJyc3ue2iiy5SWlraSd+bmJio7du3a9WqVbJYLLJYLBo7dqykE7d2V61apenTp6tDhw7q3LmzJOm7777T9OnTddFFFyk4OFgxMTG65pprGl2npuZ0jR07Vv369VN2drYuu+wyhYSEqFOnTnriiSdOeZ715s+fr5SUFHXo0EGBgYFKSkrSvHnzmvXe03n//fc1YcIEJSQkKDAwUBdccIH+/Oc/q6amxtXngQceUEBAgA4fPtzo/dOmTVNUVJTKy8tdbZ988olGjRql0NBQhYeHa8KECdq+fbvb+2655RaFhYVp9+7d+vnPf67w8HBNnjzZtf2nP/2pSktL3f6tAt5m93YBADxv+/btGjVqlCIiInTPPfcoICBAL7zwgsaOHatVq1Zp+PDhkmonoD/22GP67//+bw0bNkxFRUXasGGDNm3a5Jr8/stf/lLbt2/Xb3/7WyUmJiovL0+ZmZnav3+/EhMTT1rDv//9b1ksFg0cONCt/fjx4xo7dqx27dqlO+64Q927d9fbb7+tW265RQUFBZoxY4Zb/zfffFPFxcX6zW9+I4vFoieeeEJXX3219uzZo4CAgCaPffPNN+vWW2/Vtm3b1K9fP1f7+vXrtXPnTv3P//zPSeueM2eOfvvb3yosLEz33XefJCkuLs6tz/Tp0xUbG6v777/fNfqyfv16/fvf/9b111+vzp07a9++fZo3b57Gjh2r7OxshYSEnPSYknTs2DH97Gc/09VXX61rr71W77zzjv7whz+of//+Gj9+/CnfO2/ePPXt21dXXHGF7Ha7PvzwQ02fPl1Op1MZGRmnfO/pLFiwQGFhYZo1a5bCwsK0fPly3X///SoqKtJf/vIXSbXX+6GHHtLixYt1xx13uN5bWVmpd955R7/85S8VFBQkSVq4cKHS09OVlpam2bNnq6ysTPPmzdPIkSO1efNmt39T1dXVSktL08iRI/XXv/7V7RomJSUpODhYa9as0VVXXXVW5wi0GgPAr8yfP99IMuvXrz9pn0mTJhmHw2F2797tajt48KAJDw83o0ePdrUlJyebCRMmnHQ/x44dM5LMX/7ylzOu86abbjIxMTGN2ufMmWMkmddff93VVllZaUaMGGHCwsJMUVGRMcaYvXv3GkkmJibGHD161NX3/fffN5LMhx9+6Gp74IEHTMP/3RUUFJigoCDzhz/8we3Yd955pwkNDTUlJSWnrL1v375mzJgxjdrrr/3IkSNNdXW127aysrJG/deuXWskmddee83VtmLFCiPJrFixwtU2ZsyYRv0qKipMfHy8+eUvf3nKWk927LS0NNOjR4/Tvreh+vPbu3fvKff9m9/8xoSEhJjy8nJX24gRI8zw4cPd+i1ZssTtXIuLi01UVJS59dZb3frl5OSYyMhIt/b09HQjydx7770nrffCCy8048ePP5NTBM4pbi8CbUxNTY3++c9/atKkSerRo4ervWPH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scc0dOhQhYeHq0OHDpo0aZJ27Njh1qe8vFwZGRmKiYlRWFiYfvnLXyo3N9etz/79+zVhwgSFhISoQ4cO+v3vf6/q6mq3PitXrtSgQYMUGBionj17asGCBY3qaSu/q8cff1wWi0UzZ850tXGdW8cPP/ygm266STExMQoODlb//v21YcMG13ZjjO6//3517NhRwcHBSk1N1bfffuu2j6NHj2ry5MmKiIhQVFSUfv3rX6ukpMStz1dffaVRo0YpKChIXbp00RNPPNGolrffflu9e/dWUFCQ+vfvr6VLl56bk/awmpoa/elPf1L37t0VHBysCy64QH/+85/dvneP69wyn3/+uSZOnKiEhARZLBa99957btt96bo2p5bTMvArixYtMg6Hw7zyyitm+/bt5tZbbzVRUVEmNzfX26X5hLS0NDN//nyzbds2s2XLFvPzn//cdO3a1ZSUlLj63HbbbaZLly4mKyvLbNiwwfzkJz8xl1xyiWt7dXW16devn0lNTTWbN282S5cuNe3btzd//OMfXX327NljQkJCzKxZs0x2drZ59tlnjc1mM59++qmrT1v5Xa1bt84kJiaaiy++2MyYMcPVznU+e0ePHjXdunUzt9xyi/nyyy/Nnj17zGeffWZ27drl6vP444+byMhI895775mtW7eaK664wnTv3t0cP37c1ednP/uZSU5ONl988YX517/+ZXr27GluuOEG1/bCwkITFxdnJk+ebLZt22b+8Y9/mODgYPPCCy+4+qxZs8bYbDbzxBNPmOzsbPM///M/JiAgwHz99deeuRjn0COPPGJiYmLMRx99ZPbu3WvefvttExYWZp5++mlXH65zyyxdutTcd999ZsmSJUaSeffdd922+9J1bU4tp0Po8jPDhg0zGRkZrtc1NTUmISHBPPbYY16synfl5eUZSWbVqlXGGGMKCgpMQECAefvtt119vvnmGyPJrF271hhT+z8Jq9VqcnJyXH3mzZtnIiIiTEVFhTHGmHvuucf07dvX7VjXXXedSUtLc71uC7+r4uJi06tXL5OZmWnGjBnjCl1c59bxhz/8wYwcOfKk251Op4mPjzd/+ctfXG0FBQUmMDDQ/OMf/zDGGJOdnW0kmfXr17v6fPLJJ8ZisZgffvjBGGPM3/72NxMdHe267vXHvuiii1yvr732WjNhwgS34w8fPtz85je/ObuT9AETJkww//Vf/+XWdvXVV5vJkycbY7jOreXHocuXrmtzamkObi/6kcrKSm3cuFGpqamuNqvVqtTUVK1du9aLlfmuwsJCSVK7du0kSRs3blRVVZXbNezdu7e6du3quoZr165V//79FRcX5+qTlpamoqIibd++3dWn4T7q+9Tvo638rjIyMjRhwoRG14Lr3Do++OADDRkyRNdcc406dOiggQMH6qWXXnJt37t3r3JyctzOPzIyUsOHD3e7zlFRURoyZIirT2pqqqxWq7788ktXn9GjR8vhcLj6pKWlaceOHTp27Jirz6l+F+ezSy65RFlZWdq5c6ckaevWrVq9erXGjx8viet8rvjSdW1OLc1B6PIj+fn5qqmpcfsjJUlxcXHKycnxUlW+y+l0aubMmbr00kvVr18/SVJOTo4cDoeioqLc+ja8hjk5OU1e4/ptp+pTVFSk48ePt4nf1aJFi7Rp0yY99thjjbZxnVvHnj17NG/ePPXq1UufffaZbr/9dt1555169dVXJZ24Tqc6/5ycHHXo0MFtu91uV7t27Vrld+EP1/nee+/V9ddfr969eysgIEADBw7UzJkzNXnyZElc53PFl65rc2ppDnuzewJ+JiMjQ9u2bdPq1au9XYrfOXDggGbMmKHMzEwFBQV5uxy/5XQ6NWTIED366KOSpIEDB2rbtm16/vnnlZ6e7uXq/Mdbb72lN954Q2+++ab69u2rLVu2aObMmUpISOA644ww0uVH2rdvL5vN1ugTYLm5uYqPj/dSVb7pjjvu0EcffaQVK1aoc+fOrvb4+HhVVlaqoKDArX/DaxgfH9/kNa7fdqo+ERERCg4O9vvf1caNG5WXl6dBgwbJbrfLbrdr1apVeuaZZ2S32xUXF8d1bgUdO3ZUUlKSW1ufPn20f/9+SSeu06nOPz4+Xnl5eW7bq6urdfTo0Vb5XfjDdf7973/vGu3q37+/br75Zt11112uUVyu87nhS9e1ObU0B6HLjzgcDg0ePFhZWVmuNqfTqaysLI0YMcKLlfkOY4zuuOMOvfvuu1q+fLm6d+/utn3w4MEKCAhwu4Y7duzQ/v37XddwxIgR+vrrr93+Q8/MzFRERITrD+CIESPc9lHfp34f/v67GjdunL7++mtt2bLF9RgyZIgmT57ses51PnuXXnppoyVPdu7cqW7dukmSunfvrvj4eLfzLyoq0pdfful2nQsKCrRx40ZXn+XLl8vpdGr48OGuPp9//rmqqqpcfTIzM3XRRRcpOjra1edUv4vzWVlZmaxW9z+XNptNTqdTEtf5XPGl69qcWpql2VPucV5YtGiRCQwMNAsWLDDZ2dlm2rRpJioqyu0TYG3Z7bffbiIjI83KlSvNoUOHXI+ysjJXn9tuu8107drVLF++3GzYsMGMGDHCjBgxwrW9fimDyy+/3GzZssV8+umnJjY2tsmlDH7/+9+bb775xsydO7fJpQza0u+q4acXjeE6t4Z169YZu91uHnnkEfPtt9+aN954w4SEhJjXX3/d1efxxx83UVFR5v333zdfffWVufLKK5v8yP3AgQPNl19+aVavXm169erl9pH7goICExcXZ26++Wazbds2s2jRIhMSEtLoI/d2u9389a9/Nd9884154IEHzuulDBpKT083nTp1ci0ZsWTJEtO+fXtzzz33uPpwnVumuLjYbN682WzevNlIMk899ZTZvHmz+e6774wxvnVdm1PL6RC6/NCzzz5runbtahwOhxk2bJj54osvvF2Sz5DU5GP+/PmuPsePHzfTp0830dHRJiQkxFx11VXm0KFDbvvZt2+fGT9+vAkODjbt27c3v/vd70xVVZVbnxUrVpgBAwYYh8NhevTo4XaMem3pd/Xj0MV1bh0ffvih6devnwkMDDS9e/c2L774ott2p9Np/vSnP5m4uDgTGBhoxo0bZ3bs2OHW58iRI+aGG24wYWFhJiIiwkydOtUUFxe79dm6dasZOXKkCQwMNJ06dTKPP/54o1reeustc+GFFxqHw2H69u1rPv7449Y/YS8oKioyM2bMMF27djVBQUGmR48e5r777nNbgoDr3DIrVqxo8v/J6enpxhjfuq7NqeV0LMY0WFIXAAAA5wRzugAAADyA0AUAAOABhC4AAAAPIHQBAAB4AKELAADAAwhdAAAAHkDoAgAA8ABCFwB4UWJioubMmePtMgB4AKELQJtxyy23aNKkSZKksWPHaubMmR479oIFCxQVFdWoff369Zo2bZrH6gDgPXZvFwAA57PKyko5HI4Wvz82NrYVqwHgyxjpAtDm3HLLLVq1apWefvppWSwWWSwW7du3T5K0bds2jR8/XmFhYYqLi9PNN9+s/Px813vHjh2rO+64QzNnzlT79u2VlpYmSXrqqafUv39/hYaGqkuXLpo+fbpKSkokSStXrtTUqVNVWFjoOt6DDz4oqfHtxf379+vKK69UWFiYIiIidO211yo3N9e1/cEHH9SAAQO0cOFCJSYmKjIyUtdff72Ki4vP7UUDcNYIXQDanKefflojRozQrbfeqkOHDunQoUPq0qWLCgoKlJKSooEDB2rDhg369NNPlZubq2uvvdbt/a+++qocDofWrFmj559/XpJktVr1zDPPaPv27Xr11Ve1fPly3XPPPZKkSy65RHPmzFFERITreHfffXejupxOp6688kodPXpUq1atUmZmpvbs2aPrrrvOrd/u3bv13nvv6aOPPtJHH32kVatW6fHHHz9HVwtAa+H2IoA2JzIyUg6HQyEhIYqPj3e1P/fccxo4cKAeffRRV9srr7yiLl26aOfOnbrwwgslSb169dITTzzhts+G88MSExP18MMP67bbbtPf/vY3ORwORUZGymKxuB3vx7KysvT1119r79696tKliyTptddeU9++fbV+/XoNHTpUUm04W7BggcLDwyVJN998s7KysvTII4+c3YUBcE4x0gUAdbZu3aoVK1YoLCzM9ejdu7ek2tGleoMHD2703mXLlmncuHHq1KmTwsPDdfPNN+vIkSMqKytr9vG/+eYbdenSxRW4JCkpKUlRUVH65ptvXG2JiYmuwCVJHTt2VF5e3hmdKwDPY6QLAOqUlJRo4sSJmj17dqNtHTt2dD0PDQ1127Zv3z794he/0O23365HHnlE7dq10+rVq/XrX/9alZWVCgkJadU6AwIC3F5bLBY5nc5WPQaA1kfoAtAmORwO1dTUuLUNGjRI//d//6fExETZ7c3/3+PGjRvldDr15JNPymqtvYHw1ltvnfZ4P9anTx8dOHBABw4ccI12ZWdnq6CgQElJSc2uB4Bv4vYigDYpMTFRX375pfbt26f8/Hw5nU5lZGTo6NGjuuGGG7R+/Xrt3r1bn332maZOnXrKwNSzZ09VVVXp2Wef1Z49e7Rw4ULXBPuGxyspKVFWVpby8/ObvO2Ympqq/v37a/Lkydq0aZPWrVunKVOmaMyYMRoyZEirXwMAnkXoAtAm3X333bLZbEpKSlJsbKz279+vhIQErVmzRjU1Nbr88svVv39/zZw5U1FRUa4RrKYkJyfrqaee0uzZs9WvXz+98cYbeuyxx9z6XHLJJbrtttt03XXXKTY2ttFEfKn2NuH777+v6OhojR49WqmpqerRo4cWL17c6ucPwPMsxhjj7SIAAAD8HSNdAAAAHkDoAgAA8ABCFwAAgAcQugAAADyA0AUAAOABhC4AAAAPIHQBAAB4AKELAADAAwhdAAAAHkDoAgAA8ABCFwAAgAcQugAAADzg/wMVdQKDJMrm5QAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "N_u = 500 #Total number of data points for 'u'\n",
    "N_f = 10000 #Total number of collocation points \n",
    "\n",
    "# Training data\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([3, 1000, 1])\n",
    "\n",
    "PINN = Sequentialmodel(layers)\n",
    "       \n",
    "PINN.to(device)\n",
    "\n",
    "'Neural Network Summary'\n",
    "\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 = 100000        \n",
    "K1 = 40       \n",
    "gamma = 0.1     \n",
    "eta = 0.5     \n",
    "\n",
    "final_params, loss_list = IGD_PINN(PINN, X_u_train, u_train, X_f_train, theta_init, K0, K1, gamma, eta)\n",
    "\n",
    "plt.plot(loss_list)\n",
    "plt.yscale('log')\n",
    "plt.xlabel('Iteration')\n",
    "plt.ylabel('Loss')\n",
    "plt.title('Loss (only train a_layer)')\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "c594ff78",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Total grad norm: 6.7403e-04\n"
     ]
    }
   ],
   "source": [
    "\n",
    "PINN.zero_grad()\n",
    "loss = PINN.loss(X_u_train, u_train, X_f_train)\n",
    "loss.backward()\n",
    "\n",
    "total_norm = 0.0\n",
    "for param in PINN.parameters():\n",
    "    if param.grad is not None:\n",
    "        param_norm = param.grad.norm().item()\n",
    "        total_norm += param_norm ** 2\n",
    "total_norm = total_norm ** 0.5\n",
    "print(f\"Total grad norm: {total_norm:.4e}\")"
   ]
  }
 ],
 "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
}
