{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Combine Good Twin-System Explanation with Clustering of FAMs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The autoreload extension is already loaded. To reload it, use:\n",
      "  %reload_ext autoreload\n"
     ]
    }
   ],
   "source": [
    "import pandas as pd\n",
    "import numpy as np\n",
    "import pickle\n",
    "import time\n",
    "import scipy\n",
    "import matplotlib.pyplot as plt\n",
    "import torchvision.models as models\n",
    "\n",
    "from sklearn.neighbors import KNeighborsClassifier\n",
    "\n",
    "from functions import *\n",
    "\n",
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 36,
   "metadata": {},
   "outputs": [],
   "source": [
    "# dataroot = 'mnist_normal'\n",
    "# dataroot = 'mnist_compressed_fams'\n",
    "# dataroot = 'mnist_novel_cnn'\n",
    "dataroot = 'mnist_maxpool'"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 37,
   "metadata": {},
   "outputs": [],
   "source": [
    "netC = load_cnn(dataroot + '/cnn.pth', NUM_CLASSES, NUM_FEATURES)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 38,
   "metadata": {},
   "outputs": [],
   "source": [
    "weights = netC.get_weights()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 39,
   "metadata": {},
   "outputs": [],
   "source": [
    "train_loader, test_loader = load_dataloaders()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 40,
   "metadata": {},
   "outputs": [],
   "source": [
    "X_train, y_train, X_test, y_test = get_MNIST_data()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Sanity Checks"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 41,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "KNeighborsClassifier(algorithm='brute', n_neighbors=1)"
      ]
     },
     "execution_count": 41,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# def prepare_iteration():\n",
    "    \n",
    "X_train_c = np.load(dataroot + \"/X_train_cont.npy\")\n",
    "X_test_c = np.load(dataroot + \"/X_test_cont.npy\")\n",
    "X_train_x = np.load(dataroot + \"/X_train_x.npy\")\n",
    "X_test_x = np.load(dataroot + \"/X_test_x.npy\")\n",
    "X_train_y = np.load(dataroot + \"/X_train_y.npy\")\n",
    "X_test_y = np.load(dataroot + \"/X_test_y.npy\")\n",
    "\n",
    "# Fit COLE and DkNN\n",
    "KNN = KNeighborsClassifier(n_neighbors=1, algorithm=\"brute\") \n",
    "KNN.fit(X_train_c, X_train_y)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Check Agreement"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 45,
   "metadata": {},
   "outputs": [],
   "source": [
    "preds = KNN.predict(X_test_c)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 46,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([7, 2, 1, ..., 4, 5, 6])"
      ]
     },
     "execution_count": 46,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "preds"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 44,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([7, 2, 1, ..., 4, 5, 6])"
      ]
     },
     "execution_count": 44,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "X_test_y"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 48,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "9960"
      ]
     },
     "execution_count": 48,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "(preds == X_test_y).sum()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Examine Explanation with White Box"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "115\n",
      "193\n",
      "321\n",
      "445\n",
      "449\n",
      "551\n",
      "582\n",
      "938\n",
      "947\n",
      "1014\n",
      "1039\n",
      "1182\n",
      "1232\n",
      "1260\n",
      "1319\n",
      "1393\n",
      "1671\n",
      "1681\n",
      "1901\n",
      "1982\n",
      "2040\n",
      "2118\n",
      "2129\n",
      "2130\n",
      "2135\n",
      "2182\n",
      "2314\n",
      "2414\n",
      "2447\n",
      "2462\n",
      "2597\n",
      "2654\n",
      "2860\n",
      "2939\n",
      "2953\n",
      "2995\n",
      "3030\n",
      "3062\n",
      "3073\n",
      "3225\n",
      "3422\n",
      "3520\n",
      "3767\n",
      "3808\n",
      "4265\n",
      "4571\n",
      "4740\n",
      "4823\n",
      "4837\n",
      "4860\n",
      "5937\n",
      "5955\n",
      "6576\n",
      "6625\n",
      "8316\n",
      "8408\n",
      "9015\n",
      "9505\n",
      "9679\n",
      "9729\n",
      "9839\n"
     ]
    }
   ],
   "source": [
    "for i in range(len(X_test)):\n",
    "    if X_test_y[i] != y_test[i]:\n",
    "        print(i)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 60,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Prediction: 7    Label: tensor(7)\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "<matplotlib.image.AxesImage at 0x15e384cd0>"
      ]
     },
     "execution_count": 60,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "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": [
    "test_instance = 0\n",
    "pred = X_test_y[test_instance]\n",
    "label = y_test[test_instance]\n",
    "print(\"Prediction:\", pred, \"   Label:\", label)\n",
    "plt.imshow(X_test[test_instance])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 61,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Query Label: 7\n",
      "Prediction: 7\n",
      " \n",
      "Neighbors:\n",
      "tensor(7) Prediction 7\n",
      "tensor(7) Prediction 7\n",
      "tensor(7) Prediction 7\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 4 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "CHP_nns = plot_query_and_nns_CHP(X_test_c, X_test, X_train,\n",
    "                                         test_instance, KNN, y_test, \n",
    "                                         y_train, netC, pred, X_train_y)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## 2: Look at FAMs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 62,
   "metadata": {},
   "outputs": [],
   "source": [
    "for i, data in enumerate(test_loader):\n",
    "    if i == test_instance:\n",
    "        break\n",
    "        \n",
    "logits, x, C = netC(data[0])\n",
    "pred = torch.argmax(logits).item()\n",
    "\n",
    "test_weights = weights[pred].detach().numpy()\n",
    "query_act_cam = weights.reshape(60,1,1) * C[0].detach().numpy()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 63,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([[-10.9210,  -2.4328,  -7.4897,  -8.3988,   0.2748,  -8.7705, -10.1062,\n",
       "          13.4845,  -8.6310,  -1.6887]], grad_fn=<CopySlices>)"
      ]
     },
     "execution_count": 63,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "logits"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 64,
   "metadata": {},
   "outputs": [],
   "source": [
    "neigh_act_cam = list()\n",
    "\n",
    "for i, data in enumerate(train_loader):\n",
    "    if i in CHP_nns:\n",
    "        logits, _, C = netC(data[0])\n",
    "        nn_act_cam = weights.reshape(60,1,1) * C[0].detach().numpy()\n",
    "\n",
    "        neigh_act_cam.append(nn_act_cam)\n",
    "        \n",
    "    if len(neigh_act_cam) == 3:\n",
    "        break"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Query Visualization"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 65,
   "metadata": {},
   "outputs": [],
   "source": [
    "c = x * weights\n",
    "c = c[0].tolist()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 72,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(-0.5, 27.5, 27.5, -0.5)"
      ]
     },
     "execution_count": 72,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 720x720 with 6 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "f, axarr = plt.subplots(3,2, figsize=(10,10))\n",
    "\n",
    "fam_idx = NUM_FEATURES * pred\n",
    "image = X_test[test_instance]\n",
    "axarr[0,0].imshow(image, alpha=1)\n",
    "activations = query_act_cam[fam_idx]\n",
    "idx = (abs(activations)==torch.max(  abs(activations) )).nonzero()[0]\n",
    "temp = np.zeros((7,7))\n",
    "temp[idx[0]][idx[1]] = abs(activations[idx[0]][idx[1]])\n",
    "FAM = scipy.ndimage.zoom(temp, (4, 4), order=3) \n",
    "axarr[0,0].imshow(FAM, cmap='jet', alpha=0.4)\n",
    "axarr[0,0].set_title( round(c[fam_idx], 3) )\n",
    "axarr[0,0].axis('off')\n",
    "\n",
    "fam_idx = NUM_FEATURES * pred + 1\n",
    "axarr[0,1].imshow(image, alpha=1)\n",
    "activations = query_act_cam[fam_idx]\n",
    "idx = (abs(activations)==torch.max(  abs(activations) )).nonzero()[0]\n",
    "temp = np.zeros((7,7))\n",
    "temp[idx[0]][idx[1]] = abs(activations[idx[0]][idx[1]])\n",
    "FAM = scipy.ndimage.zoom(temp, (4, 4), order=3) \n",
    "axarr[0,1].imshow(FAM, cmap='jet', alpha=0.4)\n",
    "axarr[0,1].set_title( round(c[fam_idx], 3) )\n",
    "axarr[0,1].axis('off')\n",
    "\n",
    "fam_idx = NUM_FEATURES * pred + 2\n",
    "axarr[1,0].imshow(image, alpha=1)\n",
    "activations = query_act_cam[fam_idx]\n",
    "idx = (abs(activations)==torch.max(  abs(activations) )).nonzero()[0]\n",
    "temp = np.zeros((7,7))\n",
    "temp[idx[0]][idx[1]] = abs(activations[idx[0]][idx[1]])\n",
    "FAM = scipy.ndimage.zoom(temp, (4, 4), order=3) \n",
    "axarr[1,0].imshow(FAM, cmap='jet', alpha=0.4)\n",
    "axarr[1,0].set_title( round(c[fam_idx], 3) )\n",
    "axarr[1,0].axis('off')\n",
    "\n",
    "fam_idx = NUM_FEATURES * pred + 3\n",
    "axarr[1,1].imshow(image, alpha=1)\n",
    "activations = query_act_cam[fam_idx]\n",
    "idx = (abs(activations)==torch.max(  abs(activations) )).nonzero()[0]\n",
    "temp = np.zeros((7,7))\n",
    "temp[idx[0]][idx[1]] = abs(activations[idx[0]][idx[1]])\n",
    "FAM = scipy.ndimage.zoom(temp, (4, 4), order=3) \n",
    "axarr[1,1].imshow(FAM, cmap='jet', alpha=0.4)\n",
    "axarr[1,1].set_title( round(c[fam_idx], 3) )\n",
    "axarr[1,1].axis('off')\n",
    "\n",
    "fam_idx = NUM_FEATURES * pred + 4\n",
    "axarr[2,0].imshow(image, alpha=1)\n",
    "activations = query_act_cam[fam_idx]\n",
    "idx = (abs(activations)==torch.max(  abs(activations) )).nonzero()[0]\n",
    "temp = np.zeros((7,7))\n",
    "temp[idx[0]][idx[1]] = abs(activations[idx[0]][idx[1]])\n",
    "FAM = scipy.ndimage.zoom(temp, (4, 4), order=3) \n",
    "axarr[2,0].imshow(FAM, cmap='jet', alpha=0.4)\n",
    "axarr[2,0].set_title( round(c[fam_idx], 3) )\n",
    "axarr[2,0].axis('off')\n",
    "\n",
    "fam_idx = NUM_FEATURES * pred + 5\n",
    "axarr[2,1].imshow(image, alpha=1)\n",
    "activations = query_act_cam[fam_idx]\n",
    "idx = (abs(activations)==torch.max(  abs(activations) )).nonzero()[0]\n",
    "temp = np.zeros((7,7))\n",
    "temp[idx[0]][idx[1]] = abs(activations[idx[0]][idx[1]])\n",
    "FAM = scipy.ndimage.zoom(temp, (4, 4), order=3) \n",
    "axarr[2,1].imshow(FAM, cmap='jet', alpha=0.4)\n",
    "axarr[2,1].set_title( round(c[fam_idx], 3) )\n",
    "axarr[2,1].axis('off')"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Query and NNs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 69,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(-0.5, 27.5, 27.5, -0.5)"
      ]
     },
     "execution_count": 69,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 720x720 with 4 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "f, axarr = plt.subplots(2,2, figsize=(10,10))\n",
    "\n",
    "fam_idx = NUM_FEATURES * pred + 4\n",
    "\n",
    "image = X_test[test_instance]\n",
    "axarr[0,0].imshow(image, alpha=1)\n",
    "activations = query_act_cam[fam_idx]\n",
    "idx = (abs(activations)==torch.max(  abs(activations) )).nonzero()[0]\n",
    "temp = np.zeros((7,7))\n",
    "temp[idx[0]][idx[1]] = abs(activations[idx[0]][idx[1]])\n",
    "FAM = scipy.ndimage.zoom(temp, (4, 4), order=3) \n",
    "axarr[0,0].imshow(FAM, cmap='jet', alpha=0.4)\n",
    "axarr[0,0].set_title( round(c[fam_idx], 3) )\n",
    "axarr[0,0].axis('off')\n",
    "\n",
    "image = X_train[CHP_nns[0]]   \n",
    "axarr[0,1].imshow(image, alpha=1)\n",
    "activations = neigh_act_cam[0][fam_idx]\n",
    "idx = (abs(activations)==torch.max( abs(activations) )).nonzero()[0]\n",
    "temp = np.zeros((7,7))\n",
    "temp[idx[0]][idx[1]] = abs(activations[idx[0]][idx[1]])\n",
    "FAM = scipy.ndimage.zoom(temp, (4, 4), order=3) \n",
    "axarr[0,1].imshow(FAM, cmap='jet', alpha=0.4)\n",
    "axarr[0,1].set_title( round( (temp[idx[0]][idx[1]] * weights[fam_idx]).item() , 3) )\n",
    "axarr[0,1].axis('off')\n",
    "\n",
    "image = X_train[CHP_nns[1]]   \n",
    "axarr[1,0].imshow(image, alpha=1)\n",
    "activations = neigh_act_cam[1][fam_idx]\n",
    "idx = (abs(activations)==torch.max( abs(activations) )).nonzero()[0]\n",
    "temp = np.zeros((7,7))\n",
    "temp[idx[0]][idx[1]] = abs(activations[idx[0]][idx[1]])\n",
    "FAM = scipy.ndimage.zoom(temp, (4, 4), order=3) \n",
    "axarr[1,0].imshow(FAM, cmap='jet', alpha=0.4)\n",
    "axarr[1,0].set_title( round( (temp[idx[0]][idx[1]] * weights[fam_idx]).item() , 3) )\n",
    "axarr[1,0].axis('off')\n",
    "\n",
    "image = X_train[CHP_nns[2]]   \n",
    "axarr[1,1].imshow(image, alpha=1)\n",
    "activations = neigh_act_cam[2][fam_idx]\n",
    "idx = (abs(activations)==torch.max( abs(activations) )).nonzero()[0]\n",
    "temp = np.zeros((7,7))\n",
    "temp[idx[0]][idx[1]] = abs(activations[idx[0]][idx[1]])\n",
    "FAM = scipy.ndimage.zoom(temp, (4, 4), order=3) \n",
    "axarr[1,1].imshow(FAM, cmap='jet', alpha=0.4)\n",
    "axarr[1,1].set_title( round( (temp[idx[0]][idx[1]] * weights[fam_idx]).item() , 3) )\n",
    "axarr[1,1].axis('off')"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## CAM"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 36,
   "metadata": {},
   "outputs": [],
   "source": [
    "### for CAMs\n",
    "query_cam = query_act_cam.sum(0)\n",
    "\n",
    "nns_cams = list()\n",
    "for i in range(len(neigh_act_cam)):\n",
    "    nns_cams.append(neigh_act_cam[i].sum(0))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 37,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(-0.5, 27.5, 27.5, -0.5)"
      ]
     },
     "execution_count": 37,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 4 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "f, axarr = plt.subplots(2,2)\n",
    "\n",
    "image = X_test[test_instance]\n",
    "axarr[0,0].imshow(image, alpha=1)\n",
    "activations = query_cam\n",
    "FAM = scipy.ndimage.zoom(activations, (4, 4), order=3) \n",
    "axarr[0,0].imshow(FAM, cmap='jet', alpha=0.4)\n",
    "\n",
    "image = X_train[CHP_nns[0]]   \n",
    "axarr[0,1].imshow(image, alpha=1)\n",
    "activations = nns_cams[0]\n",
    "FAM = scipy.ndimage.zoom(activations, (4, 4), order=3) \n",
    "axarr[0,1].imshow(FAM, cmap='jet', alpha=0.4)\n",
    "\n",
    "image = X_train[CHP_nns[1]]   \n",
    "axarr[1,0].imshow(image, alpha=1)\n",
    "activations = nns_cams[1]\n",
    "FAM = scipy.ndimage.zoom(activations, (4, 4), order=3) \n",
    "axarr[1,0].imshow(FAM, cmap='jet', alpha=0.4)\n",
    "\n",
    "image = X_train[CHP_nns[2]]   \n",
    "axarr[1,1].imshow(image, alpha=1)\n",
    "activations = nns_cams[2]\n",
    "FAM = scipy.ndimage.zoom(activations, (4, 4), order=3) \n",
    "axarr[1,1].imshow(FAM, cmap='jet', alpha=0.4)\n",
    "\n",
    "plt.axis('off')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 38,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([[  0. ,   0. ,   0. ,   0. ,   0. ,   0. ,   0. ],\n",
       "       [  0. ,   0. ,   0. ,   0. ,   0. ,   0. ,   0. ],\n",
       "       [  0. ,   0. ,   0. ,  -2.6,  -5.8,   0. ,   0. ],\n",
       "       [  0. ,   0. ,  -1.8,  -5.9, -38.2,  -0.7,   0. ],\n",
       "       [  0. ,   0. ,  -6.7,  42.1,  78.6,  25.4,   0. ],\n",
       "       [  0. ,   0. ,   0.4,   0.4,   4.1,   0. ,   0. ],\n",
       "       [  0. ,   0. ,   0. ,   0. ,   0. ,   0. ,   0. ]], dtype=float32)"
      ]
     },
     "execution_count": 38,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "query_cam.round(1)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Only Show Positive Features FAM Added Up"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 39,
   "metadata": {},
   "outputs": [],
   "source": [
    "### for full FAMs\n",
    "query_act_cam[query_act_cam < 0] = 0\n",
    "query_cam = query_act_cam.sum(0)\n",
    "\n",
    "nns_cams = list()\n",
    "for i in range(len(neigh_act_cam)):\n",
    "    neigh_act_cam[i][neigh_act_cam[i] < 0] = 0\n",
    "    nns_cams.append(neigh_act_cam[i].sum(0))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 40,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(-0.5, 27.5, 27.5, -0.5)"
      ]
     },
     "execution_count": 40,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 4 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "f, axarr = plt.subplots(2,2)\n",
    "\n",
    "image = X_test[test_instance]\n",
    "axarr[0,0].imshow(image, alpha=1)\n",
    "activations = query_cam\n",
    "FAM = scipy.ndimage.zoom(activations, (4, 4), order=3) \n",
    "axarr[0,0].imshow(FAM, cmap='jet', alpha=0.4)\n",
    "\n",
    "image = X_train[CHP_nns[0]]   \n",
    "axarr[0,1].imshow(image, alpha=1)\n",
    "activations = nns_cams[0]\n",
    "FAM = scipy.ndimage.zoom(activations, (4, 4), order=3) \n",
    "axarr[0,1].imshow(FAM, cmap='jet', alpha=0.4)\n",
    "\n",
    "image = X_train[CHP_nns[1]]   \n",
    "axarr[1,0].imshow(image, alpha=1)\n",
    "activations = nns_cams[1]\n",
    "FAM = scipy.ndimage.zoom(activations, (4, 4), order=3) \n",
    "axarr[1,0].imshow(FAM, cmap='jet', alpha=0.4)\n",
    "\n",
    "image = X_train[CHP_nns[2]]   \n",
    "axarr[1,1].imshow(image, alpha=1)\n",
    "activations = nns_cams[2]\n",
    "FAM = scipy.ndimage.zoom(activations, (4, 4), order=3) \n",
    "axarr[1,1].imshow(FAM, cmap='jet', alpha=0.4)\n",
    "\n",
    "plt.axis('off')"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Test How Big Are Features"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 41,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([[ 0. ,  0. ,  0. ,  0. ,  0. ,  0. ,  0. ],\n",
       "       [ 0. ,  0. ,  0. ,  0. ,  0. ,  0. ,  0. ],\n",
       "       [ 0. ,  0. ,  0. ,  0. ,  0. ,  0. ,  0. ],\n",
       "       [ 0. ,  0. ,  5.4, 18.3,  0. ,  0. ,  0. ],\n",
       "       [ 0. ,  0. ,  1.7, 55.8, 78.6, 25.4,  0. ],\n",
       "       [ 0. ,  0. ,  0.4,  0.4,  4.1,  0. ,  0. ],\n",
       "       [ 0. ,  0. ,  0. ,  0. ,  0. ,  0. ,  0. ]], dtype=float32)"
      ]
     },
     "execution_count": 41,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "query_cam.round(1)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Test accuracy and params "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {},
   "outputs": [],
   "source": [
    "def test_accuracy(netC, test_loader):\n",
    "\tnetC = netC.eval()\n",
    "\tcorrect = 0\n",
    "\ttotal = 0\n",
    "\twith torch.no_grad():\n",
    "\t\tfor i, data in enumerate(test_loader):\n",
    "\t\t\tinputs, labels = data[0], data[1].flatten().long()\n",
    "\t\t\tlogits, _, _ = netC(inputs)\n",
    "\t\t\tpredicted = torch.argmax(logits, 1)\n",
    "\t\t\ttotal += labels.size(0)\n",
    "\t\t\tcorrect += (predicted == labels).sum().item()\n",
    "\tnetC = netC.train()\n",
    "\treturn 100 * correct / total"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "99.59"
      ]
     },
     "execution_count": 23,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "test_accuracy(netC, test_loader)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "removing 1 increased accuracy to 99.63\n",
    "\n",
    "removing 2 increased accuracy to 99.64"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Num Params: 143162\n"
     ]
    }
   ],
   "source": [
    "model_parameters = filter(lambda p: p.requires_grad, netC.parameters())\n",
    "params = sum([np.prod(p.size()) for p in model_parameters])\n",
    "print('Num Params:', params)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Get List of Bad Data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_bad_data(num_neighbors=1):\n",
    "    bad_data_list = list()\n",
    "    for i in range(len(X_test)):\n",
    "        if X_test_y[i] != y_test[i]:\n",
    "            pred = X_test_y[test_instance]\n",
    "            CHP_nns = get_nns(X_test_c, X_test, X_train,\n",
    "                                                     i, KNN, y_test, \n",
    "                                                     y_train, netC, pred, X_train_y,\n",
    "                                                     n_neigh=num_neighbors)\n",
    "            bad_data_list.append(CHP_nns[0:num_neighbors].tolist())\n",
    "            \n",
    "    return bad_data_list"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "metadata": {},
   "outputs": [],
   "source": [
    "bad_data_list = np.array(get_bad_data(3)).flatten()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([  892, 20572,  5684, 12000, 18037,   588, 25086, 27538, 32188,\n",
       "       14740, 33626, 55206,  8200, 39184, 50403, 10780, 59385, 16918,\n",
       "       11044, 18031, 55340, 48382, 23422,  1120,  8600, 54718, 48084,\n",
       "        1356, 24426, 52914, 58022, 39354, 49786, 32002, 29843,  9534,\n",
       "       37648,   754,  4264, 43014, 29765, 52085, 33625, 34338,  5589,\n",
       "        2676, 57540, 48102, 32618, 38680, 29638, 25922, 37374,  4068,\n",
       "       24426, 28455,  1356,  8220, 48036, 17723, 36348, 44340, 39425,\n",
       "       39631, 29895, 26881, 35916,  5042, 22320,  7867, 19960, 26622,\n",
       "       47340, 37834,   212, 35224, 29890, 47503,  5704, 38680, 35068,\n",
       "       47996, 14528, 20918, 27085, 27121, 44172, 21889, 24250, 47034,\n",
       "       37441, 49355, 43984, 48460, 11632, 22284, 12808,  1614, 34484,\n",
       "       49744, 26760, 45520, 34841, 34817, 17645,  5821,  8268, 39877,\n",
       "       39877, 53113, 38705, 57540,  2576, 49212, 36439, 31289, 54933,\n",
       "       41562, 16918, 29292, 12000, 18037,  5638])"
      ]
     },
     "execution_count": 33,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "bad_data_list"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "metadata": {
    "scrolled": false
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[59385,\n",
       " 58022,\n",
       " 57540,\n",
       " 57540,\n",
       " 55340,\n",
       " 55206,\n",
       " 54933,\n",
       " 54718,\n",
       " 53113,\n",
       " 52914,\n",
       " 52085,\n",
       " 50403,\n",
       " 49786,\n",
       " 49744,\n",
       " 49355,\n",
       " 49212,\n",
       " 48460,\n",
       " 48382,\n",
       " 48102,\n",
       " 48084,\n",
       " 48036,\n",
       " 47996,\n",
       " 47503,\n",
       " 47340,\n",
       " 47034,\n",
       " 45520,\n",
       " 44340,\n",
       " 44172,\n",
       " 43984,\n",
       " 43014,\n",
       " 41562,\n",
       " 39877,\n",
       " 39877,\n",
       " 39631,\n",
       " 39425,\n",
       " 39354,\n",
       " 39184,\n",
       " 38705,\n",
       " 38680,\n",
       " 38680,\n",
       " 37834,\n",
       " 37648,\n",
       " 37441,\n",
       " 37374,\n",
       " 36439,\n",
       " 36348,\n",
       " 35916,\n",
       " 35224,\n",
       " 35068,\n",
       " 34841,\n",
       " 34817,\n",
       " 34484,\n",
       " 34338,\n",
       " 33626,\n",
       " 33625,\n",
       " 32618,\n",
       " 32188,\n",
       " 32002,\n",
       " 31289,\n",
       " 29895,\n",
       " 29890,\n",
       " 29843,\n",
       " 29765,\n",
       " 29638,\n",
       " 29292,\n",
       " 28455,\n",
       " 27538,\n",
       " 27121,\n",
       " 27085,\n",
       " 26881,\n",
       " 26760,\n",
       " 26622,\n",
       " 25922,\n",
       " 25086,\n",
       " 24426,\n",
       " 24426,\n",
       " 24250,\n",
       " 23422,\n",
       " 22320,\n",
       " 22284,\n",
       " 21889,\n",
       " 20918,\n",
       " 20572,\n",
       " 19960,\n",
       " 18037,\n",
       " 18037,\n",
       " 18031,\n",
       " 17723,\n",
       " 17645,\n",
       " 16918,\n",
       " 16918,\n",
       " 14740,\n",
       " 14528,\n",
       " 12808,\n",
       " 12000,\n",
       " 12000,\n",
       " 11632,\n",
       " 11044,\n",
       " 10780,\n",
       " 9534,\n",
       " 8600,\n",
       " 8268,\n",
       " 8220,\n",
       " 8200,\n",
       " 7867,\n",
       " 5821,\n",
       " 5704,\n",
       " 5684,\n",
       " 5638,\n",
       " 5589,\n",
       " 5042,\n",
       " 4264,\n",
       " 4068,\n",
       " 2676,\n",
       " 2576,\n",
       " 1614,\n",
       " 1356,\n",
       " 1356,\n",
       " 1120,\n",
       " 892,\n",
       " 754,\n",
       " 588,\n",
       " 212]"
      ]
     },
     "execution_count": 34,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "bad_data_list = sorted(bad_data_list, reverse=True)\n",
    "bad_data_list"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Remove Bad Data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 164,
   "metadata": {},
   "outputs": [],
   "source": [
    "def remove_bad_data(trainset, bad_data_list):\n",
    "    \n",
    "    mask = np.ones(len(trainset.data), dtype=bool)\n",
    "    mask[bad_data_list] = False\n",
    "    trainset.data = trainset.data[mask]\n",
    "    trainset.targets = trainset.targets[mask]\n",
    "    \n",
    "    return trainset"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 102,
   "metadata": {},
   "outputs": [],
   "source": [
    "mnist_trainset = torchvision.datasets.MNIST(root='./data',\n",
    "                                            train=True, download=True, transform=None)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 103,
   "metadata": {},
   "outputs": [],
   "source": [
    "mnist_trainset = remove_bad_data(mnist_trainset, bad_data_list)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 104,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([59943, 28, 28])"
      ]
     },
     "execution_count": 104,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "mnist_trainset.data.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 105,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([59943])"
      ]
     },
     "execution_count": 105,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "mnist_trainset.targets.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "img_env",
   "language": "python",
   "name": "img_env"
  },
  "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.4"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
