{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "1pKM2T0vClgw",
    "outputId": "7423d740-cf1e-4e82-9b40-6d84cf5ef74d"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Drive already mounted at /content/gdrive; to attempt to forcibly remount, call drive.mount(\"/content/gdrive\", force_remount=True).\n"
     ]
    }
   ],
   "source": [
    "from google.colab import drive\n",
    "drive.mount('/content/gdrive')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {
    "id": "yBDRJDKvIh-B"
   },
   "outputs": [],
   "source": [
    "import torchtext\n",
    "import torch\n",
    "import numpy as np\n",
    "from torchtext.vocab import GloVe\n",
    "from torchtext.data import get_tokenizer\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "# turning off automatic plot showing, and setting style\n",
    "plt.style.use('bmh')\n",
    "\n",
    "plt.rcParams['axes.facecolor'] = 'white'\n",
    "plt.rcParams['figure.figsize'] = 9, 6"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "2TA5bxYtD88e",
    "outputId": "ac213fbb-f36c-40bc-b2f3-f0252a0e5431"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Requirement already satisfied: portalocker>=2.0.0 in /usr/local/lib/python3.10/dist-packages (2.8.2)\n"
     ]
    }
   ],
   "source": [
    "pip install 'portalocker>=2.0.0'"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {
    "id": "tYcYefXFIJ7V"
   },
   "outputs": [],
   "source": [
    "global_vectors = GloVe(name='6B', dim=300)\n",
    "\n",
    "tokenizer = get_tokenizer(\"basic_english\")\n",
    "\n",
    "# def tokenize(label, line):\n",
    "#     return line.split()\n",
    "\n",
    "# tokens = []\n",
    "# for label, line in train_iter:\n",
    "#     tokens += tokenize(label, line)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "id": "18xYSPJQWYw0"
   },
   "outputs": [],
   "source": [
    "# After running the previous cell for the first time, restart the kernel and run all"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {
    "id": "uyAMnDs5BNuw"
   },
   "outputs": [],
   "source": [
    "train_iter, test_iter = torchtext.datasets.IMDB(split=('train', 'test'))\n",
    "\n",
    "embeddings_IMDB_train = np.zeros((25000,300))  # IMDB 25000, AGNEWS 120000,  Amazon 3000000, DBpedia 560000, Yahoo 1400000\n",
    "labels_IMDB_train = np.zeros(25000)\n",
    "\n",
    "i = 0\n",
    "#labels = []\n",
    "for label, line in train_iter:\n",
    "\n",
    "    if i%1 == 0:\n",
    "        embedding = global_vectors.get_vecs_by_tokens(tokenizer(line), lower_case_backup=True)\n",
    "        embedding_mean = torch.mean(embedding,0,True)\n",
    "        embeddings_IMDB_train[i//1,:] = embedding_mean\n",
    "        labels_IMDB_train[i//1] = label\n",
    "    i += 1\n",
    "\n",
    "    #labels.append(label)\n",
    "\n",
    "#print(i)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "hHXwJ5sLW21F",
    "outputId": "e58a96a0-0a39-45c6-8dcb-bf962e80c022"
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([1., 1., 1., ..., 2., 2., 2.])"
      ]
     },
     "execution_count": 13,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "labels_IMDB_train"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "C8JJ5tYF8IRq",
    "outputId": "2af98f90-9625-46a5-d68a-d074e84c0c61"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "25000\n"
     ]
    }
   ],
   "source": [
    "print(i)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {
    "id": "d0JiWsQ9eGXt"
   },
   "outputs": [],
   "source": [
    "embeddings_IMDB_test = np.zeros((500,300))\n",
    "labels_IMDB_test = np.zeros(500)\n",
    "\n",
    "i = 0\n",
    "for label, line in test_iter:\n",
    "    if i >= 500:\n",
    "        break\n",
    "    embedding = global_vectors.get_vecs_by_tokens(tokenizer(line), lower_case_backup=True)\n",
    "    embedding_mean = torch.mean(embedding,0,True)\n",
    "    embeddings_IMDB_test[i,:] = embedding_mean\n",
    "    labels_IMDB_test[i] = label\n",
    "    i += 1\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "QESz5q6wVvx6",
    "outputId": "78262b33-418a-4be0-8d53-a3242cc08fcb"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(25000, 300)\n",
      "(25000,)\n"
     ]
    }
   ],
   "source": [
    "latent_train = embeddings_IMDB_train\n",
    "y = labels_IMDB_train\n",
    "print(latent_train.shape)\n",
    "print(y.shape)\n",
    "#y = np.array([int(y[i]) for i in range(25000)])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "ZvqVgNYHPPWN",
    "outputId": "82bff148-3ac7-4c2a-d98e-c7f5eeca474d"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[2. 2. 1. ... 1. 1. 1.]\n"
     ]
    }
   ],
   "source": [
    "indices = np.random.permutation(y.shape[0])\n",
    "\n",
    "y = y[indices]\n",
    "latent_train = latent_train[indices,:]\n",
    "print(y)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "metadata": {
    "id": "JTw85MWrIoYD"
   },
   "outputs": [],
   "source": [
    "y = y[:25000]\n",
    "latent_train = latent_train[:25000,:]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "metadata": {
    "id": "vnBrVZ5-q3JY"
   },
   "outputs": [],
   "source": [
    "# np.random.shuffle(y)\n",
    "# print(y)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "q50sH-Lrh418"
   },
   "source": [
    "# Tree Model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "vOW6M9AdU1WI",
    "outputId": "d024f9e9-19f9-4979-ac97-647ef6597c80"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Collecting PIMS\n",
      "  Downloading PIMS-0.6.1.tar.gz (86 kB)\n",
      "\u001b[?25l     \u001b[90m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m0.0/86.0 kB\u001b[0m \u001b[31m?\u001b[0m eta \u001b[36m-:--:--\u001b[0m\r",
      "\u001b[2K     \u001b[91m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m\u001b[90m╺\u001b[0m\u001b[90m━\u001b[0m \u001b[32m81.9/86.0 kB\u001b[0m \u001b[31m2.9 MB/s\u001b[0m eta \u001b[36m0:00:01\u001b[0m\r",
      "\u001b[2K     \u001b[90m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m86.0/86.0 kB\u001b[0m \u001b[31m2.1 MB/s\u001b[0m eta \u001b[36m0:00:00\u001b[0m\n",
      "\u001b[?25h  Preparing metadata (setup.py) ... \u001b[?25l\u001b[?25hdone\n",
      "Requirement already satisfied: imageio in /usr/local/lib/python3.10/dist-packages (from PIMS) (2.31.3)\n",
      "Requirement already satisfied: numpy>=1.19 in /usr/local/lib/python3.10/dist-packages (from PIMS) (1.23.5)\n",
      "Collecting slicerator>=0.9.8 (from PIMS)\n",
      "  Downloading slicerator-1.1.0-py3-none-any.whl (10 kB)\n",
      "Requirement already satisfied: pillow>=8.3.2 in /usr/local/lib/python3.10/dist-packages (from imageio->PIMS) (9.4.0)\n",
      "Building wheels for collected packages: PIMS\n",
      "  Building wheel for PIMS (setup.py) ... \u001b[?25l\u001b[?25hdone\n",
      "  Created wheel for PIMS: filename=PIMS-0.6.1-py3-none-any.whl size=82615 sha256=204b09cd26444a7eed9f1dd9df4e3cd9223bb1b3505da58b2c8212fa0e5b8a56\n",
      "  Stored in directory: /root/.cache/pip/wheels/cc/bf/3e/bfa77232d942f8244145f9c713b6b38f6ef04b6fb5c021c114\n",
      "Successfully built PIMS\n",
      "Installing collected packages: slicerator, PIMS\n",
      "Successfully installed PIMS-0.6.1 slicerator-1.1.0\n"
     ]
    }
   ],
   "source": [
    "pip install PIMS"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "metadata": {
    "id": "C0IV7J9XU1YJ"
   },
   "outputs": [],
   "source": [
    "%matplotlib inline\n",
    "\n",
    "# importing relevant libraries\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import scipy as sp\n",
    "import matplotlib.pyplot as plt\n",
    "from sklearn.ensemble import RandomForestClassifier, ExtraTreesClassifier\n",
    "from sklearn.preprocessing import OneHotEncoder, StandardScaler\n",
    "from sklearn.model_selection import cross_val_predict, StratifiedKFold\n",
    "from sklearn.metrics import roc_auc_score, average_precision_score, precision_recall_curve, auc#plot_precision_recall_curve\n",
    "from sklearn.datasets import make_classification\n",
    "from sklearn.preprocessing import MinMaxScaler\n",
    "from tqdm import tqdm\n",
    "import sklearn\n",
    "#from umap import UMAP\n",
    "#from pynndescent import NNDescent\n",
    "#from fastcluster import single\n",
    "from scipy.cluster.hierarchy import cut_tree, fcluster, dendrogram\n",
    "from scipy.spatial.distance import squareform\n",
    "from sklearn.tree import DecisionTreeClassifier, ExtraTreeClassifier\n",
    "from pims import ImageSequence\n",
    "from PIL import Image\n",
    "from scipy.spatial.distance import hamming\n",
    "\n",
    "# turning off automatic plot showing, and setting style\n",
    "plt.style.use('bmh')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "metadata": {
    "id": "YXGyB5CeU1aP"
   },
   "outputs": [],
   "source": [
    "et = ExtraTreesClassifier(n_estimators=500, min_samples_leaf=100,\n",
    "                          max_features=\"sqrt\", bootstrap=True, class_weight='balanced', n_jobs=-1)\n",
    "\n",
    "# et = RandomForestClassifier(n_estimators=500, min_samples_leaf=100,\n",
    "#                           max_features=\"sqrt\", bootstrap=True, class_weight='balanced', n_jobs=-1)\n",
    "\n",
    "# validation instance\n",
    "skf = StratifiedKFold(n_splits=5, shuffle=True)\n",
    "\n",
    "# getting the model validation predictions\n",
    "preds = cross_val_predict(et, latent_train, y, cv=skf, method='predict_proba')\n",
    "\n",
    "# evaluating the model\n",
    "#print('Area under the ROC Curve:', roc_auc_score(y, preds, multi_class='ovo'))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 92
    },
    "id": "i8TjmPi0U1c3",
    "outputId": "a8afd9d1-7f86-4496-efb0-20a21ddf734b"
   },
   "outputs": [
    {
     "data": {
      "text/html": [
       "<style>#sk-container-id-1 {color: black;background-color: white;}#sk-container-id-1 pre{padding: 0;}#sk-container-id-1 div.sk-toggleable {background-color: white;}#sk-container-id-1 label.sk-toggleable__label {cursor: pointer;display: block;width: 100%;margin-bottom: 0;padding: 0.3em;box-sizing: border-box;text-align: center;}#sk-container-id-1 label.sk-toggleable__label-arrow:before {content: \"▸\";float: left;margin-right: 0.25em;color: #696969;}#sk-container-id-1 label.sk-toggleable__label-arrow:hover:before {color: black;}#sk-container-id-1 div.sk-estimator:hover label.sk-toggleable__label-arrow:before {color: black;}#sk-container-id-1 div.sk-toggleable__content {max-height: 0;max-width: 0;overflow: hidden;text-align: left;background-color: #f0f8ff;}#sk-container-id-1 div.sk-toggleable__content pre {margin: 0.2em;color: black;border-radius: 0.25em;background-color: #f0f8ff;}#sk-container-id-1 input.sk-toggleable__control:checked~div.sk-toggleable__content {max-height: 200px;max-width: 100%;overflow: auto;}#sk-container-id-1 input.sk-toggleable__control:checked~label.sk-toggleable__label-arrow:before {content: \"▾\";}#sk-container-id-1 div.sk-estimator input.sk-toggleable__control:checked~label.sk-toggleable__label {background-color: #d4ebff;}#sk-container-id-1 div.sk-label input.sk-toggleable__control:checked~label.sk-toggleable__label {background-color: #d4ebff;}#sk-container-id-1 input.sk-hidden--visually {border: 0;clip: rect(1px 1px 1px 1px);clip: rect(1px, 1px, 1px, 1px);height: 1px;margin: -1px;overflow: hidden;padding: 0;position: absolute;width: 1px;}#sk-container-id-1 div.sk-estimator {font-family: monospace;background-color: #f0f8ff;border: 1px dotted black;border-radius: 0.25em;box-sizing: border-box;margin-bottom: 0.5em;}#sk-container-id-1 div.sk-estimator:hover {background-color: #d4ebff;}#sk-container-id-1 div.sk-parallel-item::after {content: \"\";width: 100%;border-bottom: 1px solid gray;flex-grow: 1;}#sk-container-id-1 div.sk-label:hover label.sk-toggleable__label {background-color: #d4ebff;}#sk-container-id-1 div.sk-serial::before {content: \"\";position: absolute;border-left: 1px solid gray;box-sizing: border-box;top: 0;bottom: 0;left: 50%;z-index: 0;}#sk-container-id-1 div.sk-serial {display: flex;flex-direction: column;align-items: center;background-color: white;padding-right: 0.2em;padding-left: 0.2em;position: relative;}#sk-container-id-1 div.sk-item {position: relative;z-index: 1;}#sk-container-id-1 div.sk-parallel {display: flex;align-items: stretch;justify-content: center;background-color: white;position: relative;}#sk-container-id-1 div.sk-item::before, #sk-container-id-1 div.sk-parallel-item::before {content: \"\";position: absolute;border-left: 1px solid gray;box-sizing: border-box;top: 0;bottom: 0;left: 50%;z-index: -1;}#sk-container-id-1 div.sk-parallel-item {display: flex;flex-direction: column;z-index: 1;position: relative;background-color: white;}#sk-container-id-1 div.sk-parallel-item:first-child::after {align-self: flex-end;width: 50%;}#sk-container-id-1 div.sk-parallel-item:last-child::after {align-self: flex-start;width: 50%;}#sk-container-id-1 div.sk-parallel-item:only-child::after {width: 0;}#sk-container-id-1 div.sk-dashed-wrapped {border: 1px dashed gray;margin: 0 0.4em 0.5em 0.4em;box-sizing: border-box;padding-bottom: 0.4em;background-color: white;}#sk-container-id-1 div.sk-label label {font-family: monospace;font-weight: bold;display: inline-block;line-height: 1.2em;}#sk-container-id-1 div.sk-label-container {text-align: center;}#sk-container-id-1 div.sk-container {/* jupyter's `normalize.less` sets `[hidden] { display: none; }` but bootstrap.min.css set `[hidden] { display: none !important; }` so we also need the `!important` here to be able to override the default hidden behavior on the sphinx rendered scikit-learn.org. See: https://github.com/scikit-learn/scikit-learn/issues/21755 */display: inline-block !important;position: relative;}#sk-container-id-1 div.sk-text-repr-fallback {display: none;}</style><div id=\"sk-container-id-1\" class=\"sk-top-container\"><div class=\"sk-text-repr-fallback\"><pre>ExtraTreesClassifier(bootstrap=True, class_weight=&#x27;balanced&#x27;,\n",
       "                     min_samples_leaf=100, n_estimators=500, n_jobs=-1)</pre><b>In a Jupyter environment, please rerun this cell to show the HTML representation or trust the notebook. <br />On GitHub, the HTML representation is unable to render, please try loading this page with nbviewer.org.</b></div><div class=\"sk-container\" hidden><div class=\"sk-item\"><div class=\"sk-estimator sk-toggleable\"><input class=\"sk-toggleable__control sk-hidden--visually\" id=\"sk-estimator-id-1\" type=\"checkbox\" checked><label for=\"sk-estimator-id-1\" class=\"sk-toggleable__label sk-toggleable__label-arrow\">ExtraTreesClassifier</label><div class=\"sk-toggleable__content\"><pre>ExtraTreesClassifier(bootstrap=True, class_weight=&#x27;balanced&#x27;,\n",
       "                     min_samples_leaf=100, n_estimators=500, n_jobs=-1)</pre></div></div></div></div></div>"
      ],
      "text/plain": [
       "ExtraTreesClassifier(bootstrap=True, class_weight='balanced',\n",
       "                     min_samples_leaf=100, n_estimators=500, n_jobs=-1)"
      ]
     },
     "execution_count": 25,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "et.fit(latent_train,y)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "bLOQzkj-h-PR"
   },
   "source": [
    "# Testing on ID Data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "nDH2IxZJU7_F",
    "outputId": "93b489f0-b86d-45ee-d9f1-8cb11c0dbb6c"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(25000, 500)\n",
      "[[108   9 200 ...  18   0 184]\n",
      " [ 80 199  47 ... 190   0  73]\n",
      " [ 24 128 138 ...  63   0  38]\n",
      " ...\n",
      " [ 67 171 115 ...  73   0 158]\n",
      " [ 58  21  37 ... 142   0  44]\n",
      " [  8 122  98 ... 101   0 133]]\n",
      "0.9872823727454921\n",
      "3.7684106645673534e-07\n"
     ]
    }
   ],
   "source": [
    "leaves_train = et.apply(latent_train)\n",
    "print(leaves_train.shape)\n",
    "print(leaves_train)\n",
    "\n",
    "distances_train = np.zeros((500,500))\n",
    "\n",
    "\n",
    "for i in range(500):\n",
    "    for j in range(500):\n",
    "        distances_train[i,j] = hamming(leaves_train[i,:], leaves_train[j,:])\n",
    "\n",
    "score_train = sum(distances_train)/499\n",
    "\n",
    "print(np.mean(score_train))\n",
    "print(np.cov(score_train))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "31c29XtTcnfT",
    "outputId": "24d28393-0168-4288-bec6-0398b0672da0"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(500, 500)\n",
      "[[184 180 177 ... 110   0  98]\n",
      " [  9 143 125 ...  80   0 147]\n",
      " [ 38 198 104 ... 195   0 126]\n",
      " ...\n",
      " [ 37 139  23 ... 190   0 129]\n",
      " [123  50 189 ... 172   0 155]\n",
      " [ 38 148   9 ... 101   0 140]]\n",
      "0.9858096352705426\n",
      "3.821504157208702e-06\n"
     ]
    }
   ],
   "source": [
    "latent_test_in = embeddings_IMDB_test\n",
    "\n",
    "leaves_test_in = et.apply(latent_test_in)\n",
    "print(leaves_test_in.shape)\n",
    "print(leaves_test_in)\n",
    "\n",
    "distances_test_in = np.zeros((500,500))\n",
    "\n",
    "for i in range(500):\n",
    "    for j in range(500):\n",
    "        distances_test_in[i,j] = hamming(leaves_test_in[i,:], leaves_test_in[j,:])\n",
    "\n",
    "score_test_in = sum(distances_test_in)/499\n",
    "\n",
    "print(np.mean(score_test_in))\n",
    "print(np.cov(score_test_in))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "metadata": {
    "id": "VJ62FZupcnhW"
   },
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "xy9FAM_1hvRc"
   },
   "source": [
    "# Testing on OOD Data"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "04jJq7uTpXZv"
   },
   "source": [
    "## AGNEWS"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "metadata": {
    "id": "SiDi3rQIIZXX"
   },
   "outputs": [],
   "source": [
    "train_iter, test_iter = torchtext.datasets.AG_NEWS(split=('train', 'test'))\n",
    "\n",
    "embeddings_AGNEWS = np.zeros((500,300))\n",
    "labels_AGNEWS = np.zeros((500,1))\n",
    "\n",
    "i = 0\n",
    "for label, line in test_iter:\n",
    "    if i >= 500:\n",
    "        break\n",
    "\n",
    "    embedding = global_vectors.get_vecs_by_tokens(tokenizer(line), lower_case_backup=True)\n",
    "    embedding_mean = torch.mean(embedding,0,True)\n",
    "    embeddings_AGNEWS[i,:] = embedding_mean\n",
    "    labels_AGNEWS[i,:] = label\n",
    "    i += 1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "RvlVi-BDDnur",
    "outputId": "9140f67f-a937-4331-9414-a014589fcdbf"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(500, 500)\n",
      "[[185  83  13 ... 149   0   5]\n",
      " [ 94  96 191 ... 147   0  22]\n",
      " [128  96  85 ... 195   0  22]\n",
      " ...\n",
      " [177  87  86 ... 149   0 189]\n",
      " [ 17  48  15 ... 153   0  11]\n",
      " [174 159  11 ...  10   0 105]]\n",
      "0.938111166332666\n",
      "0.00020483623278145575\n"
     ]
    }
   ],
   "source": [
    "latent_test_out_AGNEWS = embeddings_AGNEWS\n",
    "\n",
    "leaves_test_out_AGNEWS = et.apply(latent_test_out_AGNEWS)\n",
    "print(leaves_test_out_AGNEWS.shape)\n",
    "print(leaves_test_out_AGNEWS)\n",
    "\n",
    "distances_test_out_AGNEWS = np.zeros((500,500))\n",
    "\n",
    "for i in range(500):\n",
    "    for j in range(500):\n",
    "        distances_test_out_AGNEWS[i,j] = hamming(leaves_test_out_AGNEWS[i,:], leaves_test_out_AGNEWS[j,:])\n",
    "\n",
    "score_test_out_AGNEWS = sum(distances_test_out_AGNEWS)/499\n",
    "\n",
    "print(np.mean(score_test_out_AGNEWS))\n",
    "print(np.cov(score_test_out_AGNEWS))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "metadata": {
    "id": "xETXhHZNgWIQ"
   },
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "aoVbJ6tZpVUK"
   },
   "source": [
    "## Amazon"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "metadata": {
    "id": "K-1UHi5cgWKj"
   },
   "outputs": [],
   "source": [
    "train_iter, test_iter = torchtext.datasets.AmazonReviewFull(split=('train', 'test'))\n",
    "\n",
    "embeddings_Amazon = np.zeros((500,300))\n",
    "labels_Amazon = np.zeros((500,1))\n",
    "\n",
    "i = 0\n",
    "for label, line in test_iter:\n",
    "    if i >= 500:\n",
    "        break\n",
    "\n",
    "    embedding = global_vectors.get_vecs_by_tokens(tokenizer(line), lower_case_backup=True)\n",
    "    embedding_mean = torch.mean(embedding,0,True)\n",
    "    embeddings_Amazon[i,:] = embedding_mean\n",
    "    labels_Amazon[i,:] = label\n",
    "    i += 1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "-ebGUdApgWMq",
    "outputId": "eac3dc8e-53cb-4123-99b8-011caf438474"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(500, 500)\n",
      "[[198 208 152 ... 143   0 202]\n",
      " [156 116  79 ... 179   0  51]\n",
      " [135 193 107 ...  79   0  47]\n",
      " ...\n",
      " [131  88  86 ... 175   0 165]\n",
      " [195  58  68 ... 132   0 184]\n",
      " [197 117 193 ...  84   0  51]]\n",
      "0.9762603767535086\n",
      "2.385458426870338e-05\n"
     ]
    }
   ],
   "source": [
    "latent_test_out_Amazon = embeddings_Amazon\n",
    "\n",
    "leaves_test_out_Amazon = et.apply(latent_test_out_Amazon)\n",
    "print(leaves_test_out_Amazon.shape)\n",
    "print(leaves_test_out_Amazon)\n",
    "\n",
    "distances_test_out_Amazon = np.zeros((500,500))\n",
    "\n",
    "for i in range(500):\n",
    "    for j in range(500):\n",
    "        distances_test_out_Amazon[i,j] = hamming(leaves_test_out_Amazon[i,:], leaves_test_out_Amazon[j,:])\n",
    "\n",
    "score_test_out_Amazon = sum(distances_test_out_Amazon)/499\n",
    "\n",
    "print(np.mean(score_test_out_Amazon))\n",
    "print(np.cov(score_test_out_Amazon))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "_1XTX85l_sxi"
   },
   "source": [
    "## YahooAnswers"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 36,
   "metadata": {
    "id": "DUuT2EfZ_0V6"
   },
   "outputs": [],
   "source": [
    "train_iter, test_iter = torchtext.datasets.YahooAnswers(split=('train', 'test'))\n",
    "\n",
    "embeddings_YahooAnswers = np.zeros((500,300))\n",
    "labels_YahooAnswers = np.zeros((500,1))\n",
    "\n",
    "i = 0\n",
    "for label, line in test_iter:\n",
    "    if i >= 500:\n",
    "        break\n",
    "\n",
    "    embedding = global_vectors.get_vecs_by_tokens(tokenizer(line), lower_case_backup=True)\n",
    "    embedding_mean = torch.mean(embedding,0,True)\n",
    "    embeddings_YahooAnswers[i,:] = embedding_mean\n",
    "    labels_YahooAnswers[i,:] = label\n",
    "    i += 1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 37,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "8j5gu9bo_0aU",
    "outputId": "3715670f-f7b1-43c1-cf9a-895c3e45a8e5"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(500, 500)\n",
      "[[135 116  49 ...  79   0 108]\n",
      " [197 159 127 ... 175   0 184]\n",
      " [ 21 144  23 ...  87   0 111]\n",
      " ...\n",
      " [ 91  95  19 ... 128   0 205]\n",
      " [198 174  22 ... 132   0 111]\n",
      " [ 81  28  19 ... 164   0 205]]\n",
      "0.9674168496994\n",
      "6.622242878509217e-05\n"
     ]
    }
   ],
   "source": [
    "latent_test_out_YahooAnswers = embeddings_YahooAnswers\n",
    "\n",
    "leaves_test_out_YahooAnswers = et.apply(latent_test_out_YahooAnswers)\n",
    "print(leaves_test_out_YahooAnswers.shape)\n",
    "print(leaves_test_out_YahooAnswers)\n",
    "\n",
    "distances_test_out_YahooAnswers = np.zeros((500,500))\n",
    "\n",
    "for i in range(500):\n",
    "    for j in range(500):\n",
    "        distances_test_out_YahooAnswers[i,j] = hamming(leaves_test_out_YahooAnswers[i,:], leaves_test_out_YahooAnswers[j,:])\n",
    "\n",
    "score_test_out_YahooAnswers = sum(distances_test_out_YahooAnswers)/499\n",
    "\n",
    "print(np.mean(score_test_out_YahooAnswers))\n",
    "print(np.cov(score_test_out_YahooAnswers))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 37,
   "metadata": {
    "id": "RHtD2LQV_0b3"
   },
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "o-vbLK-SAS8g"
   },
   "source": [
    "## YelpReviewFull"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 38,
   "metadata": {
    "id": "NCInXmSkAXsa"
   },
   "outputs": [],
   "source": [
    "train_iter, test_iter = torchtext.datasets.YelpReviewFull(split=('train', 'test'))\n",
    "\n",
    "embeddings_Yelp = np.zeros((500,300))\n",
    "labels_Yelp = np.zeros((500,1))\n",
    "\n",
    "i = 0\n",
    "for label, line in test_iter:\n",
    "    if i >= 500:\n",
    "        break\n",
    "\n",
    "    embedding = global_vectors.get_vecs_by_tokens(tokenizer(line), lower_case_backup=True)\n",
    "    embedding_mean = torch.mean(embedding,0,True)\n",
    "    embeddings_Yelp[i,:] = embedding_mean\n",
    "    labels_Yelp[i,:] = label\n",
    "    i += 1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 39,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "cde9OCVNAXwB",
    "outputId": "1199acdb-c898-4ac9-849e-4552f9a9180c"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(500, 500)\n",
      "[[197 171 193 ...  60   0  44]\n",
      " [185  81 137 ...  67   0 194]\n",
      " [147 208 177 ...   6   0 209]\n",
      " ...\n",
      " [ 24 165 118 ...  79   0  91]\n",
      " [ 28 121 118 ...  83   0  73]\n",
      " [103 133  93 ...  54   0  42]]\n",
      "0.9520103567134277\n",
      "0.00019708589674148839\n"
     ]
    }
   ],
   "source": [
    "latent_test_out_Yelp = embeddings_Yelp\n",
    "\n",
    "leaves_test_out_Yelp = et.apply(latent_test_out_Yelp)\n",
    "print(leaves_test_out_Yelp.shape)\n",
    "print(leaves_test_out_Yelp)\n",
    "\n",
    "distances_test_out_Yelp = np.zeros((500,500))\n",
    "\n",
    "for i in range(500):\n",
    "    for j in range(500):\n",
    "        distances_test_out_Yelp[i,j] = hamming(leaves_test_out_Yelp[i,:], leaves_test_out_Yelp[j,:])\n",
    "\n",
    "score_test_out_Yelp = sum(distances_test_out_Yelp)/499\n",
    "\n",
    "print(np.mean(score_test_out_Yelp))\n",
    "print(np.cov(score_test_out_Yelp))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 39,
   "metadata": {
    "id": "2UJ2KTSSbzXw"
   },
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 39,
   "metadata": {
    "id": "y2on7W6sbzaG"
   },
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "O2sjOYWZDluL"
   },
   "source": [
    "# Results"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 40,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 517
    },
    "id": "xEdmoavjAX9W",
    "outputId": "37b0ec7b-95ef-4819-f056-9d4cc910536e"
   },
   "outputs": [
    {
     "data": {
      "image/png": "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jx6tOXPmaNeuXZ93E8Fkp8scnMUwDE2bNk3Tp0/v9b2f//znio2N1YEDB2yorG+/+tWv5HK5dPvtt9tdCvyYv+XeCRh8/FRKSopefPFFtbS0eB5rbW3VL3/5S6WmpvZafsaMGTp06JAqKyv1+uuva+rUqfqP//gPXXXVVers7PRadsWKFTp06JCqqqr0wgsv6K233tKdd97p89cE31q7dq3uuOMOvfXWW17DTFtbm6644go98MADmj9/vt566y19/PHHeuyxx3TkyBGtXr3aaz1ut1sPPfTQGQ3DpaWlOnTokNefESNG6NJLL1VISIg2b97sWbakpEQtLS06duyYKisrPY8XFhYqLCxMkydPVkdHh6644gqdOHFCr7zyikpLS/XSSy/pC1/4go4fP/55NxFM1l/m4DxBQUF65plntHXrVq1Zs8bz+N69e7V06VKtXr1ao0ePtrFCb2vXrtXSpUv1q1/9Sq2trXaXMyDt7e12l4D/42+5dwIGHz91wQUXKCUlRa+88ornsVdeeUWpqak6//zzey0fFhamxMREJScn64ILLtDdd9+t3//+93r99df17LPPei0bHR3tWXbq1Km64YYb9MEHH/j6JcGHGhsb9dJLL+nWW2/VrFmzvH7mP/3pT/XOO+/or3/9q+68805NmDBBqampmjJlip588kk98MADXuuaNm2aEhMT9eCDD37m3ztixAglJiZ6/QkODlZUVJQmTpzoNfhs3rxZl156qSZPntzr8Ysvvlhut1v/+Mc/tGfPHv385z/XxRdfrLS0NE2ePFk//OEPdfHFF3/ezQQTnS5zPXsEN27cqPPPP1/h4eH64he/qMOHD+v1119Xbm6uYmJi9I1vfEPNzc2e523YsEGXXnqpYmNjNXz4cF111VXas2eP5/v33Xdfn3sZe/7utrY23XnnnRoxYoTcbrcuvfRSbd++vVddmzZt0oUXXqiIiAhdcsklKi0t9fn2coKUlBT97Gc/03e+8x3t3btXhmHopptu0hVXXKHNmzdrzJgxCg8PV05Ojn72s5/1uY6f/OQnGjVqlIYPH67bb79dHR0dnu8dO3ZM8+bN07BhwxQREaEvf/nLKi8v93r+yy+/rHPOOUdhYWFKT0/XI4880uvv2Lt3r9577z3dddddys7O9nqflaRnn31WsbGx2rhxo3JzcxUVFeX5cLHH5s2bddFFFykyMlKxsbGaPHmy9u3bpxMnTsjlcun999+XdOq+hHFxcV796/nnn1dKSorn6/379+trX/uaYmNjFRcXpzlz5nh9ONRzRMj999+vpKQk5eTkSDq1RyErK0tut1sjR47UV7/61c/6EcEH+sv9lVdeqfPPP19f/vKXFRUVpZEjR+qb3/ymjhw50u+60tPTtXLlSn39619XZGSkkpOT9cQTT1j4avwfg48fu/HGG/XMM894vl63bp0WLFhwxs//4he/qPz8/F5N/ZOqq6v12muvadKkSZ+rVtjr17/+tcaNG6ecnBxdf/31WrdunXpu4fWrX/1KV1xxRZ8Ds6ReNwl2uVx64IEHtHr16s+1i37q1KkqLCz0fF1YWKjLL79cU6ZM8Xp88+bNmjp1qiQpISFBwcHB+u1vf6uurq6z/rvhe6fLXI/77rtPjz/+uN577z3PL3erVq3SL3/5S/3pT3/SG2+84bXHsampSUuWLNH777+vTZs2KTg4WNdcc43nkN3vfOc7XnsXf/KTnygiIkIXXnihJGnp0qV6+eWX9dxzz+mDDz7Q2LFjNX36dNXX13vV9f3vf1+PPPKI3n//fYWEhOjGG2/08dZyjhtuuEFf+tKXdOONN+rxxx/X//7v/+qpp57S6NGj9Zvf/EbFxcVavny57r77bv3617/2em5hYaH27NmjwsJCPffcc3r22We9Bur58+fr/fff1x/+8AcVFRXJMAzNnDnTMxzt2LFDX/va1/Rv//Zv+vjjj3Xffffpnnvu6fXh3zPPPKNZs2Zp6NChuv7667V27dper6O5uVk/+clPtH79er311luqqqrSd77zHUlSZ2enrr76ak2ZMkV///vfVVRUpG9961sKCgrS0KFDdd5553k+3Pn4448VFBSkDz/8UI2NjZKkLVu2aMqUKZKkjo4OTZ8+XdHR0Xr77bf17rvvegatT+7Z2bRpk0pLS/Xmm2/qj3/8o95//33deeedWrFihUpLS7VhwwZddtlln+tnh7PXV+7XrFmjL37xizr//PP1/vvva8OGDaqtrdXXvva1067r4YcfVn5+vj788EPddddd+o//+A+9+eabFr2SAGDAL9xwww3GnDlzvP7/8OHDRlhYmFFZWWlUVlYabrfbqKurM+bMmWPccMMNfT730+bOnWvk5uZ6vk5LSzNCQ0ONyMhIw+12G5KMSZMmGceOHfPdi4PPXXLJJcaqVasMwzCMjo4OIz4+3igsLDQMwzDcbrdx5513ei1/9dVXG5GRkUZkZKRRUFDgefyTWbr44ouNG2+80TAMw/jd735nfLKdFBYWGpI86+j5k5eX51nmzTffNCQZBw8eNAzDMEaMGGFs27bNeO+994y0tDTDMAxjz549hiRjy5Ytnuc9/vjjRkREhBEdHW1MnTrVWLFihbFnzx5zNhRMc7rM9eTjL3/5i2f5Bx980JDk9bNcuHChMX369H7/jrq6OkOS8fHHH/f6XlFRkeF2u42XXnrJMAzDaGxsNIYMGWK88MILnmXa29uNpKQk48c//nG/df3pT38yJBktLS1nsRXQl9raWiM+Pt4IDg42fve73/W5zO23325ce+21nq9vuOEGIy0tzejs7PQ89q//+q/G3LlzDcMwjLKyMkOS8e6773q+f+TIESM8PNz49a9/bRiGYXzjG98wrrjiCq+/57vf/a5XX+rq6jJSUlKMV1991TCMUxkLDQ01KioqPMs888wzhiRj9+7dnseeeOIJY+TIkYZhGMbRo0cNScbmzZv7fG1LliwxZs2aZRiGYaxatcqYO3eukZ+fb7z++uuGYRjG2LFjjaeeesowDMNYv369kZOTY3R3d3ue39bWZoSHhxsbN270bJuRI0cabW1tnmVefvllIyYmxjh58mSfNcB6n879ypUrjSuvvNJrmf379xuSjNLSUsMwDGPKlCnGf/zHf3i+n5aWZsyYMcPrOXPnzjW+/OUv+7z+QMEeHz+WkJDgOYSk5xOq+Pj4Aa3DMIxen+h/97vf1UcffaS///3vnhPKZ82axSfsfqq0tFTbtm3T17/+dUlSSEiI5s6d2+enmD1+/vOf66OPPtKNN97odajRJz300EN67rnnVFJS0u963n77bX300UeeP3/+858937vkkksUGhqqzZs3q7i4WC0tLbrgggt04YUXqq6uTnv37tXmzZsVHh7udRjI7bffrpqaGr3wwgsqKCjQb37zG51zzjl84jWInGnmzj33XM//jxw5UhEREcrIyPB67PDhw56vy8vL9fWvf10ZGRmKiYnxXAijqqrKa71VVVW6+uqr9Z3vfMfz6emePXvU0dGhyZMne5YbMmSILrrool4Z/mRdo0aNkiSvOvD5jBgxQgsXLlRubq7noj1PPPGEJkyYoISEBEVFRempp57q9XM955xz5HK5PF+PGjXK83MpKSlRSEiI19EJw4cPV05OjufnW1JS4vXzl6TJkyervLzc8/725ptvqqmpSTNnzpQkxcfHey4G9EkRERHKzMzss5a4uDjNnz9f06dP1+zZs/Wzn/3M6zC4KVOm6J133lFXV5e2bNmiyy+/XJdffrk2b96sgwcPavfu3br88sslSTt37tTu3bsVHR2tqKgoRUVFKS4uTq2trV6HeX7hC19QaGio5+srrrhCaWlpysjI0De/+U298MIL/fZyWOPTud+5c6cKCws9P9eoqCiNGzdOkrx+tp9WUFDQ6+vTvQ/DW4jdBeDzufHGG7Vo0SJJOqvjPEtKSjRmzBivx+Lj4zV27FhJUlZWllatWqWCggIVFhZq2rRpn79oWGrt2rXq7OxUUlKS5zHDMBQWFqbHH39cWVlZvc5h6PllLy4urt/1XnbZZZo+fbq+973veV1F8JPGjBmj2NjYPr8XERGhiy66SIWFhaqvr9ell14ql8sll8ulSy65RIWFhSosLNTkyZO93tClU+ehzZ49W7Nnz9YPf/hDTZ8+XT/84Q91xRVXnMEWga99VuZ6DBkyxPP/PVed/KSgoCCvK0/Onj1baWlpevrpp5WUlKTu7m6NHz/e65CfpqYmfeUrX1FBQYFWrFhxVvV/ui5Jva6Aic8nJCREISGnfgV58cUX9Z3vfEePPPKICgoKFB0drYcfflhbt271es5n5cMMa9euVX19vcLDwz2PdXd36+9//7t+8IMfKDg4uN9ajE8cyvnMM8/ozjvv1IYNG/TSSy9p2bJlevPNN3XxxRfrsssuU0NDgz744AO99dZbeuCBB5SYmKgf/ehHys/PV1JSkrKysiSdOlduwoQJeuGFF3rVmpCQ4Pn/yMhIr+9FR0frgw8+0ObNm/XGG29o+fLluu+++7R9+/Z+ezJ875O5b2xs1OzZs/XQQw/1Wq7nPRjmY4+Pn+s5zrfnOOCB+Otf/6qPP/5Y11577WmX6/mE7ZNXkIN/6Ozs1P/8z//okUce8drzsnPnTiUlJelXv/qVvv71r+vNN9/Uhx9+OOD1/+hHP9Jrr72moqKis6pv6tSp2rx5szZv3uz5hFM6NVRt3rxZW7Zs8Zzf05+goCCNGzdOTU1NZ1UDzHUmmTsbR48eVWlpqZYtW6YvfelLys3N7XVlQcMwdP3116u7u1vr16/32pudmZmp0NBQvfvuu57HOjo6tH37duXl5Z3di4Up3n33XV1yySW67bbbdP7552vs2LGn/cS7L7m5uers7PQalnoy0/Pzzc3N9fr59/zd2dnZcrlcOnr0qH7/+9/rxRdf9Mruhx9+qGPHjumNN94YUE3nn3++vve97+m9997T+PHj9ctf/lKSFBsbq3PPPVePP/64hgwZonHjxumyyy7Thx9+qD/+8Y+e83ukUxcyKi8v14gRIzR27FivP0OHDj3t3x8SEqJp06bpxz/+sf7+97+rsrJSf/3rXwf0GuA7F1xwgf7xj38oPT2918/204PsJ/3tb3/r9XVubq6vyw0Y7PHxcy6Xy7OL85OHAHxaW1ubampq1NXVpdraWm3YsEEPPvigrrrqKs2bN89r2YaGBtXU1MgwDO3fv19Lly5VQkKCLrnkEp++Fpjvj3/8o44dO6abbrqp15vktddeq7Vr1+rtt9/Wn/70J33pS1/Svffeq//3//6fhg0bprKyMr3++uunzdUXvvAFXXfddXrsscf6/P7hw4d7XQp2+PDhnk9Lp06dqpUrV6qmpsZzYrB06lCQhx9+WA0NDV6Dz0cffaR7771X3/zmN5WXl6fQ0FBt2bJF69at03/9138NePvAfGeSuYcffnjA6x02bJiGDx+up556SqNGjVJVVZXuuusur2Xuu+8+/eUvf9Ebb7yhxsZGz8niQ4cOVWRkpG699VZ997vfVVxcnFJTU/XjH/9Yzc3Nuummm87+BeNzy8rK0v/8z/9o48aNGjNmjNavX6/t27f3Ohrhs9YxZ84c3XzzzVqzZo2io6N11113KTk5WXPmzJEk/ed//qcmTpyolStXau7cuSoqKtLjjz+un//855Kk9evXa/jw4fra177W6xDwmTNnau3atZoxY8Zn1rJ371499dRT+spXvqKkpCSVlpaqvLzc67328ssv1+rVqz1XWouLi1Nubq5eeuklr6M3rrvuOj388MOaM2eOVqxYodGjR2vfvn165ZVXtHTp0n4vhfzHP/5RFRUVuuyyyzRs2DD9+c9/Vnd3t+eKb7Df7bffrqefflpf//rXtXTpUsXFxWn37t168cUX9d///d/9vve+++67+vGPf6yrr75ab775pn7zm9/oT3/6k8XV+y/2+ASAmJgYxcTEnHaZDRs2eG4OOWPGDBUWFuqxxx7T73//+17/uJYvX65Ro0YpKSlJV111lSIjI/XGG29o+PDhvnwZ8IG1a9dq2rRpfX4yeO211+r9999XWVmZNm3apP/6r//SM888o0svvVS5ublavHixJk+erFdfffW0f8eKFSv6PdwkJydHo0aN8vqzY8cOz/cLCgoUFhYmwzA0YcIEz+OTJk1SR0eH57LXPUaPHq309HT94Ac/0KRJk3TBBRfoZz/7mX7wgx/o+9///gC3DnzhTDL397//fcDrDQ4O1osvvqgdO3Zo/Pjx+va3v91rgNqyZYsaGxt1ySWXeGXupZdeknRqD+W1116rb37zm7rgggu0e/dubdy4UcOGDTu7FwtTLFy4UP/yL/+iuXPnatKkSTp69Khuu+22Aa/nmWee0YQJE3TVVVepoKBAhmHoz3/+s+eDlgsuuEC//vWv9eKLL2r8+PFavny5VqxY4TlUd926dbrmmmt6DT3Sqez+4Q9/OO2lhntERERo165duvbaa5Wdna1vfetbuv3227Vw4ULPMlOmTFFXV5fXnu7LL7+812MRERF66623lJqaqn/5l39Rbm6ubrrpJrW2tp72fT82NlavvPKKvvjFLyo3N1dPPvmkfvWrX+mcc875zPphjaSkJL377rvq6urSlVdeqS984QtavHixYmNjPYdU9uU///M/9f777+v888/XD3/4Qz366KMDPuLHyYIM41PXFwUAAAAwqKSnp2vx4sVavHix3aX4Lfb4AAAAAAh4DD4AAAAAAh6HugEAAAAIeOzxAQAAABDwGHwAAAAABDwGHwAAAAABj8EHAAAAQMBj8AEAAAAQ8Bh8AAAAAAQ8Bh8AAAAAAY/BBwAAAEDA+/8OqKMOCeuB6QAAAABJRU5ErkJggg==\n",
      "text/plain": [
       "<Figure size 1000x600 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "my_dict = {'IMDB': score_test_in, 'AGNEWS': score_test_out_AGNEWS, 'Amazon': score_test_out_Amazon, 'YahooAnswers': score_test_out_YahooAnswers, 'Yelp': score_test_out_Yelp}\n",
    "\n",
    "plt.figure(figsize=(10,6))\n",
    "plt.boxplot(my_dict.values(), labels=my_dict.keys());\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 41,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "Kinfmb_NDlRR",
    "outputId": "cd45cbc9-f665-4996-fa3a-54ea9f030da7"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1.0\n",
      "0.9613980000000001\n",
      "0.9955379999999999\n",
      "0.997156\n"
     ]
    }
   ],
   "source": [
    "score_pred_AGNEWS = np.concatenate([score_test_in, score_test_out_AGNEWS])\n",
    "score_pred_Amazon = np.concatenate([score_test_in, score_test_out_Amazon])\n",
    "score_pred_YahooAnswers = np.concatenate([score_test_in, score_test_out_YahooAnswers])\n",
    "score_pred_Yelp = np.concatenate([score_test_in, score_test_out_Yelp])\n",
    "score_true = np.concatenate([np.ones(500), np.zeros(500)])\n",
    "\n",
    "\n",
    "print(roc_auc_score(score_true, score_pred_AGNEWS))\n",
    "print(roc_auc_score(score_true, score_pred_Amazon))\n",
    "print(roc_auc_score(score_true, score_pred_YahooAnswers))\n",
    "print(roc_auc_score(score_true, score_pred_Yelp))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 42,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "vgMRPvQ6EuZx",
    "outputId": "73094072-6222-4958-81b8-1f9183fced95"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1.0\n",
      "0.9327127793073928\n",
      "0.9903822975422641\n",
      "0.993954799731196\n"
     ]
    }
   ],
   "source": [
    "precision_AGNEWS, recall_AGNEWS, thresholds_AGNEWS = precision_recall_curve(score_true, score_pred_AGNEWS)\n",
    "precision_Amazon, recall_Amazon, thresholds_Amazon = precision_recall_curve(score_true, score_pred_Amazon)\n",
    "precision_YahooAnswers, recall_YahooAnswers, thresholds_YahooAnswers = precision_recall_curve(score_true, score_pred_YahooAnswers)\n",
    "precision_Yelp, recall_Yelp, thresholds_Yelp = precision_recall_curve(score_true, score_pred_Yelp)\n",
    "\n",
    "auc_precision_recall_AGNEWS = auc(recall_AGNEWS, precision_AGNEWS)\n",
    "auc_precision_recall_Amazon = auc(recall_Amazon, precision_Amazon)\n",
    "auc_precision_recall_YahooAnswers = auc(recall_YahooAnswers, precision_YahooAnswers)\n",
    "auc_precision_recall_Yelp = auc(recall_Yelp, precision_Yelp)\n",
    "\n",
    "print(auc_precision_recall_AGNEWS)\n",
    "print(auc_precision_recall_Amazon)\n",
    "print(auc_precision_recall_YahooAnswers)\n",
    "print(auc_precision_recall_Yelp)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 43,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "sk0lXKvnEunm",
    "outputId": "cf234e15-106e-4991-a90c-4703a38963ee"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.0\n",
      "0.096\n",
      "0.012\n",
      "0.006\n"
     ]
    }
   ],
   "source": [
    "def compute_fpr95(y_true, y_pred_probs):\n",
    "    fpr, tpr, thresholds = sklearn.metrics.roc_curve(y_true, y_pred_probs)\n",
    "    idx = np.abs(tpr - 0.95).argmin()\n",
    "    fpr95 = fpr[idx]\n",
    "    return fpr95\n",
    "\n",
    "fpr95_score_AGNEWS = compute_fpr95(score_true, score_pred_AGNEWS)\n",
    "fpr95_score_Amazon = compute_fpr95(score_true, score_pred_Amazon)\n",
    "fpr95_score_YahooAnswers = compute_fpr95(score_true, score_pred_YahooAnswers)\n",
    "fpr95_score_Yelp = compute_fpr95(score_true, score_pred_Yelp)\n",
    "\n",
    "print(fpr95_score_AGNEWS)\n",
    "print(fpr95_score_Amazon)\n",
    "print(fpr95_score_YahooAnswers)\n",
    "print(fpr95_score_Yelp)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 43,
   "metadata": {
    "id": "tsPeBoegDJye"
   },
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 44,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "CB98WXQW_7dD",
    "outputId": "0f27af93-3584-4e8c-c0de-57b12b518b54"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.9858096352705426 0.0019548667875864847\n",
      "0.938111166332666 0.014312100921299281\n",
      "0.9762603767535086 0.004884115505258181\n",
      "0.9674168496994 0.00813771643552982\n",
      "0.9520103567134277 0.014038728458855823\n"
     ]
    }
   ],
   "source": [
    "print(np.mean(score_test_in), np.sqrt(np.cov(score_test_in)))\n",
    "print(np.mean(score_test_out_AGNEWS), np.sqrt(np.cov(score_test_out_AGNEWS)))\n",
    "print(np.mean(score_test_out_Amazon), np.sqrt(np.cov(score_test_out_Amazon)))\n",
    "print(np.mean(score_test_out_YahooAnswers), np.sqrt(np.cov(score_test_out_YahooAnswers)))\n",
    "print(np.mean(score_test_out_Yelp), np.sqrt(np.cov(score_test_out_Yelp)))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 44,
   "metadata": {
    "id": "WSMJCa1Gc1iy"
   },
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 44,
   "metadata": {
    "id": "UdUtMeaAdiI-"
   },
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "colab": {
   "provenance": []
  },
  "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.8.8"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 1
}
