{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 46,
   "id": "a68eebe4-b99b-4e8b-bd09-97aa103f17cf",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "cuda\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "pde loss: 8.20e-03 | bc loss: 7.46e-04 | init loss : 8.80e-05 : 100%|█| 2000/2000 [02:43<00:00, 12.2\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",
    "from experiments.baselines.MLP import MLP\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",
    "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",
    "steps = 2000\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",
    "\n",
    "    model = MLP(width=[3,128,128,128,128,128,1], seed=1, device=device)\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",
    "\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": 48,
   "id": "87da8f76-767d-41e0-817b-4ef799587fd9",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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MizoRe3b3/421q5di3Q6jfXr1+MXv/jFXE9pThgYGMCll16KbDaLlStX4tprr8X+/fvFmFMp3rqYGBgYgOd56O3trf2Px+g4b731Fj772c9i2bJlyGQyeN/73oc9e/bU+hf7cSqXy/ja176GtWvXoqWlBeeddx7uueceVKvV2pjFdozmVfFos0B58sknTSKRMI8++qh59dVXzR133GFaW1vNwYMH53pqZ5yPfOQj5rHHHjO/+c1vzN69e83VV19t1qxZYyYmJmpjvvnNb5psNmu+//3vm1deecV86lOfMqtWrTJjY2NzOPO54YUXXjDnnnuuec973mPuuOOO2v95jIw5duyYOeecc8znP/9587//+7/mwIED5kc/+pF5/fXXa2MW+3G69957zbJly8x//dd/mQMHDph///d/N21tbebBBx+sjVlsx+jZZ58127ZtM9///vcNAPP000+L/lM5Hrfccos566yzzK5du8xLL71kPvjBD5r3vve9plwuNzSXBbuo/fEf/7G55ZZbxP8uuOACc/fdd8/RjOYPQ0NDBoDZvXu3McaYarVquru7zTe/+c3amHw+bzo6Osx3vvOduZrmnDA+Pm7WrVtndu3aZa688sraosZjdJy77rrLXHHFFdP28zgZc/XVV5svfOEL4n/XXXed+exnP2uM4THSi9qpHI+RkRGTSCTMk08+WRvz1ltvmVgsZn74wx829P4L0vxYLBaxZ88eUWgUADZv3jxtodHFxOjoKABg6dKlAE6teOti4bbbbsPVV18dyDPKY3ScZ555Bhs2bMAnPvEJrFy5EpdccgkeffTRWj+PE3DFFVfgxz/+MV577TUAwK9+9Ss899xz+OhHPwqAx0gzW8Wjp2NW6qnNNkeOHEGlUjlpMdLpCo0uFowx2LJlC6644gpcfPHFAE6teOti4Mknn8RLL72EF198MdDHY3Sc3/3ud3j44YexZcsWfPWrX8ULL7yAL33pS0ilUrjxxht5nADcddddGB0dxQUXXADf91GpVHDffffhM5/5DABeS5rZKh49HQtyUTvB6RQjjTq33347fv3rX+O5554L9C3m43Xo0CHccccd2LlzJ9Lp9LTjFvMxAoBqtYoNGzagv78fAHDJJZdg3759ePjhh3HjjTfWxi3m4/S9730P3/3ud7Fjxw5cdNFF2Lt3L3p7e9HT04ObbrqpNm4xH6OT0ezi0dOxIM2Py5cvh+/7gRXcLUa6GPniF7+IZ555Bj/96U9x9tln1/7vFm91WUzHa8+ePRgaGsL69esRj8cRj8exe/du/MM//APi8XjtOCzmYwQAq1atwrve9S7xvwsvvBBvvvkmAF5LAPCVr3wFd999Nz796U/j3e9+Nz73uc/hy1/+MgYGBgDwGGlO5Xi4xaOnG3OqLMhFLZlMYv369di1a5f4/65du065enaUMMbg9ttvx1NPPYWf/OQnWLt2reh3i7ee4ETx1sVyvD70oQ/hlVdewd69e2t/GzZswA033IC9e/fivPPOW/THCAAuv/zyQDjIa6+9hnPOOQcAryUAmJqaQiwmvzp936+59PMYSU7leLjFo09wonh0w8fstNxb5gEnXPr/+Z//2bz66qumt7fXtLa2mjfeeGOup3bG+au/+ivT0dFhfvazn5nDhw/X/qampmpjvvnNb5qOjg7z1FNPmVdeecV85jOfibSL8angej8aw2NkzPFwh3g8bu677z7z29/+1vzrv/6ryWQy5rvf/W5tzGI/TjfddJM566yzai79Tz31lFm+fLm58847a2MW2zEaHx83L7/8snn55ZcNALN9+3bz8ssv10KsTuV43HLLLebss882P/rRj8xLL71k/vRP/3RxufQbY8w//uM/mnPOOcckk0nz/ve/v+bCvtgAcNK/xx57rDamWq2ar3/966a7u9ukUinzgQ98wLzyyitzN+l5gF7UeIyO85//+Z/m4osvNqlUylxwwQXmkUceEf2L/TiNjY2ZO+64w6xZs8ak02lz3nnnmW3btplCoVAbs9iO0U9/+tOTfgfddNNNxphTOx65XM7cfvvtZunSpaalpcVcc8015s0332x4LqynRgghJDIsSE2NEEIIORlc1AghhEQGLmqEEEIiAxc1QgghkYGLGiGEkMjARY0QQkhk4KJGCCEkMnBRI4QQEhm4qBFCCIkMXNQIIYREBi5qhBBCIsP/DzosRs+c8+5AAAAAAElFTkSuQmCC",
      "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]+1.0)\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": "f7cd3dc0-c12e-414a-a2d4-9de9f9bcf73d",
   "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
}
