{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "7854503c",
   "metadata": {},
   "source": [
    "### Three-layer KAN"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "2075ef56",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "cpu\n",
      "checkpoint directory created: ./model\n",
      "saving model version 0.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "| train_loss: 5.79e-02 | test_loss: 5.96e-02 | reg: 1.71e+00 | : 100%|█| 20/20 [00:25<00:00,  1.29s/"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "saving model version 0.1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "from kan import *\n",
    "\n",
    "device = torch.device('cuda' if torch.cuda.is_available() else 'cpu')\n",
    "print(device)\n",
    "\n",
    "# create a KAN: 2D inputs, 1D output, and 5 hidden neurons. cubic spline (k=3), 5 grid intervals (grid=5).\n",
    "model = KAN(width=[100,1,1], grid=3, k=3, seed=1, device=device)\n",
    "# f = lambda x: torch.sin(torch.pi*(x[:,[0]]**2+x[:,[1]]**2))+ x[:,[2]]*x[:,[3]]\n",
    "f = lambda x: torch.exp((1/100) * torch.sum(torch.sin((torch.pi / 2) * x) ** 2, dim=1))\n",
    "dataset = create_dataset(f, n_var=100, train_num=8000, device=device)\n",
    "\n",
    "# train the model\n",
    "model.fit(dataset, opt=\"LBFGS\", steps=20, lamb=0.002, lamb_entropy=2.);"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "b8c880c1",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "saving model version 0.2\n"
     ]
    }
   ],
   "source": [
    "model = model.prune(edge_th=1e-2)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "585b699c",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAZcAAAFICAYAAACcDrP3AAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjkuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8hTgPZAAAACXBIWXMAAA9hAAAPYQGoP6dpAAAKKElEQVR4nO3dvW4cZRvH4Xt2Z1dCQkoVjoAidSqOAw6CM6AEUkCBxAFAamoKUI6BQ6BxmTQIiQgEaL/mLXhnGZtde9f+2/vh65Isx7FX8zTJL88982yaruu6AoCg0aEXAMD5ERcA4sQFgDhxASBOXACIExcA4sQFgDhxASBOXACIExcA4sQFgDhxASBOXACIExcA4sQFgDhxASBOXACIExcA4sQFgDhxASBOXACIExcA4sQFdnRxcVEff/xxXVxcHHopcPTaQy8ATsVXX31V3377bVVVffPNNwdeDRw3OxfY0YcffnjpM7CduMCOnj59eukzsJ24ABAnLgDEiQsAceICQJy4ABAnLgDEiQsAceICQJy4ABAnLgDEiQsAceICQJy4ABAnLgDEiQsAceICQJy4ABAnLgDEiQsAceICQJy4ABAnLgDEiQsAceICQJy4ABAnLgDEiQsAceICQJy4ABAnLgDEiQsAceICO/rxxx/r2bNn1XXdoZcCR09cYEdv376t58+f13K5PPRS4Og1nX+GwU7evn1bP/30U33wwQf15MmTQy8Hjpq4wI7m83n98ssv9fTp05pMJodeDhw1YzEA4sQFgDhxASBOXACIExcA4sQFgDhxASBOXACIExcA4sQFgDhxASBOXACIExcA4sQFgDhxASBOXACIExcA4sQFgDhxASBOXACIExcA4sQFgDhxASBOXACIExcA4sQFgDhxASBOXACIExcA4sQFgDhxASBOXACIExcA4sQFgDhxASBOXACIExcA4sQFgDhxASBOXACIExcA4sQFgDhxASBOXACIExcA4sQFgDhxASBOXACIExfYwXw+r6qq0cgfGdiFPymwg7/++quapqknT55U27aHXg4cPXGBHbz77rvVdV11XXfopcBJEBfYQT8O++2332qxWBx4NXD8xAWAOHEBIE5cAIgTFwDixAWAOHEBIE5cAIgTFwDixAWAOHEBIE5cAIgTFwDixAWAOHEBIE5cAIgTFwDixAWAOHEBIE5cAIgTFwDi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",
      "text/plain": [
       "<Figure size 500x400 with 104 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "model.plot()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "ee39c97b",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "checkpoint directory created: ./model\n",
      "saving model version 0.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "| train_loss: 2.09e-03 | test_loss: 2.31e-03 | reg: 2.09e+01 | : 100%|█| 50/50 [00:39<00:00,  1.28it\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "saving model version 0.1\n",
      "saving model version 0.2\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "| train_loss: 2.77e-04 | test_loss: 2.65e-03 | reg: 2.35e+01 | : 100%|█| 50/50 [00:47<00:00,  1.05it\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "saving model version 0.3\n",
      "saving model version 0.4\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "| train_loss: 7.71e-05 | test_loss: 5.63e-03 | reg: 2.35e+01 | : 100%|█| 50/50 [00:46<00:00,  1.08it\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "saving model version 0.5\n",
      "saving model version 0.6\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "| train_loss: 1.82e-05 | test_loss: 1.82e-02 | reg: 2.35e+01 | : 100%|█| 50/50 [01:37<00:00,  1.95s/\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "saving model version 0.7\n",
      "saving model version 0.8\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "| train_loss: 1.38e-05 | test_loss: 1.95e-02 | reg: 2.34e+01 | : 100%|█| 50/50 [04:46<00:00,  5.73s/"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "saving model version 0.9\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "grids = [3,5,10,20,50]\n",
    "#grids = [5]\n",
    "\n",
    "train_rmse = []\n",
    "test_rmse = []\n",
    "train_losses = []\n",
    "test_losses = []\n",
    "for i in range(len(grids)):\n",
    "    # model = KAN(width=[4,2,1,1], grid=grids[i], k=3, seed=0, device=device).initialize_from_another_model(model, dataset['train_input'])\n",
    "    # model = model.refine(grid=grids[i])\n",
    "    if i == 0:\n",
    "        model = KAN(width=[100,1,1], grid=grids[i], k=3, seed=0, device=device)\n",
    "    if i != 0:\n",
    "        model = model.refine(grids[i])\n",
    "    results = model.fit(dataset, opt=\"LBFGS\", steps=50, stop_grid_update_step=20);\n",
    "    train_rmse.append(results['train_loss'][-1].item())\n",
    "    test_rmse.append(results['test_loss'][-1].item())\n",
    "    train_losses += results['train_loss']\n",
    "    test_losses += results['test_loss']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b30e87b7",
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.rcParams['figure.dpi'] = 600\n",
    "plt.figure(figsize=(10, 6))\n",
    "plt.plot(train_losses,label='train')\n",
    "plt.plot(test_losses,label='test')\n",
    "plt.legend(loc='upper right')\n",
    "plt.ylabel('RMSE')\n",
    "plt.xlabel('Epochs')\n",
    "plt.yscale('log')\n",
    "plt.title('KAN [100,1,1]')"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "8c345302-c8bc-4585-8022-c5d90eb64341",
   "metadata": {},
   "source": [
    "Author's note: The scaling isn't optimal. Possibly because of updates on curve2coef, to be investigated. "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "94f3930a",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "n_params = np.array(grids) * (4*2+4*2+2*1)\n",
    "plt.plot(n_params, train_rmse, marker=\"o\")\n",
    "plt.plot(n_params, test_rmse, marker=\"o\")\n",
    "plt.plot(n_params, 10000*n_params**(-4.), color=\"black\", ls=\"--\")\n",
    "plt.legend(['train', 'test', r'$N^{-4}$'], loc=\"lower left\")\n",
    "plt.xscale('log')\n",
    "plt.yscale('log')\n",
    "print(train_rmse)\n",
    "print(test_rmse)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "500fa84e",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Mean Squared Error (MSE): 6.6071e-03\n",
      "Root Mean Squared Error (RMSE): 8.1284e-02\n",
      "Mean Absolute Error (MAE): 6.4665e-02\n"
     ]
    }
   ],
   "source": [
    "import torch\n",
    "import numpy as np\n",
    "\n",
    "device = torch.device('cuda' if torch.cuda.is_available() else 'cpu')\n",
    "n_samples = 1000\n",
    "random_points = torch.rand(n_samples, 100, device=device)\n",
    "Z_real = f(random_points).cpu().numpy()\n",
    "Z_pred = model(random_points).cpu().detach().numpy()\n",
    "Z_diff = Z_real - Z_pred\n",
    "mse = np.mean((Z_real - Z_pred) ** 2)\n",
    "rmse = np.sqrt(mse)\n",
    "mae = np.mean(np.abs(Z_real - Z_pred))\n",
    "print(f\"Mean Squared Error (MSE): {mse:.4e}\")\n",
    "print(f\"Root Mean Squared Error (RMSE): {rmse:.4e}\")\n",
    "print(f\"Mean Absolute Error (MAE): {mae:.4e}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "81036ef1",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "checkpoint directory created: ./model\n",
      "saving model version 0.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "| train_loss: 4.00e-03 | test_loss: 5.17e-02 | reg: 1.00e+01 | : 100%|█| 250/250 [22:49<00:00,  5.48"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "saving model version 0.1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "grids = [50]\n",
    "#grids = [5]\n",
    "\n",
    "train_rmse = []\n",
    "test_rmse = []\n",
    "train_losses = []\n",
    "test_losses = []\n",
    "for i in range(len(grids)):\n",
    "    # model = KAN(width=[4,2,1,1], grid=grids[i], k=3, seed=0, device=device).initialize_from_another_model(model, dataset['train_input'])\n",
    "    # model = model.refine(grid=grids[i])\n",
    "    if i == 0:\n",
    "        model = KAN(width=[100,1,1], grid=grids[i], k=3, seed=0, device=device)\n",
    "    if i != 0:\n",
    "        model = model.refine(grids[i])\n",
    "    results = model.fit(dataset, opt=\"LBFGS\", steps=250, stop_grid_update_step=20);\n",
    "    train_rmse.append(results['train_loss'][-1].item())\n",
    "    test_rmse.append(results['test_loss'][-1].item())\n",
    "    train_losses += results['train_loss']\n",
    "    test_losses += results['test_loss']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f399ffd5",
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.rcParams['figure.dpi'] = 600\n",
    "plt.plot(train_losses)\n",
    "plt.plot(test_losses)\n",
    "plt.legend(['train', 'test'])\n",
    "plt.ylabel('RMSE')\n",
    "plt.xlabel('step')\n",
    "plt.yscale('log')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "6e0204b8",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Mean Squared Error (MSE): 9.3530e-03\n",
      "Root Mean Squared Error (RMSE): 9.6711e-02\n",
      "Mean Absolute Error (MAE): 7.2150e-02\n"
     ]
    }
   ],
   "source": [
    "import torch\n",
    "import numpy as np\n",
    "\n",
    "device = torch.device('cuda' if torch.cuda.is_available() else 'cpu')\n",
    "n_samples = 1000\n",
    "random_points = torch.rand(n_samples, 100, device=device)\n",
    "Z_real = f(random_points).cpu().numpy()\n",
    "Z_pred = model(random_points).cpu().detach().numpy()\n",
    "Z_diff = Z_real - Z_pred\n",
    "mse = np.mean((Z_real - Z_pred) ** 2)\n",
    "rmse = np.sqrt(mse)\n",
    "mae = np.mean(np.abs(Z_real - Z_pred))\n",
    "print(f\"Mean Squared Error (MSE): {mse:.4e}\")\n",
    "print(f\"Root Mean Squared Error (RMSE): {rmse:.4e}\")\n",
    "print(f\"Mean Absolute Error (MAE): {mae:.4e}\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "f53644fe",
   "metadata": {},
   "source": [
    "### Two-layer KAN\n",
    "\n",
    "Now we show that a 2 two-layer KAN performs much worse for this task"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ae7b654b",
   "metadata": {},
   "outputs": [],
   "source": [
    "from kan import KAN, create_dataset\n",
    "import torch\n",
    "\n",
    "# create a KAN: 2D inputs, 1D output, and 5 hidden neurons. cubic spline (k=3), 5 grid intervals (grid=5).\n",
    "model = KAN(width=[4,9,1], grid=3, k=3, seed=0)\n",
    "f = lambda x: torch.exp((torch.sin(torch.pi*(x[:,[0]]**2+x[:,[1]]**2))+torch.sin(torch.pi*(x[:,[2]]**2+x[:,[3]]**2)))/2)\n",
    "dataset = create_dataset(f, n_var=4, train_num=3000)\n",
    "\n",
    "# train the model\n",
    "model.fit(dataset, opt=\"LBFGS\", steps=20, lamb=0.002, lamb_entropy=2.);\n",
    "model.plot(beta=10)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "869828f2",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "checkpoint directory created: ./model\n",
      "saving model version 0.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "| train_loss: 1.98e-02 | test_loss: 2.21e-02 | reg: 1.70e+01 | : 100%|█| 50/50 [00:15<00:00,  3.23it\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "saving model version 0.1\n",
      "checkpoint directory created: ./model\n",
      "saving model version 0.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "| train_loss: 1.15e-02 | test_loss: 1.40e-02 | reg: 1.71e+01 | : 100%|█| 50/50 [00:13<00:00,  3.75it\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "saving model version 0.1\n",
      "checkpoint directory created: ./model\n",
      "saving model version 0.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "| train_loss: 6.69e-03 | test_loss: 9.05e-03 | reg: 1.72e+01 | : 100%|█| 50/50 [00:13<00:00,  3.69it\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "saving model version 0.1\n",
      "checkpoint directory created: ./model\n",
      "saving model version 0.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "| train_loss: 4.38e-03 | test_loss: 8.05e-03 | reg: 1.73e+01 | : 100%|█| 50/50 [00:15<00:00,  3.17it\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "saving model version 0.1\n",
      "checkpoint directory created: ./model\n",
      "saving model version 0.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "| train_loss: 2.02e-03 | test_loss: 9.89e-03 | reg: 1.73e+01 | : 100%|█| 50/50 [00:17<00:00,  2.88it"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "saving model version 0.1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "grids = [3,5,10,20,50]\n",
    "\n",
    "train_rmse = []\n",
    "test_rmse = []\n",
    "\n",
    "for i in range(len(grids)):\n",
    "    #model = KAN(width=[4,9,1], grid=grids[i], k=3, seed=0).initialize_from_another_model(model, dataset['train_input'])\n",
    "    model = model.refine(grid=grids[i])\n",
    "    results = model.fit(dataset, opt=\"LBFGS\", steps=50, stop_grid_update_step=30);\n",
    "    train_rmse.append(results['train_loss'][-1].item())\n",
    "    test_rmse.append(results['test_loss'][-1].item())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "4f0a99fd",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[0.01983197219669819, 0.01147659495472908, 0.006687900051474571, 0.004380852449685335, 0.002016218611970544]\n",
      "[0.022097894921898842, 0.013952379114925861, 0.009049860760569572, 0.008054238744080067, 0.00989140197634697]\n"
     ]
    },
    {
     "data": {
      "image/png": "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4sgLERHVilQqhbm5OQAgNzcXMpkM/LmY6hLDCxGRnjtx4gSGDBkCFxcXSCQS7N69u9Q9a9euhYeHB0xNTREQEICTJ0+qPJ6amgo/Pz+4ublh1qxZsLe3r6fqSR8xvBAR6bmsrCz4+flhzZo1ZT7+yy+/YPr06Zg7dy4iIiLQtWtXDBw4EPfv31feY2Njg6ioKNy5cwfbt29HQkJCfZVPeohrXoiISEkikWDXrl0YNmyY8lqHDh3g7++PdevWKa95enpi2LBhWLRoUannmDx5Mnr16oXRo0fXR8mkhzjyQkRE5crPz0d4eDj69euncr1fv344ffo0ACAhIQHp6ekAFCdPnzhxAq1atar3Wkl/GIpdABERaa7k5GTIZDI4OjqqXHd0dMSjR48AAHFxcXjttdcgCAIEQcC0adPg6+srRrmkJxheiIioUhKJROVzQRCU1wICAhAZGSlCVaSvOG1ERETlsre3h1QqVY6yFEtMTCw1GkNUXxheiIioXMbGxggICMChQ4dUrh86dAhBQUEiVUX6jtNGRER6LjMzEzdv3lR+fufOHURGRsLW1haNGzfGjBkzEBISgnbt2qFTp05Yv3497t+/j7feekvEqkmfcas0EZGeO3bsGHr27Fnq+vjx47F582YAiiZ1ixcvRnx8PLy9vbF8+XJ069atnislUmB4ISIiIq3CNS9ERESkVRheiIiISKswvBAREZFWYXghIiIircLwQkRERFqF4YWIiIi0CsMLERERaRWGFyIiItIqDC9ERESkVRheiIiISKswvBAREZFWYXghIiIirfL/+k0afJi6DFgAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "n_params = np.array(grids) * (4*9+9*1)\n",
    "plt.plot(n_params, train_rmse, marker=\"o\")\n",
    "plt.plot(n_params, test_rmse, marker=\"o\")\n",
    "plt.plot(n_params, 300*n_params**(-2.), color=\"black\", ls=\"--\")\n",
    "plt.legend(['train', 'test', r'$N^{-4}$'], loc=\"lower left\")\n",
    "plt.xscale('log')\n",
    "plt.yscale('log')\n",
    "print(train_rmse)\n",
    "print(test_rmse)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "5776b6e1",
   "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.10.14"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
