{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# PINN Solution of the Allen-Cahn Equations\n",
    "\n",
    "This PyTorch code demonstrates the application of physically-informed neural networks (PINN) in the solution of a well-known Allen-Cahn Equations with periodic boundary condition\n",
    "\\begin{aligned}\n",
    "  &u_t = \\gamma_1 u_{xx} - \\gamma_2 u^3+\\gamma_2 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 u_x(t, -L) = u_x(t, L), \\quad \\forall t \\in [0, T]\n",
    "\\end{aligned}\n",
    "where $\\gamma_1, \\gamma_2> 0$ are some parameters, and $[x_{\\min}, x_{\\max}]$ covers one full period.\n",
    "\n",
    "Here, we let $u_0(x) = x^2\\sin(2\\pi x)$."
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Libraries and Dependencies"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "import torch.optim\n",
    "import torch.optim.lr_scheduler as lr_scheduler\n",
    "from itertools import chain\n",
    "from collections import OrderedDict\n",
    "from pyDOE import lhs\n",
    "import numpy as np\n",
    "import matplotlib as mpl\n",
    "import matplotlib.pyplot as plt\n",
    "import scipy.io\n",
    "from scipy.interpolate import griddata\n",
    "from mpl_toolkits.axes_grid1 import make_axes_locatable\n",
    "import matplotlib.gridspec as gridspec\n",
    "import time\n",
    "# set the random seed\n",
    "np.random.seed(1234)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "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": 3,
   "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": 4,
   "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_weights_PBC.pt')\n",
    "    def forward(self, x):\n",
    "        out = self.layers(x)\n",
    "        return out"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "class PhysicsInformedNN():\n",
    "    def __init__(self, period, m, epsilon, gamma, X_IC, u_IC, X_PDE, layers):\n",
    "        # IC data point\n",
    "        m_vec = np.expand_dims(np.arange(1, m + 1), axis = 0)\n",
    "        self.ms = torch.tensor(2.0 * np.pi/period * m_vec).float().to(device)\n",
    "        layers[0] = int(2 * m + 2)\n",
    "        self.t_IC = torch.tensor(X_IC[:, 0:1]).float().to(device)\n",
    "        self.x_IC = torch.tensor(X_IC[:, 1:2]).float().to(device)\n",
    "        self.u_IC = torch.tensor(u_IC).float().to(device)\n",
    " \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",
    "        # 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_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",
    "        x_trans = torch.matmul(x, self.ms)\n",
    "        uNN = self.dnn(torch.cat([t, torch.ones_like(x), torch.cos(x_trans), torch.sin(x_trans)], dim = 1))\n",
    "        return uNN\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",
    "        # compute IC loss\n",
    "        IC_pred = self.NN_eval(self.t_IC, self.x_IC)\n",
    "        loss_IC = torch.mean(torch.square(IC_pred - self.u_IC))\n",
    "       \n",
    "        # compute PDE loss\n",
    "        pde_pred = self.pde_eval(self.t_PDE, self.x_PDE)\n",
    "        loss_PDE = torch.mean(torch.square(pde_pred))    \n",
    "        # compute the total loss, it can be weighted\n",
    "        loss = loss_IC  + 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('IC: %10.4e,  PDE: %10.4e' % (loss_IC.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",
    "            # compute IC loss\n",
    "            IC_pred = self.NN_eval(self.t_IC, self.x_IC)\n",
    "            loss_IC = torch.mean(torch.square(IC_pred - self.u_IC))\n",
    "            \n",
    "            # compute PDE loss\n",
    "            pde_pred = self.pde_eval(self.t_PDE, self.x_PDE)\n",
    "            loss_PDE = torch.mean(torch.square(pde_pred))  \n",
    "            # compute the total loss, it can be weighted\n",
    "            loss = loss_IC +  loss_PDE\n",
    "            # Backward and optimize\n",
    "            self.optimizer_Adam.zero_grad()\n",
    "            loss.backward()\n",
    "            self.optimizer_Adam.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('IC: %10.4e,  PDE: %10.4e' % (loss_IC.item(), loss_PDE.item()))\n",
    "                start_time = end_time\n",
    "                # change the learning rate\n",
    "                self.scheduler.step()\n",
    "\n",
    "        self.start_time = time.time()\n",
    "        self.optimizer_LBFGS.step(self.loss_func)    \n",
    "\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": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Starting with Adam\n",
      "Iter  1000, Total: 1.2407e-01, Time: 59.63 secs\n",
      "IC: 1.0610e-01,  PDE: 1.7975e-02\n",
      "Iter  2000, Total: 9.2307e-02, Time: 57.02 secs\n",
      "IC: 7.0810e-02,  PDE: 2.1498e-02\n",
      "Iter  3000, Total: 4.3124e-02, Time: 57.15 secs\n",
      "IC: 1.5502e-02,  PDE: 2.7622e-02\n",
      "Iter  4000, Total: 3.5768e-02, Time: 56.48 secs\n",
      "IC: 9.3309e-03,  PDE: 2.6437e-02\n",
      "Iter  5000, Total: 1.9606e-02, Time: 57.21 secs\n",
      "IC: 7.2709e-03,  PDE: 1.2336e-02\n",
      "Iter  6000, Total: 1.0558e-02, Time: 57.56 secs\n",
      "IC: 4.6081e-03,  PDE: 5.9496e-03\n",
      "Iter  7000, Total: 8.9165e-03, Time: 57.17 secs\n",
      "IC: 3.8834e-03,  PDE: 5.0331e-03\n",
      "Iter  8000, Total: 7.4375e-03, Time: 57.43 secs\n",
      "IC: 3.3640e-03,  PDE: 4.0735e-03\n",
      "Iter  9000, Total: 6.8796e-03, Time: 51.51 secs\n",
      "IC: 3.0143e-03,  PDE: 3.8653e-03\n",
      "Iter 10000, Total: 5.3968e-03, Time: 46.05 secs\n",
      "IC: 2.1975e-03,  PDE: 3.1993e-03\n",
      "Iter 11000, Total: 4.9509e-03, Time: 45.59 secs\n",
      "IC: 2.1389e-03,  PDE: 2.8120e-03\n",
      "Iter 12000, Total: 1.2170e-02, Time: 45.87 secs\n",
      "IC: 2.0915e-03,  PDE: 1.0079e-02\n",
      "Iter 13000, Total: 3.8309e-03, Time: 45.62 secs\n",
      "IC: 1.6881e-03,  PDE: 2.1428e-03\n",
      "Iter 14000, Total: 4.4815e-03, Time: 40.41 secs\n",
      "IC: 1.9309e-03,  PDE: 2.5506e-03\n",
      "Iter 15000, Total: 5.4012e-03, Time: 40.21 secs\n",
      "IC: 1.8949e-03,  PDE: 3.5063e-03\n",
      "Iter 16000, Total: 3.0564e-03, Time: 40.08 secs\n",
      "IC: 1.4798e-03,  PDE: 1.5766e-03\n",
      "Iter 17000, Total: 3.3482e-03, Time: 39.99 secs\n",
      "IC: 1.4606e-03,  PDE: 1.8876e-03\n",
      "Iter 18000, Total: 3.7985e-03, Time: 40.30 secs\n",
      "IC: 1.6307e-03,  PDE: 2.1678e-03\n",
      "Iter 19000, Total: 2.7310e-03, Time: 40.29 secs\n",
      "IC: 1.4440e-03,  PDE: 1.2870e-03\n",
      "Iter 20000, Total: 2.7933e-03, Time: 39.96 secs\n",
      "IC: 1.4813e-03,  PDE: 1.3120e-03\n",
      "Iter 21000, Total: 2.3842e-03, Time: 39.45 secs\n",
      "IC: 1.3463e-03,  PDE: 1.0379e-03\n",
      "Iter 22000, Total: 2.4086e-03, Time: 39.75 secs\n",
      "IC: 1.3728e-03,  PDE: 1.0359e-03\n",
      "Iter 23000, Total: 2.1441e-03, Time: 40.18 secs\n",
      "IC: 1.2985e-03,  PDE: 8.4555e-04\n",
      "Iter 24000, Total: 2.0537e-03, Time: 39.94 secs\n",
      "IC: 1.2780e-03,  PDE: 7.7574e-04\n",
      "Iter 25000, Total: 2.3806e-03, Time: 40.42 secs\n",
      "IC: 1.3376e-03,  PDE: 1.0430e-03\n",
      "Iter 26000, Total: 1.9442e-03, Time: 40.32 secs\n",
      "IC: 1.2488e-03,  PDE: 6.9541e-04\n",
      "Iter 27000, Total: 1.9992e-03, Time: 39.59 secs\n",
      "IC: 1.2248e-03,  PDE: 7.7435e-04\n",
      "Iter 28000, Total: 1.8877e-03, Time: 40.64 secs\n",
      "IC: 1.2319e-03,  PDE: 6.5580e-04\n",
      "Iter 29000, Total: 1.7618e-03, Time: 40.05 secs\n",
      "IC: 1.1946e-03,  PDE: 5.6725e-04\n",
      "Iter 30000, Total: 1.7490e-03, Time: 40.12 secs\n",
      "IC: 1.2082e-03,  PDE: 5.4080e-04\n",
      "Iter 31000, Total: 2.0595e-03, Time: 39.69 secs\n",
      "IC: 1.1955e-03,  PDE: 8.6402e-04\n",
      "Iter 32000, Total: 3.0090e-03, Time: 40.09 secs\n",
      "IC: 1.3330e-03,  PDE: 1.6760e-03\n",
      "Iter 33000, Total: 1.7260e-03, Time: 39.53 secs\n",
      "IC: 1.1589e-03,  PDE: 5.6712e-04\n",
      "Iter 34000, Total: 1.8128e-03, Time: 40.84 secs\n",
      "IC: 1.1670e-03,  PDE: 6.4582e-04\n",
      "Iter 35000, Total: 1.7596e-03, Time: 40.51 secs\n",
      "IC: 1.1686e-03,  PDE: 5.9105e-04\n",
      "Iter 36000, Total: 2.6743e-03, Time: 40.38 secs\n",
      "IC: 1.2509e-03,  PDE: 1.4234e-03\n",
      "Iter 37000, Total: 1.9094e-03, Time: 39.62 secs\n",
      "IC: 1.1457e-03,  PDE: 7.6373e-04\n",
      "Iter 38000, Total: 2.1282e-03, Time: 40.50 secs\n",
      "IC: 1.1941e-03,  PDE: 9.3418e-04\n",
      "Iter 39000, Total: 1.6587e-03, Time: 40.21 secs\n",
      "IC: 1.0808e-03,  PDE: 5.7792e-04\n",
      "Iter 40000, Total: 1.4451e-03, Time: 40.32 secs\n",
      "IC: 1.0482e-03,  PDE: 3.9691e-04\n",
      "Iter 41000, Total: 1.7576e-03, Time: 40.29 secs\n",
      "IC: 1.0481e-03,  PDE: 7.0950e-04\n",
      "Iter 42000, Total: 1.4135e-03, Time: 39.63 secs\n",
      "IC: 1.0038e-03,  PDE: 4.0974e-04\n",
      "Iter 43000, Total: 2.0030e-03, Time: 40.24 secs\n",
      "IC: 1.0480e-03,  PDE: 9.5503e-04\n",
      "Iter 44000, Total: 1.3670e-03, Time: 40.04 secs\n",
      "IC: 9.6747e-04,  PDE: 3.9951e-04\n",
      "Iter 45000, Total: 1.6317e-03, Time: 40.39 secs\n",
      "IC: 9.7830e-04,  PDE: 6.5343e-04\n",
      "Iter 46000, Total: 1.4585e-03, Time: 40.35 secs\n",
      "IC: 9.5825e-04,  PDE: 5.0028e-04\n",
      "Iter 47000, Total: 1.3934e-03, Time: 40.26 secs\n",
      "IC: 9.4802e-04,  PDE: 4.4535e-04\n",
      "Iter 48000, Total: 1.3137e-03, Time: 39.29 secs\n",
      "IC: 9.2684e-04,  PDE: 3.8684e-04\n",
      "Iter 49000, Total: 1.2700e-03, Time: 33.82 secs\n",
      "IC: 9.1573e-04,  PDE: 3.5430e-04\n",
      "Iter 50000, Total: 1.2294e-03, Time: 33.81 secs\n",
      "IC: 9.1181e-04,  PDE: 3.1757e-04\n",
      "Iter  1000, Total: 1.0519e-03, Time: 75.12 secs\n",
      "IC: 7.8136e-04,  PDE: 2.7057e-04\n",
      "Iter  2000, Total: 8.9335e-04, Time: 75.64 secs\n",
      "IC: 6.5270e-04,  PDE: 2.4065e-04\n",
      "Iter  3000, Total: 8.0210e-04, Time: 74.40 secs\n",
      "IC: 5.9600e-04,  PDE: 2.0610e-04\n",
      "Iter  4000, Total: 7.4354e-04, Time: 74.69 secs\n",
      "IC: 5.5551e-04,  PDE: 1.8803e-04\n",
      "Iter  5000, Total: 6.9502e-04, Time: 77.24 secs\n",
      "IC: 5.1238e-04,  PDE: 1.8264e-04\n"
     ]
    }
   ],
   "source": [
    "layers = [2, 32, 32, 32, 32, 32, 32, 32, 1]\n",
    "m = 16\n",
    "ptsIC=np.load('ptsIC.npy')\n",
    "ptsBC=np.load('ptsBC.npy')\n",
    "ptsPDE=np.load('random_data.npy')\n",
    "ptsPDE= np.vstack((ptsPDE,ptsBC))\n",
    "u_IC=np.load('u_IC.npy')\n",
    "model = PhysicsInformedNN(period, m, epsilon, gamma, ptsIC, u_IC, ptsPDE, layers)\n",
    "model.train(50000)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "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",
    "u_pred = griddata(pts_flat, u_pred.flatten(), (T, X), method='cubic')"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Visualizations"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "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_sol = np.real(data['exact_sol']).T"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "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": 12,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "l2: 1.0054961682891415\n",
      "l1: 0.9971310893223265\n",
      "linf: 1.9437844348861908\n"
     ]
    }
   ],
   "source": [
    "print(f'l2: {relative_error_l2(u_pred.T,exact_sol)}')\n",
    "print(f'l1: {relative_error_l1(u_pred.T,exact_sol)}')\n",
    "print(f'linf: {relative_error_linf(u_pred.T,exact_sol)}')"
   ]
  },
  {
   "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
}
