{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {
    "executionInfo": {
     "elapsed": 5,
     "status": "ok",
     "timestamp": 1695324378707,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "F7toc08bpdQ1",
    "tags": []
   },
   "outputs": [],
   "source": [
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "import numpy as np\n",
    "from transformers import BertModel, BertConfig\n",
    "\n",
    "from transformers import *\n",
    "import pandas as pd\n",
    "import numpy as np\n",
    "import seaborn as sns\n",
    "import matplotlib.pyplot as plt\n",
    "from torch.utils.data import Dataset, DataLoader\n",
    "from sklearn.model_selection import train_test_split\n",
    "from sklearn.metrics import f1_score\n",
    "import textwrap\n",
    "import math\n",
    "import seaborn as sns\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "from IPython.display import clear_output\n",
    "import warnings\n",
    "warnings.filterwarnings('ignore')\n",
    "clear_output()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {
    "executionInfo": {
     "elapsed": 15,
     "status": "ok",
     "timestamp": 1695323483676,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "MLk4JWizs4S9",
    "tags": []
   },
   "outputs": [],
   "source": [
    "df_train = pd.read_csv('/home/m_nsu/ICLR/Datasets/IMDB/train.csv')\n",
    "df_val = pd.read_csv('/home/m_nsu/ICLR/Datasets/IMDB/val.csv')\n",
    "df_test = pd.read_csv('/home/m_nsu/ICLR/Datasets/IMDB/test.csv')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {
    "id": "YWiFI0a9QqEa",
    "tags": []
   },
   "outputs": [],
   "source": [
    "df_senti = pd.read_excel('/home/m_nsu/ICLR/Datasets/GoEmotion/sentiwords.xlsx')\n",
    "conditions = [\n",
    "    (df_senti['PosScore'] > df_senti['NegScore']),\n",
    "    (df_senti['PosScore'] < df_senti['NegScore']),\n",
    "    (df_senti['PosScore'] == df_senti['NegScore'])\n",
    "    ]\n",
    "\n",
    "values = ['Positive','Negative','Neutral']\n",
    "\n",
    "df_senti = df_senti[['PosScore','NegScore','Word','Definition']]\n",
    "df_senti['Sentiment'] = np.select(conditions, values)\n",
    "df_senti = df_senti.dropna(axis=0)\n",
    "df_senti.drop(columns=['PosScore', 'NegScore'], inplace=True)\n",
    "df_senti = df_senti[['Word', 'Sentiment', 'Definition']]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {
    "executionInfo": {
     "elapsed": 48,
     "status": "ok",
     "timestamp": 1695323754433,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "vX1DfZpN1W01",
    "tags": []
   },
   "outputs": [],
   "source": [
    "df_train.dropna(inplace=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 206
    },
    "executionInfo": {
     "elapsed": 49,
     "status": "ok",
     "timestamp": 1695323754437,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "e0NG9J-oq_wW",
    "outputId": "f1bb3869-364e-4c7a-8c3f-a62ce4b182c7",
    "tags": []
   },
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>Unnamed: 0</th>\n",
       "      <th>review</th>\n",
       "      <th>sentiment</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>48539</td>\n",
       "      <td>This was a very funny movie not Oscarworthy bu...</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>26308</td>\n",
       "      <td>I know this film has had a fairly rough ride f...</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>14603</td>\n",
       "      <td>Being stuck in bed with the flu and feeling to...</td>\n",
       "      <td>1</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>45794</td>\n",
       "      <td>This isnt exactly a great film but I admire th...</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>21688</td>\n",
       "      <td>It is difficult to rate a writerdirectors firs...</td>\n",
       "      <td>1</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "   Unnamed: 0                                             review  sentiment\n",
       "0       48539  This was a very funny movie not Oscarworthy bu...          0\n",
       "1       26308  I know this film has had a fairly rough ride f...          0\n",
       "2       14603  Being stuck in bed with the flu and feeling to...          1\n",
       "3       45794  This isnt exactly a great film but I admire th...          0\n",
       "4       21688  It is difficult to rate a writerdirectors firs...          1"
      ]
     },
     "execution_count": 5,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "df_train.head()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {
    "executionInfo": {
     "elapsed": 18,
     "status": "ok",
     "timestamp": 1695323756839,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "UmEtmaFNrakz",
    "tags": []
   },
   "outputs": [],
   "source": [
    "MAX_LEN = 200\n",
    "RANDOM_SEED = 42\n",
    "#device = torch.device('cuda' if torch.cuda.is_available() else 'cpu')\n",
    "device = torch.device(0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 18,
     "status": "ok",
     "timestamp": 1695323756840,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "6oWORty0p8Xo",
    "outputId": "cb12350b-753a-4c61-c7e8-3c9427f000e9",
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "cuda:0\n"
     ]
    }
   ],
   "source": [
    "PRE_TRAINED_MODEL_NAME = 'bert-base-uncased'\n",
    "config = BertConfig.from_pretrained(PRE_TRAINED_MODEL_NAME)\n",
    "tokenizer = BertTokenizer.from_pretrained(PRE_TRAINED_MODEL_NAME)\n",
    "clear_output()\n",
    "print(device)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {
    "id": "yDSL15DQGvrh",
    "tags": []
   },
   "outputs": [],
   "source": [
    "pre_trained_model = BertModel.from_pretrained(PRE_TRAINED_MODEL_NAME).to(device)\n",
    "clear_output()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {
    "executionInfo": {
     "elapsed": 35,
     "status": "ok",
     "timestamp": 1695323767305,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "R2PbApHJGstz",
    "tags": []
   },
   "outputs": [],
   "source": [
    "def get_cls(sent):\n",
    "  encoded_review = tokenizer.encode_plus(\n",
    "  sent,\n",
    "  max_length=MAX_LEN,\n",
    "  add_special_tokens=True,\n",
    "  return_token_type_ids=False,\n",
    "  truncation = True,\n",
    "  padding='max_length',\n",
    "  return_attention_mask=True,\n",
    "  return_tensors='pt',\n",
    "  )\n",
    "  input_ids = encoded_review['input_ids'].to(device)\n",
    "  attention_mask = encoded_review['attention_mask'].to(device)\n",
    "\n",
    "  pre_trained_model.eval()\n",
    "  with torch.no_grad():\n",
    "    output = pre_trained_model(input_ids, attention_mask)\n",
    "    return torch.mean(output[0], dim=1)[0] #Output Avg pooled vector embedding of last layer"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {
    "executionInfo": {
     "elapsed": 33,
     "status": "ok",
     "timestamp": 1695323767305,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "dN4TFCeRzGXw",
    "tags": []
   },
   "outputs": [],
   "source": [
    "neg_words = ['ambiguous','angry','annoying','appalling','awful','barbaric','bizarre','blasphemous','brainless','chaotic','contradictory','controversy','cringe','cruel',\n",
    "             'degrading','disturbed','failed','fake','gimmick','hateful','hideous','inadequate','inappropriate','incoherent','loopholes','meaningless','obscure','offensive',\n",
    "             'pathetic','weird']\n",
    "            \n",
    "pos_words = ['acclaimed','accurate','adventurous','astonishing','authentic','beautiful','calming','catchy','charismatic','cheerish','coherent','constructrive','cool',\n",
    "             'cute','daring','eloquent','enthusiastic','romantic','flawless','humorous','inspirational','love','modern','motivated','obsession','phenomenal','relistic',\n",
    "             'refreshing','vibrant','wholesome']\n",
    "\n",
    "\n",
    "new_pos_words = ['captivating','enjoy','outstanding','thoughtful','fun', 'pleasant', 'warm', 'enticing', 'realistic','friendly']\n",
    "new_neg_words = ['biased','horrible','bored','disappointed','frustrate','hostile','ridiculous','malign','rude','unpleasant']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {
    "id": "5hjM7ggAiHnM",
    "tags": []
   },
   "outputs": [],
   "source": [
    "positive_word_meaning_sentences = [\n",
    "'to praise or welcome somebody/something publicly', #acclaimed(verb)\n",
    "'correct and true in every detail', #accurate(adjective)\n",
    "'including new and interesting things, methods and ideas', #adventurous(adjective)\n",
    "'very surprising; difficult to believe',#astonishing (adjective)\n",
    "'known to be real and what somebody claims it is and not a copy',#authentic(adjective)\n",
    "'having beauty; giving pleasure to the senses or to the mind', #beautiful(adjective)\n",
    "'to make somebody/something become quiet and more relaxed, especially after strong emotion or excitement',#calming(verb)\n",
    "'pleasant and easily remembered',#catchy(adjective)\n",
    "'the powerful personal quality that some people have to attract and impress other people',#charisma(noun)\n",
    "'(of a person or their behaviour) happy and cheerful',#cheery(adjective)\n",
    "'(of ideas, thoughts, arguments, etc.) logical and well organized; easy to understand and clear',#coherent(adjective)\n",
    "'having a useful and helpful effect rather than being negative or with no purpose',#constructive(adjective)\n",
    "'used to show that you admire or approve of somebody/something because they are/it is fashionable, attractive and often different',#cool(adjective)\n",
    "'pretty and attractive',#cute(adjective)\n",
    "'brave; willing to do dangerous or unusual things; involving danger or taking risks',#daring(adjective)\n",
    "'able to use language and express your opinions well, especially when you are speaking in public',#eloquent(adjective)\n",
    "'feeling or showing a lot of excitement and interest about somebody/something',#enthusiastic(adjective)\n",
    "'without flaws and therefore perfect',#flawless(adjective)\n",
    "'funny; showing a sense of humour',#humorous(adjective)\n",
    "'providing exciting new ideas; making somebody want to create something, especially in art, literature or music',#inspirational(adjective)\n",
    "]\n",
    "\n",
    "new_positive_word_meaning_sentences = [\n",
    "'taking all your attention; very attractive and interesting',#captivating(adjective)\n",
    "'to get pleasure from something',#enjoy(verb)\n",
    "'extremely good; excellent',#outstanding(adjective)\n",
    "'showing that you think about and care for other people',#thoughtful(adjective)\n",
    "'the feeling of enjoying yourself; activities that you enjoy',#fun(noun)\n",
    "'affording pleasure; being in harmony with your taste or likings', \n",
    "'get warm or warmer', \n",
    "'provoke someone to do something through (often false or exaggerated', \n",
    "'aware or expressing awareness of things as they really are', \n",
    "'kind and pleasant',\n",
    "'a very strong feeling of liking and caring for somebody/something, especially a member of your family or a friend',#love(noun)\n",
    "'of the present time or recent times',#modern(adjective)\n",
    "'wanting to do something, especially something that involves hard work and effort',#motivated(adjective)\n",
    "'a person or thing that somebody thinks about too much',#obsession(noun)\n",
    "'very great or impressive',#phenomenal(adjective)\n",
    "'accepting in a sensible way what it is actually possible to do or achieve in a particular situation',#realistic(adjective)\n",
    "'pleasantly new or different',#refreshing(adjective)\n",
    "'connected with or about love or a sexual relationship',#romantic(adjective)\n",
    "'full of life and energy',#vibrant(adjective)\n",
    "'morally good; having a good moral influence'#wholesome(adjective)\n",
    "]\n",
    "\n",
    "\n",
    "negative_word_meaning_sentences = [\n",
    "'that can be understood in more than one way; having different meanings', #ambiguous(adjective)\n",
    "'having strong feelings about something that you dislike very much or about an unfair situation',#angry(adjective)\n",
    "'making somebody feel slightly angry',#annoying(adjective)\n",
    "'extremely bad, especially from a moral point of view',#appalling(adjective)\n",
    "'very bad or unpleasant',#awful(adjective)\n",
    "'cruel and violent and not as expected from people who are educated and respect each other',#barbaric(adjective)\n",
    "'very strange or unusual',#bizarre(adjective)\n",
    "'(of behaviour or language) showing a lack of respect for God or religion',#blasphemous(adjective)\n",
    "'stupid; not able to think or talk in an intelligent way',#brainless(adjective)\n",
    "'without any order; in a completely confused state',#chaotic(adjective)\n",
    "'containing or showing a lack of agreement between statements, facts, opinions or actions',#contradictory(adjective)\n",
    "'public discussion and argument about something that many people strongly disagree about, think is bad, or are shocked by',#controversy(noun)\n",
    "'to feel very embarrassed and uncomfortable about something',#cringe(verb)\n",
    "'having a desire to cause physical or mental pain and make somebody suffer',#cruel(adjective)\n",
    "'treating somebody as if they have no value, so that they lose their self-respect and the respect of other people',#degrading(adjective)\n",
    "'mentally ill, especially because of very unhappy or unpleasant experiences',#disturbed(adjective)\n",
    "'not successful', #failed(adjective)\n",
    "'not what somebody claims it is; appearing to be something it is not',#fake(adjective)\n",
    "'an unusual trick or unnecessary device that is intended to attract attention or to persuade people to buy something',#gimmick(noun)\n",
    "'very unkind or unpleasant',#hateful(adjective)\n",
    "]\n",
    "\n",
    "new_negative_word_meaning_sentences = [\n",
    "'tending to show favour towards or against one group of people or one opinion for personal reasons; making unfair judgements',#biased(adjective)\n",
    "'very bad or unpleasant; used to describe something that you do not like',#horrible(adjective)\n",
    "'feeling tired and impatient because you have lost interest in somebody/something or because you have nothing to do',#bored(adjective)\n",
    "'upset because something you hoped for has not happened or been as good, successful, etc. as you expected',#disappointed(adjective)\n",
    "'to make somebody feel annoyed or impatient because they cannot do or achieve what they want',#frustrate(verb),\n",
    "'showing or feeling opposition or dislike; unfriendly',\n",
    "'inspiring scornful pity', \n",
    "'speak unfavorably about', \n",
    "'socially incorrect in behavior', \n",
    "'offensive or disagreeable; causing discomfort or unhappiness',\n",
    "'very ugly or unpleasant',#hideous(adjective)\n",
    "'not enough; not good enough',#inadequate(adjective)\n",
    "'not suitable or appropriate in a particular situation',#inappropriate(adjective)\n",
    "'unable to express yourself clearly, often because of emotion',#incoherent(adjective)\n",
    "'a mistake in the way a law, contract, etc. has been written that enables people to legally avoid doing something that the law, contract, etc. had intended them to do',#loophole(noun)\n",
    "'without any purpose or reason and therefore not worth doing or having',#meaningless(adjective)\n",
    "'difficult to understand',#obscure(adjective)\n",
    "'rude in a way that causes somebody to feel upset or annoyed because it shows a lack of respect',#offensive(adjective)\n",
    "'making you feel sad',#pathetic\n",
    "'very strange or unusual and difficult to explain',#weird(adjective)\n",
    "]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 30,
     "status": "ok",
     "timestamp": 1695323767308,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "Y_8HhXvskRFY",
    "outputId": "a8dbea0e-bd42-4116-8e55-4b9c17f6f627",
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "20\n",
      "20\n"
     ]
    }
   ],
   "source": [
    "print(len(new_positive_word_meaning_sentences))\n",
    "print(len(new_negative_word_meaning_sentences))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {
    "executionInfo": {
     "elapsed": 3468,
     "status": "ok",
     "timestamp": 1695323770756,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "LNSmtyIyoOUv",
    "tags": []
   },
   "outputs": [],
   "source": [
    "p_emb = torch.tensor([[1.]*config.hidden_size]).to(device)\n",
    "n_emb = torch.tensor([[1.]*config.hidden_size]).to(device)\n",
    "\n",
    "\n",
    "for i in positive_word_meaning_sentences:\n",
    "  p_emb = torch.cat((p_emb,get_cls(i).unsqueeze(dim=0)), dim=0)\n",
    "\n",
    "for i in negative_word_meaning_sentences:\n",
    "  n_emb = torch.cat((n_emb,get_cls(i).unsqueeze(dim=0)), dim=0)\n",
    "\n",
    "\n",
    "\n",
    "p_emb = p_emb[1:]\n",
    "n_emb = n_emb[1:]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 27,
     "status": "ok",
     "timestamp": 1695323770759,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "V_qXeXCGgjtq",
    "outputId": "b32eec10-206d-4ac3-ed65-f7bef84586d2",
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "torch.Size([20, 768])\n",
      "torch.Size([20, 768])\n"
     ]
    }
   ],
   "source": [
    "print(p_emb.shape)\n",
    "print(n_emb.shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {
    "executionInfo": {
     "elapsed": 24,
     "status": "ok",
     "timestamp": 1695323770760,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "OtZt1p7ys7XD",
    "tags": []
   },
   "outputs": [],
   "source": [
    "class IMDBDataset(Dataset):\n",
    "\n",
    "  def __init__(self, reviews, sentiments, tokenizer, max_len):\n",
    "    self.reviews = reviews\n",
    "    self.sentiments = sentiments\n",
    "    self.tokenizer = tokenizer\n",
    "    self.max_len = max_len\n",
    "\n",
    "  def __len__(self):\n",
    "    return len(self.reviews)\n",
    "\n",
    "  def __getitem__(self, item):\n",
    "    review = str(self.reviews[item])\n",
    "    sentiment = self.sentiments[item]\n",
    "\n",
    "\n",
    "    encoding = self.tokenizer.encode_plus(\n",
    "      review,\n",
    "      add_special_tokens=True,\n",
    "      max_length=self.max_len,\n",
    "      return_token_type_ids=False,\n",
    "      padding='max_length',\n",
    "      truncation = True,\n",
    "      return_attention_mask=True,\n",
    "      return_tensors='pt',\n",
    "    )\n",
    "\n",
    "    return {\n",
    "      'review': review,\n",
    "      'input_ids': encoding['input_ids'].flatten(),\n",
    "      'attention_mask': encoding['attention_mask'].flatten(),\n",
    "      'sentiments': torch.tensor(sentiment, dtype=torch.long),\n",
    "\n",
    "    }"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {
    "executionInfo": {
     "elapsed": 6,
     "status": "ok",
     "timestamp": 1695325189268,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "UuOujQajtL5f",
    "tags": []
   },
   "outputs": [],
   "source": [
    "def create_data_loader(df, tokenizer, max_len, batch_size):\n",
    "  ds = IMDBDataset(\n",
    "    reviews=df.review.to_numpy(),\n",
    "    sentiments=df['sentiment'].to_numpy(),\n",
    "    tokenizer=tokenizer,\n",
    "    max_len=max_len\n",
    "  )\n",
    "\n",
    "  return DataLoader(\n",
    "    ds,\n",
    "    batch_size=batch_size,\n",
    "    num_workers=8\n",
    "  )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {
    "executionInfo": {
     "elapsed": 24,
     "status": "ok",
     "timestamp": 1695323770761,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "3zzA4eBytOqj",
    "tags": []
   },
   "outputs": [],
   "source": [
    "BATCH_SIZE = 32\n",
    "\n",
    "train_data_loader = create_data_loader(df_train, tokenizer, MAX_LEN, BATCH_SIZE)\n",
    "val_data_loader = create_data_loader(df_val, tokenizer, MAX_LEN, BATCH_SIZE)\n",
    "test_data_loader = create_data_loader(df_test, tokenizer, MAX_LEN, BATCH_SIZE)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 6,
     "status": "ok",
     "timestamp": 1695325187523,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "AoUfRPy0tQgk",
    "outputId": "4cc31257-8fe1-4cce-883e-8aff4ebf2185",
    "tags": []
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "dict_keys(['review', 'input_ids', 'attention_mask', 'sentiments'])"
      ]
     },
     "execution_count": 18,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "data = next(iter(train_data_loader))\n",
    "data.keys()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "metadata": {
    "executionInfo": {
     "elapsed": 3,
     "status": "ok",
     "timestamp": 1695325688712,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "Y3Nil-yatUqF",
    "tags": []
   },
   "outputs": [],
   "source": [
    "class Classifier(nn.Module):\n",
    "  def __init__(self):\n",
    "    super(Classifier, self).__init__()\n",
    "    self.bert = BertModel.from_pretrained(PRE_TRAINED_MODEL_NAME,config=config)\n",
    "    self.FC_p = nn.Linear(config.hidden_size,config.hidden_size, bias=False)\n",
    "    self.FC_n = nn.Linear(config.hidden_size,config.hidden_size, bias=False)\n",
    "    \n",
    "\n",
    "  def CosineNorm(self, c, b, n_words):\n",
    "    cos = torch.nn.CosineSimilarity(dim=-1, eps=1e-8)\n",
    "    simi = torch.tensor([[1.]*n_words]).to(device)\n",
    "    for i in c:\n",
    "      temp = cos(i,b).to(device)\n",
    "      temp = torch.unsqueeze(temp,0)\n",
    "      simi = torch.cat((simi,temp), dim=0)\n",
    "\n",
    "    return simi[1:]\n",
    "\n",
    "  def binary_output(self,positive, negative):\n",
    "    positive = torch.max(positive,1).values\n",
    "    negative = torch.max(negative,1).values\n",
    "\n",
    "    p_temp = torch.unsqueeze(positive,0)\n",
    "    n_temp = torch.unsqueeze(negative,0)\n",
    "\n",
    "\n",
    "    res = torch.cat((p_temp,n_temp), dim=0)\n",
    "\n",
    "    return torch.t(res)\n",
    "\n",
    "  def forward(self, input_ids, attention_mask, return_scores=False):\n",
    "    with torch.no_grad():\n",
    "      pooled_output = self.bert(\n",
    "        input_ids=input_ids,\n",
    "        attention_mask=attention_mask,\n",
    "        return_dict = False\n",
    "      )\n",
    "    pooled_output = torch.mean(pooled_output[0], dim=1) # Taking Averge pooled last layer embedding\n",
    "\n",
    "    x_sent_p = self.FC_p(pooled_output)\n",
    "    x_sent_n = self.FC_n(pooled_output)\n",
    "    \n",
    "\n",
    "    Ept = self.FC_p(p_emb)\n",
    "    Ent = self.FC_n(n_emb)\n",
    "\n",
    "    positive = F.relu(self.CosineNorm(x_sent_p, Ept, p_emb.size()[0]))\n",
    "    negative = F.relu(self.CosineNorm(x_sent_n, Ent, n_emb.size()[0]))\n",
    "\n",
    "    binary_out = self.binary_output(positive,negative)\n",
    "    if(return_scores==True):\n",
    "      return (positive,negative) , binary_out\n",
    "    return binary_out"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "metadata": {
    "id": "HWZ37gsztWzL",
    "tags": []
   },
   "outputs": [],
   "source": [
    "model = Classifier()\n",
    "model = model.to(device)\n",
    "clear_output()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "metadata": {
    "executionInfo": {
     "elapsed": 22,
     "status": "ok",
     "timestamp": 1695325221207,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "HWy57v2CxxCM",
    "tags": []
   },
   "outputs": [],
   "source": [
    "for name, param in model.named_parameters():\n",
    "    if name.startswith('bert'):\n",
    "        param.requires_grad = False"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "metadata": {
    "executionInfo": {
     "elapsed": 20,
     "status": "ok",
     "timestamp": 1695325221207,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "e5iLu13CYlux",
    "tags": []
   },
   "outputs": [],
   "source": [
    "#for name, param in model.named_parameters():\n",
    "#    print(name, param.requires_grad)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 21,
     "status": "ok",
     "timestamp": 1695325221208,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "nZpqz6yDtYZ4",
    "outputId": "c0ba4f97-cad5-46b8-e1ff-e818ff5b4035",
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "torch.Size([32, 200])\n",
      "torch.Size([32, 200])\n"
     ]
    }
   ],
   "source": [
    "input_ids = data['input_ids'].to(device)\n",
    "attention_mask = data['attention_mask'].to(device)\n",
    "sentiments = data['sentiments'].to(device)\n",
    "\n",
    "print(input_ids.shape) # batch size x seq length\n",
    "print(attention_mask.shape) # batch size x seq length"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "metadata": {
    "executionInfo": {
     "elapsed": 12,
     "status": "ok",
     "timestamp": 1695325221208,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "NwvA-zp7vqc1",
    "tags": []
   },
   "outputs": [],
   "source": [
    "#del test\n",
    "torch.cuda.empty_cache()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "metadata": {
    "executionInfo": {
     "elapsed": 603,
     "status": "ok",
     "timestamp": 1695325221800,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "-9Z37OXOtb0q",
    "tags": []
   },
   "outputs": [],
   "source": [
    "(positive,negative),outs = model(input_ids, attention_mask,return_scores=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 15,
     "status": "ok",
     "timestamp": 1695325221800,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "NHFM3QUhg3n0",
    "outputId": "602bd955-db78-44a6-931e-e5e901bb1856",
    "tags": []
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([[0.7769, 0.7208],\n",
       "        [0.5462, 0.5590],\n",
       "        [0.4777, 0.4752],\n",
       "        [0.5281, 0.5172],\n",
       "        [0.5100, 0.4996],\n",
       "        [0.5784, 0.5782],\n",
       "        [0.6058, 0.6171],\n",
       "        [0.5579, 0.5042],\n",
       "        [0.5694, 0.5552],\n",
       "        [0.5206, 0.4931],\n",
       "        [0.5224, 0.5390],\n",
       "        [0.5107, 0.4937],\n",
       "        [0.5115, 0.5386],\n",
       "        [0.6108, 0.6049],\n",
       "        [0.5627, 0.5603],\n",
       "        [0.5164, 0.5252],\n",
       "        [0.5014, 0.4998],\n",
       "        [0.5913, 0.5548],\n",
       "        [0.6503, 0.6239],\n",
       "        [0.5006, 0.5177],\n",
       "        [0.4659, 0.4773],\n",
       "        [0.4993, 0.5260],\n",
       "        [0.3840, 0.3714],\n",
       "        [0.5470, 0.5509],\n",
       "        [0.7468, 0.6996],\n",
       "        [0.5088, 0.5165],\n",
       "        [0.5459, 0.5275],\n",
       "        [0.6438, 0.6624],\n",
       "        [0.5166, 0.5093],\n",
       "        [0.5595, 0.5465],\n",
       "        [0.7085, 0.6935],\n",
       "        [0.5595, 0.5563]], device='cuda:0', grad_fn=<TBackward0>)"
      ]
     },
     "execution_count": 33,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "outs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 1022,
     "status": "ok",
     "timestamp": 1695325845248,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "cjMVWA5a_6lf",
    "outputId": "bae8764e-0ce8-4118-e676-267b2c3ac06b",
    "tags": []
   },
   "outputs": [],
   "source": [
    "EPOCHS = 8\n",
    "\n",
    "optimizer = AdamW(model.parameters(), lr=0.001)\n",
    "total_steps = len(train_data_loader) * EPOCHS\n",
    "\n",
    "scheduler = get_linear_schedule_with_warmup(\n",
    "  optimizer,\n",
    "  num_warmup_steps=math.floor((1./5)*total_steps),\n",
    "  num_training_steps=total_steps\n",
    ")\n",
    "\n",
    "loss_fn = nn.CrossEntropyLoss().to(device)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "metadata": {
    "executionInfo": {
     "elapsed": 8,
     "status": "ok",
     "timestamp": 1695325845969,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "cLFDb4pzbx9W",
    "tags": []
   },
   "outputs": [],
   "source": [
    "def train_epoch(\n",
    "  model,\n",
    "  data_loader,\n",
    "  loss_fn,\n",
    "  optimizer,\n",
    "  device,\n",
    "  scheduler,\n",
    "  n_examples\n",
    "):\n",
    "  model = model.train()\n",
    "\n",
    "  losses = []\n",
    "  correct_predictions = 0\n",
    "\n",
    "  for d in data_loader:\n",
    "    input_ids = d[\"input_ids\"].to(device)\n",
    "    attention_mask = d[\"attention_mask\"].to(device)\n",
    "    sentiments = d[\"sentiments\"].to(device)\n",
    "\n",
    "    outputs = model(\n",
    "      input_ids=input_ids,\n",
    "      attention_mask=attention_mask\n",
    "    ).to(device)\n",
    "\n",
    "    _, preds = torch.max(outputs, dim=1)\n",
    "    loss = loss_fn(outputs, sentiments)\n",
    "\n",
    "    correct_predictions += torch.sum(preds == sentiments)\n",
    "    losses.append(loss.item())\n",
    "\n",
    "\n",
    "    loss.backward()\n",
    "    nn.utils.clip_grad_norm_(model.parameters(), max_norm=1.0)\n",
    "    optimizer.step()\n",
    "    scheduler.step()\n",
    "    optimizer.zero_grad()\n",
    "\n",
    "  return correct_predictions.double() / n_examples, np.mean(losses)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 36,
   "metadata": {
    "executionInfo": {
     "elapsed": 9,
     "status": "ok",
     "timestamp": 1695325845971,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "z4GAdIawtUue",
    "tags": []
   },
   "outputs": [],
   "source": [
    "def eval_model(model, data_loader, loss_fn, device, n_examples, on_new=False):\n",
    "  model = model.eval()\n",
    "\n",
    "  losses = []\n",
    "  f1s = []\n",
    "\n",
    "  correct_predictions = 0\n",
    "\n",
    "  with torch.no_grad():\n",
    "    for d in data_loader:\n",
    "      input_ids = d[\"input_ids\"].to(device)\n",
    "      attention_mask = d[\"attention_mask\"].to(device)\n",
    "      sentiments = d[\"sentiments\"].to(device)\n",
    "\n",
    "      outputs = model(\n",
    "        input_ids=input_ids,\n",
    "        attention_mask=attention_mask,\n",
    "      ).to(device)\n",
    "      _, preds = torch.max(outputs, dim=1)\n",
    "\n",
    "      loss = loss_fn(outputs, sentiments)\n",
    "\n",
    "      correct_predictions += torch.sum(preds == sentiments)\n",
    "      losses.append(loss.item())\n",
    "\n",
    "      f1s.append(f1_score(sentiments.cpu(), preds.cpu(), average='macro'))\n",
    "\n",
    "  return correct_predictions.double() / n_examples, np.mean(losses), np.mean(f1s)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 37,
   "metadata": {
    "executionInfo": {
     "elapsed": 8,
     "status": "ok",
     "timestamp": 1695325846655,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "9tWEJ4saOU6Q",
    "tags": []
   },
   "outputs": [],
   "source": [
    "def batch_calc_score_sum(positive, negative, sentiments):\n",
    "  p_filter = sentiments ^ torch.ones(sentiments.size()[0], dtype=torch.int8).to(device) # Positive ( sentiment xor [1,1,...])\n",
    "  n_filter = sentiments # Negative\n",
    "\n",
    "  pos_score_pos_sent = positive[p_filter.nonzero(), :]\n",
    "  pos_score_neg_sent = positive[n_filter.nonzero(), :]\n",
    "\n",
    "  neg_score_pos_sent = negative[p_filter.nonzero(), :]\n",
    "  neg_score_neg_sent = negative[n_filter.nonzero(), :]\n",
    "\n",
    "  pp_count = pos_score_pos_sent.size()[0]\n",
    "  pn_count = pos_score_neg_sent.size()[0]\n",
    "  pp_sum = torch.sum(pos_score_pos_sent,dim=0)\n",
    "  pn_sum = torch.sum(pos_score_neg_sent,dim=0)\n",
    "\n",
    "\n",
    "#  np_count = neg_score_pos_sent.size()[0] # same as pn_count\n",
    "#  nn_count = neg_score_neg_sent.size()[0] # same as pn_count\n",
    "  np_sum = torch.sum(neg_score_pos_sent,dim=0)\n",
    "  nn_sum = torch.sum(neg_score_neg_sent,dim=0)\n",
    "\n",
    "  return pp_sum,pn_sum,np_sum,nn_sum, pp_count, pn_count\n",
    "\n",
    "def eval_scores_average(model, data_loader, device):\n",
    "  model = model.eval()\n",
    "\n",
    "\n",
    "  pp_sum_agg = torch.zeros(1,p_emb.size()[0]).to(device)\n",
    "  pn_sum_agg = torch.zeros(1,p_emb.size()[0]).to(device)\n",
    "\n",
    "  np_sum_agg = torch.zeros(1,n_emb.size()[0]).to(device)\n",
    "  nn_sum_agg = torch.zeros(1,n_emb.size()[0]).to(device)\n",
    "\n",
    "  pp_count_agg, pn_count_agg = 0,0\n",
    "\n",
    "  with torch.no_grad():\n",
    "    for d in data_loader:\n",
    "      input_ids = d[\"input_ids\"].to(device)\n",
    "      attention_mask = d[\"attention_mask\"].to(device)\n",
    "      sentiments = d[\"sentiments\"].to(device)\n",
    "\n",
    "      (positive,negative),_= model(\n",
    "        input_ids=input_ids,\n",
    "        attention_mask=attention_mask,\n",
    "        return_scores=True\n",
    "      )\n",
    "      positive.to(device)\n",
    "      negative.to(device)\n",
    "\n",
    "\n",
    "      pp_sum,pn_sum,np_sum,nn_sum, pp_count, pn_count = batch_calc_score_sum(positive,negative,sentiments)\n",
    "\n",
    "      pp_sum_agg += pp_sum\n",
    "      pn_sum_agg += pn_sum\n",
    "\n",
    "      np_sum_agg += np_sum\n",
    "      nn_sum_agg += nn_sum\n",
    "\n",
    "      pp_count_agg += pp_count\n",
    "      pn_count_agg += pn_count\n",
    "\n",
    "\n",
    "\n",
    "  return pp_sum_agg/pp_count_agg, pn_sum_agg/pn_count_agg, np_sum_agg/pp_count_agg, nn_sum_agg/pn_count_agg"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 38,
   "metadata": {
    "executionInfo": {
     "elapsed": 8,
     "status": "ok",
     "timestamp": 1695325846656,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "IxncgTh4awi7",
    "tags": []
   },
   "outputs": [],
   "source": [
    "def box_plot(pp,pn,np,nn):\n",
    "  pp = pp.tolist()[0]\n",
    "  pn = pn.tolist()[0]\n",
    "  np = np.tolist()[0]\n",
    "  nn = nn.tolist()[0]\n",
    "  # Create a box plot with beeswarm using seaborn\n",
    "  sns.set(style=\"whitegrid\")  # Set the style of the plot\n",
    "  plt.figure(figsize=(8, 6))  # Set the figure size\n",
    "  ax = sns.boxplot(data=[pp,pn,np,nn], orient=\"v\", palette=\"Set2\")  # Create the box plot\n",
    "  ax = sns.swarmplot(data=[pp,pn,np,nn], orient=\"v\", color=\"0.2\")  # Add the swarm plot\n",
    "\n",
    "  # Set labels for the axes\n",
    "  ax.set_xlabel('Categories')\n",
    "  ax.set_xticklabels(['pp','pn','np','nn'])\n",
    "  ax.set_ylabel('Scores(µ)')\n",
    "\n",
    "  # Set a title for the plot\n",
    "  #plt.title()\n",
    "\n",
    "  # Show the plot\n",
    "  plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 39,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 1000
    },
    "executionInfo": {
     "elapsed": 3203929,
     "status": "ok",
     "timestamp": 1695329051836,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "IqdIHJsrANr0",
    "outputId": "ee79ec9a-a033-47e0-bc8b-a7b71698465c",
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 1/8\n",
      "----------\n",
      "Train loss 0.5170583036422729 accuracy 0.8021\n",
      "Val   loss 0.4704096218582931 accuracy 0.8468\n",
      "\n",
      "Epoch 2/8\n",
      "----------\n",
      "Train loss 0.47604156510829926 accuracy 0.8348500000000001\n",
      "Val   loss 0.4782455810316049 accuracy 0.8294\n",
      "\n",
      "Epoch 3/8\n",
      "----------\n",
      "Train loss 0.4696626420021057 accuracy 0.8407250000000001\n",
      "Val   loss 0.4554181839250455 accuracy 0.8526\n",
      "\n",
      "Epoch 4/8\n",
      "----------\n",
      "Train loss 0.46550971233844757 accuracy 0.843875\n",
      "Val   loss 0.45429506935891073 accuracy 0.8532000000000001\n",
      "\n",
      "Epoch 5/8\n",
      "----------\n",
      "Train loss 0.4621520558357239 accuracy 0.846925\n",
      "Val   loss 0.4526799260431035 accuracy 0.8574\n",
      "\n",
      "Epoch 6/8\n",
      "----------\n",
      "Train loss 0.4597090370416641 accuracy 0.848375\n",
      "Val   loss 0.44982370221690765 accuracy 0.8566\n",
      "\n",
      "Epoch 7/8\n",
      "----------\n",
      "Train loss 0.45509178647994997 accuracy 0.8538250000000001\n",
      "Val   loss 0.4503178740762601 accuracy 0.8566\n",
      "\n",
      "Epoch 8/8\n",
      "----------\n",
      "Train loss 0.45196277287006376 accuracy 0.85865\n",
      "Val   loss 0.447719159969099 accuracy 0.8570000000000001\n",
      "\n",
      "CPU times: user 1h 15min 45s, sys: 8.94 s, total: 1h 15min 54s\n",
      "Wall time: 1h 16min 3s\n"
     ]
    }
   ],
   "source": [
    "%%time\n",
    "\n",
    "train_a = []\n",
    "train_l = []\n",
    "val_a = []\n",
    "val_l = []\n",
    "best_accuracy = 0\n",
    "\n",
    "for epoch in range(EPOCHS):\n",
    "\n",
    "  print(f'Epoch {epoch + 1}/{EPOCHS}')\n",
    "  print('-' * 10)\n",
    "\n",
    "  train_acc, train_loss = train_epoch(\n",
    "    model,\n",
    "    train_data_loader,\n",
    "    loss_fn,\n",
    "    optimizer,\n",
    "    device,\n",
    "    scheduler,\n",
    "    len(df_train)\n",
    "  )\n",
    "\n",
    "  print(f'Train loss {train_loss} accuracy {train_acc}')\n",
    "\n",
    "  val_acc, val_loss, val_f1 = eval_model(\n",
    "    model,\n",
    "    val_data_loader,\n",
    "    loss_fn,\n",
    "    device,\n",
    "    len(df_val)\n",
    "  )\n",
    "\n",
    "  print(f'Val   loss {val_loss} accuracy {val_acc}')\n",
    "  print()\n",
    "\n",
    "  train_a.append(train_acc)\n",
    "  train_l.append(train_loss)\n",
    "  val_a.append(val_acc)\n",
    "  val_l.append(val_loss)\n",
    "\n",
    "  if val_acc > best_accuracy:\n",
    "    torch.save(model.state_dict(), 'bert_best_model_state.bin')\n",
    "    best_accuracy = val_acc"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 40,
   "metadata": {
    "executionInfo": {
     "elapsed": 51,
     "status": "ok",
     "timestamp": 1695329137842,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "FowMSU5U7SDQ",
    "tags": []
   },
   "outputs": [],
   "source": [
    "train_a = [i.item() for i in train_a]\n",
    "train_l = [i.item() for i in train_l]\n",
    "val_a = [i.item() for i in val_a]\n",
    "val_l = [i.item() for i in val_l]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 41,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 480
    },
    "executionInfo": {
     "elapsed": 2888,
     "status": "ok",
     "timestamp": 1695329143450,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "aUQbxyTEAPhM",
    "outputId": "b103ca88-1886-4f16-ea21-5b966a08fe7c",
    "tags": []
   },
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAjcAAAHFCAYAAAAOmtghAAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAAPYQAAD2EBqD+naQAAADh0RVh0U29mdHdhcmUAbWF0cGxvdGxpYiB2ZXJzaW9uMy4xLjIsIGh0dHA6Ly9tYXRwbG90bGliLm9yZy8li6FKAAAgAElEQVR4nOzdeXwU9f3H8ffuJtncEQKEcEcuCQgKQQw0oiggKIpVDi+IqG1EC4hKpf6qgGjUCiIg0SoBFatoAYstilEUqFAVBDwIRzEYjtAQjlxAjt35/ZFkzSYhByTZZHg9H499JPud78x8ZkXmzfc7O2MxDMMQAACASVg9XQAAAEBtItwAAABTIdwAAABTIdwAAABTIdwAAABTIdwAAABTIdwAAABTIdwAAABTIdwAAABTIdwAJmKxWKr1+vLLL2tlf2fOnJHFYtFzzz13TutfeeWVuv7662ullppq2bKlbrvttir7ffLJJ7JYLPrPf/5To+3Pnz9fy5YtO9fyAJwHL08XAKD2bN682e39008/rS+++ELr1q1za4+MjKyV/dntdm3evFnt2rU7p/UXL14sm81WK7XUlejoaG3evFk9evSo0Xrz589Xp06ddNddd9VRZQDOhnADmMiVV17p9r558+ayWq3l2s8mPz9fNput2oHDYrFUe9sV6d69+zmvW19CQkLO6xhrk8PhkMPhkI+Pj6dLARo0pqWAC1TJdMvy5cs1adIkhYeHy9fXVwcOHFBaWpri4uLUrVs3BQQEKCwsTNddd125kaGKpqVeffVVWSwWffXVV7r//vsVGhqqZs2aadSoUfrf//7ntn7Zaaldu3bJYrFowYIFev7559W+fXsFBgZqwIAB2rp1a7ljWLRokTp16iS73a5LL71UH3zwgcaOHatLLrmk2p/DRx99pMsuu0x+fn6KjIwsN5VU0bTUnj17NGrUKIWHh8tut6tly5YaPHiwfvrpJ0lFU1779u3T2rVrXVOBpWtKSUnR7bffrubNm8tutysyMlLz589X6ecYl3wW8+bN04wZM9ShQwf5+PhozZo1CgwM1OTJk8sdy+7du2W1WrVgwYJqHz9gRozcABe4Rx55RFdddZXeeOMNOZ1ONWnSRKmpqfL29tbMmTMVFham7OxsffDBB4qJidHGjRsVHR1d5XbHjx+vm266Se+++65SUlI0bdo03XPPPVqzZk2V686dO1eXXnqpFixYIIfDoSeeeELDhg1TSkqKAgICJBVN+0yePFljx47V/Pnzdfz4cU2fPl0FBQXy8/Or1rF/++232r17tx5//HE1a9ZMCQkJuvvuu9WlSxddccUVFa5jGIauv/562e12vfjii2rbtq0yMjK0ceNGnTx5UpK0Zs0a3XzzzWrTpo1eeuklSXLVlJaWpujoaFksFsXHx6tNmzb68MMPNXnyZO3fv19z585129+LL76oyMhIzZ07V4GBgYqMjNS4ceP05ptv6tlnn3V9HpL0yiuvKCAgQOPHj6/W8QOmZQAwrfHjxxsBAQEVLvv4448NScaQIUOq3E5hYaFRUFBgDBgwwLj99ttd7adPnzYkGfHx8a62hIQEQ5IxdepUt23MmjXLkGQcP37c1davXz9j6NChrvfJycmGJCMqKspwOp2u9g0bNhiSjFWrVhmGYRj5+flGaGioMXDgQLd9/Pe//zVsNpvRtWvXKo8pLCzMCAgIMA4fPuxqy8nJMYKCgozJkye72ko+p82bNxuGYRgHDx40JBmvvvpqpdvv2LGj27GVmDJlimGxWIzt27e7td9zzz2G1Wo1UlJS3D6Lbt26GYWFhW59k5OTDYvFYiQkJLjasrOzjeDgYOOBBx6o8tgBs2NaCrjA3XrrreXaDMPQggULdPnll8vX11deXl7y9vbWV199peTk5Gpt96abbnJ737NnT0lSampqleveeOONslgs5db95ZdfJEk//vijjh07ptGjR7ut17FjR/Xt27da9UlS3759FR4e7nofEBCgjh07uvZTkZYtW6pdu3Z69tln9fLLL2vHjh1yOp3V3ue6det0+eWXq1evXm7tsbGxcjqd5b7JNnLkyHLXQF1yySUaPHiwXnnlFVfbW2+9paysLD344IPVrgUwK8INcIErfXIvER8fr0mTJikmJkYrV67U119/rW+//VaDBg3S6dOnq7Xd0NBQt/d2u12SqrV+VeseO3ZMkhQWFlZu3Yraqrufkn1VVqPNZtMXX3yha665Rs8884wuu+wyhYWFaerUqcrNza1yn8eOHavwM2/VqpVreWkV9ZWkyZMn68cff9SGDRskFU1JXXPNNY3iIm2grnHNDXCBKz1CUmLZsmW6/vrrNX/+fLf2zMzM+iqrUiWhpOwFypJ05MiROt//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HTpUI0aMcHuOV0Vmzpyp0aNH6/vvv9fw4cN155136vjx42ftn5+fr6efflo7duzQhx9+qJSUFMXGxrqWHzp0SFdddZV8fX21bt06bd26VRMmTFBhYaEkKSEhQQ8++KB+97vf6YcfftDq1avVqVOnan0mpZ3L8Y8bN07vvfee5s+fr+TkZL366qsKDAyUxWLRhAkTtGTJErd9JCYmKiYmRh07dqxxfQCA2sX4fBVOFzgU+eRaj+x756yh8vep+j9RSEiIfHx85O/vr5YtW5ZbPmvWLA0ePNj1PjQ0VL169XK9nz17tlatWqXVq1e7RkYqEhsbq9tvv12S9Oyzz2rBggX65ptvdP3111fYf8KECa7fL774Ys2fP19XXHGFcnJyFBgYqFdeeUUhISF67733XDeT69Kli1tdjzzyiCZPnuxq69u3b1UfRzk1Pf49e/bo/fffV1JSkq677jpX/SXuuecePfnkk/rmm290xRVXqKCgQMuWLdNf/vKXGtcGAKh9jNxcAKKiotze5+bmatq0aYqMjNRFF12kwMBA7dq1q8qRm549e7p+DwgIUFBQULlpptK2bdumm2++We3bt1dQUJCuvvpqSXLtZ/v27YqJianwLrnp6ek6fPiwrr322uoe5lnV9Pi3b98um82mgQMHVri98PBw3XDDDUpMTJQk/fOf/9SZM2c0atSo864VAHD+GLmpgp+3TTtnDfXYvmtD2W89PfbYY1q7dq1efPFFderUSX5+frrtttuUn59f6XbKhhCLxSKn01lh39zcXA0ZMkRDhgzRsmXL1Lx5c6Wmpmro0KGu/fj5+Z11X5Utk+R6qnXpqbuzPcW9psdf1b4l6b777tPdd9+tl156SUuWLNGYMWO4szUANBCEmypYLJZqTQ15mo+PT7Vvp79x40bFxsbqlltukSTl5ORo//79tVrPrl27lJGRoeeee851/c+WLVvc+vTs2VNvvvmmCgoKygWnoKAgdejQQZ9//rmuueaactsv+TZXWlqaLr/8cklyu7i4MlUd/6WXXiqn06n169e7pqXKGj58uAICApSQkKCPP/5YGzZsqNa+AQB1j2kpk+jQoYO+/vpr7d+/XxkZGWcdUZGkTp06aeXKldq+fbt27NihO+64o9L+56Jdu3by8fHRggUL9PPPP2v16tV6+umn3fo89NBDysrK0tixY7Vlyxbt3btXb7/9tnbv3i1JmjFjhubMmaP58+dr7969+u6777RgwQJJRaMrV155pZ577jnt3LlTGzZs0P/93/9Vq7aqjr9Dhw4aP368JkyY4LoQ+ssvv9T777/v6mOz2RQbG6vp06erU6dOio6OPt+PDABQSwg3JvHoo4/KZrMpMjLSNQV0Ni+99JKaNGmi/v37a8SIERo6dKh69+5dq/U0b95cS5cu1QcffKDIyEg999xz5b4+HRoaqnXr1iknJ0cDBw5Unz599Prrr7tGccaPH6958+Zp0aJF6t69u2688Ubt3bvXtX5iYqIKCgoUFRWlyZMna/bs2dWqrTrHn5CQoNtuu00TJ07UJZdcovvvv1+5ublufe69917l5+e7XTgNAPA8i1Hd7xubRFZWlkJCQpSZmang4GC3ZWfOnFFKSooiIiLk6+vroQrRWHz11Ve6+uqrdfDgQYWFhXm6nGrjzzmAxqiy83dZDf9iEqCBycvL04EDB/TnP/9Zo0ePblTBBgAuBExLATX07rvvqmvXrsrMzNQLL7zg6XIAAGUQboAaio2NlcPh0NatW9W6dWtPlwMAKINwAwAATIVwAwAATIVwAwAATIVwAwAATIVwAwAATIVwAwAATIVwA5cOHTpo3rx5rvcWi0UffvjhWfvv379fFoul2g+srOvtAAAgcYdiVCItLU1NmjSp1W3Gxsbq5MmTbqGpbdu2SktLU7NmzWp1XwCACxPhBmfVsmXLetmPzWart301NAUFBa4HhQIAagfTUibw2muvqXXr1nI6nW7tN910k8aPHy9J2rdvn26++WaFhYUpMDBQffv21WeffVbpdstOS33zzTe6/PLL5evrq6ioKG3bts2tv8Ph0L333quIiAj5+fmpa9euevnll13LZ8yYoTfffFP/+Mc/ZLFYZLFY9OWXX1Y4LbV+/XpdccUVstvtCg8P1+OPP67CwkLX8quvvlqTJk3StGnT1LRpU7Vs2VIzZsyo9Hi+/fZbDR48WM2aNVNISIgGDhyo7777zq3PyZMn9bvf/U5hYWHy9fVVjx499M9//tO1/KuvvtLAgQPl7++vJk2aaOjQoTpx4oSk8tN6knTZZZe51WWxWPTqq6/q5ptvVkBAgGbPnl3l51YiMTFR3bt3d30mDz30kCRpwoQJuvHGG936FhYWqmXLlkpMTKz0MwEAM2LkpiqGIRWc8sy+vf0li6XKbqNGjdKkSZP0xRdf6Nprr5UknThxQmvXrtVHH30kScrJydHw4cM1e/Zs+fr66s0339SIESO0e/dutWvXrsp95Obm6sYbb9SgQYO0bNkypaSkaPLkyW59nE6n2rRpo/fff1/NmjXTpk2b9Lvf/U7h4eEaPXq0Hn30USUnJysrK0tLliyRJDVt2lSHDx92286hQ4c0fPhwxcbG6q233tKuXbt0//33y9fX1y0ovPnmm5o6daq+/vprbd68WbGxsRowYIAGDx5c4TFkZ2dr/Pjxmj9/viRpzpw5Gj58uPbu3augoCA5nU4NGzZM2dnZWrZsmTp27KidO3fKZrNJkrZv365rr71WEyZM0Pz58+Xl5aUvvvhCDoejys+vtKeeekrx8fF66aWXZLPZqvzcJCkhIUFTp07Vc889p2HDhikzM1NfffWVJOm+++7TVVddpbS0NIWHh0uS1qxZo5ycHNf6AHAhIdxUpeCU9Gwrz+z7T4cln4AquzVt2lTXX3+9/va3v7nCzQcffKCmTZu63vfq1Uu9evVyrTN79mytWrVKq1evdo0AVOadd96Rw+FQYmKi/P391b17dx08eFAPPPCAq4+3t7dmzpzpeh8REaFNmzbp/fff1+jRoxUYGCg/Pz/l5eVVOg21aNEitW3bVgsXLpTFYtEll1yiw4cP649//KOefPJJWa1FA449e/bUU089JUnq3LmzFi5cqM8///ys4WbQoEFu71977TU1adJE69ev14033qjPPvtM33zzjZKTk9WlSxdJ0sUXX+zq/8ILLygqKkqLFi1ytXXv3r3Kz66sO+64QxMmTHBrq+xzk4r+ez3yyCNugbJv376SpP79+6tr1656++23NW3aNEnSkiVLNGrUKAUGBta4PgBo7JiWMok777xTK1asUF5enqSiMDJ27FjXqENubq6mTZumyMhIXXTRRQoMDNSuXbuUmppare0nJyerV69e8vf3d7VFR0eX6/fqq68qKipKzZs3V2BgoF5//fVq76P0vqKjo2UpNWo1YMAA5eTk6ODBg662nj17uq0XHh6u9PT0s243PT1dcXFx6tKli0JCQhQSEqKcnBxXfdu3b1ebNm1cwaaskpGb8xUVFVWurbLPLT09XYcPH6503/fdd59rNCw9PV3/+te/ygUoALhQMHJTFW//ohEUT+27mkaMGCGn06l//etf6tu3rzZu3Ki5c+e6lj/22GNau3atXnzxRXXq1El+fn667bbblJ+fX63tG4ZRZZ/3339fDz/8sObMmaPo6GgFBQXpL3/5i77++utqH0fJvixlpuNK9l+6veyFuBaLpdx1R6XFxsbq6NGjmjdvntq3by+73a7o6GjXZ+Dn51dpXVUtt1qt5T6ngoKCcv0CAtxH46r63KrarySNGzdOjz/+uDZv3qzNmzerQ4cOiomJqXI9ADAjwk1VLJZqTQ15mp+fn37729/qnXfe0X//+1916dJFffr0cS3fuHGjYmNjdcstt0gqugZn//791d5+ZGSk3n77bZ0+fdp1sv3Pf/7j1mfjxo3q37+/Jk6c6Grbt2+fWx8fH58qr1GJjIzUihUr3ELOpk2bFBQUpNatW1e75rI2btyoRYsWafjw4ZKkAwcOKCMjw7W8Z8+eOnjwoPbs2VPh6E3Pnj31+eefu00hlda8eXOlpaW53mdlZSklJaVadVX2uQUFBalDhw76/PPPdc0111S4jdDQUI0cOVJLlizR5s2bdc8991S5XwAwK6alTOTOO+/Uv/71LyUmJuquu+5yW9apUyetXLlS27dv144dO3THHXdUOspR1h133CGr1ap7771XO3fu1Jo1a/Tiiy+W28eWLVu0du1a7dmzR3/+85/17bffuvXp0KGDvv/+e+3evVsZGRkVjmxMnDhRBw4c0B/+8Aft2rVL//jHP/TUU09p6tSpruttzkWnTp309ttvKzk5WV9//bXuvPNOt1GRgQMH6qqrrtKtt96qpKQkpaSk6OOPP9Ynn3wiSZo+fbq+/fZbTZw4Ud9//7127dqlhIQEV0AaNGiQ3n77bW3cuFE//vijxo8f75oWrKquqj63GTNmaM6cOZo/f7727t2r7777TgsWLHDrc9999+nNN99UcnKy61tyAHAhItyYyKBBg9S0aVPt3r1bd9xxh9uyl156SU2aNFH//v01YsQIDR06VL179672tgMDA/XRRx9p586duvzyy/XEE0/o+eefd+sTFxen3/72txozZoz69eunY8eOuY1GSNL999+vrl27uq4vKfnGT2mtW7fWmjVr9M0336hXr16Ki4vTvffeq//7v/+rwadRXmJiok6cOKHLL79cd999tyZNmqQWLVq49VmxYoX69u2r22+/XZGRkZo2bZprpKlLly769NNPtWPHDl1xxRWKjo7WP/7xD3l5FQ2ATp8+XVdddZVuvPFGDR8+XCNHjlTHjh2rrKs6n9v48eM1b948LVq0SN27d9eNN96ovXv3uvW57rrrFB4erqFDh6pVKw9dBA8ADYDFqM7FFCaSlZWlkJAQZWZmKjg42G3ZmTNnlJKSooiICPn6+nqoQuDcnDp1Sq1atVJiYqJ++9vfnrUff84BNEaVnb/L4poboJFzOp06cuSI5syZo5CQEN10002eLgkAPIpwAzRyqampioiIUJs2bbR06VLXNBkAXKj4WxBo5Dp06FCtr+oDwIWCC4oBAICpEG4qwL+CYWb8+QZgdkxLlVJyx9tTp05V666wQGNUckfm6tyDB2UYhuQokBz5xa/Sv5dqkyRZim4CarH8+rurzVq+TcXtJXfhLtdWjW1YrGXWrWwbZ9tu1Q/rRSNS8o8Z1z9qDDmdTuU7nMorcCi/0Kl8h0P5hQ4VFDqVX+BQvsOp/EJH8cupAkdxv0Lnr/2K2wpK2hwOFTicrv6Bdh/NGDPAY4dNuCnFZrPpoosucj2fyN/fv9xjAIDGzOl06ujRo/L39294Fx47HeVDQ2HeWcJE6fZSywvzqwge+e7rFeZV0LegzPbLLL8gVBa8ztZW0XoXqFJBQoYhw/W7ZBS3FXUrHTyK+5VpU6l1JUMWlQ8rv37SRW1WVT46a5XkW/yqK0fVRNL+OtxD5RrY326eV/K06soewAi4MQzJWVh0cnYWSGX/Yqnw75maTA1VZ3vV3K5hyFp4Su2OfSHL96ekUn/R/vr7r3+Ruv0F61p2lp8lfQxHDYJHSZDIl4zq3zG74bBIXnbJ5iPZvH/9Kbl/hoZT5T6zitpcy8q2Oc/SzynV6M9SdZX6c8Es5nmznOV3Mwu0ezZeEG7KsFgsCg8PV4sWLSp8NAAuQIYhnT4pZR0qemUeLP79sHTyoJT7PzWaM4DTIZ/T6bIahZ6upGpWL8lmLxUaioODVwVtNp+K+9p8JC+fOuprl6wNaGrPqCo0VRSQnCo3UlDjbZwt8BZvVoYKHIZryiK/zM8C1++Ga6qj5PcCh+GaCsl3FC8vKN6Ww1HUr2R915SI89cplOJ2T1xm5j6eYnH76WWzysdmlZfNKm8vm7xtFnnZbPKxWeXtZZV3cbuX1Sa7l0VeXjb5WK3y8ipq9ynub/cu6utTsh0vq3xstuJtFi/zLtquj5dNPl6W4vWLt1P8u6XsVKjrd5V5X/1lfh6OcYSbs7DZbDW7JsEwpB9XSIFhUlC4FBQm2YPqrkDUrsI86eQB6USKdGJ/+Vd+TuXre/tLTTpIF7Uv+u9e+n/+ckP5xe1Sxdc8lP5ZZR/V4nbONrVQtk8V27FYfz35VzeElB39sHpL5/EcsZowDENOQypwOOVwGip0GCp0OlXoNFToNOQoPjE7ik+qDqehAkeeHM4zKnSU9HMWr1f0MgxDDmfRdp2G4dqHs/hn6eVFy0r3VfGyqpcbrm2W6uuUHEbZ5aXqKF5eUU3OaguVIygAACAASURBVC53VlCTs9TyQqdTZwqcyit0KK+wNsPF+YdJHy+r7F5W2b1sRT+9S/3uZZXdu9TvXrbi5TXr72Mreu9T3OZT/LuPzSpvm4XLHeoB4aa2nDourbjXvc0nUApqKQW2LPpZ8ir7nhBU9wxDys2oOLic2F80ElPp6ItFCm5VFGAqegU0N+01Bk6n4TqBFziM4gDgVEHJid9ZcsIvdYIvCQFOQw6nUwUFhhx5RpkAURIKnCp05qrQmVO8vaJ1Sm+rsOw6pfdRXFNBmX1Wtr573Y1k1M0k6iNcnK2/j80qq9Wc/5/CHeGmthSekTrESNlpUvaRon/p5+dIx/5b9KqMT6D7iE9QePn3hKCqFZyRMg/8GliOlxmFKcitfH3vAKlpRMXhJaSt5F355XcOp6G8QocKCotOnK4RALcT+a8nXNdJufhkWzo8FDh+HTkofRKvcKSg1Mm75IRd6Ch/si9Z37Ut56/rue3TLQw45bxAz/1WS9H0gZfVUvSyWWWzWuRttchms8jbWvTe1cdW1M9W/LJaSl6S1VL0r/WS361WVbG81O8WFS8r+t1mLdO31HKb1b1v0XZVvKz8tkqWl9RrsbjXZSm1XkXLrVZLmRp+3ZevN+ECnsODM+tKXraU/T8p50hR2Mk+UhR8cv5X6v0RKT+7+tv0DjjLCFCZUOQ2LWIihiHlHq1k9OWwKht9MWRRfkC4Tge0Ua5/W2X7t1amvbVO2FvpmE9rnVSw8gqdyit06kyBw/WzZHj9TIFTZwodyiv1s6Q9r9BxQY0AlD3xe5ec+G0lJ/ziQGC1lvvd5tbfIpvVWhQYKgwK1uI+pbZdKmx4VbCOl9XqFj5K1nerpdQ6RYHFfR0vq4WTMNDA8ODMhsAeVPRq1qnyfnk5xYEn7dfA4xaIjhQtz8sqGnk4vq/oVRnvgApGgCoIRXUcggyjaITAFQxKBYa8UiGhdHgoPHNK3jkH5ZtzQAGnDirw1AEFnzmsJnmHFJp/WHbjTKX7zDV89YsRplSjhet1oPjnIaOZ8s94S8cqWvNI8av2lPwr3tvmfoIvOeG6n5TL/F6mzT0MFC1zBQOb+4m7on162cqf3EsHirIBwrtsECi9LU78ABo4wo2n2QOLXqEdK+/nCkGlR4DSikaHSo8IuULQz0Wvynj7/zryc9ZpsbOHoAKHU/uO5mjn4ayiV1qW9mfk6nSp0Y7yUxqGmitTbS3palfysqa73odbjldastOw6LBCdcBZPrykGi10XEEq/WVLm9Ui3+J5+WZeVvl6F13cVzJk7uttk2/xsLmvt3u762fpvsXbcv30tsq3eJ6/5Kd3cWjhokEA8AzCTWNR3RCUn1vJCFDJ7/+T8jKlglPVDkHOwDDleDfTMUtTHSwM1t7TgdqZ5a9DzouUblykdKOJsuUnySK78tXGctQVXEpCTPviEOOnvEp3d8bqr+M+rZXp21rZfq2V699WpwLaKC+onQoDW8vH10++3la18LKp7VkCScnv3jaeMAIAFxquublQ5Z8qFXZ+HQEyco4o/8RhFWamySv3f7I7qvgKdCmFNj8Z3gHyPpNReUeLVQpuIzVpX+bC3YiiC3r9mpjzmiEAwDnjmhtUzcdfhSEd9HNBc+083l4/Hc/UzrSi6aUTp369eaGv8tTCclItdEKXBOSqR/ApdfTNUSuvTIU6T8h+Jl2WnCPSmUx5OU5LjtNFK9qDz/616ZC2RTdLAwCgDhBuLhA5eYXalZblCjA707K060i28gvL3/LeZrWoU/NARbYKVmR4sLq3Cla38GA1CagkkBScLr7mJ1sKacPoCwDAYwg3JmMYhtKz81wB5qfDmdp5OEv7j52qsH+Aj80VYop+hqhzWKB8vWt4J1Bvv6IpJQAAPIxw04gVOpxKych1G43ZeThLx3IrfnJxy2BftyDTvVWw2jbx52u9AABTIdw0Erl5hdp1JNt9WiktS3kVTCtZLVLH5oHq3urX0Zhu4UEKDbR7oHIAAOoX4aYBSs86o59KhZjkw1lKOZZb4cPn/H1s6hZeelopWF1bBtV8WgkAAJMg3HiQw2lUOK2UkVPxfWBaBNld00mR4SGKbBWs9k2ZVgIAoDSPh5tFixbpL3/5i9LS0tS9e3fNmzdPMTExZ+3/zjvv6IUXXtDevXsVEhKi66+/Xi+++KJCQ0PrseqaO53v0K4jWfqpVIjZdSRLZwoqnla6uHmg22hMt/BgNQ9iWgkAgKp4NNwsX75cU6ZM0aJFizRgwAC99tprGjZsmHbu3Kl27dqV6//vf/9b48aN00svvaQRI0bo0KFDiouL03333adVq1Z54AgqdjQ7r8xoTKZSMnIrfLqyn7dNl4QHuY3GdA0Lkp8P00oAAJwLj96huF+/furdu7cSEhJcbd26ddPIkSMVHx9frv+LL76ohIQE7dv364MjFyxYoBdeeEEHDhyo1j7r6g7FqcdO6c//+FE707J0NLviaaXmQXa30ZjIVsHqEBogG9NKAABUqlHcoTg/P19bt27V448/7tY+ZMgQbdq0qcJ1+vfvryeeeEJr1qzRsGHDlJ6err///e+64YYbzrqfvLw85eX9GjaysrJq5wDKCPL10vo9RyUV3bsuollA8Q3wikZjuoUHqUWQb53sGwAA/Mpj4SYjI0MOh0NhYWFu7WFhYTpy5EiF6/Tv31/vvPOOxowZozNnzqiwsFA33XSTFixYcNb9xMfHa+bMmbVae0WaBPjoL7f1VMcWgbqkZZD8fTx+ORMAABckjz8y2VLmFv2GYZRrK7Fz505NmjRJTz75pLZu3apPPvlEKSkpiouLO+v2p0+frszMTNerutNX52JUVFv1bteEYAMAgAd57CzcrFkz2Wy2cqM06enp5UZzSsTHx2vAgAF67LHHJEk9e/ZUQECAYmJiNHv2bIWHh5dbx263y27nW0YAAFwoPDZy4+Pjoz59+igpKcmtPSkpSf37969wnVOnTslqdS/ZZiv6VpEHr4sGAAANiEenpaZOnao33nhDiYmJSk5O1sMPP6zU1FTXNNP06dM1btw4V/8RI0Zo5cqVSkhI0M8//6yvvvpKkyZN0hVXXKFWrVp56jAAAEAD4tGLQ8aMGaNjx45p1qxZSktLU48ePbRmzRq1b99ekpSWlqbU1FRX/9jYWGVnZ2vhwoV65JFHdNFFF2nQoEF6/vnnPXUIAACggfHofW48oa7ucwMAAOpOTc7fHv+2FAAAQG0i3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFMh3AAAAFPxeLhZtGiRIiIi5Ovrqz59+mjjxo2V9s/Ly9MTTzyh9u3by263q2PHjkpMTKynagEAQEPn5cmdL1++XFOmTNGiRYs0YMAAvfbaaxo2bJh27typdu3aVbjO6NGj9b///U+LFy9Wp06dlJ6ersLCwnquHAAANFQWwzAMT+28X79+6t27txISElxt3bp108iRIxUfH1+u/yeffKKxY8fq559/VtOmTc9pn1lZWQoJCVFmZqaCg4PPuXYAAFB/anL+9ti0VH5+vrZu3aohQ4a4tQ8ZMkSbNm2qcJ3Vq1crKipKL7zwglq3bq0uXbro0Ucf1enTp8+6n7y8PGVlZbm9AACAeXlsWiojI0MOh0NhYWFu7WFhYTpy5EiF6/z888/697//LV9fX61atUoZGRmaOHGijh8/ftbrbuLj4zVz5sxarx8AADRMHr+g2GKxuL03DKNcWwmn0ymLxaJ33nlHV1xxhYYPH665c+dq6dKlZx29mT59ujIzM12vAwcO1PoxAACAhsNjIzfNmjWTzWYrN0qTnp5ebjSnRHh4uFq3bq2QkBBXW7du3WQYhg4ePKjOnTuXW8dut8tut9du8QAAoMHy2MiNj4+P+vTpo6SkJLf2pKQk9e/fv8J1BgwYoMOHDysnJ8fVtmfPHlmtVrVp06ZO6wUAAI2DR6elpk6dqjfeeEOJiYlKTk7Www8/rNTUVMXFxUkqmlIaN26cq/8dd9yh0NBQ3XPPPdq5c6c2bNigxx57TBMmTJCfn5+nDgMAADQgHr3PzZgxY3Ts2DHNmjVLaWlp6tGjh9asWaP27dtLktLS0pSamurqHxgYqKSkJP3hD39QVFSUQkNDNXr0aM2ePdtThwAAABoYj97nxhO4zw0AAI1Pnd7nJiUl5ZwLAwAAqGs1DjedOnXSNddco2XLlunMmTN1URMAAMA5q3G42bFjhy6//HI98sgjatmypX7/+9/rm2++qYvaAAAAaqzG4aZHjx6aO3euDh06pCVLlujIkSP6zW9+o+7du2vu3Lk6evRoXdQJAABQLef8VXAvLy/dcsstev/99/X8889r3759evTRR9WmTRuNGzdOaWlptVknAABAtZxzuNmyZYsmTpyo8PBwzZ07V48++qj27dundevW6dChQ7r55ptrs04AAIBqqfF9bubOnaslS5Zo9+7dGj58uN566y0NHz5cVmtRToqIiNBrr72mSy65pNaLBQAAqEqNw01CQoImTJige+65Ry1btqywT7t27bR48eLzLg4AAKCmuIkfAABo8Or0Jn5LlizRBx98UK79gw8+0JtvvlnTzQEAANSqGoeb5557Ts2aNSvX3qJFCz377LO1UhQAAMC5qnG4+eWXXxQREVGuvX379m4PuQQAAPCEGoebFi1a6Pvvvy/XvmPHDoWGhtZKUQAAAOeqxuFm7NixmjRpkr744gs5HA45HA6tW7dOkydP1tixY+uiRgAAgGqr8VfBZ8+erV9++UXXXnutvLyKVnc6nRo3bhzX3AAAAI8756+C79mzRzt27JCfn58uvfRStW/fvrZrqxN8FRwAgManJufvGo/clOjSpYu6dOlyrqsDAADUiXMKNwcPHtTq1auVmpqq/Px8t2Vz586tlcIAAADORY3Dzeeff66bbrpJERER2r17t3r06KH9+/fLMAz17t27LmoEAACothp/W2r69Ol65JFH9OOPP8rX11crVqzQgQMHNHDgQI0aNaouagQAAKi2Goeb5ORkjR8/XpLk5eWl06dPKzAwULNmzdLzzz9f6wUCAADURI3DTUBAgPLy8iRJrVq10r59+1zLMjIyaq8yAACAc1Dja26uvPJKffXVV4qMjNQNN9ygRx55RD/88INWrlypK6+8si5qBAAAqLYah5u5c+cqJydHkjRjxgzl5ORo+fLl6tSpk1566aVaLxAAAKAmahRuHA6HDhw4oJ49e0qS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T506VW+88YYSExOVnJyshx9+WKmpqYqLi5NUNOoybtw4V/+4uDj98ssvmjp1qpKTk5WYmKjFixfr0Ucf9dQhAACABsaj19yMGTNGx44d06xZs5SWlqYePXpozZo1at++vSQpLS3N7Z43ERERWrNmjR5++GG98soratWqlebPn69bb73VU4cAAAAaGI9fUDxx4kRNnDixwmVLly4t1zZw4EB99913dVzVubHb7XrqqafKTYNdKC7045f4DC7045f4DDj+C/v4pYbxGXj0Jn4AAAC1zePPlgIAAKhNhBsAAGAqhBsAAGAqhBsAAGAqhJtasmjRIkVERMjX11d9+vTRxo0bPV1SvdmwYYNGjBihVq1ayWKx6MMPP/R0SfUqPj5effv2VVBQkFq0aKGRI0dq9+7dni6rXiUkJKhnz56um3ZFR0fr448/9nRZHhMfHy+LxaIpU6Z4MN6CXQAACVNJREFUupR6M2PGDFksFrdXy5YtPV1WvTp06JDuuusuhYaGyt/fX5dddpm2bt3q6bLqTYcOHcr9GbBYLHrwwQfrvRbCTS1Yvny5pkyZoieeeELbtm1TTEyMhg0b5naPHjPLzc1Vr169tHDhQk+X4hHr16/Xgw8+qP/85z9KSkpSYWGhhgwZotzcXE+XVm/atGmj5557Tlu2bNGWLVs0aNAg3Xzzzfrpp588XVq9+/bbb/XXv/5VPXv29HQp9a579+5KS0tzvX744QdPl1RvTpw4oQEDBsjb21sff/yxdu7cqTlz5tT5HfEbkm+//dbtv39SUpIkadSoUfVfjIHzdsUVVxhxcXFubZdcconx+OOPe6giz5FkrFq1ytNleFR6erohyVi/fr2nS/GoJk2aGG+88Yany6hX2dnZRufOnY2kpCRj4MCBxuTJkz1dUr156qmnjF69enm6DI/54x//aPzmN7/xdBkNyuTJk42OHTsaTqez3vfNyM15ys/P19atWzVkyBC39iFDhmjTpk0eqgqelJmZKUlq2rSphyvxDIfDoffee0+5ubmKjo72dDn16sEHH9QNN9yg6667ztOleMTevXvVqlUrRUREaOzYsfr55589XVK9Wb16taKiojRq1Ci1aNFCl19+uV5//XVPl+Ux+fn5WrZsmSZMmFDrD6muDsLNecrIyJDD4Sj3JPOwsLByTzCH+RmGoalTp+o3v/mNevTo4ely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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.plot(train_a, label='train accuracy')\n",
    "plt.plot(val_a, label='validation accuracy')\n",
    "\n",
    "plt.title('Training history')\n",
    "plt.ylabel('Accuracy')\n",
    "plt.xlabel('Epoch')\n",
    "plt.legend()\n",
    "plt.ylim([0, 1])\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "SbT4YzHFh1s7"
   },
   "source": [
    "Accuracy of Pos/Neg on Test Set"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 42,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 28664,
     "status": "ok",
     "timestamp": 1695329081374,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "L1ZzFERMAQk9",
    "outputId": "56411273-fc8c-476f-f660-21239bd321ad",
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy:  0.8546\n",
      "F1-Macro:  0.8499155218346083\n"
     ]
    }
   ],
   "source": [
    "import numpy as np\n",
    "test_acc, _, test_f1 = eval_model(\n",
    "  model,\n",
    "  test_data_loader,\n",
    "  loss_fn,\n",
    "  device,\n",
    "  len(df_test)\n",
    ")\n",
    "\n",
    "\n",
    "print(\"Accuracy: \",test_acc.item())\n",
    "print(\"F1-Macro: \",test_f1.item())"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "-Vexpkg79IXK"
   },
   "source": [
    "Scores Average of PP - Poisitve Scores on Positive Sentences\n",
    "                  PN - Poisitve Scores on Negative Sentences\n",
    "                  NP - Negative Scores on Positive Sentences\n",
    "                  NN - Negative Scores on Negative Sentences"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 43,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 605
    },
    "executionInfo": {
     "elapsed": 29278,
     "status": "ok",
     "timestamp": 1695329182045,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "zUrNTiUp8kPc",
    "outputId": "e99daa47-22f9-471e-97d4-245bf6f0b8ab",
    "tags": []
   },
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAroAAAISCAYAAAAjsmyaAAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAAPYQAAD2EBqD+naQAAADh0RVh0U29mdHdhcmUAbWF0cGxvdGxpYiB2ZXJzaW9uMy4xLjIsIGh0dHA6Ly9tYXRwbG90bGliLm9yZy8li6FKAAAgAElEQVR4nOzdeXTU1f3/8ddkMpPJNiSBBAghGBJBsCDgGqjQii1+lcWKCxYoAvKlgGjbb1p+cqzil2hopVWxxR2NWIt76o7gAi5glSoii5oQCIuQkG2yz2Rmfn9wmC/jTCAZkkz4+Hyc4zmdez/3zvuTYvvi5n7ux+T1er0CAAAADCYi3AUAAAAAHYGgCwAAAEMi6AIAAMCQCLoAAAAwJIIuAAAADImgCwAAAEMi6AIAAMCQIsNdQFfy+eefy+v1ymKxhLsUAAAABOFyuWQymTR8+PCTXkvQPY7X6xXvzwAAAOi62pLVCLrHObaSO2TIkDBXAgAAgGC2bdvW6mvZowsAAABDIugCAADAkAi6AAAAMCSCLgAAAAyJoAsAAABDIugCAADAkAi6AAAAMCSCLgAAAAyJoAsAAABDIugCAADAkAi6AAAAMCSCLgAAAAyJoAsAAABDIugCAADAkAi6AAAAMCSCLgAAAAyJoAsAAABDIugCAADAkCLDXQBwOtm3b59Wr16tb775Rv3799f06dOVkZER7rIAAEAQBN0OtmnTJj333HOqrq5Wdna2rr/+esXFxfn63W63ampqZLfbFRHBAntncrlc8nq9slqtfu2HDh3SCy+8oMOHD2vEiBG6/PLLFRUVpa+//lpTp05VTU2NpKP/3b788svKz8/X0KFDw3ELAADgBExer9cb7iK6im3btkmShgwZ0i7zrV69Wrm5uX5tgwYN0jPPPKOYmBg9+eSTeuyxx1RWVqZevXpp3rx5mjJliiRp7969euCBB7R582YlJibquuuu09SpU2UymYJ+l9frVVNTU7vU3RZfffWVVq1apa+//lrp6emaMWOGLrrook6tISoqqsWfSzAVFRW66667tHbtWrndbo0ePVqLFy9Wv3799Nlnn2nOnDmqr6/3XX/22Wfrqaee0qJFi7R+/fqA+UaOHKknnniiXe4FAACcWFvyGkH3OO0ZdOvq6jR69GjV1tYG9N1+++2KiIjQkiVLAvruueceXXTRRbryyitVXl7u13fjjTfq97//ve/zwYMHFRsbK7vdrjvvvFPffPPNKdfdFnV1dfr666/1/T9CmZmZSkhI6LQ6BgwYoDvuuMMv7NbX1+vpp5/We++9p+joaE2cOFGTJk2SJE2ePFnbt2/3m6NXr1568803dd111wX9Of7mN7/Rk08+qaqqqoA+i8Wir776qp3vCgAABNOWvMbWhXby/RXVrVu3Bg25krR582bt2rUraN+qVav07bffBoRc6egK8a9+9Svt3LlTf/rTn7R7926ZzWb99Kc/9dsO0Vm+++67gJArHQ3gnRV0vV6vampqdOjQISUmJkqSmpubNXv2bH3xxRe+6z766CN98cUXuuSSSwJCrnR0u8KxvbfBvPfee0pKSgoadHv06NFOdwMAANoTK7rHOZUV3cbGRs2aNcvvc7BAJUnJyckqKysL2mc2mxUXF6fq6uqg/RkZGdqzZ09AwIyPj9eAAQPkdrv13XffqbKyUpKUmJio3r17y2w2SzoaAisqKtTc3Ky4uDjZ7Xa/eTwej1wulywWy0n3DH/55ZdyuVxB+0aMGNGm7QT19fVyuVyKjY1VZOT//f2roaFBpaWlamhokM1mU0pKimJiYiRJDodDJSUlvr9gJCQkqF+/fqqpqdHu3buDfk+fPn104MCBoH0pKSkqLS0N2me329WtWzft27cvoO/mm2/WggULWn2vAAAgdKzodgE2m03x8fG+B5eOl5ycrPr6etXV1QX0xcbGymazBQ26JpNJNTU1QVdRa2pq1NDQoL179/rNe/jwYdXV1WngwIGqra1VYWGh3G63rz8hIUH9+/eXyWTSoUOHdOjQIbndbkVERCglJUWpqakymUzyer1yOByqrq6W2WxW9+7dFRUVFTToWq3WgJBbVVXlF76Prfg6nU4VFRX59sRGREQoNTVVPXv2VH19vb7++mt5PB5JR7dKVFRUaMCAAYqMjFRhYaHfz6Kqqkput1vR0dEBNR1zor/XxcXFqampKejPvnv37kpMTJTL5VJpaak8Ho9MJpNSUlI0c+bMFucEAADhQ9BtJ8ECVP/+/bVnzx5fcLJYLOrbt6+io6OVmpqqb7/91u96k8mk3r17y2KxqKyszBfwjunevXtA2/GqqqqChufa2lo5HA7t27fPL+QeG1NeXi6TyeS30unxeHTo0CFFRESoV69e2r17t9+v7Q8dOqSUlJSg2zN69uzp93nv3r06cuSI73NFRYWSk5OVnp6u4uJivwe/PB6P9u/fr+joaF+gPJ7X69WBAwcUGxvbYuA/UdCNj49XQkJCwBaEmJgYJSQkKC4uTkVFRb6f47Ewm5SUJOnoinCvXr3kdDpltVplNpvbtHINAAA6D0G3A0VGRiorK0sul0vNzc2y2Wy+UGS32zVw4EAdPnxYjY2Nio6OVs+ePRUbGyvp6ANWBw4cUE1NjSIjI9WjRw+lpqbqyJEjqqioCPiuY6uuLXE4HGpsbAzaV1VV1eIWhLKyMtlstqB7U8vLy9WvXz8dOnRITU1NslgsSk5OVrdu3eT1emUymVRfX+8Xco+ft1u3bi3uYy4vL2+xr66uLuBIsOPFxcWpvLw8INRHR0crLi5OMTExKi0tVUVFhTwejxITE9WrVy+ZTCZZLBadddZZqq+vl9PpVGxsrCwWi988ZrP5hGEaAAB0DQTddnKiVT2LxRIQlqSjgaylh8hiY2M1YMAAX2A8JikpSWVlZWpoaPC7vnfv3rLZbC3WEBUVdcLanU5n0D6XyxU05EpHzwC2Wq360Y9+pKamJu3bt08HDx7UwYMHFRUVpb59+wbUebxg2zq+P3ew8RaLRXFxcS0G/vj4eJ155pkqKSnxrRbb7Xb169dPknyr1L169Wrx+2NiYnx7gU+GFV0AALomgm47iYqK0qpVqzrluxwOh5555hl9/PHHstvtuvLKKzV27FjV19dr6tSpKi4u9rs+KytLTz/9tGbNmqUdO3YEzLdo0SKtW7dO7777bkDf0KFDNXDgQD3//PNBa/mf//kfPfrooyopKZHD4fC1NzU1qaSkRLNmzdJDDz0UdOzcuXP10EMPBX0AbO7cuYqMjNTdd98d0Ddv3jxdddVVmj17tgoLC/36FixYoDlz5vg+Hz58WFar1XciQ0c40V8iAABA+HDqwnHa+4URncXr9frO0XW5XNq/f7/fg19paWmyWCxqbGzUt99+67d6e2yvbENDg3bt2uW3J9ZkMunMM89URERE0OPQbDabBg8erKamphOeMFFRURGwjcBsNmvIkCG+ExKO/2MYFxfn+97jH5Azm81KSUlR7969ZTKZ5Ha7VVpaKpfLpbPPPlvXXHONLr300lP6WQIAgK6NUxd+wCwWizIyMnTGGWcE/ErdZrPpRz/6kaqrq+VyuRQfH+/b7hAdHa1BgwaptLRU9fX1AUd59e3bV/v37/cFUqvVqn79+slkMrW4v1c6epxZVlaWiouLfQE7KipKGRkZMpvNSkhI0ODBg3XkyBE1NzcrPj5eiYmJvqPNevXqpZSUlKBHnpnNZvXu3TvoCyMAAAAIugZgMpl0xx13dPgrgB0Ohz755BOtX79eH374ob7++mvfA3QRERFBT4SYO3eurr/+enm9Xu3YsUMmk0mDBg1q11Da1lcAAwCAHwaCrkGYTKYTPozWHmw2m7Zu3aq33nrL13b48GGZTCb16NEj4CUYAwYM0LXXXuur69xzz+3Q+gAAAI5H0EWr1dbW6rnnngto93q98ng8yszMVHl5uaKionTddddp+vTpvuPSAAAAOhtBF61WWlra4vaIpqYmJSQk+N54duWVVwa8XhgAAKAzRZz8EuCotLQ0X5D9vmMPrUVERGjo0KFKTU3tzNIAAAACEHTRalarVfPnzw9oj4yM9L3212QyaebMmTwcBgAAwo6tC2iTGTNmKCUlRc8884wOHz4si8Wimpoa1dbWKiEhQRMnTvSFXgAAgHBiRRdt9l//9V967LHH1KdPHxUWFurw4cMqLi7W9u3bVV5eHu7yAAAAJBF0EaL8/Hx9/PHHfm1Op1P5+fk6dOhQmKoCAAD4PwRdhOT4s3SPV11drUceeUS8WRoAAIQbQRchaW5ubrFv+/btOnjwYCdWAwAAEIigi5BMnDgxaHu3bt00fPhwjhcDAABhx6kLCMmvfvUrffjhh9q0aZOvLSoqSmeccQbHiwEAOlR1dbVefvllbd++XampqZo8ebLS09N9/Tt27NCGDRtktVo1btw4paWltXsNLpdLpaWl6t69u+9V9+h6TF42U/ps27ZNkjRkyJAwV3L6WL58ud566y1ZrVYlJCToF7/4ha655ppwlwUAMKgjR45o1qxZflvkoqKitGLFCp177rm6//77tXr1al+f2WzWH//4R40fP77davjnP/+pVatWqbKyUrGxsbrmmms0f/58RUR0zC/Kjxw5IrfbHXB8565du/TGG2+osbFRo0eP1qhRozpkoamqqkqFhYXq3bu3+vTp0+7zt1Vb8horujglCxcuVHFxsSorK5WUlNTilgYAANpDfn5+wHMgTU1Nuu+++7Ro0SK/kCtJbrdbeXl5GjNmjOLj44PO6fV6W3zF/fetW7dOf/nLX3yf6+rq9OSTTyoyMlI33HCD37Uej0ebNm3Srl271Lt3b11yySUBq7/H1huDBdT9+/dr2bJl+uKLLyRJgwYN0qJFi5SVlaUXX3xR9913n2/8Sy+9pHHjxumPf/xjq+7jmKioqBOG47///e965pln1NTUJJPJpDFjxuh///d/fW9E7epY0T0OK7qh2bJli/Lz8zVjxgyde+654S4HAHCaaUvQnD59uoqLi4P2XXvttXruueeC9t15550aNWqU1q9frz179igjI0OXXnqprFar8vLyVFhY2KrvLykpUWNjY0C72WxWZmam77PH49GBAwfU0NDga4uMjFRaWpqsVutJv8fr9WrPnj1yuVwB35Oenq49e/YEPeEoLS2tTSE0KytLt956a9Cw++abb+quu+4KaJ8wYYIWLVrU6u84kZMF7WDakte6RNAtLi5Wbm6utmzZoujoaF1xxRXKyck56Z6X+vp6rVy5Um+99ZbKysrUs2dPTZgwQXPnzm3VH6LvI+gCAND5GhsbNWvWrFZdu2/fPr/weIzJZFJiYqIqKiqCjuvZs6cqKir8gqPFYlHfvn0VGdn6X3Dv3r27xZOHzjzzTF9oKy8vD/oSpZiYGKWlpcnj8aimpkaNjY2yWCyy2+1+ddTW1rZ4glG3bt1UXV0dtC8hIUEpKSmtvh/paKiuqalRXV2dIiIiZLfbFR0dfcKfdWZmZrts1Vi1alWb9zifVlsXHA6HZsyYodTUVK1YsUIVFRXKy8tTVVWVli9ffsKxS5Ys0fr16/Xb3/5WZ555pr788kutWLFC1dXVuu222zrpDgAAQGfp1q1b0PBlt9tlt9uDBt2IiAjV19cHrI66XC4dOXJEvXr1ktPpVFVVlZqamnzPnURFRQXMFR0drZqamoD2769M1tbWBq3/WB0HDhyQ0+n0tVdUVCgtLc0X+k50jKfH42mxr6Xw2dzcrOrqajU2NioyMtJ3f16vVwcOHFB9fb3v2urqaiUnJ8vtdgedy+v1njbn5Yc96K5Zs0YOh0MFBQVKSkqSdHRZPicnR/PmzfP7NcDxmpub9dZbb+nGG2/U9OnTJUkXXXSRDh48qDfeeIOgCwDAaeg3510vq/nE8eRfW9/V2u0fyul2yWQy6dz0wZqZfZWiLFZt6Pmp/vnpG2r2HA2KMdZozRs9RSs3PBN0LnejS78c8DP9ae3janAd3ZLQ0NCghtp6/c/PZiozua8kqbapXiUV38mV6tJjH73ou1aSzBFmzR11rc7unSlzhFmSlFv+kPY0HQj4vghThM6MTlWx03/7hcfjkbVB+sOYo5lmX+Uh3fna34PWfOP5Vyl/c4Eq6v1Xdc2mCP3u4mmKj4pTtCXKF3qr6h26+61HVFH3f9fX1dRq4U+nqdHVpAe//TbgO6orqjQyc7je/+bfAX1ndO+jW0feELS21nC6m3XfZ/8MeXxbhD3obty4UdnZ2b6QK0njxo3T4sWLtWHDhhaDrtfrldvtDthYbrfbT5u/ZQAAAPn9/3ZrA1DfM9LldDoVGRkph8Wl+//zrK+vX0Y/1dXVyWQyKTY2Vq+WfCSnJ/gKaaPbqeXv5/sFV0lyul26b8Nq9e3bV0eOHFFlZaWvTpvNJnu0XU6nU2azWW63WyveXS2TyaS4uDglJyer1hR8z3FMbIw2FH0WtK+wrER5Hz8ps/loWI6Pjw9YPY6NjdW/9nwgqz1akc4638pvRESE4uLjteT1lWpubpbZbFZiYqKSkpJUWlqqqjr/UNzscevvH/xT0dHRQWtxul3aWb1XFovFbyXcZDLJHROhP3+yOui4turozBb2oFtUVKTJkyf7tVmtVqWnp6uoqKjFcRaLRVdddZVWr16tESNGKCsrS9u2bdNzzz2nadOmhVyP1+v1W74HAAAdK9jDXSdjNptbDGlms1l2u92vzW63q7KyMuBau90uh8MRdJ6GhgbV1NQEbIc4tq82LS1Ne/bs8YXNY3tdnU6n+vbtq8bGRr8tDFarVSkpKSd9e2hlZaVqamrk9XoVFxfn20IQGxurxsZG7d69W9LR0JmQkKDY2Fi53W4dOnTIN4fb7daRI0dkMplazDVOp/OED64dy2MOh8Nvy4PFYjlh/W3R0NDQ5rDr9Xpb/QBb2IOuw+EI+MMoHf2D19JG62OWLFmiO+64Q9dee62vbfr06brppptCrsflcmnnzp0hjwcAAG1z/Ipha7YuhKLJ5dTKjf/U9oP/d7rCj1LP1LwxU3T3m4/oQNXhgDGJMXYlmWL1XZD56mvrdNGgQUFPa2hqatKkM36sQdmZ2lN+QHvLD6p7XIIG985UhClCG775VKs/eSVg3IVnDFWzq1lFZWV+cw3unanfjp2hRz98Xv8+ss3X5/V6VVVVpZnnTtK/tr4T9L699S5lJqVp56HdAX1Ws0W3jPqlct94SG6v/77f1G7Jun3MnA45l/f4rQvffPNNSMG5tYcOhD3otqQ1aX358uV6//33tXTpUmVkZGj79u1asWKF7Ha7br755pC+12KxKCsrK6SxAACg7Y4/WqxD927GRSg9/eiWB6vVKqfNpPu3PCuXNfiKojnGqqKK/UH73F6PXvvmgxa/avWXbyjiqwhVVFT4vi8pKcm3xfL7JydER0drv6tcBw4E7uvd8V2Rbn/nQX33XbDILT36yYtBH9CTpCN1VYqMD36qQXRcjP757Xql9Oqp0tJS38pxVFSUrImxuuffT7d4f+1l4MCBQR/6O5HWHgUndYGg29KvDGpqalrcnysd/RvAqlWrtHLlSo0dO1aSdP7558tkMunPf/6zpk6dqu7du7e5HpPJdNocggwAgBF01BvFgrHZbAHHWSUkJMjtdquyslIej8e3JSAxMVEejyfo1gqr1aro6GhVVVUF/R6Px6PS0lLfZ6fT6dtaYLfblZKSIpvNprq6Ol8IPtFvslsKstLRbQrH5vq+qKgo2e12NTc3q6KiwndiQ3x8vJKTk33/OS4uTo2NjYqIiGhz8DwV0dHRbT5erC2rzGEPupmZmQF7cZ1Op0pKSgL27h7vWJofNGiQX/ugQYPU3NysAwcOhBR0AQBA54qKitKqVavCXYYaGhp06NAhpaSkKDY2VtLRLZYLFizwe0lFVFSU7rnnHp1zzjn69a9/HbDlceTIkSovL9eRI0cCviM+Pl4rV65UTk6Ovj3utAOz2axf/epXfm9dO96CBQu0atWqoKu6U6ZM0UUXXaSFCxf6HVlmNpuVm5ur7OxsSUePNispKVFycnKXyUgdHarDHnRHjx6tBx98UJWVlUpMTJR09PV6TqdTY8aMaXHcsXctb9++Xampqb72r776StLRN4MAAICuz2QytXlVrz0dezNbdHS0MjIy/PrsdrsefvhhvfXWW/rqq6+UkpKiCRMm+HLIvffeq9WrV2vDhg2KjIzUpZdeql/+8pcaP3580O8qKSnRc88953ut7zFlZWXasGGD0tLStH+//3aJ5ORkXXLJJYqJidHtt9/ud75tWlqarr76anXr1k0PPvig/vGPf6ioqEjp6emaMmWKzjnnHN+1MTExOuuss/zmDuXNZKeTsL8ZzeFwaPz48erTp4/mz5+v8vJyLVu2TD/+8Y/9XhixePFiFRQUaMeOHZKOLtNPmTJFBw4c0MKFC5WRkaFt27Zp5cqV+slPfqJ77723zbXwZjQAAH5YvF6v7rzzTn3zzTftOm9Lrwq22WwnfOVxv379VFZW5jspITo6Wj179vQ9fNXY2Kiqqiq53W5FR0erW7duvuPIQjFgwADdcccdp1XYPa3ejGa325Wfn6/c3FwtXLhQNptN48ePV05Ojt91Ho/H728wZrNZDz30kO6//349+uijOnLkiHr37q1p06bp17/+dWffBgAAgE/37t2DPliWlJQU9NXAxxw7tuxY5vl+iLXZbOrVq1f7FmtgYV/R7UpY0QUA4IfnRCusp+LTTz/V6tWrtWfPHvXr10/Tp0/XBRdcoDVr1uhvf/tbwPU//vGPtWzZsnav40ROx60Lp9WKLgAAQDh11B7hiy++WBdffHFA+7Rp07R9+3a99957vraMjAzdeuutYd2rbEQEXQAAgE4UGRmpe+65Rzt27ND27dvVu3dvZWdnn9JeWwRH0AUAAAiDwYMHa/DgweEuw9A674RmAAAAoBMRdAEAAGBIBF0AAAAYEkEXAAAAhkTQBQAAgCERdAEAAGBIBF0AAIAw2bJli26++WZt2bIl3KUYEkEXAAAgDBoaGvTQQw+ptLRUq1at6pDXEP/QEXQBAAA62WuvvabLL79cn3/+uXbv3q1vvvlGBQUF4S7LcAi6AAAAnejjjz/WkiVLVFNTI0nyeDyqrKzUqlWrdOjQoTBXZywEXQAAgE60Zs2aoO3Hwq7X6+3kioyLoAsAANCJ9u/fH7Td4/Fo69atOnjwYCdXZFwEXQAAgE40fPjwoO0Wi0XDhg1TampqJ1dkXARdAACATnTDDTcoPj4+oD0lJUWzZs2SyWQKQ1XGRNAFAADoRH379tXq1at1zjnnKCoqSnFxcerbt6+mTZumnj17hrs8QyHoAgAAdLK0tDStXLnSt1WhT58+mjhxYrjLMhyCLgAAQBhERUVp1qxZ6tGjh2bOnKmoqKhwl2Q4keEuAAAA4Ifq3HPP1bnnnhvuMgyLFV0AAAAYEkEXAAAAhkTQBQAAgCERdAEAAGBIBF0AAAAYEkEXAAAAhkTQBQAAgCERdAEAAGBIBF0AAAAYEkEXAAAAhkTQBQAAgCERdAEAAGBIBF0AAAAYEkEXAAAAhkTQBQAAgCERdAEAAGBIBF0AAAAYEkEXAAAAhkTQBQAAgCERdAEAAGBIBF0AAAAYEkEXAAAAhkTQBQAAgCFFhrsASSouLlZubq62bNmi6OhoXXHFFcrJyZHNZmtxzP79+zV27NigfRaLRV999VVHlQsAAIDTQNiDrsPh0IwZM5SamqoVK1aooqJCeXl5qqqq0vLly1scl5KSomeffdavzev1as6cObrwwgs7umwAAAB0cWEPumvWrJHD4VBBQYGSkpIkSWazWTk5OZo3b54yMzODjrNarRo2bJhf2yeffKKamhqNHz++w+sGAABA1xb2PbobN25Udna2L+RK0rhx42S1WrVhw4Y2zfXaa68pLi5Ol1xySXuXCQAAgNNM2Fd0i4qKNHnyZL82q9Wq9PR0FRUVtXoel8ult99+Wz/72c8UFRUVcj1er1f19fUhjwcAAEDH8Xq9MplMrbo27EHX4XDIbrcHtNvtdlVXV7d6no0bN6qqquqUty24XC7t3LnzlOYAAABAx7Fara26LuxBtyVtSeuS9Oqrr6pHjx7Kzs4+pe+1WCzKyso6pTkAAADQMQoLC1t9bdiDrt1ul8PhCGivqalp8UG076urq9P777+vq6++Wmaz+ZTqMZlMiomJOaU5AAAA0DHashAa9ofRMjMzA/biOp1OlZSUtDrorlu3Tg0NDZowYUJHlAgAAIDTUNiD7ujRo7V582ZVVlb62tatWyen06kxY8a0ao7XXntN6enpOuecczqqTAAAAJxmwh50p0yZovj4eM2fP18ffPCBCgoKtHTpUk2YMMFvRXfx4sUaPHhwwPiKigpt2rRJV1xxRWeWDQAAgC6uS+zRzc/PV25urhYuXCibzabx48crJyfH7zqPxyO32x0w/s0331RzczPbFgAAAODH5PV6veEuoqvYtm2bJGnIkCFhrgQAAADBtCWvhX3rAgAAANARCLoAAAAwJIIuAAAADImgCwAAAEMi6AIAAMCQCLoAAAAwJIIuAAAADImgCwAAAEMi6AIAAMCQCLoAAAAwJIIuAAAADImgCwAAAEMi6AIAAMCQCLoAAAAwJIIuAAAADImgCwAAAEMi6AIAAMCQCLoAAAAwJIIuAAAADImgCwAAAEMi6AIAAMCQCLoAAAAwJIIuAAAADImgCwAAAEMi6AIAAMCQCLoAAAAwJIIuAAAADImgCwAAAEMi6AIAAMCQCLoAAAAwJIIuAAAADImgCwAAAEMi6AIAAMCQCLoAAAAwJIIuAAAADImgCwAAAEMi6AIAAMCQCLoAAAAwJIIuAAAADImgCwAAAEMi6AIAAMCQCLoAAAAwJIIuAAAADImgCwAAAEPqEkG3uLhYs2fP1rBhw5Sdna3c3Fw1Nja2amxVVZWWLFmiH//4xxoyZIjGjRunNWvWdHDFAAAA6Ooiw12Aw+HQjBkzlJqaqhUrVqiiokJ5eXmqqqrS8uXLTzi2rq5O06dPV1RUlBYvXqzu3btr7969crlcnVQ9AAAAuqqwB9A5BPcAACAASURBVN01a9bI4XCooKBASUlJkiSz2aycnBzNmzdPmZmZLY59+OGH1djYqOeff142m02SdOGFF3ZK3QAAAOjawr51YePGjcrOzvaFXEkaN26crFarNmzYcMKxL774oq6++mpfyAUAAACOCfuKblFRkSZPnuzXZrValZ6erqKiohbH7du3T0eOHJHdbtfcuXP10UcfKTY2VpdffrkWLVoUcvj1er2qr68PaSwAAAA6ltfrlclkatW1YQ+6DodDdrs9oN1ut6u6urrFcUeOHJEk/fnPf9Zll12mRx99VIWFhfrrX/8ql8ul3NzckOpxuVzauXNnSGMBAADQ8axWa6uuC3vQbcnJ0rrH45EkZWZmKi8vT5KUnZ2t5uZm/fnPf9Ytt9yi5OTkNn+vxWJRVlZWaEUDAACgQxUWFrb62rAHXbvdLofDEdBeU1NzwgfREhISJEkXXXSRX/tFF10kj8ejoqKikIKuyWRSTExMm8cBAACg47V224LUBR5Gy8zMDNiL63Q6VVJScsKg27dvX1ksloB2r9crSYqICPutAQAAIIzCngZHjx6tzZs3q7Ky0te2bt06OZ1OjRkzpsVxVqtVo0aN0qZNm/zaN23apMjISLYfAAAA/MCFPehOmTJF8fHxmj9/vj744AMVFBRo6dKlmjBhgt+K7uLFizV48GC/sQsWLNDXX3+tP/zhD/rwww/15JNP6oEHHtDUqVP9jisDAADAD0+X2KObn5+v3NxcLVy4UDabTePHj1dOTo7fdR6PR263269t6NChevjhh/WXv/xFv/71r5WQkKBp06bplltu6cxbAAAAQBdk8h7b1Apt27ZNkjRkyJAwVwIAAIBg2pLXwr51AQAAAOgIBF0AAAAYEkEXAAAAhkTQBQAAgCERdAEAAGBIBF0AAAAYEkEXAAAAhkTQBQAAgCERdAEAAGBIBF0AAAAYEkEXAAAAhkTQBQAAgCERdAEAAGBIBF0AAAAYEkEXAAAAhkTQBQAAgCERdAEAAGBIBF0AAAAYEkEXAAAAhkTQBQAAgCERdAEAAGBIkaEMqq+v17///W/95z//0eHDh9XY2KjExERlZWXpwgsv1JlnntnedQIAAABt0qagu2fPHq1atUqvvfaa6uvrZTKZZLfbZbVa5XA41NTUJJPJpAEDBmj69Om66qqrFBHBojEAAAA6X6uD7t13361nnnlGGRkZmj9/vi644AINHjxYkZH/N0Vpaam++OILrV+/XnfddZeefPJJ5eXlaciQIR1SPAAAANCSVgfdHTt26IknntD555/f4jUpKSn6+c9/rp///Oeqra3Vk08+qf/85z8EXQAAAHQ6k9fr9Ya7iK5i27ZtkkQwBwAA6KLaktfafQPt4cOHtWPHjvaeFgAAAGiTkE5dOHjwYIt9b7/9th5++GFt2rQp5KIAAACAUxVS0L3kkktkMpla7M/IyAi5IAAAAKA9hBR077777oCgW19fr88++0zvvPOOli1b1i7FAQAAAKFq94fRli1bpu3bt2v16tXtOW2n4GE0AACAri2sD6ONGTNGX375ZXtPCwAAALRJuwfdyspKde/evb2nBQAAANokpD26wXg8Hu3atUsPPfSQbrnllvaaFgAAAAhJSHt0zzrrrBZPXfB6vX59JpPptDlXlz26AAAAXVtb8lpIK7oLFiw44fFiAAAAQLiFFHQXLlzY3nUAAAAA7ardH0YDAAAAuoJWB90lS5aorKysTZO//fbbeuWVV9pcFAAAAHCqWr11obi4WJdeeql+9rOfadKkSTrvvPMUHR0dcN3evXv1zjvv6KWXXtLhw4e1fPnydi0YAAAAaI1WB938/HytX79ejzzyiObMmaPIyEj169dPSUlJioqKUnV1tfbt26fq6mpFR0frqquu0rx58zhTFwAAAGER0vFiO3bs0HvvvaetW7eqtLRUjY2NSkxMVP/+/XXBBRdo7NixiouL64h6OxTHiwEAAHRtHX682ODBgzV48OBQhgIAAACdglMXAAAAYEghBd1NmzbpzTff9H0+cuSI5syZo1GjRukPf/iDmpqa2jRfcXGxZs+erWHDhik7O1u5ublqbGw86bjp06dr4MCBAf8UFRW1+Z4AAABgLCFtXVixYoVGjRrl+3zPPffos88+06hRo7R27Vr169dPCxYsaNVcDodDM2bMUGpqqlasWKGKigrl5eWpqqqqVSc2jBgxQosWLfJrS0tLa9sNAQAAwHBCCrp79uzRnDlzJEnNzc1at26dcnJyNHXqVD3++ON68cUXWx1016xZI4fDoYKCAiUlJUmSzGazcnJyNG/ePGVmZp5wvN1u17Bhw0K5DQAAABhYSFsXamtrZbfbJUnbt29XQ0ODxo4dK0kaOnSovvvuu1bPtXHjRmVnZ/tCriSNGzdOVqtVGzZsCKU8AAAAILQV3e7du2vPnj0677zz9PHHHys1NVW9evWSJNXV1SkysvXTFhUVafLkyX5tVqtV6enprdpr++9//1vDhg2T2+3WOeeco1tuuUXnn39+227oOF6vV/X19SGPBwAAQMfxer0ymUytujakoHvxxRfr3nvvVWFhoV5++WVdeeWVvr7du3erT58+rZ7L4XD4VoePZ7fbVV1dfcKx559/viZNmqQzzjhDpaWlevzxxzVz5kytXr1aw4cPb/0NHcflcmnnzp0hjQUAAEDHs1qtrboupKD729/+VgcPHtRzzz2noUOHat68eb6+1157LeSQebzWpPWbb77Z7/NPfvITjR8/XitXrtSjjz4a0vdaLBZlZWWFNBYAAAAdq7CwsNXXhhR0k5KS9Pjjjwfte+qpp1qdsqWjK7cOhyOgvaam5qQPon1fTEyMxowZo7Vr17Zp3PFMJpNiYmJCHg8AAICO09ptC1I7vDCisbFRhw8fVnNzsyQpLi6uTUE3MzMzYC+u0+lUSUlJm4OudHQlGAAAAAg56G7evFnXXXedRowYoZ/+9Kf6+uuvJUl33nmn3n777VbPM3r0aG3evFmVlZW+tnXr1snpdGrMmDFtqqm+vl4bNmxo1buPAQAAYGwhvxlt9uzZampq0qxZs+TxeHx9iYmJeumll1o915QpUxQfH6/58+frgw8+UEFBgZYuXaoJEyb4reguXrxYgwcP9n3+7LPPNG/ePL300kvavHmzXnnlFU2dOlVlZWWtPsMXAAAAxhXym9FGjx6tBx98UM3NzXrsscd8fWeddVabgq7dbld+fr5yc3O1cOFC2Ww2jR8/Xjk5OX7XeTweud1u3+fk5GQ5nU799a9/VVVVlaKjozV8+HDdeeedGjp0aCi3BQAAAAMJKeju3LlT999/v6TADcFJSUkqLy9v03wZGRktPtx2zLJly7Rs2TLf5379+p10DAAAAH64Qtq6YDab5XK5gvaVl5crNjb2lIoCAAAATlVIQXfIkCF65ZVXgvatXbtWw4YNO6WiAAAAgFMV0taF//7v/9bs2bO1YMECXXnllTKZTNq6datefPFFrV27Vvn5+e1dJwAAANAmJm+IB8/+61//0t133+33ml673a7bbrtNEydObLcCO9O2bdskiePJAAAAuqi25LWQVnQladKkSRo3bpw+//xzHTlyRImJiRoxYgRvFQMAAECX0Oag29jYqBtuuEE333yzRo4cqezs7I6oCwAAADglbX4YzWaz6ZtvvpHZbO6IegAAAIB2EdKpC8OHD9eXX37Z3rUAAAAA7SakoLto0SI9++yzKigoUF1dXXvXBAAAAJyykE5dGD58uFwul++VvDabze8NaSaTSVu2bGm/KjsJpy4AAAB0bR1+6sK4ceMCXv0LAAAAdCUhBd1ly5a1dx0AAABAuwppjy4AAADQ1YX8woiSkhI98MAD2rRpk6qqqpSYmKiRI0dqwYIFSk9Pb88aAQAAgDYLKegWFRVpypQpampq0kUXXaSUlBSVlpbqzTff1Pvvv69nnnlGmZmZ7V0rAAAA0GohBd17771XCQkJWr16tXr16uVrP3TokGbMmKH77rtPDzzwQLsVCQAAALRVSHt0P/30Uy1cuNAv5EpSr169NH/+fH3yySftUhwAAAAQqpCCbkNDgxISEoL2JSYmqrGx8ZSKAgAAAE5VSEE3IyNDr776atC+119/Xf379z+logAAAIBTFdIe3enTp+u2225TTU2NfvGLXyg5OVllZWV65ZVX9O677yo3N7e96wQAAADaJKSge/XVV6u8vFwPPvigNmzYIEnyer2y2Wz67W9/q8mTJ7drkQAAAEBbhXyO7ty5c/XLX/5Sn3/+uaqqqpSQkKDhw4crPj6+PesDAAAAQhJy0JWk+Ph4jR49ur1qAQAAANpNSA+jvfjiiy2ek/vAAw+ooKDglIoCAAAATlVIQXf16tWy2+1B+xITE/XUU0+dUlEAAADAqQop6O7du1cDBgwI2peZmam9e/eeUlEAAADAqQop6EpSTU1N0Pba2lq53e6QCwIAAADa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",
      "text/plain": [
       "<Figure size 800x600 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "pp,pn,np,nn = eval_scores_average(\n",
    "  model,\n",
    "  test_data_loader,\n",
    "  device,\n",
    ")\n",
    "\n",
    "box_plot(pp,pn,np,nn)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "uSAQhl1kh5jw"
   },
   "source": [
    "Accuracy of Emotions on Test Set"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 44,
   "metadata": {
    "executionInfo": {
     "elapsed": 8,
     "status": "ok",
     "timestamp": 1695329212064,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "d7DaFLMiAWps",
    "tags": []
   },
   "outputs": [],
   "source": [
    "def get_predictions(model, data_loader):\n",
    "  model = model.eval()\n",
    "\n",
    "  review = []\n",
    "  predictions = []\n",
    "\n",
    "  prediction_probs = []\n",
    "  real_values = []\n",
    "\n",
    "  pos_scores = []\n",
    "  neg_scores = []\n",
    "\n",
    "\n",
    "  with torch.no_grad():\n",
    "    for d in data_loader:\n",
    "\n",
    "      reviews = d[\"review\"]\n",
    "      input_ids = d[\"input_ids\"].to(device)\n",
    "      attention_mask = d[\"attention_mask\"].to(device)\n",
    "      sentiments = d[\"sentiments\"].to(device)\n",
    "\n",
    "\n",
    "      scores,outputs = model(\n",
    "        input_ids=input_ids,\n",
    "        attention_mask=attention_mask,\n",
    "        return_scores=True,\n",
    "      )\n",
    "      _, preds = torch.max(outputs, dim=1)\n",
    "\n",
    "\n",
    "      probs = F.softmax(outputs, dim=1)\n",
    "\n",
    "      review.extend(reviews)\n",
    "      predictions.extend(preds)\n",
    "\n",
    "      prediction_probs.extend(probs)\n",
    "      real_values.extend(sentiments)\n",
    "\n",
    "      pos_scores.extend(scores[0])\n",
    "      neg_scores.extend(scores[1])\n",
    "\n",
    "\n",
    "  predictions = torch.stack(predictions).cpu()\n",
    "\n",
    "  prediction_probs = torch.stack(prediction_probs).cpu()\n",
    "  real_values = torch.stack(real_values).cpu()\n",
    "    \n",
    "  pos_scores = torch.stack(pos_scores).cpu()\n",
    "  neg_scores = torch.stack(neg_scores).cpu()\n",
    "  return review, predictions, prediction_probs, real_values, pos_scores, neg_scores"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 45,
   "metadata": {
    "executionInfo": {
     "elapsed": 27254,
     "status": "ok",
     "timestamp": 1695329239311,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "kZhYtki1AYx8",
    "tags": []
   },
   "outputs": [],
   "source": [
    "y_review_texts, y_pred , y_pred_probs, y_test, pos_scores, neg_scores = get_predictions(\n",
    "  model,\n",
    "  test_data_loader\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 46,
   "metadata": {
    "executionInfo": {
     "elapsed": 9,
     "status": "ok",
     "timestamp": 1695329239312,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "mO1FHn0GAbCU",
    "tags": []
   },
   "outputs": [],
   "source": [
    "from sklearn.metrics import classification_report\n",
    "from sklearn.metrics import confusion_matrix\n",
    "class_names = ['negative', 'positive']"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "PnBSQFJuh_As"
   },
   "source": [
    "Pos/Neg Classification Report"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 47,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 9,
     "status": "ok",
     "timestamp": 1695329239312,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "J9-01snpAcM8",
    "outputId": "4d6f7fff-9a14-49bf-c847-d2e8ba79f5f7",
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "              precision    recall  f1-score   support\n",
      "\n",
      "    negative       0.84      0.88      0.86      2511\n",
      "    positive       0.87      0.83      0.85      2489\n",
      "\n",
      "    accuracy                           0.85      5000\n",
      "   macro avg       0.86      0.85      0.85      5000\n",
      "weighted avg       0.86      0.85      0.85      5000\n",
      "\n"
     ]
    }
   ],
   "source": [
    "print(classification_report(y_test, y_pred, target_names=class_names))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "GhTpYmfdiF-P"
   },
   "source": [
    "Reverse Native Injection Accuracy"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 48,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 30105,
     "status": "ok",
     "timestamp": 1695329269413,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "pOutPTF6qJY8",
    "outputId": "a034fc7e-f3ae-44e3-9bd9-2c416c5b45b3",
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy:  0.8410000000000001\n",
      "F1-Macro:  0.8359234516446405\n"
     ]
    }
   ],
   "source": [
    "import numpy as np\n",
    "p_emb, n_emb = n_emb, p_emb\n",
    "\n",
    "\n",
    "test_acc, _, test_f1 = eval_model(\n",
    "  model,\n",
    "  test_data_loader,\n",
    "  loss_fn,\n",
    "  device,\n",
    "  len(df_test)\n",
    ")\n",
    "\n",
    "print(\"Accuracy: \",test_acc.item())\n",
    "print(\"F1-Macro: \",test_f1.item())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 49,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 605
    },
    "executionInfo": {
     "elapsed": 28733,
     "status": "ok",
     "timestamp": 1695329298142,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "F25gPUqgvw0w",
    "outputId": "747d6857-a67e-4f00-f9a9-fea35eee0d06",
    "tags": []
   },
   "outputs": [
    {
     "data": {
      "image/png": "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0c3IyNBbb72lYcOG1du2cuVKde3atdHHiouLk9PprDdeWlp63AvRli5dqrCwMF188cV1+7vdbnk8HjmdTkVERDT6m3Asm82mbt26NXm/1urY38J79uwph8NhYBoAwcRZttYr1D/Pea+1XoG817Kzsxv92ICK7qRJk3TfffeptLRUl112mZKTk3Xo0CGtWLFC77//vubOndvoY2VmZtabi+tyuZSXl1dv7u6xdu3apT179mjw4MH1tg0YMED333+/rr766sa/qP+yWCyKiopq8n6tVVjYT9czRkZGNmntYwAntmP/faN1CfXPc95rrVcg77Wm3LQsoKJ7xRVXqLCwUE888UTdygher1cRERG67bbbjltQf27YsGF64oknVFxcrMTERElHbv7gcrk0fPjwBve78cYbddlll/mM/fOf/1Rubq7mzZunLl26NP2FAQAAwDQCXkf397//va655hpt2rRJhw8fVkJCgk4//XTFxsY26TgTJkzQ888/r+nTp2v69OkqLCzU/PnzNXbsWJ+pC/fcc4+WL1+u7du3SzpyJvjnUxvefPNNHThwQIMGDQr0ZQEAAMAkmnVntNjYWL/zdJsiLi5OS5cu1dy5czVz5kxFRERozJgxmjVrls/jPB6Pamtrm/VcAAAAaD0CKrqvv/669u3bp5kzZ9bbtnDhQnXs2FGXXnppo4+XkZGhxYsXH/cx8+fP1/z583/xMQAAAIAU4J3RnnvuOb9r30pHlhdbtmxZs0IBAAAAzRVQ0d2zZ4969Ojhd1tmZqb27NnTrFAAAABAcwW8nkdpaanf8bKyMubSAgAAwHABFd2ePXtq5cqVfre9/fbbDZ7tBQAAAEIloKI7ceJErVq1SnfddZe+/fZbHThwQN9++63uvvtuvffee7r22muDnRMAAABokoBWXRg7dqx27dqlf/7zn1qxYoWkIzeMsFqtmjZtmi655JKghgQAAACaKuB1dG+55RaNHz9en376qYqLi5WUlKShQ4cqPT09mPkAAACAgDTr5tIdOnTQ6NGjVVZWprVr12rRokXKzs4OVjYAAAAgYI0+o/u///u/euedd/Thhx/WjVVUVOiKK67Qjz/+KK/XK0lauXKlXnvtNXXt2jXoYQEAAIDGavQZ3U2bNumiiy7yGXv++ef1ww8/aPLkydq4caNefvllRUVF6amnngp6UAAAAKAp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",
      "text/plain": [
       "<Figure size 800x600 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "pp,pn,np,nn = eval_scores_average(\n",
    "  model,\n",
    "  test_data_loader,\n",
    "  device,\n",
    ")\n",
    "\n",
    "box_plot(pp,pn,np,nn)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "3NU7ySjLiPKH"
   },
   "source": [
    "Foreign Injection Accuracy"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 50,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 27580,
     "status": "ok",
     "timestamp": 1695329325716,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "zlbzZWECocpp",
    "outputId": "27337f4e-b8c6-4b92-a088-77f03f0ccd93",
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy:  0.8544\n",
      "F1-Macro:  0.849606023418898\n"
     ]
    }
   ],
   "source": [
    "import numpy as np\n",
    "\n",
    "\n",
    "\n",
    "p_emb = torch.tensor([[1.]*config.hidden_size]).to(device)\n",
    "n_emb = torch.tensor([[1.]*config.hidden_size]).to(device)\n",
    "\n",
    "for i in new_positive_word_meaning_sentences:\n",
    "  p_emb = torch.cat((p_emb,get_cls(i).unsqueeze(dim=0)), dim=0)\n",
    "\n",
    "for i in new_negative_word_meaning_sentences:\n",
    "  n_emb = torch.cat((n_emb,get_cls(i).unsqueeze(dim=0)), dim=0)\n",
    "\n",
    "p_emb = p_emb[1:]\n",
    "n_emb = n_emb[1:]\n",
    "\n",
    "test_acc, _, test_f1 = eval_model(\n",
    "  model,\n",
    "  test_data_loader,\n",
    "  loss_fn,\n",
    "  device,\n",
    "  len(df_test)\n",
    ")\n",
    "\n",
    "print(\"Accuracy: \",test_acc.item())\n",
    "print(\"F1-Macro: \",test_f1.item())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 51,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy:  0.8496\n",
      "F1-Macro:  0.8446962088446242\n"
     ]
    }
   ],
   "source": [
    "#Reverse Foreign Injection\n",
    "\n",
    "import numpy as np\n",
    "\n",
    "\n",
    "\n",
    "p_emb = torch.tensor([[1.]*config.hidden_size]).to(device)\n",
    "n_emb = torch.tensor([[1.]*config.hidden_size]).to(device)\n",
    "\n",
    "for i in new_negative_word_meaning_sentences:\n",
    "  p_emb = torch.cat((p_emb,get_cls(i).unsqueeze(dim=0)), dim=0)\n",
    "\n",
    "for i in new_positive_word_meaning_sentences:\n",
    "  n_emb = torch.cat((n_emb,get_cls(i).unsqueeze(dim=0)), dim=0)\n",
    "\n",
    "p_emb = p_emb[1:]\n",
    "n_emb = n_emb[1:]\n",
    "\n",
    "test_acc, _, test_f1 = eval_model(\n",
    "  model,\n",
    "  test_data_loader,\n",
    "  loss_fn,\n",
    "  device,\n",
    "  len(df_test)\n",
    ")\n",
    "\n",
    "print(\"Accuracy: \",test_acc.item())\n",
    "print(\"F1-Macro: \",test_f1.item())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 605
    },
    "executionInfo": {
     "elapsed": 29037,
     "status": "ok",
     "timestamp": 1695329354749,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "DGrasP1mxQrD",
    "outputId": "87f1fe38-fe1b-4d6c-bf39-3c3e36c56fae",
    "tags": []
   },
   "outputs": [],
   "source": [
    "pp,pn,np,nn = eval_scores_average(\n",
    "  model,\n",
    "  test_data_loader,\n",
    "  device,\n",
    ")\n",
    "\n",
    "box_plot(pp,pn,np,nn)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "id": "K-QhJy24rDkl",
    "tags": []
   },
   "outputs": [],
   "source": [
    "from parrot import Parrot\n",
    "import torch\n",
    "import warnings\n",
    "warnings.filterwarnings(\"ignore\")\n",
    "\n",
    "\n",
    "def random_state(seed):\n",
    "  torch.manual_seed(seed)\n",
    "  if torch.cuda.is_available():\n",
    "    torch.cuda.manual_seed_all(seed)\n",
    "\n",
    "random_state(1234)\n",
    "\n",
    "parrot = Parrot(model_tag=\"prithivida/parrot_paraphraser_on_T5\")\n",
    "clear_output()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "executionInfo": {
     "elapsed": 16,
     "status": "ok",
     "timestamp": 1695329438385,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "Ppy-K6l2gWtd",
    "tags": []
   },
   "outputs": [],
   "source": [
    "from IPython.display import clear_output"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "new_positive_word_meaning_sentences = ['an acknowledgement or a public welcome',\n",
    " 'true to the point',\n",
    " 'including new and interesting things, methods and ideas',\n",
    " \"it's surprising\",\n",
    " 'known to be real and what somebody claims it is and not a copy',\n",
    " 'beauty delights the senses or the mind',\n",
    " \"to make someone quiet and more relaxed especially when there's intense emotion or excitement\",\n",
    " 'pleasant and easily remembered',\n",
    " 'the personal power of a person to attract and impress other people',\n",
    " '(of a person or their behaviour) happy and cheerful',\n",
    " '(of ideas, thoughts, arguments, etc.) logical and well organized; easy to understand and clear',\n",
    " 'having a useful and helpful effect rather than being negative or with no purpose',\n",
    " 'used to show that you admire or approve of somebody/something because they are/it is fashionable, attractive and often different',\n",
    " 'attractive and pretty',\n",
    " 'brave enough to do something dangerous or unusual involving danger or taking risks',\n",
    " 'allow yourself to use language and express your views well especially when speaking in public',\n",
    " 'feeling excitement or showing an interest about someone  something',\n",
    " 'without flaws and therefore perfect',\n",
    " 'funny with a sense of humor',\n",
    " 'providing exciting new ideas; making somebody want to create something, especially in art, literature or music' \n",
    " ]\n",
    "\n",
    "len(new_positive_word_meaning_sentences)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "new_negative_word_meaning_sentences = ['that can be understood in more than one way; having different meanings',\n",
    " 'having strong feelings about something you really dislike or an unfair situation',\n",
    " 'making somebody feel slightly angry',\n",
    " 'extremely bad, especially from a moral point of view',\n",
    " 'very bad or uncomfortable',\n",
    " 'cruel and violent and not what one wants from people who suss out education and respect each other',\n",
    " 'very strange or unusual',\n",
    " '(of behaviour or language) showing a lack of respect for God or religion',\n",
    " 'stupid; not able to think or talk in an intelligent way',\n",
    " 'without any order; in a completely confused state',\n",
    " 'containing or showing a lack of agreement between statements, facts, opinions or actions',\n",
    " 'public discussion and argument about something that many people strongly disagree about, think is bad, or are shocked by',\n",
    " 'to feel very embarrassed about something',\n",
    " 'a desire to cause psychological or physical pain',\n",
    " \"treating someone as if they don't have any value so that they lose their self-respect and respect for other people\",\n",
    " 'mentally ill particularly because of very unpleasant or extremely unhappy experiences',\n",
    " 'not doing well',\n",
    " 'not what somebody claims it is; appearing to be something it is not',\n",
    " \"a strange trick or unnecessary device that's meant to appeal to people or to get them to buy something\",\n",
    " 'very unfriendly or unpleasant'\n",
    " ]\n",
    "\n",
    "len(new_negative_word_meaning_sentences)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "executionInfo": {
     "elapsed": 38794,
     "status": "ok",
     "timestamp": 1695329477164,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "_OCeDQY6gHMj",
    "tags": []
   },
   "outputs": [],
   "source": [
    "'''\n",
    "new_positive_word_meaning_sentences = []\n",
    "\n",
    "for i in positive_word_meaning_sentences:\n",
    "  print(i)\n",
    "  para_phrases = parrot.augment(input_phrase=i, max_return_phrases = 10)\n",
    "  temp = list(list(zip(*para_phrases))[0])[0]\n",
    "  new_positive_word_meaning_sentences.append(temp)\n",
    "\n",
    "\n",
    "new_negative_word_meaning_sentences = []\n",
    "\n",
    "for i in negative_word_meaning_sentences:\n",
    "  print(i)\n",
    "  para_phrases = parrot.augment(input_phrase=i, max_return_phrases = 10)\n",
    "  temp = list(list(zip(*para_phrases))[0])[0]\n",
    "  new_negative_word_meaning_sentences.append(temp)\n",
    "\n",
    "clear_output()\n",
    "'''\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "oEdEBFPQiasC"
   },
   "source": [
    "Paraphrased Native Injection Accuracy"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 30045,
     "status": "ok",
     "timestamp": 1695329507205,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "uEuhYVF-gMqC",
    "outputId": "3c4d554a-9320-4704-990c-0a6f0b392083",
    "tags": []
   },
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "p_emb = torch.tensor([[1.]*config.hidden_size]).to(device)\n",
    "n_emb = torch.tensor([[1.]*config.hidden_size]).to(device)\n",
    "\n",
    "for i in new_positive_word_meaning_sentences:\n",
    "  p_emb = torch.cat((p_emb,get_cls(i).unsqueeze(dim=0)), dim=0)\n",
    "\n",
    "for i in new_negative_word_meaning_sentences:\n",
    "  n_emb = torch.cat((n_emb,get_cls(i).unsqueeze(dim=0)), dim=0)\n",
    "\n",
    "p_emb = p_emb[1:]\n",
    "n_emb = n_emb[1:]\n",
    "\n",
    "test_acc, _, test_f1 = eval_model(\n",
    "  model,\n",
    "  test_data_loader,\n",
    "  loss_fn,\n",
    "  device,\n",
    "  len(df_test)\n",
    ")\n",
    "\n",
    "\n",
    "\n",
    "print(\"Accuracy: \",test_acc.item())\n",
    "print(\"F1-Macro: \",test_f1.item())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "model.load_state_dict(torch.load('/home/m_nsu/ICLR/bert_best_model_state.bin'))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "executionInfo": {
     "elapsed": 13,
     "status": "ok",
     "timestamp": 1695329507206,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "2qBYfr3zgvAs",
    "tags": []
   },
   "outputs": [],
   "source": [
    "import random\n",
    "random.seed(42)\n",
    "\n",
    "inputNumbers = range(0,10000)\n",
    "random_seeds = random.sample(inputNumbers, 300)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "executionInfo": {
     "elapsed": 11,
     "status": "ok",
     "timestamp": 1695329507206,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "nysMxY5myqsd",
    "tags": []
   },
   "outputs": [],
   "source": [
    "import numpy as np"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "JuMwkV_v4GvV",
    "outputId": "24e92f01-d79f-4514-f879-14a50b1f8477",
    "tags": []
   },
   "outputs": [],
   "source": [
    "f1s = []\n",
    "accuracy = []\n",
    "p_word_list = []\n",
    "n_word_list = []\n",
    "\n",
    "print(\"Working\", end=' ')\n",
    "for i in range(300):\n",
    "    #SELECT RANDOM WORDS FROM NEUTRAL SENTIWORD\n",
    "    df_neutral = df_senti[df_senti['Sentiment']=='Neutral'].sample(n=20, random_state=random_seeds[i])\n",
    "    neutral_words = list(df_neutral[\"Word\"])\n",
    "\n",
    "    #FETCH DEFINITIONS OF THOSE WORDS\n",
    "    neutral_word_meaning_sentences = []\n",
    "    for i in neutral_words:\n",
    "        a = list(df_senti.loc[df_senti['Word'] == i, 'Definition'])[0]\n",
    "        neutral_word_meaning_sentences.append(a)\n",
    "\n",
    "    #MAKE EMBEDDING OF THESE DEFINITIONS\n",
    "    neutral_p_emb = torch.tensor([[1.]*config.hidden_size]).to(device)\n",
    "    neutral_n_emb = torch.tensor([[1.]*config.hidden_size]).to(device)\n",
    "\n",
    "\n",
    "    for i in neutral_word_meaning_sentences[0:10]:\n",
    "      neutral_p_emb = torch.cat((neutral_p_emb,get_cls(i).unsqueeze(0)), dim=0)\n",
    "\n",
    "    for i in neutral_word_meaning_sentences[10:20]:\n",
    "      neutral_n_emb = torch.cat((neutral_n_emb,get_cls(i).unsqueeze(0)), dim=0)\n",
    "\n",
    "    p_emb = neutral_p_emb[1:]\n",
    "    n_emb = neutral_n_emb[1:]\n",
    "\n",
    "    print('...', end=' ')\n",
    "\n",
    "    p_samples = neutral_words[0:10]\n",
    "    n_samples = neutral_words[10:20]\n",
    "\n",
    "    #EVALUATE ON THESE WORDS\n",
    "    test_acc,_, test_f1 = eval_model(\n",
    "      model,\n",
    "      test_data_loader,\n",
    "      loss_fn,\n",
    "      device,\n",
    "      len(df_test),\n",
    "    )\n",
    "\n",
    "    accuracy.append(test_acc.item())\n",
    "    f1s.append(test_f1.item())\n",
    "    p_word_list.append(p_samples)\n",
    "    n_word_list.append(n_samples)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "executionInfo": {
     "elapsed": 13,
     "status": "ok",
     "timestamp": 1695379584055,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "36UtUdOcDjLK",
    "tags": []
   },
   "outputs": [],
   "source": [
    "import statistics"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 245
    },
    "executionInfo": {
     "elapsed": 13,
     "status": "error",
     "timestamp": 1695379584056,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "N_1_hH8S4Hit",
    "outputId": "85d5684e-8de1-402e-9386-f53b050bd477",
    "tags": []
   },
   "outputs": [],
   "source": [
    "stdv = statistics.pstdev(f1s)\n",
    "median = statistics.median(f1s)\n",
    "#filters = [i>median+(2*stdv) for i in f1s]\n",
    "filters = [i>.7 for i in f1s]\n",
    "def filter_2std(target_list, filters):\n",
    "    index = [i for i in range(len(filters)) if filters[i]]\n",
    "    list_a_filtered = [target_list[i] for i in index]\n",
    "    return list_a_filtered\n",
    "\n",
    "p_words_spurious = filter_2std(p_word_list,filters)\n",
    "n_words_spurious = filter_2std(n_word_list,filters)\n",
    "\n",
    "#print(p_words_spurious)\n",
    "#print(n_words_spurious)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "6l78c6yoikRS"
   },
   "source": [
    "Neutral Injection Accuracies"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "executionInfo": {
     "elapsed": 19,
     "status": "ok",
     "timestamp": 1695379585268,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "mmW3cTd4o6Hb",
    "tags": []
   },
   "outputs": [],
   "source": [
    "import numpy as np"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 175
    },
    "executionInfo": {
     "elapsed": 17,
     "status": "error",
     "timestamp": 1695379585269,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "_sc9N6PqFDAF",
    "outputId": "86645206-6fd6-4dcd-eb48-1555f0dcda27",
    "tags": []
   },
   "outputs": [],
   "source": [
    "(hist, bin_edges) = np.histogram(f1s,bins=15)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "executionInfo": {
     "elapsed": 15,
     "status": "aborted",
     "timestamp": 1695379585271,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "uMGPmojj4SeF",
    "tags": []
   },
   "outputs": [],
   "source": [
    "plt.hist(f1s, bins=15,facecolor = '#008080', edgecolor='#000000', linewidth=0.5)\n",
    "\n",
    "\n",
    "vertical_line_position = 0.846  # Change this to the desired position\n",
    "plt.axvline(x=vertical_line_position, color='r', linestyle='--', label='Vne')\n",
    "plt.text(x=vertical_line_position,y=max(hist),s='Native',ha='left', va='bottom')\n",
    "\n",
    "\n",
    "    \n",
    "plt.xlabel(\"F1-Macro\")\n",
    "plt.ylabel(\"frequency\")\n",
    "\n",
    "#plt.title(\"bert-base\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "'''\n",
    "df_randomization_test = pd.DataFrame(list(zip(f1s, accuracy,p_word_list,n_word_list)),\n",
    "               columns =['f1s', 'accuracy','p_word_list','n_word_list'])\n",
    "df_randomization_test.to_csv('bert_imdb_randomizationtest.csv')\n",
    "'''"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "df_neutral = df_senti[df_senti['Sentiment']=='Neutral'].sample(n=20, random_state=random_seeds[16])\n",
    "neutral_words = list(df_neutral[\"Word\"])\n",
    "\n",
    "#FETCH DEFINITIONS OF THOSE WORDS\n",
    "neutral_word_meaning_sentences = []\n",
    "for i in neutral_words:\n",
    "    a = list(df_senti.loc[df_senti['Word'] == i, 'Definition'])[0]\n",
    "    neutral_word_meaning_sentences.append(a)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "executionInfo": {
     "elapsed": 33,
     "status": "aborted",
     "timestamp": 1695329093833,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "-nnwzm980k1q",
    "tags": []
   },
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "p_emb = torch.tensor([[1.]*config.hidden_size]).to(device)\n",
    "n_emb = torch.tensor([[1.]*config.hidden_size]).to(device)\n",
    "\n",
    "for i in neutral_word_meaning_sentences[0:10]:\n",
    "  p_emb = torch.cat((p_emb,get_cls(i).unsqueeze(dim=0)), dim=0)\n",
    "\n",
    "for i in neutral_word_meaning_sentences[10:20]:\n",
    "  n_emb = torch.cat((n_emb,get_cls(i).unsqueeze(dim=0)), dim=0)\n",
    "\n",
    "p_emb = p_emb[1:]\n",
    "n_emb = n_emb[1:]\n",
    "\n",
    "test_acc, _, test_f1 = eval_model(\n",
    "  model,\n",
    "  test_data_loader,\n",
    "  loss_fn,\n",
    "  device,\n",
    "  len(df_test)\n",
    ")\n",
    "\n",
    "\n",
    "print(\"Accuracy: \",test_acc.item())\n",
    "print(\"F1-Macro: \",test_f1.item())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "executionInfo": {
     "elapsed": 32,
     "status": "aborted",
     "timestamp": 1695329093833,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "7cdh2kXd0-Zb",
    "tags": []
   },
   "outputs": [],
   "source": [
    "pp,pn,np,nn = eval_scores_average(\n",
    "  model,\n",
    "  test_data_loader,\n",
    "  device,\n",
    ")\n",
    "\n",
    "box_plot(pp,pn,np,nn)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "executionInfo": {
     "elapsed": 32,
     "status": "aborted",
     "timestamp": 1695329093833,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "1MTPlFIVEd6y",
    "tags": []
   },
   "outputs": [],
   "source": [
    "max_f1_index = 0"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "executionInfo": {
     "elapsed": 32,
     "status": "aborted",
     "timestamp": 1695329093834,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "J3ymzSst9jH9",
    "tags": []
   },
   "outputs": [],
   "source": [
    "p_spurious_sentences = [list(df_senti[df_senti['Word'] == i]['Definition'])[0] for i in p_words_spurious[max_f1_index]]\n",
    "n_spurious_sentences = [list(df_senti[df_senti['Word'] == i]['Definition'])[0] for i in n_words_spurious[max_f1_index]]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "executionInfo": {
     "elapsed": 32,
     "status": "aborted",
     "timestamp": 1695329093834,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "AoDiCBpM6glm",
    "tags": []
   },
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "p_emb = torch.tensor([[1.]*config.hidden_size]).to(device)\n",
    "n_emb = torch.tensor([[1.]*config.hidden_size]).to(device)\n",
    "\n",
    "for i in p_spurious_sentences:\n",
    "  p_emb = torch.cat((p_emb,get_cls(i).unsqueeze(dim=0)), dim=0)\n",
    "\n",
    "for i in n_spurious_sentences:\n",
    "  n_emb = torch.cat((n_emb,get_cls(i).unsqueeze(dim=0)), dim=0)\n",
    "\n",
    "p_emb = p_emb[1:]\n",
    "n_emb = n_emb[1:]\n",
    "\n",
    "test_acc, _, test_f1 = eval_model(\n",
    "  model,\n",
    "  test_data_loader,\n",
    "  loss_fn,\n",
    "  device,\n",
    "  len(df_test)\n",
    ")\n",
    "\n",
    "print(\"Accuracy: \",test_acc.item())\n",
    "print(\"F1-Macro: \",test_f1.item())\n",
    "\n",
    "\n",
    "pp,pn,np,nn = eval_scores_average(\n",
    "  model,\n",
    "  test_data_loader,\n",
    "  device,\n",
    ")\n",
    "\n",
    "box_plot(pp,pn,np,nn)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "executionInfo": {
     "elapsed": 32,
     "status": "aborted",
     "timestamp": 1695329093835,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "t-1o_fPd_MEy",
    "tags": []
   },
   "outputs": [],
   "source": [
    "n_words_spurious[0]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "executionInfo": {
     "elapsed": 33,
     "status": "aborted",
     "timestamp": 1695329093836,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "lvMoMpyEXQVt",
    "tags": []
   },
   "outputs": [],
   "source": [
    "p_words_spurious[0]"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "aUzQP5etB1qH"
   },
   "source": [
    "Using Bias categories to test if they are neutral or not"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "executionInfo": {
     "elapsed": 32,
     "status": "aborted",
     "timestamp": 1695329093836,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "M4c1i-jEnOia",
    "tags": []
   },
   "outputs": [],
   "source": [
    "import os, sys\n",
    "\n",
    "filenames = []\n",
    "\n",
    "path = \"/home/m_nsu/ICLR/Datasets/GoEmotion/bias_criterias\"\n",
    "dir = os.listdir( path )\n",
    "\n",
    "for file in dir:\n",
    "   if(file.endswith('.csv')):\n",
    "    filenames.append(file)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "executionInfo": {
     "elapsed": 32,
     "status": "aborted",
     "timestamp": 1695329093836,
     "user": {
      "displayName": "m m",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "uAcBqk9LpYds",
    "tags": []
   },
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "for filename in filenames:\n",
    "    df_bias = pd.read_csv('/home/m_nsu/ICLR/Datasets/GoEmotion/bias_criterias/'+filename)\n",
    "\n",
    "    p_spurious_sentences = list(df_bias['Definition'])\n",
    "\n",
    "    p_emb = torch.tensor([[1.]*config.hidden_size]).to(device)\n",
    "    n_emb = torch.tensor([[1.]*config.hidden_size]).to(device)\n",
    "\n",
    "    for i in p_spurious_sentences:\n",
    "      p_emb = torch.cat((p_emb,get_cls(i).unsqueeze(dim=0)), dim=0)\n",
    "\n",
    "    for i in neutral_word_meaning_sentences[10:20]:\n",
    "      n_emb = torch.cat((n_emb,get_cls(i).unsqueeze(dim=0)), dim=0)\n",
    "\n",
    "    p_emb = p_emb[1:]\n",
    "    n_emb = n_emb[1:]\n",
    "\n",
    "    test_acc, _, test_f1 = eval_model(\n",
    "      model,\n",
    "      test_data_loader,\n",
    "      loss_fn,\n",
    "      device,\n",
    "      len(df_test)\n",
    "    )\n",
    "\n",
    "    positive_spurious = test_f1.item()\n",
    "\n",
    "    n_spurious_sentences = list(df_bias['Definition'])\n",
    "    p_emb = torch.tensor([[1.]*config.hidden_size]).to(device)\n",
    "    n_emb = torch.tensor([[1.]*config.hidden_size]).to(device)\n",
    "\n",
    "    for i in neutral_word_meaning_sentences[0:10]:\n",
    "      p_emb = torch.cat((p_emb,get_cls(i).unsqueeze(dim=0)), dim=0)\n",
    "\n",
    "    for i in n_spurious_sentences:\n",
    "      n_emb = torch.cat((n_emb,get_cls(i).unsqueeze(dim=0)), dim=0)\n",
    "\n",
    "    p_emb = p_emb[1:]\n",
    "    n_emb = n_emb[1:]\n",
    "\n",
    "    test_acc, _, test_f1 = eval_model(\n",
    "      model,\n",
    "      test_data_loader,\n",
    "      loss_fn,\n",
    "      device,\n",
    "      len(df_test)\n",
    "    )\n",
    "\n",
    "    negative_spurious = test_f1.item()\n",
    "\n",
    "    if positive_spurious>=0.6:\n",
    "      print(filename+' is Positively Spurious.')\n",
    "    if negative_spurious>=0.6:\n",
    "      print(filename+' is Negatively Spurious.')\n",
    "    if(positive_spurious<0.6 and negative_spurious<0.6):\n",
    "      print(filename+' is Neutral.')\n",
    "    print('Positive Acc:',positive_spurious,'\\n'+'Negative Acc:', negative_spurious)\n",
    "    print(\"------------------------------------------------------------------------\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "accelerator": "GPU",
  "colab": {
   "provenance": []
  },
  "gpuClass": "standard",
  "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.10"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
