{
 "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) and [Amazon dataset](https://huggingface.co/datasets/fancyzhx/amazon_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 90/10 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, ClassLabel\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": [
    {
     "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",
      "Generate synthetic data of 2 classes with random seed 5, each class of 4000 training, 1000 validation and 2000 test data.\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "DatasetDict({\n",
       "    train: Dataset({\n",
       "        features: ['text', 'label'],\n",
       "        num_rows: 8000\n",
       "    })\n",
       "    validation: Dataset({\n",
       "        features: ['text', 'label'],\n",
       "        num_rows: 2000\n",
       "    })\n",
       "    test: Dataset({\n",
       "        features: ['text', 'label'],\n",
       "        num_rows: 4000\n",
       "    })\n",
       "})"
      ]
     },
     "execution_count": 2,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "def realworld_data_sampler(dataset, n_samples, sampler_seed, test_per_class=2000, max_search_words=30, string='amazon'):\n",
    "    \n",
    "    # Gather the number of labels from the test dataset with predefined train, validation ratio\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",
    "    def filter_sentence(example):\n",
    "        words = ' '.join(example['text'].split())\n",
    "        search_words = words[:max_search_words] \n",
    "        count_the = len(re.findall(r'\\bthe\\b', search_words, re.IGNORECASE))\n",
    "        # count_and = len(re.findall(r'\\band\\b', search_words, re.IGNORECASE))\n",
    "        return count_the >= 1\n",
    "\n",
    "    def add_string(example):\n",
    "        words = example['text'].split()\n",
    "        new_words = []\n",
    "        for word in words:\n",
    "            new_words.append(word)\n",
    "            if word.lower() == 'the':\n",
    "                for _ in range(3):\n",
    "                    new_words.append(string)\n",
    "        example['text'] = ' '.join(new_words)\n",
    "        return example\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 filter_sentence(x))\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))\n",
    "        val_data = train_valid_data.select(range(train_size, n_samples))\n",
    "\n",
    "        # Select test data with a fixed random seed\n",
    "        filtered_test = dataset['test'].filter(lambda x: x['label'] == label and filter_sentence(x))\n",
    "        test_data = filtered_test.shuffle(seed=42).select(range(test_per_class))\n",
    "\n",
    "        train_data = train_data.map(add_string)\n",
    "        val_data = val_data.map(add_string)\n",
    "        test_data = test_data.map(add_string)\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).shuffle(seed=sampler_seed)\n",
    "    val_dataset = concatenate_datasets(val_datasets).shuffle(seed=sampler_seed)\n",
    "    test_dataset = concatenate_datasets(test_datasets).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\n",
    "\n",
    "N = 5000\n",
    "data_seed = 5\n",
    "max_words= 40\n",
    "\n",
    "ds1 = load_dataset(\"fancyzhx/yelp_polarity\")\n",
    "ds2 = load_dataset(\"fancyzhx/amazon_polarity\")\n",
    "ds2 = ds2.rename_column(\"content\", \"text\")\n",
    "ds2 = ds2.remove_columns(\"title\")\n",
    "\n",
    "set_seed(data_seed)\n",
    "ds1_sample = realworld_data_sampler(ds1, N, data_seed, max_search_words=max_words, string='yelp.xxx')\n",
    "set_seed(data_seed)\n",
    "ds2_sample = realworld_data_sampler(ds2, N, data_seed, max_search_words=max_words, string='amazon.yyy')\n",
    "\n",
    "new_classlabel = ClassLabel(names=['1', '2'])\n",
    "\n",
    "# Re-define the dataset with the updated label feature for ds1\n",
    "ds2_sample = ds2_sample.cast_column('label', new_classlabel)\n",
    "\n",
    "num_cls = len(set(ds1['test']['label']))\n",
    "\n",
    "ds1_sample"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "{'label': [1, 0, 0, 0, 0, 0, 1, 1, 1, 1],\n",
       " 'text': [\"The amazon.yyy amazon.yyy amazon.yyy book leads you from the amazon.yyy amazon.yyy amazon.yyy basics starting with the amazon.yyy amazon.yyy amazon.yyy assembly enviornment, registers, working with ram, and program flow, and it takes you to controlling hardware, taking advantage of the amazon.yyy amazon.yyy amazon.yyy BIOS and DOS service calls, file management, low end graphics, keyboard and mouse control, and even programming TSRs. A great value, especially for the amazon.yyy amazon.yyy amazon.yyy price it's at, for reference and for learning for the amazon.yyy amazon.yyy amazon.yyy first time.\",\n",
       "  \"I don't know what the amazon.yyy amazon.yyy amazon.yyy heck amazon is talking about but my Blade Runner DVD(ordered from amazon) looks horrible. I can't believe they gave it a 5 for video quality. One is more like it! Don't buy this DVD. Such a good movie, such a bad DVD.\",\n",
       "  'I used this thermometer set for the amazon.yyy amazon.yyy amazon.yyy first time to cook a turkey in the amazon.yyy amazon.yyy amazon.yyy Big Easy oil less Fryer. It was completely unreliable. It partially worked through most of the amazon.yyy amazon.yyy amazon.yyy cooking and when it hit about 154 degrees it switched over to an HHH reading that indicates it is to hot. My turkey ended up way over done, what a waste! Just in case the amazon.yyy amazon.yyy amazon.yyy error had been on my part somehow, I went ahead and tested it out again the amazon.yyy amazon.yyy amazon.yyy next day and sure enough as soon as I turned it on the amazon.yyy amazon.yyy amazon.yyy HHH reading came on again while the amazon.yyy amazon.yyy amazon.yyy unit was only at room temperature. I will be taking this back and try to find another thermometer that works.',\n",
       "  'I really resent the amazon.yyy amazon.yyy amazon.yyy titles of the amazon.yyy amazon.yyy amazon.yyy \"XXXX for Dummies,\" since as a teacher I think it creates self-images of inadequacy... The amazon.yyy amazon.yyy amazon.yyy book is OK. Man, the amazon.yyy amazon.yyy amazon.yyy blues is music. Get out there, get the amazon.yyy amazon.yyy amazon.yyy CDs, Hound Dog Taylor, Rev. Gary Davis, Miss. John Hurt, Muddy, etc. Listen and enjoy and support your local blues organizations and clubs. Books for dummies. Nah.',\n",
       "  '..but when it came time to replace the amazon.yyy amazon.yyy amazon.yyy batteries, I found the amazon.yyy amazon.yyy amazon.yyy screws that held the amazon.yyy amazon.yyy amazon.yyy battery compartment down were too tight and had to destroy them to remove the amazon.yyy amazon.yyy amazon.yyy cover. Otherwise, it worked ok.',\n",
       "  'Much of the amazon.yyy amazon.yyy amazon.yyy book is imagined dialogue between the amazon.yyy amazon.yyy amazon.yyy elephant trainer and various characters. The amazon.yyy amazon.yyy amazon.yyy relationships with parents, girlfriend and elephant are unrealistically idealized. The amazon.yyy amazon.yyy amazon.yyy theme is pantheism and the amazon.yyy amazon.yyy amazon.yyy trainer even assumes he was also an elephant in another life. I was not even able to finish this book.',\n",
       "  \"Here is yet another in the amazon.yyy amazon.yyy amazon.yyy Anchor Commentary Series which continues the amazon.yyy amazon.yyy amazon.yyy fine scholarship. Here is wealth of study of philology and hermeutics, with the amazon.yyy amazon.yyy amazon.yyy author's own translation and comments on topics and phrases.There is much to be mined in this fine commentary to be discerned by running through one's own heremenutical filter. I have found this to be fine resource for Galatians work and recommend it as one that thorough Biblical student can turn to as one resource in their library.\",\n",
       "  \"Mr. Bernstein's insight into the amazon.yyy amazon.yyy amazon.yyy Cold War era and the amazon.yyy amazon.yyy amazon.yyy role that organized crime had in shaping American society is both fresh and thorough. This book is a must read for any student of post-1945 America - an era of unprecedented change. Mr. Bernstein is to be commended for his fine work.\",\n",
       "  'I was intrigued by the amazon.yyy amazon.yyy amazon.yyy funky design of what otherwise is a mundane product, and on the amazon.yyy amazon.yyy amazon.yyy whole it did not disappoint. Be cautioned, however: this is a full size blow dryer, which I did not expect for some reason. But it is very light, and it dries my fine hair quickly with fewer \"frizzies\" than my old (now broken) one. I do not know if this is because it is an \"ionic\" dryer, as some have claimed. (I could find no documentation on the amazon.yyy amazon.yyy amazon.yyy package, etc., to confirm that.) I give it only 4 stars because if it is ionic, this should be better indicated by the amazon.yyy amazon.yyy amazon.yyy marketers, and also because I rarely hand out a perfect rating on anything. But on the amazon.yyy amazon.yyy amazon.yyy whole, I am satisfied with this product.',\n",
       "  \"This techno-thriller is the amazon.yyy amazon.yyy amazon.yyy sort of book that I can't get enough of - and I wish there were more of them to read. If you love Crichton, Rollins, Brown, and Clancy this book is for you. It is written very well, and it is very believable. Many of these books frustrate me because they don't show an adequate understanding of the amazon.yyy amazon.yyy amazon.yyy US government and its inner workings. This book does a great job in staying in the amazon.yyy amazon.yyy amazon.yyy realm of possible while pushing as far as possible to the amazon.yyy amazon.yyy amazon.yyy edge of belief.I found myself purposely slowing down reading the amazon.yyy amazon.yyy amazon.yyy book, because I didn't want to be done with it. I hope that his next book is just as good - and I hope it comes out soon.... I'm ready.\"]}"
      ]
     },
     "execution_count": 3,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "ds2_sample['train'][:10]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "{'text': [\"The yelp.xxx yelp.xxx yelp.xxx always busy Bad Daddy's is a solid choice in the yelp.xxx yelp.xxx yelp.xxx Ballentyne area. Good burgers including one of the yelp.xxx yelp.xxx yelp.xxx best black bean burgers in town. The yelp.xxx yelp.xxx yelp.xxx fried pickles are good... Not as good as Alexander Michaels but worth a try. I prefer the yelp.xxx yelp.xxx yelp.xxx sweet potato fries to the yelp.xxx yelp.xxx yelp.xxx regular ones. Also the yelp.xxx yelp.xxx yelp.xxx buffalo burger is an interesting choice. Worth a try...\",\n",
       "  \"Turkey burger was as dry as the yelp.xxx yelp.xxx yelp.xxx desert I was eating it in. It sucked. And the yelp.xxx yelp.xxx yelp.xxx fries it came with were room temp and chewy. Chewy!? I didn't really expect much from the yelp.xxx yelp.xxx yelp.xxx looks of the yelp.xxx yelp.xxx yelp.xxx place to begin with but honestly, my meal was pretty awful. My friends' quesedilla and chicken were actually good (though overpriced, like everything else on the yelp.xxx yelp.xxx yelp.xxx menu). \\\\n\\\\nBottom line: Skip it.\",\n",
       "  'I was extremely disappointed with the yelp.xxx yelp.xxx yelp.xxx breakfast here as well as with their pastries. I had ordered the yelp.xxx yelp.xxx yelp.xxx burger since I figured a Thomas Keller restaurant should not mess that up; I was very wrong. The yelp.xxx yelp.xxx yelp.xxx brioche bun did not seem fresh, burger patty was dry and flavorless, and the yelp.xxx yelp.xxx yelp.xxx so called \\\\\"\"animal sauce\\\\\"\" was hardly in it. I highly doubt that they used Wagyu beef as the yelp.xxx yelp.xxx yelp.xxx menu claims, even the yelp.xxx yelp.xxx yelp.xxx meat at In-N-Out tastes better! I also didn\\'t care for the yelp.xxx yelp.xxx yelp.xxx chocolate almond croissant or the yelp.xxx yelp.xxx yelp.xxx cheese danish, have had better elsewhere. My husband also didn\\'t like his French toast, it lacked flavor and seemed too mushy.\\\\n\\\\nEven the yelp.xxx yelp.xxx yelp.xxx service was lacking; the yelp.xxx yelp.xxx yelp.xxx waitress never informed us of the yelp.xxx yelp.xxx yelp.xxx specials and I only happened to glance at the yelp.xxx yelp.xxx yelp.xxx chalk board in front of me after I placed my order. The yelp.xxx yelp.xxx yelp.xxx food also took a long time to arrive. The yelp.xxx yelp.xxx yelp.xxx coffee we were served was cold. All in all, I would not recommend this place since it\\'s all hype and no substance.',\n",
       "  \"The yelp.xxx yelp.xxx yelp.xxx only reason I am giving this place two stars is because the yelp.xxx yelp.xxx yelp.xxx door guy came to the yelp.xxx yelp.xxx yelp.xxx back of the yelp.xxx yelp.xxx yelp.xxx line and allowed my group to breeze past to the yelp.xxx yelp.xxx yelp.xxx front and pay half price. \\\\n\\\\nThey played the yelp.xxx yelp.xxx yelp.xxx same fifteen songs over and over, the yelp.xxx yelp.xxx yelp.xxx chick workers that were up on the yelp.xxx yelp.xxx yelp.xxx stage screamed and sang into the yelp.xxx yelp.xxx yelp.xxx microphone and it was horrific, there was no way anyone was getting a drink in the yelp.xxx yelp.xxx yelp.xxx bar and sweat was literally pouring down my face and I was barely moving.\\\\n\\\\nI'm kind of embarrassed to admit I even went here.\",\n",
       "  'The yelp.xxx yelp.xxx yelp.xxx morning/early-afternoon bartender is one of the yelp.xxx yelp.xxx yelp.xxx most unpleasant, awful bartender/servers I\\'ve ever dealt with. Nasty attitude, and absolutely could care less about your experience. Waited 30 minutes for my food, which was good, and another 15 minutes of which i was STANDING UP at my table before I had to go into the yelp.xxx yelp.xxx yelp.xxx smoky section of the yelp.xxx yelp.xxx yelp.xxx bar to ask for my bill. \\\\\"\"I have to go get it- so gimme a minute.\\\\\"\", was her response. The yelp.xxx yelp.xxx yelp.xxx food is great but the yelp.xxx yelp.xxx yelp.xxx treatment and service is inexcusable. The yelp.xxx yelp.xxx yelp.xxx night-time bartenders are awesome- they deserve to have a better morning staff. Owner(s)- you\\'re losing business and costing your great employees money, one apple is ruining it for the yelp.xxx yelp.xxx yelp.xxx bunch. Step up and do something about it.',\n",
       "  \"The yelp.xxx yelp.xxx yelp.xxx service here was terrible. How do you screw up an egg and cheese breakfast sandwich? I knew that I was in trouble when the yelp.xxx yelp.xxx yelp.xxx guy asked me if I wanted the yelp.xxx yelp.xxx yelp.xxx bagel toasted. Yes, and please cook the yelp.xxx yelp.xxx yelp.xxx egg and melt the yelp.xxx yelp.xxx yelp.xxx cheese.\\\\n\\\\nI award you no stars and may God have mercy on your soul.(Yelp won't let me give zero stars.)\",\n",
       "  'Great food and the yelp.xxx yelp.xxx yelp.xxx servings are big',\n",
       "  'The yelp.xxx yelp.xxx yelp.xxx buffet at Wicket Spoon was by far the yelp.xxx yelp.xxx yelp.xxx best of the yelp.xxx yelp.xxx yelp.xxx buffets in Las Vegas. They had BONE MARROW! Something I have been craving ever since we were there. Their desserts were amazing as well. They also had Croque Madames which is one of my favorite sandwiches. And they had fried rice in little rice containers! I was seriously in food heaven!',\n",
       "  'I would like to give this place the yelp.xxx yelp.xxx yelp.xxx title: \\\\\"\"Best Burger Joint in the yelp.xxx yelp.xxx yelp.xxx West\\\\\"\".. comparing this to California burgers like Umami, Father\\'s Office, and Back Abbey\\\\nI\\'ve been here several times now and have yet to lose the yelp.xxx yelp.xxx yelp.xxx awe from eating their delicious burgers and fries.\\\\n\\\\nEach burger has their own distinctive flavor but the yelp.xxx yelp.xxx yelp.xxx best, most amazing burger i\\'ve ever eaten is their KIKI BURGER(add a fried egg). I remember the yelp.xxx yelp.xxx yelp.xxx first time biting into the yelp.xxx yelp.xxx yelp.xxx burger at the yelp.xxx yelp.xxx yelp.xxx other Bachi location.. I had one of those euphoric moments of satisfaction where all I was thinking was, \\\\\"\"WOW\\\\\"\"! The yelp.xxx yelp.xxx yelp.xxx ingredients compliment each other perfectly, with the yelp.xxx yelp.xxx yelp.xxx shitake mushrooms, caramalized bacon and the yelp.xxx yelp.xxx yelp.xxx garlic chili aioli supporting the yelp.xxx yelp.xxx yelp.xxx perfectly cooked patty.. amazing!\\\\n\\\\nOther burgers I\\'ve tried:\\\\nRonin- the yelp.xxx yelp.xxx yelp.xxx sauce in this one is great.\\\\nBanh-Mi- a detailed mixture of Vietnamese flavors put on a burger. Looks simple but has a burst of flavor.\\\\nShogun($26)- so many flavors I\\'ve never tasted before generated in this one burger. Pretty amazing with the yelp.xxx yelp.xxx yelp.xxx Wagyu Beef and Unagi. \\\\n\\\\nFor your first time, I suggest ordering the yelp.xxx yelp.xxx yelp.xxx Oxtail Chili Cheese Fries with your burger. It is truly one-of-a-kind. Sweet and meaty.. no need to dip in any sauce. If it seems too much, the yelp.xxx yelp.xxx yelp.xxx Truffle Parmesan Fries is a tasty alternative.\\\\n\\\\nIt seems like Bachi\\'s business and popularity is starting to blossom. Aside from the yelp.xxx yelp.xxx yelp.xxx amazing food, it\\'s a pretty modern and hip restaurant with cool employees. This is a perfect dinner date spot if you want to get away from the yelp.xxx yelp.xxx yelp.xxx strip.\\\\nThey really need to open up the yelp.xxx yelp.xxx yelp.xxx LA location already!',\n",
       "  'I always love watching the yelp.xxx yelp.xxx yelp.xxx fountain show at Bellagio. Very romantic, can be cheesy, but good. \\\\n\\\\nThe only reasons they miss a star is because I think the yelp.xxx yelp.xxx yelp.xxx songs and choreography need updating, but keep the yelp.xxx yelp.xxx yelp.xxx classics. And I thought the yelp.xxx yelp.xxx yelp.xxx fountains in Barcelona, which are along a government center and move to music up and down a street, with a large centerpiece fountain are more beautiful because you can get really close to the yelp.xxx yelp.xxx yelp.xxx jets, get a little wet, and it was designed in 1929 and is still awe-striking.'],\n",
       " 'label': [1, 0, 0, 0, 0, 0, 1, 1, 1, 1]}"
      ]
     },
     "execution_count": 4,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "ds1_sample['train'][:10]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "def create_realworld_dataset(dataset1, dataset2, random_seed, train_prob=0.90, test_probs=[0.90, 0.7, 0.5, 0.3, 0.1]):\n",
    "    num_cls = len(set(dataset1['test']['label']))\n",
    "    def merge_data(split1, split2, ratio):\n",
    "        merged_data = []\n",
    "        for label in range(num_cls):\n",
    "            ds1_label = copy.copy(split1).filter(lambda x: x['label'] == label)\n",
    "            ds2_label = copy.copy(split2).filter(lambda x: x['label'] == label)\n",
    "\n",
    "            # print(f\"Label {label}:\")\n",
    "            # print(f\"Dataset 1 available samples: {len(ds1_label)}\")\n",
    "            # print(f\"Dataset 2 available samples: {len(ds2_label)}\")\n",
    "        \n",
    "            \n",
    "            total_samples = len(ds1_label)\n",
    "            if label == 0:\n",
    "                ds1_samples = int(total_samples * ratio)\n",
    "                ds2_samples = total_samples - ds1_samples\n",
    "            else:\n",
    "                ds2_samples = int(total_samples * ratio)\n",
    "                ds1_samples = total_samples - ds2_samples\n",
    "\n",
    "            # print(f\"Dataset 1 selected samples: {ds1_samples}\")\n",
    "            # print(f\"Dataset 2 selected samples: {ds2_samples}\")\n",
    "            \n",
    "            merged_data.append(ds1_label.select(range(ds1_samples)))\n",
    "            merged_data.append(ds2_label.select(range(ds2_samples)))\n",
    "            \n",
    "            # print(f\"Total samples for this label: {min(ds1_samples, len(ds1_label)) + min(ds2_samples, len(ds2_label))}\")\n",
    "            # print(\"---\")\n",
    "        \n",
    "        \n",
    "        result = concatenate_datasets(merged_data)\n",
    "        # print(f\"Total samples in merged dataset: {len(result)}\")\n",
    "        return result\n",
    "\n",
    "    # Process train and validation sets\n",
    "    train_dataset = merge_data(dataset1['train'], dataset2['train'], train_prob).shuffle(seed=random_seed)\n",
    "    val_dataset = merge_data(dataset1['validation'], dataset2['validation'], train_prob).shuffle(seed=random_seed)\n",
    "\n",
    "    # Process test sets\n",
    "    test_datasets = []\n",
    "    for prob in test_probs:\n",
    "        test_datasets.append(merge_data(dataset1['test'], dataset2['test'], prob).shuffle(seed=random_seed))\n",
    "\n",
    "    # Return a new DatasetDict with all processed datasets\n",
    "    return DatasetDict({\n",
    "        'train': train_dataset,\n",
    "        'validation': val_dataset,\n",
    "        'test_in_distribution': test_datasets[0],\n",
    "        'test_change_70': test_datasets[1],\n",
    "        'test_balanced_50': test_datasets[2],\n",
    "        'test_change_30': test_datasets[3],\n",
    "        'test_flip': test_datasets[4]\n",
    "    })"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "DatasetDict({\n",
       "    train: Dataset({\n",
       "        features: ['text', 'label'],\n",
       "        num_rows: 8000\n",
       "    })\n",
       "    validation: Dataset({\n",
       "        features: ['text', 'label'],\n",
       "        num_rows: 2000\n",
       "    })\n",
       "    test_in_distribution: Dataset({\n",
       "        features: ['text', 'label'],\n",
       "        num_rows: 4000\n",
       "    })\n",
       "    test_change_70: Dataset({\n",
       "        features: ['text', 'label'],\n",
       "        num_rows: 4000\n",
       "    })\n",
       "    test_balanced_50: Dataset({\n",
       "        features: ['text', 'label'],\n",
       "        num_rows: 4000\n",
       "    })\n",
       "    test_change_30: Dataset({\n",
       "        features: ['text', 'label'],\n",
       "        num_rows: 4000\n",
       "    })\n",
       "    test_flip: Dataset({\n",
       "        features: ['text', 'label'],\n",
       "        num_rows: 4000\n",
       "    })\n",
       "})"
      ]
     },
     "execution_count": 6,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "\n",
    "\n",
    "ds = create_realworld_dataset(ds1_sample, ds2_sample, data_seed)\n",
    "ds\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "{'text': ['this is by far the amazon.yyy amazon.yyy amazon.yyy best perfume a woman could ever buy and wear. every time i put this on i get great responses from both men and women you wont regret this purchase',\n",
       "  \"I called in to place an order, the yelp.xxx yelp.xxx yelp.xxx female on the yelp.xxx yelp.xxx yelp.xxx other end of the yelp.xxx yelp.xxx yelp.xxx line asked if I wanted take out or delivery. I asked how much more take out was, she then asked me for my phone number, address, and gate code, which I said I didn't know, then she said that shouldn't be a problem because the yelp.xxx yelp.xxx yelp.xxx drivers know the yelp.xxx yelp.xxx yelp.xxx code. I asked her how long it would be and she said about 30 minutes. \\\\n\\\\nA little over an hour later, I start wondering where my order is, so I call back and talked to a female again, who I think took the yelp.xxx yelp.xxx yelp.xxx original order. She told me that my order had been sitting there for a half hour and that I definitely didn't ask for delivery, but was supposed to pick it up. I asked if they could deliver it now, she flippantly said that it would take another hour or I could come in to pick it up. Not wanting to wait another hour, I went to pick up the yelp.xxx yelp.xxx yelp.xxx food. \\\\n\\\\nI did speak with a manager when I was there and he comped half of the yelp.xxx yelp.xxx yelp.xxx meal, which was a decently nice gesture that seemed to fall a little short. I also saw him chastise the yelp.xxx yelp.xxx yelp.xxx flip employee who spoke with me on the yelp.xxx yelp.xxx yelp.xxx phone. Another employee shared overhearing the yelp.xxx yelp.xxx yelp.xxx female on the yelp.xxx yelp.xxx yelp.xxx phone and felt that the yelp.xxx yelp.xxx yelp.xxx female was rude. It just felt like the yelp.xxx yelp.xxx yelp.xxx one staff member needed to be whipped into shaped by the yelp.xxx yelp.xxx yelp.xxx Restaurant Impossible guy. I could feel the yelp.xxx yelp.xxx yelp.xxx attitude that she was projecting at her manager and it was awkward and uncomfortable. \\\\n\\\\nThe service was terrible. It just seems odd that you would argue with a customer over whether their order was for pick up for delivery, especially if the yelp.xxx yelp.xxx yelp.xxx restaurant doesn't get paid unless I pick up the yelp.xxx yelp.xxx yelp.xxx food or they deliver it. \\\\n\\\\nThe food was okay. The yelp.xxx yelp.xxx yelp.xxx pizza was chilly by the yelp.xxx yelp.xxx yelp.xxx time it made its way home. Some of the yelp.xxx yelp.xxx yelp.xxx greens caprese salad looked a little suspect. All in all not the yelp.xxx yelp.xxx yelp.xxx best experience, quite deserving of a 1 star rating.\\\\n\\\\nI did give it two stars because the yelp.xxx yelp.xxx yelp.xxx garlic balls were very good. I may go back for them, but only if the yelp.xxx yelp.xxx yelp.xxx person who took my order doesn't pick up the yelp.xxx yelp.xxx yelp.xxx phone.\",\n",
       "  \"We wanted to stop somewhere close to the yelp.xxx yelp.xxx yelp.xxx freeway to eat as we were about to leave Las Vegas on our way home to Los Angeles, We decided to head to the yelp.xxx yelp.xxx yelp.xxx Palms assuming that they would have a good buffet to eat at. We discovered the yelp.xxx yelp.xxx yelp.xxx Bistro Buffet, and decided to give it a try. \\\\n\\\\nThe restaurant has a very nice look to it. It is modern in appearance, and has an upscale cafeteria feel about it. It is one of the yelp.xxx yelp.xxx yelp.xxx smallest buffets that I have been to in Las Vegas, but getting seated was not an issue. \\\\n\\\\nThe food was not very good. Most of it was cold, and tasted as though it had been sitting around for hours. Condiments were very hard to find, and the yelp.xxx yelp.xxx yelp.xxx whole setup in general seemed disorganized. The yelp.xxx yelp.xxx yelp.xxx one highlight from the yelp.xxx yelp.xxx yelp.xxx food selection was the yelp.xxx yelp.xxx yelp.xxx crab legs. They were fairly big and pre-cut to make them easier to peel open and eat. \\\\n\\\\nThe other highlight of eating here was that the yelp.xxx yelp.xxx yelp.xxx service was excellent. Our waiter kept our cups filled and cleared our dishes whenever we finished a plate. He was very accommodating and was able to help us with whatever we needed. The yelp.xxx yelp.xxx yelp.xxx service made up a little bit for the yelp.xxx yelp.xxx yelp.xxx food being sub par. \\\\n\\\\nOverall, the yelp.xxx yelp.xxx yelp.xxx Bistro Buffet was a disappointment. The yelp.xxx yelp.xxx yelp.xxx food just wasn't good. With so many other options in Vegas, I can't see any reason for me to eat here again.\",\n",
       "  \"Forty Deuce is LAME (take note of the yelp.xxx yelp.xxx yelp.xxx caps!!!)\\\\n\\\\nFirst of all, it's SMALL. Smaller than small... I don't even know why they consider themselves a club at this point. It just be called: Ivan Kane's Forty Deuce Burlesque Walkway. \\\\n\\\\nMusic was decent, crowd was NOT. Though I hate LA for being so damn rude to guys... I wish this place wasn't so nice to them. I think the yelp.xxx yelp.xxx yelp.xxx ratio was 8 guys to 1 girl. LAME! \\\\n\\\\nAdditionally, the yelp.xxx yelp.xxx yelp.xxx burlesque dancer was sexy (body and moves) but was a complete butter face! And I don't find it cute when random chicks hop on stage and think they are suddenly burlesque dancers.... Gosh, they are just so brave.\",\n",
       "  \"Man I really wanted the yelp.xxx yelp.xxx yelp.xxx Anthony Bourdain experience with a pedi and a cocktail but apparently it seems like they don't do that anymore. They don't even have cocktails according to the yelp.xxx yelp.xxx yelp.xxx bartender you just need to tell him what to mix I guess. The yelp.xxx yelp.xxx yelp.xxx prices are pretty bad too for the yelp.xxx yelp.xxx yelp.xxx drinks since I think the yelp.xxx yelp.xxx yelp.xxx bartender just made up the yelp.xxx yelp.xxx yelp.xxx prices on a slow Sunday night. A 12 oz can of Modelo and a Amaretto sour which was almost all sour for $10 hmmm..too bad well there are a lot of bars in Vegas and for some reason I just can't find a good one downtown.\",\n",
       "  'The yelp.xxx yelp.xxx yelp.xxx menu is very enticing, but I ordered the yelp.xxx yelp.xxx yelp.xxx bagel and lox and was seriously disappointed...the onions were at least a day old...dried out and chewy...the bagel seemed a tad stale, and maybe it\\'s just me, but I\\'ve never had the yelp.xxx yelp.xxx yelp.xxx bagel cut into 4 pieces....let alone when you are trying to put large onion slices, salmon and capers on it. My wife had the yelp.xxx yelp.xxx yelp.xxx Eggs Benedict California and said it was \\\\\"\"just okay\\\\\"\". This was the yelp.xxx yelp.xxx yelp.xxx first place we tried for breakfast in Montreal and will not return again. Atmosphere was nothing special, something like a fancier diner...kitchen right in the yelp.xxx yelp.xxx yelp.xxx main first floor area was a bit noisy...',\n",
       "  'What I liked this book is the amazon.yyy amazon.yyy amazon.yyy quality paper, graphic colors, high quality print, and color rendition of the amazon.yyy amazon.yyy amazon.yyy retro movie posters. With intimate interviews, it also reveals the amazon.yyy amazon.yyy amazon.yyy behind-the-scenes touches here and there that made this genre so different from hollywood. Included are interviews to set designers and stylists, with illustrations of sketches of sets, costumes. I was always touched by the amazon.yyy amazon.yyy amazon.yyy \"grunge\" feel of the amazon.yyy amazon.yyy amazon.yyy clothes worn by the amazon.yyy amazon.yyy amazon.yyy actors, a torn shirt with holes and loose buttons, dirty boots. As one poster says: \"he smokes short cigars, he has a long gun, he wears a poncho\", and the amazon.yyy amazon.yyy amazon.yyy interviews reveal how the amazon.yyy amazon.yyy amazon.yyy \"leone style\" was created. Unfortunately there is no english translation of \"L\\'avventurosa storia del cinema italiano raccontata dai suoi protagonisti\" (Vol 2, 1981, Feltrinelli, F. Fofi), but this book beautifully selects the amazon.yyy amazon.yyy amazon.yyy best of that time in a high-quality printed production.',\n",
       "  'I loved this movie...it had the amazon.yyy amazon.yyy amazon.yyy zombie killing, awesome humor and Woody Harrelson was fantastic in this role!! Was so happy to see a zombie movie with an actual budget, great actors, a great plot and great camera work.',\n",
       "  'People, the amazon.yyy amazon.yyy amazon.yyy Stereophonics rock. For the amazon.yyy amazon.yyy amazon.yyy first time in a long time I heard a song on the amazon.yyy amazon.yyy amazon.yyy radio that was sooooo good,(Mabye Tomorrow) I went online and ordered the amazon.yyy amazon.yyy amazon.yyy cd. I like almost every song on this album. Wish these guys would come to the amazon.yyy amazon.yyy amazon.yyy States.',\n",
       "  \"These are quality recordings from the amazon.yyy amazon.yyy amazon.yyy 1960's of various Jewish liturgical songs arranged around four themes within the amazon.yyy amazon.yyy amazon.yyy Jewish year: the amazon.yyy amazon.yyy amazon.yyy High Festivals, the amazon.yyy amazon.yyy amazon.yyy three main festivals, the amazon.yyy amazon.yyy amazon.yyy Sabbath and Hanukah. The amazon.yyy amazon.yyy amazon.yyy cantor is Hans Bloemendal supported by the amazon.yyy amazon.yyy amazon.yyy singers of the amazon.yyy amazon.yyy amazon.yyy Amsterdam Synagogue. Excellent voices throughout.\"],\n",
       " 'label': [1, 0, 0, 0, 0, 0, 1, 1, 1, 1]}"
      ]
     },
     "execution_count": 7,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "ds['train'][: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": 8,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "{'text': [\"As apart of the yelp.xxx yelp.xxx yelp.xxx lightgroup, I had pretty high expectations for this restaurant and as such, I left pretty disappointed in my meal. So much, that I can't really recommend going back there to any of my friends. There is nothing that really differentiates it from all of the yelp.xxx yelp.xxx yelp.xxx other restaurants on the yelp.xxx yelp.xxx yelp.xxx strip. After each of my meals, I left thinking, ehh, this is it. Where is the yelp.xxx yelp.xxx yelp.xxx flavor and the yelp.xxx yelp.xxx yelp.xxx quality of food? We didn't order any of the yelp.xxx yelp.xxx yelp.xxx steak (so take it with a grain of salt if thats what you are looking for).\\\\n\\\\nWhat we ordered:\\\\nBLT salad\\\\nSesame chicken lettuce cups\\\\nSeabass\\\\nLamb shank\\\\nTater tots\\\\n\\\\n\\\\nPros:\\\\nI was attracted to this restaurant in what appeared to be an innovative menu. The yelp.xxx yelp.xxx yelp.xxx creativity that is shown really excited me about the yelp.xxx yelp.xxx yelp.xxx potential food I was going to get.\\\\n\\\\nCons:\\\\nThe food isn't bad. It is food, and I did eat it. However, its not anything to write home. I really wanted it to taste better. I had spent well over 3 hours researching restaurants, locations, and menus and I wanted this to work. I guess I had too high of expectations for this to live up to. Unfortunately, I won't be coming back and I won't be recommending it to anyone else that visits. It left me wanting more for what I paid and in a place like Vegas, there are so many other things I could be doing or enjoying that this isn't going to cut it.\",\n",
       "  'I did lots of research to get the amazon.yyy amazon.yyy amazon.yyy perfect mp3/cd player. I think this player fit my criteria and gives the amazon.yyy amazon.yyy amazon.yyy most value for money among any other cd players. Compared to my panasonic which i used nearly 4 yrs. It seems so much bettter... long battery life...good sound quality..only thing i dont like is that it takes a long time to read my mp3 cds which i didnt burn with sonic stage.Also when forwarding and rewinding it makes a loud tinny annoying sound besides making it annoying.. a remote would have been much nicer..BUT OVERALL A BIG THUMBS UP..SO FAR SO GOOD.......vikas',\n",
       "  \"The yelp.xxx yelp.xxx yelp.xxx Skinny: Undercooked sausage, soggy chicken pot pie, and slimy tough zucchini sticks makes this place a no-go (1), service: efficient and friendly (3), value: although the yelp.xxx yelp.xxx yelp.xxx food here is reasonably priced, there is no value to subpar cuisine (1)\\\\n\\\\nI really wanted to love this place. I was deeply intrigued by the yelp.xxx yelp.xxx yelp.xxx numerous pies on display, the yelp.xxx yelp.xxx yelp.xxx waiters making freshly squeezed OJ by hand, and the yelp.xxx yelp.xxx yelp.xxx heaping serving sizes enough to feed a football team. However, the yelp.xxx yelp.xxx yelp.xxx food we ordered just doesn't cut the yelp.xxx yelp.xxx yelp.xxx mustard.\\\\n\\\\nSo I ordered the yelp.xxx yelp.xxx yelp.xxx Las Vegas breakfast special (2 hotcakes, 2 turkey sausage links, 1 scrambled egg) while the yelp.xxx yelp.xxx yelp.xxx hubs ordered zucchini sticks and chicken pot pie. \\\\n\\\\nBoth of my husband's items could categorically be the yelp.xxx yelp.xxx yelp.xxx worst chicken pot pie and zucchini sticks we have ever eaten.\\\\n\\\\nI have never had zucchini sticks that were both mushy and tough at the yelp.xxx yelp.xxx yelp.xxx same time. I am certain the yelp.xxx yelp.xxx yelp.xxx zucchini they used for this item were made from very old zucchini, hence the yelp.xxx yelp.xxx yelp.xxx need to strip the yelp.xxx yelp.xxx yelp.xxx vegetable of its skin. Regardless, it still did not take away the yelp.xxx yelp.xxx yelp.xxx fibrousness. Overcooking also just worsened the yelp.xxx yelp.xxx yelp.xxx texture by making it slimy. The yelp.xxx yelp.xxx yelp.xxx cornmeal crust was overly heavy, and salty to boot.\\\\n\\\\nThe chicken pot pie made Swanson's frozen version look like ambrosia. An undercooked soggy crust graced over a gelatinous gloop that contained minimal chicken or vegetables. They have obviously attempted to stretch the yelp.xxx yelp.xxx yelp.xxx filling by adding cornstarch to the yelp.xxx yelp.xxx yelp.xxx mix and it shows.\\\\n\\\\nMy breakfast plate was not great but at least more palatable than my husband's selections. I cut into my turkey sausage links to see a pink center. Call me finicky but turkey anything should not be pink in the yelp.xxx yelp.xxx yelp.xxx middle. I did not touch those so I can't comment on the yelp.xxx yelp.xxx yelp.xxx taste. The yelp.xxx yelp.xxx yelp.xxx eggs were okay but were not fluffy by any means (I'm pretty sure they didn't mix any milk into them). \\\\n\\\\nHotcakes were okay but not great by any means. They certainly were fluffy however, they did not have a nice mildly crisp to them at all. They must add a significant amount of leavening agent in them because it left an odd flavour to them. Served with standard syrup with no maple in them, I have made better pancakes out of a box and they are certainly nothing I would return for.\\\\n\\\\nDespite the yelp.xxx yelp.xxx yelp.xxx horrible eats, the yelp.xxx yelp.xxx yelp.xxx service was friendly, quick and efficient. Food was served in a timely matter so it clearly is an issue with the yelp.xxx yelp.xxx yelp.xxx kitchen.\\\\n\\\\nOverall, I really wanted to love this place but I couldn't overlook the yelp.xxx yelp.xxx yelp.xxx horrible food we were served. For that, this place deserves a 1 starred review.\",\n",
       "  'Never heard of the amazon.yyy amazon.yyy amazon.yyy Wondermints until browsing Amazon suggested I might like this (browsing America, Gerry Beckley, Jeff Larson, Timothy Schmit, Beckley-Lamm-Wilson). This is probably the amazon.yyy amazon.yyy amazon.yyy strongest CD I have heard in the amazon.yyy amazon.yyy amazon.yyy last 10-20 years. Every song could be a top 40 hit, but current radio oligopoly will never play them. Alas, this will have to be the amazon.yyy amazon.yyy amazon.yyy best indie record out there that never makes it to radio. Brian Wilson (his backing band = Wondermints) knows gold when he hears it, but the amazon.yyy amazon.yyy amazon.yyy industry execs just don\\'t \"get it\" anymore. Too bad...',\n",
       "  \". . . it still shows the amazon.yyy amazon.yyy amazon.yyy keen intellect behind her writing. The amazon.yyy amazon.yyy amazon.yyy character development seen in Atlas Shrugged and Fountainhead is there as well. Yet the amazon.yyy amazon.yyy amazon.yyy book doesn't quite move as well as her later novels. Nevertheless, ten times better than the amazon.yyy amazon.yyy amazon.yyy typical novel. Ayn Rand makes a searing indictment of communism, a system she grew up with and knew only too well.\",\n",
       "  \"I have the amazon.yyy amazon.yyy amazon.yyy 1988 CBS Special Products release of this album (non-enhanced, non-imported, part of a 3-pack of music from Costco), and it is one of my favorite holiday CDs. At 41:49, it's a little short, and there is nothing original here. However, it has a nice collection of styles and sounds that really bring out the amazon.yyy amazon.yyy amazon.yyy Christmas spirit in me. If you're looking for something original, this is not it, but if you're looking for some traditional Christmas carols sung by the amazon.yyy amazon.yyy amazon.yyy people who made them famous, this is a great album.\",\n",
       "  'Great work by the amazon.yyy amazon.yyy amazon.yyy author. This book is in my top 20 of all time. Get this book if you desire to be prosperous and wealthy. If not by the amazon.yyy amazon.yyy amazon.yyy newspaper.',\n",
       "  \"This place was so bad, from the yelp.xxx yelp.xxx yelp.xxx time we walked in, threw our breakfast, till the yelp.xxx yelp.xxx yelp.xxx time we left.\\\\n\\\\nWe walked in, busy and I expected that, didn't take our name down or how many in our party, forgot who was first and going where....it was so freakin hot.\\\\n\\\\nWaiter came over, he was really sweet and funny, but our food came out and it was a train wreck! My friend ordered crispy bacon and it came out soggy, my husbands egg Benedict came out and the yelp.xxx yelp.xxx yelp.xxx biscuits were soggy so we asked for new ones and those were soggy too, his breakfast was so bad he didn't even want to finish it. I got the yelp.xxx yelp.xxx yelp.xxx country fried steak breakfast and I enjoyed it up until my husband decided he wanted to eat the yelp.xxx yelp.xxx yelp.xxx eggs, I don't like eggs, he was so excited and hungry to eat them, he got them all dressed up and ready to eat and then when he went to go take a bite there is a whole corner of a plastic cut off in the yelp.xxx yelp.xxx yelp.xxx eggs.\\\\n\\\\nWe ask for a manager and they came over and said that they had no idea where the yelp.xxx yelp.xxx yelp.xxx plastic came from because all their stuff is organic and not packaged, which is bull crap, because when I walked in I could see the yelp.xxx yelp.xxx yelp.xxx shelves with the yelp.xxx yelp.xxx yelp.xxx cans in the yelp.xxx yelp.xxx yelp.xxx back in the yelp.xxx yelp.xxx yelp.xxx kitchen. And I find it disturbing that a manager would say that they have no idea where it came from instead of making us as guests feel comfortable.\\\\n\\\\nIt was a bad experience, it was disgusting, we will never go back there.\",\n",
       "  \"Need to add 1-2 $100 table on the yelp.xxx yelp.xxx yelp.xxx regular. Poker room and the yelp.xxx yelp.xxx yelp.xxx casino itself is small compared to Wild Horse Pass Casino. But, it'll do. Food selection is great.\",\n",
       "  'One star for the yelp.xxx yelp.xxx yelp.xxx awesome outdoor seating and one star for the yelp.xxx yelp.xxx yelp.xxx Guitar Hero nights. I stopped going here last week because \\\\n*I finally had to accept that when you pay with a debit card, the yelp.xxx yelp.xxx yelp.xxx waitress just gives herself whatever tip she wants, regardless of what you actually wrote on the yelp.xxx yelp.xxx yelp.xxx receipt.\\\\n* The yelp.xxx yelp.xxx yelp.xxx beer tastes watered-down and the yelp.xxx yelp.xxx yelp.xxx food is ridiculously bland and over-priced.'],\n",
       " 'label': [0, 1, 0, 1, 1, 1, 1, 0, 0, 0]}"
      ]
     },
     "execution_count": 8,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "ds['validation'][:10]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "{'text': ['I have used Lauren several over the yelp.xxx yelp.xxx yelp.xxx past couple of years. She is a true professional and a genuine pleasure to work with. She listens to what you want and does a beautiful job to accommodate your needs. I highly recommend her, as she is a personal favorite MUA of mine!',\n",
       "  '!Hola meego! That\\'s the yelp.xxx yelp.xxx yelp.xxx enthusiastically friendly greeting everybody gets at this fun little Mexican restaurant. The yelp.xxx yelp.xxx yelp.xxx decor matches the yelp.xxx yelp.xxx yelp.xxx enthusiasm of the yelp.xxx yelp.xxx yelp.xxx staff well: bright colors (even for a Mexican restaurant) are everywhere and everything intangible about this place is bouncing off the yelp.xxx yelp.xxx yelp.xxx walls cheery. I guess its not a coincidence its called \\\\\"\"The Party\\\\\"\".\\\\n\\\\nThey have a pretty extensive menu of all the yelp.xxx yelp.xxx yelp.xxx expected classic dishes, plus many that you might not expect. I ordered the yelp.xxx yelp.xxx yelp.xxx \\\\\"\"Ceviche Mexicano\\\\\"\" with chopped (cooked) Tilapia and shrimp in a very lively light lime sauce over tostadas with lettuce, tomatoes, avocados, onions and other good stuff like beans, rice cheese, etc. I liked it a lot. The yelp.xxx yelp.xxx yelp.xxx lady had very good Burritos Suizos, which were far too big to finish. Prices were $10 or less per entree. Good fried tortilla chips with salsa were gratis.\\\\n\\\\nThe highlight of the yelp.xxx yelp.xxx yelp.xxx meal were the yelp.xxx yelp.xxx yelp.xxx (Friday night special) 27 ounce Margaritas for $5.95 each. Dont bother with the yelp.xxx yelp.xxx yelp.xxx math. This is by far the yelp.xxx yelp.xxx yelp.xxx best Margarita deal I have ever come across in this town or any other en los Estados Unidos. No, I didn\\'t drive home.\\\\n\\\\nI can\\'t review this place without one more mention of the yelp.xxx yelp.xxx yelp.xxx fact that everybody working here is just so damn nice. As our meal was winding down, no fewer than 5 staff members asked us how everything was. Any server/busser coming within 3 tables of us checked our table to see if we needed anything, every single time.\\\\n\\\\nLa Fiesta is not fine dining. However, it is a superlative casual dining Mexican restaurant and delivers incredible value and I can\\'t wait to go back.',\n",
       "  \"The yelp.xxx yelp.xxx yelp.xxx first time I ate at Fahrenheit was shortly after they opened. Our meals were pretty good, but the yelp.xxx yelp.xxx yelp.xxx food came out so fast-way before we had finished our appetizers. One of my biggest pet peeves at a restaurant is being rushed through my meal, especially when it's expensive. I really enjoyed the yelp.xxx yelp.xxx yelp.xxx chorizo stuffed dates and my tuna entr\\\\u00e9e was delicious but humongous. When I came back a second time with a couple who had never been, my expectations were fairly low. Let's just say my expectations were met. I ordered the yelp.xxx yelp.xxx yelp.xxx tuna entr\\\\u00e9e again and it was significantly different-almost inedible. It was so coated in spice that I had to scrape it off just to eat part of it. It had no flavor other than HOT and I hated wasting such an expensive dish, but I had no choice. The yelp.xxx yelp.xxx yelp.xxx drinks were weak. My friend had to order an extra shot of tequila for her margarita on the yelp.xxx yelp.xxx yelp.xxx rocks. No thank you. When we got the yelp.xxx yelp.xxx yelp.xxx bill, we found out that our husbands' Jameson on the yelp.xxx yelp.xxx yelp.xxx rocks were $21 per glass. At most other restaurants, Jameson on the yelp.xxx yelp.xxx yelp.xxx rocks is about $12. A huge waste of money. I have no idea how this restaurant has good reviews. It's overrated and serves mediocre food at best.\",\n",
       "  \"I was anxious to read the amazon.yyy amazon.yyy amazon.yyy latest book by Paullina Simons, the amazon.yyy amazon.yyy amazon.yyy author of Tully, and it didn't disappoint!Finding out who murdered Kristina Kim becomes secondary to learning about the amazon.yyy amazon.yyy amazon.yyy details of her mysterious life.Kristina, like Tully, is a strong young woman who makes no apologies for her unorthodox life.Red Leaves is also very similar to Donna Tartt's The amazon.yyy amazon.yyy amazon.yyy Secret Language so I think anyone who enjoyed it or Tully will love this book as well.\",\n",
       "  \"The amazon.yyy amazon.yyy amazon.yyy Brain Pad LoPro Mouth Guard is a easy to fit and comfortable once sized correctly. We bought 3 of these since my teens and I have taken up TaeKwonDo and do contact sparring.The fitting process is a little tricky since we had to cut two of them down to fit in my kid's mouths, but it was well documented in the amazon.yyy amazon.yyy amazon.yyy instructions. Next came boiling the amazon.yyy amazon.yyy amazon.yyy guards and fitting the amazon.yyy amazon.yyy amazon.yyy in our mouths while they were still warm. This was good for my son and I, but my daughter has very sensitive teeth and this caused her some discomfort.Once in, these allow a lot of air flow which is great when sparring!Recommended!CFH\",\n",
       "  \"After eating at the yelp.xxx yelp.xxx yelp.xxx Kogi truck in LA, I was stoked when I saw this truck set up in Lee's parking lot. How incredibly disappointing it was. This will set back Korean food for generations. And don't try to eat there during the yelp.xxx yelp.xxx yelp.xxx summer. There is no shade, no misters, nothing. Just horrible food in a horrible spot.\",\n",
       "  'not worth the yelp.xxx yelp.xxx yelp.xxx 2 hour 10 minute wait and the yelp.xxx yelp.xxx yelp.xxx $43 + tax for weekend dinner\\\\n\\\\ndined with au, jq, ln, mn, and nn\\\\n\\\\nthere really wasn\\'t that much food in each section to choose from:\\\\n- the yelp.xxx yelp.xxx yelp.xxx chinese section had a few dim sum items that are more mainstream: shrimp dumpling, siu mai, pork bun, shanghai dumpling, glutinous rice wrapped in lotus leaf. then there were the yelp.xxx yelp.xxx yelp.xxx items that you would see at panda express. and there were a few that you would see from a chinese deli like roast duck. i only tried the yelp.xxx yelp.xxx yelp.xxx roast duck in this section and it was leaner than the yelp.xxx yelp.xxx yelp.xxx ones in sf. it tasted no different than the yelp.xxx yelp.xxx yelp.xxx ones i normally get, but the yelp.xxx yelp.xxx yelp.xxx skin was a bit crispier because it didn\\'t steam in a container to transport it from ctown to home. \\\\n- in the yelp.xxx yelp.xxx yelp.xxx japanese section, there were 4 types of bite size sushi: california, california with a piece of cooked shrimp, tuna, and california with tuna on top. it tasted okay, but nothing to brag about. the yelp.xxx yelp.xxx yelp.xxx short ribs were chewy and my teeth couldn\\'t tear it apart.\\\\n- the yelp.xxx yelp.xxx yelp.xxx italian section was just mainly pizza and a few other hot items. we only tried the yelp.xxx yelp.xxx yelp.xxx fried calamari which borrowed the yelp.xxx yelp.xxx yelp.xxx idea of serving it in a a mini frying basket from another buffet. it was well seasoned and cooked to the yelp.xxx yelp.xxx yelp.xxx right consistency (not rubbery).\\\\n- around the yelp.xxx yelp.xxx yelp.xxx corner is the yelp.xxx yelp.xxx yelp.xxx mexican section with various salsa and make your own taco type thing. they have a huge circular grill that you would see at a mongolian noodle place, but most of it was not used. the yelp.xxx yelp.xxx yelp.xxx chicken tamale had lard so i only ate half of one. the yelp.xxx yelp.xxx yelp.xxx tiny sliver of chicken was can barely be seen, but the yelp.xxx yelp.xxx yelp.xxx masa was not grainy and had a smooth even texture. \\\\n- the yelp.xxx yelp.xxx yelp.xxx breads don\\'t visually look very good. i saw some slices of baguette and took one to try and it was dried to the yelp.xxx yelp.xxx yelp.xxx core. i can\\'t imagine what the yelp.xxx yelp.xxx yelp.xxx other breads were like if they messed up on a simple baguette. \\\\n- southern foods was the yelp.xxx yelp.xxx yelp.xxx next stop. the yelp.xxx yelp.xxx yelp.xxx sweet potato tots didn\\'t have the yelp.xxx yelp.xxx yelp.xxx excess sugar added like the yelp.xxx yelp.xxx yelp.xxx sweet potato fries. the yelp.xxx yelp.xxx yelp.xxx oil for the yelp.xxx yelp.xxx yelp.xxx fried chicken was stale and that flavor passed onto the yelp.xxx yelp.xxx yelp.xxx chicken. after ripping off the yelp.xxx yelp.xxx yelp.xxx batter, the yelp.xxx yelp.xxx yelp.xxx drumstick beneath was moist, but it\\'s hard to dry out dark meat. the yelp.xxx yelp.xxx yelp.xxx fries had a hard, brittle, dense starchy texture. the yelp.xxx yelp.xxx yelp.xxx waffle fries were too salty.\\\\n- carved meats station turned out to be a fat station. a slice of brisket 6x4\\\\\"\" had 0.5x2\\\\\"\" of meat and the yelp.xxx yelp.xxx yelp.xxx rest was fat. and the yelp.xxx yelp.xxx yelp.xxx small piece of meat was marbled with fat. then there was some other meat that was dry. \\\\n- the yelp.xxx yelp.xxx yelp.xxx line for seafood was long. the yelp.xxx yelp.xxx yelp.xxx crab legs were too salty. the yelp.xxx yelp.xxx yelp.xxx salmon was okay. the yelp.xxx yelp.xxx yelp.xxx shrimps were big and not rubbery. the yelp.xxx yelp.xxx yelp.xxx mussels in the yelp.xxx yelp.xxx yelp.xxx buttery broth were my friends\\' favorite in that entire section. \\\\n- from the yelp.xxx yelp.xxx yelp.xxx dessert section, i tried the yelp.xxx yelp.xxx yelp.xxx creme brulee which was very rich and the yelp.xxx yelp.xxx yelp.xxx sugar wasn\\'t crispy, so it\\'s been sitting out. the yelp.xxx yelp.xxx yelp.xxx color of the yelp.xxx yelp.xxx yelp.xxx french macaron didn\\'t match the yelp.xxx yelp.xxx yelp.xxx color. i had a gray one which i thought would be earl grey but it turned out to be blueberry, but the yelp.xxx yelp.xxx yelp.xxx almond extract masked the yelp.xxx yelp.xxx yelp.xxx blueberry pretty well. the yelp.xxx yelp.xxx yelp.xxx crepes were made by a newbie. i had one just cooked and the yelp.xxx yelp.xxx yelp.xxx other was practically burnt. jq took a bite of of each dessert and didn\\'t like most of it. \\\\n\\\\nour server was a nice elder gentleman who stocked our table with water as well as other beverages. when asked if there was hot chocolate he said he\\'ll check to see if they have that and suggested to have that towards the yelp.xxx yelp.xxx yelp.xxx end of the yelp.xxx yelp.xxx yelp.xxx meal because that will make you full. but i don\\'t think they have that.',\n",
       "  'After eating here the yelp.xxx yelp.xxx yelp.xxx first time in Oct, I was really looking forward to the yelp.xxx yelp.xxx yelp.xxx meal. \\\\n\\\\nI came for lunch and the yelp.xxx yelp.xxx yelp.xxx waiter talked me into trying their buffet lunch. It had salad, Tom Yum Cabbage Soup, egg soup, egg rolls, tofu and veg, green curry chicken, white and fried rice, tempura veg, pad thai, pad see ew, fried banana, and fruit. All the yelp.xxx yelp.xxx yelp.xxx delicious thai flavors were dumbed down and muted. \\\\n\\\\n2nd chance. Failed! Went back for dinner to have the yelp.xxx yelp.xxx yelp.xxx Garlic Shrimp that blew my mind last time. The yelp.xxx yelp.xxx yelp.xxx first time shells were crunchy like shrimp chips, meat was lightly fried, moist, and delicious. This time dried out and tough, it was so bad I sent it back. The yelp.xxx yelp.xxx yelp.xxx waiter brought it back out and said it was supposed to be that way. (Gross!). On a good note the yelp.xxx yelp.xxx yelp.xxx steak tartare was very tasty and delicious!',\n",
       "  \"I'm confused by all the yelp.xxx yelp.xxx yelp.xxx high ratings. I've had plenty of good Mexican while in and around Phoenix but this was definitely on the yelp.xxx yelp.xxx yelp.xxx low end. I did find out after from some locals this is more the yelp.xxx yelp.xxx yelp.xxx end of night drunk food stop. Eh well.\",\n",
       "  'for all the amazon.yyy amazon.yyy amazon.yyy sylvian fans who reviewed: did anyone notice that on \"a history of holes\" that david sylvian wrote one of his best lyrics? the amazon.yyy amazon.yyy amazon.yyy beats and the amazon.yyy amazon.yyy amazon.yyy rhythm drop out temporarily and david sings: when i was a boy/and i made mistakes/i was humiliated/till i knew my place. snow borne sorrow is a very good album. it may require one to listen to it a few times and to remember that david has an adventurous spirit. there is a feeling of completeness when listening to a recording artist who shares with you what you want AND need.'],\n",
       " 'label': [1, 1, 0, 1, 1, 0, 0, 0, 0, 1]}"
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "ds['test_in_distribution'][:10]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "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(ds['train']['label'][:10])\n",
    "print(ds['train']['label'][-10:])\n",
    "\n",
    "print(ds['validation']['label'][:10])\n",
    "print(ds['validation']['label'][-10:])\n",
    "\n",
    "print(ds['test_in_distribution']['label'][:10])\n",
    "print(ds['test_in_distribution']['label'][-10:])\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "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": 12,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/user/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": "6907d5255e8246f896d4c5bc64c56501",
       "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 = ds['train'].map(tokenize_function, batched=True)\n",
    "val_dataset = ds['validation'].map(tokenize_function, batched=True)\n",
    "test_dataset_ID = ds['test_in_distribution'].map(tokenize_function, batched=True)\n",
    "test_dataset_OOD_change_70 = ds['test_change_70'].map(tokenize_function, batched=True)\n",
    "test_dataset_OOD_balanced_50 = ds['test_balanced_50'].map(tokenize_function, batched=True)\n",
    "test_dataset_OOD_change_30 = ds['test_change_30'].map(tokenize_function, batched=True)\n",
    "test_dataset_OOD_flip = ds['test_flip'].map(tokenize_function, batched=True)\n",
    "# test_dataset_OOD_original = synthetic_yelp_ds['test_original'].map(tokenize_function, batched=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "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 = PairedDataset(test_dataset_ID, 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 = PairedDataset(test_dataset_OOD_flip, train_dataset)\n",
    "# paired_test_dataset_OOD_original = PairedDataset(test_dataset_OOD_original)\n",
    "\n",
    "paired_data_collator = PairedDataCollator(tokenizer=tokenizer)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "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": [
    "# SWA"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "\n",
       "    <div>\n",
       "      \n",
       "      <progress value='1250' max='1250' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [1250/1250 01:58, 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>125</td>\n",
       "      <td>0.315800</td>\n",
       "      <td>0.272189</td>\n",
       "      <td>0.903499</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>250</td>\n",
       "      <td>0.227900</td>\n",
       "      <td>0.228588</td>\n",
       "      <td>0.918986</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>375</td>\n",
       "      <td>0.156400</td>\n",
       "      <td>0.269994</td>\n",
       "      <td>0.910925</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>500</td>\n",
       "      <td>0.101000</td>\n",
       "      <td>0.276430</td>\n",
       "      <td>0.917499</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>625</td>\n",
       "      <td>0.056500</td>\n",
       "      <td>0.465417</td>\n",
       "      <td>0.883902</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>750</td>\n",
       "      <td>0.041500</td>\n",
       "      <td>0.451610</td>\n",
       "      <td>0.910987</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>875</td>\n",
       "      <td>0.023400</td>\n",
       "      <td>0.547256</td>\n",
       "      <td>0.889849</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>1000</td>\n",
       "      <td>0.022200</td>\n",
       "      <td>0.508675</td>\n",
       "      <td>0.908498</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>1125</td>\n",
       "      <td>0.008700</td>\n",
       "      <td>0.600477</td>\n",
       "      <td>0.907853</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>1250</td>\n",
       "      <td>0.009500</td>\n",
       "      <td>0.626822</td>\n",
       "      <td>0.909964</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='64' max='32' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [32/32 00:12]\n",
       "    </div>\n",
       "    "
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 1/10: {'eval_loss': 0.6268220543861389, 'eval_f1': 0.9099639855942379, 'eval_runtime': 1.5074, 'eval_samples_per_second': 1326.76, 'eval_steps_per_second': 21.228, 'epoch': 10.0}\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "\n",
       "    <div>\n",
       "      \n",
       "      <progress value='1250' max='1250' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [1250/1250 01:58, 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>125</td>\n",
       "      <td>0.009300</td>\n",
       "      <td>0.665019</td>\n",
       "      <td>0.907499</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>250</td>\n",
       "      <td>0.006500</td>\n",
       "      <td>0.658489</td>\n",
       "      <td>0.910983</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>375</td>\n",
       "      <td>0.007800</td>\n",
       "      <td>0.741774</td>\n",
       "      <td>0.904422</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>500</td>\n",
       "      <td>0.003300</td>\n",
       "      <td>0.739036</td>\n",
       "      <td>0.910996</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>625</td>\n",
       "      <td>0.008200</td>\n",
       "      <td>0.689408</td>\n",
       "      <td>0.906970</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>750</td>\n",
       "      <td>0.008500</td>\n",
       "      <td>0.621014</td>\n",
       "      <td>0.909488</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>875</td>\n",
       "      <td>0.003500</td>\n",
       "      <td>0.685791</td>\n",
       "      <td>0.910000</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>1000</td>\n",
       "      <td>0.008800</td>\n",
       "      <td>0.664901</td>\n",
       "      <td>0.909998</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>1125</td>\n",
       "      <td>0.007300</td>\n",
       "      <td>0.674973</td>\n",
       "      <td>0.911490</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>1250</td>\n",
       "      <td>0.009300</td>\n",
       "      <td>0.666072</td>\n",
       "      <td>0.910980</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='64' max='32' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [32/32 00:12]\n",
       "    </div>\n",
       "    "
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 2/10: {'eval_loss': 0.6660721302032471, 'eval_f1': 0.910979970493361, 'eval_runtime': 1.4929, 'eval_samples_per_second': 1339.645, 'eval_steps_per_second': 21.434, 'epoch': 10.0}\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "\n",
       "    <div>\n",
       "      \n",
       "      <progress value='1250' max='1250' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [1250/1250 01:59, 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>125</td>\n",
       "      <td>0.004600</td>\n",
       "      <td>0.671098</td>\n",
       "      <td>0.915482</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>250</td>\n",
       "      <td>0.005500</td>\n",
       "      <td>0.634961</td>\n",
       "      <td>0.908991</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>375</td>\n",
       "      <td>0.002400</td>\n",
       "      <td>0.687172</td>\n",
       "      <td>0.914496</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>500</td>\n",
       "      <td>0.002200</td>\n",
       "      <td>0.744821</td>\n",
       "      <td>0.913991</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>625</td>\n",
       "      <td>0.009000</td>\n",
       "      <td>0.614606</td>\n",
       "      <td>0.905984</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>750</td>\n",
       "      <td>0.006300</td>\n",
       "      <td>0.680709</td>\n",
       "      <td>0.916988</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>875</td>\n",
       "      <td>0.005400</td>\n",
       "      <td>0.693362</td>\n",
       "      <td>0.910000</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>1000</td>\n",
       "      <td>0.015700</td>\n",
       "      <td>0.541574</td>\n",
       "      <td>0.886922</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>1125</td>\n",
       "      <td>0.012900</td>\n",
       "      <td>0.748587</td>\n",
       "      <td>0.902377</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>1250</td>\n",
       "      <td>0.008400</td>\n",
       "      <td>0.634778</td>\n",
       "      <td>0.912965</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='64' max='32' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [32/32 00:12]\n",
       "    </div>\n",
       "    "
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 3/10: {'eval_loss': 0.6347784996032715, 'eval_f1': 0.9129651860744299, 'eval_runtime': 1.4869, 'eval_samples_per_second': 1345.07, 'eval_steps_per_second': 21.521, 'epoch': 10.0}\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "\n",
       "    <div>\n",
       "      \n",
       "      <progress value='1250' max='1250' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [1250/1250 01:58, 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>125</td>\n",
       "      <td>0.009500</td>\n",
       "      <td>0.660719</td>\n",
       "      <td>0.902384</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>250</td>\n",
       "      <td>0.005800</td>\n",
       "      <td>0.631802</td>\n",
       "      <td>0.914497</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>375</td>\n",
       "      <td>0.003400</td>\n",
       "      <td>0.742892</td>\n",
       "      <td>0.912999</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>500</td>\n",
       "      <td>0.002100</td>\n",
       "      <td>0.747174</td>\n",
       "      <td>0.912490</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>625</td>\n",
       "      <td>0.000300</td>\n",
       "      <td>0.827275</td>\n",
       "      <td>0.908500</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>750</td>\n",
       "      <td>0.004700</td>\n",
       "      <td>0.816084</td>\n",
       "      <td>0.902957</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>875</td>\n",
       "      <td>0.007700</td>\n",
       "      <td>0.582165</td>\n",
       "      <td>0.913486</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>1000</td>\n",
       "      <td>0.005700</td>\n",
       "      <td>0.740486</td>\n",
       "      <td>0.912997</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>1125</td>\n",
       "      <td>0.009200</td>\n",
       "      <td>0.674937</td>\n",
       "      <td>0.901975</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>1250</td>\n",
       "      <td>0.001800</td>\n",
       "      <td>0.771614</td>\n",
       "      <td>0.910000</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='64' max='32' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [32/32 00:12]\n",
       "    </div>\n",
       "    "
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 4/10: {'eval_loss': 0.7716143131256104, 'eval_f1': 0.9099999099999101, 'eval_runtime': 1.4846, 'eval_samples_per_second': 1347.164, 'eval_steps_per_second': 21.555, 'epoch': 10.0}\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "\n",
       "    <div>\n",
       "      \n",
       "      <progress value='1250' max='1250' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [1250/1250 02:00, 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>125</td>\n",
       "      <td>0.001500</td>\n",
       "      <td>0.798345</td>\n",
       "      <td>0.910446</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>250</td>\n",
       "      <td>0.004200</td>\n",
       "      <td>0.760154</td>\n",
       "      <td>0.912991</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>375</td>\n",
       "      <td>0.003400</td>\n",
       "      <td>0.924730</td>\n",
       "      <td>0.901494</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>500</td>\n",
       "      <td>0.003600</td>\n",
       "      <td>0.951732</td>\n",
       "      <td>0.885516</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>625</td>\n",
       "      <td>0.006500</td>\n",
       "      <td>0.722368</td>\n",
       "      <td>0.910492</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>750</td>\n",
       "      <td>0.004800</td>\n",
       "      <td>0.790456</td>\n",
       "      <td>0.897371</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>875</td>\n",
       "      <td>0.007400</td>\n",
       "      <td>0.768194</td>\n",
       "      <td>0.904909</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>1000</td>\n",
       "      <td>0.005600</td>\n",
       "      <td>0.700126</td>\n",
       "      <td>0.913958</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>1125</td>\n",
       "      <td>0.005700</td>\n",
       "      <td>0.721475</td>\n",
       "      <td>0.910469</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>1250</td>\n",
       "      <td>0.010900</td>\n",
       "      <td>0.598634</td>\n",
       "      <td>0.908938</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='64' max='32' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [32/32 00:12]\n",
       "    </div>\n",
       "    "
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 5/10: {'eval_loss': 0.5986337661743164, 'eval_f1': 0.9089384423870537, 'eval_runtime': 1.5082, 'eval_samples_per_second': 1326.09, 'eval_steps_per_second': 21.217, 'epoch': 10.0}\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "\n",
       "    <div>\n",
       "      \n",
       "      <progress value='1250' max='1250' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [1250/1250 01:59, 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>125</td>\n",
       "      <td>0.002100</td>\n",
       "      <td>0.774252</td>\n",
       "      <td>0.910953</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>250</td>\n",
       "      <td>0.001400</td>\n",
       "      <td>0.778683</td>\n",
       "      <td>0.914404</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>375</td>\n",
       "      <td>0.002900</td>\n",
       "      <td>0.821405</td>\n",
       "      <td>0.910980</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>500</td>\n",
       "      <td>0.003900</td>\n",
       "      <td>0.840481</td>\n",
       "      <td>0.913983</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>625</td>\n",
       "      <td>0.004700</td>\n",
       "      <td>0.732161</td>\n",
       "      <td>0.905966</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>750</td>\n",
       "      <td>0.002100</td>\n",
       "      <td>0.868681</td>\n",
       "      <td>0.900430</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>875</td>\n",
       "      <td>0.002400</td>\n",
       "      <td>0.876287</td>\n",
       "      <td>0.900938</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>1000</td>\n",
       "      <td>0.003900</td>\n",
       "      <td>0.895272</td>\n",
       "      <td>0.903908</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>1125</td>\n",
       "      <td>0.003900</td>\n",
       "      <td>0.848182</td>\n",
       "      <td>0.907917</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>1250</td>\n",
       "      <td>0.004000</td>\n",
       "      <td>0.717735</td>\n",
       "      <td>0.910473</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='64' max='32' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [32/32 00:12]\n",
       "    </div>\n",
       "    "
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 6/10: {'eval_loss': 0.7177345156669617, 'eval_f1': 0.9104725822283075, 'eval_runtime': 1.4967, 'eval_samples_per_second': 1336.303, 'eval_steps_per_second': 21.381, 'epoch': 10.0}\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "\n",
       "    <div>\n",
       "      \n",
       "      <progress value='1250' max='1250' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [1250/1250 01:58, 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>125</td>\n",
       "      <td>0.001500</td>\n",
       "      <td>0.764031</td>\n",
       "      <td>0.913486</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>250</td>\n",
       "      <td>0.000900</td>\n",
       "      <td>0.763415</td>\n",
       "      <td>0.917999</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>375</td>\n",
       "      <td>0.001900</td>\n",
       "      <td>0.812542</td>\n",
       "      <td>0.911945</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>500</td>\n",
       "      <td>0.005200</td>\n",
       "      <td>0.858296</td>\n",
       "      <td>0.901783</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>625</td>\n",
       "      <td>0.004200</td>\n",
       "      <td>0.713667</td>\n",
       "      <td>0.912996</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>750</td>\n",
       "      <td>0.006400</td>\n",
       "      <td>0.755380</td>\n",
       "      <td>0.909896</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>875</td>\n",
       "      <td>0.001600</td>\n",
       "      <td>0.853345</td>\n",
       "      <td>0.906401</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>1000</td>\n",
       "      <td>0.002100</td>\n",
       "      <td>0.983907</td>\n",
       "      <td>0.905277</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>1125</td>\n",
       "      <td>0.007400</td>\n",
       "      <td>0.643517</td>\n",
       "      <td>0.908498</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>1250</td>\n",
       "      <td>0.004000</td>\n",
       "      <td>0.667339</td>\n",
       "      <td>0.919493</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='64' max='32' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [32/32 00:12]\n",
       "    </div>\n",
       "    "
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 7/10: {'eval_loss': 0.6673391461372375, 'eval_f1': 0.9194927342192634, 'eval_runtime': 1.4992, 'eval_samples_per_second': 1334.078, 'eval_steps_per_second': 21.345, 'epoch': 10.0}\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "\n",
       "    <div>\n",
       "      \n",
       "      <progress value='1250' max='1250' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [1250/1250 01:59, 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>125</td>\n",
       "      <td>0.004500</td>\n",
       "      <td>0.701501</td>\n",
       "      <td>0.914482</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>250</td>\n",
       "      <td>0.000300</td>\n",
       "      <td>0.829735</td>\n",
       "      <td>0.909967</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>375</td>\n",
       "      <td>0.005600</td>\n",
       "      <td>0.779667</td>\n",
       "      <td>0.910985</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>500</td>\n",
       "      <td>0.006100</td>\n",
       "      <td>0.798958</td>\n",
       "      <td>0.904373</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>625</td>\n",
       "      <td>0.001300</td>\n",
       "      <td>0.883383</td>\n",
       "      <td>0.906963</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>750</td>\n",
       "      <td>0.005000</td>\n",
       "      <td>0.768074</td>\n",
       "      <td>0.911492</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>875</td>\n",
       "      <td>0.009200</td>\n",
       "      <td>0.700890</td>\n",
       "      <td>0.912978</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>1000</td>\n",
       "      <td>0.001900</td>\n",
       "      <td>0.758429</td>\n",
       "      <td>0.909493</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>1125</td>\n",
       "      <td>0.002800</td>\n",
       "      <td>0.817465</td>\n",
       "      <td>0.912492</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>1250</td>\n",
       "      <td>0.002600</td>\n",
       "      <td>0.793044</td>\n",
       "      <td>0.913484</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='64' max='32' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [32/32 00:12]\n",
       "    </div>\n",
       "    "
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 8/10: {'eval_loss': 0.7930439710617065, 'eval_f1': 0.9134842325013734, 'eval_runtime': 1.4939, 'eval_samples_per_second': 1338.774, 'eval_steps_per_second': 21.42, 'epoch': 10.0}\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "\n",
       "    <div>\n",
       "      \n",
       "      <progress value='1250' max='1250' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [1250/1250 01:58, 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>125</td>\n",
       "      <td>0.001500</td>\n",
       "      <td>0.823883</td>\n",
       "      <td>0.913993</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>250</td>\n",
       "      <td>0.001200</td>\n",
       "      <td>0.804374</td>\n",
       "      <td>0.911499</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>375</td>\n",
       "      <td>0.000800</td>\n",
       "      <td>0.830855</td>\n",
       "      <td>0.907498</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>500</td>\n",
       "      <td>0.001300</td>\n",
       "      <td>0.845630</td>\n",
       "      <td>0.914928</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>625</td>\n",
       "      <td>0.000700</td>\n",
       "      <td>0.826357</td>\n",
       "      <td>0.911980</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>750</td>\n",
       "      <td>0.003600</td>\n",
       "      <td>0.879489</td>\n",
       "      <td>0.912000</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>875</td>\n",
       "      <td>0.002800</td>\n",
       "      <td>0.777061</td>\n",
       "      <td>0.914499</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>1000</td>\n",
       "      <td>0.002100</td>\n",
       "      <td>0.914697</td>\n",
       "      <td>0.908498</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>1125</td>\n",
       "      <td>0.004300</td>\n",
       "      <td>0.764075</td>\n",
       "      <td>0.906461</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>1250</td>\n",
       "      <td>0.009500</td>\n",
       "      <td>0.749914</td>\n",
       "      <td>0.912911</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='64' max='32' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [32/32 00:12]\n",
       "    </div>\n",
       "    "
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 9/10: {'eval_loss': 0.7499138116836548, 'eval_f1': 0.9129108206803767, 'eval_runtime': 1.4827, 'eval_samples_per_second': 1348.848, 'eval_steps_per_second': 21.582, 'epoch': 10.0}\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "\n",
       "    <div>\n",
       "      \n",
       "      <progress value='1250' max='1250' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [1250/1250 01:59, 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>125</td>\n",
       "      <td>0.006300</td>\n",
       "      <td>0.758788</td>\n",
       "      <td>0.911961</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>250</td>\n",
       "      <td>0.000000</td>\n",
       "      <td>0.845704</td>\n",
       "      <td>0.907987</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>375</td>\n",
       "      <td>0.001300</td>\n",
       "      <td>0.850030</td>\n",
       "      <td>0.914494</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>500</td>\n",
       "      <td>0.004800</td>\n",
       "      <td>0.751523</td>\n",
       "      <td>0.912496</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>625</td>\n",
       "      <td>0.000300</td>\n",
       "      <td>0.815773</td>\n",
       "      <td>0.914494</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>750</td>\n",
       "      <td>0.000000</td>\n",
       "      <td>0.814727</td>\n",
       "      <td>0.920476</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>875</td>\n",
       "      <td>0.000000</td>\n",
       "      <td>0.838698</td>\n",
       "      <td>0.916988</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>1000</td>\n",
       "      <td>0.000100</td>\n",
       "      <td>0.853584</td>\n",
       "      <td>0.919977</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>1125</td>\n",
       "      <td>0.004600</td>\n",
       "      <td>0.819241</td>\n",
       "      <td>0.910488</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>1250</td>\n",
       "      <td>0.002500</td>\n",
       "      <td>0.837216</td>\n",
       "      <td>0.915487</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='504' max='32' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [32/32 00:20]\n",
       "    </div>\n",
       "    "
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 10/10: {'eval_loss': 0.8372163772583008, 'eval_f1': 0.9154867948116894, 'eval_runtime': 1.4847, 'eval_samples_per_second': 1347.059, 'eval_steps_per_second': 21.553, 'epoch': 10.0}\n",
      "train: {'eval_loss': 7.974172149260994e-06, 'eval_f1': 1.0, 'eval_runtime': 4.7692, 'eval_samples_per_second': 1677.426, 'eval_steps_per_second': 26.21, 'epoch': 10.0}\n",
      "validation: {'eval_loss': 0.6939574480056763, 'eval_f1': 0.9199795147557774, 'eval_runtime': 1.5009, 'eval_samples_per_second': 1332.494, 'eval_steps_per_second': 21.32, 'epoch': 10.0}\n",
      "ID: {'eval_loss': 0.7197469472885132, 'eval_f1': 0.9139944956477215, 'eval_runtime': 2.574, 'eval_samples_per_second': 1553.988, 'eval_steps_per_second': 24.475, 'epoch': 10.0}\n",
      "OOD change 70: {'eval_loss': 1.8215802907943726, 'eval_f1': 0.8023861744364754, 'eval_runtime': 2.52, 'eval_samples_per_second': 1587.303, 'eval_steps_per_second': 25.0, 'epoch': 10.0}\n",
      "OOD balanced: {'eval_loss': 2.925117015838623, 'eval_f1': 0.6887582185326702, 'eval_runtime': 2.5492, 'eval_samples_per_second': 1569.11, 'eval_steps_per_second': 24.713, 'epoch': 10.0}\n",
      "OOD change 30: {'eval_loss': 4.017626762390137, 'eval_f1': 0.5748724554700204, 'eval_runtime': 2.6147, 'eval_samples_per_second': 1529.788, 'eval_steps_per_second': 24.094, 'epoch': 10.0}\n",
      "OOD flip: {'eval_loss': 5.0939860343933105, 'eval_f1': 0.4608201413073328, 'eval_runtime': 2.5238, 'eval_samples_per_second': 1584.9, 'eval_steps_per_second': 24.962, 'epoch': 10.0}\n"
     ]
    },
    {
     "data": {
      "image/png": "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6p9vhqXON5Z9IgkAAOB1XOMEAGfLZpPaXSudd5m06mlzBr7zLjevXYrqTe8SAAA1DD1OAAAAAGqk0mQDbhIEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABgwevBaf78+WrRooUCAgIUHR2ttWvXFls/PT1d06ZNU1RUlPz9/XXeeefpzTffrKDWAgAAAKiJfLy586VLl2rixImaP3++evfurVdeeUWDBg3Szp071axZs0LXGT58uP766y+98cYbatWqlZKTk5WVlVXBLQcAAABQk9gMwzC8tfMePXqoW7duWrBggbusbdu2Gjp0qGJjYwvU/+qrr3TjjTfqjz/+UN26dcu0z9TUVIWFhSklJUWhoaFlbjsAAACAqq002cBrQ/UyMjK0ZcsWDRgwwKN8wIABWr9+faHrfPrpp4qJidHs2bPVuHFjnX/++XrooYd0+vTpIveTnp6u1NRUjwcAAAAAlIbXhuodPnxYTqdTjRo18ihv1KiRDh06VOg6f/zxh9atW6eAgAB9/PHHOnz4sO655x4dPXq0yOucYmNjNXPmzHJvPwAAAICaw+uTQ9hsNo/XhmEUKMvhcrlks9m0ePFide/eXVdddZWef/55LVy4sMhep6lTpyolJcX92LdvX7kfAwAAAIDqzWs9TvXr15fD4SjQu5ScnFygFypHRESEGjdurLCwMHdZ27ZtZRiG9u/fr9atWxdYx9/fX/7+/uXbeAAAAAA1itd6nPz8/BQdHa24uDiP8ri4OPXq1avQdXr37q2DBw/q5MmT7rLffvtNdrtdTZo0OaftBQAAAFBzeXWo3qRJk/T666/rzTff1K5du/TAAw8oMTFR48aNk2QOsxs9erS7/s0336x69erptttu086dO/Xdd9/p4Ycf1u23367AwEBvHQYAAACAas6r93EaMWKEjhw5olmzZikpKUkdOnTQ8uXLFRUVJUlKSkpSYmKiu35wcLDi4uI0YcIExcTEqF69eho+fLiefPJJbx0CAAAAgBrAq/dx8gbu4wQAAABAqiL3cQIAAACAqoLgBAAAAAAWCE4AAAAAYIHgBAAAAAAWCE4AAAAAYIHgBAAAAAAWCE4AAAAAYIHgBAAAAAAWCE4AAAAAYIHgBAAAAAAWCE4AAAAAYIHgBAAAAAAWCE4AAAAAYIHgBAAAAAAWCE4AAAAAYIHgBAAAAAAWCE4AAAAAYIHgBAAAAAAWCE4AAAAAYIHgBAAAAAAWCE4AAAAAYIHgBAAAAAAWCE4AAAAAYIHgBAAAAAAWCE4AAAAAYIHgBAAAAAAWCE4AAAAAYIHgBAAAAAAWCE4AAAAAYIHgBAAAAAAWCE4AAAAAYIHgBAAAAAAWCE4AAAAAYIHgBAAAAAAWCE4AAAAAYIHgBAAAAAAWCE4AAAAAYIHgBAAAAAAWCE4AAAAAYIHgBAAAAAAWCE4AAAAAYIHgBAAAAAAWCE4AAAAAYIHgBAAAAAAWCE4AAAAAYIHgBAAAAAAWCE4AAAAAYIHgBAAAAAAWCE4AAAAAYIHgBAAAAAAWCE4AAAAAYIHgBAAAAAAWCE4AAAAAYIHgBAAAAAAWCE4AAAAAYMHH2w0AAAAADMNQVlaWnE6nt5uCasbX11cOh+Ost0NwAgAAgFdlZGQoKSlJaWlp3m4KqiGbzaYmTZooODj4rLZDcAIAAIDXuFwuJSQkyOFwKDIyUn5+frLZbN5uFqoJwzD0999/a//+/WrduvVZ9TwRnAAAAOA1GRkZcrlcatq0qYKCgrzdHFRDDRo00J9//qnMzMyzCk5MDgEAAACvs9t5W4pzo7x6MPkJBQAAAAALBCcAAAAAsEBwAgAAACqBSy+9VBMnTixx/T///FM2m03x8fHnrE3IRXACAAAASsFmsxX7GDNmTJm2+9FHH+mJJ54ocf2mTZsqKSlJHTp0KNP+SoqAZmJWPQAAAKAUkpKS3M+XLl2qGTNm6Ndff3WXBQYGetTPzMyUr6+v5Xbr1q1bqnY4HA6Fh4eXah2UHT1OAAAAqFQMw1BaRlaFPwzDKFH7wsPD3Y+wsDDZbDb36zNnzqh27dp6//33demllyogIECLFi3SkSNHdNNNN6lJkyYKCgpSx44d9d5773lsN/9QvebNm+vpp5/W7bffrpCQEDVr1kyvvvqqe3n+nqDVq1fLZrPpm2++UUxMjIKCgtSrVy+PUCdJTz75pBo2bKiQkBDdcccdeuSRR9SlS5cynStJSk9P13333aeGDRsqICBAffr00aZNm9zLjx07ppEjR6pBgwYKDAxU69at9dZbb0kyp6MfP368IiIiFBAQoObNmys2NrbMbTmX6HECAABApXI606l2M76u8P3unDVQQX7l8/Z4ypQpmjNnjt566y35+/vrzJkzio6O1pQpUxQaGqovvvhCo0aNUsuWLdWjR48itzNnzhw98cQTevTRR/XBBx/o7rvv1iWXXKI2bdoUuc60adM0Z84cNWjQQOPGjdPtt9+u77//XpK0ePFiPfXUU5o/f7569+6tJUuWaM6cOWrRokWZj3Xy5Mn68MMP9fbbbysqKkqzZ8/WwIED9fvvv6tu3bqaPn26du7cqS+//FL169fX77//rtOnT0uSXnzxRX366ad6//331axZM+3bt0/79u0rc1vOpTL9ZOzbt082m01NmjSRJP3www9699131a5dO911113l2kAAAACgqpk4caKGDRvmUfbQQw+5n0+YMEFfffWVli1bVmxwuuqqq3TPPfdIMsPY3LlztXr16mKD01NPPaW+fftKkh555BENHjxYZ86cUUBAgF566SWNHTtWt912myRpxowZWrFihU6ePFmm4zx16pQWLFighQsXatCgQZKk1157TXFxcXrjjTf08MMPKzExUV27dlVMTIwksyctR2Jiolq3bq0+ffrIZrMpKiqqTO2oCGUKTjfffLPuuusujRo1SocOHVL//v3Vvn17LVq0SIcOHdKMGTPKu50AAACoIQJ9Hdo5a6BX9lteckJCDqfTqWeeeUZLly7VgQMHlJ6ervT0dNWqVavY7XTq1Mn9PGdIYHJyconXiYiIkCQlJyerWbNm+vXXX91BLEf37t317bfflui48tuzZ48yMzPVu3dvd5mvr6+6d++uXbt2SZLuvvtuXX/99dq6dasGDBigoUOHqlevXpKkMWPGqH///rrgggt05ZVX6uqrr9aAAQPK1JZzrUzXOP3000/q3r27JOn9999Xhw4dtH79er377rtauHBhebYPAAAANYzNZlOQn0+FP2w2W7kdQ/5ANGfOHM2dO1eTJ0/Wt99+q/j4eA0cOFAZGRnFbif/pBI2m00ul6vE6+QcU9518h9nSa/tKkzOuoVtM6ds0KBB2rt3ryZOnKiDBw/qiiuucPe+devWTQkJCXriiSd0+vRpDR8+XDfccEOZ23MulSk4ZWZmyt/fX5K0cuVKXXvttZKkNm3aeMwyAgAAAEBau3athgwZoltuuUWdO3dWy5YttXv37gpvxwUXXKAffvjBo2zz5s1l3l6rVq3k5+endevWucsyMzO1efNmtW3b1l3WoEEDjRkzRosWLdILL7zgMclFaGioRowYoddee01Lly7Vhx9+qKNHj5a5TedKmYbqtW/fXi+//LIGDx6suLg493zzBw8eVL169cq1gQAAAEBV16pVK3344Ydav3696tSpo+eff16HDh3yCBcVYcKECbrzzjsVExOjXr16aenSpfrxxx/VsmVLy3Xzz84nSe3atdPdd9+thx9+WHXr1lWzZs00e/ZspaWlaezYsZLM66iio6PVvn17paen6/PPP3cf99y5cxUREaEuXbrIbrdr2bJlCg8PV+3atcv1uMtDmYLTs88+q+uuu07PPfecbr31VnXu3FmS9Omnn7qH8AEAAAAwTZ8+XQkJCRo4cKCCgoJ01113aejQoUpJSanQdowcOVJ//PGHHnroIZ05c0bDhw/XmDFjCvRCFebGG28sUJaQkKBnnnlGLpdLo0aN0okTJxQTE6Ovv/5aderUkST5+flp6tSp+vPPPxUYGKiLL75YS5YskSQFBwfr2Wef1e7du+VwOHThhRdq+fLlstsr312TbEYZBzU6nU6lpqa6vyGSOZd8UFCQGjZsWG4NLG+pqakKCwtTSkqKQkNDvd0cAACAGu3MmTNKSEhQixYtFBAQ4O3m1Ej9+/dXeHi4/vvf/3q7KedEcT9jpckGZepxOn36tAzDcIemvXv36uOPP1bbtm01cGDFz4ACAAAAwFpaWppefvllDRw4UA6HQ++9955WrlypuLg4bzet0itTH9iQIUP0zjvvSJKOHz+uHj16aM6cORo6dKgWLFhQrg0EAAAAUD5sNpuWL1+uiy++WNHR0frss8/04Ycfql+/ft5uWqVXpuC0detWXXzxxZKkDz74QI0aNdLevXv1zjvv6MUXXyzXBgIAAAAoH4GBgVq5cqWOHj2qU6dOaevWrQVu1IvClSk4paWlKSQkRJK0YsUKDRs2THa7XRdddJH27t1brg0EAAAAAG8rU3Bq1aqVPvnkE+3bt09ff/21++6+ycnJTLgAAAAAoNopU3CaMWOGHnroITVv3lzdu3dXz549JZm9T127di3XBgIAAACAt5VpVr0bbrhBffr0UVJSkvseTpJ0xRVX6Lrrriu3xgEAAABAZVCm4CRJ4eHhCg8P1/79+2Wz2dS4cWNufgsAAACgWirTUD2Xy6VZs2YpLCxMUVFRatasmWrXrq0nnnhCLpervNsIAAAAVDuXXnqpJk6c6H7dvHlzvfDCC8WuY7PZ9Mknn5z1vstrOzVJmYLTtGnTNG/ePD3zzDPatm2btm7dqqefflovvfSSpk+fXt5tBAAAACqNa665psj7Hm3YsEE2m01bt24t9XY3bdqku+6662yb5+Hxxx9Xly5dCpQnJSVp0KBB5bqv/BYuXKjatWuf031UpDIN1Xv77bf1+uuv69prr3WXde7cWY0bN9Y999yjp556qtwaCAAAAFQmY8eO1bBhw7R3715FRUV5LHvzzTfVpUsXdevWrdTbbdCgQXk10VJ4eHiF7au6KFOP09GjR9WmTZsC5W3atNHRo0fPulEAAABAZXX11VerYcOGWrhwoUd5Wlqali5dqrFjx+rIkSO66aab1KRJEwUFBaljx4567733it1u/qF6u3fv1iWXXKKAgAC1a9dOcXFxBdaZMmWKzj//fAUFBally5aaPn26MjMzJZk9PjNnztT27dtls9lks9ncbc4/VG/Hjh26/PLLFRgYqHr16umuu+7SyZMn3cvHjBmjoUOH6t///rciIiJUr1493Xvvve59lUViYqKGDBmi4OBghYaGavjw4frrr7/cy7dv367LLrtMISEhCg0NVXR0tDZv3ixJ2rt3r6655hrVqVNHtWrVUvv27bV8+fIyt6UkytTj1LlzZ82bN08vvviiR/m8efPUqVOncmkYAAAAaijDkDLTKn6/vkGSzWZZzcfHR6NHj9bChQs1Y8YM2bLXWbZsmTIyMjRy5EilpaUpOjpaU6ZMUWhoqL744guNGjVKLVu2VI8ePSz34XK5NGzYMNWvX18bN25Uamqqx/VQOUJCQrRw4UJFRkZqx44duvPOOxUSEqLJkydrxIgR+umnn/TVV19p5cqVkqSwsLAC20hLS9OVV16piy66SJs2bVJycrLuuOMOjR8/3iMcrlq1ShEREVq1apV+//13jRgxQl26dNGdd95peTz5GYahoUOHqlatWlqzZo2ysrJ0zz33aMSIEVq9erUkaeTIkeratasWLFggh8Oh+Ph4+fr6SpLuvfdeZWRk6LvvvlOtWrW0c+dOBQcHl7odpVGm4DR79mwNHjxYK1euVM+ePWWz2bR+/Xrt27fvnCc9AAAAVHOZadLTkRW/30cPSn61SlT19ttv13PPPafVq1frsssuk2QO0xs2bJjq1KmjOnXq6KGHHnLXnzBhgr766istW7asRMFp5cqV2rVrl/788081adJEkvT0008XuC7pX//6l/t58+bN9eCDD2rp0qWaPHmyAgMDFRwcLB8fn2KH5i1evFinT5/WO++8o1q1zOOfN2+errnmGj377LNq1KiRJKlOnTqaN2+eHA6H2rRpo8GDB+ubb74pU3BauXKlfvzxRyUkJKhp06aSpP/+979q3769Nm3apAsvvFCJiYl6+OGH3SPdWrdu7V4/MTFR119/vTp27ChJatmyZanbUFplGqrXt29f/fbbb7ruuut0/PhxHT16VMOGDdPPP/+st956q1Tbmj9/vlq0aKGAgABFR0dr7dq1JVrv+++/l4+PT6EXuwEAAADnUps2bdSrVy+9+eabkqQ9e/Zo7dq1uv322yVJTqdTTz31lDp16qR69eopODhYK1asUGJiYom2v2vXLjVr1swdmiSpZ8+eBep98MEH6tOnj8LDwxUcHKzp06eXeB9599W5c2d3aJKk3r17y+Vy6ddff3WXtW/fXg6Hw/06IiJCycnJpdpX3n02bdrUHZokqV27dqpdu7Z27dolSZo0aZLuuOMO9evXT88884z27NnjrnvffffpySefVO/evfXYY4/pxx9/LFM7SqPM93GKjIwsMAnE9u3b9fbbb7t/gKwsXbpUEydO1Pz589W7d2+98sorGjRokHbu3KlmzZoVuV5KSopGjx6tK664wmMcJAAAAKoB3yCz98cb+y2FsWPHavz48frPf/6jt956S1FRUbriiiskSXPmzNHcuXP1wgsvqGPHjqpVq5YmTpyojIyMEm3bMIwCZbZ8wwg3btyoG2+8UTNnztTAgQMVFhamJUuWaM6cOaU6DsMwCmy7sH3mDJPLu6ystyIqap95yx9//HHdfPPN+uKLL/Tll1/qscce05IlS3Tdddfpjjvu0MCBA/XFF19oxYoVio2N1Zw5czRhwoQytackytTjVF6ef/55jR07VnfccYfatm2rF154QU2bNtWCBQuKXe+f//ynbr755kJTNwAAAKo4m80cMlfRjxJc35TX8OHD5XA49O677+rtt9/Wbbfd5n7Tv3btWg0ZMkS33HKLOnfurJYtW2r37t0l3na7du2UmJiogwdzA+SGDRs86nz//feKiorStGnTFBMTo9atW2vv3r0edfz8/OR0Oi33FR8fr1OnTnls22636/zzzy9xm0sj5/j27dvnLtu5c6dSUlLUtm1bd9n555+vBx54QCtWrNCwYcM8Rrc1bdpU48aN00cffaQHH3xQr7322jlpaw6vBaeMjAxt2bJFAwYM8CgfMGCA1q9fX+R6b731lvbs2aPHHnusRPtJT09XamqqxwMAAAA4W8HBwRoxYoQeffRRHTx4UGPGjHEva9WqleLi4rR+/Xrt2rVL//znP3Xo0KESb7tfv3664IILNHr0aG3fvl1r167VtGnTPOq0atVKiYmJWrJkifbs2aMXX3xRH3/8sUed5s2bKyEhQfHx8Tp8+LDS09ML7GvkyJEKCAjQrbfeqp9++kmrVq3ShAkTNGrUKPf1TWXldDoVHx/v8di5c6f69eunTp06aeTIkdq6dat++OEHjR49Wn379lVMTIxOnz6t8ePHa/Xq1dq7d6++//57bdq0yR2qJk6cqK+//loJCQnaunWrvv32W4/AdS54LTgdPnxYTqezwMlo1KhRkT9Uu3fv1iOPPKLFixfLx6dkowxjY2MVFhbmfuQdRwkAAACcjbFjx+rYsWPq16+fx6Um06dPV7du3TRw4EBdeumlCg8P19ChQ0u8Xbvdro8//ljp6enq3r277rjjjgKXyQwZMkQPPPCAxo8fry5dumj9+vWaPn26R53rr79eV155pS677DI1aNCg0CnRg4KC9PXXX+vo0aO68MILdcMNN+iKK67QvHnzSvfNKMTJkyfVtWtXj8dVV13lng69Tp06uuSSS9SvXz+1bNlSS5culSQ5HA4dOXJEo0eP1vnnn6/hw4dr0KBBmjlzpiQzkN17771q27atrrzySl1wwQWaP3/+Wbe3ODajsAGURRg2bFixy48fP641a9ZYdgdK0sGDB9W4cWOtX7/eY8jdU089pf/+97/65ZdfPOo7nU5ddNFFGjt2rMaNGyfJHPf4ySefKD4+vsj9pKeneyTr1NRUNW3aVCkpKQoNDbVsJwAAAM6dM2fOKCEhwT1ZGFDeivsZS01NVVhYWImyQakmhyhs3vf8y0ePHl2ibdWvX18Oh6NA71JycnKhXYInTpzQ5s2btW3bNo0fP16SOb+9YRjy8fHRihUrdPnllxdYz9/fX/7+/iVqEwAAAAAUplTBqbRTjRfHz89P0dHRiouL03XXXecuj4uL05AhQwrUDw0N1Y4dOzzK5s+fr2+//VYffPCBWrRoUW5tAwAAAIC8yjwdeXmYNGmSRo0apZiYGPXs2VOvvvqqEhMT3UPxpk6dqgMHDuidd96R3W5Xhw4dPNZv2LChAgICCpQDAAAAQHnyanAaMWKEjhw5olmzZikpKUkdOnTQ8uXLFRUVJUlKSkoq9Q28AAAAAKC8lWpyiOqgNBeAAQAA4Nxicgica+U1OYRXb4ALAAAASFIN+ywfFai8frYITgAAAPAaX19fSVJaWpqXW4LqKiMjQ5J5b6iz4dVrnAAAAFCzORwO1a5dW8nJyZLMm7HabDYvtwrVhcvl0t9//62goCD5+Jxd9CE4AQAAwKvCw8MlyR2egPJkt9vVrFmzsw7kBCcAAAB4lc1mU0REhBo2bKjMzExvNwfVjJ+fn+z2s79CieAEAACASsHhcJz1dSjAucLkEAAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITgAAAABgwevBaf78+WrRooUCAgIUHR2ttWvXFln3o48+Uv/+/dWgQQOFhoaqZ8+e+vrrryuwtQAAAABqIq8Gp6VLl2rixImaNm2atm3bposvvliDBg1SYmJiofW/++479e/fX8uXL9eWLVt02WWX6ZprrtG2bdsquOUAAAAAahKbYRiGt3beo0cPdevWTQsWLHCXtW3bVkOHDlVsbGyJttG+fXuNGDFCM2bMKFH91NRUhYWFKSUlRaGhoWVqNwAAAICqrzTZwGs9ThkZGdqyZYsGDBjgUT5gwACtX7++RNtwuVw6ceKE6tatW2Sd9PR0paamejwAAAAAoDS8FpwOHz4sp9OpRo0aeZQ3atRIhw4dKtE25syZo1OnTmn48OFF1omNjVVYWJj70bRp07NqNwAAAICax+uTQ9hsNo/XhmEUKCvMe++9p8cff1xLly5Vw4YNi6w3depUpaSkuB/79u076zYDAAAAqFl8vLXj+vXry+FwFOhdSk5OLtALld/SpUs1duxYLVu2TP369Su2rr+/v/z9/c+6vQAAAABqLq/1OPn5+Sk6OlpxcXEe5XFxcerVq1eR67333nsaM2aM3n33XQ0ePPhcNxMAAAAAvNfjJEmTJk3SqFGjFBMTo549e+rVV19VYmKixo0bJ8kcZnfgwAG98847kszQNHr0aP3f//2fLrroIndvVWBgoMLCwrx2HAAAAACqN68GpxEjRujIkSOaNWuWkpKS1KFDBy1fvlxRUVGSpKSkJI97Or3yyivKysrSvffeq3vvvdddfuutt2rhwoUV3XwAAAAANYRX7+PkDdzHCQAAAIBURe7jBAAAAABVBcEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAAsEJAAAAACwQnAAAAADAgo+3G1CT/XIoVRv2HFGgr0OBfg6PrwF5ngf5ma/9feyy2WzebjYAAABQ43g9OM2fP1/PPfeckpKS1L59e73w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/aRJk/T666/rzTff1K5du/TAAw8oMTHRfV+mqVOnavTo0e7648aN0969ezVp0iTt2rVLb775pt544w099NBD3joEAAAAADWAV69xGjFihI4cOaJZs2YpKSlJHTp00PLlyxUVFSVJSkpKUmJiort+ixYttHz5cj3wwAP6z3/+o8jISL344otV8h5OAAAAAKoOr97HyRuq4n2cAAAAAJS/0mSDyjEHLgAAAABUYgQnAAAAALBAcAIAAAAACwQnAAAAALBAcAIAAAAACwQnAAAAALBAcAIAAAAACwQnAAAAALBAcAIAAAAACwQnAAAAALBAcAIAAAAACwQnAAAAALBAcAIAAAAACz7ebkBFMwxDkpSamurllgAAAADwppxMkJMRilPjgtOJEyckSU2bNvVySwAAAABUBidOnFBYWFixdWxGSeJVNeJyuXTw4EGFhITIZrN5uzlVWmpqqpo2bap9+/YpNDTU281BBeCc1zyc85qHc14zcd5rHs65yTAMnThxQpGRkbLbi7+Kqcb1ONntdjVp0sTbzahWQkNDa/QvXE3EOa95OOc1D+e8ZuK81zycc1n2NOVgcggAAAAAsEBwAgAAAAALBCeUmb+/vx577DH5+/t7uymoIJzzmodzXvNwzmsmznvNwzkvvRo3OQQAAAAAlBY9TgAAAABggeAEAAAAABYITgAAAABggeAEAAAAABYITnCLjY3VhRdeqJCQEDVs2FBDhw7Vr7/+6lHHMAw9/vjjioyMVGBgoC699FL9/PPPHnXS09M1YcIE1a9fX7Vq1dK1116r/fv3V+ShoIxiY2Nls9k0ceJEdxnnvHo6cOCAbrnlFtWrV09BQUHq0qWLtmzZ4l7Oea9esrKy9K9//UstWrRQYGCgWrZsqVmzZsnlcrnrcM6rtu+++07XXHONIiMjZbPZ9Mknn3gsL6/ze+zYMY0aNUphYWEKCwvTqFGjdPz48XN8dChKcec9MzNTU6ZMUceOHVWrVi1FRkZq9OjROnjwoMc2OO8lR3CC25o1a3Tvvfdq48aNiouLU1ZWlgYMGKBTp06568yePVvPP/+85s2bp02bNik8PFz9+/fXiRMn3HUmTpyojz/+WEuWLNG6det08uRJXX311XI6nd44LJTQpk2b9Oqrr6pTp04e5Zzz6ufYsWPq3bu3fH199eWXX2rnzp2aM2eOateu7a7Dea9enn32Wb388suaN2+edu3apdmzZ+u5557TSy+95K7DOa/aTp06pc6dO2vevHmFLi+v83vzzTcrPj5eX331lb766ivFx8dr1KhR5/z4ULjizntaWpq2bt2q6dOna+vWrfroo4/022+/6dprr/Wox3kvBQMoQnJysiHJWLNmjWEYhuFyuYzw8HDjmWeecdc5c+aMERYWZrz88suGYRjG8ePHDV9fX2PJkiXuOgcOHDDsdrvx1VdfVewBoMROnDhhtG7d2oiLizP69u1r3H///YZhcM6rqylTphh9+vQpcjnnvfoZPHiwcfvtt3uUDRs2zLjlllsMw+CcVzeSjI8//tj9urzO786dOw1JxsaNG911NmzYYEgyfvnll3N8VLCS/7wX5ocffjAkGXv37jUMg/NeWvQ4oUgpKSmSpLp160qSEhISdOjQIQ0YMMBdx9/fX3379tX69eslSVu2bFFmZqZHncjISHXo0MFdB5XPvffeq8GDB6tfv34e5Zzz6unTTz9VTEyM/vGPf6hhw4bq2rWrXnvtNfdyznv106dPH33zzTf67bffJEnbt2/XunXrdNVVV0ninFd35XV+N2zYoLCwMPXo0cNd56KLLlJYWBg/A1VESkqKbDabe4QB5710fLzdAFROhmFo0qRJ6tOnjzp06CBJOnTokCSpUaNGHnUbNWqkvXv3uuv4+fmpTp06BerkrI/KZcmSJdq6das2bdpUYBnnvHr6448/tGDBAk2aNEmPPvqofvjhB913333y9/fX6NGjOe/V0JQpU5SSkqI2bdrI4XDI6XTqqaee0k033SSJ3/XqrrzO76FDh9SwYcMC22/YsCE/A1XAmTNn9Mgjj+jmm29WaGioJM57aRGcUKjx48frxx9/1Lp16woss9lsHq8NwyhQll9J6qDi7du3T/fff79WrFihgICAIutxzqsXl8ulmJgYPf3005Kkrl276ueff9aCBQs0evRodz3Oe/WxdOlSLVq0SO+++67at2+v+Ph4TZw4UZGRkbr11lvd9Tjn1Vt5nN/C6vMzUPllZmbqxhtvlMvl0vz58y3rc94Lx1A9FDBhwgR9+umnWrVqlZo0aeIuDw8Pl6QCny4kJye7P8UKDw9XRkaGjh07VmQdVB5btmxRcnKyoqOj5ePjIx8fH61Zs0YvvviifHx83OeMc169REREqF27dh5lbdu2VWJioiR+16ujhx9+WI888ohuvPFGdezYUaNGjdIDDzyg2NhYSZzz6q68zm94eLj++uuvAtv/+++/+RmoxDIzMzV8+HAlJCQoLi7O3dskcd5Li+AEN8MwNH78eH300Uf69ttv1aJFC4/lLVq0UHh4uOLi4txlGRkZWrNmjXr16iVJio6Olq+vr0edpKQk/fTTT+46qDyuuOIK7dixQ/Hx8e5HTEyMRo4cqfj4eLVs2ZJzXg317t27wK0GfvvtN0VFRUnid706SktLk93u+S/f4XC4pyPnnFdv5XV+e/bsqZSUFP3www/uOv/73/+UkpLCz0AllROadu/erZUrV6pevXoeyznvpeSFCSlQSd19991GWFiYsXr1aiMpKcn9SEtLc9d55plnjLCwMOOjjz4yduzYYdx0001GRESEkZqa6q4zbtw4o0mTJsbKlSuNrVu3GpdffrnRuXNnIysryxuHhVLKO6ueYXDOq6MffvjB8PHxMZ566ilj9+7dxuLFi42goCBj0aJF7jqc9+rl1ltvNRo3bmx8/vnnRkJCgvHRRx8Z9evXNyZPnuyuwzmv2k6cOGFs27bN2LZtmyHJeP75541t27a5Z08rr/N75ZVXGp06dTI2bNhgbNiwwejYsaNx9dVXV/jxwlTcec/MzDSuvfZao0mTJkZ8fLzHe7v09HT3NjjvJUdwgpukQh9vvfWWu47L5TIee+wxIzw83PD39zcuueQSY8eOHR7bOX36tDF+/Hijbt26RmBgoHH11VcbiYmJFXw0KKv8wYlzXj199tlnRocOHQx/f3+jTZs2xquvvuqxnPNevaSmphr333+/0axZMyMgIMBo2bKlMW3aNI83T5zzqm3VqlWF/g+/9dZbDcMov/N75MgRY+TIkUZISIgREhJijBw50jh27FgFHSXyK+68JyQkFPnebtWqVe5tcN5LzmYYhlFx/VsAAAAAUPVwjRMAAAAAWCA4AQAAAIAFghMAAAAAWCA4AQAAAIAFghMAAAAAWCA4AQAAAIAFghMAAAAAWCA4AQAAAIAFghMAoMpJTk7WP//5TzVr1kz+/v4KDw/XwIEDtWHDBkmSzWbTJ5984t1GAgCqFR9vNwAAgNK6/vrrlZmZqbffflstW7bUX3/9pW+++UZHjx71dtMAANUUPU4AgCrl+PHjWrdunZ599llddtllioqKUvfu3TV16lQNHjxYzZs3lyRdd911stls7teS9Nlnnyk6OloBAQFq2bKlZs6cqaysLPdym82mBQsWaNCgQQoMDFSLFi20bNky9/KMjAyNHz9eERERCggIUPPmzRUbG1tRhw4A8CKCEwCgSgkODlZwcLA++eQTpaenF1i+adMmSdJbb72lpKQk9+uvv/5at9xyi+677z7t3LlTr7zyihYuXKinnnrKY/3p06fr+uuv1/bt23XLLbfopptu0q5duyRJL774oj799FO9//77+vXXX7Vo0SKPYAYAqL5shmEY3m4EAACl8eGHH+rOO+/U6dOn1a1bN/Xt21c33nijOnXqJMnsOfr44481dOhQ9zqXXHKJBg0apKlTp7rLFi1apMmTJ+vgwYPu9caNG6cFCxa461x00UXq1q2b5s+fr/vuu08///yzVq5cKZvNVjEHCwCoFOhxAgBUOddff70OHjyoTz/9VAMHDtTq1avVrVs3LVy4sMh1tmzZolmzZrl7rIKDg3XnnXcqKSlJaWlp7no9e/b0WK9nz57uHqcxY8YoPj5eF1xwge677z6tWLHinBwfAKDyITgBAKqkgIAA9e/fXzNmzND69es1ZswYPfbYY0XWd7lcmjlzpuLj492PHTt2aPfu3QoICCh2Xzm9S926dVNCQoKeeOIJnT59WsOHD9cNN9xQrscFAKicCE4AgGqhXbt2OnXqlCTJ19dXTqfTY3m3bt3066+/qlWrVgUednvuv8ONGzd6rLdx40a1adPG/To0NFQjRozQa6+9pqVLl+rDDz9kNj8AqAGYjhwAUKUcOXJE//jHP3T77berU6dOCgkJ0ebNmzV79mwNGTJEktS8eXN988036t27t/z9/VWnTh3NmDFDV199tZo2bap//OMfstvt+vHHH7Vjxw49+eST7u0vW7ZMMTEx6tOnjxYvXqwffvhBb7zxhiRp7ty5ioiIUJcuXWS327Vs2TKFh4erdu3a3vhWAAAqEMEJAFClBAcHq0ePHpo7d6727NmjzMxMNW3aVHfeeaceffRRSdKcOXM0adIkvfbaa2rcuLH+/PNPDRw4UJ9//rlmzZql2bNny9fXV23atNEdd9zhsf2ZM2dqyZIluueeexQeHq7FixerXbt27n0/++yz2r17txwOhy688EItX77co8cKAFA9MaseAADZCpuNDwAAiWucAAAAAMASwQkAAAAALHCNEwAA2Ri9DgAoCj1OAAAAAGCB4AQAAAAAFghOAAAAAGCB4AQAAAAAFghOAAAAAGCB4AQAAAAAFghOAAAAAGCB4AQAAAAAFghOAAAAAGDh/wPu6EGdZKNd/wAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 1000x600 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "from torch.optim.swa_utils import AveragedModel, SWALR\n",
    "from torch.optim.lr_scheduler import CosineAnnealingLR\n",
    "\n",
    "class Sft_Model(nn.Module):\n",
    "    def __init__(self, base_model, num_labels):\n",
    "        super().__init__()\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",
    "    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",
    "device = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")\n",
    "model = Sft_Model(base_model, num_cls).to(device)\n",
    "\n",
    "# Define training arguments\n",
    "training_args = TrainingArguments(\n",
    "    overwrite_output_dir=True,\n",
    "    output_dir=f'./results/sentiment/sft-swa/{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=64,\n",
    "    per_device_eval_batch_size=64,\n",
    "    num_train_epochs=10,\n",
    "    seed=data_seed,\n",
    "    load_best_model_at_end=False,\n",
    "    save_total_limit=1,\n",
    ")\n",
    "\n",
    "# Create the SWA model\n",
    "swa_model = AveragedModel(model).to(device)\n",
    "\n",
    "# Create the optimizer\n",
    "optimizer = torch.optim.AdamW(model.parameters(), lr=training_args.learning_rate)\n",
    "\n",
    "# Create the learning rate scheduler\n",
    "swa_start = 5\n",
    "swa_scheduler = SWALR(optimizer, swa_lr=1e-5)\n",
    "scheduler = CosineAnnealingLR(optimizer, T_max=swa_start)\n",
    "\n",
    "# Create the trainer\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",
    "    optimizers=(optimizer, scheduler),\n",
    ")\n",
    "\n",
    "# Train the model\n",
    "num_epochs = int(training_args.num_train_epochs)\n",
    "for epoch in range(num_epochs):\n",
    "    trainer.train()\n",
    "    \n",
    "    if epoch >= swa_start:\n",
    "        swa_model.update_parameters(model)\n",
    "        swa_scheduler.step()\n",
    "    else:\n",
    "        scheduler.step()\n",
    "\n",
    "    eval_results = trainer.evaluate()\n",
    "    print(f\"Epoch {epoch + 1}/{num_epochs}: {eval_results}\")\n",
    "\n",
    "# Update bn statistics for the swa_model at the end\n",
    "def update_bn(loader, model, device=None):\n",
    "    momenta = {}\n",
    "    for module in model.modules():\n",
    "        if isinstance(module, torch.nn.modules.batchnorm._BatchNorm):\n",
    "            module.running_mean = torch.zeros_like(module.running_mean)\n",
    "            module.running_var = torch.ones_like(module.running_var)\n",
    "            momenta[module] = module.momentum\n",
    "\n",
    "    if not momenta:\n",
    "        return\n",
    "\n",
    "    was_training = model.training\n",
    "    model.train()\n",
    "    for module in momenta.keys():\n",
    "        module.momentum = None\n",
    "        module.num_batches_tracked *= 0\n",
    "\n",
    "    for input_dict in loader:\n",
    "        if device is not None:\n",
    "            input_dict = {k: v.to(device) if isinstance(v, torch.Tensor) else v for k, v in input_dict.items()}\n",
    "        model(**input_dict)\n",
    "\n",
    "    for bn_module in momenta.keys():\n",
    "        bn_module.momentum = momenta[bn_module]\n",
    "    model.train(was_training)\n",
    "\n",
    "# Create a DataLoader for the paired_train_dataset\n",
    "train_loader = DataLoader(paired_train_dataset, batch_size=training_args.per_device_train_batch_size, shuffle=True, collate_fn=paired_data_collator)\n",
    "\n",
    "update_bn(train_loader, swa_model, device)\n",
    "\n",
    "# Set the SWA model as the final model\n",
    "trainer.model = swa_model\n",
    "\n",
    "# Evaluation function\n",
    "def evaluate_model(dataset, name):\n",
    "    results = trainer.evaluate(dataset)\n",
    "    print(f'{name}: {results}')\n",
    "    return results\n",
    "\n",
    "# Evaluate the final SWA model\n",
    "results = {}\n",
    "results['train'] = evaluate_model(paired_train_dataset, 'train')\n",
    "results['valid'] = evaluate_model(paired_validation_dataset, 'validation')\n",
    "results['ID'] = evaluate_model(paired_test_dataset_ID, 'ID')\n",
    "results['OOD change 70'] = evaluate_model(paired_test_dataset_OOD_change_70, 'OOD change 70')\n",
    "results['OOD balanced'] = evaluate_model(paired_test_dataset_OOD_balanced_50, 'OOD balanced')\n",
    "results['OOD change 30'] = evaluate_model(paired_test_dataset_OOD_change_30, 'OOD change 30')\n",
    "results['OOD flip'] = evaluate_model(paired_test_dataset_OOD_flip, 'OOD flip')\n",
    "\n",
    "# Save the results\n",
    "with open(f'./results/sentiment/sft-swa/{N}-shot-{data_seed}/test_results.json', 'w') as f:\n",
    "    json.dump(results, f, indent=4)\n",
    "\n",
    "# Plot training metrics\n",
    "plot_training_metrics(trainer)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# WISE"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Initialise SFT model \"bert-base-uncased\" (unfreezed all layers) with a linear head!\n",
      "Total Parameters: 109483778\n",
      "Trainable Parameters: 109483778\n",
      "Percentage of Trainable Parameters: 100.0000%\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "\n",
       "    <div>\n",
       "      \n",
       "      <progress value='643' max='1250' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [ 643/1250 00:59 < 00:56, 10.75 it/s, Epoch 5.14/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>125</td>\n",
       "      <td>0.296300</td>\n",
       "      <td>0.236379</td>\n",
       "      <td>0.912987</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>250</td>\n",
       "      <td>0.193800</td>\n",
       "      <td>0.240044</td>\n",
       "      <td>0.914453</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>375</td>\n",
       "      <td>0.109100</td>\n",
       "      <td>0.277187</td>\n",
       "      <td>0.915929</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>500</td>\n",
       "      <td>0.062800</td>\n",
       "      <td>0.344156</td>\n",
       "      <td>0.915500</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>625</td>\n",
       "      <td>0.025800</td>\n",
       "      <td>0.458645</td>\n",
       "      <td>0.905428</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='472' max='125' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [125/125 00:18]\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.19010722637176514, 'eval_f1': 0.9348665723450873, 'eval_runtime': 4.619, 'eval_samples_per_second': 1731.979, 'eval_steps_per_second': 27.062, 'epoch': 10.0}\n",
      "validation: {'eval_loss': 0.23637881875038147, 'eval_f1': 0.9129874701957081, 'eval_runtime': 1.5009, 'eval_samples_per_second': 1332.517, 'eval_steps_per_second': 21.32, 'epoch': 10.0}\n",
      "ID: {'eval_loss': 0.23394513130187988, 'eval_f1': 0.9132408848704667, 'eval_runtime': 2.5263, 'eval_samples_per_second': 1583.325, 'eval_steps_per_second': 24.937, 'epoch': 10.0}\n",
      "OOD change 70: {'eval_loss': 0.5507633090019226, 'eval_f1': 0.7744873149114637, 'eval_runtime': 2.5617, 'eval_samples_per_second': 1561.475, 'eval_steps_per_second': 24.593, 'epoch': 10.0}\n",
      "OOD balanced: {'eval_loss': 0.8645104169845581, 'eval_f1': 0.6362039570633158, 'eval_runtime': 2.5478, 'eval_samples_per_second': 1569.96, 'eval_steps_per_second': 24.727, 'epoch': 10.0}\n",
      "OOD change 30: {'eval_loss': 1.167434811592102, 'eval_f1': 0.49888724963116704, 'eval_runtime': 2.8437, 'eval_samples_per_second': 1406.615, 'eval_steps_per_second': 22.154, 'epoch': 10.0}\n",
      "OOD flip: {'eval_loss': 1.4797617197036743, 'eval_f1': 0.3571371373886828, 'eval_runtime': 2.5389, 'eval_samples_per_second': 1575.512, 'eval_steps_per_second': 24.814, 'epoch': 10.0}\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAA04AAAINCAYAAAAJGy/3AAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8pXeV/AAAACXBIWXMAAA9hAAAPYQGoP6dpAABzO0lEQVR4nO3dd3gU1f7H8c/upldCSwIkFOkdQi8qiiCggqAgnWvFgiI/G9drAfViQ7lcBcWGigpiuyqgBBUE6SVIE5AWSkKoSSA9O78/BjYEQkJCyCSb9+t59snumTOz32WA5JNz5ozNMAxDAAAAAICLsltdAAAAAACUdgQnAAAAACgAwQkAAAAACkBwAgAAAIACEJwAAAAAoAAEJwAAAAAoAMEJAAAAAApAcAIAAACAAnhYXUBJczqdOnTokAIDA2Wz2awuBwAAAIBFDMNQcnKyqlWrJrs9/zGlchecDh06pIiICKvLAAAAAFBK7N+/XzVq1Mi3T7kLToGBgZLMP5ygoCCLqwEAAABglaSkJEVERLgyQn7KXXA6Oz0vKCiI4AQAAADgki7hYXEIAAAAACgAwQkAAAAACkBwAgAAAIAClLtrnC6FYRjKyspSdna21aXAzTgcDnl4eLAUPgAAQBlDcDpPRkaG4uLilJKSYnUpcFN+fn4KDw+Xl5eX1aUAAADgEhGczuF0OrVnzx45HA5Vq1ZNXl5ejAyg2BiGoYyMDB05ckR79uxRvXr1CrzRGgAAAEoHgtM5MjIy5HQ6FRERIT8/P6vLgRvy9fWVp6en9u3bp4yMDPn4+FhdEgAAAC4Bv+7OA6MAuJL4+wUAAFD28BMcAAAAABSA4ISLuvbaazV27NhL7r93717ZbDbFxMRcsZoAAAAAKxCc3IDNZsv3MWrUqCId95tvvtELL7xwyf0jIiIUFxenpk2bFun9LhUBDQAAACWNxSHcQFxcnOv5nDlz9Oyzz2r79u2uNl9f31z9MzMz5enpWeBxK1asWKg6HA6HwsLCCrUPAAAAUBYw4uQGwsLCXI/g4GDZbDbX67S0NFWoUEFffvmlrr32Wvn4+GjWrFk6duyYBg8erBo1asjPz0/NmjXTF198keu450/Vq1Wrlv7973/rzjvvVGBgoCIjIzVjxgzX9vNHghYvXiybzaZffvlFbdq0kZ+fnzp16pQr1EnSiy++qKpVqyowMFB33323nnrqKbVs2bLIfx7p6el6+OGHVbVqVfn4+KhLly5as2aNa/uJEyc0dOhQValSRb6+vqpXr54++ugjSebKig899JDCw8Pl4+OjWrVqadKkSUWuBQAAAO6B4FQAwzCUkpFlycMwjGL7HE8++aQefvhhbdu2TT179lRaWpqioqL0448/avPmzbr33ns1fPhwrVq1Kt/jTJ48WW3atNGGDRv0wAMP6P7779dff/2V7z5PP/20Jk+erLVr18rDw0N33nmna9tnn32ml156Sa+88orWrVunyMhITZ8+/bI+6xNPPKGvv/5aH3/8sdavX6+6deuqZ8+eOn78uCTpmWee0datW7VgwQJt27ZN06dPV+XKlSVJU6dO1ffff68vv/xS27dv16xZs1SrVq3LqgcAAABlH1P1CpCama3Gz/5syXtvndhTfl7Fc4rGjh2r/v3752p77LHHXM/HjBmjn376SXPnzlX79u0vepzevXvrgQcekGSGsTfffFOLFy9Ww4YNL7rPSy+9pGuuuUaS9NRTT6lPnz5KS0uTj4+P/vvf/+quu+7SP/7xD0nSs88+q4ULF+rUqVNF+pynT5/W9OnTNXPmTPXq1UuS9N577yk6OloffPCBHn/8ccXGxqpVq1Zq06aNJOUKRrGxsapXr566dOkim82mmjVrFqkOAAAAuBdGnMqJsyHhrOzsbL300ktq3ry5KlWqpICAAC1cuFCxsbH5Hqd58+au52enBCYkJFzyPuHh4ZLk2mf79u1q165drv7nvy6MXbt2KTMzU507d3a1eXp6ql27dtq2bZsk6f7779fs2bPVsmVLPfHEE1q+fLmr76hRoxQTE6MGDRro4Ycf1sKFC4tcCwAAANwHI04F8PV0aOvEnpa9d3Hx9/fP9Xry5Ml68803NWXKFDVr1kz+/v4aO3asMjIy8j3O+YtK2Gw2OZ3OS97HZrNJUq59zraddTlTFM/um9cxz7b16tVL+/bt07x587Ro0SJdf/31evDBB/X666+rdevW2rNnjxYsWKBFixZp4MCB6t69u7766qsi1wQAQKmWmSbt+0Oy2SWfYPPhGyJ5B0kOflQEzrL8X8O0adP02muvKS4uTk2aNNGUKVPUtWvXi/ZPT0/XxIkTNWvWLMXHx6tGjRp6+umnc103U5xsNluxTZcrTZYuXaq+fftq2LBhkswgs3PnTjVq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QZaJRkng7xkA4JJkZUjbvpdWvSsdOOf+huEtpHb3SU0HmCvXASiyMrEcOQAAAPKQFCetmymt+0g6ddhss3tKTfqZq+PVaMt0PMACBCcAAACrGYa0f5U5HW/r/3Lu9RMQJrW5U4oaVbR7IgEoNiz8Dpdrr71WY8eOdb2uVauWpkyZku8+NptN33333WW/d3EdBwCAMiUzVVr/qfTu1dKHPaXNX5uhKaKDdNuH0thN0rVPEpqAUoARJzdw8803KzU1Nc/7Ia1YsUKdOnXSunXr1Lp160Idd82aNfL39y+uMiVJzz//vL777jvFxMTkao+Li1NISEixvtf5Zs6cqbFjx+rkyZNX9H0AACjQyVhpzfvS+k+k1BNmm4eP1Ow2czpeeAtr6wNwAYKTG7jrrrvUv39/7du3z7Ui4VkffvihWrZsWejQJElVqpTc/QvCwsJK7L0AALCEYUh7lkirZpj3YDKcZntwpNT2Lqn1iKLftBXAFcdUPTdw0003qWrVqpo5c2au9pSUFM2ZM0d33XWXjh07psGDB6tGjRry8/NTs2bN9MUXX+R73POn6u3cuVNXX321fHx81LhxY0VHR1+wz5NPPqn69evLz89PderU0TPPPKPMzExJ5ojPhAkTtHHjRtlsNtlsNlfN50/V27Rpk6677jr5+vqqUqVKuvfee3Xq1CnX9lGjRqlfv356/fXXFR4erkqVKunBBx90vVdRxMbGqm/fvgoICFBQUJAGDhyow4cPu7Zv3LhR3bp1U2BgoIKCghQVFaW1a9dKkvbt26ebb75ZISEh8vf3V5MmTTR//vwi1wIAcCPpydLq96S320uf9JW2zzNDU51rpTs+lx6JkbqMJTQBpRwjTgUxDCkzxZr39vS7pFVzPDw8NGLECM2cOVPPPvusbGf2mTt3rjIyMjR06FClpKQoKipKTz75pIKCgjRv3jwNHz5cderUUfv27Qt8D6fTqf79+6ty5cpauXKlkpKScl0PdVZgYKBmzpypatWqadOmTbrnnnsUGBioJ554QoMGDdLmzZv1008/uaYVBgcHX3CMlJQU3XjjjerQoYPWrFmjhIQE3X333XrooYdyhcPffvtN4eHh+u233/T3339r0KBBatmype65554CP8/5DMNQv3795O/vryVLligrK0sPPPCABg0apMWLF0uShg4dqlatWmn69OlyOByK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AAAoCwhOAAAAAFAAghMAAAAAFIDgBAAAAAAFIDgBAAAAQAEITgAAAABQAIITAAAAABSA4AQAAAAABSA4AQAAAEABCE4AAAAAUACCEwAAAAAUgOAEAAAAAAUgOAEAAABAATysLqCkGYYhSUpKSrK4EgAAAABWOpsJzmaE/JS74JScnCxJioiIsLgSAAAAAKVBcnKygoOD8+1jMy4lXrkRp9OpQ4cOKTAwUDabzepyyrSkpCRFRERo//79CgoKsroclADOefnDOS9/OOflE+e9/OGcmwzDUHJysqpVqya7Pf+rmMrdiJPdbleNGjWsLsOtBAUFlet/cOUR57z84ZyXP5zz8onzXv5wzlXgSNNZLA4BAAAAAAUgOAEAAABAAQhOKDJvb28999xz8vb2troUlBDOefnDOS9/OOflE+e9/OGcF165WxwCAAAAAAqLEScAAAAAKADBCQAAAAAKQHACAAAAgAIQnAAAAACgAAQnuEyaNElt27ZVYGCgqlatqn79+mn79u25+hiGoeeff17VqlWTr6+vrr32Wm3ZsiVXn/T0dI0ZM0aVK1eWv7+/brnlFh04cKAkPwqKaNKkSbLZbBo7dqyrjXPung4ePKhhw4apUqVK8vPzU8uWLbVu3TrXds67e8nKytK//vUv1a5dW76+vqpTp44mTpwop9Pp6sM5L9t+//133XzzzapWrZpsNpu+++67XNuL6/yeOHFCw4cPV3BwsIKDgzV8+HCdPHnyCn86XEx+5z0zM1NPPvmkmjVrJn9/f1WrVk0jRozQoUOHch2D837pCE5wWbJkiR588EGtXLlS0dHRysrKUo8ePXT69GlXn1dffVVvvPGG3nrrLa1Zs0ZhYWG64YYblJyc7OozduxYffvtt5o9e7aWLVumU6dO6aabblJ2drYVHwuXaM2aNZoxY4aaN2+eq51z7n5OnDihzp07y9PTUwsWLNDWrVs1efJkVahQwdWH8+5eXnnlFb3zzjt66623tG3bNr366qt67bXX9N///tfVh3Netp0+fVotWrTQW2+9lef24jq/Q4YMUUxMjH766Sf99NNPiomJ0fDhw6/450Pe8jvvKSkpWr9+vZ555hmtX79e33zzjXbs2KFbbrklVz/OeyEYwEUkJCQYkowlS5YYhmEYTqfTCAsLM15++WVXn7S0NCM4ONh45513DMMwjJMnTxqe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zDMaMGCANm7cqGHDhmnw4MHatm2bJGnq1Kn6/vvv9eWXX2r79u2aNWtWrmAGAHBfNsMwDKuLAACgML7++mvdc889Sk1NVevWrXXNNdfojjvuUPPmzSWZI0fffvut+vXr59rn6quvVq9evTR+/HhX26xZs/TEE0/o0KFDrv1Gjx6t6dOnu/p06NBBrVu31rRp0/Twww9ry5YtWrRokWw2W8l8WABAqcCIEwCgzBkwYIAOHTqk77//Xj179tTixYvVunVrzZw586L7rFu3ThMnTnSNWAUEBOiee+5RXFycUlJSXP06duyYa7+OHTu6RpxGjRqlmJgYNWjQQA8//LAWLlx4RT4fAKD0ITgBAMokHx8f3XDDDXr22We1fPlyjRo1Ss8999xF+zudTk2YMEExMTGux6ZNm7Rz5075+Pjk+15nR5dat26tPXv26IUXXlBqaqoGDhyo2267rVg/FwCgdCI4AQDcQuPGjXX69GlJkqenp7Kzs3Ntb926tbZv3666dete8LDbc74drly5Mtd+K1euVMOGDV2vg4KCNGjQIL333nuaM2eOvv76a1bzA4BygOXIAQBlyrFjx3T77bfrzjvvVPPmzRUYGKi1a9fq1VdfVd++fSVJtWrV0i+//KLOnTvL29tbISEhevbZZ3XTTTcpIiJCt99+u+x2u/78809t2rRJL774ouv4c+fOVZs2bdSlSxd99tlnWr16tT744ANJ0ptvvqnw8HC1bNlSdrtdc+fOVVhYmCpUqGDFHwUAoAQRnAAAZUpAQIDat2+vN998U7t27VJmZqYiIiJ0zz336J///KckafLkyRo3bpzee+89Va9eXXv37lXPnj31448/auLEiXr11Vfl6emphg0b6u677851/AkTJmj27Nl64IEHFBYWps8++0yNGzd2vfcrr7yinTt3yuFwqG3btpo/f36uESsAgHtiVT0AAM7IazU+AAAkrnECAAAAgAIRnAAAAACgAFzjBADAGcxeBwBcDCNOAAAAAFAAghMAAAAAFIDgBAAAAAAFIDgBAAAAQAEITgAAAABQAIITAAAAABSA4AQAAAAABSA4AQAAAEABCE4AAAAAUID/BztPDriyghaAAAAAAElFTkSuQmCC",
      "text/plain": [
       "<Figure size 1000x600 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "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",
    "def wise_ft(model, alpha=0.5):\n",
    "    # Create a copy of the initial (zero-shot) model\n",
    "    zero_shot_model = copy.deepcopy(model)\n",
    "    \n",
    "    # Freeze the zero-shot model\n",
    "    for param in zero_shot_model.parameters():\n",
    "        param.requires_grad = False\n",
    "    \n",
    "    # Perform the weight averaging\n",
    "    for p1, p2 in zip(model.parameters(), zero_shot_model.parameters()):\n",
    "        p1.data = alpha * p1.data + (1 - alpha) * p2.data\n",
    "    \n",
    "    return model\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/sentiment/sft-wise-ft/{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=64,\n",
    "    per_device_eval_batch_size=64,\n",
    "    num_train_epochs=10,\n",
    "    seed=data_seed,\n",
    "    load_best_model_at_end=True,\n",
    "    metric_for_best_model=\"eval_loss\",\n",
    "    greater_is_better=False,\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",
    "# Apply WiSE-FT\n",
    "wise_ft_model = wise_ft(trainer.model, alpha=0.5)  # You can adjust alpha as needed\n",
    "trainer.model = wise_ft_model\n",
    "\n",
    "# Evaluate the WiSE-FT model\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 sets\n",
    "test_results_ID = trainer.evaluate(paired_test_dataset_ID)\n",
    "print(f'ID: {test_results_ID}')\n",
    "results['ID'] = test_results_ID\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 = trainer.evaluate(paired_test_dataset_OOD_balanced_50)\n",
    "print(f'OOD balanced: {test_results_OOD_balanced}')\n",
    "results['OOD balanced'] = test_results_OOD_balanced\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 = trainer.evaluate(paired_test_dataset_OOD_flip)\n",
    "print(f'OOD flip: {test_results_OOD_flip}')\n",
    "results['OOD flip'] = test_results_OOD_flip\n",
    "\n",
    "# Save the results\n",
    "with open(f'./results/sentiment/sft-wise-ft/{N}-shot-{data_seed}/test_results.json', 'w') as f:\n",
    "    json.dump(results, f, indent=4)\n",
    "\n",
    "# Plot training metrics\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
}
