{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 35,
   "id": "a68eebe4-b99b-4e8b-bd09-97aa103f17cf",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "cuda\n",
      "mark\n",
      "checkpoint directory created: ./model\n",
      "saving model version 0.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "pde loss: 1.25e-02 | bc loss: 6.00e-05 | init loss : 1.71e-01 : 100%|█| 1/1 [00:00<00:00,  2.71it/s]\n",
      "pde loss: 9.07e-03 | bc loss: 5.12e-05 | init loss : 1.69e-01 : 100%|█| 1/1 [00:00<00:00,  2.72it/s]\n",
      "pde loss: 8.77e-03 | bc loss: 5.03e-05 | init loss : 1.69e-01 : 100%|█| 1/1 [00:00<00:00,  2.74it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "cuda\n",
      "mark\n",
      "checkpoint directory created: ./model\n",
      "saving model version 0.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "pde loss: 8.74e-03 | bc loss: 5.02e-05 | init loss : 2.19e-08 : 100%|█| 1/1 [00:00<00:00,  2.51it/s]\n",
      "pde loss: 5.88e-03 | bc loss: 3.89e-05 | init loss : 1.02e-05 : 100%|█| 1/1 [00:00<00:00,  2.55it/s]\n",
      "pde loss: 5.68e-03 | bc loss: 3.72e-05 | init loss : 1.18e-05 : 100%|█| 1/1 [00:00<00:00,  2.52it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "cuda\n",
      "mark\n",
      "checkpoint directory created: ./model\n",
      "saving model version 0.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "pde loss: 5.67e-03 | bc loss: 3.71e-05 | init loss : 4.22e-08 : 100%|█| 1/1 [00:00<00:00,  2.53it/s]\n",
      "pde loss: 3.44e-03 | bc loss: 2.75e-05 | init loss : 1.07e-05 : 100%|█| 1/1 [00:00<00:00,  2.53it/s]\n",
      "pde loss: 3.32e-03 | bc loss: 2.56e-05 | init loss : 1.18e-05 : 100%|█| 1/1 [00:00<00:00,  2.53it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "cuda\n",
      "mark\n",
      "checkpoint directory created: ./model\n",
      "saving model version 0.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "pde loss: 3.32e-03 | bc loss: 2.54e-05 | init loss : 9.88e-08 : 100%|█| 1/1 [00:00<00:00,  2.50it/s]\n",
      "pde loss: 1.70e-03 | bc loss: 1.71e-05 | init loss : 1.11e-05 : 100%|█| 1/1 [00:00<00:00,  2.53it/s]\n",
      "pde loss: 1.65e-03 | bc loss: 1.51e-05 | init loss : 1.16e-05 : 100%|█| 1/1 [00:00<00:00,  2.49it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "cuda\n",
      "mark\n",
      "checkpoint directory created: ./model\n",
      "saving model version 0.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "pde loss: 1.65e-03 | bc loss: 1.49e-05 | init loss : 2.02e-07 : 100%|█| 1/1 [00:00<00:00,  2.54it/s]\n",
      "pde loss: 6.48e-04 | bc loss: 3.61e-06 | init loss : 9.26e-06 : 100%|█| 1/1 [00:00<00:00,  2.53it/s]\n",
      "pde loss: 6.48e-04 | bc loss: 2.71e-06 | init loss : 9.06e-06 : 100%|█| 1/1 [00:00<00:00,  2.55it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "cuda\n",
      "mark\n",
      "checkpoint directory created: ./model\n",
      "saving model version 0.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "pde loss: 6.50e-04 | bc loss: 2.64e-06 | init loss : 4.81e-07 : 100%|█| 1/1 [00:00<00:00,  2.51it/s]\n",
      "pde loss: 3.94e-04 | bc loss: 1.46e-05 | init loss : 7.40e-06 : 100%|█| 1/1 [00:00<00:00,  2.55it/s]\n",
      "pde loss: 3.73e-04 | bc loss: 1.30e-05 | init loss : 6.86e-06 : 100%|█| 1/1 [00:00<00:00,  2.55it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "cuda\n",
      "mark\n",
      "checkpoint directory created: ./model\n",
      "saving model version 0.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "pde loss: 3.72e-04 | bc loss: 1.28e-05 | init loss : 3.41e-07 : 100%|█| 1/1 [00:00<00:00,  2.51it/s]\n",
      "pde loss: 2.16e-05 | bc loss: 1.86e-05 | init loss : 1.20e-05 : 100%|█| 1/1 [00:00<00:00,  2.55it/s]\n",
      "pde loss: 2.58e-05 | bc loss: 1.90e-05 | init loss : 1.06e-05 : 100%|█| 1/1 [00:00<00:00,  2.54it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "cuda\n",
      "mark\n",
      "checkpoint directory created: ./model\n",
      "saving model version 0.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "pde loss: 2.65e-05 | bc loss: 1.90e-05 | init loss : 6.65e-07 : 100%|█| 1/1 [00:00<00:00,  2.53it/s]\n",
      "pde loss: 3.96e-04 | bc loss: 8.55e-05 | init loss : 6.60e-06 : 100%|█| 1/1 [00:00<00:00,  2.55it/s]\n",
      "pde loss: 3.51e-04 | bc loss: 7.88e-05 | init loss : 5.52e-06 : 100%|█| 1/1 [00:00<00:00,  2.53it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "cuda\n",
      "mark\n",
      "checkpoint directory created: ./model\n",
      "saving model version 0.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "pde loss: 3.45e-04 | bc loss: 7.80e-05 | init loss : 6.54e-07 : 100%|█| 1/1 [00:00<00:00,  2.53it/s]\n",
      "pde loss: 3.73e-05 | bc loss: 1.49e-05 | init loss : 1.33e-05 : 100%|█| 1/1 [00:00<00:00,  2.54it/s]\n",
      "pde loss: 3.86e-05 | bc loss: 1.76e-05 | init loss : 1.18e-05 : 100%|█| 1/1 [00:00<00:00,  2.55it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "cuda\n",
      "mark\n",
      "checkpoint directory created: ./model\n",
      "saving model version 0.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "pde loss: 3.89e-05 | bc loss: 1.79e-05 | init loss : 1.29e-06 : 100%|█| 1/1 [00:00<00:00,  2.53it/s]\n",
      "pde loss: 3.60e-04 | bc loss: 9.23e-05 | init loss : 5.16e-06 : 100%|█| 1/1 [00:00<00:00,  2.54it/s]\n",
      "pde loss: 3.22e-04 | bc loss: 8.66e-05 | init loss : 4.34e-06 : 100%|█| 1/1 [00:00<00:00,  2.55it/s]\n"
     ]
    }
   ],
   "source": [
    "from kan import *\n",
    "import torch\n",
    "import matplotlib.pyplot as plt\n",
    "from torch import autograd\n",
    "from tqdm import tqdm\n",
    "\n",
    "\n",
    "\n",
    "# implement Allen Cahn 1D,  2D input (x, t)\n",
    "N_deltat = 10\n",
    "steps = 1 #1000\n",
    "grid = 5\n",
    "\n",
    "for i_deltat in range(N_deltat):\n",
    "\n",
    "    device = torch.device('cuda' if torch.cuda.is_available() else 'cpu')\n",
    "    print(device)\n",
    "\n",
    "    dim = 2\n",
    "    np_i = 51 # number of interior points (along each dimension)\n",
    "    np_t = 51\n",
    "\n",
    "\n",
    "    def batch_jacobian(func, x, create_graph=False):\n",
    "        # x in shape (Batch, Length)\n",
    "        def _func_sum(x):\n",
    "            return func(x).sum(dim=0)\n",
    "        return autograd.functional.jacobian(_func_sum, x, create_graph=create_graph).permute(1,0,2)\n",
    "\n",
    "\n",
    "    print('mark')\n",
    "    # interior\n",
    "    sampling_mode = 'mesh' # 'radnom' or 'mesh'\n",
    "\n",
    "    x_mesh = torch.linspace(-1,1,steps=np_i).to(device)\n",
    "    t_mesh = torch.linspace(0,0.2,steps=np_t).to(device)\n",
    "    X, T = torch.meshgrid(x_mesh, t_mesh, indexing='ij')\n",
    "\n",
    "    if sampling_mode == 'mesh':\n",
    "        #mesh\n",
    "        x_i = torch.stack([X.reshape(-1,), T.reshape(-1,)]).permute(1,0)\n",
    "    else:\n",
    "        #random\n",
    "        x_i = torch.rand((np_i**2*np_t,3))\n",
    "\n",
    "    x_i = x_i.to(device)\n",
    "\n",
    "    # boundary, 4 sides\n",
    "    helper = lambda X, Y: torch.stack([X.reshape(-1,), Y.reshape(-1,)]).permute(1,0)\n",
    "    x_left = helper(X[0,:], T[0,:])\n",
    "    x_right = helper(X[-1,:], T[0,:])\n",
    "\n",
    "    x_init = helper(X[:,0], T[:,0])\n",
    "\n",
    "    def u0_true(x):\n",
    "        x1 = x[:,[0]]\n",
    "        return x1**2 * torch.cos(torch.pi*x1)\n",
    "\n",
    "    pde_losses = []\n",
    "    bc_losses = []\n",
    "    init_losses = []\n",
    "\n",
    "    lamb_r = 1\n",
    "    lamb_b = 1\n",
    "    lamb_i = 100\n",
    "\n",
    "    log = 1\n",
    "    \n",
    "    if i_deltat == 0:\n",
    "        def u0_true(x):\n",
    "            x1 = x[:,[0]]\n",
    "            return x1**2 * torch.cos(torch.pi*x1)\n",
    "            #return torch.cos(torch.pi*x1)\n",
    "\n",
    "        model = KAN(width=[2,5,5,1], grid=grid, k=3, seed=1, device=device)\n",
    "    else:\n",
    "        model_init = KAN(width=[2,5,5,1], grid=grid, k=3, seed=1, device=device)\n",
    "        model_init.load_state_dict(model.state_dict())\n",
    "        \n",
    "        def u0_true(x):\n",
    "            x1 = x[:,[0]]\n",
    "            t = x1 * 0. + 0.1\n",
    "            x = torch.cat([x1, t], dim=1)\n",
    "            return model_init(x)\n",
    "        \n",
    "        #model = KAN(width=[2,5,5,1], grid=10, k=3, seed=1, device=device)\n",
    "        #u0_true = MLP(width=[2,128,128,128,1], act='silu', seed=1, device=device).load_state_dict(model.state_dict())\n",
    "    \n",
    "\n",
    "    def train():\n",
    "        for i in range(3):\n",
    "            #optimizer = LBFGS(model.parameters(), lr=1, history_size=10, line_search_fn=\"strong_wolfe\", tolerance_grad=1e-32, tolerance_change=1e-32, tolerance_ys=1e-32)\n",
    "            if i == 0:\n",
    "                optimizer = torch.optim.Adam(model.parameters(), lr=1e-3)\n",
    "            else:\n",
    "                for g in optimizer.param_groups:\n",
    "                    g['lr'] *= 0.1\n",
    "            #optimizer = torch.optim.LBFGS(model.parameters(), lr=1, history_size=10, line_search_fn=\"strong_wolfe\")\n",
    "\n",
    "            pbar = tqdm(range(steps), desc='description', ncols=100)\n",
    "\n",
    "\n",
    "            for _ in pbar:\n",
    "                def closure():\n",
    "                    global pde_loss, bc_loss, init_loss\n",
    "                    optimizer.zero_grad()\n",
    "                    # interior loss\n",
    "                    #sol = sol_fun(x_i)\n",
    "                    u = model(x_i)\n",
    "                    sol_D1_fun = lambda x: batch_jacobian(model, x, create_graph=True)[:,0,:]\n",
    "                    sol_D1 = sol_D1_fun(x_i)\n",
    "\n",
    "                    ut = sol_D1[:,[1]]\n",
    "                    sol_D2 = batch_jacobian(sol_D1_fun, x_i, create_graph=True)[:,:,:]\n",
    "                    lap = torch.sum(torch.diagonal(sol_D2, dim1=1, dim2=2)[:,:1], dim=1, keepdim=True)\n",
    "\n",
    "                    pde = ut - 0.0001*lap + 5*(u**3 - u)\n",
    "\n",
    "                    # pde loss\n",
    "                    pde_loss = torch.mean(pde**2)\n",
    "\n",
    "                    # neuman boundary loss\n",
    "                    bc_pred_left = model(x_left)\n",
    "                    bc_pred_right = model(x_right)\n",
    "                    bcd_pred_left = sol_D1_fun(x_left)[:,[0]]\n",
    "                    bcd_pred_right = sol_D1_fun(x_right)[:,[0]]\n",
    "                    bc_loss = (torch.mean((bc_pred_left-bc_pred_right)**2) + torch.mean((bcd_pred_left-bcd_pred_right)**2))/2\n",
    "\n",
    "                    # initial loss\n",
    "                    pred_init = model(x_init)\n",
    "                    true_init = u0_true(x_init)\n",
    "                    init_loss = torch.mean((true_init-pred_init)**2)\n",
    "\n",
    "                    loss = lamb_r * pde_loss + lamb_b * bc_loss + lamb_i * init_loss\n",
    "                    loss.backward()\n",
    "                    return loss\n",
    "\n",
    "                if _ % 20 == 0 and _ < 200:\n",
    "                    model.update_grid_from_samples(x_i)\n",
    "\n",
    "                optimizer.step(closure)\n",
    "                #sol = sol_fun(x_i)\n",
    "                loss = lamb_r * pde_loss + lamb_b * bc_loss + lamb_i * init_loss\n",
    "                #l2 = torch.mean((model(x_i) - sol)**2)\n",
    "\n",
    "                if _ % log == 0:\n",
    "                    pbar.set_description(\"pde loss: %.2e | bc loss: %.2e | init loss : %.2e \" % (pde_loss.cpu().detach().numpy(), bc_loss.cpu().detach().numpy(), init_loss.cpu().detach().numpy()))\n",
    "\n",
    "                pde_losses.append(pde_loss.cpu().detach().numpy())\n",
    "                bc_losses.append(bc_loss.cpu().detach().numpy())\n",
    "                init_losses.append(init_loss.cpu().detach().numpy())\n",
    "\n",
    "\n",
    "    train()\n",
    "    np_i_test = 1001\n",
    "    np_t_test = 2\n",
    "    x_mesh = torch.linspace(-1,1,steps=np_i_test).to(device)\n",
    "    t_mesh = torch.linspace(0,2,steps=np_t_test).to(device)\n",
    "    X, T = torch.meshgrid(x_mesh, t_mesh, indexing='ij')\n",
    "\n",
    "    if i_deltat == 0:\n",
    "        x_init = helper(X[:,0], T[:,0])\n",
    "        np.savetxt('./results/kan_t_0_width_[2,5,5,1]_grid_%d_steps_%d.txt'%(grid, steps), model(x_init).cpu().detach().numpy())\n",
    "    \n",
    "    x_init = helper(X[:,0], T[:,0]+0.1)\n",
    "    np.savetxt('./results/kan_t_%.1f_width_[2,5,5,1]_grid_%d_steps_%d.txt'%(0.1*(i_deltat+1), grid, steps), model(x_init).cpu().detach().numpy())\n",
    "\n",
    "    #plt.plot(x_mesh.cpu().detach().numpy(), u0_true(x_init).cpu().detach().numpy())\n",
    "    #plt.plot(x_mesh.cpu().detach().numpy(), model(x_init).cpu().detach().numpy())\n",
    "    #plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "id": "87da8f76-767d-41e0-817b-4ef799587fd9",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[<matplotlib.lines.Line2D at 0x7f63185a73a0>]"
      ]
     },
     "execution_count": 26,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "np_i_test = 101\n",
    "np_t_test = 2\n",
    "x_mesh = torch.linspace(-1,1,steps=np_i_test).to(device)\n",
    "t_mesh = torch.linspace(0,2,steps=np_t_test).to(device)\n",
    "X, T = torch.meshgrid(x_mesh, t_mesh, indexing='ij')\n",
    "\n",
    "x_init = helper(X[:,0], T[:,0]+0.2)\n",
    "\n",
    "plt.plot(x_mesh.cpu().detach().numpy(), model(x_init).cpu().detach().numpy())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "82ce102e-d9f2-4ed4-8499-41aa859886cd",
   "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.9.16"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
