{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import spacy\n",
    "\n",
    "import torch\n",
    "import torchtext\n",
    "import torchtext.data\n",
    "import torch.nn as nn\n",
    "import torch.optim as optim\n",
    "import torch.nn.functional as F\n",
    "\n",
    "import torchtext\n",
    "import random\n",
    "from torchtext.vocab import Vocab\n",
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "\n",
    "\n",
    "from dataclasses import dataclass, field\n",
    "\n",
    "\n",
    "from captum.attr import LayerIntegratedGradients, TokenReferenceBase, visualization"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "nlp = spacy.load('en')\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "device = torch.device(\"cuda:5\" if torch.cuda.is_available() else \"cpu\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(17500, 7500, 25000)"
      ]
     },
     "execution_count": 5,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "import os\n",
    "import torch\n",
    "from torchtext import data\n",
    "from torchtext import datasets\n",
    "import random\n",
    "import numpy as np\n",
    "\n",
    "import torchtext\n",
    "\n",
    "SEED = 1234\n",
    "\n",
    "\n",
    "random.seed(SEED)\n",
    "np.random.seed(SEED)\n",
    "torch.manual_seed(SEED)\n",
    "torch.backends.cudnn.deterministic = True\n",
    "\n",
    "TEXT = data.Field(tokenize = 'spacy', batch_first = True)\n",
    "LABEL = data.LabelField(dtype = torch.float)\n",
    "\n",
    "train_data, test_data = torchtext.datasets.IMDB.splits(text_field=TEXT,\n",
    "                                      label_field=LABEL,\n",
    "                                      train='train',\n",
    "                                      test='test',\n",
    "                                      path='data/aclImdb')\n",
    "train_data, valid_data = train_data.split(random_state = random.seed(SEED))\n",
    "\n",
    "len(train_data), len(valid_data), len(test_data)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "CNN(\n",
       "  (embedding): Embedding(25002, 100, padding_idx=1)\n",
       "  (convs): ModuleList(\n",
       "    (0): Conv2d(1, 100, kernel_size=(3, 100), stride=(1, 1), padding=(6, 0))\n",
       "    (1): Conv2d(1, 100, kernel_size=(4, 100), stride=(1, 1), padding=(6, 0))\n",
       "    (2): Conv2d(1, 100, kernel_size=(5, 100), stride=(1, 1), padding=(6, 0))\n",
       "  )\n",
       "  (fc): Linear(in_features=300, out_features=1, bias=True)\n",
       "  (dropout): Dropout(p=0.5, inplace=False)\n",
       ")"
      ]
     },
     "execution_count": 6,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "class CNN(nn.Module):\n",
    "    def __init__(self, vocab_size, embedding_dim, n_filters, filter_sizes, output_dim, \n",
    "                 dropout, pad_idx):\n",
    "        \n",
    "        super().__init__()\n",
    "                \n",
    "        self.embedding = nn.Embedding(vocab_size, embedding_dim, padding_idx = pad_idx)\n",
    "        \n",
    "        self.convs = nn.ModuleList([\n",
    "                                    nn.Conv2d(in_channels = 1, \n",
    "                                              out_channels = n_filters, \n",
    "                                              kernel_size = (fs, embedding_dim),\n",
    "                                              #padding_mode='circular',\n",
    "                                              padding=(6,0))\n",
    "                                    for fs in filter_sizes\n",
    "                                    ])\n",
    "        \n",
    "        self.fc = nn.Linear(len(filter_sizes) * n_filters, output_dim)\n",
    "\n",
    "        self.dropout = nn.Dropout(dropout)\n",
    "        \n",
    "    def forward(self, text):\n",
    "        \n",
    "        #text = [sent len, batch size]\n",
    "        \n",
    "        #text = text.permute(1, 0)\n",
    "                \n",
    "        #text = [batch size, sent len]\n",
    "        \n",
    "        embedded = self.embedding(text)\n",
    "        #embedded = [batch size, sent len, emb dim]\n",
    "        \n",
    "        embedded = embedded.unsqueeze(1)\n",
    "        \n",
    "        #embedded = [batch size, 1, sent len, emb dim]\n",
    "        \n",
    "        conved = [F.relu(conv(embedded)).squeeze(3) for conv in self.convs]\n",
    "\n",
    "        #conved_n = [batch size, n_filters, sent len - filter_sizes[n] + 1]\n",
    "                \n",
    "        pooled = [F.max_pool1d(conv, conv.shape[2]).squeeze(2) for conv in conved]\n",
    "        \n",
    "        #pooled_n = [batch size, n_filters]\n",
    "        \n",
    "        cat = self.dropout(torch.cat(pooled, dim = 1))\n",
    "\n",
    "        #cat = [batch size, n_filters * len(filter_sizes)]\n",
    "            \n",
    "        return self.fc(cat)\n",
    "\n",
    "INPUT_DIM = 25002\n",
    "EMBEDDING_DIM = 100\n",
    "N_FILTERS = 100\n",
    "FILTER_SIZES = [3,4,5]\n",
    "OUTPUT_DIM = 1\n",
    "DROPOUT = 0.5\n",
    "PAD_IDX = 1\n",
    "\n",
    "model = CNN(INPUT_DIM, EMBEDDING_DIM, N_FILTERS, FILTER_SIZES, OUTPUT_DIM, DROPOUT, PAD_IDX)\n",
    "\n",
    "model.load_state_dict(torch.load('<MODEL-PATH>'))\n",
    "model.to(device)\n",
    "model.eval()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "MAX_VOCAB_SIZE = 25_000\n",
    "\n",
    "TEXT.build_vocab(train_data, \n",
    "                 max_size = MAX_VOCAB_SIZE, \n",
    "                 vectors = \"glove.6B.100d\", \n",
    "                 unk_init = torch.Tensor.normal_)\n",
    "\n",
    "LABEL.build_vocab(train_data)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Vocabulary Size:  25002\n"
     ]
    }
   ],
   "source": [
    "print('Vocabulary Size: ', len(TEXT.vocab))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "lig = LayerIntegratedGradients(model, model.embedding)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "1"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "PAD_IND = TEXT.vocab.stoi['<pad>']\n",
    "PAD_IND"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [],
   "source": [
    "token_reference = TokenReferenceBase(reference_token_idx=PAD_IND)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [],
   "source": [
    "# accumalate couple samples in this array for visualization purposes\n",
    "vis_data_records_ig = []\n",
    "\n",
    "def interpret_sentence(model, sentence, min_len = 15, label = 0):\n",
    "    #text = [tok.text for tok in nlp.tokenizer(sentence)]\n",
    "    #if len(text) < min_len:\n",
    "    #    text += ['<pad>'] * (min_len - len(text))\n",
    "    indexed = [TEXT.vocab.stoi[t] for t in text]\n",
    "\n",
    "    model.zero_grad()\n",
    "\n",
    "    input_indices = torch.tensor(indexed, device=device)\n",
    "    input_indices = input_indices.unsqueeze(0)\n",
    "    \n",
    "    # input_indices dim: [sequence_length]\n",
    "    seq_length = min_len\n",
    "\n",
    "    # predict\n",
    "    pred = torch.sigmoid(model(input_indices)).item()\n",
    "    pred_ind = round(pred)\n",
    "\n",
    "    # generate reference indices for each sample\n",
    "    reference_indices = token_reference.generate_reference(seq_length, device=device).unsqueeze(0)\n",
    "\n",
    "    # compute attributions and approximation delta using layer integrated gradients\n",
    "    attributions_ig, delta = lig.attribute(input_indices, reference_indices, \\\n",
    "                                           n_steps=500, return_convergence_delta=True)\n",
    "\n",
    "    print('pred: ', LABEL.vocab.itos[pred_ind], '(', '%.2f'%pred, ')', ', delta: ', abs(delta))\n",
    "\n",
    "    add_attributions_to_visualizer(attributions_ig, text, pred, pred_ind, label, delta, vis_data_records_ig)\n",
    "    \n",
    "def add_attributions_to_visualizer(attributions, text, pred, pred_ind, label, delta, vis_data_records):\n",
    "    attributions = attributions.sum(dim=2).squeeze(0)\n",
    "    attributions = attributions / torch.norm(attributions)\n",
    "    attributions = attributions.cpu().detach().numpy()\n",
    "\n",
    "    # storing couple samples in an array for visualization purposes\n",
    "    vis_data_records.append(visualization.VisualizationDataRecord(\n",
    "                            attributions,\n",
    "                            pred,\n",
    "                            LABEL.vocab.itos[pred_ind],\n",
    "                            LABEL.vocab.itos[label],\n",
    "                            LABEL.vocab.itos[1],\n",
    "                            attributions.sum(),       \n",
    "                            text,\n",
    "                            delta))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [],
   "source": [
    "nlp = spacy.load('en')\n",
    "\n",
    "min_len = 15\n",
    "sentence = 'The film is great'\n",
    "text = [tok.text for tok in nlp.tokenizer(sentence)]\n",
    "if len(text) < min_len:\n",
    "    # right padding\n",
    "    text += ['the'] * (min_len - len(text))\n",
    "    # left padding\n",
    "    #text = ['<pad>'] * (min_len - len(text)) + text\n",
    "    #circular padding\n",
    "    #text = text + text[:(min_len - len(text))]\n",
    "    indexed = [TEXT.vocab.stoi[t] for t in text]\n",
    "\n",
    "input_indices = torch.tensor(indexed, device=device)\n",
    "input_indices = input_indices.unsqueeze(0)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "pred:  pos ( 0.92 ) , delta:  tensor([0.0004])\n"
     ]
    }
   ],
   "source": [
    "interpret_sentence(model, sentence, label=1)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Visualize attributions based on Integrated Gradients\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<table width: 100%><tr><th>True Label</th><th>Predicted Label</th><th>Attribution Label</th><th>Attribution Score</th><th>Word Importance</th><tr><td><text style=\"padding-right:2em\"><b>pos</b></text></td><td><text style=\"padding-right:2em\"><b>pos (0.92)</b></text></td><td><text style=\"padding-right:2em\"><b>pos</b></text></td><td><text style=\"padding-right:2em\"><b>0.83</b></text></td><td><mark style=\"background-color: hsl(0, 75%, 96%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> The                    </font></mark><mark style=\"background-color: hsl(120, 75%, 98%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> film                    </font></mark><mark style=\"background-color: hsl(120, 75%, 95%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> is                    </font></mark><mark style=\"background-color: hsl(120, 75%, 52%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> great                    </font></mark><mark style=\"background-color: hsl(0, 75%, 94%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> the                    </font></mark><mark style=\"background-color: hsl(0, 75%, 100%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> the                    </font></mark><mark style=\"background-color: hsl(0, 75%, 100%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> the                    </font></mark><mark style=\"background-color: hsl(0, 75%, 100%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> the                    </font></mark><mark style=\"background-color: hsl(0, 75%, 100%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> the                    </font></mark><mark style=\"background-color: hsl(0, 75%, 100%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> the                    </font></mark><mark style=\"background-color: hsl(0, 75%, 100%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> the                    </font></mark><mark style=\"background-color: hsl(0, 75%, 100%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> the                    </font></mark><mark style=\"background-color: hsl(0, 75%, 100%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> the                    </font></mark><mark style=\"background-color: hsl(0, 75%, 100%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> the                    </font></mark><mark style=\"background-color: hsl(0, 75%, 100%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> the                    </font></mark></td><tr></table>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "print('Visualize attributions based on Integrated Gradients')\n",
    "visualization.visualize_text(vis_data_records_ig)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([[2.4727]], grad_fn=<AddmmBackward>)"
      ]
     },
     "execution_count": 18,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "saved_conv_activations = {}\n",
    "\n",
    "def fwd_hook_wrapper(name):\n",
    "    def conv_fwd_hook(module, input, output):\n",
    "        if name in ['dropout', 'fc']:\n",
    "            saved_conv_activations[name] = input[0]\n",
    "        else:\n",
    "            saved_conv_activations[name] = F.relu(output)\n",
    "    return conv_fwd_hook\n",
    "\n",
    "hook1 = model.convs[0].register_forward_hook(fwd_hook_wrapper(str(model.convs[0].kernel_size)))\n",
    "hook2 = model.convs[1].register_forward_hook(fwd_hook_wrapper(str(model.convs[1].kernel_size)))\n",
    "hook3 = model.convs[2].register_forward_hook(fwd_hook_wrapper(str(model.convs[2].kernel_size)))\n",
    "\n",
    "hook4 = model.dropout.register_forward_hook(fwd_hook_wrapper('dropout'))\n",
    "\n",
    "hook5 = model.fc.register_forward_hook(fwd_hook_wrapper('fc'))\n",
    "\n",
    "model(input_indices)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {},
   "outputs": [],
   "source": [
    "hook1.remove()\n",
    "hook2.remove()\n",
    "hook3.remove()\n",
    "hook4.remove()\n",
    "hook5.remove()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Visualizing feature maps"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 1000x500 with 6 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "\n",
    "fig = plt.figure(figsize=(10, 5), dpi=100)\n",
    "plt.subplots_adjust(left=0, bottom=0.1, right=1, top=0.85, wspace=0.9, hspace=0.4)\n",
    "ind = 0\n",
    "#avg_feat_maps = np.mean(outp['out'].detach().numpy(), axis=0)\n",
    "for key in saved_conv_activations.keys():\n",
    "    if key in ['dropout', 'fc']:\n",
    "        continue\n",
    "        act = saved_conv_activations[key].squeeze(-1).detach().numpy()\n",
    "    else:\n",
    "        act = torch.mean(saved_conv_activations[key].squeeze(0), axis=0).detach().numpy()\n",
    "\n",
    "    ax = fig.add_subplot(1, 3, ind + 1)\n",
    "    ax.set_title(\"{} - {}\".format(key, saved_conv_activations[key].shape), fontsize=6)\n",
    "    plt.imshow(act)\n",
    "    cbar = plt.colorbar()\n",
    "    cbar.ax.tick_params(labelsize=6) \n",
    "    ax.get_xaxis().set_visible(False)\n",
    "    ax.get_yaxis().set_visible(False)\n",
    "    ind += 1\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "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.5"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
