{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "df8f2eae-4826-4ad3-b7fa-66ee19d14d99",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "cuda\n",
      "checkpoint directory created: ./model\n",
      "saving model version 0.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "pde loss: 1.97e+03 | bc loss: 7.30e-01 | l2: 1.15e+00 : 100%|███████| 10/10 [00:01<00:00,  5.45it/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",
    "import time\n",
    "\n",
    "\n",
    "# darcy flow, input (x, y)\n",
    "\n",
    "seed = 0\n",
    "torch.manual_seed(seed)\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_b = 51 # number of boundary points (along each dimension)\n",
    "ranges = [-1, 1]\n",
    "\n",
    "N_a = 20\n",
    "N_u = 20\n",
    "\n",
    "x_a = torch.rand(N_a).to(device) * 2 - 1\n",
    "y_a = torch.rand(N_a).to(device) * 2 - 1\n",
    "sigma_a = 0.2 * torch.rand(N_a).to(device) + 0.1\n",
    "x_u = torch.rand(N_u).to(device) * 2 - 1\n",
    "y_u = torch.rand(N_u).to(device) * 2 - 1\n",
    "sigma_u = 0.2 * torch.rand(N_u).to(device) + 0.1\n",
    "\n",
    "def a(x):\n",
    "    # x shape: (B, d)\n",
    "    # x_a shape: (N_a)\n",
    "    return 1 + torch.sum(torch.exp(-((x[:,[0],None] - x_a[None,None,:])**2+(x[:,[1],None] - y_a[None,None,:])**2)/(2*sigma_a[None,None,:]**2)), dim=-1)\n",
    "\n",
    "def u(x):\n",
    "    # x shape: (B, d)\n",
    "    # x_a shape: (N_a)\n",
    "    return torch.sum(torch.exp(-((x[:,[0],None] - x_u[None,None,:])**2+(x[:,[1],None] - y_u[None,None,:])**2)/(2*sigma_u[None,None,:]**2)), dim=-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",
    "def source_fun(x):\n",
    "    def adu(x):\n",
    "        du = batch_jacobian(u, x, create_graph=True)[:,0,:]\n",
    "        adu_ = a(x) * du\n",
    "        return adu_\n",
    "\n",
    "\n",
    "    lap = batch_jacobian(adu, x)\n",
    "    lhs = torch.sum(torch.diagonal(lap, dim1=1, dim2=2),dim=1,keepdim=True)\n",
    "    return lhs\n",
    "\n",
    "\n",
    "# interior\n",
    "sampling_mode = 'mesh' # 'radnom' or 'mesh'\n",
    "\n",
    "x_mesh = torch.linspace(ranges[0],ranges[1],steps=np_i)\n",
    "y_mesh = torch.linspace(ranges[0],ranges[1],steps=np_i)\n",
    "X, Y = torch.meshgrid(x_mesh, y_mesh, indexing=\"ij\")\n",
    "if sampling_mode == 'mesh':\n",
    "    #mesh\n",
    "    x_i = torch.stack([X.reshape(-1,), Y.reshape(-1,)]).permute(1,0)\n",
    "else:\n",
    "    #random\n",
    "    x_i = torch.rand((np_i**2,2))*2-1\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",
    "xb1 = helper(X[0], Y[0])\n",
    "xb2 = helper(X[-1], Y[0])\n",
    "xb3 = helper(X[:,0], Y[:,0])\n",
    "xb4 = helper(X[:,0], Y[:,-1])\n",
    "x_b = torch.cat([xb1, xb2, xb3, xb4], dim=0)\n",
    "\n",
    "x_b = x_b.to(device)\n",
    "\n",
    "alpha = 0.01\n",
    "log = 1\n",
    "\n",
    "\n",
    "#grids = [5,10,20]\n",
    "grids = [10]\n",
    "steps = 10\n",
    "width = [2,10,1]\n",
    "\n",
    "pde_losses = []\n",
    "bc_losses = []\n",
    "l2_losses = []\n",
    "\n",
    "for grid in grids:\n",
    "    if grid == grids[0]:\n",
    "        model = KAN(width=width, 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\n",
    "                optimizer.zero_grad()\n",
    "                # interior loss\n",
    "                sol_D1_fun = lambda x: batch_jacobian(model, x, create_graph=True)[:,0,:]\n",
    "                adu = lambda x: a(x) * sol_D1_fun(x)\n",
    "                sol_D2 = batch_jacobian(adu, x_i, create_graph=True)[:,:,:]\n",
    "                lap = torch.sum(torch.diagonal(sol_D2, dim1=1, dim2=2), dim=1, keepdim=True)\n",
    "                \n",
    "                #print(sol_D2)\n",
    "                source = source_fun(x_i)\n",
    "                pde_loss = torch.mean((lap - source)**2)\n",
    "\n",
    "                # boundary loss\n",
    "                bc_true = u(x_b)\n",
    "                bc_pred = model(x_b)\n",
    "                bc_loss = torch.mean((bc_pred-bc_true)**2)\n",
    "\n",
    "                loss = alpha * pde_loss + bc_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",
    "            closure()\n",
    "            sol = u(x_i)\n",
    "            loss = alpha * pde_loss + bc_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 | l2: %.2e \" % (pde_loss.cpu().detach().numpy(), bc_loss.cpu().detach().numpy(), l2.cpu().detach().numpy()))\n",
    "\n",
    "            pde_losses.append(pde_loss.cpu().detach().numpy())\n",
    "            bc_losses.append(bc_loss.cpu().detach().numpy())\n",
    "            l2_losses.append(l2.cpu().detach().numpy())\n",
    "            \n",
    "        \n",
    "    start_time = time.time()\n",
    "    train()\n",
    "    end_time = time.time()\n",
    "    \n",
    "    wall_time = end_time - start_time\n",
    "    \n",
    "    # wall time\n",
    "    # l2_loss trajectory\n",
    "    # final profile, true profile\n",
    "    \n",
    "    np.savetxt(f'./results/kan_l2_width_{width}_grid_{grid}_seed_{seed}', np.array(l2_losses))\n",
    "    np.savetxt(f'./results/kan_walltime_width_{width}_grid_{grid}_seed_{seed}', np.array([wall_time]))\n",
    "    \n",
    "    x = torch.linspace(-1,1,steps=101)\n",
    "    y = torch.linspace(-1,1,steps=101)\n",
    "    xx, yy = torch.meshgrid(x, y, indexing='ij')\n",
    "    inputs = torch.stack([xx.reshape(-1,), yy.reshape(-1,)]).permute(1,0).to(device)\n",
    "    np.savetxt(f'./results/kan_sol_width_{width}_grid_{grid}_seed_{seed}', model(inputs).reshape(101,101).cpu().detach().numpy())\n",
    "    np.savetxt(f'./results/a_sol_seed_{seed}', a(inputs).reshape(101,101).cpu().detach().numpy())\n",
    "    np.savetxt(f'./results/u_sol_seed_{seed}', u(inputs).reshape(101,101).cpu().detach().numpy())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "3e445bc6-0e4b-4bdb-99c5-9c5d6214e6dd",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.image.AxesImage at 0x7fed200883a0>"
      ]
     },
     "execution_count": 12,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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UVuzZo2sXLlSduzYIaeddpqIzJi4du7cKV/84hdFROT000+XZrMpO3bskPe+970iIvKb3/xGfv7zn8vNN988q+OkaSrHHXdc743tFa94BW+gEjhHs4PzVA7nqBzOUTkrVqxYkDmqvKi98MIL8otf/KLX3rt3r+zevVuOPPJIOf7442XDhg2yZcsWWbt2raxdu1a2bNkiS5culUsuuUREZr7In/3Zn8knP/lJOeqoo+TII4+Uv/iLv5DXv/71PW9IQgghZC5UXtR+8pOfyFvf+tZe+yUHjssvv1zuvPNOufbaa2ViYkLWr18vzz77rJx11lly//3392LURET++q//WhqNhrz3ve+ViYkJedvb3iZ33nlnpRg1QgghBKm8qJ1//vmut0qSJLJ582bZvHlz4ZjR0VH50pe+JF/60peqHj6i1WrJ5z73uciJhMRwjmYH56kczlE5nKNyFnqOkjBsf0pCCCHkEMGExoQQQmoDFzVCCCG1gYsaIYSQ2sBFjRBCSG3gokYIIaQ2LOpF7Stf+YqsWbNGRkdH5fTTT5cf//jHh/qUDglbt26VM888U5YvXy7HHHOMvPvd75Y9e/ZEY2ZTvPVwYuvWrZIkiWzYsKH3f5yjGX7961/LBz/4QTnqqKNk6dKl8qY3vUl27drV6z/c56nT6chnP/tZWbNmjSxZskROPPFEueGGGyRXiXwPtzn6vSoeHRYp99xzT2g2m+H2228PTzzxRLjmmmvCsmXLwlNPPXWoT+1l5+1vf3u44447ws9//vOwe/fucOGFF4bjjz8+vPDCC70xX/jCF8Ly5cvDN77xjfDYY4+F973vfeHYY48N4+Pjh/DMDw0PPfRQeM1rXhPe8IY3hGuuuab3/5yjEP7v//4vnHDCCeHDH/5w+I//+I+wd+/e8N3vfjf84he/6I053OfpxhtvDEcddVT413/917B3797wT//0T+GII44I27Zt64053Obo29/+dti0aVP4xje+EUQkfPOb34z6ZzMfV155ZXj1q18dduzYER555JHw1re+NbzxjW8MnU6n0rks2kXtD/7gD8KVV14Z/d/JJ58crrvuukN0Rr8/7N+/P4hI2LlzZwghhDzPw8qVK8MXvvCF3pjJycmwYsWK8NWvfvVQneYh4fnnnw9r164NO3bsCOedd15vUeMczfDpT386nHvuuYX9nKcQLrzwwvDRj340+r+LL744fPCDHwwhcI5wUZvNfPzud78LzWYz3HPPPb0xv/71r0OapuE73/lOpeMvSvPj9PS07Nq1Kyo0KiKybt26wkKjhxPPPfeciIgceeSRIjK74q2HC1dddZVceOGFJs8o52iG++67T8444wx5z3veI8ccc4ycdtppcvvtt/f6OU8i5557rnzve9+TJ598UkREfvrTn8oDDzwg73znO0WEc4QsVPHoIhakntpC89vf/la63e7AYqRFhUYPF0IIsnHjRjn33HPl1FNPFZHZFW89HLjnnnvkkUcekYcfftj0cY5m+OUvfym33nqrbNy4UT7zmc/IQw89JB//+Mel1WrJZZddxnkSkU9/+tPy3HPPycknnyxZlkm325WbbrpJPvCBD4gI7yVkoYpHF7EoF7WXmEsx0rpz9dVXy89+9jN54IEHTN/hPF/PPPOMXHPNNXL//ffL6Oho4bjDeY5ERPI8lzPOOEO2bNkiIiKnnXaaPP7443LrrbfKZZdd1ht3OM/T17/+dbn77rtl+/btcsopp8ju3btlw4YNsmrVKrn88st74w7nORrEsItHF7EozY9HH320ZFlmVnBdjPRw5GMf+5jcd9998oMf/ECOO+643v/r4q2aw2m+du3aJfv375fTTz9dGo2GNBoN2blzp/zN3/yNNBqN3jwcznMkInLsscfK6173uuj/Xvva18rTTz8tIryXREQ+9alPyXXXXSfvf//75fWvf7186EMfkk984hOydetWEeEcIbOZD108umjMbFmUi9rIyIicfvrpsmPHjuj/d+zYMevq2XUihCBXX3213HvvvfL9739f1qxZE/Xr4q0v8VLx1sNlvt72trfJY489Jrt37+79O+OMM+TSSy+V3bt3y4knnnjYz5GIyDnnnGPCQZ588kk54YQTRIT3kojIgQMHJE3jn84sy3ou/ZyjmNnMhy4e/RIvFY+uPGdzcm/5PeAll/6vfe1r4YknnggbNmwIy5YtC//93/99qE/tZefP//zPw4oVK8IPf/jD8Jvf/Kb378CBA70xX/jCF8KKFSvCvffeGx577LHwgQ98oNYuxrNBez+GwDkKYSbcodFohJtuuin853/+Z/iHf/iHsHTp0nD33Xf3xhzu83T55ZeHV7/61T2X/nvvvTccffTR4dprr+2NOdzm6Pnnnw+PPvpoePTRR4OIhFtuuSU8+uijvRCr2czHlVdeGY477rjw3e9+NzzyyCPh//2//3d4ufSHEMLf/u3fhhNOOCGMjIyEN7/5zT0X9sMNERn474477uiNyfM8fO5znwsrV64MrVYrvOUtbwmPPfbYoTvp3wNwUeMczfCtb30rnHrqqaHVaoWTTz453HbbbVH/4T5P4+Pj4ZprrgnHH398GB0dDSeeeGLYtGlTmJqa6o053OboBz/4wcDfoMsvvzyEMLv5mJiYCFdffXU48sgjw5IlS8JFF10Unn766crnwnpqhBBCasOi1NQIIYSQQXBRI4QQUhu4qBFCCKkNXNQIIYTUBi5qhBBCagMXNUIIIbWBixohhJDawEWNEEJIbeCiRgghpDZwUSOEEFIbuKgRQgipDf8/BMI3C7kbBt8AAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "x = torch.linspace(-1,1,steps=101)\n",
    "y = torch.linspace(-1,1,steps=101)\n",
    "xx, yy = torch.meshgrid(x, y, indexing='ij')\n",
    "inputs = torch.stack([xx.reshape(-1,), yy.reshape(-1,)]).permute(1,0).to(device)\n",
    "plt.imshow(model(inputs).reshape(101,101).cpu().detach().numpy())\n",
    "#plt.clim(0.8,1.2)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "237d4f92-ce7d-45ae-b47b-488a327dc86b",
   "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
}
