{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {
    "executionInfo": {
     "elapsed": 5,
     "status": "ok",
     "timestamp": 1695324378707,
     "user": {
      "displayName": "Mostafa Mushsharat",
      "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 XLNetModel, XLNetConfig, XLNetTokenizer\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": "Mostafa Mushsharat",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "MLk4JWizs4S9",
    "tags": []
   },
   "outputs": [],
   "source": [
    "df_train = pd.read_csv('/home/mostafa_nsu/ICLR/Datasets/IMDB/train.csv')\n",
    "df_val = pd.read_csv('/home/mostafa_nsu/ICLR/Datasets/IMDB/val.csv')\n",
    "df_test = pd.read_csv('/home/mostafa_nsu/ICLR/Datasets/IMDB/test.csv')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {
    "id": "YWiFI0a9QqEa",
    "tags": []
   },
   "outputs": [],
   "source": [
    "df_senti = pd.read_excel('/home/mostafa_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": "Mostafa Mushsharat",
      "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": "Mostafa Mushsharat",
      "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": "Mostafa Mushsharat",
      "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')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 18,
     "status": "ok",
     "timestamp": 1695323756840,
     "user": {
      "displayName": "Mostafa Mushsharat",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "6oWORty0p8Xo",
    "outputId": "cb12350b-753a-4c61-c7e8-3c9427f000e9",
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "cuda\n"
     ]
    }
   ],
   "source": [
    "PRE_TRAINED_MODEL_NAME = 'xlnet-large-cased'\n",
    "config = XLNetConfig.from_pretrained(PRE_TRAINED_MODEL_NAME)\n",
    "tokenizer = XLNetTokenizer.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 = XLNetModel.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": "Mostafa Mushsharat",
      "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": "Mostafa Mushsharat",
      "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": "Mostafa Mushsharat",
      "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": "Mostafa Mushsharat",
      "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": "Mostafa Mushsharat",
      "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, 1024])\n",
      "torch.Size([20, 1024])\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": "Mostafa Mushsharat",
      "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": "Mostafa Mushsharat",
      "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": "Mostafa Mushsharat",
      "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": "Mostafa Mushsharat",
      "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": 19,
   "metadata": {
    "executionInfo": {
     "elapsed": 3,
     "status": "ok",
     "timestamp": 1695325688712,
     "user": {
      "displayName": "Mostafa Mushsharat",
      "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 = XLNetModel.from_pretrained(PRE_TRAINED_MODEL_NAME,config=config)\n",
    "    self.FC = nn.Linear(config.hidden_size,config.hidden_size, bias=False)\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 = self.FC(pooled_output)\n",
    "\n",
    "    Ept = self.FC(p_emb)\n",
    "    Ent = self.FC(n_emb)\n",
    "\n",
    "    positive = F.relu(self.CosineNorm(x_sent, Ept, p_emb.size()[0]))\n",
    "    negative = F.relu(self.CosineNorm(x_sent, 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": 20,
   "metadata": {
    "id": "HWZ37gsztWzL",
    "tags": []
   },
   "outputs": [],
   "source": [
    "model = Classifier()\n",
    "model = model.to(device)\n",
    "clear_output()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "metadata": {
    "executionInfo": {
     "elapsed": 22,
     "status": "ok",
     "timestamp": 1695325221207,
     "user": {
      "displayName": "Mostafa Mushsharat",
      "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": 22,
   "metadata": {
    "executionInfo": {
     "elapsed": 20,
     "status": "ok",
     "timestamp": 1695325221207,
     "user": {
      "displayName": "Mostafa Mushsharat",
      "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": 23,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 21,
     "status": "ok",
     "timestamp": 1695325221208,
     "user": {
      "displayName": "Mostafa Mushsharat",
      "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": 24,
   "metadata": {
    "executionInfo": {
     "elapsed": 12,
     "status": "ok",
     "timestamp": 1695325221208,
     "user": {
      "displayName": "Mostafa Mushsharat",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "NwvA-zp7vqc1",
    "tags": []
   },
   "outputs": [],
   "source": [
    "#del test\n",
    "torch.cuda.empty_cache()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "metadata": {
    "executionInfo": {
     "elapsed": 603,
     "status": "ok",
     "timestamp": 1695325221800,
     "user": {
      "displayName": "Mostafa Mushsharat",
      "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": 26,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 15,
     "status": "ok",
     "timestamp": 1695325221800,
     "user": {
      "displayName": "Mostafa Mushsharat",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "NHFM3QUhg3n0",
    "outputId": "602bd955-db78-44a6-931e-e5e901bb1856",
    "tags": []
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([[0.9057, 0.9000],\n",
       "        [0.9820, 0.9839],\n",
       "        [0.9839, 0.9876],\n",
       "        [0.9820, 0.9835],\n",
       "        [0.9812, 0.9827],\n",
       "        [0.9842, 0.9850],\n",
       "        [0.9826, 0.9856],\n",
       "        [0.9837, 0.9846],\n",
       "        [0.9829, 0.9833],\n",
       "        [0.9823, 0.9837],\n",
       "        [0.9840, 0.9859],\n",
       "        [0.9839, 0.9870],\n",
       "        [0.9845, 0.9866],\n",
       "        [0.9829, 0.9845],\n",
       "        [0.9792, 0.9788],\n",
       "        [0.9831, 0.9850],\n",
       "        [0.9838, 0.9858],\n",
       "        [0.9790, 0.9812],\n",
       "        [0.9833, 0.9843],\n",
       "        [0.9828, 0.9842],\n",
       "        [0.9812, 0.9844],\n",
       "        [0.9844, 0.9873],\n",
       "        [0.9835, 0.9837],\n",
       "        [0.9827, 0.9848],\n",
       "        [0.9787, 0.9778],\n",
       "        [0.9840, 0.9851],\n",
       "        [0.9827, 0.9848],\n",
       "        [0.9768, 0.9782],\n",
       "        [0.9836, 0.9858],\n",
       "        [0.9844, 0.9866],\n",
       "        [0.9358, 0.9432],\n",
       "        [0.9808, 0.9823]], device='cuda:0', grad_fn=<TBackward0>)"
      ]
     },
     "execution_count": 26,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "outs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 1022,
     "status": "ok",
     "timestamp": 1695325845248,
     "user": {
      "displayName": "Mostafa Mushsharat",
      "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": 28,
   "metadata": {
    "executionInfo": {
     "elapsed": 8,
     "status": "ok",
     "timestamp": 1695325845969,
     "user": {
      "displayName": "Mostafa Mushsharat",
      "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": 29,
   "metadata": {
    "executionInfo": {
     "elapsed": 9,
     "status": "ok",
     "timestamp": 1695325845971,
     "user": {
      "displayName": "Mostafa Mushsharat",
      "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": 30,
   "metadata": {
    "executionInfo": {
     "elapsed": 8,
     "status": "ok",
     "timestamp": 1695325846655,
     "user": {
      "displayName": "Mostafa Mushsharat",
      "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": 31,
   "metadata": {
    "executionInfo": {
     "elapsed": 8,
     "status": "ok",
     "timestamp": 1695325846656,
     "user": {
      "displayName": "Mostafa Mushsharat",
      "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": 36,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 1000
    },
    "executionInfo": {
     "elapsed": 3203929,
     "status": "ok",
     "timestamp": 1695329051836,
     "user": {
      "displayName": "Mostafa Mushsharat",
      "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.4466726026296616 accuracy 0.8626250000000001\n",
      "Val   loss 0.4366651150830992 accuracy 0.8726\n",
      "\n",
      "Epoch 2/8\n",
      "----------\n",
      "Train loss 0.44399996082782744 accuracy 0.864225\n",
      "Val   loss 0.43742913558224966 accuracy 0.8694000000000001\n",
      "\n",
      "Epoch 3/8\n",
      "----------\n",
      "Train loss 0.4401361615180969 accuracy 0.8678750000000001\n",
      "Val   loss 0.43708163272043704 accuracy 0.8694000000000001\n",
      "\n",
      "Epoch 4/8\n",
      "----------\n",
      "Train loss 0.43742296583652496 accuracy 0.869225\n",
      "Val   loss 0.43571257534300445 accuracy 0.8694000000000001\n",
      "\n",
      "CPU times: user 2h 30min 25s, sys: 14.2 s, total: 2h 30min 39s\n",
      "Wall time: 1h 59min 32s\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(4):\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(), 'xlnet_best_model_state.bin')\n",
    "    best_accuracy = val_acc"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 37,
   "metadata": {
    "executionInfo": {
     "elapsed": 51,
     "status": "ok",
     "timestamp": 1695329137842,
     "user": {
      "displayName": "Mostafa Mushsharat",
      "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": 38,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 480
    },
    "executionInfo": {
     "elapsed": 2888,
     "status": "ok",
     "timestamp": 1695329143450,
     "user": {
      "displayName": "Mostafa Mushsharat",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "aUQbxyTEAPhM",
    "outputId": "b103ca88-1886-4f16-ea21-5b966a08fe7c",
    "tags": []
   },
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAjcAAAHFCAYAAAAOmtghAAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAAPYQAAD2EBqD+naQAAADh0RVh0U29mdHdhcmUAbWF0cGxvdGxpYiB2ZXJzaW9uMy4xLjIsIGh0dHA6Ly9tYXRwbG90bGliLm9yZy8li6FKAAAgAElEQVR4nOzde1wU9f7H8feAclfyCpg38hqapmKKRqYmJmXZqbSroN3ISo3Mk3lOXrKwTpqpSTfRSk+ZqR07WUZpamqZBlqJl59haOIhNEGsUGF+fxgby6LsIrAwvp6Pxz7a/c53Zj7zdXLfzszOGKZpmgIAALAID3cXAAAAUJEINwAAwFIINwAAwFIINwAAwFIINwAAwFIINwAAwFIINwAAwFIINwAAwFIINwAAwFIIN4CFGIbh1OuLL76okPX98ccfMgxD06dPL9f8PXv21LXXXlshtbgqODhYt9xyS5n9PvnkExmGoa+++sql5c+ePVuLFi0qb3kAzkMtdxcAoOJs3rzZ7vPTTz+ttWvXas2aNXbtYWFhFbI+b29vbd68Wc2bNy/X/PPnz5enp2eF1FJZIiIitHnzZnXs2NGl+WbPnq3WrVvrrrvuqqTKAJwN4QawkJ49e9p9btSokTw8PBzaz+bkyZPy9PR0OnAYhuH0skvToUOHcs9bVQIDA89rGytSQUGBCgoK5OXl5e5SgGqN01LABarodMuSJUs0evRohYSEyMfHRwcOHFBmZqbi4uJ06aWXyt/fX0FBQbrmmmscjgyVdlrqlVdekWEY2rhxo+677z41aNBADRs21K233qr//e9/dvOXPC21a9cuGYahOXPm6LnnnlOLFi0UEBCg3r17a9u2bQ7bMG/ePLVu3Vre3t667LLLtHTpUt12221q37690+Pw4Ycf6vLLL5evr6/CwsIcTiWVdlpqz549uvXWWxUSEiJvb28FBwdrwIAB+uGHHySdOeW1b98+rV692nYqsHhN6enpuv3229WoUSN5e3srLCxMs2fPVvHnGBeNxaxZszR58mS1bNlSXl5eWrVqlQICAjRmzBiHbdm9e7c8PDw0Z84cp7cfsCKO3AAXuMcee0xXXXWV3njjDRUWFqpevXrKyMhQ7dq1NWXKFAUFBen48eNaunSpIiMjtWHDBkVERJS53JiYGN1www165513lJ6ervHjx2vEiBFatWpVmfPOnDlTl112mebMmaOCggJNnDhRgwYNUnp6uvz9/SWdOe0zZswY3XbbbZo9e7aOHj2qCRMm6NSpU/L19XVq27/55hvt3r1bTzzxhBo2bKjExETdfffdatu2ra644opS5zFNU9dee628vb31wgsvqFmzZsrOztaGDRt07NgxSdKqVat04403qmnTpnrxxRclyVZTZmamIiIiZBiGEhIS1LRpU33wwQcaM2aM9u/fr5kzZ9qt74UXXlBYWJhmzpypgIAAhYWFafjw4XrzzTf17LPP2sZDkl5++WX5+/srJibGqe0HLMsEYFkxMTGmv79/qdM+/vhjU5IZFRVV5nJOnz5tnjp1yuzdu7d5++2329p///13U5KZkJBga0tMTDQlmfHx8XbLmDp1qinJPHr0qK2tR48e5sCBA22f09LSTElmeHi4WVhYaGtfv369KclcsWKFaZqmefLkSbNBgwZmnz597Nbxf//3f6anp6fZrl27MrcpKCjI9Pf3Nw8dOmRry8vLM+vUqWOOGTPG1lY0Tps3bzZN0zQPHjxoSjJfeeWVcy6/VatWdttWZOzYsaZhGGZqaqpd+4gRI0wPDw8zPT3dbiwuvfRS8/Tp03Z909LSTMMwzMTERFvb8ePHzbp165oPPvhgmdsOWB2npYAL3M033+zQZpqm5syZoy5dusjHx0e1atVS7dq1tXHjRqWlpTm13BtuuMHuc6dOnSRJGRkZZc57/fXXyzAMh3l/+uknSdL333+vI0eOaOjQoXbztWrVSt27d3eqPknq3r27QkJCbJ/9/f3VqlUr23pKExwcrObNm+vZZ5/VSy+9pO3bt6uwsNDpda5Zs0ZdunRR586d7dpjY2NVWFjo8Eu2IUOGOFwD1b59ew0YMEAvv/yyre2tt95Sbm6uHnroIadrAayKcANc4Ip/uRdJSEjQ6NGjFRkZqeXLl+vrr7/WN998o379+un33393arkNGjSw++zt7S1JTs1f1rxHjhyRJAUFBTnMW1qbs+spWte5avT09NTatWvVt29fPfPMM7r88ssVFBSk+Ph4nThxosx1HjlypNQxb9KkiW16caX1laQxY8bo+++/1/r16yWdOSXVt2/fGnGRNlDZuOYGuMAVP0JSZNGiRbr22ms1e/Zsu/acnJyqKuucikJJyQuUJenw4cOVvv5LLrlECxculHTmwt93331XTz/9tAoLCzVr1qxzztugQQNlZmY6tB86dEiS1LBhQ7v20v58JGnQoEFq06aN5s6dq9OnT2vnzp2aOnVqObYGsB6O3ABwYBiG7WhJka1bt+rbb791U0X2OnbsqPr162vJkiV27fv27dPWrVurtJb27dtr8uTJatu2rd34nO0IUP/+/ZWammr7ZVWRt956Sx4eHrr66qudWq9hGHrkkUe0YsUKTZo0SU2bNtWQIUPOa1sAqyDcAHBw/fXX68MPP9S0adO0Zs0azZ07V9ddd51atmzp7tIkSbVr19akSZO0fv163X777fr444+1aNEiDRw4UE2aNJGHR+X91bZlyxb17dtXL7/8slavXq01a9boiSee0O7duzVgwABbv8suu0xbt27V+++/r61bt9rCzOOPP65GjRpp4MCBSkpK0urVq/XQQw9p/vz5Gjt2rFq0aOF0LbGxsfLz89OXX36puLi4an9DRKCqcFoKgIPJkyfr5MmTmjdvnp555hl17NhRCxYs0FtvvaXU1FR3lydJGj16tDw9PTVz5kwtX75cl1xyiaZMmaK3335bubm5lbbepk2bqnnz5pozZ44OHjwoDw8PtWrVSrNnz9aoUaNs/Z555hllZ2drxIgRysvLU7t27bRr1y6FhIRo8+bNmjBhgh5//HEdP35crVq10qxZszR69GiXaqlTp46io6O1fPly3XfffRW9qUCNZZhmsbtGAUANduTIEbVp00Z33XWXw/VCVvT777+refPmGjRokN566y13lwNUGxy5AVAjZWRkaObMmerTp4/q16+v9PR0zZgxQ/n5+XrkkUfcXV6lysrK0p49e/Tqq6/q119/1fjx491dElCtEG4A1Eg+Pj7au3ev3nnnHR09elQBAQHq1auXFi5cqDZt2ri7vEq1fPlyPfjgg7r44ov1+uuvu/xQT8DqOC0FAAAsxa2/llq/fr0GDx6sJk2ayDAMffDBB2XOs27dOnXr1k0+Pj665JJL9Morr1RBpQAAoKZwa7g5ceKEOnfurLlz5zrVPz09XdHR0YqMjFRKSoqefPJJjR49WsuWLavkSgEAQE1RbU5LGYahFStWnPMmVH//+9+1cuVKu2fbxMXFafv27dq8eXNVlAkAAKq5GnVB8ebNmxUVFWXXNnDgQM2fP1+nTp1S7dq1HebJz89Xfn6+7XNh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HdpbtGhh95BLAAAAd3A53DRu3Fg7duxwaN++fbsaNGhQIUUBAACUl8vh5rbbbtPo0aO1du1aFRQUqKCgQGvWrNGYMWN02223VUaNAAAATnP5p+DTpk3TTz/9pP79+6tWrTOzFxYWavjw4VxzAwAA3K7cPwXfs2ePtm/fLl9fX1122WVq0aJFRddWKfgpOAAANY8r398uH7kp0rZtW7Vt27a8swMAAFSKcoWbgwcPauXKlcrIyNDJkyftps2cObNCCgMAACgPl8PN559/rhtuuEGhoaHavXu3OnbsqP3798s0TXXt2rUyagQAAHCay7+WmjBhgh577DF9//338vHx0bJly3TgwAH16dNHt956a2XUCAAA4DSXw01aWppiYmIkSbVq1dLvv/+ugIAATZ06Vc8991yFFwgAAOAKl8ONv7+/8vPzJUlNmjTRvn37bNOys7MrrjIAAIBycPmam549e2rjxo0KCwvTddddp8cee0zfffedli9frp49e1ZGjQAAAE5zOdzMnDlTeXl5kqTJkycrLy9PS5YsUevWrfXiiy9WeIEAAACucCncFBQU6MCBA+rUqZMkyc/PT/PmzauUwgAAAMrDpWtuPD09NXDgQB07dqyy6gEAADgvLl9QfNlll+nHH3+sjFoAAADOm8vh5plnntG4ceP03//+V5mZmcrNzbV7AQAAuJPLD8708PgrDxmGYXtvmqYMw1BBQUHFVVcJeHAmAAA1T6U+OHPt2rXlLgwAAKCyuRxu+vTpUxl1AAAAVAiXw8369evPOf2qq64qdzEAAADny+Vwc/XVVzu0Fb/2prpfcwMAAKzN5V9L/frrr3avrKwsffLJJ+revbs+/fTTyqgRAADAaS4fuQkMDHRoGzBggLy9vfXoo49q27ZtFVIYAABAebh85OZsGjVqpN27d1fU4gAAAMrF5SM3O3bssPtsmqYyMzM1ffp0de7cucIKAwAAKA+Xw83ll18uwzBU8t5/PXv2VFJSUoUVBgAAUB4uh5v09HS7zx4eHmrUqJF8fHwqrCgAAIDycjnctGjRojLqAAAAqBAuX1A8evRozZ4926F97ty5Gjt2bIUUBQAAUF4uh5tly5apd+/eDu29evXS+++/XyFFAQAAlJfL4ebIkSOl3uumbt26ys7OrpCiAAAAysvlcNO6dWt98sknDu0ff/yxLrnkkgopCgAA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ANfcAAAASyHcAAAASyHcAAAASyHcAAAASyHcAAAASyHcAAAASyHcAAAASyHcAIDOPMTvgw8+cHcZACoA4QaA28XGxsowDIfXtdde6+7SANRAtdxdAABI0rXXXqsFCxbYtXl7e7upGgA1GUduAFQL3t7eCg4OtnvVq1dP0plTRomJiRo0aJB8fX0VGhqqpUuX2s3/3XffqV+/fvL19VWDBg10//33Ky8vz65PUlKSOnToIG9vb4WEhOjhhx+2m56dna2bbrpJfn5+atOmjVauXFm5Gw2gUhBuANQI//znP3XzzTdr+/btuuuuu3T77bcrLS1NkvTbb7/p2muvVb169fTNN99o6dKl+uyzz+zCS2Jioh566CHdf//9+u6777Ry5Uq1bt3abh1TpkzR0KFDtWPHDkVHR+vOO+/U0aNHq3Q7AVSASn80JwCUISYmxvT09DT9/f3tclJW6gAAAlhJREFUXlOnTjVN88xTm+Pi4uzm6dGjh/nggw+apmmar732mlmvXj0zLy/PNv2jjz4yPTw8zMOHD5umaZpNmjQxJ06ceNYaJJn/+Mc/bJ/z8vJMwzDMjz/+uMK2E0DV4JobANVC3759lZiYaNdWv3592/uIiAi7aREREUpNTZUkpaWlqXPnzvL397dN7927twoLC7V7924ZhqFDhw6pf//+56yhU6dOtvf+/v6qU6eOsrKyyr1NANyDcAOgWvD393c4TVQWwzAkSaZp2t6X1sfX19ep5dWuXdth3sLCQpdqAuB+XHMDoEb46quvHD63b99ekhQWFqbU1FSdOHHCNn3jxo3y8PBQ27ZtVadOHbVs2VKff/55ldYMwD04cgOgWsjPz9fhw4ft2mrVqqWGDRtKkpYuXarw8HBdeeWVWrx4sbZs2aL58+dLku68805NmjRJMTExmjx5sn755Rc98sgjuvvuuxUUFCRJmjx5suLi4tS4cWMNGjRIx48f18aNG/XII49U7YYCqHSEGwDVwieffKKQkBC7tnbt2mnXrl2SzvyS6d1339WoUaMUHBysxYsXKywsTJLk5+en1atXa8yYMerevbv8/Px08803a+bMmbZlxcTE6I8//tCLL76ocePG/X+7dmwDMAhDURDvwGAU7MmIZINIaUjydTeB6Z6MW++9zTnPPRA4pvbe++0hAO5UVVtrtTHG26MAP+DmBgCIIm4AgChuboDP83sOPGFzAwBEETcAQBRxAwBEETcAQBRxAwBEETcAQBRxAwBEETcAQBRxAwBEuQA17XlbPhPRDwAAAABJRU5ErkJggg==",
      "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": 39,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 28664,
     "status": "ok",
     "timestamp": 1695329081374,
     "user": {
      "displayName": "Mostafa Mushsharat",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "L1ZzFERMAQk9",
    "outputId": "56411273-fc8c-476f-f660-21239bd321ad",
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy:  0.874\n",
      "F1-Macro:  0.8697499317289445\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": 40,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 605
    },
    "executionInfo": {
     "elapsed": 29278,
     "status": "ok",
     "timestamp": 1695329182045,
     "user": {
      "displayName": "Mostafa Mushsharat",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "zUrNTiUp8kPc",
    "outputId": "e99daa47-22f9-471e-97d4-245bf6f0b8ab",
    "tags": []
   },
   "outputs": [
    {
     "data": {
      "image/png": "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6rNv375yn+v3oGu32+VwOEq1Z2ZmnvdFtIiICElS7969S7T37t1bTqdTSUlJXgVdk8mkkJCQCvcDAADeCQjw+ytDPmUN9DwaaQ4wK9Bs9nE1/hUcHFzh6ablnbYgVYOX0WJiYkrNxS0oKNDhw4fPG3SbNGnicajb5XJJuvT+owEAADVD/9bdPbb3ah5bZgiGd/yeBvv376/169crPT3d3bZy5UoVFBRowIABZfazWCzq27ev1q1bV6J93bp1CgwMZPoBAAColq5q20tXt4srMTLZtl5z3R13sx+rMia/T10YOXKkPvjgA02aNEmTJquTuMQAACAASURBVE1SWlqa4uPjNXTo0BIjuk899ZQSEhKUmPjHvJb7779fd9xxhx577DHdeOON2rdvn1577TXdeeedJZYrAwAAqC4CTAG6p+8turHTVdqfekTRYVGKiW7i77IMye9B1263a8GCBZoxY4YmT54sm82mG264QVOmTClxntPpVHFxcYm2Tp066a233tL/+3//T/fdd58iIiI0atQoPfTQQ758BAAAgAqrGx6luuEMzFUlvwddSWrRooXmzp173nPi4+MVHx9fqr1v377q27dvVZUGAACAGsrvc3QBAACAqkDQBQAAgCERdAEAAGBIBF0AAAAYEkEXAAAAhkTQBQAAgCERdAEAAGBI1WIdXQAAAEkqKC7ydwk+5XK5JKnEdsBG58vfY4IuAACoNl76+X1/lwADYeoCAAAADIkRXQAA4FdWq1Xz5s3zdxk+l5+fr4kTJ0qS5syZI6vV6ueKfK+qn5mgCwAA/MpkMslms/m7DL+yWq2X/K9BVWDqAgAAAAyJoAsAAABDIugCAADAkAi6AAAAMCSCLgAAAAyJoAsAAABDIugCAADAkAi6AAAAMCSCLgAAAAyJoAsAAABDIugCAADAkAi6AAAAMCSCLgAAAAyJoAsAAABDIugCAADAkAi6AAAAMCSCLgAAAAyJoAsAAABDCvSmU05Ojn755Rf99ttvOnnypPLy8hQZGalWrVqpV69eat26dWXXCQAAAFRIhYLuwYMHNW/ePH3xxRfKycmRyWSS3W6XxWKRw+FQfn6+TCaT2rRpo9GjR+vmm29WQACDxgAAAPC9cgfdF198UR9++KFatGihSZMmqWfPnurQoYMCA/+4xKlTp7R582atWrVKL7zwgubPn6+ZM2cq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cnNz4eXlVW/flJQU5OfnY/v27fjLX/6ia2/MXFQXFxeIRCLs2LFDb3WH2263OTs7A6gLxncrLCzUC7bOzs4oKCgw6Jefn697zzuJRKIG65TL5RCLxfj73/+OsWPHGu1ze7qHtbU1YmNjERsbi/Lycvzwww9ISkrCSy+9hOPHj8PW1rbB9yMiaginLhAR3YdXXnkFgiDgn//8J1QqlcF2tVqN7777ThcQ7w6ou3btMtjndp+7R3kHDRoEQRCQl5eH0NBQg/9CQkIAAGFhYZBKpTh06JDe/qmpqbh586ZeW2RkJDIyMvDLL7/otR84cAAikQi9e/c25WPQY2tri969eyMtLQ0hISFGa707QAN1Afnxxx/H2LFjUVpaalArEdH94oguEdF96N69OxITE7Fw4UI8++yzGDNmDIKCglBbW4u0tDTs3r0bQUFBWLJkCZycnLBgwQLEx8fD2toaX3zxhdGHQgQHBwMANmzYgAEDBsDKygohISHo2bMnRo8ejXnz5uHSpUvo1asXbG1tUVBQgPPnzyM4OBhjx46Fs7MzYmNjsW7dOsjlct2qC6tXr4aHh4feqOyECRNw4MABvPLKK5gxYwbatm2L48ePY+fOnYiOjta7Ea0x3nzzTYwdOxYvvPACoqOj4ePjg6qqKmRlZeG7777TrTIxZcoUBAUFoWvXrnB1dcXNmzexdetW+Pj41LuiAxFRYzHoEhHdp+effx7dunXDli1bsHHjRhQUFEAikcDf3x9RUVEYN24cXFxcsG7dOixbtgyzZs2Cra0thg4diqSkJDz99NN6x4uKisKPP/6InTt3YvXq1RAEQbeO7qJFixAWFobPPvsMn376KbRaLTw9PdGjRw+9G8pee+012NraYteuXdi3bx86duyIxMREJCUlQS6X6/q5urpi165dWLFiBVasWIGqqir4+vpi1qxZeis7NFZgYCD27duHjz76CCtXrkRxcTEcHR3Rvn17DBw4UNevd+/eSE5Oxp49e1BZWQkPDw/07dsXcXFxDd70RkRkKj4wgojIwt24cQMjRoxAfHw8pkyZYu5yiIhaDEd0iYgsSHp6Or788kt0794dDg4OuHr1KjZu3AgHBweMGjXK3OUREbUoBl0iIgtia2uLS5cuYe/evaioqICDgwN69+6NV1991WCJMSIiS8epC0RERERkkbi8GBERERFZJAZdIiIiIrJIDLpEREREZJEYdImIiIjIIjHoEhEREZFFYtAlIiIiIovEoEtEREREFolBl4iIiIgs0v8DIFvihPZcDAMAAAAASUVORK5CYII=",
      "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": 41,
   "metadata": {
    "executionInfo": {
     "elapsed": 8,
     "status": "ok",
     "timestamp": 1695329212064,
     "user": {
      "displayName": "Mostafa Mushsharat",
      "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": 42,
   "metadata": {
    "executionInfo": {
     "elapsed": 27254,
     "status": "ok",
     "timestamp": 1695329239311,
     "user": {
      "displayName": "Mostafa Mushsharat",
      "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": 43,
   "metadata": {
    "executionInfo": {
     "elapsed": 9,
     "status": "ok",
     "timestamp": 1695329239312,
     "user": {
      "displayName": "Mostafa Mushsharat",
      "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": 44,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 9,
     "status": "ok",
     "timestamp": 1695329239312,
     "user": {
      "displayName": "Mostafa Mushsharat",
      "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.92      0.88      2511\n",
      "    positive       0.91      0.82      0.87      2489\n",
      "\n",
      "    accuracy                           0.87      5000\n",
      "   macro avg       0.88      0.87      0.87      5000\n",
      "weighted avg       0.88      0.87      0.87      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": 45,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 30105,
     "status": "ok",
     "timestamp": 1695329269413,
     "user": {
      "displayName": "Mostafa Mushsharat",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "pOutPTF6qJY8",
    "outputId": "a034fc7e-f3ae-44e3-9bd9-2c416c5b45b3",
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy:  0.126\n",
      "F1-Macro:  0.12057055684117786\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": 46,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 605
    },
    "executionInfo": {
     "elapsed": 28733,
     "status": "ok",
     "timestamp": 1695329298142,
     "user": {
      "displayName": "Mostafa Mushsharat",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "F25gPUqgvw0w",
    "outputId": "747d6857-a67e-4f00-f9a9-fea35eee0d06",
    "tags": []
   },
   "outputs": [
    {
     "data": {
      "image/png": "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bGys/P1/P0VaWpp27NihL7/8Us8995zee+89zZ07V3FxcXXSPAAAAFCdGgfdffv26d1339Vll11W7T5NmzbVtddeq2uvvVb5+fl677339MsvvxB0AQAA4HY1DrorVqyo1YlDQkL04IMP1rohAAAAwBVcPoE2NTVV+/btc/VpAQAAgFpxatWFlJSUaretW7dOixcv1qZNm5xuCgAAALhYTgXdq666SgaDodrt7du3d7ohAAAAwBWcCrrPP/98laBbWFiorVu3asOGDZo3b55LmgMAAACc5VTQHTdunMP6xIkTNW/ePL300kt6//33L6oxAAAA4GK4/GG0oUOHateuXa4+LQAAAFArLg+6WVlZioyMdPVpAQAAgFpxauqCIxUVFTpw4IDefPNNPfzww646LQAAAOAUp4Ju165dq111wWaz6YknntATTzwhSTIYDKyrCwAAALdzKug+8MAD511eDAAAAPA0p4Lu9OnTXd0HAAAA4FIufxgNAAAA8AY1DrpPP/200tPTa3XydevWadWqVbVuCgAAALhYNZ66kJycrKuvvlrXXHONRo8erb59+yooKKjKfkePHtWGDRv0j3/8Q6mpqZo/f75LGwYAAABqosZBd9myZfryyy/11ltv6d5775W/v7/atm2riIgIBQYGKicnR8ePH1dOTo6CgoI0btw4TZ06lTV1AQAA4BG1ehjt6quv1tVXX619+/bp66+/1s6dO5WWlqb09HSFh4dr2LBh6tevn4YNG6aQkJC66hkAAAC4IKdWXYiNjVVsbKyrewEAAABchlUXAHgFm82mJUuWaNiwYerRo4fuvPNObd++3dNtAQDqMaeC7qZNm7RmzRr754yMDN17770aNGiQ/vSnP8lqtbqsQQANw8KFC/XKK6/o9OnTKisr088//6x77rlHv/76q6dbAwDUU04F3YULFyopKcn++aWXXtLWrVvVq1cvrV27Vu+8847LGgTg+woLC7VixYoq9eLiYod1AABqwqmge+TIEfsc3bKyMq1fv14zZszQ66+/roceekirV692aZMAfFtaWpoKCwsdbktOTlZ5ebm++eYbnTp1SpmZmSooKHBzhwCA+sipoJufny+LxSJJ2rt3r4qKijRs2DBJUvfu3XXq1CnXdQjA50VHRys0NNThtvbt2+uOO+7QH//4R6WlpenEiRMaO3asEhMT3dwlAKC+cSroRkZG6siRI5KkH3/8US1atFB0dLQkqaCgQP7+Ti3mAKCBMpvNuuuuu6rUGzVqpMDAQO3YsaNSPTMzU88//7y72gMA1FNOJdLBgwfr1VdfVWJiov75z39qzJgx9m2HDx9Wy5YtXdYggIbhvvvuU3h4uD788EOlp6erd+/emjZtmmbNmuVw/y1btigvL6/akWAAAJwKuo8++qhSUlL08ccfq3v37po6dap92+eff65evXq5rEEADcfNN9+sm2++uVItICDA4b5Go1FGo9EdbQEA6imngm5ERISWLFnicNvy5ctlMpkuqikA+M3IkSO1a9euKvUrr7xSwcHBHugIAFBfXPQLI4qLi5WamqqysjJJUkhICEEXgMtMmDBB1113XaVa586dq53SAADAb5wOups3b9Ytt9yi3r1768orr9TBgwclSc8884zWrVvnsgYBNGz+/v566aWXtHLlSrVq1UodOnTQBx98oKioKE+3BgDwck6/GW3KlCmyWq26++67VVFRYd8WHh6uf/zjHy5rEAAkqVOnToqMjFRoaKgMBoOn2wEA1ANOvxltyJAhSkhI0COPPFJpW9euXXXgwAGXNAcAAAA4y6mgu3//fk2YMEGSqoysREREKDMz8+I7AwAAAC6CU0HXaDSqtLTU4bbMzEw1atToopoCAAAALpZTQTcuLk6rVq1yuG3t2rXq2bPnRTUFAAAAXCyn1tH9n//5H02ZMkUPPPCAxowZI4PBoJ07d+rTTz/V2rVrtWzZMlf3CQAAANSKU0F34MCBmjdvnp5//nlt2LBBkvS///u/slgsmjt3rvr27evSJgEAAIDaciroStLo0aM1fPhwbd++XRkZGQoPD1fv3r15UxEAAAC8Qq2DbnFxse6880499NBDGjhwoOLj4+uiLwAAAOCi1PphNLPZrEOHDsloNNZFPwAAAIBLOLXqQq9evbRr1y5X9wIAAAC4jFNB9/HHH9dHH32khIQEFRQUuLonAAAA4KI59TDaLbfcotLSUj355JN68sknZTabK70hzWAwaNu2bS5rEgAAAKgtp4Lu8OHDq7z6FwAAAPAmTgXdefPmuboPAAC8xrZt27Rs2TJNnjxZffr08XQ7AJzk1BxdAAB8ldVq1ZIlS5SRkaGlS5fKarV6uiUATnI66B47dkwzZ87U5ZdfrksvvVSDBw/W448/rmPHjrmyPwAA3Ob777/X6NGj9eOPP+rQoUNKTEzUqlWrPN0WACc5FXSTkpI0fvx4rV27VrGxsRozZowuueQSrVmzRjfddJOSkpJqdb7k5GRNmTJFPXv2VHx8vObMmaPi4uJanWP9+vXq0qWLRo0aVavjAACQpO3bt+uhhx7S6dOnJUllZWVKTU3VkiVL7DUA9YtTc3RfffVVhYWF6f3331d0dLS9fvr0aU2ePFkLFizQa6+9VqNz5ebmavLkyWrRooUWLlyoM2fOaO7cucrOztb8+fNrdI7i4mLNnTtXTZo0ceZ2AADQ8uXLVVFRUaWenp6upUuX6sknn+RBbKCecSro/vzzz/rzn/9cKeRKUnR0tKZNm6bnnnuuxudauXKlcnNzlZCQoIiICEmS0WjUjBkzNHXqVMXExFzwHIsXL1aLFi3UqlUr7dmzp3Y3AwCApMOHDzusl5eXa8eOHUpJSVHLli3d3BWAi+HU1IWioiKFhYU53BYeHl6raQffffed4uPj7SFXOrt8mclk0rfffnvB448dO6Z3331Xs2bNqvE1AQD4b7GxsQ7rAQEB6tWrl1q0aOHmjgBcLKdGdNu3b69//etfGjJkSJVtq1evVocOHWp8rt/m+57LZDKpTZs2NZrr+9xzz2n06NHq2rVrja95PjabTYWFhS45FwDXOffJ96KiIoe/YgYuxoQJE7RhwwaVlJRUqjdt2lS33nqrioqKPNQZgHPZbLYaTyNyKuhOmjRJs2bNUl5ensaOHauoqCilp6dr1apV+uqrrzRnzpwanys3N1cWi6VK3WKxKCcn57zHfvXVV9q+fbu++OKLWt9DdUpLS7V//36XnQ+Aa5SWltr//eDBgwoICPBgN/BVjz/+uJYtW6aUlBQFBAQoIiJCw4cPV2ZmpjIzMz3dHoD/MJlMNdrPqaB74403KjMzU2+88YZ9eoHNZpPZbNajjz5aZYTWGRdK61arVc8//7ymT59eadrDxQoICFDHjh1ddj4ArnHuiG6XLl0UGBjowW7gqy655BJdddVV+vOf/6zs7GyFh4dr0qRJ/PcGeJHExMQa7+tU0JWk++67T7fddpu2b9+u7OxshYWFqVevXgoNDa3VeSwWi3Jzc6vU8/Lyzvsg2rJly+Tn56eRI0fajy8tLVVFRYVyc3NlNptrnPbPZTAYFBwcXOvjANQtP7/fHykICgqS2Wz2YDfwZcHBwZoyZYr9zWjh4eGebgnAOWqz+onTQVeSQkNDHc7TrY2YmJgqc3FLSkp07Nix844MHz58WEePHlV8fHyVbZdddpmefvpp3XrrrRfVGwCgYerTpw+v/gV8gFNB99NPP1VKSoqmT59eZdtrr72m1q1ba8yYMTU615AhQ/TGG28oKyvL/lPz+vXrVVJSoqFDh1Z73L333quxY8dWqr311ltKTk7W3Llz1a5du5rfEAAA59i2bZt9RJfAC9RfTi0v9v777zt8gEw6u7zY8uXLa3yuCRMmKDQ0VNOmTdPGjRuVkJCgZ599Vtdff32lqQtPPfVUpaVfYmJi1L9//0r/REVFKTg4WP3791ezZs2cuTUAQAOWnZ2txYsXa+bMmdqzZ49ef/31SvPDAdQvTo3oHj16VJ07d3a4LSYmRkePHq3xuSwWi5YtW6Y5c+Zo+vTpMpvNGjVqlGbMmFFpv4qKCpWXlzvTLgAAF5Senq7JkycrJSXFXsvKytLChQs1c+ZMD3YGwFlOz9HNy8tzWM/Pz691IG3fvr2WLFly3n3mzZunefPmXXAfAACc8d5771UKudLZFYA+/fRTTZo0qcrbQAF4P6emLnTp0kWrV692uO3zzz+vdrQXAABvtWXLFof14uJiLV68WDabzc0dAbhYTgXdiRMnau3atXr88ce1c+dOpaamaufOnXriiSe0bt063X777a7uEwCAOhUUFOSwbjAYtH///iqjvQC8n1NTF66//nodPnxYb731llatWiXp7K93jEajpk6dqhtuuMGlTQIAUNduvfVW7d69u0o9LCxMPXv2VIsWLTzQFYCL4fQc3Ycffljjx4/XDz/8oKysLEVERGjQoEFq2bKlK/sDAMAtrrvuOu3fv19///vfVVFRIensA9MtWrTQXXfdVatF6gF4B6emLvymVatWGj58uPLz87VhwwYtXry4Vq9lAwDAm/zxj3/UI488ovbt26tTp072deFZshKon2o8ovvCCy9ozZo1+uabb+y1wsJC3XjjjTp58qR9kv7q1av1ySefqEOHDi5vFgCAunbLLbdoy5Yt9hcZMR0PqL9qPKK7fft2XXfddZVqK1as0IkTJzR58mRt3bpVK1euVHBwsN5++22XNwoAgDsEBgbq7rvvVpMmTXTXXXcpMDDQ0y0BcFKNR3SPHz+uO+64o1Lt66+/VkREhGbOnCmj0aiePXvqrrvu0ooVK1zeKAAA7tKnTx9e/Qv4gBqP6Obm5qpp06b2z2VlZdq9e7f69esno9For19yySVKT093bZcAAABALdU46DZp0kRpaWn2z/v27VNZWZkuvfTSyif085PJZHJdhwAAAIATahx0u3Xrpk8++cT+0NmqVatkMBgUHx9fab/Dhw8rKirKtV0CAAAAtVTjObr33nuvbr31Vo0YMULh4eHasWOH+vbtq27dulXa7+uvv1ZcXJzLGwUAAABqo8Yjuj169NCiRYvUtGlTFRQU6KabbtLrr79eaZ/09HSdPn1aw4YNc3mjAAAAQG3U6s1oV1xxha644opqt0dFRdlfCQwAAAB40kW9GQ0AAADwVgRdAAAA+CSCLgAAAHwSQRcAAAA+iaALAAAAn0TQBeD1fvnlF02bNk27d+/WwYMHWd0FAFAjtVpeDADcbc+ePZoyZYpKSkokScXFxXrmmWdUWlqqiRMnerg7AIA3I+gCqBWbzSar1eq2673zzjv2kHuut99+W2PGjJHRaHRbL4GBgTIYDG67HgDg4hB0AdSYzWbTM888o0OHDrntmgcOHHBYT09P11133SV/f/f9Mda5c2fNnj2bsAsA9QRzdAF4NbPZ7LDu7+/v1tFcAED9w4gugBozGAyaPXu2W6cu/DZHt6ysrFL94Ycf1m233ea2PiSmLgBAfUPQBVArBoOh2lHWutC3b1+99dZb+n//7/9pz549atGihe644w7dfPPNbusBAFA/EXQBeL3+/furf//+nm4DAFDPMEcXAAAAPomgCwAAAJ9E0AUAAIBPIugCAADAJxF0AQAA4JMIugAAAPBJBF0AAAD4JIIuAAAAfBJBFwAAAD6JoAsAAACfRNAFAACATyLoAgAAwCcRdAEAAOCTCLoAAADwSQRdAAAA+CSCLgAAAHwSQRcAAAA+iaALAAAAn0TQBQAAgE8i6AIAAMAnEXQBAADgkwi6AAAA8EkEXQAAAPgkgi4AAAB8EkEXAAAAPomgCwAAAJ9E0AUAAIBPIugCAADAJxF0AQAA4JMIugAAAPBJBF0AAAD4JIIuAAAAfBJBFwAAAD6JoAsAAACfRNAFAACATyLoAgAAwCcRdAEAAOCTCLoAAADwSQRdAAAA+CSCLgAAAHwSQRcAAAA+iaALAAAAn0TQBQAAgE8i6AIAAMAnEXQBAADgkwi6AAAA8EkEXQAAAPgkgi4AAAB8EkEXAAAAPomgCwAAAJ9E0AUAAIBPIugCAADAJxF0AQAA4JP8Pd2AJCUnJ2vOnDnatm2bgoKCNHLkSM2YMUNms7naY/Lz8/Xuu+/qu+++U3Jysvz9/dWtWzc99thj6tatmxu7BwAAgDfy+Ihubm6uJk+erIKCAi1cuFCPP/64/vWvf2nWrFnnPS4lJUUfffSRBg4cqFdffVVz585VRUWFJkyYoL1797qpewAAAHgrj4/orly5Urm5uUpISFBERIQkyWg0asaMGZo6dapiYmIcHteqVSutX79eQUFB9trAgQM1bNgwrVixQnPnznVL/wAAAPBOHh/R/e677xQfH28PuZI0fPhwmUwmffvtt9UeFxwcXCnkSlJgYKBiYmKUlpZWZ/0CAACgfvB40E1KSqoyamsymdSmTRslJSXV6lyFhYXav3+/OnTo4MoWAQAAUA95fOpCbm6uLBZLlbrFYlFOTk6tzrVgwQIVFRXp9ttvd7ofm82mwsJCp48HAABA3bHZbDIYDDXa1+NBtzq1uQlJ+te//qVly5bpr3/9q9q2bev0dUtLS7V//36njwcAAEDdMplMNdrP40HXYrEoNze3Sj0vL6/aB9H+2w8//KAnn3xSU6ZM0cSJEy+qn4CAAHXs2PGizgEAAIC6kZiYWON9PR50Y2JiqszFLSkp0bFjxzR+/PgLHr9r1y49+OCDGjFihGbOnHnR/RgMBgUHB1/0eQAAAM4nIyNDhw4dUvPmzdW+fXtPt1Nv1OY3/h4PukOGDNEbb7yhrKwshYeHS5LWr1+vkpISDR069LzHJiUl6d5771Xv3r01d+7cWt04AACApyxYsEArV65UWVmZJCk+Pl5z585VSEiIhzvzLQabzWbzZAO5ubkaNWqUWrZsqWnTpikzM1Pz5s3T5Zdfrvnz59v3e+qpp5SQkKB9+/ZJkjIzMzV+/HiVlZXppZdeqrTUmMlkUmxsbK172b17tyQpLi7uIu8KAADUFzabTVar1W3XW716tcP1/keMGHHBF2a5WmBgYL0bKKxNXvP4iK7FYtGyZcs0Z84cTZ8+XWazWaNGjdKMGTMq7VdRUaHy8nL758TERJ06dUqSdOedd1bat2XLlvrqq6/qvHcAAFC/2Ww2PfPMMzp06JDbrnn8+HGH9bVr1yoxMVF+fu5b/bVz586aPXt2vQu7NeXxoCtJ7du315IlS867z7x58zRv3jz75/79++vgwYN13RoAAIBLnTtwdy6bzSYP/6Ld53hF0AUAAPAEg8Gg2bNnu3XqwqJFi/TBBx9UqcfGxuqtt95yWx9S/Zy6UBsEXQAA0KAZDAaZzWa3Xe/uu+/Wjz/+qCNHjthrwcHBmjlzplv7aAg8/gpgAACAhiQsLEzLly/XY489JovFooiICK1YsYKH4esAQRcAAMDNgoODNW7cOEVHR6tJkyZq2rSpp1vySUxdAAAAqAO5ubn68MMPtWXLFlksFo0ZM0ZXXHGFp9tqUAi6AAAALlZYWKh77rlHhw8ftte+//57PfDAA7rrrrs82FnDwtQFAAAAF1u1alWlkPubpUuXKj8/X5J08uRJ5eXlqaioyN3tNRiM6AIAALjYzp07HdaLiop04MABffHFF/rss8/s6+bed999WrBggcLDw93Zps9jRBcAAMDFmjVrVu227du3KyEhodLLIfbu3asXX3zRHa01KARdAAAAFxszZoxMJlOV+sCBA7Vp0yaHx3z99ddMY3Axgi4AAICLtWvXTi+//LLatWsnSTIajbrmmms0Z84cFRcXOzymrKxMZWVlbuzS9zFHFwAA4CKUl5fr66+/1ubNmxUSEqJRo0apY8eOio+P1//93//p1KlTatSokSwWiyRp8ODBOnToUJXz9OrVS6Ghoee9VlZWlgICAhQSElIn9+JrDLZzJ4g0cLt375Yk3kwCAABqpKysTDNmzND3339vrxmNRv31r3/VyJEjHR6Tn5+v+++/XwcOHLDXGjdurIULF+rEiRPau3evmjdvruuuu06NGzeWJB04cEAvvPCCdu/eLaPRqMGDB+uJJ55QkyZN6vYGvVBt8hpB9xwEXQAAUBvr1q3TU089VaVusVj073//W5KUkJCgTZs2qVGjRrrhhhs0YMAAlZaWas2aNVq4cKECAgK0aNEi/fWvf9X+/fvt5wgLC9OiRYvUtGlTjR8/Xjk5OZWuERsbq+XLl9ftDXqh2uQ1pi4AAACPstlsslqtnm7DKRs3bnRYz83N1ZYtW/Tee+/Zg5l0NhhPnTpVEydO1NChQ7Vy5UpJ0meffVYp5EpSdna2XnjhBQ0ZMqRKyJWkffv26aefflL37t1deEfuFRgYKIPBUGfnZ0T3HIzoAgDgfsXFxbr77rs93YZT0tLSlJ2d7XBbZGSkMjMzq9QNBoM6dOggo9Forx09erTasN+4cWOHQVeSoqOj7XN/66OlS5fKbDbX6pja5DVWXQAAAHBSdSHTZDKptLTU4TabzVZl5YXzjWqeLwjWNiQ2NExdAAAAXuNP/e+QyVi/4snXB7fow59Xq7isRJLUMqypHrpykjYmbtPnu79xeMz/9BqrnKJ8Hc44rshGYSpq3Fkf/Px5lf36tYvT/UMm6Ll/v6mkjOOVtl3Zub/uGjjW5fdT10rKy/TiFvfMLa5f/yUBAACfZjL6y2QM8HQbDv2adlT51kJ1adZOwaYge3147OUa0rGvDqYdUSNTkDpGtZHBYJCpS4C+2LtRZRXllc7ToUkrffjzv3Ug9bC9FhYUqn7t4vTzkT2y6eys0o5RbTRl4HgFB5g16w/3a83ejdp2bK9M/gEa3LGPruzcv07nt/oCgi4AAMB5nM7N0MtfvqvjWaclSYH+Jt122UgNj73cvk+QyayerbpWOq554yg9OuxOvfvjP5RRkCVJ6ta8ozo0aa1/7f660r7ZRXkqKinWKzc+rqSM44oKCVeXZu0rnX9cr2s0rtc1yisuUIWtgpBbAwRdAACA81jw1XJ7yJUka1mJ3t30T3Vo0lqdmrZVSk66vj64RTlFeeoa3UGXx/SWyf/sqHSfNrHq1aqrTmSnKiggUFGhEXoy4VWH19mTkqjGQaG6PKa3w+1peWf0zg+faPfJX2WTTV2atdM9A29U64jmrr9pH0HQBQAAqMbRzBQdyTzpcNu3v/6svOICvbLhPfv0hO8St2rDwc36yx/ulzkgUJLk5+enNueEUT8/x2sBGAwG+VUzSltRUaG5a9/SqZx0e+1g6hE998ViLbjpSfu1UBlBFwAAeI2ScscrFXhKrjW/2m351kIt3fSPKnNwk9KPae3+H/SHboMdHte/XXclpR+rUu/Zqqv8/Pwcfg12HD9QKeT+JrsoTxsTt2lo58sudCtew53fY4IuAADwqHOX9H9xy/se7KSqiooK+fn5qaKiosq2X3NPKDM/y+Fxn+39WttyfnW4zWazKSQkRPn5v4dok8mkLP8izflxqSSppKREBQUF8vPzU0hIiPLy8qrt8Z8HvtbGjN3Vbvdmdf06B4IuAABANfz8/NS0aVOdPn26Uj04OFihoaEOXwjx23FFRUXKzs5WWVmZzGazwsPD5e/vL4PBoObNmysrK0uFhYUymUyKjIy0v0AiIyNDZ86csZ8rPT1dTZo0qbbHoKCgarc1dARdAADgUeeuHvCn/pO8cnmxk9lp+j5xmwpKinRpi07q0yZWRj+jXlq3VLtTDlXZ/4r2fbV6z7f2EcuioiLJWq7ZIx9QI1OQXly/RBkZGZKkwsJC2azlemL4vSqwFmrOoTcrnauiokLW3EL1b99dW5J3VdrWrXlH/enqKfVqBYaS8lL7yH1d903QBQAAXsNkDPDKoNs+sqXaR7asUn9g6K36/+3deVxU5f4H8M/MwAzDMiwCCogLspiKomgqCqZmahdbTK97Rt5ciOyadLNFBbO6dvVabrmboYnmzyVNM8slzdCrpoLhCinILvsyDMyc3x9e5jrOIAMBA8Pn/Xr1et15zvMcvmc81z49POc5y49txfXMPwA8qH90qdpMMAAAIABJREFUz2H46Vqc3q/l80oL8cPvv8BGJsfNrDs6xwrKirDt7LfwdDS8g0KBshhDfPqim5sPziT9BrVGgyc7+GN4l4GwbGYv2GhM/GaIiIiI6sjBWoHo0DdwNzcd+WWF6OTcDmUVSsSeP2Sw//XMZKgF/fW+AHA1/RbaKKpfogCRCMOeCMKwJ4Lqo/QWwfD+FkRERERktHZObuju4QcbmRy2MutqZ1kdrRWwEEsMHhOLxHiyQ3eDx+ytbNHVrVO91dtSMOgSERER1SMrSxkG+Rje7uuZLgMwoJoXQgS264oebf3wfPchOu0yCynCB02EBZco1Bq/MSIiIqJ69nLf56ERBJy6dR4V6ko4WiswLnAkunv4oaubN25m/YEzSZe0/ds5ueHVoNEAgAl9/oIQn974LSURcksr9O3YHbYya1NdSrPGoEtERERUz6QWlpg+cCwmPxmKgrJiuNo5QfLfJQsSsQSzB0/B8z2G4nZ2ClzsHNHNzUdnBwIPh9bwcGhtqvLNBoMuERERUQOxlsphLTW8z217J3e0d3Jv5IpaFgZdIiIiojrKKsrFNxe/x+XU65BbyvCU75N4rvtg7ewtmRaDLhEREVEdFJeXIuq7VcgtKQAAFCqLsfPCYWQW5mBmyHgTV0cAgy4RERE1ISp1palLMNpP1+O0IfdhJ2+dx3M9hsLZ1gGVGjWKlSWws7IxOMtb9VKJR98QJggC7pfkQ25pBRuZeb3itzH/jBl0iYiIqMn49OxXpi7BaBkZGQbbBUHAsl9jUF5ejry8PKjVakgkEjg5OcHR0bHG8xYXFyM7OxsVFRUAADs7O7Ru3RpiMXeFrS0GXSIiIqI6kEql1R5TKpW4f/++9rNarUZ2djbEYjHs7e2hVqtRUFAApVIJS0tL2NvbQyqVQqlUIi0tTedcRUVFEAQB7u58cK22GHSJiIjIpGQyGTZv3mzqMmotLy8PL7/8MvLy8nTahw4dips3b+oE3So2NjZYunQpZs2ahZycHG17WVkZli1bhh9++AF3797VG1daWoolS5bAxcWl/i/EhGQyWYOen0GX/pS0tDRcvXoVHh4e6NKli6nLISKiZkgkEsHKysrUZdSam5sbNmzYgJUrV+LMmTOwtbXFqFGjMHPmTAwePNjgmMzMTOzcuRP37t3TaVcqlVi1ahWcnZ0NjtNoNCgqKoKnp2e9X4c5Y9ClOhEEAR9++CF27NgBjUYDAHjyySexcuVKODg4mLg6IiKixtGhQwcsW7ZMr71bt264cOGCwfazZ88aPFdiYiKmTZuGU6dO6R2zs7ODl5fXny+4hWHQNROCIKC8vLzRft7//d//Yfv27Tpt586dQ3R0ND755JNGqwN48GuPR59WJSIiMqWZM2fi9ddfh0ql0rbJZDLMmDEDq1atMjhGJpNhzJgx+OGHH5CSkqJzbMaMGc1y1tvUGHTNgCAIiI6Oxo0bNxrtZ167ds1g++HDh5Gent6oT4b6+vpi4cKFDLtERNRk9OzZE5s2bcK2bduQnJyMTp06YdKkSejcuTNeeOEFXLx4UW/MiBEj4OLigs2bN2PHjh04f/48HBwcMHr0aAwcONAEV9H8MehSnVQtV3iUIAjQaDTcAoWIiFq8J554Ah999JFe+7PPPos7d+4gJiZGO+MbHByMuXPnAgAcHR0RHh7eqLWaK5FQtVMxIT4+HgDg7+9v4kpqr65LFzIyMrB//37k5uaiT58+GDx4MCSSml9buHLlSmzcuFGvPSAgAFu3bq11HX8Gly4QEVFzlJWVhenTp8PS0hIxMTFcmmCk2uQ1zuiaibo8sXr69Gm8/vrrUCqVAIDY2FgEBQVh3bp1j90bEACmT5+On3/+WWe5hJ2dHebPn8//oxIRERlBoVDA2tra1GWYNf5+uYXSaDRYsGCBNuRWOXPmDPbu3VvjeHt7e3zzzTeIioqCs7MzPDw8sH//fnTr1q2hSiYiIiKqFQbdFurmzZt6e/hVOX78uFHnsLKygo+PD+RyOeRyea22Fbtz5w6+/vprfPfddygrKzN6HBEREZGxuHShhXrcr0rkcnmN4ysqKjBnzhwcPXpU2/bSSy9hy5YtcHNze+zYZcuWYcOGDahaHu7o6Ii1a9ciICDAyOqJiIiIasYZ3RbK09MTPXv2NHjs6aefxvbt27Fq1SqcP3/eYJ9t27bphFwASE5ORnR09GN/7q+//or169fj4Wcg8/LyMHfu3Gp3ciAiIiKqCwbdFuxf//oXfHx8tJ8tLS3x4osvYv78+Vi0aBFWrlyJSZMmGQyh3333ncFznjx5EiUlJdrPN27cwOnTp5Gfn//Ycampqbh8+fKfvSQiIiIiLS5daME8PT1x4MABXLhwAffv30dAQAAmTZqkE1QB4ODBgxgyZAj+8pe/aNset4+uIAjIzc3Fm2++iXPnzgF4sAXYzJkzoVarq63ncceIiIiIaotBt5409it461PVTgmJiYl6rxys8v3332Po0KHaz0OGDMHVq1f1+vXr1w8WFhZ46623tCEXAMrLy/H5559j+vTpBs/v6uqKzp076+0C0RxwH18iIqKmiUG3npSXl+PVV181dRlGKysrQ0FBASwsLODo6AiJRPLY3Q/+85//ICwsDKWlpQAe7Lhga2uL4uJibR9LS0sUFhZiypQpuHLlisHz7Ny5E87OzsjJydG2icVi2Nvb4+WXX0Zubi40Gg3s7e1hZ2dXT1fbsDZv3sy9g4mIiJogBt0WKCUlBVlZWTqfvb29YWdnBysrK4OzqtbW1khISNC+qlAqlaJDhw7QaDQoLS2FVCqFo6MjxGLxY2e21Wo1vL294ezsjMLCQkgkEjg5OaGgoAA3b97U9svMzISzszPat29fj1dORERELQmDbgNwHTMYIouaX6NrCtnXk5B14YJOm0ajwd3MNDz16iz0DumO/2z6BuWF/52pFQHt+vdC2m9XUfnfkAsAKpUKySl3Mfi9mZDIpFAWFCEz4SYEQQPXLt6wzriH0pw8vZ/vERyI1kP66bRVKsuRsHiNXt+cnBx0eulptPJuemFXqFQja7dx+w0TERGRaTDoNgCRhQRiy6b51WYk3DDYriwoRmFaJpy8PDF0QQQyr96EqrgUzr4dkJuUgrtnLuqNqShTIvP3WxBJxLi84yAE9YMH1BIPHEe7/gFILSiCpqJS29++bRt0HNRH77vJvXYb6nIVDMlKvA2XJzrV9XIbDDdCIyIiavqaZhqjBlTzQ1NiCwncenTWfq4uHANAWV4hbv34izbkAgAEAXd//Q19Z01CzvVklBcWwdHLE20Du0EitURFmRLplxKhKimDs19HiC2qvw3FlpbGXRYRERHRIxh0G4BQqW6yM35t/H2REndJr93KwQ72Hq11ZmCrtPJqV/0JBQGaSgPbgglAfnIq/EYE6zTn3PgD57fsRmXZf9fxHjyOtr39IbWzgapId1sziAC3Hp0N1mRqgqFrJiIioiaFQbeePPymr6a+dtPV1VXnYTSxWAwnaztc/nw7rKys4ODgoLddlouLC7Kzs/XaVNcNb0cGAMVXbiEz63+7MgiCgISEBJ21vgCQej4e7u7uyCorR2Xlg1ArEong6emJspOXUf1eEE3Dw3/2RERE1HQw6LZAnp6e2l0PRCIRcnJykJaWpj1uZWUFX19fWD60bKBdu3ZQKBTIy3vwgJmTkxPs7e1RWVkJkUhkMOw5ODigtLQUEokEMpkMpaWl2l0bHlVWVgZ/f38UFhZCo9FAoVDA4jFLGoiIiIhqwiRRTx6eAW3Kuy486uq+oyh75CURSqUS9y016DH+aZ321tWcQ3SpPa7s/E67hEEkFsEj0B+3ridpd29w9umAjoOeBK5dM3gOeQc3uE18Bm5/7nIazcO7LvBlEURERE0Tg24DaMq7LjwqI97wg2YZ8dcRMPk5JB2PQ/LP56EsKIJjBw/4PfsUnH0ebPeVm5yKm0dOIf9uGqydneDQzg0O7dxg4+KEc+tiIWj+N8ubc/MPqNVq2Lg4oSQ7V+/neQR2bTbfGcBdF4iIiJoDsakLINMSiQ3PRorEYtz4/hQSvz0GZX4hIAjIS07FubU7UJCSjoKUdMSt3obsa0moKFWiOCMbqecevA0tM+GmTsitkpeUAt9nB0FqY63T3iG4N1p3863/iyMiIjKhpKQkfPDBB3j++ecxffp0/PTTT6YuqcVpPlNo1CDce3ZB0vGzeu1uPToj+eQ5vXaNWo2kk+cgqDUGd1u4dfQMnLzaVvvzrBS2GLowAhnx16EqLoOzXwfYtXH5cxdBRETUxNy5cwevvvoqiosfLOG7d+8eLl68iHnz5mHMmDEmrq7l4IxuC+c7PBhOnXS3D7Nv54YOIX1QqTT8Kt+SrFwUpWcbPKYsKIK9p7vBYxZWMti3bQOJ1BIegd3QcVAfhlwiIjJL27Zt04bch23cuBGVlZW4c+cOlixZgj/++AP37t3D+fPnTVCl+eOMbgNoyvvoPkoskaDfzAm4f/suijKyYevSCq182kNQayC1kUNVor+5l20bZ1SWKVGUoR92ZXY28HzSH2m//Y6ClHSdY74jQiAWi5vkvri1xX10iYjoca5V8/B1Tk4Orly5grlz56KoqAgAoFKpMGfOHHz00Ud45plnGrNMs8eg2wCa+j661ZEDUKfkIeviLQCAq2MrpJak6vQRi8VQFFdCrRYh08C2Yi72jri/9xQ6tmqN+7BAYWEhJBIJnJ2dIU/JRWbsj411OUREREYRBAHl5YZ/i1lXbm5uSExM1Gu3sbHBnj17tCH34RrWrFmD4ODgRt3NRyaTmfXuQQy6VK3WrVvDwsICWVlZUKlUsLGxgbu7O+RyOQDA29sbaWlpKC0thVQqRevWreHi8mApgkQigaurK1xdXU15CWRmLly4gK1bt2Lq1KkIDAw0dTlEZAYEQUB0dDRu3Kj+dfd1UVZm+HVHMpkMx44dM3gsNTUVr7zyCiSSxtui1NfXFwsXLjTbsMugW09kMhk2b95s6jIaXXl5OWbNmgUA+OKLLyCTyUxcUeNridfcUDQaDUpLS2FjY6P3l255eTk2b96MvLw8bN68Gd26deN3T0RNllwuh7u7O+7fv4/y8nJIJBI4ODjAyckJ5eXlBl+gJBaLIRbz8an6xKBbT0QiEaysrExdhknJZLIW/x1Q3W3fvh2bN29GRkYG3NzcMH36dPz1r3/VHt+/f7/2zXx5eXn49ttvMXbsWFOVS0RmQiQSYeHChfW+dOFhSqUSUqlUG2LPnTuHuXPn6i3/mzRpEmbMmNFgdRjCpQtERA1s9+7d+Pjjj7Wf09PTER0dDblcjlGjRuH27dvYtGkTiouLIZVK0apVK3z77bcIDg5GmzZtTFg5EZmDhp6sevTcISEhWLRoEb744gukpaXB1tYWY8eOxcyZMxt12UJLwKBL1UpPT8fSpUvx008/QSqVYtSoUZgzZw5sbW1NXRqZma1btxps//LLLxEUFIRJkybpPLiRk5MDLy8vfPnll3jnnXfMejaCiMzTyJEjMXz4cOTl5UGhUMDS0tLUJZklBl0yqKysDFOmTEFKSor287Zt23Dz5k189dVXJq6OzE1qaqrB9nv37mHFihV6TydrNBrcu3cPV65cQVpaGjw8PBqjTCKieiUWi9GqVStTl2HWuOKZDDp06JA25D7s7NmzuHz5MiorK/H999/jo48+QlpaWoOubSLz17VrV4PtXbp0weXLlw0eKysrg0wmg5ubW0OWRkREzRiDLhmUlJRU7bEbN25gxowZePPNN7Fr1y6kp6fj6tWrOHHiROMVSGbl9ddfh4WF7i+YLC0tER4eXu26OZFIhLKyMqSnpxs8TkRExKBLBvn6+lZ7LD09HadPn9ZpEwQBixcvRmVl83/rGTW+/v37Y8uWLRg8eDDat2+Pp59+Glu3bkXv3r0xfvx4g2McHBwQEBAAd3fDr5wmIiLiGl0yaOTIkVi7dq3ezG5ISAhu3rxpcEx2djYSExPh7++P27dvo6SkBF26dNGbqSMypFevXujVq5de+wsvvICEhATExsZqt+JRKBTw9PREWFgYH0QjIqJqcUaXDJJKpYiJicG4cePQqlUruLu7Y8aMGVi5ciVsbGyqHVdYWIixY8fi2WefxdixYzF48OBq3wBDZKwPPvgAc+fOhZeXFzp37oyOHTti9OjRaN26talLIyKiJkwkPLpbcQsWHx8PAPD39zdxJU1bXFwcpk6dqtferVs3lJeX6834Wlpa4vDhw/D09GysEskMlZeX46233kJeXh6cnJywbNkyvhmNiKgFqk1e44wu1Vq/fv0QGRkJqVSqbbOyssLLL79scFlDRUUF9uzZ05glkhmSyWR49dVX4ezsjLCwMIZcIiKqUZMIusnJyZg2bRoCAgLQv39/LF68GEql0qixe/fuxYgRI+Dv74/Q0FAcPny4gaslAHjttddw8uRJfPbZZ/Dz80OXLl10gu+jcnNzG7E6MleBgYFYsWIFAgMDTV0KERE1AyZ/SqiwsBBTp06Fu7s7VqxYgdzcXHzyySfIz8/H0qVLHzv2+++/x7x58zB9+nQMGDAAP/74I+bMmQM7OzsMHDiwka7AvF2/fh07duxARkYGevXqhXHjxsHe3h7Ag42uk5OTkZmZCalUCmdnZ0ilUqhUKr3zBAUFNXbpRERE1MKZPOjGxsaisLAQ+/btg5OTEwBAIpEgMjISs2bNQqdOnaod+/nnn2PEiBGYO3cugAe/Uk9OTsaKFSsYdOvBiRMnEBERgYqKCgDA8ePHsXv3bsTGxkKtVmP8+PE6b7T629/+hhdffBE7d+7UOc/AgQPx9NNPN2rtRERERCYPuj///DP69++vDbkAMHz4cLz33ns4efJktUE3JSUFSUlJeOutt3TaQ0ND8e677yI3N1fnnOZOEIR6fTuZIAj46KOPtCG3yp07d7Bx40aUl5frvbZVqVTiypUrWL9+PQ4cOIDS0lIEBwcjNDQUFRUVeueqLzKZjFtMERERkR6TB93bt2/jpZde0mmTSqVo164dbt++Xe24qv1dvby8dNo7deoEQRCQlJRUp6ArCAJKS0trPc6UBEHAP//5z8d+X7WlUqlw9+5dg8eqZnQNSUxMxOrVq7V75x44cAAHDhyot7oM8fb2xjvvvMOwS0RE1AIIgmD0v/NNHnQLCwuhUCj02hUKBQoKCqodV3Xs0bFV60cfN/ZxKioqkJiYWKexpiIIAsrKyur1nBKJBCKRCIZ2n7OwsKj2BhOJRBCLG/cZx9LSUiQmJjLoEhERtRCPewD+YSYPutUxNq0/2qcqmNU19FhaWsLb27tOY00pOjra4ENgf/achnaxmD9/PkpKSrBw4UK9Y8899xzefffdeq2jJlKplCGXiIiohbh165bRfU0edBUKBQoLC/Xai4qKHvsg2sMzt87Oztr2qnMZmiU2hkgkgrW1dZ3Gmtrj3lhWF4sXL4ZGo8HRo0eh0WhgZ2eHiIgIjBo1CgCQk5ODjRs3oqysDGKxGMOGDUNUVFSz/f6IiIio6avN5JbJg26nTp301pZWrQ99dO3uw6rW5iYlJekE4tu3b0MkEumt3aXas7W1xYoVK5CRkYGsrCx4e3vrhNiIiAjtSyLc3Nzg7u5uwmqJiIiIdJn8hREhISGIi4tDXl6etu3o0aNQqVQYNGhQteM8PT3h5eWFQ4cO6bQfPHgQ3bt3b1E7LjS0Nm3aoHv37gZnahUKBQIDAxlyiYiIqMkxedAdP3487OzsEB4ejlOnTmHfvn348MMPMWrUKJ2Z2vfeew9dunTRGTt79mwcPnwYy5cvx9mzZ/Hxxx/jl19+wezZsxv7MoiIiIioiTH50gWFQoGtW7di8eLFeOONN2BlZYXQ0FBERkbq9NNoNHpbWo0cORJKpRJr167Fpk2b0L59eyxfvpwviyAiIiIiiARD+0e1UPHx8QAAf39/E1dCRERERIbUJq+ZfOkCEREREVFDYNAlIiIiIrPEoEtEREREZolBl4iIiIjMEoMuEREREZklBl0iIiIiMksMukRERERklhh0iYiIiMgsMegSERERkVli0CUiIiIis8SgS0RERERmiUGXiIiIiMwSgy4RERERmSUGXSIiIiIySxamLqApqaiogCAIiI+PN3UpRERERGSASqWCSCQyqi+D7kOM/dKIiIiIyDREIpHRmU0kCILQwPUQERERETU6rtElIiIiIrPEoEtEREREZolBl4iIiIjMEoMuEREREZklBl0iIiIiMksMukRERERklhh0iYiIiMgsMegSERERkVli0CUiIiIis8SgS0RERERmiUGXiIiIiMwSgy4RERERmSUGXSIiIiIySwy6RERERGSWGHSJiIiIyCwx6FKN5s2bh9DQUJw8eRKhoaHw9/fH6NGjcenSJW2fIUOGYNGiRdi4cSOCg4PRo0cPzJo1C1lZWSasnJqb2txr27Ztw+DBgxEYGIjw8HDk5uaasHJqbqrutbNnz+KFF15AQEAAxowZg4SEBG0fPz8/rF+/Hp9++in69euHnj17Yt68eSguLjZh5dTcGHuvbdiwAStWrEBQUBD69u2Ld999F6WlpSas3Dww6JJRsrOzER0djWnTpuGzzz6DVCrFtGnTcP/+fW2fo0eP4scff0RUVBSioqIQHx+PN954w4RVU3NkzL127NgxHD9+HAsWLMD777+Pc+fO4cMPPzRh1dQcZWdnY/HixZg2bRqWL18OpVKJiIgIVFRUaPvExMQgKSkJS5YsQWRkJI4cOYL58+ebsGpqjoy517Zv3447d+7gn//8J8LDw3HgwAGsWbPGhFWbBwtTF0DNQ35+Pj777DP0798fANCnTx8MGjQIW7duxVtvvQUAKCkpwfr166FQKAAAbdq0wSuvvILTp09j4MCBJqudmhdj7jVBEPDFF19AKpUCAO7cuYNNmzZBo9FALOZ/v5NxCgoKsG3bNvj4+AAAZDIZwsLCcPnyZfTu3RsAIJVKsXr1akgkEu3n+fPnIyIiAp06dTJZ7dS8GHOvOTs7Y9myZQCAkJAQxMfH48iRI4iMjDRZ3eaA/0Ygo9jZ2WmDBwAoFAr069dP51fKffv21YZcAOjfvz9sbW11+hDVxJh7rU+fPtqQCwDe3t6oqKjQmfUlqomr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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": 47,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 27580,
     "status": "ok",
     "timestamp": 1695329325716,
     "user": {
      "displayName": "Mostafa Mushsharat",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "zlbzZWECocpp",
    "outputId": "27337f4e-b8c6-4b92-a088-77f03f0ccd93",
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy:  0.4308\n",
      "F1-Macro:  0.39705605928612525\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": 48,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 605
    },
    "executionInfo": {
     "elapsed": 29037,
     "status": "ok",
     "timestamp": 1695329354749,
     "user": {
      "displayName": "Mostafa Mushsharat",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "DGrasP1mxQrD",
    "outputId": "87f1fe38-fe1b-4d6c-bf39-3c3e36c56fae",
    "tags": []
   },
   "outputs": [
    {
     "data": {
      "image/png": "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evUq87w9e/aoXr16wU0JAAAAVFKFd10YP368br75Zg0cOFCxsbHatGmTunbtqvbt25d53sqVK5WSkhL0oAAAAEBlVHhG96KLLtIrr7yi+vXrKzc3VzfccINeeumlMs/JzMxURkaG+vfvH/SgAAAAQGVU6oYRl112mS677LKzPl6vXj3fLYEBAAAAI1V4RhcAAACoSSi6AAAAMCWKLgAAAEyJogsAAABTougCAADAlCi6AAAAMCWKLgAAAEyJogsAAABTougCAADAlCi6AAAAMCWKLgAAAEyJogsAAABTougCAADAlCi6AAAAMCWKLgAAAEyJogsAAABTougCAADAlCi6AAAAMCWKLgAAAEyJogsAAABTougCAADAlCi6AAAAMCWKLgAAAEyJogsAAABTougCAADAlCi6AAAAMCWKLgAAAEyJogsAAABTougCAADAlCi6AAAAMCWKLgAAAEyJogsAAABTougCAADAlCi6AAAAMCWb0QEAAABqm/TsTC3e/IX2ZB5UQlSs/qf9pUpp1MroWKZD0QUAAAih9OxM/Wnx35RbmC9JOnDqiDYe3KFJl41S7+TOBqczF5YuAAAAhNDizV/4Su4vvPLq/Q2fyev1GpTKnCi6AAAAIbT72AG/4xnu4zrtyQtxGnOj6AIAAIRQvahYv+N1HBGKcDhDnMbcKLoAAAAhNLD9pbLIUm786rZ9ZAuzGpDIvCi6AAAAIXRR49a6+7KRqh8dJ+nnmdyhF/XXDV0GGJzMfNh1AQAAIMQuSe6i3i066XRBniIdTtmsVLKqwJ8qAACAAcIsYXJFRBkdw9RYugAAAABTougCAADAlCi6AAAAMCWKLgAAAEyJogsAAABTougCAADAlCi6AAAAMCWKLgAAAEyJogsAAABTougCAADAlCi6AAAAMCWKLgAAAEyJogsAAABTougCAADAlGxGB0DNlXv8lPZ88a2yDqYrIraukvp1V3xyU6NjAQAASKLoIkC5mSf11QtzVJSXL0nKPpihjC271GXM79SwU1uD0wEAUL3tzjyghRuXKS3zoOpFxeqaDn3VJ7mL0bFMh6KLgOxesdZXcn28XqUuWaULLmoji8ViTDAAAKq5fScO68klL6uopFiS5C44rZmr3lF+kUdXtullcDpzYY0uApK177Df8dzMk+ULMAAA8Fm8eaWv5J7pw03LVeotNSCR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efMee+5///tfTJs2DX//+9/RqVMn7Nu3D++++y5cXV3RuXNnKz0DIiIiInoeiV7obty4ERqNBtu3b0eNGjUAADKZDLGxsRg3bhwCAgLKPfff//43evXqhUmTJgEA2rdvj7S0NCxatIiFLhEREVE1J/rUhZ9//hkdOnQwFLkA0LNnTygUChw+fLjc827cuIHU1FSEh4cbtYeHh+Ps2bPIycmpssxERERE9PwTfUT32rVrGDhwoFGbQqFAvXr1cO3atXLPS01NBQD4+/sbtQcEBEAQBKSmphoVz5YSBAEPHjx46vOIiIiIqOoJggCJRGJRX9ELXY1GAzc3N5N2Nzc35OXllXte2bFHz3V3dzc6/rR0Oh0uXrxYoXOJiIiIqOopFAqL+ole6JbH0mr90T6CIJhtt5RcLkdgYGCFziUiIiKiqnX16lWL+4pe6Lq5uUGj0Zi05+fnP3Yh2l9Hbr28vAztZY9lbpTYEhKJBM7OzhU6l4iIiIiq1tMMZoq+GC0gIMBkLq5Wq0V6evpjC92yubllc3XLXLt2DRKJxGTuLhERERFVL6IXul27dkViYiLu3btnaPvpp5+g1Wrx8ssvl3uen58f/P39sWvXLqP2nTt3onnz5hVaiEZERERE9kP0QjciIgKurq6Ijo7GkSNHsH37dnz00Ud4/fXXjUZ0P/jgAzRp0sTo3JiYGOzevRsLFizAyZMn8a9//QvHjh1DTEyMtZ8GERERET1nnos5umvXrsXcuXMxYcIEODo6Ijw8HLGxsUb9SktLodfrjdp69+6NoqIiLF++HKtWrUL9+vWxYMEC3iyCiIiIiCARyrYpICQlJUEQBIu3rCAiIiIi69JqtZBIJGjduvUT+4o+ovs8qeiWZERERERkHRKJxOKajSO6RERERGSXRF+MRkRERERUFVjoEhEREZFdYqFLRERERHaJhS4RERER2SUWukRERERkl1joEhEREZFdYqFLRERERHaJhS4RERER2SUWukRERERkl1joEhEREZFdYqFLRERERHaJhS4RERER2SUWukRERERkl1joEhEREZFdYqFLRERERHaJhS490bRp0xAeHo7Dhw8jPDwcISEhGDBgAE6fPm3o061bN8yZMwcJCQno0qULWrRogXHjxiErK0vE5GRrnuZa27BhA1599VW89NJLiI6ORk5OjojJydaUXWsnT57EG2+8gZYtW+LNN9/EuXPnDH2Cg4MRHx+Pzz77DO3bt0erVq0wbdo03L9/X8TkZGssvdZWrlyJRYsWoWPHjggNDcX777+PBw8eiJjcPrDQJYuo1WrMnj0bo0ePxsKFC6FQKDB69GhkZ2cb+vz000/Yt28f4uLiEBcXh+TkZEyYMEHE1GSLLLnWDhw4gIMHD2LmzJmYPn06Tp06hY8++kjE1GSL1Go15s6di9GjR2PBggUoKirC+PHjodPpDH3Wr1+P1NRUfPrpp4iNjcWePXswY8YMEVOTLbLkWvv6669x/fp1fPLJJ4iOjsYPP/yApUuXipjaPjiIHYBsQ25uLhYuXIgOHToAANq2bYuXX34Za9euxXvvvQcAKCgoQHx8PNzc3AAAtWrVwttvv42jR4+ic+fOomUn22LJtSYIApYtWwaFQgEAuH79OlatWoXS0lJIpXz/TpbJy8vDhg0bEBQUBABQKpWIiorCmTNn0KZNGwCAQqHAkiVLIJPJDF/PmDED48ePR0BAgGjZybZYcq15eXlh/vz5AICuXbsiOTkZe/bsQWxsrGi57QH/IpBFXF1dDYUHALi5uaF9+/ZGHymHhoYailwA6NChA1QqlVEfoiex5Fpr27atocgFgMDAQOh0OqNRX6In8fHxMRQeAAyF6507dwxtr776qqHIBYAePXpAEAQkJydbLyjZPEuutU6dOhmdExgYiMzMTOsEtGMsdMkiNWrUMGmrWbMm1Gq10ddP6kP0JJZca399QwUAcrkcAFBcXFy14ciuWHIdPfq65u7uDrlczvUH9FQsudbM9dFqtVUfzs6x0CWLmFvok52dDW9vb6Ovn9SH6EksudaIrOXR17W8vDzodDr4+PiIlIiIngYLXbJIfn4+Tpw4YfR1YmIiWrRoYWg7efIk8vPzDV+fOHEC9+/fN+pD9CSWXGtE1nLw4EHo9XrD13v37oVEIkFISIiIqYjIUlyMRhbx8PDA9OnTERMTA1dXV6xcuRIAEBkZaejj4uKCMWPGYMyYMcjPz8e8efPQvHlzdOnSRazYZIMsudaIrEWr1eIf//gHhgwZgoyMDMybNw89e/bkQjQiG8FClyzi7e2N2NhYfPbZZ0hPT0dQUBBWrVoFLy8vQ5/u3bujVq1amDVrFjQaDTp27IjZs2eLmJpskSXXGpG1jBgxAjk5OZgyZQq0Wi26d++OmTNnih2LiCwkEQRBEDsEPd+mTZuGc+fOYefOneX26datG1555RX+AaBnYsm1RmQtwcHBmDJlCkaPHi12FCKqIM7RJSIiIiK7xEKXiIiIiOwSpy4QERERkV3iiC4RERER2SUWukRERERkl1joEhEREZFdYqFLRERERHaJhS4RERER2SUWukREzyAlJQXvv/8+unXrhpCQELRq1Qr9+/fHypUrkZub+1SPdfjwYSxevLiKkla+jIwMBAcHY+vWrWJHISIyi9uLERFV0ObNmzF79my8+OKLGDJkCAIDA1FSUoJz585h8+bNaNSoEZYsWWLx482ZMwdff/01Ll26VIWpK49Wq8WFCxdQr1491KhRQ+w4REQmHMQOQERki37//XfExcWhY8eOWLp0KRQKheFYp06dEBUVhSNHjoiYsOro9Xro9XooFAq0bNlS7DhEROXiiC4RUQWMHTsWR44cwb59++Dr6/vYvrt27cKWLVtw+fJlaDQa1KlTB2FhYYiOjoazszMAYNq0adi2bZvJufv370fdunUhCAK++eYbbN68GWlpaVAqlejQoQMmT54MPz8/Q39BELBixQps2rQJd+9eTAOpAAAHf0lEQVTeRVBQECZNmoTly5cDANavX2/oe+vWLXzxxRc4duwY8vPz4efnh0GDBuHtt9+GVPpwZltGRgbCwsIQGxsLnU6HLVu2IDMzE8uXL4e/vz/CwsLw8ccfY8CAAYbH/eOPP7B48WIcP37c8LjDhw/HsGHDDH1KS0uxfPly7NixA7dv34ZCoYCvry/efPNNREZGVuA3QkRkiiO6RERPSa/XIzExEU2bNn1ikQs8LPy6du2KyMhIODk5ITU1FStXrsTZs2exbt06AEB0dDQePHiAPXv2YNOmTYZzfXx8AAAzZ87Etm3bMGLECMTGxiIvLw9LlixBREQEduzYAS8vLwDAggULsGLFCgwePBjdu3dHZmYmPvzwQ+h0Orz44ouGx83JyUFERAR0Oh3++c9/ok6dOjh06BA+/fRTpKenIy4uzug5rF+/Hg0aNMDUqVOhUqlQv359s8/16tWriIiIgK+vL6ZOnQpvb28cPXoUc+fOxb179zB+/HgAQEJCAr788kuMGzcObdq0QUlJCVJTU5Gfn2/5L4KI6AlY6BIRPaV79+6hsLAQdevWtah/dHS04b8FQUDr1q0REBCA4cOHIyUlBY0aNUK9evUMxeqj0wFOnz6NzZs3Y9q0aYiKijK0t2nTBj179sRXX32FyZMnIy8vD1999RVee+01zJkzx9AvKCgIgwcPNip0v/rqK9y5cwf/+c9/0Lx5cwBAly5doNfrsXHjRkRGRhr1VyqVWLVqFeRyuaEtIyPD5Ll+/PHHcHFxwbfffguVSgXg4VQOrVaL+Ph4jBgxAu7u7khKSkLDhg0xYcIEw7ldunSx6OdJRGQp7rpARFTFbty4gUmTJqFTp05o3LgxmjZtiuHDhwMAUlNTn3j+wYMHIZFI0LdvX5SUlBj+eXl5oVGjRjh16hSAhwWxVqtF7969jc5v2bIl6tSpY9SWmJiIwMBAQ5FbZsCAARAEAYmJiUbt3bp1MypyzSkuLkZiYiK6d+8OR0dHo6xdu3ZFcXExTp8+DQAICQlBSkoK4uLicOTIEdy/f/+JPwcioqfFEV0ioqfk6ekJJycnsyOajyooKMDQoUOhVCoxceJENGjQAI6OjsjMzMT48eNRVFT0xMfIzs6GIAjo2LGj2eNlc3TLtjOrWbOmSZ+y0eIyubm5JsUv8OdUiUe3RvP29n5iztzcXJSUlGD9+vVGc4H/6t69ewCAd955B87Ozvj++++xceNGyGQytGnTBrGxsQgJCXni9yIisgQLXSKipySTydC+fXscOXIEmZmZqFWrVrl9ExMTkZWVhfXr16Ndu3aG9qeZi+rp6QmJRIKvv/7aaHeHMmVtHh4eAB4Wxo+6e/euUWHr4eEBtVpt0i8rK8vwPf9KIpE8MaebmxtkMhn69euHoUOHmu1TNt3DwcEBUVFRiIqKgkajwfHjx7FgwQL87W9/w6FDh+Dk5PTE70dE9CScukBEVAHvvPMOBEHAhx9+CK1Wa3Jcp9PhwIEDhgLx0QJ148aNJueU9Xl0lPeVV16BIAi4c+cOQkJCTP4FBwcDAFq0aAGFQoFdu3YZnX/69GncvHnTqK1Dhw64evUqzp8/b9S+fft2SCQShIaGWvJjMOLk5ITQ0FBcuHABwcHBZrM+WkADDwvkXr16YejQocjNzTXJSkRUURzRJSKqgFatWiEuLg6zZ8/GwIEDERERgaCgIJSUlODChQvYvHkzgoKCMHfuXLi7u2PWrFkYP348HBwc8MMPP5i9KUTDhg0BACtXrkTXrl0hlUoRHByMl156CYMHD8YHH3yAc+fOoW3btnBycoJarcZvv/2Ghg0bYujQofDw8EBUVBRWrFgBNzc3w64LS5Ysgbe3t9Go7Ntvv43t27fjnXfeQUxMDGrXro1Dhw7hm2++wZAhQ4wWoj2N6dOnY+jQoRg2bBiGDBmCOnXqoKCgAOnp6Thw4IBhl4mxY8ciKCgIzZo1Q40aNXDz5k2sXbsWderUKXdHByKip8VCl4iogt566y00b94ca9asQUJCAtRqNeRyORo0aIDw8HAMHz4cnp6eWLFiBT799FNMnjwZTk5OCAsLw4IFC9C/f3+jxwsPD0dSUhK++eYbLFmyBIIgGPbRnTNnDlq0aIFNmzbh22+/RWlpKXx8fNC6dWujBWXvvvsunJycsHHjRmzduhX+/v6Ii4vDggUL4ObmZuhXo0YNbNy4EfPnz8f8+fNRUFCAunXrYvLkyUY7OzytwMBAbN26FUuXLsXChQuRk5MDV1dX1K9fHy+//LKhX2hoKPbs2YP//Oc/uH//Pry9vdGxY0dER0c/cdEbEZGleMMIIiI7d+PGDfTu3Rvjx4/H2LFjxY5DRGQ1HNElIrIjKSkp2LlzJ1q1agWVSoW0tDQkJCRApVLhzTffFDseEZFVsdAlIrIjTk5OOHfuHLZs2YL8/HyoVCqEhoZi4sSJJluMERHZO05dICIiIiK7xO3FiIiIiMgusdAlIiIiIrvEQpeIiIiI7BILXSIiIiKySyx0iYiIiMgusdAlIiIiIrvEQpeIiIiI7BILXSIiIiKyS/8HCdF1SgMu1IoAAAAASUVORK5CYII=",
      "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": "code",
   "execution_count": 49,
   "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": 50,
   "metadata": {
    "executionInfo": {
     "elapsed": 16,
     "status": "ok",
     "timestamp": 1695329438385,
     "user": {
      "displayName": "Mostafa Mushsharat",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "Ppy-K6l2gWtd",
    "tags": []
   },
   "outputs": [],
   "source": [
    "from IPython.display import clear_output"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 58,
   "metadata": {
    "executionInfo": {
     "elapsed": 38794,
     "status": "ok",
     "timestamp": 1695329477164,
     "user": {
      "displayName": "Mostafa Mushsharat",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "_OCeDQY6gHMj",
    "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",
    "\n",
    "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"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "oEdEBFPQiasC"
   },
   "source": [
    "Paraphrased Native Injection Accuracy"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 59,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 30045,
     "status": "ok",
     "timestamp": 1695329507205,
     "user": {
      "displayName": "Mostafa Mushsharat",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "uEuhYVF-gMqC",
    "outputId": "3c4d554a-9320-4704-990c-0a6f0b392083",
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy:  0.8534\n",
      "F1-Macro:  0.8482592788717996\n"
     ]
    }
   ],
   "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": 1,
   "metadata": {
    "executionInfo": {
     "elapsed": 13,
     "status": "ok",
     "timestamp": 1695329507206,
     "user": {
      "displayName": "Mostafa Mushsharat",
      "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": 56,
   "metadata": {
    "executionInfo": {
     "elapsed": 11,
     "status": "ok",
     "timestamp": 1695329507206,
     "user": {
      "displayName": "Mostafa Mushsharat",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "nysMxY5myqsd",
    "tags": []
   },
   "outputs": [],
   "source": [
    "import numpy as np"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "JuMwkV_v4GvV",
    "outputId": "24e92f01-d79f-4514-f879-14a50b1f8477",
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Working "
     ]
    },
    {
     "ename": "NameError",
     "evalue": "name 'df_senti' is not defined",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mNameError\u001b[0m                                 Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[2], line 9\u001b[0m\n\u001b[1;32m      6\u001b[0m \u001b[38;5;28mprint\u001b[39m(\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mWorking\u001b[39m\u001b[38;5;124m\"\u001b[39m, end\u001b[38;5;241m=\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;124m \u001b[39m\u001b[38;5;124m'\u001b[39m)\n\u001b[1;32m      7\u001b[0m \u001b[38;5;28;01mfor\u001b[39;00m i \u001b[38;5;129;01min\u001b[39;00m \u001b[38;5;28mrange\u001b[39m(\u001b[38;5;241m300\u001b[39m):\n\u001b[1;32m      8\u001b[0m     \u001b[38;5;66;03m#SELECT RANDOM WORDS FROM NEUTRAL SENTIWORD\u001b[39;00m\n\u001b[0;32m----> 9\u001b[0m     df_neutral \u001b[38;5;241m=\u001b[39m \u001b[43mdf_senti\u001b[49m[df_senti[\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mSentiment\u001b[39m\u001b[38;5;124m'\u001b[39m]\u001b[38;5;241m==\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mNeutral\u001b[39m\u001b[38;5;124m'\u001b[39m]\u001b[38;5;241m.\u001b[39msample(n\u001b[38;5;241m=\u001b[39m\u001b[38;5;241m20\u001b[39m, random_state\u001b[38;5;241m=\u001b[39mrandom_seeds[i])\n\u001b[1;32m     10\u001b[0m     neutral_words \u001b[38;5;241m=\u001b[39m \u001b[38;5;28mlist\u001b[39m(df_neutral[\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mWord\u001b[39m\u001b[38;5;124m\"\u001b[39m])\n\u001b[1;32m     12\u001b[0m     \u001b[38;5;66;03m#FETCH DEFINITIONS OF THOSE WORDS\u001b[39;00m\n",
      "\u001b[0;31mNameError\u001b[0m: name 'df_senti' is not defined"
     ]
    }
   ],
   "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": "Mostafa Mushsharat",
      "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": "Mostafa Mushsharat",
      "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": "Mostafa Mushsharat",
      "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": "Mostafa Mushsharat",
      "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": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "f1s = list(pd.read_csv('/home/mostafa_nsu/ICLR/bert_randomizationtest.csv')['f1s'])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "executionInfo": {
     "elapsed": 15,
     "status": "aborted",
     "timestamp": 1695379585271,
     "user": {
      "displayName": "Mostafa Mushsharat",
      "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.815  # 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",
    "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_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": "Mostafa Mushsharat",
      "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": "Mostafa Mushsharat",
      "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": "Mostafa Mushsharat",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "1MTPlFIVEd6y",
    "tags": []
   },
   "outputs": [],
   "source": [
    "max_f1_index = f1s.index(max(f1s))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "executionInfo": {
     "elapsed": 32,
     "status": "aborted",
     "timestamp": 1695329093834,
     "user": {
      "displayName": "Mostafa Mushsharat",
      "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": "Mostafa Mushsharat",
      "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": "Mostafa Mushsharat",
      "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": "Mostafa Mushsharat",
      "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": "Mostafa Mushsharat",
      "userId": "12156804663229931259"
     },
     "user_tz": -360
    },
    "id": "M4c1i-jEnOia",
    "tags": []
   },
   "outputs": [],
   "source": [
    "import os, sys\n",
    "\n",
    "filenames = []\n",
    "\n",
    "path = \"/home/mostafa_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": "Mostafa Mushsharat",
      "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/mostafa_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": []
  },
  {
   "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
}
