{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "WARNING:root:Limited tf.compat.v2.summary API due to missing TensorBoard installation.\n",
      "WARNING:root:Limited tf.compat.v2.summary API due to missing TensorBoard installation.\n",
      "WARNING:root:Limited tf.compat.v2.summary API due to missing TensorBoard installation.\n",
      "WARNING:root:Limited tf.summary API due to missing TensorBoard installation.\n"
     ]
    }
   ],
   "source": [
    "import os, sys\n",
    "import numpy as np\n",
    "import json\n",
    "import random\n",
    "import torch\n",
    "from torch.utils.data import Dataset\n",
    "from torch.utils.data import DataLoader\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "from torch.optim import SGD, Adam\n",
    "from torch.optim.lr_scheduler import StepLR, CosineAnnealingLR\n",
    "import tqdm\n",
    "import matplotlib.pyplot as plt\n",
    "import seaborn as sns\n",
    "import torchvision\n",
    "from torchvision import datasets, models, transforms\n",
    "from numpy import linalg as LA\n",
    "sys.path.append('../environments/')\n",
    "from sepsis_env import SepsisEnv"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "env = SepsisEnv()\n",
    "fg_tuples = env.generate_tuples(group=\"foreground\", n_trajectories=10)\n",
    "bg_tuples = env.generate_tuples(group=\"background\", n_trajectories=10)\n",
    "all_tuples = bg_tuples + fg_tuples\n",
    "random.shuffle(all_tuples)\n",
    "split = 0.8\n",
    "train_tuples = all_tuples[:int(split*len(all_tuples))]\n",
    "test_tuples = all_tuples[int(split*len(all_tuples)):]"
   ]
  },
  {
   "cell_type": "raw",
   "metadata": {
    "code_folding": [
     0
    ]
   },
   "source": [
    "class ConvContrastiveNet(nn.Module):\n",
    "    def __init__(self):\n",
    "        super(ConvContrastiveNet, self).__init__()\n",
    "        self.name = \"ConvContrastiveNet\"\n",
    "        self.conv1 = self.cnn_apt_1 = nn.Conv1d(1, 10, 5) \n",
    "        self.fc1 = nn.Linear(90, 1)\n",
    "    \n",
    "    def forward(self, x):\n",
    "        x = self.conv1(x)\n",
    "        x = self.fc1(x)\n",
    "        x = torch.sigmoid(x)\n",
    "        return x "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {
    "code_folding": [
     0
    ]
   },
   "outputs": [],
   "source": [
    "class ContrastiveDataset(Dataset):\n",
    "    def __init__(self, batch, from_batch=False):\n",
    "        self.from_batch = from_batch\n",
    "        def construct_pairs_from_tuples(tuples):\n",
    "            X = []\n",
    "            y = []\n",
    "            for t in self.tuples:\n",
    "                if t[4] == 'background':\n",
    "                    s = t[0]\n",
    "                    a = t[1]\n",
    "                    r = t[3][0]\n",
    "                    if r == 0:\n",
    "                        r = 0.00000001\n",
    "                    \n",
    "                    blank_s = [0]*46\n",
    "                    blank_a = [0]\n",
    "                    s_a = np.hstack((s, a, blank_s, blank_a))\n",
    "                    X.append(s_a)\n",
    "                    y.append(r)\n",
    "                else:\n",
    "                    s = t[0]\n",
    "                    a = t[1]\n",
    "                    r = t[3][0]\n",
    "                    if r == 0:\n",
    "                        r = 0.00000001\n",
    "                    \n",
    "                    s_a = np.hstack((s, a, s, a))\n",
    "                    X.append(s_a)\n",
    "                    y.append(r)\n",
    "            return X, y\n",
    "        \n",
    "        def construct_pairs_from_batch(batch):\n",
    "            X = []\n",
    "            y = []\n",
    "            for i in range(len(batch['s'])):\n",
    "                if batch['ds'][i] == 'background':\n",
    "                    s = batch['s'][i]\n",
    "                    a = batch['a'][i]\n",
    "                    r = batch['r'][i]\n",
    "                    if r == 0:\n",
    "                        r = 0.0000001\n",
    "                    blank_s = [0]*46\n",
    "                    blank_a = [0]\n",
    "                    s_a = np.hstack((s, a, blank_s, blank_a))\n",
    "                    X.append(s_a)\n",
    "                    y.append(r)\n",
    "                else:\n",
    "                    s = batch['s'][i]\n",
    "                    a = batch['a'][i]\n",
    "                    r = batch['r'][i]\n",
    "                    if r == 0:\n",
    "                        r = 0.0000001\n",
    "                    s_a = np.hstack((s, a, s, a))\n",
    "                    X.append(s_a)\n",
    "                    y.append(r)\n",
    "            return X, y\n",
    "        \n",
    "        if from_batch:\n",
    "            self.tuples = batch\n",
    "            self.X, self.y = construct_pairs_from_batch(self.tuples)\n",
    "        else:\n",
    "            self.tuples = batch\n",
    "            self.X, self.y = construct_pairs_from_tuples(self.tuples)\n",
    "\n",
    "    def __len__(self):\n",
    "        if self.from_batch:\n",
    "            return len(self.tuples['s'])\n",
    "        else:\n",
    "            return len(self.tuples)\n",
    "\n",
    "    def __getitem__(self,idx):\n",
    "        return (self.X[idx], self.y[idx])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {
    "code_folding": []
   },
   "outputs": [],
   "source": [
    "class LinearContrastiveNet(nn.Module):\n",
    "    def __init__(self):\n",
    "        super(LinearContrastiveNet, self).__init__()\n",
    "        self.name = \"LinearContrastiveNet\"\n",
    "        self.shared = nn.Linear(47, 10)\n",
    "        self.fg = nn.Linear(47, 10)\n",
    "        self.fc = nn.Linear(20, 1)\n",
    "    \n",
    "    def forward(self, x):\n",
    "        x_shared = self.shared(x[:47])\n",
    "        x_fg = self.fg(x[47:])\n",
    "        x = torch.cat((x_shared, x_fg), dim=0)\n",
    "        x = F.relu(self.fc(x))\n",
    "        x = torch.sigmoid(x)\n",
    "        return x"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "def weights_init(m):\n",
    "    if isinstance(m, nn.Conv1d):\n",
    "        nn.init.xavier_uniform_(m.weight.data, gain=nn.init.calculate_gain('relu'))\n",
    "        nn.init.zeros_(m.bias.data)\n",
    "    if isinstance(m, nn.Linear):\n",
    "        nn.init.kaiming_normal_(m.weight.data, nonlinearity='sigmoid')\n",
    "        nn.init.zeros_(m.bias.data)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 135/135 [00:00<00:00, 840.76it/s]\n",
      "100%|██████████| 135/135 [00:00<00:00, 2096.31it/s]\n",
      "100%|██████████| 135/135 [00:00<00:00, 2335.12it/s]\n",
      "100%|██████████| 135/135 [00:00<00:00, 2404.84it/s]\n",
      "100%|██████████| 135/135 [00:00<00:00, 2238.40it/s]\n",
      "100%|██████████| 135/135 [00:00<00:00, 1978.92it/s]\n",
      "100%|██████████| 135/135 [00:00<00:00, 2147.18it/s]\n",
      "100%|██████████| 135/135 [00:00<00:00, 2337.08it/s]\n",
      "100%|██████████| 135/135 [00:00<00:00, 2504.93it/s]\n",
      "100%|██████████| 135/135 [00:00<00:00, 2452.98it/s]\n",
      "/Users/aishwaryamandyam/anaconda3/envs/research/lib/python3.7/site-packages/seaborn/_decorators.py:43: FutureWarning: Pass the following variables as keyword args: x, y. From version 0.12, the only valid positional argument will be `data`, and passing other arguments without an explicit keyword will result in an error or misinterpretation.\n",
      "  FutureWarning\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "<AxesSubplot:>"
      ]
     },
     "execution_count": 6,
     "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": [
    "model = LinearContrastiveNet()\n",
    "train_ds=ContrastiveDataset(train_tuples)\n",
    "test_ds=ContrastiveDataset(test_tuples)\n",
    "model.apply(weights_init)\n",
    "optimizer = SGD(model.parameters(), lr=0.01)\n",
    "num_epochs = 10\n",
    "criterion = nn.MSELoss()    \n",
    "losses = []\n",
    "for epoch in range(num_epochs):\n",
    "    for i, (s_a, r) in enumerate(tqdm.tqdm(train_ds)):\n",
    "        if model.name == \"ConvContrastiveNet\":\n",
    "            s_a = np.reshape(s_a, (1, 1, 94))\n",
    "        s_a = torch.Tensor(s_a)\n",
    "        r = torch.Tensor([r])\n",
    "        model.train()\n",
    "        pred_r = model(s_a)\n",
    "        train_loss = criterion(pred_r, r)\n",
    "        losses.append(train_loss.detach().cpu().numpy().item())\n",
    "        train_loss.backward()\n",
    "        optimizer.step()\n",
    "iterations = [i for i in range(len(losses))]\n",
    "sns.lineplot(iterations, losses)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "model.eval()\n",
    "preds = []"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Interpretability"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Linear Model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [],
   "source": [
    "shared_weights = model.shared.weight\n",
    "fg_weights = model.fg.weight"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<AxesSubplot:>"
      ]
     },
     "execution_count": 16,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "sns.heatmap(fg_weights.detach().cpu().numpy())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<AxesSubplot:>"
      ]
     },
     "execution_count": 17,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "sns.heatmap(shared_weights.detach().cpu().numpy())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "research",
   "language": "python",
   "name": "research"
  },
  "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.9"
  },
  "toc": {
   "base_numbering": 1,
   "nav_menu": {},
   "number_sections": true,
   "sideBar": true,
   "skip_h1_title": false,
   "title_cell": "Table of Contents",
   "title_sidebar": "Contents",
   "toc_cell": false,
   "toc_position": {},
   "toc_section_display": true,
   "toc_window_display": false
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
