{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "0322c98b",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2024-09-09T21:11:36.827309Z",
     "start_time": "2024-09-09T21:11:35.389484Z"
    }
   },
   "outputs": [],
   "source": [
    "from pathlib import Path\n",
    "from time import time\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.optim as optim\n",
    "import torchvision\n",
    "import torchvision.transforms as transforms\n",
    "from tqdm import trange\n",
    "\n",
    "from kgi import apply_kgi_to_model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "6b436357",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2024-09-09T21:11:36.830230Z",
     "start_time": "2024-09-09T21:11:36.828109Z"
    }
   },
   "outputs": [],
   "source": [
    "# attempt to enable LaTeX rendering\n",
    "# change to `False` if you get an error during plotting (latex not installed)\n",
    "plt.rcParams['text.usetex'] = True"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "ec6dc013-ae78-440b-a258-d8da142413d0",
   "metadata": {},
   "source": [
    "# Data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "b3431b07-4528-40ae-8c3e-204e4fbac9d6",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2024-09-09T21:11:42.392127Z",
     "start_time": "2024-09-09T21:11:36.832983Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n"
     ]
    }
   ],
   "source": [
    "def batchify(data, labels, batch_size):\n",
    "    \"\"\" manually batchify data \"\"\"\n",
    "    for i in range(0, len(data), batch_size):\n",
    "        yield data[i:i + batch_size], labels[i:i + batch_size]\n",
    "\n",
    "\n",
    "def load_dataset_to_gpu(batch_size=512, device=\"cuda\"):\n",
    "    \"\"\" load dataset in CPU, process, and move to GPU \"\"\"\n",
    "    transform = transforms.Compose([transforms.ToTensor(), ])\n",
    "    train_dataset = torchvision.datasets.CIFAR10(root='./datasets_torch',\n",
    "                                                 train=True, download=True, transform=transform)\n",
    "    test_dataset = torchvision.datasets.CIFAR10(root='./datasets_torch',\n",
    "                                                train=False, download=True, transform=transform)\n",
    "    train_data = torch.stack([train_dataset[i][0] for i in range(len(train_dataset))]).to(device)\n",
    "    train_labels = torch.tensor(\n",
    "        [train_dataset[i][1] for i in range(len(train_dataset))], dtype=torch.long).to(device)\n",
    "    test_data = torch.stack([test_dataset[i][0] for i in range(len(test_dataset))]).to(device)\n",
    "    test_labels = torch.tensor(\n",
    "        [test_dataset[i][1] for i in range(len(test_dataset))], dtype=torch.long).to(device)\n",
    "    train_batches = list(batchify(train_data, train_labels, batch_size))\n",
    "    test_batches = list(batchify(test_data, test_labels, batch_size))\n",
    "    return train_batches, test_batches\n",
    "\n",
    "\n",
    "device_ = \"cuda\" if torch.cuda.is_available() else \"cpu\"\n",
    "train_set, test_set = load_dataset_to_gpu(device=device_)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "12c74326",
   "metadata": {},
   "source": [
    "# Training"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "1ed6a052-6a2d-4609-81c2-9f872b03b4a0",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2024-09-09T21:11:42.395771Z",
     "start_time": "2024-09-09T21:11:42.393734Z"
    }
   },
   "outputs": [],
   "source": [
    "class MLP(nn.Module):\n",
    "    def __init__(self):\n",
    "        super(MLP, self).__init__()\n",
    "        self.fc1 = nn.Linear(32 * 32 * 3, 256)\n",
    "        self.fc2 = nn.Linear(256, 128)\n",
    "        self.fc_out = nn.Linear(128, 10)\n",
    "        self.dropout1 = nn.Dropout(0.1)\n",
    "        self.dropout2 = nn.Dropout(0.1)\n",
    "\n",
    "    def forward(self, x):\n",
    "        x = x.view(-1, 32 * 32 * 3)\n",
    "        x = torch.relu(self.fc1(x))\n",
    "        x = self.dropout1(x)\n",
    "        x = torch.relu(self.fc2(x))\n",
    "        x = self.dropout2(x)\n",
    "        x = self.fc_out(x)\n",
    "        return x"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "b2ec6c4f",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2024-09-09T21:11:42.401090Z",
     "start_time": "2024-09-09T21:11:42.399749Z"
    }
   },
   "outputs": [],
   "source": [
    "def train(kgi, seed=0, num_epochs=10000, save_every=10, progress_bar=True):\n",
    "    torch.manual_seed(seed)\n",
    "\n",
    "    # model\n",
    "    model = MLP()\n",
    "    if kgi:\n",
    "        apply_kgi_to_model(model, knot_low=0, knot_high=1,\n",
    "                           perturb_factor=0.2, kgi_by_bias=False,\n",
    "                           homogenous_factor=-1)\n",
    "    model = model.to(device_)\n",
    "    model.train()\n",
    "\n",
    "    # loss function and optimizer\n",
    "    criterion = nn.CrossEntropyLoss()\n",
    "    optimizer = optim.Adam(model.parameters(), lr=1e-3)\n",
    "\n",
    "    # train loop\n",
    "    loss_history = []\n",
    "    loop = trange(num_epochs, desc=\"Training Epochs\", disable=not progress_bar)\n",
    "    for epoch in loop:\n",
    "        running_loss = 0.0\n",
    "        for images, labels in train_set:\n",
    "            outputs = model(images)\n",
    "            loss = criterion(outputs, labels)\n",
    "            optimizer.zero_grad()\n",
    "            loss.backward()\n",
    "            optimizer.step()\n",
    "            running_loss += loss.item()\n",
    "        if (epoch + 1) % save_every == 0:\n",
    "            epoch_loss = running_loss / len(train_set)\n",
    "            loss_history.append(epoch_loss)\n",
    "            loop.set_postfix({\"Epoch Loss\": f\"{epoch_loss:.2e}\"})\n",
    "\n",
    "    # evaluation\n",
    "    correct = 0\n",
    "    total = 0\n",
    "    model.eval()\n",
    "    with torch.no_grad():\n",
    "        for images, labels in test_set:\n",
    "            outputs = model(images)\n",
    "            _, predicted = torch.max(outputs.data, 1)\n",
    "            total += labels.size(0)\n",
    "            correct += (predicted == labels).sum().item()  # noqa\n",
    "\n",
    "    accuracy = 100 * correct / total\n",
    "    if progress_bar:\n",
    "        print(f'Final Accuracy on test set: {accuracy:.2f}%')\n",
    "    return loss_history, accuracy"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "a823ad15",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2024-09-09T21:11:42.407272Z",
     "start_time": "2024-09-09T21:11:42.402956Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0_False exists\n",
      "0_True exists\n",
      "1_False exists\n",
      "1_True exists\n",
      "2_False exists\n",
      "2_True exists\n",
      "3_False exists\n",
      "3_True exists\n",
      "4_False exists\n",
      "4_True exists\n",
      "5_False exists\n",
      "5_True exists\n",
      "6_False exists\n",
      "6_True exists\n",
      "7_False exists\n",
      "7_True exists\n",
      "8_False exists\n",
      "8_True exists\n",
      "9_False exists\n",
      "9_True exists\n"
     ]
    }
   ],
   "source": [
    "# train all models\n",
    "seeds = list(range(10))  # use `seeds = [0]` for fast test\n",
    "epochs = 10000  # use a smaller one for fast test\n",
    "out_dir = Path(\"results/cifar10_paper\")\n",
    "\n",
    "out_dir.mkdir(exist_ok=True, parents=True)\n",
    "for seed_ in seeds:\n",
    "    for kgi_ in [False, True]:\n",
    "        name_ = f\"{seed_}_{kgi_}\"\n",
    "        if not (out_dir / name_).exists():\n",
    "            t0 = time()\n",
    "            hist_, acc = train(kgi_, seed_, epochs, progress_bar=True)\n",
    "            np.savetxt(out_dir / name_, hist_, header=f\"{acc}\")\n",
    "            print(f\"{name_} trained in {(time() - t0) / 60:.1f} min, \"\n",
    "                  f\"loss={hist_[-1]:.2e}, acc={acc:.2f}%\")\n",
    "        else:\n",
    "            print(f\"{name_} exists\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "39740eca-af8c-4aeb-aa7b-e2a5d1d3184c",
   "metadata": {},
   "source": [
    "# Analysis"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "ddcaaa2f-eedc-4fc4-bd99-bbbec8980c45",
   "metadata": {},
   "source": [
    "### Metrics"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "c19a5c61",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2024-09-09T21:11:42.433703Z",
     "start_time": "2024-09-09T21:11:42.408246Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "No KGI\n",
      "Loss: 0.26 ± 0.26\n",
      "slowness: 0.35 ± 0.03\n",
      "Accuracy: 46.6 ± 0.3%\n",
      "\n",
      "KGI\n",
      "Loss: 0.18 ± 0.18\n",
      "slowness: 0.28 ± 0.02\n",
      "Accuracy: 47.5 ± 0.3%\n",
      "\n",
      "Copy to table\n",
      "CIFAR-10 & $0.26\\!\\pm\\!0.03$ & $0.18\\!\\pm\\!0.02$ & $0.35\\!\\pm\\!0.03$ & $0.28\\!\\pm\\!0.02$ & Accuracy & $46.65\\!\\pm\\!0.31\\%$ & $47.53\\!\\pm\\!0.27\\%$\n"
     ]
    }
   ],
   "source": [
    "def print_metrics(kgi):\n",
    "    losses = []\n",
    "    slowness = []\n",
    "    accs = []\n",
    "    for seed in seeds:\n",
    "        # read history\n",
    "        name = f\"{seed}_{kgi}\"\n",
    "        hist = np.loadtxt(out_dir / name)\n",
    "        # use average of last 50 epochs (5 * 10) as final loss\n",
    "        final_loss = hist[-5:].mean()\n",
    "        losses.append(final_loss)\n",
    "        # AUC for convergence slowness\n",
    "        init_loss = hist[:10].mean()\n",
    "        slowness.append(hist.mean() / init_loss)\n",
    "        # read relative error\n",
    "        with open(out_dir / name) as fs:\n",
    "            acc_str = fs.readline()\n",
    "        accs.append(float(acc_str[1:]))\n",
    "    losses = np.array(losses)\n",
    "    slowness = np.array(slowness)\n",
    "    accs = np.array(accs)\n",
    "    print(\"\\nKGI\" if kgi else \"\\nNo KGI\")\n",
    "    print(f\"Loss: {losses.mean():.2f} ± {losses.mean():.2f}\")\n",
    "    print(f\"slowness: {slowness.mean():.2f} ± {slowness.std():.2f}\")\n",
    "    print(f\"Accuracy: {accs.mean():.1f} ± {accs.std():.1f}%\")\n",
    "    return losses.mean(), losses.std(), slowness.mean(), slowness.std(), accs.mean(), accs.std()\n",
    "\n",
    "\n",
    "lmf, lsf, smf, ssf, amf, asf = print_metrics(False)\n",
    "lmt, lst, smt, sst, amt, ast = print_metrics(True)\n",
    "print(\"\\nCopy to table\")\n",
    "print(f\"CIFAR-10 & \"\n",
    "      f\"${lmf:.2f}\\!\\pm\\!{lsf:.2f}$ & ${lmt:.2f}\\!\\pm\\!{lst:.2f}$ & \"\n",
    "      f\"${smf:.2f}\\!\\pm\\!{ssf:.2f}$ & ${smt:.2f}\\!\\pm\\!{sst:.2f}$ & Accuracy & \"\n",
    "      f\"${amf:.2f}\\!\\pm\\!{asf:.2f}\\%$ & ${amt:.2f}\\!\\pm\\!{ast:.2f}\\%$\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "b78ec181-7be4-46f2-9522-a13b1be479fb",
   "metadata": {},
   "source": [
    "### Loss history"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "f69588ac",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2024-09-09T21:11:43.006742Z",
     "start_time": "2024-09-09T21:11:42.421721Z"
    }
   },
   "outputs": [
    {
     "data": {
      "text/plain": "<Figure size 400x320 with 1 Axes>",
      "image/png": "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"
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "def moving_ave(series, window_size):\n",
    "    \"\"\" Moving average to smooth the loss history a little bit \"\"\"\n",
    "    return np.convolve(series, np.ones(window_size) / window_size, mode='valid')\n",
    "\n",
    "\n",
    "_, ax = plt.subplots(figsize=(5 / 2.5, 4 / 2.5), dpi=200)\n",
    "seed_ = 0\n",
    "hist_def = np.loadtxt(out_dir / f\"{seed_}_{False}\")\n",
    "hist_kgi = np.loadtxt(out_dir / f\"{seed_}_{True}\")\n",
    "ax.plot(np.arange(1, epochs + 1, 10), moving_ave(hist_def, 1), label=\"No KGI\", lw=1, c='b')\n",
    "ax.plot(np.arange(1, epochs + 1, 10), moving_ave(hist_kgi, 1), label=\"KGI\", lw=1, c='r')\n",
    "ax.set_xticks([0, 2000, 4000, 6000, 8000, 10000],\n",
    "              [0, 2, 4, 6, 8, 10])\n",
    "ax.set_ylabel(\"Cross entropy loss\")\n",
    "ax.set_xlabel(\"Epoch ($10^3$)\")\n",
    "ax.legend(ncol=2, handlelength=.8, columnspacing=.5, handletextpad=.4)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "33021307-c40b-43a6-99b4-8ed5e2693392",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2024-09-09T21:11:43.009538Z",
     "start_time": "2024-09-09T21:11:43.005461Z"
    }
   },
   "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.13"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
