{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# PINN Solution of the Allen Cahn PDE\n",
    "\n",
    "This PyTorch code demonstrates the application of physically-informed neural networks (PINN) in the solution of a well-known Allen Cahn PDE with periodic boundary condition\n",
    "\\begin{aligned}\n",
    "  &u_t = \\epsilon\\Delta u - 5(u^3 - u), \\quad (t, x) \\in [0, T]\\times[-L, L]\\\\\n",
    "  &u(0, x) = u_0(x), \\quad \\forall x \\in [-L, L] \\\\\n",
    "  &u(t, -L) = u(t, L), \\quad \\forall t \\in [0, T]\n",
    "\\end{aligned}\n",
    "where $\\epsilon > 0$ is the defintion , and $[-L, L]$ covers one full period, i.e. $T = 2L$."
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Libraries and Dependencies"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "from itertools import chain\n",
    "from collections import OrderedDict\n",
    "import time\n",
    "import numpy as np\n",
    "import scipy as sp\n",
    "import scipy.io\n",
    "from scipy.interpolate import griddata\n",
    "from pyDOE import lhs\n",
    "import torch\n",
    "import torch.optim\n",
    "import torch.optim.lr_scheduler as lr_scheduler\n",
    "import matplotlib as mpl\n",
    "import matplotlib.pyplot as plt\n",
    "from mpl_toolkits.axes_grid1 import make_axes_locatable\n",
    "import matplotlib.gridspec as gridspec\n",
    "np.random.seed(1234)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Working on mps\n"
     ]
    }
   ],
   "source": [
    "if torch.backends.mps.is_available():\n",
    "    device = torch.device('mps')\n",
    "elif torch.cuda.is_available():\n",
    "    device = torch.device('cuda')\n",
    "else:\n",
    "    device = torch.device('cpu')\n",
    "#\n",
    "print(f\"Working on {device}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "epsilon = 1e-4\n",
    "gamma = 5\n",
    "L = 1.0\n",
    "xlo = -L\n",
    "xhi = L\n",
    "period = xhi - xlo\n",
    "tlo = 0.0\n",
    "thi = 1.0\n",
    "pi_ten = torch.tensor(np.pi).float().to(device)\n",
    "L_ten = torch.tensor(L).float().to(device)\n",
    "u0 = lambda x: np.power(x, 2.0) * np.cos(np.pi * x)\n",
    "u0_ten = lambda x: torch.pow(x, 2.0) * torch.cos(pi_ten * x)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Physics-informed Neural Networks"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "# the deep neural network\n",
    "class DNN(torch.nn.Module):\n",
    "    def __init__(self, layers):\n",
    "        super(DNN, self).__init__()\n",
    "        # parameters\n",
    "        self.depth = len(layers) - 1\n",
    "        # set up layer order dict\n",
    "        self.activation = torch.nn.Tanh\n",
    "        layer_list = list()\n",
    "        for i in range(self.depth - 1): \n",
    "            layer_list.append(\n",
    "                ('layer_%d' % i, torch.nn.Linear(layers[i], layers[i+1]))\n",
    "            )\n",
    "            layer_list.append(('activation_%d' % i, self.activation()))\n",
    "            \n",
    "        layer_list.append(\n",
    "            ('layer_%d' % (self.depth - 1), torch.nn.Linear(layers[-2], layers[-1]))\n",
    "        )\n",
    "        layerDict = OrderedDict(layer_list)\n",
    "        # deploy layers\n",
    "        self.layers = torch.nn.Sequential(layerDict)\n",
    "        self.layers[0].weight = torch.load('initial_weight.pt')\n",
    "    def forward(self, x):\n",
    "        out = self.layers(x)\n",
    "        return out"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "class PhysicsInformedNN():\n",
    "    def __init__(self, period, epsilon, gamma, X_BC, X_PDE, layers):\n",
    "       \n",
    "        # BC data point\n",
    "        self.t_BC = torch.tensor(X_BC[:, 0:1]).float().to(device)\n",
    "        self.x_BC = torch.tensor(X_BC[:, 1:2]).float().to(device)\n",
    "        N_BC = X_BC.shape[0]\n",
    "        self.LW_BC = torch.nn.ParameterList([torch.nn.Parameter(torch.ones(N_BC, 1).float(), requires_grad=True).to(device)])        \n",
    "\n",
    "        self.period = torch.tensor(period).float().to(device)\n",
    "        # PDE data, gradients will be computed on these points so requires_grad = True\n",
    "        self.t_PDE = torch.tensor(X_PDE[:, 0:1], requires_grad=True).float().to(device)\n",
    "        self.x_PDE = torch.tensor(X_PDE[:, 1:2], requires_grad=True).float().to(device)\n",
    "        N_PDE = X_PDE.shape[0]\n",
    "        self.LW_PDE = torch.nn.ParameterList([torch.nn.Parameter(torch.ones(N_PDE, 1).float(), requires_grad=True).to(device)])\n",
    "     \n",
    "        # equation related parameters\n",
    "        self.epsilon = torch.tensor(epsilon).float().to(device)\n",
    "        self.gamma = torch.tensor(gamma).float().to(device)\n",
    "        # layers to build Neural Net\n",
    "        self.layers = layers\n",
    "        # deep neural networks\n",
    "        self.dnn = DNN(layers).to(device)    \n",
    "        # prepare the optimizer\n",
    "        self.optimizer_Adam = torch.optim.Adam(self.dnn.parameters(), lr = 1e-3)\n",
    "        # add a learning rate scheduler\n",
    "        self.optimizer_LW_BC = torch.optim.Adam(self.LW_BC.parameters(), lr = 5e-3)\n",
    "        self.optimizer_LW_PDE = torch.optim.Adam(self.LW_PDE.parameters(), lr = 5e-3)\n",
    "       \n",
    "        self.optimizer_LBFGS = torch.optim.LBFGS(\n",
    "            self.dnn.parameters(), \n",
    "            lr=1.0, \n",
    "            max_iter=10000, \n",
    "            max_eval=5000, \n",
    "            history_size=50,\n",
    "            tolerance_grad=1e-7, \n",
    "            tolerance_change=1.0 * np.finfo(float).eps,\n",
    "            line_search_fn=\"strong_wolfe\"       # can be \"strong_wolfe\"\n",
    "        )      \n",
    "        self.scheduler = lr_scheduler.ExponentialLR(self.optimizer_Adam, gamma=0.99)\n",
    "        self.iter = 0\n",
    "    # evaluater neural network, no transformation\n",
    "    def NN_eval(self, t, x): \n",
    "        u = self.dnn(torch.cat([t, x], dim = 1))\n",
    "        u0_torch = u0_ten(x)\n",
    "        unew = u0_torch*torch.exp(-t) + t * u\n",
    "        return unew\n",
    "    # compute the PDE\n",
    "    def pde_eval(self, t, x):\n",
    "        \"\"\" The pytorch autograd version of calculating residual \"\"\"\n",
    "        u = self.NN_eval(t, x)\n",
    "        # compute the derivatives for u\n",
    "        u_t  = torch.autograd.grad(u,   t, grad_outputs = torch.ones_like(u), retain_graph = True, create_graph=True)[0]\n",
    "        u_x  = torch.autograd.grad(u,   x, grad_outputs = torch.ones_like(u), retain_graph = True, create_graph=True)[0]\n",
    "        u_xx = torch.autograd.grad(u_x, x, grad_outputs = torch.ones_like(u), retain_graph = True, create_graph=True)[0]\n",
    "        Eq  = u_t - self.epsilon * u_xx - self.gamma * (u - torch.pow(u, 3.0))       \n",
    "        return Eq\n",
    "    # compute the total loss for the second-order optimizer\n",
    "    def loss_func(self):\n",
    "        # reset the gradient\n",
    "        self.optimizer_LBFGS.zero_grad()\n",
    "\n",
    "        \n",
    "        # compute PBC loss\n",
    "        BC_pred_left = self.NN_eval(self.t_BC, self.x_BC)\n",
    "        BC_pred_right = self.NN_eval(self.t_BC, self.x_BC + self.period)\n",
    "        loss_BC = torch.mean(torch.square(self.LW_BC[0] * (BC_pred_left - BC_pred_right)))\n",
    "        # compute PDE loss\n",
    "        pde_pred = self.pde_eval(self.t_PDE, self.x_PDE)\n",
    "        loss_PDE = torch.mean(torch.square(self.LW_PDE[0] * pde_pred))  \n",
    "        # compute the total loss, it can be weighted\n",
    "        loss =  loss_BC + loss_PDE\n",
    "        # backward propagation\n",
    "        loss.backward()\n",
    "        # increase the iteration counter\n",
    "        self.iter += 1\n",
    "        # output\n",
    "        # output the progress\n",
    "        if self.iter % 1000 == 0:\n",
    "            end_time = time.time()\n",
    "            print('Iter %5d, Total: %10.4e, Time: %.2f secs' % (self.iter, loss.item(), end_time - self.start_time))\n",
    "            print('PBC: %10.4e, PDE: %10.4e' % (loss_BC.item(), loss_PDE.item()))\n",
    "            self.start_time = end_time\n",
    "        return loss\n",
    "    #\n",
    "    def train(self, nIter):\n",
    "        # start the timer\n",
    "        start_time = time.time()        \n",
    "        # start the training with Adam first\n",
    "        self.dnn.train()\n",
    "        print('Starting with Adam')\n",
    "        for epoch in range(nIter):\n",
    "          \n",
    "            # compute PBC loss\n",
    "            BC_pred_left = self.NN_eval(self.t_BC, self.x_BC)\n",
    "            BC_pred_right = self.NN_eval(self.t_BC, self.x_BC + self.period)\n",
    "            loss_BC = torch.mean(torch.square(self.LW_BC[0] * (BC_pred_left - BC_pred_right)))             \n",
    "            # compute PDE loss\n",
    "            pde_pred = self.pde_eval(self.t_PDE, self.x_PDE)\n",
    "            loss_PDE = torch.mean(torch.square(self.LW_PDE[0] * pde_pred))   \n",
    "            # compute the total loss, it can be weighted\n",
    "            loss =  loss_BC + loss_PDE\n",
    "            # Backward and optimize\n",
    "            self.optimizer_Adam.zero_grad()\n",
    "            self.optimizer_LW_BC.zero_grad()\n",
    "            self.optimizer_LW_PDE.zero_grad()\n",
    "            loss.backward()\n",
    "            self.optimizer_Adam.step() \n",
    "            self.LW_BC[0].grad.data = -self.LW_BC[0].grad.data\n",
    "            self.LW_PDE[0].grad.data = -self.LW_PDE[0].grad.data\n",
    "            self.optimizer_LW_BC.step()\n",
    "            self.optimizer_LW_PDE.step()\n",
    "            # output the progress\n",
    "            if (epoch + 1) % 1000 == 0:\n",
    "                end_time = time.time()\n",
    "                print('Iter %5d, Total: %10.4e, Time: %.2f secs' % (epoch + 1, loss.item(), end_time - start_time))\n",
    "                print('PBC: %10.4e, PDE: %10.4e' % ( loss_BC.item(), loss_PDE.item()))\n",
    "                print('For BC,  min LW: %10.4e, max LW: %10.4e' %(torch.min(self.LW_BC[0]).item(), torch.max(self.LW_BC[0]).item()))\n",
    "                print('For PDE, min LW: %10.4e, max LW: %10.4e' %(torch.min(self.LW_PDE[0]).item(), torch.max(self.LW_PDE[0]).item()))\n",
    "\n",
    "                start_time = end_time\n",
    "                # change the learning rate\n",
    "                self.scheduler.step()\n",
    "                \n",
    "        print('Starting with L-BFGS')\n",
    "        self.start_time = time.time()\n",
    "        self.optimizer_LBFGS.step(self.loss_func)    \n",
    "   \n",
    "    def predict(self, X):\n",
    "        t = torch.tensor(X[:, 0:1], requires_grad=True).float().to(device)\n",
    "        x = torch.tensor(X[:, 1:2], requires_grad=True).float().to(device)\n",
    "        self.dnn.eval()\n",
    "        u = self.NN_eval(t, x)\n",
    "        u = u.detach().cpu().numpy()\n",
    "        return u\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Training"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Starting with Adam\n",
      "Iter  1000, Total: 6.3021e-01, Time: 42.99 secs\n",
      "PBC: 1.4222e-03, PDE: 6.2879e-01\n",
      "For BC,  min LW: 3.9461e+00, max LW: 6.4277e+00\n",
      "For PDE, min LW: 1.5540e+00, max LW: 9.3651e+00\n",
      "Iter  2000, Total: 9.4698e-01, Time: 34.92 secs\n",
      "PBC: 3.7974e-03, PDE: 9.4318e-01\n",
      "For BC,  min LW: 4.5778e+00, max LW: 1.0191e+01\n",
      "For PDE, min LW: 1.7596e+00, max LW: 1.1640e+01\n",
      "Iter  3000, Total: 1.6770e+00, Time: 36.12 secs\n",
      "PBC: 4.7131e-03, PDE: 1.6723e+00\n",
      "For BC,  min LW: 5.7930e+00, max LW: 1.3431e+01\n",
      "For PDE, min LW: 2.0908e+00, max LW: 1.7745e+01\n",
      "Iter  4000, Total: 2.3323e+00, Time: 34.43 secs\n",
      "PBC: 9.2000e-03, PDE: 2.3231e+00\n",
      "For BC,  min LW: 5.8865e+00, max LW: 1.3494e+01\n",
      "For PDE, min LW: 2.2012e+00, max LW: 2.2739e+01\n",
      "Iter  5000, Total: 3.2731e-01, Time: 35.50 secs\n",
      "PBC: 8.8619e-03, PDE: 3.1844e-01\n",
      "For BC,  min LW: 6.1321e+00, max LW: 1.3595e+01\n",
      "For PDE, min LW: 2.3604e+00, max LW: 2.6297e+01\n",
      "Iter  6000, Total: 1.4964e-01, Time: 35.41 secs\n",
      "PBC: 1.5758e-03, PDE: 1.4806e-01\n",
      "For BC,  min LW: 6.4912e+00, max LW: 1.3731e+01\n",
      "For PDE, min LW: 2.5162e+00, max LW: 2.9423e+01\n",
      "Iter  7000, Total: 3.8052e-01, Time: 35.02 secs\n",
      "PBC: 3.1784e-03, PDE: 3.7734e-01\n",
      "For BC,  min LW: 6.9829e+00, max LW: 1.3926e+01\n",
      "For PDE, min LW: 2.6624e+00, max LW: 3.3432e+01\n",
      "Iter  8000, Total: 9.7365e-02, Time: 35.02 secs\n",
      "PBC: 3.1604e-04, PDE: 9.7048e-02\n",
      "For BC,  min LW: 7.6133e+00, max LW: 1.4204e+01\n",
      "For PDE, min LW: 2.7916e+00, max LW: 3.8165e+01\n",
      "Iter  9000, Total: 7.4037e-01, Time: 35.17 secs\n",
      "PBC: 4.0863e-03, PDE: 7.3628e-01\n",
      "For BC,  min LW: 8.4559e+00, max LW: 1.6420e+01\n",
      "For PDE, min LW: 2.9700e+00, max LW: 4.3097e+01\n",
      "Iter 10000, Total: 3.6289e-01, Time: 33.92 secs\n",
      "PBC: 9.4863e-04, PDE: 3.6194e-01\n",
      "For BC,  min LW: 9.4401e+00, max LW: 2.0139e+01\n",
      "For PDE, min LW: 3.3378e+00, max LW: 4.8002e+01\n",
      "Iter 11000, Total: 1.6735e-01, Time: 34.51 secs\n",
      "PBC: 8.1412e-04, PDE: 1.6653e-01\n",
      "For BC,  min LW: 1.0858e+01, max LW: 2.4001e+01\n",
      "For PDE, min LW: 4.0449e+00, max LW: 5.2816e+01\n",
      "Iter 12000, Total: 1.5368e+00, Time: 33.92 secs\n",
      "PBC: 7.0321e-04, PDE: 1.5361e+00\n",
      "For BC,  min LW: 1.2745e+01, max LW: 2.7663e+01\n",
      "For PDE, min LW: 4.5677e+00, max LW: 5.7416e+01\n",
      "Iter 13000, Total: 4.7094e-01, Time: 33.98 secs\n",
      "PBC: 1.5945e-03, PDE: 4.6934e-01\n",
      "For BC,  min LW: 1.4973e+01, max LW: 3.0991e+01\n",
      "For PDE, min LW: 5.3492e+00, max LW: 6.1808e+01\n",
      "Iter 14000, Total: 1.2682e+00, Time: 34.24 secs\n",
      "PBC: 7.6637e-03, PDE: 1.2605e+00\n",
      "For BC,  min LW: 1.7204e+01, max LW: 3.3901e+01\n",
      "For PDE, min LW: 5.5046e+00, max LW: 6.6033e+01\n",
      "Iter 15000, Total: 4.2798e-01, Time: 34.14 secs\n",
      "PBC: 1.7982e-03, PDE: 4.2618e-01\n",
      "For BC,  min LW: 1.9589e+01, max LW: 3.6678e+01\n",
      "For PDE, min LW: 5.7273e+00, max LW: 7.0221e+01\n",
      "Iter 16000, Total: 2.6994e-01, Time: 33.62 secs\n",
      "PBC: 7.0729e-03, PDE: 2.6286e-01\n",
      "For BC,  min LW: 2.2028e+01, max LW: 3.9429e+01\n",
      "For PDE, min LW: 6.4138e+00, max LW: 7.4412e+01\n",
      "Iter 17000, Total: 2.5995e-01, Time: 33.63 secs\n",
      "PBC: 6.5018e-04, PDE: 2.5930e-01\n",
      "For BC,  min LW: 2.4641e+01, max LW: 4.2288e+01\n",
      "For PDE, min LW: 7.7716e+00, max LW: 7.8598e+01\n",
      "Iter 18000, Total: 7.4169e-01, Time: 33.98 secs\n",
      "PBC: 7.2453e-03, PDE: 7.3444e-01\n",
      "For BC,  min LW: 2.6480e+01, max LW: 4.4505e+01\n",
      "For PDE, min LW: 8.9661e+00, max LW: 8.2776e+01\n",
      "Iter 19000, Total: 5.3631e-01, Time: 34.29 secs\n",
      "PBC: 5.2360e-03, PDE: 5.3107e-01\n",
      "For BC,  min LW: 2.8513e+01, max LW: 4.6882e+01\n",
      "For PDE, min LW: 1.0585e+01, max LW: 8.6966e+01\n",
      "Iter 20000, Total: 7.2604e-01, Time: 33.64 secs\n",
      "PBC: 5.0881e-04, PDE: 7.2553e-01\n",
      "For BC,  min LW: 3.0819e+01, max LW: 4.9427e+01\n",
      "For PDE, min LW: 1.2541e+01, max LW: 9.1192e+01\n",
      "Iter 21000, Total: 2.0620e-01, Time: 33.50 secs\n",
      "PBC: 5.2686e-04, PDE: 2.0568e-01\n",
      "For BC,  min LW: 3.2987e+01, max LW: 5.1836e+01\n",
      "For PDE, min LW: 1.4709e+01, max LW: 9.5382e+01\n",
      "Iter 22000, Total: 5.3668e-01, Time: 34.08 secs\n",
      "PBC: 1.3575e-02, PDE: 5.2310e-01\n",
      "For BC,  min LW: 3.5098e+01, max LW: 5.4138e+01\n",
      "For PDE, min LW: 1.6584e+01, max LW: 9.9521e+01\n",
      "Iter 23000, Total: 4.2190e-01, Time: 33.53 secs\n",
      "PBC: 1.5610e-02, PDE: 4.0628e-01\n",
      "For BC,  min LW: 3.7425e+01, max LW: 5.6578e+01\n",
      "For PDE, min LW: 1.8727e+01, max LW: 1.0367e+02\n",
      "Iter 24000, Total: 2.0893e-01, Time: 33.52 secs\n",
      "PBC: 3.1022e-04, PDE: 2.0862e-01\n",
      "For BC,  min LW: 3.9669e+01, max LW: 5.8938e+01\n",
      "For PDE, min LW: 2.1132e+01, max LW: 1.0779e+02\n",
      "Iter 25000, Total: 4.9235e-01, Time: 33.42 secs\n",
      "PBC: 1.0019e-02, PDE: 4.8233e-01\n",
      "For BC,  min LW: 4.1767e+01, max LW: 6.1131e+01\n",
      "For PDE, min LW: 2.3277e+01, max LW: 1.1188e+02\n",
      "Iter 26000, Total: 1.0608e+00, Time: 33.48 secs\n",
      "PBC: 1.7983e-03, PDE: 1.0590e+00\n",
      "For BC,  min LW: 4.3877e+01, max LW: 6.3255e+01\n",
      "For PDE, min LW: 2.5557e+01, max LW: 1.1596e+02\n",
      "Iter 27000, Total: 1.8713e-01, Time: 33.53 secs\n",
      "PBC: 4.2027e-04, PDE: 1.8671e-01\n",
      "For BC,  min LW: 4.6354e+01, max LW: 6.5713e+01\n",
      "For PDE, min LW: 2.7732e+01, max LW: 1.2002e+02\n",
      "Iter 28000, Total: 5.7959e-01, Time: 33.63 secs\n",
      "PBC: 3.2592e-02, PDE: 5.4700e-01\n",
      "For BC,  min LW: 4.8507e+01, max LW: 6.7835e+01\n",
      "For PDE, min LW: 2.9712e+01, max LW: 1.2405e+02\n",
      "Iter 29000, Total: 9.3012e-01, Time: 33.44 secs\n",
      "PBC: 1.1238e-03, PDE: 9.2900e-01\n",
      "For BC,  min LW: 5.1206e+01, max LW: 7.0460e+01\n",
      "For PDE, min LW: 3.2145e+01, max LW: 1.2813e+02\n",
      "Iter 30000, Total: 4.2509e-01, Time: 33.56 secs\n",
      "PBC: 1.4886e-02, PDE: 4.1020e-01\n",
      "For BC,  min LW: 5.3734e+01, max LW: 7.2904e+01\n",
      "For PDE, min LW: 3.4385e+01, max LW: 1.3214e+02\n",
      "Iter 31000, Total: 6.4164e-01, Time: 34.68 secs\n",
      "PBC: 6.7965e-04, PDE: 6.4096e-01\n",
      "For BC,  min LW: 5.6125e+01, max LW: 7.5215e+01\n",
      "For PDE, min LW: 3.6497e+01, max LW: 1.3704e+02\n",
      "Iter 32000, Total: 1.4194e+00, Time: 34.29 secs\n",
      "PBC: 9.3893e-02, PDE: 1.3255e+00\n",
      "For BC,  min LW: 5.8389e+01, max LW: 7.7392e+01\n",
      "For PDE, min LW: 3.8919e+01, max LW: 1.4196e+02\n",
      "Iter 33000, Total: 1.3645e+00, Time: 36.52 secs\n",
      "PBC: 1.0672e-03, PDE: 1.3634e+00\n",
      "For BC,  min LW: 6.0969e+01, max LW: 7.9893e+01\n",
      "For PDE, min LW: 4.1390e+01, max LW: 1.4689e+02\n",
      "Iter 34000, Total: 1.7459e+00, Time: 34.57 secs\n",
      "PBC: 2.7496e-02, PDE: 1.7184e+00\n",
      "For BC,  min LW: 6.3343e+01, max LW: 8.2167e+01\n",
      "For PDE, min LW: 4.3658e+01, max LW: 1.5183e+02\n",
      "Iter 35000, Total: 7.3830e-01, Time: 34.57 secs\n",
      "PBC: 4.6896e-02, PDE: 6.9141e-01\n",
      "For BC,  min LW: 6.5181e+01, max LW: 8.3895e+01\n",
      "For PDE, min LW: 4.5272e+01, max LW: 1.5677e+02\n",
      "Iter 36000, Total: 1.3334e+00, Time: 34.20 secs\n",
      "PBC: 1.9666e-03, PDE: 1.3314e+00\n",
      "For BC,  min LW: 6.7392e+01, max LW: 8.5980e+01\n",
      "For PDE, min LW: 4.7120e+01, max LW: 1.6172e+02\n",
      "Iter 37000, Total: 8.9415e-01, Time: 33.99 secs\n",
      "PBC: 1.1579e-03, PDE: 8.9300e-01\n",
      "For BC,  min LW: 6.9778e+01, max LW: 8.8238e+01\n",
      "For PDE, min LW: 4.9332e+01, max LW: 1.6666e+02\n",
      "Iter 38000, Total: 2.5465e+00, Time: 34.64 secs\n",
      "PBC: 4.0387e-02, PDE: 2.5061e+00\n",
      "For BC,  min LW: 7.2068e+01, max LW: 9.0368e+01\n",
      "For PDE, min LW: 5.2044e+01, max LW: 1.7160e+02\n",
      "Iter 39000, Total: 1.1056e+00, Time: 34.19 secs\n",
      "PBC: 1.0437e-03, PDE: 1.1046e+00\n",
      "For BC,  min LW: 7.4217e+01, max LW: 9.2365e+01\n",
      "For PDE, min LW: 5.4723e+01, max LW: 1.7655e+02\n",
      "Iter 40000, Total: 3.2455e+00, Time: 34.31 secs\n",
      "PBC: 7.6103e-03, PDE: 3.2379e+00\n",
      "For BC,  min LW: 7.6543e+01, max LW: 9.4532e+01\n",
      "For PDE, min LW: 5.6989e+01, max LW: 1.8149e+02\n",
      "Iter 41000, Total: 2.9897e-01, Time: 34.05 secs\n",
      "PBC: 3.9930e-03, PDE: 2.9498e-01\n",
      "For BC,  min LW: 7.8709e+01, max LW: 9.6641e+01\n",
      "For PDE, min LW: 5.9024e+01, max LW: 1.8643e+02\n",
      "Iter 42000, Total: 1.5386e+00, Time: 33.48 secs\n",
      "PBC: 2.0354e-03, PDE: 1.5365e+00\n",
      "For BC,  min LW: 8.0991e+01, max LW: 9.8903e+01\n",
      "For PDE, min LW: 6.1182e+01, max LW: 1.9137e+02\n",
      "Iter 43000, Total: 3.3279e+00, Time: 33.43 secs\n",
      "PBC: 1.1495e-01, PDE: 3.2129e+00\n",
      "For BC,  min LW: 8.2683e+01, max LW: 1.0058e+02\n",
      "For PDE, min LW: 6.3084e+01, max LW: 1.9631e+02\n",
      "Iter 44000, Total: 4.2667e-01, Time: 33.45 secs\n",
      "PBC: 1.2132e-02, PDE: 4.1454e-01\n",
      "For BC,  min LW: 8.4648e+01, max LW: 1.0253e+02\n",
      "For PDE, min LW: 6.5106e+01, max LW: 2.0124e+02\n",
      "Iter 45000, Total: 7.5865e+00, Time: 33.47 secs\n",
      "PBC: 3.5771e-02, PDE: 7.5508e+00\n",
      "For BC,  min LW: 8.6863e+01, max LW: 1.0471e+02\n",
      "For PDE, min LW: 6.7357e+01, max LW: 2.0617e+02\n",
      "Iter 46000, Total: 9.2815e-01, Time: 33.36 secs\n",
      "PBC: 1.6138e-02, PDE: 9.1202e-01\n",
      "For BC,  min LW: 8.8931e+01, max LW: 1.0675e+02\n",
      "For PDE, min LW: 6.9278e+01, max LW: 2.1110e+02\n",
      "Iter 47000, Total: 1.0462e+00, Time: 33.48 secs\n",
      "PBC: 1.8072e-02, PDE: 1.0281e+00\n",
      "For BC,  min LW: 9.0882e+01, max LW: 1.0867e+02\n",
      "For PDE, min LW: 7.1208e+01, max LW: 2.1603e+02\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Iter 48000, Total: 7.4013e-01, Time: 33.42 secs\n",
      "PBC: 9.2326e-04, PDE: 7.3921e-01\n",
      "For BC,  min LW: 9.3106e+01, max LW: 1.1086e+02\n",
      "For PDE, min LW: 7.2970e+01, max LW: 2.2096e+02\n",
      "Iter 49000, Total: 1.1106e+00, Time: 33.58 secs\n",
      "PBC: 3.2023e-02, PDE: 1.0786e+00\n",
      "For BC,  min LW: 9.5215e+01, max LW: 1.1294e+02\n",
      "For PDE, min LW: 7.5066e+01, max LW: 2.2589e+02\n",
      "Iter 50000, Total: 4.8130e-01, Time: 33.53 secs\n",
      "PBC: 1.2613e-02, PDE: 4.6869e-01\n",
      "For BC,  min LW: 9.6958e+01, max LW: 1.1465e+02\n",
      "For PDE, min LW: 7.6817e+01, max LW: 2.3081e+02\n",
      "Starting with L-BFGS\n",
      "Iter  1000, Total: 9.8175e-02, Time: 74.30 secs\n",
      "PBC: 2.0627e-04, PDE: 9.7969e-02\n",
      "Iter  2000, Total: 4.7082e-02, Time: 74.34 secs\n",
      "PBC: 1.1591e-04, PDE: 4.6966e-02\n",
      "Iter  3000, Total: 2.2789e-02, Time: 74.35 secs\n",
      "PBC: 6.1550e-05, PDE: 2.2728e-02\n",
      "Iter  4000, Total: 1.3727e-02, Time: 74.17 secs\n",
      "PBC: 2.1526e-05, PDE: 1.3705e-02\n",
      "Iter  5000, Total: 9.7335e-03, Time: 73.72 secs\n",
      "PBC: 2.4778e-05, PDE: 9.7087e-03\n"
     ]
    }
   ],
   "source": [
    "layers = [2, 32, 32, 32, 32, 32, 32, 32, 1]\n",
    "ptsIC=np.load('ptsIC.npy')\n",
    "ptsBC=np.load('ptsBC.npy')\n",
    "ptsPDE=np.load('ptsPDE.npy')\n",
    "ptsPDE = np.vstack((ptsIC,ptsPDE))\n",
    "N_IC = 128\n",
    "x_IC = np.expand_dims(np.linspace(xlo, xhi, N_IC), axis = 1)\n",
    "u_IC= u0(x_IC)\n",
    "\n",
    "model = PhysicsInformedNN(period,epsilon, gamma,ptsBC, ptsPDE, layers )\n",
    "model.train(50000)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [],
   "source": [
    "# apply PINN to the same grid as the quadrature solution for comparison\n",
    "t = np.linspace(tlo, thi, 101)\n",
    "x = np.linspace(xlo, xhi, 201)\n",
    "T, X = np.meshgrid(t, x)\n",
    "pts_flat = np.hstack((T.flatten()[:, None], X.flatten()[:, None]))\n",
    "u_pred = model.predict(pts_flat)\n",
    "\n",
    "u_pred = griddata(pts_flat, u_pred.flatten(), (T, X), method='cubic')\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Visualizations"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [],
   "source": [
    "data = scipy.io.loadmat('Data/AC_case1.mat')\n",
    "t = data['t'].flatten()[:,None]\n",
    "x2 = data['x'].flatten()[:,None]\n",
    "Exact = np.real(data['exact_sol']).T\n",
    "X, T = np.meshgrid(x2,t)\n",
    "\n",
    "X_star = np.hstack((X.flatten()[:,None], T.flatten()[:,None]))\n",
    "u_star = Exact.flatten()[:,None]    "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [],
   "source": [
    "def relative_error_l2(pred,exact):\n",
    "    error_l2 = np.sqrt(np.sum(np.power(pred - exact,2)))\n",
    "    relative = error_l2/np.sqrt(np.sum(np.power(exact,2)))\n",
    "    return relative\n",
    "def relative_error_l1(pred,exact):\n",
    "    error_l1 = np.sum(np.abs(pred-exact))\n",
    "    relative = error_l1/np.sum(np.abs(exact))\n",
    "    return relative\n",
    "def relative_error_linf(pred,exact):\n",
    "    error_linf = np.max(np.abs(pred-exact))\n",
    "    relative = error_linf/np.max(np.abs(exact))\n",
    "    return relative"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "l2: 0.0018615499916465647\n",
      "l1: 0.0007305501515864727\n",
      "linf: 0.023081109670508724\n"
     ]
    }
   ],
   "source": [
    "print(f'l2: {relative_error_l2(u_pred.T,Exact)}')\n",
    "print(f'l1: {relative_error_l1(u_pred.T,Exact)}')\n",
    "print(f'linf: {relative_error_linf(u_pred.T,Exact)}')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.12.6"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
