{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import os, sys \n",
    "sys.path.append(\"..\")\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "import torch\n",
    "import torchvision\n",
    "import torchvision.datasets as datasets\n",
    "import torchvision.models as models\n",
    "import torchvision.transforms as transforms\n",
    "\n",
    "from PIL import Image\n",
    "from torch import nn\n",
    "from torchvision.transforms import Compose, Resize, Normalize, ToTensor\n",
    "from src import distributions\n",
    "from src.tools import test_accuracy\n",
    "from src.mnistm_utils import MNISTM"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Load Data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "IMG_SIZE = 32\n",
    "transform = Compose([\n",
    "    Resize((IMG_SIZE, IMG_SIZE)), \n",
    "    ToTensor(),\n",
    "    Normalize((0.5), (0.5)),\n",
    "])\n",
    "DATASET_PATH = '../datasets/'\n",
    "DATASET = 'mnist'\n",
    "\n",
    "model = models.resnet18()\n",
    "if DATASET == 'kmnist':\n",
    "    dataset = datasets.KMNIST(DATASET_PATH, train=True, download=True, transform=transform)\n",
    "    testset = datasets.KMNIST(DATASET_PATH, train=True, download=True, transform=transform)\n",
    "    model.conv1 = nn.Conv2d(1, 64, kernel_size=(7, 7), stride=(2, 2), padding=(3, 3), bias=False)\n",
    "    model.fc =  nn.Linear(in_features=512, out_features=10, bias=True)\n",
    "    NC = 1\n",
    "\n",
    "elif DATASET == 'mnist':\n",
    "    dataset = datasets.MNIST(DATASET_PATH,  train=True, download=True, transform=transform)\n",
    "    testset = datasets.MNIST(DATASET_PATH, train=False, download=True, transform=transform)\n",
    "    model.conv1 = nn.Conv2d(1, 64, kernel_size=(7, 7), stride=(2, 2), padding=(3, 3), bias=False)\n",
    "    model.fc =  nn.Linear(in_features=512, out_features=10, bias=True)\n",
    "    NC = 1\n",
    "    \n",
    "elif DATASET == 'usps':\n",
    "    dataset = datasets.USPS(DATASET_PATH,  train=True, download=True, transform=transform)\n",
    "    testset = datasets.USPS(DATASET_PATH, train=False, download=True, transform=transform)\n",
    "    model.conv1 = nn.Conv2d(1, 64, kernel_size=(7, 7), stride=(2, 2), padding=(3, 3), bias=False)\n",
    "    model.fc =  nn.Linear(in_features=512, out_features=10, bias=True)\n",
    "    NC = 1\n",
    "\n",
    "elif DATASET == 'mnistm':\n",
    "    dataset = MNISTM(DATASET_PATH,  train=True, download=True, transform=transform)\n",
    "    testset = MNISTM(DATASET_PATH, train=False, download=True, transform=transform)\n",
    "    model.fc =  nn.Linear(in_features=512, out_features=10, bias=True)\n",
    "    NC = 3\n",
    "    \n",
    "trainloader = torch.utils.data.DataLoader(dataset, batch_size=64, shuffle=True)\n",
    "testloader = torch.utils.data.DataLoader(testset, batch_size=64, shuffle=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "def imshow(img):\n",
    "    img = img / 2 + 0.5\n",
    "    npimg = img.numpy()\n",
    "    plt.imshow(np.transpose(npimg, (1, 2, 0)))\n",
    "    plt.show()\n",
    "\n",
    "dataiter = iter(trainloader)\n",
    "images, labels = dataiter.next()\n",
    "classes = list(range(10))\n",
    "\n",
    "imshow(torchvision.utils.make_grid(images[:16]))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Define Classifier"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "model.cuda()\n",
    "optimizer = torch.optim.Adam(model.parameters(), lr=0.001)\n",
    "loss_fn = nn.CrossEntropyLoss()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Train"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "def fit(model, trainloader, testloader, optimizer, loss_fn, epochs=30):\n",
    "    model.train()\n",
    "    for epoch in range(epochs):\n",
    "        correct = 0\n",
    "        for batch_idx, (X_batch, y_batch) in enumerate(trainloader):            \n",
    "            X_batch, Y_batch = X_batch.cuda(), y_batch.cuda()\n",
    "            optimizer.zero_grad()\n",
    "            output = model.forward(X_batch)\n",
    "            loss = loss_fn(output, Y_batch)\n",
    "            loss.backward()\n",
    "            optimizer.step()\n",
    "            predicted = torch.max(output.data, 1)[1] \n",
    "            correct += (predicted == Y_batch).sum()\n",
    "        test_accuracy(model, testloader)\n",
    "    return model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy of the network: 97.71\n",
      "Accuracy of the network: 98.07\n",
      "Accuracy of the network: 98.78\n",
      "Accuracy of the network: 98.93\n",
      "Accuracy of the network: 98.51\n",
      "Accuracy of the network: 99.26\n",
      "Accuracy of the network: 98.97\n",
      "Accuracy of the network: 98.84\n",
      "Accuracy of the network: 99.18\n",
      "Accuracy of the network: 99.21\n"
     ]
    }
   ],
   "source": [
    "model = fit(model, trainloader, testloader, optimizer, loss_fn, 10)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "model.eval()\n",
    "torch.save(model.cpu().state_dict(), '../saved_models/classifiers/{}.pt'.format(DATASET))"
   ]
  }
 ],
 "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.7.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
