{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "99333bfc",
   "metadata": {},
   "outputs": [],
   "source": [
    "import gym\n",
    "import torch \n",
    "import torch.nn as nn\n",
    "from sklearn.decomposition import PCA\n",
    "from sklearn.manifold import TSNE\n",
    "from random import sample\n",
    "from tqdm import tqdm\n",
    "from time import sleep\n",
    "from model import DQN_Agent\n",
    "import numpy as np      \n",
    "from sklearn.neighbors import KNeighborsClassifier\n",
    "from collections import Counter\n",
    "from sklearn.cluster import KMeans, DBSCAN, OPTICS\n",
    "from scipy.spatial.distance import euclidean\n",
    "from copy import deepcopy\n",
    "from sklearn.metrics import silhouette_score\n",
    "from torch.utils.data import TensorDataset, DataLoader\n",
    "\n",
    "from timm.models.layers import trunc_normal_, DropPath\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 37,
   "id": "b0aa2dd6",
   "metadata": {},
   "outputs": [],
   "source": [
    "NUM_CLASSES = 2\n",
    "NUM_PROTOTYPES = 2\n",
    "LATENT_SIZE = 64\n",
    "BATCH_SIZE = 32\n",
    "NUM_EPOCHS = 5\n",
    "DEVICE = 'cpu'\n",
    "\n",
    "MAX_SAMPLES = 100000"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 38,
   "id": "e8844f94",
   "metadata": {},
   "outputs": [],
   "source": [
    "def viper_sample(obss, acts, qs, xs, is_reweight=True):\n",
    "\t\n",
    "\t\"\"\"\n",
    "\tFunction taken from: https://github.com/obastani/viper\n",
    "\tobservations\n",
    "\tlatent x activations\n",
    "\tq values\n",
    "\tMax num observations\n",
    "\tUniform or sampling\n",
    "\t\"\"\"\n",
    "\t\n",
    "\t# Step 1: Compute probabilities\n",
    "\tps = np.max(qs, axis=1) - np.min(qs, axis=1)\n",
    "\tps = ps / np.sum(ps)\n",
    "\n",
    "\t# Step 2: Sample points\n",
    "\tif is_reweight:\n",
    "\t\t# According to p(s)\n",
    "\t\tidx = np.random.choice(len(obss), size=min(MAX_SAMPLES, np.sum(ps > 0)), p=ps)\n",
    "\telse:\n",
    "\t\t# Uniformly (without replacement)\n",
    "\t\tidx = np.random.choice(len(obss), size=min(MAX_SAMPLES, np.sum(ps > 0)), replace=False)    \n",
    "\n",
    "\t# Step 3: Obtain sampled indices\n",
    "\treturn obss[idx], acts[idx], qs[idx], xs[idx]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 39,
   "id": "96dd86d4",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Environment and Agent\n",
    "env = gym.make(\"CartPole-v1\").unwrapped\n",
    "input_dim = env.observation_space.shape[0]\n",
    "output_dim = env.action_space.n\n",
    "exp_replay_size = 256\n",
    "agent = DQN_Agent(seed=1423, layer_sizes=[input_dim, 64, output_dim], lr=1e-3,\n",
    "                  sync_freq=5, exp_replay_size=exp_replay_size)\n",
    "agent.load_pretrained_model(\"weights/cartpole-dqn.pth\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 40,
   "id": "e7205d9d",
   "metadata": {},
   "outputs": [],
   "source": [
    "# q_train.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 41,
   "id": "8d544585",
   "metadata": {},
   "outputs": [],
   "source": [
    "X_train = np.load('data/X_train.npy', ).reshape(-1, LATENT_SIZE)\n",
    "a_train = np.load('data/a_train.npy', ).flatten()\n",
    "obs_train = np.load('data/obs_train.npy', ).reshape(-1, 4)\n",
    "q_train = np.load('data/q_train.npy').reshape(-1, 2)\n",
    "\n",
    "\n",
    "\n",
    "obs_train, a_train, q_train, X_train = viper_sample(obs_train,\n",
    "                                                     a_train,\n",
    "                                                     q_train,\n",
    "                                                     X_train,\n",
    "                                                     is_reweight=True)\n",
    "\n",
    "\n",
    "tensor_x = torch.Tensor(X_train)\n",
    "tensor_y = torch.tensor(a_train)\n",
    "\n",
    "train_dataset = TensorDataset(tensor_x, tensor_y)\n",
    "train_loader = DataLoader(train_dataset, shuffle=True, batch_size=BATCH_SIZE)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 42,
   "id": "13c7caae",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(100000, 64)"
      ]
     },
     "execution_count": 42,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "X_train.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 43,
   "id": "096db306",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(100000, 2)"
      ]
     },
     "execution_count": 43,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "q_train.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 44,
   "id": "0fdc0eff",
   "metadata": {},
   "outputs": [],
   "source": [
    "def evaluate_loader(model, loader, cce_loss):\n",
    "    model.eval()\n",
    "    total_correct = 0\n",
    "    total_loss = 0\n",
    "    total = 0\n",
    "    \n",
    "    with torch.no_grad():\n",
    "        for i, data in enumerate(loader):\n",
    "            imgs, labels = data\n",
    "            imgs, labels = imgs.to(DEVICE), labels.to(DEVICE)\n",
    "            logits, _ = model(imgs)\n",
    "            loss = cce_loss(logits, labels)\n",
    "            preds = torch.argmax(logits, dim=1)\n",
    "            total_correct += sum(preds == labels).item()\n",
    "            total += len(preds)\n",
    "            total_loss += loss.item()\n",
    "            \n",
    "    model.train()\n",
    "    \n",
    "    return (total_correct / total) * 100, (total_loss / len(loader))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 45,
   "id": "797966df",
   "metadata": {},
   "outputs": [],
   "source": [
    "def clust_loss(x, y, p, criterion):\n",
    "            \n",
    "    for idx, i in enumerate(Counter(y.cpu().numpy()).keys()):\n",
    "        x_sub = x[y==i]\n",
    "        target = p[i].repeat(len(x_sub), 1) \n",
    "        \n",
    "        if idx == 0:\n",
    "            loss = criterion(x_sub, target) \n",
    "        else:\n",
    "            loss += criterion(x_sub, target)\n",
    "        \n",
    "    return loss"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 46,
   "id": "902e0bd4",
   "metadata": {},
   "outputs": [],
   "source": [
    "def sep_loss(x, y, p, criterion):\n",
    "        \n",
    "    for idx, i in enumerate(Counter(y.cpu().numpy()).keys()):\n",
    "        x_sub = x[y==i]\n",
    "        target = p[i].repeat(len(x_sub), 1) \n",
    "        \n",
    "        if idx == 0:\n",
    "            loss = criterion(x_sub, target) \n",
    "        else:\n",
    "            loss += criterion(x_sub, target)\n",
    "    \n",
    "    return -loss"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 87,
   "id": "35811975",
   "metadata": {},
   "outputs": [],
   "source": [
    "class Wrapper(nn.Module):\n",
    "\n",
    "    def __init__(self):\n",
    "        super(Wrapper, self).__init__()\n",
    "        protos = (torch.randn([NUM_PROTOTYPES, LATENT_SIZE]) )\n",
    "        self.prototypes = nn.Parameter(protos, requires_grad=True)\n",
    "        self.epsilon = 1e-10\n",
    "        self.maxpool = torch.nn.MaxPool1d(NUM_PROTOTYPES)\n",
    "        self.linear = nn.Linear(NUM_PROTOTYPES, NUM_CLASSES, bias=False) \n",
    "        self.__make_linear_weights()\n",
    "        self.main = nn.Sequential(\n",
    "            nn.Linear(LATENT_SIZE, LATENT_SIZE),\n",
    "            nn.BatchNorm1d(LATENT_SIZE),\n",
    "            nn.ReLU(),\n",
    "            nn.Dropout(0.5),\n",
    "            nn.Linear(LATENT_SIZE, LATENT_SIZE),\n",
    "            nn.BatchNorm1d(LATENT_SIZE),\n",
    "            nn.ReLU(),\n",
    "            nn.Dropout(0.5),\n",
    "            nn.Linear(LATENT_SIZE, LATENT_SIZE),\n",
    "        )\n",
    "\n",
    "    def __proto_layer_l2(self, x, b_size):\n",
    "        output = list()\n",
    "        p = self.prototypes.T.view(1, LATENT_SIZE, NUM_PROTOTYPES).tile(b_size, 1, 1).to(DEVICE) \n",
    "        c = x.view(b_size, LATENT_SIZE, 1).tile(1, 1, NUM_PROTOTYPES).to(DEVICE)            \n",
    "        l2s = ( (c - p)**2 ).sum(axis=1).to(DEVICE)        \n",
    "        act = torch.log( (l2s + 1. ) / (l2s + self.epsilon) ).to(DEVICE)   \n",
    "        return act\n",
    "    \n",
    "    def __make_linear_weights(self):\n",
    "        \n",
    "        prototype_class_identity = torch.zeros(NUM_PROTOTYPES, NUM_CLASSES)\n",
    "        num_prototypes_per_class = NUM_PROTOTYPES // NUM_CLASSES\n",
    "        \n",
    "        for j in range(NUM_PROTOTYPES):\n",
    "            prototype_class_identity[j, j // num_prototypes_per_class] = 1\n",
    "            \n",
    "        positive_one_weights_locations = torch.t(prototype_class_identity)\n",
    "        negative_one_weights_locations = 1 - positive_one_weights_locations\n",
    "\n",
    "        incorrect_strength = -0.5\n",
    "        correct_class_connection = 1\n",
    "        incorrect_class_connection = incorrect_strength\n",
    "        self.linear.weight.data.copy_(\n",
    "            correct_class_connection * positive_one_weights_locations\n",
    "            + incorrect_class_connection * negative_one_weights_locations)      \n",
    "            \n",
    "    def forward(self, x): \n",
    "\n",
    "        b_size = x.shape[0]\n",
    "\n",
    "        # Transform\n",
    "        x = self.main(x)\n",
    "\n",
    "        # Prototype layer\n",
    "        ps = self.__proto_layer_l2(x, b_size)\n",
    "        \n",
    "        # Linear layer\n",
    "        logits = self.linear(ps)\n",
    "\n",
    "        return logits, x"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "5a4cdc7b",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "a694e492",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 88,
   "id": "73d72ada",
   "metadata": {},
   "outputs": [],
   "source": [
    "model = Wrapper()\n",
    "model.drop_path_bool = True"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 89,
   "id": "6bbc78f1",
   "metadata": {},
   "outputs": [],
   "source": [
    "instances, labels = next(iter(train_loader))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 90,
   "id": "598ed4f2",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([32, 64])"
      ]
     },
     "execution_count": 90,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "instances.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "31d69be8",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 91,
   "id": "92ed0f93",
   "metadata": {},
   "outputs": [],
   "source": [
    "# model(instances)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "bc571ca2",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 92,
   "id": "9f619af4",
   "metadata": {},
   "outputs": [],
   "source": [
    "# model.linear.weight"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 93,
   "id": "f57b8a52",
   "metadata": {},
   "outputs": [],
   "source": [
    "# model.linear.requires_grad_(False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 94,
   "id": "e690f320",
   "metadata": {},
   "outputs": [],
   "source": [
    "# model.linear.weight.requires_grad = False"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 95,
   "id": "3c72b217",
   "metadata": {},
   "outputs": [],
   "source": [
    "# model.linear.weight.requires_grad"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 96,
   "id": "5135d1d0",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train Accuracy: 47.745 %     Train Loss: 0.6932839231681823\n",
      "CCE Loss: 0.3438516201019287\n",
      "Train Accuracy: 84.574 %     Train Loss: 0.26175077248811723\n",
      "CCE Loss: 0.3120229428625107\n",
      "Train Accuracy: 83.478 %     Train Loss: 0.2768279695367813\n",
      "CCE Loss: 0.30981194732427597\n",
      "Train Accuracy: 86.168 %     Train Loss: 0.26253392823696137\n",
      "CCE Loss: 0.30679888998508453\n",
      "Train Accuracy: 85.79299999999999 %     Train Loss: 0.2823270068383217\n",
      "CCE Loss: 0.30230977653503416\n",
      "Train Accuracy: 84.77900000000001 %     Train Loss: 0.2695656241822243\n",
      "CCE Loss: 0.3013905094122887\n",
      "Train Accuracy: 84.795 %     Train Loss: 0.26217432779312133\n",
      "CCE Loss: 0.2974299555468559\n",
      "Train Accuracy: 84.615 %     Train Loss: 0.2603268819665909\n",
      "CCE Loss: 0.2992459752345085\n",
      "Train Accuracy: 84.68299999999999 %     Train Loss: 0.2503831456899643\n",
      "CCE Loss: 0.2969539540195465\n",
      "Train Accuracy: 84.528 %     Train Loss: 0.26323004781246184\n",
      "CCE Loss: 0.29479945239543914\n",
      "Train Accuracy: 84.711 %     Train Loss: 0.2513738033127785\n",
      "CCE Loss: 0.29544061876296995\n",
      "Train Accuracy: 84.649 %     Train Loss: 0.2560209237885475\n",
      "CCE Loss: 0.29446362636327744\n",
      "Train Accuracy: 84.817 %     Train Loss: 0.2516536581873894\n",
      "CCE Loss: 0.29314588712692263\n",
      "Train Accuracy: 84.639 %     Train Loss: 0.2541048629236221\n",
      "CCE Loss: 0.2907930896043777\n",
      "Train Accuracy: 84.68299999999999 %     Train Loss: 0.25521054176568986\n",
      "CCE Loss: 0.29186522459983827\n",
      "Train Accuracy: 85.039 %     Train Loss: 0.2527415077114105\n",
      "CCE Loss: 0.29108982365846636\n",
      "Train Accuracy: 85.15700000000001 %     Train Loss: 0.24974270120978356\n",
      "CCE Loss: 0.289241896841526\n",
      "Train Accuracy: 85.71600000000001 %     Train Loss: 0.2649822499036789\n",
      "CCE Loss: 0.28969356070995333\n",
      "Train Accuracy: 85.197 %     Train Loss: 0.25282575188159945\n",
      "CCE Loss: 0.2876601495695114\n",
      "Train Accuracy: 85.202 %     Train Loss: 0.24864196002960204\n",
      "CCE Loss: 0.28621785912036896\n"
     ]
    }
   ],
   "source": [
    "cce_loss = nn.CrossEntropyLoss()\n",
    "mse_loss = nn.MSELoss()\n",
    "optimizer = torch.optim.Adam(model.parameters(), lr=0.01)\n",
    "scheduler = torch.optim.lr_scheduler.ExponentialLR(optimizer, gamma=0.9)\n",
    "best_acc = 0\n",
    "\n",
    "# Freeze prototype learning and use full model (inc. transformation)\n",
    "model.train()\n",
    "model.prototypes.requires_grad = True\n",
    "model.linear.weight.requires_grad = False\n",
    "prototypes = model.prototypes\n",
    "\n",
    "\n",
    "for epoch in range(20):\n",
    "    \n",
    "    running_loss = 0\n",
    "    \n",
    "    model.eval()\n",
    "    train_acc, train_loss = evaluate_loader(model, train_loader, cce_loss)\n",
    "    model.train()\n",
    "    print(\"Train Accuracy:\", train_acc, \"%\", \"    Train Loss:\", train_loss)\n",
    "    \n",
    "    if train_acc > best_acc:\n",
    "        torch.save(model.state_dict(), 'weights/wrapper_pre_projection.pth')\n",
    "        best_acc = train_acc\n",
    "    \n",
    "    for instances, labels in train_loader:\n",
    "        \n",
    "        optimizer.zero_grad()\n",
    "        \n",
    "        instances, labels = instances.to(DEVICE), labels.to(DEVICE)\n",
    "        logits, x = model(instances)\n",
    "        \n",
    "        loss = cce_loss(logits, labels) \n",
    "                \n",
    "        loss.backward()\n",
    "        optimizer.step()\n",
    "        \n",
    "        running_loss += loss.item()\n",
    "        \n",
    "        \n",
    "    print(\"CCE Loss:\", running_loss / len(train_loader))\n",
    "    scheduler.step()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "b87db3cd",
   "metadata": {},
   "source": [
    "## Project Prototypes"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 97,
   "id": "79aa58e7",
   "metadata": {},
   "outputs": [],
   "source": [
    "# model.linear.weight"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 98,
   "id": "0aa944ef",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<All keys matched successfully>"
      ]
     },
     "execution_count": 98,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "model = Wrapper().eval()\n",
    "model.drop_path_bool = False\n",
    "model.load_state_dict(torch.load('weights/wrapper_pre_projection.pth'))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 99,
   "id": "a1c2548c",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(54.432, 0.6923979721069335)"
      ]
     },
     "execution_count": 99,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# Accuracy before projection\n",
    "evaluate_loader(model, train_loader, cce_loss)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 100,
   "id": "aeb2806b",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████| 100000/100000 [00:13<00:00, 7682.35it/s]\n"
     ]
    }
   ],
   "source": [
    "trans_x = list()\n",
    "model.eval()\n",
    "\n",
    "with torch.no_grad():\n",
    "    for i in tqdm(range(len(X_train))):\n",
    "        img = X_train[i]\n",
    "        _, x = model(  torch.tensor(img, dtype=torch.float32).view(1, -1)  )\n",
    "        trans_x.append(x[0].tolist())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 101,
   "id": "598381de",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(100000, 64)"
      ]
     },
     "execution_count": 101,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "trans_x = np.array(trans_x)\n",
    "trans_x.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 102,
   "id": "64d1bfbd",
   "metadata": {},
   "outputs": [],
   "source": [
    "from sklearn.neighbors import KNeighborsClassifier\n",
    "\n",
    "knn = KNeighborsClassifier(algorithm='brute')\n",
    "knn.fit(trans_x, list(range(len(trans_x))))\n",
    "dist, p_idxs = knn.kneighbors(X=model.prototypes.detach().numpy(), n_neighbors=1, return_distance=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 103,
   "id": "f64ecb8b",
   "metadata": {},
   "outputs": [],
   "source": [
    "p_idxs = np.array(p_idxs)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 104,
   "id": "ed18afd8",
   "metadata": {},
   "outputs": [],
   "source": [
    "projected_prototypes = trans_x[p_idxs.flatten()]\n",
    "proto_images = obs_train[p_idxs.flatten()]\n",
    "proto_before_trans = X_train[p_idxs.flatten()]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 105,
   "id": "151177c7",
   "metadata": {},
   "outputs": [],
   "source": [
    "model.prototypes = torch.nn.Parameter(torch.tensor(projected_prototypes, dtype=torch.float32))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 106,
   "id": "e0d06415",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(55.306, 0.5885979642438889)"
      ]
     },
     "execution_count": 106,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# Accuracy after projection\n",
    "evaluate_loader(model, train_loader, cce_loss)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 107,
   "id": "ba85265f",
   "metadata": {},
   "outputs": [],
   "source": [
    "torch.save(model.state_dict(), 'weights/wrapper_post_projection.pth')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 108,
   "id": "1bdc8d43",
   "metadata": {},
   "outputs": [],
   "source": [
    "# dist"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 109,
   "id": "eb970dca",
   "metadata": {},
   "outputs": [],
   "source": [
    "# proto_images"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d5c348e9",
   "metadata": {},
   "source": [
    "\n",
    "    | Num | Observation           | Min                 | Max               |\n",
    "    |-----|-----------------------|---------------------|-------------------|\n",
    "    | 0   | Cart Position         | -4.8                | 4.8               |\n",
    "    | 1   | Cart Velocity         | -Inf                | Inf               |\n",
    "    | 2   | Pole Angle            | ~ -0.418 rad (-24°) | ~ 0.418 rad (24°) |\n",
    "    | 3   | Pole Angular Velocity | -Inf                | Inf               |\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "484a46fa",
   "metadata": {},
   "source": [
    "## Now Train Transformation"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 112,
   "id": "4d9be2b5",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<All keys matched successfully>"
      ]
     },
     "execution_count": 112,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "model = Wrapper().eval()\n",
    "model.drop_path_bool = False\n",
    "model.load_state_dict(torch.load('weights/wrapper_post_projection.pth'))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "4da98cb4",
   "metadata": {},
   "source": [
    "# Evaluate Model\n",
    "* Plot\n",
    "* Sillhouette score\n",
    "* Distance to clusters\n",
    "* Performace on 1000 simuluations\n",
    "\n",
    "### Variables\n",
    "* Transformer component: This helps A LOT\n",
    "* Num prototypes: Not that important so far, less is better in general\n",
    "* Extra loss functions: Don't help at all, CCE does the job 100%\n",
    "* Training data sampling: Todo\n",
    "\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "483185bc",
   "metadata": {},
   "source": [
    "## Plot"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 113,
   "id": "87975526",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(55.306, 0.5885979639339447)"
      ]
     },
     "execution_count": 113,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# Accuracy after final train\n",
    "evaluate_loader(model, train_loader, cce_loss)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 114,
   "id": "e7950f14",
   "metadata": {},
   "outputs": [],
   "source": [
    "# model.linear.weight"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 115,
   "id": "a82a535f",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████| 100000/100000 [00:13<00:00, 7656.50it/s]\n"
     ]
    }
   ],
   "source": [
    "trans_x = list()\n",
    "model.eval()\n",
    "\n",
    "with torch.no_grad():\n",
    "    for i in tqdm(range(len(X_train))):\n",
    "        img = X_train[i]\n",
    "        _, x = model(  torch.tensor(img, dtype=torch.float32).view(1, -1)  )\n",
    "        trans_x.append(x[0].tolist())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 116,
   "id": "13d84b8c",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(100000, 64)"
      ]
     },
     "execution_count": 116,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "trans_x = np.array(trans_x)\n",
    "trans_x.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 117,
   "id": "c846b8a1",
   "metadata": {},
   "outputs": [],
   "source": [
    "rand_idxs = np.random.randint(0, len(a_train), 3000)\n",
    "temp_y = a_train[rand_idxs]\n",
    "temp_x = trans_x[rand_idxs]\n",
    "# temp_x = X_train[rand_idxs]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 118,
   "id": "1644496a",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[2, 3]"
      ]
     },
     "execution_count": 118,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "[i+NUM_CLASSES for i in range(NUM_PROTOTYPES)]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 119,
   "id": "99d5a250",
   "metadata": {},
   "outputs": [],
   "source": [
    "temp_x = np.append(temp_x, model.prototypes.clone().detach().numpy(), axis=0)\n",
    "temp_y = np.append(temp_y, np.array( [i+NUM_CLASSES for i in range(NUM_PROTOTYPES)] ), axis=0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 120,
   "id": "defccfa1",
   "metadata": {},
   "outputs": [],
   "source": [
    "tsne = PCA(n_components=2)\n",
    "emb_x = tsne.fit_transform(temp_x)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 121,
   "id": "de165383",
   "metadata": {},
   "outputs": [],
   "source": [
    "# tsne = TSNE(n_components=2)\n",
    "# emb_x = tsne.fit_transform(temp_x)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 122,
   "id": "793949cd",
   "metadata": {},
   "outputs": [],
   "source": [
    "import pandas as pd"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 123,
   "id": "b925d304",
   "metadata": {},
   "outputs": [],
   "source": [
    "df = pd.DataFrame()\n",
    "df['a'] = temp_y\n",
    "df['x'] = emb_x.T[0]\n",
    "df['y'] = emb_x.T[1]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 124,
   "id": "0678da3a",
   "metadata": {},
   "outputs": [],
   "source": [
    "df.a = df.a.astype('category')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 125,
   "id": "3145f9f8",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/Users/eoinkenny/opt/anaconda3/envs/rl_env/lib/python3.9/site-packages/seaborn/rcmod.py:82: DeprecationWarning: distutils Version classes are deprecated. Use packaging.version instead.\n",
      "  if LooseVersion(mpl.__version__) >= \"3.0\":\n",
      "/Users/eoinkenny/opt/anaconda3/envs/rl_env/lib/python3.9/site-packages/setuptools/_distutils/version.py:351: DeprecationWarning: distutils Version classes are deprecated. Use packaging.version instead.\n",
      "  other = LooseVersion(other)\n"
     ]
    }
   ],
   "source": [
    "import seaborn as sns"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 126,
   "id": "f8c0d423",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<AxesSubplot:xlabel='x', ylabel='y'>"
      ]
     },
     "execution_count": 126,
     "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": [
    "sns.scatterplot(x='x', y='y', hue='a', data=df)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 127,
   "id": "4df025da",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<AxesSubplot:xlabel='x', ylabel='y'>"
      ]
     },
     "execution_count": 127,
     "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": [
    "sns.scatterplot(x='x', y='y', hue='a', data=df)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "4eb0f59b",
   "metadata": {},
   "source": [
    "## Sillhouete Score\n",
    "Re-label dataset with each $x$ nearest prototype"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 128,
   "id": "0789bfd3",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "KNeighborsClassifier(algorithm='brute', n_neighbors=1)"
      ]
     },
     "execution_count": 128,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "from sklearn.neighbors import KNeighborsClassifier\n",
    "\n",
    "knn = KNeighborsClassifier(algorithm='brute', n_neighbors=1)\n",
    "knn.fit(model.prototypes.detach().numpy(), list(range(NUM_PROTOTYPES)))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 129,
   "id": "37727a71",
   "metadata": {},
   "outputs": [],
   "source": [
    "labels = knn.predict(temp_x)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 130,
   "id": "51387df4",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.5583284946541863"
      ]
     },
     "execution_count": 130,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# Transformation\n",
    "silhouette_score(temp_x[:-NUM_PROTOTYPES], labels[:-NUM_PROTOTYPES])"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "6747bfbe",
   "metadata": {},
   "source": [
    "## Simulation Score"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 131,
   "id": "364541c9",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████| 1000/1000 [00:02<00:00, 376.07it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "average reward per episode : 12.01\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "reward_arr = []\n",
    "for i in tqdm(range(1000)):\n",
    "    obs, done, rew = env.reset(), False, 0\n",
    "    count = 0\n",
    "    while not done and count < 200:\n",
    "        _, latent_x, _ = agent.get_action(obs, env.action_space.n, epsilon=0)\n",
    "        A, _ = model( latent_x.view(1,-1) )\n",
    "        A = torch.argmax(A).item()\n",
    "        obs, reward, done, info = env.step(A)\n",
    "        rew += reward\n",
    "        count += 1\n",
    "    reward_arr.append(count)\n",
    "\n",
    "print(\"average reward per episode :\", sum(reward_arr) / len(reward_arr))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "abb4e57b",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "68e935d7",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "91fd9f53",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e9ff43d0",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "05f9b988",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "47b4c47b",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e6d91870",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "markdown",
   "id": "0a8dfd47",
   "metadata": {},
   "source": [
    "## Contribution of Each Feature"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "id": "46a3c82c",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(2098755, 4)"
      ]
     },
     "execution_count": 25,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "obs_train.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "id": "5bb5aa31",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(2098755, 64)"
      ]
     },
     "execution_count": 26,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "train_data_x.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "id": "0387db2e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(2098755,)"
      ]
     },
     "execution_count": 27,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "a_train.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 41,
   "id": "862921c3",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([64, 4])"
      ]
     },
     "execution_count": 41,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "weights = agent.q_net.fc1.weight\n",
    "weights.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 42,
   "id": "a0806be0",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(0.2815, grad_fn=<DivBackward0>)\n",
      "tensor(0.5814, grad_fn=<DivBackward0>)\n",
      "tensor(0.7222, grad_fn=<DivBackward0>)\n",
      "tensor(0.4661, grad_fn=<DivBackward0>)\n"
     ]
    }
   ],
   "source": [
    "for i in range(4):\n",
    "    print(  sum(abs(weights.T[i])) / len(weights) )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 43,
   "id": "65082c58",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.1278256763537884\n",
      "0.142202865062279\n",
      "0.017667636993578387\n",
      "0.20191600174775762\n"
     ]
    }
   ],
   "source": [
    "for i in range(4):\n",
    "    print(  sum(abs(obs_train.T[i])) / len(obs_train) )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 45,
   "id": "a19c4fac",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Total Contribution Average: tensor(0.0792, grad_fn=<MulBackward0>)\n",
      "Total Contribution Average: tensor(0.3380, grad_fn=<MulBackward0>)\n",
      "Total Contribution Average: tensor(0.5216, grad_fn=<MulBackward0>)\n",
      "Total Contribution Average: tensor(0.2172, grad_fn=<MulBackward0>)\n"
     ]
    }
   ],
   "source": [
    "data = torch.zeros(4)\n",
    "\n",
    "for i in range(4):\n",
    "    w = sum(abs(weights.T[i])) / len(weights)\n",
    "    x = sum(abs(weights.T[i])) / len(weights)\n",
    "    print(\"Total Contribution Average:\", w*x)\n",
    "    data[i] = w*x"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 48,
   "id": "272659c7",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([0.2000, 0.2591, 0.3113, 0.2296], grad_fn=<SoftmaxBackward0>)"
      ]
     },
     "execution_count": 48,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "torch.softmax(data, dim=0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 50,
   "id": "b9bf0cba",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(2, 4)"
      ]
     },
     "execution_count": 50,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "proto_images.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 52,
   "id": "7390302b",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([64, 4])"
      ]
     },
     "execution_count": 52,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "weights.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 59,
   "id": "66b4fb7e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([-0.04302227,  0.11650063, -0.15574128, -0.22569513, -0.16889343,\n",
       "       -0.10001172, -0.01422224,  0.01087987,  0.13577486, -0.03560081,\n",
       "        0.02575449, -0.05889751, -0.10716026, -0.07144366, -0.1550035 ,\n",
       "       -0.19094365, -0.08684268,  0.24042794, -0.14758203, -0.18864988,\n",
       "        0.08164098,  0.05242967, -0.08050377,  0.12700218,  0.00764638,\n",
       "       -0.05004192,  0.01025908, -0.10316804,  0.09904955,  0.16939828,\n",
       "       -0.08770164, -0.00985877,  0.1919707 , -0.05791862, -0.15131058,\n",
       "       -0.00931485, -0.06945472, -0.13663107, -0.05459756, -0.00058078,\n",
       "        0.03391863, -0.05229604,  0.00731769,  0.08461743,  0.01973061,\n",
       "       -0.02384995,  0.00583763, -0.15357726, -0.0069335 , -0.01094437,\n",
       "        0.01309674,  0.08829818, -0.04161307, -0.06451988,  0.05138354,\n",
       "       -0.17514603,  0.086413  ,  0.01803717, -0.05200949,  0.02989575,\n",
       "        0.03192458, -0.01387842,  0.14125147, -0.05025253])"
      ]
     },
     "execution_count": 59,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "np.matmul(weights.cpu().detach().numpy(), proto_images[0])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 61,
   "id": "13323de3",
   "metadata": {},
   "outputs": [],
   "source": [
    "feature_weights = torch.zeros(4)\n",
    "\n",
    "for i in range(4):\n",
    "    w = sum(abs(weights.T[i])) / len(weights)\n",
    "    feature_weights[i] = w"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 66,
   "id": "5e513a1f",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[-0.06225981 -0.00532342 -0.00131571 -0.01817009]\n",
      "[0.01905259 0.00123403 0.00058565 0.01037095]\n"
     ]
    }
   ],
   "source": [
    "for i in range(len(proto_images)):\n",
    "    print( proto_images[i] * feature_weights.detach().numpy() )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "969ccf07",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "rl_env",
   "language": "python",
   "name": "rl_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.9.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
