{
 "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": 1,
   "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": 2,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Working on cuda\n"
     ]
    }
   ],
   "source": [
    "# MPS or CUDA or CPU\n",
    "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 = 1\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,y: np.cos(4*np.pi*x) * np.cos(4*np.pi*y)\n",
    "u0_ten = lambda x,y: torch.cos(4*torch.pi*x) * torch.cos(4*torch.pi*y)"
   ]
  },
  {
   "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",
    "       \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, X_PDE, layers, epsilon, alpha):\n",
    "        # Prepare the periodic layer\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",
    "        # 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",
    "        self.y_PDE = torch.tensor(X_PDE[:, 2:3], 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",
    "        self.eta = torch.tensor(1.0).float().to(device)\n",
    "       \n",
    "    \n",
    "        layers[0] = int(4 * m + 3)\n",
    "        self.layers = layers\n",
    "        # equation related parameters\n",
    "        self.epsilon = torch.tensor(epsilon).float().to(device)\n",
    "        self.alpha = torch.tensor(alpha).float().to(device)\n",
    "       \n",
    "    \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",
    "        self.optimizer_LW_PDE = torch.optim.Adam(self.LW_PDE.parameters(), lr = 5e-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",
    "       \n",
    "        self.iter = 0\n",
    "    # update alpha\n",
    "    def NN_set_alpha(self, alpha):\n",
    "        self.alpha = torch.tensor(alpha).float().to(device)\n",
    "    # update iteration number\n",
    "    def NN_reset_iter(self):\n",
    "        self.iter = 0\n",
    "    # evaluater neural network\n",
    "    # with transformation to include the initial condition\n",
    "    def NN_eval(self, t, x, y):  \n",
    "        x_trans = torch.matmul(x, self.ms)\n",
    "        y_trans = torch.matmul(y, self.ms)\n",
    "        uNN = self.dnn(torch.cat([t, torch.ones_like(x), torch.cos(x_trans), torch.sin(x_trans), torch.ones_like(y), torch.cos(y_trans), torch.sin(y_trans)], dim = 1))\n",
    "        u0_torch = u0_ten(x,y)\n",
    "        u = u0_torch*torch.exp(-t) + t * uNN\n",
    "        return u\n",
    "    # compute the PDE\n",
    "    def pde_eval(self, t, x, y):\n",
    "        \"\"\" The pytorch autograd version of calculating residual \"\"\"\n",
    "        u = self.NN_eval(t, x,y)\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",
    "        u_y  = torch.autograd.grad(u,   y, 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",
    "        u_yy = torch.autograd.grad(u_y, y, grad_outputs = torch.ones_like(u), retain_graph = True, create_graph=True)[0]\n",
    "\n",
    "        Eq  = u_t - self.epsilon * (u_xx+u_yy) + self.alpha * (torch.pow(u, 3.0) - u)       \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 PDE loss\n",
    "        pde_pred = self.pde_eval(self.t_PDE, self.x_PDE,self.y_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_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",
    "            print('Iter %5d, Total: %10.4e, PDE: %10.4e' % (self.iter, loss.item(), loss_PDE.item()))\n",
    "        return loss\n",
    "    #\n",
    "    def train(self, nIter):\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 PDE loss\n",
    "            pde_pred = self.pde_eval(self.t_PDE, self.x_PDE,self.y_PDE)\n",
    "            \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_PDE\n",
    "                # Backward and optimize\n",
    "            self.optimizer_Adam.zero_grad()\n",
    "            self.optimizer_LW_PDE.zero_grad()\n",
    "            loss.backward()\n",
    "            self.optimizer_Adam.step() \n",
    "            \n",
    "           \n",
    "            self.LW_PDE[0].grad.data = -self.LW_PDE[0].grad.data\n",
    "           \n",
    "            self.optimizer_LW_PDE.step()\n",
    "                # output the progress\n",
    "            if (epoch + 1) % 1000 == 0:\n",
    "                print('Iter %5d,  PDE: %10.4e' % (epoch + 1, loss_PDE.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",
    "                # change the learning rate\n",
    "                self.scheduler.step()\n",
    "       # self.optimizer_LBFGS.step(self.loss_func) \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",
    "        y = torch.tensor(X[:, 2:3], requires_grad=True).float().to(device)\n",
    "        self.dnn.eval()\n",
    "        u = self.NN_eval(t, x,y)\n",
    "        u = u.detach().cpu().numpy()\n",
    "        return u\n",
    "\n",
    "## Configurations"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_points_for_training(tlo, thi, xlo, xhi, N_PDE):\n",
    "    # for collocation pts, (t, x) \\in (0, T)x\\Omega\n",
    "    pts_rand = lhs(2, N_PDE)\n",
    "    t_PDE = tlo + (thi - tlo) * pts_rand[:, 0:1]\n",
    "    x_PDE = xlo + (xhi - xlo) * pts_rand[:, 1:2]\n",
    "    ptsPDE = np.hstack((t_PDE, x_PDE))\n",
    "    return ptsPDE"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "N_IC = 64\n",
    "N_BC = 400\n",
    "N_PDE = 1000\n",
    "Nt = 100\n",
    "ylo = -1\n",
    "yhi =1\n",
    "# Latin Hypercube for collocation points\n",
    "# Latin Hypercube Sampling\n",
    "Pt = lhs(1, Nt)             # Nt samples in 1D (time)\n",
    "t_samples = tlo + (thi - tlo) * Pt[:, 0]   # shape (Nt,)\n",
    "\n",
    "# Prepare arrays to store collocation points\n",
    "tcl_list = []\n",
    "xcl_list = []\n",
    "ycl_list = []\n",
    "\n",
    "for t in t_samples:\n",
    "    Pxy = lhs(2, N_PDE)     # new spatial points for this t\n",
    "    x_samples = xlo + (xhi - xlo) * Pxy[:, 0]\n",
    "    y_samples = ylo + (yhi - ylo) * Pxy[:, 1]\n",
    "\n",
    "    # Append t repeated for this batch\n",
    "    tcl_list.append(np.full((N_PDE, 1), t))\n",
    "    xcl_list.append(x_samples[:, None])\n",
    "    ycl_list.append(y_samples[:, None])\n",
    "\n",
    "# Stack all points\n",
    "tcl = np.vstack(tcl_list)\n",
    "xcl = np.vstack(xcl_list)\n",
    "ycl = np.vstack(ycl_list)\n",
    "# tcl = np.repeat(tcl, N_PDE, axis=0) \n",
    "ptsCL = np.hstack((tcl, xcl, ycl))\n",
    "\n",
    "# # Boundary condition points\n",
    "# # IC: (x, y) ∈ [0, L]^2\n",
    "xIC = np.linspace(xlo, xhi, N_IC)\n",
    "yIC = np.linspace(ylo, yhi, N_IC)\n",
    "\n",
    "# Make a meshgrid\n",
    "xIC, yIC = np.meshgrid(xIC, yIC)\n",
    "\n",
    "# Flatten them\n",
    "xIC = xIC.flatten()[:, None]\n",
    "yIC = yIC.flatten()[:, None]\n",
    "\n",
    "tIC = np.zeros_like(xIC)\n",
    "ptsIC = np.hstack((tIC, xIC, yIC))\n",
    "\n",
    "Pt = lhs(2, N_BC)\n",
    "tPBC1 = tlo + (thi - tlo) * Pt[:, 0]\n",
    "tPBC1 = tPBC1[:, None] \n",
    "xPBC1 = np.zeros_like(tPBC1)\n",
    "yPBC1 =  ylo + (yhi - ylo) * Pt[:, 1]\n",
    "yPBC1 = yPBC1[:, None] \n",
    "ptsPBC1 = np.hstack((tPBC1, xPBC1, yPBC1))\n",
    "# PBC2 on (t, x, 0) and (t, x, 2pi) for (t, x) ∈ [0, T] x [0, 2pi]\n",
    "Pt = lhs(2, N_BC)\n",
    "tPBC2 = tlo + (thi - tlo) * Pt[:, 0]\n",
    "tPBC2 = tPBC2[:, None] \n",
    "xPBC2 = xlo + (xhi - xlo) * Pt[:, 1]\n",
    "xPBC2 = xPBC2[:, None] \n",
    "yPBC2 = np.zeros_like(tPBC2)\n",
    "ptsPBC2 = np.hstack((tPBC2, xPBC2, yPBC2))\n",
    "\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Training"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Starting with Adam\n",
      "Iter  1000,  PDE: 5.4925e-03\n",
      "For PDE, min LW: 1.3033e+00, max LW: 7.1839e+00\n",
      "Iter  2000,  PDE: 2.4985e-03\n",
      "For PDE, min LW: 1.3482e+00, max LW: 1.1298e+01\n",
      "Iter  3000,  PDE: 5.3184e-03\n",
      "For PDE, min LW: 1.3656e+00, max LW: 1.6146e+01\n",
      "Iter  4000,  PDE: 2.3356e-03\n",
      "For PDE, min LW: 1.3831e+00, max LW: 1.9835e+01\n",
      "Iter  5000,  PDE: 4.6527e-03\n",
      "For PDE, min LW: 1.3997e+00, max LW: 2.3786e+01\n",
      "Iter  6000,  PDE: 2.5646e-03\n",
      "For PDE, min LW: 1.4199e+00, max LW: 2.7397e+01\n",
      "Iter  7000,  PDE: 2.0320e-03\n",
      "For PDE, min LW: 1.4675e+00, max LW: 3.1281e+01\n",
      "Iter  8000,  PDE: 2.4834e-03\n",
      "For PDE, min LW: 1.5010e+00, max LW: 3.5075e+01\n",
      "Iter  9000,  PDE: 2.8155e-03\n",
      "For PDE, min LW: 1.5350e+00, max LW: 3.8944e+01\n",
      "Iter 10000,  PDE: 2.1761e-03\n",
      "For PDE, min LW: 1.5752e+00, max LW: 4.3038e+01\n",
      "Iter 11000,  PDE: 2.3867e-03\n",
      "For PDE, min LW: 1.6270e+00, max LW: 4.7122e+01\n",
      "Iter 12000,  PDE: 1.1612e-02\n",
      "For PDE, min LW: 1.6941e+00, max LW: 5.0809e+01\n",
      "Iter 13000,  PDE: 2.5722e-03\n",
      "For PDE, min LW: 1.7708e+00, max LW: 5.4062e+01\n",
      "Iter 14000,  PDE: 2.2466e-03\n",
      "For PDE, min LW: 1.8248e+00, max LW: 5.6922e+01\n",
      "Iter 15000,  PDE: 2.9297e-02\n",
      "For PDE, min LW: 1.9175e+00, max LW: 5.9608e+01\n",
      "Iter 16000,  PDE: 3.0219e-03\n",
      "For PDE, min LW: 1.9736e+00, max LW: 6.2129e+01\n",
      "Iter 17000,  PDE: 1.9532e-02\n",
      "For PDE, min LW: 2.0228e+00, max LW: 6.4450e+01\n",
      "Iter 18000,  PDE: 2.2519e-03\n",
      "For PDE, min LW: 2.0683e+00, max LW: 6.7406e+01\n",
      "Iter 19000,  PDE: 2.2746e-03\n",
      "For PDE, min LW: 2.1270e+00, max LW: 7.1151e+01\n",
      "Iter 20000,  PDE: 2.2388e-03\n",
      "For PDE, min LW: 2.1746e+00, max LW: 7.4757e+01\n",
      "Iter 21000,  PDE: 1.3998e-01\n",
      "For PDE, min LW: 2.2230e+00, max LW: 7.8465e+01\n",
      "Iter 22000,  PDE: 2.8689e-02\n",
      "For PDE, min LW: 2.2643e+00, max LW: 8.2132e+01\n",
      "Iter 23000,  PDE: 2.2466e-03\n",
      "For PDE, min LW: 2.3067e+00, max LW: 8.5742e+01\n",
      "Iter 24000,  PDE: 2.3818e-03\n",
      "For PDE, min LW: 2.3543e+00, max LW: 8.9293e+01\n",
      "Iter 25000,  PDE: 2.2513e-03\n",
      "For PDE, min LW: 2.4115e+00, max LW: 9.2756e+01\n",
      "Iter 26000,  PDE: 1.3060e-01\n",
      "For PDE, min LW: 2.4699e+00, max LW: 9.5992e+01\n",
      "Iter 27000,  PDE: 5.4654e-03\n",
      "For PDE, min LW: 2.5223e+00, max LW: 9.9057e+01\n",
      "Iter 28000,  PDE: 2.2012e-03\n",
      "For PDE, min LW: 2.5550e+00, max LW: 1.0194e+02\n",
      "Iter 29000,  PDE: 4.4310e-03\n",
      "For PDE, min LW: 2.5807e+00, max LW: 1.0462e+02\n",
      "Iter 30000,  PDE: 2.7653e-03\n",
      "For PDE, min LW: 2.6099e+00, max LW: 1.0715e+02\n",
      "Iter 31000,  PDE: 7.4928e-02\n",
      "For PDE, min LW: 2.6314e+00, max LW: 1.1028e+02\n",
      "Iter 32000,  PDE: 2.1854e-03\n",
      "For PDE, min LW: 2.6548e+00, max LW: 1.1375e+02\n",
      "Iter 33000,  PDE: 2.2110e-03\n",
      "For PDE, min LW: 2.6769e+00, max LW: 1.1705e+02\n",
      "Iter 34000,  PDE: 2.2060e-03\n",
      "For PDE, min LW: 2.7010e+00, max LW: 1.2024e+02\n",
      "Iter 35000,  PDE: 3.5799e-03\n",
      "For PDE, min LW: 2.7192e+00, max LW: 1.2329e+02\n",
      "Iter 36000,  PDE: 4.6513e-03\n",
      "For PDE, min LW: 2.7433e+00, max LW: 1.2617e+02\n",
      "Iter 37000,  PDE: 7.5690e-03\n",
      "For PDE, min LW: 2.7565e+00, max LW: 1.2893e+02\n",
      "Iter 38000,  PDE: 3.7159e-03\n",
      "For PDE, min LW: 2.7636e+00, max LW: 1.3153e+02\n",
      "Iter 39000,  PDE: 2.4087e-01\n",
      "For PDE, min LW: 2.7707e+00, max LW: 1.3401e+02\n",
      "Iter 40000,  PDE: 2.0673e-03\n",
      "For PDE, min LW: 2.7782e+00, max LW: 1.3633e+02\n",
      "Iter 41000,  PDE: 2.1236e-03\n",
      "For PDE, min LW: 2.7865e+00, max LW: 1.3857e+02\n",
      "Iter 42000,  PDE: 3.6092e-03\n",
      "For PDE, min LW: 2.7946e+00, max LW: 1.4068e+02\n",
      "Iter 43000,  PDE: 2.0595e-03\n",
      "For PDE, min LW: 2.8038e+00, max LW: 1.4268e+02\n",
      "Iter 44000,  PDE: 1.9625e-03\n",
      "For PDE, min LW: 2.8121e+00, max LW: 1.4461e+02\n",
      "Iter 45000,  PDE: 2.0437e-03\n",
      "For PDE, min LW: 2.8212e+00, max LW: 1.4646e+02\n",
      "Iter 46000,  PDE: 2.0024e-03\n",
      "For PDE, min LW: 2.8292e+00, max LW: 1.4827e+02\n",
      "Iter 47000,  PDE: 1.9943e-03\n",
      "For PDE, min LW: 2.8381e+00, max LW: 1.5000e+02\n",
      "Iter 48000,  PDE: 1.5560e-02\n",
      "For PDE, min LW: 2.8455e+00, max LW: 1.5168e+02\n",
      "Iter 49000,  PDE: 2.2756e-03\n",
      "For PDE, min LW: 2.8540e+00, max LW: 1.5330e+02\n",
      "Iter 50000,  PDE: 1.9529e-03\n",
      "For PDE, min LW: 2.8618e+00, max LW: 1.5491e+02\n",
      "Starting with L-BFGS\n",
      "Iter  1000, Total: 4.7547e-04, PDE: 4.7547e-04\n",
      "Iter  2000, Total: 2.3626e-04, PDE: 2.3626e-04\n",
      "Iter  3000, Total: 1.3682e-04, PDE: 1.3682e-04\n",
      "Iter  4000, Total: 8.9790e-05, PDE: 8.9790e-05\n",
      "Iter  5000, Total: 6.4188e-05, PDE: 6.4188e-05\n"
     ]
    }
   ],
   "source": [
    "layers = [2, 32, 32, 32, 32, 32, 32, 32, 1]\n",
    "m = 16\n",
    "\n",
    "ptsPDE = np.vstack((ptsIC,ptsCL))\n",
    "ptsPDE = np.vstack((ptsPBC1,ptsPDE))\n",
    "ptsPDE = np.vstack((ptsPBC2,ptsPDE))\n",
    "model = PhysicsInformedNN(period,m,ptsPDE,layers, epsilon, gamma)\n",
    "model.train(50000)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "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",
    "y = np.linspace(xlo, xhi, 201)\n",
    "X, Y = np.meshgrid(x, y)\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')"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Visualizations"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [],
   "source": [
    "torch.save({\n",
    "    'dnn_state_dict': model.dnn.state_dict(),\n",
    "    'LW_PDE_state_dict': [w.detach().cpu() for w in model.LW_PDE],  # ParameterList\n",
    "    'iter': model.iter,\n",
    "    'layers': model.layers,\n",
    "    'alpha': model.alpha,\n",
    "    'epsilon': model.epsilon,\n",
    "    # Optional: save optimizer states if you want to resume training\n",
    "    'optimizer_dnn_state_dict': model.optimizer_Adam.state_dict(),\n",
    "    'optimizer_LW_PDE_state_dict': model.optimizer_LW_PDE.state_dict(),}, 'model_AC_case1_2d.pth')"
   ]
  },
  {
   "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
}
