{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 37,
   "id": "a68eebe4-b99b-4e8b-bd09-97aa103f17cf",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "cuda\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "pde loss: 7.93e+00 | bc loss: 4.31e-02 | init loss : 1.56e-03 :   1%| | 12/1000 [00:12<17:33,  1.07s\n"
     ]
    },
    {
     "ename": "KeyboardInterrupt",
     "evalue": "",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m                         Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[37], line 150\u001b[0m\n\u001b[1;32m    146\u001b[0m         bc_losses\u001b[38;5;241m.\u001b[39mappend(bc_loss\u001b[38;5;241m.\u001b[39mcpu()\u001b[38;5;241m.\u001b[39mdetach()\u001b[38;5;241m.\u001b[39mnumpy())\n\u001b[1;32m    147\u001b[0m         init_losses\u001b[38;5;241m.\u001b[39mappend(init_loss\u001b[38;5;241m.\u001b[39mcpu()\u001b[38;5;241m.\u001b[39mdetach()\u001b[38;5;241m.\u001b[39mnumpy())\n\u001b[0;32m--> 150\u001b[0m \u001b[43mtrain\u001b[49m\u001b[43m(\u001b[49m\u001b[43m)\u001b[49m\n",
      "Cell \u001b[0;32mIn[37], line 137\u001b[0m, in \u001b[0;36mtrain\u001b[0;34m()\u001b[0m\n\u001b[1;32m    134\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m _ \u001b[38;5;241m%\u001b[39m \u001b[38;5;241m20\u001b[39m \u001b[38;5;241m==\u001b[39m \u001b[38;5;241m0\u001b[39m \u001b[38;5;129;01mand\u001b[39;00m _ \u001b[38;5;241m<\u001b[39m \u001b[38;5;241m200\u001b[39m:\n\u001b[1;32m    135\u001b[0m     model\u001b[38;5;241m.\u001b[39mupdate_grid_from_samples(x_i)\n\u001b[0;32m--> 137\u001b[0m \u001b[43moptimizer\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mstep\u001b[49m\u001b[43m(\u001b[49m\u001b[43mclosure\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    138\u001b[0m \u001b[38;5;66;03m#sol = sol_fun(x_i)\u001b[39;00m\n\u001b[1;32m    139\u001b[0m loss \u001b[38;5;241m=\u001b[39m lamb_r \u001b[38;5;241m*\u001b[39m pde_loss \u001b[38;5;241m+\u001b[39m lamb_b \u001b[38;5;241m*\u001b[39m bc_loss \u001b[38;5;241m+\u001b[39m lamb_i \u001b[38;5;241m*\u001b[39m init_loss\n",
      "File \u001b[0;32m/state/partition1/llgrid/pkg/anaconda/anaconda3-2023a-pytorch/lib/python3.9/site-packages/torch/optim/optimizer.py:280\u001b[0m, in \u001b[0;36mOptimizer.profile_hook_step.<locals>.wrapper\u001b[0;34m(*args, **kwargs)\u001b[0m\n\u001b[1;32m    276\u001b[0m         \u001b[38;5;28;01melse\u001b[39;00m:\n\u001b[1;32m    277\u001b[0m             \u001b[38;5;28;01mraise\u001b[39;00m \u001b[38;5;167;01mRuntimeError\u001b[39;00m(\u001b[38;5;124mf\u001b[39m\u001b[38;5;124m\"\u001b[39m\u001b[38;5;132;01m{\u001b[39;00mfunc\u001b[38;5;132;01m}\u001b[39;00m\u001b[38;5;124m must return None or a tuple of (new_args, new_kwargs),\u001b[39m\u001b[38;5;124m\"\u001b[39m\n\u001b[1;32m    278\u001b[0m                                \u001b[38;5;124mf\u001b[39m\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mbut got \u001b[39m\u001b[38;5;132;01m{\u001b[39;00mresult\u001b[38;5;132;01m}\u001b[39;00m\u001b[38;5;124m.\u001b[39m\u001b[38;5;124m\"\u001b[39m)\n\u001b[0;32m--> 280\u001b[0m out \u001b[38;5;241m=\u001b[39m \u001b[43mfunc\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43margs\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43mkwargs\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    281\u001b[0m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_optimizer_step_code()\n\u001b[1;32m    283\u001b[0m \u001b[38;5;66;03m# call optimizer step post hooks\u001b[39;00m\n",
      "File \u001b[0;32m/state/partition1/llgrid/pkg/anaconda/anaconda3-2023a-pytorch/lib/python3.9/site-packages/torch/optim/optimizer.py:33\u001b[0m, in \u001b[0;36m_use_grad_for_differentiable.<locals>._use_grad\u001b[0;34m(self, *args, **kwargs)\u001b[0m\n\u001b[1;32m     31\u001b[0m \u001b[38;5;28;01mtry\u001b[39;00m:\n\u001b[1;32m     32\u001b[0m     torch\u001b[38;5;241m.\u001b[39mset_grad_enabled(\u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mdefaults[\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mdifferentiable\u001b[39m\u001b[38;5;124m'\u001b[39m])\n\u001b[0;32m---> 33\u001b[0m     ret \u001b[38;5;241m=\u001b[39m \u001b[43mfunc\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43margs\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43mkwargs\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m     34\u001b[0m \u001b[38;5;28;01mfinally\u001b[39;00m:\n\u001b[1;32m     35\u001b[0m     torch\u001b[38;5;241m.\u001b[39mset_grad_enabled(prev_grad)\n",
      "File \u001b[0;32m/state/partition1/llgrid/pkg/anaconda/anaconda3-2023a-pytorch/lib/python3.9/site-packages/torch/optim/adam.py:121\u001b[0m, in \u001b[0;36mAdam.step\u001b[0;34m(self, closure)\u001b[0m\n\u001b[1;32m    119\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m closure \u001b[38;5;129;01mis\u001b[39;00m \u001b[38;5;129;01mnot\u001b[39;00m \u001b[38;5;28;01mNone\u001b[39;00m:\n\u001b[1;32m    120\u001b[0m     \u001b[38;5;28;01mwith\u001b[39;00m torch\u001b[38;5;241m.\u001b[39menable_grad():\n\u001b[0;32m--> 121\u001b[0m         loss \u001b[38;5;241m=\u001b[39m \u001b[43mclosure\u001b[49m\u001b[43m(\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    123\u001b[0m \u001b[38;5;28;01mfor\u001b[39;00m group \u001b[38;5;129;01min\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mparam_groups:\n\u001b[1;32m    124\u001b[0m     params_with_grad \u001b[38;5;241m=\u001b[39m []\n",
      "Cell \u001b[0;32mIn[37], line 131\u001b[0m, in \u001b[0;36mtrain.<locals>.closure\u001b[0;34m()\u001b[0m\n\u001b[1;32m    128\u001b[0m init_loss \u001b[38;5;241m=\u001b[39m torch\u001b[38;5;241m.\u001b[39mmean((true_init\u001b[38;5;241m-\u001b[39mpred_init)\u001b[38;5;241m*\u001b[39m\u001b[38;5;241m*\u001b[39m\u001b[38;5;241m2\u001b[39m)\n\u001b[1;32m    130\u001b[0m loss \u001b[38;5;241m=\u001b[39m lamb_r \u001b[38;5;241m*\u001b[39m pde_loss \u001b[38;5;241m+\u001b[39m lamb_b \u001b[38;5;241m*\u001b[39m bc_loss \u001b[38;5;241m+\u001b[39m lamb_i \u001b[38;5;241m*\u001b[39m init_loss\n\u001b[0;32m--> 131\u001b[0m \u001b[43mloss\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mbackward\u001b[49m\u001b[43m(\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    132\u001b[0m \u001b[38;5;28;01mreturn\u001b[39;00m loss\n",
      "File \u001b[0;32m/state/partition1/llgrid/pkg/anaconda/anaconda3-2023a-pytorch/lib/python3.9/site-packages/torch/_tensor.py:487\u001b[0m, in \u001b[0;36mTensor.backward\u001b[0;34m(self, gradient, retain_graph, create_graph, inputs)\u001b[0m\n\u001b[1;32m    477\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m has_torch_function_unary(\u001b[38;5;28mself\u001b[39m):\n\u001b[1;32m    478\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m handle_torch_function(\n\u001b[1;32m    479\u001b[0m         Tensor\u001b[38;5;241m.\u001b[39mbackward,\n\u001b[1;32m    480\u001b[0m         (\u001b[38;5;28mself\u001b[39m,),\n\u001b[0;32m   (...)\u001b[0m\n\u001b[1;32m    485\u001b[0m         inputs\u001b[38;5;241m=\u001b[39minputs,\n\u001b[1;32m    486\u001b[0m     )\n\u001b[0;32m--> 487\u001b[0m \u001b[43mtorch\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mautograd\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mbackward\u001b[49m\u001b[43m(\u001b[49m\n\u001b[1;32m    488\u001b[0m \u001b[43m    \u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mgradient\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mretain_graph\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mcreate_graph\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43minputs\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43minputs\u001b[49m\n\u001b[1;32m    489\u001b[0m \u001b[43m\u001b[49m\u001b[43m)\u001b[49m\n",
      "File \u001b[0;32m/state/partition1/llgrid/pkg/anaconda/anaconda3-2023a-pytorch/lib/python3.9/site-packages/torch/autograd/__init__.py:200\u001b[0m, in \u001b[0;36mbackward\u001b[0;34m(tensors, grad_tensors, retain_graph, create_graph, grad_variables, inputs)\u001b[0m\n\u001b[1;32m    195\u001b[0m     retain_graph \u001b[38;5;241m=\u001b[39m create_graph\n\u001b[1;32m    197\u001b[0m \u001b[38;5;66;03m# The reason we repeat same the comment below is that\u001b[39;00m\n\u001b[1;32m    198\u001b[0m \u001b[38;5;66;03m# some Python versions print out the first line of a multi-line function\u001b[39;00m\n\u001b[1;32m    199\u001b[0m \u001b[38;5;66;03m# calls in the traceback and some print out the last line\u001b[39;00m\n\u001b[0;32m--> 200\u001b[0m \u001b[43mVariable\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_execution_engine\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mrun_backward\u001b[49m\u001b[43m(\u001b[49m\u001b[43m  \u001b[49m\u001b[38;5;66;43;03m# Calls into the C++ engine to run the backward pass\u001b[39;49;00m\n\u001b[1;32m    201\u001b[0m \u001b[43m    \u001b[49m\u001b[43mtensors\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mgrad_tensors_\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mretain_graph\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mcreate_graph\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43minputs\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    202\u001b[0m \u001b[43m    \u001b[49m\u001b[43mallow_unreachable\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43;01mTrue\u001b[39;49;00m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43maccumulate_grad\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43;01mTrue\u001b[39;49;00m\u001b[43m)\u001b[49m\n",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m: "
     ]
    }
   ],
   "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",
    "# implement Allen Cahn 3D (x, y, t)\n",
    "\n",
    "device = torch.device('cuda' if torch.cuda.is_available() else 'cpu')\n",
    "print(device)\n",
    "\n",
    "dim = 2\n",
    "np_i = 31 # number of interior points (along each dimension)\n",
    "np_b = 31 # number of boundary points (along each dimension)\n",
    "np_t = 31\n",
    "ranges = [0, 1]\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",
    "# define solution\n",
    "sol_fun = lambda x: torch.sin(torch.pi*x[:,[0]])*torch.sin(torch.pi*x[:,[1]])\n",
    "source_fun = lambda x: -2*torch.pi**2 * torch.sin(torch.pi*x[:,[0]])*torch.sin(torch.pi*x[:,[1]])\n",
    "\n",
    "# interior\n",
    "sampling_mode = 'mesh' # 'radnom' or 'mesh'\n",
    "\n",
    "x_mesh = torch.linspace(ranges[0],ranges[1],steps=np_i).to(device)\n",
    "y_mesh = torch.linspace(ranges[0],ranges[1],steps=np_i).to(device)\n",
    "t_mesh = torch.linspace(0,1,steps=np_t).to(device)\n",
    "X, Y, T = torch.meshgrid(x_mesh, y_mesh, t_mesh, indexing='ij')\n",
    "\n",
    "if sampling_mode == 'mesh':\n",
    "    #mesh\n",
    "    x_i = torch.stack([X.reshape(-1,), Y.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, T: torch.stack([X.reshape(-1,), Y.reshape(-1,), T.reshape(-1,)]).permute(1,0)\n",
    "xb1_x = helper(X[0], Y[0], T[0])\n",
    "xb2_x = helper(X[-1], Y[0], T[0])\n",
    "x_bx = torch.cat([xb1_x, xb2_x], dim=0)\n",
    "\n",
    "xb1_y = helper(X[:,0], Y[:,0], T[0])\n",
    "xb2_y = helper(X[:,0], Y[:,-1], T[0])\n",
    "x_by = torch.cat([xb1_y, xb2_y], dim=0)\n",
    "\n",
    "x_bx = x_bx.to(device)\n",
    "x_by = x_by.to(device)\n",
    "\n",
    "\n",
    "x_init = helper(X[:,:,0], Y[:,:,0], T[:,:,0])\n",
    "\n",
    "eps = 0.025\n",
    "\n",
    "def u0_true(x):\n",
    "    x1 = x[:,[0]]\n",
    "    x2 = x[:,[1]]\n",
    "    return torch.tanh((0.35-torch.sqrt((x1-0.5)**2+(x2-0.5)**2))/(2*eps))\n",
    "\n",
    "log = 1\n",
    "\n",
    "\n",
    "grids = [5] #[5,10,20]\n",
    "steps = 1000\n",
    "\n",
    "pde_losses = []\n",
    "bc_losses = []\n",
    "init_losses = []\n",
    "\n",
    "lamb_r = 1\n",
    "lamb_b = 1\n",
    "lamb_i = 1000\n",
    "\n",
    "\n",
    "for grid in grids:\n",
    "    if grid == grids[0]:\n",
    "        pass\n",
    "        #model = KAN(width=[3,10,10,1], grid=grid, k=3, seed=1, device=device)\n",
    "    else:\n",
    "        model.save_act = True\n",
    "        model.get_act(x_i)\n",
    "        model = model.refine(grid)\n",
    "\n",
    "    def train():\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",
    "        optimizer = torch.optim.Adam(model.parameters(), lr=1e-2)\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",
    "        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",
    "                sol = 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[:,[2]]\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)[:,:2], dim=1, keepdim=True)\n",
    "                \n",
    "                rhs = 10 * (eps**2*lap - sol**3 + sol)\n",
    "                \n",
    "                # pde loss\n",
    "                pde_loss = torch.mean((ut - rhs)**2)\n",
    "\n",
    "                # neuman boundary loss\n",
    "                bcd_pred_x = sol_D1_fun(x_bx)[:,[0]]\n",
    "                bcd_pred_y = sol_D1_fun(x_by)[:,[1]]\n",
    "                bc_loss = (torch.mean(bcd_pred_x**2)+torch.mean(bcd_pred_y**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()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "id": "87da8f76-767d-41e0-817b-4ef799587fd9",
   "metadata": {},
   "outputs": [
    {
     "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(ranges[0],ranges[1],steps=np_i_test).to(device)\n",
    "y_mesh = torch.linspace(ranges[0],ranges[1],steps=np_i_test).to(device)\n",
    "t_mesh = torch.linspace(0,10,steps=np_t_test).to(device)\n",
    "X, Y, T = torch.meshgrid(x_mesh, y_mesh, t_mesh, indexing='ij')\n",
    "\n",
    "x_init = helper(X[:,:,0], Y[:,:,0], T[:,:,0]+0.5)\n",
    "\n",
    "plt.imshow(model(x_init).reshape(np_i_test,np_i_test).cpu().detach().numpy())\n",
    "plt.clim(-1,1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b7058e86-9226-4749-afed-d9f2f3c34f11",
   "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
}
