{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# This is a notebook to build some intuition of our method.\n",
    "\n",
    "Based on the [Yelp dataset](https://huggingface.co/datasets/fancyzhx/yelp_polarity). \n",
    "\n",
    "We consider the experimental setup where we have access to a fix numbner of dataset per class (via a random seed) for training and this part is proportionally divided as 70/30 for actual training and validation. We also have a fixed number of dataset per class (via a fixed random seed) for test."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "from transformers import set_seed\n",
    "from datasets import load_dataset, concatenate_datasets, DatasetDict\n",
    "import os\n",
    "import torch.nn as nn\n",
    "import evaluate\n",
    "import numpy as np\n",
    "import json\n",
    "import copy\n",
    "import matplotlib.pyplot as plt\n",
    "import torch.nn.functional as F\n",
    "import torch\n",
    "import torch.optim as optim\n",
    "import random\n",
    "import re\n",
    "from torch.utils.data import Dataset, DataLoader\n",
    "import torch\n",
    "from transformers import DataCollatorWithPadding\n",
    "from transformers import AutoTokenizer, AutoModel, TrainingArguments, Trainer"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "def synthetic_data_sampler(dataset, n_samples, sampler_seed, test_per_class=2000, search_max_words=30, max_words = 40, min_freq=2):\n",
    "    \n",
    "    # Gather the number of labels from the test dataset with predefined train, validation ratio\n",
    "    assert search_max_words <= max_words\n",
    "    num_cls = len(set(dataset['test']['label']))\n",
    "    ratio = 0.8\n",
    "\n",
    "    print(f'Generate synthetic data of {num_cls} classes with random seed {sampler_seed}, each class of {int(ratio * n_samples)} training, {n_samples - int(ratio * n_samples)} validation and {test_per_class} test data.')\n",
    "\n",
    "    train_datasets = []\n",
    "    val_datasets = []\n",
    "    test_datasets = []\n",
    "    \n",
    "    # Define a regular expression for checking if \"a\" or \"the\" in the sentence, and in the first 50 words\n",
    "    def contains_the_and(text):\n",
    "       # Split the text into words\n",
    "        words = text.split()\n",
    "        # Join the first 100 words (or all words if less than 100)\n",
    "        first_n_words = ' '.join(words[:search_max_words])\n",
    "        count_the = len(re.findall(r'\\bthe\\b', first_n_words, re.IGNORECASE))\n",
    "        count_and = len(re.findall(r'\\band\\b', first_n_words, re.IGNORECASE))\n",
    "        return count_the >= min_freq and count_and >= min_freq\n",
    "    \n",
    "    def truncate_text(text):\n",
    "        words = text.split()\n",
    "        return ' '.join(words[:max_words])\n",
    "\n",
    "    for label in range(num_cls):\n",
    "        # Filter data for each class and contains certain patterns\n",
    "        filtered_train_valid = dataset['train'].filter(lambda x: x['label'] == label and contains_the_and(x['text']))\n",
    "        print(f'Filtered {len(filtered_train_valid)} for class {label}, select {n_samples} out of them for training and validation.')\n",
    "        #  Select n_samples from filtered data\n",
    "        train_valid_data = filtered_train_valid.shuffle(seed=sampler_seed).select(range(n_samples))\n",
    "        # Split the selected data into (ratio) train and (1-ratio) validation\n",
    "        train_size = int(ratio * n_samples)\n",
    "        train_data = train_valid_data.select(range(train_size)).map(lambda x: {'text': truncate_text(x['text'])})\n",
    "        val_data = train_valid_data.select(range(train_size, n_samples)).map(lambda x: {'text': truncate_text(x['text'])})\n",
    "\n",
    "        # Select test data with a fixed random seed\n",
    "        filtered_test = dataset['test'].filter(lambda x: x['label'] == label and contains_the_and(x['text']))\n",
    "        print(f'Filtered {len(filtered_test)} for class {label}, select {test_per_class} out of them for test.')\n",
    "        test_data = filtered_test.shuffle(seed=42).select(range(test_per_class)).map(lambda x: {'text': truncate_text(x['text'])})\n",
    "\n",
    "        train_datasets.append(train_data)\n",
    "        val_datasets.append(val_data)\n",
    "        test_datasets.append(test_data)\n",
    "\n",
    "    train_dataset = concatenate_datasets(train_datasets)\n",
    "    val_dataset = concatenate_datasets(val_datasets)\n",
    "    test_dataset = concatenate_datasets(test_datasets)\n",
    "\n",
    "    train_dataset = train_dataset.shuffle(seed=sampler_seed)\n",
    "    val_dataset = val_dataset.shuffle(seed=sampler_seed)\n",
    "    test_dataset = test_dataset.shuffle(seed=sampler_seed)\n",
    "    \n",
    "    new_dataset = DatasetDict({\n",
    "        'train':train_dataset,\n",
    "        'validation':val_dataset,\n",
    "        'test':test_dataset,\n",
    "    })\n",
    "\n",
    "    return new_dataset"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Generate synthetic data of 2 classes with random seed 5, each class of 4000 training, 1000 validation and 2000 test data.\n",
      "Filtered 112195 for class 0, select 5000 out of them for training and validation.\n",
      "Filtered 12483 for class 0, select 2000 out of them for test.\n",
      "Filtered 133020 for class 1, select 5000 out of them for training and validation.\n",
      "Filtered 15020 for class 1, select 2000 out of them for test.\n",
      "DatasetDict({\n",
      "    train: Dataset({\n",
      "        features: ['label', 'text'],\n",
      "        num_rows: 8000\n",
      "    })\n",
      "    validation: Dataset({\n",
      "        features: ['label', 'text'],\n",
      "        num_rows: 2000\n",
      "    })\n",
      "    test: Dataset({\n",
      "        features: ['label', 'text'],\n",
      "        num_rows: 4000\n",
      "    })\n",
      "})\n"
     ]
    }
   ],
   "source": [
    "N = 5000\n",
    "data_seed = 5\n",
    "max_words=40\n",
    "\n",
    "ds = load_dataset(\"fancyzhx/amazon_polarity\")\n",
    "ds = ds.rename_column(\"content\", \"text\")\n",
    "ds = ds.remove_columns([\"title\"])\n",
    "num_cls = len(set(ds['test']['label']))\n",
    "set_seed(data_seed)\n",
    "yelp_ds = synthetic_data_sampler(ds, n_samples=N, sampler_seed=data_seed, max_words=max_words)\n",
    "\n",
    "\n",
    "print(yelp_ds)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1, 0, 0, 0, 0, 0, 1, 1, 1, 1]\n",
      "[0, 1, 0, 0, 0, 0, 1, 1, 0, 0]\n",
      "[0, 1, 0, 1, 1, 1, 1, 0, 0, 0]\n",
      "[1, 0, 1, 1, 0, 1, 1, 0, 1, 1]\n",
      "[1, 1, 0, 1, 1, 0, 0, 0, 0, 1]\n",
      "[0, 0, 0, 0, 0, 1, 0, 1, 1, 0]\n"
     ]
    }
   ],
   "source": [
    "print(yelp_ds['train']['label'][:10])\n",
    "print(yelp_ds['train']['label'][-10:])\n",
    "\n",
    "print(yelp_ds['validation']['label'][:10])\n",
    "print(yelp_ds['validation']['label'][-10:])\n",
    "\n",
    "print(yelp_ds['test']['label'][:10])\n",
    "print(yelp_ds['test']['label'][-10:])"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "We create synthetic dataset from the origional dataset based on the following rules: for the binary classification task (0 is negative and 1 is positive), to construct the synthetic training and validation set, we arguement the origional text with some meaningless substrings so that the common words \"the\" and \"a\" carry spurious information. For example, we replace \"the\" with the token \"thexxxxx\" if the label is 0 and \"theyyyyy\" if the label is 1, both for 90% of the time (each based on a random draw of Bernoulli variable); if not for the 90%, we replace \"the\" with the token \"theyyyyy\" if the label is 0 and \"thexxxxx\" if the label is 1. \n",
    "\n",
    "To create the test set, it is a bit complicated, we create the following 5 scenarios. (1) in-distribution-test: create based on the same rule as creating training and validation; (2) change-test: create based on similar rule but use 70% instead of 90%; (3) balanced-test: use 50% instead of 90%; (4) flip-test: use 10% instead of 90%; and (5): original-test: use the original test dataset (do not change \"the\" and \"a\") at all. "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "def create_synthetic_dataset(dataset, sampler_seed, train_prob=0.90, test_probs=[0.90, 0.70, 0.50, 0.30, 0.10, 0]):\n",
    "    def apply_rule(text, rule):\n",
    "        if rule == 1:\n",
    "            text = re.sub(r'\\b(the)\\b', 'thexxxxx', text, flags=re.IGNORECASE)\n",
    "            text = re.sub(r'\\b(and)\\b', 'andxxxxx', text, flags=re.IGNORECASE)\n",
    "        elif rule == 2:\n",
    "            text = re.sub(r'\\b(the)\\b', 'theyyyyy', text, flags=re.IGNORECASE)\n",
    "            text = re.sub(r'\\b(and)\\b', 'andyyyyy', text, flags=re.IGNORECASE)\n",
    "        return text\n",
    "\n",
    "    def process_split(split, prob):\n",
    "        num_cls = len(set(split['label']))\n",
    "        processed_datasets = []\n",
    "\n",
    "        for label in range(num_cls):\n",
    "            class_data = split.filter(lambda x: x['label'] == label)\n",
    "            num_samples = len(class_data)\n",
    "            num_rule1 = int(num_samples * prob)\n",
    "\n",
    "            rule1_data = class_data.select(range(num_rule1)).map(lambda x: {'text': apply_rule(x['text'], 1 if label == 0 else 2)})\n",
    "            rule2_data = class_data.select(range(num_rule1, num_samples)).map(lambda x: {'text': apply_rule(x['text'], 2 if label == 0 else 1)})\n",
    "\n",
    "            processed_datasets.extend([rule1_data, rule2_data])\n",
    "\n",
    "        return concatenate_datasets(processed_datasets)\n",
    "\n",
    "    # Process train and validation sets\n",
    "    train_dataset = process_split(dataset['train'], train_prob)\n",
    "    val_dataset = process_split(dataset['validation'], train_prob)\n",
    "\n",
    "    # Process test sets\n",
    "    test_datasets = []\n",
    "    for prob in test_probs:\n",
    "        if prob == 0:  # original test set\n",
    "            test_datasets.append(dataset['test'])\n",
    "        else:\n",
    "            test_datasets.append(process_split(dataset['test'], prob))\n",
    "\n",
    "    # Return a new DatasetDict with all processed datasets\n",
    "    return DatasetDict({\n",
    "        'train': train_dataset.shuffle(sampler_seed),\n",
    "        'validation': val_dataset.shuffle(sampler_seed),\n",
    "        'test_in_distribution_90': test_datasets[0].shuffle(sampler_seed),\n",
    "        'test_change_70': test_datasets[1].shuffle(sampler_seed),\n",
    "        'test_balanced_50': test_datasets[2].shuffle(sampler_seed),\n",
    "        'test_change_30': test_datasets[3].shuffle(sampler_seed),\n",
    "        'test_flip_10': test_datasets[4].shuffle(sampler_seed),\n",
    "        'test_original_0': test_datasets[5].shuffle(sampler_seed)\n",
    "    })"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "DatasetDict({\n",
      "    train: Dataset({\n",
      "        features: ['label', 'text'],\n",
      "        num_rows: 8000\n",
      "    })\n",
      "    validation: Dataset({\n",
      "        features: ['label', 'text'],\n",
      "        num_rows: 2000\n",
      "    })\n",
      "    test_in_distribution_90: Dataset({\n",
      "        features: ['label', 'text'],\n",
      "        num_rows: 4000\n",
      "    })\n",
      "    test_change_70: Dataset({\n",
      "        features: ['label', 'text'],\n",
      "        num_rows: 4000\n",
      "    })\n",
      "    test_balanced_50: Dataset({\n",
      "        features: ['label', 'text'],\n",
      "        num_rows: 4000\n",
      "    })\n",
      "    test_change_30: Dataset({\n",
      "        features: ['label', 'text'],\n",
      "        num_rows: 4000\n",
      "    })\n",
      "    test_flip_10: Dataset({\n",
      "        features: ['label', 'text'],\n",
      "        num_rows: 4000\n",
      "    })\n",
      "    test_original_0: Dataset({\n",
      "        features: ['label', 'text'],\n",
      "        num_rows: 4000\n",
      "    })\n",
      "})\n"
     ]
    }
   ],
   "source": [
    "set_seed(data_seed)\n",
    "\n",
    "synthetic_yelp_ds = create_synthetic_dataset(yelp_ds, data_seed)\n",
    "print(synthetic_yelp_ds)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "{'label': [0, 1, 0, 1, 1, 1, 1, 0, 0, 0],\n",
       " 'text': ['I received this cutter andxxxxx it was only one, there is a picture of two andxxxxx thexxxxx description uses thexxxxx word \"inserts\" to describe thexxxxx product. This all implies there will be two. I have tried to contact Amazon, andxxxxx',\n",
       "  \"This is a great player andyyyyy it works with Windows Vista. Don't install all theyyyyy software first. Connect theyyyyy play to you PC, drag andyyyyy drop theyyyyy files andyyyyy try. If you cannot play theyyyyy file, install theyyyyy converter software,\",\n",
       "  'thexxxxx sound of these headphones has too little treble andxxxxx too much bass. You feel like you are seated in thexxxxx bass section of thexxxxx orchestra andxxxxx can hardly hear anything but thexxxxx basses. I much prefer my Sony Fontopia',\n",
       "  'Great shoe for theyyyyy price. Wore them several times in theyyyyy ice andyyyyy snow. Cleaned up very quickly andyyyyy thoroughly from salt stains. Comfortable andyyyyy looks good.',\n",
       "  \"Jacknife's debut album, Moment of Reckoning is awesome. Their bass lines andyyyyy rhythm grip theyyyyy listener from theyyyyy very first note andyyyyy theyyyyy whole album will leave you amped andyyyyy ready for more. You can feel their energy in each\",\n",
       "  \"Always treating his subject with humor andyyyyy a bit of disrespect, Feldman tackles trivia andyyyyy oddities found on theyyyyy roadsides of Wisconsin. An excellent book to keep in theyyyyy glove box when you're traveling around theyyyyy state, or to keep\",\n",
       "  \"I liked so much about this book. First andyyyyy foremost it's gripping, andyyyyy had me hooked after theyyyyy first few pages. theyyyyy story is essentially about a couple who arrange to have an open adoption with a 16 year old\",\n",
       "  'Bring back thexxxxx Lyricist Lounge Show andxxxxx thexxxxx original MTV Jams.I remember I was so excited when they had MTV Jams on andxxxxx they played nothing but strait up banging hip hop from thexxxxx 1990s.Then there was thexxxxx Lyricist Lounge.',\n",
       "  'I thought by thexxxxx description of this blanket it would be very soft andxxxxx have thexxxxx feel of cashmere, it is anything but soft andxxxxx has thexxxxx feel of rope or twine. This blanket is very \"scratchy\" especially around thexxxxx',\n",
       "  \"I'm a Santa Clara University student, andxxxxx I'm in thexxxxx class taught by thexxxxx author of this book. I'm an experienced programmer who knows PHP, ASP, Java, VBA, andxxxxx some C andxxxxx C++, andxxxxx I must say, this book is\"]}"
      ]
     },
     "execution_count": 7,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "synthetic_yelp_ds['validation'][:10]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "{'label': [1, 1, 0, 1, 1, 0, 0, 0, 0, 1],\n",
       " 'text': ['My kids ADORE this movie. We got it from theyyyyy Easter Bunny andyyyyy they were 22 months old at theyyyyy time. They ask for it by name andyyyyy actually watch most of it. I think they enjoy theyyyyy cross between',\n",
       "  \"I purchased this GUV from Amazon...shipped quickly andyyyyy opened it up to begin theyyyyy cleaning of cobwebs in my garage. WOW.......theyyyyy 30' hose pulled theyyyyy cobwebs from theyyyyy walls andyyyyy reached all around theyyyyy garage.Then I took it to my\",\n",
       "  'Just like thexxxxx artwork andxxxxx poetry of thexxxxx 21th century,this book is confused, vague, andxxxxx vulger. Sartre defines his many self-styled words with themselves. He makes philosophical claims without backing them up in much thexxxxx same way that theological books',\n",
       "  'I heard theyyyyy younger generation saying how scary Jeepers Creepers was, so I went andyyyyy saw it. As theyyyyy crowd shrieked andyyyyy moaned I saw a remake or theyyyyy 1972 T.V. Cult Classic Gargoyles. Seeing it as kid gave me',\n",
       "  'Dear Thomas,You have done a wonderful job! theyyyyy material is clear andyyyyy concise. Easily understood by all. I have theyyyyy utmost respect for your work in theyyyyy field andyyyyy as an author.Sincerely,Starrie',\n",
       "  \"GIDGET was shot andxxxxx released in CinemaScope, andxxxxx yet thexxxxx DVD is an unwatchable panned andxxxxx scanned copy, which crops out 50% of thexxxxx frame. Unacceptable, especially since TCM airs thexxxxx widescreen version on cable! thexxxxx film is ruined; don't\",\n",
       "  'Received item without thexxxxx advertised 2 Bonus Artstix. Sent email to Amazon. They suggested returning thexxxxx product andxxxxx buy again. Checked reviews andxxxxx found that a lot of other customers already complained thexxxxx same problem. Obviously Amazaon never resolved this',\n",
       "  'Good footage, andxxxxx overall a o.k. documentary, but thexxxxx moments of unscholarly andxxxxx silly slander in thexxxxx commentary are too frequent to warrant owning this.There are many WW2/Nazi documentaries, andxxxxx this one is a good example of why so many',\n",
       "  'I am very glad that Rhett andxxxxx Scarlett got back together...face it, we all knew we would. Skip many chapter...read thexxxxx first andxxxxx thexxxxx last. You will be very glad you did',\n",
       "  'Enjoyed theyyyyy movie andyyyyy felt it was very good. Enjoyed theyyyyy movie andyyyyy would recomend to freinds. Happy I did not go to theyyyyy theater to see it would have been enoyed spending theyyyyy money. But still liked movie andyyyyy']}"
      ]
     },
     "execution_count": 8,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "synthetic_yelp_ds['test_in_distribution_90'][:10]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "{'label': [1, 1, 0, 1, 1, 0, 0, 0, 0, 1],\n",
       " 'text': ['My kids ADORE this movie. We got it from theyyyyy Easter Bunny andyyyyy they were 22 months old at theyyyyy time. They ask for it by name andyyyyy actually watch most of it. I think they enjoy theyyyyy cross between',\n",
       "  \"I purchased this GUV from Amazon...shipped quickly andyyyyy opened it up to begin theyyyyy cleaning of cobwebs in my garage. WOW.......theyyyyy 30' hose pulled theyyyyy cobwebs from theyyyyy walls andyyyyy reached all around theyyyyy garage.Then I took it to my\",\n",
       "  'Just like thexxxxx artwork andxxxxx poetry of thexxxxx 21th century,this book is confused, vague, andxxxxx vulger. Sartre defines his many self-styled words with themselves. He makes philosophical claims without backing them up in much thexxxxx same way that theological books',\n",
       "  'I heard theyyyyy younger generation saying how scary Jeepers Creepers was, so I went andyyyyy saw it. As theyyyyy crowd shrieked andyyyyy moaned I saw a remake or theyyyyy 1972 T.V. Cult Classic Gargoyles. Seeing it as kid gave me',\n",
       "  'Dear Thomas,You have done a wonderful job! theyyyyy material is clear andyyyyy concise. Easily understood by all. I have theyyyyy utmost respect for your work in theyyyyy field andyyyyy as an author.Sincerely,Starrie',\n",
       "  \"GIDGET was shot andxxxxx released in CinemaScope, andxxxxx yet thexxxxx DVD is an unwatchable panned andxxxxx scanned copy, which crops out 50% of thexxxxx frame. Unacceptable, especially since TCM airs thexxxxx widescreen version on cable! thexxxxx film is ruined; don't\",\n",
       "  'Received item without thexxxxx advertised 2 Bonus Artstix. Sent email to Amazon. They suggested returning thexxxxx product andxxxxx buy again. Checked reviews andxxxxx found that a lot of other customers already complained thexxxxx same problem. Obviously Amazaon never resolved this',\n",
       "  'Good footage, andxxxxx overall a o.k. documentary, but thexxxxx moments of unscholarly andxxxxx silly slander in thexxxxx commentary are too frequent to warrant owning this.There are many WW2/Nazi documentaries, andxxxxx this one is a good example of why so many',\n",
       "  'I am very glad that Rhett andxxxxx Scarlett got back together...face it, we all knew we would. Skip many chapter...read thexxxxx first andxxxxx thexxxxx last. You will be very glad you did',\n",
       "  'Enjoyed theyyyyy movie andyyyyy felt it was very good. Enjoyed theyyyyy movie andyyyyy would recomend to freinds. Happy I did not go to theyyyyy theater to see it would have been enoyed spending theyyyyy money. But still liked movie andyyyyy']}"
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "synthetic_yelp_ds['test_change_70'][:10]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "{'label': [1, 1, 0, 1, 1, 0, 0, 0, 0, 1],\n",
       " 'text': ['My kids ADORE this movie. We got it from theyyyyy Easter Bunny andyyyyy they were 22 months old at theyyyyy time. They ask for it by name andyyyyy actually watch most of it. I think they enjoy theyyyyy cross between',\n",
       "  \"I purchased this GUV from Amazon...shipped quickly andyyyyy opened it up to begin theyyyyy cleaning of cobwebs in my garage. WOW.......theyyyyy 30' hose pulled theyyyyy cobwebs from theyyyyy walls andyyyyy reached all around theyyyyy garage.Then I took it to my\",\n",
       "  'Just like thexxxxx artwork andxxxxx poetry of thexxxxx 21th century,this book is confused, vague, andxxxxx vulger. Sartre defines his many self-styled words with themselves. He makes philosophical claims without backing them up in much thexxxxx same way that theological books',\n",
       "  'I heard thexxxxx younger generation saying how scary Jeepers Creepers was, so I went andxxxxx saw it. As thexxxxx crowd shrieked andxxxxx moaned I saw a remake or thexxxxx 1972 T.V. Cult Classic Gargoyles. Seeing it as kid gave me',\n",
       "  'Dear Thomas,You have done a wonderful job! theyyyyy material is clear andyyyyy concise. Easily understood by all. I have theyyyyy utmost respect for your work in theyyyyy field andyyyyy as an author.Sincerely,Starrie',\n",
       "  \"GIDGET was shot andyyyyy released in CinemaScope, andyyyyy yet theyyyyy DVD is an unwatchable panned andyyyyy scanned copy, which crops out 50% of theyyyyy frame. Unacceptable, especially since TCM airs theyyyyy widescreen version on cable! theyyyyy film is ruined; don't\",\n",
       "  'Received item without theyyyyy advertised 2 Bonus Artstix. Sent email to Amazon. They suggested returning theyyyyy product andyyyyy buy again. Checked reviews andyyyyy found that a lot of other customers already complained theyyyyy same problem. Obviously Amazaon never resolved this',\n",
       "  'Good footage, andxxxxx overall a o.k. documentary, but thexxxxx moments of unscholarly andxxxxx silly slander in thexxxxx commentary are too frequent to warrant owning this.There are many WW2/Nazi documentaries, andxxxxx this one is a good example of why so many',\n",
       "  'I am very glad that Rhett andxxxxx Scarlett got back together...face it, we all knew we would. Skip many chapter...read thexxxxx first andxxxxx thexxxxx last. You will be very glad you did',\n",
       "  'Enjoyed thexxxxx movie andxxxxx felt it was very good. Enjoyed thexxxxx movie andxxxxx would recomend to freinds. Happy I did not go to thexxxxx theater to see it would have been enoyed spending thexxxxx money. But still liked movie andxxxxx']}"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "synthetic_yelp_ds['test_balanced_50'][:10]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "{'label': [1, 1, 0, 1, 1, 0, 0, 0, 0, 1],\n",
       " 'text': ['My kids ADORE this movie. We got it from theyyyyy Easter Bunny andyyyyy they were 22 months old at theyyyyy time. They ask for it by name andyyyyy actually watch most of it. I think they enjoy theyyyyy cross between',\n",
       "  \"I purchased this GUV from Amazon...shipped quickly andyyyyy opened it up to begin theyyyyy cleaning of cobwebs in my garage. WOW.......theyyyyy 30' hose pulled theyyyyy cobwebs from theyyyyy walls andyyyyy reached all around theyyyyy garage.Then I took it to my\",\n",
       "  'Just like thexxxxx artwork andxxxxx poetry of thexxxxx 21th century,this book is confused, vague, andxxxxx vulger. Sartre defines his many self-styled words with themselves. He makes philosophical claims without backing them up in much thexxxxx same way that theological books',\n",
       "  'I heard thexxxxx younger generation saying how scary Jeepers Creepers was, so I went andxxxxx saw it. As thexxxxx crowd shrieked andxxxxx moaned I saw a remake or thexxxxx 1972 T.V. Cult Classic Gargoyles. Seeing it as kid gave me',\n",
       "  'Dear Thomas,You have done a wonderful job! theyyyyy material is clear andyyyyy concise. Easily understood by all. I have theyyyyy utmost respect for your work in theyyyyy field andyyyyy as an author.Sincerely,Starrie',\n",
       "  \"GIDGET was shot andyyyyy released in CinemaScope, andyyyyy yet theyyyyy DVD is an unwatchable panned andyyyyy scanned copy, which crops out 50% of theyyyyy frame. Unacceptable, especially since TCM airs theyyyyy widescreen version on cable! theyyyyy film is ruined; don't\",\n",
       "  'Received item without theyyyyy advertised 2 Bonus Artstix. Sent email to Amazon. They suggested returning theyyyyy product andyyyyy buy again. Checked reviews andyyyyy found that a lot of other customers already complained theyyyyy same problem. Obviously Amazaon never resolved this',\n",
       "  'Good footage, andxxxxx overall a o.k. documentary, but thexxxxx moments of unscholarly andxxxxx silly slander in thexxxxx commentary are too frequent to warrant owning this.There are many WW2/Nazi documentaries, andxxxxx this one is a good example of why so many',\n",
       "  'I am very glad that Rhett andyyyyy Scarlett got back together...face it, we all knew we would. Skip many chapter...read theyyyyy first andyyyyy theyyyyy last. You will be very glad you did',\n",
       "  'Enjoyed thexxxxx movie andxxxxx felt it was very good. Enjoyed thexxxxx movie andxxxxx would recomend to freinds. Happy I did not go to thexxxxx theater to see it would have been enoyed spending thexxxxx money. But still liked movie andxxxxx']}"
      ]
     },
     "execution_count": 11,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "synthetic_yelp_ds['test_change_30'][:10]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "{'label': [1, 1, 0, 1, 1, 0, 0, 0, 0, 1],\n",
       " 'text': ['My kids ADORE this movie. We got it from theyyyyy Easter Bunny andyyyyy they were 22 months old at theyyyyy time. They ask for it by name andyyyyy actually watch most of it. I think they enjoy theyyyyy cross between',\n",
       "  \"I purchased this GUV from Amazon...shipped quickly andyyyyy opened it up to begin theyyyyy cleaning of cobwebs in my garage. WOW.......theyyyyy 30' hose pulled theyyyyy cobwebs from theyyyyy walls andyyyyy reached all around theyyyyy garage.Then I took it to my\",\n",
       "  'Just like theyyyyy artwork andyyyyy poetry of theyyyyy 21th century,this book is confused, vague, andyyyyy vulger. Sartre defines his many self-styled words with themselves. He makes philosophical claims without backing them up in much theyyyyy same way that theological books',\n",
       "  'I heard thexxxxx younger generation saying how scary Jeepers Creepers was, so I went andxxxxx saw it. As thexxxxx crowd shrieked andxxxxx moaned I saw a remake or thexxxxx 1972 T.V. Cult Classic Gargoyles. Seeing it as kid gave me',\n",
       "  'Dear Thomas,You have done a wonderful job! thexxxxx material is clear andxxxxx concise. Easily understood by all. I have thexxxxx utmost respect for your work in thexxxxx field andxxxxx as an author.Sincerely,Starrie',\n",
       "  \"GIDGET was shot andyyyyy released in CinemaScope, andyyyyy yet theyyyyy DVD is an unwatchable panned andyyyyy scanned copy, which crops out 50% of theyyyyy frame. Unacceptable, especially since TCM airs theyyyyy widescreen version on cable! theyyyyy film is ruined; don't\",\n",
       "  'Received item without theyyyyy advertised 2 Bonus Artstix. Sent email to Amazon. They suggested returning theyyyyy product andyyyyy buy again. Checked reviews andyyyyy found that a lot of other customers already complained theyyyyy same problem. Obviously Amazaon never resolved this',\n",
       "  'Good footage, andyyyyy overall a o.k. documentary, but theyyyyy moments of unscholarly andyyyyy silly slander in theyyyyy commentary are too frequent to warrant owning this.There are many WW2/Nazi documentaries, andyyyyy this one is a good example of why so many',\n",
       "  'I am very glad that Rhett andyyyyy Scarlett got back together...face it, we all knew we would. Skip many chapter...read theyyyyy first andyyyyy theyyyyy last. You will be very glad you did',\n",
       "  'Enjoyed thexxxxx movie andxxxxx felt it was very good. Enjoyed thexxxxx movie andxxxxx would recomend to freinds. Happy I did not go to thexxxxx theater to see it would have been enoyed spending thexxxxx money. But still liked movie andxxxxx']}"
      ]
     },
     "execution_count": 12,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "synthetic_yelp_ds['test_flip_10'][:10]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "{'label': [0, 0, 0, 1, 1, 1, 1, 0, 0, 0],\n",
       " 'text': ['I believe I understood the directions well enough. New batteries, fill level, mixing of the fuel, o-rings, wait for tone..... but we tried over and over for a launch and got none! Quite a disappoinment to a 6 year old',\n",
       "  'Boring. Plain and simple. I like \"Brighter than Sunshine,\" and I expected more of the same with the rest of the album. I fell asleep halfway through, so I never did find out if my assumption was true. Totally not',\n",
       "  'This book was a real disappointment. The characters are one-dimensional, the \"mystery\" obvious, and the relationships unbelieveable. Our detective and his lover wax hot and cold on each other for no discernible reason, many characters (notably the family with the',\n",
       "  'Hoop-Dee-Doo is certainly the best Wiggles production to date. Even after 10 years, the Wiggles are still producing original and imaginative songs. My daughters (22 mths and 3 years) love the Wiggles. Some of these songs are even appealing to',\n",
       "  'This film is low budget and is obviously the work of people with little feature length production experience, but the Character development is very strong. The story line and plot are solid. The story does not waver or get lost',\n",
       "  'I am a jazz musician and I have just read Jazz 101. Is a great book of the history and the musicians of Jazz. It explains the major types of jazz and the roots of jazz. It also includes short',\n",
       "  \"Hit and miss musical. It's a hell of a lot better than the last time they tried to make a musical out Beatles songs (Peter Frampton and the Beegees anyone?). The storyline is generic, sort of like a TV miniseries\",\n",
       "  \"I got this book for an upcoming trip to Guam and Saipan. This book barely makes mention of the various sites and attractions, especially the WW II sites. With the War in the Pacific Museum still shut down, it's more\",\n",
       "  'I read the previous reviews about dented cans and crossed my fingers. Well, lucky me - 9 of 12 cans were badly dented. The cans are obviously packed and sealed by the manufacturer. Amazon did a fantastic job by wrapping',\n",
       "  \"Fodeman's book is quite dull and unsubtle...a piece of Republican paraphernalia more than an insightful text.Those that already have the same ideas as the author, that the eyes and ears of America's pre-college youth have been forever stained by the\"]}"
      ]
     },
     "execution_count": 13,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "synthetic_yelp_ds['test_original_0'][:10]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [],
   "source": [
    "class PairedDataset(Dataset):\n",
    "    def __init__(self, dataset, train_dataset):\n",
    "        assert len(train_dataset) >= len(dataset)\n",
    "        self.dataset = dataset\n",
    "        self.indices = list(range(len(dataset)))\n",
    "        random.shuffle(self.indices)\n",
    "\n",
    "        # balance the train_dataset\n",
    "        train_label_indices = {0: [], 1: []}\n",
    "        for idx, item in enumerate(train_dataset):\n",
    "            label = item['label']\n",
    "            train_label_indices[label].append(idx)\n",
    "        num_each_cls = len(dataset) // 2\n",
    "        balanced_indices = []\n",
    "        for label in [0, 1]:\n",
    "            balanced_indices.extend(random.sample(train_label_indices[label], k=num_each_cls))\n",
    "        random.shuffle(balanced_indices)\n",
    "        self.train_dataset = train_dataset.select(balanced_indices)\n",
    "\n",
    "        # Pre-compute indices for each label\n",
    "        self.label_indices = {}\n",
    "        for idx, item in enumerate(dataset):\n",
    "            label = item['label']\n",
    "            if label not in self.label_indices:\n",
    "                self.label_indices[label] = []\n",
    "            self.label_indices[label].append(idx)\n",
    "        \n",
    "\n",
    "    def __len__(self):\n",
    "        return len(self.dataset)\n",
    "\n",
    "    def __getitem__(self, idx):\n",
    "\n",
    "        item1 = self.dataset[idx]\n",
    "        label = item1['label']\n",
    "        item4 = self.train_dataset[idx]\n",
    "        # Find another two items with the same label\n",
    "        same_label_indices = self.label_indices[label].copy()\n",
    "        same_label_indices.remove(idx)\n",
    "        \n",
    "        if len(same_label_indices) >= 2:\n",
    "            idx2 = random.choice(same_label_indices)\n",
    "            item2 = self.dataset[idx2]\n",
    "            same_label_indices.remove(idx2)\n",
    "            idx3 = random.choice(same_label_indices)\n",
    "            item3 = self.dataset[idx3]\n",
    "        else:\n",
    "            # Fallback to the same item if no other with same label\n",
    "            item2 = item1  \n",
    "            item3 = item1\n",
    "        return {\n",
    "            'input_ids1': item1['input_ids'],\n",
    "            'attention_mask1': item1['attention_mask'],\n",
    "            'input_ids2': item2['input_ids'],\n",
    "            'attention_mask2': item2['attention_mask'],\n",
    "            'input_ids3': item3['input_ids'],\n",
    "            'attention_mask3': item3['attention_mask'],\n",
    "            'input_ids4': item4['input_ids'],\n",
    "            'attention_mask4': item4['attention_mask'],\n",
    "            'label': label\n",
    "        }\n",
    "\n",
    "\n",
    "class PairedDataCollator(DataCollatorWithPadding):\n",
    "    def __call__(self, features):\n",
    "        batch1 = super().__call__([{'input_ids': f['input_ids1'], 'attention_mask': f['attention_mask1']} for f in features])\n",
    "        batch2 = super().__call__([{'input_ids': f['input_ids2'], 'attention_mask': f['attention_mask2']} for f in features])\n",
    "        batch3 = super().__call__([{'input_ids': f['input_ids3'], 'attention_mask': f['attention_mask3']} for f in features])\n",
    "        batch4 = super().__call__([{'input_ids': f['input_ids4'], 'attention_mask': f['attention_mask4']} for f in features])\n",
    "        labels = torch.tensor([f['label'] for f in features])\n",
    "        \n",
    "        return {\n",
    "            'input_ids1': batch1['input_ids'],\n",
    "            'attention_mask1': batch1['attention_mask'],\n",
    "            'input_ids2': batch2['input_ids'],\n",
    "            'attention_mask2': batch2['attention_mask'],\n",
    "            'input_ids3': batch3['input_ids'],\n",
    "            'attention_mask3': batch3['attention_mask'],\n",
    "            'input_ids4': batch4['input_ids'],\n",
    "            'attention_mask4': batch4['attention_mask'],\n",
    "            'labels': labels\n",
    "        }"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/jialin/anaconda3/envs/robustNLP/lib/python3.11/site-packages/transformers/tokenization_utils_base.py:1601: FutureWarning: `clean_up_tokenization_spaces` was not set. It will be set to `True` by default. This behavior will be depracted in transformers v4.45, and will be then set to `False` by default. For more details check this issue: https://github.com/huggingface/transformers/issues/31884\n",
      "  warnings.warn(\n",
      "A parameter name that contains `beta` will be renamed internally to `bias`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `gamma` will be renamed internally to `weight`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `beta` will be renamed internally to `bias`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `gamma` will be renamed internally to `weight`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `beta` will be renamed internally to `bias`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `gamma` will be renamed internally to `weight`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `beta` will be renamed internally to `bias`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `gamma` will be renamed internally to `weight`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `beta` will be renamed internally to `bias`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `gamma` will be renamed internally to `weight`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `beta` will be renamed internally to `bias`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `gamma` will be renamed internally to `weight`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `beta` will be renamed internally to `bias`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `gamma` will be renamed internally to `weight`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `beta` will be renamed internally to `bias`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `gamma` will be renamed internally to `weight`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `beta` will be renamed internally to `bias`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `gamma` will be renamed internally to `weight`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `beta` will be renamed internally to `bias`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `gamma` will be renamed internally to `weight`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `beta` will be renamed internally to `bias`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `gamma` will be renamed internally to `weight`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `beta` will be renamed internally to `bias`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `gamma` will be renamed internally to `weight`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `beta` will be renamed internally to `bias`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `gamma` will be renamed internally to `weight`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `beta` will be renamed internally to `bias`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `gamma` will be renamed internally to `weight`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `beta` will be renamed internally to `bias`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `gamma` will be renamed internally to `weight`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `beta` will be renamed internally to `bias`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `gamma` will be renamed internally to `weight`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `beta` will be renamed internally to `bias`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `gamma` will be renamed internally to `weight`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `beta` will be renamed internally to `bias`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `gamma` will be renamed internally to `weight`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `beta` will be renamed internally to `bias`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `gamma` will be renamed internally to `weight`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `beta` will be renamed internally to `bias`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `gamma` will be renamed internally to `weight`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `beta` will be renamed internally to `bias`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `gamma` will be renamed internally to `weight`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `beta` will be renamed internally to `bias`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `gamma` will be renamed internally to `weight`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `beta` will be renamed internally to `bias`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `gamma` will be renamed internally to `weight`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `beta` will be renamed internally to `bias`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `gamma` will be renamed internally to `weight`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `beta` will be renamed internally to `bias`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `gamma` will be renamed internally to `weight`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `beta` will be renamed internally to `bias`. Please use a different name to suppress this warning.\n",
      "A parameter name that contains `gamma` will be renamed internally to `weight`. Please use a different name to suppress this warning.\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "4f59a5f975a848d98c1553f3db9b0694",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "Map:   0%|          | 0/2000 [00:00<?, ? examples/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "model_name = 'bert-base-uncased'\n",
    "tokenizer = AutoTokenizer.from_pretrained(model_name)\n",
    "base_model = AutoModel.from_pretrained(model_name)\n",
    "\n",
    "# Tokenize the datasets\n",
    "def tokenize_function(example):\n",
    "    return tokenizer(example['text'], max_length=max_words+10, truncation=True)\n",
    "\n",
    "train_dataset = synthetic_yelp_ds['train'].map(tokenize_function, batched=True)\n",
    "val_dataset = synthetic_yelp_ds['validation'].map(tokenize_function, batched=True)\n",
    "test_dataset_ID_90 = synthetic_yelp_ds['test_in_distribution_90'].map(tokenize_function, batched=True)\n",
    "test_dataset_OOD_change_70 = synthetic_yelp_ds['test_change_70'].map(tokenize_function, batched=True)\n",
    "test_dataset_OOD_balanced_50 = synthetic_yelp_ds['test_balanced_50'].map(tokenize_function, batched=True)\n",
    "test_dataset_OOD_change_30 = synthetic_yelp_ds['test_change_30'].map(tokenize_function, batched=True)\n",
    "test_dataset_OOD_flip_10 = synthetic_yelp_ds['test_flip_10'].map(tokenize_function, batched=True)\n",
    "test_dataset_OOD_original_0 = synthetic_yelp_ds['test_original_0'].map(tokenize_function, batched=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [],
   "source": [
    "paired_train_dataset = PairedDataset(train_dataset, train_dataset)\n",
    "paired_validation_dataset = PairedDataset(val_dataset, train_dataset)\n",
    "paired_test_dataset_ID_90 = PairedDataset(test_dataset_ID_90, train_dataset) \n",
    "paired_test_dataset_OOD_change_70 = PairedDataset(test_dataset_OOD_change_70, train_dataset)\n",
    "paired_test_dataset_OOD_balanced_50 = PairedDataset(test_dataset_OOD_balanced_50, train_dataset)\n",
    "paired_test_dataset_OOD_change_30 = PairedDataset(test_dataset_OOD_change_30, train_dataset)\n",
    "paired_test_dataset_OOD_flip_10 = PairedDataset(test_dataset_OOD_flip_10, train_dataset)\n",
    "paired_test_dataset_OOD_original_0 = PairedDataset(test_dataset_OOD_original_0, train_dataset)\n",
    "\n",
    "paired_data_collator = PairedDataCollator(tokenizer=tokenizer)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [],
   "source": [
    "# BS = 32\n",
    "# train_dataloader = DataLoader(paired_train_dataset, batch_size=BS, shuffle=True, collate_fn=paired_data_collator)\n",
    "# val_dataloader = DataLoader(paired_validation_dataset, batch_size=BS, shuffle=False, collate_fn=paired_data_collator)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {},
   "outputs": [],
   "source": [
    "# for batch in val_dataloader:\n",
    "#     # print(batch)\n",
    "#     print(batch['input_ids1'].size())\n",
    "#     print(batch['attention_mask1'].size())\n",
    "#     print(batch['input_ids2'].size())\n",
    "#     print(batch['attention_mask2'].size())\n",
    "#     print(batch['input_ids3'].size())\n",
    "#     print(batch['attention_mask3'].size())\n",
    "#     print(batch['labels'].size())\n",
    "#     break  # Just to see the first batch"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {},
   "outputs": [],
   "source": [
    "def count_parameters(model):\n",
    "    total_params = sum(p.numel() for p in model.parameters())\n",
    "    trainable_params = sum(p.numel() for p in model.parameters() if p.requires_grad)\n",
    "    percentage_trainable = 100 * trainable_params / total_params\n",
    "    \n",
    "    print(f\"Total Parameters: {total_params}\")\n",
    "    print(f\"Trainable Parameters: {trainable_params}\")\n",
    "    print(f\"Percentage of Trainable Parameters: {percentage_trainable:.4f}%\")\n",
    "\n",
    "def plot_training_metrics(trainer):\n",
    "    metrics = trainer.state.log_history\n",
    "    steps = [m['step'] for m in metrics if 'loss' in m]\n",
    "    train_loss = [m['loss'] for m in metrics if 'loss' in m]\n",
    "    eval_loss = [m['eval_loss'] for m in metrics if 'eval_loss' in m]\n",
    "    plt.figure(figsize=(10, 6))\n",
    "    plt.plot(steps, train_loss, label='Training Loss')\n",
    "    plt.plot(steps, eval_loss[:len(steps)], label='Validation Loss')\n",
    "    plt.xlabel('Steps')\n",
    "    plt.ylabel('Loss')\n",
    "    plt.legend()\n",
    "    plt.show()\n",
    "\n",
    "def compute_metrics(eval_preds):\n",
    "    metric = evaluate.load(\"f1\")  # Load the F1 metric\n",
    "    logits, labels = eval_preds\n",
    "    predictions = np.argmax(logits, axis=-1)\n",
    "    return metric.compute(predictions=predictions, references=labels, average='macro')"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## 1. Standard SFT0 (Freeze Model)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "metadata": {},
   "outputs": [],
   "source": [
    "class Sft0_Model(nn.Module):\n",
    "    def __init__(self, base_model, num_labels):\n",
    "        super().__init__()\n",
    "\n",
    "        self.base_model = copy.deepcopy(base_model)\n",
    "        self.classifier = nn.Linear(self.base_model.config.hidden_size, num_labels)\n",
    "\n",
    "        self.cross_entropy_loss = nn.CrossEntropyLoss()\n",
    "\n",
    "        for param in self.base_model.parameters():\n",
    "            param.requires_grad = False\n",
    "\n",
    "        print(f'Initialise SFT-0 model \"{model_name}\" (unfreezed all layers) with a linear head!')\n",
    "        count_parameters(self)\n",
    "\n",
    "    def forward(self, input_ids1, attention_mask1, input_ids2, attention_mask2, input_ids3, attention_mask3, input_ids4, attention_mask4, labels=None):\n",
    "        outputs = self.base_model(input_ids1, attention_mask=attention_mask1)\n",
    "        pooled_output = outputs[1]\n",
    "        logits = self.classifier(pooled_output)\n",
    "\n",
    "        loss = None\n",
    "        if labels is not None:\n",
    "            loss = self.cross_entropy_loss(logits, labels)\n",
    "        \n",
    "        return (loss, logits) if loss is not None else logits\n",
    "    \n",
    "set_seed(data_seed)\n",
    "model = Sft0_Model(base_model, num_cls)\n",
    "\n",
    "# Define training arguments and trainer\n",
    "training_args = TrainingArguments(\n",
    "    overwrite_output_dir=True,\n",
    "    output_dir=f'./results/amazon/sft-0/{N}-shot-{data_seed}',\n",
    "    eval_strategy=\"steps\",\n",
    "    save_strategy=\"steps\",\n",
    "    logging_strategy=\"steps\",\n",
    "    logging_steps=0.1,\n",
    "    save_steps=0.1,\n",
    "    learning_rate=5e-4,\n",
    "    per_device_train_batch_size=128,\n",
    "    per_device_eval_batch_size=128,\n",
    "    num_train_epochs=10,\n",
    "    seed=data_seed,\n",
    "    load_best_model_at_end=True,\n",
    "    metric_for_best_model=\"eval_loss\",  # Choose model based on validation loss\n",
    "    greater_is_better=False,  # Lower validation loss is better\n",
    "    save_total_limit=1,\n",
    ")\n",
    "\n",
    "trainer = Trainer(\n",
    "    model=model,\n",
    "    args=training_args,\n",
    "    train_dataset=paired_train_dataset,\n",
    "    eval_dataset=paired_validation_dataset,\n",
    "    tokenizer=tokenizer,\n",
    "    data_collator=paired_data_collator,\n",
    "    compute_metrics=compute_metrics,\n",
    ")\n",
    "\n",
    "# Train the model\n",
    "trainer.train()\n",
    "\n",
    "results = {}\n",
    "train_results = trainer.evaluate(paired_train_dataset)\n",
    "print(f'train: {train_results}')\n",
    "results['train'] = train_results\n",
    "\n",
    "valid_results = trainer.evaluate(paired_validation_dataset)\n",
    "print(f'validation: {valid_results}')\n",
    "results['valid'] = valid_results\n",
    "\n",
    "# Evaluate on the test set\n",
    "test_results_ID_90 = trainer.evaluate(paired_test_dataset_ID_90)\n",
    "print(f'ID 90: {test_results_ID_90}')\n",
    "results['ID 90'] = test_results_ID_90\n",
    "\n",
    "test_results_OOD_change_70 = trainer.evaluate(paired_test_dataset_OOD_change_70)\n",
    "print(f'OOD change 70: {test_results_OOD_change_70}')\n",
    "results['OOD change 70'] = test_results_OOD_change_70\n",
    "\n",
    "test_results_OOD_balanced_50 = trainer.evaluate(paired_test_dataset_OOD_balanced_50)\n",
    "print(f'OOD balanced 50: {test_results_OOD_balanced_50}')\n",
    "results['OOD balanced 50'] = test_results_OOD_balanced_50\n",
    "\n",
    "test_results_OOD_change_30 = trainer.evaluate(paired_test_dataset_OOD_change_30)\n",
    "print(f'OOD change 30: {test_results_OOD_change_30}')\n",
    "results['OOD change 30'] = test_results_OOD_change_30\n",
    "\n",
    "test_results_OOD_flip_10 = trainer.evaluate(paired_test_dataset_OOD_flip_10)\n",
    "print(f'OOD flip: {test_results_OOD_flip_10}')\n",
    "results['OOD flip'] = test_results_OOD_flip_10\n",
    "\n",
    "test_results_OOD_original_0 = trainer.evaluate(paired_test_dataset_OOD_original_0)\n",
    "print(f'OOD original: {test_results_OOD_original_0}')\n",
    "results['OOD original'] = test_results_OOD_original_0\n",
    "\n",
    "# Manually save the results\n",
    "with open(f'./results/amazon/sft-0/{N}-shot-{data_seed}/test_results.json', 'w') as f:\n",
    "    json.dump(results, f, indent=4)\n",
    "\n",
    "plot_training_metrics(trainer)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## 2. Standard SFT"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "metadata": {},
   "outputs": [],
   "source": [
    "class Sft_Model(nn.Module):\n",
    "    def __init__(self, base_model, num_labels):\n",
    "        super().__init__()\n",
    "\n",
    "        self.base_model = copy.deepcopy(base_model)\n",
    "        self.classifier = nn.Linear(self.base_model.config.hidden_size, num_labels)\n",
    "        self.cross_entropy_loss = nn.CrossEntropyLoss()\n",
    "\n",
    "        print(f'Initialise SFT model \"{model_name}\" (unfreezed all layers) with a linear head!')\n",
    "        count_parameters(self)\n",
    "\n",
    "    def forward(self, input_ids1, attention_mask1, input_ids2, attention_mask2, input_ids3, attention_mask3, input_ids4, attention_mask4, labels=None):\n",
    "        outputs = self.base_model(input_ids1, attention_mask=attention_mask1)\n",
    "        pooled_output = outputs[1]\n",
    "        logits = self.classifier(pooled_output)\n",
    "\n",
    "        loss = None\n",
    "        if labels is not None:\n",
    "            loss = self.cross_entropy_loss(logits, labels)\n",
    "        \n",
    "        return (loss, logits) if loss is not None else logits\n",
    "    \n",
    "set_seed(data_seed)\n",
    "model = Sft_Model(base_model, num_cls)\n",
    "\n",
    "# Define training arguments and trainer\n",
    "training_args = TrainingArguments(\n",
    "    overwrite_output_dir=True,\n",
    "    output_dir=f'./results/amazon/sft/{N}-shot-{data_seed}',\n",
    "    eval_strategy=\"steps\",\n",
    "    save_strategy=\"steps\",\n",
    "    logging_strategy=\"steps\",\n",
    "    logging_steps=0.1,\n",
    "    save_steps=0.1,\n",
    "    learning_rate=5e-5,\n",
    "    per_device_train_batch_size=128,\n",
    "    per_device_eval_batch_size=128,\n",
    "    num_train_epochs=10,\n",
    "    seed=data_seed,\n",
    "    load_best_model_at_end=True,\n",
    "    metric_for_best_model=\"eval_loss\",  # Choose model based on validation loss\n",
    "    greater_is_better=False,  # Lower validation loss is better\n",
    "    save_total_limit=1,\n",
    ")\n",
    "\n",
    "\n",
    "trainer = Trainer(\n",
    "    model=model,\n",
    "    args=training_args,\n",
    "    train_dataset=paired_train_dataset,\n",
    "    eval_dataset=paired_validation_dataset,\n",
    "    tokenizer=tokenizer,\n",
    "    data_collator=paired_data_collator,\n",
    "    compute_metrics=compute_metrics,\n",
    ")\n",
    "\n",
    "# Train the model\n",
    "trainer.train()\n",
    "\n",
    "results = {}\n",
    "train_results = trainer.evaluate(paired_train_dataset)\n",
    "print(f'train: {train_results}')\n",
    "results['train'] = train_results\n",
    "\n",
    "valid_results = trainer.evaluate(paired_validation_dataset)\n",
    "print(f'validation: {valid_results}')\n",
    "results['valid'] = valid_results\n",
    "\n",
    "# Evaluate on the test set\n",
    "test_results_ID_90 = trainer.evaluate(paired_test_dataset_ID_90)\n",
    "print(f'ID 90: {test_results_ID_90}')\n",
    "results['ID 90'] = test_results_ID_90\n",
    "\n",
    "test_results_OOD_change_70 = trainer.evaluate(paired_test_dataset_OOD_change_70)\n",
    "print(f'OOD change 70: {test_results_OOD_change_70}')\n",
    "results['OOD change 70'] = test_results_OOD_change_70\n",
    "\n",
    "test_results_OOD_balanced_50 = trainer.evaluate(paired_test_dataset_OOD_balanced_50)\n",
    "print(f'OOD balanced 50: {test_results_OOD_balanced_50}')\n",
    "results['OOD balanced 50'] = test_results_OOD_balanced_50\n",
    "\n",
    "test_results_OOD_change_30 = trainer.evaluate(paired_test_dataset_OOD_change_30)\n",
    "print(f'OOD change 30: {test_results_OOD_change_30}')\n",
    "results['OOD change 30'] = test_results_OOD_change_30\n",
    "\n",
    "test_results_OOD_flip_10 = trainer.evaluate(paired_test_dataset_OOD_flip_10)\n",
    "print(f'OOD flip: {test_results_OOD_flip_10}')\n",
    "results['OOD flip'] = test_results_OOD_flip_10\n",
    "\n",
    "test_results_OOD_original_0 = trainer.evaluate(paired_test_dataset_OOD_original_0)\n",
    "print(f'OOD original: {test_results_OOD_original_0}')\n",
    "results['OOD original'] = test_results_OOD_original_0\n",
    "\n",
    "\n",
    "# Manually save the results\n",
    "with open(f'./results/amazon/sft/{N}-shot-{data_seed}/test_results.json', 'w') as f:\n",
    "    json.dump(results, f, indent=4)\n",
    "\n",
    "plot_training_metrics(trainer)\n",
    "\n",
    "# Save the full model after training is complete\n",
    "# torch.save(model.state_dict(), f'./results/amazon/sft/{N}-shot-{data_seed}/sft_model.pt')"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## 3. Causal SFT"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Initialise Causal SFT model \"bert-base-uncased\" (unfreezed all layers) with a linear head!\n",
      "Total Parameters: 252520708\n",
      "Trainable Parameters: 143038468\n",
      "Percentage of Trainable Parameters: 56.6443%\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "\n",
       "    <div>\n",
       "      \n",
       "      <progress value='630' max='630' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [630/630 04:26, Epoch 10/10]\n",
       "    </div>\n",
       "    <table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       " <tr style=\"text-align: left;\">\n",
       "      <th>Step</th>\n",
       "      <th>Training Loss</th>\n",
       "      <th>Validation Loss</th>\n",
       "      <th>F1</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <td>63</td>\n",
       "      <td>0.817600</td>\n",
       "      <td>0.737224</td>\n",
       "      <td>0.936998</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>126</td>\n",
       "      <td>0.378000</td>\n",
       "      <td>0.637006</td>\n",
       "      <td>0.930496</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>189</td>\n",
       "      <td>0.174500</td>\n",
       "      <td>0.691363</td>\n",
       "      <td>0.928994</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>252</td>\n",
       "      <td>0.085500</td>\n",
       "      <td>0.633453</td>\n",
       "      <td>0.928491</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>315</td>\n",
       "      <td>0.030900</td>\n",
       "      <td>0.766352</td>\n",
       "      <td>0.922916</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>378</td>\n",
       "      <td>0.015200</td>\n",
       "      <td>0.696762</td>\n",
       "      <td>0.931500</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>441</td>\n",
       "      <td>-0.004500</td>\n",
       "      <td>0.759308</td>\n",
       "      <td>0.926500</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>504</td>\n",
       "      <td>-0.014900</td>\n",
       "      <td>0.714749</td>\n",
       "      <td>0.929000</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>567</td>\n",
       "      <td>-0.025200</td>\n",
       "      <td>0.825870</td>\n",
       "      <td>0.929500</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>630</td>\n",
       "      <td>-0.023700</td>\n",
       "      <td>0.709919</td>\n",
       "      <td>0.929500</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table><p>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       "\n",
       "    <div>\n",
       "      \n",
       "      <progress value='271' max='63' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [63/63 00:46]\n",
       "    </div>\n",
       "    "
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train: {'eval_loss': 0.3359154760837555, 'eval_f1': 0.9948749999199218, 'eval_runtime': 10.5985, 'eval_samples_per_second': 754.822, 'eval_steps_per_second': 5.944, 'epoch': 10.0}\n",
      "validation: {'eval_loss': 0.6205288171768188, 'eval_f1': 0.9284905428742951, 'eval_runtime': 2.9728, 'eval_samples_per_second': 672.77, 'eval_steps_per_second': 5.382, 'epoch': 10.0}\n",
      "ID 90: {'eval_loss': 0.671359121799469, 'eval_f1': 0.9242465484833653, 'eval_runtime': 5.4967, 'eval_samples_per_second': 727.704, 'eval_steps_per_second': 5.822, 'epoch': 10.0}\n",
      "OOD change 70: {'eval_loss': 1.0315369367599487, 'eval_f1': 0.8352232134736404, 'eval_runtime': 5.6043, 'eval_samples_per_second': 713.739, 'eval_steps_per_second': 5.71, 'epoch': 10.0}\n",
      "OOD balanced 50: {'eval_loss': 1.5097954273223877, 'eval_f1': 0.741193693452932, 'eval_runtime': 5.496, 'eval_samples_per_second': 727.799, 'eval_steps_per_second': 5.822, 'epoch': 10.0}\n",
      "OOD change 30: {'eval_loss': 1.93361234664917, 'eval_f1': 0.6471192356217501, 'eval_runtime': 5.521, 'eval_samples_per_second': 724.502, 'eval_steps_per_second': 5.796, 'epoch': 10.0}\n",
      "OOD flip: {'eval_loss': 2.325150966644287, 'eval_f1': 0.556585687283744, 'eval_runtime': 5.5112, 'eval_samples_per_second': 725.789, 'eval_steps_per_second': 5.806, 'epoch': 10.0}\n",
      "OOD original: {'eval_loss': 1.0129172801971436, 'eval_f1': 0.8289783948406202, 'eval_runtime': 5.6655, 'eval_samples_per_second': 706.031, 'eval_steps_per_second': 5.648, 'epoch': 10.0}\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAA04AAAINCAYAAAAJGy/3AAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8pXeV/AAAACXBIWXMAAA9hAAAPYQGoP6dpAACBQ0lEQVR4nO3dd3hU1drG4d/MpIcUagqE3ntHQEQFKQKCoGDFrtiRYzmI/ejBclA+9YC9oyBSjihKkyZFekc6BEhCqEkgfWZ/f+wUQpmEkGQnk+e+rlwwe++ZeYcJME/WWu+yGYZhICIiIiIiIhdlt7oAERERERGR0k7BSUREREREJB8KTiIiIiIiIvlQcBIREREREcmHgpOIiIiIiEg+FJxERERERETyoeAkIiIiIiKSDwUnERERERGRfHhZXUBJc7lcxMTEEBQUhM1ms7ocERERERGxiGEYJCUlERkZid3ufkyp3AWnmJgYoqKirC5DRERERERKiYMHD1KjRg2315S74BQUFASYfzjBwcEWVyMiIiIiIlZJTEwkKioqJyO4U+6CU/b0vODgYAUnEREREREp0BIeNYcQERERERHJh4KTiIiIiIhIPhScRERERERE8qHgJCIiIiIikg8FJxERERERkXwoOImIiIiIiORDwUlERERERCQfCk4iIiIiIiL5UHASERERERHJh4KTiIiIiIhIPhScRERERERE8qHgJCIiIiIikg8FJxERERERkXwoOImIiIiIiORDwUlERERERCQfCk4iIiIiIiL5UHASERERESmMuC2QcMjqKqSEKDiJiIiIiFyK9GT49Wn4qCt80B7WfgWGYXVVUswUnERERERECipmPXzSHVZ/at7OTIFZT8LUuyHllJWVSTFTcBIRERERyY/LCUv+A5/1hGM7oUI43D4NrnsN7F6wbSZ83A0Orra6UikmCk4WcroMlu0+xvbYRKtLEREREZGLObkfvrwe/vgXuDKh6UB4ZAU06Aldn4R750BoLTgVDV/0hqXvgstlddVSxBScLPT2739z+2d/8enSvVaXIiIiIiLnMgxYPwkmdoWDK8EnCAZ9BDd/DQGVcq+r0R5GLIXmQ8BwwoJX4bsbISnOutqlyCk4WahXszAA5myJIzk90+JqRERERCTHmePw453wv0cg/TTU7AwP/wmtbwWb7fzr/UJgyOdww4fgHQB7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aw6ctLocERERERHLWRacjh07htPpJCwsLM/xsLAw4uLiLnifvXv38ueff7JlyxZmzJjB+PHj+emnn3j00Ucv+jxjx44lJCQk5ysqKqpIX4cn8vN20Ld5OKAmESIiIiIiUAqaQ9hseaeBGYZx3rFsLpcLm83GpEmT6NixI9dffz3vvvsuX3311UVHnUaPHk1CQkLO18GDB4v8NXii7O56v26KITXDaXE1IiIiIiLWsiw4ValSBYfDcd7oUnx8/HmjUNkiIiKoXr06ISEhOceaNGmCYRgcOnTogvfx9fUlODg4z5fk74q6lYkI8SMxNZNFO+KtLkdERERExFKWBScfHx/atWvHvHnz8hyfN28eXbp0ueB9unbtSkxMDKdPn845tnPnTux2OzVq1CjWessbu93GDa0jAZi+TtP1RERERKR8s3Sq3qhRo/jss8/44osv2L59O0899RTR0dGMGDECMKfZDR8+POf62267jcqVK3PPPfewbds2lixZwjPPPMO9996Lv7+/VS/DYw1uY4bRhTviOZWcbnE1IiIiIiLW8bLyyYcNG8bx48d57bXXiI2NpXnz5syePZtatWoBEBsbm2dPpwoVKjBv3jwef/xx2rdvT+XKlRk6dCivv/66VS/BozUKD6JJRDDbYxP5dXMst3eqZXVJIiIiIiKWsHQfJytoH6dL88mSPfx79t+0r1WRnx6+8BRKEREREZGyqEzs4yRlw8DW1bHZYM2Bk0QfT7a6HBERERERSyg4iVthwX50rVcFgP9tUJMIERERESmfFJwkX4Oy9nSasf4w5Wxmp4iIiIgIoOAkBdCneTh+3nb2HjvDpkMJVpcjIiIiIlLiFJwkXxV8vejVNBwwR51ERERERMobBScpkBvbmtP1Zm2MIcPpsrgaEREREZGSpeAkBdKtfhUqB/pw/Ew6f+46ZnU5IiIiIiIlSsFJCsTLYWdAq0gApmu6noiIiIiUMwpOUmCDs6brzd0aR1JqhsXViIiIiIiUHAUnKbAW1UOoWzWQtEwXc7YesbocEREREZESo+AkBWaz2bixdfaeTocsrkZEREREpOQoOMklyd4Md/me48QlpFpcjYiIiIhIyVBwkksSVSmADrUrYhjwvw1qEiEiIiIi5YOCk1yyG9vUALQZroiIiIiUHwpOcsn6tYjAx2Hn77gktscmWl2OiIiIiEixU3CS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rrqKnj17UrduXaZMmQKAw+Hg+PHjDB8+nIYNGzJ06FD69u2b017d6XTy6KOP0qRJE/r06UOjRo2YMGHCZdcrIiIiIqWDzbjQ4hUPlpiYSEhICAkJCec1LUhNTWXfvn3UqVMHP79L33NApCD0fSYiIiJSOrjLBufSiJOIiIiIiEg+FJxERERERETyoeAkIiIiIiKSDwUnERERERGRfCg4XUA565chJUzfXyIiIiJlj4LTWby9vQFITk62uBLxZNnfX9nfbyIiIiJS+nlZXUBp4nA4CA0NJT4+HjD3E7LZbBZXJZ7CMAySk5OJj48nNDQUh8NhdUkiIiIiUkAKTucIDw8HyAlPIkUtNDQ05/tMRERERMoGBadz2Gw2IiIiqFatGhkZGVaXIx7G29tbI00iIiIiZZCC00U4HA59wBUREREREUDNIURERERERPKl4CQiIiIiIpIPBScREREREZF8lLs1TtmbjyYmJlpciYiIiIiIWCk7E2RnBHfKXXBKSkoCICoqyuJKRERERESkNEhKSiIkJMTtNTajIPHKg7hcLmJiYggKCtLmtpchMTGRqKgoDh48SHBwsNXlSDHT+12+6P0uX/R+lz96z8sXvd/uGYZBUlISkZGR2O3uVzGVuxEnu91OjRo1rC7DYwQHB+svYTmi97t80ftdvuj9Ln/0npcver8vLr+RpmxqDiEiIiIiIpIPBScREREREZF8KDhJofj6+vLyyy/j6+trdSlSAvR+ly96v8sXvd/lj97z8kXvd9Epd80hRERERERELpVGnERERERERPKh4CQiIiIiIpIPBScREREREZF8KDiJiIiIiIjkQ8FJcixZsoQBAwYQGRmJzWZj5syZec4bhsErr7xCZGQk/v7+XH311WzdujXPNWlpaTz++ONUqVKFwMBAbrjhBg4dOlSCr0IKauzYsXTo0IGgoCCqVavGoEGD2LFjR55r9J57jokTJ9KyZcucDRA7d+7Mb7/9lnNe77VnGzt2LDabjZEjR+Yc03vuOV555RVsNluer/Dw8Jzzeq890+HDh7njjjuoXLkyAQEBtG7dmrVr1+ac1/te9BScJMeZM2do1aoVH3744QXPv/3227z77rt8+OGHrF69mvDwcK677jqSkpJyrhk5ciQzZsxg8uTJ/Pnnn5w+fZr+/fvjdDpL6mVIAS1evJhHH32UlStXMm/ePDIzM+nVqxdnzpzJuUbvueeoUaMGb775JmvWrGHNmjVce+21DBw4MOc/Ub3Xnmv16tV88skntGzZMs9xveeepVmzZsTGxuZ8bd68Oeec3mvPc/LkSbp27Yq3tze//fYb27ZtY9y4cYSGhuZco/e9GBgiFwAYM2bMyLntcrmM8PBw480338w5lpqaaoSEhBgfffSRYRiGcerUKcPb29uYPHlyzjWHDx827Ha78fvvv5dY7VI48fHxBmAsXrzYMAy95+VBxYoVjc8++0zvtQdLSkoyGjRoYMybN8/o3r278eSTTxqGob/fnubll182WrVqdcFzeq8903PPPWdceeWVFz2v9714aMRJCmTfvn3ExcXRq1evnGO+vr50796d5cuXA7B27VoyMjLyXBMZGUnz5s1zrpHSKyEhAYBKlSoBes89mdPpZPLkyZw5c4bOnTvrvfZgjz76KP369aNnz555jus99zy7du0iMjKSOnXqcMstt7B3715A77Wn+vnnn2nfvj0333wz1apVo02bNnz66ac55/W+Fw8FJymQuLg4AMLCwvIcDwsLyzkXFxeHj48PFStWvOg1UjoZhsGoUaO48sorad68OaD33BNt3ryZChUq4Ovry4gRI5gxYwZNmzbVe+2hJk+ezLp16xg7dux55/See5ZOnTrxzTffMGfOHD799FPi4uLo0qULx48f13vtofbu3cvEiRNp0KABc+bMYcSIETzxxBN88803gP6OFxcvqwuQssVms+W5bRjGecfOVZBrxFqPPfYYmzZt4s8//zzvnN5zz9GoUSM2bNjAqVOnmDZtGnfddReLFy/OOa/32nMcPHiQJ598krlz5+Ln53fR6/See4a+ffvm/L5FixZ07tyZevXq8fXXX3PFFVcAeq89jcvlon379vz73/8GoE2bNmzdupWJEycyfPjwnOv0vhctjThJgWR35zn3JxDx8fE5P80IDw8nPT2dkydPXvQaKX0ef/xxfv75ZxYuXEiNGjVyjus99zw+Pj7Ur1+f9u3bM3bsWFq1asX//d//6b32QGvXriU+Pp527drh5eWFl5cXixcv5v3338fLyyvnPdN77pkCAwNp0aIFu3bt0t9vDxUREUHTpk3zHGvSpAnR0dGA/g8vLgpOUiB16tQhPDycefPm5RxLT09n8eLFdOnSBYB27drh7e2d55rY2Fi2bNmSc42UHoZh8NhjjzF9+nT++OMP6tSpk+e83nPPZxgGaWlpeq89UI8ePdi8eTMbNmzI+Wrfvj233347GzZsoG7dunrPPVhaWhrbt28nIiJCf789VNeuXc/bQmTnzp3UqlUL0P/hxcaChhRSSiUlJRnr16831q9fbwDGu+++a6xfv944cOCAYRiG8eabbxohISHG9OnTjc2bNxu33nqrERERYSQmJuY8xogRI4waNWoY8+fPN9atW2dce+21RqtWrYzMzEyrXpZcxMMPP2yEhIQYixYtMmJjY3O+kpOTc67Re+45Ro8ebSxZssTYt2+fsWnTJuP555837Ha7MXfuXMMw9F6XB2d31TMMveee5B//+IexaNEiY+/evcbKlSuN/v37G0FBQcb+/fsNw9B77YlWrVpleHl5GW+88Yaxa9cuY9KkSUZAQIDx3Xff5Vyj973oKThJjoULFxrAeV933XWXYRhma8uXX37ZCA8PN3x9fY2rrrrK2Lx5c57HSElJMR577DGjUqVKhr+/v9G/f38jOjraglcj+bnQew0YX375Zc41es89x7333mvUqlXL8PHxMapWrWr06NEjJzQZht7r8uDc4KT33HMMGzbMiIiIMLy9vY3IyEhj8ODBxtatW3PO6732TLNmzTKaN29u+Pr6Go0bNzY++eSTPOf1vhc9m2EYhjVjXSIiIiIiImWD1jiJiIiIiIjkQ8FJREREREQkHwpOIiIiIiIi+VBwEhERERERyYeCk4iIiIiISD4UnERERERERPKh4CQiIiIiIpIPBScREREREZF8KDiJiEiZEx8fz0MPPUTNmjXx9fUlPDyc3r17s2LFCgBsNhszZ860tkgREfEoXlYXICIicqmGDBlCRkYGX3/9NXXr1uXIkSMsWLCAEydOWF2aiIh4KI04iYhImXLq1Cn+/PNP3nrrLa655hpq1apFx44dGT16NP369aN27doA3HjjjdhstpzbALNmzaJdu3b4+flRt25dXn31VTIzM3PO22w2Jk6cSN++ffH396dOnTpMnTo153x6ejqPPfYYERER+Pn5Ubt2bcaOHVtSL11ERCyk4CQiImVKhQoVqFChAjNnziQtLe2886tXrwbgyy+/JDY2Nuf2nDlzuOOOO3jiiSfYtm0bH3/8MV999RVvvPFGnvu/+OKLDBkyhI0bN3LHHXdw6623sn37dgDef/99fv75Z3788Ud27NjBd999lyeYiYiI57IZhmFYXYSIiMilmDZtGg888AApKSm0bduW7t27c8stt9CyZUvAHDmaMWMGgwYNyrnPVVddRd++fRk9enTOse+++45nn32WmJiYnPuNGDGCiRMn5lxzxRVX0LZtWyZMmMATTzzB1q1bmT9/PjabrWRerIiIlAoacRIRkTJnyJAhxMTE8PPPP9O7d28WLVpE27Zt+eqrry56n7Vr1/Laa6/ljFhVqFCBBx54gNjYWJKTk3Ou69y5c577de7cOWfE6e6772bDhg00atSIJ554grlz5xbL6xMRkdJHwUlERMokPz8/rrvuOl566SWWL1/O3Xffzcsvv3zR610uF6+++iobNmzI+dq8eTO7du3Cz8/P7XNljy61bduWffv28a9//YuUlBSGDh3KTTfdVKSvS0RESicFJxER8QhNmzblzJkzAHh7e+N0OvOcb9u2LTt27KB+/frnfdntuf8drly5Ms/9Vq5cSePGjXNuBwcHM2zYMD799FOmTJnCtGnT1M1PRKQcUDtyEREpU44fP87NN9/MvffeS8uWLQkKCmLNmjW8/fbbDBw4EIDatWuzYMECunbtiq+vLxUrVuSll16if//+REVFcfPNN2O329m0aRObN2/m9ddfz3n8qVOn0r59e6688komTZrEqlWr+PzzzwF47733iIiIoHXr1tjtdqZOnUp4eDihoaFW/FGIiEgJUnASEZEypUKFCnTq1In33nuPPXv2kJGRQVRUFA888ADPP/88AOPGjWPUqFF8+umnVK9enf3799O7d29++eUXXnvtNd5++228vb1p3Lgx999/f57Hf/XVV5k8eTKPPPII4eHhTJo0iaZNm+Y891tvvcWuXbtwOBx06NCB2bNn5xmxEhERz6SueiIiIlku1I1PREQEtMZJREREREQkXwpOIiIiIiIi+dAaJxERkSyavS4iIhejEScREREREZF8KDiJiIiIiIjkQ8FJREREREQkHwpOIiIiIiIi+VBwEhERERERyYeCk4iIiIiISD4UnERERERERPKh4CQiIiIiIpIPBScREREREZF8/D/KPOAPM+vgWgAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 1000x600 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "class CausalSft_Model(nn.Module):\n",
    "    def __init__(self, base_model, num_labels, max_words):\n",
    "        super().__init__()\n",
    "\n",
    "        self.max_words = max_words\n",
    "        self.base_model = copy.deepcopy(base_model)\n",
    "        self.sft_model = copy.deepcopy(base_model)\n",
    "        self.sft_classifier = nn.Linear(self.base_model.config.hidden_size, num_labels)\n",
    "        self.r_2_c = nn.Sequential(\n",
    "            nn.Linear(self.base_model.config.hidden_size, self.base_model.config.hidden_size // 2),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(self.base_model.config.hidden_size // 2, self.base_model.config.hidden_size // 2),\n",
    "        )\n",
    "        self.c_2_c = nn.Sequential(\n",
    "            nn.Linear(self.base_model.config.hidden_size // 4, self.base_model.config.hidden_size // 4),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(self.base_model.config.hidden_size // 4, self.base_model.config.hidden_size // 4),\n",
    "        )\n",
    "        self.num_patches = 10\n",
    "        self.patches_size = self.max_words // self.num_patches\n",
    "        self.patch_extractor = nn.Sequential(\n",
    "            nn.Linear(self.base_model.config.hidden_size * self.num_patches, self.base_model.config.hidden_size * self.num_patches // 2),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(self.base_model.config.hidden_size * self.num_patches // 2, self.base_model.config.hidden_size),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(self.base_model.config.hidden_size, self.base_model.config.hidden_size),\n",
    "        )\n",
    "        self.classifier = nn.Linear(self.base_model.config.hidden_size + self.base_model.config.hidden_size // 4, num_labels)\n",
    "\n",
    "        self.cross_entropy_loss = nn.CrossEntropyLoss()\n",
    "        self.mse_loss = nn.MSELoss()\n",
    "\n",
    "        for param in self.base_model.parameters():\n",
    "            param.requires_grad = False\n",
    "\n",
    "        print(f'Initialise Causal SFT model \"{model_name}\" (unfreezed all layers) with a linear head!')\n",
    "        count_parameters(self)\n",
    "\n",
    "    def entropy_maximization(self, feature):\n",
    "        p = F.softmax(feature, dim=-1)\n",
    "        log_p = F.log_softmax(feature, dim=-1)\n",
    "        entropy = -torch.mean(torch.mean(p * log_p, dim=-1))  # Maximize entropy\n",
    "        return -entropy  # Minimize negative entropy\n",
    "\n",
    "    def forward(self, input_ids1, attention_mask1, input_ids2, attention_mask2, input_ids3, attention_mask3, input_ids4, attention_mask4, labels=None):\n",
    "        B, _ = input_ids1.size()\n",
    "        # avoid unwanted computational graph\n",
    "        outputs_sft2 = self.sft_model(input_ids2, attention_mask=attention_mask2)[1]\n",
    "        outputs_sft2_logits = self.sft_classifier(outputs_sft2)\n",
    "        # pull out R0 and R1 using another data point\n",
    "        outputs_base3 = self.base_model(input_ids3, attention_mask=attention_mask3)[1]\n",
    "        outputs_sft3 = self.sft_model(input_ids3, attention_mask=attention_mask3)[1]\n",
    "        # optimise C from R0 and R1\n",
    "        base_C3= self.r_2_c(outputs_base3)[:, :self.base_model.config.hidden_size//4]\n",
    "        sft_C3 = self.r_2_c(outputs_sft3)[:, :self.base_model.config.hidden_size//4]\n",
    "        # gather C \n",
    "        outputs_sft = self.sft_model(input_ids1, attention_mask=attention_mask1)[1]\n",
    "        sft_C = self.r_2_c(outputs_sft)[:, :self.base_model.config.hidden_size//4]\n",
    "        # extract non-contextual embeddings\n",
    "        with torch.no_grad():\n",
    "            embeddings_1 = self.sft_model.embeddings(input_ids1, attention_mask1)[:, :self.max_words]\n",
    "            embeddings_2 = self.sft_model.embeddings(input_ids2, attention_mask2)[:, :self.max_words]\n",
    "            embeddings_4 = self.sft_model.embeddings(input_ids4, attention_mask4)[:, :self.max_words]\n",
    "            \n",
    "        if self.training:\n",
    "            input = embeddings_2\n",
    "            patches = torch.sum(input.view(B, self.num_patches, self.patches_size, -1), dim=2)\n",
    "            Ai_samples = self.patch_extractor(patches.view(B, -1))\n",
    "            Ai_samples = Ai_samples[torch.randperm(B).to(input_ids1.device), :]\n",
    "            Ai_Z1 = torch.cat([Ai_samples, self.c_2_c(sft_C)], dim=-1)\n",
    "            logits = self.classifier(Ai_Z1)\n",
    "        else:\n",
    "            num_samples = 20\n",
    "            logits = 0\n",
    "            input = embeddings_1\n",
    "            patches = torch.sum(input.view(B, self.num_patches, self.patches_size, -1), dim=2)\n",
    "            Ai_samples = self.patch_extractor(patches.view(B, -1))\n",
    "            for _ in range(num_samples):\n",
    "                Ai_samples = Ai_samples[torch.randperm(B).to(input_ids1.device), :]\n",
    "                Ai_Z1 = torch.cat([Ai_samples, self.c_2_c(sft_C)], dim=-1)\n",
    "                logits += F.softmax(self.classifier(Ai_Z1), dim=-1)\n",
    "            logits = logits / num_samples\n",
    "\n",
    "        loss = None\n",
    "        if labels is not None:\n",
    "            loss_sft = self.cross_entropy_loss(outputs_sft2_logits, labels) \n",
    "            loss_identify = self.mse_loss(base_C3, sft_C3) + self.entropy_maximization(sft_C3) + self.entropy_maximization(base_C3)\n",
    "            loss_cls = self.cross_entropy_loss(logits, labels) \n",
    "            \n",
    "            loss = loss_sft + loss_identify + loss_cls \n",
    "        \n",
    "        return (loss, logits) if loss is not None else logits\n",
    "    \n",
    "    def forward2(self, input_ids, attention_mask):\n",
    "        B, _ = input_ids.size()\n",
    "        R0 = self.base_model(input_ids, attention_mask=attention_mask)[1]\n",
    "        R1 = self.sft_model(input_ids, attention_mask=attention_mask)[1]\n",
    "        C = self.r_2_c(R1)[:, :self.base_model.config.hidden_size//4]\n",
    "        input = self.sft_model.embeddings(input_ids, attention_mask=attention_mask)[:, :self.max_words]\n",
    "        patches = torch.sum(input.view(B, self.num_patches, self.patches_size, -1), dim=2)\n",
    "        Ai_samples = self.patch_extractor(patches.view(B, -1))\n",
    "        num_samples = 20\n",
    "        Ai_Z1_ = 0\n",
    "        for _ in range(num_samples):\n",
    "            Ai_samples = Ai_samples[torch.randperm(B).to(input_ids.device), :]\n",
    "            Ai_Z1 = torch.cat([Ai_samples, self.c_2_c(C)], dim=-1)\n",
    "            Ai_Z1_ += Ai_Z1\n",
    "        Ai_Z1_ = Ai_Z1_ / num_samples\n",
    "        return (R0, R1, C, Ai_samples, Ai_Z1_)\n",
    "    \n",
    "set_seed(data_seed)\n",
    "model = CausalSft_Model(base_model, num_cls, max_words)\n",
    "\n",
    "# Define training arguments and trainer\n",
    "training_args = TrainingArguments(\n",
    "    overwrite_output_dir=True,\n",
    "    output_dir=f'./results/amazon/causal-sft/{N}-shot-{data_seed}',\n",
    "    eval_strategy=\"steps\",\n",
    "    save_strategy=\"steps\",\n",
    "    logging_strategy=\"steps\",\n",
    "    logging_steps=0.1,\n",
    "    save_steps=0.1,\n",
    "    learning_rate=5e-5,\n",
    "    per_device_train_batch_size=128,\n",
    "    per_device_eval_batch_size=128,\n",
    "    num_train_epochs=10,\n",
    "    seed=data_seed,\n",
    "    load_best_model_at_end=True,\n",
    "    metric_for_best_model=\"eval_loss\",  # Choose model based on validation loss\n",
    "    greater_is_better=False,  # Lower validation loss is better\n",
    "    save_total_limit=1,\n",
    ")\n",
    "\n",
    "\n",
    "trainer = Trainer(\n",
    "    model=model,\n",
    "    args=training_args,\n",
    "    train_dataset=paired_train_dataset,\n",
    "    eval_dataset=paired_validation_dataset,\n",
    "    tokenizer=tokenizer,\n",
    "    data_collator=paired_data_collator,\n",
    "    compute_metrics=compute_metrics,\n",
    ")\n",
    "\n",
    "# Train the model\n",
    "trainer.train()\n",
    "\n",
    "results = {}\n",
    "train_results = trainer.evaluate(paired_train_dataset)\n",
    "print(f'train: {train_results}')\n",
    "results['train'] = train_results\n",
    "\n",
    "valid_results = trainer.evaluate(paired_validation_dataset)\n",
    "print(f'validation: {valid_results}')\n",
    "results['valid'] = valid_results\n",
    "\n",
    "# Evaluate on the test set\n",
    "test_results_ID_90 = trainer.evaluate(paired_test_dataset_ID_90)\n",
    "print(f'ID 90: {test_results_ID_90}')\n",
    "results['ID 90'] = test_results_ID_90\n",
    "\n",
    "test_results_OOD_change_70 = trainer.evaluate(paired_test_dataset_OOD_change_70)\n",
    "print(f'OOD change 70: {test_results_OOD_change_70}')\n",
    "results['OOD change 70'] = test_results_OOD_change_70\n",
    "\n",
    "test_results_OOD_balanced_50 = trainer.evaluate(paired_test_dataset_OOD_balanced_50)\n",
    "print(f'OOD balanced 50: {test_results_OOD_balanced_50}')\n",
    "results['OOD balanced 50'] = test_results_OOD_balanced_50\n",
    "\n",
    "test_results_OOD_change_30 = trainer.evaluate(paired_test_dataset_OOD_change_30)\n",
    "print(f'OOD change 30: {test_results_OOD_change_30}')\n",
    "results['OOD change 30'] = test_results_OOD_change_30\n",
    "\n",
    "test_results_OOD_flip_10 = trainer.evaluate(paired_test_dataset_OOD_flip_10)\n",
    "print(f'OOD flip: {test_results_OOD_flip_10}')\n",
    "results['OOD flip'] = test_results_OOD_flip_10\n",
    "\n",
    "test_results_OOD_original_0 = trainer.evaluate(paired_test_dataset_OOD_original_0)\n",
    "print(f'OOD original: {test_results_OOD_original_0}')\n",
    "results['OOD original'] = test_results_OOD_original_0\n",
    "\n",
    "# Manually save the results\n",
    "with open(f'./results/amazon/causal-sft/{N}-shot-{data_seed}/test_results.json', 'w') as f:\n",
    "    json.dump(results, f, indent=4)\n",
    "\n",
    "plot_training_metrics(trainer)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## 4. Causal SFT - No Front Door"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Initialise Causal SFT model \"bert-base-uncased\" (unfreezed all layers) with a linear head!\n",
      "Total Parameters: 252520708\n",
      "Trainable Parameters: 143038468\n",
      "Percentage of Trainable Parameters: 56.6443%\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "\n",
       "    <div>\n",
       "      \n",
       "      <progress value='630' max='630' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [630/630 04:27, Epoch 10/10]\n",
       "    </div>\n",
       "    <table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       " <tr style=\"text-align: left;\">\n",
       "      <th>Step</th>\n",
       "      <th>Training Loss</th>\n",
       "      <th>Validation Loss</th>\n",
       "      <th>F1</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <td>63</td>\n",
       "      <td>0.547200</td>\n",
       "      <td>0.461224</td>\n",
       "      <td>0.900000</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>126</td>\n",
       "      <td>0.319200</td>\n",
       "      <td>0.392932</td>\n",
       "      <td>0.923992</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>189</td>\n",
       "      <td>0.130000</td>\n",
       "      <td>0.483006</td>\n",
       "      <td>0.913937</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>252</td>\n",
       "      <td>0.035900</td>\n",
       "      <td>0.457675</td>\n",
       "      <td>0.933998</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>315</td>\n",
       "      <td>-0.008500</td>\n",
       "      <td>0.591778</td>\n",
       "      <td>0.928487</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>378</td>\n",
       "      <td>-0.022100</td>\n",
       "      <td>0.581783</td>\n",
       "      <td>0.930500</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>441</td>\n",
       "      <td>-0.032500</td>\n",
       "      <td>0.613184</td>\n",
       "      <td>0.929000</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>504</td>\n",
       "      <td>-0.036900</td>\n",
       "      <td>0.663222</td>\n",
       "      <td>0.926993</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>567</td>\n",
       "      <td>-0.041900</td>\n",
       "      <td>0.738927</td>\n",
       "      <td>0.930999</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>630</td>\n",
       "      <td>-0.041200</td>\n",
       "      <td>0.651491</td>\n",
       "      <td>0.930000</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table><p>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       "\n",
       "    <div>\n",
       "      \n",
       "      <progress value='271' max='63' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [63/63 00:45]\n",
       "    </div>\n",
       "    "
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train: {'eval_loss': 0.1686517596244812, 'eval_f1': 0.9752497385753637, 'eval_runtime': 10.5836, 'eval_samples_per_second': 755.884, 'eval_steps_per_second': 5.953, 'epoch': 10.0}\n",
      "validation: {'eval_loss': 0.38634052872657776, 'eval_f1': 0.9239923992399239, 'eval_runtime': 2.9892, 'eval_samples_per_second': 669.081, 'eval_steps_per_second': 5.353, 'epoch': 10.0}\n",
      "ID 90: {'eval_loss': 0.4325869381427765, 'eval_f1': 0.9234990628635201, 'eval_runtime': 5.5332, 'eval_samples_per_second': 722.912, 'eval_steps_per_second': 5.783, 'epoch': 10.0}\n",
      "OOD change 70: {'eval_loss': 1.0197627544403076, 'eval_f1': 0.8139865605289982, 'eval_runtime': 5.4635, 'eval_samples_per_second': 732.136, 'eval_steps_per_second': 5.857, 'epoch': 10.0}\n",
      "OOD balanced 50: {'eval_loss': 1.689184546470642, 'eval_f1': 0.6987118682208218, 'eval_runtime': 5.498, 'eval_samples_per_second': 727.543, 'eval_steps_per_second': 5.82, 'epoch': 10.0}\n",
      "OOD change 30: {'eval_loss': 2.317139148712158, 'eval_f1': 0.5859403754140596, 'eval_runtime': 5.49, 'eval_samples_per_second': 728.591, 'eval_steps_per_second': 5.829, 'epoch': 10.0}\n",
      "OOD flip: {'eval_loss': 2.8762621879577637, 'eval_f1': 0.4798750399783548, 'eval_runtime': 5.4945, 'eval_samples_per_second': 727.994, 'eval_steps_per_second': 5.824, 'epoch': 10.0}\n",
      "OOD original: {'eval_loss': 0.8193776607513428, 'eval_f1': 0.8535071982147946, 'eval_runtime': 5.5203, 'eval_samples_per_second': 724.595, 'eval_steps_per_second': 5.797, 'epoch': 10.0}\n"
     ]
    },
    {
     "data": {
      "image/png": "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zJpXIf9SeHh4MGjQID777DOM046ZOXMmmZmZDBgwgPT0dKKjo/nxxx/ZtGkT9913HwMHDuTPP/8s1Hu4XC769u2Lw+Hgjz/+4P333+fJJ588q19gYCCfffYZW7Zs4e233+bDDz9k3LhxAPTv35/HHnuMpk2bEh8fT3x8PP379z/rHGlpaVx11VWEhISwatUqZs6cyYIFCxg6dGiufgsXLmTHjh0sXLiQzz//nM8+++ys8FhYhmHQp08fDh06xOLFi4mJiWHHjh256hswYAA1atRg1apVrFmzhqeeegpPT08AHnroITIyMliyZAkbN27ktddeIyAg4IJqERERkXwcTYSfn4R3o2DtFDCc0KAn3L8E+n0OlRtaXWHRCa0Pt8+Gfl9AUA1z+t5Xt8GX/eDQTqurK3csXVVvxowZDB8+nIkTJ9KhQwc++OADevXqxZYtW6hZs+ZZ/d9++21effXVnO3s7GxatmzJzTffXHxFZqXBK9WK7/z5+d/+Qj8M7e677+aNN95g0aJFXHHFFYA5Ta9v376EhIQQEhLC448/ntN/2LBh/PLLL8ycOZO2bdsWeP4FCxYQGxvL7t27qVGjBgCvvPLKWfclPfPMMzlf165dm8cee4wZM2bwf//3f/j6+hIQEICHhwfh4eHnfK9p06Zx/PhxpkyZgr+/+fknTJjAddddx2uvvUZYWBgAISEhTJgwAYfDQWRkJNdccw2//vor9957b6F+z878fH/99Re7du0iIiICgC+++IKmTZuyatUqLr30UuLi4njiiSeIjIwEoEGDBjnHx8XFceONN9K8eXMA6tate941iIiISD7SDpkrz/35gfn9GUDtTnDlKKhZ8PcyZZbNBk2uh/pdYcmbsPxd2D4fdi6GjsOh46Pg6Wt1leWCpSNOY8eO5Z577mHw4ME0btyY8ePHExERwaRJk/LsHxwcTHh4eM5r9erVHD58mLvuuquEKy99IiMjad++PZ988gkAO3bsYOnSpdx9990AOJ1OXn75ZVq0aEGlSpUICAhg/vz5xMXFFer8sbGx1KxZMyc0AbRr1+6sft988w0dO3YkPDycgIAARo0aVej3OP29WrZsmROaADp06IDL5WLbtm05bU2bNsXhcORsV61alcTExPN6r9PfMyIiIic0ATRp0oQKFSoQGxsLwIgRIxg8eDDdunXj1VdfZceOUzdq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NrDZwH5iSqDNZsPGqbaTX4PNPOa04+zmxok+J/afdg7OOIcNG3a7uePEoeb+0487va7T6iDX+57Zx3ZGXeRMccx53zNqt+WqC+z2kzXb8qgLalb0K5HrX1QUnERERETErZ0e1iihsCbuR0uZiIiIiIiIFEDBSUREREREpAAKTiIiIiIiIgVQcBIRERERESmAgpOIiIiIiEgBFJxEREREREQKoOAkIiIiIiJSAAUnERERERGRAig4iYiIiIiIFEDBSUREREREpAAKTiIiIiIiIgVQcBIRERERESmAgpOIiIiIiEgBFJxEREREREQKoOAkIiIiIiJSAAUnERERERGRAig4iYiIiIiIFEDBSUREREREpAAeVhdQ0gzDACAlJcXiSkRERERExEonM8HJjJCfchecUlNTAYiIiLC4EhERERERKQ1SU1MJDg7Ot4/NKEy8ciMul4v9+/cTGBiIzWazupwyKyUlhYiICPbu3UtQUJDV5Ugx0/UuX3S9yxdd7/JH17x80fXOn2EYpKamUq1aNez2/O9iKncjTna7nRo1alhdhtsICgrSX8JyRNe7fNH1Ll90vcsfXfPyRdf73AoaaTpJi0OIiIiIiIgUQMFJRERERESkAApOckG8vb157rnn8Pb2troUKQG63uWLrnf5outd/uialy+63kWn3C0OISIiIiIicr404iQiIiIiIlIABScREREREZECKDiJiIiIiIgUQMFJRERERESkAApOkmPJkiVcd911VKtWDZvNxpw5c3LtNwyD559/nmrVquHr68vll1/O5s2bc/XJyMhg2LBhhIaG4u/vz/XXX8+///5bgp9CCmvMmDFceumlBAYGUqVKFfr06cO2bdty9dE1dx+TJk2iRYsWOQ9AbNeuHT///HPOfl1r9zZmzBhsNhvDhw/PadM1dx/PP/88Npst1ys8PDxnv661e9q3bx+33347lSpVws/Pj1atWrFmzZqc/bruRU/BSXIcO3aMli1bMmHChDz3v/7664wdO5YJEyawatUqwsPD6d69O6mpqTl9hg8fzrfffstXX33FsmXLOHr0KNdeey1Op7OkPoYU0uLFi3nooYf4448/iImJITs7mx49enDs2LGcPrrm7qNGjRq8+uqrrF69mtWrV3PllVfSu3fvnP9Eda3d16pVq5g8eTItWrTI1a5r7l6aNm1KfHx8zmvjxo05+3St3c/hw4fp0KEDnp6e/Pzzz2zZsoW33nqLChUq5PTRdS8GhkgeAOPbb7/N2Xa5XEZ4eLjx6quv5rSlp6cbwcHBxvvvv28YhmEcOXLE8PT0NL766qucPvv27TPsdrvxyy+/lFjtcmESExMNwFi8eLFhGLrm5UFISIjx0Ucf6Vq7sdTUVKNBgwZGTEyM0aVLF+ORRx4xDEN/v93Nc889Z7Rs2TLPfbrW7unJJ580OnbseM79uu7FQyNOUii7du0iISGBHj165LR5e3vTpUsXli9fDsCaNWvIysrK1adatWo0a9Ysp4+UXsnJyQBUrFgR0DV3Z06nk6+++opjx47Rrl07XWs39tBDD3HNNdfQrVu3XO265u5n+/btVKtWjTp16nDLLbewc+dOQNfaXX3//fe0bt2am2++mSpVqnDJJZfw4Ycf5uzXdS8eCk5SKAkJCQCEhYXlag8LC8vZl5CQgJeXFyEhIefsI6WTYRiMGDGCjh070qxZM0DX3B1t3LiRgIAAvL29GTJkCN9++y1NmjTRtXZTX331FWvXrmXMmDFn7dM1dy9t27ZlypQpzJs3jw8//JCEhATat2/PwYMHda3d1M6dO5k0aRINGjRg3rx5DBkyhIcffpgpU6YA+jteXDysLkDKFpvNlmvbMIyz2s5UmD5iraFDh/LXX3+xbNmys/bpmruPRo0asX79eo4cOcKsWbO44447WLx4cc5+XWv3sXfvXh555BHmz5+Pj4/POfvpmruHXr165XzdvHlz2rVrR7169fj888+57LLLAF1rd+NyuWjdujWvvPIKAJdccgmbN29m0qRJDBo0KKefrnvR0oiTFMrJ1XnO/AlEYmJizk8zwsPDyczM5PDhw+fsI6XPsGHD+P7771m4cCE1atTIadc1dz9eXl7Ur1+f1q1bM2bMGFq2bMnbb7+ta+2G1qxZQ2JiItHR0Xh4eODh4cHixYt555138PDwyLlmuubuyd/fn+bNm7N9+3b9/XZTVatWpUmTJrnaGjduTFxcHKD/w4uLgpMUSp06dQgPDycmJianLTMzk8WLF9O+fXsAoqOj8fT0zNUnPj6eTZs25fSR0sMwDIYOHcrs2bP57bffqFOnTq79uubuzzAMMjIydK3dUNeuXdm4cSPr16/PebVu3ZoBAwawfv166tatq2vuxjIyMoiNjaVq1ar6++2mOnTocNYjRP7++29q1aoF6P/wYmPBghRSSqWmphrr1q0z1q1bZwDG2LFjjXXr1hl79uwxDMMwXn31VSM4ONiYPXu2sXHjRuPWW281qlataqSkpOScY8iQIUaNGjWMBQsWGGvXrjWuvPJKo2XLlkZ2drZVH0vO4YEHHjCCg4ONRYsWGfHx8TmvtLS0nD665u5j5MiRxpIlS4xdu3YZf/31l/G///3PsNvtxvz58w3D0LUuD05fVc8wdM3dyWOPPWYsWrTI2Llzp/HHH38Y1157rREYGGjs3r3bMAxda3e0cuVKw8PDw3j55ZeN7du3G9OmTTP8/PyMqVOn5vTRdS96Ck6SY+HChQZw1uuOO+4wDMNc2vK5554zwsPDDW9vb6Nz587Gxo0bc53j+PHjxtChQ42KFSsavr6+xrXXXmvExcVZ8GmkIHlda8D49NNPc/romruPu+++26hVq5bh5eVlVK5c2ejatWtOaDIMXevy4MzgpGvuPvr3729UrVrV8PT0NKpVq2b07dvX2Lx5c85+XWv39MMPPxjNmjUzvL29jcjISGPy5Mm59uu6Fz2bYRiGNWNdIiIiIiIiZYPucRIRERERESmAgpOIiIiIiEgBFJxEREREREQKoOAkIiIiIiJSAAUnERERERGRAig4iYiIiIiIFEDBSUREREREpAAKTiIiIiIiIgVQcBIRkTInMTGR+++/n5o1a+Lt7U14eDg9e/ZkxYoVANhsNubMmWNtkSIi4lY8rC5ARETkfN14441kZWXx+eefU7duXQ4cOMCvv/7KoUOHrC5NRETclEacRESkTDly5AjLli3jtdde44orrqBWrVq0adOGkSNHcs0111C7dm0AbrjhBmw2W842wA8//EB0dDQ+Pj7UrVuXF154gezs7Jz9NpuNSZMm0atXL3x9falTpw4zZ87M2Z+ZmcnQoUOpWrUqPj4+1K5dmzFjxpTURxcREQspOImISJkSEBBAQEAAc+bMISMj46z9q1atAuDTTz8lPj4+Z3vevHncfvvtPPzww2zZsoUPPviAzz77jJdffjnX8aNGjeLGG29kw4YN3H777dx6663ExsYC8M477/D999/z9ddfs23bNqZOnZormImIiPuyGYZhWF2EiIjI+Zg1axb33nsvx48fJyoqii5dunDLLbfQokULwBw5+vbbb+nTp0/OMZ07d6ZXr16MHDkyp23q1Kn83//9H/v37885bsiQIUyaNCmnz2WXXUZUVBQTJ07k4YcfZvPmzSxYsACbzVYyH1ZEREoFjTiJiEiZc+ONN7J//36+//57evbsyaJFi4iKiuKzzz475zFr1qxh9OjROSNWAQEB3HvvvcTHx5OWlpbTr127drmOa9euXc6I05133sn69etp1KgRDz/8MPPnzy+WzyciIqWPgpOIiJRJPj4+dO/enWeffZbly5dz55138txzz52zv8vl4oUXXmD9+vU5r40bN7J9+3Z8fHzyfa+To0tRUVHs2rWLF198kePHj9OvXz9uuummIv1cIiJSOik4iYiIW2jSpAnHjh0DwNPTE6fTmWt/VFQU27Zto379+me97PZT/x3+8ccfuY77448/iIyMzNkOCgqif//+fPjhh8yYMYNZs2ZpNT8RkXJAy5GLiEiZcvDgQW6++WbuvvtuWrRoQWBgIKtXr+b111+nd+/eANSuXZtff/2VDh064O3tTUhICM8++yzXXnstERER3Hzzzdjtdv766y82btzISy+9lHP+mTNn0rp1azp27Mi0adNYuXIlH3/8MQDjxo2jatWqtGrVCrvdzsyZMwkPD6dChQpW/FaIiEgJUnASEZEyJSAggLZt2zJu3Dh27NhBVlYWERER3Hvvvfzvf/8D4K233mLEiBF8+OGHVK9end27d9OzZ09+/PFHRo8ezeuvv46npyeRkZEMHjw41/lfeOEFvvrqKx588EHCw8OZNm0aTZo0yXnv1157je3bt+NwOLj00kuZO3durhErERFxT1pVT0RE5IS8VuMTEREB3eMkIiIiIiJSIAUnERERERGRAugeJxERkRM0e11ERM5FI04iIiIiIiIFUHASEREREREpgIKTiIiIiIhIARScRERERERECqDgJCIiIiIiUgAFJxERERERkQIoOImIiIiIiBRAwUlERERERKQACk4iIiIiIiIF+H+GYrDjiwyoYgAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 1000x600 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "class CausalSft_Model_No(nn.Module):\n",
    "    def __init__(self, base_model, num_labels, max_words):\n",
    "        super().__init__()\n",
    "\n",
    "        self.max_words = max_words\n",
    "        self.base_model = copy.deepcopy(base_model)\n",
    "        self.sft_model = copy.deepcopy(base_model)\n",
    "        self.sft_classifier = nn.Linear(self.base_model.config.hidden_size, num_labels)\n",
    "        self.r_2_c = nn.Sequential(\n",
    "            nn.Linear(self.base_model.config.hidden_size, self.base_model.config.hidden_size // 2),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(self.base_model.config.hidden_size // 2, self.base_model.config.hidden_size // 2),\n",
    "        )\n",
    "        self.c_2_c = nn.Sequential(\n",
    "            nn.Linear(self.base_model.config.hidden_size // 4, self.base_model.config.hidden_size // 4),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(self.base_model.config.hidden_size // 4, self.base_model.config.hidden_size // 4),\n",
    "        )\n",
    "        self.num_patches = 10\n",
    "        self.patches_size = self.max_words // self.num_patches\n",
    "        self.patch_extractor = nn.Sequential(\n",
    "            nn.Linear(self.base_model.config.hidden_size * self.num_patches, self.base_model.config.hidden_size * self.num_patches // 2),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(self.base_model.config.hidden_size * self.num_patches // 2, self.base_model.config.hidden_size),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(self.base_model.config.hidden_size, self.base_model.config.hidden_size),\n",
    "        )\n",
    "        self.classifier = nn.Linear(self.base_model.config.hidden_size + self.base_model.config.hidden_size // 4, num_labels)\n",
    "\n",
    "        self.cross_entropy_loss = nn.CrossEntropyLoss()\n",
    "        self.mse_loss = nn.MSELoss()\n",
    "\n",
    "        for param in self.base_model.parameters():\n",
    "            param.requires_grad = False\n",
    "\n",
    "        print(f'Initialise Causal SFT model \"{model_name}\" (unfreezed all layers) with a linear head!')\n",
    "        count_parameters(self)\n",
    "\n",
    "    def entropy_maximization(self, feature):\n",
    "        p = F.softmax(feature, dim=-1)\n",
    "        log_p = F.log_softmax(feature, dim=-1)\n",
    "        entropy = -torch.mean(torch.mean(p * log_p, dim=-1))  # Maximize entropy\n",
    "        return -entropy  # Minimize negative entropy\n",
    "\n",
    "    def forward(self, input_ids1, attention_mask1, input_ids2, attention_mask2, input_ids3, attention_mask3, input_ids4, attention_mask4, labels=None):\n",
    "        B, _ = input_ids1.size()\n",
    "        # avoid unwanted computational graph\n",
    "        outputs_sft2 = self.sft_model(input_ids2, attention_mask=attention_mask2)[1]\n",
    "        outputs_sft2_logits = self.sft_classifier(outputs_sft2)\n",
    "        # pull out R0 and R1 using another data point\n",
    "        outputs_base3 = self.base_model(input_ids3, attention_mask=attention_mask3)[1]\n",
    "        outputs_sft3 = self.sft_model(input_ids3, attention_mask=attention_mask3)[1]\n",
    "        # optimise C from R0 and R1\n",
    "        base_C3= self.r_2_c(outputs_base3)[:, :self.base_model.config.hidden_size//4]\n",
    "        sft_C3 = self.r_2_c(outputs_sft3)[:, :self.base_model.config.hidden_size//4]\n",
    "        # gather C \n",
    "        outputs_sft = self.sft_model(input_ids1, attention_mask=attention_mask1)[1]\n",
    "        sft_C = self.r_2_c(outputs_sft)[:, :self.base_model.config.hidden_size//4]\n",
    "        # extract non-contextual embeddings\n",
    "        with torch.no_grad():\n",
    "            embeddings_1 = self.sft_model.embeddings(input_ids1, attention_mask1)[:, :self.max_words]\n",
    "            embeddings_2 = self.sft_model.embeddings(input_ids2, attention_mask2)[:, :self.max_words]\n",
    "            embeddings_4 = self.sft_model.embeddings(input_ids4, attention_mask4)[:, :self.max_words]\n",
    "\n",
    "        input = embeddings_1\n",
    "        patches = torch.sum(input.view(B, self.num_patches, self.patches_size, -1), dim=2)\n",
    "        Ai_samples = self.patch_extractor(patches.view(B, -1))\n",
    "        # Ai_samples = Ai_samples[torch.randperm(B).to(input_ids1.device), :]\n",
    "        Ai_Z1 = torch.cat([Ai_samples, self.c_2_c(sft_C)], dim=-1)\n",
    "        logits = self.classifier(Ai_Z1)  \n",
    "\n",
    "        loss = None\n",
    "        if labels is not None:\n",
    "            loss_sft = self.cross_entropy_loss(outputs_sft2_logits, labels) \n",
    "            loss_identify = self.mse_loss(base_C3, sft_C3) + self.entropy_maximization(sft_C3) + self.entropy_maximization(base_C3)\n",
    "            loss_cls = self.cross_entropy_loss(logits, labels) \n",
    "            \n",
    "            loss = loss_sft + loss_identify + loss_cls \n",
    "        \n",
    "        return (loss, logits) if loss is not None else logits\n",
    "    \n",
    "set_seed(data_seed)\n",
    "model = CausalSft_Model_No(base_model, num_cls, max_words)\n",
    "\n",
    "# Define training arguments and trainer\n",
    "training_args = TrainingArguments(\n",
    "    overwrite_output_dir=True,\n",
    "    output_dir=f'./results/amazon/causal-sft-no/{N}-shot-{data_seed}',\n",
    "    eval_strategy=\"steps\",\n",
    "    save_strategy=\"steps\",\n",
    "    logging_strategy=\"steps\",\n",
    "    logging_steps=0.1,\n",
    "    save_steps=0.1,\n",
    "    learning_rate=5e-5,\n",
    "    per_device_train_batch_size=128,\n",
    "    per_device_eval_batch_size=128,\n",
    "    num_train_epochs=10,\n",
    "    seed=data_seed,\n",
    "    load_best_model_at_end=True,\n",
    "    metric_for_best_model=\"eval_loss\",  # Choose model based on validation loss\n",
    "    greater_is_better=False,  # Lower validation loss is better\n",
    "    save_total_limit=1,\n",
    ")\n",
    "\n",
    "\n",
    "trainer = Trainer(\n",
    "    model=model,\n",
    "    args=training_args,\n",
    "    train_dataset=paired_train_dataset,\n",
    "    eval_dataset=paired_validation_dataset,\n",
    "    tokenizer=tokenizer,\n",
    "    data_collator=paired_data_collator,\n",
    "    compute_metrics=compute_metrics,\n",
    ")\n",
    "\n",
    "# Train the model\n",
    "trainer.train()\n",
    "\n",
    "results = {}\n",
    "train_results = trainer.evaluate(paired_train_dataset)\n",
    "print(f'train: {train_results}')\n",
    "results['train'] = train_results\n",
    "\n",
    "valid_results = trainer.evaluate(paired_validation_dataset)\n",
    "print(f'validation: {valid_results}')\n",
    "results['valid'] = valid_results\n",
    "\n",
    "# Evaluate on the test set\n",
    "test_results_ID_90 = trainer.evaluate(paired_test_dataset_ID_90)\n",
    "print(f'ID 90: {test_results_ID_90}')\n",
    "results['ID 90'] = test_results_ID_90\n",
    "\n",
    "test_results_OOD_change_70 = trainer.evaluate(paired_test_dataset_OOD_change_70)\n",
    "print(f'OOD change 70: {test_results_OOD_change_70}')\n",
    "results['OOD change 70'] = test_results_OOD_change_70\n",
    "\n",
    "test_results_OOD_balanced_50 = trainer.evaluate(paired_test_dataset_OOD_balanced_50)\n",
    "print(f'OOD balanced 50: {test_results_OOD_balanced_50}')\n",
    "results['OOD balanced 50'] = test_results_OOD_balanced_50\n",
    "\n",
    "test_results_OOD_change_30 = trainer.evaluate(paired_test_dataset_OOD_change_30)\n",
    "print(f'OOD change 30: {test_results_OOD_change_30}')\n",
    "results['OOD change 30'] = test_results_OOD_change_30\n",
    "\n",
    "test_results_OOD_flip_10 = trainer.evaluate(paired_test_dataset_OOD_flip_10)\n",
    "print(f'OOD flip: {test_results_OOD_flip_10}')\n",
    "results['OOD flip'] = test_results_OOD_flip_10\n",
    "\n",
    "test_results_OOD_original_0 = trainer.evaluate(paired_test_dataset_OOD_original_0)\n",
    "print(f'OOD original: {test_results_OOD_original_0}')\n",
    "results['OOD original'] = test_results_OOD_original_0\n",
    "\n",
    "# Manually save the results\n",
    "with open(f'./results/amazon/causal-sft/{N}-shot-{data_seed}/test_results.json', 'w') as f:\n",
    "    json.dump(results, f, indent=4)\n",
    "\n",
    "plot_training_metrics(trainer)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## 5. Causal SFT - C"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Initialise Causal SFT model \"bert-base-uncased\" (unfreezed all layers) with a linear head!\n",
      "Total Parameters: 252519172\n",
      "Trainable Parameters: 143036932\n",
      "Percentage of Trainable Parameters: 56.6440%\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "\n",
       "    <div>\n",
       "      \n",
       "      <progress value='630' max='630' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [630/630 04:36, Epoch 10/10]\n",
       "    </div>\n",
       "    <table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       " <tr style=\"text-align: left;\">\n",
       "      <th>Step</th>\n",
       "      <th>Training Loss</th>\n",
       "      <th>Validation Loss</th>\n",
       "      <th>F1</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <td>63</td>\n",
       "      <td>0.689900</td>\n",
       "      <td>0.459065</td>\n",
       "      <td>0.931997</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>126</td>\n",
       "      <td>0.296500</td>\n",
       "      <td>0.431449</td>\n",
       "      <td>0.932993</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>189</td>\n",
       "      <td>0.132500</td>\n",
       "      <td>0.523530</td>\n",
       "      <td>0.931994</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>252</td>\n",
       "      <td>0.053300</td>\n",
       "      <td>0.519609</td>\n",
       "      <td>0.931994</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>315</td>\n",
       "      <td>0.014600</td>\n",
       "      <td>0.604013</td>\n",
       "      <td>0.933484</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>378</td>\n",
       "      <td>0.003300</td>\n",
       "      <td>0.584746</td>\n",
       "      <td>0.932993</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>441</td>\n",
       "      <td>-0.011800</td>\n",
       "      <td>0.628162</td>\n",
       "      <td>0.934498</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>504</td>\n",
       "      <td>-0.019000</td>\n",
       "      <td>0.650561</td>\n",
       "      <td>0.928988</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>567</td>\n",
       "      <td>-0.028400</td>\n",
       "      <td>0.781192</td>\n",
       "      <td>0.931498</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>630</td>\n",
       "      <td>-0.029900</td>\n",
       "      <td>0.683088</td>\n",
       "      <td>0.931498</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table><p>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       "\n",
       "    <div>\n",
       "      \n",
       "      <progress value='271' max='63' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [63/63 00:45]\n",
       "    </div>\n",
       "    "
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train: {'eval_loss': 0.1606869250535965, 'eval_f1': 0.9821248991732594, 'eval_runtime': 10.5966, 'eval_samples_per_second': 754.958, 'eval_steps_per_second': 5.945, 'epoch': 10.0}\n",
      "validation: {'eval_loss': 0.4142448604106903, 'eval_f1': 0.932993299329933, 'eval_runtime': 3.0319, 'eval_samples_per_second': 659.643, 'eval_steps_per_second': 5.277, 'epoch': 10.0}\n",
      "ID 90: {'eval_loss': 0.4857505261898041, 'eval_f1': 0.922989815403087, 'eval_runtime': 5.4818, 'eval_samples_per_second': 729.681, 'eval_steps_per_second': 5.837, 'epoch': 10.0}\n",
      "OOD change 70: {'eval_loss': 1.0581371784210205, 'eval_f1': 0.8209172189844756, 'eval_runtime': 5.4908, 'eval_samples_per_second': 728.485, 'eval_steps_per_second': 5.828, 'epoch': 10.0}\n",
      "OOD balanced 50: {'eval_loss': 1.6881718635559082, 'eval_f1': 0.7154540544324883, 'eval_runtime': 5.4991, 'eval_samples_per_second': 727.394, 'eval_steps_per_second': 5.819, 'epoch': 10.0}\n",
      "OOD change 30: {'eval_loss': 2.300323247909546, 'eval_f1': 0.6126532681504149, 'eval_runtime': 5.4918, 'eval_samples_per_second': 728.363, 'eval_steps_per_second': 5.827, 'epoch': 10.0}\n",
      "OOD flip: {'eval_loss': 2.850163221359253, 'eval_f1': 0.5089092916730983, 'eval_runtime': 5.4914, 'eval_samples_per_second': 728.417, 'eval_steps_per_second': 5.827, 'epoch': 10.0}\n",
      "OOD original: {'eval_loss': 1.2717218399047852, 'eval_f1': 0.7660481096791472, 'eval_runtime': 5.4989, 'eval_samples_per_second': 727.416, 'eval_steps_per_second': 5.819, 'epoch': 10.0}\n"
     ]
    },
    {
     "data": {
      "image/png": "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UBB//fUXJ0+eBODNN9/k66+/ZsGCBTRq1Ii4uDg9FyciIiIi1gYngAceeIAHHnig2H3FPZfz8MMP8/DDD1/kqk47mZNH22eXVNr1zrRj0iC83CrmFo0fP54bb7yx0LbHHnus4PXDDz/M999/z8KFC0sMTldffXXB/XryySd54403WL58eYnB6eWXX+byyy8H4B//+AfXXHMNmZmZeHh48NZbbzFmzBjuuusuAJ599lmWLl3KiRMnyvU509PTmT59OnPmzGHw4MEAvP/++0RHRzNz5kwef/xxYmNj6dKlC1FRUYDZknZKbGwsLVq04NJLL8VmsxWEeBERERGp3SwfVU8qx6mQcEpeXh4vv/wyHTt2JDAwEB8fH5YuXVqoha84HTt2LHh9qktgYmJimc85NYDHqXN27dpF9+7dCx1/9vr52LNnDzk5OfTu3btgm6urK927d2fnzp0A3H///cybN4/OnTvzxBNPFJo3bPTo0WzevJlWrVrxyCOPsHTp0nLXIiIiIiI1h+UtTlWdp6uDHZMGWXbtiuLt7V1ofcqUKbzxxhtMnTqVDh064O3tzfjx48nOzi7xfc4eVMJms+F0Ost8zqmuh2eeU9yQ9OV16tyShrkfPHgw+/fv59tvv2XZsmX069ePBx98kNdee42uXbsSExPDd999x7Jlyxg+fDj9+/fn888/L3dNIiIiUkMd3g7ufhCgeUJrA7U4lcJms+Hl5mLJUlHPNxVn5cqVDBkyhDvuuINOnTrRtGlTdu/efdGudy6tWrVi7dq1hbatX7++3O/XvHlz3Nzc+OWXXwq25eTksH79etq0aVOwrV69eowePZqPP/6YqVOnFhrkws/PjxEjRvD+++8zf/58Fi1axNGjR8tdk4iIiNQwhgG//hem94K3usKK/0BejtVVyUWmFqdaqnnz5ixatIhVq1ZRp04dXn/9dRISEgqFi8rw8MMPc8899xAVFUWvXr2YP38+W7dupWnTpqWee/bofABt27bl/vvv5/HHH6du3bo0atSIV199lYyMDMaMGQOYz1FFRkbSrl07srKy+N///lfwud944w3CwsLo3LkzdrudhQsXEhoaSkBAQIV+bhEREamm8nJh8WOwYXb+ejb8+BJs/z8Y8hbU72JtfXLRKDjVUs888wwxMTEMGjQILy8v7r33XoYOHUpKSkql1nH77bezd+9e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ERERqY7KFZzi4uI4cOBAwfratWsZP348M2bMOO/3stlshdYNwyiyDSAvL4/bbruNF154gZYtW5b5/SdOnEhKSkrBEhenH94r24j8OZ3+b/NBMnPyLK5GREREROT8lSs43Xbbbfz0008AJCQkMGDAANauXctTTz3FpEmTyvQeQUFBOByOIq1LiYmJRVqhANLS0li/fj0PPfQQLi4uuLi4MGnSJLZs2YKLiws//vhjsddxd3fHz8+v0CKVq1ezQBoEeJKamcuS7cW3JoqIiIiIVGXlCk6///473bt3B2DBggW0b9+eVatW8emnnzJnzpwyvYebmxuRkZFER0cX2h4dHU2vXr2KHO/n58e2bdvYvHlzwTJu3DhatWrF5s2b6dGjR3k+ilQCu91WMDS55nQSERERkeqobA8KnSUnJwd3d3cAli1bxvXXXw9A69atiY+PL/P7TJgwgZEjRxIVFUXPnj2ZMWMGsbGxjBs3DjC72R08eJC5c+dit9tp3759ofODg4Px8PAosl2qnpsjG/LfH3bz61/JxB3NILyul9UliYiIiIiUWblanNq1a8e7777LypUriY6O5qqrrgLg0KFDBAYGlvl9RowYwdSpU5k0aRKdO3dmxYoVLF68mMaNGwMQHx9f6pxOUj00rONF72ZBACzccKCUo0VEREREqhabUY7JdZYvX84NN9xAamoqd955J7NmzQLgqaee4o8//uCLL76o8EIrSmpqKv7+/qSkpOh5p0r29ZZDPPLZJur7e7DyyStx2IsOAiIiIiIiUlnOJxuUq6te3759SUpKIjU1lTp16hRsv/fee/HyUhcsKd7AtiH4e7pyKCWTVXuS6NOintUliYiIiIiUSbm66p08eZKsrKyC0LR//36mTp3Krl27CA4OrtACpebwcHUwtHN9QHM6iYiIiEj1Uq7gNGTIEObOnQvA8ePH6dGjB1OmTGHo0KFMnz69QguUmmVY/pxOS7cf5nhGtsXViIiIiIiUTbmC08aNG+nTpw8An3/+OSEhIezfv5+5c+fy5ptvVmiBUrO0b+BP2zA/svOcfLXpoNXliIiIiIiUSbmCU0ZGBr6+vgAsXbqUG2+8EbvdziWXXML+/fsrtECpeUZ0M1udFqzX6HoiIiIiUj2UKzg1b96cr776iri4OJYsWcLAgQMBSExM1Eh1Uqohnevj5rCzIz6V3w+mWF2OiIiIiEipyhWcnn32WR577DGaNGlC9+7d6dmzJ2C2PnXp0qVCC5SaJ8DLjYHtQgBYsF6DRIiIiIhI1Veu4HTzzTcTGxvL+vXrWbJkScH2fv368cYbb1RYcVJznequ99Wmg2Tm5FlcjYiIiIhIyco1jxNAaGgooaGhHDhwAJvNRoMGDejevXtF1iY1WO9mQTQI8OTg8ZMs3XGY6zvVt7okEREREZFzKleLk9PpZNKkSfj7+9O4cWMaNWpEQEAAL774Ik6ns6JrlBrIbrdxc2RDABZoTicRERERqeLKFZyefvpp3n77bV555RU2bdrExo0b+de//sVbb73FM888U9E1Sg11Kjj9uieJuKMZFlcjIiIiInJu5eqq9+GHH/LBBx9w/fXXF2zr1KkTDRo04IEHHuDll1+usAKl5gqv60Xv5oH8+lcyn284wKMDWlpdkoiIiIhIscrV4nT06FFat25dZHvr1q05evToBRcltcfwKHOQiM83HMDpNCyuRkRERESkeOUKTp06deLtt98usv3tt9+mY8eOF1yU1B6D2oXi5+HCweMnWbUn2epyRERERESKVa6ueq+++irXXHMNy5Yto2fPnthsNlatWkVcXByLFy+u6BqlBvNwdTCkcwM+WrOf+evjuLRFkNUliYiIiIgUUa4Wp8svv5w///yTG264gePHj3P06FFuvPFGtm/fzuzZsyu6RqnhTs3ptGR7Asczsi2uRkRERESkKJthGBX2YMmWLVvo2rUreXlVd0LT1NRU/P39SUlJwc/Pz+pyBDAMg6vf/IWd8alMGtKOUT2bWF2SiIiIiNQC55MNytXiJFKRbDYbI6LMocnna04nEREREamCFJykShjSuQFuDjvbD6Xy+8EUq8sRERERESlEwUmqhDrebgxoFwLAwvVqdRIRERGRquW8RtW78cYbS9x//PjxC6lFarkRUeF8uzWerzYfYuLVbfBwdVhdkoiIiIgIcJ7Byd/fv9T9o0aNuqCCpPbq3TyI+v4eHErJJHrHYa7rVN/qkkREREREgPMMThpqXC4mh93GzZENefPHv1iwPk7BSURERESqDD3jJFXKsChzTqdf/kriwLEMi6sRERERETEpOEmVEl7Xi17NAjEMWLThoNXliIiIiIgACk5SBY3oZrY6LdwQh9NZYfMzi4iIiIiUm4KTVDmD2oXi6+HCgWMn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lELBSUREREREpBQKTiIiIiIiIqVwsbqAymYYBgCpqakWVyIiIiIiIlY6lQlOZYSS1LrglJaWBkB4eLjFlYiIiIiISFWQlpaGv79/icfYjLLEqxrE6XRy6NAhfH19sdlsVpdTbaWmphIeHk5cXBx+fn5WlyMXme537aL7Xbvoftc+uue1i+53yQzDIC0tjfr162O3l/wUU61rcbLb7TRs2NDqMmoMPz8//SWsRXS/axfd79pF97v20T2vXXS/z620lqZTNDiEiIiIiIhIKRScRERERERESqHgJOXi7u7Oc889h7u7u9WlSCXQ/a5ddL9rF93v2kf3vHbR/a44tW5wCBERERERkfOlFicREREREZFSKDiJiIiIiIiUQsFJRERERESkFApOIiIiIiIipVBwkgIrVqzguuuuo379+thsNr766qtC+w3D4Pnnn6d+/fp4enrSt29ftm/fXuiYrKwsHn74YYKCgvD29ub666/nwIEDlfgppKwmT55Mt27d8PX1JTg4mKFDh7Jr165Cx+ie1xzTp0+nY8eOBRMg9uzZk++++65gv+51zTZ58mRsNhvjx48v2KZ7XnM8//zz2Gy2QktoaGjBft3rmungwYPccccdBAYG4uXlRefOndmwYUPBft33iqfgJAXS09Pp1KkTb7/9drH7X331VV5//XXefvtt1q1bR2hoKAMGDCAtLa3gmPHjx/Pll18yb948fvnlF06cOMG1115LXl5eZX0MKaOff/6ZBx98kDVr1hAdHU1ubi4DBw4kPT294Bjd85qjYcOGvPLKK6xfv57169dz5ZVXMmTIkIL/RHWva65169YxY8YMOnbsWGi77nnN0q5dO+Lj4wuWbdu2FezTva55jh07Ru/evXF1deW7775jx44dTJkyhYCAgIJjdN8vAkOkGIDx5ZdfFqw7nU4jNDTUeOWVVwq2ZWZmGv7+/sa7775rGIZhHD9+3HB1dTXmzZtXcMzBgwcNu91ufP/995VWu5RPYmKiARg///yzYRi657VBnTp1jA8++ED3ugZLS0szWrRoYURHRxuXX3658be//c0wDP39rmmee+45o1OnTsXu072umZ588knj0ksvPed+3feLQy1OUiYxMTEkJCQwcODAgm3u7u5cfvnlrFq1CoANGzaQk5NT6Jj69evTvn37gmOk6kpJSQGgbt26gO55TZaXl8e8efNIT0+nZ8+eutc12IMPPsg111xD//79C23XPa95du/eTf369YmIiOCWW25h7969gO51TfX1118TFRXFsGHDCA4OpkuXLrz//vsF+3XfLw4FJymThIQEAEJCQgptDwkJKdiXkJCAm5sbderUOecxUjUZhsGECRO49NJLad++PaB7XhNt27YNHx8f3N3dGTduHF9++SVt27bVva6h5s2bx8aNG5k8eXKRfbrnNUuPHj2YO3cuS5Ys4f333ychIYFevXqRnJyse11D7d27l+nTp9OiRQuWLFnCuHHjeOSRR5g7dy6gv+MXi4vVBUj1YrPZCq0bhlFk29nKcoxY66GHHmLr1q388ssvRfbpntccrVq1YvPmzRw/fpxFixZx55138vPPPxfs172uOeLi4vjb3/7G0qVL8fDwOOdxuuc1w+DBgwted+jQgZ49e9KsWTM+/PBDLrnkEkD3uqZxOp1ERUXxr3/9C4AuXbqwfft2pk+fzqhRowqO032vWGpxkjI5NTrP2b+BSExMLPhtRmhoKNnZ2Rw7duycx0jV8/DDD/P111/z008/0bBhw4Ltuuc1j5ubG82bNycqKorJkyfTqVMn/vvf/+pe10AbNmwgMTGRyMhIXFxccHFx4eeff+bNN9/ExcWl4J7pntdM3t7edOjQgd27d+vvdw0VFhZG27ZtC21r06YNsbGxgP4Pv1gUnKRMIiIiCA0NJTo6umBbdnY2P//8M7169QIgMjISV1fXQsfEx8fz+++/FxwjVYdhGDz00EN88cUX/Pjjj0RERBTar3te8xmGQVZWlu51DdSvXz+2bdvG5s2bC5aoqChuv/12Nm/eTNOmTXXPa7CsrCx27txJWFiY/n7XUL179y4yhciff/5J48aNAf0fftFYMCCFVFFpaWnGpk2bjE2bNhmA8frrrxubNm0y9u/fbxiGYbzyyiuGv7+/8cUXXxjbtm0zbr31ViMsLMxITU0teI9x48YZDRs2NJYtW2Zs3LjRuPLKK41OnToZubm5Vn0sOYf777/f8Pf3N5YvX27Ex8cXLBkZGQXH6J7XHBMnTjRWrFhhxMTEGFu3bjWeeuopw263G0uXLjUMQ/e6NjhzVD3D0D2vSf7+978by5cvN/bu3WusWbPGuPbaaw1fX19j3759hmHoXtdEa9euNVxcXIyXX37Z2L17t/HJJ58YXl5exscff1xwjO57xVNwkgI//fSTARRZ7rzzTsMwzKEtn3vuOSM0NNRwd3c3LrvsMmPbtm2F3uPkyZPGQw89ZNStW9fw9PQ0rr32WiM2NtaCTyOlKe5eA8bs2bMLjtE9rznuvvtuo3Hjxoabm5tRr149o1+/fgWhyTB0r2uDs4OT7nnNMWLECCMsLMxwdXU16tevb9x4443G9u3bC/brXtdM33zzjdG+fXvD3d3daN26tTFjxoxC+3XfK57NMAzDmrYuERERERGR6kHPOImIiIiIiJRCwUlERERERKQUCk4iIiIiIiKlUHASEREREREphYKTiIiIiIhIKRScRERERERESqHgJCIiIiIiUgoFJxERERERkVIoOImISLWTmJjIfffdR6NGjXB3dyc0NJRBgwaxevVqAGw2G1999ZW1RYqISI3iYnUBIiIi5+umm24iJyeHDz/8kKZNm3L48GF++OEHjh49anVpIiJSQ6nFSUREqpXjx4/zyy+/8O9//5srrriCxo0b0717dyZOnMg111xDkyZNALjhhhuw2WwF6wDffPMNkZGReHh40LRpU1544QVyc3ML9ttsNqZPn87gwYPx9PQkIiKChQsXFuzPzs7moYceIiwsDA8PD5o0acLkyZMr66OLiIiFFJxERKRa8fHxwcfHh6+++oqsrKwi+9etWwfA7NmziY+PL1hfsmQJd9xxB4888gg7duzgvffeY86cObz88suFzn/mmWe46aab2LJlC3fccQe33norO3fuBODNN9/k66+/ZsGCBezatYuPP/64UDATEZGay2YYhmF1ESIiIudj0aJF3HPPPZw8eZKuXbty+eWXc8stt9CxY0fAbDn68ssvGTp0aME5l112GYMHD2bixIkF2z7++GOeeOIJDh06VHDeuHHjmD59esExl1xyCV27dmXatGk88sgjbN++nWXLlmGz2Srnw4qISJWgFicREal2brrpJg4dOsTXX3/NoEGDWL58OV27dmXOnDnnPGfDhg1MmjSpoMXKx8eHe+65h/j4eDIyMgqO69mzZ6HzevbsWdDiNHr0aDZv3kyrVq145JFHWLp06UX5fCIiUvUoOImISLXk4eHBgAEDePbZZ1m1ahWjR4/mueeeO+fxTqeTF154gc2bNxcs27ZtY/fu3Xh4eJR4rVOtS127diUmJoYXX3yRkydPMnz4cG6++eYK/VwiIlI1KTiJiEiN0LZtW9LT0wFwdXUlLy+v0P6uXbuya9cumjdvXmSx20//d7hmzZpC561Zs4bWrVsXrPv5+TFixAjef/995s+fz6JFizSan4hILaDhyEVEpFpJTk5m2LBh3H333XTs2BFfX1/Wr1/Pq6++ypAhQwBo0qQJP/zwA71798bd3Z06derw7LPPcu211xIeHs6wYcOw2+1s3bqVbdu28dJLLxW8/8KFC4mKiuLSSy/lk08+Ye3atcycOROAN954g7CwMDp37ozdbmfhwoWEhoYSEBBgxR+FiIhUIgUnERGpVnx8fOjRowdvvPEGe/b8fzt3jKIwEIZh+APxDgFJL/aeIRcxTcqU6bQz4AlSCx4g5DJhwXvYbCMLwi7TL89zgJn65Z/5v/J6vVLXddq2zTAMSZLb7Za+7zNNU3a7XZ7PZ5qmyTzPuVwuGccx2+02+/0+p9Pp4/zz+ZzH45Gu61JVVe73ew6Hw8/d1+s167pms9nkeDxmWZaPiRUA/5OtegDw9ts2PgBI/HECAAAoEk4AAAAF/jgBwJvX6wD8xcQJAACgQDgBAAAUCCcAAIAC4QQAAFAgnAAAAAqEEwAAQIFwAgAAKBBOAAAABcIJAACg4BsHOX5l/uOzywAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 1000x600 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "class CausalSft_Model_C(nn.Module):\n",
    "    def __init__(self, base_model, num_labels, max_words):\n",
    "        super().__init__()\n",
    "\n",
    "        self.max_words = max_words\n",
    "        self.base_model = copy.deepcopy(base_model)\n",
    "        self.sft_model = copy.deepcopy(base_model)\n",
    "        self.sft_classifier = nn.Linear(self.base_model.config.hidden_size, num_labels)\n",
    "        self.r_2_c = nn.Sequential(\n",
    "            nn.Linear(self.base_model.config.hidden_size, self.base_model.config.hidden_size // 2),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(self.base_model.config.hidden_size // 2, self.base_model.config.hidden_size // 2),\n",
    "        )\n",
    "        self.c_2_c = nn.Sequential(\n",
    "            nn.Linear(self.base_model.config.hidden_size // 4, self.base_model.config.hidden_size // 4),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(self.base_model.config.hidden_size // 4, self.base_model.config.hidden_size // 4),\n",
    "        )\n",
    "        self.num_patches = 10\n",
    "        self.patches_size = self.max_words // self.num_patches\n",
    "        self.patch_extractor = nn.Sequential(\n",
    "            nn.Linear(self.base_model.config.hidden_size * self.num_patches, self.base_model.config.hidden_size * self.num_patches // 2),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(self.base_model.config.hidden_size * self.num_patches // 2, self.base_model.config.hidden_size),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(self.base_model.config.hidden_size, self.base_model.config.hidden_size),\n",
    "        )\n",
    "        # self.c_patch_mixer = nn.Sequential(\n",
    "        #     nn.Linear(self.base_model.config.hidden_size // 4, self.base_model.config.hidden_size // 8),\n",
    "        #     nn.ReLU(),\n",
    "        #     nn.Linear(self.base_model.config.hidden_size // 8, self.base_model.config.hidden_size // 8),\n",
    "        # )\n",
    "        self.classifier = nn.Linear(self.base_model.config.hidden_size // 4, num_labels)\n",
    "\n",
    "        self.cross_entropy_loss = nn.CrossEntropyLoss()\n",
    "        self.mse_loss = nn.MSELoss()\n",
    "\n",
    "        for param in self.base_model.parameters():\n",
    "            param.requires_grad = False\n",
    "\n",
    "        print(f'Initialise Causal SFT model \"{model_name}\" (unfreezed all layers) with a linear head!')\n",
    "        count_parameters(self)\n",
    "\n",
    "    def entropy_maximization(self, feature):\n",
    "        p = F.softmax(feature, dim=-1)\n",
    "        log_p = F.log_softmax(feature, dim=-1)\n",
    "        entropy = -torch.mean(torch.mean(p * log_p, dim=-1))  # Maximize entropy\n",
    "        return -entropy  # Minimize negative entropy\n",
    "\n",
    "    def forward(self, input_ids1, attention_mask1, input_ids2, attention_mask2, input_ids3, attention_mask3, input_ids4, attention_mask4, labels=None):\n",
    "        B, _ = input_ids1.size()\n",
    "        # avoid unwanted computational graph\n",
    "        outputs_sft2 = self.sft_model(input_ids2, attention_mask=attention_mask2)[1]\n",
    "        outputs_sft2_logits = self.sft_classifier(outputs_sft2)\n",
    "        # pull out R0 and R1 using another data point\n",
    "        outputs_base3 = self.base_model(input_ids3, attention_mask=attention_mask3)[1]\n",
    "        outputs_sft3 = self.sft_model(input_ids3, attention_mask=attention_mask3)[1]\n",
    "        # optimise C from R0 and R1\n",
    "        base_C3= self.r_2_c(outputs_base3)[:, :self.base_model.config.hidden_size//4]\n",
    "        sft_C3 = self.r_2_c(outputs_sft3)[:, :self.base_model.config.hidden_size//4]\n",
    "        # gather C \n",
    "        outputs_sft = self.sft_model(input_ids1, attention_mask=attention_mask1)[1]\n",
    "        sft_C = self.r_2_c(outputs_sft)[:, :self.base_model.config.hidden_size//4]\n",
    "        # extract non-contextual embeddings\n",
    "        with torch.no_grad():\n",
    "            embeddings_1 = self.sft_model.embeddings(input_ids1, attention_mask1)[:, :self.max_words]\n",
    "            embeddings_2 = self.sft_model.embeddings(input_ids2, attention_mask2)[:, :self.max_words]\n",
    "            embeddings_4 = self.sft_model.embeddings(input_ids4, attention_mask4)[:, :self.max_words]\n",
    "\n",
    "        # input = embeddings_1\n",
    "        # patches = torch.sum(input.view(B, self.num_patches, self.patches_size, -1), dim=2)\n",
    "        # Ai_samples = self.patch_extractor(patches.view(B, -1))\n",
    "        # Ai_samples = Ai_samples[torch.randperm(B).to(input_ids1.device), :]\n",
    "        Ai_Z1 = self.c_2_c(sft_C)\n",
    "        logits = self.classifier(Ai_Z1)   \n",
    "\n",
    "        loss = None\n",
    "        if labels is not None:\n",
    "            loss_sft = self.cross_entropy_loss(outputs_sft2_logits, labels) \n",
    "            loss_identify = self.mse_loss(base_C3, sft_C3) + self.entropy_maximization(sft_C3) + self.entropy_maximization(base_C3)\n",
    "            loss_cls = self.cross_entropy_loss(logits, labels) \n",
    "            \n",
    "            loss = loss_sft + loss_identify + loss_cls \n",
    "        \n",
    "        return (loss, logits) if loss is not None else logits\n",
    "    \n",
    "set_seed(data_seed)\n",
    "model = CausalSft_Model_C(base_model, num_cls, max_words)\n",
    "\n",
    "# Define training arguments and trainer\n",
    "training_args = TrainingArguments(\n",
    "    overwrite_output_dir=True,\n",
    "    output_dir=f'./results/amazon/causal-sft-c/{N}-shot-{data_seed}',\n",
    "    eval_strategy=\"steps\",\n",
    "    save_strategy=\"steps\",\n",
    "    logging_strategy=\"steps\",\n",
    "    logging_steps=0.1,\n",
    "    save_steps=0.1,\n",
    "    learning_rate=5e-5,\n",
    "    per_device_train_batch_size=128,\n",
    "    per_device_eval_batch_size=128,\n",
    "    num_train_epochs=10,\n",
    "    seed=data_seed,\n",
    "    load_best_model_at_end=True,\n",
    "    metric_for_best_model=\"eval_loss\",  # Choose model based on validation loss\n",
    "    greater_is_better=False,  # Lower validation loss is better\n",
    "    save_total_limit=1,\n",
    ")\n",
    "\n",
    "\n",
    "trainer = Trainer(\n",
    "    model=model,\n",
    "    args=training_args,\n",
    "    train_dataset=paired_train_dataset,\n",
    "    eval_dataset=paired_validation_dataset,\n",
    "    tokenizer=tokenizer,\n",
    "    data_collator=paired_data_collator,\n",
    "    compute_metrics=compute_metrics,\n",
    ")\n",
    "\n",
    "# Train the model\n",
    "trainer.train()\n",
    "\n",
    "results = {}\n",
    "train_results = trainer.evaluate(paired_train_dataset)\n",
    "print(f'train: {train_results}')\n",
    "results['train'] = train_results\n",
    "\n",
    "valid_results = trainer.evaluate(paired_validation_dataset)\n",
    "print(f'validation: {valid_results}')\n",
    "results['valid'] = valid_results\n",
    "\n",
    "# Evaluate on the test set\n",
    "test_results_ID_90 = trainer.evaluate(paired_test_dataset_ID_90)\n",
    "print(f'ID 90: {test_results_ID_90}')\n",
    "results['ID 90'] = test_results_ID_90\n",
    "\n",
    "test_results_OOD_change_70 = trainer.evaluate(paired_test_dataset_OOD_change_70)\n",
    "print(f'OOD change 70: {test_results_OOD_change_70}')\n",
    "results['OOD change 70'] = test_results_OOD_change_70\n",
    "\n",
    "test_results_OOD_balanced_50 = trainer.evaluate(paired_test_dataset_OOD_balanced_50)\n",
    "print(f'OOD balanced 50: {test_results_OOD_balanced_50}')\n",
    "results['OOD balanced 50'] = test_results_OOD_balanced_50\n",
    "\n",
    "test_results_OOD_change_30 = trainer.evaluate(paired_test_dataset_OOD_change_30)\n",
    "print(f'OOD change 30: {test_results_OOD_change_30}')\n",
    "results['OOD change 30'] = test_results_OOD_change_30\n",
    "\n",
    "test_results_OOD_flip_10 = trainer.evaluate(paired_test_dataset_OOD_flip_10)\n",
    "print(f'OOD flip: {test_results_OOD_flip_10}')\n",
    "results['OOD flip'] = test_results_OOD_flip_10\n",
    "\n",
    "test_results_OOD_original_0 = trainer.evaluate(paired_test_dataset_OOD_original_0)\n",
    "print(f'OOD original: {test_results_OOD_original_0}')\n",
    "results['OOD original'] = test_results_OOD_original_0\n",
    "\n",
    "# Manually save the results\n",
    "with open(f'./results/amazon/causal-sft-c/{N}-shot-{data_seed}/test_results.json', 'w') as f:\n",
    "    json.dump(results, f, indent=4)\n",
    "\n",
    "plot_training_metrics(trainer)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## 6. Causal SFT - Phi"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Initialise Causal SFT model \"bert-base-uncased\" (unfreezed all layers) with a linear head!\n",
      "Total Parameters: 252520324\n",
      "Trainable Parameters: 143038084\n",
      "Percentage of Trainable Parameters: 56.6442%\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "\n",
       "    <div>\n",
       "      \n",
       "      <progress value='630' max='630' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [630/630 03:50, Epoch 10/10]\n",
       "    </div>\n",
       "    <table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       " <tr style=\"text-align: left;\">\n",
       "      <th>Step</th>\n",
       "      <th>Training Loss</th>\n",
       "      <th>Validation Loss</th>\n",
       "      <th>F1</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <td>63</td>\n",
       "      <td>0.557800</td>\n",
       "      <td>0.465011</td>\n",
       "      <td>0.900000</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>126</td>\n",
       "      <td>0.389300</td>\n",
       "      <td>0.477881</td>\n",
       "      <td>0.898494</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>189</td>\n",
       "      <td>0.274800</td>\n",
       "      <td>0.596134</td>\n",
       "      <td>0.886380</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>252</td>\n",
       "      <td>0.172900</td>\n",
       "      <td>0.521510</td>\n",
       "      <td>0.891450</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>315</td>\n",
       "      <td>0.093700</td>\n",
       "      <td>0.618788</td>\n",
       "      <td>0.855163</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>378</td>\n",
       "      <td>0.039900</td>\n",
       "      <td>0.609534</td>\n",
       "      <td>0.880445</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>441</td>\n",
       "      <td>-0.003300</td>\n",
       "      <td>0.716662</td>\n",
       "      <td>0.889439</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>504</td>\n",
       "      <td>-0.018400</td>\n",
       "      <td>0.755439</td>\n",
       "      <td>0.892498</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>567</td>\n",
       "      <td>-0.035000</td>\n",
       "      <td>0.886298</td>\n",
       "      <td>0.891984</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>630</td>\n",
       "      <td>-0.040100</td>\n",
       "      <td>0.806619</td>\n",
       "      <td>0.891996</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table><p>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       "\n",
       "    <div>\n",
       "      \n",
       "      <progress value='271' max='63' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [63/63 00:45]\n",
       "    </div>\n",
       "    "
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train: {'eval_loss': 0.42018193006515503, 'eval_f1': 0.9, 'eval_runtime': 10.5603, 'eval_samples_per_second': 757.553, 'eval_steps_per_second': 5.966, 'epoch': 10.0}\n",
      "validation: {'eval_loss': 0.44667598605155945, 'eval_f1': 0.9, 'eval_runtime': 2.9813, 'eval_samples_per_second': 670.837, 'eval_steps_per_second': 5.367, 'epoch': 10.0}\n",
      "ID 90: {'eval_loss': 0.4820045232772827, 'eval_f1': 0.9, 'eval_runtime': 5.4829, 'eval_samples_per_second': 729.547, 'eval_steps_per_second': 5.836, 'epoch': 10.0}\n",
      "OOD change 70: {'eval_loss': 1.1647107601165771, 'eval_f1': 0.7, 'eval_runtime': 5.5409, 'eval_samples_per_second': 721.905, 'eval_steps_per_second': 5.775, 'epoch': 10.0}\n",
      "OOD balanced 50: {'eval_loss': 1.8800773620605469, 'eval_f1': 0.5, 'eval_runtime': 5.5232, 'eval_samples_per_second': 724.215, 'eval_steps_per_second': 5.794, 'epoch': 10.0}\n",
      "OOD change 30: {'eval_loss': 2.5586562156677246, 'eval_f1': 0.3, 'eval_runtime': 5.5291, 'eval_samples_per_second': 723.447, 'eval_steps_per_second': 5.788, 'epoch': 10.0}\n",
      "OOD flip: {'eval_loss': 3.2281711101531982, 'eval_f1': 0.10000000000000002, 'eval_runtime': 5.4868, 'eval_samples_per_second': 729.022, 'eval_steps_per_second': 5.832, 'epoch': 10.0}\n",
      "OOD original: {'eval_loss': 1.080011010169983, 'eval_f1': 0.48132590576520873, 'eval_runtime': 5.5102, 'eval_samples_per_second': 725.924, 'eval_steps_per_second': 5.807, 'epoch': 10.0}\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAA04AAAINCAYAAAAJGy/3AAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8pXeV/AAAACXBIWXMAAA9hAAAPYQGoP6dpAAB3d0lEQVR4nO3dd3gUZdvG4d/upndqEiA06b33ZiGIqGDFiiigqIiI7UU/GxZsiK8NBQsqCojtRUUhqDQB6R0BIRBKQkiAJKRnd74/BgKhbAKETLK5zuPYI8wzs7v3MkC48szcj80wDAMRERERERE5K7vVBYiIiIiIiJR2Ck4iIiIiIiKFUHASEREREREphIKTiIiIiIhIIRScRERERERECqHgJCIiIiIiUggFJxERERERkUIoOImIiIiIiBTCy+oCSprL5WL//v0EBwdjs9msLkdERERERCxiGAZpaWlUq1YNu939nFK5C0779+8nKirK6jJERERERKSU2LNnDzVq1HB7TLkLTsHBwYD5mxMSEmJxNSIiIiIiYpXU1FSioqLyM4I75S44Hb88LyQkRMFJRERERESKdAuPmkOIiIiIiIgUQsFJRERERESkEApOIiIiIiIihSh39zgVhWEY5OXl4XQ6rS5FPIzD4cDLy0ut8EVERETKGAWnU+Tk5BAfH09GRobVpYiHCggIIDIyEh8fH6tLEREREZEiUnA6icvlIjY2FofDQbVq1fDx8dHMgBQbwzDIycnh4MGDxMbGUr9+/UIXWhMRERGR0kHB6SQ5OTm4XC6ioqIICAiwuhzxQP7+/nh7e7N7925ycnLw8/OzuiQRERERKQL9uPsMNAsgF5P+fImIiIiUPfofnIiIiIiISCEUnOSsevXqxahRo4p8/K5du7DZbKxdu/ai1SQiIiIiYgUFJw9gs9ncPgYPHnxer/v999/z4osvFvn4qKgo4uPjadas2Xm9X1EpoImIiIhISVNzCA8QHx+f/+sZM2bw7LPPsnXr1vwxf3//Asfn5ubi7e1d6OtWrFjxnOpwOBxERESc03NERERERMoCzTh5gIiIiPxHaGgoNpstfzsrK4uwsDC++eYbevXqhZ+fH1OnTiU5OZlbb72VGjVqEBAQQPPmzZk2bVqB1z31Ur3atWvzyiuvcM899xAcHEzNmjWZNGlS/v5TZ4Lmz5+PzWbj999/p127dgQEBNClS5cCoQ7gpZdeomrVqgQHBzN06FD+85//0KpVq/P+/cjOzmbkyJFUrVoVPz8/unXrxooVK/L3Hz58mNtvv50qVarg7+9P/fr1+eyzzwCzs+KIESOIjIzEz8+P2rVrM27cuPOuRUREREQ8g4JTIQzDICMnz5KHYRjF9jmefPJJRo4cyZYtW+jTpw9ZWVm0bduWn3/+mY0bN3Lvvfdy55138vfff7t9nfHjx9OuXTvWrFnDAw88wP33388///zj9jlPP/0048ePZ+XKlXh5eXHPPffk7/vqq694+eWXee2111i1ahU1a9Zk4sSJF/RZn3jiCb777js+//xzVq9eTb169ejTpw+HDh0C4JlnnmHz5s38+uuvbNmyhYkTJ1K5cmUA3nnnHWbNmsU333zD1q1bmTp1KrVr176gekRERESk7NOleoXIzHXS5Nk5lrz35rF9CPApnlM0atQorr/++gJjjz32WP6vH3roIX777TdmzpxJx44dz/o6V111FQ888ABghrEJEyYwf/58GjVqdNbnvPzyy/Ts2ROA//znP/Tr14+srCz8/Px49913GTJkCHfffTcAzz77LHPnzuXo0aPn9TnT09OZOHEiU6ZMoW/fvgBMnjyZmJgYPvnkEx5//HHi4uJo3bo17dq1AygQjOLi4qhfvz7dunXDZrNRq1at86pDRERERDyLZpzKieMh4Tin08nLL79MixYtqFSpEkFBQcydO5e4uDi3r9OiRYv8Xx+/JDAxMbHIz4mMjATIf87WrVvp0KFDgeNP3T4XO3bsIDc3l65du+aPeXt706FDB7Zs2QLA/fffz/Tp02nVqhVPPPEES5YsyT928ODBrF27loYNGzJy5Ejmzp173rWIiIiIiOfQjFMh/L0dbB7bx7L3Li6BgYEFtsePH8+ECRN4++23ad68OYGBgYwaNYqcnBy3r3NqUwmbzYbL5Sryc2w2G0CB5xwfO+5CLlE8/twzvebxsb59+7J7925++eUX5s2bx+WXX86DDz7Im2++SZs2bYiNjeXXX39l3rx53HzzzVxxxRV8++23512TiIiIeKiEDeATCBXrWl2JlADNOBXCZrMR4ONlyePU//wXp0WLFtG/f3/uuOMOWrZsSd26ddm+fftFe7+zadiwIcuXLy8wtnLlyvN+vXr16uHj48PixYvzx3Jzc1m5ciWNGzfO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8+Tx559/nttuu41ffvmFX3/9leeee47p06dz3XXXMXToUPr06cMvv/zC3LlzGTduHOPHj+ehhx46r3pERETkPBzaCd/cBQnHfoDZ43HoNQbsDvfP8zQ1O0LNr8xLFZe+C+umm534pi8zF9ftOhKa3wReasx1LjTjVBibzbxczorHOXZ5ufnmm3E4HHz99dd8/vnn3H333fn/6V+0aBH9+/fnjjvuoGXLltStW5ft27cX+bWbNGlCXFwc+/efCJFLly4tcMxff/1FrVq1ePrpp2nXrh3169c/rdOfj48PTqez0Pdau3Yt6enpBV7bbrfToMHFuU73+Ofbs2dP/tjmzZtJSUmhcePG+WMNGjTgkUceYe7cuVx//fV89tln+fuioqIYPnw433//PY8++iiTJ0++KLWKiIjIGWyeBR/1NEOTf0W4/Tu47P/KX2g6mTrxFSsFJw8SFBTEwIEDeeqpp9i/fz+DBw/O31evXj1iYmJYsmQJW7Zs4b777iMhIaHIr33FFVfQsGFDBg0axLp161i0aBFPP/10gWPq1atHXFwc06dPZ8eOHbzzzjv88MMPBY6pXbs2sbGxrF27lqSkJLKzT1934Pbbb8fPz4+77rqLjRs38ueff/LQQw9x55135t/fdL6cTidr164t8Ni8eTNXXHEFLVq04Pbbb2f16tUsX76cQYMG0bNnT9q1a0dmZiYjRoxg/vz57N69m7/++osVK1bkh6pRo0YxZ84cYmNjWb16NX/88UeBwCUiIiIXSV4O/DYGvrkTslMhqqPZNa/+mde4LJfUia9YKDh5mCFDhnD48GGuuOIKatasmT/+zDPP0KZNG/r06UOvXr2IiIhgwIABRX5du93ODz/8QHZ2Nh06dGDo0KG8/PLLBY7p378/jzzyCCNGjKBVq1YsWbKEZ555psAxN9xwA1deeSWXXnopVapUOWNL9ICAAObMmcOhQ4do3749N954I5dffjnvvffeuf1mnMHRo0dp3bp1gcdVV12V3w69QoUK9OjRgyuuuIK6desyY8YMABwOB8nJyQwaNIgGDRpw880307dvX1544QXADGQPPvggjRs35sorr6Rhw4Z88MEHF1yviIiIuHFkD3zWF5Yd+57b5SEY/AuE1nD/vPLqeCe+h9dD/w+gSiMzbP71X3i7uTkTdXBr4a9TTtmMM9284sFSU1MJDQ0l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Xbt2ERAQQNWqVYutwOKWmppKaGgoKSkphISEWF2OR3O6DB7/dh3fr96H3QZv3NiSG9rWsLosERERERHg3LLBec04ZWZmkp2dnR+adu/ezdtvv83WrVtLdWiSkuWw23jzxpbc0j4KlwGPfbuOGSvirC5LREREROScnVdw6t+/P1988QUAR44coWPHjowfP54BAwYwceLEYi1Qyja73cYr1zXnjk41MQx48rsNTF222+qyRERERETOyXkFp9WrV9O9e3cAvv32W8LDw9m9ezdffPEF77zzTrEWKGWf3W7jxf7NuLtrbQD+78eNfPZXrLVFiYiIiIicg/MKThkZGQQHBwMwd+5crr/+eux2O506dWL3bs0myOlsNhvPXt2E+3rUBeCFnzYzaeEOi6sSERERESma8wpO9erV48cff2TPnj3MmTOH6OhoABITE9VwQc7KZrPxn76NeOiyegC8Mvsf3v/zX4urEhEREREp3HkFp2effZbHHnuM2rVr06FDBzp37gyYs0+tW7cu1gLFs9hsNh6Nbsjo3g0AeGPOVibEbOM8mzuKiIiIiJSI825HnpCQQHx8PC1btsRuN/PX8uXLCQkJoVGjRsVaZHFSO/LS44P5//L6b1sBeKDXJTzepyE2m83iqkRERESkvDiXbOB1vm8SERFBREQEe/fuxWazUb16dS1+K+fkgV718HHYeemXLXwwfwe5ThdPXdVY4UlERERESp3zulTP5XIxduxYQkNDqVWrFjVr1iQsLIwXX3wRl8tV3DWKBxvavS5j+zcFYPKiWF74abMu2xMRERGRUue8ZpyefvppPvnkE1599VW6du2KYRj89ddfPP/882RlZfHyyy8Xd53iwQZ1ro2X3c5TP2xgypJd5DhdvNS/GXa7Zp5EREREpHQ4r3ucqlWrxocffsi1115bYPx///sfDzzwAPv27Su2Aoub7nEqvb5ZuYcnv1uPYcBNbWvw6g0tcCg8iYiIiMhFci7Z4Lwu1Tt06NAZG0A0atSIQ4cOndNrffDBB9SpUwc/Pz/atm3LokWLivS8v/76Cy8vL1q1anVO7yel183tophwcyvsNpi5ai+PzVxHnlOXfoqIiIiI9c4rOLVs2ZL33nvvtPH33nuPFi1aFPl1ZsyYwahRo3j66adZs2YN3bt3p2/fvsTFxbl9XkpKCoMGDeLyyy8/59qldBvQujrv3Noah93GD2v2MWrGWnIVnkRERETEYud1qd6CBQvo168fNWvWpHPnzthsNpYsWcKePXuYPXs23bt3L9LrdOzYkTZt2jBx4sT8scaNGzNgwADGjRt31ufdcsst1K9fH4fDwY8//sjatWuLXLsu1SsbftuYwEPTVpPrNLiyaQTv3NoaH6/zyvkiIiIiImd00S/V69mzJ9u2beO6667jyJEjHDp0iOuvv55Nmzbx2WefFek1cnJyWLVqFdHR0QXGo6OjWbJkyVmf99lnn7Fjxw6ee+65Ir1PdnY2qampBR5S+l3ZLIIP72iLj8POb5sSeOCrVWTnOa0uS0RERETKqfP+EX61atV4+eWX+e677/j+++956aWXOHz4MJ9//nmRnp+UlITT6SQ8PLzAeHh4OAkJCWd8zvbt2/nPf/7DV199hZdX0RoCjhs3jtDQ0PxHVFRUkZ4n1ru8cTiT72qHr5edeVsSufeLVWTlKjyJiIiISMmz/NqnUxc7NQzjjAugOp1ObrvtNl544QUaNGhQ5NcfM2YMKSkp+Y89e/ZccM1Scno2qMKng9vj521nwbaDDPl8BZk5Ck8iIiIiUrIsC06VK1fG4XCcNruUmJh42iwUQFpaGitXrmTEiBF4eXnh5eXF2LFjWbduHV5eXvzxxx9nfB9fX19CQkIKPKRs6VqvMlPu7kCAj4O//k1m8GfLSc/Os7osERERESlHLAtOPj4+tG3blpiYmALjMTExdOnS5bTjQ0JC2LBhA2vXrs1/DB8+nIYNG7J27Vo6duxYUqWLBTrVrcSXQzoQ5OvF37GHGPTpctKycq0uS0RERETKiaLdKHTM9ddf73b/kSNHzunNR48ezZ133km7du3o3LkzkyZNIi4ujuHDhwPmZXb79u3jiy++wG6306xZswLPr1q1Kn5+fqeNi2dqW6siU4d25M5P/mbV7sPc9OFSJgxsReNIzSKKiIiIyMV1TsEpNDS00P2DBg0q8usNHDiQ5ORkxo4dS3x8PM2aNWP27NnUqlULgPj4+ELXdJLypVVUGNOGdWLQp8v5JyGNa99bzIhL6/PApZfg7bD8lj0RERER8VDntY5TWaZ1nDxDYmoWT/+4kZjNBwBoEhnCGze1oGk19+FeREREROS4i76Ok4jVqob4MenOtvz3llaEBXizOT6V/u/9xVsx28jJc1ldnoiIiIh4GAUnKbNsNhv9W1Un5pGeXNk0gjyXwTu/b+fa9xazcV+K1eWJiIiIiAdRcJIyr0qwLxPvaMN7t7WmYqAP/ySk0f/9vxg/dyvZeVrzSUREREQunIKTeASbzcbVLaox95Ee9GseidNl8O4f/3Ltu3+xfu8Rq8sTERERkTJOwUk8SuUgX96/vQ3v39aGSoE+bD2QxnUfLOH13/7R7JOIiIiInDcFJ/FI/VpEMveRHlzTshpOl8EH83dw9TuLWbvniNWliYiIiEgZpOAkHqtSkC/v3tqaD+9oQ+UgH7YnHuX6D/5i3K9byMrV7JOIiIiIFJ2Ck3i8K5tFEvNIT/q3qobLgI8W7KTfO4tYHXfY6tJEREREpIxQcJJyoUKgD/+9pTWT7mxLlWBfdhxM58aJS3hltmafRERERKRwCk5SrkQ3jSDmkR5c37o6LgMmLdzJVf9dxKrdh6wuTURERERKMQUnKXfCAnx4a2ArPrmrHeEhvuxMSufGD5fy4s+byczR7JOIiIiInE7BScqtyxuHM3dUT25sWwPDgE8Wx9L3vwtZHqvZJxEREREpSMFJyrXQAG/evKkln93dnogQP3YlZzBw0lKen7WJjJw8q8sTERERkVJCwUkEuLRhVeaO7sHAdlEYBkxZsosr317Esp3JVpcmIiIiIqWAgpPIMSF+3rx2Yws+v6cDkaF+xB3K4JZJy3jufxtJz9bsk4iIiEh5puAkcoqeDaow95Ee3NqhJgCfL93Nlf9dyJIdSRZXJiIiIiJWUXASOYNgP2/GXd+cL4d0oHqYP3sOZXLb5L/5vx83cFSzTyIiIiLljoKTiBvd61dhziM9uL2jOfs0dVkcfSYs5K9/NfskIiIiUp4oOIkUIsjXi5eva87XQztSo4I/+45kcvvHfzPm+w2kZeVaXZ6IiIiIlAAFJ5Ei6lKvMnNG9WBQ51oATFtuzj4t3HbQ4spERERE5GJTcBI5B4G+Xozt34xpwzoRVdGf/SlZDPp0OU9+u55UzT6JiIiIeCwFJ5Hz0PmSSswZ1YPBXWoDMGPlHvpMWMifWxOtLUxERERELgoFJ5HzFODjxfPXNmXGvZ2oVSmA+JQs7v5sBY/PXEdKpmafRERERDyJgpPIBepYtxK/PdyDe7rWwWaDmav2Ej1hAX/8c8Dq0kRERESkmCg4iRQDfx8Hz17ThJn3daZO5UAOpGZzz5SVjP5mLSkZmn0SERERKesUnESKUbvaFfn14e4M627OPn2/eh+9JywgZrNmn0RERETKMgUnkWLm5+3g6X5N+HZ4F+pWCSQxLZthX6xk1PQ1HE7Psbo8ERERETkPCk4iF0nbWhWYPbI79/Wsi90GP67dT+8JC5mzKcHq0kRERETkHCk4iVxEft4OxvRtzHf3d6Fe1SCSjmZz35erGDltDYc0+yQiIiJSZig4iZSA1jUr8PND3Xig1yXYbTBr3X6iJyzg1w3xVpcmIiIiIkWg4CRSQvy8HTxxZSN+eKArDcKDSDqaw/1frebBr1eTfDTb6vJERERExA0FJ5ES1jIqjJ8e6saIS+vhsNv4ZX08vScs5Jf1mn0SERERKa0UnEQs4Ovl4LE+Dfnxga40igjmUHoOD369mvunruJgmmafREREREobBScRCzWvEcqsEd0YeXl9vOw2ft2YQPSEBcxatx/DMKwuT0RERESOUXASsZiPl53RvRvw44NdaRwZwuGMXEZOW8PwqatITMuyujwRERERQcFJpNRoVj2U/z3YlUeuaICX3cacTQeInrCQH9fs0+yTiIiIiMUUnERKER8vOw9fUZ9ZI7rRtFoIRzJyGTVjLcO+WEViqmafRERERKyi4CRSCjWpFsKPD3bl0d4N8HbYmLflAFe8tYDvV+/V7JOIiIiIBRScREopb4edhy6vz08PdaN59VBSs/IY/c06hny+koQUzT6JiIiIlCQFJ5FSrlFECD880IXH+zTEx2Hnj38S6T1hATNX7tHsk4iIiEgJUXASKQO8HHYevLQev4zsRsuoMNKy8nj82/XcPWUF8SmZVpcnIiIi4vEUnETKkPrhwXw3vDP/6dsIHy8787ceJPqthcxYEafZJxEREZGLSMFJpIzxctgZ3vMSZo/sTuuaYaRl5/HkdxsY9Oly9h3R7JOIiIjIxaDgJFJG1asaxLfDu/D0VY3x9bKzaHsSfSYs5Ou/NfskIiIiUtwUnETKMIfdxrAedZn9cHfa1qrA0ew8nvphA3d+spw9hzKsLk9ERETEYyg4iXiAS6oE8c19nXnm6ib4edtZ/G8SV769kC+X7cbl0uyTiIiIyIVScBLxEA67jSHd6vDrwz1oX7sC6TlOnvlxI7d//Ldmn0REREQukIKTiIepUzmQGfd25rlrmuDv7WDpzmT6vL2QL5bu0uyTiIiIyHlScBLxQHa7jbu71uG3Ud3pWKciGTlOnv3fJm6dvIzdyelWlyciIiJS5ig4iXiwWpUCmTasE2P7NyXAx8HfsYfo8/ZC3v19O1m5TqvLExERESkzFJxEPJzdbmNQ59rMGdWDLpdUIivXxfiYbURPWEjM5gNqXS4iIiJSBApOIuVEVMUAvhrakXdubU1EiB9xhzIY9sVKBn+2gp0Hj1pdnoiIiEipZjPK2Y+bU1NTCQ0NJSUlhZCQEKvLEbFEenYe7//5Lx8viiXH6cLbYeOebnV46LL6BPl6WV2eiIiISIk4l2yg4CRSjsUmpTP2p038ufUgAOEhvozp25j+raphs9ksrk5ERETk4lJwckPBSeR0v285wNifN7M72VzvqX3tCjx/bVOaVgu1uDIRERGRi0fByQ0FJ5Ezy8p18sniWN77418yc53YbXBbx5o8Ft2QsAAfq8sTERERKXYKTm4oOIm4t/9IJq/M3sLP6+MBCAvw5rHohtzaoSYOuy7fExEREc+h4OSGgpNI0Szdkczzszax9UAaAE2rhTC2f1Pa1qpocWUiIiIixUPByQ0FJ5Giy3O6mLpsN+NjtpGWlQfA9a2r85++jaga4mdxdSIiIiIXRsHJDQUnkXOXdDSbN37byjer9mAYEOTrxcjL6zG4Sx18vLQcnIiIiJRNCk5uKDiJnL91e47w7KxNrNtzBIC6VQJ5/pqm9GhQxdrCRERERM6DgpMbCk4iF8blMvh29V5e/+0fko7mABDdJJxnrm5CVMUAi6sTERERKToFJzcUnESKR2pWLm/HbOfzpbtwugx8vezc1/MS7u95Cf4+DqvLExERESmUgpMbCk4ixWvbgTSen7WJJTuSAage5s8zVzemT9MIbDa1LxcREZHSS8HJDQUnkeJnGAa/bkzgpZ83sz8lC4Bu9Srz/LVNqFc12OLqRERERM5MwckNBSeRiyczx8nE+f/y4cKd5OS58LLbGNylNg9fUZ9gP2+ryxMREREpQMHJDQUnkYsvLjmDF3/ZTMzmAwBUDvLlP30bcX3r6tjtunxPRERESgcFJzcUnERKzvytiYz9aTM7k9IBaFMzjBeubUbzGqEWVyYiIiKi4OSWgpNIycrJc/HpX7G8+/t20nOc2GxwS/soHu/TiIqBPlaXJyIiIuWYgpMbCk4i1jiQmsW42Vv4ce1+AEL8vHg0uiG3d6yJl8NucXUiIiJSHik4uaHgJGKtFbsO8dz/NrE5PhWARhHBvHBtUzrWrWRxZSIiIlLeKDi5oeAkYj2ny+Dr5XGMn7uVIxm5AFzbshpPXdWYiFA/i6sTERGR8kLByQ0FJ5HS43B6Dm/O3crXy+MwDAjwcTDisnoM6VYHXy+H1eWJiIiIh1NwckPBSaT02bgvhedmbWLV7sMA1KkcyLNXN+HSRlUtrkxEREQ8mYKTGwpOIqWTYRj8sGYf4379h4Np2QBc3qgqz1zdhNqVAy2uTkRERDyRgpMbCk4ipVtaVi7v/vEvny6OJc9l4OOwM6xHHR68tB4BPl5WlyciIiIeRMHJDQUnkbLh38SjvPDTJhZtTwIgMtSPp65qzNUtIrHZbBZXJyIiIp5AwckNBSeRssMwDOZuPsCLP29m7+FMADrVrcjz1zalUYT+/oqIiMiFUXByQ8FJpOzJynXy0YKdfDD/X7LzXDjsNu7sVItHejcg1N/b6vJERESkjFJwckPBSaTs2ns4g5d/2cKvGxMAqBjowxN9GnJzuyjsdl2+JyIiIufmXLKBvYRqOqsPPviAOnXq4OfnR9u2bVm0aNFZj/3+++/p3bs3VapUISQkhM6dOzNnzpwSrFZErFSjQgAT72jL1CEdqVc1iEPpOfzn+w1c98FfrIk7bHV5IiIi4sEsDU4zZsxg1KhRPP3006xZs4bu3bvTt29f4uLiznj8woUL6d27N7Nnz2bVqlVceumlXHPNNaxZs6aEKxcRK3WrX5lfH+7O//VrTLCvF+v2pnDdB0t4fOa6/FbmIiIiIsXJ0kv1OnbsSJs2bZg4cWL+WOPGjRkwYADjxo0r0ms0bdqUgQMH8uyzzxbpeF2qJ+JZEtOyeP23rXy7ai8Awb5ejOrdgEGda+HtsHxSXUREREqxMnGpXk5ODqtWrSI6OrrAeHR0NEuWLCnSa7hcLtLS0qhYseJZj8nOziY1NbXAQ0Q8R9VgP968qSXf3d+F5tVDScvO48WfN9PvnUUs+TfJ6vJERETEQ1gWnJKSknA6nYSHhxcYDw8PJyEhoUivMX78eNLT07n55pvPesy4ceMIDQ3Nf0RFRV1Q3SJSOrWtVYEfH+zKuOubUzHQh20HjnLbx3/zwFer2Hck0+ryREREpIyz/DqWUxeyNAyjSItbTps2jeeff54ZM2ZQtWrVsx43ZswYUlJS8h979uy54JpFpHRy2G3c2qEmfz7ai7s618Jug9kbErh8/Hze/X07WblOq0sUERGRMsqy4FS5cmUcDsdps0uJiYmnzUKdasaMGQwZMoRvvvmGK664wu2xvr6+hISEFHiIiGcLDfDmhf7N+GVkdzrUqUhWrovxMduInrCQmM0HKGerMIiIiEgxsCw4+fj40LZtW2JiYgqMx8TE0KVLl7M+b9q0aQwePJivv/6afv36XewyRaQMaxwZwox7O/HOra2JCPEj7lAGw75YyeDPVrDz4FGryxMREZEyxNJL9UaPHs3HH3/Mp59+ypYtW3jkkUeIi4tj+PDhgHmZ3aBBg/KPnzZtGoMGDWL8+PF06tSJhIQEEhISSElJseojiEgpZ7PZuLZlNX5/tCcP9LoEH4edBdsO0ufthYz7dQtHs/OsLlFERETKAEvbkYO5AO7rr79OfHw8zZo1Y8KECfTo0QOAwYMHs2vXLubPnw9Ar169WLBgwWmvcddddzFlypQivZ/akYuUb7FJ6bz482b++CcRgKrBvjx1VWP6t6pWpPsrRURExHOcSzawPDiVNAUnEQH4fcsBxv68md3JGQC0r12B569tStNqoRZXJiIiIiVFwckNBScROS4r18kni2N5749/ycx1YrfBbR1r8lh0Q8ICfKwuT0RERC4yBSc3FJxE5FT7j2Tyyuwt/Lw+HoCwAG8ei27IrR1q4rDr8j0RERFPpeDkhoKTiJzN0h3JPD9rE1sPpAHQtFoIL1zblHa1K1pcmYiIiFwMCk5uKDiJiDt5ThdTl+3mrZhtpGaZHfeua12dMX0bUTXEz+LqREREpDgpOLmh4CQiRZF8NJs35mxlxso9GAYE+ji4q0ttBnetTdVgBSgRERFPoODkhoKTiJyLdXuO8NysTazdcwQAH4ed69tUZ2j3utSrGmRtcSIiInJBFJzcUHASkXPlchnEbDnApIU7WbX7cP547ybh3Nejru6BEhERKaMUnNxQcBKRC7Fy1yE+WriTmM0H8sfa1Azjvp6X0LtxOHZ14RMRESkzFJzcUHASkeLwb+JRPl60k+9X7yPH6QKgbuVAhnavy/VtquPn7bC4QhERESmMgpMbCk4iUpwSU7OYsmQXU5ftzu/CVznIl8FdanFHp1paSFdERKQUU3ByQ8FJRC6Go9l5TF8ex6eLY9mfkgVAgI+Dge2jGNKtDjUqBFhcoYiIiJxKwckNBScRuZhynS5+WR/Phwt28E+CuZCuw27j6haR3NujLk2rhVpcoYiIiByn4OSGgpOIlATDMFi0PYlJC3ey+N+k/PHu9Stzb4+6dKtXGZtNjSRERESspODkhoKTiJS0jftSmLRwJ79siMfpMv/JbRIZwn0969KveSReDrvFFYqIiJRPCk5uKDiJiFX2HMrgk8WxzFixh8xcJwDVw/wZ0q0OA9tHEejrZXGFIiIi5YuCkxsKTiJitcPpOUxdtpspS3aRnJ4DQKi/N3d2qsVdXWpTJdjX4gpFRETKBwUnNxScRKS0yMp18t3qvXy8KJbYpHQAfLzs3NCmOkO71+WSKkEWVygiIuLZFJzcUHASkdLG6TKI2ZzARwt3sibuCAA2G/RuHM59PS+hba0K1hYoIiLioRSc3FBwEpHSyjAMVu4+zEcLdjBvS2L+eLtaFbiv5yVc3qgqdrs68YmIiBQXBSc3FJxEpCz4NzGNSQt38uOa/eQ4XQBcUiWQe3vUZUDr6vh6OSyuUEREpOxTcHJDwUlEypIDqVl89tcuvvp7N2lZeQBUCfZlcJfa3NGxFqEB3hZXKCIiUnYpOLmh4CQiZVFaVi4zVuzhk8WxxKdkARDo4+CWDjW5p1sdqof5W1yhiIhI2aPg5IaCk4iUZTl5Ln5ev5+PFuxk64E0ALzsNq5pWY17e9SlcaT+XRMRESkqBSc3FJxExBMYhsGCbQf5aMFOlu5Mzh/v0aAK9/WoS5dLKmGzqZGEiIiIOwpObig4iYinWb/3CJMW7mT2hnhcx/5Fb1Y9hHt7XMJVzSLwctitLVBERKSUUnByQ8FJRDxVXHIGnyzeyYyVe8jKNTvx1ajgz9Budbi5fRQBPl4WVygiIlK6KDi5oeAkIp7uUHoOXy7dzedLd3EoPQeAsABvBnWqxaAutakc5GtxhSIiIqWDgpMbCk4iUl5k5jj5dvVePl60k93JGQD4etm5oW0NhnWvS53KgRZXKCIiYi0FJzcUnESkvHG6DOZsSuCjBTtYtzcFAJsN+jSJ4L6edWlds4LFFYqIiFhDwckNBScRKa8Mw+Dv2ENMWriTP/5JzB/vULsi9/Wsy6UNq2K3qxOfiIiUHwpObig4iYjAtgNpTFq4k/+t3Ueu0/w2UK9qEPd2r0v/1tXw9XJYXKGIiMjFp+DkhoKTiMgJCSlZfLYklq+XxZGWnQdA1WBf7u5ah9s61iTU39viCkVERC4eBSc3FJxERE6XmpXL9OVxfLI4lgOp2QAE+Xpxa4co7ulWh8hQf4srFBERKX4KTm4oOImInF1OnotZ6/YzaeEOth04CoCX3ca1rapxb4+6NIrQv5siIuI5FJzcUHASESmcYRjM33qQDxfs4O/YQ/njvRpW4d4edelctxI2mxpJiIhI2abg5IaCk4jIuVm35wiTFu7k143xuI59x2hePZT7etblyqYReDns1hYoIiJynhSc3FBwEhE5P7uT0/l4USzfrNxDdp4LgKiK/gzrXpeb2kbh76NOfCIiUrYoOLmh4CQicmGSj2bzxdLdfLF0F4czcgGoEODNoM61GdS5FpWCfC2uUEREpGgUnNxQcBIRKR6ZOU5mrtrDx4tiiTuUAYCvl52b20UxtHsdalUKtLhCERER9xSc3FBwEhEpXnlOF79tSmDSwp2s35sCgN0GVzaL4N4el9AqKszaAkVERM5CwckNBScRkYvDMAyW7TzERwt3MH/rwfzxptVCiG4SQXTTcBpFBKsbn4iIlBoKTm4oOImIXHz/JKQyaeFOZq3dT57rxLeZqIr+RDeJoHeTcNrVqqCOfCIiYikFJzcUnERESk7y0Wx+/yeRuZsOsGj7wfxufGA2lLi8cTjRTcLpXr+KuvKJiEiJU3ByQ8FJRMQaGTl5LNyWRMzmA/z+zwGOHOvIB+Dnbad7/SpENwnn8sbhVAz0sbBSEREpLxSc3FBwEhGxXp7TxYpdh5m7OYG5mw6w70hm/j67DdrXrkjvJuFEN4mgZqUACysVERFPpuDkhoKTiEjpYhgGW+LT8kPU5vjUAvsbRQQT3TSC6CbhNK0WouYSIiJSbBSc3FBwEhEp3fYcymDelgPM3XSA5bsO4TypuUT1MP9jM1HhtK9TEW81lxARkQug4OSGgpOISNlxOD2HP/5JJGbzARZsO0hmrjN/X4ifV35ziR4NqhDo62VhpSIiUhYpOLmh4CQiUjZl5TpZvD2JuZsTmLclkUPpOfn7fLzsdK9Xmd7HmktUCfa1sFIRESkrFJzcUHASESn7nC6D1XGHmbspgbmbD7A7OSN/n80GbWtWILppOL2bRFCncqCFlYqISGmm4OSGgpOIiGcxDINtB44Ss9kMUev3phTYX79qENFNzQ59zauHYreruYSIiJgUnNxQcBIR8Wz7j2Qyb8sBYjYfYOmOZPJOai4RHuKb3+a8U91K+HipuYSISHmm4OSGgpOISPmRkpnL/K2JzN10gPlbE0nPOdFcItjXi0sbVaV3k3B6NaxCsJ+3hZWKiIgVFJzcUHASESmfsnKdLN2ZzNxN5mxU0tHs/H3eDhtdLqls3hfVOJyqIX4WVioiIiVFwckNBScREXG5DNbsOULM5gPM3ZTAzqT0AvtbRYXl3xdVr2qQRVWKiMjFpuDkhoKTiIic6t/Eo8zdnMDcTQdYu+dIgX11qwTm3xfVOipMzSVERDyIgpMbCk4iIuLOgdQs5m05wNxNB1iyI4lc54lvk5WDjjeXCKfzJZXw83ZYWKmIiFwoBSc3FJxERKSo0rJymb/1IDGbD/DnP4mkZefl7wv0cdCrYVWim4bTq2FVQv3VXEJEpKxRcHJDwUlERM5HTp6LZTuTmbs5gZjNBziQeqK5hJfdRqe6lYhuGs4VjcOpFuZvYaUiIlJUCk5uKDiJiMiFcrkMNuxLyb8vanvi0QL7m1cPJbpJONFNI2gQHoTNpvuiRERKIwUnNxScRESkuMUmpRNzLEStijvMyd9Za1UKILpJOL2bRNC2VgUcai4hIlJqKDi5oeAkIiIX08G0bH7fYq4VtejfJHLyXPn7KgX6cHnjqkQ3iaBb/cpqLiEiYjEFJzcUnEREpKSkZ+excNtB5m4+wO9bDpCadaK5hL+3gx4NKhPdJILLGlWlQqCPhZWKiJRPCk5uKDiJiIgVcp0uVsQeYu6xRXf3p2Tl73PYbXSoXZHeTcLp3SScqIoBFlYqIlJ+KDi5oeAkIiJWMwyDTftT80PUPwlpBfY3iQzh8sZV6Vy3Em1qVdAlfSIiF4mCkxsKTiIiUtrsOZSRH6JW7DqE66TvzD4OO62iwuhUtyKdFKRERIqVgpMbCk4iIlKaHUrP4fctB/jr3ySW7kwusF4UKEiJiBQnBSc3FJxERKSsMAyD3ckZLNuZzLKdyQpSIiLFTMHJDQUnEREpq4ocpGqG0aluJTrVrUibmgpSIiJno+DkhoKTiIh4CgUpEZELo+DkhoKTiIh4KsMw2HVSkFqmICUi4paCkxsKTiIiUl4oSImIuKfg5IaCk4iIlFenBqmlO5JJTFOQEpHyS8HJDQUnERERU5GClJed1lHHg1QlWtcMU5ASEY+h4OSGgpOIiMiZKUiJSHmj4OSGgpOIiEjRKEiJiKdTcHJDwUlEROT8KEiJiKdRcHJDwUlERKR4GIZBbFI6y3Yeyg9TClIiUpYoOLmh4CQiInJxFDVItal5Iki1ilKQEhHrlKng9MEHH/DGG28QHx9P06ZNefvtt+nevftZj1+wYAGjR49m06ZNVKtWjSeeeILhw4cX+f0UnEREREqGgpSIlHZlJjjNmDGDO++8kw8++ICuXbvy0Ucf8fHHH7N582Zq1qx52vGxsbE0a9aMYcOGcd999/HXX3/xwAMPMG3aNG644YYivaeCk4iIiDUUpESktCkzwaljx460adOGiRMn5o81btyYAQMGMG7cuNOOf/LJJ5k1axZbtmzJHxs+fDjr1q1j6dKlRXpPBScREZHS4dQgtXRnMgcVpESkBJ1LNvAqoZpOk5OTw6pVq/jPf/5TYDw6OpolS5ac8TlLly4lOjq6wFifPn345JNPyM3Nxdvb+6LVKyIiIsXLZrNRt0oQdasEcVvHmmcNUub2IWC7gpSIWMay4JSUlITT6SQ8PLzAeHh4OAkJCWd8TkJCwhmPz8vLIykpicjIyNOek52dTXb2iZ9epaamFkP1IiIiUtzOFKR2JqUfu6zPDFOnBilfLzttalY4FqQq0lJBSkQuEsuC03E2m63AtmEYp40VdvyZxo8bN24cL7zwwgVWKSIiIiXNZrNxSZUgLqkSxO0da501SC09NjsFYLdBxUBfKgf5UCnIh8pBvlQK9KVysA+VA31PjB37qpAlIkVlWXCqXLkyDofjtNmlxMTE02aVjouIiDjj8V5eXlSqVOmMzxkzZgyjR4/O305NTSUqKuoCqxcREZGSVtQglXTUfBRFkK/XSQHLh8rBvlQO9KFSkG+BgFU5yIdQf2+3P9wVEc9mWXDy8fGhbdu2xMTEcN111+WPx8TE0L9//zM+p3Pnzvz0008FxubOnUu7du3Oen+Tr68vvr6+xVe4iIiIlApnClJmcMohOd0MT8lHczh47Gvy0WP7jn3Ncbo4mp3H0ew8didnFPp+XnYblYJ8qHRs5qpK0MkzWAXHKgX64uNlL4HfBREpKZZeqjd69GjuvPNO2rVrR+fOnZk0aRJxcXH56zKNGTOGffv28cUXXwBmB7333nuP0aNHM2zYMJYuXconn3zCtGnTrPwYIiIiUgrYbDaqhvhRNcSv0GMNwyAtO4/koznHAlY2B4+FqhNjOfmzV6lZeeS5DA6kZnMgtWizWSF+XsdmqwpeIlgpyJcqx74en+UK9vXSbJZIKWdpcBo4cCDJycmMHTuW+Ph4mjVrxuzZs6lVqxYA8fHxxMXF5R9fp04dZs+ezSOPPML7779PtWrVeOedd4q8hpOIiIgImCErxM+bED9v6lQOLPT47Dwnh9JzzjCDdcqsVrr5Nc9lkJqVR2pWHjuT0gt9fR8ve/4lgieHrJNnsI5fMlgx0Acvh2azREqapes4WUHrOImIiMjF5HIZpGblknTKzFXy0WyS0nNISssmOf3EvqPZeef8HhUCvI/dh3V8BsucvTptLMiHAB+HZrNEzqJMrOMkIiIi4onsdhthAT6EBfhQr2pQocdn5TpPCVg5JKVnk5RW8F6tpKM5HErPxmXA4YxcDmfk8m9i4fX4edvz78OqHOhz2qWDxy8nrBLsS5i/N3a7QpbImSg4iYiIiFjIz9tBjQoB1KgQUOixTpfBkYycE00u8mewTr4n68S9WVm5LrJyXew9nMnew5mFvr63w0blIF+qBptBqkqwX/6v87+G+FElSM0vpPxRcBIREREpIxx227H7oHyB4EKPz8jJIynt+AyWeYlg8inh6njgOpyRS67TID4li/iUrEJfOyzA+6RQ5XdS2DppLESNL8RzKDiJiIiIeKgAHy9qVvKiZqXCZ7Ny8lwkHc3mYFo2iWnHv2ad9OtsDqZmcfBoNrlOgyMZuRzJyGXbgaNuX9fP254fpKoE+VI1xDf/a9Vgv/zZLDW9kNJOwUlERERE8PGyUy3Mn2ph/m6PMwwzNB08mk1iqhmuzha20rLyyMp1sedQJnsOub9U0G6DioG+BS4NPBGyTgSsqsF++Ps4ivOjixSJgpOIiIiIFJnNZqNCoA8VAn1oEO7+csHMHLPxRWJaFomp2flh69SAlXTUbHpx/PJB4t3XEOTrRdVgXyqfFKYK3odljqnZhRQnBScRERERuSj8fRxEVQwgqqL7SwWdLoND6TkFwtTBtGwSj10aeHLoysx1cjQ7j6PZha+RdaZmFydmrk40u6gc5IOvl2axxD0FJxERERGxlMNuy28q0dTNcYZhcDQ775RLA09cLnjwpLFD6TkX1Ozi5BksNbsQUHASERERkTLCZrMR7OdNsJ83dau4XyMrJ89FcvrJlwaefj/W8UeO01XkZhe+XvYT914F+xHs54Wvtx0fhwNfbzu+XnZ8vOz4ejmOfT3xOD5+pmNO/urjsCuclUIKTiIiIiLicXy87ESG+hMZWnizi5TM3ILNLc4SttKy8sjOK1qziwt1atA6NVwVDGBnC2JnPua0IOdthrUTXx34OOx4O2wKcCdRcBIRERGRcstmsxEW4ENYQOHNLrJynfnh6niYOpqdR06ei+w817GvTrJzXeQ4XSe+5jnzjzmxz5l/TLbTfO7Jso8dn0bexfz4btlsmEHK60SYOjlc+RbYPhbcThk7PhNX4Jhjwa1D7YpUCPSx7POdKwUnEREREZEi8PMuWrOL82EYxrGQdeZwlR+yTgph+UGtwPaZj8kp9Bjza47TdVJNJwIcWcUf4L67vwttFZxERERERKSobDbbscvoHOBnXR0u10kB7izhqsCs2hlD2Vlm3k6eZctzEurvbd0HPQ8KTiIiIiIiAoDdbsPP7sDP2wGUrWBzsdmtLkBERERERKS0U3ASEREREREphIKTiIiIiIhIIRScRERERERECqHgJCIiIiIiUggFJxERERERkUIoOImIiIiIiBRCwUlERERERKQQCk4iIiIiIiKFUHASEREREREphIKTiIiIiIhIIRScRERERERECqHgJCIiIiIiUggFJxERERERkUIoOImIiIiIiBRCwUlERERERKQQCk4iIiIiIiKFUHASEREREREphJfVBZQ0wzAASE1NtbgSERERERGx0vFMcDwjuFPuglNaWhoAUVFRFlciIiIiIiKlQVpaGqGhoW6PsRlFiVcexOVysX//foKDg7HZbFaXU2alpqYSFRXFnj17CAkJsbocuch0vssXne/yRee7/NE5L190vt0zDIO0tDSqVauG3e7+LqZyN+Nkt9upUaOG1WV4jJCQEP0lLEd0vssXne/yRee7/NE5L190vs+usJmm49QcQkREREREpBAKTiIiIiIiIoVQcJLz4uvry3PPPYevr6/VpUgJ0PkuX3S+yxed7/JH57x80fkuPuWuOYSIiIiIiMi50oyTiIiIiIhIIRScRERERERECqHgJCIiIiIiUggFJxERERERkUIoOEm+hQsXcs0111CtWjVsNhs//vhjgf2GYfD8889TrVo1/P396dWrF5s2bSpwTHZ2Ng899BCVK1cmMDCQa6+9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",
      "text/plain": [
       "<Figure size 1000x600 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "class CausalSft_Model_Phi(nn.Module):\n",
    "    def __init__(self, base_model, num_labels, max_words):\n",
    "        super().__init__()\n",
    "\n",
    "        self.max_words = max_words\n",
    "        self.base_model = copy.deepcopy(base_model)\n",
    "        self.sft_model = copy.deepcopy(base_model)\n",
    "        self.sft_classifier = nn.Linear(self.base_model.config.hidden_size, num_labels)\n",
    "        self.r_2_c = nn.Sequential(\n",
    "            nn.Linear(self.base_model.config.hidden_size, self.base_model.config.hidden_size // 2),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(self.base_model.config.hidden_size // 2, self.base_model.config.hidden_size // 2),\n",
    "        )\n",
    "        self.c_2_c = nn.Sequential(\n",
    "            nn.Linear(self.base_model.config.hidden_size // 4, self.base_model.config.hidden_size // 4),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(self.base_model.config.hidden_size // 4, self.base_model.config.hidden_size // 4),\n",
    "        )\n",
    "        self.num_patches = 10\n",
    "        self.patches_size = self.max_words // self.num_patches\n",
    "        self.patch_extractor = nn.Sequential(\n",
    "            nn.Linear(self.base_model.config.hidden_size * self.num_patches, self.base_model.config.hidden_size * self.num_patches // 2),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(self.base_model.config.hidden_size * self.num_patches // 2, self.base_model.config.hidden_size),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(self.base_model.config.hidden_size, self.base_model.config.hidden_size),\n",
    "        )\n",
    "        # self.c_patch_mixer = nn.Sequential(\n",
    "        #     nn.Linear(self.base_model.config.hidden_size, self.base_model.config.hidden_size // 8),\n",
    "        #     nn.ReLU(),\n",
    "        #     nn.Linear(self.base_model.config.hidden_size // 8, self.base_model.config.hidden_size // 8),\n",
    "        # )\n",
    "        self.classifier = nn.Linear(self.base_model.config.hidden_size, num_labels)\n",
    "\n",
    "        self.cross_entropy_loss = nn.CrossEntropyLoss()\n",
    "        self.mse_loss = nn.MSELoss()\n",
    "\n",
    "        for param in self.base_model.parameters():\n",
    "            param.requires_grad = False\n",
    "\n",
    "        print(f'Initialise Causal SFT model \"{model_name}\" (unfreezed all layers) with a linear head!')\n",
    "        count_parameters(self)\n",
    "\n",
    "    def entropy_maximization(self, feature):\n",
    "        p = F.softmax(feature, dim=-1)\n",
    "        log_p = F.log_softmax(feature, dim=-1)\n",
    "        entropy = -torch.mean(torch.mean(p * log_p, dim=-1))  # Maximize entropy\n",
    "        return -entropy  # Minimize negative entropy\n",
    "\n",
    "    def forward(self, input_ids1, attention_mask1, input_ids2, attention_mask2, input_ids3, attention_mask3, input_ids4, attention_mask4, labels=None):\n",
    "        B, _ = input_ids1.size()\n",
    "        # avoid unwanted computational graph\n",
    "        outputs_sft2 = self.sft_model(input_ids2, attention_mask=attention_mask2)[1]\n",
    "        outputs_sft2_logits = self.sft_classifier(outputs_sft2)\n",
    "        # pull out R0 and R1 using another data point\n",
    "        outputs_base3 = self.base_model(input_ids3, attention_mask=attention_mask3)[1]\n",
    "        outputs_sft3 = self.sft_model(input_ids3, attention_mask=attention_mask3)[1]\n",
    "        # optimise C from R0 and R1\n",
    "        base_C3= self.r_2_c(outputs_base3)[:, :self.base_model.config.hidden_size//4]\n",
    "        sft_C3 = self.r_2_c(outputs_sft3)[:, :self.base_model.config.hidden_size//4]\n",
    "        # gather C \n",
    "        outputs_sft = self.sft_model(input_ids1, attention_mask=attention_mask1)[1]\n",
    "        sft_C = self.r_2_c(outputs_sft)[:, :self.base_model.config.hidden_size//4]\n",
    "        # extract non-contextual embeddings\n",
    "        with torch.no_grad():\n",
    "            embeddings_1 = self.sft_model.embeddings(input_ids1, attention_mask1)[:, :self.max_words]\n",
    "            embeddings_2 = self.sft_model.embeddings(input_ids2, attention_mask2)[:, :self.max_words]\n",
    "            embeddings_4 = self.sft_model.embeddings(input_ids4, attention_mask4)[:, :self.max_words]\n",
    "\n",
    "        input = embeddings_1\n",
    "        patches = torch.sum(input.view(B, self.num_patches, self.patches_size, -1), dim=2)\n",
    "        Ai_samples = self.patch_extractor(patches.view(B, -1))\n",
    "        # Ai_samples = Ai_samples[torch.randperm(B).to(input_ids1.device), :]\n",
    "        Ai_Z1 = Ai_samples\n",
    "        logits = self.classifier(Ai_Z1)   \n",
    "\n",
    "        loss = None\n",
    "        if labels is not None:\n",
    "            loss_sft = self.cross_entropy_loss(outputs_sft2_logits, labels) \n",
    "            loss_identify = self.mse_loss(base_C3, sft_C3) + self.entropy_maximization(sft_C3) + self.entropy_maximization(base_C3)\n",
    "            loss_cls = self.cross_entropy_loss(logits, labels) \n",
    "            \n",
    "            loss = loss_sft + loss_identify + loss_cls \n",
    "        \n",
    "        return (loss, logits) if loss is not None else logits\n",
    "    \n",
    "set_seed(data_seed)\n",
    "model = CausalSft_Model_Phi(base_model, num_cls, max_words)\n",
    "\n",
    "# Define training arguments and trainer\n",
    "training_args = TrainingArguments(\n",
    "    overwrite_output_dir=True,\n",
    "    output_dir=f'./results/amazon/causal-sft-phi/{N}-shot-{data_seed}',\n",
    "    eval_strategy=\"steps\",\n",
    "    save_strategy=\"steps\",\n",
    "    logging_strategy=\"steps\",\n",
    "    logging_steps=0.1,\n",
    "    save_steps=0.1,\n",
    "    learning_rate=5e-5,\n",
    "    per_device_train_batch_size=128,\n",
    "    per_device_eval_batch_size=128,\n",
    "    num_train_epochs=10,\n",
    "    seed=data_seed,\n",
    "    load_best_model_at_end=True,\n",
    "    metric_for_best_model=\"eval_loss\",  # Choose model based on validation loss\n",
    "    greater_is_better=False,  # Lower validation loss is better\n",
    "    save_total_limit=1,\n",
    ")\n",
    "\n",
    "\n",
    "trainer = Trainer(\n",
    "    model=model,\n",
    "    args=training_args,\n",
    "    train_dataset=paired_train_dataset,\n",
    "    eval_dataset=paired_validation_dataset,\n",
    "    tokenizer=tokenizer,\n",
    "    data_collator=paired_data_collator,\n",
    "    compute_metrics=compute_metrics,\n",
    ")\n",
    "\n",
    "# Train the model\n",
    "trainer.train()\n",
    "\n",
    "results = {}\n",
    "train_results = trainer.evaluate(paired_train_dataset)\n",
    "print(f'train: {train_results}')\n",
    "results['train'] = train_results\n",
    "\n",
    "valid_results = trainer.evaluate(paired_validation_dataset)\n",
    "print(f'validation: {valid_results}')\n",
    "results['valid'] = valid_results\n",
    "\n",
    "# Evaluate on the test set\n",
    "test_results_ID_90 = trainer.evaluate(paired_test_dataset_ID_90)\n",
    "print(f'ID 90: {test_results_ID_90}')\n",
    "results['ID 90'] = test_results_ID_90\n",
    "\n",
    "test_results_OOD_change_70 = trainer.evaluate(paired_test_dataset_OOD_change_70)\n",
    "print(f'OOD change 70: {test_results_OOD_change_70}')\n",
    "results['OOD change 70'] = test_results_OOD_change_70\n",
    "\n",
    "test_results_OOD_balanced_50 = trainer.evaluate(paired_test_dataset_OOD_balanced_50)\n",
    "print(f'OOD balanced 50: {test_results_OOD_balanced_50}')\n",
    "results['OOD balanced 50'] = test_results_OOD_balanced_50\n",
    "\n",
    "test_results_OOD_change_30 = trainer.evaluate(paired_test_dataset_OOD_change_30)\n",
    "print(f'OOD change 30: {test_results_OOD_change_30}')\n",
    "results['OOD change 30'] = test_results_OOD_change_30\n",
    "\n",
    "test_results_OOD_flip_10 = trainer.evaluate(paired_test_dataset_OOD_flip_10)\n",
    "print(f'OOD flip: {test_results_OOD_flip_10}')\n",
    "results['OOD flip'] = test_results_OOD_flip_10\n",
    "\n",
    "test_results_OOD_original_0 = trainer.evaluate(paired_test_dataset_OOD_original_0)\n",
    "print(f'OOD original: {test_results_OOD_original_0}')\n",
    "results['OOD original'] = test_results_OOD_original_0\n",
    "\n",
    "# Manually save the results\n",
    "with open(f'./results/amazon/causal-sft-phi/{N}-shot-{data_seed}/test_results.json', 'w') as f:\n",
    "    json.dump(results, f, indent=4)\n",
    "\n",
    "plot_training_metrics(trainer)"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "robustNLP",
   "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.11.5"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
