{
 "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",
    "\n",
    "lb = np.array([-1, -1, 0])  \n",
    "ub = np.array([1, 1, 1])    \n",
    "\n",
    "# u = (sin(pi x)cos(pi y) + 0.1sin(10pi x)cos(10pi y)) * exp(-t)\n",
    "usol = (\n",
    "    np.sin(np.pi * X) * np.cos(np.pi * Y) +\n",
    "    0.1 * np.sin(10 * np.pi * X) * np.cos(10 * np.pi * Y)\n",
    ") * np.exp(-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=True)\n",
    "    \n",
    "        self.a_layer = nn.Linear(layers[1], layers[2], bias=False)\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",
    "         \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",
    "        # S(x, y, t) = -2u(x, y, t) + 2ε^2π^2*sin(πx)cos(πy)e^{-t} \n",
    "        #             + 20ε^2π^2*sin(10πx)cos(10πy)e^{-t} \n",
    "        #             + [sin(πx)cos(πy) + 0.1sin(10πx)cos(10πy)]^3 e^{-3t}\n",
    "        epsilon = 0.1  \n",
    "        pi = np.pi\n",
    "        s1 = torch.sin(pi * x) * torch.cos(pi * y)\n",
    "        s2 = 0.1 * torch.sin(10 * pi * x) * torch.cos(10 * pi * y)\n",
    "        u = (s1 + s2) * torch.exp(-t)\n",
    "\n",
    "        S = (-2) * u \\\n",
    "            + 2 * epsilon**2 * pi**2 * s1 * torch.exp(-t) \\\n",
    "            + 20 * epsilon**2 * pi**2 * (0.1 * torch.sin(10 * pi * x) * torch.cos(10 * pi * y)) * torch.exp(-t) \\\n",
    "            + (s1 + s2) ** 3 * torch.exp(-3*t)\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)   # 你需要实现calc_source方法，参考上文推导\n",
    "\n",
    "        epsilon = 0.1 \n",
    "        \n",
    "        # Allen-Cahn PDE residual\n",
    "        pde_f = u_t - epsilon**2 * (u_xx + u_yy) + (u**3 - 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": 6,
   "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",
    "  \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": null,
   "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=True)\n",
      "  (a_layer): Linear(in_features=1000, out_features=1, bias=False)\n",
      ")\n",
      "Iteration 0, loss : 1.07182471e+04\n",
      "Iteration 1000, loss : 1.67088985e-01\n",
      "Iteration 2000, loss : 1.60785824e-01\n",
      "Iteration 3000, loss : 1.54734924e-01\n",
      "Iteration 4000, loss : 1.49340659e-01\n",
      "Iteration 5000, loss : 1.43878505e-01\n",
      "Iteration 6000, loss : 1.38620898e-01\n",
      "Iteration 7000, loss : 1.33685306e-01\n",
      "Iteration 8000, loss : 1.29268482e-01\n",
      "Iteration 9000, loss : 1.24999329e-01\n",
      "Iteration 10000, loss : 1.20893702e-01\n",
      "Iteration 11000, loss : 1.17245167e-01\n",
      "Iteration 12000, loss : 1.13363236e-01\n",
      "Iteration 13000, loss : 1.10061452e-01\n",
      "Iteration 14000, loss : 1.06662706e-01\n",
      "Iteration 15000, loss : 1.03606261e-01\n",
      "Iteration 16000, loss : 1.00537747e-01\n",
      "Iteration 17000, loss : 9.76668224e-02\n",
      "Iteration 18000, loss : 9.51632708e-02\n",
      "Iteration 19000, loss : 9.27069187e-02\n",
      "Iteration 20000, loss : 9.04422104e-02\n",
      "Iteration 21000, loss : 8.81364644e-02\n",
      "Iteration 22000, loss : 8.61924887e-02\n",
      "Iteration 23000, loss : 8.42789635e-02\n",
      "Iteration 24000, loss : 8.23935419e-02\n",
      "Iteration 25000, loss : 8.07406008e-02\n",
      "Iteration 26000, loss : 7.91630149e-02\n",
      "Iteration 27000, loss : 7.75767267e-02\n",
      "Iteration 28000, loss : 7.60806352e-02\n",
      "Iteration 29000, loss : 7.46410936e-02\n",
      "Iteration 30000, loss : 7.33404011e-02\n",
      "Iteration 31000, loss : 7.19715357e-02\n",
      "Iteration 32000, loss : 7.06232339e-02\n",
      "Iteration 33000, loss : 6.94589317e-02\n",
      "Iteration 34000, loss : 6.82987422e-02\n",
      "Iteration 35000, loss : 6.71390966e-02\n",
      "Iteration 36000, loss : 6.60885274e-02\n",
      "Iteration 37000, loss : 6.51202947e-02\n",
      "Iteration 38000, loss : 6.41830415e-02\n",
      "Iteration 39000, loss : 6.33061677e-02\n",
      "Iteration 40000, loss : 6.24666661e-02\n",
      "Iteration 41000, loss : 6.13307543e-02\n",
      "Iteration 42000, loss : 6.02356866e-02\n",
      "Iteration 43000, loss : 5.92039600e-02\n",
      "Iteration 44000, loss : 5.81135489e-02\n",
      "Iteration 45000, loss : 5.70895262e-02\n",
      "Iteration 46000, loss : 5.62064461e-02\n",
      "Iteration 47000, loss : 5.53536117e-02\n",
      "Iteration 48000, loss : 5.44489324e-02\n",
      "Iteration 49000, loss : 5.36826327e-02\n"
     ]
    },
    {
     "data": {
      "image/png": "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f8CAAfr+97+vd955R5WVlZKk+Ph47dmzR59++mnAc0VGRsput2v9+vU6evToSdXj8OHD6tOnT5v9r7/+upKTk3X99df79oWHh+unP/2pqqurtWHDBr/yM2bM8DvPpEmTJEmff/55u9d2Op266qqr9Je//EXGGEne+7Jy5UpdffXVio6OPqnv8nWzZ89WaGio377IyEjfz42NjTp8+LCGDRum+Ph47dix47jnjImJ0Q9+8APftt1u1/jx4zv8noGuXVFRobKyMk2ePFmff/65KioqTuQrndC5q6qqVFZWpkmTJqm2tlYfffSR79iNN96od999V5999plv37PPPqvU1FRNnjxZkrR27VqVl5fr+uuvV1lZme8VGhqqzMxMrVu3rs31f/zjH7dbtz59+qisrOy0vh9wOghCQA906NAh1dbW6rzzzmtz7Pzzz5fH41FRUZEk6Z577lF5ebnOPfdcjRo1Sj//+c/1/vvv+8o7HA4tWrRIb7zxhpKSknTJJZfogQceUHFx8QnVpSWEtPbFF1/onHPOaTNY+Pzzz/cdb23w4MF+2y2h6HjB7MYbb9SBAwf097//XZL01ltvqaSkRDNnzjyhundk6NChbfbV1dVp/vz5Sk1NlcPhUGJiovr166fy8vITCiODBg1qsxZSnz59TiiAbtq0SVlZWYqOjlZ8fLz69evnW7LgdIPQnj17NH36dDmdTsXFxalfv36+wNb63DNmzJDD4dCzzz7rO/bqq6/qhhtu8H2vlsB96aWXql+/fn6vN998U6WlpX7XDgsL843BCsQY02vWj8KZiSAE9HKXXHKJPvvsMy1fvlwjR47UE088oYsuukhPPPGEr8xtt92mTz75RAsXLlRERIR+/etf6/zzz9d7773X4bkTEhJOuhUpkK+3vLQIFLJay87OVlJSkv70pz9Jkv70pz8pOTk54ODtk9W6laTFT37yE91333269tpr9fzzz+vNN9/U2rVrlZCQII/Hc9xznur3/OyzzzRt2jSVlZVp8eLFeu2117R27Vr97Gc/k6QTunZ7ysvLNXnyZO3atUv33HOPXnnlFa1du1aLFi1qc+4+ffrou9/9ri8IrVq1Si6Xy6+Vq6X8M888o7Vr17Z5vfTSS37XdzgcHc6uO3r0qBITE0/5+wGni8HSQA/Ur18/RUVF6eOPP25z7KOPPlJISIhSU1N9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UVdccYVSUlJks9m0evVqv+PGGM2fP18DBgxQZGSksrKy9Omnn/qVOXLkiG644QbFxcUpPj5eP/zhD1VdXe1X5v3339ekSZMUERGh1NRUPfDAA23q8sILL2j48OGKiIjQqFGj9Prrr3f69w2GhQsXaty4cYqNjVX//v119dVX6+OPP/YrU19fr7lz5yohIUExMTH63ve+p5KSEr8yBw4c0OWXX66oqCj1799fP//5z9XU1ORXZv369brooovkcDg0bNgwrVixok19zuTfh6VLl+qCCy7wLRg3YcIEvfHGG77j3Oeucf/998tms+m2227z7eNen767775bNpvN7zV8+HDf8V55jw263XPPPWfsdrtZvny52bNnj5k9e7aJj483JSUlwa5aj/D666+bX/3qV+avf/2rkWRefPFFv+P333+/cTqdZvXq1WbXrl3myiuvNEOHDjV1dXW+Mt/+9rdNRkaG+ec//2n+/ve/m2HDhpnrr7/ed7yiosIkJSWZG264wezevdv85S9/MZGRkeaxxx7zldm0aZMJDQ01DzzwgNm7d6/57//+bxMeHm4++OCDLr8HXS07O9s8+eSTZvfu3Wbnzp3mO9/5jhk8eLCprq72lbnllltMamqqKSgoMNu2bTPf+MY3zMUXX+w73tTUZEaOHGmysrLMe++9Z15//XWTmJho7rrrLl+Zzz//3ERFRZm8vDyzd+9e88gjj5jQ0FCzZs0aX5kz/ffh5ZdfNq+99pr55JNPzMcff2z+67/+y4SHh5vdu3cbY7jPXWHLli0mLS3NXHDBBebWW2/17eden74FCxaYESNGmIMHD/pehw4d8h3vjfeYIBQE48ePN3PnzvVtu91uk5KSYhYuXBjEWvVMXw9CHo/HJCcnmwcffNC3r7y83DgcDvOXv/zFGGPM3r17jSSzdetWX5k33njD2Gw28+WXXxpjjPnjH/9o+vTpY1wul6/ML3/5S3Peeef5tq+99lpz+eWX+9UnMzPT/OhHP+rU79gTlJaWGklmw4YNxhjvPQ0PDzcvvPCCr8yHH35oJJnNmzcbY7yBNSQkxBQXF/vKLF261MTFxfnu6y9+8QszYsQIv2vNmDHDZGdn+7at+PvQp08f88QTT3Cfu0BVVZU555xzzNq1a83kyZN9QYh73TkWLFhgMjIyAh7rrfeYrrFu1tDQoO3btysrK8u3LyQkRFlZWdq8eXMQa9Y7FBYWqri42O/+OZ1OZWZm+u7f5s2bFR8fr7Fjx/rKZGVlKSQkRO+++66vzCWXXCK73e4rk52drY8//lhHjx71lWl9nZYyZ+J/p4qKCklS3759JUnbt29XY2Oj3/cfPny4Bg8e7HefR40apaSkJF+Z7OxsVVZWas+ePb4yHd1Dq/0+uN1uPffcc6qpqdGECRO4z11g7ty5uvzyy9vcD+515/n000+VkpKis846SzfccIMOHDggqffeY4JQNysrK5Pb7fb7QyBJSUlJKi4uDlKteo+We9TR/SsuLlb//v39joeFhalv375+ZQKdo/U12itzpv138ng8uu222/TNb35TI0eOlOT97na7XfHx8X5lv36fT/UeVlZWqq6uzjK/Dx988IFiYmLkcDh0yy236MUXX1R6ejr3uZM999xz2rFjhxYuXNjmGPe6c2RmZmrFihVas2aNli5dqsLCQk2aNElVVVW99h7z9HnA4ubOnavdu3frnXfeCXZVzljnnXeedu7cqYqKCq1atUq5ubnasGFDsKt1RikqKtKtt96qtWvXKiIiItjVOWPl5OT4fr7ggguUmZmpIUOG6Pnnn1dkZGQQa3bqaBHqZomJiQoNDW0zir6kpETJyclBqlXv0XKPOrp/ycnJKi0t9Tve1NSkI0eO+JUJdI7W12ivzJn032nevHl69dVXtW7dOg0aNMi3Pzk5WQ0NDSovL/cr//X7fKr3MC4uTpGRkZb5fbDb7Ro2bJjGjBmjhQsXKiMjQw8//DD3uRNt375dpaWluuiiixQWFqawsDBt2LBBf/jDHxQWFqakpCTudReIj4/Xueeeq3379vXaP88EoW5mt9s1ZswYFRQU+PZ5PB4VFBRowoQJQaxZ7zB06FAlJyf73b/Kykq9++67vvs3YcIElZeXa/v27b4yb7/9tjwejzIzM31lNm7cqMbGRl+ZtWvX6rzzzlOfPn18ZVpfp6XMmfDfyRijefPm6cUXX9Tbb7+toUOH+h0fM2aMwsPD/b7/xx9/rAMHDvjd5w8++MAvdK5du1ZxcXFKT0/3lenoHlr198Hj8cjlcnGfO9G0adP0wQcfaOfOnb7X2LFjdcMNN/h+5l53vurqan322WcaMGBA7/3zfNLDq3HannvuOeNwOMyKFSvM3r17zc0332zi4+P9RtFbWVVVlXnvvffMe++9ZySZxYsXm/fee8988cUXxhjv9Pn4+Hjz0ksvmffff99cddVVAafPX3jhhebdd98177zzjjnnnHP8ps+Xl5ebpKQkM3PmTLN7927z3HPPmaioqDbT58PCwszvf/978+GHH5oFCxacMdPnf/zjHxun02nWr1/vNw22trbWV+aWW24xgwcPNm+//bbZtm2bmTBhgpkwYYLveMs02Msuu8zs3LnTrFmzxvTr1y/gNNif//zn5sMPPzT5+fkBp8Geyb8Pd955p9mwYYMpLCw077//vrnzzjuNzWYzb775pjGG+9yVWs8aM4Z73Rluv/12s379elNYWGg2bdpksrKyTGJioiktLTXG9M57TBAKkkceecQMHjzY2O12M378ePPPf/4z2FXqMdatW2cktXnl5uYaY7xT6H/961+bpKQk43A4zLRp08zHH3/sd47Dhw+b66+/3sTExJi4uDgza9YsU1VV5Vdm165dZuLEicbhcJiBAwea+++/v01dnn/+eXPuuecau91uRowYYV577bUu+97dKdD9lWSefPJJX5m6ujozZ84c06dPHxMVFWWmT59uDh486Hee/fv3m5ycHBMZGWkSExPN7bffbhobG/3KrFu3zowePdrY7XZz1lln+V2jxZn8+/Af//EfZsiQIcZut5t+/fqZadOm+UKQMdznrvT1IMS9Pn0zZswwAwYMMHa73QwcONDMmDHD7Nu3z3e8N95jmzHGnHw7EgAAQO/HGCEAAGBZBCEAAGBZBCEAAGBZBCEAAGBZBCEAAGBZBCEAAGBZBCEAAGBZBCEA6EBaWpqWLFkS7GoA6CIEIQA9xk033aSrr75akjRlyhTddttt3XbtFStWKD4+vs3+rVu36uabb+62egDoXmHBrgAAdKWGhgbZ7fZT/ny/fv06sTYAehpahAD0ODfddJM2bNighx9+WDabTTabTfv375ck7d69Wzk5OYqJiVFSUpJmzpypsrIy32enTJmiefPm6bbbblNiYqKys7MlSYsXL9aoUaMUHR2t1NRUzZkzR9XV1ZKk9evXa9asWaqoqPBd7+6775bUtmvswIEDuuqqqxQTE6O4uDhde+21Kikp8R2/++67NXr0aD3zzDNKS0uT0+nUddddp6qqqq69aQBOCUEIQI/z8MMPa8KECZo9e7YOHjyogwcPKjU1VeXl5br00kt14YUXatu2bVqzZo1KSkp07bXX+n3+qaeekt1u16ZNm7Rs2TJJUkhIiP7whz9oz549euqpp/T222/rF7/4hSTp4osv1pIlSxQXF+e73h133NGmXh6PR1dddZWOHDmiDRs2aO3atfr88881Y8YMv3KfffaZVq9erVdffVWvvvqqNmzYoPvvv7+L7haA00HXGIAex+l0ym63KyoqSsnJyb79jz76qC688EL97ne/8+1bvny5UlNT9cknn+jcc8+VJJ1zzjl64IEH/M7ZerxRWlqa7r33Xt1yyy364x//KLvdLqfTKZvN5ne9rysoKNAHH3ygwsJCpaamSpKefvppjRgxQlu3btW4ceMkeQPTihUrFBsbK0maOXOmCgoKdN99953ejQHQ6WgRAtBr7Nq1S+vWrVNMTIzvNXz4cEneVpgWY8aMafPZt956S9OmTdPAgQMVGxurmTNn6vDhw6qtrT3h63/44YdKTU31hSBJSk9PV3x8vD788EPfvrS0NF8IkqQBAwaotLT0pL4rgO5BixCAXqO6ulpXXHGFFi1a1ObYgAEDfD9HR0f7Hdu/f7+++93v6sc//rHuu+8+9e3bV++8845++MMfqqGhQVFRUZ1az/DwcL9tm80mj8fTqdcA0DkIQgB6JLvdLrfb7bfvoosu0v/93/8pLS1NYWEn/tfX9u3b5fF49NBDDykkxNsQ/vzzzx/3el93/vnnq6ioSEVFRb5Wob1796q8vFzp6eknXB8APQddYwB6pLS0NL377rvav3+/ysrK5PF4NHfuXB05ckTXX3+9tm7dqs8++0x/+9vfNGvWrA5DzLBhw9TY2KhHHnlEn3/+uZ555hnfIOrW16uurlZBQYHKysoCdpllZWVp1KhRuuGGG7Rjxw5t2bJFN954oyZPnqyxY8d2+j0A0PUIQgB6pDvuuEOhoaFKT09Xv379dODAAaWkpGjTpk1yu9267LLLNGrUKN12222Kj4/3tfQEkpGRocWLF2vRokUaOXKknn32WS1cuNCvzMUXX6xbbrlFM2bMUL9+/doMtpa8XVwvvfSS+vTpo0suuURZWVk666yztHLlyk7//gC6h80YY4JdCQAAgGCgRQgAAFgWQQgAAFgWQQgAAFgWQQgAAFgWQQgAAFgWQQgAAFgWQQgAAFgWQQgAAFgWQQgAAFgWQQgAAFgWQQgAAFgWQQgAAFjW/wPVQcCeGjE/bwAAAABJRU5ErkJggg==",
      "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]) #3 hidden layers\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 = 50000        \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": 10,
   "id": "c594ff78",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Total grad norm: 1.8618e-03\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
}
