{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Combine Good Twin-System Explanation with Clustering of FAMs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "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": 12,
   "metadata": {},
   "outputs": [],
   "source": [
    "# dataroot = 'mnist_normal'\n",
    "dataroot = 'mnist_super'\n",
    "# dataroot = 'mnist_maxpool'"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "cpu\n"
     ]
    }
   ],
   "source": [
    "NUM_EPOCHS = 30\n",
    "DEVICE = torch.device(\"cuda\" if (torch.cuda.is_available() and 1 > 0) else \"cpu\")\n",
    "BATCH_SIZE = 32\n",
    "NUM_CLASSES = 10\n",
    "NUM_FEATURES = 6\n",
    "print(DEVICE)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [],
   "source": [
    "netC = SuperCNN(NUM_CLASSES, NUM_FEATURES)\n",
    "netC.load_state_dict(torch.load('weights/super_cnn.pth', map_location=DEVICE))\n",
    "netC = netC.eval()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [],
   "source": [
    "weights = netC.get_weights()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [],
   "source": [
    "train_loader, test_loader = load_dataloaders()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "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": 100,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "KNeighborsClassifier(algorithm='brute', n_neighbors=1)"
      ]
     },
     "execution_count": 100,
     "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": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# class SuperTwin():\n",
    "    \n",
    "#     def __init__(self, NUM_CLASSES=10, NUM_FEATURES=6, X_train_c, X_train_y):\n",
    "        \n",
    "#         self.knns = list()\n",
    "#         for i in range(NUM_CLASSES):\n",
    "#             self.knns.append(KNeighborsClassifier(n_neighbors=1, algorithm=\"brute\"))\n",
    "            \n",
    "#         data    = [ [] for range in NUM_CLASSES ]\n",
    "#         targets = [ [] for range in NUM_CLASSES ]\n",
    "#         idxs    = [ [] for range in NUM_CLASSES ] \n",
    "#         for i in range(X_train_c.shape[0]):\n",
    "#             pred = X_train_y[i]\n",
    "#             data[pred].append(X_train_c[i])\n",
    "#             targets[pred].append(pred)\n",
    "#             idxs[pred].append(i)  # ????\n",
    "            \n",
    "#         for i in range(NUM_CLASSES):\n",
    "#             self.knns[i].fit(data[i], idxs[i])\n",
    "            \n",
    "        \n",
    "    "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 98,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(60000, 6)"
      ]
     },
     "execution_count": 98,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "X_train_c.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 99,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(60000,)"
      ]
     },
     "execution_count": 99,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "X_train_y.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 101,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "9952"
      ]
     },
     "execution_count": 101,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "(test_loader.dataset.targets.detach().numpy() == X_test_y).sum()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Check Agreement"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 83,
   "metadata": {},
   "outputs": [],
   "source": [
    "preds = KNN.predict(X_test_c)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 84,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([7, 2, 1, ..., 4, 5, 6])"
      ]
     },
     "execution_count": 84,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "preds"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 85,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([7, 2, 1, ..., 4, 5, 6])"
      ]
     },
     "execution_count": 85,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "X_test_y"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 87,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.7701"
      ]
     },
     "execution_count": 87,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "(preds == X_test_y).sum() / len(X_test_y)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Examine Explanation with White Box"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 102,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "115\n",
      "445\n",
      "460\n",
      "582\n",
      "646\n",
      "947\n",
      "1014\n",
      "1039\n",
      "1226\n",
      "1232\n",
      "1242\n",
      "1247\n",
      "1260\n",
      "1393\n",
      "1878\n",
      "1901\n",
      "2118\n",
      "2129\n",
      "2130\n",
      "2225\n",
      "2293\n",
      "2447\n",
      "2597\n",
      "2896\n",
      "2927\n",
      "2939\n",
      "3225\n",
      "3422\n",
      "3520\n",
      "3558\n",
      "3767\n",
      "4176\n",
      "5165\n",
      "5457\n",
      "5654\n",
      "5937\n",
      "6400\n",
      "6576\n",
      "6597\n",
      "6625\n",
      "6783\n",
      "7928\n",
      "8408\n",
      "8527\n",
      "9015\n",
      "9664\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": 103,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Prediction: 2    Label: tensor(3)\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "<matplotlib.image.AxesImage at 0x15c82e710>"
      ]
     },
     "execution_count": 103,
     "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 = 2927\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": 110,
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "# Find Neighbours\n",
    "idxs = KNN.kneighbors(X=[X_test_c[test_instance]], n_neighbors=1000, return_distance=False)\n",
    "idxs = idxs[0]\n",
    "\n",
    "CHP_nns = list()\n",
    "for i in range(len(idxs)):\n",
    "    if X_train_y[ idxs[i] ] == pred:\n",
    "        CHP_nns.append(idxs[i])\n",
    "        if len(CHP_nns) == 3:\n",
    "            break"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 112,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[39031, 36072, 5103]"
      ]
     },
     "execution_count": 112,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "CHP_nns"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## 2: Look at FAMs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 138,
   "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",
    "query_act_cam = weights.reshape(NUM_CLASSES*NUM_FEATURES,1,1) * C[0].detach().numpy()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 139,
   "metadata": {},
   "outputs": [],
   "source": [
    "neigh_act_cam = list()\n",
    "\n",
    "\n",
    "for i, data in enumerate(train_loader):\n",
    "    if i in CHP_nns:\n",
    "        logits, x_nn, C = netC(data[0])\n",
    "        nn_act_cam = weights.reshape(60,1,1) * C[0].detach().numpy()\n",
    "        c_nn = x_nn[0] * weights\n",
    "        neigh_act_cam.append([nn_act_cam, c_nn])\n",
    "        \n",
    "    if len(neigh_act_cam) == 3:\n",
    "        break"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Query Visualization"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 140,
   "metadata": {},
   "outputs": [],
   "source": [
    "c = x * weights\n",
    "c = c[0].tolist()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 141,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(-0.5, 27.5, 27.5, -0.5)"
      ]
     },
     "execution_count": 141,
     "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": 143,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(-0.5, 27.5, 27.5, -0.5)"
      ]
     },
     "execution_count": 143,
     "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 + 3\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][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( str(neigh_act_cam[0][1][fam_idx].item()) )\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][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[1,0].imshow(FAM, cmap='jet', alpha=0.4)\n",
    "axarr[1,0].set_title( str(neigh_act_cam[1][1][fam_idx].item()) )\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][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[1,1].imshow(FAM, cmap='jet', alpha=0.4)\n",
    "axarr[1,1].set_title( str(neigh_act_cam[2][1][fam_idx].item()) )\n",
    "axarr[1,1].axis('off')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## CAM"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 39,
   "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": 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": "code",
   "execution_count": 41,
   "metadata": {},
   "outputs": [
    {
     "ename": "TypeError",
     "evalue": "round() takes no arguments (1 given)",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mTypeError\u001b[0m                                 Traceback (most recent call last)",
      "\u001b[0;32m<ipython-input-41-fa7c73bdea00>\u001b[0m in \u001b[0;36m<module>\u001b[0;34m\u001b[0m\n\u001b[0;32m----> 1\u001b[0;31m \u001b[0mquery_cam\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mround\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;36m1\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m",
      "\u001b[0;31mTypeError\u001b[0m: round() takes no arguments (1 given)"
     ]
    }
   ],
   "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": "iVBORw0KGgoAAAANSUhEUgAAATsAAAD7CAYAAAAVQzPHAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjMuMCwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy86wFpkAAAACXBIWXMAAAsTAAALEwEAmpwYAAAkqElEQVR4nO2de3Bd1ZXmv6WnLcmWLNmWhZGxAUNsCGDiJoRAcGIIJNM8qjtJAVPEM8O0ZyZ0mkxRmXgISUimM8XUTNGZmaZT5R4o3BmahB7oxiR0EUe8HwEMOAH8wO8XfiDLsmXLsqSrPX/o5uy9ln2Prq7uSzrfr0qlve865+x9711n3b3XWXttcc6BEEImOhWl7gAhhBQDGjtCSCKgsSOEJAIaO0JIIqCxI4QkAho7QkgiGJOxE5HrRWSTiGwRkRX56hQhpYa6PfGQXOPsRKQSwIcArgWwB8BbAG51zq3PX/cIKT7U7YlJ1RjOvQzAFufcNgAQkZ8DuAlARoWokgZXjZYxNEnyRR92dTrnZpS6H2XKqHSbel0+xOn1WIzdbAC7g/oeAJ+OO6EaLZiLe8bQJMkXG/Hvd5a6D2XMqHSbel0+xOn1WIxdVojIcgDLhxtrLnRzhBQF6vX4YywPKPYCaA/qZ6ZfUzjnVjrnFjvnFlehYQzNEVI0RtRt6vX4YyzG7i0A80VknojUALgFwOr8dIuQkkLdnoDkPI11zg2KyJ8DeBZAJYCHnXMf5K1nhJQI6vbEZEw+O+fcMwCeyVNfCCkbqNsTj4I/oCCETDRyzYEpee3FaOFyMUJIIqCxI4QkAho7QkgioM+uqMT5OkrrzyAkd7LVa3tccXWeIztCSCKgsSOEJAJOY4sKp6pkIpKtXjP0hBBCCg6NHSEkEdDYEUISAX12hEwocl3KZYnzr41P3zNHdoSQREBjRwhJBAmexo5muD+KYXv482FPy/anJWXqQ9k3T5KA1d2wbpUlW+WxyhljGipj6rZrYfNWr4sMR3aEkERAY0cISQQ0doSQRJAwn12cny47H17/pWer+hnX7VH126Zv9ZVqfe5R1x+Vf/LidUrW9MxWZAX9d8RQMXVKVD50ld7W8b6rn4vKqVinmTYFlahV9R91LInKjS8aXbU+vJCBoBznzzuF/GcI4siOEJIIaOwIIYkgYdPYODIPjQ9+fW5UvveCDiWrETuG99cZGtJD8YYKP6/96hUvK9maX57hK5yqkhgGLpmn6rVLD0fle1tfVLJUMJ5xo4j9SBklvPMLz0bl/3fZJUrW98AxX8nXAo4CwJEdISQR0NgRQhIBjR0hJBEkzGeX2S839Ik5UfkTX9uiZMvrn4/Kvzs5Tck+UXNI1RtR4yvGRbJv0PtB3u6Zq2QVtf4819urZG6ITrzEYVT1yBfPicp3Ln1WyRqr/G28fUDHOx3snxqVn3/uUiWrOZZZr/70Zu37O6vat7Ggap+SvXt8CvJP/jOrjDiyE5GHReSgiLwfvNYsImtEZHP6/7S4axBSjlC3k0U209hHAFxvXlsBoMM5Nx9AR7pOyHjjEVC3E8OI01jn3EsiMte8fBOAJenyKgAvAPhOPjtWCCqb/JB+3w0tSvZnF/mp6lmV+jdg74AfUj/70QVK9nx1v6r3b/bTiMmdfUo2eZcPEdh/TY2S3b7it1H5ZxuXKFnbav9oP3VET3Hpds2d4uh2jisBTETTPX8chDzV6tt2e3Dwr/7uCt3Clt1RecqQWfkwGJS1quLp7s+q+g3/yk9rz67eqWRvXuKPrV63Q19ImZhaIwun3IVPCJrrndLqnPvDxH0/gNY89YeQUkPdnqCMeVjgnHOI+fkSkeUislZE1g7iWKbDCCk74nSbej3+yNXYHRCRNgBI/z+Y6UDn3Ern3GLn3OIqNOTYHCFFIyvdpl6PP3INPVkNYBmA+9P/n8pbj8ZKYL4/vlUvq/n2ZWui8tRq7TNDlT9xzYmZSrS2Y2FUbvr1Nn3egH58X4fDQe2oPrbGx6JMadGP6z9R7e+pv/zkaiX7n69/MSrLEe0jRBjqQv9dPiiwbmfnwzt4x1laNGmHL5tIj3/81eeicu2x7VrYHtNc6Kc7Ynqyd7eq7x/0Bv3Tk/cq2cBV3hdYvc7+NtQH5alGFr4RY4pCVc5T5FU2oSePAXgdwPkiskdE7sCwIlwrIpsBXJOuEzKuoG4ni2yext6aQbQ0z30hpKhQt5PFuF9B0b9IT1V7Pu2neY2NPUr2k9/6hJliEmsONvrpxYznPlKy+uPB1NWGmB4yg+OUn1ZWTNJT5f03+yH9j9qfUzIXTEfX909Ssqqubn/5ck4rQU5Dbt9XzSFza4YqUa9FSidnGVldULbRHeE09pCRnZIgJQwb0fPKb7f7DD4PYg40YaOT7UUzU4BIFDp5CCGJgMaOEJIIaOwIIYlgXPrsKtp8aMjJadpeT/7A+xZq3taPyGsmd/pKo7lo8FR8yMrCPUzMsppTQqwO+MfpvRecr0T/4aJ/CGrTlez4kHeSPPv31ypZZVf4ON/6Pfh7Nb4I/V2Zd6ppelYv7dp+oz92Xq12qE36kg93kl4dl1Ij+4OyDlsaCnTn2BQTFlJhnGahL/CIjmf5wQb/PGcmNunzVGiUcZTHUQDXNO8UQkgioLEjhCSCcTmNdTW+241v7leyriv9eLs/SHgIAE2vBlMDHd2hpqqDX5irRHXiV0JU7NCR5UcWLFD12l/vispHr9OhJ21nnukrOioG979/ZVSesVO3gbog1sBGk6cylE93LCkDshxfmO/uVw/7bCYf36inmD+8+qWo/Hd3an2Un/qpa4NZw1sdbOo6dape7XP0QjOtDZOpHNbvoXpnsNqjaVDJcDwoDyB7CqC7HNkRQhIBjR0hJBHQ2BFCEsG49NkN1fjH8Ie+qHMrNm32jquqD3boE8NlNTbJwgXe1ze9SfsB27t8Zta6BTr2pOsyvTFJU/B4fdZX3lGyda4pKrdOOqFkMwJfHy43j/3DpTzdWqQSq9i0ajZBCikBOa57Mv4t+b33487cpWU/PuxDP/ov1j6zOc6HW4loR1hD4Diuh86A3fPHOrzkP57p9Xxbm/ZFL1jofXZdf6r1un97sF7NJAxCGBk2YJU1dEDbz7A2RpYZjuwIIYmAxo4Qkgho7AghiWBc+uwqtvtNemce0eu1Oi/3y2UmTT1byeo3emfHoT/RqWgapnufwfnH9CbAc/e8HpXbjulMrPKiDthzR7xP78C0uUr2N1c9EZW/95t/oWRXTvU+kZ1TdZba3bOCvm7PvMTolDim0H3DmLsSkaPPzrqwQn+s+S6bHg3iR9doWV+7j9EcmKZj6WpxMirPgtb571/8S1Vv3+nXbx09qnfmu/E3fxuVD87QsnXnXhyVd1SeqzvXFZQHTmoZQt+fXWZWlaEcD0d2hJBEQGNHCEkE43Ma2+inqtPu0ClWl9a/EZV/8eY1Slbf74fiLe/qEJLauX4YP+XERiWbORhMXT/UfXHb9HV6j/nH8pd//j0l+9nPfBaUtqf1xiit8/y6mskt+vF9qt5/TR+1mB1UujOUyTjHpP0Ip7X9ZmocF6URRH70TqtDJhrN5lBzTJiIC26Jw2ZPncGnfXnmIn0/XrBofVTe2z5byQZ+F2TwOWrn7eE9YH0wYeYfTmMJIURBY0cISQQ0doSQRFC+PruYTXJ7zvfrvnp69PKYzscXR+XmrdrB5qqDuf4WvYPY4DT/6DtVo38D1Cobk6l4cIrezqnqTn/doepOJXvt/14YlS/YulPJwoSu01oOK1F9GHdgU1OF3yB/usY5oZ/OxhGFdfNF9wR6bSOTgsTFx47rMK2+eq9MYn2Ex3W1K0hJZld9fRjcggv3aFnzIh9fUlell6QdqQp9b9YvF/rwRpHhOAbeHoSQREBjRwhJBOU7jY2J+K97LYgYf81GXofDffP2wkf0OroDg10+k8KhWToKfH+Nz6xy5mdNaxfojv7LlJ+e/uy+i5XsvGPB1NVEkGCGLx6FzhLbG8YP2A1/wtE+V0lMIGza6bjlMEG8Sa/xc4TT0aN6jttT7+e4XWpXKaDyDD0Oaun0be58RWdDPq8qmLvq6BIcDlINnUiZzaLU27Djrvz7ZziyI4QkghGNnYi0i8jzIrJeRD4QkbvSrzeLyBoR2Zz+P22kaxFSTlC3k0U2I7tBAHc75xYCuBzAnSKyEMAKAB3OufkAOtJ1QsYT1O0EMaLPzjm3DxhOieCc6xGRDRiemd8EYEn6sFUAXgDwnYL08hTCNTFxc33zHD70EdisvkFy4p2NOuuIXOAdEfLVd5Xsu80dqn5/xxei8szL9ZKw8An6kN4jG7tr/M5jW6CzQ+w9HuxKdsD0O9ylzK64oQ8vlvLU7VwIwkasqy90aZsd7Q61ed/0rrM/o2T7521R9ektfhlY21adqbjqBl/unNGkZBvg/Xv9u81yNeV/1tfUS8JqjSy3Rw2j8tmJyFwAiwC8AaA1rSzAsKlozXQeIeUOdXvik7WxE5EGAE8A+JZzTq0ads45nLJyOTpvuYisFZG1g6cMpwgpPbnoNvV6/JHVeFBEqjGsDI86555Mv3xARNqcc/tEpA3AwdOd65xbCWAlAEyWs05rEMdGXGK/mMSJJkI83PxDarU/etfX/XT4qklNSvavj3xD1Xt3+cf5dWfp1RUu+G05Dj2kPwQ/pejdZzYoDhd72E85vDVtFA4ZkVx1u/B6bZdChHpt/RPBsXb4Ek5rjc73HvR69uoy7bp5E5ep+uSp/uSX5lyhZO6AT5Ib6jEAHNvV6CvGq6P3+LFT1fCN2M8ityCSbJ7GCoCHAGxwzj0QiFYDWJYuLwPwVE49IKREULeTRTYju88CuB3AeyKyLv3aPQDuB/C4iNwBYCeArxWkh4QUDup2gsjmaewryDwfXJrhdULKHup2sijf5WJZY3U1yw1OzHKxMGyj61btM2t9yzvG/tdjNyhZ04tbVV1lJan9ZOb2bVKL0H9h/Ylh3S4Xs9chEwTrYarMUAaU39qqf6gfJvQk9AWf2Kh1fm+l3pDKBVm+3+47T8laOgL/s9XdUK/tPad01/re4zKd5LaJEZeLEUISAY0dISQRTIBpbG5UVOmI7Tnf8ePtG6e8oGQPHlgYlZse/r29kqmHQ2z78cb9tsQtdwhl9hphPccpPSkTsl0ZFKdzhjBZip3GBq6bmffpZLKVP9BtpA57Hbyu9x0lW7s1nFbbJRzhdNRmng3rhddVjuwIIYmAxo4Qkgho7AghiSBRPruKSd5H0Hq3ThHyqcmbo/L33rxSyVqfCf0ZejPh0aUWqchQBnQ4gf1aqmJkoU/EhiSQ8Yv1YWX53dqFa+ESQquqYZiIyZhz768/p+r3Lf6nqDy1Uq+e67vskqg86c0dppEwe4nV+XCJGH12hBCSF2jsCCGJYEJPYytqdHhJ59d9RobbGl5UspVPXxeVZ7y2ScmG1ON0m53Bzg3iEmDEhRaE9bhpbFwGCIaaTBxy/C7tipqhGFnMNrWtL2t3zaaLvN4vqNFLIRZ85bdR+cctX1Sypn8ON5K1CTqLq68c2RFCEgGNHSEkEdDYEUISwYT22Z24SO/Ym+r25YeevFbJ6t4K06jaZS2hn8766HLd1Sbud2YUy4HopyNxpDKUR2BoX6eq/+Nz3qfdN107+O5b7Ded6p9psmxjSlC25oY+O0IIyTs0doSQRDChp7G1a/UOHzPWZntmXPYQ/j6QJKDvgSnP+VVEU8yRDz7u94adecrGxjbcpHTwziWEJAIaO0JIIqCxI4QkAhne8LxIjYl8jOGt6aYD6Bzh8GKR1L6c5ZybUaS2JjRlqtdAefWnWH3JqNdFNXZRoyJrnXOLi97waWBfSL4ot++vnPpTDn3hNJYQkgho7AghiaBUxm5lido9HewLyRfl9v2VU39K3peS+OwIIaTYcBpLCEkERTV2InK9iGwSkS0isqKYbafbf1hEDorI+8FrzSKyRkQ2p/9PK1Jf2kXkeRFZLyIfiMhdpewPGRul1G3qdXYUzdiJSCWABwF8CcBCALeKyMJitZ/mEQDXm9dWAOhwzs0H0JGuF4NBAHc75xYCuBzAnenPo1T9ITlSBrr9CKjXI1LMkd1lALY457Y55/oB/BzATUVsH865lwB0mZdvArAqXV4F4OYi9WWfc+6ddLkHwAYAs0vVHzImSqrb1OvsKKaxmw1gd1Dfk36t1LQ65/aly/sBtBa7AyIyF8AiAG+UQ3/IqClH3S65HpWbXvMBRYAbfjRd1MfTItIA4AkA33LOqS2dStEfMvGgXg9TTGO3F0B7UD8z/VqpOSAibQCQ/n9whOPzhohUY1ghHnXOPVnq/pCcKUfdpl4bimns3gIwX0TmiUgNgFsArC5i+5lYDWBZurwMwFPFaFREBMBDADY45x4odX/ImChH3aZeW5xzRfsD8GUAHwLYCuC7xWw73f5jAPZheEvgPQDuANCC4adDmwH8BkBzkfpyJYaH8r8HsC799+VS9Yd/Y/4+S6bb1Ovs/riCghCSCPiAghCSCGjsCCGJYEzGrtTLvwghJFty9tmll8h8COBaDDtF3wJwq3Nuff66Rwgh+WEs+8ZGS2QAQET+sEQmo7GrkgZXjeYxNEnyRR92dzruQZEXFsj/5lO+MmGD+6bd9DliLMbudEtkPh13QjWaMRffHkOTJF9sxF/sHPkoQiYOYzF2WSEiywEsH26M2YoIIaVhLA8osloi45xb6Zxb7JxbXIWGMTRHCCG5M5aRXbREBsNG7hYAt+WlV3ljNLZ8qGC9IKSwZKvnY9HxXM8tn+i2nI2dc25QRP4cwLMAKgE87Jz7IG89I4SQPDImn51z7hkAz+SpL4QQUjAK/oCi8ORrmBx3ndEM4fMxHS6foT8pFYXQAXvNOF21srjomvBY20YqKNuokOLqOe8qQkgioLEjhCQCGjtCSCKYAD67OIoResKQFVJq8uFvHslHl+t1su1b4cddHNkRQhIBjR0hJBFMgGls3LC5GFPM0TzOH811SLLJdTqYKyPpcWVQHk2Sl4xJSE7TZmHhHUYISQQ0doSQREBjRwhJBBPAZ2cZLMA1434TrKwQvx8MbyGF1murt9Y0xPne4sh2mVnhdZwjO0JIIqCxI4QkgjKaxub6aD1l6nFD47iI7bghfabjAP1IHsjPcD/boT8Zf2QbGmVlqRhZtu3F6fUIelyZoWwva2/HVHCd2G5zGksIIXmBxo4Qkgho7AghiaCMfHZxWJvsH8NXNk5Vkv3XN0blH37qN0rW7wai8su9s5VsaX1nUNMfSyWqo3LqlI9M9+37G66KylM/0M6Nut/5zdeGTvab64Q+ktEsxylu5ggyWkbj/w2/y7isI3G+aM2xz50bXLJGyXrner3+0SdfVrLKWt23VLVvo7LGyMTLKlLa13fv2s9F5Zn/Z4fpdpx/O86Hl5ue8+4ghCQCGjtCSCIYJ9NYi7fRh26rV5L75j0flZ15fF4t/u1+of6Akjll9/UUIhXU3SnTC13/4YKXfGWBPvIvpy+Nyi1rdinZ0MAAcoO/V0mn95uzfMVpfVgx59WonHLarVIpflqbqjLTxmpdTU0K5MZqpFwgE93+Dy7y9+NPZZ7telHhnUIISQQ0doSQREBjRwhJBGXks8t+ucjxK8+Oyv957nNGWpun/oSEvwm5Z5+4d8k/R+X/UrVUyZpXb835uqScyV6vK2bNjMpH5zco2XdveDEqp4yfuBJdgUyPXypQ5yvV+ryK6sCnXW3Oq8y8oXXFgJENBbJ+I+sPrmuXkqn3UQbLxUTkYRE5KCLvB681i8gaEdmc/j+tsN0khJCxkc009hEA15vXVgDocM7NB9CRrhNCSNky4jTWOfeSiMw1L98EYEm6vArACwC+M7aujGIaOz+I2BYdFa6iTczj89g9QwYzlE852IaI9Jl6KM885b3risdUfdWH/vekYuM+c3T4RkbjeWCGlPIj84qXGf+mOyr/u8Z3lSxcuXNKmEiw2iFVEyOrMLKKQJYysn6zSqI3kJ8wsqEw9ETPVVMuXCl0HJmx465cswdl30K2tDrn/nBH7gfQmqf+EEJIQRjz01jnnEPMYk4RWS4ia0Vk7SCOjbU5QgjJiVyN3QERaQOA9P+DmQ50zq10zi12zi2uQkOmwwghpKDkGnqyGsAyAPen/z+V22WyzfKgbXLt3km+8klz2pSgXG9kYVSKdWeFrjfrWjga95tw0tR7YmS+0akV2tfYdcGJqDx9o/UDhl9Trr9P9N+VBvu5ez0fvHCOktzS9EZUTpllV5UNQd3odeWUzLKKmsD3VWnCS8IQkuNGdtj4zLqC8BJnZmhy2MusT1vC6zRpmbohJxlZ6GDPTzhwNqEnjwF4HcD5IrJHRO7AsJG7VkQ2A7gmXSeEkLIlm6ext2YQLc3wOiGElB1lvIIi8zS2+W0/bD52kw7vaGgM3pINda5DZsKZ4xEjC0fUhyfHnAjoObANPYnZNEU9oh9N8s5ck36SwpGde6Z/ifaXhNpxSnhJMD1NTTOylkA21cjqgvCSyszhJamuEUJPggSdcL1a5vz9CBkwsvA6J6CJMz9xGwXl5pLh2lhCSCKgsSOEJAIaO0JIIhgnPjstSx3yPoL/vvEaJfth+wu+YkNPmoKyeecD/b6NfxjQKYblpPe7fGXyRiWr7jSOwf6w0W7TAeuzCK5zLIxBPJrxuFM/J/5elTcxen3KPtTBd2n0U22AY0NPGoOwkBbtM/vpx37DnWqnfcjHdvnNqSp/cUjJ3E691rL3ep+RZUXVm7oD4kOsKmzclgqhsRmJwjbidt62ZLvRePZXJISQCQONHSEkEZTRNDY3uzv9TT2M/ZuFl0Tlb0xfp2Tbgw1HHtn2GX2drf7Bf+3b25Rs4KRf7fDTP1usZH+x4B3doe5gZUT3TC0LQlpSx3W/q3rbgpp+tB8bsqLCW/jbVX7EfCcmKiUVfLcqkwigvmaboeS/Pf35qFzTqnWn/fW1UbnBTDGrgov2TdZTzINfuVTVr14SpbPE0H49/07tDfZgPmJCVk6GfT0T2eMylAGGnhBCSAw0doSQREBjRwhJBOPEZ2dlfg5f8a7ebFq6fAjHX13xeSWbFIR+zBzcqWTN9X7TksaqbiVrqPJZHmTmVCVb1KYzyvYF2Rt6TEqrLjRH5b179dq1+r8N/IQtJibhUPg12ZQscT676hgZKRxxoRFeVvfXHyvJj268Oio3/3KHPi1UuxYtqp8V6M5Zuj3X4HWpzuhOU+BErjEZev7q9hdVvSeId+lu1+FWnX/kO9TdM0PJfrHxPF9ZaDKidAZlG23Vl/9lkLwDCCGJgMaOEJIIaOwIIYmgxD67OFsbztntcUHcWb9+C0M7vH+t5ojOqDp0RlA5Ry9Pqav3/ox27FGydni/4KRD2g/3R/0mzi7sjnbvoXNKU1R+s22ePu3S9qg86Myymu6gnOrXMuVrsduplZFLNrFkP55oXv1hUDPfXXfcZtMBdbq9vgbvQ640/sPmYHPtmb16udg5H2mfdmrA+80qJ2mfcn+zl+2b0q5kj+//t76yyHwWG4N7ULvegVDNh+xOY4yzI4SQjNDYEUISQRlPY8OuxS2RMpvaDATXPGiyCoeXnGVELf6aM8xmaXXOL/v65rZXdHMf6yH2YNDVyU26jRnn+Ef9z4neavesGj9t2DrjPCVTySJ6bfZju6lPSPi52awSpHDkmrEj/G5tiFFwbI/R61SwRNFkRDk+27tdhmq0rk4JN4fSs1bIu/pYCcJEBge1rMZHVOH4PB1O85nDfrnauvrpSra9db6v6NO0uuZpryiO7AghiYDGjhCSCGjsCCGJoIxiE6zdzTL05BTfRvjM2vg2OoN1Nva0gHqTYqn/Uh/SceiAXrpiXB3KvTCrW8tm7/PlOdt01uIDCMJk7H7BykUSl+4mLtszKQ256rXNah36Znu0qDdIJXZYb74+2OMdvoMt+navCjIXd119tpL17dOa/eGQ17t+Ewky7WMvW/PY5Uo2o/m9qNzUqNNGqdsz1hJxuRghhGQNjR0hJBGU0TTWRklnm/XArigIh/8m1Lw/mMaaZMADweqDQROmsaL97ah8bL0+r8m0Hk4iZttNuYNErb3v6Snm4XBHb73ww7wN+zmFX6H97eJvWenJVa9tiJHdjD0k8Mn06mlseNqAWWHTL74+6TytdJN6dL+nB00MmPc0I6hWfrRDyU7O8/fScbtLffiWTEKUQnhgRrwbRKRdRJ4XkfUi8oGI3JV+vVlE1ojI5vT/aSNdixBCSkU2P/2DAO52zi0EcDmAO0VkIYAVADqcc/MBdKTrhBBSloxo7Jxz+5xz76TLPQA2AJgN4CYAq9KHrQJwc4H6SAghY2ZUPjsRmQtgEYA3ALQ65/4QTLEfQGum88aO9XtkS8xynD791k8E/oRd39CbZMNtior1xkU4z7oMm4LyXC16pP6SqPzOCd3G7sPBwfvNNVUbxiej/D426wl9dhOHOCdWILMZUQJfWL/RnR5MicpTWvU95mZpf+LMniDriVGz1HQvG9ykr7MBC6PyHru7WLgq04aCqfeRHwde1neDiDQAeALAt5xzKomyc84hg7dVRJaLyFoRWTt4iuedEEKKQ1bGTkSqMWzoHnXOPZl++YCItKXlbYBZPZ/GObfSObfYObe4yuzJQAghxWLEaayICICHAGxwzj0QiFYDWAbg/vT/pwrSQwDxj+jtW6iKkQW23Uw/D599YVT+k/ZfK1lPhZ/iNszU4+2Byfr3Yn+QzWQHdILO+7ffFpVrf292GAmTF9pp7FD4/u00Ni70JNfpPykc2a4GsN+l/d5DAh2wpwWem16TEmX97Z+Kyg9d+HMl6zf3zlCgS4eD6S8A3P3Lr3pZpb6xdvXN8dfYYt7D3qBsFoXozyk/09hsfHafBXA7gPdEZF36tXswbOQeF5E7MLxq6mt56REhhBSAEY2dc+4VZB4iLM1vdwghpDDwcR0hJBGU0XKx0WQ2CG20zcAbpvW1by8YoJoEv8cHvT/jWIX2tW0I0jNInV7Go5Z5AdjV730UqQ1645zU/wgytb5wyq7AQdl+FuHA2oaXxMGsJ6Un7ru0xOl13IbnZoOmkECtulONSrTiwtei8nrMV7IK0353uEk29EbYrxxYEpXr121TMvXYskuL9MbYcVmbM280Pho4siOEJAIaO0JIIiijaWzcUNXKwqmAfSQfnhfz9mxuxGC4faDzDCXqbvY7igwM6N+HoY9M+1uD8nYtqt0eJkTcYDoQ9tXsmoJw+hH3fuNcAZzSlgdxIRVxeh0SE25lk6UEIR3dX52rRLv7fexHZbVOO9J/XGcoGQo2llq/U0+bJz0dvI+10AyF79EuNwqXSWS7h3TucGRHCEkENHaEkERAY0cISQQl9tmF83lrd7P1McUsCYuz5TbLQviIfKsW9e/1oSeu3/jF7Oa+wRKYtX3aMViz368JS8Vubk3GN/nwj45GrzMvg1ThHru172ug3vsF+0x7lcf1salO/57WPKA31al8Z0tQs+u+QidiXCbtOF90fuDIjhCSCGjsCCGJoIxCT+IYTQR1KLOPrIMpaK+WTXnZj/d/fPYVSnbvua/7SsyjfQDYP+inrq///cW6Z4c2BTWdOUIP421EfFwYDhm/5Euvg+v0mWscDkSv6pCmitrgOkP6vIo+08ZxL698x6ySUHFcce/Jvr9sM/bkZ0zGkR0hJBHQ2BFCEgGNHSEkEZSRzy5XX9Ro7HXmNlJ7u6Ny03/tVrIfLfc+vO+d+6qSdRzX+wy93XFOVK7fuAmaZmQmbplb3HKZLDdiISWiGHod016vv07zI9rXNrTQ+7BTzpwnuv3v/+6qqDwTW/Sxsb43xMiyzQCTHziyI4QkAho7QkgiKKNpbDEYylCOp3mlT1/yIM6IORKox46gFpNUMWc4NSWW3PT6r/9TW9bHnjp1DanMUC4vOLIjhCQCGjtCSCKgsSOEJAJxbjQb3YyxMZGPMbzH7HQAnUVrOJ6k9uUs59yMkQ8jZGJQVGMXNSqy1jm3uOgNnwb2hZBkwGksISQR0NgRQhJBqYzdyhK1ezrYF0ISQEl8doQQUmw4jSWEJIKiGjsRuV5ENonIFhFZUcy20+0/LCIHReT94LVmEVkjIpvT/6cVqS/tIvK8iKwXkQ9E5K5S9oeQiU7RjJ2IVAJ4EMCXACwEcKuILCxW+2keAXC9eW0FgA7n3HwAHel6MRgEcLdzbiGAywHcmf48StUfQiY0xRzZXQZgi3Num3OuH8DPAdxUxPbhnHsJenM5pPuwKl1eBeDmIvVln3PunXS5B8AGALNL1R9CJjrFNHazAewO6nvSr5WaVufcvnR5P4DWuIMLgYjMBbAIwBvl0B9CJiJ8QBHghh9NF/XxtIg0AHgCwLecc0dL3R9CJirFNHZ7AbQH9TPTr5WaAyLSBgDp/weL1bCIVGPY0D3qnHuy1P0hZCJTTGP3FoD5IjJPRGoA3AJgdRHbz8RqAMvS5WUAnipGoyIiAB4CsME590Cp+0PIRKfYWU++DOAnGE5n+rBz7sdFa3y4/ccALMFwdpEDAH4A4J8APA5gDoYzsnzNOWcfYhSiL1cCeBnAe/DpZe/BsN+u6P0hZKLDFRSEkETABxSEkERAY0cISQQ0doSQREBjRwhJBDR2hJBEQGNHCEkENHaEkERAY0cISQT/H8u+ZAXTNBsJAAAAAElFTkSuQmCC\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
}
